UncountablePythonSDK 0.0.115__py3-none-any.whl → 0.0.142.dev0__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.

Potentially problematic release.


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

Files changed (119) hide show
  1. docs/conf.py +52 -5
  2. docs/index.md +107 -4
  3. docs/integration_examples/create_ingredient.md +43 -0
  4. docs/integration_examples/create_output.md +56 -0
  5. docs/integration_examples/index.md +6 -0
  6. docs/justfile +1 -1
  7. docs/requirements.txt +3 -2
  8. examples/basic_auth.py +7 -0
  9. examples/integration-server/jobs/materials_auto/example_cron.py +3 -0
  10. examples/integration-server/jobs/materials_auto/example_http.py +19 -7
  11. examples/integration-server/jobs/materials_auto/example_instrument.py +100 -0
  12. examples/integration-server/jobs/materials_auto/example_parse.py +140 -0
  13. examples/integration-server/jobs/materials_auto/example_predictions.py +61 -0
  14. examples/integration-server/jobs/materials_auto/example_runsheet_wh.py +57 -16
  15. examples/integration-server/jobs/materials_auto/profile.yaml +27 -0
  16. examples/integration-server/pyproject.toml +4 -4
  17. examples/oauth.py +7 -0
  18. pkgs/argument_parser/__init__.py +1 -0
  19. pkgs/argument_parser/_is_namedtuple.py +3 -0
  20. pkgs/argument_parser/argument_parser.py +22 -3
  21. pkgs/serialization_util/serialization_helpers.py +3 -1
  22. pkgs/type_spec/builder.py +66 -19
  23. pkgs/type_spec/builder_types.py +9 -0
  24. pkgs/type_spec/config.py +26 -5
  25. pkgs/type_spec/cross_output_links.py +10 -16
  26. pkgs/type_spec/emit_open_api.py +72 -22
  27. pkgs/type_spec/emit_open_api_util.py +1 -0
  28. pkgs/type_spec/emit_python.py +76 -12
  29. pkgs/type_spec/emit_typescript.py +48 -32
  30. pkgs/type_spec/emit_typescript_util.py +44 -6
  31. pkgs/type_spec/load_types.py +2 -2
  32. pkgs/type_spec/open_api_util.py +16 -1
  33. pkgs/type_spec/parts/base.ts.prepart +4 -0
  34. pkgs/type_spec/type_info/emit_type_info.py +37 -4
  35. pkgs/type_spec/ui_entry_actions/generate_ui_entry_actions.py +1 -0
  36. pkgs/type_spec/value_spec/__main__.py +2 -2
  37. pkgs/type_spec/value_spec/emit_python.py +6 -1
  38. uncountable/core/client.py +10 -3
  39. uncountable/integration/cli.py +175 -23
  40. uncountable/integration/executors/executors.py +1 -2
  41. uncountable/integration/executors/generic_upload_executor.py +1 -1
  42. uncountable/integration/http_server/types.py +3 -1
  43. uncountable/integration/job.py +35 -3
  44. uncountable/integration/queue_runner/command_server/__init__.py +4 -0
  45. uncountable/integration/queue_runner/command_server/command_client.py +89 -0
  46. uncountable/integration/queue_runner/command_server/command_server.py +117 -5
  47. uncountable/integration/queue_runner/command_server/constants.py +4 -0
  48. uncountable/integration/queue_runner/command_server/protocol/command_server.proto +51 -0
  49. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2.py +34 -11
  50. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2.pyi +102 -1
  51. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2_grpc.py +180 -0
  52. uncountable/integration/queue_runner/command_server/types.py +44 -1
  53. uncountable/integration/queue_runner/datastore/datastore_sqlite.py +189 -8
  54. uncountable/integration/queue_runner/datastore/interface.py +13 -0
  55. uncountable/integration/queue_runner/datastore/model.py +8 -1
  56. uncountable/integration/queue_runner/job_scheduler.py +85 -21
  57. uncountable/integration/queue_runner/queue_runner.py +10 -2
  58. uncountable/integration/queue_runner/types.py +2 -0
  59. uncountable/integration/queue_runner/worker.py +28 -29
  60. uncountable/integration/scheduler.py +121 -23
  61. uncountable/integration/server.py +36 -6
  62. uncountable/integration/telemetry.py +129 -8
  63. uncountable/integration/webhook_server/entrypoint.py +2 -0
  64. uncountable/types/__init__.py +38 -0
  65. uncountable/types/api/entity/create_or_update_entity.py +1 -0
  66. uncountable/types/api/entity/export_entities.py +13 -0
  67. uncountable/types/api/entity/list_aggregate.py +79 -0
  68. uncountable/types/api/entity/list_entities.py +25 -0
  69. uncountable/types/api/entity/set_barcode.py +43 -0
  70. uncountable/types/api/entity/transition_entity_phase.py +2 -1
  71. uncountable/types/api/files/download_file.py +15 -1
  72. uncountable/types/api/integrations/__init__.py +1 -0
  73. uncountable/types/api/integrations/publish_realtime_data.py +41 -0
  74. uncountable/types/api/integrations/push_notification.py +49 -0
  75. uncountable/types/api/integrations/register_sockets_token.py +41 -0
  76. uncountable/types/api/listing/__init__.py +1 -0
  77. uncountable/types/api/listing/fetch_listing.py +57 -0
  78. uncountable/types/api/notebooks/__init__.py +1 -0
  79. uncountable/types/api/notebooks/add_notebook_content.py +119 -0
  80. uncountable/types/api/outputs/get_output_organization.py +173 -0
  81. uncountable/types/api/recipes/edit_recipe_inputs.py +1 -1
  82. uncountable/types/api/recipes/get_recipe_output_metadata.py +2 -2
  83. uncountable/types/api/recipes/get_recipes_data.py +29 -0
  84. uncountable/types/api/recipes/lock_recipes.py +2 -1
  85. uncountable/types/api/recipes/set_recipe_total.py +59 -0
  86. uncountable/types/api/recipes/unlock_recipes.py +2 -1
  87. uncountable/types/api/runsheet/export_default_runsheet.py +44 -0
  88. uncountable/types/api/uploader/complete_async_parse.py +46 -0
  89. uncountable/types/api/user/__init__.py +1 -0
  90. uncountable/types/api/user/get_current_user_info.py +40 -0
  91. uncountable/types/async_batch_processor.py +266 -0
  92. uncountable/types/async_batch_t.py +5 -0
  93. uncountable/types/client_base.py +432 -2
  94. uncountable/types/client_config.py +1 -0
  95. uncountable/types/client_config_t.py +10 -0
  96. uncountable/types/entity_t.py +9 -1
  97. uncountable/types/exports_t.py +1 -0
  98. uncountable/types/integration_server_t.py +2 -0
  99. uncountable/types/integration_session.py +10 -0
  100. uncountable/types/integration_session_t.py +60 -0
  101. uncountable/types/integrations.py +10 -0
  102. uncountable/types/integrations_t.py +62 -0
  103. uncountable/types/listing.py +46 -0
  104. uncountable/types/listing_t.py +533 -0
  105. uncountable/types/notices.py +8 -0
  106. uncountable/types/notices_t.py +37 -0
  107. uncountable/types/notifications.py +11 -0
  108. uncountable/types/notifications_t.py +74 -0
  109. uncountable/types/queued_job.py +2 -0
  110. uncountable/types/queued_job_t.py +20 -2
  111. uncountable/types/sockets.py +20 -0
  112. uncountable/types/sockets_t.py +169 -0
  113. uncountable/types/uploader.py +24 -0
  114. uncountable/types/uploader_t.py +222 -0
  115. {uncountablepythonsdk-0.0.115.dist-info → uncountablepythonsdk-0.0.142.dev0.dist-info}/METADATA +5 -2
  116. {uncountablepythonsdk-0.0.115.dist-info → uncountablepythonsdk-0.0.142.dev0.dist-info}/RECORD +118 -79
  117. docs/quickstart.md +0 -19
  118. {uncountablepythonsdk-0.0.115.dist-info → uncountablepythonsdk-0.0.142.dev0.dist-info}/WHEEL +0 -0
  119. {uncountablepythonsdk-0.0.115.dist-info → uncountablepythonsdk-0.0.142.dev0.dist-info}/top_level.txt +0 -0
