unique-orchestrator 2026.28.0.dev0__tar.gz → 2026.28.0.dev2__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 (32) hide show
  1. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/PKG-INFO +2 -2
  2. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/pyproject.toml +2 -2
  3. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/config.py +3 -5
  4. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_config_services_validators.py +3 -5
  5. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_loop_debug_params.py +1 -0
  6. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/unique_ai.py +36 -2
  7. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/unique_ai_builder.py +1 -4
  8. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/README.md +0 -0
  9. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/__init__.py +0 -0
  10. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/_builders/__init__.py +0 -0
  11. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/_builders/inject_tool_reminders.py +0 -0
  12. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/_builders/loop_iteration_runner.py +0 -0
  13. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/_builders/open_file_setup.py +0 -0
  14. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/_builders/skill_setup.py +0 -0
  15. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/prompts/generic_reference_prompt.jinja2 +0 -0
  16. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/prompts/system_prompt.jinja2 +0 -0
  17. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/prompts/user_message_prompt.jinja2 +0 -0
  18. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/settings.py +0 -0
  19. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_agent_config_rjsf_ui_schema.py +0 -0
  20. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_build_loop_iteration_runner.py +0 -0
  21. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_qwen_max_loop_iterations.py +0 -0
  22. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_register_code_interpreter_postprocessors.py +0 -0
  23. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_async_modify.py +0 -0
  24. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_execution_timing.py +0 -0
  25. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_get_filtered_user_metadata.py +0 -0
  26. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_log_tool_calls.py +0 -0
  27. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_persist_tool_calls.py +0 -0
  28. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_plan_or_execute_include.py +0 -0
  29. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_reference_order.py +0 -0
  30. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_unique_ai_render_system_prompt_user_instructions.py +0 -0
  31. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/tests/test_utils_resolve_other_options.py +0 -0
  32. {unique_orchestrator-2026.28.0.dev0 → unique_orchestrator-2026.28.0.dev2}/unique_orchestrator/utils.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: unique-orchestrator
3
- Version: 2026.28.0.dev0
3
+ Version: 2026.28.0.dev2
4
4
  Summary:
5
5
  Author: Andreas Hauri
6
6
  Author-email: Andreas Hauri <andreas.hauri@unique.ai>
@@ -17,7 +17,7 @@ Requires-Dist: unique-deep-research>=2026.28.0.dev0,<2026.28.0rc0
17
17
  Requires-Dist: unique-web-search>=2026.28.0.dev0,<2026.28.0rc0
18
18
  Requires-Dist: unique-swot>=2026.28.0.dev0,<2026.28.0rc0
19
19
  Requires-Dist: unique-skill-tool>=2026.28.0.dev0,<2026.28.0rc0
20
- Requires-Dist: unique-user-memory>=2026.28.0.dev0,<2026.28.0rc0
20
+ Requires-Dist: unique-user-memory>=2026.28.0.dev1,<2026.28.0rc0
21
21
  Requires-Dist: openai>=1.109.1,<3
22
22
  Requires-Python: >=3.12, <4
23
23
  Description-Content-Type: text/markdown
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "unique_orchestrator"
3
- version = "2026.28.0.dev0"
3
+ version = "2026.28.0.dev2"
4
4
  description = ""
5
5
  readme = "README.md"
6
6
  license = { text = "Proprietary" }
@@ -21,7 +21,7 @@ dependencies = [
21
21
  "unique-web-search>=2026.28.0.dev0,<2026.28.0rc0",
22
22
  "unique-swot>=2026.28.0.dev0,<2026.28.0rc0",
23
23
  "unique-skill-tool>=2026.28.0.dev0,<2026.28.0rc0",
24
- "unique-user-memory>=2026.28.0.dev0,<2026.28.0rc0",
24
+ "unique-user-memory>=2026.28.0.dev1,<2026.28.0rc0",
25
25
  "openai>=1.109.1,<3",
26
26
  ]
27
27
 
@@ -292,11 +292,9 @@ class UniqueAIServices(BaseToolConfig):
292
292
  ToolProgressReporterConfig()
293
293
  )
294
294
 
