janito 2.33.0__py3-none-any.whl → 3.0.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 (140) hide show
  1. janito/cli/cli_commands/check_tools.py +212 -0
  2. janito/cli/cli_commands/list_plugins.py +52 -43
  3. janito/cli/core/getters.py +3 -0
  4. janito/cli/main_cli.py +9 -12
  5. janito/drivers/openai/driver.py +1 -0
  6. janito/drivers/zai/driver.py +1 -0
  7. janito/llm/auth_utils.py +14 -5
  8. janito/plugin_system/__init__.py +10 -0
  9. janito/{plugins → plugin_system}/base.py +5 -2
  10. janito/{plugins/core_loader_fixed.py → plugin_system/core_loader.py} +45 -26
  11. janito/plugin_system/core_loader_fixed.py +149 -0
  12. janito/plugins/__init__.py +31 -12
  13. janito/plugins/auto_loader_fixed.py +12 -11
  14. janito/plugins/builtin.py +15 -1
  15. janito/plugins/core/__init__.py +7 -0
  16. janito/plugins/core/codeanalyzer/__init__.py +43 -0
  17. janito/plugins/core/filemanager/__init__.py +124 -0
  18. janito/plugins/core/filemanager/tools/create_file.py +87 -0
  19. janito/plugins/core/filemanager/tools/replace_text_in_file.py +270 -0
  20. janito/plugins/core/imagedisplay/__init__.py +14 -0
  21. janito/plugins/core/imagedisplay/plugin.py +51 -0
  22. janito/plugins/core/imagedisplay/tools/__init__.py +1 -0
  23. janito/plugins/core/imagedisplay/tools/show_image.py +83 -0
  24. janito/{tools/adapters/local → plugins/core/imagedisplay/tools}/show_image_grid.py +13 -5
  25. janito/plugins/core/system/__init__.py +23 -0
  26. janito/plugins/core_adapter.py +11 -9
  27. janito/plugins/dev/__init__.py +7 -0
  28. janito/plugins/dev/pythondev/__init__.py +37 -0
  29. janito/plugins/dev/visualization/__init__.py +23 -0
  30. janito/plugins/discovery.py +5 -5
  31. janito/plugins/example_plugin.py +108 -0
  32. janito/plugins/manager.py +1 -1
  33. janito/plugins/tools/__init__.py +10 -0
  34. janito/{tools/adapters/local → plugins/tools}/ask_user.py +3 -3
  35. janito/plugins/tools/copy_file.py +87 -0
  36. janito/plugins/tools/core_tools_plugin.py +88 -0
  37. janito/plugins/tools/create_directory.py +70 -0
  38. janito/{tools/adapters/local → plugins/tools}/create_file.py +4 -4
  39. janito/plugins/tools/decorators.py +19 -0
  40. janito/plugins/tools/delete_text_in_file.py +134 -0
  41. janito/{tools/adapters/local → plugins/tools}/fetch_url.py +3 -3
  42. janito/plugins/tools/find_files.py +143 -0
  43. janito/plugins/tools/get_file_outline/__init__.py +7 -0
  44. janito/plugins/tools/get_file_outline/core.py +122 -0
  45. janito/plugins/tools/get_file_outline/java_outline.py +47 -0
  46. janito/plugins/tools/get_file_outline/markdown_outline.py +14 -0
  47. janito/plugins/tools/get_file_outline/python_outline.py +303 -0
  48. janito/plugins/tools/get_file_outline/search_outline.py +36 -0
  49. janito/plugins/tools/move_file.py +131 -0
  50. janito/plugins/tools/open_html_in_browser.py +51 -0
  51. janito/plugins/tools/open_url.py +37 -0
  52. janito/plugins/tools/python_code_run.py +172 -0
  53. janito/plugins/tools/python_command_run.py +171 -0
  54. janito/plugins/tools/python_file_run.py +172 -0
  55. janito/plugins/tools/read_chart.py +259 -0
  56. janito/plugins/tools/read_files.py +58 -0
  57. janito/plugins/tools/remove_directory.py +55 -0
  58. janito/plugins/tools/remove_file.py +58 -0
  59. janito/{tools/adapters/local → plugins/tools}/replace_text_in_file.py +4 -4
  60. janito/plugins/tools/run_bash_command.py +183 -0
  61. janito/plugins/tools/run_powershell_command.py +218 -0
  62. janito/plugins/tools/search_text/__init__.py +7 -0
  63. janito/plugins/tools/search_text/core.py +205 -0
  64. janito/plugins/tools/search_text/match_lines.py +67 -0
  65. janito/plugins/tools/search_text/pattern_utils.py +73 -0
  66. janito/plugins/tools/search_text/traverse_directory.py +145 -0
  67. janito/{tools/adapters/local → plugins/tools}/show_image.py +15 -6
  68. janito/plugins/tools/show_image_grid.py +85 -0
  69. janito/plugins/tools/validate_file_syntax/__init__.py +7 -0
  70. janito/plugins/tools/validate_file_syntax/core.py +114 -0
  71. janito/plugins/tools/validate_file_syntax/css_validator.py +35 -0
  72. janito/plugins/tools/validate_file_syntax/html_validator.py +100 -0
  73. janito/plugins/tools/validate_file_syntax/jinja2_validator.py +50 -0
  74. janito/plugins/tools/validate_file_syntax/js_validator.py +27 -0
  75. janito/plugins/tools/validate_file_syntax/json_validator.py +6 -0
  76. janito/plugins/tools/validate_file_syntax/markdown_validator.py +109 -0
  77. janito/plugins/tools/validate_file_syntax/ps1_validator.py +32 -0
  78. janito/plugins/tools/validate_file_syntax/python_validator.py +5 -0
  79. janito/plugins/tools/validate_file_syntax/xml_validator.py +11 -0
  80. janito/plugins/tools/validate_file_syntax/yaml_validator.py +6 -0
  81. janito/plugins/tools/view_file.py +172 -0
  82. janito/plugins/ui/__init__.py +7 -0
  83. janito/plugins/ui/userinterface/__init__.py +16 -0
  84. janito/plugins/ui/userinterface/tools/ask_user.py +110 -0
  85. janito/plugins/web/__init__.py +7 -0
  86. janito/plugins/web/webtools/__init__.py +33 -0
  87. janito/plugins/web/webtools/tools/fetch_url.py +458 -0
  88. janito/tools/__init__.py +31 -7
  89. janito/tools/adapters/__init__.py +6 -1
  90. janito/tools/adapters/local/__init__.py +7 -70
  91. janito/tools/cli_initializer.py +88 -0
  92. janito/tools/function_adapter.py +93 -16
  93. janito/tools/initialize.py +70 -0
  94. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/METADATA +1 -2
  95. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/RECORD +139 -71
  96. janito/plugins/core_loader.py +0 -120
  97. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/__init__.py +0 -0
  98. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/core.py +0 -0
  99. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/java_outline.py +0 -0
  100. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/markdown_outline.py +0 -0
  101. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/python_outline.py +0 -0
  102. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/get_file_outline/search_outline.py +0 -0
  103. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/search_text/__init__.py +0 -0
  104. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/search_text/core.py +0 -0
  105. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/search_text/match_lines.py +0 -0
  106. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/search_text/pattern_utils.py +0 -0
  107. /janito/{tools/adapters/local → plugins/core/codeanalyzer/tools}/search_text/traverse_directory.py +0 -0
  108. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/copy_file.py +0 -0
  109. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/create_directory.py +0 -0
  110. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/delete_text_in_file.py +0 -0
  111. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/find_files.py +0 -0
  112. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/move_file.py +0 -0
  113. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/read_files.py +0 -0
  114. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/remove_directory.py +0 -0
  115. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/remove_file.py +0 -0
  116. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/__init__.py +0 -0
  117. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/core.py +0 -0
  118. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/css_validator.py +0 -0
  119. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/html_validator.py +0 -0
  120. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/jinja2_validator.py +0 -0
  121. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/js_validator.py +0 -0
  122. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/json_validator.py +0 -0
  123. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/markdown_validator.py +0 -0
  124. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/ps1_validator.py +0 -0
  125. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/python_validator.py +0 -0
  126. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/xml_validator.py +0 -0
  127. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/validate_file_syntax/yaml_validator.py +0 -0
  128. /janito/{tools/adapters/local → plugins/core/filemanager/tools}/view_file.py +0 -0
  129. /janito/{tools/adapters/local → plugins/core/system/tools}/run_bash_command.py +0 -0
  130. /janito/{tools/adapters/local → plugins/core/system/tools}/run_powershell_command.py +0 -0
  131. /janito/{tools/adapters/local → plugins/dev/pythondev/tools}/python_code_run.py +0 -0
  132. /janito/{tools/adapters/local → plugins/dev/pythondev/tools}/python_command_run.py +0 -0
  133. /janito/{tools/adapters/local → plugins/dev/pythondev/tools}/python_file_run.py +0 -0
  134. /janito/{tools/adapters/local → plugins/dev/visualization/tools}/read_chart.py +0 -0
  135. /janito/{tools/adapters/local → plugins/web/webtools/tools}/open_html_in_browser.py +0 -0
  136. /janito/{tools/adapters/local → plugins/web/webtools/tools}/open_url.py +0 -0
  137. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/WHEEL +0 -0
  138. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/entry_points.txt +0 -0
  139. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/licenses/LICENSE +0 -0
  140. {janito-2.33.0.dist-info → janito-3.0.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,143 @@
1
+ from janito.tools.tool_base import ToolBase, ToolPermissions
2
+ from janito.report_events import ReportAction
3
+ from janito.plugins.tools.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
@@ -0,0 +1,7 @@
1
+ """
2
+ File outline tools for janito.
3
+ """
4
+
5
+ from .core import GetFileOutline
6
+
7
+ __all__ = ["GetFileOutline"]
@@ -0,0 +1,122 @@
1
+ from janito.tools.adapters.local.adapter import register_local_tool
2
+ from .python_outline import parse_python_outline
3
+ from .markdown_outline import parse_markdown_outline
4
+ from janito.formatting import OutlineFormatter
5
+ from .java_outline import parse_java_outline
6
+ import os
7
+ from janito.tools.path_utils import expand_path
8
+ from janito.tools.tool_base import ToolBase, ToolPermissions
9
+ from janito.report_events import ReportAction
10
+ from janito.tools.tool_utils import display_path, pluralize
11
+ from janito.i18n import tr
12
+
13
+ from janito.plugins.tools.decorators import register_core_tool
14
+ from janito.tools.loop_protection_decorator import protect_against_loops
15
+
16
+
17
+ @register_core_tool
18
+ class GetFileOutline(ToolBase):
19
+ """
20
+ Get an outline of a file's structure. Supports Python and Markdown files.
21
+
22
+ Args:
23
+ path (str): Path to the file to outline.
24
+ """
25
+
26
+ permissions = ToolPermissions(read=True)
27
+ tool_name = "get_file_outline"
28
+
29
+ @protect_against_loops(max_calls=5, time_window=10.0, key_field="path")
30
+ def run(self, path: str) -> str:
31
+ try:
32
+ path = expand_path(path)
33
+ self.report_action(
34
+ tr(
35
+ "📄 Outline file '{disp_path}' ...",
36
+ disp_path=display_path(path),
37
+ ),
38
+ ReportAction.READ,
39
+ )
40
+ ext = os.path.splitext(path)[1].lower()
41
+ with open(path, "r", encoding="utf-8", errors="replace") as f:
42
+ lines = f.readlines()
43
+ return self._outline_by_extension(ext, lines)
44
+ except Exception as e:
45
+ self.report_error(
46
+ tr("❌ Error reading file: {error}", error=e),
47
+ ReportAction.READ,
48
+ )
49
+ return tr("Error reading file: {error}", error=e)
50
+
51
+ def _outline_by_extension(self, ext, lines):
52
+ if ext == ".py":
53
+ outline_items = parse_python_outline(lines)
54
+ outline_type = "python"
55
+ table = OutlineFormatter.format_outline_table(outline_items)
56
+ self.report_success(
57
+ tr(
58
+ "✅ Outlined {count} {item_word}",
59
+ count=len(outline_items),
60
+ item_word=pluralize("item", len(outline_items)),
61
+ ),
62
+ ReportAction.READ,
63
+ )
64
+ return (
65
+ tr(
66
+ "Outline: {count} items ({outline_type})\n",
67
+ count=len(outline_items),
68
+ outline_type=outline_type,
69
+ )
70
+ + table
71
+ )
72
+ elif ext == ".md":
73
+ outline_items = parse_markdown_outline(lines)
74
+ outline_type = "markdown"
75
+ table = OutlineFormatter.format_markdown_outline_table(outline_items)
76
+ self.report_success(
77
+ tr(
78
+ "✅ Outlined {count} {item_word}",
79
+ count=len(outline_items),
80
+ item_word=pluralize("item", len(outline_items)),
81
+ ),
82
+ ReportAction.READ,
83
+ )
84
+ return (
85
+ tr(
86
+ "Outline: {count} items ({outline_type})\n",
87
+ count=len(outline_items),
88
+ outline_type=outline_type,
89
+ )
90
+ + table
91
+ )
92
+ elif ext == ".java":
93
+ outline_items = parse_java_outline(lines)
94
+ outline_type = "java"
95
+ table = OutlineFormatter.format_outline_table(outline_items)
96
+ self.report_success(
97
+ tr(
98
+ "✅ Outlined {count} {item_word}",
99
+ count=len(outline_items),
100
+ item_word=pluralize("item", len(outline_items)),
101
+ ),
102
+ ReportAction.READ,
103
+ )
104
+ return (
105
+ tr(
106
+ "Outline: {count} items ({outline_type})\n",
107
+ count=len(outline_items),
108
+ outline_type=outline_type,
109
+ )
110
+ + table
111
+ )
112
+ else:
113
+ outline_type = "default"
114
+ self.report_success(
115
+ tr("✅ Outlined {count} items", count=len(lines)),
116
+ ReportAction.READ,
117
+ )
118
+ return tr(
119
+ "Outline: {count} lines ({outline_type})\nFile has {count} lines.",
120
+ count=len(lines),
121
+ outline_type=outline_type,
122
+ )
@@ -0,0 +1,47 @@
1
+ import re
2
+ from typing import List, Dict
3
+
4
+
5
+ def parse_java_outline(lines: List[str]) -> List[Dict]:
6
+ """
7
+ Parses Java source code lines and extracts classes and methods with their signatures.
8
+ Returns a list of outline items: {type, name, return_type, parameters, generics, line}
9
+ """
10
+ outline = []
11
+ class_pattern = re.compile(r"\bclass\s+(\w+)(\s*<[^>]+>)?")
12
+ # Match methods with or without visibility modifiers (including package-private)
13
+ method_pattern = re.compile(
14
+ r"^(?:\s*(public|protected|private)\s+)?(?:static\s+)?([\w<>\[\]]+)\s+(\w+)\s*\(([^)]*)\)"
15
+ )
16
+ current_class = None
17
+ for idx, line in enumerate(lines, 1):
18
+ class_match = class_pattern.search(line)
19
+ if class_match:
20
+ class_name = class_match.group(1)
21
+ generics = class_match.group(2) or ""
22
+ outline.append(
23
+ {
24
+ "type": "class",
25
+ "name": class_name,
26
+ "generics": generics.strip("<>") if generics else None,
27
+ "line": idx,
28
+ }
29
+ )
30
+ current_class = class_name
31
+ else:
32
+ method_match = method_pattern.search(line)
33
+ if method_match:
34
+ return_type = method_match.group(2)
35
+ method_name = method_match.group(3)
36
+ params = method_match.group(4)
37
+ outline.append(
38
+ {
39
+ "type": "method",
40
+ "class": current_class,
41
+ "name": method_name,
42
+ "return_type": return_type,
43
+ "parameters": params.strip(),
44
+ "line": idx,
45
+ }
46
+ )
47
+ return outline
@@ -0,0 +1,14 @@
1
+ import re
2
+ from typing import List
3
+
4
+
5
+ def parse_markdown_outline(lines: List[str]):
6
+ header_pat = re.compile(r"^(#+)\s+(.*)")
7
+ outline = []
8
+ for idx, line in enumerate(lines):
9
+ match = header_pat.match(line)
10
+ if match:
11
+ level = len(match.group(1))
12
+ title = match.group(2).strip()
13
+ outline.append({"level": level, "title": title, "line": idx + 1})
14
+ return outline
@@ -0,0 +1,303 @@
1
+ import re
2
+ from typing import List
3
+
4
+
5
+ def handle_assignment(idx, assign_match, outline):
6
+ var_name = assign_match.group(2)
7
+ var_type = "const" if var_name.isupper() else "var"
8
+ outline.append(
9
+ {
10
+ "type": var_type,
11
+ "name": var_name,
12
+ "start": idx + 1,
13
+ "end": idx + 1,
14
+ "parent": "",
15
+ "docstring": "",
16
+ }
17
+ )
18
+
19
+
20
+ def handle_main(idx, outline):
21
+ outline.append(
22
+ {
23
+ "type": "main",
24
+ "name": "__main__",
25
+ "start": idx + 1,
26
+ "end": idx + 1,
27
+ "parent": "",
28
+ "docstring": "",
29
+ }
30
+ )
31
+
32
+
33
+ def close_stack_objects(idx, indent, stack, obj_ranges):
34
+ while stack and indent < stack[-1][2]:
35
+ popped = stack.pop()
36
+ obj_ranges.append((popped[0], popped[1], popped[3], idx, popped[4], popped[2]))
37
+
38
+
39
+ def close_last_top_obj(idx, last_top_obj, stack, obj_ranges):
40
+ if last_top_obj and last_top_obj in stack:
41
+ stack.remove(last_top_obj)
42
+ obj_ranges.append(
43
+ (
44
+ last_top_obj[0],
45
+ last_top_obj[1],
46
+ last_top_obj[3],
47
+ idx,
48
+ last_top_obj[4],
49
+ last_top_obj[2],
50
+ )
51
+ )
52
+ return None
53
+ return last_top_obj
54
+
55
+
56
+ def handle_class(idx, class_match, indent, stack, last_top_obj):
57
+ name = class_match.group(2)
58
+ parent = stack[-1][1] if stack and stack[-1][0] == "class" else ""
59
+ obj = ("class", name, indent, idx + 1, parent)
60
+ stack.append(obj)
61
+ if indent == 0:
62
+ last_top_obj = obj
63
+ return last_top_obj
64
+
65
+
66
+ def handle_function(idx, func_match, indent, stack, last_top_obj):
67
+ name = func_match.group(2)
68
+ parent = ""
69
+ for s in reversed(stack):
70
+ if s[0] == "class" and indent > s[2]:
71
+ parent = s[1]
72
+ break
73
+ obj = ("function", name, indent, idx + 1, parent)
74
+ stack.append(obj)
75
+ if indent == 0:
76
+ last_top_obj = obj
77
+ return last_top_obj
78
+
79
+
80
+ def process_line(idx, line, regexes, stack, obj_ranges, outline, last_top_obj):
81
+ class_pat, func_pat, assign_pat, main_pat = regexes
82
+ class_match = class_pat.match(line)
83
+ func_match = func_pat.match(line)
84
+ assign_match = assign_pat.match(line)
85
+ indent = len(line) - len(line.lstrip())
86
+ # If a new top-level class or function starts, close the previous one
87
+ if (class_match or func_match) and indent == 0 and last_top_obj:
88
+ last_top_obj = close_last_top_obj(idx, last_top_obj, stack, obj_ranges)
89
+ if class_match:
90
+ last_top_obj = handle_class(idx, class_match, indent, stack, last_top_obj)
91
+ elif func_match:
92
+ last_top_obj = handle_function(idx, func_match, indent, stack, last_top_obj)
93
+ elif assign_match and indent == 0:
94
+ handle_assignment(idx, assign_match, outline)
95
+ main_match = main_pat.match(line)
96
+ if main_match:
97
+ handle_main(idx, outline)
98
+ close_stack_objects(idx, indent, stack, obj_ranges)
99
+ return last_top_obj
100
+
101
+
102
+ def extract_signature_and_decorators(lines, start_idx):
103
+ """
104
+ Extracts the signature line and leading decorators for a given function/class/method.
105
+ Returns (signature:str, decorators:List[str], signature_lineno:int)
106
+ """
107
+ decorators = []
108
+ sig_line = None
109
+ sig_lineno = start_idx
110
+ for i in range(start_idx - 1, -1, -1):
111
+ striped = lines[i].strip()
112
+ if striped.startswith("@"):
113
+ decorators.insert(0, striped)
114
+ sig_lineno = i
115
+ elif not striped:
116
+ continue
117
+ else:
118
+ break
119
+ # Find the signature line itself
120
+ for k in range(start_idx, len(lines)):
121
+ striped = lines[k].strip()
122
+ if striped.startswith("def ") or striped.startswith("class "):
123
+ sig_line = striped
124
+ sig_lineno = k
125
+ break
126
+ return sig_line, decorators, sig_lineno
127
+
128
+
129
+ def extract_docstring(lines, start_idx, end_idx):
130
+ """Extracts a docstring from lines[start_idx:end_idx] if present."""
131
+ for i in range(start_idx, min(end_idx, len(lines))):
132
+ line = lines[i].lstrip()
133
+ if not line:
134
+ continue
135
+ if line.startswith('"""') or line.startswith("'''"):
136
+ quote = line[:3]
137
+ doc = line[3:]
138
+ if doc.strip().endswith(quote):
139
+ return doc.strip()[:-3].strip()
140
+ docstring_lines = [doc]
141
+ for j in range(i + 1, min(end_idx, len(lines))):
142
+ line = lines[j]
143
+ if line.strip().endswith(quote):
144
+ docstring_lines.append(line.strip()[:-3])
145
+ return "\n".join([d.strip() for d in docstring_lines]).strip()
146
+ docstring_lines.append(line)
147
+ break
148
+ else:
149
+ break
150
+ return ""
151
+
152
+
153
+ def build_outline_entry(obj, lines, outline):
154
+ obj_type, name, start, end, parent, indent = obj
155
+ # Determine if this is a method
156
+ if obj_type == "function" and parent:
157
+ outline_type = "method"
158
+ elif obj_type == "function":
159
+ outline_type = "function"
160
+ else:
161
+ outline_type = obj_type
162
+ docstring = extract_docstring(lines, start, end)
163
+ outline.append(
164
+ {
165
+ "type": outline_type,
166
+ "name": name,
167
+ "start": start,
168
+ "end": end,
169
+ "parent": parent,
170
+ "docstring": docstring,
171
+ }
172
+ )
173
+
174
+
175
+ def process_lines(lines, regexes):
176
+ outline = []
177
+ stack = []
178
+ obj_ranges = []
179
+ last_top_obj = None
180
+ for idx, line in enumerate(lines):
181
+ last_top_obj = process_line(
182
+ idx, line, regexes, stack, obj_ranges, outline, last_top_obj
183
+ )
184
+ # Close any remaining open objects
185
+ for popped in stack:
186
+ obj_ranges.append(
187
+ (popped[0], popped[1], popped[3], len(lines), popped[4], popped[2])
188
+ )
189
+ return outline, obj_ranges
190
+
191
+
192
+ def build_outline(obj_ranges, lines, outline):
193
+ for obj in obj_ranges:
194
+ build_outline_entry(obj, lines, outline)
195
+ return outline
196
+
197
+
198
+ def parse_python_outline(lines: List[str]):
199
+ class_pat = re.compile(r"^(\s*)class\s+(\w+)")
200
+ func_pat = re.compile(r"^(\s*)def\s+(\w+)")
201
+ assign_pat = re.compile(r"^(\s*)([A-Za-z_][A-Za-z0-9_]*)\s*=.*")
202
+ main_pat = re.compile(r"^\s*if\s+__name__\s*==\s*[\'\"]__main__[\'\"]\s*:")
203
+ outline = []
204
+ stack = []
205
+ obj_ranges = []
206
+ last_top_obj = None
207
+ for idx, line in enumerate(lines):
208
+ class_match = class_pat.match(line)
209
+ func_match = func_pat.match(line)
210
+ assign_match = assign_pat.match(line)
211
+ indent = len(line) - len(line.lstrip())
212
+ parent = ""
213
+ for s in reversed(stack):
214
+ if s[0] == "class" and indent > s[2]:
215
+ parent = s[1]
216
+ break
217
+ if class_match:
218
+ obj = ("class", class_match.group(2), idx + 1, None, parent, indent)
219
+ stack.append(obj)
220
+ last_top_obj = obj
221
+ elif func_match:
222
+ obj = ("function", func_match.group(2), idx + 1, None, parent, indent)
223
+ stack.append(obj)
224
+ last_top_obj = obj
225
+ elif assign_match and indent == 0:
226
+ outline.append(
227
+ {
228
+ "type": "const" if assign_match.group(2).isupper() else "var",
229
+ "name": assign_match.group(2),
230
+ "start": idx + 1,
231
+ "end": idx + 1,
232
+ "parent": "",
233
+ "signature": line.strip(),
234
+ "decorators": [],
235
+ "docstring": "",
236
+ }
237
+ )
238
+ if line.strip().startswith("if __name__ == "):
239
+ outline.append(
240
+ {
241
+ "type": "main",
242
+ "name": "__main__",
243
+ "start": idx + 1,
244
+ "end": idx + 1,
245
+ "parent": "",
246
+ "signature": line.strip(),
247
+ "decorators": [],
248
+ "docstring": "",
249
+ }
250
+ )
251
+ # Close stack objects if indent falls back
252
+ while stack and indent <= stack[-1][5] and idx + 1 > stack[-1][2]:
253
+ finished = stack.pop()
254
+ outline_entry = finished[:2] + (
255
+ finished[2],
256
+ idx + 1,
257
+ finished[4],
258
+ finished[5],
259
+ )
260
+ build_outline_entry(outline_entry, lines, outline)
261
+ # Close any remaining objects
262
+ while stack:
263
+ finished = stack.pop()
264
+ outline_entry = finished[:2] + (
265
+ finished[2],
266
+ len(lines),
267
+ finished[4],
268
+ finished[5],
269
+ )
270
+ build_outline_entry(outline_entry, lines, outline)
271
+ return outline
272
+
273
+ class_pat = re.compile(r"^(\s*)class\s+(\w+)")
274
+ func_pat = re.compile(r"^(\s*)def\s+(\w+)")
275
+ assign_pat = re.compile(r"^(\s*)([A-Za-z_][A-Za-z0-9_]*)\s*=.*")
276
+ main_pat = re.compile(r"^\s*if\s+__name__\s*==\s*[\'\"]__main__[\'\"]\s*:")
277
+ regexes = (class_pat, func_pat, assign_pat, main_pat)
278
+ outline, obj_ranges = process_lines(lines, regexes)
279
+ return build_outline(obj_ranges, lines, outline)
280
+
281
+
282
+ def extract_docstring(lines, start_idx, end_idx):
283
+ """Extracts a docstring from lines[start_idx:end_idx] if present."""
284
+ for i in range(start_idx, min(end_idx, len(lines))):
285
+ line = lines[i].lstrip()
286
+ if not line:
287
+ continue
288
+ if line.startswith('"""') or line.startswith("'''"):
289
+ quote = line[:3]
290
+ doc = line[3:]
291
+ if doc.strip().endswith(quote):
292
+ return doc.strip()[:-3].strip()
293
+ docstring_lines = [doc]
294
+ for j in range(i + 1, min(end_idx, len(lines))):
295
+ line = lines[j]
296
+ if line.strip().endswith(quote):
297
+ docstring_lines.append(line.strip()[:-3])
298
+ return "\n".join([d.strip() for d in docstring_lines]).strip()
299
+ docstring_lines.append(line)
300
+ break
301
+ else:
302
+ break
303
+ return ""
@@ -0,0 +1,36 @@
1
+ from janito.tools.tool_base import ToolBase, ToolPermissions
2
+ from janito.report_events import ReportAction
3
+ from janito.tools.loop_protection_decorator import protect_against_loops
4
+
5
+
6
+ class SearchOutlineTool(ToolBase):
7
+ """
8
+ Tool for searching outlines in files.
9
+
10
+ Args:
11
+ path (str): Path to the file for which to generate an outline.
12
+ Returns:
13
+ str: Outline search result or status message.
14
+ """
15
+
16
+ permissions = ToolPermissions(read=True)
17
+ tool_name = "search_outline"
18
+
19
+ @protect_against_loops(max_calls=5, time_window=10.0, key_field="path")
20
+ def run(self, path: str) -> str:
21
+ from janito.tools.tool_utils import display_path
22
+ from janito.i18n import tr
23
+
24
+ self.report_action(
25
+ tr(
26
+ "🔍 Searching for outline in '{disp_path}'",
27
+ disp_path=display_path(path),
28
+ ),
29
+ ReportAction.READ,
30
+ )
31
+ # ... rest of implementation ...
32
+ # Example warnings and successes:
33
+ # self.report_warning(tr("No files found with supported extensions."))
34
+ # self.report_warning(tr("Error reading {path}: {error}", path=path, error=e))
35
+ # self.report_success(tr("✅ {count} {match_word} found", count=len(output), match_word=pluralize('match', len(output))))
36
+ pass