dbos 0.22.0a5__tar.gz → 0.22.0a7__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.

Potentially problematic release.


This version of dbos might be problematic. Click here for more details.

Files changed (90) hide show
  1. {dbos-0.22.0a5 → dbos-0.22.0a7}/PKG-INFO +1 -1
  2. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_admin_server.py +21 -0
  3. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_sys_db.py +3 -1
  4. {dbos-0.22.0a5 → dbos-0.22.0a7}/pyproject.toml +1 -1
  5. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_admin_server.py +22 -1
  6. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_dbos.py +13 -5
  7. {dbos-0.22.0a5 → dbos-0.22.0a7}/LICENSE +0 -0
  8. {dbos-0.22.0a5 → dbos-0.22.0a7}/README.md +0 -0
  9. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/__init__.py +0 -0
  10. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_app_db.py +0 -0
  11. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_classproperty.py +0 -0
  12. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_cloudutils/authentication.py +0 -0
  13. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_cloudutils/cloudutils.py +0 -0
  14. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_cloudutils/databases.py +0 -0
  15. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_context.py +0 -0
  16. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_core.py +0 -0
  17. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_croniter.py +0 -0
  18. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_db_wizard.py +0 -0
  19. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_dbos.py +0 -0
  20. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_dbos_config.py +0 -0
  21. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_error.py +0 -0
  22. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_fastapi.py +0 -0
  23. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_flask.py +0 -0
  24. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_kafka.py +0 -0
  25. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_kafka_message.py +0 -0
  26. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_logger.py +0 -0
  27. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/env.py +0 -0
  28. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/script.py.mako +0 -0
  29. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/04ca4f231047_workflow_queues_executor_id.py +0 -0
  30. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/50f3227f0b4b_fix_job_queue.py +0 -0
  31. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/5c361fc04708_added_system_tables.py +0 -0
  32. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/a3b18ad34abe_added_triggers.py +0 -0
  33. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/d76646551a6b_job_queue_limiter.py +0 -0
  34. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/d76646551a6c_workflow_queue.py +0 -0
  35. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_migrations/versions/eab0cc1d9a14_job_queue.py +0 -0
  36. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_outcome.py +0 -0
  37. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_queue.py +0 -0
  38. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_recovery.py +0 -0
  39. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_registrations.py +0 -0
  40. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_request.py +0 -0
  41. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_roles.py +0 -0
  42. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_scheduler.py +0 -0
  43. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_schemas/__init__.py +0 -0
  44. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_schemas/application_database.py +0 -0
  45. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_schemas/system_database.py +0 -0
  46. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_serialization.py +0 -0
  47. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/README.md +0 -0
  48. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/__package/__init__.py +0 -0
  49. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/__package/main.py +0 -0
  50. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/__package/schema.py +0 -0
  51. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/alembic.ini +0 -0
  52. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/dbos-config.yaml.dbos +0 -0
  53. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/migrations/env.py.dbos +0 -0
  54. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/migrations/script.py.mako +0 -0
  55. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/migrations/versions/2024_07_31_180642_init.py +0 -0
  56. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_templates/dbos-db-starter/start_postgres_docker.py +0 -0
  57. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_tracer.py +0 -0
  58. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/_workflow_commands.py +0 -0
  59. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/cli/_github_init.py +0 -0
  60. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/cli/_template_init.py +0 -0
  61. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/cli/cli.py +0 -0
  62. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/dbos-config.schema.json +0 -0
  63. {dbos-0.22.0a5 → dbos-0.22.0a7}/dbos/py.typed +0 -0
  64. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/__init__.py +0 -0
  65. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/atexit_no_ctor.py +0 -0
  66. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/atexit_no_launch.py +0 -0
  67. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/classdefs.py +0 -0
  68. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/conftest.py +0 -0
  69. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/more_classdefs.py +0 -0
  70. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/queuedworkflow.py +0 -0
  71. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_async.py +0 -0
  72. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_classdecorators.py +0 -0
  73. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_concurrency.py +0 -0
  74. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_config.py +0 -0
  75. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_croniter.py +0 -0
  76. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_failures.py +0 -0
  77. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_fastapi.py +0 -0
  78. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_fastapi_roles.py +0 -0
  79. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_flask.py +0 -0
  80. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_kafka.py +0 -0
  81. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_outcome.py +0 -0
  82. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_package.py +0 -0
  83. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_queue.py +0 -0
  84. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_scheduler.py +0 -0
  85. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_schema_migration.py +0 -0
  86. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_singleton.py +0 -0
  87. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_spans.py +0 -0
  88. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_sqlalchemy.py +0 -0
  89. {dbos-0.22.0a5 → dbos-0.22.0a7}/tests/test_workflow_cmds.py +0 -0
  90. {dbos-0.22.0a5 → dbos-0.22.0a7}/version/__init__.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: dbos
3
- Version: 0.22.0a5
3
+ Version: 0.22.0a7
4
4
  Summary: Ultra-lightweight durable execution in Python
5
5
  Author-Email: "DBOS, Inc." <contact@dbos.dev>
6
6
  License: MIT
@@ -16,6 +16,7 @@ if TYPE_CHECKING:
16
16
  _health_check_path = "/dbos-healthz"
17
17
  _workflow_recovery_path = "/dbos-workflow-recovery"
18
18
  _deactivate_path = "/deactivate"
19
+ _workflow_queues_metadata_path = "/dbos-workflow-queues-metadata"
19
20
  # /workflows/:workflow_id/cancel
20
21
  # /workflows/:workflow_id/resume
21
22
  # /workflows/:workflow_id/restart
