🚨 removed linter warnings for Python code
This commit is contained in:
parent
f4a55f26b0
commit
a66b2d20c6
2 changed files with 259 additions and 245 deletions
10
develop/amalgamate/CHANGES.md
Normal file
10
develop/amalgamate/CHANGES.md
Normal 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`
|
|
@ -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()
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue