diff --git a/changelog/selections_from_parent_dir.dd b/changelog/selections_from_parent_dir.dd new file mode 100644 index 0000000..923d8e4 --- /dev/null +++ b/changelog/selections_from_parent_dir.dd @@ -0,0 +1,16 @@ +`dub.selections.json` files are now looked up in parent directories too + +In case the root package directory doesn't contain a `dub.selections.json` +file, dub now looks in parent directories too and potentially uses the +first (deepest) one it finds - if and only if that JSON file contains an +optional new `"inheritable": true` flag. + +This allows using a 'central' `dub.selections.json` file for a repository +containing multiple dub projects, making it automatically apply to all +builds in that source tree if located in the repository root directory +(unless a local `dub.selections.json` overrides it). + +Such an inherited selections file is never mutated when running dub for a +nested project, i.e., changes are always saved to a *local* +`dub.selections.json` file. E.g., when running `dub upgrade` for a nested +project. diff --git a/source/dub/dub.d b/source/dub/dub.d index 2b42fb7..98247b8 100644 --- a/source/dub/dub.d +++ b/source/dub/dub.d @@ -522,7 +522,7 @@ /// Loads a specific package as the main project package (can be a sub package) void loadPackage(Package pack) { - auto selections = Project.loadSelections(pack, m_packageManager); + auto selections = Project.loadSelections(pack.path, m_packageManager); m_project = new Project(m_packageManager, pack, selections); } diff --git a/source/dub/packagemanager.d b/source/dub/packagemanager.d index e3b9ff4..fd0c69e 100644 --- a/source/dub/packagemanager.d +++ b/source/dub/packagemanager.d @@ -60,6 +60,15 @@ } } +/// A SelectionsFile associated with its file-system path. +struct SelectionsFileLookupResult { + /// The absolute path to the dub.selections.json file + /// (potentially inherited from a parent directory of the root package). + NativePath absolutePath; + /// The parsed dub.selections.json file. + SelectionsFile selectionsFile; +} + /// The PackageManager can retrieve present packages and get / remove /// packages. class PackageManager { @@ -1151,32 +1160,50 @@ * However, due to it being a filesystem interaction, it is managed * from the `PackageManager`. * - * Under normal conditions, either `null` is returned (if no selection - * file exists or parsing encountered an error), or a `SelectionFile` - * is returned. Note that the returned `SelectionsFile` might use an - * unsupported version (see `SelectionsFile` documentation). - * - * Note that this is currently not part of the public API, and won't be - * until there is a clear need for it. - * * Params: - * project = The `Package` for which to load a selection file. + * absProjectPath = The absolute path to the root package/project for + * which to load the selections file. * * Returns: - * The parsed `SelectionsFile`, or a null instance if none exists. + * Either `null` (if no selections file exists or parsing encountered an error), + * or a `SelectionsFileLookupResult`. Note that the nested `SelectionsFile` + * might use an unsupported version (see `SelectionsFile` documentation). */ - package final Nullable!SelectionsFile readSelections(in Package project) { + Nullable!SelectionsFileLookupResult readSelections(in NativePath absProjectPath) + in (absProjectPath.absolute) { import dub.internal.configy.Read; - const path = (project.path ~ "dub.selections.json"); - if (!this.existsFile(path)) - return typeof(return).init; + alias N = typeof(return); + + // check for dub.selections.json in root project dir first, then walk up its + // parent directories and look for inheritable dub.selections.json files + const path = this.findSelections(absProjectPath); + if (path.empty) return N.init; const content = this.readText(path); // TODO: Remove `StrictMode.Warn` after v1.40 release // The default is to error, but as the previous parser wasn't // complaining, we should first warn the user. - return wrapException(parseConfigString!SelectionsFile( + auto selections = wrapException(parseConfigString!SelectionsFile( content, path.toNativeString(), StrictMode.Warn)); + // Could not parse file + if (selections.isNull()) + return N.init; + // Non-inheritable selections found + if (!path.startsWith(absProjectPath) && !selections.get().inheritable) + return N.init; + return N(SelectionsFileLookupResult(path, selections.get())); + } + + /// Helper function to walk up the filesystem and find `dub.selections.json` + private NativePath findSelections(in NativePath dir) + { + const path = dir ~ "dub.selections.json"; + if (this.existsFile(path)) + return path; + if (!dir.hasParentPath) + return NativePath.init; + return this.findSelections(dir.parentPath); + } /** @@ -1208,12 +1235,14 @@ { Json json = selectionsToJSON(s); assert(json.type == Json.Type.object); - assert(json.length == 2); + assert(json.length == 2 || json.length == 3); assert(json["versions"].type != Json.Type.undefined); auto result = appender!string(); result.put("{\n\t\"fileVersion\": "); result.writeJsonString(json["fileVersion"]); + if (s.inheritable) + result.put(",\n\t\"inheritable\": true"); result.put(",\n\t\"versions\": {"); auto vers = json["versions"].get!(Json[string]); bool first = true; @@ -1234,6 +1263,8 @@ { Json serialized = Json.emptyObject; serialized["fileVersion"] = s.fileVersion; + if (s.inheritable) + serialized["inheritable"] = true; serialized["versions"] = Json.emptyObject; foreach (p, dep; s.versions) serialized["versions"][p] = dep.toJson(true); diff --git a/source/dub/project.d b/source/dub/project.d index d733f35..575e8eb 100644 --- a/source/dub/project.d +++ b/source/dub/project.d @@ -76,7 +76,7 @@ /// Ditto this(PackageManager package_manager, Package pack) { - auto selections = Project.loadSelections(pack, package_manager); + auto selections = Project.loadSelections(pack.path, package_manager); this(package_manager, pack, selections); } @@ -98,32 +98,34 @@ * is returned. * * Params: - * pack = Package to load the selection file from. + * packPath = Absolute path of the Package to load the selection file from. * * Returns: * Always a non-null instance. */ - static package SelectedVersions loadSelections(in Package pack, PackageManager mgr) + static package SelectedVersions loadSelections(in NativePath packPath, PackageManager mgr) { import dub.version_; import dub.internal.dyaml.stdsumtype; - auto selected = mgr.readSelections(pack); - // Parsing error that will be displayed to the user or just no selections - if (selected.isNull()) + auto lookupResult = mgr.readSelections(packPath); + if (lookupResult.isNull()) // no file, or parsing error (displayed to the user) return new SelectedVersions(); - return selected.get().content.match!( + auto r = lookupResult.get(); + return r.selectionsFile.content.match!( (Selections!0 s) { logWarnTag("Unsupported version", - "File %s/dub.selections.json has fileVersion %s, which " ~ - "is not yet supported by DUB %s.", - pack.path, s.fileVersion, dubVersion); + "File %s has fileVersion %s, which is not yet supported by DUB %s.", + r.absolutePath, s.fileVersion, dubVersion); logWarn("Ignoring selections file. Use a newer DUB version " ~ "and set the appropriate toolchainRequirements in your recipe file"); return new SelectedVersions(); }, - (Selections!1 s) => new SelectedVersions(s), + (Selections!1 s) { + auto selectionsDir = r.absolutePath.parentPath; + return new SelectedVersions(s, selectionsDir.relativeTo(packPath)); + }, ); } @@ -1874,6 +1876,23 @@ this.m_bare = false; } + /** Constructs a new non-empty version selection, prefixing relative path + selections with the specified prefix. + + To be used in cases where the "dub.selections.json" file isn't located + in the root package directory. + */ + public this(Selections!1 data, NativePath relPathPrefix) + { + this(data); + if (relPathPrefix.empty) return; + foreach (ref dep; m_selections.versions.byValue) { + const depPath = dep.path; + if (!depPath.empty && !depPath.absolute) + dep = Dependency(relPathPrefix ~ depPath); + } + } + /** Constructs a new version selection from JSON data. The structure of the JSON document must match the contents of the @@ -1918,6 +1937,7 @@ { m_selections.fileVersion = versions.m_selections.fileVersion; m_selections.versions = versions.m_selections.versions.dup; + m_selections.inheritable = versions.m_selections.inheritable; m_dirty = true; } @@ -2078,6 +2098,8 @@ clear(); m_selections.fileVersion = fileVersion; scope(failure) clear(); + if (auto p = "inheritable" in json) + m_selections.inheritable = p.get!bool; foreach (string p, dep; json["versions"]) m_selections.versions[p] = dependencyFromJson(dep); } diff --git a/source/dub/recipe/selection.d b/source/dub/recipe/selection.d index 11758e8..f77df43 100644 --- a/source/dub/recipe/selection.d +++ b/source/dub/recipe/selection.d @@ -45,6 +45,18 @@ } /** + * Whether this dub.selections.json can be inherited by nested projects + * without local dub.selections.json + */ + public bool inheritable () const @safe pure nothrow @nogc + { + return this.content.match!( + (const Selections!0 _) => false, + (const Selections!1 s) => s.inheritable, + ); + } + + /** * The content of this selections file * * The underlying content can be accessed using @@ -102,6 +114,10 @@ else static if (Version == 1) { /// The selected package and their matching versions public SelectedDependency[string] versions; + + /// Whether this dub.selections.json can be inherited by nested projects + /// without local dub.selections.json + @Optional public bool inheritable; } else static assert(false, "This version is not supported"); @@ -191,6 +207,7 @@ (Selections!1 s) => s, (s) { assert(0); return Selections!(1).init; }, ); + assert(!s.inheritable); assert(s.versions.length == 5); assert(s.versions["simple"] == Dependency(Version("1.5.6"))); assert(s.versions["branch"] == Dependency(Version("~master"))); @@ -199,6 +216,23 @@ assert(s.versions["repository"] == Dependency(Repository("git+https://github.com/dlang/dub", "123456123456123456"))); } +// with optional `inheritable` Boolean +unittest +{ + import dub.internal.configy.Read : parseConfigString; + + immutable string content = `{ + "fileVersion": 1, + "inheritable": true, + "versions": { + "simple": "1.5.6", + } +}`; + + auto s = parseConfigString!SelectionsFile(content, "/dev/null"); + assert(s.inheritable); +} + // Test reading an unsupported version unittest { diff --git a/source/dub/test/base.d b/source/dub/test/base.d index f58ca7c..81e96fa 100644 --- a/source/dub/test/base.d +++ b/source/dub/test/base.d @@ -699,6 +699,7 @@ protected inout(FSEntry) lookup(NativePath path) inout return scope { auto relp = this.relativePath(path); + relp.normalize(); // try to get rid of `..` if (relp.empty) return this; auto segments = relp.bySegment; diff --git a/source/dub/test/selections_from_parent_dir.d b/source/dub/test/selections_from_parent_dir.d new file mode 100644 index 0000000..bf90c3f --- /dev/null +++ b/source/dub/test/selections_from_parent_dir.d @@ -0,0 +1,148 @@ +/******************************************************************************* + + Test inheritable flag of selections file + + Selections files can have an `inheritable` flag that is used to have + a central selections file, e.g. in the case of a monorepo. + +*******************************************************************************/ + +module dub.test.selections_from_parent_dir; + +version (unittest): + +import dub.test.base; +import std.string : replace; + +// dub.selections.json can be inherited from parent directories +unittest +{ + const pkg1Dir = TestDub.ProjectPath ~ "pkg1"; + const pkg2Dir = TestDub.ProjectPath ~ "pkg2"; + const path = TestDub.ProjectPath ~ "dub.selections.json"; + const dubSelectionsJsonContent = `{ + "fileVersion": 1, + "inheritable": true, + "versions": { + "pkg1": {"path":"pkg1"} + } +} +`; + + scope dub = new TestDub((scope FSEntry fs) { + fs.mkdir(pkg1Dir).writeFile(NativePath("dub.sdl"), `name "pkg1" +targetType "none"`); + fs.mkdir(pkg2Dir).writeFile(NativePath("dub.sdl"), `name "pkg2" +targetType "library" + +# don't specify a path, require inherited dub.selections.json to make it path-based (../pkg1) +dependency "pkg1" version="*"`); + + // important: dub.selections.json in *parent* directory + fs.writeFile(path, dubSelectionsJsonContent); + }, pkg2Dir.toNativeString()); // pkg2 is our root package + + dub.loadPackage(); + assert(dub.project.hasAllDependencies()); + // the relative path should have been adjusted (`pkg1` => `../pkg1`) + assert(dub.project.selections.getSelectedVersion(PackageName("pkg1")).path == NativePath("../pkg1")); + + // invoking `dub upgrade` for the pkg2 root package should generate a local dub.selections.json, + // leaving the inherited one untouched + dub.upgrade(UpgradeOptions.select); + const nestedPath = pkg2Dir ~ "dub.selections.json"; + assert(dub.fs.existsFile(nestedPath)); + assert(dub.fs.readFile(path) == dubSelectionsJsonContent, + "Inherited dub.selections.json modified after dub upgrade!"); + const nestedContent = dub.fs.readText(nestedPath); + assert(nestedContent == dubSelectionsJsonContent.replace(`{"path":"pkg1"}`, `{"path":"../pkg1"}`), + "Unexpected nestedContent:\n" ~ nestedContent); +} + +// a non-inheritable dub.selections.json breaks the inheritance chain +unittest +{ + const root = TestDub.ProjectPath ~ "root"; + const root_a = root ~ "a"; + const root_a_b = root_a ~ "b"; + + scope dub_ = new TestDub((scope FSEntry fs) { + // inheritable root/dub.selections.json + fs.mkdir(root).writeFile(NativePath("dub.selections.json"), `{ + "fileVersion": 1, + "inheritable": true, + "versions": { + "dub": "1.38.0" + } +} +`); + // non-inheritable root/a/dub.selections.json + fs.mkdir(root_a).writeFile(NativePath("dub.selections.json"), `{ + "fileVersion": 1, + "versions": { + "dub": "1.37.0" + } +} +`); + // We need packages for `loadPackage` + fs.mkdir(root_a_b); + fs.writeFile(root_a_b ~ `dub.json`, + `{"name":"ab","dependencies":{"dub":"~>1.0"}}`); + fs.writeFile(root_a ~ `dub.json`, + `{"name":"a","dependencies":{"dub":"~>1.0"}}`); + fs.writeFile(root ~ `dub.json`, + `{"name":"r","dependencies":{"dub":"~>1.0"}}`); + fs.writePackageFile(`dub`, `1.37.0`, `{"name":"dub","version":"1.37.0"}`); + fs.writePackageFile(`dub`, `1.38.0`, `{"name":"dub","version":"1.38.0"}`); + }); + + // no selections for root/a/b/ + { + auto dub = dub_.newTest(); + const result = dub.packageManager.readSelections(root_a_b); + assert(result.isNull()); + dub.loadPackage(root_a_b); + assert(!dub.project.hasAllDependencies()); + } + + // local selections for root/a/ + { + auto dub = dub_.newTest(); + const result = dub.packageManager.readSelections(root_a); + assert(!result.isNull()); + assert(result.get().absolutePath == root_a ~ "dub.selections.json"); + assert(!result.get().selectionsFile.inheritable); + dub.loadPackage(root_a); + assert(dub.project.hasAllDependencies()); + assert(dub.project.dependencies()[0].name == "dub"); + assert(dub.project.dependencies()[0].version_ == Version("1.37.0")); + } + + // local selections for root/ + { + auto dub = dub_.newTest(); + const result = dub.packageManager.readSelections(root); + assert(!result.isNull()); + assert(result.get().absolutePath == root ~ "dub.selections.json"); + assert(result.get().selectionsFile.inheritable); + dub.loadPackage(root); + assert(dub.project.hasAllDependencies()); + assert(dub.project.dependencies()[0].name == "dub"); + assert(dub.project.dependencies()[0].version_ == Version("1.38.0")); + } + + // after removing non-inheritable root/a/dub.selections.json: inherited root selections for root/a/b/ + { + auto dub = dub_.newTest((scope FSEntry fs) { + fs.removeFile(root_a ~ "dub.selections.json"); + }); + const result = dub.packageManager.readSelections(root_a_b); + assert(!result.isNull()); + assert(result.get().absolutePath == root ~ "dub.selections.json"); + assert(result.get().selectionsFile.inheritable); + dub.loadPackage(root_a_b); + assert(dub.project.hasAllDependencies()); + assert(dub.project.dependencies()[0].name == "dub"); + assert(dub.project.dependencies()[0].version_ == Version("1.38.0")); + } +}