aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGravatar Alex Lam S.L <alexlamsl@gmail.com> 2023-07-18 02:06:31 +0300
committerGravatar GitHub <noreply@github.com> 2023-07-17 16:06:31 -0700
commit9f031b36425580350f35fcd1082ad9445cab163e (patch)
treef9b8c892e592aacce0e27346f2ef228e9d061892
parentca3b7fa3c9179a764ce7a7bf6c52de3b3cd1f232 (diff)
downloadbun-9f031b36425580350f35fcd1082ad9445cab163e.tar.gz
bun-9f031b36425580350f35fcd1082ad9445cab163e.tar.zst
bun-9f031b36425580350f35fcd1082ad9445cab163e.zip
[install] handle duplicated workspace declarations gracefully (#3662)
fixes #3644
-rw-r--r--src/install/lockfile.zig31
-rw-r--r--test/cli/install/bun-install.test.ts172
2 files changed, 197 insertions, 6 deletions
diff --git a/src/install/lockfile.zig b/src/install/lockfile.zig
index 843381228..202cfddc0 100644
--- a/src/install/lockfile.zig
+++ b/src/install/lockfile.zig
@@ -2533,7 +2533,7 @@ pub const Package = extern struct {
dependency_version.value.workspace = path;
} else {
const workspace = dependency_version.value.workspace.slice(buf);
- const path = string_builder.append(
+ var path = string_builder.append(
String,
if (strings.eqlComptime(workspace, "*")) "*" else Path.relative(
FileSystem.instance.top_level_dir,
@@ -2547,13 +2547,32 @@ pub const Package = extern struct {
),
),
);
- dependency_version.value.workspace = path;
+ defer dependency_version.value.workspace = path;
var workspace_entry = try lockfile.workspace_paths.getOrPut(allocator, @truncate(u32, external_name.hash));
if (workspace_entry.found_existing) {
- log.addErrorFmt(&source, logger.Loc.Empty, allocator, "Workspace name \"{s}\" already exists", .{
- external_name.slice(buf),
- }) catch {};
- return error.InstallFailed;
+ const old_path = workspace_entry.value_ptr.*;
+
+ if (strings.eqlComptime(workspace, "*")) {
+ path = old_path;
+ return null;
+ } else if (strings.eqlComptime(old_path.slice(buf), "*")) brk: {
+ workspace_entry.value_ptr.* = path;
+ for (package_dependencies[0..dependencies_count]) |*package_dep| {
+ if (package_dep.name_hash == external_name.hash) {
+ if (package_dep.version.tag != .workspace) break :brk;
+ package_dep.version.value.workspace = path;
+ return null;
+ }
+ }
+ return error.InstallFailed;
+ } else if (strings.eql(old_path.slice(buf), path.slice(buf))) {
+ return null;
+ } else {
+ log.addErrorFmt(&source, logger.Loc.Empty, allocator, "Workspace name \"{s}\" already exists", .{
+ external_name.slice(buf),
+ }) catch {};
+ return error.InstallFailed;
+ }
}
workspace_entry.value_ptr.* = path;
},
diff --git a/test/cli/install/bun-install.test.ts b/test/cli/install/bun-install.test.ts
index f2483c8d2..f1d8e0d17 100644
--- a/test/cli/install/bun-install.test.ts
+++ b/test/cli/install/bun-install.test.ts
@@ -4589,3 +4589,175 @@ it("should handle trustedDependencies", async () => {
expect(await file(join(package_dir, "node_modules", "moo", "package.json")).text()).toEqual(moo_package);
await access(join(package_dir, "bun.lockb"));
});
+
+it("should handle `workspaces:*` and `workspace:*` gracefully", async () => {
+ await writeFile(
+ join(package_dir, "package.json"),
+ JSON.stringify({
+ name: "foo",
+ workspaces: ["*"],
+ dependencies: {
+ bar: "workspace:*",
+ },
+ }),
+ );
+ await mkdir(join(package_dir, "bar"));
+ const bar_package = JSON.stringify({
+ name: "bar",
+ version: "0.0.1",
+ });
+ await writeFile(join(package_dir, "bar", "package.json"), bar_package);
+ const { stdout, stderr, exited } = spawn({
+ cmd: [bunExe(), "install"],
+ cwd: package_dir,
+ stdout: null,
+ stdin: "pipe",
+ stderr: "pipe",
+ env,
+ });
+ expect(stderr).toBeDefined();
+ const err = await new Response(stderr).text();
+ expect(err).toContain("Saved lockfile");
+ expect(stdout).toBeDefined();
+ const out = await new Response(stdout).text();
+ expect(out.replace(/\s*\[[0-9\.]+m?s\]\s*$/, "").split(/\r?\n/)).toEqual([
+ " + bar@workspace:bar",
+ "",
+ " 1 packages installed",
+ ]);
+ expect(await exited).toBe(0);
+ expect(requested).toBe(0);
+ expect(await readdirSorted(join(package_dir, "node_modules"))).toEqual([".cache", "bar"]);
+ expect(await readlink(join(package_dir, "node_modules", "bar"))).toBe(join("..", "bar"));
+ expect(await file(join(package_dir, "node_modules", "bar", "package.json")).text()).toEqual(bar_package);
+ await access(join(package_dir, "bun.lockb"));
+});
+
+it("should handle `workspaces:bar` and `workspace:*` gracefully", async () => {
+ await writeFile(
+ join(package_dir, "package.json"),
+ JSON.stringify({
+ name: "foo",
+ workspaces: ["bar"],
+ dependencies: {
+ bar: "workspace:*",
+ },
+ }),
+ );
+ await mkdir(join(package_dir, "bar"));
+ const bar_package = JSON.stringify({
+ name: "bar",
+ version: "0.0.1",
+ });
+ await writeFile(join(package_dir, "bar", "package.json"), bar_package);
+ const { stdout, stderr, exited } = spawn({
+ cmd: [bunExe(), "install"],
+ cwd: package_dir,
+ stdout: null,
+ stdin: "pipe",
+ stderr: "pipe",
+ env,
+ });
+ expect(stderr).toBeDefined();
+ const err = await new Response(stderr).text();
+ expect(err).toContain("Saved lockfile");
+ expect(stdout).toBeDefined();
+ const out = await new Response(stdout).text();
+ expect(out.replace(/\s*\[[0-9\.]+m?s\]\s*$/, "").split(/\r?\n/)).toEqual([
+ " + bar@workspace:bar",
+ "",
+ " 1 packages installed",
+ ]);
+ expect(await exited).toBe(0);
+ expect(requested).toBe(0);
+ expect(await readdirSorted(join(package_dir, "node_modules"))).toEqual([".cache", "bar"]);
+ expect(await readlink(join(package_dir, "node_modules", "bar"))).toBe(join("..", "bar"));
+ expect(await file(join(package_dir, "node_modules", "bar", "package.json")).text()).toEqual(bar_package);
+ await access(join(package_dir, "bun.lockb"));
+});
+
+it("should handle `workspaces:*` and `workspace:bar` gracefully", async () => {
+ await writeFile(
+ join(package_dir, "package.json"),
+ JSON.stringify({
+ name: "foo",
+ workspaces: ["*"],
+ dependencies: {
+ bar: "workspace:bar",
+ },
+ }),
+ );
+ await mkdir(join(package_dir, "bar"));
+ const bar_package = JSON.stringify({
+ name: "bar",
+ version: "0.0.1",
+ });
+ await writeFile(join(package_dir, "bar", "package.json"), bar_package);
+ const { stdout, stderr, exited } = spawn({
+ cmd: [bunExe(), "install"],
+ cwd: package_dir,
+ stdout: null,
+ stdin: "pipe",
+ stderr: "pipe",
+ env,
+ });
+ expect(stderr).toBeDefined();
+ const err = await new Response(stderr).text();
+ expect(err).toContain("Saved lockfile");
+ expect(stdout).toBeDefined();
+ const out = await new Response(stdout).text();
+ expect(out.replace(/\s*\[[0-9\.]+m?s\]\s*$/, "").split(/\r?\n/)).toEqual([
+ " + bar@workspace:bar",
+ "",
+ " 1 packages installed",
+ ]);
+ expect(await exited).toBe(0);
+ expect(requested).toBe(0);
+ expect(await readdirSorted(join(package_dir, "node_modules"))).toEqual([".cache", "bar"]);
+ expect(await readlink(join(package_dir, "node_modules", "bar"))).toBe(join("..", "bar"));
+ expect(await file(join(package_dir, "node_modules", "bar", "package.json")).text()).toEqual(bar_package);
+ await access(join(package_dir, "bun.lockb"));
+});
+
+it("should handle `workspaces:bar` and `workspace:bar` gracefully", async () => {
+ await writeFile(
+ join(package_dir, "package.json"),
+ JSON.stringify({
+ name: "foo",
+ workspaces: ["bar"],
+ dependencies: {
+ bar: "workspace:bar",
+ },
+ }),
+ );
+ await mkdir(join(package_dir, "bar"));
+ const bar_package = JSON.stringify({
+ name: "bar",
+ version: "0.0.1",
+ });
+ await writeFile(join(package_dir, "bar", "package.json"), bar_package);
+ const { stdout, stderr, exited } = spawn({
+ cmd: [bunExe(), "install"],
+ cwd: package_dir,
+ stdout: null,
+ stdin: "pipe",
+ stderr: "pipe",
+ env,
+ });
+ expect(stderr).toBeDefined();
+ const err = await new Response(stderr).text();
+ expect(err).toContain("Saved lockfile");
+ expect(stdout).toBeDefined();
+ const out = await new Response(stdout).text();
+ expect(out.replace(/\s*\[[0-9\.]+m?s\]\s*$/, "").split(/\r?\n/)).toEqual([
+ " + bar@workspace:bar",
+ "",
+ " 1 packages installed",
+ ]);
+ expect(await exited).toBe(0);
+ expect(requested).toBe(0);
+ expect(await readdirSorted(join(package_dir, "node_modules"))).toEqual([".cache", "bar"]);
+ expect(await readlink(join(package_dir, "node_modules", "bar"))).toBe(join("..", "bar"));
+ expect(await file(join(package_dir, "node_modules", "bar", "package.json")).text()).toEqual(bar_package);
+ await access(join(package_dir, "bun.lockb"));
+});