Skip to content

Commit

Permalink
add nox task to output dependencies
Browse files Browse the repository at this point in the history
  • Loading branch information
Jannis-Mittenzwei committed Jan 10, 2025
1 parent 14be6eb commit eed8091
Show file tree
Hide file tree
Showing 3 changed files with 188 additions and 0 deletions.
182 changes: 182 additions & 0 deletions exasol/toolbox/nox/_dependencies.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,182 @@
from __future__ import annotations

import json
from collections import defaultdict
from dataclasses import dataclass
from inspect import cleandoc
from pathlib import Path

import nox
import tomlkit
from nox import Session


@dataclass(frozen=True)
class Package:
name: str
package_link: str
license: str
version: str
license_link: str


def _dependencies(toml_str: str) -> dict[str, list]:
toml = tomlkit.loads(toml_str)
poetry = toml.get("tool", {}).get("poetry", {})
dependencies: dict[str, list] = {}

packages = poetry.get("dependencies", {})
dependencies["project"] = []
for package in packages:
dependencies["project"].append(package)

packages = poetry.get("dev", {}).get("dependencies", {})
dependencies["dev"] = []
for package in packages:
dependencies["dependencies"].append(package)

groups = poetry.get("group", {})
for group in groups:
packages = groups.get(group, {}).get("dependencies")
dependencies[group] = []
for package in packages:
dependencies[group].append(package)
return dependencies


def _licenses(session: Session) -> None:
session.run(
"poetry",
"run",
"pip-licenses",
"--format=json",
"--output-file=packages.json",
"--with-system",
"--with-urls",
)


def _normalize(_license: str) -> str:
def is_multi_license(l):
return ";" in l

def select_most_permissive(l: str) -> str:
licenses = [_normalize(l.strip()) for l in l.split(";")]
priority = defaultdict(
lambda: 9999,
{
"Unlicensed": 0,
"BSD": 1,
"MIT": 2,
"MPLv2": 3,
"LGPLv2": 4,
"GPLv2": 5,
"GPLv3": 6,
},
)
priority_to_license = defaultdict(
lambda: "Unknown", {v: k for k, v in priority.items()}
)
selected = min(*[priority[lic] for lic in licenses])
return priority_to_license[int(selected)]

mapping = {
"BSD License": "BSD",
"MIT License": "MIT",
"The Unlicensed (Unlicensed)": "Unlicensed",
"Mozilla Public License 2.0 (MPL 2.0)": "MPLv2",
"GNU Lesser General Public License v2 (LGPLv2)": "LGPLv2",
"GNU General Public License v2 (GPLv2)": "GPLv2",
"GNU General Public License v3 (GPLv3)": "GPLv3",
}

if is_multi_license(_license):
return select_most_permissive(_license)

if _license not in mapping:
return _license

return mapping[_license]


def _packages_from_json(file: Path) -> list[Package]:
packages = json.loads(file.read_text())
packages_list = []
for package in packages:
packages_list.append(
Package(
name=package["Name"],
package_link=package["URL"],
version=package["Version"],
license=_normalize(package["License"]),
license_link="",
)
)
return packages_list


def dependency(group: str, group_packages: list, packages: list[Package]) -> str:
def header(group: str):
group = "".join([word.capitalize() for word in group.strip().split()])
text = f"## {group} Dependencies\n"
text += "---\n"
text += "|Package|version|Licence|\n"
text += "|---|---|---|\n"
return text

def rows(group_packages: list, packages: list[Package]) -> str:
def _normalize_package_name(name: str) -> str:
_name = name.lower()
while "_" in _name:
_name = _name.replace("_", "-")
return _name

text = ""
for package in group_packages:
consistent = filter(
lambda elem: (_normalize_package_name(elem.name) == package),
packages,
)
for content in consistent:
text += f"|[{content.name}]({content.package_link})"
text += f"|{content.version}"
text += f"|[{content.license}]({content.license_link})|\n"
text += "\n"
return text

_template = cleandoc(
"""
{header}{rows}
"""
)
return _template.format(
header=header(group), rows=rows(group_packages, packages)
)


def packages_markdown(dependencies: dict[str, list], packages: list[Package]) -> str:
def heading():
text = "# Dependecies\n"
text += "---\n"
return text

template = cleandoc(
"""
{heading}{rows}
"""
)
rows = ""

for group in dependencies:
rows += dependency(group, dependencies[group], packages)
return template.format(heading=heading(), rows=rows)


@nox.session(name="dependency:licenses", python=False)
def dependency_licenses(session: Session) -> None:
"""returns the packages and their licenses"""
toml = Path("pyproject.toml")
dependencies = _dependencies(toml.read_text())
_licenses(session)
package_infos = _packages_from_json(Path("packages.json"))
print(packages_markdown(dependencies, package_infos))
5 changes: 5 additions & 0 deletions exasol/toolbox/nox/tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -78,5 +78,10 @@ def check(session: Session) -> None:
from exasol.toolbox.nox._artifacts import (
check_artifacts
)

from exasol.toolbox.nox._dependencies import (
dependency_licenses,
)

# isort: on
# fmt: on
1 change: 1 addition & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@ sphinx-design = ">=0.5.0,<1"
typer = {extras = ["all"], version = ">=0.7.0"}
bandit = {extras = ["toml"], version = "^1.7.9"}
jinja2 = "^3.1.4"
pip-licenses = "^5.0.0"

[tool.poetry.group.dev.dependencies]
autoimport = "^1.4.0"
Expand Down

0 comments on commit eed8091

Please sign in to comment.