/** SDL format support for PackageRecipe Copyright: © 2014-2015 rejectedsoftware e.K. License: Subject to the terms of the MIT license, as written in the included LICENSE.txt file. Authors: Sönke Ludwig */ module dub.recipe.sdl; import dub.compilers.compiler; import dub.dependency; import dub.internal.dyaml.stdsumtype; import dub.internal.logging; import dub.internal.sdlang; import dub.internal.vibecompat.inet.path; import dub.recipe.packagerecipe; import std.algorithm : map; import std.array : array; import std.conv; import std.string : startsWith, format; deprecated("Use `parseSDL(PackageRecipe, string, PackageName, string)` instead") void parseSDL(ref PackageRecipe recipe, string sdl, string parent_name, string filename) { parseSDL(recipe, parseSource(sdl, filename), PackageName(parent_name)); } deprecated("Use `parseSDL(PackageRecipe, Tag, PackageName)` instead") void parseSDL(ref PackageRecipe recipe, Tag sdl, string parent_name) { parseSDL(recipe, sdl, PackageName(parent_name)); } void parseSDL(ref PackageRecipe recipe, string sdl, in PackageName parent, string filename) { parseSDL(recipe, parseSource(sdl, filename), parent); } void parseSDL(ref PackageRecipe recipe, Tag sdl, in PackageName parent = PackageName.init) { Tag[] subpacks; Tag[] configs; // parse top-level fields foreach (n; sdl.all.tags) { enforceSDL(n.name.length > 0, "Anonymous tags are not allowed at the root level.", n); switch (n.fullName) { default: break; case "name": recipe.name = n.stringTagValue; break; case "version": recipe.version_ = n.stringTagValue; break; case "description": recipe.description = n.stringTagValue; break; case "homepage": recipe.homepage = n.stringTagValue; break; case "authors": recipe.authors ~= n.stringArrayTagValue; break; case "copyright": recipe.copyright = n.stringTagValue; break; case "license": recipe.license = n.stringTagValue; break; case "subPackage": subpacks ~= n; break; case "configuration": configs ~= n; break; case "buildType": auto name = n.stringTagValue(true); BuildSettingsTemplate bt; parseBuildSettings(n, bt, parent); recipe.buildTypes[name] = bt; break; case "toolchainRequirements": parseToolchainRequirements(recipe.toolchainRequirements, n); break; case "x:ddoxFilterArgs": recipe.ddoxFilterArgs ~= n.stringArrayTagValue; break; case "x:ddoxTool": recipe.ddoxTool = n.stringTagValue; break; } } enforceSDL(recipe.name.length > 0, "The package \"name\" field is missing or empty.", sdl); const full_name = parent.toString().length ? PackageName(parent.toString() ~ ":" ~ recipe.name) : PackageName(recipe.name); // parse general build settings parseBuildSettings(sdl, recipe.buildSettings, full_name); // parse configurations recipe.configurations.length = configs.length; foreach (i, n; configs) { parseConfiguration(n, recipe.configurations[i], full_name); } // finally parse all sub packages recipe.subPackages.length = subpacks.length; foreach (i, n; subpacks) { if (n.values.length) { recipe.subPackages[i].path = n.stringTagValue; } else { enforceSDL(n.attributes.length == 0, "No attributes allowed for inline sub package definitions.", n); parseSDL(recipe.subPackages[i].recipe, n, full_name); } } } Tag toSDL(const scope ref PackageRecipe recipe) { Tag ret = new Tag; void add(T)(string field, T value) { ret.add(new Tag(null, field, [Value(value)])); } add("name", recipe.name); if (recipe.version_.length) add("version", recipe.version_); if (recipe.description.length) add("description", recipe.description); if (recipe.homepage.length) add("homepage", recipe.homepage); if (recipe.authors.length) ret.add(new Tag(null, "authors", recipe.authors.map!(a => Value(a)).array)); if (recipe.copyright.length) add("copyright", recipe.copyright); if (recipe.license.length) add("license", recipe.license); foreach (name, settings; recipe.buildTypes) { auto t = new Tag(null, "buildType", [Value(name)]); t.add(settings.toSDL()); ret.add(t); } if (!recipe.toolchainRequirements.empty) { ret.add(toSDL(recipe.toolchainRequirements)); } if (recipe.ddoxFilterArgs.length) ret.add(new Tag("x", "ddoxFilterArgs", recipe.ddoxFilterArgs.map!(a => Value(a)).array)); if (recipe.ddoxTool.length) ret.add(new Tag("x", "ddoxTool", [Value(recipe.ddoxTool)])); ret.add(recipe.buildSettings.toSDL()); foreach(config; recipe.configurations) ret.add(config.toSDL()); foreach (i, subPackage; recipe.subPackages) { if (subPackage.path !is null) { add("subPackage", subPackage.path); } else { auto t = subPackage.recipe.toSDL(); t.name = "subPackage"; ret.add(t); } } return ret; } private void parseBuildSettings(Tag settings, ref BuildSettingsTemplate bs, in PackageName name) { foreach (setting; settings.all.tags) parseBuildSetting(setting, bs, name); } private void parseBuildSetting(Tag setting, ref BuildSettingsTemplate bs, in PackageName name) { switch (setting.fullName) { default: break; case "dependency": parseDependency(setting, bs, name); break; case "systemDependencies": bs.systemDependencies = setting.stringTagValue; break; case "targetType": bs.targetType = setting.stringTagValue.to!TargetType; break; case "targetName": bs.targetName = setting.stringTagValue; break; case "targetPath": bs.targetPath = setting.stringTagValue; break; case "workingDirectory": bs.workingDirectory = setting.stringTagValue; break; case "subConfiguration": auto args = setting.stringArrayTagValue; enforceSDL(args.length == 2, "Expecting package and configuration names as arguments.", setting); bs.subConfigurations[expandPackageName(args[0], name, setting)] = args[1]; break; case "dflags": setting.parsePlatformStringArray(bs.dflags); break; case "lflags": setting.parsePlatformStringArray(bs.lflags); break; case "libs": setting.parsePlatformStringArray(bs.libs); break; case "sourceFiles": setting.parsePlatformStringArray(bs.sourceFiles); break; case "sourcePaths": setting.parsePlatformStringArray(bs.sourcePaths); break; case "cSourcePaths": setting.parsePlatformStringArray(bs.cSourcePaths); break; case "excludedSourceFiles": setting.parsePlatformStringArray(bs.excludedSourceFiles); break; case "mainSourceFile": bs.mainSourceFile = setting.stringTagValue; break; case "injectSourceFiles": setting.parsePlatformStringArray(bs.injectSourceFiles); break; case "copyFiles": setting.parsePlatformStringArray(bs.copyFiles); break; case "extraDependencyFiles": setting.parsePlatformStringArray(bs.extraDependencyFiles); break; case "versions": setting.parsePlatformStringArray(bs.versions); break; case "debugVersions": setting.parsePlatformStringArray(bs.debugVersions); break; case "x:versionFilters": setting.parsePlatformStringArray(bs.versionFilters); break; case "x:debugVersionFilters": setting.parsePlatformStringArray(bs.debugVersionFilters); break; case "importPaths": setting.parsePlatformStringArray(bs.importPaths); break; case "cImportPaths": setting.parsePlatformStringArray(bs.cImportPaths); break; case "stringImportPaths": setting.parsePlatformStringArray(bs.stringImportPaths); break; case "preGenerateCommands": setting.parsePlatformStringArray(bs.preGenerateCommands); break; case "postGenerateCommands": setting.parsePlatformStringArray(bs.postGenerateCommands); break; case "preBuildCommands": setting.parsePlatformStringArray(bs.preBuildCommands); break; case "postBuildCommands": setting.parsePlatformStringArray(bs.postBuildCommands); break; case "preRunCommands": setting.parsePlatformStringArray(bs.preRunCommands); break; case "postRunCommands": setting.parsePlatformStringArray(bs.postRunCommands); break; case "environments": setting.parsePlatformStringAA(bs.environments); break; case "buildEnvironments": setting.parsePlatformStringAA(bs.buildEnvironments); break; case "runEnvironments": setting.parsePlatformStringAA(bs.runEnvironments); break; case "preGenerateEnvironments": setting.parsePlatformStringAA(bs.preGenerateEnvironments); break; case "postGenerateEnvironments": setting.parsePlatformStringAA(bs.postGenerateEnvironments); break; case "preBuildEnvironments": setting.parsePlatformStringAA(bs.preBuildEnvironments); break; case "postBuildEnvironments": setting.parsePlatformStringAA(bs.postBuildEnvironments); break; case "preRunEnvironments": setting.parsePlatformStringAA(bs.preRunEnvironments); break; case "postRunEnvironments": setting.parsePlatformStringAA(bs.postRunEnvironments); break; case "buildRequirements": setting.parsePlatformEnumArray!BuildRequirement(bs.buildRequirements); break; case "buildOptions": setting.parsePlatformEnumArray!BuildOption(bs.buildOptions); break; } } private void parseDependency(Tag t, ref BuildSettingsTemplate bs, in PackageName name) { enforceSDL(t.values.length != 0, "Missing dependency name.", t); enforceSDL(t.values.length == 1, "Multiple dependency names.", t); auto pkg = expandPackageName(t.values[0].expect!string(t), name, t); enforceSDL(pkg !in bs.dependencies, "The dependency '"~pkg~"' is specified more than once.", t); Dependency dep = Dependency.Any; auto attrs = t.attributes; if ("path" in attrs) { dep = Dependency(NativePath(attrs["path"][0].value.expect!string(t, t.fullName ~ " path"))); } else if ("repository" in attrs) { enforceSDL("version" in attrs, "Missing version specification.", t); dep = Dependency(Repository(attrs["repository"][0].value.expect!string(t, t.fullName ~ " repository"), attrs["version"][0].value.expect!string(t, t.fullName ~ " version"))); } else { enforceSDL("version" in attrs, "Missing version specification.", t); dep = Dependency(attrs["version"][0].value.expect!string(t, t.fullName ~ " version")); } if ("optional" in attrs) dep.optional = attrs["optional"][0].value.expect!bool(t, t.fullName ~ " optional"); if ("default" in attrs) dep.default_ = attrs["default"][0].value.expect!bool(t, t.fullName ~ " default"); bs.dependencies[pkg] = dep; BuildSettingsTemplate dbs; parseBuildSettings(t, bs.dependencies[pkg].settings, name); } private void parseConfiguration(Tag t, ref ConfigurationInfo ret, in PackageName name) { ret.name = t.stringTagValue(true); foreach (f; t.tags) { switch (f.fullName) { default: parseBuildSetting(f, ret.buildSettings, name); break; case "platforms": ret.platforms ~= f.stringArrayTagValue; break; } } } private Tag toSDL(const scope ref ConfigurationInfo config) { auto ret = new Tag(null, "configuration", [Value(config.name)]); if (config.platforms.length) ret.add(new Tag(null, "platforms", config.platforms[].map!(p => Value(p)).array)); ret.add(config.buildSettings.toSDL()); return ret; } private Tag[] toSDL(const scope ref BuildSettingsTemplate bs) { Tag[] ret; void add(string name, string value, string namespace = null) { ret ~= new Tag(namespace, name, [Value(value)]); } void adda(string name, string suffix, in string[] values, string namespace = null) { ret ~= new Tag(namespace, name, values[].map!(v => Value(v)).array, suffix.length ? [new Attribute(null, "platform", Value(suffix))] : null); } void addaa(string name, string suffix, in string[string] values, string namespace = null) { foreach (k, v; values) { ret ~= new Tag(namespace, name, [Value(k), Value(v)], suffix.length ? [new Attribute(null, "platform", Value(suffix))] : null); } } string[] toNameArray(T, U)(U bits) if(is(T == enum)) { string[] ret; foreach (m; __traits(allMembers, T)) if (bits & __traits(getMember, T, m)) ret ~= m; return ret; } foreach (pack, d; bs.dependencies) { Attribute[] attribs; d.visit!( (const Repository r) { attribs ~= new Attribute(null, "repository", Value(r.toString())); attribs ~= new Attribute(null, "version", Value(r.ref_)); }, (const NativePath p) { attribs ~= new Attribute(null, "path", Value(p.toString())); }, (const VersionRange v) { attribs ~= new Attribute(null, "version", Value(v.toString())); }, ); if (d.optional) attribs ~= new Attribute(null, "optional", Value(true)); auto t = new Tag(null, "dependency", [Value(pack)], attribs); if (d.settings !is typeof(d.settings).init) t.add(d.settings.toSDL()); ret ~= t; } if (bs.systemDependencies !is null) add("systemDependencies", bs.systemDependencies); if (bs.targetType != TargetType.autodetect) add("targetType", bs.targetType.to!string()); if (bs.targetPath.length) add("targetPath", bs.targetPath); if (bs.targetName.length) add("targetName", bs.targetName); if (bs.workingDirectory.length) add("workingDirectory", bs.workingDirectory); if (bs.mainSourceFile.length) add("mainSourceFile", bs.mainSourceFile); foreach (pack, conf; bs.subConfigurations) ret ~= new Tag(null, "subConfiguration", [Value(pack), Value(conf)]); foreach (suffix, arr; bs.dflags) adda("dflags", suffix, arr); foreach (suffix, arr; bs.lflags) adda("lflags", suffix, arr); foreach (suffix, arr; bs.libs) adda("libs", suffix, arr); foreach (suffix, arr; bs.sourceFiles) adda("sourceFiles", suffix, arr); foreach (suffix, arr; bs.sourcePaths) adda("sourcePaths", suffix, arr); foreach (suffix, arr; bs.cSourcePaths) adda("cSourcePaths", suffix, arr); foreach (suffix, arr; bs.excludedSourceFiles) adda("excludedSourceFiles", suffix, arr); foreach (suffix, arr; bs.injectSourceFiles) adda("injectSourceFiles", suffix, arr); foreach (suffix, arr; bs.copyFiles) adda("copyFiles", suffix, arr); foreach (suffix, arr; bs.extraDependencyFiles) adda("extraDependencyFiles", suffix, arr); foreach (suffix, arr; bs.versions) adda("versions", suffix, arr); foreach (suffix, arr; bs.debugVersions) adda("debugVersions", suffix, arr); foreach (suffix, arr; bs.versionFilters) adda("versionFilters", suffix, arr, "x"); foreach (suffix, arr; bs.debugVersionFilters) adda("debugVersionFilters", suffix, arr, "x"); foreach (suffix, arr; bs.importPaths) adda("importPaths", suffix, arr); foreach (suffix, arr; bs.cImportPaths) adda("cImportPaths", suffix, arr); foreach (suffix, arr; bs.stringImportPaths) adda("stringImportPaths", suffix, arr); foreach (suffix, arr; bs.preGenerateCommands) adda("preGenerateCommands", suffix, arr); foreach (suffix, arr; bs.postGenerateCommands) adda("postGenerateCommands", suffix, arr); foreach (suffix, arr; bs.preBuildCommands) adda("preBuildCommands", suffix, arr); foreach (suffix, arr; bs.postBuildCommands) adda("postBuildCommands", suffix, arr); foreach (suffix, arr; bs.preRunCommands) adda("preRunCommands", suffix, arr); foreach (suffix, arr; bs.postRunCommands) adda("postRunCommands", suffix, arr); foreach (suffix, aa; bs.environments) addaa("environments", suffix, aa); foreach (suffix, aa; bs.buildEnvironments) addaa("buildEnvironments", suffix, aa); foreach (suffix, aa; bs.runEnvironments) addaa("runEnvironments", suffix, aa); foreach (suffix, aa; bs.preGenerateEnvironments) addaa("preGenerateEnvironments", suffix, aa); foreach (suffix, aa; bs.postGenerateEnvironments) addaa("postGenerateEnvironments", suffix, aa); foreach (suffix, aa; bs.preBuildEnvironments) addaa("preBuildEnvironments", suffix, aa); foreach (suffix, aa; bs.postBuildEnvironments) addaa("postBuildEnvironments", suffix, aa); foreach (suffix, aa; bs.preRunEnvironments) addaa("preRunEnvironments", suffix, aa); foreach (suffix, aa; bs.postRunEnvironments) addaa("postRunEnvironments", suffix, aa); foreach (suffix, bits; bs.buildRequirements) adda("buildRequirements", suffix, toNameArray!BuildRequirement(bits)); foreach (suffix, bits; bs.buildOptions) adda("buildOptions", suffix, toNameArray!BuildOption(bits)); return ret; } private void parseToolchainRequirements(ref ToolchainRequirements tr, Tag tag) { foreach (attr; tag.attributes) tr.addRequirement(attr.name, attr.value.expect!string(tag)); } private Tag toSDL(const ref ToolchainRequirements tr) { Attribute[] attrs; if (tr.dub != VersionRange.Any) attrs ~= new Attribute("dub", Value(tr.dub.toString())); if (tr.frontend != VersionRange.Any) attrs ~= new Attribute("frontend", Value(tr.frontend.toString())); if (tr.dmd != VersionRange.Any) attrs ~= new Attribute("dmd", Value(tr.dmd.toString())); if (tr.ldc != VersionRange.Any) attrs ~= new Attribute("ldc", Value(tr.ldc.toString())); if (tr.gdc != VersionRange.Any) attrs ~= new Attribute("gdc", Value(tr.gdc.toString())); return new Tag(null, "toolchainRequirements", null, attrs); } private string expandPackageName(string name, in PackageName parent, Tag tag) { import std.algorithm : canFind; if (!name.startsWith(":")) return name; enforceSDL(!parent.sub.length, "Short-hand packages syntax not " ~ "allowed within sub packages: %s -> %s".format(parent, name), tag); return parent.toString() ~ name; } private string stringTagValue(Tag t, bool allow_child_tags = false) { enforceSDL(t.values.length > 0, format("Missing string value for '%s'.", t.fullName), t); enforceSDL(t.values.length == 1, format("Expected only one value for '%s'.", t.fullName), t); enforceSDL(allow_child_tags || t.tags.length == 0, format("No child tags allowed for '%s'.", t.fullName), t); // Q: should attributes be disallowed, or just ignored for forward compatibility reasons? //enforceSDL(t.attributes.length == 0, format("No attributes allowed for '%s'.", t.fullName), t); return t.values[0].expect!string(t); } private T expect(T)( Value value, Tag errorInfo, string customFieldName = null, string file = __FILE__, int line = __LINE__ ) { return value.match!( (T v) => v, (fallback) { enforceSDL(false, format("Expected value of type " ~ T.stringof ~ " for '%s', but got %s.", customFieldName.length ? customFieldName : errorInfo.fullName, typeof(fallback).stringof), errorInfo, file, line); return T.init; } ); } private string[] stringArrayTagValue(Tag t, bool allow_child_tags = false) { enforceSDL(allow_child_tags || t.tags.length == 0, format("No child tags allowed for '%s'.", t.fullName), t); // Q: should attributes be disallowed, or just ignored for forward compatibility reasons? //enforceSDL(t.attributes.length == 0, format("No attributes allowed for '%s'.", t.fullName), t); string[] ret; foreach (i, v; t.values) { ret ~= v.expect!string(t, text(t.fullName, "[", i, "]")); } return ret; } private string getPlatformSuffix(Tag t, string file = __FILE__, int line = __LINE__) { string platform; if ("platform" in t.attributes) platform = t.attributes["platform"][0].value.expect!string(t, t.fullName ~ " platform", file, line); return platform; } private void parsePlatformStringArray(Tag t, ref string[][string] dst) { string platform = t.getPlatformSuffix; dst[platform] ~= t.values.map!(v => v.expect!string(t)).array; } private void parsePlatformStringAA(Tag t, ref string[string][string] dst) { string platform = t.getPlatformSuffix; enforceSDL(t.values.length == 2, format("Values for '%s' must be 2 required.", t.fullName), t); dst[platform][t.values[0].expect!string(t)] = t.values[1].expect!string(t); } private void parsePlatformEnumArray(E, Es)(Tag t, ref Es[string] dst) { string platform = t.getPlatformSuffix; foreach (v; t.values) { if (platform !in dst) dst[platform] = Es.init; dst[platform] |= v.expect!string(t).to!E; } } private void enforceSDL(bool condition, lazy string message, Tag tag, string file = __FILE__, int line = __LINE__) { if (!condition) { throw new Exception(format("%s(%s): Error: %s", tag.location.file, tag.location.line + 1, message), file, line); } } // Just a wrapper around `parseSDL` for easier testing version (unittest) private void parseSDLTest(ref PackageRecipe recipe, string sdl) { parseSDL(recipe, parseSource(sdl, "testfile"), PackageName.init); } unittest { // test all possible fields auto sdl = `name "projectname"; description "project description"; homepage "http://example.com" authors "author 1" "author 2" authors "author 3" copyright "copyright string" license "license string" version "1.0.0" subPackage { name "subpackage1" } subPackage { name "subpackage2" dependency "projectname:subpackage1" version="*" } subPackage "pathsp3" configuration "config1" { platforms "windows" "linux" targetType "library" } configuration "config2" { platforms "windows-x86" targetType "executable" } buildType "debug" { dflags "-g" "-debug" } buildType "release" { dflags "-release" "-O" } toolchainRequirements dub="~>1.11.0" dmd="~>2.082" x:ddoxFilterArgs "-arg1" "-arg2" x:ddoxFilterArgs "-arg3" x:ddoxTool "ddoxtool" dependency ":subpackage1" optional=false path="." { dflags "-g" "-debug" } dependency "somedep" version="1.0.0" optional=true systemDependencies "system dependencies" targetType "executable" targetName "target name" targetPath "target path" workingDirectory "working directory" subConfiguration ":subpackage2" "library" buildRequirements "allowWarnings" "silenceDeprecations" buildOptions "verbose" "ignoreUnknownPragmas" libs "lib1" "lib2" libs "lib3" sourceFiles "source1" "source2" sourceFiles "source3" sourcePaths "sourcepath1" "sourcepath2" sourcePaths "sourcepath3" cSourcePaths "csourcepath1" "csourcepath2" cSourcePaths "csourcepath3" excludedSourceFiles "excluded1" "excluded2" excludedSourceFiles "excluded3" mainSourceFile "main source" injectSourceFiles "finalbinarysourcefile.d" "extrafile" copyFiles "copy1" "copy2" copyFiles "copy3" extraDependencyFiles "extradepfile1" "extradepfile2" extraDependencyFiles "extradepfile3" versions "version1" "version2" versions "version3" debugVersions "debug1" "debug2" debugVersions "debug3" x:versionFilters "version1" "version2" x:versionFilters "version3" x:versionFilters x:debugVersionFilters "debug1" "debug2" x:debugVersionFilters "debug3" x:debugVersionFilters importPaths "import1" "import2" importPaths "import3" cImportPaths "cimport1" "cimport2" cImportPaths "cimport3" stringImportPaths "string1" "string2" stringImportPaths "string3" preGenerateCommands "preg1" "preg2" preGenerateCommands "preg3" postGenerateCommands "postg1" "postg2" postGenerateCommands "postg3" preBuildCommands "preb1" "preb2" preBuildCommands "preb3" postBuildCommands "postb1" "postb2" postBuildCommands "postb3" preRunCommands "prer1" "prer2" preRunCommands "prer3" postRunCommands "postr1" "postr2" postRunCommands "postr3" environments "Var1" "env" buildEnvironments "Var2" "buildEnv" runEnvironments "Var3" "runEnv" preGenerateEnvironments "Var4" "preGenEnv" postGenerateEnvironments "Var5" "postGenEnv" preBuildEnvironments "Var6" "preBuildEnv" postBuildEnvironments "Var7" "postBuildEnv" preRunEnvironments "Var8" "preRunEnv" postRunEnvironments "Var9" "postRunEnv" dflags "df1" "df2" dflags "df3" lflags "lf1" "lf2" lflags "lf3" `; PackageRecipe rec1; parseSDLTest(rec1, sdl); PackageRecipe rec; parseSDL(rec, rec1.toSDL()); // verify that all fields are serialized properly assert(rec.name == "projectname"); assert(rec.description == "project description"); assert(rec.homepage == "http://example.com"); assert(rec.authors == ["author 1", "author 2", "author 3"]); assert(rec.copyright == "copyright string"); assert(rec.license == "license string"); assert(rec.version_ == "1.0.0"); assert(rec.subPackages.length == 3); assert(rec.subPackages[0].path == ""); assert(rec.subPackages[0].recipe.name == "subpackage1"); assert(rec.subPackages[1].path == ""); assert(rec.subPackages[1].recipe.name == "subpackage2"); assert(rec.subPackages[1].recipe.buildSettings.dependencies.length == 1); assert("projectname:subpackage1" in rec.subPackages[1].recipe.buildSettings.dependencies); assert(rec.subPackages[2].path == "pathsp3"); assert(rec.configurations.length == 2); assert(rec.configurations[0].name == "config1"); assert(rec.configurations[0].platforms == ["windows", "linux"]); assert(rec.configurations[0].buildSettings.targetType == TargetType.library); assert(rec.configurations[1].name == "config2"); assert(rec.configurations[1].platforms == ["windows-x86"]); assert(rec.configurations[1].buildSettings.targetType == TargetType.executable); assert(rec.buildTypes.length == 2); assert(rec.buildTypes["debug"].dflags == ["": ["-g", "-debug"]]); assert(rec.buildTypes["release"].dflags == ["": ["-release", "-O"]]); assert(rec.toolchainRequirements.dub == VersionRange.fromString("~>1.11.0")); assert(rec.toolchainRequirements.frontend == VersionRange.Any); assert(rec.toolchainRequirements.dmd == VersionRange.fromString("~>2.82.0")); assert(rec.toolchainRequirements.ldc == VersionRange.Any); assert(rec.toolchainRequirements.gdc == VersionRange.Any); assert(rec.ddoxFilterArgs == ["-arg1", "-arg2", "-arg3"], rec.ddoxFilterArgs.to!string); assert(rec.ddoxTool == "ddoxtool"); assert(rec.buildSettings.dependencies.length == 2); assert(rec.buildSettings.dependencies["projectname:subpackage1"].optional == false); assert(rec.buildSettings.dependencies["projectname:subpackage1"].path == NativePath(".")); assert(rec.buildSettings.dependencies["projectname:subpackage1"].settings.dflags == ["":["-g", "-debug"]]); assert(rec.buildSettings.dependencies["somedep"].version_.toString() == "1.0.0"); assert(rec.buildSettings.dependencies["somedep"].optional == true); assert(rec.buildSettings.systemDependencies == "system dependencies"); assert(rec.buildSettings.targetType == TargetType.executable); assert(rec.buildSettings.targetName == "target name"); assert(rec.buildSettings.targetPath == "target path"); assert(rec.buildSettings.workingDirectory == "working directory"); assert(rec.buildSettings.subConfigurations.length == 1); assert(rec.buildSettings.subConfigurations["projectname:subpackage2"] == "library"); assert(rec.buildSettings.buildRequirements == ["": cast(Flags!BuildRequirement)(BuildRequirement.allowWarnings | BuildRequirement.silenceDeprecations)]); assert(rec.buildSettings.buildOptions == ["": cast(Flags!BuildOption)(BuildOption.verbose | BuildOption.ignoreUnknownPragmas)]); assert(rec.buildSettings.libs == ["": ["lib1", "lib2", "lib3"]]); assert(rec.buildSettings.sourceFiles == ["": ["source1", "source2", "source3"]]); assert(rec.buildSettings.sourcePaths == ["": ["sourcepath1", "sourcepath2", "sourcepath3"]]); assert(rec.buildSettings.cSourcePaths == ["": ["csourcepath1", "csourcepath2", "csourcepath3"]]); assert(rec.buildSettings.excludedSourceFiles == ["": ["excluded1", "excluded2", "excluded3"]]); assert(rec.buildSettings.mainSourceFile == "main source"); assert(rec.buildSettings.sourceFiles == ["": ["source1", "source2", "source3"]]); assert(rec.buildSettings.injectSourceFiles == ["": ["finalbinarysourcefile.d", "extrafile"]]); assert(rec.buildSettings.extraDependencyFiles == ["": ["extradepfile1", "extradepfile2", "extradepfile3"]]); assert(rec.buildSettings.versions == ["": ["version1", "version2", "version3"]]); assert(rec.buildSettings.debugVersions == ["": ["debug1", "debug2", "debug3"]]); assert(rec.buildSettings.versionFilters == ["": ["version1", "version2", "version3"]]); assert(rec.buildSettings.debugVersionFilters == ["": ["debug1", "debug2", "debug3"]]); assert(rec.buildSettings.importPaths == ["": ["import1", "import2", "import3"]]); assert(rec.buildSettings.cImportPaths == ["": ["cimport1", "cimport2", "cimport3"]]); assert(rec.buildSettings.stringImportPaths == ["": ["string1", "string2", "string3"]]); assert(rec.buildSettings.preGenerateCommands == ["": ["preg1", "preg2", "preg3"]]); assert(rec.buildSettings.postGenerateCommands == ["": ["postg1", "postg2", "postg3"]]); assert(rec.buildSettings.preBuildCommands == ["": ["preb1", "preb2", "preb3"]]); assert(rec.buildSettings.postBuildCommands == ["": ["postb1", "postb2", "postb3"]]); assert(rec.buildSettings.preRunCommands == ["": ["prer1", "prer2", "prer3"]]); assert(rec.buildSettings.postRunCommands == ["": ["postr1", "postr2", "postr3"]]); assert(rec.buildSettings.environments == ["": ["Var1": "env"]]); assert(rec.buildSettings.buildEnvironments == ["": ["Var2": "buildEnv"]]); assert(rec.buildSettings.runEnvironments == ["": ["Var3": "runEnv"]]); assert(rec.buildSettings.preGenerateEnvironments == ["": ["Var4": "preGenEnv"]]); assert(rec.buildSettings.postGenerateEnvironments == ["": ["Var5": "postGenEnv"]]); assert(rec.buildSettings.preBuildEnvironments == ["": ["Var6": "preBuildEnv"]]); assert(rec.buildSettings.postBuildEnvironments == ["": ["Var7": "postBuildEnv"]]); assert(rec.buildSettings.preRunEnvironments == ["": ["Var8": "preRunEnv"]]); assert(rec.buildSettings.postRunEnvironments == ["": ["Var9": "postRunEnv"]]); assert(rec.buildSettings.dflags == ["": ["df1", "df2", "df3"]]); assert(rec.buildSettings.lflags == ["": ["lf1", "lf2", "lf3"]]); } unittest { // test platform identifiers auto sdl = `name "testproject" dflags "-a" "-b" platform="windows-x86" dflags "-c" platform="windows-x86" dflags "-e" "-f" dflags "-g" dflags "-h" "-i" platform="linux" dflags "-j" platform="linux" `; PackageRecipe rec; parseSDLTest(rec, sdl); assert(rec.buildSettings.dflags.length == 3); assert(rec.buildSettings.dflags["windows-x86"] == ["-a", "-b", "-c"]); assert(rec.buildSettings.dflags[""] == ["-e", "-f", "-g"]); assert(rec.buildSettings.dflags["linux"] == ["-h", "-i", "-j"]); } unittest { // test for missing name field import std.exception; auto sdl = `description "missing name"`; PackageRecipe rec; assertThrown(parseSDLTest(rec, sdl)); } unittest { // test single value fields import std.exception; PackageRecipe rec; assertThrown!Exception(parseSDLTest(rec, `name "hello" "world"`)); assertThrown!Exception(parseSDLTest(rec, `name`)); assertThrown!Exception(parseSDLTest(rec, `name 10`)); assertThrown!Exception(parseSDLTest(rec, `name "hello" { world }`)); assertThrown!Exception(parseSDLTest(rec, `name "" versions "hello" 10`)); } unittest { // test basic serialization PackageRecipe p; p.name = "test"; p.authors = ["foo", "bar"]; p.buildSettings.dflags["windows"] = ["-a"]; p.buildSettings.lflags[""] = ["-b", "-c"]; auto sdl = toSDL(p).toSDLDocument(); assert(sdl == `name "test" authors "foo" "bar" dflags "-a" platform="windows" lflags "-b" "-c" `); } unittest { auto sdl = "name \"test\"\nsourcePaths"; PackageRecipe rec; parseSDLTest(rec, sdl); assert("" in rec.buildSettings.sourcePaths); } unittest { auto sdl = "name \"test\"\ncSourcePaths"; PackageRecipe rec; parseSDLTest(rec, sdl); assert("" in rec.buildSettings.cSourcePaths); } unittest { auto sdl = `name "test" dependency "package" repository="git+https://some.url" version="12345678" `; PackageRecipe rec; parseSDLTest(rec, sdl); auto dependency = rec.buildSettings.dependencies["package"]; assert(!dependency.repository.empty); assert(dependency.repository.ref_ == "12345678"); } unittest { PackageRecipe p; p.name = "test"; auto repository = Repository("git+https://some.url", "12345678"); p.buildSettings.dependencies["package"] = Dependency(repository); auto sdl = toSDL(p).toSDLDocument(); assert(sdl == `name "test" dependency "package" repository="git+https://some.url" version="12345678" `); }