docs/conf.py CHANGED
@@ -8,6 +8,11 @@
8
8
 
9
9
  import datetime
10
10
 
11
+ from docutils import nodes # type: ignore[import-untyped]
12
+ from sphinx.addnodes import pending_xref # type: ignore[import-not-found]
13
+ from sphinx.application import Sphinx # type: ignore[import-not-found]
14
+ from sphinx.environment import BuildEnvironment # type: ignore[import-not-found]
15
+
11
16
  project = "Uncountable SDK"
12
17
  copyright = f"{datetime.datetime.now(tz=datetime.UTC).date().year}, Uncountable Inc"
13
18
  author = "Uncountable Inc"
@@ -22,22 +27,23 @@ extensions = [
22
27
  "sphinx_copybutton",
23
28
  "sphinx_favicon",
24
29
  ]
25
- myst_enable_extensions = ["fieldlist", "deflist"]
30
+ myst_enable_extensions = ["fieldlist", "deflist", "colon_fence"]
26
31
 
27
32
  autoapi_dirs = ["../uncountable"]
28
33
  autoapi_options = [
29
34
  "members",
35
+ "inherited-members",
30
36
  "undoc-members",
31
- "show-inheritance",
32
- "show-module-summary",
33
- "imported-members",
34
37
  ]
