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.
- {plain-0.62.1 → plain-0.64.0}/.gitignore +0 -2
- {plain-0.62.1 → plain-0.64.0}/PKG-INFO +1 -1
- {plain-0.62.1 → plain-0.64.0}/plain/CHANGELOG.md +25 -1
- {plain-0.62.1 → plain-0.64.0}/plain/chores/README.md +1 -1
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/request.py +1 -1
- {plain-0.62.1 → plain-0.64.0}/plain/cli/formatting.py +12 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/upgrade.py +0 -1
- {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/https.py +4 -17
- {plain-0.62.1 → plain-0.64.0}/plain/runtime/global_settings.py +3 -9
- {plain-0.62.1 → plain-0.64.0}/plain/signing.py +6 -2
- {plain-0.62.1 → plain-0.64.0}/plain/test/README.md +1 -1
- {plain-0.62.1 → plain-0.64.0}/plain/utils/crypto.py +0 -5
- {plain-0.62.1 → plain-0.64.0}/plain/views/README.md +4 -4
- {plain-0.62.1 → plain-0.64.0}/pyproject.toml +2 -1
- {plain-0.62.1 → plain-0.64.0}/LICENSE +0 -0
- {plain-0.62.1 → plain-0.64.0}/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/AGENTS.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/__main__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/compile.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/finders.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/fingerprints.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/urls.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/assets/views.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/chores/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/chores/registry.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/docs.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/llmdocs.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/md.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/agent/prompt.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/build.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/changelog.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/chores.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/core.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/docs.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/install.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/output.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/preflight.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/print.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/registry.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/scaffold.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/settings.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/shell.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/startup.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/urls.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/cli/utils.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/csrf/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/csrf/middleware.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/csrf/views.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/debug.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/exceptions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/boundfield.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/exceptions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/fields.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/forms/forms.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/cookie.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/multipartparser.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/request.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/http/response.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/base.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/locks.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/move.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/temp.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/uploadedfile.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/uploadhandler.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/files/utils.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/base.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/exception.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/handlers/wsgi.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/headers.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/internal/middleware/slash.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/json.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/configure.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/debug.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/formatters.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/loggers.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/logs/utils.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/packages/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/packages/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/packages/config.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/packages/registry.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/paginator.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/files.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/messages.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/registry.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/security.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/preflight/urls.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/runtime/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/runtime/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/runtime/user_settings.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/runtime/utils.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/signals/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/signals/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/dispatcher.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/signals/dispatch/license.txt +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/AGENTS.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/core.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/environments.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/extensions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/filters.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/templates/jinja/globals.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/test/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/test/client.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/test/encoding.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/test/exceptions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/converters.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/exceptions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/patterns.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/resolvers.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/routers.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/urls/utils.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/README.md +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/cache.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/datastructures.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/dateparse.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/deconstruct.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/decorators.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/duration.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/encoding.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/functional.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/hashable.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/html.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/http.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/inspect.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/ipv6.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/itercompat.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/module_loading.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/regex_helper.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/safestring.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/text.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/timesince.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/timezone.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/utils/tree.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/validators.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/base.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/errors.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/exceptions.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/forms.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/objects.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/redirect.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/views/templates.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/plain/wsgi.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/.gitignore +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/app/.gitignore +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/app/settings.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/app/test/__init__.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/app/test/default_settings.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/app/urls.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/conftest.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/test_cli.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/test_csrf.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/test_logs.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/test_runtime.py +0 -0
- {plain-0.62.1 → plain-0.64.0}/tests/test_wsgi.py +0 -0
@@ -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.
|
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.
|
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.
|
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
|
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
|
-
|
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
|
-
|
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
|
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
|
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
|
|
@@ -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.
|
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.
|
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.
|
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.
|
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.
|
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
|
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
|