{ lib, pkgs }: let inherit (import ./semver.nix { inherit lib ireplace; }) satisfiesSemver; inherit (builtins) genList length; # Replace a list entry at defined index with set value ireplace = idx: value: list: ( genList (i: if i == idx then value else (builtins.elemAt list i)) (length list) ); # Compare a semver expression with a version isCompatible = version: let operators = { "||" = cond1: cond2: cond1 || cond2; "," = cond1: cond2: cond1 && cond2; # , means && "&&" = cond1: cond2: cond1 && cond2; }; splitRe = "(" + (builtins.concatStringsSep "|" (builtins.map (x: lib.replaceStrings [ "|" ] [ "\\|" ] x) (lib.attrNames operators))) + ")"; in expr: let tokens = builtins.filter (x: x != "") (builtins.split splitRe expr); combine = acc: v: let isOperator = builtins.typeOf v == "list"; operator = if isOperator then (builtins.elemAt v 0) else acc.operator; in if isOperator then (acc // { inherit operator; }) else { inherit operator; state = operators."${operator}" acc.state (satisfiesSemver version v); }; initial = { operator = "&&"; state = true; }; in if expr == "" then true else (builtins.foldl' combine initial tokens).state; fromTOML = builtins.fromTOML or ( toml: builtins.fromJSON ( builtins.readFile ( pkgs.runCommand "from-toml" { inherit toml; allowSubstitutes = false; preferLocalBuild = true; } '' ${pkgs.remarshal}/bin/remarshal \ -if toml \ -i <(echo "$toml") \ -of json \ -o $out '' ) ) ); readTOML = path: fromTOML (builtins.readFile path); # # Returns the appropriate manylinux dependencies and string representation for the file specified # getManyLinuxDeps = f: let ml = pkgs.pythonManylinuxPackages; in if lib.strings.hasInfix "manylinux1" f then { pkg = [ ml.manylinux1 ]; str = "1"; } else if lib.strings.hasInfix "manylinux2010" f then { pkg = [ ml.manylinux2010 ]; str = "2010"; } else if lib.strings.hasInfix "manylinux2014" f then { pkg = [ ml.manylinux2014 ]; str = "2014"; } else { pkg = []; str = null; }; # Fetch the artifacts from the PyPI index. Since we get all # info we need from the lock file we don't use nixpkgs' fetchPyPi # as it modifies casing while not providing anything we don't already # have. # # Args: # pname: package name # file: filename including extension # hash: SRI hash # kind: Language implementation and version tag https://www.python.org/dev/peps/pep-0427/#file-name-convention fetchFromPypi = lib.makeOverridable ( { pname, file, hash, kind }: pkgs.fetchurl { url = "https://files.pythonhosted.org/packages/${kind}/${lib.toLower (builtins.substring 0 1 file)}/${pname}/${file}"; inherit hash; } ); getBuildSystemPkgs = { pythonPackages , pyProject }: let buildSystem = lib.getAttrFromPath [ "build-system" "build-backend" ] pyProject; drvAttr = builtins.elemAt (builtins.split "\\.|:" buildSystem) 0; in if buildSystem == "" then [] else ( [ pythonPackages.${drvAttr} or (throw "unsupported build system ${buildSystem}") ] ); # Find gitignore files recursively in parent directory stopping with .git findGitIgnores = path: let parent = path + "/.."; gitIgnore = path + "/.gitignore"; isGitRoot = builtins.pathExists (path + "/.git"); hasGitIgnore = builtins.pathExists gitIgnore; gitIgnores = if hasGitIgnore then [ gitIgnore ] else []; in lib.optionals (! isGitRoot) (findGitIgnores parent) ++ gitIgnores; /* Provides a source filtering mechanism that: - Filters gitignore's - Filters pycache/pyc files - Uses cleanSourceFilter to filter out .git/.hg, .o/.so, editor backup files & nix result symlinks */ cleanPythonSources = { src }: let gitIgnores = findGitIgnores src; pycacheFilter = name: type: (type == "directory" && ! lib.strings.hasInfix "__pycache__" name) || (type == "regular" && ! lib.strings.hasSuffix ".pyc" name) ; in lib.cleanSourceWith { filter = lib.cleanSourceFilter; src = lib.cleanSourceWith { filter = pkgs.nix-gitignore.gitignoreFilterPure pycacheFilter gitIgnores src; inherit src; }; }; in { inherit fetchFromPypi getManyLinuxDeps isCompatible readTOML getBuildSystemPkgs satisfiesSemver cleanPythonSources ; }