38
+ autoapi_root = "api"
35
39
  autoapi_ignore = ["*integration*"]
36
40
  autodoc_typehints = "description"
41
+ autoapi_member_order = "groupwise"
42
+ autoapi_own_page_level = "class"
37
43
 
38
44
  exclude_patterns = ["_build", "Thumbs.db", ".DS_Store"]
39
45
 
40
-
46
+ python_use_unqualified_type_names = True
41
47
  # -- Options for HTML output -------------------------------------------------
42
48
  # https://www.sphinx-doc.org/en/master/usage/configuration.html#options-for-html-output
43
49
 
@@ -55,3 +61,44 @@ favicons = [
55
61
  "favicons/mstile-150x150.png",
56
62
  "favicons/safari-pinned-tab.svg",
57
63
  ]
64
+
65
+
66
+ def _hook_missing_reference(
67
+ _app: Sphinx, _env: BuildEnvironment, node: pending_xref, contnode: nodes.Text
68
+ ) -> nodes.reference | None:
69
+ """
70
+ Manually resolve reference when autoapi reference resolution fails.
71
+ This is necessary because autoapi does not fully support type aliases.
72
+ """
73
+ # example reftarget value: uncountable.types.identifier_t.IdentifierKey
74
+ target = node.get("reftarget", "")
75
+
76
+ # example refdoc value: api/uncountable/types/generic_upload_t/GenericUploadStrategy
77
+ current_doc = node.get("refdoc", "")
78
+
79
+ if not target.startswith("uncountable"):
80
+ return None
81
+
82
+ target_module, target_name = target.rsplit(".", 1)
83
+
84
+ # construct relative path from current doc page to target page
85
+ relative_segments_to_root = [".." for _ in current_doc.split("/")]
86
+ relative_segments_to_target = target_module.split(".")
87
+
88
+ # example full relative path: ../../../../../api/uncountable/types/identifier_t/#uncountable.types.identifier_t.IdentifierKey
89
+ full_relative_path = "/".join([
90
+ *relative_segments_to_root,
91
+ autoapi_root,
92
+ *relative_segments_to_target,
93
+ f"#{target}",
94
+ ])
95
+
96
+ return nodes.reference(
97
+ text=target_name if python_use_unqualified_type_names else target,
98
+ children=[contnode],
99
+ refuri=full_relative_path,
100
+ )
101
+
102
+
103
+ def setup(app: Sphinx) -> None:
104
+ app.connect("missing-reference", _hook_missing_reference)
docs/index.md CHANGED
@@ -3,11 +3,114 @@
3
3
  ![PyPI - Python Version](https://img.shields.io/pypi/pyversions/UncountablePythonSDK)
4
4
 
5
5
 
6
- `Uncountable Python SDK` is a python package that allows interacting with the uncountable platform
6
+ The Uncountable Python SDK is a python package that provides a wrapper around the Uncountable REST API.
7
7
 
8
- ```{toctree}
9
- :maxdepth: 2
8
+ Using this SDK provides the following advantages:
9
+
10
+ - In-editor parameter/type safety
11
+ - Automatic parsing of response data.
12
+ - Reduced code boilerplate
13
+ - Helper methods
14
+
15
+ ## Getting Started
16
+ The first step in any integration is to create a [Client](uncountable.core.client.Client) object. The client provides access to all available SDK methods, and includes built-in request authentication & error propagation.
17
+
18
+ ### Creating a Client
19
+ Create a client using one of the supported authentication mechanisms. API credentials can be generated by a member of Uncountable staff.
20
+
21
+ ::::{tab-set}
22
+ :::{tab-item} Basic Auth
23
+ ```{literalinclude} ../examples/basic_auth.py
24
+ ```
25
+ :::
26
+
27
+ :::{tab-item} OAuth
28
+ ```{literalinclude} ../examples/oauth.py
29
+ ```
30
+ :::
31
+ ::::
32
+
33
+ The provided code examples assume that a Client has been created and stored in the `client` variable
34
+
35
+
36
+ ### Basic Usage
37
+
38
+ :::{dropdown} List Ingredient Names & IDs
39
+ ```{code-block} python
40
+ from uncountable.types import entity_t, id_source_t
41
+
42
+ client.list_id_source(
43
+ spec=id_source_t.IdSourceSpecEntity(entity_type=entity_t.EntityType.INGREDIENT),
44
+ search_label="",
45
+ )
46
+ ```
47
+ Example Response:
48
+ ```code
49
+ Data(
50
+ results=[
51
+ IdName(id=1, name='Filler'),
52
+ IdName(id=2, name='Calcium Oxide 2'),
53
+ IdName(id=3, name='Carbon Black'),
54
+ ]
55
+ )
56
+ ```
57
+ :::
58
+
59
+ :::{dropdown} Create an Experiment
60
+ ```{code-block} python
61
+ client.create_recipe(material_family_id=1, workflow_id=1, name="Example Recipe")
62
+ ```
63
+ Example Response:
64
+ ```code
65
+ Data(result_id=52271)
66
+ ```
67
+ :::
68
+
69
+ :::{dropdown} Upload a file
70
+ ```{code-block} python
71
+ from uncountable.core.file_upload import MediaFileUpload
72
+
73
+ client.upload_files(file_uploads=[MediaFileUpload(path="/path/to/local/example_file.pdf")])
74
+ ```
75
+ Example Response:
76
+ ```code
77
+ [
78
+ UploadedFile(name='example_file.pdf', file_id=718)
79
+ ]
80
+ ```
81
+ :::
82
+
83
+ [More examples](integration_examples/index)
84
+
85
+ ## Errors
86
+ Client methods will raise Exceptions when the API returns codes in the `3xx`, `4xx` or `5xx` ranges. Ensure all method calls are wrapped in Exception handling logic.
87
+
88
+ ## Pagination
89
+ Many of the Uncountable APIs require pagination to fetch more than 100 results at once. The following code snippet implements pagination to fetch the Names & IDs of all Projects:
90
+ :::{dropdown} Pagination Example
91
+ ```{code-block} python
92
+ from uncountable.types import entity_t, id_source_t
93
+ from uncountable.types.api.id_source.list_id_source import IdName
10
94
 
11
- quickstart
95
+ def fetch_all_projects(client: Client) -> list[IdName]:
96
+ projects: list[IdName] = []
97
+ while True:
98
+ response = client.list_id_source(
99
+ spec=IdSourceSpecEntity(entity_type=entity_t.EntityType.PROJECT),
100
+ search_label="",
101
+ offset=len(projects),
102
+ )
103
+ projects.extend(response.results)
104
+ if len(response.results) < 100:
105
+ return projects
12
106
  ```
107
+ :::
13
108
 
109
+
110
+ ```{toctree}
111
+ :hidden:
112
+ Overview <self>
113
+ Available SDK Methods <api/uncountable/core/client/Client>
114
+ integration_examples/index
115
+ SDK Reference <api/uncountable/index>
116
+ ```
@@ -0,0 +1,43 @@
1
+ # Create an Ingredient
2
+
3
+ Use the `create_or_update_entity` method to create Ingredients.
4
+
5
+ The following fields are required when creating an Ingredient:
6
+ - `name`: The name of the Ingredient
7
+ - `core_ingredient_ingredientMaterialFamilies`: The list of material families in which to include the Ingredient
8
+
9
+ The reference name of the default definition of Ingredients is `uncIngredient`
10
+
11
+ This is an example of a minimal ingredient creation call
12
+
13
+ ```{code-block} python
14
+ from uncountable.types import entity_t, field_values_t, identifier_t
15
+
16
+ client.create_or_update_entity(
17
+ entity_type=entity_t.EntityType.INGREDIENT,
18
+ definition_key=identifier_t.IdentifierKeyRefName(ref_name="uncIngredient"),
19
+ field_values=[
20
+ field_values_t.FieldArgumentValue(
21
+ field_key=identifier_t.IdentifierKeyRefName(
22
+ ref_name="core_ingredient_ingredientMaterialFamilies"
23
+ ),
24
+ value=field_values_t.FieldValueIds(
25
+ entity_type=entity_t.EntityType.MATERIAL_FAMILY,
26
+ identifier_keys=[identifier_t.IdentifierKeyId(id=1)],
27
+ ),
28
+ ),
29
+ field_values_t.FieldArgumentValue(
30
+ field_key=identifier_t.IdentifierKeyRefName(ref_name="name"),
31
+ value=field_values_t.FieldValueText(value="Example Ingredient"),
32
+ ),
33
+ ],
34
+ )
35
+ ```
36
+
37
+ Example Response:
38
+ ```{code}
39
+ Data(modification_made=True, result_id=3124, entity=None, result_values=None)
40
+ ```
41
+
42
+ Optional fields:
43
+ - `core_ingredient_quantityType`: The quantity type of the ingredient (default is `numeric`)
@@ -0,0 +1,56 @@
1
+ # Create an Output
2
+
3
+ Use the `create_or_update_entity` method to create Outputs.
4
+
5
+ The following fields are required when creating an Output:
6
+ - `name`: The name of the Output
7
+ - `core_output_unitsId`: The unit the output is measured in
8
+ - `core_output_outputMaterialFamilies`: The list of material families in which to include the Output
9
+ - `core_output_quantityType`: The quantity type of the output
10
+
11
+ The reference name of the default definition of Ingredients is `unc_output_definition`
12
+
13
+ This is an example of a minimal output creation call
14
+
15
+ ```{code-block} python
16
+ from uncountable.types import entity_t, field_values_t, identifier_t
17
+
18
+ client.create_or_update_entity(
19
+ entity_type=entity_t.EntityType.OUTPUT,
20
+ definition_key=identifier_t.IdentifierKeyRefName(ref_name="unc_output_definition"),
21
+ field_values=[
22
+ field_values_t.FieldArgumentValue(
23
+ field_key=identifier_t.IdentifierKeyRefName(ref_name="name"),
24
+ value=field_values_t.FieldValueText(value="Example Output"),
25
+ ),
26
+ field_values_t.FieldArgumentValue(
27
+ field_key=identifier_t.IdentifierKeyRefName(ref_name="core_output_unitsId"),
28
+ value=field_values_t.FieldValueId(
29
+ entity_type=entity_t.EntityType.UNITS,
30
+ identifier_key=identifier_t.IdentifierKeyId(id=1),
31
+ ),
32
+ ),
33
+ field_values_t.FieldArgumentValue(
34
+ field_key=identifier_t.IdentifierKeyRefName(
35
+ ref_name="core_output_outputMaterialFamilies"
36
+ ),
37
+ value=field_values_t.FieldValueIds(
38
+ entity_type=entity_t.EntityType.MATERIAL_FAMILY,
39
+ identifier_keys=[identifier_t.IdentifierKeyId(id=1)],
40
+ ),
41
+ ),
42
+ field_values_t.FieldArgumentValue(
43
+ field_key=identifier_t.IdentifierKeyRefName(
44
+ ref_name="core_output_quantityType"
45
+ ),
46
+ value=field_values_t.FieldValueFieldOption(value="numeric"),
47
+ ),
48
+ ],
49
+ )
50
+ ```
51
+
52
+ Example Response:
53
+
54
+ ```{code}
55
+ Data(modification_made=True, result_id=653, entity=None, result_values=None)
56
+ ```
@@ -0,0 +1,6 @@
1
+ # Integration Examples
2
+
3
+ ```{toctree}
4
+ create_ingredient
5
+ create_output
6
+ ```
docs/justfile CHANGED
@@ -1,5 +1,5 @@
1
1
  docs-setup-python:
2
- curl -LsSf https://astral.sh/uv/0.7.1/install.sh | sh
2
+ curl -LsSf https://astral.sh/uv/0.8.4/install.sh | sh
3
3
  uv pip install -r requirements.txt
4
4
 
5
5
  docs-clean:
docs/requirements.txt CHANGED
@@ -1,8 +1,9 @@
1
- furo==2024.8.6
2
- myst-parser==4.0.0
1
+ furo==2025.9.25
2
+ myst-parser==4.0.1
3
3
  sphinx-autoapi==3.6.0
4
4
  sphinx-copybutton==0.5.2
5
5
  Sphinx==8.2.0
6
6
  sphinx_design==0.6.1
7
7
  sphinx-favicon==1.0.1
8
8
  astroid==3.3.8
9
+ docutils==0.21.2
examples/basic_auth.py ADDED
@@ -0,0 +1,7 @@
1
+ from uncountable.core.client import Client
2
+ from uncountable.core.types import AuthDetailsApiKey
3
+
4
+ client = Client(
5
+ base_url="https://app.uncountable.com",
6
+ auth_details=AuthDetailsApiKey(api_id="x", api_secret_key="x"),
7
+ )
@@ -1,3 +1,5 @@
1
+ import time
2
+
1
3
  from uncountable.integration.job import CronJob, JobArguments, register_job
2
4
  from uncountable.types import entity_t
3
5
  from uncountable.types.job_definition_t import JobResult
@@ -15,4 +17,5 @@ class MyCronJob(CronJob):
15
17
  if field_val.field_ref_name == "name":
16
18
  name = field_val.value
17
19
  args.logger.log_info(f"material family found with name: {name}")
20
+ time.sleep(20)
18
21
  return JobResult(success=True)
@@ -4,6 +4,7 @@ from uncountable.integration.http_server import (
4
4
  GenericHttpRequest,
5
5
  GenericHttpResponse,
6
6
  )
7
+ from uncountable.integration.http_server.types import HttpException
7
8
  from uncountable.integration.job import CustomHttpJob, register_job
8
9
  from uncountable.types import job_definition_t
9
10
 
@@ -14,22 +15,33 @@ class ExampleWebhookPayload:
14
15
  message: str
15
16
 
16
17
 
18
+ _EXPECTED_USER_ID = 1
19
+
20
+
17
21
  @register_job
18
22
  class HttpExample(CustomHttpJob):
19
23
  @staticmethod
20
24
  def validate_request(
21
25
  *,
22
- request: GenericHttpRequest,
23
- job_definition: job_definition_t.HttpJobDefinitionBase,
24
- profile_meta: job_definition_t.ProfileMetadata,
26
+ request: GenericHttpRequest, # noqa: ARG004
27
+ job_definition: job_definition_t.HttpJobDefinitionBase, # noqa: ARG004
28
+ profile_meta: job_definition_t.ProfileMetadata, # noqa: ARG004
25
29
  ) -> None:
26
- return None
30
+ if (
31
+ CustomHttpJob.get_validated_oauth_request_user_id(
32
+ request=request, profile_metadata=profile_meta
33
+ )
34
+ != _EXPECTED_USER_ID
35
+ ):
36
+ raise HttpException(
37
+ message="unauthorized; invalid oauth token", error_code=401
38
+ )
27
39
 
28
40
  @staticmethod
29
41
  def handle_request(
30
42
  *,
31
- request: GenericHttpRequest,
32
- job_definition: job_definition_t.HttpJobDefinitionBase,
33
- profile_meta: job_definition_t.ProfileMetadata,
43
+ request: GenericHttpRequest, # noqa: ARG004
44
+ job_definition: job_definition_t.HttpJobDefinitionBase, # noqa: ARG004
45
+ profile_meta: job_definition_t.ProfileMetadata, # noqa: ARG004
34
46
  ) -> GenericHttpResponse:
35
47
  return GenericHttpResponse(response="OK", status_code=200)
@@ -0,0 +1,100 @@
1
+ import json
2
+ import time
3
+ from dataclasses import dataclass
4
+ from decimal import Decimal
5
+
6
+ from uncountable.integration.job import JobArguments, WebhookJob, register_job
7
+ from uncountable.types import (
8
+ base_t,
9
+ identifier_t,
10
+ job_definition_t,
11
+ sockets_t,
12
+ )
13
+ from uncountable.types.integration_session_t import IntegrationSessionInstrument
14
+ from websockets.sync.client import connect
15
+ from websockets.typing import Data
16
+
17
+ from pkgs.argument_parser.argument_parser import CachedParser
18
+ from pkgs.serialization_util import serialize_for_api
19
+
20
+
21
+ @dataclass(kw_only=True)
22
+ class InstrumentPayload:
23
+ equipment_id: base_t.ObjectId
24
+
25
+
26
+ @register_job
27
+ class InstrumentExample(WebhookJob[InstrumentPayload]):
28
+ def run(
29
+ self, args: JobArguments, payload: InstrumentPayload
30
+ ) -> job_definition_t.JobResult:
31
+ parser: CachedParser[sockets_t.SocketResponse] = CachedParser(
32
+ sockets_t.SocketResponse # type:ignore[arg-type]
33
+ )
34
+
35
+ def parse_message(message: Data) -> sockets_t.SocketEventData | None:
36
+ try:
37
+ return parser.parse_api(json.loads(message)).data
38
+ except ValueError:
39
+ return None
40
+
41
+ integration_session = IntegrationSessionInstrument(
42
+ equipment_key=identifier_t.IdentifierKeyId(id=payload.equipment_id)
43
+ )
44
+ registration_info = args.client.register_sockets_token(
45
+ socket_request=sockets_t.SocketRequestIntegrationSession(
46
+ integration_session=integration_session
47
+ )
48
+ ).response
49
+ token = registration_info.token
50
+ room_key = registration_info.room_key
51
+ args.logger.log_info(f"Token: {token}")
52
+
53
+ with connect(
54
+ "ws://host.docker.internal:8765",
55
+ additional_headers={
56
+ "Authorization": f"Bearer {token}",
57
+ "X-UNC-EXTERNAL": "true",
58
+ },
59
+ ) as ws:
60
+ ws.send(
61
+ json.dumps(
62
+ serialize_for_api(
63
+ sockets_t.JoinRoomWithTokenSocketClientMessage(token=token)
64
+ )
65
+ )
66
+ )
67
+ for i in range(10):
68
+ args.logger.log_info("Sending reading...")
69
+ ws.send(
70
+ json.dumps(
71
+ serialize_for_api(
72
+ sockets_t.SendInstrumentReadingClientMessage(
73
+ value=Decimal(i * 100), room_key=room_key
74
+ )
75
+ )
76
+ )
77
+ )
78
+ time.sleep(0.75)
79
+
80
+ while True:
81
+ message = parse_message(ws.recv())
82
+ match message:
83
+ case sockets_t.UsersInRoomUpdatedEventData():
84
+ num_users = len(message.user_ids)
85
+ if num_users <= 1:
86
+ break
87
+ else:
88
+ args.logger.log_info(
89
+ f"Session still open, {num_users} users in room."
90
+ )
91
+ case _:
92
+ args.logger.log_info("Session still open...")
93
+ continue
94
+
95
+ args.logger.log_info("Session closed.")
96
+ return job_definition_t.JobResult(success=True)
97
+
98
+ @property
99
+ def webhook_payload_type(self) -> type:
100
+ return InstrumentPayload
@@ -0,0 +1,140 @@
1
+ from dataclasses import dataclass
2
+ from decimal import Decimal
3
+
4
+ from uncountable.integration.job import JobArguments, WebhookJob, register_job
5
+ from uncountable.types import (
6
+ base_t,
7
+ entity_t,
8
+ generic_upload_t,
9
+ identifier_t,
10
+ job_definition_t,
11
+ notifications_t,
12
+ uploader_t,
13
+ )
14
+
15
+
16
+ @dataclass(kw_only=True)
17
+ class ParsePayload:
18
+ async_job_id: base_t.ObjectId
19
+
20
+
21
+ @register_job
22
+ class ParseExample(WebhookJob[ParsePayload]):
23
+ def run(
24
+ self, args: JobArguments, payload: ParsePayload
25
+ ) -> job_definition_t.JobResult:
26
+ user_id: base_t.ObjectId | None = None
27
+ recipe_id: base_t.ObjectId | None = None
28
+ file_name: str | None = None
29
+ data = args.client.get_entities_data(
30
+ entity_ids=[payload.async_job_id], entity_type=entity_t.EntityType.ASYNC_JOB
31
+ )
32
+ for field_value in data.entity_details[0].field_values:
33
+ if field_value.field_ref_name == "core_async_job_jobData":
34
+ assert isinstance(field_value.value, dict)
35
+ assert isinstance(field_value.value["user_id"], int)
36
+ user_id = field_value.value["user_id"]
37
+ elif (
38
+ field_value.field_ref_name
39
+ == "unc_async_job_custom_parser_recipe_ids_in_view"
40
+ ):
41
+ if field_value.value is None:
42
+ continue
43
+ assert isinstance(field_value.value, list)
44
+ if len(field_value.value) > 0:
45
+ assert isinstance(field_value.value[0], int)
46
+ recipe_id = field_value.value[0]
47
+ elif field_value.field_ref_name == "unc_async_job_custom_parser_input_file":
48
+ assert isinstance(field_value.value, list)
49
+ assert len(field_value.value) == 1
50
+ assert isinstance(field_value.value[0], dict)
51
+ assert isinstance(field_value.value[0]["name"], str)
52
+ file_name = field_value.value[0]["name"]
53
+
54
+ assert user_id is not None
55
+ assert file_name is not None
56
+
57
+ dummy_parsed_file_data: list[uploader_t.ParsedFileData] = [
58
+ uploader_t.ParsedFileData(
59
+ file_name=file_name,
60
+ file_structures=[
61
+ uploader_t.DataChannel(
62
+ type=uploader_t.StructureElementType.CHANNEL,
63
+ channel=uploader_t.TextChannelData(
64
+ name="column1",
65
+ type=uploader_t.ChannelType.TEXT_CHANNEL,
66
+ data=[
67
+ uploader_t.StringValue(value="value1"),
68
+ uploader_t.StringValue(value="value4"),
69
+ uploader_t.StringValue(value="value7"),
70
+ ],
71
+ ),
72
+ ),
73
+ uploader_t.DataChannel(
74
+ type=uploader_t.StructureElementType.CHANNEL,
75
+ channel=uploader_t.TextChannelData(
76
+ name="column2",
77
+ type=uploader_t.ChannelType.TEXT_CHANNEL,
78
+ data=[
79
+ uploader_t.StringValue(value="value2"),
80
+ uploader_t.StringValue(value="value5"),
81
+ uploader_t.StringValue(value="value8"),
82
+ ],
83
+ ),
84
+ ),
85
+ uploader_t.DataChannel(
86
+ type=uploader_t.StructureElementType.CHANNEL,
87
+ channel=uploader_t.TextChannelData(
88
+ name="column3",
89
+ type=uploader_t.ChannelType.TEXT_CHANNEL,
90
+ data=[
91
+ uploader_t.StringValue(value="value3"),
92
+ uploader_t.StringValue(value="value6"),
93
+ uploader_t.StringValue(value="value9"),
94
+ ],
95
+ ),
96
+ ),
97
+ uploader_t.HeaderEntry(
98
+ type=uploader_t.StructureElementType.HEADER,
99
+ value=uploader_t.TextHeaderData(
100
+ name="file_source",
101
+ type=uploader_t.HeaderType.TEXT_HEADER,
102
+ data=uploader_t.StringValue(value="my_file_to_upload.xlsx"),
103
+ ),
104
+ ),
105
+ uploader_t.HeaderEntry(
106
+ type=uploader_t.StructureElementType.HEADER,
107
+ value=uploader_t.NumericHeaderData(
108
+ name="file structure number",
109
+ data=uploader_t.DecimalValue(value=Decimal(99)),
110
+ ),
111
+ ),
112
+ ],
113
+ )
114
+ ]
115
+
116
+ complete_async_parse_req = args.batch_processor.complete_async_parse(
117
+ parsed_file_data=dummy_parsed_file_data,
118
+ async_job_key=identifier_t.IdentifierKeyId(id=payload.async_job_id),
119
+ upload_destination=generic_upload_t.UploadDestinationRecipe(
120
+ recipe_key=identifier_t.IdentifierKeyId(id=recipe_id or 1)
121
+ ),
122
+ )
123
+
124
+ args.batch_processor.push_notification(
125
+ depends_on=[complete_async_parse_req.batch_reference],
126
+ notification_targets=[
127
+ notifications_t.NotificationTargetUser(
128
+ user_key=identifier_t.IdentifierKeyId(id=user_id)
129
+ )
130
+ ],
131
+ subject="Upload complete",
132
+ message="Your file has been uploaded",
133
+ display_notice=True,
134
+ )
135
+
136
+ return job_definition_t.JobResult(success=True)
137
+
138
+ @property
139
+ def webhook_payload_type(self) -> type:
140
+ return ParsePayload