UncountablePythonSDK 0.0.7__py3-none-any.whl → 0.0.92__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 (311) hide show
  1. UncountablePythonSDK-0.0.92.dist-info/METADATA +61 -0
  2. UncountablePythonSDK-0.0.92.dist-info/RECORD +301 -0
  3. {UncountablePythonSDK-0.0.7.dist-info → UncountablePythonSDK-0.0.92.dist-info}/WHEEL +1 -1
  4. {UncountablePythonSDK-0.0.7.dist-info → UncountablePythonSDK-0.0.92.dist-info}/top_level.txt +1 -1
  5. docs/.gitignore +1 -0
  6. docs/conf.py +57 -0
  7. docs/index.md +13 -0
  8. docs/justfile +12 -0
  9. docs/quickstart.md +19 -0
  10. docs/requirements.txt +7 -0
  11. docs/static/favicons/android-chrome-192x192.png +0 -0
  12. docs/static/favicons/android-chrome-512x512.png +0 -0
  13. docs/static/favicons/apple-touch-icon.png +0 -0
  14. docs/static/favicons/browserconfig.xml +9 -0
  15. docs/static/favicons/favicon-16x16.png +0 -0
  16. docs/static/favicons/favicon-32x32.png +0 -0
  17. docs/static/favicons/manifest.json +18 -0
  18. docs/static/favicons/mstile-150x150.png +0 -0
  19. docs/static/favicons/safari-pinned-tab.svg +32 -0
  20. docs/static/logo_blue.png +0 -0
  21. examples/async_batch.py +35 -0
  22. examples/create_entity.py +22 -17
  23. examples/download_files.py +26 -0
  24. examples/edit_recipe_inputs.py +50 -0
  25. examples/integration-server/jobs/materials_auto/example_cron.py +18 -0
  26. examples/integration-server/jobs/materials_auto/example_wh.py +15 -0
  27. examples/integration-server/jobs/materials_auto/profile.yaml +43 -0
  28. examples/integration-server/pyproject.toml +224 -0
  29. examples/invoke_uploader.py +26 -0
  30. examples/set_recipe_metadata_file.py +40 -0
  31. examples/set_recipe_output_file_sdk.py +26 -0
  32. examples/upload_files.py +18 -0
  33. pkgs/argument_parser/__init__.py +5 -0
  34. pkgs/argument_parser/_is_enum.py +1 -6
  35. pkgs/argument_parser/argument_parser.py +232 -76
  36. pkgs/argument_parser/case_convert.py +4 -3
  37. pkgs/filesystem_utils/__init__.py +20 -0
  38. pkgs/filesystem_utils/_blob_session.py +137 -0
  39. pkgs/filesystem_utils/_gdrive_session.py +309 -0
  40. pkgs/filesystem_utils/_local_session.py +69 -0
  41. pkgs/filesystem_utils/_s3_session.py +117 -0
  42. pkgs/filesystem_utils/_sftp_session.py +147 -0
  43. pkgs/filesystem_utils/file_type_utils.py +91 -0
  44. pkgs/filesystem_utils/filesystem_session.py +39 -0
  45. pkgs/py.typed +0 -0
  46. pkgs/serialization/__init__.py +8 -1
  47. pkgs/serialization/annotation.py +64 -0
  48. pkgs/serialization/opaque_key.py +1 -1
  49. pkgs/serialization/serial_alias.py +47 -0
  50. pkgs/serialization/serial_class.py +65 -50
  51. pkgs/serialization/serial_generic.py +16 -0
  52. pkgs/serialization/serial_union.py +84 -0
  53. pkgs/serialization/yaml.py +57 -0
  54. pkgs/serialization_util/__init__.py +7 -7
  55. pkgs/serialization_util/_get_type_for_serialization.py +1 -3
  56. pkgs/serialization_util/convert_to_snakecase.py +27 -0
  57. pkgs/serialization_util/dataclasses.py +14 -0
  58. pkgs/serialization_util/serialization_helpers.py +118 -73
  59. pkgs/strenum_compat/strenum_compat.py +1 -9
  60. pkgs/type_spec/actions_registry/__init__.py +0 -0
  61. pkgs/type_spec/actions_registry/__main__.py +126 -0
  62. pkgs/type_spec/actions_registry/emit_typescript.py +182 -0
  63. pkgs/type_spec/builder.py +475 -89
  64. pkgs/type_spec/config.py +24 -19
  65. pkgs/type_spec/emit_io_ts.py +5 -2
  66. pkgs/type_spec/emit_open_api.py +266 -32
  67. pkgs/type_spec/emit_open_api_util.py +32 -13
  68. pkgs/type_spec/emit_python.py +601 -150
  69. pkgs/type_spec/emit_typescript.py +74 -273
  70. pkgs/type_spec/emit_typescript_util.py +239 -5
  71. pkgs/type_spec/load_types.py +55 -10
  72. pkgs/type_spec/open_api_util.py +30 -41
  73. pkgs/type_spec/parts/base.py.prepart +4 -3
  74. pkgs/type_spec/type_info/emit_type_info.py +178 -16
  75. pkgs/type_spec/util.py +11 -11
  76. pkgs/type_spec/value_spec/__main__.py +3 -3
  77. pkgs/type_spec/value_spec/convert_type.py +8 -1
  78. pkgs/type_spec/value_spec/emit_python.py +13 -4
  79. uncountable/__init__.py +1 -2
  80. uncountable/core/__init__.py +12 -2
  81. uncountable/core/async_batch.py +37 -0
  82. uncountable/core/client.py +293 -43
  83. uncountable/core/environment.py +41 -0
  84. uncountable/core/file_upload.py +135 -0
  85. uncountable/core/types.py +17 -0
  86. uncountable/integration/__init__.py +0 -0
  87. uncountable/integration/cli.py +49 -0
  88. uncountable/integration/construct_client.py +51 -0
  89. uncountable/integration/cron.py +29 -0
  90. uncountable/integration/db/__init__.py +0 -0
  91. uncountable/integration/db/connect.py +18 -0
  92. uncountable/integration/db/session.py +25 -0
  93. uncountable/integration/entrypoint.py +13 -0
  94. uncountable/integration/executors/__init__.py +0 -0
  95. uncountable/integration/executors/executors.py +148 -0
  96. uncountable/integration/executors/generic_upload_executor.py +284 -0
  97. uncountable/integration/executors/script_executor.py +25 -0
  98. uncountable/integration/job.py +87 -0
  99. uncountable/integration/queue_runner/__init__.py +0 -0
  100. uncountable/integration/queue_runner/command_server/__init__.py +24 -0
  101. uncountable/integration/queue_runner/command_server/command_client.py +68 -0
  102. uncountable/integration/queue_runner/command_server/command_server.py +64 -0
  103. uncountable/integration/queue_runner/command_server/protocol/__init__.py +0 -0
  104. uncountable/integration/queue_runner/command_server/protocol/command_server.proto +22 -0
  105. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2.py +40 -0
  106. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2.pyi +38 -0
  107. uncountable/integration/queue_runner/command_server/protocol/command_server_pb2_grpc.py +129 -0
  108. uncountable/integration/queue_runner/command_server/types.py +52 -0
  109. uncountable/integration/queue_runner/datastore/__init__.py +3 -0
  110. uncountable/integration/queue_runner/datastore/datastore_sqlite.py +93 -0
  111. uncountable/integration/queue_runner/datastore/interface.py +19 -0
  112. uncountable/integration/queue_runner/datastore/model.py +17 -0
  113. uncountable/integration/queue_runner/job_scheduler.py +163 -0
  114. uncountable/integration/queue_runner/queue_runner.py +26 -0
  115. uncountable/integration/queue_runner/types.py +7 -0
  116. uncountable/integration/queue_runner/worker.py +119 -0
  117. uncountable/integration/scan_profiles.py +67 -0
  118. uncountable/integration/scheduler.py +150 -0
  119. uncountable/integration/secret_retrieval/__init__.py +3 -0
  120. uncountable/integration/secret_retrieval/retrieve_secret.py +93 -0
  121. uncountable/integration/server.py +117 -0
  122. uncountable/integration/telemetry.py +209 -0
  123. uncountable/integration/webhook_server/entrypoint.py +170 -0
  124. uncountable/types/__init__.py +151 -5
  125. uncountable/types/api/batch/execute_batch.py +15 -7
  126. uncountable/types/api/batch/execute_batch_load_async.py +42 -0
  127. uncountable/types/api/chemical/__init__.py +1 -0
  128. uncountable/types/api/chemical/convert_chemical_formats.py +63 -0
  129. uncountable/types/api/entity/create_entities.py +23 -10
  130. uncountable/types/api/entity/create_entity.py +21 -12
  131. uncountable/types/api/entity/get_entities_data.py +19 -29
  132. uncountable/types/api/entity/grant_entity_permissions.py +48 -0
  133. uncountable/types/api/entity/list_entities.py +28 -20
  134. uncountable/types/api/entity/lock_entity.py +45 -0
  135. uncountable/types/api/entity/resolve_entity_ids.py +19 -7
  136. uncountable/types/api/entity/set_entity_field_values.py +44 -0
  137. uncountable/types/api/entity/set_values.py +13 -28
  138. uncountable/types/api/entity/transition_entity_phase.py +80 -0
  139. uncountable/types/api/entity/unlock_entity.py +44 -0
  140. uncountable/types/api/equipment/__init__.py +1 -0
  141. uncountable/types/api/equipment/associate_equipment_input.py +44 -0
  142. uncountable/types/api/field_options/__init__.py +1 -0
  143. uncountable/types/api/field_options/upsert_field_options.py +55 -0
  144. uncountable/types/api/files/__init__.py +1 -0
  145. uncountable/types/api/files/download_file.py +77 -0
  146. uncountable/types/api/id_source/__init__.py +1 -0
  147. uncountable/types/api/id_source/list_id_source.py +56 -0
  148. uncountable/types/api/id_source/match_id_source.py +54 -0
  149. uncountable/types/api/input_groups/get_input_group_names.py +18 -7
  150. uncountable/types/api/inputs/create_inputs.py +25 -24
  151. uncountable/types/api/inputs/get_input_data.py +37 -31
  152. uncountable/types/api/inputs/get_input_names.py +20 -9
  153. uncountable/types/api/inputs/get_inputs_data.py +33 -27
  154. uncountable/types/api/inputs/set_input_attribute_values.py +18 -13
  155. uncountable/types/api/inputs/set_input_category.py +44 -0
  156. uncountable/types/api/inputs/set_input_subcategories.py +45 -0
  157. uncountable/types/api/inputs/set_intermediate_type.py +50 -0
  158. uncountable/types/api/material_families/__init__.py +1 -0
  159. uncountable/types/api/material_families/update_entity_material_families.py +48 -0
  160. uncountable/types/api/outputs/get_output_data.py +38 -29
  161. uncountable/types/api/outputs/get_output_names.py +20 -9
  162. uncountable/types/api/outputs/resolve_output_conditions.py +23 -10
  163. uncountable/types/api/permissions/__init__.py +1 -0
  164. uncountable/types/api/permissions/set_core_permissions.py +105 -0
  165. uncountable/types/api/project/get_projects.py +23 -19
  166. uncountable/types/api/project/get_projects_data.py +26 -43
  167. uncountable/types/api/recipe_links/__init__.py +1 -0
  168. uncountable/types/api/recipe_links/create_recipe_link.py +46 -0
  169. uncountable/types/api/recipe_links/remove_recipe_link.py +45 -0
  170. uncountable/types/api/recipe_metadata/get_recipe_metadata_data.py +21 -10
  171. uncountable/types/api/recipes/add_recipe_to_project.py +42 -0
  172. uncountable/types/api/recipes/archive_recipes.py +42 -0
  173. uncountable/types/api/recipes/associate_recipe_as_input.py +44 -0
  174. uncountable/types/api/recipes/associate_recipe_as_lot.py +43 -0
  175. uncountable/types/api/recipes/clear_recipe_outputs.py +42 -0
  176. uncountable/types/api/recipes/create_recipe.py +51 -0
  177. uncountable/types/api/recipes/create_recipes.py +25 -24
  178. uncountable/types/api/recipes/disassociate_recipe_as_input.py +42 -0
  179. uncountable/types/api/recipes/edit_recipe_inputs.py +283 -0
  180. uncountable/types/api/recipes/get_column_calculation_values.py +58 -0
  181. uncountable/types/api/recipes/get_curve.py +13 -27
  182. uncountable/types/api/recipes/get_recipe_calculations.py +21 -21
  183. uncountable/types/api/recipes/get_recipe_links.py +14 -6
  184. uncountable/types/api/recipes/get_recipe_names.py +18 -7
  185. uncountable/types/api/recipes/get_recipe_output_metadata.py +18 -19
  186. uncountable/types/api/recipes/get_recipes_data.py +83 -144
  187. uncountable/types/api/recipes/lock_recipes.py +63 -0
  188. uncountable/types/api/recipes/remove_recipe_from_project.py +42 -0
  189. uncountable/types/api/recipes/set_recipe_inputs.py +21 -11
  190. uncountable/types/api/recipes/set_recipe_metadata.py +43 -0
  191. uncountable/types/api/recipes/set_recipe_output_annotations.py +115 -0
  192. uncountable/types/api/recipes/set_recipe_output_file.py +56 -0
  193. uncountable/types/api/recipes/set_recipe_outputs.py +28 -15
  194. uncountable/types/api/recipes/set_recipe_tags.py +109 -0
  195. uncountable/types/api/recipes/unarchive_recipes.py +41 -0
  196. uncountable/types/api/recipes/unlock_recipes.py +50 -0
  197. uncountable/types/api/triggers/__init__.py +1 -0
  198. uncountable/types/api/triggers/run_trigger.py +43 -0
  199. uncountable/types/api/uploader/__init__.py +1 -0
  200. uncountable/types/api/uploader/invoke_uploader.py +47 -0
  201. uncountable/types/async_batch.py +13 -0
  202. uncountable/types/async_batch_processor.py +384 -0
  203. uncountable/types/async_batch_t.py +97 -0
  204. uncountable/types/async_jobs.py +9 -0
  205. uncountable/types/async_jobs_t.py +53 -0
  206. uncountable/types/auth_retrieval.py +12 -0
  207. uncountable/types/auth_retrieval_t.py +75 -0
  208. uncountable/types/base.py +5 -78
  209. uncountable/types/base_t.py +85 -0
  210. uncountable/types/calculations.py +8 -0
  211. uncountable/types/calculations_t.py +27 -0
  212. uncountable/types/chemical_structure.py +8 -0
  213. uncountable/types/chemical_structure_t.py +28 -0
  214. uncountable/types/client_base.py +1115 -76
  215. uncountable/types/client_config.py +8 -0
  216. uncountable/types/client_config_t.py +26 -0
  217. uncountable/types/curves.py +10 -0
  218. uncountable/types/curves_t.py +51 -0
  219. uncountable/types/entity.py +8 -266
  220. uncountable/types/entity_t.py +393 -0
  221. uncountable/types/experiment_groups.py +8 -0
  222. uncountable/types/experiment_groups_t.py +27 -0
  223. uncountable/types/field_values.py +17 -23
  224. uncountable/types/field_values_t.py +204 -0
  225. uncountable/types/fields.py +8 -0
  226. uncountable/types/fields_t.py +28 -0
  227. uncountable/types/generic_upload.py +15 -0
  228. uncountable/types/generic_upload_t.py +119 -0
  229. uncountable/types/id_source.py +12 -0
  230. uncountable/types/id_source_t.py +68 -0
  231. uncountable/types/identifier.py +11 -0
  232. uncountable/types/identifier_t.py +63 -0
  233. uncountable/types/input_attributes.py +8 -0
  234. uncountable/types/input_attributes_t.py +30 -0
  235. uncountable/types/inputs.py +11 -0
  236. uncountable/types/inputs_t.py +83 -0
  237. uncountable/types/integration_server.py +9 -0
  238. uncountable/types/integration_server_t.py +42 -0
  239. uncountable/types/job_definition.py +27 -0
  240. uncountable/types/job_definition_t.py +260 -0
  241. uncountable/types/outputs.py +8 -0
  242. uncountable/types/outputs_t.py +30 -0
  243. uncountable/types/overrides.py +10 -0
  244. uncountable/types/overrides_t.py +49 -0
  245. uncountable/types/permissions.py +8 -0
  246. uncountable/types/permissions_t.py +46 -0
  247. uncountable/types/phases.py +8 -0
  248. uncountable/types/phases_t.py +27 -0
  249. uncountable/types/post_base.py +8 -0
  250. uncountable/types/post_base_t.py +30 -0
  251. uncountable/types/queued_job.py +16 -0
  252. uncountable/types/queued_job_t.py +123 -0
  253. uncountable/types/recipe_identifiers.py +12 -0
  254. uncountable/types/recipe_identifiers_t.py +76 -0
  255. uncountable/types/recipe_inputs.py +9 -0
  256. uncountable/types/recipe_inputs_t.py +30 -0
  257. uncountable/types/recipe_links.py +4 -44
  258. uncountable/types/recipe_links_t.py +54 -0
  259. uncountable/types/recipe_metadata.py +10 -0
  260. uncountable/types/recipe_metadata_t.py +58 -0
  261. uncountable/types/recipe_output_metadata.py +8 -0
  262. uncountable/types/recipe_output_metadata_t.py +28 -0
  263. uncountable/types/recipe_tags.py +8 -0
  264. uncountable/types/recipe_tags_t.py +27 -0
  265. uncountable/types/recipe_workflow_steps.py +14 -0
  266. uncountable/types/recipe_workflow_steps_t.py +95 -0
  267. uncountable/types/recipes.py +8 -0
  268. uncountable/types/recipes_t.py +25 -0
  269. uncountable/types/response.py +8 -0
  270. uncountable/types/response_t.py +26 -0
  271. uncountable/types/secret_retrieval.py +12 -0
  272. uncountable/types/secret_retrieval_t.py +75 -0
  273. uncountable/types/units.py +8 -0
  274. uncountable/types/units_t.py +27 -0
  275. uncountable/types/users.py +8 -0
  276. uncountable/types/users_t.py +28 -0
  277. uncountable/types/webhook_job.py +9 -0
  278. uncountable/types/webhook_job_t.py +37 -0
  279. uncountable/types/workflows.py +9 -0
  280. uncountable/types/workflows_t.py +39 -0
  281. UncountablePythonSDK-0.0.7.dist-info/METADATA +0 -27
  282. UncountablePythonSDK-0.0.7.dist-info/RECORD +0 -119
  283. examples/recipe-import/importer.py +0 -39
  284. type_spec/external/api/batch/execute_batch.yaml +0 -56
  285. type_spec/external/api/entity/create_entities.yaml +0 -33
  286. type_spec/external/api/entity/create_entity.yaml +0 -39
  287. type_spec/external/api/entity/get_entities_data.yaml +0 -55
  288. type_spec/external/api/entity/list_entities.yaml +0 -62
  289. type_spec/external/api/entity/resolve_entity_ids.yaml +0 -29
  290. type_spec/external/api/entity/set_values.yaml +0 -45
  291. type_spec/external/api/input_groups/get_input_group_names.yaml +0 -29
  292. type_spec/external/api/inputs/create_inputs.yaml +0 -61
  293. type_spec/external/api/inputs/get_input_data.yaml +0 -108
  294. type_spec/external/api/inputs/get_input_names.yaml +0 -38
  295. type_spec/external/api/inputs/get_inputs_data.yaml +0 -95
  296. type_spec/external/api/inputs/set_input_attribute_values.yaml +0 -37
  297. type_spec/external/api/outputs/get_output_data.yaml +0 -103
  298. type_spec/external/api/outputs/get_output_names.yaml +0 -35
  299. type_spec/external/api/outputs/resolve_output_conditions.yaml +0 -50
  300. type_spec/external/api/project/get_projects.yaml +0 -52
  301. type_spec/external/api/project/get_projects_data.yaml +0 -86
  302. type_spec/external/api/recipe_metadata/get_recipe_metadata_data.yaml +0 -41
  303. type_spec/external/api/recipes/create_recipes.yaml +0 -60
  304. type_spec/external/api/recipes/get_curve.yaml +0 -50
  305. type_spec/external/api/recipes/get_recipe_calculations.yaml +0 -49
  306. type_spec/external/api/recipes/get_recipe_links.yaml +0 -26
  307. type_spec/external/api/recipes/get_recipe_names.yaml +0 -29
  308. type_spec/external/api/recipes/get_recipe_output_metadata.yaml +0 -49
  309. type_spec/external/api/recipes/get_recipes_data.yaml +0 -372
  310. type_spec/external/api/recipes/set_recipe_inputs.yaml +0 -36
  311. type_spec/external/api/recipes/set_recipe_outputs.yaml +0 -56
