janito 2.3.0__py3-none-any.whl → 2.4.0__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.
Files changed (150) hide show
  1. janito/__init__.py +6 -6
  2. janito/_version.py +57 -0
  3. janito/agent/setup_agent.py +92 -18
  4. janito/agent/templates/profiles/system_prompt_template_developer.txt.j2 +44 -0
  5. janito/cli/chat_mode/bindings.py +21 -2
  6. janito/cli/chat_mode/chat_entry.py +2 -3
  7. janito/cli/chat_mode/prompt_style.py +5 -0
  8. janito/cli/chat_mode/session.py +80 -94
  9. janito/cli/chat_mode/session_profile_select.py +80 -0
  10. janito/cli/chat_mode/shell/autocomplete.py +21 -21
  11. janito/cli/chat_mode/shell/commands/__init__.py +13 -7
  12. janito/cli/chat_mode/shell/commands/_priv_check.py +5 -0
  13. janito/cli/chat_mode/shell/commands/clear.py +12 -12
  14. janito/cli/chat_mode/shell/commands/conversation_restart.py +30 -0
  15. janito/cli/chat_mode/shell/commands/execute.py +42 -0
  16. janito/cli/chat_mode/shell/commands/help.py +6 -3
  17. janito/cli/chat_mode/shell/commands/model.py +28 -0
  18. janito/cli/chat_mode/shell/commands/multi.py +51 -51
  19. janito/cli/chat_mode/shell/commands/read.py +37 -0
  20. janito/cli/chat_mode/shell/commands/tools.py +45 -18
  21. janito/cli/chat_mode/shell/commands/write.py +37 -0
  22. janito/cli/chat_mode/shell/commands.bak.zip +0 -0
  23. janito/cli/chat_mode/shell/input_history.py +62 -62
  24. janito/cli/chat_mode/shell/session.bak.zip +0 -0
  25. janito/cli/chat_mode/toolbar.py +44 -27
  26. janito/cli/cli_commands/list_models.py +35 -35
  27. janito/cli/cli_commands/list_providers.py +9 -9
  28. janito/cli/cli_commands/list_tools.py +86 -53
  29. janito/cli/cli_commands/model_selection.py +50 -50
  30. janito/cli/cli_commands/set_api_key.py +19 -19
  31. janito/cli/cli_commands/show_config.py +51 -51
  32. janito/cli/cli_commands/show_system_prompt.py +105 -62
  33. janito/cli/config.py +5 -6
  34. janito/cli/core/__init__.py +4 -4
  35. janito/cli/core/event_logger.py +59 -59
  36. janito/cli/core/runner.py +25 -18
  37. janito/cli/core/setters.py +10 -1
  38. janito/cli/core/unsetters.py +54 -54
  39. janito/cli/main_cli.py +28 -5
  40. janito/cli/prompt_core.py +18 -2
  41. janito/cli/prompt_setup.py +56 -0
  42. janito/cli/single_shot_mode/__init__.py +6 -6
  43. janito/cli/single_shot_mode/handler.py +14 -73
  44. janito/cli/verbose_output.py +1 -1
  45. janito/config.py +5 -5
  46. janito/config_manager.py +13 -0
  47. janito/drivers/anthropic/driver.py +113 -113
  48. janito/drivers/dashscope.bak.zip +0 -0
  49. janito/drivers/openai/README.md +20 -0
  50. janito/drivers/openai_responses.bak.zip +0 -0
  51. janito/event_bus/event.py +2 -2
  52. janito/formatting_token.py +54 -54
  53. janito/i18n/__init__.py +35 -35
  54. janito/i18n/messages.py +23 -23
  55. janito/i18n/pt.py +46 -47
  56. janito/llm/README.md +23 -0
  57. janito/llm/__init__.py +5 -5
  58. janito/llm/agent.py +507 -443
  59. janito/llm/driver.py +8 -0
  60. janito/llm/driver_config_builder.py +34 -34
  61. janito/llm/driver_input.py +12 -12
  62. janito/llm/message_parts.py +60 -60
  63. janito/llm/model.py +38 -38
  64. janito/llm/provider.py +196 -196
  65. janito/provider_registry.py +8 -6
  66. janito/providers/anthropic/model_info.py +22 -22
  67. janito/providers/anthropic/provider.py +2 -0
  68. janito/providers/azure_openai/provider.py +3 -0
  69. janito/providers/dashscope.bak.zip +0 -0
  70. janito/providers/deepseek/__init__.py +1 -1
  71. janito/providers/deepseek/model_info.py +16 -16
  72. janito/providers/deepseek/provider.py +94 -91
  73. janito/providers/google/provider.py +3 -0
  74. janito/providers/mistralai/provider.py +3 -0
  75. janito/providers/openai/provider.py +4 -0
  76. janito/providers/registry.py +26 -26
  77. janito/shell.bak.zip +0 -0
  78. janito/tools/DOCSTRING_STANDARD.txt +33 -0
  79. janito/tools/README.md +3 -0
  80. janito/tools/__init__.py +20 -6
  81. janito/tools/adapters/__init__.py +1 -1
  82. janito/tools/adapters/local/__init__.py +65 -62
  83. janito/tools/adapters/local/adapter.py +18 -35
  84. janito/tools/adapters/local/ask_user.py +101 -102
  85. janito/tools/adapters/local/copy_file.py +84 -84
  86. janito/tools/adapters/local/create_directory.py +69 -69
  87. janito/tools/adapters/local/create_file.py +82 -82
  88. janito/tools/adapters/local/delete_text_in_file.py +2 -2
  89. janito/tools/adapters/local/fetch_url.py +97 -97
  90. janito/tools/adapters/local/find_files.py +139 -138
  91. janito/tools/adapters/local/get_file_outline/__init__.py +1 -1
  92. janito/tools/adapters/local/get_file_outline/core.py +117 -117
  93. janito/tools/adapters/local/get_file_outline/java_outline.py +40 -40
  94. janito/tools/adapters/local/get_file_outline/markdown_outline.py +14 -14
  95. janito/tools/adapters/local/get_file_outline/python_outline.py +303 -303
  96. janito/tools/adapters/local/get_file_outline/python_outline_v2.py +156 -156
  97. janito/tools/adapters/local/get_file_outline/search_outline.py +33 -33
  98. janito/tools/adapters/local/move_file.py +2 -2
  99. janito/tools/adapters/local/open_html_in_browser.py +2 -1
  100. janito/tools/adapters/local/open_url.py +2 -2
  101. janito/tools/adapters/local/python_code_run.py +166 -166
  102. janito/tools/adapters/local/python_command_run.py +164 -164
  103. janito/tools/adapters/local/python_file_run.py +163 -163
  104. janito/tools/adapters/local/remove_directory.py +2 -2
  105. janito/tools/adapters/local/remove_file.py +2 -2
  106. janito/tools/adapters/local/replace_text_in_file.py +2 -2
  107. janito/tools/adapters/local/run_bash_command.py +176 -176
  108. janito/tools/adapters/local/run_powershell_command.py +219 -219
  109. janito/tools/adapters/local/search_text/__init__.py +1 -1
  110. janito/tools/adapters/local/search_text/core.py +201 -201
  111. janito/tools/adapters/local/search_text/pattern_utils.py +73 -73
  112. janito/tools/adapters/local/search_text/traverse_directory.py +145 -145
  113. janito/tools/adapters/local/validate_file_syntax/__init__.py +1 -1
  114. janito/tools/adapters/local/validate_file_syntax/core.py +106 -106
  115. janito/tools/adapters/local/validate_file_syntax/css_validator.py +35 -35
  116. janito/tools/adapters/local/validate_file_syntax/html_validator.py +93 -93
  117. janito/tools/adapters/local/validate_file_syntax/js_validator.py +27 -27
  118. janito/tools/adapters/local/validate_file_syntax/json_validator.py +6 -6
  119. janito/tools/adapters/local/validate_file_syntax/markdown_validator.py +109 -109
  120. janito/tools/adapters/local/validate_file_syntax/ps1_validator.py +32 -32
  121. janito/tools/adapters/local/validate_file_syntax/python_validator.py +5 -5
  122. janito/tools/adapters/local/validate_file_syntax/xml_validator.py +11 -11
  123. janito/tools/adapters/local/validate_file_syntax/yaml_validator.py +6 -6
  124. janito/tools/adapters/local/view_file.py +168 -167
  125. janito/tools/inspect_registry.py +17 -17
  126. janito/tools/outline_file.bak.zip +0 -0
  127. janito/tools/permissions.py +45 -0
  128. janito/tools/permissions_parse.py +12 -0
  129. janito/tools/tool_base.py +118 -105
  130. janito/tools/tool_events.py +58 -58
  131. janito/tools/tool_run_exception.py +12 -12
  132. janito/tools/tool_use_tracker.py +81 -81
  133. janito/tools/tool_utils.py +43 -45
  134. janito/tools/tools_adapter.py +25 -20
  135. janito/tools/tools_schema.py +104 -104
  136. {janito-2.3.0.dist-info → janito-2.4.0.dist-info}/METADATA +425 -388
  137. janito-2.4.0.dist-info/RECORD +195 -0
  138. janito/agent/templates/profiles/system_prompt_template_base_pt.txt.j2 +0 -13
  139. janito/agent/templates/profiles/system_prompt_template_main.txt.j2 +0 -37
  140. janito/cli/chat_mode/shell/commands/edit.py +0 -25
  141. janito/cli/chat_mode/shell/commands/exec.py +0 -27
  142. janito/cli/chat_mode/shell/commands/termweb_log.py +0 -92
  143. janito/cli/termweb_starter.py +0 -122
  144. janito/termweb/app.py +0 -95
  145. janito/version.py +0 -4
  146. janito-2.3.0.dist-info/RECORD +0 -181
  147. {janito-2.3.0.dist-info → janito-2.4.0.dist-info}/WHEEL +0 -0
  148. {janito-2.3.0.dist-info → janito-2.4.0.dist-info}/entry_points.txt +0 -0
  149. {janito-2.3.0.dist-info → janito-2.4.0.dist-info}/licenses/LICENSE +0 -0
  150. {janito-2.3.0.dist-info → janito-2.4.0.dist-info}/top_level.txt +0 -0