295
- user_memory_config: Annotated[
296
- UserMemoryConfig, RJSFMetaTag.SpecialWidget.hidden()
297
- ] = Field(
298
- title="User Memory",
299
- description="Configuration for persistent user memory.",
295
+ user_memory_config: UserMemoryConfig = Field(
296
+ title="User Memory (Experimental)",
297
+ description="Configuration for persistent user memory. Experimental feature - requires activation via feature flag.",
300
298
  default_factory=UserMemoryConfig,
301
299
  )
302
300
 
@@ -467,18 +467,17 @@ class TestUniqueAIConfigInjectTodoToolValidator:
467
467
 
468
468
 
469
469
  class TestUniqueAIConfigUserMemory:
470
- def test_user_memory_config_defaults_to_disabled(self):
470
+ def test_user_memory_config_defaults(self):
471
471
  config = UniqueAIConfig()
472
472
 
473
473
  assert isinstance(config.agent.services.user_memory_config, UserMemoryConfig)
474
- assert config.agent.services.user_memory_config.enabled is False
474
+ assert config.agent.services.user_memory_config.max_tokens == 2000
475
475
 
476
- def test_user_memory_config_parses_enabled_payload(self):
476
+ def test_user_memory_config_parses_payload(self):
477
477
  config = UniqueAIConfig(
478
478
  agent={
479
479
  "services": {
480
480
  "user_memory_config": {
481
- "enabled": True,
482
481
  "max_tokens": 1500,
483
482
  }
484
483
  }
@@ -486,7 +485,6 @@ class TestUniqueAIConfigUserMemory:
486
485
  )
487
486
 
488
487
  memory_config = config.agent.services.user_memory_config
489
- assert memory_config.enabled is True
490
488
  assert memory_config.max_tokens == 1500
491
489
 
492
490
  def test_allow_user_memory_defaults_to_false(self):
@@ -352,6 +352,7 @@ class TestRunLoopDebugParams:
352
352
  call.args[0] for call in ua._debug_info_manager.add.call_args_list
353
353
  ]
354
354
  assert "loop_params" in keys_written
355
+ assert "skills" in keys_written
355
356
 
356
357
  @pytest.mark.ai
357
358
  @pytest.mark.asyncio
@@ -6,6 +6,7 @@ from typing import Any, cast, overload
6
6
 
7
7
  import jinja2
8
8
  from typing_extensions import deprecated
9
+ from unique_skill_tool.service import SkillTool
9
10
  from unique_toolkit.agentic.debug_info_manager.debug_info_manager import (
10
11
  DebugInfoManager,
11
12
  )
@@ -35,7 +36,7 @@ from unique_toolkit.agentic.tools.tool_manager import (
35
36
  SafeTaskExecutor,
36
37
  ToolManager,
37
38
  )
38
- from unique_toolkit.app.schemas import ChatEvent, McpServer
39
+ from unique_toolkit.app.schemas import ChatEvent, McpServer, SkillReference
39
40
  from unique_toolkit.chat.cancellation import CancellationEvent
40
41
  from unique_toolkit.chat.service import ChatService
41
42
  from unique_toolkit.content import Content
@@ -143,7 +144,9 @@ class UniqueAI:
143
144
  self._content_service = content_service
144
145
  self._uploaded_documents = uploaded_documents or []
145
146
  self._user_memory_text = user_memory_text
146
- self._skill_choices = getattr(event.payload, "skill_choices", [])
147
+ self._skill_choices: list[SkillReference] = getattr(
148
+ event.payload, "skill_choices", []
149
+ )
147
150
 
148
151
  self._debug_info_manager = debug_info_manager
149
152
  self._reference_manager = reference_manager
@@ -313,6 +316,10 @@ class UniqueAI:
313
316
  },
314
317
  )
315
318
  self._debug_info_manager.add("loop_params", self._loop_debug_params)
319
+ self._debug_info_manager.add(
320
+ "skills",
321
+ self._get_activated_skills_debug_info(),
322
+ )
316
323
 
317
324
  tool_names = [
318
325
  tool["name"] for tool in self._debug_info_manager.get()["tools"]
@@ -349,6 +356,33 @@ class UniqueAI:
349
356
  }
350
357
  )
351
358
 
359
+ def _get_activated_skills_debug_info(self) -> list[dict[str, str | bool]]:
360
+ skill_tool = self._tool_manager.get_tool_by_name(SkillTool.name)
361
+ if not isinstance(skill_tool, SkillTool):
362
+ return []
363
+
364
+ forced_content_ids = {
365
+ choice.content_id for choice in self._skill_choices if choice.content_id
366
+ }
367
+ forced_names = {choice.name for choice in self._skill_choices if choice.name}
368
+
369
+ skills_debug_info: dict[str, dict[str, str | bool]] = {}
370
+ for skill in skill_tool.activated_skills:
371
+ skills_debug_info.setdefault(
372
+ skill.name,
373
+ {
374
+ "name": skill.name,
375
+ "content_id": skill.content_id,
376
+ "is_forced": (
377
+ skill.content_id in forced_content_ids
378
+ if skill.content_id
379
+ else skill.name in forced_names
380
+ ),
381
+ },
382
+ )
383
+
384
+ return list(skills_debug_info.values())
385
+
352
386
  # @track()
353
387
  async def _plan_or_execute(self) -> LanguageModelStreamResponse:
354
388
  self._logger.info("Planning or executing the loop.")
@@ -360,10 +360,7 @@ async def _build_common(
360
360
  )
361
361
 
362
362
  user_memory_text = ""
363
- if (
364
- config.space.allow_user_memory
365
- or config.agent.services.user_memory_config.enabled
366
- ):
363
+ if config.space.allow_user_memory:
367
364
  user_memory_state = await load_user_memory(
368
365
  event=event,
369
366
  config=config.agent.services.user_memory_config,