Skip to content

Commit

Permalink
refactoring code snippet into separate file
Browse files Browse the repository at this point in the history
  • Loading branch information
domanchi committed Mar 10, 2019
1 parent b307e86 commit 0e0e3f7
Show file tree
Hide file tree
Showing 3 changed files with 149 additions and 73 deletions.
102 changes: 30 additions & 72 deletions detect_secrets/core/audit.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,6 @@
from __future__ import print_function
from __future__ import unicode_literals

import codecs
import itertools
import json
import os
import subprocess
Expand All @@ -15,6 +13,7 @@
from ..plugins.high_entropy_strings import HighEntropyStringsPlugin
from .baseline import merge_results
from .bidirectional_iterator import BidirectionalIterator
from .code_snippet import CodeSnippetHighlighter
from .color import AnsiColor
from .color import colorize
from .common import write_baseline_to_file
Expand Down Expand Up @@ -454,92 +453,59 @@ def _get_secret_with_context(
:raises: SecretNotFoundOnSpecifiedLineError
"""
secret_line_idx = secret['line_number'] - 1
end_line = secret_line_idx + lines_of_context + 1

if secret_line_idx <= lines_of_context:
start_line = 0
index_of_secret_in_output = secret_line_idx
else:
start_line = secret_line_idx - lines_of_context
index_of_secret_in_output = lines_of_context

with codecs.open(filename, encoding='utf-8') as file:
output = list(
itertools.islice(
file.read().splitlines(),
start_line,
end_line,
),
)
snippet = CodeSnippetHighlighter().get_code_snippet(
filename,
secret['line_number'],
lines_of_context=lines_of_context,
)

try:
output[index_of_secret_in_output] = _highlight_secret(
output[index_of_secret_in_output],
secret['line_number'],
raw_secret_value = get_raw_secret_value(
snippet.target_line,
secret,
filename,
plugin_settings,
filename,
)

snippet.highlight_line(raw_secret_value)
except SecretNotFoundOnSpecifiedLineError:
if not force:
raise

output[index_of_secret_in_output] = '{}'.format(
colorize(
output[index_of_secret_in_output],
AnsiColor.BOLD,
),
snippet.target_line = colorize(
snippet.target_line,
AnsiColor.BOLD,
)

# Adding line numbers
return '\n'.join(
map(
lambda x: '{}:{}'.format(
colorize(
str(int(x[0]) + start_line + 1),
AnsiColor.LIGHT_GREEN,
),
x[1],
),
enumerate(output),
),
)
return str(snippet.add_line_numbers())


def _highlight_secret(
def get_raw_secret_value(
secret_line,
secret_lineno,
secret,
filename,
plugin_settings,
filename,
):
"""
:type secret_line: str
:param secret_line: the line on which the secret is found
:type secret_lineno: int
:param secret_lineno: secret_line's line number in the source file
:type secret: dict
:param secret: see caller's docstring
:type filename: str
:param filename: this is needed, because PotentialSecret uses this
as a means of comparing whether two secrets are equal.
:type plugin_settings: list
:param plugin_settings: see caller's docstring
:rtype: str
:returns: secret_line, but with the actual secret highlighted.
:type filename: str
:param filename: this is needed, because PotentialSecret uses this
as a means of comparing whether two secrets are equal.
"""
plugin = initialize.from_secret_type(
secret['type'],
plugin_settings,
)

for raw_secret in _raw_secret_generator(
for raw_secret in raw_secret_generator(
plugin,
secret_line,
filetype=determine_file_type(filename),
Expand All @@ -553,26 +519,18 @@ def _highlight_secret(
# There could be more than two secrets on the same line.
# We only want to highlight the right one.
if secret_obj.secret_hash == secret['hashed_secret']:
break
return raw_secret
else:
raise SecretNotFoundOnSpecifiedLineError(secret_lineno)

index_of_secret = secret_line.lower().index(raw_secret.lower())
end_of_secret = index_of_secret + len(raw_secret)
return '{}{}{}'.format(
secret_line[:index_of_secret],
colorize(
# copy the secret out of the line because .lower() from secret
# generator may be different from the original value:
secret_line[index_of_secret:end_of_secret],
AnsiColor.RED_BACKGROUND,
),
secret_line[index_of_secret + len(raw_secret):],
)
raise SecretNotFoundOnSpecifiedLineError(secret['line_number'])


def _raw_secret_generator(plugin, secret_line, filetype):
"""Generates raw secrets by re-scanning the line, with the specified plugin"""
def raw_secret_generator(plugin, secret_line, filetype):
"""Generates raw secrets by re-scanning the line, with the specified plugin
:type plugin: BasePlugin
:type secret_line: str
:type filetype: FileType
"""
for raw_secret in plugin.secret_generator(secret_line, filetype=filetype):
yield raw_secret

Expand Down
118 changes: 118 additions & 0 deletions detect_secrets/core/code_snippet.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
import codecs
import itertools

from .color import AnsiColor
from .color import colorize


class CodeSnippetHighlighter:

def get_code_snippet(self, filename, line_number, lines_of_context=5):
"""
:type filename: str
:type line_number: int
:param line_number: line which you want to focus on
:type lines_of_context: int
:param lines_of_context: how many lines to display around the line you want
to focus on.
:rtype: CodeSnippet
"""
secret_line_index = line_number - 1
end_line = secret_line_index + lines_of_context + 1

if secret_line_index <= lines_of_context:
start_line = 0
index_of_secret_in_output = secret_line_index
else:
start_line = secret_line_index - lines_of_context
index_of_secret_in_output = lines_of_context

return CodeSnippet(
list(
itertools.islice(
self._get_lines_in_file(filename),
start_line,
end_line,
)
),
start_line,
index_of_secret_in_output,
)

def _get_lines_in_file(self, filename):
"""
:rtype: list
"""
with codecs.open(filename, encoding='utf-8') as file:
return file.read().splitlines()


class CodeSnippet:

def __init__(self, snippet, start_line, target_index):
"""
:type snippet: list
:param snippet: lines of code extracted from file
:type start_line: int
:param start_line: first line number in segment
:type target_index: int
:param target_index: index in snippet of target line
"""
self.lines = snippet
self.start_line = start_line
self.target_index = target_index

@property
def target_line(self):
return self.lines[self.target_index]

@target_line.setter
def target_line(self, value):
self.lines[self.target_index] = value

def add_line_numbers(self):
for index, line in enumerate(self.lines):
self.lines[index] = '{}:{}'.format(
self.get_line_number(self.start_line + index + 1),
line,
)

return self

def highlight_line(self, payload):
"""
:type payload: str
:param payload: string to highlight, on chosen line
"""
index_of_payload = self.target_line.lower().index(payload.lower())
end_of_payload = index_of_payload + len(payload)

self.target_line = '{}{}{}'.format(
self.target_line[:index_of_payload],
self.apply_highlight(self.target_line[index_of_payload:end_of_payload]),
self.target_line[end_of_payload:],
)

return self

def get_line_number(self, line_number):
"""Broken out, for custom colorization."""
return colorize(
str(line_number),
AnsiColor.LIGHT_GREEN,
)

def apply_highlight(self, payload):
"""Broken out, for custom colorization."""
return colorize(
payload,
AnsiColor.RED_BACKGROUND,
)

def __str__(self):
return '\n'.join(self.lines)
2 changes: 1 addition & 1 deletion tests/core/audit_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -519,7 +519,7 @@ def mock_open(
string.ascii_letters[:(end_line - secret_line)][::-1],
),
)
return mock_open_base(data, 'detect_secrets.core.audit.codecs.open')
return mock_open_base(data, 'detect_secrets.core.code_snippet.codecs.open')

@staticmethod
def _make_string_into_individual_lines(string):
Expand Down

0 comments on commit 0e0e3f7

Please sign in to comment.