streamlit-octostar-utils 0.5.0.dev2__tar.gz → 0.5.0.dev4__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (45) hide show
  1. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/PKG-INFO +1 -1
  2. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/pyproject.toml +1 -1
  3. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/nifi.py +107 -17
  4. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/LICENSE +0 -0
  5. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/README.md +0 -0
  6. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/__init__.py +0 -0
  7. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/__init__.py +0 -0
  8. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/celery.py +0 -0
  9. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/contents.py +0 -0
  10. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/fastapi.py +0 -0
  11. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parallelism.py +0 -0
  12. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/__init__.py +0 -0
  13. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/combine_fields.py +0 -0
  14. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/entities_parser.py +0 -0
  15. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/generics.py +0 -0
  16. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/info.py +0 -0
  17. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/linkchart_functions.py +0 -0
  18. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/matches.py +0 -0
  19. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/parameters.py +0 -0
  20. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/rules.py +0 -0
  21. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/api_crafter/parser/signals.py +0 -0
  22. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/__init__.py +0 -0
  23. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/dict.py +0 -0
  24. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/filetypes.py +0 -0
  25. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/threading/__init__.py +0 -0
  26. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/threading/key_queue.py +0 -0
  27. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/core/timestamp.py +0 -0
  28. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/nlp/__init__.py +0 -0
  29. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/nlp/custom_recognizers.py +0 -0
  30. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/nlp/language.py +0 -0
  31. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/nlp/ner.py +0 -0
  32. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/octostar/__init__.py +0 -0
  33. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/octostar/client.py +0 -0
  34. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/octostar/context.py +0 -0
  35. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/octostar/permissions.py +0 -0
  36. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/ontology/__init__.py +0 -0
  37. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/ontology/inheritance.py +0 -0
  38. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/ontology/relationships.py +0 -0
  39. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/ontology/validation.py +0 -0
  40. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/style/__init__.py +0 -0
  41. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/style/common.py +0 -0
  42. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/threading/__init__.py +0 -0
  43. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/threading/async_task_manager.py +0 -0
  44. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/threading/session_callback_manager.py +0 -0
  45. {streamlit_octostar_utils-0.5.0.dev2 → streamlit_octostar_utils-0.5.0.dev4}/streamlit_octostar_utils/threading/session_state_hot_swapper.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: streamlit-octostar-utils
3
- Version: 0.5.0.dev2
3
+ Version: 0.5.0.dev4
4
4
  Summary:
5
5
  License: MIT
6
6
  License-File: LICENSE
@@ -5,7 +5,7 @@ include = '\.pyi?$'
5
5
 
6
6
  [tool.poetry]
7
7
  name = "streamlit-octostar-utils"
8
- version = "0.5.0-dev.2"
8
+ version = "0.5.0-dev.4"
9
9
  description = ""
10
10
  license = "MIT"
11
11
  authors = ["Octostar"]
@@ -1,4 +1,5 @@
1
1
  from copy import deepcopy
2
+ import hashlib
2
3
  import uuid
3
4
  from functools import wraps
4
5
  from contextlib import contextmanager
@@ -72,7 +73,7 @@ class NifiEntityModel(BaseModel):
72
73
  relationships: List[str]
73
74
  label_keys: List[str]
74
75
 
75
- jwt: str
76
+ jwt: Optional[str] = None
76
77
  ontology_name: str
77
78
  ontology_info: OntologyInfoModel
78
79
  entity_timestamp: Optional[str]
@@ -388,6 +389,66 @@ class NifiFragmenter(object):
388
389
  ]
389
390
  return parent_fn(tree, child_results)
390
391
 
392
+ _FRAGMENTS_NAMESPACE = b"octostar.pipeline.fragments"
393
+
394
+ @staticmethod
395
+ def _create_deterministic_uuid(namespace: bytes, data: bytes):
396
+ return str(
397
+ uuid.uuid5(
398
+ uuid.UUID(bytes=hashlib.md5(namespace).digest()),
399
+ hashlib.md5(data).hexdigest(),
400
+ )
401
+ )
402
+
403
+ @staticmethod
404
+ def create_fragment_uuid(os_entity_uid, stable_fragment_identifier, processor_name, os_workspace):
405
+ """Create a deterministic UUID for a fragment.
406
+
407
+ The UUID is stable across re-runs for the same entity, fragment
408
+ identifier, processor, and workspace -- enabling idempotent fragment
409
+ creation while ensuring uniqueness across workspaces.
410
+
411
+ Args:
412
+ os_entity_uid: The parent entity UID.
413
+ stable_fragment_identifier: A unique and stable identifier within
414
+ the fragmentation (e.g. page range, keyframe number, face index).
415
+ processor_name: Name of the NiFi processor.
416
+ os_workspace: The workspace UID the fragment will be written to.
417
+
418
+ Returns:
419
+ A deterministic UUID string.
420
+ """
421
+ return NifiFragmenter._create_deterministic_uuid(
422
+ NifiFragmenter._FRAGMENTS_NAMESPACE,
423
+ (processor_name + "::" + os_workspace + "::" + os_entity_uid + "::" + stable_fragment_identifier).encode("utf-8"),
424
+ )
425
+
426
+ @staticmethod
427
+ def resolve_source_entity_uid(entity, fragment_root_source=None) -> str:
428
+ """Resolve the source_entity_uid to use for child fragments.
429
+
430
+ When fragment_root_source is set (a fragment name or stack index),
431
+ the UID is looked up via entity.get_fragment_root_uid() -- this is
432
+ necessary when the current entity is a clone that may not be persisted.
433
+ Otherwise falls back to the entity's own UID.
434
+
435
+ Args:
436
+ entity: A NifiEntity or NifiEntityProxy.
437
+ fragment_root_source: None to use the entity's own UID, an int
438
+ to index into the fragments stack, or a string fragmenter
439
+ keylist name.
440
+
441
+ Returns:
442
+ The resolved source entity UID string.
443
+ """
444
+ if fragment_root_source is None:
445
+ return entity.record["os_entity_uid"]
446
+ try:
447
+ idx = int(fragment_root_source)
448
+ return entity.get_fragment_root_uid(idx)
449
+ except (ValueError, TypeError):
450
+ return entity.get_fragment_root_uid(fragment_root_source)
451
+
391
452
 
