Skip to content
This repository was archived by the owner on Apr 4, 2024. It is now read-only.

Commit 417eaf6

Browse files
authored
recordCall and CallStack/CallLocation (#35 closes #31)
2 parents 2aededc + 336d916 commit 417eaf6

File tree

2 files changed

+126
-0
lines changed

2 files changed

+126
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
from typing import List, Optional
2+
from selfie_lib.CommentTracker import SnapshotFileLayout
3+
import inspect
4+
from functools import total_ordering
5+
6+
7+
@total_ordering
8+
class CallLocation:
9+
def __init__(self, file_name: Optional[str], line: int):
10+
self._file_name = file_name
11+
self._line = line
12+
13+
@property
14+
def file_name(self) -> Optional[str]:
15+
return self._file_name
16+
17+
@property
18+
def line(self) -> int:
19+
return self._line
20+
21+
def with_line(self, line: int) -> "CallLocation":
22+
return CallLocation(self._file_name, line)
23+
24+
def ide_link(self, layout: "SnapshotFileLayout") -> str:
25+
return f"File: {self._file_name}, Line: {self._line}"
26+
27+
def same_path_as(self, other: "CallLocation") -> bool:
28+
if not isinstance(other, CallLocation):
29+
return False
30+
return self._file_name == other.file_name
31+
32+
def source_filename_without_extension(self) -> str:
33+
if self._file_name is not None:
34+
return self._file_name.rsplit(".", 1)[0]
35+
return ""
36+
37+
def __lt__(self, other) -> bool:
38+
if not isinstance(other, CallLocation):
39+
return NotImplemented
40+
return (self._file_name, self._line) < (other.file_name, other.line)
41+
42+
def __eq__(self, other) -> bool:
43+
if not isinstance(other, CallLocation):
44+
return NotImplemented
45+
return (self._file_name, self._line) == (other.file_name, other.line)
46+
47+
48+
class CallStack:
49+
def __init__(self, location: CallLocation, rest_of_stack: List[CallLocation]):
50+
self.location = location
51+
self.rest_of_stack = rest_of_stack
52+
53+
def ide_link(self, layout: "SnapshotFileLayout") -> str:
54+
links = [self.location.ide_link(layout)] + [
55+
loc.ide_link(layout) for loc in self.rest_of_stack
56+
]
57+
return "\n".join(links)
58+
59+
60+
def recordCall(callerFileOnly: bool = False) -> CallStack:
61+
stack_frames = inspect.stack()[1:]
62+
63+
if callerFileOnly:
64+
caller_file = stack_frames[0].filename
65+
stack_frames = [
66+
frame for frame in stack_frames if frame.filename == caller_file
67+
]
68+
69+
call_locations = [
70+
CallLocation(frame.filename, frame.lineno) for frame in stack_frames
71+
]
72+
73+
location = call_locations[0]
74+
rest_of_stack = call_locations[1:]
75+
76+
return CallStack(location, rest_of_stack)
+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
from unittest.mock import Mock
2+
from selfie_lib.WriteTracker import CallLocation, CallStack, recordCall
3+
import os
4+
5+
6+
def test_call_location_ide_link():
7+
layout = Mock()
8+
location = CallLocation(file_name="example.py", line=10)
9+
expected_link = "File: example.py, Line: 10"
10+
11+
assert location.ide_link(layout) == expected_link
12+
13+
14+
def test_call_stack_ide_link():
15+
layout = Mock()
16+
location1 = CallLocation(file_name="example1.py", line=10)
17+
location2 = CallLocation(file_name="example2.py", line=20)
18+
call_stack = CallStack(location=location1, rest_of_stack=[location2])
19+
20+
expected_links = "File: example1.py, Line: 10\nFile: example2.py, Line: 20"
21+
assert call_stack.ide_link(layout) == expected_links
22+
23+
24+
def test_record_call_with_caller_file_only_false():
25+
call_stack = recordCall(False)
26+
27+
assert (
28+
len(call_stack.rest_of_stack) > 0
29+
), "Expected the rest of stack to contain more than one CallLocation"
30+
31+
expected_call_location_str = "File: RecordCall_test.py, Line: 25"
32+
33+
if call_stack.location.file_name is not None:
34+
actual_file_name = os.path.basename(call_stack.location.file_name)
35+
actual_call_location_str = (
36+
f"File: {actual_file_name}, Line: {call_stack.location.line}"
37+
)
38+
else:
39+
actual_call_location_str = "File name is None"
40+
41+
assert (
42+
actual_call_location_str == expected_call_location_str
43+
), f"Expected call location to be '{expected_call_location_str}' but was '{actual_call_location_str}'"
44+
45+
46+
def test_record_call_with_caller_file_only_true():
47+
call_stack = recordCall(True)
48+
assert (
49+
len(call_stack.rest_of_stack) >= 0
50+
), "Expected the rest of stack to potentially contain only the caller's file location"

0 commit comments

Comments
 (0)