janito/llm/driver.py CHANGED
@@ -19,6 +19,14 @@ class LLMDriver(ABC):
19
19
  except Exception:
20
20
  pass
21
21
 
22
+ def clear_input_queue(self):
23
+ """Remove all items from the input queue."""
24
+ try:
25
+ while True:
26
+ self.input_queue.get_nowait()
27
+ except Exception:
28
+ pass
29
+
22
30
  """
23
31
  Abstract base class for LLM drivers (threaded, queue-based).
24
32
  Subclasses must implement:
@@ -1,34 +1,34 @@
1
- from typing import Type, Dict, Any
2
- from janito.llm.driver_config import LLMDriverConfig
3
-
4
-
5
- def build_llm_driver_config(
6
- config: Dict[str, Any], driver_class: Type
7
- ) -> LLMDriverConfig:
8
- """
9
- Build an LLMDriverConfig instance for the given driver class based on its declared driver_fields.
10
- Only fills fields missing from given config; does not overwrite fields already provided.
11
- Any config fields not in driver_fields or LLMDriverConfig fields go into .extra.
12
- """
13
- driver_fields = getattr(driver_class, "driver_fields", None)
14
- if driver_fields is None:
15
- driver_fields = set(LLMDriverConfig.__dataclass_fields__.keys()) - {
16
- "model",
17
- "extra",
18
- }
19
- base_info = {}
20
- extra = {}
21
- for k, v in (config or {}).items():
22
- if k in driver_fields and k in LLMDriverConfig.__dataclass_fields__:
23
- base_info[k] = v
24
- else:
25
- extra[k] = v
26
- # Only set missing fields, do NOT overwrite those from CLI/user
27
- for field in driver_fields:
28
- if field not in base_info and field in LLMDriverConfig.__dataclass_fields__:
29
- base_info[field] = (
30
- None # Optional: replace None with provider/driver default if wanted
31
- )
32
- return LLMDriverConfig(
33
- model=config.get("model") or config.get("model_name"), extra=extra, **base_info
34
- )
1
+ from typing import Type, Dict, Any
2
+ from janito.llm.driver_config import LLMDriverConfig
3
+
4
+
5
+ def build_llm_driver_config(
6
+ config: Dict[str, Any], driver_class: Type
7
+ ) -> LLMDriverConfig:
8
+ """
9
+ Build an LLMDriverConfig instance for the given driver class based on its declared driver_fields.
10
+ Only fills fields missing from given config; does not overwrite fields already provided.
11
+ Any config fields not in driver_fields or LLMDriverConfig fields go into .extra.
12
+ """
13
+ driver_fields = getattr(driver_class, "driver_fields", None)
14
+ if driver_fields is None:
15
+ driver_fields = set(LLMDriverConfig.__dataclass_fields__.keys()) - {
16
+ "model",
17
+ "extra",
18
+ }
19
+ base_info = {}
20
+ extra = {}
21
+ for k, v in (config or {}).items():
22
+ if k in driver_fields and k in LLMDriverConfig.__dataclass_fields__:
23
+ base_info[k] = v
24
+ else:
25
+ extra[k] = v
26
+ # Only set missing fields, do NOT overwrite those from CLI/user
27
+ for field in driver_fields:
28
+ if field not in base_info and field in LLMDriverConfig.__dataclass_fields__:
29
+ base_info[field] = (
30
+ None # Optional: replace None with provider/driver default if wanted
31
+ )
32
+ return LLMDriverConfig(
33
+ model=config.get("model") or config.get("model_name"), extra=extra, **base_info
34
+ )
@@ -1,12 +1,12 @@
1
- from dataclasses import dataclass, field
2
- from typing import Optional
3
- import threading
4
- from janito.llm.driver_config import LLMDriverConfig
5
- from janito.conversation_history import LLMConversationHistory
6
-
7
-
8
- @dataclass
9
- class DriverInput:
10
- config: LLMDriverConfig
11
- conversation_history: LLMConversationHistory
12
- cancel_event: Optional[threading.Event] = field(default=None)
1
+ from dataclasses import dataclass, field
2
+ from typing import Optional
3
+ import threading
4
+ from janito.llm.driver_config import LLMDriverConfig
5
+ from janito.conversation_history import LLMConversationHistory
6
+
7
+
8
+ @dataclass
9
+ class DriverInput:
10
+ config: LLMDriverConfig
11
+ conversation_history: LLMConversationHistory
12
+ cancel_event: Optional[threading.Event] = field(default=None)
@@ -1,60 +1,60 @@
1
- import attr
2
-
3
-
4
- class MessagePart:
5
- """
6
- Base class for all driver message parts.
7
- """
8
-
9
- type: str
10
-
11
-
12
- @attr.s(auto_attribs=True, kw_only=True)
13
- class TextMessagePart(MessagePart):
14
- content: str = ""
15
-
16
-
17
- @attr.s(auto_attribs=True, kw_only=True)
18
- class InlineDataMessagePart(MessagePart):
19
- content: bytes = b""
20
-
21
-
22
- @attr.s(auto_attribs=True, kw_only=True)
23
- class FileDataMessagePart(MessagePart):
24
- content: str = ""
25
-
26
-
27
- @attr.s(auto_attribs=True, kw_only=True)
28
- class VideoMetadataMessagePart(MessagePart):
29
- content: dict = attr.Factory(dict)
30
-
31
-
32
- @attr.s(auto_attribs=True, kw_only=True)
33
- class CodeExecutionResultMessagePart(MessagePart):
34
- content: str = ""
35
- stdout: str = ""
36
- stderr: str = ""
37
-
38
-
39
- @attr.s(auto_attribs=True, kw_only=True)
40
- class ExecutableCodeMessagePart(MessagePart):
41
- content: str = ""
42
-
43
-
44
- @attr.s(auto_attribs=True, kw_only=True)
45
- class FunctionCallMessagePart(MessagePart):
46
- tool_call_id: str = ""
47
- function: object = (
48
- None # Should match OpenAI SDK structure (with arguments as JSON string)
49
- )
50
-
51
-
52
- @attr.s(auto_attribs=True, kw_only=True)
53
- class FunctionResponseMessagePart(MessagePart):
54
- name: str = ""
55
- content: dict = attr.Factory(dict)
56
-
57
-
58
- @attr.s(auto_attribs=True, kw_only=True)
59
- class ThoughtMessagePart(MessagePart):
60
- content: bool = False
1
+ import attr
2
+
3
+
4
+ class MessagePart:
5
+ """
6
+ Base class for all driver message parts.
7
+ """
8
+
9
+ type: str
10
+
11
+
12
+ @attr.s(auto_attribs=True, kw_only=True)
13
+ class TextMessagePart(MessagePart):
14
+ content: str = ""
15
+
16
+
17
+ @attr.s(auto_attribs=True, kw_only=True)
18
+ class InlineDataMessagePart(MessagePart):
19
+ content: bytes = b""
20
+
21
+
22
+ @attr.s(auto_attribs=True, kw_only=True)
23
+ class FileDataMessagePart(MessagePart):
24
+ content: str = ""
25
+
26
+
27
+ @attr.s(auto_attribs=True, kw_only=True)
28
+ class VideoMetadataMessagePart(MessagePart):
29
+ content: dict = attr.Factory(dict)
30
+
31
+
32
+ @attr.s(auto_attribs=True, kw_only=True)
33
+ class CodeExecutionResultMessagePart(MessagePart):
34
+ content: str = ""
35
+ stdout: str = ""
36
+ stderr: str = ""
37
+
38
+
39
+ @attr.s(auto_attribs=True, kw_only=True)
40
+ class ExecutableCodeMessagePart(MessagePart):
41
+ content: str = ""
42
+
43
+
44
+ @attr.s(auto_attribs=True, kw_only=True)
45
+ class FunctionCallMessagePart(MessagePart):
46
+ tool_call_id: str = ""
47
+ function: object = (
48
+ None # Should match OpenAI SDK structure (with arguments as JSON string)
49
+ )
50
+
51
+
52
+ @attr.s(auto_attribs=True, kw_only=True)
53
+ class FunctionResponseMessagePart(MessagePart):
54
+ name: str = ""
55
+ content: dict = attr.Factory(dict)
56
+
57
+
58
+ @attr.s(auto_attribs=True, kw_only=True)
59
+ class ThoughtMessagePart(MessagePart):
60
+ content: bool = False
janito/llm/model.py CHANGED
@@ -1,38 +1,38 @@
1
- from dataclasses import dataclass, field
2
- from typing import Any, Optional
3
-
4
-
5
- @dataclass
6
- class LLMModelInfo:
7
- name: str
8
- context: Any = "N/A"
9
- max_input: Any = "N/A"
10
- max_cot: Any = "N/A"
11
- max_response: Any = "N/A"
12
- thinking_supported: Any = "N/A"
13
- default_temp: float = 0.2
14
- open: Optional[Any] = None
15
- category: Optional[str] = None
16
- driver: Optional[str] = None
17
- # This enables arbitrary provider-specific metadata
18
- other: dict = field(default_factory=dict)
19
-
20
- def to_dict(self) -> dict:
21
- d = self.__dict__.copy()
22
- if not self.open:
23
- d.pop("open")
24
- if not self.category:
25
- d.pop("category")
26
- if not self.driver:
27
- d.pop("driver")
28
- if not self.other:
29
- d.pop("other")
30
- return d
31
-
32
- @staticmethod
33
- def get_model_info(model_specs):
34
- """
35
- Standard get_model_info implementation for all providers:
36
- returns a list of model info dicts, one per model in the given MODEL_SPECS dict.
37
- """
38
- return [m.to_dict() for m in model_specs.values()]
1
+ from dataclasses import dataclass, field
2
+ from typing import Any, Optional
3
+
4
+
5
+ @dataclass
6
+ class LLMModelInfo:
7
+ name: str
8
+ context: Any = "N/A"
9
+ max_input: Any = "N/A"
10
+ max_cot: Any = "N/A"
11
+ max_response: Any = "N/A"
12
+ thinking_supported: Any = "N/A"
13
+ default_temp: float = 0.2
14
+ open: Optional[Any] = None
15
+ category: Optional[str] = None
16
+ driver: Optional[str] = None
17
+ # This enables arbitrary provider-specific metadata
18
+ other: dict = field(default_factory=dict)
19
+
20
+ def to_dict(self) -> dict:
21
+ d = self.__dict__.copy()
22
+ if not self.open:
23
+ d.pop("open")
24
+ if not self.category:
25
+ d.pop("category")
26
+ if not self.driver:
27
+ d.pop("driver")
28
+ if not self.other:
29
+ d.pop("other")
30
+ return d
31
+
32
+ @staticmethod
33
+ def get_model_info(model_specs):
34
+ """
35
+ Standard get_model_info implementation for all providers:
36
+ returns a list of model info dicts, one per model in the given MODEL_SPECS dict.
37
+ """
38
+ return [m.to_dict() for m in model_specs.values()]