janito 3.6.1__py3-none-any.whl → 3.7.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.
- janito/README.md +4 -7
- janito/cli/chat_mode/bindings.py +0 -50
- janito/cli/chat_mode/session.py +1 -12
- janito/cli/chat_mode/shell/commands/multi.py +0 -5
- janito/cli/chat_mode/shell/commands/security/allowed_sites.py +33 -47
- janito/cli/cli_commands/list_plugins.py +43 -52
- janito/cli/cli_commands/list_tools.py +1 -6
- janito/cli/core/getters.py +0 -3
- janito/cli/core/model_guesser.py +24 -40
- janito/cli/main_cli.py +13 -10
- janito/cli/prompt_core.py +9 -47
- janito/cli/rich_terminal_reporter.py +4 -4
- janito/docs/GETTING_STARTED.md +9 -8
- janito/drivers/openai/driver.py +0 -1
- janito/drivers/zai/driver.py +0 -1
- janito/i18n/it.py +46 -46
- janito/llm/agent.py +16 -32
- janito/llm/auth_utils.py +5 -14
- janito/llm/driver.py +0 -8
- janito/plugins/__init__.py +12 -31
- janito/plugins/auto_loader.py +11 -12
- janito/plugins/auto_loader_fixed.py +11 -12
- janito/{plugin_system → plugins}/base.py +2 -5
- janito/plugins/builtin.py +1 -15
- janito/plugins/core_adapter.py +11 -89
- janito/plugins/core_loader.py +120 -0
- janito/plugins/core_loader_fixed.py +125 -0
- janito/plugins/discovery.py +5 -5
- janito/plugins/discovery_core.py +9 -14
- janito/plugins/manager.py +1 -1
- janito/providers/__init__.py +0 -1
- janito/providers/moonshot/model_info.py +7 -7
- janito/providers/moonshot/provider.py +1 -1
- janito/tools/__init__.py +7 -41
- janito/tools/adapters/__init__.py +1 -6
- janito/tools/adapters/local/__init__.py +70 -7
- janito/{plugins/tools/core → tools/adapters/local}/ask_user.py +3 -3
- janito/tools/adapters/local/copy_file.py +87 -0
- janito/tools/adapters/local/create_file.py +138 -0
- janito/{plugins/tools/core → tools/adapters/local}/fetch_url.py +23 -20
- janito/tools/adapters/local/move_file.py +131 -0
- janito/{plugins/tools/core → tools/adapters/local}/python_code_run.py +7 -23
- janito/{plugins/tools/core → tools/adapters/local}/python_command_run.py +5 -21
- janito/{plugins/tools/core → tools/adapters/local}/python_file_run.py +5 -21
- janito/tools/adapters/local/remove_file.py +58 -0
- janito/{plugins/tools/core → tools/adapters/local}/replace_text_in_file.py +4 -4
- janito/{plugins/tools/core → tools/adapters/local}/run_bash_command.py +3 -3
- janito/{plugins/tools/core → tools/adapters/local}/run_powershell_command.py +3 -3
- janito/{plugins/tools/core → tools/adapters/local}/show_image.py +6 -15
- janito/{plugins/core/imagedisplay/tools → tools/adapters/local}/show_image_grid.py +5 -13
- janito/tools/adapters/local/view_file.py +172 -0
- janito/tools/function_adapter.py +65 -0
- janito/tools/loop_protection_decorator.py +117 -114
- janito-3.7.0.dist-info/METADATA +84 -0
- janito-3.7.0.dist-info/RECORD +264 -0
- janito/cli/cli_commands/check_tools.py +0 -212
- janito/data/blocked.txt +0 -96
- janito/llm/cancellation_manager.py +0 -63
- janito/llm/enter_cancellation.py +0 -107
- janito/plugin_system/__init__.py +0 -10
- janito/plugin_system/core_loader.py +0 -217
- janito/plugin_system/core_loader_fixed.py +0 -225
- janito/plugins/core/__init__.py +0 -7
- janito/plugins/core/codeanalyzer/__init__.py +0 -43
- janito/plugins/core/filemanager/__init__.py +0 -124
- janito/plugins/core/filemanager/tools/copy_file.py +0 -87
- janito/plugins/core/filemanager/tools/create_file.py +0 -87
- janito/plugins/core/filemanager/tools/move_file.py +0 -131
- janito/plugins/core/filemanager/tools/remove_file.py +0 -58
- janito/plugins/core/filemanager/tools/replace_text_in_file.py +0 -270
- janito/plugins/core/filemanager/tools/view_file.py +0 -172
- janito/plugins/core/imagedisplay/__init__.py +0 -14
- janito/plugins/core/imagedisplay/plugin.py +0 -51
- janito/plugins/core/imagedisplay/tools/__init__.py +0 -1
- janito/plugins/core/imagedisplay/tools/show_image.py +0 -83
- janito/plugins/core/system/__init__.py +0 -23
- janito/plugins/core/system/tools/run_bash_command.py +0 -204
- janito/plugins/core/system/tools/run_powershell_command.py +0 -234
- janito/plugins/dev/__init__.py +0 -7
- janito/plugins/dev/pythondev/__init__.py +0 -37
- janito/plugins/dev/visualization/__init__.py +0 -23
- janito/plugins/example_plugin.py +0 -108
- janito/plugins/tools/__init__.py +0 -39
- janito/plugins/tools/core/__init__.py +0 -65
- janito/plugins/tools/core/copy_file.py +0 -87
- janito/plugins/tools/core/create_directory.py +0 -70
- janito/plugins/tools/core/create_file.py +0 -138
- janito/plugins/tools/core/decorators.py +0 -19
- janito/plugins/tools/core/delete_text_in_file.py +0 -134
- janito/plugins/tools/core/find_files.py +0 -143
- janito/plugins/tools/core/get_file_outline/__init__.py +0 -7
- janito/plugins/tools/core/get_file_outline/core.py +0 -122
- janito/plugins/tools/core/get_file_outline/java_outline.py +0 -47
- janito/plugins/tools/core/get_file_outline/markdown_outline.py +0 -14
- janito/plugins/tools/core/get_file_outline/python_outline.py +0 -303
- janito/plugins/tools/core/get_file_outline/search_outline.py +0 -36
- janito/plugins/tools/core/move_file.py +0 -131
- janito/plugins/tools/core/open_html_in_browser.py +0 -51
- janito/plugins/tools/core/open_url.py +0 -37
- janito/plugins/tools/core/read_chart.py +0 -259
- janito/plugins/tools/core/read_files.py +0 -58
- janito/plugins/tools/core/remove_directory.py +0 -55
- janito/plugins/tools/core/remove_file.py +0 -58
- janito/plugins/tools/core/search_text/__init__.py +0 -7
- janito/plugins/tools/core/search_text/core.py +0 -205
- janito/plugins/tools/core/search_text/match_lines.py +0 -67
- janito/plugins/tools/core/search_text/pattern_utils.py +0 -73
- janito/plugins/tools/core/search_text/traverse_directory.py +0 -145
- janito/plugins/tools/core/show_image_grid.py +0 -85
- janito/plugins/tools/core/validate_file_syntax/__init__.py +0 -7
- janito/plugins/tools/core/validate_file_syntax/core.py +0 -114
- janito/plugins/tools/core/validate_file_syntax/css_validator.py +0 -35
- janito/plugins/tools/core/validate_file_syntax/html_validator.py +0 -100
- janito/plugins/tools/core/validate_file_syntax/jinja2_validator.py +0 -50
- janito/plugins/tools/core/validate_file_syntax/js_validator.py +0 -27
- janito/plugins/tools/core/validate_file_syntax/json_validator.py +0 -6
- janito/plugins/tools/core/validate_file_syntax/markdown_validator.py +0 -109
- janito/plugins/tools/core/validate_file_syntax/ps1_validator.py +0 -32
- janito/plugins/tools/core/validate_file_syntax/python_validator.py +0 -5
- janito/plugins/tools/core/validate_file_syntax/xml_validator.py +0 -11
- janito/plugins/tools/core/validate_file_syntax/yaml_validator.py +0 -6
- janito/plugins/tools/core/view_file.py +0 -172
- janito/plugins/ui/__init__.py +0 -7
- janito/plugins/ui/userinterface/__init__.py +0 -16
- janito/plugins/ui/userinterface/tools/ask_user.py +0 -110
- janito/plugins/web/__init__.py +0 -7
- janito/plugins/web/webtools/__init__.py +0 -23
- janito/providers/together/__init__.py +0 -1
- janito/providers/together/model_info.py +0 -69
- janito/providers/together/provider.py +0 -108
- janito/tools/blocked_sites.py +0 -74
- janito/tools/cli_initializer.py +0 -88
- janito/tools/initialize.py +0 -70
- janito-3.6.1.dist-info/METADATA +0 -228
- janito-3.6.1.dist-info/RECORD +0 -339
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/create_directory.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/delete_text_in_file.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/find_files.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/__init__.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/core.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/java_outline.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/markdown_outline.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/python_outline.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/get_file_outline/search_outline.py +0 -0
- /janito/{plugins/web/webtools/tools → tools/adapters/local}/open_html_in_browser.py +0 -0
- /janito/{plugins/web/webtools/tools → tools/adapters/local}/open_url.py +0 -0
- /janito/{plugins/dev/visualization/tools → tools/adapters/local}/read_chart.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/read_files.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/remove_directory.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/search_text/__init__.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/search_text/core.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/search_text/match_lines.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/search_text/pattern_utils.py +0 -0
- /janito/{plugins/core/codeanalyzer/tools → tools/adapters/local}/search_text/traverse_directory.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/__init__.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/core.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/css_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/html_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/jinja2_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/js_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/json_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/markdown_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/ps1_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/python_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/xml_validator.py +0 -0
- /janito/{plugins/core/filemanager/tools → tools/adapters/local}/validate_file_syntax/yaml_validator.py +0 -0
- {janito-3.6.1.dist-info → janito-3.7.0.dist-info}/WHEEL +0 -0
- {janito-3.6.1.dist-info → janito-3.7.0.dist-info}/entry_points.txt +0 -0
- {janito-3.6.1.dist-info → janito-3.7.0.dist-info}/licenses/LICENSE +0 -0
- {janito-3.6.1.dist-info → janito-3.7.0.dist-info}/top_level.txt +0 -0
@@ -1,65 +0,0 @@
|
|
1
|
-
"""
|
2
|
-
Core tools for janito.
|
3
|
-
|
4
|
-
This package contains the essential tools that provide basic functionality
|
5
|
-
for file operations, code execution, web scraping, and system interactions.
|
6
|
-
"""
|
7
|
-
|
8
|
-
from .fetch_url import FetchUrl
|
9
|
-
from .ask_user import AskUser
|
10
|
-
from .copy_file import CopyFile
|
11
|
-
from .create_directory import CreateDirectory
|
12
|
-
from .create_file import CreateFile
|
13
|
-
from .delete_text_in_file import DeleteTextInFile
|
14
|
-
from .find_files import FindFiles
|
15
|
-
from .move_file import MoveFile
|
16
|
-
from .open_html_in_browser import OpenHtmlInBrowser
|
17
|
-
from .open_url import OpenUrl
|
18
|
-
from .python_code_run import PythonCodeRun
|
19
|
-
from .python_command_run import PythonCommandRun
|
20
|
-
from .python_file_run import PythonFileRun
|
21
|
-
from .read_chart import ReadChart
|
22
|
-
from .read_files import ReadFiles
|
23
|
-
from .remove_directory import RemoveDirectory
|
24
|
-
from .remove_file import RemoveFile
|
25
|
-
from .replace_text_in_file import ReplaceTextInFile
|
26
|
-
from .run_bash_command import RunBashCommand
|
27
|
-
from .run_powershell_command import RunPowershellCommand
|
28
|
-
from .show_image import ShowImage
|
29
|
-
from .show_image_grid import ShowImageGrid
|
30
|
-
from .view_file import ViewFile
|
31
|
-
from .validate_file_syntax.core import ValidateFileSyntax
|
32
|
-
from .get_file_outline.core import GetFileOutline
|
33
|
-
from .search_text.core import SearchText
|
34
|
-
from .decorators import get_core_tools, register_core_tool
|
35
|
-
|
36
|
-
__all__ = [
|
37
|
-
'FetchUrl',
|
38
|
-
'AskUser',
|
39
|
-
'CopyFile',
|
40
|
-
'CreateDirectory',
|
41
|
-
'CreateFile',
|
42
|
-
'DeleteTextInFile',
|
43
|
-
'FindFiles',
|
44
|
-
'MoveFile',
|
45
|
-
'OpenHtmlInBrowser',
|
46
|
-
'OpenUrl',
|
47
|
-
'PythonCodeRun',
|
48
|
-
'PythonCommandRun',
|
49
|
-
'PythonFileRun',
|
50
|
-
'ReadChart',
|
51
|
-
'ReadFiles',
|
52
|
-
'RemoveDirectory',
|
53
|
-
'RemoveFile',
|
54
|
-
'ReplaceTextInFile',
|
55
|
-
'RunBashCommand',
|
56
|
-
'RunPowershellCommand',
|
57
|
-
'ShowImage',
|
58
|
-
'ShowImageGrid',
|
59
|
-
'ViewFile',
|
60
|
-
'ValidateFileSyntax',
|
61
|
-
'GetFileOutline',
|
62
|
-
'SearchText',
|
63
|
-
'get_core_tools',
|
64
|
-
'register_core_tool',
|
65
|
-
]
|
@@ -1,87 +0,0 @@
|
|
1
|
-
import os
|
2
|
-
from janito.tools.path_utils import expand_path
|
3
|
-
import shutil
|
4
|
-
from typing import List, Union
|
5
|
-
from .decorators import register_core_tool
|
6
|
-
from janito.tools.tool_base import ToolBase, ToolPermissions
|
7
|
-
from janito.tools.tool_utils import display_path
|
8
|
-
from janito.report_events import ReportAction
|
9
|
-
from janito.i18n import tr
|
10
|
-
|
11
|
-
|
12
|
-
@register_core_tool
|
13
|
-
class CopyFile(ToolBase):
|
14
|
-
"""
|
15
|
-
Copy one or more files to a target directory, or copy a single file to a new file.
|
16
|
-
Args:
|
17
|
-
sources (str): Space-separated path(s) to the file(s) to copy.
|
18
|
-
For multiple sources, provide a single string with paths separated by spaces.
|
19
|
-
target (str): Destination path. If copying multiple sources, this must be an existing directory.
|
20
|
-
overwrite (bool, optional): Overwrite existing files. Default: False.
|
21
|
-
Recommended only after reading the file to be overwritten.
|
22
|
-
Returns:
|
23
|
-
str: Status string for each copy operation.
|
24
|
-
"""
|
25
|
-
|
26
|
-
permissions = ToolPermissions(read=True, write=True)
|
27
|
-
tool_name = "copy_file"
|
28
|
-
|
29
|
-
def run(self, sources: str, target: str, overwrite: bool = False) -> str:
|
30
|
-
source_list = [expand_path(src) for src in sources.split() if src]
|
31
|
-
target = expand_path(target)
|
32
|
-
messages = []
|
33
|
-
if len(source_list) > 1:
|
34
|
-
if not os.path.isdir(target):
|
35
|
-
return tr(
|
36
|
-
"❗ Target must be an existing directory when copying multiple files: '{target}'",
|
37
|
-
target=display_path(target),
|
38
|
-
)
|
39
|
-
for src in source_list:
|
40
|
-
if not os.path.isfile(src):
|
41
|
-
messages.append(
|
42
|
-
tr(
|
43
|
-
"❗ Source file does not exist: '{src}'",
|
44
|
-
src=display_path(src),
|
45
|
-
)
|
46
|
-
)
|
47
|
-
continue
|
48
|
-
dst = os.path.join(target, os.path.basename(src))
|
49
|
-
messages.append(self._copy_one(src, dst, overwrite=overwrite))
|
50
|
-
else:
|
51
|
-
src = source_list[0]
|
52
|
-
if os.path.isdir(target):
|
53
|
-
dst = os.path.join(target, os.path.basename(src))
|
54
|
-
else:
|
55
|
-
dst = target
|
56
|
-
messages.append(self._copy_one(src, dst, overwrite=overwrite))
|
57
|
-
return "\n".join(messages)
|
58
|
-
|
59
|
-
def _copy_one(self, src, dst, overwrite=False) -> str:
|
60
|
-
disp_src = display_path(src)
|
61
|
-
disp_dst = display_path(dst)
|
62
|
-
if not os.path.isfile(src):
|
63
|
-
return tr("❗ Source file does not exist: '{src}'", src=disp_src)
|
64
|
-
if os.path.exists(dst) and not overwrite:
|
65
|
-
return tr(
|
66
|
-
"❗ Target already exists: '{dst}'. Set overwrite=True to replace.",
|
67
|
-
dst=disp_dst,
|
68
|
-
)
|
69
|
-
try:
|
70
|
-
os.makedirs(os.path.dirname(dst), exist_ok=True)
|
71
|
-
shutil.copy2(src, dst)
|
72
|
-
note = (
|
73
|
-
"\n⚠️ Overwrote existing file. (recommended only after reading the file to be overwritten)"
|
74
|
-
if (os.path.exists(dst) and overwrite)
|
75
|
-
else ""
|
76
|
-
)
|
77
|
-
self.report_success(
|
78
|
-
tr("✅ Copied '{src}' to '{dst}'", src=disp_src, dst=disp_dst)
|
79
|
-
)
|
80
|
-
return tr("✅ Copied '{src}' to '{dst}'", src=disp_src, dst=disp_dst) + note
|
81
|
-
except Exception as e:
|
82
|
-
return tr(
|
83
|
-
"❗ Copy failed from '{src}' to '{dst}': {err}",
|
84
|
-
src=disp_src,
|
85
|
-
dst=disp_dst,
|
86
|
-
err=str(e),
|
87
|
-
)
|
@@ -1,70 +0,0 @@
|
|
1
|
-
from .decorators import register_core_tool
|
2
|
-
|
3
|
-
from janito.tools.tool_utils import display_path
|
4
|
-
from janito.tools.tool_base import ToolBase, ToolPermissions
|
5
|
-
from janito.report_events import ReportAction
|
6
|
-
from janito.i18n import tr
|
7
|
-
import os
|
8
|
-
from janito.tools.path_utils import expand_path
|
9
|
-
|
10
|
-
|
11
|
-
@register_core_tool
|
12
|
-
class CreateDirectory(ToolBase):
|
13
|
-
"""
|
14
|
-
Create a new directory at the specified path.
|
15
|
-
Args:
|
16
|
-
path (str): Path for the new directory.
|
17
|
-
Returns:
|
18
|
-
str: Status message indicating the result. Example:
|
19
|
-
- "5c5 Successfully created the directory at ..."
|
20
|
-
- "5d7 Cannot create directory: ..."
|
21
|
-
"""
|
22
|
-
|
23
|
-
permissions = ToolPermissions(write=True)
|
24
|
-
tool_name = "create_directory"
|
25
|
-
|
26
|
-
def run(self, path: str) -> str:
|
27
|
-
path = expand_path(path)
|
28
|
-
disp_path = display_path(path)
|
29
|
-
self.report_action(
|
30
|
-
tr("📁 Create directory '{disp_path}' ...", disp_path=disp_path),
|
31
|
-
ReportAction.CREATE,
|
32
|
-
)
|
33
|
-
try:
|
34
|
-
if os.path.exists(path):
|
35
|
-
if not os.path.isdir(path):
|
36
|
-
self.report_error(
|
37
|
-
tr(
|
38
|
-
"❌ Path '{disp_path}' exists and is not a directory.",
|
39
|
-
disp_path=disp_path,
|
40
|
-
)
|
41
|
-
)
|
42
|
-
return tr(
|
43
|
-
"❌ Path '{disp_path}' exists and is not a directory.",
|
44
|
-
disp_path=disp_path,
|
45
|
-
)
|
46
|
-
self.report_error(
|
47
|
-
tr(
|
48
|
-
"❗ Directory '{disp_path}' already exists.",
|
49
|
-
disp_path=disp_path,
|
50
|
-
)
|
51
|
-
)
|
52
|
-
return tr(
|
53
|
-
"❗ Cannot create directory: '{disp_path}' already exists.",
|
54
|
-
disp_path=disp_path,
|
55
|
-
)
|
56
|
-
os.makedirs(path, exist_ok=True)
|
57
|
-
self.report_success(tr("✅ Directory created"))
|
58
|
-
return tr(
|
59
|
-
"✅ Successfully created the directory at '{disp_path}'.",
|
60
|
-
disp_path=disp_path,
|
61
|
-
)
|
62
|
-
except Exception as e:
|
63
|
-
self.report_error(
|
64
|
-
tr(
|
65
|
-
"❌ Error creating directory '{disp_path}': {error}",
|
66
|
-
disp_path=disp_path,
|
67
|
-
error=e,
|
68
|
-
)
|
69
|
-
)
|
70
|
-
return tr("❌ Cannot create directory: {error}", error=e)
|
@@ -1,138 +0,0 @@
|
|
1
|
-
import os
|
2
|
-
from janito.tools.path_utils import expand_path
|
3
|
-
from .decorators import register_core_tool
|
4
|
-
|
5
|
-
from janito.tools.tool_utils import display_path
|
6
|
-
from janito.tools.tool_base import ToolBase, ToolPermissions
|
7
|
-
from janito.report_events import ReportAction
|
8
|
-
from janito.i18n import tr
|
9
|
-
from janito.tools.loop_protection_decorator import protect_against_loops
|
10
|
-
|
11
|
-
from .validate_file_syntax.core import validate_file_syntax
|
12
|
-
|
13
|
-
|
14
|
-
@register_core_tool
|
15
|
-
class CreateFile(ToolBase):
|
16
|
-
"""
|
17
|
-
Create a new file with specified content at the given path.
|
18
|
-
|
19
|
-
This tool provides comprehensive file creation capabilities with built-in safety features,
|
20
|
-
automatic syntax validation, and detailed feedback. It handles path expansion, directory
|
21
|
-
creation, encoding issues, and provides clear status messages for both success and failure cases.
|
22
|
-
|
23
|
-
Key Features:
|
24
|
-
- Automatic directory creation for nested paths
|
25
|
-
- UTF-8 encoding with error handling for special characters
|
26
|
-
- Built-in syntax validation for common file types (Python, JavaScript, JSON, YAML, etc.)
|
27
|
-
- Loop protection to prevent excessive file creation
|
28
|
-
- Detailed error messages with context
|
29
|
-
- Safe overwrite protection with preview of existing content
|
30
|
-
- Cross-platform path handling (Windows, macOS, Linux)
|
31
|
-
|
32
|
-
Args:
|
33
|
-
path (str): Target file path. Supports relative and absolute paths, with automatic
|
34
|
-
expansion of user home directory (~) and environment variables.
|
35
|
-
Examples: "src/main.py", "~/Documents/config.json", "$HOME/.env"
|
36
|
-
content (str): File content to write. Empty string creates empty file.
|
37
|
-
Supports any text content including Unicode characters, newlines,
|
38
|
-
and binary-safe text representation.
|
39
|
-
overwrite (bool, optional): If True, allows overwriting existing files. Default: False.
|
40
|
-
When False, prevents accidental overwrites by checking
|
41
|
-
file existence and showing current content. Always review
|
42
|
-
existing content before enabling overwrite.
|
43
|
-
|
44
|
-
Returns:
|
45
|
-
str: Detailed status message including:
|
46
|
-
- Success confirmation with line count
|
47
|
-
- File path (display-friendly format)
|
48
|
-
- Syntax validation results
|
49
|
-
- Existing content preview (when overwrite blocked)
|
50
|
-
- Error details (when creation fails)
|
51
|
-
|
52
|
-
Raises:
|
53
|
-
No direct exceptions - all errors are caught and returned as user-friendly messages.
|
54
|
-
Common error cases include: permission denied, invalid path format, disk full,
|
55
|
-
or file exists (when overwrite=False).
|
56
|
-
|
57
|
-
Security Features:
|
58
|
-
- Loop protection: Prevents repeated create calls for the same file path within a short window (1 allowed per 10 seconds)
|
59
|
-
- Path traversal prevention: Validates and sanitizes file paths
|
60
|
-
- Permission checking: Respects file system permissions
|
61
|
-
- Atomic writes: Prevents partial file creation on errors
|
62
|
-
|
63
|
-
Examples:
|
64
|
-
Basic file creation:
|
65
|
-
>>> create_file("hello.py", "print('Hello, World!')")
|
66
|
-
✅ Created file 1 lines.
|
67
|
-
✅ Syntax OK
|
68
|
-
|
69
|
-
Creating nested directories:
|
70
|
-
>>> create_file("src/utils/helpers.py", "def helper(): pass")
|
71
|
-
✅ Created file 2 lines.
|
72
|
-
✅ Syntax OK
|
73
|
-
|
74
|
-
Overwrite protection:
|
75
|
-
>>> create_file("existing.txt", "new content")
|
76
|
-
❗ Cannot create file: file already exists at 'existing.txt'.
|
77
|
-
--- Current file content ---
|
78
|
-
old content
|
79
|
-
|
80
|
-
Note: After successful creation, automatic syntax validation is performed based on
|
81
|
-
file extension. Results are appended to the return message for immediate feedback.
|
82
|
-
"""
|
83
|
-
|
84
|
-
permissions = ToolPermissions(write=True)
|
85
|
-
tool_name = "create_file"
|
86
|
-
|
87
|
-
@protect_against_loops(max_calls=1, time_window=3600.0, key_field="path")
|
88
|
-
def run(self, path: str, content: str, overwrite: bool = False) -> str:
|
89
|
-
path = expand_path(path)
|
90
|
-
disp_path = display_path(path)
|
91
|
-
if os.path.exists(path) and not overwrite:
|
92
|
-
try:
|
93
|
-
with open(path, "r", encoding="utf-8", errors="replace") as f:
|
94
|
-
existing_content = f.read()
|
95
|
-
except Exception as e:
|
96
|
-
existing_content = f"[Error reading file: {e}]"
|
97
|
-
return tr(
|
98
|
-
"❗ Cannot create file: file already exists at '{disp_path}'.\n--- Current file content ---\n{existing_content}",
|
99
|
-
disp_path=disp_path,
|
100
|
-
existing_content=existing_content,
|
101
|
-
)
|
102
|
-
# Determine if we are overwriting an existing file
|
103
|
-
is_overwrite = os.path.exists(path) and overwrite
|
104
|
-
if is_overwrite:
|
105
|
-
# Overwrite branch: log only overwrite warning (no create message)
|
106
|
-
self.report_action(
|
107
|
-
tr("⚠️ Overwriting file '{disp_path}'", disp_path=disp_path),
|
108
|
-
ReportAction.CREATE,
|
109
|
-
)
|
110
|
-
dir_name = os.path.dirname(path)
|
111
|
-
if dir_name:
|
112
|
-
os.makedirs(dir_name, exist_ok=True)
|
113
|
-
if not is_overwrite:
|
114
|
-
# Create branch: log file creation message
|
115
|
-
self.report_action(
|
116
|
-
tr("📝 Create file '{disp_path}' ...", disp_path=disp_path),
|
117
|
-
ReportAction.CREATE,
|
118
|
-
)
|
119
|
-
with open(path, "w", encoding="utf-8", errors="replace") as f:
|
120
|
-
f.write(content)
|
121
|
-
new_lines = content.count("\n") + 1 if content else 0
|
122
|
-
self.report_success(
|
123
|
-
tr("✅ {new_lines} lines", new_lines=new_lines), ReportAction.CREATE
|
124
|
-
)
|
125
|
-
# Perform syntax validation and append result
|
126
|
-
validation_result = validate_file_syntax(path)
|
127
|
-
if is_overwrite:
|
128
|
-
# Overwrite branch: return minimal overwrite info to user
|
129
|
-
return (
|
130
|
-
tr("✅ {new_lines} lines", new_lines=new_lines)
|
131
|
-
+ f"\n{validation_result}"
|
132
|
-
)
|
133
|
-
else:
|
134
|
-
# Create branch: return detailed create success to user
|
135
|
-
return (
|
136
|
-
tr("✅ Created file {new_lines} lines.", new_lines=new_lines)
|
137
|
-
+ f"\n{validation_result}"
|
138
|
-
)
|
@@ -1,19 +0,0 @@
|
|
1
|
-
"""
|
2
|
-
Decorators for core tools registration.
|
3
|
-
"""
|
4
|
-
|
5
|
-
from typing import Type
|
6
|
-
|
7
|
-
# Registry for core tools
|
8
|
-
_core_tools_registry = []
|
9
|
-
|
10
|
-
|
11
|
-
def register_core_tool(cls: Type):
|
12
|
-
"""Decorator to register a core tool."""
|
13
|
-
_core_tools_registry.append(cls)
|
14
|
-
return cls
|
15
|
-
|
16
|
-
|
17
|
-
def get_core_tools():
|
18
|
-
"""Get all registered core tools."""
|
19
|
-
return list(_core_tools_registry)
|
@@ -1,134 +0,0 @@
|
|
1
|
-
from janito.tools.tool_base import ToolBase, ToolPermissions
|
2
|
-
from janito.report_events import ReportAction
|
3
|
-
from .decorators import register_core_tool
|
4
|
-
from janito.i18n import tr
|
5
|
-
import shutil
|
6
|
-
from .validate_file_syntax.core import validate_file_syntax
|
7
|
-
|
8
|
-
|
9
|
-
@register_core_tool
|
10
|
-
class DeleteTextInFile(ToolBase):
|
11
|
-
"""
|
12
|
-
Delete all occurrences of text between start_marker and end_marker (inclusive) in a file, using exact string markers.
|
13
|
-
|
14
|
-
Args:
|
15
|
-
path (str): Path to the file to modify.
|
16
|
-
start_marker (str): The starting delimiter string.
|
17
|
-
end_marker (str): The ending delimiter string.
|
18
|
-
|
19
|
-
Returns:
|
20
|
-
str: Status message indicating the result.
|
21
|
-
"""
|
22
|
-
|
23
|
-
permissions = ToolPermissions(read=True, write=True)
|
24
|
-
tool_name = "delete_text_in_file"
|
25
|
-
|
26
|
-
def run(
|
27
|
-
self,
|
28
|
-
path: str,
|
29
|
-
start_marker: str,
|
30
|
-
end_marker: str,
|
31
|
-
backup: bool = False,
|
32
|
-
) -> str:
|
33
|
-
from janito.tools.tool_utils import display_path
|
34
|
-
|
35
|
-
disp_path = display_path(path)
|
36
|
-
info_msg = tr(
|
37
|
-
"📝 Delete text in {disp_path} between markers: '{start_marker}' ... '{end_marker}'",
|
38
|
-
disp_path=disp_path,
|
39
|
-
start_marker=start_marker,
|
40
|
-
end_marker=end_marker,
|
41
|
-
)
|
42
|
-
self.report_action(info_msg, ReportAction.CREATE)
|
43
|
-
try:
|
44
|
-
content = self._read_file_content(path)
|
45
|
-
occurrences, match_lines = self._find_marker_blocks(
|
46
|
-
content, start_marker, end_marker
|
47
|
-
)
|
48
|
-
if occurrences == 0:
|
49
|
-
self.report_warning(
|
50
|
-
tr(" ℹ️ No blocks found between markers."), ReportAction.CREATE
|
51
|
-
)
|
52
|
-
return tr(
|
53
|
-
"No blocks found between markers in {path}.",
|
54
|
-
path=path,
|
55
|
-
)
|
56
|
-
|
57
|
-
new_content, deleted_blocks = self._delete_blocks(
|
58
|
-
content, start_marker, end_marker
|
59
|
-
)
|
60
|
-
self._write_file_content(path, new_content)
|
61
|
-
validation_result = validate_file_syntax(path)
|
62
|
-
self._report_success(match_lines)
|
63
|
-
return tr(
|
64
|
-
"Deleted {count} block(s) between markers in {path}. ",
|
65
|
-
count=deleted_blocks,
|
66
|
-
path=path,
|
67
|
-
) + (f"\n{validation_result}" if validation_result else "")
|
68
|
-
except Exception as e:
|
69
|
-
self.report_error(tr(" ❌ Error: {error}", error=e), ReportAction.REPLACE)
|
70
|
-
return tr("Error deleting text: {error}", error=e)
|
71
|
-
|
72
|
-
def _read_file_content(self, path):
|
73
|
-
with open(path, "r", encoding="utf-8", errors="replace") as f:
|
74
|
-
return f.read()
|
75
|
-
|
76
|
-
def _find_marker_blocks(self, content, start_marker, end_marker):
|
77
|
-
"""Find all blocks between start_marker and end_marker, return count and starting line numbers."""
|
78
|
-
lines = content.splitlines(keepends=True)
|
79
|
-
joined = "".join(lines)
|
80
|
-
match_lines = []
|
81
|
-
idx = 0
|
82
|
-
occurrences = 0
|
83
|
-
while True:
|
84
|
-
start_idx = joined.find(start_marker, idx)
|
85
|
-
if start_idx == -1:
|
86
|
-
break
|
87
|
-
end_idx = joined.find(end_marker, start_idx + len(start_marker))
|
88
|
-
if end_idx == -1:
|
89
|
-
break
|
90
|
-
upto = joined[:start_idx]
|
91
|
-
line_no = upto.count("\n") + 1
|
92
|
-
match_lines.append(line_no)
|
93
|
-
idx = end_idx + len(end_marker)
|
94
|
-
occurrences += 1
|
95
|
-
return occurrences, match_lines
|
96
|
-
|
97
|
-
def _delete_blocks(self, content, start_marker, end_marker):
|
98
|
-
"""Delete all blocks between start_marker and end_marker (inclusive)."""
|
99
|
-
count = 0
|
100
|
-
new_content = content
|
101
|
-
while True:
|
102
|
-
start_idx = new_content.find(start_marker)
|
103
|
-
if start_idx == -1:
|
104
|
-
break
|
105
|
-
end_idx = new_content.find(end_marker, start_idx + len(start_marker))
|
106
|
-
if end_idx == -1:
|
107
|
-
break
|
108
|
-
new_content = (
|
109
|
-
new_content[:start_idx] + new_content[end_idx + len(end_marker) :]
|
110
|
-
)
|
111
|
-
count += 1
|
112
|
-
return new_content, count
|
113
|
-
|
114
|
-
def _backup_file(self, path, backup_path):
|
115
|
-
shutil.copy2(path, backup_path)
|
116
|
-
|
117
|
-
def _write_file_content(self, path, content):
|
118
|
-
with open(path, "w", encoding="utf-8", errors="replace") as f:
|
119
|
-
f.write(content)
|
120
|
-
|
121
|
-
def _report_success(self, match_lines):
|
122
|
-
if match_lines:
|
123
|
-
lines_str = ", ".join(str(line_no) for line_no in match_lines)
|
124
|
-
self.report_success(
|
125
|
-
tr(
|
126
|
-
" ✅ deleted block(s) starting at line(s): {lines_str}",
|
127
|
-
lines_str=lines_str,
|
128
|
-
),
|
129
|
-
ReportAction.CREATE,
|
130
|
-
)
|
131
|
-
else:
|
132
|
-
self.report_success(
|
133
|
-
tr(" ✅ deleted block(s) (lines unknown)"), ReportAction.CREATE
|
134
|
-
)
|
@@ -1,143 +0,0 @@
|
|
1
|
-
from janito.tools.tool_base import ToolBase, ToolPermissions
|
2
|
-
from janito.report_events import ReportAction
|
3
|
-
from .decorators import register_core_tool
|
4
|
-
from janito.tools.tool_utils import pluralize, display_path
|
5
|
-
from janito.dir_walk_utils import walk_dir_with_gitignore
|
6
|
-
from janito.i18n import tr
|
7
|
-
import fnmatch
|
8
|
-
import os
|
9
|
-
from janito.tools.path_utils import expand_path
|
10
|
-
from janito.tools.loop_protection_decorator import protect_against_loops
|
11
|
-
|
12
|
-
|
13
|
-
@register_core_tool
|
14
|
-
class FindFiles(ToolBase):
|
15
|
-
"""
|
16
|
-
Find files or directories in one or more directories matching a pattern. Respects .gitignore.
|
17
|
-
|
18
|
-
If a path is an existing file, it is checked against the provided pattern(s) and included in the results if it matches. This allows find_files to be used to look for a specific set of filenames in a single call, as well as searching directories.
|
19
|
-
|
20
|
-
Args:
|
21
|
-
paths (str): String of one or more paths (space-separated) to search in. Each path can be a directory or a file.
|
22
|
-
pattern (str): File pattern(s) to match. Multiple patterns can be separated by spaces. Uses Unix shell-style wildcards (fnmatch), e.g. '*.py', 'data_??.csv', '[a-z]*.txt'.
|
23
|
-
- If the pattern ends with '/' or '\', only matching directory names (with trailing slash) are returned, not the files within those directories. For example, pattern '*/' will return only directories at the specified depth.
|
24
|
-
max_depth (int, optional): Maximum directory depth to search. If None, unlimited recursion. If 0, only the top-level directory. If 1, only the root directory (matches 'find . -maxdepth 1').
|
25
|
-
include_gitignored (bool, optional): If True, includes files/directories ignored by .gitignore. Defaults to False.
|
26
|
-
Returns:
|
27
|
-
str: Newline-separated list of matching file paths. Example:
|
28
|
-
"/path/to/file1.py\n/path/to/file2.py"
|
29
|
-
"Warning: Empty file pattern provided. Operation skipped."
|
30
|
-
"""
|
31
|
-
|
32
|
-
permissions = ToolPermissions(read=True)
|
33
|
-
tool_name = "find_files"
|
34
|
-
|
35
|
-
def _match_directories(self, root, dirs, pat):
|
36
|
-
dir_output = set()
|
37
|
-
dir_pat = pat.rstrip("/\\")
|
38
|
-
for d in dirs:
|
39
|
-
if fnmatch.fnmatch(d, dir_pat):
|
40
|
-
dir_output.add(os.path.join(root, d) + os.sep)
|
41
|
-
return dir_output
|
42
|
-
|
43
|
-
def _match_files(self, root, files, pat):
|
44
|
-
file_output = set()
|
45
|
-
for filename in fnmatch.filter(files, pat):
|
46
|
-
file_output.add(os.path.join(root, filename))
|
47
|
-
return file_output
|
48
|
-
|
49
|
-
def _match_dirs_without_slash(self, root, dirs, pat):
|
50
|
-
dir_output = set()
|
51
|
-
for d in fnmatch.filter(dirs, pat):
|
52
|
-
dir_output.add(os.path.join(root, d))
|
53
|
-
return dir_output
|
54
|
-
|
55
|
-
def _handle_path(self, directory, patterns):
|
56
|
-
dir_output = set()
|
57
|
-
filename = os.path.basename(directory)
|
58
|
-
for pat in patterns:
|
59
|
-
# Only match files, not directories, for file paths
|
60
|
-
if not (pat.endswith("/") or pat.endswith("\\")):
|
61
|
-
if fnmatch.fnmatch(filename, pat):
|
62
|
-
dir_output.add(directory)
|
63
|
-
break
|
64
|
-
return dir_output
|
65
|
-
|
66
|
-
def _handle_directory_path(
|
67
|
-
self, directory, patterns, max_depth, include_gitignored
|
68
|
-
):
|
69
|
-
dir_output = set()
|
70
|
-
for root, dirs, files in walk_dir_with_gitignore(
|
71
|
-
directory,
|
72
|
-
max_depth=max_depth,
|
73
|
-
include_gitignored=include_gitignored,
|
74
|
-
):
|
75
|
-
for pat in patterns:
|
76
|
-
if pat.endswith("/") or pat.endswith("\\"):
|
77
|
-
dir_output.update(self._match_directories(root, dirs, pat))
|
78
|
-
else:
|
79
|
-
dir_output.update(self._match_files(root, files, pat))
|
80
|
-
dir_output.update(self._match_dirs_without_slash(root, dirs, pat))
|
81
|
-
return dir_output
|
82
|
-
|
83
|
-
def _report_search(self, pattern, disp_path, depth_msg):
|
84
|
-
self.report_action(
|
85
|
-
tr(
|
86
|
-
"🔍 Search for files '{pattern}' in '{disp_path}'{depth_msg} ...",
|
87
|
-
pattern=pattern,
|
88
|
-
disp_path=disp_path,
|
89
|
-
depth_msg=depth_msg,
|
90
|
-
),
|
91
|
-
ReportAction.READ,
|
92
|
-
)
|
93
|
-
|
94
|
-
def _report_success(self, count):
|
95
|
-
self.report_success(
|
96
|
-
tr(
|
97
|
-
" ✅ {count} {file_word}",
|
98
|
-
count=count,
|
99
|
-
file_word=pluralize("file", count),
|
100
|
-
),
|
101
|
-
ReportAction.READ,
|
102
|
-
)
|
103
|
-
|
104
|
-
def _format_output(self, directory, dir_output):
|
105
|
-
if directory.strip() == ".":
|
106
|
-
dir_output = {
|
107
|
-
p[2:] if (p.startswith("./") or p.startswith(".\\")) else p
|
108
|
-
for p in dir_output
|
109
|
-
}
|
110
|
-
return sorted(dir_output)
|
111
|
-
|
112
|
-
@protect_against_loops(max_calls=5, time_window=10.0, key_field="paths")
|
113
|
-
def run(
|
114
|
-
self,
|
115
|
-
paths: str,
|
116
|
-
pattern: str,
|
117
|
-
max_depth: int = None,
|
118
|
-
include_gitignored: bool = False,
|
119
|
-
) -> str:
|
120
|
-
if not pattern:
|
121
|
-
self.report_warning(tr("ℹ️ Empty file pattern provided."), ReportAction.READ)
|
122
|
-
return tr("Warning: Empty file pattern provided. Operation skipped.")
|
123
|
-
patterns = pattern.split()
|
124
|
-
results = []
|
125
|
-
for directory in [expand_path(p) for p in paths.split()]:
|
126
|
-
disp_path = display_path(directory)
|
127
|
-
depth_msg = (
|
128
|
-
tr(" (max depth: {max_depth})", max_depth=max_depth)
|
129
|
-
if max_depth is not None and max_depth > 0
|
130
|
-
else ""
|
131
|
-
)
|
132
|
-
self._report_search(pattern, disp_path, depth_msg)
|
133
|
-
dir_output = set()
|
134
|
-
if os.path.isfile(directory):
|
135
|
-
dir_output = self._handle_path(directory, patterns)
|
136
|
-
elif os.path.isdir(directory):
|
137
|
-
dir_output = self._handle_directory_path(
|
138
|
-
directory, patterns, max_depth, include_gitignored
|
139
|
-
)
|
140
|
-
self._report_success(len(dir_output))
|
141
|
-
results.extend(self._format_output(directory, dir_output))
|
142
|
-
result = "\n".join(results)
|
143
|
-
return result
|