mach-nix/mach_nix/resolver/resolvelib_resolver.py

86 lines
3.2 KiB
Python
Raw Normal View History

2020-04-22 09:28:58 +00:00
from dataclasses import dataclass
from typing import Iterable, List
import resolvelib
from packaging.version import Version
from mach_nix.data.providers import DependencyProviderBase
2020-08-08 09:59:55 +00:00
from mach_nix.data.nixpkgs import NixpkgsIndex
2020-04-22 09:28:58 +00:00
from mach_nix.requirements import Requirement
from mach_nix.resolver import Resolver, ResolvedPkg
from mach_nix.versions import filter_versions
from mach_nix.visualize import print_deps
2020-04-22 09:28:58 +00:00
@dataclass
class Candidate:
name: str
ver: Version
extras: tuple
# Implement logic so the resolver understands the requirement format.
class Provider:
2020-08-08 09:59:55 +00:00
def __init__(self, nixpkgs: NixpkgsIndex, deps_db: DependencyProviderBase):
2020-04-22 09:28:58 +00:00
self.nixpkgs = nixpkgs
self.deps_db = deps_db
def get_extras_for(self, dependency):
return tuple(sorted(dependency.extras))
def get_base_requirement(self, candidate):
return Requirement("{}=={}".format(candidate.name, candidate.ver))
def identify(self, dependency):
return dependency.name
def get_preference(self, resolution, candidates, information):
return len(candidates)
def find_matches(self, req):
all = self.deps_db.available_versions(req.key)
matching_versions = filter_versions(all, req.specs)
return [Candidate(name=req.name, ver=ver, extras=req.extras) for ver in matching_versions]
2020-04-22 09:28:58 +00:00
def is_satisfied_by(self, requirement, candidate):
if not set(requirement.extras).issubset(set(candidate.extras)):
return False
return bool(len(list(filter_versions([candidate.ver], requirement.specs))))
def get_dependencies(self, candidate):
install_requires, setup_requires = self.deps_db.get_pkg_reqs(candidate.name, candidate.ver, candidate.extras)
deps = install_requires + setup_requires
return deps
class ResolvelibResolver(Resolver):
2020-08-08 09:59:55 +00:00
def __init__(self, nixpkgs: NixpkgsIndex, deps_provider: DependencyProviderBase):
2020-04-22 09:28:58 +00:00
self.nixpkgs = nixpkgs
self.deps_provider = deps_provider
2020-04-22 09:28:58 +00:00
def resolve(self, reqs: Iterable[Requirement]) -> List[ResolvedPkg]:
2020-04-22 09:28:58 +00:00
reporter = resolvelib.BaseReporter()
result = resolvelib.Resolver(Provider(self.nixpkgs, self.deps_provider), reporter).resolve(reqs, max_rounds=1000)
2020-04-22 09:28:58 +00:00
nix_py_pkgs = []
for name in result.graph._forwards.keys():
if name is None:
continue
ver = result.mapping[name].ver
install_requires, setup_requires = self.deps_provider.get_pkg_reqs(
name, ver, extras=result.mapping[name].extras)
provider_info = self.deps_provider.get_provider_info(name, ver)
prop_build_inputs = list({req.key for req in install_requires})
2020-04-22 09:28:58 +00:00
build_inputs = list({req.key for req in setup_requires})
is_root = name in result.graph._forwards[None]
nix_py_pkgs.append(ResolvedPkg(
name=name,
ver=ver,
build_inputs=build_inputs,
prop_build_inputs=prop_build_inputs,
is_root=is_root,
provider_info=provider_info,
extras_selected=list(result.mapping[name].extras)
2020-04-22 09:28:58 +00:00
))
2020-08-08 09:59:55 +00:00
print_deps(nix_py_pkgs, self.nixpkgs)
2020-04-22 09:28:58 +00:00
return nix_py_pkgs