plain 0.62.1__tar.gz → 0.64.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.
Files changed (179) hide show
  1. {plain-0.62.1 → plain-0.64.0}/.gitignore +0 -2
  2. {plain-0.62.1 → plain-0.64.0}/PKG-INFO +1 -1
  3. {plain-0.62.1 → plain-0.64.0}/plain/CHANGELOG.md +25 -1
  4. {plain-0.62.1 → plain-0.64.0}/plain/chores/README.md +1 -1
  5. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/request.py +1 -1
  6. {plain-0.62.1 → plain-0.64.0}/plain/cli/formatting.py +12 -0
  7. {plain-0.62.1 → plain-0.64.0}/plain/cli/upgrade.py +0 -1
  8. {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/https.py +4 -17
  9. {plain-0.62.1 → plain-0.64.0}/plain/runtime/global_settings.py +3 -9
  10. {plain-0.62.1 → plain-0.64.0}/plain/signing.py +6 -2
  11. {plain-0.62.1 → plain-0.64.0}/plain/test/README.md +1 -1
  12. {plain-0.62.1 → plain-0.64.0}/plain/utils/crypto.py +0 -5
  13. {plain-0.62.1 → plain-0.64.0}/plain/views/README.md +4 -4
  14. {plain-0.62.1 → plain-0.64.0}/pyproject.toml +2 -1
  15. {plain-0.62.1 → plain-0.64.0}/LICENSE +0 -0
  16. {plain-0.62.1 → plain-0.64.0}/README.md +0 -0
  17. {plain-0.62.1 → plain-0.64.0}/plain/AGENTS.md +0 -0
  18. {plain-0.62.1 → plain-0.64.0}/plain/README.md +0 -0
  19. {plain-0.62.1 → plain-0.64.0}/plain/__main__.py +0 -0
  20. {plain-0.62.1 → plain-0.64.0}/plain/assets/README.md +0 -0
  21. {plain-0.62.1 → plain-0.64.0}/plain/assets/__init__.py +0 -0
  22. {plain-0.62.1 → plain-0.64.0}/plain/assets/compile.py +0 -0
  23. {plain-0.62.1 → plain-0.64.0}/plain/assets/finders.py +0 -0
  24. {plain-0.62.1 → plain-0.64.0}/plain/assets/fingerprints.py +0 -0
  25. {plain-0.62.1 → plain-0.64.0}/plain/assets/urls.py +0 -0
  26. {plain-0.62.1 → plain-0.64.0}/plain/assets/views.py +0 -0
  27. {plain-0.62.1 → plain-0.64.0}/plain/chores/__init__.py +0 -0
  28. {plain-0.62.1 → plain-0.64.0}/plain/chores/registry.py +0 -0
  29. {plain-0.62.1 → plain-0.64.0}/plain/cli/README.md +0 -0
  30. {plain-0.62.1 → plain-0.64.0}/plain/cli/__init__.py +0 -0
  31. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/__init__.py +0 -0
  32. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/docs.py +0 -0
  33. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/llmdocs.py +0 -0
  34. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/md.py +0 -0
  35. {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/prompt.py +0 -0
  36. {plain-0.62.1 → plain-0.64.0}/plain/cli/build.py +0 -0
  37. {plain-0.62.1 → plain-0.64.0}/plain/cli/changelog.py +0 -0
  38. {plain-0.62.1 → plain-0.64.0}/plain/cli/chores.py +0 -0
  39. {plain-0.62.1 → plain-0.64.0}/plain/cli/core.py +0 -0
  40. {plain-0.62.1 → plain-0.64.0}/plain/cli/docs.py +0 -0
  41. {plain-0.62.1 → plain-0.64.0}/plain/cli/install.py +0 -0
  42. {plain-0.62.1 → plain-0.64.0}/plain/cli/output.py +0 -0
  43. {plain-0.62.1 → plain-0.64.0}/plain/cli/preflight.py +0 -0
  44. {plain-0.62.1 → plain-0.64.0}/plain/cli/print.py +0 -0
  45. {plain-0.62.1 → plain-0.64.0}/plain/cli/registry.py +0 -0
  46. {plain-0.62.1 → plain-0.64.0}/plain/cli/scaffold.py +0 -0
  47. {plain-0.62.1 → plain-0.64.0}/plain/cli/settings.py +0 -0
  48. {plain-0.62.1 → plain-0.64.0}/plain/cli/shell.py +0 -0
  49. {plain-0.62.1 → plain-0.64.0}/plain/cli/startup.py +0 -0
  50. {plain-0.62.1 → plain-0.64.0}/plain/cli/urls.py +0 -0
  51. {plain-0.62.1 → plain-0.64.0}/plain/cli/utils.py +0 -0
  52. {plain-0.62.1 → plain-0.64.0}/plain/csrf/README.md +0 -0
  53. {plain-0.62.1 → plain-0.64.0}/plain/csrf/middleware.py +0 -0
  54. {plain-0.62.1 → plain-0.64.0}/plain/csrf/views.py +0 -0
  55. {plain-0.62.1 → plain-0.64.0}/plain/debug.py +0 -0
  56. {plain-0.62.1 → plain-0.64.0}/plain/exceptions.py +0 -0
  57. {plain-0.62.1 → plain-0.64.0}/plain/forms/README.md +0 -0
  58. {plain-0.62.1 → plain-0.64.0}/plain/forms/__init__.py +0 -0
  59. {plain-0.62.1 → plain-0.64.0}/plain/forms/boundfield.py +0 -0
  60. {plain-0.62.1 → plain-0.64.0}/plain/forms/exceptions.py +0 -0
  61. {plain-0.62.1 → plain-0.64.0}/plain/forms/fields.py +0 -0
  62. {plain-0.62.1 → plain-0.64.0}/plain/forms/forms.py +0 -0
  63. {plain-0.62.1 → plain-0.64.0}/plain/http/README.md +0 -0
  64. {plain-0.62.1 → plain-0.64.0}/plain/http/__init__.py +0 -0
  65. {plain-0.62.1 → plain-0.64.0}/plain/http/cookie.py +0 -0
  66. {plain-0.62.1 → plain-0.64.0}/plain/http/multipartparser.py +0 -0
  67. {plain-0.62.1 → plain-0.64.0}/plain/http/request.py +0 -0
  68. {plain-0.62.1 → plain-0.64.0}/plain/http/response.py +0 -0
  69. {plain-0.62.1 → plain-0.64.0}/plain/internal/__init__.py +0 -0
  70. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/__init__.py +0 -0
  71. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/base.py +0 -0
  72. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/locks.py +0 -0
  73. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/move.py +0 -0
  74. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/temp.py +0 -0
  75. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/uploadedfile.py +0 -0
  76. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/uploadhandler.py +0 -0
  77. {plain-0.62.1 → plain-0.64.0}/plain/internal/files/utils.py +0 -0
  78. {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/__init__.py +0 -0
  79. {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/base.py +0 -0
  80. {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/exception.py +0 -0
  81. {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/wsgi.py +0 -0
  82. {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/__init__.py +0 -0
  83. {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/headers.py +0 -0
  84. {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/slash.py +0 -0
  85. {plain-0.62.1 → plain-0.64.0}/plain/json.py +0 -0
  86. {plain-0.62.1 → plain-0.64.0}/plain/logs/README.md +0 -0
  87. {plain-0.62.1 → plain-0.64.0}/plain/logs/__init__.py +0 -0
  88. {plain-0.62.1 → plain-0.64.0}/plain/logs/configure.py +0 -0
  89. {plain-0.62.1 → plain-0.64.0}/plain/logs/debug.py +0 -0
  90. {plain-0.62.1 → plain-0.64.0}/plain/logs/formatters.py +0 -0
  91. {plain-0.62.1 → plain-0.64.0}/plain/logs/loggers.py +0 -0
  92. {plain-0.62.1 → plain-0.64.0}/plain/logs/utils.py +0 -0
  93. {plain-0.62.1 → plain-0.64.0}/plain/packages/README.md +0 -0
  94. {plain-0.62.1 → plain-0.64.0}/plain/packages/__init__.py +0 -0
  95. {plain-0.62.1 → plain-0.64.0}/plain/packages/config.py +0 -0
  96. {plain-0.62.1 → plain-0.64.0}/plain/packages/registry.py +0 -0
  97. {plain-0.62.1 → plain-0.64.0}/plain/paginator.py +0 -0
  98. {plain-0.62.1 → plain-0.64.0}/plain/preflight/README.md +0 -0
  99. {plain-0.62.1 → plain-0.64.0}/plain/preflight/__init__.py +0 -0
  100. {plain-0.62.1 → plain-0.64.0}/plain/preflight/files.py +0 -0
  101. {plain-0.62.1 → plain-0.64.0}/plain/preflight/messages.py +0 -0
  102. {plain-0.62.1 → plain-0.64.0}/plain/preflight/registry.py +0 -0
  103. {plain-0.62.1 → plain-0.64.0}/plain/preflight/security.py +0 -0
  104. {plain-0.62.1 → plain-0.64.0}/plain/preflight/urls.py +0 -0
  105. {plain-0.62.1 → plain-0.64.0}/plain/runtime/README.md +0 -0
  106. {plain-0.62.1 → plain-0.64.0}/plain/runtime/__init__.py +0 -0
  107. {plain-0.62.1 → plain-0.64.0}/plain/runtime/user_settings.py +0 -0
  108. {plain-0.62.1 → plain-0.64.0}/plain/runtime/utils.py +0 -0
  109. {plain-0.62.1 → plain-0.64.0}/plain/signals/README.md +0 -0
  110. {plain-0.62.1 → plain-0.64.0}/plain/signals/__init__.py +0 -0
  111. {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/__init__.py +0 -0
  112. {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/dispatcher.py +0 -0
  113. {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/license.txt +0 -0
  114. {plain-0.62.1 → plain-0.64.0}/plain/templates/AGENTS.md +0 -0
  115. {plain-0.62.1 → plain-0.64.0}/plain/templates/README.md +0 -0
  116. {plain-0.62.1 → plain-0.64.0}/plain/templates/__init__.py +0 -0
  117. {plain-0.62.1 → plain-0.64.0}/plain/templates/core.py +0 -0
  118. {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/__init__.py +0 -0
  119. {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/environments.py +0 -0
  120. {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/extensions.py +0 -0
  121. {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/filters.py +0 -0
  122. {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/globals.py +0 -0
  123. {plain-0.62.1 → plain-0.64.0}/plain/test/__init__.py +0 -0
  124. {plain-0.62.1 → plain-0.64.0}/plain/test/client.py +0 -0
  125. {plain-0.62.1 → plain-0.64.0}/plain/test/encoding.py +0 -0
  126. {plain-0.62.1 → plain-0.64.0}/plain/test/exceptions.py +0 -0
  127. {plain-0.62.1 → plain-0.64.0}/plain/urls/README.md +0 -0
  128. {plain-0.62.1 → plain-0.64.0}/plain/urls/__init__.py +0 -0
  129. {plain-0.62.1 → plain-0.64.0}/plain/urls/converters.py +0 -0
  130. {plain-0.62.1 → plain-0.64.0}/plain/urls/exceptions.py +0 -0
  131. {plain-0.62.1 → plain-0.64.0}/plain/urls/patterns.py +0 -0
  132. {plain-0.62.1 → plain-0.64.0}/plain/urls/resolvers.py +0 -0
  133. {plain-0.62.1 → plain-0.64.0}/plain/urls/routers.py +0 -0
  134. {plain-0.62.1 → plain-0.64.0}/plain/urls/utils.py +0 -0
  135. {plain-0.62.1 → plain-0.64.0}/plain/utils/README.md +0 -0
  136. {plain-0.62.1 → plain-0.64.0}/plain/utils/__init__.py +0 -0
  137. {plain-0.62.1 → plain-0.64.0}/plain/utils/cache.py +0 -0
  138. {plain-0.62.1 → plain-0.64.0}/plain/utils/datastructures.py +0 -0
  139. {plain-0.62.1 → plain-0.64.0}/plain/utils/dateparse.py +0 -0
  140. {plain-0.62.1 → plain-0.64.0}/plain/utils/deconstruct.py +0 -0
  141. {plain-0.62.1 → plain-0.64.0}/plain/utils/decorators.py +0 -0
  142. {plain-0.62.1 → plain-0.64.0}/plain/utils/duration.py +0 -0
  143. {plain-0.62.1 → plain-0.64.0}/plain/utils/encoding.py +0 -0
  144. {plain-0.62.1 → plain-0.64.0}/plain/utils/functional.py +0 -0
  145. {plain-0.62.1 → plain-0.64.0}/plain/utils/hashable.py +0 -0
  146. {plain-0.62.1 → plain-0.64.0}/plain/utils/html.py +0 -0
  147. {plain-0.62.1 → plain-0.64.0}/plain/utils/http.py +0 -0
  148. {plain-0.62.1 → plain-0.64.0}/plain/utils/inspect.py +0 -0
  149. {plain-0.62.1 → plain-0.64.0}/plain/utils/ipv6.py +0 -0
  150. {plain-0.62.1 → plain-0.64.0}/plain/utils/itercompat.py +0 -0
  151. {plain-0.62.1 → plain-0.64.0}/plain/utils/module_loading.py +0 -0
  152. {plain-0.62.1 → plain-0.64.0}/plain/utils/regex_helper.py +0 -0
  153. {plain-0.62.1 → plain-0.64.0}/plain/utils/safestring.py +0 -0
  154. {plain-0.62.1 → plain-0.64.0}/plain/utils/text.py +0 -0
  155. {plain-0.62.1 → plain-0.64.0}/plain/utils/timesince.py +0 -0
  156. {plain-0.62.1 → plain-0.64.0}/plain/utils/timezone.py +0 -0
  157. {plain-0.62.1 → plain-0.64.0}/plain/utils/tree.py +0 -0
  158. {plain-0.62.1 → plain-0.64.0}/plain/validators.py +0 -0
  159. {plain-0.62.1 → plain-0.64.0}/plain/views/__init__.py +0 -0
  160. {plain-0.62.1 → plain-0.64.0}/plain/views/base.py +0 -0
  161. {plain-0.62.1 → plain-0.64.0}/plain/views/errors.py +0 -0
  162. {plain-0.62.1 → plain-0.64.0}/plain/views/exceptions.py +0 -0
  163. {plain-0.62.1 → plain-0.64.0}/plain/views/forms.py +0 -0
  164. {plain-0.62.1 → plain-0.64.0}/plain/views/objects.py +0 -0
  165. {plain-0.62.1 → plain-0.64.0}/plain/views/redirect.py +0 -0
  166. {plain-0.62.1 → plain-0.64.0}/plain/views/templates.py +0 -0
  167. {plain-0.62.1 → plain-0.64.0}/plain/wsgi.py +0 -0
  168. {plain-0.62.1 → plain-0.64.0}/tests/.gitignore +0 -0
  169. {plain-0.62.1 → plain-0.64.0}/tests/app/.gitignore +0 -0
  170. {plain-0.62.1 → plain-0.64.0}/tests/app/settings.py +0 -0
  171. {plain-0.62.1 → plain-0.64.0}/tests/app/test/__init__.py +0 -0
  172. {plain-0.62.1 → plain-0.64.0}/tests/app/test/default_settings.py +0 -0
  173. {plain-0.62.1 → plain-0.64.0}/tests/app/urls.py +0 -0
  174. {plain-0.62.1 → plain-0.64.0}/tests/conftest.py +0 -0
  175. {plain-0.62.1 → plain-0.64.0}/tests/test_cli.py +0 -0
  176. {plain-0.62.1 → plain-0.64.0}/tests/test_csrf.py +0 -0
  177. {plain-0.62.1 → plain-0.64.0}/tests/test_logs.py +0 -0
  178. {plain-0.62.1 → plain-0.64.0}/tests/test_runtime.py +0 -0
  179. {plain-0.62.1 → plain-0.64.0}/tests/test_wsgi.py +0 -0
@@ -11,8 +11,6 @@ plain*/tests/.plain
11
11
  # Ottobot
12
12
  .aider*
13
13
 
14
- /llms-full.txt
15
-
16
14
  # Plain temp dirs
17
15
  .plain
18
16
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: plain
3
- Version: 0.62.1
3
+ Version: 0.64.0
4
4
  Summary: A web framework for building products with Python.
5
5
  Author-email: Dave Gaeddert <dave.gaeddert@dropseed.dev>
6
6
  License-File: LICENSE
@@ -1,5 +1,29 @@
1
1
  # plain changelog
2
2
 
3
+ ## [0.64.0](https://github.com/dropseed/plain/releases/plain@0.64.0) (2025-09-19)
4
+
5
+ ### What's changed
6
+
7
+ - Added `plain-build` command as a standalone executable ([4b39ca4](https://github.com/dropseed/plain/commit/4b39ca4599))
8
+ - Removed `constant_time_compare` utility function in favor of `hmac.compare_digest` ([55f3f55](https://github.com/dropseed/plain/commit/55f3f5596d))
9
+ - CLI now forces colors in CI environments (GitHub Actions, GitLab CI, etc.) for better output visibility ([56f7d2b](https://github.com/dropseed/plain/commit/56f7d2b312))
10
+
11
+ ### Upgrade instructions
12
+
13
+ - Replace any usage of `plain.utils.crypto.constant_time_compare` with `hmac.compare_digest` or `secrets.compare_digest`
14
+
15
+ ## [0.63.0](https://github.com/dropseed/plain/releases/plain@0.63.0) (2025-09-12)
16
+
17
+ ### What's changed
18
+
19
+ - Model manager attribute renamed from `objects` to `query` throughout codebase ([037a239](https://github.com/dropseed/plain/commit/037a239ef4))
20
+ - Simplified HTTPS redirect middleware by removing `HTTPS_REDIRECT_EXEMPT_PATHS` and `HTTPS_REDIRECT_HOST` settings ([d264cd3](https://github.com/dropseed/plain/commit/d264cd306b))
21
+ - Database backups are now created automatically during migrations when `DEBUG=True` unless explicitly disabled ([c802307](https://github.com/dropseed/plain/commit/c8023074e9))
22
+
23
+ ### Upgrade instructions
24
+
25
+ - Remove any `HTTPS_REDIRECT_EXEMPT_PATHS` and `HTTPS_REDIRECT_HOST` settings from your configuration - the HTTPS redirect middleware now performs a blanket redirect. For advanced redirect logic, write custom middleware.
26
+
3
27
  ## [0.62.1](https://github.com/dropseed/plain/releases/plain@0.62.1) (2025-09-09)
4
28
 
5
29
  ### What's changed
@@ -126,7 +150,7 @@
126
150
 
127
151
  - Update your URL patterns from `<int:pk>` to `<int:id>` in your URLconf
128
152
  - Update view code that accesses `self.url_kwargs["pk"]` to use `self.url_kwargs["id"]` instead
129
- - Replace any QuerySet filters using `pk` with `id` (e.g., `Model.objects.get(pk=1)` becomes `Model.objects.get(id=1)`)
153
+ - Replace any QuerySet filters using `pk` with `id` (e.g., `Model.query.get(pk=1)` becomes `Model.query.get(id=1)`)
130
154
 
131
155
  ## [0.54.1](https://github.com/dropseed/plain/releases/plain@0.54.1) (2025-07-20)
132
156
 
@@ -27,7 +27,7 @@ def clear_expired():
27
27
  """
28
28
  Delete sessions that have expired.
29
29
  """
30
- result = Session.objects.filter(expires_at__lt=timezone.now()).delete()
30
+ result = Session.query.filter(expires_at__lt=timezone.now()).delete()
31
31
  return f"{result[0]} expired sessions deleted"
32
32
  ```
33
33
 
@@ -64,7 +64,7 @@ def request(path, method, data, user_id, follow, content_type, headers):
64
64
 
65
65
  # Get the user
66
66
  try:
67
- user = User.objects.get(id=user_id)
67
+ user = User.query.get(id=user_id)
68
68
  client.force_login(user)
69
69
  click.secho(
70
70
  f"Authenticated as user {user_id}", fg="green", dim=True
@@ -1,3 +1,5 @@
1
+ import os
2
+
1
3
  import click
2
4
  from click.formatting import iter_rows, measure_table, term_len, wrap_text
3
5
 
@@ -59,3 +61,13 @@ class PlainHelpFormatter(click.HelpFormatter):
59
61
 
60
62
  class PlainContext(click.Context):
61
63
  formatter_class = PlainHelpFormatter
64
+
65
+ def __init__(self, *args, **kwargs):
66
+ super().__init__(*args, **kwargs)
67
+
68
+ # Force colors in CI environments
69
+ if any(
70
+ os.getenv(var)
71
+ for var in ["CI", "FORCE_COLOR", "GITHUB_ACTIONS", "GITLAB_CI"]
72
+ ) and not any(os.getenv(var) for var in ["NO_COLOR", "PYTEST_CURRENT_TEST"]):
73
+ self.color = True
@@ -153,7 +153,6 @@ def build_prompt(before_after: dict[str, tuple[str | None, str | None]]) -> str:
153
153
  " - Process ALL packages before testing or validation",
154
154
  " - After all packages are updated, run `uv run plain fix --unsafe-fixes` and `uv run plain pre-commit` to check results",
155
155
  " - DO NOT commit any changes",
156
- " - DO NOT run `plain migrate` with the `--no-backup` option",
157
156
  " - Keep code changes minimal and focused - avoid unnecessary comments",
158
157
  "",
159
158
  "3. **Available tools:**",
@@ -1,5 +1,3 @@
1
- import re
2
-
3
1
  from plain.http import ResponseRedirect
4
2
  from plain.runtime import settings
5
3
 
@@ -8,16 +6,12 @@ class HttpsRedirectMiddleware:
8
6
  def __init__(self, get_response):
9
7
  self.get_response = get_response
10
8
 
11
- # Settings for https (compile regexes once)
9
+ # Settings for HTTPS
12
10
  self.https_redirect_enabled = settings.HTTPS_REDIRECT_ENABLED
13
- self.https_redirect_host = settings.HTTPS_REDIRECT_HOST
14
- self.https_redirect_exempt = [
15
- re.compile(r) for r in settings.HTTPS_REDIRECT_EXEMPT_PATHS
16
- ]
17
11
 
18
12
  def __call__(self, request):
19
13
  """
20
- Rewrite the URL based on settings.APPEND_SLASH
14
+ Perform a blanket HTTP→HTTPS redirect when enabled.
21
15
  """
22
16
 
23
17
  if redirect_response := self.maybe_https_redirect(request):
@@ -26,15 +20,8 @@ class HttpsRedirectMiddleware:
26
20
  return self.get_response(request)
27
21
 
28
22
  def maybe_https_redirect(self, request):
29
- if (
30
- self.https_redirect_enabled
31
- and not request.is_https()
32
- and not any(
33
- pattern.search(request.path_info)
34
- for pattern in self.https_redirect_exempt
35
- )
36
- ):
37
- host = self.https_redirect_host or request.get_host()
23
+ if self.https_redirect_enabled and not request.is_https():
24
+ host = request.get_host()
38
25
  return ResponseRedirect(
39
26
  f"https://{host}{request.get_full_path()}", status_code=301
40
27
  )
@@ -37,17 +37,11 @@ DEFAULT_RESPONSE_HEADERS = {
37
37
  "X-Frame-Options": "DENY",
38
38
  }
39
39
 
40
- # Whether to redirect all non-HTTPS requests to HTTPS.
40
+ # Whether to redirect all non-HTTPS requests to HTTPS (blanket redirect).
41
+ # For anything more advanced (custom host, path exemptions, etc.), write
42
+ # your own middleware.
41
43
  HTTPS_REDIRECT_ENABLED = True
42
44
 
43
- # Regex patterns for paths that should be exempt from HTTPS redirect
44
- # Examples: [r"^/health$", r"/api/internal/.*", r"/dev/.*"]
45
- HTTPS_REDIRECT_EXEMPT_PATHS: list[str] = []
46
-
47
- # Custom host to redirect to for HTTPS. If None, uses the same host as the request.
48
- # Useful for redirecting to a different domain (e.g., "secure.example.com")
49
- HTTPS_REDIRECT_HOST = None
50
-
51
45
  # If your Plain app is behind a proxy that sets a header to specify secure
52
46
  # connections, AND that proxy ensures that user-submitted headers with the
53
47
  # same name are ignored (so that people can't spoof it), set this value to
@@ -35,12 +35,14 @@ These functions make use of all of them.
35
35
 
36
36
  import base64
37
37
  import datetime
38
+ import hmac
38
39
  import json
39
40
  import time
40
41
  import zlib
41
42
 
42
43
  from plain.runtime import settings
43
- from plain.utils.crypto import constant_time_compare, salted_hmac
44
+ from plain.utils.crypto import salted_hmac
45
+ from plain.utils.encoding import force_bytes
44
46
  from plain.utils.regex_helper import _lazy_re_compile
45
47
 
46
48
  _SEP_UNSAFE = _lazy_re_compile(r"^[A-z0-9-_=]*$")
@@ -196,7 +198,9 @@ class Signer:
196
198
  raise BadSignature(f'No "{self.sep}" found in value')
197
199
  value, sig = signed_value.rsplit(self.sep, 1)
198
200
  for key in [self.key, *self.fallback_keys]:
199
- if constant_time_compare(sig, self.signature(value, key)):
201
+ if hmac.compare_digest(
202
+ force_bytes(sig), force_bytes(self.signature(value, key))
203
+ ):
200
204
  return value
201
205
  raise BadSignature(f'Signature "{sig}" does not match')
202
206
 
@@ -24,7 +24,7 @@ def test_client_example():
24
24
  assert client.session["example"] == "value"
25
25
 
26
26
  # Logging in
27
- user = User.objects.first()
27
+ user = User.query.first()
28
28
  client.force_login(user)
29
29
  response = client.get("/protected/")
30
30
  assert response.status_code == 200
@@ -62,11 +62,6 @@ def get_random_string(length, allowed_chars=RANDOM_STRING_CHARS):
62
62
  return "".join(secrets.choice(allowed_chars) for i in range(length))
63
63
 
64
64
 
65
- def constant_time_compare(val1, val2):
66
- """Return True if the two strings are equal, False otherwise."""
67
- return secrets.compare_digest(force_bytes(val1), force_bytes(val2))
68
-
69
-
70
65
  def pbkdf2(password, salt, iterations, dklen=0, digest=None):
71
66
  """Return the hash of password using pbkdf2."""
72
67
  if digest is None:
@@ -179,7 +179,7 @@ class ExampleDetailView(DetailView):
179
179
  template_name = "detail.html"
180
180
 
181
181
  def get_object(self):
182
- return MyObjectClass.objects.get(
182
+ return MyObjectClass.query.get(
183
183
  id=self.url_kwargs["id"],
184
184
  user=self.request.user, # Limit access
185
185
  )
@@ -197,7 +197,7 @@ class ExampleUpdateView(UpdateView):
197
197
  success_url = "."
198
198
 
199
199
  def get_object(self):
200
- return MyObjectClass.objects.get(
200
+ return MyObjectClass.query.get(
201
201
  id=self.url_kwargs["id"],
202
202
  user=self.request.user, # Limit access
203
203
  )
@@ -211,7 +211,7 @@ class ExampleDeleteView(DeleteView):
211
211
  # Just POST to this view to delete the object.
212
212
 
213
213
  def get_object(self):
214
- return MyObjectClass.objects.get(
214
+ return MyObjectClass.query.get(
215
215
  id=self.url_kwargs["id"],
216
216
  user=self.request.user, # Limit access
217
217
  )
@@ -221,7 +221,7 @@ class ExampleListView(ListView):
221
221
  template_name = "list.html"
222
222
 
223
223
  def get_objects(self):
224
- return MyObjectClass.objects.filter(
224
+ return MyObjectClass.query.filter(
225
225
  user=self.request.user, # Limit access
226
226
  )
227
227
  ```
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "plain"
3
- version = "0.62.1"
3
+ version = "0.64.0"
4
4
  description = "A web framework for building products with Python."
5
5
  authors = [{name = "Dave Gaeddert", email = "dave.gaeddert@dropseed.dev"}]
6
6
  readme = "README.md"
@@ -18,6 +18,7 @@ plain = "plain.cli.core:cli"
18
18
  # Make this directly available without loading Plain,
19
19
  # for use in upgrade scripts.
20
20
  plain-changelog = "plain.cli.changelog:changelog"
21
+ plain-build = "plain.cli.build:build"
21
22
 
22
23
  [tool.uv]
23
24
  dev-dependencies = [
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
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