flagsmith-common 2.2.1__tar.gz → 2.2.3__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 (91) hide show
  1. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/PKG-INFO +1 -1
  2. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/pyproject.toml +1 -1
  3. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/main.py +8 -4
  4. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/utils.py +47 -1
  5. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/processor.py +16 -4
  6. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/LICENSE +0 -0
  7. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/README.md +0 -0
  8. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/__init__.py +0 -0
  9. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/__init__.py +0 -0
  10. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/app.py +0 -0
  11. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/cli/__init__.py +0 -0
  12. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/cli/healthcheck.py +0 -0
  13. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/logging.py +0 -0
  14. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/management/__init__.py +0 -0
  15. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/management/commands/__init__.py +0 -0
  16. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/management/commands/docgen.py +0 -0
  17. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/management/commands/start.py +0 -0
  18. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/management/commands/waitfordb.py +0 -0
  19. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/metrics.py +0 -0
  20. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/middleware.py +0 -0
  21. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/templates/docgen-metrics.md +0 -0
  22. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/urls.py +0 -0
  23. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/core/views.py +0 -0
  24. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/environments/permissions.py +0 -0
  25. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/__init__.py +0 -0
  26. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/multivariate/__init__.py +0 -0
  27. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/multivariate/serializers.py +0 -0
  28. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/serializers.py +0 -0
  29. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/versioning/__init__.py +0 -0
  30. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/features/versioning/serializers.py +0 -0
  31. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/__init__.py +0 -0
  32. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/conf.py +0 -0
  33. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/constants.py +0 -0
  34. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/logging.py +0 -0
  35. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/metrics.py +0 -0
  36. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/middleware.py +0 -0
  37. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/gunicorn/utils.py +0 -0
  38. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/migrations/__init__.py +0 -0
  39. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/migrations/helpers/__init__.py +0 -0
  40. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/migrations/helpers/postgres_helpers.py +0 -0
  41. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/organisations/permissions.py +0 -0
  42. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/projects/permissions.py +0 -0
  43. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/prometheus/__init__.py +0 -0
  44. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/prometheus/utils.py +0 -0
  45. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/py.typed +0 -0
  46. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/test_tools/__init__.py +0 -0
  47. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/test_tools/plugin.py +0 -0
  48. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/test_tools/types.py +0 -0
  49. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/test_tools/utils.py +0 -0
  50. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/common/types.py +0 -0
  51. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/__init__.py +0 -0
  52. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/admin.py +0 -0
  53. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/apps.py +0 -0
  54. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/decorators.py +0 -0
  55. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/exceptions.py +0 -0
  56. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/health.py +0 -0
  57. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/managers.py +0 -0
  58. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/metrics.py +0 -0
  59. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0001_initial.py +0 -0
  60. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0002_healthcheckmodel.py +0 -0
  61. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0003_add_completed_to_task.py +0 -0
  62. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0004_recreate_task_indexes.py +0 -0
  63. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0005_update_conditional_index_conditions.py +0 -0
  64. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0006_auto_20230221_0802.py +0 -0
  65. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0007_add_is_locked.py +0 -0
  66. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0008_add_get_task_to_process_function.py +0 -0
  67. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0009_add_recurring_task_run_first_run_at.py +0 -0
  68. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0010_task_priority.py +0 -0
  69. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0011_add_priority_to_get_tasks_to_process.py +0 -0
  70. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0012_add_locked_at_and_timeout.py +0 -0
  71. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/0013_add_last_picked_at.py +0 -0
  72. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/__init__.py +0 -0
  73. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/0008_get_recurring_tasks_to_process.sql +0 -0
  74. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/0008_get_tasks_to_process.sql +0 -0
  75. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/0011_get_tasks_to_process.sql +0 -0
  76. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/0012_get_recurringtasks_to_process.sql +0 -0
  77. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/0013_get_recurringtasks_to_process.sql +0 -0
  78. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/migrations/sql/__init__.py +0 -0
  79. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/models.py +0 -0
  80. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/monitoring.py +0 -0
  81. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/py.typed +0 -0
  82. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/routers.py +0 -0
  83. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/serializers.py +0 -0
  84. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/task_registry.py +0 -0
  85. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/task_run_method.py +0 -0
  86. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/tasks.py +0 -0
  87. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/threads.py +0 -0
  88. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/types.py +0 -0
  89. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/urls.py +0 -0
  90. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/utils.py +0 -0
  91. {flagsmith_common-2.2.1 → flagsmith_common-2.2.3}/src/task_processor/views.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: flagsmith-common
3
- Version: 2.2.1
3
+ Version: 2.2.3
4
4
  Summary: Flagsmith's common library
5
5
  License-Expression: BSD-3-Clause
6
6
  License-File: LICENSE
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "flagsmith-common"
3
- version = "2.2.1"
3
+ version = "2.2.3"
4
4
  description = "Flagsmith's common library"
5
5
  requires-python = ">=3.11,<4.0"
