forked from dftbplus/dftbplus
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Improve import manipulation of Fortran files
- Loading branch information
Showing
2 changed files
with
199 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,110 @@ | ||
#!/usr/bin/env python3 | ||
|
||
"""Simple script to ckeck usage of imports in Fortran files""" | ||
|
||
import argparse | ||
import fnmatch | ||
import os | ||
import re | ||
|
||
|
||
_DESCRIPTION = "Checks the imports in DFTB+ Fortran file(s) for usage" | ||
|
||
_PAT_USE_MODULE = re.compile( | ||
r"""^(?P<indent>[ ]*)use | ||
(?P<attrib>(?:\s*,\s*intrinsic)?) | ||
(?P<separator>[ ]*::[ ]*|[ ]*) | ||
(?P<name>\w+) | ||
(?P<rest>.*?(?:&[ ]*\n(?:[ ]*&)?.*?)*)\n | ||
""", | ||
re.VERBOSE | re.MULTILINE, | ||
) | ||
|
||
WORD_PATTERN = re.compile(r"""\w+""") | ||
|
||
|
||
def main(): | ||
"""Main script driver.""" | ||
args = parse_arguments() | ||
filenames = [] | ||
if args.folders: | ||
filenames = get_files(args.folders) | ||
if args.files: | ||
filenames += args.files | ||
for fname in filenames: | ||
with open(fname, "r", encoding="utf-8") as file: | ||
file_txt = file.read() | ||
matches = list(_PAT_USE_MODULE.finditer(file_txt)) | ||
matches.reverse() | ||
rest_list = [] | ||
for match in matches: | ||
if match["rest"]: | ||
rest_list.append(match["rest"]) | ||
file_txt = file_txt[: match.start()] + file_txt[match.end() :] | ||
if args.case_sensitive: | ||
word_set = set(match.group() for match in WORD_PATTERN.finditer(file_txt)) | ||
else: | ||
word_set = set(match.group().lower() for match in WORD_PATTERN.finditer(file_txt)) | ||
import_set = get_import_set(rest_list, args.case_sensitive) | ||
unused_imports = import_set - word_set | ||
if unused_imports: | ||
print(f"{fname} contains the following unsued imports: ", unused_imports) | ||
else: | ||
print(f"{fname} OK") | ||
|
||
|
||
def parse_arguments(): | ||
"""Parses the command line arguments""" | ||
parser = argparse.ArgumentParser(description=_DESCRIPTION) | ||
msg = "File to process" | ||
parser.add_argument("--files", nargs="+", metavar="FILE", help=msg) | ||
msg = "Folder to process" | ||
parser.add_argument("--folders", nargs="+", metavar="FOLDER", help=msg) | ||
msg = "Case sensitive checking" | ||
parser.add_argument("-c", dest="case_sensitive", action="store_true", help=msg) | ||
args = parser.parse_args() | ||
if not (args.folders or args.files): | ||
parser.error("No Files/Folders specified, add '--files' or '--folders'") | ||
return args | ||
|
||
|
||
def get_files(folders): | ||
"""Find all '*F90' or '*f90' files in folders""" | ||
file_list = [] | ||
for folder in folders: | ||
for root, _, files in os.walk(folder): | ||
for file in files: | ||
if fnmatch.fnmatch(file, "*.[fF]90"): | ||
file_list.append(os.path.join(root, file)) | ||
return file_list | ||
|
||
|
||
def get_import_set(rest_list, case_sensitive): | ||
"""Creates set of imported methods and functions""" | ||
import_set = set() | ||
for rest in rest_list: | ||
if "only" not in rest.lower(): | ||
continue | ||
_, imports = re.split(r"only\s*\:", rest, 1) | ||
imports_list = imports.split(",") | ||
for imp in imports_list: | ||
imp = re.sub(r"\&\s*&", "", imp).strip() | ||
if "operator" in imp.lower(): | ||
continue | ||
if "assignment" in imp.lower(): | ||
continue | ||
if "=>" in imp: | ||
if case_sensitive: | ||
import_set.add(imp.split("=>")[0]) | ||
else: | ||
import_set.add(imp.split("=>")[0].lower()) | ||
else: | ||
if case_sensitive: | ||
import_set.add(imp) | ||
else: | ||
import_set.add(imp.lower()) | ||
return import_set | ||
|
||
|
||
if __name__ == "__main__": | ||
main() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters