Skip to content

Commit b56869a

Browse files
authored
Allow dependecies to use the workspace: protocol and add "workspaceProtocol": "require" config option (#204)
1 parent 3c9641c commit b56869a

9 files changed

+140
-7
lines changed

.changeset/config.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
11
{
2-
"$schema": "https://unpkg.com/@changesets/config@0.2.1/schema.json",
2+
"$schema": "https://unpkg.com/@changesets/config@3.0.0/schema.json",
33
"changelog": [
44
"@changesets/changelog-github",
55
{ "repo": "Thinkmill/manypkg" }
66
],
77
"commit": false,
88
"linked": [],
9-
"access": "public"
9+
"access": "public",
10+
"baseBranch": "main"
1011
}

.changeset/forty-needles-doubt.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@manypkg/cli": patch
3+
---
4+
5+
Allow dependecies to use the `workspace:` protocol and support adding `"workspaceProtocol": "require"` to the `manypkg` config to require it.

README.md

+13-2
Original file line numberDiff line numberDiff line change
@@ -100,8 +100,7 @@ So that only a single version of an external package will be installed because h
100100

101101
### How it's fixed
102102

103-
The most commonly used range of the dependency is set as the range at every non-peer dependency place it is depended on.
104-
If for some reason, every range is used the same amount of times, they'll all be fixed to the highest version.
103+
The most commonly used range of the dependency is set as the range at every non-peer dependency place it is depended on. If for some reason, every range is used the same amount of times, they'll all be fixed to the highest version.
105104

106105
### Examples
107106

@@ -287,6 +286,18 @@ Having a `repository` field is helpful so there is a link to the source of a pac
287286

288287
This is fixed by setting the correct URL.
289288

289+
## `workspace:` protocol required
290+
291+
If `"workspaceProtocol": "require"` is set in the `manypkg` config in the root `package.json`, all dependencies on internal packages are required to use the `workspace:` protocol.
292+
293+
### Why it's a rule
294+
295+
If you want to enforce the usage of the `workspace:` protocol.
296+
297+
#### How it's fixed
298+
299+
Dependencies are changed to `workspace:^`. Anything else is also allowed after the `workspace:` though.
300+
290301
## License
291302

292303
Copyright (c) 2023 Thinkmill Labs Pty Ltd. Licensed under the MIT License.

packages/cli/src/checks/INTERNAL_MISMATCH.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -22,10 +22,11 @@ export default makeCheck<ErrorType>({
2222
for (let depName in deps) {
2323
let range = deps[depName];
2424
let dependencyWorkspace = allWorkspaces.get(depName);
25-
25+
2626
if (
2727
dependencyWorkspace !== undefined &&
2828
!range.startsWith("npm:") &&
29+
!range.startsWith("workspace:") &&
2930
!semver.satisfies(dependencyWorkspace.packageJson.version, range)
3031
) {
3132
errors.push({
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
import { makeCheck, NORMAL_DEPENDENCY_TYPES } from "./utils";
2+
import { Package } from "@manypkg/get-packages";
3+
4+
export type ErrorType = {
5+
type: "WORKSPACE_REQUIRED";
6+
workspace: Package;
7+
depType: typeof NORMAL_DEPENDENCY_TYPES[number];
8+
depName: string;
9+
};
10+
11+
export default makeCheck<ErrorType>({
12+
validate: (workspace, allWorkspaces, root, opts) => {
13+
if (opts.workspaceProtocol !== "require") return [];
14+
let errors: ErrorType[] = [];
15+
for (let depType of NORMAL_DEPENDENCY_TYPES) {
16+
let deps = workspace.packageJson[depType];
17+
if (deps) {
18+
for (let depName in deps) {
19+
if (
20+
allWorkspaces.has(depName) &&
21+
!deps[depName].startsWith("workspace:")
22+
) {
23+
errors.push({
24+
type: "WORKSPACE_REQUIRED",
25+
workspace,
26+
depName,
27+
depType,
28+
});
29+
}
30+
}
31+
}
32+
}
33+
34+
return errors;
35+
},
36+
fix: (error) => {
37+
let deps = error.workspace.packageJson[error.depType];
38+
if (deps && deps[error.depName]) {
39+
deps[error.depName] = "workspace:^";
40+
}
41+
return { requiresInstall: true };
42+
},
43+
print: (error) =>
44+
`${error.workspace.packageJson.name} has a dependency on ${error.depName} without using the workspace: protocol but this project requires using the workspace: protocol, please change it to workspace:^ or etc.`,
45+
type: "all",
46+
});

packages/cli/src/checks/__tests__/INTERNAL_MISMATCH.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,16 @@ describe("internal mismatch", () => {
1414
let errors = makeCheck.validate(dependsOnOne, ws, rootWorkspace, {});
1515
expect(errors.length).toEqual(0);
1616
});
17+
it("should allow workspace: protocol", () => {
18+
let ws = getWS();
19+
let dependsOnOne = getFakeWS("depends-on-one");
20+
dependsOnOne.packageJson.dependencies = {
21+
"pkg-1": "workspace:^",
22+
};
23+
ws.set("depends-on-one", dependsOnOne);
24+
let errors = makeCheck.validate(dependsOnOne, ws, rootWorkspace, {});
25+
expect(errors.length).toEqual(0);
26+
});
1727
it("should error if internal version is not compatible", () => {
1828
let ws = getWS();
1929
let dependsOnOne = getFakeWS("depends-on-one");
@@ -100,5 +110,4 @@ describe("internal mismatch", () => {
100110
expect(errors.length).toEqual(0);
101111
}
102112
);
103-
104113
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import { getWS, getFakeWS, getRootWS } from "../../test-helpers";
2+
import makeCheck from "../WORKSPACE_REQUIRED";
3+
let rootWorkspace = getRootWS();
4+
5+
test("should not error if not using workspaceProtocol: require", () => {
6+
let ws = getWS();
7+
let dependsOnOne = getFakeWS("depends-on-one");
8+
dependsOnOne.packageJson.dependencies = {
9+
"pkg-1": "^1.0.0",
10+
};
11+
ws.set("depends-on-one", dependsOnOne);
12+
let errors = makeCheck.validate(dependsOnOne, ws, rootWorkspace, {});
13+
expect(errors.length).toEqual(0);
14+
});
15+
16+
test("should error if using workspaceProtocol: require", () => {
17+
let ws = getWS();
18+
let dependsOnOne = getFakeWS("depends-on-one");
19+
dependsOnOne.packageJson.dependencies = {
20+
"pkg-1": "^1.0.0",
21+
};
22+
ws.set("depends-on-one", dependsOnOne);
23+
let errors = makeCheck.validate(dependsOnOne, ws, rootWorkspace, {
24+
workspaceProtocol: "require",
25+
});
26+
expect(errors).toEqual([
27+
{
28+
type: "WORKSPACE_REQUIRED",
29+
workspace: dependsOnOne,
30+
depName: "pkg-1",
31+
depType: "dependencies",
32+
},
33+
]);
34+
});
35+
36+
test("should fix if using workspaceProtocol: require", () => {
37+
let ws = getWS();
38+
let dependsOnOne = getFakeWS("depends-on-one");
39+
dependsOnOne.packageJson.dependencies = {
40+
"pkg-1": "^1.0.0",
41+
};
42+
ws.set("depends-on-one", dependsOnOne);
43+
const errors = makeCheck.validate(dependsOnOne, ws, rootWorkspace, {
44+
workspaceProtocol: "require",
45+
});
46+
expect(errors).toHaveLength(1);
47+
const result = makeCheck.fix(errors[0], {
48+
workspaceProtocol: "require",
49+
});
50+
expect(dependsOnOne.packageJson.dependencies).toEqual({
51+
"pkg-1": "workspace:^",
52+
});
53+
expect(result).toEqual({ requiresInstall: true });
54+
});

packages/cli/src/checks/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import MULTIPLE_DEPENDENCY_TYPES from "./MULTIPLE_DEPENDENCY_TYPES";
66
import ROOT_HAS_DEV_DEPENDENCIES from "./ROOT_HAS_DEV_DEPENDENCIES";
77
import UNSORTED_DEPENDENCIES from "./UNSORTED_DEPENDENCIES";
88
import INCORRECT_REPOSITORY_FIELD from "./INCORRECT_REPOSITORY_FIELD";
9+
import WORKSPACE_REQUIRED from "./WORKSPACE_REQUIRED";
910

1011
export let checks = {
1112
EXTERNAL_MISMATCH,
@@ -16,4 +17,5 @@ export let checks = {
1617
ROOT_HAS_DEV_DEPENDENCIES,
1718
UNSORTED_DEPENDENCIES,
1819
INCORRECT_REPOSITORY_FIELD,
20+
WORKSPACE_REQUIRED,
1921
};

packages/cli/src/checks/utils.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,11 @@ export const DEPENDENCY_TYPES = [
1515
"peerDependencies",
1616
] as const;
1717

18-
export type Options = { defaultBranch?: string; ignoredRules?: string[] };
18+
export type Options = {
19+
defaultBranch?: string;
20+
ignoredRules?: string[];
21+
workspaceProtocol?: "allow" | "require";
22+
};
1923

2024
type RootCheck<ErrorType> = {
2125
type: "root";

0 commit comments

Comments
 (0)