6
6
  dependencies = [
@@ -2,14 +2,15 @@ import contextlib
2
2
  import logging
3
3
  import os
4
4
  import sys
5
- import tempfile
6
5
  import typing
7
6
 
8
7
  from django.core.management import (
9
8
  execute_from_command_line as django_execute_from_command_line,
10
9
  )
10
+ from environs import Env
11
11
 
12
12
  from common.core.cli import healthcheck
13
+ from common.core.utils import TemporaryDirectory
13
14
 
14
15
  logger = logging.getLogger(__name__)
15
16
 
@@ -30,6 +31,7 @@ def ensure_cli_env() -> typing.Generator[None, None, None]:
30
31
  main()
31
32
  ```
32
33
  """
34
+ env = Env()
33
35
  ctx = contextlib.ExitStack()
34
36
 
35
37
  # TODO @khvn26 Move logging setup to here
@@ -43,9 +45,11 @@ def ensure_cli_env() -> typing.Generator[None, None, None]:
43
45
  os.environ.setdefault("DJANGO_SETTINGS_MODULE", "settings.dev")
44
46
 
45
47
  # Set up Prometheus' multiprocess mode
46
- if "PROMETHEUS_MULTIPROC_DIR" not in os.environ:
47
- prometheus_multiproc_dir_name = tempfile.mkdtemp()
48
-
48
+ if not env.str("PROMETHEUS_MULTIPROC_DIR", ""):
49
+ delete = not env.bool("PROMETHEUS_MULTIPROC_DIR_KEEP", False)
50
+ prometheus_multiproc_dir_name = ctx.enter_context(
51
+ TemporaryDirectory(delete=delete)
52
+ )
49
53
  logger.info(
50
54
  "Created %s for Prometheus multi-process mode",
51
55
  prometheus_multiproc_dir_name,
@@ -2,9 +2,18 @@ import json
2
2
  import logging
3
3
  import pathlib
4
4
  import random
5
+ import sys
6
+ import tempfile
5
7
  from functools import lru_cache
6
8
  from itertools import cycle
7
- from typing import Iterator, Literal, NotRequired, TypedDict, TypeVar, get_args
9
+ from typing import (
10
+ Iterator,
11
+ Literal,
12
+ NotRequired,
13
+ TypedDict,
14
+ TypeVar,
15
+ get_args,
16
+ )
8
17
 
9
18
  from django.conf import settings
10
19
  from django.contrib.auth import get_user_model
@@ -186,3 +195,40 @@ def using_database_replica(
186
195
  return manager
187
196
 
188
197
  return manager.db_manager(chosen_replica)
198
+
199
+
200
+ if sys.version_info >= (3, 12):
201
+ # Already has the desired behavior; re-export for uniform imports.
202
+ TemporaryDirectory = tempfile.TemporaryDirectory
203
+ else:
204
+ import contextlib
205
+ from typing import ContextManager, Generator
206
+
207
+ def TemporaryDirectory(
208
+ suffix: str | None = None,
209
+ prefix: str | None = None,
210
+ dir: str | None = None,
211
+ *,
212
+ delete: bool = True,
213
+ ) -> ContextManager[str]:
214
+ """
215
+ Create a temporary directory with optional cleanup control.
216
+
217
+ This wrapper exists because Python 3.12 changed TemporaryDirectory's behavior
218
+ by adding a 'delete' parameter, which doesn't exist in Python 3.11. This
219
+ function provides a consistent API across both versions.
220
+
221
+ When delete=True, uses the stdlib's TemporaryDirectory (auto-cleanup).
222
+ When delete=False, creates a directory with mkdtemp that persists after
223
+ the context manager exits, matching Python 3.12's delete=False behavior.
224
+
225
+ See https://docs.python.org/3.12/library/tempfile.html#tempfile.TemporaryDirectory for usage details.
226
+ """
227
+ if delete:
228
+ return tempfile.TemporaryDirectory(suffix, prefix, dir)
229
+
230
+ @contextlib.contextmanager
231
+ def _tmpdir() -> Generator[str, None, None]:
232
+ yield tempfile.mkdtemp(suffix, prefix, dir)
233
+
234
+ return _tmpdir()
@@ -128,12 +128,17 @@ def _run_task(
128
128
  )
129
129
  task_run: AnyTaskRun = task.task_runs.model(started_at=timezone.now(), task=task) # type: ignore[attr-defined]
130
130
  result: str
131
+ executor = None
131
132
 
132
133
  try:
133
- with ThreadPoolExecutor(max_workers=1) as executor:
134
- future = executor.submit(task.run)
135
- timeout = task.timeout.total_seconds() if task.timeout else None
136
- future.result(timeout=timeout) # Wait for completion or timeout
134
+ # Use explicit executor management to avoid blocking on shutdown
135
+ # when tasks timeout but continue running in worker threads.
136
+ # The default context manager behavior (wait=True) would block
137
+ # the TaskRunner thread indefinitely waiting for stuck workers.
138
+ executor = ThreadPoolExecutor(max_workers=1)
139
+ future = executor.submit(task.run)
140
+ timeout = task.timeout.total_seconds() if task.timeout else None
141
+ future.result(timeout=timeout) # Wait for completion or timeout
137
142
 
138
143
  task_run.result = result = TaskResult.SUCCESS.value
139
144
  task_run.finished_at = timezone.now()
@@ -176,6 +181,13 @@ def _run_task(
176
181
  delay_until,
177
182
  )
178
183
 
184
+ finally:
185
+ # Always shutdown the executor without waiting for worker threads.
186
+ # This prevents the TaskRunner thread from blocking indefinitely
187
+ # when a task times out but continues running in a worker thread.
188
+ if executor is not None:
189
+ executor.shutdown(wait=False)
190
+
179
191
  labels = {
180
192
  "task_identifier": task_identifier,
181
193
  "task_type": registered_task.task_type.value.lower(),