summaryrefslogtreecommitdiff
path: root/cmd2/history.py
diff options
context:
space:
mode:
Diffstat (limited to 'cmd2/history.py')
-rw-r--r--cmd2/history.py124
1 files changed, 59 insertions, 65 deletions
diff --git a/cmd2/history.py b/cmd2/history.py
index bc6c32ce..78377c06 100644
--- a/cmd2/history.py
+++ b/cmd2/history.py
@@ -5,7 +5,9 @@ History management classes
import re
from typing import (
- List,
+ Callable,
+ Dict,
+ Optional,
Union,
)
@@ -27,7 +29,6 @@ class HistoryItem:
_ex_listformat = ' {:>4}x {}'
statement = attr.ib(default=None, validator=attr.validators.instance_of(Statement))
- idx = attr.ib(default=None, validator=attr.validators.instance_of(int))
def __str__(self):
"""A convenient human readable representation of the history item"""
@@ -50,20 +51,24 @@ class HistoryItem:
"""
return self.statement.expanded_command_line
- def pr(self, script=False, expanded=False, verbose=False) -> str:
+ def pr(self, idx: int, script: bool = False, expanded: bool = False, verbose: bool = False) -> str:
"""Represent this item in a pretty fashion suitable for printing.
If you pass verbose=True, script and expanded will be ignored
+ :param idx: The 1-based index of this item in the history list
+ :param script: True if formatting for a script (No item numbers)
+ :param expanded: True if expanded command line should be printed
+ :param verbose: True if expanded and raw should both appear when they are different
:return: pretty print string version of a HistoryItem
"""
if verbose:
raw = self.raw.rstrip()
expanded = self.expanded
- ret_str = self._listformat.format(self.idx, raw)
+ ret_str = self._listformat.format(idx, raw)
if raw != expanded:
- ret_str += '\n' + self._ex_listformat.format(self.idx, expanded)
+ ret_str += '\n' + self._ex_listformat.format(idx, expanded)
else:
if expanded:
ret_str = self.expanded
@@ -80,7 +85,7 @@ class HistoryItem:
# Display a numbered list if not writing to a script
if not script:
- ret_str = self._listformat.format(self.idx, ret_str)
+ ret_str = self._listformat.format(idx, ret_str)
return ret_str
@@ -121,7 +126,7 @@ class History(list):
:param new: Statement object which will be composed into a HistoryItem
and added to the end of the list
"""
- history_item = HistoryItem(new, len(self) + 1)
+ history_item = HistoryItem(new)
super().append(history_item)
def clear(self) -> None:
@@ -129,13 +134,12 @@ class History(list):
super().clear()
self.start_session()
- def get(self, index: Union[int, str]) -> HistoryItem:
+ def get(self, index: int) -> HistoryItem:
"""Get item from the History list using 1-based indexing.
- :param index: optional item to get (index as either integer or string)
+ :param index: optional item to get
:return: a single :class:`~cmd2.history.HistoryItem`
"""
- index = int(index)
if index == 0:
raise IndexError('The first command in history is command 1.')
elif index < 0:
@@ -155,8 +159,7 @@ class History(list):
# This regex will match 1, -1, 10, -10, but not 0 or -0.
#
# (?P<separator>:|(\.{2,}))? create a capture group named 'separator' which matches either
- # a colon or two periods. This group is optional so we can
- # match a string like '3'
+ # a colon or two periods.
#
# (?P<end>-?[1-9]{1}\d*)? create a capture group named 'end' which matches an
# optional minus sign, followed by exactly one non-zero
@@ -168,19 +171,18 @@ class History(list):
# \s*$ match any whitespace at the end of the input. This is here so
# you don't have to trim the input
#
- spanpattern = re.compile(r'^\s*(?P<start>-?[1-9]\d*)?(?P<separator>:|(\.{2,}))?(?P<end>-?[1-9]\d*)?\s*$')
+ spanpattern = re.compile(r'^\s*(?P<start>-?[1-9]\d*)?(?P<separator>:|(\.{2,}))(?P<end>-?[1-9]\d*)?\s*$')
- def span(self, span: str, include_persisted: bool = False) -> List[HistoryItem]:
- """Return an index or slice of the History list,
+ def span(self, span: str, include_persisted: bool = False) -> Dict[int, HistoryItem]:
+ """Return a slice of the History list
:param span: string containing an index or a slice
:param include_persisted: if True, then retrieve full results including from persisted history
- :return: a list of HistoryItems
+ :return: a dictionary of history items keyed by their 1-based index in ascending order,
+ or an empty dictionary if no results were found
This method can accommodate input in any of these forms:
- a
- -a
a..b or a:b
a.. or a:
..a or :a
@@ -197,89 +199,67 @@ class History(list):
- off by one errors
"""
- if span.lower() in ('*', '-', 'all'):
- span = ':'
results = self.spanpattern.search(span)
if not results:
# our regex doesn't match the input, bail out
raise ValueError('History indices must be positive or negative integers, and may not be zero.')
- sep = results.group('separator')
start = results.group('start')
if start:
- start = self._zero_based_index(start)
+ start = min(self._zero_based_index(start), len(self) - 1)
+ if start < 0:
+ start = max(0, len(self) + start)
+ else:
+ start = 0 if include_persisted else self.session_start_index
+
end = results.group('end')
if end:
- end = int(end)
- # modify end so it's inclusive of the last element
- if end == -1:
- # -1 as the end means include the last command in the array, which in pythonic
- # terms means to not provide an ending index. If you put -1 as the ending index
- # python excludes the last item in the list.
- end = None
- elif end < -1:
- # if the ending is smaller than -1, make it one larger so it includes
- # the element (python native indices exclude the last referenced element)
- end += 1
-
- if start is not None and end is not None:
- # we have both start and end, return a slice of history
- result = self[start:end]
- elif start is not None and sep is not None:
- # take a slice of the array
- result = self[start:]
- elif end is not None and sep is not None:
- if include_persisted:
- result = self[:end]
- else:
- result = self[self.session_start_index : end]
- elif start is not None:
- # there was no separator so it's either a positive or negative integer
- result = [self[start]]
+ end = min(int(end), len(self))
+ if end < 0:
+ end = max(0, len(self) + end + 1)
else:
- # we just have a separator, return the whole list
- if include_persisted:
- result = self[:]
- else:
- result = self[self.session_start_index :]
- return result
+ end = len(self)
- def str_search(self, search: str, include_persisted: bool = False) -> List[HistoryItem]:
+ return self._build_result_dictionary(start, end)
+
+ def str_search(self, search: str, include_persisted: bool = False) -> Dict[int, HistoryItem]:
"""Find history items which contain a given string
:param search: the string to search for
:param include_persisted: if True, then search full history including persisted history
- :return: a list of history items, or an empty list if the string was not found
+ :return: a dictionary of history items keyed by their 1-based index in ascending order,
+ or an empty dictionary if the string was not found
"""
- def isin(history_item):
+ def isin(history_item: HistoryItem) -> bool:
"""filter function for string search of history"""
sloppy = utils.norm_fold(search)
inraw = sloppy in utils.norm_fold(history_item.raw)
inexpanded = sloppy in utils.norm_fold(history_item.expanded)
return inraw or inexpanded
- search_list = self if include_persisted else self[self.session_start_index :]
- return [item for item in search_list if isin(item)]
+ start = 0 if include_persisted else self.session_start_index
+ return self._build_result_dictionary(start, len(self), isin)
- def regex_search(self, regex: str, include_persisted: bool = False) -> List[HistoryItem]:
+ def regex_search(self, regex: str, include_persisted: bool = False) -> Dict[int, HistoryItem]:
"""Find history items which match a given regular expression
:param regex: the regular expression to search for.
:param include_persisted: if True, then search full history including persisted history
- :return: a list of history items, or an empty list if the string was not found
+ :return: a dictionary of history items keyed by their 1-based index in ascending order,
+ or an empty dictionary if the regex was not matched
"""
regex = regex.strip()
if regex.startswith(r'/') and regex.endswith(r'/'):
regex = regex[1:-1]
finder = re.compile(regex, re.DOTALL | re.MULTILINE)
- def isin(hi):
+ def isin(hi: HistoryItem) -> bool:
"""filter function for doing a regular expression search of history"""
- return finder.search(hi.raw) or finder.search(hi.expanded)
+ return bool(finder.search(hi.raw) or finder.search(hi.expanded))
- search_list = self if include_persisted else self[self.session_start_index :]
- return [itm for itm in search_list if isin(itm)]
+ start = 0 if include_persisted else self.session_start_index
+ return self._build_result_dictionary(start, len(self), isin)
def truncate(self, max_length: int) -> None:
"""Truncate the length of the history, dropping the oldest items if necessary
@@ -294,3 +274,17 @@ class History(list):
elif len(self) > max_length:
last_element = len(self) - max_length
del self[0:last_element]
+
+ def _build_result_dictionary(
+ self, start: int, end: int, filter_func: Optional[Callable[[HistoryItem], bool]] = None
+ ) -> Dict[int, HistoryItem]:
+ """
+ Build history search results
+ :param start: start index to search from
+ :param end: end index to stop searching (exclusive)
+ """
+ results: Dict[int, HistoryItem] = dict()
+ for index in range(start, end):
+ if filter_func is None or filter_func(self[index]):
+ results[index + 1] = self[index]
+ return results