examples/create_entity.py CHANGED
@@ -1,18 +1,23 @@
1
- from uncountable.client import HTTPRequestsClient, AuthDetailsApiKey
2
- from uncountable.types.api import external_create_entities
1
+ from uncountable.core import AuthDetailsOAuth, Client
2
+ from uncountable.core.client import ClientConfig
3
+ from uncountable.types import (
4
+ entity_t,
5
+ field_values_t,
6
+ )
3
7
 
4
-
5
- if __name__ == "__main__":
6
- client = HTTPRequestsClient(
7
- "https://app.uncountable.com", AuthDetailsApiKey(api_id="X", api_secret_key="X")
8
- )
9
- connection = client.connect()
10
- entities = connection.external_create_entities(
11
- external_create_entities.Arguments(
12
- definition_id=1,
13
- entity_type="lab_request",
14
- entities_to_create=[
15
- external_create_entities.EntityToCreate(field_values=None)
16
- ],
17
- )
18
- )
8
+ client = Client(
9
+ base_url="https://app.uncountable.com",
10
+ auth_details=AuthDetailsOAuth(refresh_token="x"),
11
+ config=ClientConfig(allow_insecure_tls=False),
12
+ )
13
+ entities = client.create_entity(
14
+ definition_id=24,
15
+ entity_type=entity_t.EntityType.LAB_REQUEST,
16
+ field_values=[
17
+ field_values_t.FieldRefNameValue(
18
+ field_ref_name="name", value="SDK Lab Request"
19
+ ),
20
+ field_values_t.FieldRefNameValue(field_ref_name="materialFamilyId", value=1),
21
+ ],
22
+ )
23
+ print(entities)
@@ -0,0 +1,26 @@
1
+ import os
2
+ from pprint import pprint
3
+
4
+ from uncountable.core import AuthDetailsApiKey, Client
5
+ from uncountable.types import download_file_t, entity_t, identifier_t
6
+
7
+ client = Client(
8
+ base_url="http://localhost:5000",
9
+ auth_details=AuthDetailsApiKey(
10
+ api_id=os.environ["UNC_API_ID"],
11
+ api_secret_key=os.environ["UNC_API_SECRET_KEY"],
12
+ ),
13
+ )
14
+
15
+ file_query = download_file_t.FileDownloadQueryEntityField(
16
+ entity=entity_t.EntityIdentifier(
17
+ type=entity_t.EntityType.LAB_REQUEST,
18
+ identifier_key=identifier_t.IdentifierKeyId(id=2375),
19
+ ),
20
+ field_key=identifier_t.IdentifierKeyRefName(ref_name="attachments"),
21
+ )
22
+
23
+ downloaded = client.download_files(
24
+ file_query=file_query,
25
+ )
26
+ pprint(downloaded)
@@ -0,0 +1,50 @@
1
+ import os
2
+ from decimal import Decimal
3
+
4
+ from uncountable.core import AsyncBatchProcessor, AuthDetailsApiKey, Client
5
+ from uncountable.types import (
6
+ edit_recipe_inputs_t,
7
+ recipe_workflow_steps_t,
8
+ )
9
+ from uncountable.types.identifier import IdentifierKeyBatchReference, IdentifierKeyId
10
+ from uncountable.types.recipe_identifiers import (
11
+ RecipeIdentifiers,
12
+ )
13
+ from uncountable.types.recipe_inputs import QuantityBasis
14
+
15
+ client = Client(
16
+ base_url=os.environ["UNC_BASE_URL"],
17
+ auth_details=AuthDetailsApiKey(
18
+ api_id=os.environ["UNC_API_ID"], api_secret_key=os.environ["UNC_API_SECRET_KEY"]
19
+ ),
20
+ )
21
+ batch_loader = AsyncBatchProcessor(client=client)
22
+ recipe_identifiers: RecipeIdentifiers = []
23
+ req = batch_loader.create_recipe(
24
+ material_family_id=1, workflow_id=1, identifiers=recipe_identifiers
25
+ )
26
+ created_recipe_reference = req.batch_reference
27
+ edits: list[edit_recipe_inputs_t.RecipeInputEdit] = []
28
+ edits.append(
29
+ edit_recipe_inputs_t.RecipeInputEditAddInput(
30
+ quantity_basis=QuantityBasis.MASS,
31
+ ingredient_key=IdentifierKeyId(id=1),
32
+ value_numeric=Decimal("56.7"),
33
+ )
34
+ )
35
+ edits.append(
36
+ edit_recipe_inputs_t.RecipeInputEditChangeBasisViewed(
37
+ quantity_basis=QuantityBasis.VOLUME, ingredient_key=IdentifierKeyId(id=1)
38
+ )
39
+ )
40
+ edits.append(
41
+ edit_recipe_inputs_t.RecipeInputEditAddInstructions(
42
+ instructions="Mix for 3 minutes"
43
+ )
44
+ )
45
+ batch_loader.edit_recipe_inputs(
46
+ recipe_key=IdentifierKeyBatchReference(reference=created_recipe_reference),
47
+ edits=edits,
48
+ recipe_workflow_step_identifier=recipe_workflow_steps_t.RecipeWorkflowStepIdentifierDefault(),
49
+ )
50
+ job_id = batch_loader.send()
@@ -0,0 +1,18 @@
1
+ from uncountable.integration.job import CronJob, JobArguments, register_job
2
+ from uncountable.types import entity_t
3
+ from uncountable.types.job_definition_t import JobResult
4
+
5
+
6
+ @register_job
7
+ class MyCronJob(CronJob):
8
+ def run(self, args: JobArguments) -> JobResult:
9
+ matfam = args.client.get_entities_data(
10
+ entity_ids=[1],
11
+ entity_type=entity_t.EntityType.MATERIAL_FAMILY,
12
+ ).entity_details[0]
13
+ name = None
14
+ for field_val in matfam.field_values:
15
+ if field_val.field_ref_name == "name":
16
+ name = field_val.value
17
+ args.logger.log_info(f"material family found with name: {name}")
18
+ return JobResult(success=True)
@@ -0,0 +1,15 @@
1
+ import typing
2
+
3
+ from uncountable.integration.job import JobArguments, WebhookJob, register_job
4
+ from uncountable.types.job_definition_t import JobResult
5
+
6
+
7
+ @register_job
8
+ class WebhookExample(WebhookJob):
9
+ @property
10
+ def payload_type(self) -> typing.Any:
11
+ return super().payload_type
12
+
13
+ def run(self, args: JobArguments, payload: typing.Any) -> JobResult:
14
+ args.logger.log_info(f"webhook invoked with payload: {payload}")
15
+ return JobResult(success=True)
@@ -0,0 +1,43 @@
1
+ base_url: http://host.docker.internal:5000
2
+ auth_retrieval:
3
+ type: basic
4
+ api_id_secret:
5
+ type: env
6
+ env_key: API_ID
7
+ api_key_secret:
8
+ type: env
9
+ env_key: API_SECRET_KEY
10
+ jobs:
11
+ - id: example_cron
12
+ enabled: true
13
+ type: cron
14
+ name: MyCron - Example
15
+ cron_spec: "* * * * *"
16
+ executor:
17
+ type: script
18
+ import_path: example_cron
19
+ logging_settings:
20
+ enabled: true
21
+ share_with_user_groups:
22
+ - admin:
23
+ type: ref_name
24
+ ref_name: admin
25
+
26
+ - id: example_wh1
27
+ type: webhook
28
+ name: Webhook 1
29
+ signature_key_secret:
30
+ type: env
31
+ env_key: WH1_SIGNATURE_KEY
32
+ executor:
33
+ type: script
34
+ import_path: example_wh
35
+ - id: example_wh2
36
+ type: webhook
37
+ name: Webhook 2
38
+ signature_key_secret:
39
+ type: env
40
+ env_key: WH2_SIGNATURE_KEY
41
+ executor:
42
+ type: script
43
+ import_path: example_wh
@@ -0,0 +1,224 @@
1
+ [project]
2
+ # begin_project_name
3
+ # DO NOT MODIFY
4
+ name = "integration-server-testing"
5
+ # end_project_name
6
+ dynamic = ["version"]
7
+ dependencies = [
8
+ "mypy == 1.*",
9
+ "ruff == 0.*",
10
+ "openpyxl == 3.*",
11
+ "more_itertools == 10.*",
12
+ "types-paramiko ==3.5.0.20240918",
13
+ "types-openpyxl == 3.*",
14
+ "types-pysftp == 0.*",
15
+ "types-pytz == 2024.*",
16
+ "types-requests == 2.*",
17
+ "types-simplejson == 3.*",
18
+ "pandas-stubs",
19
+ "xlrd == 2.*"
20
+ ]
21
+
22
+ [tool.mypy]
23
+ strict = true
24
+ show_error_codes = true
25
+ check_untyped_defs = true
26
+ disallow_any_generics = false
27
+ disallow_incomplete_defs = true
28
+ disallow_subclassing_any = false
29
+ disallow_untyped_calls = false
30
+ disallow_untyped_decorators = true
31
+ disallow_untyped_defs = true
32
+ implicit_reexport = true
33
+ no_implicit_optional = true
34
+ no_implicit_reexport = true
35
+ strict_equality = true
36
+ warn_redundant_casts = true
37
+ warn_unused_ignores = false
38
+ warn_no_return = true
39
+ warn_return_any = true
40
+ local_partial_types = true
41
+ warn_unreachable = false
42
+ warn_unused_configs = true
43
+ exclude = [
44
+ "env"
45
+ ]
46
+
47
+ [tool.ruff]
48
+ preview = true
49
+ lint.select = ['ALL']
50
+ lint.ignore = [
51
+ "A001", # Variable shadows python built-in. Nice to have
52
+ "A002", # Variable shadows python built-in. Nice to have
53
+ "A003", # Class attribute shadows python built-in. Nice to have
54
+ "ANN", # Missing type annotations. Skip in favor of mypy
55
+ "ARG", # Unused arguments. Skip
56
+ "B006", # Do not use mutable data structures for argument defaults. Priority.
57
+ "B023", # Function doesn't bind loop variable. Priority
58
+ "B904", # Exception handling raise from. Skip
59
+ "BLE", # Blind exception. Should Add
60
+ "C401", # Unecessary generator. Skip
61
+ "C403", # Unnecessary list comprehension. Skip
62
+ "C405", # Unnecessary list. rewrite as set. Skip
63
+ "C408", # Unnecessary dict call. Skip
64
+ "C416", # Unnecessary list comprehension. Skip
65
+ "C417", # Unnecessary use of map. Should Add
66
+ "COM812", # Trailing comma. Too opinionated. Skip
67
+ "CPY", # Copyright notices. Skip
68
+ "D", # Docstrings. Skip
69
+ "DOC", # Docstrings. Skip
70
+ "DTZ001", # Missing tzinfo. Should add
71
+ "DTZ002", # Use of today instead of now. Should add
72
+ "DTZ003", # Use of utcnow not allowed. Should add
73
+ "DTZ004", # Use of utcfrom timestamp not allowed. Should add
74
+ "DTZ005", # now without tz not allowed. Should add
75
+ "DTZ006", # use of from timestamp without tz not allowed. Should add
76
+ "DTZ007", # replace timezone after strptime. Should add
77
+ "DTZ011", # Use of today not allowed. Should add
78
+ "E501", # Line length. Skip
79
+ "EM101", # exception must not use a string literal. skip
80
+ "EM102", # exception must not use an f string. skip
81
+ "EM103", # exception must not use a format string. skip
82
+ "ERA", # Rules around commented code. Skip
83
+ "EXE002", # Executable files need shebang. Skip
84
+ "F841", # TEMPORARY DISABLE TO CLEAR UP BUILD. Unused variables. Prioritize.
85
+ "FBT001", # boolean positional args. Skip
86
+ "FBT002", # boolean default positional. Skip
87
+ "FBT003", # boolean positional. Skip
88
+ "FIX001", # Fixme comment. Skip
89
+ "FIX002", # todo comment. Skip
90
+ "FIX004", # hack comment. Skip
91
+ "FLY002", # Use F-String instead of join. Should Add
92
+ "FURB101", # replace read function. Skip
93
+ "FURB103", # opinionated file opening. Skip
94
+ "FURB113", # extend vs append. Skip
95
+ "FURB118", # opinionated rules about lambdas. skip
96
+ "FURB140", # starmap vs generator. Skip
97
+ "G004", # avoid f strings in log statements. Should add
98
+ "INP001", # implicit imports. use __init__. Skip
99
+ "ISC001", # rule conflicts with formatter. Skip
100
+ "ISC003", # Explicitly concatenated string should be implicitly concatenated. Skip
101
+ "N801", # class name case. Skip
102
+ "N802", # function name lower case. Skip
103
+ "N803", # argument name lower cas. Skip
104
+ "N804", # cls_ first variable. Should add
105
+ "N806", # lower case variable. Skip
106
+ "N812", # case consistency on import. Skip
107
+ "N815", # mix casing, skip
108
+ "N817", # don't use acronym. Should add
109
+ "N818", # exception override should be named with error. should add
110
+ "N999", # module naming. Should add
111
+ "PD002", # avoid inplace. Skip
112
+ "PD004", # .notna. Should add
113
+ "PD010", # .pivottable. Should add
114
+ "PD011", # use .to_numpy. Skip
115
+ "PD015", # use .merge. Should add
116
+ "PD901", # avoid generic df name. Skip
117
+ "PERF203", # avoid try except in loop. Skip
118
+ "PERF401", # use list comprehension. Skip
119
+ "PERF402", # use list.copy. Skip
120
+ "PERF403", # use dictionary comprehension. Skip
121
+ "S307", # avoid eval. Should add
122
+ "PGH003", # ignore specific rules. Skip
123
+ "PGH004", # ignore specific rules. Skip
124
+ "PLC0414", # import alias doesn't rename. Skip
125
+ "PLC0415", # import at top level. Skip
126
+ "PLC1901", # falsy empty string. Skip
127
+ "PLR0124", # self comparison. Skip
128
+ "PLR0402", # alias import. Skip
129
+ "PLR0911", # too many return statements. Skip
130
+ "PLR0912", # too many branches. Skip
131
+ "PLR0913", # too many arguments. Skip
132
+ "PLR0915", # too many statements. Skip
133
+ "PLR0916", # too many expressions. Skip
134
+ "PLR0917", # too many positional arguments. Skip
135
+ "PLR1714", # multiple comparisons. Skip
136
+ "PLR2004", # magic values. skip
137
+ "PLR5501", # use elif instead of else if. skip
138
+ "PLR6104", # use augmented assignment. confuses strings, wait for fix. skip
139
+ "PLR6201", # use a set literal for membership test. skip
140
+ "PLR6301", # method inference. skip
141
+ "PLW0108", # inline lambda. skip
142
+ "PLW0120", # else without break. skip
143
+ "PLW0602", # global withotu assignment. consider
144
+ "PLW0603", # discourage global. skip
145
+ "PLW1510", # subprocess run without check arg. skip
146
+ "PLW1514", # open without encoding. skip
147
+ "PLW2901", # for loop variable overwritten. skip
148
+ "PT001", # change ficture decorator. should add
149
+ "PT011", # use more specific raises decorator. should add
150
+ "PT018", # break down complex assertsions. skip
151
+ "PTH", # enforces new python path library usage. skip
152
+ "PYI030", # turn union of literals to just literal with multiple. should add
153
+ "PYI041", # use float instead of int | float. should remove floats
154
+ "RET", # enforces return semantics. break up and re-evaluate later
155
+ "RSE102", # unnecessary parens on exception. skip.
156
+ "RUF001", # ambiguous characters. skip
157
+ "RUF005", # prefer star to concatenation. skip
158
+ "RUF010", # use explicit conversion flag. skip
159
+ "RUF012", # type with class var. should add
160
+ "RUF015", # prefer next for single element. skip
161
+ "RUF100", # unused noqa. should add
162
+ "S", # should split up
163
+ "S101", # allow asserts. skip
164
+ "SIM102", # use single if instead of nested. skip
165
+ "SIM105", # use exception supress. skip
166
+ "SIM108", # use ternary. skip
167
+ "SIM110", # use any/all. should add
168
+ "SIM112", # case of env variable. skip
169
+ "SIM114", # combine if branches. skip
170
+ "SIM115", # use context handler for files. should add
171
+ "SIM117", # use single with. should add
172
+ "SLF001", # private member access. should add
173
+ "SLOT000", # subclass of string should define slot. should add
174
+ "T201", # print. skip
175
+ "T203", # pprint. skip
176
+ "TCH001", # move application import into type checking. skip
177
+ "TCH002", # move third party import into type checking. consider
178
+ "TCH003", # move library into type checking lbock. consider
179
+ "TD", # todo comments. skip
180
+ "TID252", # eliminate relative imports from parents. skip
181
+ "TRY002", # create your own exception. skip
182
+ "TRY003", # avoid long messages. skip
183
+ "TRY004", # prefer type error. should add
184
+ "TRY201", # raise without exception name. consider
185
+ "TRY300", # move statement to else block. consider
186
+ "TRY301", # move raise to inner function. skip
187
+ "UP007", # use union for type annotations. should add
188
+ "UP013", # eliminate typeddict. should add
189
+ "UP031", # use format specifiers. should add
190
+ "B905", # zip strict. we have a deprecated pattern here, consider
191
+ "UP036", # outdated version block, consider
192
+ "RUF007", # prefer itertools.pairwise over zip, consider
193
+ "RUF022", # __all__ is not sorted. skip due to isort complication
194
+ "UP017", # use datetime.UTC, TODO add back in
195
+ "UP035", # replacing List with list, TODO add back in
196
+ "UP038", # isinstance X | Y instead of (X, Y), TODO add back in
197
+ # ## FROM RUFF UPGRADE
198
+ "PLC2701", # private name imports. should add
199
+ "PLR1702", # too many nested blocks -- add with config. skip
200
+ "C420", # unnecessary dict comprehension. skip
201
+ "SIM113", # index variable should use enumerate. should add.
202
+ "PLR0914" # too many local variables. configure below. should add with config
203
+ ]
204
+ line-length = 90
205
+ target-version = "py311"
206
+
207
+ exclude = [
208
+ 'env'
209
+ ]
210
+ [tool.ruff.lint.per-file-ignores]
211
+ "__init__.py" = ["F401"]
212
+
213
+ [tool.ruff.lint.isort]
214
+ split-on-trailing-comma = true
215
+
216
+ [tool.ruff.lint.mccabe]
217
+ max-complexity = 130 # goal would be to bring this down to ~50 or so
218
+
219
+ [tool.ruff.lint.pylint]
220
+ max-locals=50
221
+
222
+ [tool.setuptools]
223
+ py-modules = []
224
+
@@ -0,0 +1,26 @@
1
+ import os
2
+
3
+ from uncountable.core import AuthDetailsApiKey, Client, MediaFileUpload
4
+ from uncountable.types import generic_upload_t
5
+ from uncountable.types.identifier_t import IdentifierKeyId
6
+
7
+ client = Client(
8
+ base_url="http://localhost:5000",
9
+ auth_details=AuthDetailsApiKey(
10
+ api_id=os.environ["UNC_API_ID"],
11
+ api_secret_key=os.environ["UNC_API_SECRET_KEY"],
12
+ ),
13
+ )
14
+ uploaded_file = client.upload_files(
15
+ file_uploads=[
16
+ MediaFileUpload(path="~/Downloads/my_file_to_upload.csv"),
17
+ ]
18
+ )[0]
19
+
20
+ client.invoke_uploader(
21
+ file_id=uploaded_file.file_id,
22
+ uploader_key=IdentifierKeyId(id=48),
23
+ destination=generic_upload_t.UploadDestinationMaterialFamily(
24
+ material_family_key=IdentifierKeyId(id=7)
25
+ ),
26
+ )
@@ -0,0 +1,40 @@
1
+ import os
2
+
3
+ from uncountable.core import (
4
+ AsyncBatchProcessor,
5
+ AuthDetailsApiKey,
6
+ Client,
7
+ MediaFileUpload,
8
+ )
9
+ from uncountable.types import recipe_metadata_t
10
+ from uncountable.types.identifier_t import IdentifierKeyBatchReference
11
+
12
+ client = Client(
13
+ base_url="http://localhost:5000",
14
+ auth_details=AuthDetailsApiKey(
15
+ api_id=os.environ["UNC_API_ID"],
16
+ api_secret_key=os.environ["UNC_API_SECRET_KEY"],
17
+ ),
18
+ )
19
+ uploaded_file = client.upload_files(
20
+ file_uploads=[
21
+ MediaFileUpload(path="Downloads/my_file_to_upload.csv"),
22
+ ]
23
+ )[0]
24
+
25
+ batch_processor = AsyncBatchProcessor(client=client)
26
+
27
+ recipe_batch_identifier = batch_processor.create_recipe(
28
+ material_family_id=1, workflow_id=1
29
+ ).batch_reference
30
+
31
+ batch_processor.set_recipe_metadata(
32
+ recipe_key=IdentifierKeyBatchReference(reference=recipe_batch_identifier),
33
+ recipe_metadata=[
34
+ recipe_metadata_t.MetadataValue(
35
+ metadata_id=102, value_file_ids=[uploaded_file.file_id]
36
+ )
37
+ ],
38
+ )
39
+
40
+ batch_processor.send()
@@ -0,0 +1,26 @@
1
+ import os
2
+ from pathlib import Path
3
+
4
+ import uncountable.types.api.recipes.set_recipe_output_file as set_recipe_output_file_t
5
+ from uncountable.core import AuthDetailsApiKey, Client, MediaFileUpload
6
+
7
+ client = Client(
8
+ base_url="http://localhost:5000",
9
+ auth_details=AuthDetailsApiKey(
10
+ api_id=os.environ["UNC_API_ID"],
11
+ api_secret_key=os.environ["UNC_API_SECRET_KEY"],
12
+ ),
13
+ )
14
+ uploaded_file = client.upload_files(
15
+ file_uploads=[
16
+ MediaFileUpload(
17
+ path=str((Path.home() / "Downloads" / "my_file_to_upload.csv").absolute())
18
+ ),
19
+ ]
20
+ )[0]
21
+
22
+ client.set_recipe_output_file(
23
+ output_file_data=set_recipe_output_file_t.RecipeOutputFileValue(
24
+ recipe_id=58070, output_id=148, file_id=uploaded_file.file_id, experiment_num=1
25
+ )
26
+ )
@@ -0,0 +1,18 @@
1
+ import os
2
+ from pprint import pprint
3
+
4
+ from uncountable.core import AuthDetailsApiKey, Client, MediaFileUpload
5
+
6
+ client = Client(
7
+ base_url="http://localhost:5000",
8
+ auth_details=AuthDetailsApiKey(
9
+ api_id=os.environ["UNC_API_ID"],
10
+ api_secret_key=os.environ["UNC_API_SECRET_KEY"],
11
+ ),
12
+ )
13
+ uploaded = client.upload_files(
14
+ file_uploads=[
15
+ MediaFileUpload(path="Downloads/file"),
16
+ ]
17
+ )
18
+ pprint(uploaded)
@@ -1,6 +1,11 @@
1
1
  from .argument_parser import CachedParser as CachedParser