392
453
  class NifiEntityBatch(object):
393
454
  def __init__(self, entities, config, config_key):
@@ -1125,6 +1186,48 @@ class NifiEntity(object):
1125
1186
  self.children.append(child_entity_proxy)
1126
1187
  return child_entity_proxy
1127
1188
 
1189
+ def prepare_contents(
1190
+ self,
1191
+ file: Union[Contents, bytes],
1192
+ entity_type: str,
1193
+ filetype: str,
1194
+ temp_blob_key: Optional[str] = None,
1195
+ ) -> Contents:
1196
+ """Build a Contents object from raw bytes, mirroring the same
1197
+ MemoryContents / TemporaryAttachmentContents logic used internally.
1198
+
1199
+ Thread-safe: each call creates independent objects keyed by
1200
+ *temp_blob_key* (for large blobs) and does not mutate the entity tree.
1201
+ The returned Contents can later be passed to ``add_child_fragment``
1202
+ or ``add_child_file`` to skip the redundant bytes→Contents conversion.
1203
+
1204
+ Args:
1205
+ file: A Contents object (returned as-is) or raw bytes.
1206
+ entity_type: Concept / entity type for the contents metadata.
1207
+ filetype: MIME type for the contents metadata.
1208
+ temp_blob_key: Unique key for the temporary blob when the data
1209
+ exceeds the in-memory threshold. Defaults to a random UUID.
1210
+ """
1211
+ if isinstance(file, Contents):
1212
+ return file
1213
+ if len(file) <= MAX_IN_MEMORY_SIZE_BYTES:
1214
+ return MemoryContents(
1215
+ entity_type=entity_type,
1216
+ filetype=filetype,
1217
+ initial_data=file,
1218
+ )
1219
+ temp_filename = temp_blob_key or f"tmp_{uuid.uuid4()}"
1220
+ temp_contents = TemporaryAttachmentContents(
1221
+ entity_type=entity_type,
1222
+ filetype=filetype,
1223
+ filename=temp_filename,
1224
+ client=self.context.client,
1225
+ )
1226
+ temp_contents.truncate(0)
1227
+ temp_contents.write(file)
1228
+ temp_contents.flush()
1229
+ return temp_contents
1230
+
1128
1231
  def _add_child_attachable(
1129
1232
  self,
1130
1233
  os_workspace,
@@ -1152,24 +1255,11 @@ class NifiEntity(object):
1152
1255
  )
1153
1256
  if isinstance(file, Contents):
1154
1257
  child_entity._contents = file
1155
- elif len(file) <= MAX_IN_MEMORY_SIZE_BYTES:
1156
- child_entity._contents = MemoryContents(
1157
- entity_type=os_entity_type,
1158
- filetype=filetype,
1159
- initial_data=file,
1160
- )
1161
1258
  else:
1162
- temp_filename = f"tmp_{child_entity.record['os_entity_uid']}"
1163
- temp_contents = TemporaryAttachmentContents(
1164
- entity_type=os_entity_type,
1165
- filetype=filetype,
1166
- filename=temp_filename,
1167
- client=self.context.client,
1259
+ child_entity._contents = self.prepare_contents(
1260
+ file, os_entity_type, filetype,
1261
+ temp_blob_key=f"tmp_{child_entity.record['os_entity_uid']}",
1168
1262
  )
1169
- temp_contents.truncate(0)
1170
- temp_contents.write(file)
1171
- temp_contents.flush()
1172
- child_entity._contents = temp_contents
1173
1263
  child_entity.sync_params = recursive_update_dict(
1174
1264
  child_entity.sync_params,
1175
1265
  {NifiContextManager.SyncFlag.WRITE_CONTENTS: True},