Merge pull request #53 from nix-community/dev

refactoring + CI Tests + go translator
This commit is contained in:
DavHau 2021-11-17 12:47:42 +07:00 committed by GitHub
commit c5f6bd5a17
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
24 changed files with 861 additions and 403 deletions

40
.github/workflows/tests.yml vendored Normal file
View File

@ -0,0 +1,40 @@
name: "Test dream2nix"
on:
pull_request:
push:
jobs:
pure-tests:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2.4.0
- uses: cachix/install-nix-action@v15
with:
install_url: https://nixos-nix-install-tests.cachix.org/serve/w659aglf1hfvkj5wj696q9x8r19p6b7k/install
install_options: '--tarball-url-prefix https://nixos-nix-install-tests.cachix.org/serve'
extra_nix_config: |
access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}
- uses: cachix/cachix-action@v10
with:
name: nix-community
signingKey: '${{ secrets.CACHIX_SIGNING_KEY }}'
- run: nix flake check
impure-tests:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2.4.0
- uses: cachix/install-nix-action@v15
with:
install_url: https://nixos-nix-install-tests.cachix.org/serve/w659aglf1hfvkj5wj696q9x8r19p6b7k/install
install_options: '--tarball-url-prefix https://nixos-nix-install-tests.cachix.org/serve'
extra_nix_config: |
access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}
- uses: cachix/cachix-action@v10
with:
name: nix-community
signingKey: '${{ secrets.CACHIX_SIGNING_KEY }}'
- run: nix run .#tests-impure

1
.gitignore vendored
View File

@ -1,3 +1,4 @@
.*/ .*/
!.github/
result result
interpreter interpreter

22
ci.nix Normal file
View File

@ -0,0 +1,22 @@
let
b = builtins;
flakeCompatSrc = b.fetchurl "https://raw.githubusercontent.com/edolstra/flake-compat/12c64ca55c1014cdc1b16ed5a804aa8576601ff2/default.nix";
flake = (import flakeCompatSrc { src = ./.; }).defaultNix;
pkgs = import flake.inputs.nixpkgs {};
recurseIntoAll = b.mapAttrs (name: val: pkgs.recurseIntoAttrs val);
in
# {
# inherit flake;
# }
# // (recurseIntoAll {
# checks = flake.checks.x86_64-linux;
# })
# hercules ci's nix version cannot fetch submodules and crashes
{
inherit (pkgs) hello;
}

View File

