57 lines
1.6 KiB
Python
Executable File
57 lines
1.6 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
# Extract the dependencies from the setup.py files
|
|
# and save the result to requirements.txt.
|
|
#
|
|
# You can filter any extra require by adding the name as argument.
|
|
#
|
|
# Examples:
|
|
# tools/extract_requirements.py
|
|
# tools/extract_requirements.py dev
|
|
|
|
|
|
import ast
|
|
from glob import glob
|
|
from pathlib import Path
|
|
from sys import argv
|
|
|
|
|
|
class RemoveJoinedStr(ast.NodeTransformer):
|
|
def visit_JoinedStr(self, _node): # pylint: disable=invalid-name
|
|
pass
|
|
|
|
|
|
for setup in glob("*/setup.py"):
|
|
setup_path = Path(setup)
|
|
requirements_path = setup_path.parent / "requirements.txt"
|
|
|
|
lines = [
|
|
"# Please do not edit this file, edit the setup.py file!",
|
|
"# This file is auto-generated by tools/extract_requirements.py.",
|
|
]
|
|
|
|
requires = []
|
|
|
|
for node in ast.walk(ast.parse(setup_path.read_text(encoding="utf-8"))):
|
|
if (
|
|
isinstance(node, ast.Expr)
|
|
and isinstance(node.value, ast.Call)
|
|
and isinstance(node.value.func, ast.Name)
|
|
and node.value.func.id == "setup"
|
|
):
|
|
for keyword in node.value.keywords:
|
|
if keyword.arg == "install_requires":
|
|
requires.extend(ast.literal_eval(keyword.value))
|
|
|
|
if keyword.arg == "extras_require":
|
|
extras = ast.literal_eval(RemoveJoinedStr().visit(keyword.value))
|
|
|
|
for key, values in extras.items():
|
|
if key in argv:
|
|
continue
|
|
requires.extend(values)
|
|
|
|
lines.extend(sorted(requires))
|
|
|
|
requirements_path.write_text("\n".join(lines) + "\n", encoding="utf-8")
|