2
+ from .argument_parser import ParserError as ParserError
3
+ from .argument_parser import ParserExtraFieldsError as ParserExtraFieldsError
4
+ from .argument_parser import ParserFunction as ParserFunction
2
5
  from .argument_parser import ParserOptions as ParserOptions
6
+ from .argument_parser import SourceEncoding as SourceEncoding
3
7
  from .argument_parser import build_parser as build_parser
8
+ from .argument_parser import is_union as is_union
4
9
  from .case_convert import camel_to_snake_case as camel_to_snake_case
5
10
  from .case_convert import kebab_to_pascal_case as kebab_to_pascal_case
6
11
  from .case_convert import snake_to_camel_case as snake_to_camel_case
@@ -3,14 +3,9 @@ import typing
3
3
  from enum import Enum
4
4
 
5
5
 
6
- def is_string_enum_class(object_type: typing.Type[typing.Any]) -> bool:
6
+ def is_string_enum_class(object_type: type[typing.Any]) -> bool:
7
7
  return (
8
8
  inspect.isclass(object_type)
9
9
  and issubclass(object_type, Enum)
10
10
  and issubclass(object_type, str)
11
11
  )
12
-
13
-
14
- def is_string_enum_instance(obj: typing.Any) -> bool:
15
- object_type = type(obj)
16
- return is_string_enum_class(object_type)