janito 1.12.2__py3-none-any.whl → 1.13.1__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/__init__.py +1 -1
- janito/agent/tools/python_command_runner.py +80 -28
- janito/agent/tools/python_file_runner.py +79 -27
- janito/agent/tools/python_stdin_runner.py +83 -28
- janito/agent/tools/replace_text_in_file.py +5 -7
- janito/agent/tools/run_bash_command.py +172 -82
- janito/agent/tools/run_powershell_command.py +149 -111
- janito/agent/tools/search_text/core.py +1 -1
- janito/agent/tools/search_text/pattern_utils.py +1 -1
- janito/agent/tools/validate_file_syntax/core.py +8 -8
- janito/agent/tools/validate_file_syntax/markdown_validator.py +2 -2
- janito/cli/arg_parser.py +5 -0
- janito/cli/cli_main.py +28 -17
- janito/shell/commands/utility.py +3 -0
- janito/shell/main.py +56 -31
- janito/shell/prompt/session_setup.py +1 -0
- janito/shell/ui/interactive.py +7 -2
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/METADATA +4 -3
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/RECORD +23 -24
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/WHEEL +1 -1
- janito/shell/commands.py +0 -40
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/entry_points.txt +0 -0
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/licenses/LICENSE +0 -0
- {janito-1.12.2.dist-info → janito-1.13.1.dist-info}/top_level.txt +0 -0
janito/__init__.py
CHANGED
@@ -1 +1 @@
|
|
1
|
-
__version__ = "1.
|
1
|
+
__version__ = "1.13.1"
|
@@ -7,6 +7,7 @@ from janito.agent.tool_base import ToolBase
|
|
7
7
|
from janito.agent.tools_utils.action_type import ActionType
|
8
8
|
from janito.agent.tool_registry import register_tool
|
9
9
|
from janito.i18n import tr
|
10
|
+
from janito.agent.runtime_config import runtime_config
|
10
11
|
|
11
12
|
|
12
13
|
@register_tool(name="python_command_runner")
|
@@ -22,26 +23,13 @@ class PythonCommandRunnerTool(ToolBase):
|
|
22
23
|
|
23
24
|
def run(self, code: str, timeout: int = 60) -> str:
|
24
25
|
if not code.strip():
|
25
|
-
self.report_warning(tr("
|
26
|
+
self.report_warning(tr("539 Empty code provided."))
|
26
27
|
return tr("Warning: Empty code provided. Operation skipped.")
|
27
28
|
self.report_info(
|
28
|
-
ActionType.EXECUTE, tr("
|
29
|
+
ActionType.EXECUTE, tr("40d Running: python -c ...\n{code}\n", code=code)
|
29
30
|
)
|
30
31
|
try:
|
31
|
-
|
32
|
-
tempfile.NamedTemporaryFile(
|
33
|
-
mode="w+",
|
34
|
-
prefix="python_cmd_stdout_",
|
35
|
-
delete=False,
|
36
|
-
encoding="utf-8",
|
37
|
-
) as stdout_file,
|
38
|
-
tempfile.NamedTemporaryFile(
|
39
|
-
mode="w+",
|
40
|
-
prefix="python_cmd_stderr_",
|
41
|
-
delete=False,
|
42
|
-
encoding="utf-8",
|
43
|
-
) as stderr_file,
|
44
|
-
):
|
32
|
+
if runtime_config.get("all_out"):
|
45
33
|
process = subprocess.Popen(
|
46
34
|
[sys.executable, "-c", code],
|
47
35
|
stdout=subprocess.PIPE,
|
@@ -52,24 +40,88 @@ class PythonCommandRunnerTool(ToolBase):
|
|
52
40
|
encoding="utf-8",
|
53
41
|
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
54
42
|
)
|
55
|
-
|
56
|
-
|
43
|
+
stdout_accum = []
|
44
|
+
stderr_accum = []
|
45
|
+
|
46
|
+
def read_stream(stream, report_func, accum):
|
47
|
+
for line in stream:
|
48
|
+
accum.append(line)
|
49
|
+
report_func(line)
|
50
|
+
|
51
|
+
stdout_thread = threading.Thread(
|
52
|
+
target=read_stream,
|
53
|
+
args=(process.stdout, self.report_stdout, stdout_accum),
|
54
|
+
)
|
55
|
+
stderr_thread = threading.Thread(
|
56
|
+
target=read_stream,
|
57
|
+
args=(process.stderr, self.report_stderr, stderr_accum),
|
57
58
|
)
|
58
|
-
|
59
|
-
|
59
|
+
stdout_thread.start()
|
60
|
+
stderr_thread.start()
|
61
|
+
try:
|
62
|
+
return_code = process.wait(timeout=timeout)
|
63
|
+
except subprocess.TimeoutExpired:
|
64
|
+
process.kill()
|
65
|
+
self.report_error(
|
66
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
67
|
+
)
|
60
68
|
return tr(
|
61
69
|
"Code timed out after {timeout} seconds.", timeout=timeout
|
62
70
|
)
|
63
|
-
|
64
|
-
|
71
|
+
stdout_thread.join()
|
72
|
+
stderr_thread.join()
|
65
73
|
self.report_success(
|
66
|
-
tr("
|
67
|
-
)
|
68
|
-
return self._format_result(
|
69
|
-
stdout_file.name, stderr_file.name, return_code
|
74
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
70
75
|
)
|
76
|
+
stdout = "".join(stdout_accum)
|
77
|
+
stderr = "".join(stderr_accum)
|
78
|
+
result = f"Return code: {return_code}\n--- STDOUT ---\n{stdout}"
|
79
|
+
if stderr and stderr.strip():
|
80
|
+
result += f"\n--- STDERR ---\n{stderr}"
|
81
|
+
return result
|
82
|
+
else:
|
83
|
+
with (
|
84
|
+
tempfile.NamedTemporaryFile(
|
85
|
+
mode="w+",
|
86
|
+
prefix="python_cmd_stdout_",
|
87
|
+
delete=False,
|
88
|
+
encoding="utf-8",
|
89
|
+
) as stdout_file,
|
90
|
+
tempfile.NamedTemporaryFile(
|
91
|
+
mode="w+",
|
92
|
+
prefix="python_cmd_stderr_",
|
93
|
+
delete=False,
|
94
|
+
encoding="utf-8",
|
95
|
+
) as stderr_file,
|
96
|
+
):
|
97
|
+
process = subprocess.Popen(
|
98
|
+
[sys.executable, "-c", code],
|
99
|
+
stdout=subprocess.PIPE,
|
100
|
+
stderr=subprocess.PIPE,
|
101
|
+
text=True,
|
102
|
+
bufsize=1,
|
103
|
+
universal_newlines=True,
|
104
|
+
encoding="utf-8",
|
105
|
+
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
106
|
+
)
|
107
|
+
stdout_lines, stderr_lines = self._stream_process_output(
|
108
|
+
process, stdout_file, stderr_file
|
109
|
+
)
|
110
|
+
return_code = self._wait_for_process(process, timeout)
|
111
|
+
if return_code is None:
|
112
|
+
return tr(
|
113
|
+
"Code timed out after {timeout} seconds.", timeout=timeout
|
114
|
+
)
|
115
|
+
stdout_file.flush()
|
116
|
+
stderr_file.flush()
|
117
|
+
self.report_success(
|
118
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
119
|
+
)
|
120
|
+
return self._format_result(
|
121
|
+
stdout_file.name, stderr_file.name, return_code
|
122
|
+
)
|
71
123
|
except Exception as e:
|
72
|
-
self.report_error(tr("
|
124
|
+
self.report_error(tr("534 Error: {error}", error=e))
|
73
125
|
return tr("Error running code: {error}", error=e)
|
74
126
|
|
75
127
|
def _stream_process_output(self, process, stdout_file, stderr_file):
|
@@ -107,7 +159,7 @@ class PythonCommandRunnerTool(ToolBase):
|
|
107
159
|
except subprocess.TimeoutExpired:
|
108
160
|
process.kill()
|
109
161
|
self.report_error(
|
110
|
-
tr("
|
162
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
111
163
|
)
|
112
164
|
return None
|
113
165
|
|
@@ -7,6 +7,7 @@ from janito.agent.tool_base import ToolBase
|
|
7
7
|
from janito.agent.tools_utils.action_type import ActionType
|
8
8
|
from janito.agent.tool_registry import register_tool
|
9
9
|
from janito.i18n import tr
|
10
|
+
from janito.agent.runtime_config import runtime_config
|
10
11
|
|
11
12
|
|
12
13
|
@register_tool(name="python_file_runner")
|
@@ -23,23 +24,10 @@ class PythonFileRunnerTool(ToolBase):
|
|
23
24
|
def run(self, file_path: str, timeout: int = 60) -> str:
|
24
25
|
self.report_info(
|
25
26
|
ActionType.EXECUTE,
|
26
|
-
tr("
|
27
|
+
tr("680 Running: python {file_path}", file_path=file_path),
|
27
28
|
)
|
28
29
|
try:
|
29
|
-
|
30
|
-
tempfile.NamedTemporaryFile(
|
31
|
-
mode="w+",
|
32
|
-
prefix="python_file_stdout_",
|
33
|
-
delete=False,
|
34
|
-
encoding="utf-8",
|
35
|
-
) as stdout_file,
|
36
|
-
tempfile.NamedTemporaryFile(
|
37
|
-
mode="w+",
|
38
|
-
prefix="python_file_stderr_",
|
39
|
-
delete=False,
|
40
|
-
encoding="utf-8",
|
41
|
-
) as stderr_file,
|
42
|
-
):
|
30
|
+
if runtime_config.get("all_out"):
|
43
31
|
process = subprocess.Popen(
|
44
32
|
[sys.executable, file_path],
|
45
33
|
stdout=subprocess.PIPE,
|
@@ -50,24 +38,88 @@ class PythonFileRunnerTool(ToolBase):
|
|
50
38
|
encoding="utf-8",
|
51
39
|
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
52
40
|
)
|
53
|
-
|
54
|
-
|
41
|
+
stdout_accum = []
|
42
|
+
stderr_accum = []
|
43
|
+
|
44
|
+
def read_stream(stream, report_func, accum):
|
45
|
+
for line in stream:
|
46
|
+
accum.append(line)
|
47
|
+
report_func(line)
|
48
|
+
|
49
|
+
stdout_thread = threading.Thread(
|
50
|
+
target=read_stream,
|
51
|
+
args=(process.stdout, self.report_stdout, stdout_accum),
|
52
|
+
)
|
53
|
+
stderr_thread = threading.Thread(
|
54
|
+
target=read_stream,
|
55
|
+
args=(process.stderr, self.report_stderr, stderr_accum),
|
55
56
|
)
|
56
|
-
|
57
|
-
|
57
|
+
stdout_thread.start()
|
58
|
+
stderr_thread.start()
|
59
|
+
try:
|
60
|
+
return_code = process.wait(timeout=timeout)
|
61
|
+
except subprocess.TimeoutExpired:
|
62
|
+
process.kill()
|
63
|
+
self.report_error(
|
64
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
65
|
+
)
|
58
66
|
return tr(
|
59
67
|
"Code timed out after {timeout} seconds.", timeout=timeout
|
60
68
|
)
|
61
|
-
|
62
|
-
|
69
|
+
stdout_thread.join()
|
70
|
+
stderr_thread.join()
|
63
71
|
self.report_success(
|
64
|
-
tr("
|
65
|
-
)
|
66
|
-
return self._format_result(
|
67
|
-
stdout_file.name, stderr_file.name, return_code
|
72
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
68
73
|
)
|
74
|
+
stdout = "".join(stdout_accum)
|
75
|
+
stderr = "".join(stderr_accum)
|
76
|
+
result = f"Return code: {return_code}\n--- STDOUT ---\n{stdout}"
|
77
|
+
if stderr and stderr.strip():
|
78
|
+
result += f"\n--- STDERR ---\n{stderr}"
|
79
|
+
return result
|
80
|
+
else:
|
81
|
+
with (
|
82
|
+
tempfile.NamedTemporaryFile(
|
83
|
+
mode="w+",
|
84
|
+
prefix="python_file_stdout_",
|
85
|
+
delete=False,
|
86
|
+
encoding="utf-8",
|
87
|
+
) as stdout_file,
|
88
|
+
tempfile.NamedTemporaryFile(
|
89
|
+
mode="w+",
|
90
|
+
prefix="python_file_stderr_",
|
91
|
+
delete=False,
|
92
|
+
encoding="utf-8",
|
93
|
+
) as stderr_file,
|
94
|
+
):
|
95
|
+
process = subprocess.Popen(
|
96
|
+
[sys.executable, file_path],
|
97
|
+
stdout=subprocess.PIPE,
|
98
|
+
stderr=subprocess.PIPE,
|
99
|
+
text=True,
|
100
|
+
bufsize=1,
|
101
|
+
universal_newlines=True,
|
102
|
+
encoding="utf-8",
|
103
|
+
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
104
|
+
)
|
105
|
+
stdout_lines, stderr_lines = self._stream_process_output(
|
106
|
+
process, stdout_file, stderr_file
|
107
|
+
)
|
108
|
+
return_code = self._wait_for_process(process, timeout)
|
109
|
+
if return_code is None:
|
110
|
+
return tr(
|
111
|
+
"Code timed out after {timeout} seconds.", timeout=timeout
|
112
|
+
)
|
113
|
+
stdout_file.flush()
|
114
|
+
stderr_file.flush()
|
115
|
+
self.report_success(
|
116
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
117
|
+
)
|
118
|
+
return self._format_result(
|
119
|
+
stdout_file.name, stderr_file.name, return_code
|
120
|
+
)
|
69
121
|
except Exception as e:
|
70
|
-
self.report_error(tr("
|
122
|
+
self.report_error(tr("534 Error: {error}", error=e))
|
71
123
|
return tr("Error running file: {error}", error=e)
|
72
124
|
|
73
125
|
def _stream_process_output(self, process, stdout_file, stderr_file):
|
@@ -105,7 +157,7 @@ class PythonFileRunnerTool(ToolBase):
|
|
105
157
|
except subprocess.TimeoutExpired:
|
106
158
|
process.kill()
|
107
159
|
self.report_error(
|
108
|
-
tr("
|
160
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
109
161
|
)
|
110
162
|
return None
|
111
163
|
|
@@ -7,6 +7,7 @@ from janito.agent.tool_base import ToolBase
|
|
7
7
|
from janito.agent.tools_utils.action_type import ActionType
|
8
8
|
from janito.agent.tool_registry import register_tool
|
9
9
|
from janito.i18n import tr
|
10
|
+
from janito.agent.runtime_config import runtime_config
|
10
11
|
|
11
12
|
|
12
13
|
@register_tool(name="python_stdin_runner")
|
@@ -22,27 +23,14 @@ class PythonStdinRunnerTool(ToolBase):
|
|
22
23
|
|
23
24
|
def run(self, code: str, timeout: int = 60) -> str:
|
24
25
|
if not code.strip():
|
25
|
-
self.report_warning(tr("
|
26
|
+
self.report_warning(tr("ℹ️ Empty code provided."))
|
26
27
|
return tr("Warning: Empty code provided. Operation skipped.")
|
27
28
|
self.report_info(
|
28
29
|
ActionType.EXECUTE,
|
29
|
-
tr("
|
30
|
+
tr("5e1 Running: python (stdin mode) ...\n{code}\n", code=code),
|
30
31
|
)
|
31
32
|
try:
|
32
|
-
|
33
|
-
tempfile.NamedTemporaryFile(
|
34
|
-
mode="w+",
|
35
|
-
prefix="python_stdin_stdout_",
|
36
|
-
delete=False,
|
37
|
-
encoding="utf-8",
|
38
|
-
) as stdout_file,
|
39
|
-
tempfile.NamedTemporaryFile(
|
40
|
-
mode="w+",
|
41
|
-
prefix="python_stdin_stderr_",
|
42
|
-
delete=False,
|
43
|
-
encoding="utf-8",
|
44
|
-
) as stderr_file,
|
45
|
-
):
|
33
|
+
if runtime_config.get("all_out"):
|
46
34
|
process = subprocess.Popen(
|
47
35
|
[sys.executable],
|
48
36
|
stdin=subprocess.PIPE,
|
@@ -54,24 +42,91 @@ class PythonStdinRunnerTool(ToolBase):
|
|
54
42
|
encoding="utf-8",
|
55
43
|
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
56
44
|
)
|
57
|
-
|
58
|
-
|
45
|
+
stdout_accum = []
|
46
|
+
stderr_accum = []
|
47
|
+
|
48
|
+
def read_stream(stream, report_func, accum):
|
49
|
+
for line in stream:
|
50
|
+
accum.append(line)
|
51
|
+
report_func(line)
|
52
|
+
|
53
|
+
stdout_thread = threading.Thread(
|
54
|
+
target=read_stream,
|
55
|
+
args=(process.stdout, self.report_stdout, stdout_accum),
|
56
|
+
)
|
57
|
+
stderr_thread = threading.Thread(
|
58
|
+
target=read_stream,
|
59
|
+
args=(process.stderr, self.report_stderr, stderr_accum),
|
59
60
|
)
|
60
|
-
|
61
|
-
|
61
|
+
stdout_thread.start()
|
62
|
+
stderr_thread.start()
|
63
|
+
process.stdin.write(code)
|
64
|
+
process.stdin.close()
|
65
|
+
try:
|
66
|
+
return_code = process.wait(timeout=timeout)
|
67
|
+
except subprocess.TimeoutExpired:
|
68
|
+
process.kill()
|
69
|
+
self.report_error(
|
70
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
71
|
+
)
|
62
72
|
return tr(
|
63
73
|
"Code timed out after {timeout} seconds.", timeout=timeout
|
64
74
|
)
|
65
|
-
|
66
|
-
|
75
|
+
stdout_thread.join()
|
76
|
+
stderr_thread.join()
|
67
77
|
self.report_success(
|
68
|
-
tr("
|
69
|
-
)
|
70
|
-
return self._format_result(
|
71
|
-
stdout_file.name, stderr_file.name, return_code
|
78
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
72
79
|
)
|
80
|
+
stdout = "".join(stdout_accum)
|
81
|
+
stderr = "".join(stderr_accum)
|
82
|
+
result = f"Return code: {return_code}\n--- STDOUT ---\n{stdout}"
|
83
|
+
if stderr and stderr.strip():
|
84
|
+
result += f"\n--- STDERR ---\n{stderr}"
|
85
|
+
return result
|
86
|
+
else:
|
87
|
+
with (
|
88
|
+
tempfile.NamedTemporaryFile(
|
89
|
+
mode="w+",
|
90
|
+
prefix="python_stdin_stdout_",
|
91
|
+
delete=False,
|
92
|
+
encoding="utf-8",
|
93
|
+
) as stdout_file,
|
94
|
+
tempfile.NamedTemporaryFile(
|
95
|
+
mode="w+",
|
96
|
+
prefix="python_stdin_stderr_",
|
97
|
+
delete=False,
|
98
|
+
encoding="utf-8",
|
99
|
+
) as stderr_file,
|
100
|
+
):
|
101
|
+
process = subprocess.Popen(
|
102
|
+
[sys.executable],
|
103
|
+
stdin=subprocess.PIPE,
|
104
|
+
stdout=subprocess.PIPE,
|
105
|
+
stderr=subprocess.PIPE,
|
106
|
+
text=True,
|
107
|
+
bufsize=1,
|
108
|
+
universal_newlines=True,
|
109
|
+
encoding="utf-8",
|
110
|
+
env={**os.environ, "PYTHONIOENCODING": "utf-8"},
|
111
|
+
)
|
112
|
+
stdout_lines, stderr_lines = self._stream_process_output(
|
113
|
+
process, stdout_file, stderr_file, code
|
114
|
+
)
|
115
|
+
return_code = self._wait_for_process(process, timeout)
|
116
|
+
if return_code is None:
|
117
|
+
return tr(
|
118
|
+
"Code timed out after {timeout} seconds.", timeout=timeout
|
119
|
+
)
|
120
|
+
stdout_file.flush()
|
121
|
+
stderr_file.flush()
|
122
|
+
self.report_success(
|
123
|
+
tr("197 Return code {return_code}", return_code=return_code)
|
124
|
+
)
|
125
|
+
return self._format_result(
|
126
|
+
stdout_file.name, stderr_file.name, return_code
|
127
|
+
)
|
73
128
|
except Exception as e:
|
74
|
-
self.report_error(tr("
|
129
|
+
self.report_error(tr("534 Error: {error}", error=e))
|
75
130
|
return tr("Error running code via stdin: {error}", error=e)
|
76
131
|
|
77
132
|
def _stream_process_output(self, process, stdout_file, stderr_file, code):
|
@@ -111,7 +166,7 @@ class PythonStdinRunnerTool(ToolBase):
|
|
111
166
|
except subprocess.TimeoutExpired:
|
112
167
|
process.kill()
|
113
168
|
self.report_error(
|
114
|
-
tr("
|
169
|
+
tr("6d1 Timed out after {timeout} seconds.", timeout=timeout)
|
115
170
|
)
|
116
171
|
return None
|
117
172
|
|
@@ -93,7 +93,7 @@ class ReplaceTextInFileTool(ToolBase):
|
|
93
93
|
final_msg += f"\n{validation_result}"
|
94
94
|
return final_msg
|
95
95
|
except Exception as e:
|
96
|
-
self.report_error(tr("
|
96
|
+
self.report_error(tr(" ❌ Error"))
|
97
97
|
return tr("Error replacing text: {error}", error=e)
|
98
98
|
|
99
99
|
def _read_file_content(self, file_path):
|
@@ -147,12 +147,12 @@ class ReplaceTextInFileTool(ToolBase):
|
|
147
147
|
if replaced_count == 0:
|
148
148
|
warning = tr(" [Warning: Search text not found in file]")
|
149
149
|
if not file_changed:
|
150
|
-
self.report_warning(tr("
|
150
|
+
self.report_warning(tr(" ℹ️ No changes made. [not found]"))
|
151
151
|
concise_warning = tr(
|
152
152
|
"No changes made. The search text was not found. Expand your search context with surrounding lines if needed."
|
153
153
|
)
|
154
154
|
if occurrences > 1 and replaced_count == 0:
|
155
|
-
self.report_warning(tr("
|
155
|
+
self.report_warning(tr(" ℹ️ No changes made. [not unique]"))
|
156
156
|
concise_warning = tr(
|
157
157
|
"No changes made. The search text is not unique. Expand your search context with surrounding lines to ensure uniqueness."
|
158
158
|
)
|
@@ -162,11 +162,9 @@ class ReplaceTextInFileTool(ToolBase):
|
|
162
162
|
"""Report success with line numbers where replacements occurred."""
|
163
163
|
if match_lines:
|
164
164
|
lines_str = ", ".join(str(line_no) for line_no in match_lines)
|
165
|
-
self.report_success(
|
166
|
-
tr(" \u2705 replaced at {lines_str}", lines_str=lines_str)
|
167
|
-
)
|
165
|
+
self.report_success(tr(" ✅ replaced at {lines_str}", lines_str=lines_str))
|
168
166
|
else:
|
169
|
-
self.report_success(tr("
|
167
|
+
self.report_success(tr(" ✅ replaced (lines unknown)"))
|
170
168
|
|
171
169
|
def _get_line_delta_str(self, content, new_content):
|
172
170
|
"""Return a string describing the net line change after replacement."""
|