@ -1,5 +1,21 @@
{ {
"nodes": { "nodes": {
"gomod2nix": {
"flake": false,
"locked": {
"lastModified": 1627572165,
"narHash": "sha256-MFpwnkvQpauj799b4QTBJQFEddbD02+Ln5k92QyHOSk=",
"owner": "tweag",
"repo": "gomod2nix",
"rev": "67f22dd738d092c6ba88e420350ada0ed4992ae8",
"type": "github"
},
"original": {
"owner": "tweag",
"repo": "gomod2nix",
"type": "github"
}
},
"mach-nix": { "mach-nix": {
"flake": false, "flake": false,
"locked": { "locked": {
@ -81,6 +97,7 @@
}, },
"root": { "root": {
"inputs": { "inputs": {
"gomod2nix": "gomod2nix",
"mach-nix": "mach-nix", "mach-nix": "mach-nix",
"nix-parsec": "nix-parsec", "nix-parsec": "nix-parsec",
"nixpkgs": "nixpkgs", "nixpkgs": "nixpkgs",

View File

@ -4,6 +4,9 @@
inputs = { inputs = {
nixpkgs.url = "nixpkgs/nixos-unstable"; nixpkgs.url = "nixpkgs/nixos-unstable";
# required for builder go/gomod2nix
gomod2nix = { url = "github:tweag/gomod2nix"; flake = false; };
# required for translator nodejs/pure/package-lock # required for translator nodejs/pure/package-lock
nix-parsec = { url = "github:nprindle/nix-parsec"; flake = false; }; nix-parsec = { url = "github:nprindle/nix-parsec"; flake = false; };
@ -19,6 +22,7 @@
outputs = { outputs = {
self, self,
gomod2nix,
mach-nix, mach-nix,
nix-parsec, nix-parsec,
nixpkgs, nixpkgs,
@ -127,7 +131,14 @@
# all apps including cli, install, etc. # all apps including cli, install, etc.
apps = forAllSystems (system: pkgs: apps = forAllSystems (system: pkgs:
dream2nixFor."${system}".apps.flakeApps dream2nixFor."${system}".apps.flakeApps // {
tests-impure = {
type = "app";
program =
b.toString
(dream2nixFor."${system}".callPackageDream ./tests/impure {});
};
}
); );
# a dev shell for working on dream2nix # a dev shell for working on dream2nix
@ -151,7 +162,7 @@
''; '';
}); });
checks = forAllSystems (system: pkgs: import ./checks.nix { checks = forAllSystems (system: pkgs: import ./tests/pure {
inherit lib pkgs; inherit lib pkgs;
dream2nix = dream2nixFor."${system}"; dream2nix = dream2nixFor."${system}";
}); });

View File

@ -23,14 +23,17 @@ class AddCommand(Command):
arguments = [ arguments = [
argument( argument(
"source", "source",
"source of the package, can be a path, tarball URL, or flake-style spec") "Sources of the packages. Can be a paths, tarball URLs, or flake-style specs",
# multiple=True
)
] ]
options = [ options = [
option("translator", None, "which translator to use", flag=False), option("translator", None, "which translator to use", flag=False),
option("target", None, "target file/directory for the dream-lock.json", flag=False), option("target", None, "target file/directory for the dream-lock.json", flag=False),
option("attribute-name", None, "attribute name for new new pakcage", flag=False),
option( option(
"--packages-root", "packages-root",
None, None,
"Put package under a new directory inside packages-root", "Put package under a new directory inside packages-root",
flag=False flag=False
@ -58,14 +61,92 @@ class AddCommand(Command):
self.line(f"\n{self.description}\n") self.line(f"\n{self.description}\n")
# parse extra args # parse extra args
specified_extra_args = self.parse_extra_args()
# ensure packages-root
packages_root = self.find_packages_root()
lock, sourceSpec, specified_extra_args, translator =\
self.translate_from_source(specified_extra_args, self.argument("source"))
# get package name and version from lock
mainPackageName = lock['_generic']['mainPackageName']
mainPackageVersion = lock['_generic']['mainPackageVersion']
# calculate output directory and attribute name
mainPackageDirName = self.define_attribute_name(mainPackageName)
# calculate output files
filesToCreate, output = self.calc_outputs(mainPackageDirName, packages_root)
outputDreamLock = f"{output}/dream-lock.json"
outputDefaultNix = f"{output}/default.nix"
# add translator information to lock
self.extend_with_translator_info(lock, specified_extra_args, translator)
# add main package source
self.add_main_source(lock, mainPackageName, mainPackageVersion, sourceSpec)
# clean up dependency graph
if 'dependencies' in lock['_generic']:
self.postprocess_dep_graph(lock)
# calculate combined hash if --combined was specified
if self.option('combined'):
self.aggregate_hashes(lock, outputDreamLock)
# validate dream lock format
checkLockJSON(lock)
# format dream lock
lockStr = self.format_lock_str(lock)
# save dream lock file
with open(outputDreamLock, 'w') as f:
f.write(lockStr)
print(f"Created {output}/dream-lock.json")
# create default.nix
if 'default.nix' in filesToCreate:
self.create_default_nix(lock, output, outputDefaultNix, sources[0])
# add new package to git
if config['isRepo']:
sp.run(["git", "add", "-N", output])
def translate_from_source(self, specified_extra_args, source):
# get source path and spec
source, sourceSpec = self.parse_source(source)
# select translator
translator = self.select_translator(source)
# raise error if any specified extra arg is unknown
specified_extra_args = self.declare_extra_args(specified_extra_args, translator)
# do the translation and produce dream lock
lock = self.run_translate(source, specified_extra_args, translator)
return lock, sourceSpec, specified_extra_args, translator
def parse_extra_args(self):
specified_extra_args = { specified_extra_args = {
arg[0]: arg[1] for arg in map( arg[0]: arg[1] for arg in map(
lambda e: e.split('='), lambda e: e.split('='),
self.option("arg"), self.option("arg"),
) )
} }
return specified_extra_args
# ensure packages-root def create_default_nix(self, lock, output, outputDefaultNix, source):
template = callNixFunction(
'apps.apps.cli.templateDefaultNix',
dream2nixLocationRelative=os.path.relpath(dream2nix_src, output),
dreamLock=lock,
sourcePathRelative=os.path.relpath(source, os.path.dirname(outputDefaultNix))
)
with open(outputDefaultNix, 'w') as defaultNix:
defaultNix.write(template)
print(f"Created {output}/default.nix")
def find_packages_root(self):
if self.option("packages-root"): if self.option("packages-root"):
packages_root = self.option("packages-root") packages_root = self.option("packages-root")
elif config['packagesDir']: elif config['packagesDir']:
@ -77,50 +158,266 @@ class AddCommand(Command):
f"Packages direcotry {packages_root} does not exist. Please create.", f"Packages direcotry {packages_root} does not exist. Please create.",
file=sys.stderr, file=sys.stderr,
) )
return packages_root
# verify source def format_lock_str(self, lock):
source = self.argument("source") lockStr = json.dumps(lock, indent=2, sort_keys=True)
if not source and not config['packagesDir']: lockStr = lockStr \
source = os.path.realpath('./.') .replace("[\n ", "[ ") \
.replace("\"\n ]", "\" ]") \
.replace(",\n ", ", ")
return lockStr
def aggregate_hashes(self, lock, outputDreamLock):
print("Building FOD of combined sources to retrieve output hash")
# remove hashes from lock file and init sourcesCombinedHash with empty string
strip_hashes_from_lock(lock)
lock['_generic']['sourcesCombinedHash'] = ""
with open(outputDreamLock, 'w') as f:
json.dump(lock, f, indent=2)
# compute FOD hash of combined sources
proc = sp.run(
[
"nix", "build", "--impure", "-L", "--expr",
f"(import {dream2nix_src} {{}}).fetchSources {{ dreamLock = {outputDreamLock}; }}"
],
capture_output=True,
)
# read the output hash from the failed build log
match = re.search(r"FOD_PATH=(.*=)", proc.stderr.decode())
if not match:
print(proc.stderr.decode())
print(proc.stdout.decode())
raise Exception("Could not find FOD hash in FOD log")
hash = match.groups()[0]
print(f"Computed FOD hash: {hash}")
# store the hash in the lock
lock['_generic']['sourcesCombinedHash'] = hash
def postprocess_dep_graph(self, lock):
depGraph = lock['_generic']['dependencies']
# remove empty entries
if 'dependencies' in lock['_generic']:
for pname, deps in depGraph.copy().items():
if not deps:
del depGraph[pname]
# mark cyclic dependencies
edges = set()
for pname, versions in depGraph.items():
for version, deps in versions.items():
for dep in deps:
edges.add(((pname, version), tuple(dep)))
G = nx.DiGraph(sorted(list(edges)))
cycle_count = 0
removed_edges = []
for pname, versions in depGraph.items():
for version in versions.keys():
key = (pname, version)
try:
while True:
cycle = nx.find_cycle(G, key)
cycle_count += 1
node_from, node_to = cycle[-1][0], cycle[-1][1]
G.remove_edge(node_from, node_to)
removed_edges.append((node_from, node_to))
except nx.NetworkXNoCycle:
continue
lock['cyclicDependencies'] = {}
if removed_edges:
cycles_text = 'Detected Cyclic dependencies:'
for node, removed in removed_edges:
n_name, n_ver = node[0], node[1]
r_name, r_ver = removed[0], removed[1]
cycles_text += \
f"\n {n_name}#{n_ver} -> {r_name}#{r_ver}"
if n_name not in lock['cyclicDependencies']:
lock['cyclicDependencies'][n_name] = {}
if n_ver not in lock['cyclicDependencies'][n_name]:
lock['cyclicDependencies'][n_name][n_ver] = []
lock['cyclicDependencies'][n_name][n_ver].append(removed)
print(cycles_text)
def add_main_source(self, lock, mainPackageName, mainPackageVersion, sourceSpec):
mainSource = sourceSpec.copy()
if not mainSource:
mainSource = dict(
type="unknown",
)
if mainPackageName not in lock['sources']:
lock['sources'][mainPackageName] = {
mainPackageVersion: mainSource
}
else:
lock['sources'][mainPackageName][mainPackageVersion] = mainSource
def extend_with_translator_info(self, lock, specified_extra_args, translator):
t = translator
lock['_generic']['translatedBy'] = f"{t['subsystem']}.{t['type']}.{t['name']}"
lock['_generic']['translatorParams'] = " ".join(
[
'--translator',
f"{translator['subsystem']}.{translator['type']}.{translator['name']}",
] + (
["--combined"] if self.option('combined') else []
) + [
f"--arg {n}={v}" for n, v in specified_extra_args.items()
])
def calc_outputs(self, mainPackageDirName, packages_root):
if self.option('target'):
if self.option('target').startswith('/'):
output = self.option('target')
else:
output = f"{packages_root}/{self.option('target')}"
else:
output = f"{packages_root}/{mainPackageDirName}"
# collect files to create
filesToCreate = ['dream-lock.json']
if not os.path.isdir(output):
os.mkdir(output)
existingFiles = set(os.listdir(output))
if not self.option('no-default-nix') \
and not 'default.nix' in existingFiles \
and not config['packagesDir']:
if self.confirm(
'Create a default.nix for debugging purposes',
default=True):
filesToCreate.append('default.nix')
# overwrite existing files only if --force is set
if self.option('force'):
for f in filesToCreate:
if os.path.isfile(f):
os.remove(f)
# raise error if any file exists already
else:
if any(f in existingFiles for f in filesToCreate):
print( print(
f"Source not specified. Defaulting to current directory: {source}", f"output directory {output} already contains a 'default.nix' "
"or 'dream-lock.json'. Resolve via one of these:\n"
" - use --force to overwrite files\n"
" - use --target to specify another target dir",
file=sys.stderr, file=sys.stderr,
) )
# else:
# print(
# f"Source not specified. Defaulting to current directory: {source}",
# file=sys.stderr,
# )
# check if source is valid fetcher spec
sourceSpec = {}
# handle source shortcuts
if source.partition(':')[0].split('+')[0] in os.environ.get("fetcherNames", None).split()\
or source.startswith('http'):
print(f"fetching source for '{source}'")
sourceSpec =\
callNixFunction("fetchers.translateShortcut", shortcut=source)
source =\
buildNixFunction("fetchers.fetchShortcut", shortcut=source, extract=True)
# handle source paths
else:
# check if source path exists
if not os.path.exists(source):
print(f"Input source '{source}' does not exist", file=sys.stdout)
exit(1) exit(1)
source = os.path.realpath(source) output = os.path.realpath(output)
# handle source from dream-lock.json return filesToCreate, output
if source.endswith('dream-lock.json'):
print(f"fetching source defined via existing dream-lock.json")
with open(source) as f:
sourceDreamLock = json.load(f)
sourceMainPackageName = sourceDreamLock['_generic']['mainPackageName']
sourceMainPackageVersion = sourceDreamLock['_generic']['mainPackageVersion']
sourceSpec =\
sourceDreamLock['sources'][sourceMainPackageName][sourceMainPackageVersion]
source = \
buildNixFunction("fetchers.fetchSource", source=sourceSpec, extract=True)
# select translator def define_attribute_name(self, mainPackageName):
attributeName = self.option('attribute-name')
if attributeName:
return attributeName
attributeName = mainPackageName.strip('@').replace('/', '-')
# verify / change main package dir name
print(f"Current package attribute name is: {attributeName}")
new_name = self.ask(
"Specify new attribute name or leave empty to keep current:"
)
if new_name:
attributeName = newName
return attributeName
def run_translate(self, source, specified_extra_args, translator):
# build the translator bin
t = translator
translator_path = buildNixAttribute(
f"translators.translators.{t['subsystem']}.{t['type']}.{t['name']}.translateBin"
)
# direct outputs of translator to temporary file
with tempfile.NamedTemporaryFile("r") as output_temp_file:
# arguments for calling the translator nix module
translator_input = dict(
inputFiles=[],
inputDirectories=[source],
outputFile=output_temp_file.name,
)
translator_input.update(specified_extra_args)
# dump translator arguments to json file and execute translator
print("\nTranslating project metadata")
with tempfile.NamedTemporaryFile("w") as input_json_file:
json.dump(translator_input, input_json_file, indent=2)
input_json_file.seek(0) # flushes write cache
# execute translator
sp.run(
[f"{translator_path}", input_json_file.name]
)
# raise error if output wasn't produced
if not output_temp_file.read():
raise Exception(f"Translator failed to create dream-lock.json")
# read produced lock file
with open(output_temp_file.name) as f:
lock = json.load(f)
return lock
def declare_extra_args(self, specified_extra_args, translator):
unknown_extra_args = set(specified_extra_args.keys()) - set(translator['extraArgs'].keys())
if unknown_extra_args:
print(
f"Invalid extra args for translator '{translator['name']}': "
f" {', '.join(unknown_extra_args)}"
"\nPlease remove these parameters",
file=sys.stderr
)
exit(1)
# transform flags to bool
for argName, argVal in specified_extra_args.copy().items():
if translator['extraArgs'][argName]['type'] == 'flag':
if argVal.lower() in ('yes', 'y', 'true'):
specified_extra_args[argName] = True
elif argVal.lower() in ('no', 'n', 'false'):
specified_extra_args[argName] = False
else:
print(
f"Invalid value {argVal} for argument {argName}",
file=sys.stderr
)
specified_extra_args = \
{k: (bool(v) if translator['extraArgs'][k]['type'] == 'flag' else v) \
for k, v in specified_extra_args.items()}
# on non-interactive session, assume defaults for unspecified extra args
if not self.io.is_interactive():
specified_extra_args.update(
{n: (True if v['type'] == 'flag' else v['default']) \
for n, v in translator['extraArgs'].items() \
if n not in specified_extra_args and 'default' in v}
)
unspecified_extra_args = \
{n: v for n, v in translator['extraArgs'].items() \
if n not in specified_extra_args}
# raise error if any extra arg unspecified in non-interactive session
if unspecified_extra_args:
if not self.io.is_interactive():
print(
f"Please specify the following extra arguments required by translator '{translator['name']}' :\n" \
', '.join(unspecified_extra_args.keys()),
file=sys.stderr
)
exit(1)
# interactively retrieve answers for unspecified extra arguments
else:
print(f"\nThe translator '{translator['name']}' requires additional options")
for arg_name, arg in unspecified_extra_args.items():
print('')
if arg['type'] == 'flag':
print(f"Please specify '{arg_name}'")
specified_extra_args[arg_name] = self.confirm(f"{arg['description']}:", False)
else:
print(f"Please specify '{arg_name}': {arg['description']}")
print(f"Example values: " + ', '.join(arg['examples']))
if 'default' in arg:
print(f"Leave empty for default ({arg['default']})")
while True:
specified_extra_args[arg_name] = self.ask(f"{arg_name}:", arg.get('default'))
if specified_extra_args[arg_name]:
break
return specified_extra_args
def select_translator(self, source):
translatorsSorted = sorted( translatorsSorted = sorted(
list_translators_for_source(source), list_translators_for_source(source),
key=lambda t: ( key=lambda t: (
@ -159,296 +456,42 @@ class AddCommand(Command):
except IndexError: except IndexError:
print(f"Could not find translator '{'.'.join(translator)}'", file=sys.stderr) print(f"Could not find translator '{'.'.join(translator)}'", file=sys.stderr)
exit(1) exit(1)
return translator
# raise error if any specified extra arg is unknown def parse_source(self, source):
unknown_extra_args = set(specified_extra_args.keys()) - set(translator['extraArgs'].keys()) # verify source
if unknown_extra_args: if not source and not config['packagesDir']:
source = os.path.realpath('./.')
print( print(
f"Invalid extra args for translator '{translator['name']}': " f"Source not specified. Defaulting to current directory: {source}",
f" {', '.join(unknown_extra_args)}"
"\nPlease remove these parameters",
file=sys.stderr
)
exit(1)
# transform flags to bool
for argName, argVal in specified_extra_args.copy().items():
if translator['extraArgs'][argName]['type'] == 'flag':
if argVal.lower() in ('yes', 'y', 'true'):
specified_extra_args[argName] = True
elif argVal.lower() in ('no', 'n', 'false'):
specified_extra_args[argName] = False
else:
print(
f"Invalid value {argVal} for argument {argName}",
file=sys.stderr
)
specified_extra_args =\
{k: (bool(v) if translator['extraArgs'][k]['type'] == 'flag' else v ) \
for k, v in specified_extra_args.items()}
# on non-interactive session, assume defaults for unspecified extra args
if not self.io.is_interactive():
specified_extra_args.update(
{n: (True if v['type'] == 'flag' else v['default']) \
for n, v in translator['extraArgs'].items() \
if n not in specified_extra_args and 'default' in v}
)
unspecified_extra_args = \
{n: v for n, v in translator['extraArgs'].items() \
if n not in specified_extra_args}
# raise error if any extra arg unspecified in non-interactive session
if unspecified_extra_args:
if not self.io.is_interactive():
print(
f"Please specify the following extra arguments required by translator '{translator['name']}' :\n" \
', '.join(unspecified_extra_args.keys()),
file=sys.stderr
)
exit(1)
# interactively retrieve answers for unspecified extra arguments
else:
print(f"\nThe translator '{translator['name']}' requires additional options")
for arg_name, arg in unspecified_extra_args.items():
print('')
if arg['type'] == 'flag':
print(f"Please specify '{arg_name}'")
specified_extra_args[arg_name] = self.confirm(f"{arg['description']}:", False)
else:
print(f"Please specify '{arg_name}': {arg['description']}")
print(f"Example values: " + ', '.join(arg['examples']))
if 'default' in arg:
print(f"Leave empty for default ({arg['default']})")
while True:
specified_extra_args[arg_name] = self.ask(f"{arg_name}:", arg.get('default'))
if specified_extra_args[arg_name]:
break
# build the translator bin
t = translator
translator_path = buildNixAttribute(
f"translators.translators.{t['subsystem']}.{t['type']}.{t['name']}.translateBin"
)
# direct outputs of translator to temporary file
with tempfile.NamedTemporaryFile("r") as output_temp_file:
# arguments for calling the translator nix module
translator_input = dict(
inputFiles=[],
inputDirectories=[source],
outputFile=output_temp_file.name,
)
translator_input.update(specified_extra_args)
# dump translator arguments to json file and execute translator
print("\nTranslating project metadata")
with tempfile.NamedTemporaryFile("w") as input_json_file:
json.dump(translator_input, input_json_file, indent=2)
input_json_file.seek(0) # flushes write cache
# execute translator
sp.run(
[f"{translator_path}/bin/run", input_json_file.name]
)
# raise error if output wasn't produced
if not output_temp_file.read():
raise Exception(f"Translator failed to create dream-lock.json")
# read produced lock file
with open(output_temp_file.name) as f:
lock = json.load(f)
# get package name and version from lock
mainPackageName = lock['_generic']['mainPackageName']
mainPackageVersion = lock['_generic']['mainPackageVersion']
# calculate output directory
mainPackageDirName = mainPackageName.strip('@').replace('/', '-')
# verify / change main package dir name
def update_name(mainPackageDirName):
print(f"Current package attribute name is: {mainPackageDirName}")
new_name = self.ask(
"Specify new attribute name or leave empty to keep current:"
)
if new_name:
return new_name
return mainPackageDirName
mainPackageDirName = update_name(mainPackageDirName)
if self.option('target'):
if self.option('target').startswith('/'):
output = self.option('target')
else:
output = f"{packages_root}/{self.option('target')}"
else:
output = f"{packages_root}/{mainPackageDirName}"
# collect files to create
filesToCreate = ['dream-lock.json']
if not os.path.isdir(output):
os.mkdir(output)
existingFiles = set(os.listdir(output))
if not self.option('no-default-nix')\
and not 'default.nix' in existingFiles\
and not config['packagesDir']:
if self.confirm(
'Create a default.nix for debugging purposes',
default=True):
filesToCreate.append('default.nix')
# overwrite existing files only if --force is set
if self.option('force'):
for f in filesToCreate:
if os.path.isfile(f):
os.remove(f)
# raise error if any file exists already
else:
if any(f in existingFiles for f in filesToCreate):
print(
f"output directory {output} already contains a 'default.nix' "
"or 'dream-lock.json'. Resolve via one of these:\n"
" - use --force to overwrite files\n"
" - use --target to specify another target dir",
file=sys.stderr, file=sys.stderr,
) )
exit(1) # check if source is a valid fetcher spec
output = os.path.realpath(output) sourceSpec = {}
outputDreamLock = f"{output}/dream-lock.json" # handle source shortcuts
outputDefaultNix = f"{output}/default.nix" if source.partition(':')[0].split('+')[0] in os.environ.get("fetcherNames", None).split() \
or source.startswith('http'):
# write translator information to lock file print(f"fetching source for '{source}'")
combined = self.option('combined') sourceSpec = \
lock['_generic']['translatedBy'] = f"{t['subsystem']}.{t['type']}.{t['name']}" callNixFunction("fetchers.translateShortcut", shortcut=source)
lock['_generic']['translatorParams'] = " ".join([ source = \
'--translator', buildNixFunction("fetchers.fetchShortcut", shortcut=source, extract=True)
f"{translator['subsystem']}.{translator['type']}.{translator['name']}", # handle source paths
] + (
["--combined"] if combined else []
) + [
f"--arg {n}={v}" for n, v in specified_extra_args.items()
])
# add main package source
mainSource = sourceSpec.copy()
if not mainSource:
mainSource = dict(
type="unknown",
)
if mainPackageName not in lock['sources']:
lock['sources'][mainPackageName] = {
mainPackageVersion: mainSource
}
else: else:
lock['sources'][mainPackageName][mainPackageVersion] = mainSource # check if source path exists
if not os.path.exists(source):
# clean up dependency graph print(f"Input source '{source}' does not exist", file=sys.stdout)
# remove empty entries exit(1)
if 'dependencies' in lock['_generic']: source = os.path.realpath(source)
depGraph = lock['_generic']['dependencies'] # handle source from dream-lock.json
if 'dependencies' in lock['_generic']: if source.endswith('dream-lock.json'):
for pname, deps in depGraph.copy().items(): print(f"fetching source defined via existing dream-lock.json")
if not deps: with open(source) as f:
del depGraph[pname] sourceDreamLock = json.load(f)
sourceMainPackageName = sourceDreamLock['_generic']['mainPackageName']
# remove cyclic dependencies sourceMainPackageVersion = sourceDreamLock['_generic']['mainPackageVersion']
edges = set() sourceSpec = \
for pname, versions in depGraph.items(): sourceDreamLock['sources'][sourceMainPackageName][sourceMainPackageVersion]
for version, deps in versions.items(): source = \
for dep in deps: buildNixFunction("fetchers.fetchSource", source=sourceSpec, extract=True)
edges.add(((pname, version), tuple(dep))) return source, sourceSpec
G = nx.DiGraph(sorted(list(edges)))
cycle_count = 0
removed_edges = []
for pname, versions in depGraph.items():
for version in versions.keys():
key = (pname, version)
try:
while True:
cycle = nx.find_cycle(G, key)
cycle_count += 1
# remove_dependecy(indexed_pkgs, G, cycle[-1][0], cycle[-1][1])
node_from, node_to = cycle[-1][0], cycle[-1][1]
G.remove_edge(node_from, node_to)
removed_edges.append((node_from, node_to))
except nx.NetworkXNoCycle:
continue
lock['cyclicDependencies'] = {}
if removed_edges:
cycles_text = 'Detected Cyclic dependencies:'
for node, removed in removed_edges:
n_name, n_ver = node[0], node[1]
r_name, r_ver = removed[0], removed[1]
cycles_text +=\
f"\n {n_name}#{n_ver} -> {r_name}#{r_ver}"
if n_name not in lock['cyclicDependencies']:
lock['cyclicDependencies'][n_name] = {}
if n_ver not in lock['cyclicDependencies'][n_name]:
lock['cyclicDependencies'][n_name][n_ver] = []
lock['cyclicDependencies'][n_name][n_ver].append(removed)
print(cycles_text)
# calculate combined hash if --combined was specified
if combined:
print("Building FOD of combined sources to retrieve output hash")
# remove hashes from lock file and init sourcesCombinedHash with empty string
strip_hashes_from_lock(lock)
lock['_generic']['sourcesCombinedHash'] = ""
with open(outputDreamLock, 'w') as f:
json.dump(lock, f, indent=2)
# compute FOD hash of combined sources
proc = sp.run(
[
"nix", "build", "--impure", "-L", "--expr",
f"(import {dream2nix_src} {{}}).fetchSources {{ dreamLock = {outputDreamLock}; }}"
],
capture_output=True,
)
# read the output hash from the failed build log
match = re.search(r"FOD_PATH=(.*=)", proc.stderr.decode())
if not match:
print(proc.stderr.decode())
print(proc.stdout.decode())
raise Exception("Could not find FOD hash in FOD log")
hash = match.groups()[0]
print(f"Computed FOD hash: {hash}")
# store the hash in the lock
lock['_generic']['sourcesCombinedHash'] = hash
# re-write dream-lock.json
checkLockJSON(lock)
lockStr = json.dumps(lock, indent=2, sort_keys = True)
lockStr = lockStr\
.replace("[\n ", "[ ")\
.replace("\"\n ]", "\" ]")\
.replace(",\n ", ", ")
with open(outputDreamLock, 'w') as f:
f.write(lockStr)
# create default.nix
template = callNixFunction(
'apps.apps.cli.templateDefaultNix',
dream2nixLocationRelative=os.path.relpath(dream2nix_src, output),
dreamLock = lock,
sourcePathRelative = os.path.relpath(source, os.path.dirname(outputDefaultNix))
)
# with open(f"{dream2nix_src}/apps/cli2/templateDefault.nix") as template:
if 'default.nix' in filesToCreate:
with open(outputDefaultNix, 'w') as defaultNix:
defaultNix.write(template)
print(f"Created {output}/default.nix")
print(f"Created {output}/dream-lock.json")
if config['isRepo']:
sp.run(["git", "add", "-N", output])

View File

@ -66,7 +66,7 @@ class UpdateCommand(Command):
dreamLock=dreamLockFile, dreamLock=dreamLockFile,
updater=updater, updater=updater,
) )
update_proc = sp.run([f"{update_script}/bin/run"], capture_output=True) update_proc = sp.run([f"{update_script}"], capture_output=True)
version = update_proc.stdout.decode().strip() version = update_proc.stdout.decode().strip()
print(f"Updating from version {old_version} to {version}") print(f"Updating from version {old_version} to {version}")

View File

@ -23,8 +23,7 @@ let
in in
{ {
program = program =
let utils.writePureShellScript
script = utils.writePureShellScript
[ [
gitMinimal gitMinimal
nix nix
@ -39,8 +38,6 @@ in
fetcherNames="${b.toString (lib.attrNames fetchers.fetchers)}" \ fetcherNames="${b.toString (lib.attrNames fetchers.fetchers)}" \
${cliPython}/bin/python ${./.}/cli.py "$@" ${cliPython}/bin/python ${./.}/cli.py "$@"
''; '';
in
"${script}/bin/run";
templateDefaultNix = templateDefaultNix =
{ {

View File

@ -0,0 +1,21 @@
{
lib,
pkgs,
externals,
...
}:
{
fetchedSources,
dreamLock,
}:
let
gomod2nixTOML = fetchedSources.mapAttrs
dependencyObject.goName;
in
externals.gomod2nixBuilder rec {
pname = dreamLock.generic.mainPackage;
version = dreamLock.sources."${pname}".version;
src = fetchedSources."${pname}";
modules = ./gomod2nix.toml;
}

View File

@ -57,6 +57,7 @@ let
# like pkgs.callPackage, but includes all the dream2nix modules # like pkgs.callPackage, but includes all the dream2nix modules
callPackageDream = f: args: pkgs.callPackage f (args // { callPackageDream = f: args: pkgs.callPackage f (args // {
inherit apps;
inherit builders; inherit builders;
inherit callPackageDream; inherit callPackageDream;
inherit config; inherit config;
@ -409,6 +410,7 @@ in
inherit inherit
apps apps
builders builders
callPackageDream
dream2nixWithExternals dream2nixWithExternals
fetchers fetchers
fetchSources fetchSources

View File

@ -12,7 +12,7 @@
{ {
# A derivation which outputs an executable at `/bin/run`. # A derivation which outputs a single executable at `$out`.
# The executable will be called by dream2nix for translation # The executable will be called by dream2nix for translation
# The input format is specified in /specifications/translator-call-example.json. # The input format is specified in /specifications/translator-call-example.json.
# The first arg `$1` will be a json file containing the input parameters # The first arg `$1` will be a json file containing the input parameters

View File

@ -95,23 +95,62 @@ let
# flat list of all translators # flat list of all translators
translatorsList = lib.collect (v: v ? translateBin) translators; translatorsList = lib.collect (v: v ? translateBin) translators;
# json file exposing all existing translators to CLI including their special args # returns the list of translators including their special args
# and adds a flag `compatible` to each translator indicating
# if the translator is compatible to all given paths
translatorsForInput = translatorsForInput =
{ {
inputDirectories, inputDirectories,
inputFiles, inputFiles,
}@args: }@args:
lib.forEach translatorsList lib.forEach translatorsList
(t: { (t: rec {
inherit (t) inherit (t)
name name
extraArgs extraArgs
subsystem subsystem
type type
; ;
compatible = t.compatiblePaths args == args; compatiblePaths = t.compatiblePaths args;
compatible = compatiblePaths == args;
}); });
# also includes subdirectories of the given paths up to a certain depth
# to check for translator compatibility
translatorsForInputRecursive =
{
inputDirectories,
inputFiles,
depth ? 2,
}:
let
listDirsRec = dir: depth:
let
subDirs = (utils.listDirs dir);
in
if depth == 0 then
subDirs
else
subDirs
++
(lib.flatten
(map
(subDir: listDirsRec subDir (depth -1))
subDirs));
dirsToCheck =
lib.flatten
(map
(inputDir: listDirsRec inputDir depth)
inputDirectories);
in
translatorsForInput {
inputDirectories = dirsToCheck;
inherit inputFiles;
};
# pupulates a translators special args with defaults # pupulates a translators special args with defaults
getextraArgsDefaults = extraArgsDef: getextraArgsDefaults = extraArgsDef:
lib.mapAttrs lib.mapAttrs

View File

@ -0,0 +1,68 @@
{
# dream2nix utils
utils,
dream2nixWithExternals,
bash,
coreutils,
jq,
lib,
nix,
writeScriptBin,
...
}:
{
# the input format is specified in /specifications/translator-call-example.json
# this script receives a json file including the input paths and specialArgs
translateBin = utils.writePureShellScript
[
bash
coreutils
jq
nix
]
''
# accroding to the spec, the translator reads the input from a json file
jsonInput=$1
# read the json input
outputFile=$(${jq}/bin/jq '.outputFile' -c -r $jsonInput)
inputDirectory=$(${jq}/bin/jq '.inputDirectories | .[0]' -c -r $jsonInput)
tmpBuild=$(mktemp -d)
cd $tmpBuild
cp -r $inputDirectory/* .
chmod -R +w .
# This should be in sync with gomod2nix version in flake.lock
nix run github:tweag/gomod2nix/67f22dd738d092c6ba88e420350ada0ed4992ae8
nix eval --show-trace --impure --raw --expr "import ${./translate.nix} ${dream2nixWithExternals} ./." > $outputFile
'';
# From a given list of paths, this function returns all paths which can be processed by this translator.
# This allows the framework to detect if the translator is compatible with the given inputs
# to automatically select the right translator.
compatiblePaths =
{
inputDirectories,
inputFiles,
}@args:
{
inputDirectories = lib.filter
(utils.containsMatchingFile [ ''go\.sum'' ''go\.mod'' ])
args.inputDirectories;
inputFiles = [];
};
# If the translator requires additional arguments, specify them here.
# There are only two types of arguments:
# - string argument (type = "argument")
# - boolean flag (type = "flag")
# String arguments contain a default value and examples. Flags do not.
extraArgs = {};
}

View File

@ -0,0 +1,66 @@
dream2nixWithExternals:
cwd:
let
dream2nix = import dream2nixWithExternals { };
b = builtins;
parsed = b.fromTOML (builtins.readFile "${cwd}/gomod2nix.toml");
pkgs = import <nixpkgs> { };
lib = pkgs.lib;
serializePackages = inputData:
lib.mapAttrsToList
(goName: depAttrs: depAttrs // { inherit goName; })
parsed;
translated = dream2nix.utils.simpleTranslate "gomod2nix" rec {
inputData = parsed;
mainPackageName =
let
firstLine = (b.elemAt (lib.splitString "\n" (b.readFile "${cwd}/go.mod")) 0);
in
lib.last (lib.splitString "/" (b.elemAt (lib.splitString " " firstLine) 1));
mainPackageVersion = "unknown";
subsystemName = "go";
subsystemAttrs = { };
inherit serializePackages;
mainPackageDependencies =
lib.forEach
(serializePackages parsed)
(dep: {
name = getName dep;
version = getVersion dep;
});
getOriginalID = dependencyObject:
null;
getName = dependencyObject:
dependencyObject.goName;
getVersion = dependencyObject:
lib.removePrefix "v" dependencyObject.sumVersion;
getDependencies = dependencyObject: getDepByNameVer: dependenciesByOriginalID:
[];
getSourceType = dependencyObject: "git";
sourceConstructors = {
git = dependencyObject:
{
type = "git";
version = getVersion dependencyObject;
hash = dependencyObject.fetch.sha256;
url = dependencyObject.fetch.url;
rev = dependencyObject.fetch.rev;
};
};
};
in
dream2nix.utils.dreamLock.toJSON translated

View File

@ -36,7 +36,7 @@
jq ".inputDirectories[0] = \"$(pwd)\"" -c -r $jsonInput > ./newJsonInput jq ".inputDirectories[0] = \"$(pwd)\"" -c -r $jsonInput > ./newJsonInput
${translators.translators.nodejs.pure.package-lock.translateBin}/bin/run $(realpath ./newJsonInput) ${translators.translators.nodejs.pure.package-lock.translateBin} $(realpath ./newJsonInput)
''; '';

View File

@ -2,6 +2,7 @@
lib, lib,
externals, externals,
nodejs,
translatorName, translatorName,
utils, utils,
... ...
@ -16,9 +17,10 @@
# extraArgs # extraArgs
name, name,
noDev, noDev,
nodejs,
peer, peer,
... ...
}: }@args:
let let
b = builtins; b = builtins;
@ -60,9 +62,7 @@
subsystemName = "nodejs"; subsystemName = "nodejs";
subsystemAttrs = { subsystemAttrs = { nodejsVersion = args.nodejs; };
nodejsVersion = 14;
};
mainPackageDependencies = mainPackageDependencies =
lib.mapAttrsToList lib.mapAttrsToList
@ -297,6 +297,16 @@
type = "flag"; type = "flag";
}; };
nodejs = {
description = "nodejs version to use for building";
default = lib.elemAt (lib.splitString "." nodejs.version) 0;
examples = [
"14"
"16"
];
type = "argument";
};
peer = { peer = {
description = "Include peer dependencies"; description = "Include peer dependencies";
type = "flag"; type = "flag";

View File

@ -4,8 +4,9 @@
fetchzip, fetchzip,
lib, lib,
nix, nix,
pkgs,
runCommand, runCommand,
writeScriptBin, writeScript,
# dream2nix inputs # dream2nix inputs
callPackageDream, callPackageDream,
@ -58,6 +59,8 @@ rec {
listFiles = path: lib.attrNames (lib.filterAttrs (n: v: v == "regular") (builtins.readDir path)); listFiles = path: lib.attrNames (lib.filterAttrs (n: v: v == "regular") (builtins.readDir path));
listDirs = path: lib.attrNames (lib.filterAttrs (n: v: v == "directory") (builtins.readDir path));
# directory names of a given directory # directory names of a given directory
dirNames = dir: lib.attrNames (lib.filterAttrs (name: type: type == "directory") (builtins.readDir dir)); dirNames = dir: lib.attrNames (lib.filterAttrs (name: type: type == "directory") (builtins.readDir dir));
@ -87,11 +90,13 @@ rec {
b.readFile hashFile; b.readFile hashFile;
# builder to create a shell script that has it's own PATH # builder to create a shell script that has it's own PATH
writePureShellScript = availablePrograms: script: writeScriptBin "run" '' writePureShellScript = availablePrograms: script: writeScript "script.sh" ''
#!${bash}/bin/bash #!${bash}/bin/bash
set -Eeuo pipefail set -Eeuo pipefail
export PATH="${lib.makeBinPath availablePrograms}" export PATH="${lib.makeBinPath availablePrograms}"
export NIX_PATH=nixpkgs=${pkgs.path}
tmpdir=$(${coreutils}/bin/mktemp -d) tmpdir=$(${coreutils}/bin/mktemp -d)
cd $tmpdir cd $tmpdir

34
tests/impure/default.nix Normal file
View File

@ -0,0 +1,34 @@
{
lib,
# dream2nix
callPackageDream,
utils,
...
}:
let
l = lib // builtins;
allTestFiles =
l.attrNames
(l.filterAttrs
(name: type: type == "regular" && l.hasPrefix "test_" name)
(l.readDir ./.));
allTests =
l.map
(file: callPackageDream ("${./.}/${file}") {})
allTestFiles;
executeAll = utils.writePureShellScript
[]
''
for test in ${toString allTests}; do
$test
done
'';
in
executeAll

23
tests/impure/test_go.nix Normal file
View File

@ -0,0 +1,23 @@
{
lib,
# dream2nix
apps,
utils,
...
}:
let
l = lib // builtins;
cli = apps.cli.program;
in
utils.writePureShellScript
[]
''
${cli} add github:tweag/gomod2nix/67f22dd738d092c6ba88e420350ada0ed4992ae8 \
--no-default-nix \
--translator gomod2nix \
--attribute-name gomod2nix
''

View File

@ -0,0 +1,26 @@
{
lib,
# dream2nix
apps,
utils,
...
}:
let
l = lib // builtins;
cli = apps.cli.program;
in
utils.writePureShellScript
[]
''
${cli} add github:mattermost/mattermost-webapp/v6.1.0 \
--no-default-nix \
--translator package-lock \
--attribute-name mattermost-webapp \
--arg name="{automatic}" \
--arg noDev=false \
--arg nodejs=14
''

View File

@ -0,0 +1,27 @@
{
lib,
# dream2nix
apps,
utils,
...
}:
let
l = lib // builtins;
cli = apps.cli.program;
in
utils.writePureShellScript
[]
''
${cli} add github:prettier/prettier/2.4.1 \
--no-default-nix \
--translator yarn-lock \
--attribute-name prettier \
--arg name="{automatic}" \
--arg noDev=false \
--arg nodejs=14 \
--arg peer=false
''

View File

@ -28,8 +28,14 @@ let
url = "https://github.com/prettier/prettier/tarball/2.4.1"; url = "https://github.com/prettier/prettier/tarball/2.4.1";
sha256 = "19b37qakhlsnr2n5bgv83aih5npgzbad1d2p2rs3zbq5syqbxdyi"; sha256 = "19b37qakhlsnr2n5bgv83aih5npgzbad1d2p2rs3zbq5syqbxdyi";
}; };
cmds = outputs: [ cmds = outputs:
"${outputs.defaultPackage}/bin/prettier --version | grep -q 2.4.1 && mkdir $out" let
prettier = outputs.defaultPackage.overrideAttrs (old: {
dontBuild = true;
});
in
[
"${prettier}/bin/prettier --version | grep -q 2.4.1 && mkdir $out"
]; ];
}; };
}; };