-
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathproject.ts
220 lines (209 loc) · 7.21 KB
/
project.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
import { WriteStream } from 'fs';
import { resolve } from 'path';
import { getWorkspaceLocations } from '@metamask/action-utils';
import { WriteStreamLike, fileExists } from './fs.js';
import {
Package,
readMonorepoRootPackage,
readMonorepoWorkspacePackage,
updatePackageChangelog,
} from './package.js';
import { getRepositoryHttpsUrl, getTagNames, restoreFiles } from './repo.js';
import { SemVer } from './semver.js';
import { PackageManifestFieldNames } from './package-manifest.js';
import { ReleaseSpecification } from './release-specification.js';
/**
* The release version of the root package of a monorepo extracted from its
* version string.
*
* @property ordinaryNumber - The number assigned to the release if it
* introduces new changes that haven't appeared in any previous release; it will
* be 0 if there haven't been any releases yet.
* @property backportNumber - A backport release is a change ported from one
* ordinary release to a previous ordinary release. This, then, is the number
* which identifies this release relative to other backport releases under the
* same ordinary release, starting from 1; it will be 0 if there aren't any
* backport releases for the ordinary release yet.
*/
type ReleaseVersion = {
ordinaryNumber: number;
backportNumber: number;
};
/**
* Represents the entire codebase on which this tool is operating.
*
* @property directoryPath - The directory in which the project lives.
* @property repositoryUrl - The public URL of the Git repository where the
* codebase for the project lives.
* @property rootPackage - Information about the root package (assuming that the
* project is a monorepo).
* @property workspacePackages - Information about packages that are referenced
* via workspaces (assuming that the project is a monorepo).
*/
export type Project = {
directoryPath: string;
repositoryUrl: string;
rootPackage: Package;
workspacePackages: Record<string, Package>;
isMonorepo: boolean;
releaseVersion: ReleaseVersion;
};
/**
* Given a SemVer version object, interprets the "major" part of the version
* as the ordinary release number and the "minor" part as the backport release
* number in the context of the ordinary release.
*
* @param packageVersion - The version of the package.
* @returns An object containing the ordinary and backport numbers in the
* version.
*/
function examineReleaseVersion(packageVersion: SemVer): ReleaseVersion {
return {
ordinaryNumber: packageVersion.major,
backportNumber: packageVersion.minor,
};
}
/**
* Collects information about a monorepo — its root package as well as any
* packages within workspaces specified via the root `package.json`.
*
* @param projectDirectoryPath - The path to the project.
* @param args - Additional arguments.
* @param args.stderr - A stream that can be used to write to standard error.
* @param args.fetchRemote - Whether to synchronize local tags with remote.
* @returns An object that represents information about the project.
* @throws if the project does not contain a root `package.json` (polyrepo and
* monorepo) or if any of the workspaces specified in the root `package.json` do
* not have `package.json`s (monorepo only).
*/
export async function readProject(
projectDirectoryPath: string,
{ fetchRemote, stderr }: { fetchRemote?: boolean; stderr: WriteStreamLike },
): Promise<Project> {
const repositoryUrl = await getRepositoryHttpsUrl(projectDirectoryPath);
const tagNames = await getTagNames(projectDirectoryPath, fetchRemote);
const rootPackage = await readMonorepoRootPackage({
packageDirectoryPath: projectDirectoryPath,
projectDirectoryPath,
projectTagNames: tagNames,
});
const releaseVersion = examineReleaseVersion(
rootPackage.validatedManifest.version,
);
const workspaceDirectories = await getWorkspaceLocations(
rootPackage.validatedManifest[PackageManifestFieldNames.Workspaces],
projectDirectoryPath,
true,
);
const workspacePackages = (
await Promise.all(
workspaceDirectories.map(async (directory) => {
return await readMonorepoWorkspacePackage({
packageDirectoryPath: resolve(projectDirectoryPath, directory),
rootPackageName: rootPackage.validatedManifest.name,
rootPackageVersion: rootPackage.validatedManifest.version,
projectDirectoryPath,
projectTagNames: tagNames,
stderr,
});
}),
)
).reduce(
(obj, pkg) => {
return { ...obj, [pkg.validatedManifest.name]: pkg };
},
{} as Record<string, Package>,
);
const isMonorepo = Object.keys(workspacePackages).length > 0;
return {
directoryPath: projectDirectoryPath,
repositoryUrl,
rootPackage,
workspacePackages,
isMonorepo,
releaseVersion,
};
}
/**
* Updates the changelog files of all packages that have changes since latest release to include those changes.
*
* @param args - The arguments.
* @param args.project - The project.
* @param args.stderr - A stream that can be used to write to standard error.
* @param args.fetchRemote - Whether to synchronize local tags with remote.
* @returns The result of writing to the changelog.
*/
export async function updateChangelogsForChangedPackages({
project,
stderr,
fetchRemote,
}: {
project: Pick<
Project,
'directoryPath' | 'repositoryUrl' | 'workspacePackages'
>;
stderr: Pick<WriteStream, 'write'>;
fetchRemote?: boolean | undefined;
}): Promise<void> {
await Promise.all(
Object.values(project.workspacePackages)
.filter(
({ hasChangesSinceLatestRelease }) => hasChangesSinceLatestRelease,
)
.map((pkg) =>
updatePackageChangelog({
project,
package: pkg,
stderr,
fetchRemote,
}),
),
);
}
/**
* Restores the changelogs of unreleased packages which has changes since latest release.
*
* @param args - The arguments.
* @param args.project - The project.
* @param args.releaseSpecification - A parsed version of the release spec
* entered by the user.
* @param args.defaultBranch - The name of the default branch in the repository.
* @returns The result of writing to the changelog.
*/
export async function restoreChangelogsForSkippedPackages({
project: { directoryPath, workspacePackages },
releaseSpecification,
defaultBranch,
}: {
project: Pick<
Project,
'directoryPath' | 'repositoryUrl' | 'workspacePackages'
>;
releaseSpecification: ReleaseSpecification;
defaultBranch: string;
}): Promise<void> {
const existingSkippedPackageChangelogPaths = (
await Promise.all(
Object.entries(workspacePackages).map(async ([name, pkg]) => {
const changelogPath = pkg.changelogPath.replace(
`${directoryPath}/`,
'',
);
const shouldInclude =
pkg.hasChangesSinceLatestRelease &&
!releaseSpecification.packages[name] &&
(await fileExists(pkg.changelogPath));
return [changelogPath, shouldInclude] as const;
}),
)
)
.filter(([_, shouldInclude]) => shouldInclude)
.map(([changelogPath]) => changelogPath);
if (existingSkippedPackageChangelogPaths.length > 0) {
await restoreFiles(
directoryPath,
defaultBranch,
existingSkippedPackageChangelogPaths,
);
}
}