🚨 removed linter warnings for Python code

This commit is contained in:
Niels Lohmann 2018-01-13 11:15:23 +01:00
parent f4a55f26b0
commit a66b2d20c6
No known key found for this signature in database
GPG key ID: 7F3CEA63AE251B69
2 changed files with 259 additions and 245 deletions

View file

@ -0,0 +1,10 @@
The following changes have been made to the code with respect to <https://github.com/edlund/amalgamate/commit/c91f07eea1133aa184f652b8f1398eaf03586208>:
- Resolved inspection results from PyCharm:
- replaced tabs with spaces
- added encoding annotation
- reindented file to remove trailing whitespaces
- unused import `sys`
- membership check
- made function from `_is_within`
- removed unused variable `actual_path`

View file

@ -1,4 +1,5 @@
#!/usr/bin/env python #!/usr/bin/env python
# coding=utf-8
# amalgamate.py - Amalgamate C source and header files. # amalgamate.py - Amalgamate C source and header files.
# Copyright (c) 2012, Erik Edlund <erik.edlund@32767.se> # Copyright (c) 2012, Erik Edlund <erik.edlund@32767.se>
@ -37,259 +38,262 @@ import datetime
import json import json
import os import os
import re import re
import sys
class Amalgamation(object): class Amalgamation(object):
# Prepends self.source_path to file_path if needed. # Prepends self.source_path to file_path if needed.
def actual_path(self, file_path): def actual_path(self, file_path):
if not os.path.isabs(file_path): if not os.path.isabs(file_path):
file_path = os.path.join(self.source_path, file_path) file_path = os.path.join(self.source_path, file_path)
return file_path return file_path
# Search included file_path in self.include_paths and # Search included file_path in self.include_paths and
# in source_dir if specified. # in source_dir if specified.
def find_included_file(self, file_path, source_dir): def find_included_file(self, file_path, source_dir):
search_dirs = self.include_paths[:] search_dirs = self.include_paths[:]
if source_dir: if source_dir:
search_dirs.insert(0, source_dir) search_dirs.insert(0, source_dir)
for search_dir in search_dirs: for search_dir in search_dirs:
search_path = os.path.join(search_dir, file_path) search_path = os.path.join(search_dir, file_path)
if os.path.isfile(self.actual_path(search_path)): if os.path.isfile(self.actual_path(search_path)):
return search_path return search_path
return None return None
def __init__(self, args): def __init__(self, args):
with open(args.config, 'r') as f: with open(args.config, 'r') as f:
config = json.loads(f.read()) config = json.loads(f.read())
for key in config: for key in config:
setattr(self, key, config[key]) setattr(self, key, config[key])
self.verbose = args.verbose == "yes" self.verbose = args.verbose == "yes"
self.prologue = args.prologue self.prologue = args.prologue
self.source_path = args.source_path self.source_path = args.source_path
self.included_files = [] self.included_files = []
# Generate the amalgamation and write it to the target file. # Generate the amalgamation and write it to the target file.
def generate(self): def generate(self):
amalgamation = "" amalgamation = ""
if self.prologue: if self.prologue:
with open(self.prologue, 'r') as f: with open(self.prologue, 'r') as f:
amalgamation += datetime.datetime.now().strftime(f.read()) amalgamation += datetime.datetime.now().strftime(f.read())
if self.verbose: if self.verbose:
print("Config:") print("Config:")
print(" target = {0}".format(self.target)) print(" target = {0}".format(self.target))
print(" working_dir = {0}".format(os.getcwd())) print(" working_dir = {0}".format(os.getcwd()))
print(" include_paths = {0}".format(self.include_paths)) print(" include_paths = {0}".format(self.include_paths))
print("Creating amalgamation:") print("Creating amalgamation:")
for file_path in self.sources: for file_path in self.sources:
# Do not check the include paths while processing the source # Do not check the include paths while processing the source
# list, all given source paths must be correct. # list, all given source paths must be correct.
actual_path = self.actual_path(file_path) # actual_path = self.actual_path(file_path)
print(" - processing \"{0}\"".format(file_path)) print(" - processing \"{0}\"".format(file_path))
t = TranslationUnit(file_path, self, True) t = TranslationUnit(file_path, self, True)
amalgamation += t.content amalgamation += t.content
with open(self.target, 'w') as f: with open(self.target, 'w') as f:
f.write(amalgamation) f.write(amalgamation)
print("...done!\n")
if self.verbose:
print("Files processed: {0}".format(self.sources))
print("Files included: {0}".format(self.included_files))
print("")
def _is_within(match, matches):
for m in matches:
if match.start() > m.start() and \
match.end() < m.end():
return True
return False
print("...done!\n")
if self.verbose:
print("Files processed: {0}".format(self.sources))
print("Files included: {0}".format(self.included_files))
print("")
class TranslationUnit(object): class TranslationUnit(object):
# // C++ comment.
cpp_comment_pattern = re.compile(r"//.*?\n")
# // C++ comment. # /* C comment. */
cpp_comment_pattern = re.compile(r"//.*?\n") c_comment_pattern = re.compile(r"/\*.*?\*/", re.S)
# /* C comment. */ # "complex \"stri\\\ng\" value".
c_comment_pattern = re.compile(r"/\*.*?\*/", re.S) string_pattern = re.compile("[^']" r'".*?(?<=[^\\])"', re.S)
# "complex \"stri\\\ng\" value". # Handle simple include directives. Support for advanced
string_pattern = re.compile("[^']" r'".*?(?<=[^\\])"', re.S) # directives where macros and defines needs to expanded is
# not a concern right now.
include_pattern = re.compile(
r'#\s*include\s+(<|")(?P<path>.*?)("|>)', re.S)
# Handle simple include directives. Support for advanced # #pragma once
# directives where macros and defines needs to expanded is pragma_once_pattern = re.compile(r'#\s*pragma\s+once', re.S)
# not a concern right now.
include_pattern = re.compile(
r'#\s*include\s+(<|")(?P<path>.*?)("|>)', re.S)
# #pragma once # Search for pattern in self.content, add the match to
pragma_once_pattern = re.compile(r'#\s*pragma\s+once', re.S) # contexts if found and update the index accordingly.
def _search_content(self, index, pattern, contexts):
match = pattern.search(self.content, index)
if match:
contexts.append(match)
return match.end()
return index + 2
# Search for pattern in self.content, add the match to # Return all the skippable contexts, i.e., comments and strings
# contexts if found and update the index accordingly. def _find_skippable_contexts(self):
def _search_content(self, index, pattern, contexts): # Find contexts in the content in which a found include
match = pattern.search(self.content, index) # directive should not be processed.
if match: skippable_contexts = []
contexts.append(match)
return match.end()
return index + 2
# Return all the skippable contexts, i.e., comments and strings # Walk through the content char by char, and try to grab
def _find_skippable_contexts(self): # skippable contexts using regular expressions when found.
# Find contexts in the content in which a found include i = 1
# directive should not be processed. content_len = len(self.content)
skippable_contexts = [] while i < content_len:
j = i - 1
current = self.content[i]
previous = self.content[j]
# Walk through the content char by char, and try to grab if current == '"':
# skippable contexts using regular expressions when found. # String value.
i = 1 i = self._search_content(j, self.string_pattern,
content_len = len(self.content) skippable_contexts)
while i < content_len: elif current == '*' and previous == '/':
j = i - 1 # C style comment.
current = self.content[i] i = self._search_content(j, self.c_comment_pattern,
previous = self.content[j] skippable_contexts)
elif current == '/' and previous == '/':
# C++ style comment.
i = self._search_content(j, self.cpp_comment_pattern,
skippable_contexts)
else:
# Skip to the next char.
i += 1
if current == '"': return skippable_contexts
# String value.
i = self._search_content(j, self.string_pattern,
skippable_contexts)
elif current == '*' and previous == '/':
# C style comment.
i = self._search_content(j, self.c_comment_pattern,
skippable_contexts)
elif current == '/' and previous == '/':
# C++ style comment.
i = self._search_content(j, self.cpp_comment_pattern,
skippable_contexts)
else:
# Skip to the next char.
i += 1
return skippable_contexts # Returns True if the match is within list of other matches
# Returns True if the match is within list of other matches # Removes pragma once from content
def _is_within(self, match, matches): def _process_pragma_once(self):
for m in matches: content_len = len(self.content)
if match.start() > m.start() and \ if content_len < len("#include <x>"):
match.end() < m.end(): return 0
return True
return False
# Removes pragma once from content # Find contexts in the content in which a found include
def _process_pragma_once(self): # directive should not be processed.
content_len = len(self.content) skippable_contexts = self._find_skippable_contexts()
if content_len < len("#include <x>"):
return 0
# Find contexts in the content in which a found include pragmas = []
# directive should not be processed. pragma_once_match = self.pragma_once_pattern.search(self.content)
skippable_contexts = self._find_skippable_contexts() while pragma_once_match:
if not _is_within(pragma_once_match, skippable_contexts):
pragmas.append(pragma_once_match)
pragmas = [] pragma_once_match = self.pragma_once_pattern.search(self.content,
pragma_once_match = self.pragma_once_pattern.search(self.content) pragma_once_match.end())
while pragma_once_match:
if not self._is_within(pragma_once_match, skippable_contexts):
pragmas.append(pragma_once_match)
pragma_once_match = self.pragma_once_pattern.search(self.content, # Handle all collected pragma once directives.
pragma_once_match.end()) prev_end = 0
tmp_content = ''
for pragma_match in pragmas:
tmp_content += self.content[prev_end:pragma_match.start()]
prev_end = pragma_match.end()
tmp_content += self.content[prev_end:]
self.content = tmp_content
# Handle all collected pragma once directives. # Include all trivial #include directives into self.content.
prev_end = 0 def _process_includes(self):
tmp_content = '' content_len = len(self.content)
for pragma_match in pragmas: if content_len < len("#include <x>"):
tmp_content += self.content[prev_end:pragma_match.start()] return 0
prev_end = pragma_match.end()
tmp_content += self.content[prev_end:]
self.content = tmp_content
# Include all trivial #include directives into self.content. # Find contexts in the content in which a found include
def _process_includes(self): # directive should not be processed.
content_len = len(self.content) skippable_contexts = self._find_skippable_contexts()
if content_len < len("#include <x>"):
return 0
# Find contexts in the content in which a found include # Search for include directives in the content, collect those
# directive should not be processed. # which should be included into the content.
skippable_contexts = self._find_skippable_contexts() includes = []
include_match = self.include_pattern.search(self.content)
while include_match:
if not _is_within(include_match, skippable_contexts):
include_path = include_match.group("path")
search_same_dir = include_match.group(1) == '"'
found_included_path = self.amalgamation.find_included_file(
include_path, self.file_dir if search_same_dir else None)
if found_included_path:
includes.append((include_match, found_included_path))
# Search for include directives in the content, collect those include_match = self.include_pattern.search(self.content,
# which should be included into the content. include_match.end())
includes = []
include_match = self.include_pattern.search(self.content)
while include_match:
if not self._is_within(include_match, skippable_contexts):
include_path = include_match.group("path")
search_same_dir = include_match.group(1) == '"'
found_included_path = self.amalgamation.find_included_file(
include_path, self.file_dir if search_same_dir else None)
if found_included_path:
includes.append((include_match, found_included_path))
include_match = self.include_pattern.search(self.content, # Handle all collected include directives.
include_match.end()) prev_end = 0
tmp_content = ''
for include in includes:
include_match, found_included_path = include
tmp_content += self.content[prev_end:include_match.start()]
tmp_content += "// {0}\n".format(include_match.group(0))
if found_included_path not in self.amalgamation.included_files:
t = TranslationUnit(found_included_path, self.amalgamation, False)
tmp_content += t.content
prev_end = include_match.end()
tmp_content += self.content[prev_end:]
self.content = tmp_content
# Handle all collected include directives. return len(includes)
prev_end = 0
tmp_content = ''
for include in includes:
include_match, found_included_path = include
tmp_content += self.content[prev_end:include_match.start()]
tmp_content += "// {0}\n".format(include_match.group(0))
if not found_included_path in self.amalgamation.included_files:
t = TranslationUnit(found_included_path, self.amalgamation, False)
tmp_content += t.content
prev_end = include_match.end()
tmp_content += self.content[prev_end:]
self.content = tmp_content
return len(includes) # Make all content processing
def _process(self):
if not self.is_root:
self._process_pragma_once()
self._process_includes()
# Make all content processing def __init__(self, file_path, amalgamation, is_root):
def _process(self): self.file_path = file_path
if not self.is_root: self.file_dir = os.path.dirname(file_path)
self._process_pragma_once() self.amalgamation = amalgamation
self._process_includes() self.is_root = is_root
def __init__(self, file_path, amalgamation, is_root): self.amalgamation.included_files.append(self.file_path)
self.file_path = file_path
self.file_dir = os.path.dirname(file_path)
self.amalgamation = amalgamation
self.is_root = is_root
self.amalgamation.included_files.append(self.file_path) actual_path = self.amalgamation.actual_path(file_path)
if not os.path.isfile(actual_path):
raise IOError("File not found: \"{0}\"".format(file_path))
with open(actual_path, 'r') as f:
self.content = f.read()
self._process()
actual_path = self.amalgamation.actual_path(file_path)
if not os.path.isfile(actual_path):
raise IOError("File not found: \"{0}\"".format(file_path))
with open(actual_path, 'r') as f:
self.content = f.read()
self._process()
def main(): def main():
description = "Amalgamate C source and header files." description = "Amalgamate C source and header files."
usage = " ".join([ usage = " ".join([
"amalgamate.py", "amalgamate.py",
"[-v]", "[-v]",
"-c path/to/config.json", "-c path/to/config.json",
"-s path/to/source/dir", "-s path/to/source/dir",
"[-p path/to/prologue.(c|h)]" "[-p path/to/prologue.(c|h)]"
]) ])
argsparser = argparse.ArgumentParser( argsparser = argparse.ArgumentParser(
description=description, usage=usage) description=description, usage=usage)
argsparser.add_argument("-v", "--verbose", dest="verbose", argsparser.add_argument("-v", "--verbose", dest="verbose",
choices=["yes", "no"], metavar="", help="be verbose") choices=["yes", "no"], metavar="", help="be verbose")
argsparser.add_argument("-c", "--config", dest="config", argsparser.add_argument("-c", "--config", dest="config",
required=True, metavar="", help="path to a JSON config file") required=True, metavar="", help="path to a JSON config file")
argsparser.add_argument("-s", "--source", dest="source_path", argsparser.add_argument("-s", "--source", dest="source_path",
required=True, metavar="", help="source code path") required=True, metavar="", help="source code path")
argsparser.add_argument("-p", "--prologue", dest="prologue", argsparser.add_argument("-p", "--prologue", dest="prologue",
required=False, metavar="", help="path to a C prologue file") required=False, metavar="", help="path to a C prologue file")
amalgamation = Amalgamation(argsparser.parse_args())
amalgamation.generate()
amalgamation = Amalgamation(argsparser.parse_args())
amalgamation.generate()
if __name__ == "__main__": if __name__ == "__main__":
main() main()