@@ -64,6 +65,26 @@ class AdminRequestHandler(BaseHTTPRequestHandler):
64
65
  self.send_response(200)
65
66
  self._end_headers()
66
67
  self.wfile.write("deactivated".encode("utf-8"))
68
+ elif self.path == _workflow_queues_metadata_path:
69
+ queue_metadata_array = []
70
+ from ._dbos import _get_or_create_dbos_registry
71
+
72
+ registry = _get_or_create_dbos_registry()
73
+ for queue in registry.queue_info_map.values():
74
+ queue_metadata = {
75
+ "name": queue.name,
76
+ "concurrency": queue.concurrency,
77
+ "workerConcurrency": queue.worker_concurrency,
78
+ "rateLimit": queue.limiter,
79
+ }
80
+ # Remove keys with None values
81
+ queue_metadata = {
82
+ k: v for k, v in queue_metadata.items() if v is not None
83
+ }
84
+ queue_metadata_array.append(queue_metadata)
85
+ self.send_response(200)
86
+ self._end_headers()
87
+ self.wfile.write(json.dumps(queue_metadata_array).encode("utf-8"))
67
88
  else:
68
89
  self.send_response(404)
69
90
  self._end_headers()
@@ -22,8 +22,8 @@ import sqlalchemy as sa
22
22
  import sqlalchemy.dialects.postgresql as pg
23
23
  from alembic import command
24
24
  from alembic.config import Config
25
- from sqlalchemy import or_
26
25
  from sqlalchemy.exc import DBAPIError
26
+ from sqlalchemy.sql import func
27
27
 
28
28
  from . import _serialization
29
29
  from ._dbos_config import ConfigFile
@@ -307,6 +307,7 @@ class SystemDatabase:
307
307
  recovery_attempts=(
308
308
  SystemSchema.workflow_status.c.recovery_attempts + 1
309
309
  ),
310
+ updated_at=func.extract("epoch", func.now()) * 1000,
310
311
  ),
311
312
  )
312
313
  )
@@ -404,6 +405,7 @@ class SystemDatabase:
404
405
  status=status["status"],
405
406
  output=status["output"],
406
407
  error=status["error"],
408
+ updated_at=func.extract("epoch", func.now()) * 1000,
407
409
  ),
408
410
  )
409
411
  )
@@ -27,7 +27,7 @@ dependencies = [
27
27
  ]
28
28
  requires-python = ">=3.9"
29
29
  readme = "README.md"
30
- version = "0.22.0a5"
30
+ version = "0.22.0a7"
31
31
 
32
32
  [project.license]
33
33
  text = "MIT"
@@ -5,7 +5,7 @@ import uuid
5
5
  import requests
6
6
 
7
7
  # Public API
8
- from dbos import DBOS, ConfigFile, SetWorkflowID, _workflow_commands
8
+ from dbos import DBOS, ConfigFile, Queue, SetWorkflowID, _workflow_commands
9
9
 
10
10
 
11
11
  def test_admin_endpoints(dbos: DBOS) -> None:
@@ -23,6 +23,27 @@ def test_admin_endpoints(dbos: DBOS) -> None:
23
23
  assert response.status_code == 200
24
24
  assert response.json() == []
25
25
 
26
+ # Test GET /dbos-workflow-queues-metadata
27
+ Queue("q1")
28
+ Queue("q2", concurrency=1)
29
+ Queue("q3", concurrency=1, worker_concurrency=1)
30
+ Queue("q4", concurrency=1, worker_concurrency=1, limiter={"limit": 0, "period": 0})
31
+ response = requests.get(
32
+ "http://localhost:3001/dbos-workflow-queues-metadata", timeout=5
33
+ )
34
+ assert response.status_code == 200
35
+ assert response.json() == [
36
+ {"name": "q1"},
37
+ {"name": "q2", "concurrency": 1},
38
+ {"name": "q3", "concurrency": 1, "workerConcurrency": 1},
39
+ {
40
+ "name": "q4",
41
+ "concurrency": 1,
42
+ "workerConcurrency": 1,
43
+ "rateLimit": {"limit": 0, "period": 0},
44
+ },
45
+ ]
46
+
26
47
  # Test GET not found
27
48
  response = requests.get("http://localhost:3001/stuff", timeout=5)
28
49
  assert response.status_code == 404
@@ -71,14 +71,22 @@ def test_simple_workflow_attempts_counter(dbos: DBOS) -> None:
71
71
 
72
72
  wfuuid = str(uuid.uuid4())
73
73
  with dbos._sys_db.engine.connect() as c:
74
- stmt = sa.select(SystemSchema.workflow_status.c.recovery_attempts).where(
75
- SystemSchema.workflow_status.c.workflow_uuid == wfuuid
76
- )
74
+ stmt = sa.select(
75
+ SystemSchema.workflow_status.c.recovery_attempts,
76
+ SystemSchema.workflow_status.c.created_at,
77
+ SystemSchema.workflow_status.c.updated_at,
78
+ ).where(SystemSchema.workflow_status.c.workflow_uuid == wfuuid)
77
79
  for i in range(10):
78
80
  with SetWorkflowID(wfuuid):
79
81
  noop()
80
- result = c.execute(stmt).scalar()
81
- assert result == i + 1
82
+ result = c.execute(stmt).fetchone()
83
+ assert result is not None
84
+ recovery_attempts, created_at, updated_at = result
85
+ assert recovery_attempts == i + 1
86
+ if i == 0:
87
+ assert created_at == updated_at
88
+ else:
89
+ assert updated_at > created_at
82
90
 
83
91
 
84
92
  def test_child_workflow(dbos: DBOS) -> None:
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes