karrio-server 2025.5rc34__py3-none-any.whl → 2025.5rc36__py3-none-any.whl
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.
- karrio/server/VERSION +1 -1
- karrio/server/settings/apm.py +9 -2
- karrio/server/settings/base.py +13 -0
- karrio/server/settings/cache.py +21 -5
- karrio/server/settings/constance.py +29 -5
- karrio/server/settings/workers.py +110 -109
- karrio/server/static/karrio/carriers/teleship_icon.svg +3 -0
- karrio/server/static/karrio/js/karrio.js +358 -260
- karrio/server/static/karrio/js/karrio.js.map +1 -1
- {karrio_server-2025.5rc34.dist-info → karrio_server-2025.5rc36.dist-info}/METADATA +1 -1
- {karrio_server-2025.5rc34.dist-info → karrio_server-2025.5rc36.dist-info}/RECORD +14 -13
- {karrio_server-2025.5rc34.dist-info → karrio_server-2025.5rc36.dist-info}/WHEEL +0 -0
- {karrio_server-2025.5rc34.dist-info → karrio_server-2025.5rc36.dist-info}/entry_points.txt +0 -0
- {karrio_server-2025.5rc34.dist-info → karrio_server-2025.5rc36.dist-info}/top_level.txt +0 -0
karrio/server/VERSION
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
2025.
|
|
1
|
+
2025.5rc36
|
karrio/server/settings/apm.py
CHANGED
|
@@ -49,12 +49,19 @@ if SENTRY_DSN:
|
|
|
49
49
|
dsn=SENTRY_DSN,
|
|
50
50
|
integrations=integrations,
|
|
51
51
|
# Set traces_sample_rate to 1.0 to capture 100%
|
|
52
|
-
# of transactions for
|
|
53
|
-
# We recommend adjusting this value in production,
|
|
52
|
+
# of transactions for tracing.
|
|
54
53
|
traces_sample_rate=1.0,
|
|
54
|
+
# Set profile_session_sample_rate to 1.0 to profile 100%
|
|
55
|
+
# of profile sessions.
|
|
56
|
+
profile_session_sample_rate=1.0,
|
|
57
|
+
# Set profile_lifecycle to "trace" to automatically
|
|
58
|
+
# run the profiler on when there is an active transaction
|
|
59
|
+
profile_lifecycle="trace",
|
|
55
60
|
# If you wish to associate users to errors (assuming you are using
|
|
56
61
|
# django.contrib.auth) you may enable sending PII data.
|
|
57
62
|
send_default_pii=True,
|
|
63
|
+
# Enable sending logs to Sentry
|
|
64
|
+
enable_logs=True,
|
|
58
65
|
)
|
|
59
66
|
|
|
60
67
|
|
karrio/server/settings/base.py
CHANGED
|
@@ -67,6 +67,7 @@ CORS_ALLOW_HEADERS = list(default_headers) + [
|
|
|
67
67
|
# HTTPS configuration
|
|
68
68
|
if USE_HTTPS is True:
|
|
69
69
|
global SECURE_SSL_REDIRECT
|
|
70
|
+
global SECURE_REDIRECT_EXEMPT
|
|
70
71
|
global SECURE_PROXY_SSL_HEADER
|
|
71
72
|
global SESSION_COOKIE_SECURE
|
|
72
73
|
global SECURE_HSTS_SECONDS
|
|
@@ -75,6 +76,8 @@ if USE_HTTPS is True:
|
|
|
75
76
|
global SECURE_HSTS_PRELOAD
|
|
76
77
|
|
|
77
78
|
SECURE_SSL_REDIRECT = True
|
|
79
|
+
# Exempt health check endpoint from HTTPS redirect for Kubernetes probes
|
|
80
|
+
SECURE_REDIRECT_EXEMPT = [r'^status/$']
|
|
78
81
|
SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
|
|
79
82
|
SESSION_COOKIE_SECURE = True
|
|
80
83
|
SECURE_HSTS_SECONDS = 1
|
|
@@ -385,6 +388,16 @@ AUTH_PASSWORD_VALIDATORS = [
|
|
|
385
388
|
AUTH_USER_MODEL = "user.User"
|
|
386
389
|
|
|
387
390
|
|
|
391
|
+
# Session configuration
|
|
392
|
+
# Use a unique session cookie name to prevent conflicts with other applications
|
|
393
|
+
# (e.g., ORY Kratos sessions in the shipping-app frontend)
|
|
394
|
+
SESSION_COOKIE_NAME = config("SESSION_COOKIE_NAME", default="karrio_sessionid")
|
|
395
|
+
SESSION_COOKIE_PATH = config("SESSION_COOKIE_PATH", default="/")
|
|
396
|
+
SESSION_COOKIE_HTTPONLY = True
|
|
397
|
+
SESSION_COOKIE_SAMESITE = config("SESSION_COOKIE_SAMESITE", default="Lax")
|
|
398
|
+
# SESSION_COOKIE_DOMAIN is intentionally not set to allow per-host cookies
|
|
399
|
+
|
|
400
|
+
|
|
388
401
|
# Internationalization
|
|
389
402
|
# https://docs.djangoproject.com/en/3.0/topics/i18n/
|
|
390
403
|
|
karrio/server/settings/cache.py
CHANGED
|
@@ -14,9 +14,9 @@ DETACHED_WORKER = config("DETACHED_WORKER", default=False, cast=bool)
|
|
|
14
14
|
# Detect if running as a worker process (via run_huey command)
|
|
15
15
|
IS_WORKER_PROCESS = any("run_huey" in arg for arg in sys.argv)
|
|
16
16
|
|
|
17
|
-
# Skip default Redis cache configuration
|
|
18
|
-
#
|
|
19
|
-
SKIP_DEFAULT_CACHE =
|
|
17
|
+
# Skip default Redis cache configuration only for worker processes
|
|
18
|
+
# API servers should use Redis cache even when DETACHED_WORKER is True
|
|
19
|
+
SKIP_DEFAULT_CACHE = IS_WORKER_PROCESS
|
|
20
20
|
|
|
21
21
|
# Redis configuration - REDIS_URL takes precedence and supersedes granular env vars
|
|
22
22
|
REDIS_URL = config("REDIS_URL", default=None)
|
|
@@ -60,8 +60,16 @@ else:
|
|
|
60
60
|
|
|
61
61
|
# Configure Django cache if Redis is available and not in worker mode
|
|
62
62
|
if REDIS_HOST is not None and not SKIP_DEFAULT_CACHE:
|
|
63
|
-
|
|
64
|
-
|
|
63
|
+
# Configure connection pool with max_connections to prevent exhaustion
|
|
64
|
+
# Default: 50 connections per process (2 Gunicorn workers = 100 total)
|
|
65
|
+
# Azure Redis Basic: 256 max connections total
|
|
66
|
+
REDIS_CACHE_MAX_CONNECTIONS = config(
|
|
67
|
+
"REDIS_CACHE_MAX_CONNECTIONS", default=50, cast=int
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
pool_kwargs = {"max_connections": REDIS_CACHE_MAX_CONNECTIONS}
|
|
71
|
+
if REDIS_SSL:
|
|
72
|
+
pool_kwargs["ssl_cert_reqs"] = None
|
|
65
73
|
|
|
66
74
|
CACHES = {
|
|
67
75
|
"default": {
|
|
@@ -78,6 +86,14 @@ if REDIS_HOST is not None and not SKIP_DEFAULT_CACHE:
|
|
|
78
86
|
"KEY_PREFIX": REDIS_PREFIX,
|
|
79
87
|
}
|
|
80
88
|
}
|
|
89
|
+
|
|
90
|
+
# Django cache health check uses the cache backend directly
|
|
91
|
+
# Only add Redis health check if REDIS_URL environment variable is set
|
|
92
|
+
# When using granular params, the cache check is sufficient
|
|
93
|
+
if config("REDIS_URL", default=None) is not None:
|
|
94
|
+
HEALTH_CHECK_APPS += ["health_check.contrib.redis"]
|
|
95
|
+
INSTALLED_APPS += ["health_check.contrib.redis"]
|
|
96
|
+
|
|
81
97
|
print(f"Redis cache connection initialized")
|
|
82
98
|
elif SKIP_DEFAULT_CACHE:
|
|
83
99
|
print(
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
"""
|
|
1
|
+
"""Dynamic configuration editable on runtime powered by django-constance."""
|
|
2
2
|
|
|
3
3
|
from decouple import config
|
|
4
4
|
import karrio.references as ref
|
|
@@ -22,6 +22,7 @@ GOOGLE_CLOUD_API_KEY = config("GOOGLE_CLOUD_API_KEY", default="")
|
|
|
22
22
|
CANADAPOST_ADDRESS_COMPLETE_API_KEY = config(
|
|
23
23
|
"CANADAPOST_ADDRESS_COMPLETE_API_KEY", default=""
|
|
24
24
|
)
|
|
25
|
+
|
|
25
26
|
# data retention env in days
|
|
26
27
|
ORDER_DATA_RETENTION = config("ORDER_DATA_RETENTION", default=183, cast=int)
|
|
27
28
|
TRACKER_DATA_RETENTION = config("TRACKER_DATA_RETENTION", default=183, cast=int)
|
|
@@ -29,7 +30,9 @@ SHIPMENT_DATA_RETENTION = config("SHIPMENT_DATA_RETENTION", default=183, cast=in
|
|
|
29
30
|
API_LOGS_DATA_RETENTION = config("API_LOGS_DATA_RETENTION", default=92, cast=int)
|
|
30
31
|
|
|
31
32
|
# registry config
|
|
32
|
-
ENABLE_ALL_PLUGINS_BY_DEFAULT = config(
|
|
33
|
+
ENABLE_ALL_PLUGINS_BY_DEFAULT = config(
|
|
34
|
+
"ENABLE_ALL_PLUGINS_BY_DEFAULT", default=True if base.DEBUG else False, cast=bool
|
|
35
|
+
)
|
|
33
36
|
|
|
34
37
|
# Create feature flags config only for modules that exist
|
|
35
38
|
FEATURE_FLAGS_CONFIG = {
|
|
@@ -161,8 +164,21 @@ PLUGIN_REGISTRY = {
|
|
|
161
164
|
config(f"{ext.upper()}_ENABLED", default=True, cast=bool),
|
|
162
165
|
f"{metadata.get('label')} plugin",
|
|
163
166
|
bool,
|
|
164
|
-
)
|
|
165
|
-
|
|
167
|
+
)
|
|
168
|
+
for ext, metadata in ref.PLUGIN_METADATA.items()
|
|
169
|
+
},
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
# Collect plugin system configs from ref.SYSTEM_CONFIGS
|
|
173
|
+
# Format: Dict[str, Tuple[default_value, description, type]]
|
|
174
|
+
PLUGIN_SYSTEM_CONFIG = {
|
|
175
|
+
key: (config(key, default=default_value, cast=value_type), description, value_type)
|
|
176
|
+
for key, (default_value, description, value_type) in ref.SYSTEM_CONFIGS.items()
|
|
177
|
+
}
|
|
178
|
+
PLUGIN_SYSTEM_CONFIG_FIELDSETS = {
|
|
179
|
+
f"{metadata.get('label')} Config": tuple(metadata.get("system_config", {}).keys())
|
|
180
|
+
for _, metadata in ref.PLUGIN_METADATA.items()
|
|
181
|
+
if metadata.get("system_config")
|
|
166
182
|
}
|
|
167
183
|
|
|
168
184
|
|
|
@@ -218,6 +234,7 @@ CONSTANCE_CONFIG = {
|
|
|
218
234
|
),
|
|
219
235
|
**{k: v for k, v in FEATURE_FLAGS_CONFIG.items() if v is not None},
|
|
220
236
|
**PLUGIN_REGISTRY,
|
|
237
|
+
**PLUGIN_SYSTEM_CONFIG,
|
|
221
238
|
}
|
|
222
239
|
|
|
223
240
|
CONSTANCE_CONFIG_FIELDSETS = {
|
|
@@ -241,5 +258,12 @@ CONSTANCE_CONFIG_FIELDSETS = {
|
|
|
241
258
|
),
|
|
242
259
|
"Feature Flags": tuple(FEATURE_FLAGS_FIELDSET),
|
|
243
260
|
"Registry Config": ("ENABLE_ALL_PLUGINS_BY_DEFAULT",),
|
|
244
|
-
"Registry Plugins": tuple(
|
|
261
|
+
"Registry Plugins": tuple(
|
|
262
|
+
[
|
|
263
|
+
k
|
|
264
|
+
for k in PLUGIN_REGISTRY.keys()
|
|
265
|
+
if not k in ("ENABLE_ALL_PLUGINS_BY_DEFAULT",)
|
|
266
|
+
]
|
|
267
|
+
),
|
|
268
|
+
**PLUGIN_SYSTEM_CONFIG_FIELDSETS,
|
|
245
269
|
}
|
|
@@ -22,117 +22,118 @@ WORKER_IMMEDIATE_MODE = decouple.config(
|
|
|
22
22
|
)
|
|
23
23
|
|
|
24
24
|
# Detect if running as a worker process (via run_huey command)
|
|
25
|
-
# Workers always need Huey configured regardless of DETACHED_WORKER setting
|
|
26
25
|
IS_WORKER_PROCESS = any("run_huey" in arg for arg in sys.argv)
|
|
27
26
|
|
|
28
|
-
#
|
|
29
|
-
#
|
|
30
|
-
#
|
|
31
|
-
#
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
# Parse REDIS_URL or construct from individual parameters
|
|
41
|
-
if REDIS_URL is not None:
|
|
42
|
-
from urllib.parse import urlparse
|
|
43
|
-
|
|
44
|
-
parsed = urlparse(REDIS_URL)
|
|
45
|
-
|
|
46
|
-
# Extract values from REDIS_URL (these supersede granular env vars)
|
|
47
|
-
REDIS_HOST = parsed.hostname
|
|
48
|
-
REDIS_PORT = parsed.port or 6379
|
|
49
|
-
REDIS_USERNAME = parsed.username or "default"
|
|
50
|
-
REDIS_PASSWORD = parsed.password
|
|
51
|
-
|
|
52
|
-
# Determine SSL from URL scheme (rediss:// means SSL is enabled)
|
|
53
|
-
REDIS_SCHEME = (
|
|
54
|
-
parsed.scheme if parsed.scheme in ("redis", "rediss") else "redis"
|
|
55
|
-
)
|
|
56
|
-
REDIS_SSL = REDIS_SCHEME == "rediss"
|
|
57
|
-
|
|
58
|
-
else:
|
|
59
|
-
# Fall back to individual parameters
|
|
60
|
-
REDIS_HOST = decouple.config("REDIS_HOST", default=None)
|
|
61
|
-
REDIS_PORT = decouple.config("REDIS_PORT", default=None)
|
|
62
|
-
REDIS_PASSWORD = decouple.config("REDIS_PASSWORD", default=None)
|
|
63
|
-
REDIS_USERNAME = decouple.config("REDIS_USERNAME", default="default")
|
|
64
|
-
REDIS_SSL = decouple.config("REDIS_SSL", default=False, cast=bool)
|
|
65
|
-
|
|
66
|
-
# Configure HUEY based on available Redis configuration
|
|
67
|
-
if REDIS_HOST is not None:
|
|
68
|
-
# Calculate max connections based on environment
|
|
69
|
-
# Each worker replica needs: (workers_per_replica + 1 scheduler) connections
|
|
70
|
-
# Formula: (worker_replicas * (threads_per_worker + 1)) + api_connections + buffer
|
|
71
|
-
# Example: 100 connections = (5 replicas * (8 workers + 1 scheduler)) + 40 API + 15 buffer
|
|
72
|
-
REDIS_MAX_CONNECTIONS = decouple.config(
|
|
73
|
-
"REDIS_MAX_CONNECTIONS", default=100, cast=int
|
|
74
|
-
)
|
|
75
|
-
|
|
76
|
-
# Connection pool configuration with timeouts
|
|
77
|
-
# Use BlockingConnectionPool to wait for available connections instead of failing immediately
|
|
78
|
-
pool_kwargs = {
|
|
79
|
-
"host": REDIS_HOST,
|
|
80
|
-
"port": REDIS_PORT,
|
|
81
|
-
"max_connections": REDIS_MAX_CONNECTIONS,
|
|
82
|
-
"timeout": 20, # Wait up to 20 seconds for an available connection
|
|
83
|
-
# Timeout settings to prevent hung connections
|
|
84
|
-
"socket_timeout": 10, # Command execution timeout (seconds)
|
|
85
|
-
"socket_connect_timeout": 10, # Connection establishment timeout (seconds)
|
|
86
|
-
# Keep connections alive to prevent closure by firewalls/load balancers
|
|
87
|
-
"socket_keepalive": True,
|
|
88
|
-
# Retry on transient failures
|
|
89
|
-
"retry_on_timeout": True,
|
|
90
|
-
}
|
|
27
|
+
# Always configure Huey for both API servers and workers
|
|
28
|
+
# API servers need to enqueue tasks even when DETACHED_WORKER=True
|
|
29
|
+
# Only the worker pods will consume tasks
|
|
30
|
+
# Redis configuration - REDIS_URL takes precedence and supersedes granular env vars
|
|
31
|
+
REDIS_URL = decouple.config("REDIS_URL", default=None)
|
|
32
|
+
|
|
33
|
+
# Parse REDIS_URL or construct from individual parameters
|
|
34
|
+
if REDIS_URL is not None:
|
|
35
|
+
from urllib.parse import urlparse
|
|
36
|
+
|
|
37
|
+
parsed = urlparse(REDIS_URL)
|
|
91
38
|
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
settings
|
|
130
|
-
|
|
131
|
-
|
|
39
|
+
# Extract values from REDIS_URL (these supersede granular env vars)
|
|
40
|
+
REDIS_HOST = parsed.hostname
|
|
41
|
+
REDIS_PORT = parsed.port or 6379
|
|
42
|
+
REDIS_USERNAME = parsed.username or "default"
|
|
43
|
+
REDIS_PASSWORD = parsed.password
|
|
44
|
+
|
|
45
|
+
# Determine SSL from URL scheme (rediss:// means SSL is enabled)
|
|
46
|
+
REDIS_SCHEME = (
|
|
47
|
+
parsed.scheme if parsed.scheme in ("redis", "rediss") else "redis"
|
|
48
|
+
)
|
|
49
|
+
REDIS_SSL = REDIS_SCHEME == "rediss"
|
|
50
|
+
|
|
51
|
+
else:
|
|
52
|
+
# Fall back to individual parameters
|
|
53
|
+
REDIS_HOST = decouple.config("REDIS_HOST", default=None)
|
|
54
|
+
REDIS_PORT = decouple.config("REDIS_PORT", default=None)
|
|
55
|
+
REDIS_PASSWORD = decouple.config("REDIS_PASSWORD", default=None)
|
|
56
|
+
REDIS_USERNAME = decouple.config("REDIS_USERNAME", default="default")
|
|
57
|
+
REDIS_SSL = decouple.config("REDIS_SSL", default=False, cast=bool)
|
|
58
|
+
|
|
59
|
+
# Configure HUEY based on available Redis configuration
|
|
60
|
+
if REDIS_HOST is not None:
|
|
61
|
+
# Calculate max connections based on environment
|
|
62
|
+
# Each worker replica needs: (workers_per_replica + 1 scheduler) connections
|
|
63
|
+
# Formula: (worker_replicas * (threads_per_worker + 1)) + api_connections + buffer
|
|
64
|
+
# Example: 100 connections = (5 replicas * (8 workers + 1 scheduler)) + 40 API + 15 buffer
|
|
65
|
+
REDIS_MAX_CONNECTIONS = decouple.config(
|
|
66
|
+
"REDIS_MAX_CONNECTIONS", default=100, cast=int
|
|
67
|
+
)
|
|
68
|
+
|
|
69
|
+
# Connection pool configuration with timeouts
|
|
70
|
+
# Use BlockingConnectionPool to wait for available connections instead of failing immediately
|
|
71
|
+
pool_kwargs = {
|
|
72
|
+
"host": REDIS_HOST,
|
|
73
|
+
"port": REDIS_PORT,
|
|
74
|
+
"max_connections": REDIS_MAX_CONNECTIONS,
|
|
75
|
+
"timeout": 20, # Wait up to 20 seconds for an available connection
|
|
76
|
+
# Timeout settings to prevent hung connections
|
|
77
|
+
"socket_timeout": 10, # Command execution timeout (seconds)
|
|
78
|
+
"socket_connect_timeout": 10, # Connection establishment timeout (seconds)
|
|
79
|
+
# Keep connections alive to prevent closure by firewalls/load balancers
|
|
80
|
+
"socket_keepalive": True,
|
|
81
|
+
# Retry on transient failures
|
|
82
|
+
"retry_on_timeout": True,
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
# Add TCP keepalive options if available (Linux/Unix only)
|
|
86
|
+
try:
|
|
87
|
+
pool_kwargs["socket_keepalive_options"] = {
|
|
88
|
+
socket.TCP_KEEPIDLE: 60, # Start keepalive after 60s idle
|
|
89
|
+
socket.TCP_KEEPINTVL: 10, # Keepalive interval
|
|
90
|
+
socket.TCP_KEEPCNT: 3, # Keepalive probes before timeout
|
|
132
91
|
}
|
|
92
|
+
except AttributeError:
|
|
93
|
+
# TCP keepalive constants not available on this platform
|
|
94
|
+
pass
|
|
95
|
+
|
|
96
|
+
# Add authentication if provided
|
|
97
|
+
if REDIS_PASSWORD:
|
|
98
|
+
pool_kwargs["password"] = REDIS_PASSWORD
|
|
99
|
+
if REDIS_USERNAME:
|
|
100
|
+
pool_kwargs["username"] = REDIS_USERNAME
|
|
101
|
+
|
|
102
|
+
# Add SSL/TLS configuration if enabled
|
|
103
|
+
if REDIS_SSL:
|
|
104
|
+
# Use SSLConnection class for SSL/TLS connections
|
|
105
|
+
pool_kwargs["connection_class"] = redis.SSLConnection
|
|
106
|
+
pool_kwargs["ssl_cert_reqs"] = None # For Azure Redis compatibility
|
|
107
|
+
|
|
108
|
+
# Use BlockingConnectionPool to wait for connections instead of raising errors immediately
|
|
109
|
+
pool = redis.BlockingConnectionPool(**pool_kwargs)
|
|
110
|
+
|
|
111
|
+
HUEY = huey.RedisHuey(
|
|
112
|
+
"default",
|
|
113
|
+
connection_pool=pool,
|
|
114
|
+
**({"immediate": WORKER_IMMEDIATE_MODE} if WORKER_IMMEDIATE_MODE else {}),
|
|
115
|
+
)
|
|
133
116
|
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
117
|
+
else:
|
|
118
|
+
# No Redis configured, use SQLite
|
|
119
|
+
WORKER_DB_DIR = decouple.config("WORKER_DB_DIR", default=settings.WORK_DIR)
|
|
120
|
+
WORKER_DB_FILE_NAME = os.path.join(WORKER_DB_DIR, "tasks.sqlite3")
|
|
121
|
+
|
|
122
|
+
settings.DATABASES["workers"] = {
|
|
123
|
+
"NAME": WORKER_DB_FILE_NAME,
|
|
124
|
+
"ENGINE": "django.db.backends.sqlite3",
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
# SQLite-specific Huey configuration
|
|
128
|
+
# WAL mode (Write-Ahead Logging) enables better concurrency for multiple workers
|
|
129
|
+
# Increased timeout helps handle lock contention under concurrent access
|
|
130
|
+
HUEY = huey.SqliteHuey(
|
|
131
|
+
name="default",
|
|
132
|
+
filename=WORKER_DB_FILE_NAME,
|
|
133
|
+
# Storage-specific kwargs for better concurrent access handling
|
|
134
|
+
journal_mode="wal", # Enable Write-Ahead Logging for better concurrency
|
|
135
|
+
timeout=30, # Increase timeout to 30s to handle lock contention with multiple workers
|
|
136
|
+
cache_mb=16, # Increase cache size for better performance (default: 8MB)
|
|
137
|
+
fsync=False, # Disable forced fsync for better performance (default: False)
|
|
138
|
+
**({"immediate": WORKER_IMMEDIATE_MODE} if WORKER_IMMEDIATE_MODE else {}),
|
|
139
|
+
)
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="1200" height="1200" xml:space="preserve" version="1.1" viewBox="0 0 1200 1200">
|
|
2
|
+
<image width="1200" height="1200" xlink:href=""/>
|
|
3
|
+
</svg>
|