mirror of
https://github.com/dgtlmoon/changedetection.io.git
synced 2025-10-30 14:17:40 +00:00
130 lines
5.4 KiB
Python
130 lines
5.4 KiB
Python
import difflib
|
|
from typing import List, Iterator, Union
|
|
|
|
# https://github.com/dgtlmoon/changedetection.io/issues/821#issuecomment-1241837050
|
|
#HTML_ADDED_STYLE = "background-color: #d2f7c2; color: #255d00;"
|
|
#HTML_CHANGED_INTO_STYLE = "background-color: #dafbe1; color: #116329;"
|
|
#HTML_CHANGED_STYLE = "background-color: #ffd6cc; color: #7a2000;"
|
|
#HTML_REMOVED_STYLE = "background-color: #ffebe9; color: #82071e;"
|
|
|
|
# @todo - In the future we can make this configurable
|
|
HTML_ADDED_STYLE = "background-color: #eaf2c2; color: #406619"
|
|
HTML_REMOVED_STYLE = "background-color: #fadad7; color: #b30000"
|
|
HTML_CHANGED_STYLE = HTML_REMOVED_STYLE
|
|
HTML_CHANGED_INTO_STYLE = HTML_ADDED_STYLE
|
|
|
|
|
|
# These get set to html or telegram type or discord compatible or whatever in handler.py
|
|
# Something that cant get escaped to HTML by accident
|
|
REMOVED_PLACEMARKER_OPEN = '@removed_PLACEMARKER_OPEN'
|
|
REMOVED_PLACEMARKER_CLOSED = '@removed_PLACEMARKER_CLOSED'
|
|
|
|
ADDED_PLACEMARKER_OPEN = '@added_PLACEMARKER_OPEN'
|
|
ADDED_PLACEMARKER_CLOSED = '@added_PLACEMARKER_CLOSED'
|
|
|
|
CHANGED_PLACEMARKER_OPEN = '@changed_PLACEMARKER_OPEN'
|
|
CHANGED_PLACEMARKER_CLOSED = '@changed_PLACEMARKER_CLOSED'
|
|
|
|
CHANGED_INTO_PLACEMARKER_OPEN = '@changed_into_PLACEMARKER_OPEN'
|
|
CHANGED_INTO_PLACEMARKER_CLOSED = '@changed_into_PLACEMARKER_CLOSED'
|
|
|
|
def same_slicer(lst: List[str], start: int, end: int) -> List[str]:
|
|
"""Return a slice of the list, or a single element if start == end."""
|
|
return lst[start:end] if start != end else [lst[start]]
|
|
|
|
def customSequenceMatcher(
|
|
before: List[str],
|
|
after: List[str],
|
|
include_equal: bool = False,
|
|
include_removed: bool = True,
|
|
include_added: bool = True,
|
|
include_replaced: bool = True,
|
|
include_change_type_prefix: bool = True
|
|
) -> Iterator[List[str]]:
|
|
"""
|
|
Compare two sequences and yield differences based on specified parameters.
|
|
|
|
Args:
|
|
before (List[str]): Original sequence
|
|
after (List[str]): Modified sequence
|
|
include_equal (bool): Include unchanged parts
|
|
include_removed (bool): Include removed parts
|
|
include_added (bool): Include added parts
|
|
include_replaced (bool): Include replaced parts
|
|
include_change_type_prefix (bool): Add prefixes to indicate change types
|
|
Yields:
|
|
List[str]: Differences between sequences
|
|
"""
|
|
cruncher = difflib.SequenceMatcher(isjunk=lambda x: x in " \t", a=before, b=after)
|
|
|
|
|
|
|
|
for tag, alo, ahi, blo, bhi in cruncher.get_opcodes():
|
|
if include_equal and tag == 'equal':
|
|
yield before[alo:ahi]
|
|
elif include_removed and tag == 'delete':
|
|
if include_change_type_prefix:
|
|
yield [f'{REMOVED_PLACEMARKER_OPEN}{line}{REMOVED_PLACEMARKER_CLOSED}' for line in same_slicer(before, alo, ahi)]
|
|
else:
|
|
yield same_slicer(before, alo, ahi)
|
|
elif include_replaced and tag == 'replace':
|
|
if include_change_type_prefix:
|
|
yield [f'{CHANGED_PLACEMARKER_OPEN}{line}{CHANGED_PLACEMARKER_CLOSED}' for line in same_slicer(before, alo, ahi)] + \
|
|
[f'{CHANGED_INTO_PLACEMARKER_OPEN}{line}{CHANGED_INTO_PLACEMARKER_CLOSED}' for line in same_slicer(after, blo, bhi)]
|
|
else:
|
|
yield same_slicer(before, alo, ahi) + same_slicer(after, blo, bhi)
|
|
elif include_added and tag == 'insert':
|
|
if include_change_type_prefix:
|
|
yield [f'{ADDED_PLACEMARKER_OPEN}{line}{ADDED_PLACEMARKER_CLOSED}' for line in same_slicer(after, blo, bhi)]
|
|
else:
|
|
yield same_slicer(after, blo, bhi)
|
|
|
|
|
|
def render_diff(
|
|
previous_version_file_contents: str,
|
|
newest_version_file_contents: str,
|
|
include_equal: bool = False,
|
|
include_removed: bool = True,
|
|
include_added: bool = True,
|
|
include_replaced: bool = True,
|
|
line_feed_sep: str = "\n",
|
|
include_change_type_prefix: bool = True,
|
|
patch_format: bool = False
|
|
) -> str:
|
|
"""
|
|
Render the difference between two file contents.
|
|
|
|
Args:
|
|
previous_version_file_contents (str): Original file contents
|
|
newest_version_file_contents (str): Modified file contents
|
|
include_equal (bool): Include unchanged parts
|
|
include_removed (bool): Include removed parts
|
|
include_added (bool): Include added parts
|
|
include_replaced (bool): Include replaced parts
|
|
line_feed_sep (str): Separator for lines in output
|
|
include_change_type_prefix (bool): Add prefixes to indicate change types
|
|
patch_format (bool): Use patch format for output
|
|
Returns:
|
|
str: Rendered difference
|
|
"""
|
|
newest_lines = [line.rstrip() for line in newest_version_file_contents.splitlines()]
|
|
previous_lines = [line.rstrip() for line in previous_version_file_contents.splitlines()] if previous_version_file_contents else []
|
|
|
|
if patch_format:
|
|
patch = difflib.unified_diff(previous_lines, newest_lines)
|
|
return line_feed_sep.join(patch)
|
|
|
|
rendered_diff = customSequenceMatcher(
|
|
before=previous_lines,
|
|
after=newest_lines,
|
|
include_equal=include_equal,
|
|
include_removed=include_removed,
|
|
include_added=include_added,
|
|
include_replaced=include_replaced,
|
|
include_change_type_prefix=include_change_type_prefix
|
|
)
|
|
|
|
def flatten(lst: List[Union[str, List[str]]]) -> str:
|
|
return line_feed_sep.join(flatten(x) if isinstance(x, list) else x for x in lst)
|
|
|
|
return flatten(rendered_diff) |