relion 0.3.0 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.js +1 -64
- package/dist/index.d.ts +2 -3
- package/dist/index.js +17 -642
- package/package.json +5 -5
package/dist/cli.js
CHANGED
|
@@ -1,64 +1 @@
|
|
|
1
|
-
import
|
|
2
|
-
import { pathToFileURL } from "node:url";
|
|
3
|
-
import relion from "./index.js";
|
|
4
|
-
import { cli } from "cleye";
|
|
5
|
-
|
|
6
|
-
//#region src/cli.ts
|
|
7
|
-
let config;
|
|
8
|
-
try {
|
|
9
|
-
const configFileURL = pathToFileURL(resolve(process.cwd(), "relion.config.ts")).href;
|
|
10
|
-
config = (await import(configFileURL)).default;
|
|
11
|
-
} catch (error) {
|
|
12
|
-
throw new Error(`Error loading config: ${error.message}`);
|
|
13
|
-
}
|
|
14
|
-
const argv = cli({
|
|
15
|
-
name: "relion",
|
|
16
|
-
flags: {
|
|
17
|
-
bump: {
|
|
18
|
-
alias: "b",
|
|
19
|
-
type: Boolean,
|
|
20
|
-
description: "Bump the version",
|
|
21
|
-
default: false
|
|
22
|
-
},
|
|
23
|
-
changelog: {
|
|
24
|
-
alias: "l",
|
|
25
|
-
type: Boolean,
|
|
26
|
-
description: "Generate a changelog",
|
|
27
|
-
default: false
|
|
28
|
-
},
|
|
29
|
-
commit: {
|
|
30
|
-
alias: "c",
|
|
31
|
-
type: Boolean,
|
|
32
|
-
description: "Create a commit",
|
|
33
|
-
default: false
|
|
34
|
-
},
|
|
35
|
-
tag: {
|
|
36
|
-
alias: "t",
|
|
37
|
-
type: Boolean,
|
|
38
|
-
description: "Create a tag",
|
|
39
|
-
default: false
|
|
40
|
-
},
|
|
41
|
-
profile: {
|
|
42
|
-
alias: "p",
|
|
43
|
-
type: String,
|
|
44
|
-
description: "Use config profile"
|
|
45
|
-
},
|
|
46
|
-
dryRun: {
|
|
47
|
-
alias: "d",
|
|
48
|
-
type: Boolean,
|
|
49
|
-
description: "Run without making any changes",
|
|
50
|
-
default: false
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
});
|
|
54
|
-
if (!argv.flags.bump) config.bump = false;
|
|
55
|
-
if (!argv.flags.changelog) config.changelog = false;
|
|
56
|
-
if (!argv.flags.commit) config.commit = false;
|
|
57
|
-
if (!argv.flags.tag) config.tag = false;
|
|
58
|
-
if (argv.flags.profile) config.profile = argv.flags.profile;
|
|
59
|
-
if (argv.flags.dryRun) config.dryRun = argv.flags.dryRun;
|
|
60
|
-
await relion(config);
|
|
61
|
-
|
|
62
|
-
//#endregion
|
|
63
|
-
export { };
|
|
64
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpLmpzIiwibmFtZXMiOlsiY29uZmlnOiBVc2VyQ29uZmlnIl0sInNvdXJjZXMiOlsiLi4vc3JjL2NsaS50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByZXNvbHZlIH0gZnJvbSAnbm9kZTpwYXRoJ1xuaW1wb3J0IHsgcGF0aFRvRmlsZVVSTCB9IGZyb20gJ25vZGU6dXJsJ1xuaW1wb3J0IHJlbGlvbiBmcm9tICcuL2luZGV4LmpzJ1xuaW1wb3J0IHR5cGUgeyBVc2VyQ29uZmlnIH0gZnJvbSAnQC90eXBlcydcbmltcG9ydCB7IGNsaSB9IGZyb20gJ2NsZXllJ1xuXG5sZXQgY29uZmlnOiBVc2VyQ29uZmlnXG5cbnRyeSB7XG5cdGNvbnN0IGNvbmZpZ0ZpbGVVUkwgPSBwYXRoVG9GaWxlVVJMKHJlc29sdmUocHJvY2Vzcy5jd2QoKSwgJ3JlbGlvbi5jb25maWcudHMnKSkuaHJlZlxuXHRjb25maWcgPSAoKGF3YWl0IGltcG9ydChjb25maWdGaWxlVVJMKSkgYXMgeyBkZWZhdWx0OiBVc2VyQ29uZmlnIH0pLmRlZmF1bHRcbn0gY2F0Y2ggKGVycm9yKSB7XG5cdHRocm93IG5ldyBFcnJvcihgRXJyb3IgbG9hZGluZyBjb25maWc6ICR7KGVycm9yIGFzIEVycm9yKS5tZXNzYWdlfWApXG59XG5cbmNvbnN0IGFyZ3YgPSBjbGkoe1xuXHRuYW1lOiAncmVsaW9uJyxcblx0ZmxhZ3M6IHtcblx0XHRidW1wOiB7XG5cdFx0XHRhbGlhczogJ2InLFxuXHRcdFx0dHlwZTogQm9vbGVhbixcblx0XHRcdGRlc2NyaXB0aW9uOiAnQnVtcCB0aGUgdmVyc2lvbicsXG5cdFx0XHRkZWZhdWx0OiBmYWxzZSxcblx0XHR9LFxuXHRcdGNoYW5nZWxvZzoge1xuXHRcdFx0YWxpYXM6ICdsJyxcblx0XHRcdHR5cGU6IEJvb2xlYW4sXG5cdFx0XHRkZXNjcmlwdGlvbjogJ0dlbmVyYXRlIGEgY2hhbmdlbG9nJyxcblx0XHRcdGRlZmF1bHQ6IGZhbHNlLFxuXHRcdH0sXG5cdFx0Y29tbWl0OiB7XG5cdFx0XHRhbGlhczogJ2MnLFxuXHRcdFx0dHlwZTogQm9vbGVhbixcblx0XHRcdGRlc2NyaXB0aW9uOiAnQ3JlYXRlIGEgY29tbWl0Jyxcblx0XHRcdGRlZmF1bHQ6IGZhbHNlLFxuXHRcdH0sXG5cdFx0dGFnOiB7XG5cdFx0XHRhbGlhczogJ3QnLFxuXHRcdFx0dHlwZTogQm9vbGVhbixcblx0XHRcdGRlc2NyaXB0aW9uOiAnQ3JlYXRlIGEgdGFnJyxcblx0XHRcdGRlZmF1bHQ6IGZhbHNlLFxuXHRcdH0sXG5cdFx0cHJvZmlsZToge1xuXHRcdFx0YWxpYXM6ICdwJyxcblx0XHRcdHR5cGU6IFN0cmluZyxcblx0XHRcdGRlc2NyaXB0aW9uOiAnVXNlIGNvbmZpZyBwcm9maWxlJyxcblx0XHR9LFxuXHRcdGRyeVJ1bjoge1xuXHRcdFx0YWxpYXM6ICdkJyxcblx0XHRcdHR5cGU6IEJvb2xlYW4sXG5cdFx0XHRkZXNjcmlwdGlvbjogJ1J1biB3aXRob3V0IG1ha2luZyBhbnkgY2hhbmdlcycsXG5cdFx0XHRkZWZhdWx0OiBmYWxzZSxcblx0XHR9LFxuXHR9LFxufSlcblxuaWYgKCFhcmd2LmZsYWdzLmJ1bXApIGNvbmZpZy5idW1wID0gZmFsc2VcbmlmICghYXJndi5mbGFncy5jaGFuZ2Vsb2cpIGNvbmZpZy5jaGFuZ2Vsb2cgPSBmYWxzZVxuaWYgKCFhcmd2LmZsYWdzLmNvbW1pdCkgY29uZmlnLmNvbW1pdCA9IGZhbHNlXG5pZiAoIWFyZ3YuZmxhZ3MudGFnKSBjb25maWcudGFnID0gZmFsc2VcbmlmIChhcmd2LmZsYWdzLnByb2ZpbGUpIGNvbmZpZy5wcm9maWxlID0gYXJndi5mbGFncy5wcm9maWxlXG5pZiAoYXJndi5mbGFncy5kcnlSdW4pIGNvbmZpZy5kcnlSdW4gPSBhcmd2LmZsYWdzLmRyeVJ1blxuXG5hd2FpdCByZWxpb24oY29uZmlnKSJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBTUEsSUFBSUE7QUFFSixJQUFJO0NBQ0gsTUFBTSxnQkFBZ0IsY0FBYyxRQUFRLFFBQVEsT0FBTyxxQkFBcUI7QUFDaEYsV0FBVyxNQUFNLE9BQU8sZ0JBQTRDO1NBQzVELE9BQU87QUFDZixPQUFNLElBQUksTUFBTSx5QkFBMEIsTUFBZ0I7O0FBRzNELE1BQU0sT0FBTyxJQUFJO0NBQ2hCLE1BQU07Q0FDTixPQUFPO0VBQ04sTUFBTTtHQUNMLE9BQU87R0FDUCxNQUFNO0dBQ04sYUFBYTtHQUNiLFNBQVM7O0VBRVYsV0FBVztHQUNWLE9BQU87R0FDUCxNQUFNO0dBQ04sYUFBYTtHQUNiLFNBQVM7O0VBRVYsUUFBUTtHQUNQLE9BQU87R0FDUCxNQUFNO0dBQ04sYUFBYTtHQUNiLFNBQVM7O0VBRVYsS0FBSztHQUNKLE9BQU87R0FDUCxNQUFNO0dBQ04sYUFBYTtHQUNiLFNBQVM7O0VBRVYsU0FBUztHQUNSLE9BQU87R0FDUCxNQUFNO0dBQ04sYUFBYTs7RUFFZCxRQUFRO0dBQ1AsT0FBTztHQUNQLE1BQU07R0FDTixhQUFhO0dBQ2IsU0FBUzs7OztBQUtaLElBQUksQ0FBQyxLQUFLLE1BQU0sS0FBTSxRQUFPLE9BQU87QUFDcEMsSUFBSSxDQUFDLEtBQUssTUFBTSxVQUFXLFFBQU8sWUFBWTtBQUM5QyxJQUFJLENBQUMsS0FBSyxNQUFNLE9BQVEsUUFBTyxTQUFTO0FBQ3hDLElBQUksQ0FBQyxLQUFLLE1BQU0sSUFBSyxRQUFPLE1BQU07QUFDbEMsSUFBSSxLQUFLLE1BQU0sUUFBUyxRQUFPLFVBQVUsS0FBSyxNQUFNO0FBQ3BELElBQUksS0FBSyxNQUFNLE9BQVEsUUFBTyxTQUFTLEtBQUssTUFBTTtBQUVsRCxNQUFNLE9BQU8ifQ==
|
|
1
|
+
import{resolve as e}from"node:path";import{pathToFileURL as t}from"node:url";import n from"./index.js";import{cli as r}from"cleye";let i;try{let n=t(e(process.cwd(),`relion.config.ts`)).href;i=(await import(n)).default}catch(e){throw Error(`Error loading config: ${e.message}`)}const a=r({name:`relion`,flags:{bump:{alias:`b`,type:Boolean,description:`Bump the version`,default:!1},changelog:{alias:`l`,type:Boolean,description:`Generate a changelog`,default:!1},commit:{alias:`c`,type:Boolean,description:`Create a commit`,default:!1},tag:{alias:`t`,type:Boolean,description:`Create a tag`,default:!1},profile:{alias:`p`,type:String,description:`Use config profile`},dryRun:{alias:`d`,type:Boolean,description:`Run without making any changes`,default:!1},latest:{alias:`L`,type:Boolean,description:`Use the latest-release commit range in changelog`,default:!1}}});a.flags.bump||(i.bump=!1),a.flags.changelog||(i.changelog=!1),a.flags.commit||(i.commit=!1),a.flags.tag||(i.tag=!1),a.flags.profile&&(i.profile=a.flags.profile),a.flags.dryRun&&(i.dryRun=a.flags.dryRun),a.flags.latest&&i.changelog&&(i.changelog===!0&&(i.changelog={}),i.changelog.commitRange=`latest-release`),await n(i);export{};
|
package/dist/index.d.ts
CHANGED
|
@@ -101,7 +101,7 @@ interface ResolvedContext extends Required<Context> {
|
|
|
101
101
|
commits: Commit[];
|
|
102
102
|
releases: ReleaseWithGroupedCommits[] | null;
|
|
103
103
|
}
|
|
104
|
-
type CommitRange = 'all' | 'unreleased' | {
|
|
104
|
+
type CommitRange = 'all' | 'unreleased' | 'latest-release' | {
|
|
105
105
|
versionTag: string;
|
|
106
106
|
} | {
|
|
107
107
|
from: 'firstCommit' | (string & {});
|
|
@@ -227,5 +227,4 @@ interface DefaultChangelogSections extends ChangelogSectionsMap {
|
|
|
227
227
|
declare function relion(userConfig: UserConfig): Promise<void>;
|
|
228
228
|
declare const defineConfig: (config: UserConfig) => UserConfig;
|
|
229
229
|
//#endregion
|
|
230
|
-
export { BumpFiles, ChangelogOptions, ChangelogSectionDefinition, ChangelogSectionsMap, Commit, CommitMessage, CommitOptions, CommitRange, CommitsParser, CompleteChangelogOptions, CompleteCommitOptions, CompleteCommitsParser, CompleteTagOptions, Context, ContextualConfig, Contributor, DefaultChangelogSections, DefaultVersionedFile, FalseOrComplete, GithubUserInfo, GpgSig, GpgSigCode, MergedConfig, RawCommit, RawReference, RefLabel, Reference, ReleaseContext, ReleaseType, ReleaseWithFlatCommits, ReleaseWithGroupedCommits, RepoInfo, ResolvedChangelogOptions, ResolvedChangelogSection, ResolvedConfig, ResolvedContext, TagOptions, TransformedConfig, UserConfig, VersionedFile, relion as default, defineConfig };
|
|
231
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.d.ts","names":[],"sources":["../src/types/config.d.ts","../src/enums.d.ts","../src/types/commit.d.ts","../src/types/changelog.d.ts","../src/relion.d.ts"],"sourcesContent":["import type { Commit, RawCommit, ChangelogSectionDefinition, ReleaseWithGroupedCommits } from '@/types'\nimport type { HelperDeclareSpec } from 'handlebars'\n\nexport type FalseOrComplete<T> = false | Required<T>\n\nexport interface UserConfig {\n\tbump?: boolean | BumpFiles\n\tchangelog?: boolean | ChangelogOptions\n\tcommit?: boolean | CommitOptions\n\ttag?: boolean | TagOptions\n\tnewTagFormat?: string\n\tversionSourceFile?: string | VersionedFile\n\treleaseVersion?: string\n\treleaseType?: ReleaseType\n\tzeroMajorBreakingIsMinor?: boolean\n\tcontext?: Context\n\tcommitsParser?: CommitsParser\n\tprevReleaseTagPattern?: RegExp\n\tdryRun?: boolean\n\tprofile?: string\n\t[profile: `_${string}`]: UserConfig | undefined\n}\n\ntype OptionalKeys = 'releaseVersion' | 'releaseType' | 'context' | 'profile'\nexport interface MergedConfig extends Omit<Required<UserConfig>, OptionalKeys>, Pick<UserConfig, OptionalKeys> {\n\tchangelog: FalseOrComplete<ChangelogOptions>\n\tcommit: FalseOrComplete<CommitOptions>\n\ttag: FalseOrComplete<TagOptions>\n\tcommitsParser: CompleteCommitsParser\n}\nexport interface TransformedConfig extends Omit<MergedConfig, 'changelog'> {\n\tversionSourceFile: VersionedFile\n\tbump: FalseOrComplete<VersionedFile[]>\n\tchangelog: false | ResolvedChangelogOptions\n}\n\nexport interface ContextualConfig extends TransformedConfig {\n\tcontext: ResolvedContext\n}\nexport type ResolvedConfig = ContextualConfig\n\nexport type BumpFiles = (string | VersionedFile)[]\n\nexport interface ChangelogOptions {\n\tstdout?: boolean\n\toutputFile?: string\n\tcommitRange?: CommitRange\n\tsections?: ChangelogSectionDefinition[]\n\theader?: string\n\tprevReleaseHeaderPattern?: RegExp\n\thelpers?: HelperDeclareSpec\n\tpartials?: Record<string, string>\n}\nexport type CompleteChangelogOptions = Required<ChangelogOptions>\nexport interface ResolvedChangelogOptions extends Omit<CompleteChangelogOptions, 'partials'> {\n\tcompiledPartials: Record<string, HandlebarsTemplateDelegate>\n}\n\nexport interface CommitOptions {\n\tmessage?: string\n\tsignOff?: boolean\n\tgpgSign?: boolean\n\tstageAll?: boolean\n\textraArgs?: string\n}\nexport type CompleteCommitOptions = Required<CommitOptions>\n\nexport interface TagOptions {\n\tname?: string\n\tmessage?: string\n\tgpgSign?: boolean\n\tforce?: boolean\n\textraArgs?: string\n}\nexport type CompleteTagOptions = Required<TagOptions>\n\nexport interface CommitsParser {\n\tbreakingChangesPattern?: RegExp\n\tbreakingChangeListPattern?: RegExp\n\theaderPattern?: RegExp\n\ttagPattern?: RegExp\n\tcoAuthorPattern?: RegExp\n\tghEmailPattern?: RegExp\n\trefPattern?: RegExp\n\trefActionPattern?: RegExp\n\trefLabelPattern?: RegExp\n\tremoteUrlPattern?: RegExp\n\tsignerPattern?: RegExp\n\tdateSource?: 'authorDate' | 'committerDate'\n\tdateFormat?: string\n}\nexport type CompleteCommitsParser = Required<CommitsParser>\n\nexport interface RepoInfo {\n\thost?: string\n\towner?: string\n\tname?: string\n\thomepage?: string\n}\nexport interface Context {\n\tcommits?: Commit[] | RawCommit[]\n\tcurrentVersion?: string\n\tcurrentTag?: string\n\tnewVersion?: string\n\tnewTag?: string\n\trepo?: RepoInfo\n\t[key: string]: unknown\n}\nexport interface ResolvedContext extends Required<Context> {\n\tcommits: Commit[]\n\treleases: ReleaseWithGroupedCommits[] | null\n}\n\nexport type CommitRange =\n\t| 'all'\n\t| 'unreleased'\n\t| { versionTag: string }\n\t| { from: 'firstCommit' | (string & {}) }\n\t| { to: 'HEAD' | (string & {}) }\n\t| { from: 'firstCommit' | (string & {}), to: 'HEAD' | (string & {}) }\n\nexport type ReleaseType = 'major' | 'minor' | 'patch'\n\nexport interface VersionedFile {\n\tfilePath: string\n\tversionPattern: RegExp\n}\nexport interface DefaultVersionedFile extends Omit<VersionedFile, 'filePath'> {\n\tfilePathRegex: RegExp\n}","export declare enum GpgSigLabel {\n    G = \"valid\",\n    B = \"bad\",\n    U = \"valid, unknown validity\",\n    X = \"valid, expired\",\n    Y = \"valid, made by expired key\",\n    R = \"valid, made by revoked key\",\n    E = \"cannot check (missing key)\",\n    N = \"no signature\"\n}\nexport declare enum RefType {\n    issue = \"issue\",\n    pr = \"PR\"\n}\n","import type { GpgSigLabel, RefType } from '@/enums'\n\nexport type RawCommit = CommitMessageString | {\n\thash?: string\n\tmessage: string\n\ttagRefs?: string\n\tauthorName?: string\n\tauthorEmail?: string\n\tauthorTs?: string\n\tcommitterName?: string\n\tcommitterEmail?: string\n\tcommitterTs?: string\n\tgpgSigCode?: GpgSigCode\n\tgpgSigKeyId?: string\n}\n\nexport interface Commit extends CommitMessage {\n\thash?: string\n\ttags?: string[]\n\tauthors?: Contributor[]\n\tcommitter?: Contributor\n\trefs?: Reference[]\n\tgpgSig?: GpgSig\n\tdate?: string\n}\n\nexport interface CommitMessage {\n\ttype: string\n\tscope?: string\n\tsubject: string\n\tbody?: string\n\tbreakingChanges?: string | string[]\n\tfooter?: string\n}\ntype CommitMessageString = string\n\nexport interface Reference {\n\taction: string\n\ttype?: RefType\n\towner?: string\n\trepo?: string\n\tnumber: string\n}\n\nexport type RefLabel = Pick<Reference, 'owner' | 'repo' | 'number'>\n\nexport interface Contributor {\n\tname: string\n\temail: string\n\thasSignedOff?: boolean\n\tghLogin?: string\n\tghUrl?: string\n}\n\nexport type GithubUserInfo = Pick<Contributor, 'ghLogin' | 'ghUrl'>\n\nexport interface GpgSig {\n\tcode: GpgSigCode\n\tlabel: GpgSigLabel\n\tkeyId?: string\n}\n\nexport type GpgSigCode = 'G' | 'B' | 'U' | 'X' | 'Y' | 'R' | 'E' | 'N'\n\nexport interface RawReference {\n\taction: string\n\tlabels: string\n}","import type { Commit, ResolvedContext } from '@/types'\n\nexport interface ChangelogSectionDefinition {\n\ttitle: string\n\tcommitType: 'breaking' | '*' | (string & {}) | string[]\n\tfilter?: (commit: Commit) => boolean\n}\n\nexport interface ResolvedChangelogSection {\n\ttitle: string\n\tcommits: Commit[]\n}\n\nexport interface ReleaseWithFlatCommits {\n\ttag: string\n\tversion?: string\n\tdate?: string\n\tcommits: Commit[]\n}\n\nexport interface ReleaseWithGroupedCommits extends Omit<ReleaseWithFlatCommits, 'commits'> {\n\tcommitGroups: ResolvedChangelogSection[]\n}\n\nexport interface ReleaseContext extends ReleaseWithGroupedCommits, ResolvedContext {\n\tprevTag?: string\n\tprevVersion?: string\n}\n\nexport type ChangelogSectionsMap = Record<string, ChangelogSectionDefinition>\n\nexport interface DefaultChangelogSections extends ChangelogSectionsMap {\n\t[Symbol.iterator](): Iterator<ChangelogSectionDefinition>\n}","import type { UserConfig } from '@/types';\nexport default function relion(userConfig: UserConfig): Promise<void>;\nexport declare const defineConfig: (config: UserConfig) => UserConfig;\n"],"mappings":";;;;;;;;AAGA,IAAW,aAAa;CAAC;OAAU;OAAgB;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;;;AAEnD,IAAW,eAAe;CAAC;OAAC;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;;AAC5B,IAAW,oBAAe;CAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;;AAC1B,IAAW,mBAAmB;CAAC;OAAO;OAAA;;AACtC,IAAW,iBAAiB,CAAC,UAAG;AAChC,IAAW,YAAY,CAAC,UAAE;AAC1B,IAAW,mBAAU;CAAA;OAAA;OAAA;OAAA;OAAA;OAAA;;AACrB,IAAW,2BAA2B;CAAC;OAAG;OAAA;;AAC1C,IAAW,2BAAY;CAAA;OAAA;OAAA;OAAA;OAAA;;AACvB,IAAW,gBAAc,CAAA;AACzB,IAAW,wBAAuB;CAAA;OAAA;OAAA;;AAClC,IAAW,aAAM,CAAA;AACjB,IAAW,qBAAkB;CAAA;OAAA;OAAA;;AAC7B,IAAW,gBAAgB;CAAC;OAAE;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;OAAA;;AAC9B,IAAW,wBAAK;CAAA;OAAA;OAAA;;AAChB,IAAW,WAAK,CAAA;AAChB,IAAW,UAAU;CAAC;OAAU;OAAc;OAAC;;AAC/C,IAAA,kBAAA;CAAA;OAAA;OAAA;OAAA;OAAA;;;AAEA,IAAW,cAAc,CAAC;AAC1B,IAAW,gBAAgB,CAAC,UAAU;AACtC,IAAW,uBAAuB;CAAC;OAAS;OAAA;OAAA;;;;;ACzB5C,IAAW,cAAc,CAAC;AAC1B,IAAW,UAAI,CAAA;;;;;;;;;ACCf,IAAW,SAAS;CAAC;OAAU;OAAe;OAAA;OAAA;OAAA;;AAC9C,IAAW,gBAAE,CAAA;AACb,IAAI,sBAAW,CAAA;AACf,IAAW,YAAK,CAAA,UAAA;AAChB,IAAW,WAAQ;CAAA;OAAA;OAAA;;AACnB,IAAW,cAAS,CAAA;AACpB,IAAW,iBAAM;CAAA;OAAA;OAAA;;AACjB,IAAW,SAAS;CAAC;OAAC;OAAA;;AACtB,IAAW,aAAY,CAAA;AACvB,IAAW,eAAS,CAAA;;;;;ACTpB,IAAW,2BAA2B,CAAC,SAAK;AAC5C,IAAW,yBAAE,CAAA,SAAA;AACb,IAAW,4BAA4B;CAAC;OAAS;OAAM;OAAA;;AACvD,IAAW,iBAAiB;CAAC;OAAO;OAAA;;AACpC,IAAA,uBAAA;CAAA;OAAA;OAAA;;;;;;;;;;;ACLA,IAAI,SAAS;CAAC;OAAS;OAAkB;;AACzC,IAAW,eAAe;CAAC;OAAS;OAAkB"}
|
|
230
|
+
export { BumpFiles, ChangelogOptions, ChangelogSectionDefinition, ChangelogSectionsMap, Commit, CommitMessage, CommitOptions, CommitRange, CommitsParser, CompleteChangelogOptions, CompleteCommitOptions, CompleteCommitsParser, CompleteTagOptions, Context, ContextualConfig, Contributor, DefaultChangelogSections, DefaultVersionedFile, FalseOrComplete, GithubUserInfo, GpgSig, GpgSigCode, MergedConfig, RawCommit, RawReference, RefLabel, Reference, ReleaseContext, ReleaseType, ReleaseWithFlatCommits, ReleaseWithGroupedCommits, RepoInfo, ResolvedChangelogOptions, ResolvedChangelogSection, ResolvedConfig, ResolvedContext, TagOptions, TransformedConfig, UserConfig, VersionedFile, relion as default, defineConfig };
|
package/dist/index.js
CHANGED
|
@@ -1,652 +1,27 @@
|
|
|
1
|
-
import
|
|
2
|
-
import Handlebars from "handlebars";
|
|
3
|
-
import semver from "semver";
|
|
4
|
-
import { execSync } from "node:child_process";
|
|
1
|
+
import{existsSync as e,readFileSync as t,writeFileSync as n}from"node:fs";import r from"handlebars";import i from"semver";import{execSync as a}from"node:child_process";const o=e=>{if(!e.bump)return;let r=e.bump,i=e.context.newVersion;r.forEach(r=>{let a=t(r.filePath,`utf8`),o=a.replace(r.versionPattern,`$1${i}$3`);e.dryRun||n(r.filePath,o,`utf8`),console.log(`Updated version in '${r.filePath}' to '${i}'`)})},s={bump:!1,changelog:!1,commit:!1,tag:!1,versionSourceFile:`./package.json`,newTagFormat:`v{{newVersion}}`,prevReleaseTagPattern:/^v(?<version>\d+\.\d+\.\d+)/,zeroMajorBreakingIsMinor:!0,dryRun:!1,context:{commitHyperlink:!0,refHyperlink:!0},commitsParser:{headerPattern:/^(?<type>\w+)(?:\((?<scope>.+)\))?(?<bang>!)?: (?<subject>.+)/s,breakingChangesPattern:/BREAKING CHANGES?:\s*(?<content>.+)/s,breakingChangeListPattern:/- (.+)/g,tagPattern:/tag: (?<tag>.*?)[,)]/g,coAuthorPattern:/Co-authored-by: (?<name>.+?) <(?<email>.+)>/g,signerPattern:/Signed-off-by: (?<name>.+?) <(?<email>.+)>/g,ghEmailPattern:/^(?:\d+\+)?(?<username>.+)@users\.noreply\.github\.com$/,remoteUrlPattern:/^(https:\/\/|git@)(?<host>[^/:]+)[/:](?<owner>.+?)\/(?<name>.+?)(?:\..*)?$/,refPattern:/^(?<action>.+?) (?<labels>.+)$/gm,refLabelPattern:/(?:(?<owner>\S+?)\/(?<repo>\S+?))?#(?<number>\d+)/g,refActionPattern:/Fixes|Closes|Refs/i,dateSource:`authorDate`,dateFormat:`YYYY-MM-DD`}},c={breaking:{title:`⚠️ BREAKING CHANGES`,commitType:`breaking`},feat:{title:`✨ Features`,commitType:`feat`},fix:{title:`🩹 Fixes`,commitType:`fix`},perf:{title:`⚡ Performance`,commitType:`perf`},refactor:{title:`🚜 Refactoring`,commitType:`refactor`},docs:{title:`📚 Documentation`,commitType:`docs`},style:{title:`🎨 Formatting`,commitType:`style`},build:{title:`📦 Build`,commitType:`build`},ci:{title:`🚀 CI`,commitType:`ci`},revert:{title:`♻️ Reverts`,commitType:`revert`},deps:{title:`🧩 Dependencies`,commitType:`chore`,filter:e=>!!e.scope?.includes(`deps`)},chore:{title:`🛠️ Chores`,commitType:`chore`},test:{title:`🧪 Tests`,commitType:`test`},misc:{title:`⚙️ Miscellaneous`,commitType:`*`},[Symbol.iterator](){return Object.values(this)[Symbol.iterator]()}},l={stdout:!1,outputFile:`./CHANGELOG.md`,commitRange:`unreleased`,sections:[...c],header:`# Changelog
|
|
5
2
|
|
|
6
|
-
//#region src/lifecycles/bump.ts
|
|
7
|
-
const bump = (config) => {
|
|
8
|
-
if (!config.bump) return;
|
|
9
|
-
const bumpFiles = config.bump, newVersion = config.context.newVersion;
|
|
10
|
-
bumpFiles.forEach((versionedFile) => {
|
|
11
|
-
const fileContent = readFileSync(versionedFile.filePath, "utf8");
|
|
12
|
-
const updatedContent = fileContent.replace(versionedFile.versionPattern, `$1${newVersion}$3`);
|
|
13
|
-
if (!config.dryRun) writeFileSync(versionedFile.filePath, updatedContent, "utf8");
|
|
14
|
-
console.log(`Updated version in '${versionedFile.filePath}' to '${newVersion}'`);
|
|
15
|
-
});
|
|
16
|
-
};
|
|
17
3
|
|
|
18
|
-
|
|
19
|
-
//#region src/defaults.ts
|
|
20
|
-
const defaultConfig = {
|
|
21
|
-
bump: false,
|
|
22
|
-
changelog: false,
|
|
23
|
-
commit: false,
|
|
24
|
-
tag: false,
|
|
25
|
-
versionSourceFile: "./package.json",
|
|
26
|
-
newTagFormat: "v{{newVersion}}",
|
|
27
|
-
prevReleaseTagPattern: /^v(?<version>\d+\.\d+\.\d+)/,
|
|
28
|
-
zeroMajorBreakingIsMinor: true,
|
|
29
|
-
dryRun: false,
|
|
30
|
-
context: {
|
|
31
|
-
commitHyperlink: true,
|
|
32
|
-
refHyperlink: true
|
|
33
|
-
},
|
|
34
|
-
commitsParser: {
|
|
35
|
-
headerPattern: /^(?<type>\w+)(?:\((?<scope>.+)\))?(?<bang>!)?: (?<subject>.+)/s,
|
|
36
|
-
breakingChangesPattern: /BREAKING CHANGES?:\s*(?<content>.+)/s,
|
|
37
|
-
breakingChangeListPattern: /- (.+)/g,
|
|
38
|
-
tagPattern: /tag: (?<tag>.*?)[,)]/g,
|
|
39
|
-
coAuthorPattern: /Co-authored-by: (?<name>.+?) <(?<email>.+)>/g,
|
|
40
|
-
signerPattern: /Signed-off-by: (?<name>.+?) <(?<email>.+)>/g,
|
|
41
|
-
ghEmailPattern: /^(?:\d+\+)?(?<username>.+)@users\.noreply\.github\.com$/,
|
|
42
|
-
remoteUrlPattern: /^(https:\/\/|git@)(?<host>[^/:]+)[/:](?<owner>.+?)\/(?<name>.+?)(?:\..*)?$/,
|
|
43
|
-
refPattern: /^(?<action>.+?) (?<labels>.+)$/gm,
|
|
44
|
-
refLabelPattern: /(?:(?<owner>\S+?)\/(?<repo>\S+?))?#(?<number>\d+)/g,
|
|
45
|
-
refActionPattern: /Fixes|Closes|Refs/i,
|
|
46
|
-
dateSource: "authorDate",
|
|
47
|
-
dateFormat: "YYYY-MM-DD"
|
|
48
|
-
}
|
|
49
|
-
};
|
|
50
|
-
const defaultChangelogSections = {
|
|
51
|
-
breaking: {
|
|
52
|
-
title: "⚠️ BREAKING CHANGES",
|
|
53
|
-
commitType: "breaking"
|
|
54
|
-
},
|
|
55
|
-
feat: {
|
|
56
|
-
title: "✨ Features",
|
|
57
|
-
commitType: "feat"
|
|
58
|
-
},
|
|
59
|
-
fix: {
|
|
60
|
-
title: "🩹 Fixes",
|
|
61
|
-
commitType: "fix"
|
|
62
|
-
},
|
|
63
|
-
perf: {
|
|
64
|
-
title: "⚡ Performance",
|
|
65
|
-
commitType: "perf"
|
|
66
|
-
},
|
|
67
|
-
refactor: {
|
|
68
|
-
title: "🚜 Refactoring",
|
|
69
|
-
commitType: "refactor"
|
|
70
|
-
},
|
|
71
|
-
docs: {
|
|
72
|
-
title: "📚 Documentation",
|
|
73
|
-
commitType: "docs"
|
|
74
|
-
},
|
|
75
|
-
style: {
|
|
76
|
-
title: "🎨 Formatting",
|
|
77
|
-
commitType: "style"
|
|
78
|
-
},
|
|
79
|
-
build: {
|
|
80
|
-
title: "📦 Build",
|
|
81
|
-
commitType: "build"
|
|
82
|
-
},
|
|
83
|
-
ci: {
|
|
84
|
-
title: "🚀 CI",
|
|
85
|
-
commitType: "ci"
|
|
86
|
-
},
|
|
87
|
-
revert: {
|
|
88
|
-
title: "♻️ Reverts",
|
|
89
|
-
commitType: "revert"
|
|
90
|
-
},
|
|
91
|
-
deps: {
|
|
92
|
-
title: "🧩 Dependencies",
|
|
93
|
-
commitType: "chore",
|
|
94
|
-
filter: (commit$1) => !!commit$1.scope?.includes("deps")
|
|
95
|
-
},
|
|
96
|
-
chore: {
|
|
97
|
-
title: "🛠️ Chores",
|
|
98
|
-
commitType: "chore"
|
|
99
|
-
},
|
|
100
|
-
test: {
|
|
101
|
-
title: "🧪 Tests",
|
|
102
|
-
commitType: "test"
|
|
103
|
-
},
|
|
104
|
-
misc: {
|
|
105
|
-
title: "⚙️ Miscellaneous",
|
|
106
|
-
commitType: "*"
|
|
107
|
-
},
|
|
108
|
-
[Symbol.iterator]() {
|
|
109
|
-
return Object.values(this)[Symbol.iterator]();
|
|
110
|
-
}
|
|
111
|
-
};
|
|
112
|
-
const defaultChangelogOptions = {
|
|
113
|
-
stdout: false,
|
|
114
|
-
outputFile: "./CHANGELOG.md",
|
|
115
|
-
commitRange: "unreleased",
|
|
116
|
-
sections: [...defaultChangelogSections],
|
|
117
|
-
header: "# Changelog\n\n\n",
|
|
118
|
-
prevReleaseHeaderPattern: /^##.*?\d+\.\d+\.\d+/m,
|
|
119
|
-
helpers: { repeat: (string, n) => string.repeat(n) },
|
|
120
|
-
partials: {}
|
|
121
|
-
};
|
|
122
|
-
const defaultCommitOptions = {
|
|
123
|
-
message: "release({{repo.name}}): {{newTag}}",
|
|
124
|
-
signOff: false,
|
|
125
|
-
gpgSign: false,
|
|
126
|
-
stageAll: true,
|
|
127
|
-
extraArgs: ""
|
|
128
|
-
};
|
|
129
|
-
const defaultTagOptions = {
|
|
130
|
-
name: "{{newTag}}",
|
|
131
|
-
message: "release({{repo.name}}): {{newTag}}",
|
|
132
|
-
gpgSign: false,
|
|
133
|
-
force: false,
|
|
134
|
-
extraArgs: ""
|
|
135
|
-
};
|
|
136
|
-
const defaultVersionedFiles = [{
|
|
137
|
-
filePathRegex: /package\.json$/,
|
|
138
|
-
versionPattern: /(^.*?"version".*?")(.*?)(")/s
|
|
139
|
-
}, {
|
|
140
|
-
filePathRegex: /package-lock\.json$/,
|
|
141
|
-
versionPattern: /(^.*?"version".*?"|"packages".*?"".*"version".*?")(.*?)(")/gs
|
|
142
|
-
}];
|
|
4
|
+
`,prevReleaseHeaderPattern:/^##.*?\d+\.\d+\.\d+/m,helpers:{repeat:(e,t)=>e.repeat(t)},partials:{}},u={message:`release({{repo.name}}): {{newTag}}`,signOff:!1,gpgSign:!1,stageAll:!0,extraArgs:``},d={name:`{{newTag}}`,message:`release({{repo.name}}): {{newTag}}`,gpgSign:!1,force:!1,extraArgs:``},f=[{filePathRegex:/package\.json$/,versionPattern:/(^.*?"version".*?")(.*?)(")/s},{filePathRegex:/package-lock\.json$/,versionPattern:/(^.*?"version".*?"|"packages".*?"".*"version".*?")(.*?)(")/gs}],p=async e=>{let t=m(e),n=h(t),r=g(n),i=await _(r),a=x(i);return a},m=e=>{let t=e.profile;if(!t)return e;let n=e[`_${t}`];if(!n)throw Error(`Profile "${t}" not found in configuration.`);let r=(t,...r)=>{let i=e=>Object.prototype.toString.call(e)===`[object Object]`,a=(e,t)=>{let n=i(e),r=i(t);if(n&&r)return{...e,...t};if(!n&&r)return t;if(n&&!r)return e},o=e[t],s=n[t],c=a(o,s);if(c!==void 0)return r.forEach(e=>{c[e]=a(o?.[e],s?.[e])}),c};return{...e,...n,commitsParser:r(`commitsParser`),changelog:r(`changelog`,`partials`,`helpers`),commit:r(`commit`),tag:r(`tag`),context:r(`context`)}},h=e=>{let t=(e,t,n,...r)=>{if(t==null||t===!1)return!1;if(t===!0)return n;if(typeof t!=`object`)throw Error(`Invalid value for ${e}. It should be a boolean or an object.`);let i={...n,...t};return r.forEach(e=>i[e]={...n[e],...t[e]}),i};return{...s,...e,commitsParser:{...s.commitsParser,...e.commitsParser},changelog:t(`changelog`,e.changelog,l,`partials`,`helpers`),commit:t(`commit`,e.commit,u),tag:t(`tag`,e.tag,d)}},g=e=>{let t=e=>{let t=f.find(t=>t.filePathRegex.test(e));if(t)return{filePath:e,versionPattern:t.versionPattern};throw Error(`File ${e} doesn't match any default versioned files. Please provide a custom version pattern for this file.`)},n=e=>{if(e===!1)return!1;if(e===!0)return[i];if(Array.isArray(e))return[i,...e.map(e=>typeof e==`string`?t(e):e)];throw Error(`Invalid value for bump. It should be a boolean or an array.`)},i=typeof e.versionSourceFile==`string`?t(e.versionSourceFile):e.versionSourceFile;return{...e,versionSourceFile:i,bump:n(e.bump),changelog:e.changelog===!1?!1:{...e.changelog,compiledPartials:Object.fromEntries(Object.entries(e.changelog.partials).map(([e,t])=>[e,r.compile(t)]))}}},_=async e=>{let t=e.context??{},n=F(e.commitsParser.remoteUrlPattern);t.repo={...n,...t.repo};let r=e.changelog?e.changelog.commitRange:`unreleased`;t.commits=e.context?.commits?await Promise.all(e.context.commits.map(async t=>typeof t==`object`&&`message`in t||typeof t==`string`?(await O([t],e.commitsParser,e.prevReleaseTagPattern))[0]:t)):await O(r,e.commitsParser,e.prevReleaseTagPattern),t.currentVersion??=C(e.versionSourceFile),t.currentTag??=R(e.prevReleaseTagPattern)[0],t.newVersion??=await w(e,t.currentVersion),t.newTag??=S(e.newTagFormat,t);let i={...e,context:t};return t.releases=e.changelog?v(t.commits,e.changelog.sections,i):null,i},v=(e,t,n)=>{let r={};return e.forEach(e=>{let t=e.tags?.find(e=>n.prevReleaseTagPattern.test(e));if(t)r[t]??={tag:t,version:n.prevReleaseTagPattern.exec(t)?.groups?.version,date:e.date,commits:[e]};else{let t=Object.keys(r).at(-1);t?r[t].commits.push(e):r[n.context.newTag]={tag:n.context.newTag,version:n.context.newVersion,date:e.date,commits:[e]}}}),Object.values(r).map(e=>y(e,t))},y=(e,t)=>{let{commits:n,...r}=e;return{...r,commitGroups:b(n,t)}},b=(e,t)=>{let n=Object.fromEntries(t.map(e=>[e.title,[]]));return e.forEach(e=>{let r=!!e.breakingChanges,i=!1,a=!1;for(let o of t){if(o.filter&&!o.filter(e))continue;let t=[o.commitType].flat();if(r&&!a&&t.includes(`breaking`)){n[o.title].push(e),a=!0;continue}if(!i&&(t.includes(e.type)||t.includes(`*`))&&(n[o.title].push(e),i=!0),i&&(!r||a))return}}),Object.keys(n).forEach(e=>{n[e].length||delete n[e]}),Object.entries(n).map(([e,t])=>({title:e,commits:t}))},x=e=>({...e,commit:e.commit?{...e.commit,message:S(e.commit.message,e.context)}:e.commit,tag:e.tag?{...e.tag,name:S(e.tag.name,e.context),message:S(e.tag.message,e.context)}:e.tag}),S=(e,t)=>{let n=r.compile(e);return n(t)},C=e=>{let n=t(e.filePath,`utf8`),r=e.versionPattern.exec(n)?.[2];if(!r)throw Error(`Version not found in '${e.filePath}' with pattern '${e.versionPattern}'`);if(!i.valid(r))throw Error(`Invalid version format in '${e.filePath}': '${r}'`);return console.log(`Current version from '${e.filePath}': '${r}'`),r},w=async(e,t)=>{if(e.releaseVersion){if(!i.valid(e.releaseVersion))throw Error(`Invalid release version format: '${e.releaseVersion}'`);return e.releaseVersion}let n;if(e.releaseType)n=e.releaseType;else{let r=await O(`unreleased`,e.commitsParser,e.prevReleaseTagPattern);n=T(r),e.zeroMajorBreakingIsMinor&&i.major(t)===0&&n===`major`&&(n=`minor`)}let r=E(t,n);return console.log(`Determined new version: '${r}' (release type: '${n}')`),r},T=e=>{let t=e.some(e=>e.breakingChanges);if(t)return`major`;let n=e.some(e=>e.type===`feat`);return n?`minor`:`patch`},E=(e,t)=>i.inc(e,t)??(()=>{throw Error(`Failed to calculate new version from '${e}' with release type '${t}'`)})();let D=function(e){return e.G=`valid`,e.B=`bad`,e.U=`valid, unknown validity`,e.X=`valid, expired`,e.Y=`valid, made by expired key`,e.R=`valid, made by revoked key`,e.E=`cannot check (missing key)`,e.N=`no signature`,e}({});const O=async(e,t,n)=>{let r=Array.isArray(e)?e:I(e,n),i=t;return(await Promise.all(r.map(async e=>{typeof e==`string`&&(e={message:e});let{hash:t,tagRefs:n}=e,r=e.message.trim();if(!r)throw Error(`Message is missing for commit: ${JSON.stringify(e)}`);let a;try{a=k(r,i)}catch(e){console.warn(`Error parsing commit '${t??`<no hash>`}':`,e.message);return}let{type:o,scope:s,subject:c,body:l,breakingChanges:u,footer:d}=a,f=n?[...n.matchAll(i.tagPattern)].map(e=>e.groups?.tag??``):[],p=d?[...d.matchAll(i.signerPattern)].map(e=>e.groups):[],m=[],h=e=>{m.some(t=>t.email===e.email)||m.push(e)},g=e.authorName&&e.authorEmail?j({name:e.authorName,email:e.authorEmail},p):void 0;g&&h(g);let _=e.committerName&&e.committerEmail?j({name:e.committerName,email:e.committerEmail},p):void 0;_&&h(_);let v=d?[...d.matchAll(i.coAuthorPattern)].map(e=>e.groups).map(e=>j(e,p)):[];v.forEach(e=>h(e));let y=await M(d??``,i),b=e.gpgSigCode?{code:e.gpgSigCode,label:D[e.gpgSigCode],keyId:e.gpgSigKeyId}:void 0,x=e[i.dateSource===`committerDate`?`committerTs`:`authorTs`];typeof x==`string`&&(x=N(new Date(x*1e3),i.dateFormat));let S={hash:t,type:o,scope:s,subject:c,body:l,breakingChanges:u,footer:d,committer:_,gpgSig:b,date:x,tags:f.length?f:void 0,authors:m.length?m:void 0,refs:y.length?y:void 0};return S}))).filter(e=>e!==void 0)},k=(e,t)=>{let[n,...r]=e.split(`
|
|
143
5
|
|
|
144
|
-
|
|
145
|
-
//#region src/utils/config-resolver.ts
|
|
146
|
-
const resolveConfig = async (userConfig) => {
|
|
147
|
-
const profileMergedConfig = mergeProfileConfig(userConfig);
|
|
148
|
-
const mergedConfig = mergeWithDefaults(profileMergedConfig);
|
|
149
|
-
const transformedConfig = transformVersionedFiles(mergedConfig);
|
|
150
|
-
const contextualConfig = await fillContext(transformedConfig);
|
|
151
|
-
const finalConfig = resolveTemplates(contextualConfig);
|
|
152
|
-
return finalConfig;
|
|
153
|
-
};
|
|
154
|
-
const mergeProfileConfig = (baseConfig) => {
|
|
155
|
-
const profileName = baseConfig.profile;
|
|
156
|
-
if (!profileName) return baseConfig;
|
|
157
|
-
const profileConfig = baseConfig[`_${profileName}`];
|
|
158
|
-
if (!profileConfig) throw new Error(`Profile "${profileName}" not found in configuration.`);
|
|
159
|
-
const mergeOption = (propKey, ...nestedPropKeys) => {
|
|
160
|
-
const isPlainObject = (value) => Object.prototype.toString.call(value) === "[object Object]";
|
|
161
|
-
const mergeObjects = (baseObj, overrideObject) => {
|
|
162
|
-
const isBasePlainObject = isPlainObject(baseObj);
|
|
163
|
-
const isOverridePlainObject = isPlainObject(overrideObject);
|
|
164
|
-
if (isBasePlainObject && isOverridePlainObject) return {
|
|
165
|
-
...baseObj,
|
|
166
|
-
...overrideObject
|
|
167
|
-
};
|
|
168
|
-
else if (!isBasePlainObject && isOverridePlainObject) return overrideObject;
|
|
169
|
-
else if (isBasePlainObject && !isOverridePlainObject) return baseObj;
|
|
170
|
-
};
|
|
171
|
-
const baseConfigProp = baseConfig[propKey];
|
|
172
|
-
const profileConfigProp = profileConfig[propKey];
|
|
173
|
-
const result = mergeObjects(baseConfigProp, profileConfigProp);
|
|
174
|
-
if (result === void 0) return void 0;
|
|
175
|
-
nestedPropKeys.forEach((key) => {
|
|
176
|
-
result[key] = mergeObjects(baseConfigProp?.[key], profileConfigProp?.[key]);
|
|
177
|
-
});
|
|
178
|
-
return result;
|
|
179
|
-
};
|
|
180
|
-
return {
|
|
181
|
-
...baseConfig,
|
|
182
|
-
...profileConfig,
|
|
183
|
-
commitsParser: mergeOption("commitsParser"),
|
|
184
|
-
changelog: mergeOption("changelog", "partials", "helpers"),
|
|
185
|
-
commit: mergeOption("commit"),
|
|
186
|
-
tag: mergeOption("tag"),
|
|
187
|
-
context: mergeOption("context")
|
|
188
|
-
};
|
|
189
|
-
};
|
|
190
|
-
const mergeWithDefaults = (userConfig) => {
|
|
191
|
-
const resolveOptions = (optionsName, options, defaults, ...subObjects) => {
|
|
192
|
-
if (options == void 0 || options === false) return false;
|
|
193
|
-
if (options === true) return defaults;
|
|
194
|
-
if (typeof options !== "object") throw new Error(`Invalid value for ${optionsName}. It should be a boolean or an object.`);
|
|
195
|
-
const result = {
|
|
196
|
-
...defaults,
|
|
197
|
-
...options
|
|
198
|
-
};
|
|
199
|
-
subObjects.forEach((subObjectKey) => result[subObjectKey] = {
|
|
200
|
-
...defaults[subObjectKey],
|
|
201
|
-
...options[subObjectKey]
|
|
202
|
-
});
|
|
203
|
-
return result;
|
|
204
|
-
};
|
|
205
|
-
return {
|
|
206
|
-
...defaultConfig,
|
|
207
|
-
...userConfig,
|
|
208
|
-
commitsParser: {
|
|
209
|
-
...defaultConfig.commitsParser,
|
|
210
|
-
...userConfig.commitsParser
|
|
211
|
-
},
|
|
212
|
-
changelog: resolveOptions("changelog", userConfig.changelog, defaultChangelogOptions, "partials", "helpers"),
|
|
213
|
-
commit: resolveOptions("commit", userConfig.commit, defaultCommitOptions),
|
|
214
|
-
tag: resolveOptions("tag", userConfig.tag, defaultTagOptions)
|
|
215
|
-
};
|
|
216
|
-
};
|
|
217
|
-
const transformVersionedFiles = (config) => {
|
|
218
|
-
const resolveVersionedFile = (filePath) => {
|
|
219
|
-
const matchingDefaultVersionFile = defaultVersionedFiles.find((defaultFile) => defaultFile.filePathRegex.test(filePath));
|
|
220
|
-
if (matchingDefaultVersionFile) return {
|
|
221
|
-
filePath,
|
|
222
|
-
versionPattern: matchingDefaultVersionFile.versionPattern
|
|
223
|
-
};
|
|
224
|
-
else throw new Error(`File ${filePath} doesn't match any default versioned files. Please provide a custom version pattern for this file.`);
|
|
225
|
-
};
|
|
226
|
-
const resolveBump = (bump$1) => {
|
|
227
|
-
if (bump$1 === false) return false;
|
|
228
|
-
if (bump$1 === true) return [versionSourceFile];
|
|
229
|
-
if (Array.isArray(bump$1)) return [versionSourceFile, ...bump$1.map((bumpFile) => typeof bumpFile === "string" ? resolveVersionedFile(bumpFile) : bumpFile)];
|
|
230
|
-
throw new Error("Invalid value for bump. It should be a boolean or an array.");
|
|
231
|
-
};
|
|
232
|
-
const versionSourceFile = typeof config.versionSourceFile === "string" ? resolveVersionedFile(config.versionSourceFile) : config.versionSourceFile;
|
|
233
|
-
return {
|
|
234
|
-
...config,
|
|
235
|
-
versionSourceFile,
|
|
236
|
-
bump: resolveBump(config.bump),
|
|
237
|
-
changelog: config.changelog === false ? false : {
|
|
238
|
-
...config.changelog,
|
|
239
|
-
compiledPartials: Object.fromEntries(Object.entries(config.changelog.partials).map(([key, template]) => [key, Handlebars.compile(template)]))
|
|
240
|
-
}
|
|
241
|
-
};
|
|
242
|
-
};
|
|
243
|
-
const fillContext = async (config) => {
|
|
244
|
-
const resolvedContext = config.context ?? {};
|
|
245
|
-
const repoInfo = getRepoInfo(config.commitsParser.remoteUrlPattern);
|
|
246
|
-
resolvedContext.repo = {
|
|
247
|
-
...repoInfo,
|
|
248
|
-
...resolvedContext.repo
|
|
249
|
-
};
|
|
250
|
-
const commitRange = config.changelog ? config.changelog.commitRange : "unreleased";
|
|
251
|
-
resolvedContext.commits = config.context?.commits ? await Promise.all(config.context.commits.map(async (commit$1) => {
|
|
252
|
-
return typeof commit$1 === "object" && "message" in commit$1 || typeof commit$1 === "string" ? (await parseCommits([commit$1], config.commitsParser, config.prevReleaseTagPattern))[0] : commit$1;
|
|
253
|
-
})) : await parseCommits(commitRange, config.commitsParser, config.prevReleaseTagPattern);
|
|
254
|
-
resolvedContext.currentVersion ??= parseVersion(config.versionSourceFile);
|
|
255
|
-
resolvedContext.currentTag ??= getVersionTags(config.prevReleaseTagPattern)[0];
|
|
256
|
-
resolvedContext.newVersion ??= await determineNextVersion(config, resolvedContext.currentVersion);
|
|
257
|
-
resolvedContext.newTag ??= compileTemplate(config.newTagFormat, resolvedContext);
|
|
258
|
-
const contextualConfig = {
|
|
259
|
-
...config,
|
|
260
|
-
context: resolvedContext
|
|
261
|
-
};
|
|
262
|
-
resolvedContext.releases = config.changelog ? groupCommitsByReleases(resolvedContext.commits, config.changelog.sections, contextualConfig) : null;
|
|
263
|
-
return contextualConfig;
|
|
264
|
-
};
|
|
265
|
-
const groupCommitsByReleases = (commits, sections, config) => {
|
|
266
|
-
const releases = {};
|
|
267
|
-
commits.forEach((commit$1) => {
|
|
268
|
-
const releaseTag = commit$1.tags?.find((tag$1) => config.prevReleaseTagPattern.test(tag$1));
|
|
269
|
-
if (releaseTag) releases[releaseTag] ??= {
|
|
270
|
-
tag: releaseTag,
|
|
271
|
-
version: config.prevReleaseTagPattern.exec(releaseTag)?.groups?.version,
|
|
272
|
-
date: commit$1.date,
|
|
273
|
-
commits: [commit$1]
|
|
274
|
-
};
|
|
275
|
-
else {
|
|
276
|
-
const latestReleaseTag = Object.keys(releases).at(-1);
|
|
277
|
-
if (latestReleaseTag) releases[latestReleaseTag].commits.push(commit$1);
|
|
278
|
-
else releases[config.context.newTag] = {
|
|
279
|
-
tag: config.context.newTag,
|
|
280
|
-
version: config.context.newVersion,
|
|
281
|
-
date: commit$1.date,
|
|
282
|
-
commits: [commit$1]
|
|
283
|
-
};
|
|
284
|
-
}
|
|
285
|
-
});
|
|
286
|
-
return Object.values(releases).map((release) => groupReleaseCommitsBySections(release, sections));
|
|
287
|
-
};
|
|
288
|
-
const groupReleaseCommitsBySections = (release, sections) => {
|
|
289
|
-
const { commits,...releaseWithoutCommits } = release;
|
|
290
|
-
return {
|
|
291
|
-
...releaseWithoutCommits,
|
|
292
|
-
commitGroups: groupCommitsBySections(commits, sections)
|
|
293
|
-
};
|
|
294
|
-
};
|
|
295
|
-
const groupCommitsBySections = (commits, sections) => {
|
|
296
|
-
const commitGroups = Object.fromEntries(sections.map((section) => [section.title, []]));
|
|
297
|
-
commits.forEach((commit$1) => {
|
|
298
|
-
const isBreaking = !!commit$1.breakingChanges;
|
|
299
|
-
let isGrouped = false;
|
|
300
|
-
let isBreakingGrouped = false;
|
|
301
|
-
for (const section of sections) {
|
|
302
|
-
if (section.filter && !section.filter(commit$1)) continue;
|
|
303
|
-
const sectionTypes = [section.commitType].flat();
|
|
304
|
-
if (isBreaking && !isBreakingGrouped && sectionTypes.includes("breaking")) {
|
|
305
|
-
commitGroups[section.title].push(commit$1);
|
|
306
|
-
isBreakingGrouped = true;
|
|
307
|
-
continue;
|
|
308
|
-
}
|
|
309
|
-
if (!isGrouped && (sectionTypes.includes(commit$1.type) || sectionTypes.includes("*"))) {
|
|
310
|
-
commitGroups[section.title].push(commit$1);
|
|
311
|
-
isGrouped = true;
|
|
312
|
-
}
|
|
313
|
-
if (isGrouped && (!isBreaking || isBreakingGrouped)) return;
|
|
314
|
-
}
|
|
315
|
-
});
|
|
316
|
-
Object.keys(commitGroups).forEach((key) => {
|
|
317
|
-
if (!commitGroups[key].length) delete commitGroups[key];
|
|
318
|
-
});
|
|
319
|
-
return Object.entries(commitGroups).map(([title, commits$1]) => ({
|
|
320
|
-
title,
|
|
321
|
-
commits: commits$1
|
|
322
|
-
}));
|
|
323
|
-
};
|
|
324
|
-
const resolveTemplates = (config) => {
|
|
325
|
-
return {
|
|
326
|
-
...config,
|
|
327
|
-
commit: config.commit ? {
|
|
328
|
-
...config.commit,
|
|
329
|
-
message: compileTemplate(config.commit.message, config.context)
|
|
330
|
-
} : config.commit,
|
|
331
|
-
tag: config.tag ? {
|
|
332
|
-
...config.tag,
|
|
333
|
-
name: compileTemplate(config.tag.name, config.context),
|
|
334
|
-
message: compileTemplate(config.tag.message, config.context)
|
|
335
|
-
} : config.tag
|
|
336
|
-
};
|
|
337
|
-
};
|
|
338
|
-
const compileTemplate = (value, context) => {
|
|
339
|
-
const compile = Handlebars.compile(value);
|
|
340
|
-
return compile(context);
|
|
341
|
-
};
|
|
6
|
+
`),i=t.headerPattern.exec(n);if(!i?.groups)throw Error(`Commit header '${n}' doesn't match expected format`);let{type:a,scope:o,bang:s,subject:c}=i.groups,l,u=r.find(e=>t.breakingChangesPattern.test(e));u?(l=A(u,t),r.splice(r.indexOf(u),1)):s&&(l=c);let d=r.findIndex(e=>e.match(t.refActionPattern)??e.match(t.coAuthorPattern)??e.match(t.signerPattern)),[f,p]=d===-1?[r.join(`
|
|
342
7
|
|
|
343
|
-
|
|
344
|
-
//#region src/utils/version-manager.ts
|
|
345
|
-
const parseVersion = (versionedFile) => {
|
|
346
|
-
const fileContent = readFileSync(versionedFile.filePath, "utf8");
|
|
347
|
-
const version = versionedFile.versionPattern.exec(fileContent)?.[2];
|
|
348
|
-
if (!version) throw new Error(`Version not found in '${versionedFile.filePath}' with pattern '${versionedFile.versionPattern}'`);
|
|
349
|
-
if (!semver.valid(version)) throw new Error(`Invalid version format in '${versionedFile.filePath}': '${version}'`);
|
|
350
|
-
console.log(`Current version from '${versionedFile.filePath}': '${version}'`);
|
|
351
|
-
return version;
|
|
352
|
-
};
|
|
353
|
-
const determineNextVersion = async (config, currentVersion) => {
|
|
354
|
-
if (config.releaseVersion) {
|
|
355
|
-
if (!semver.valid(config.releaseVersion)) throw new Error(`Invalid release version format: '${config.releaseVersion}'`);
|
|
356
|
-
return config.releaseVersion;
|
|
357
|
-
}
|
|
358
|
-
let releaseType;
|
|
359
|
-
if (config.releaseType) releaseType = config.releaseType;
|
|
360
|
-
else {
|
|
361
|
-
const unreleasedCommits = await parseCommits("unreleased", config.commitsParser, config.prevReleaseTagPattern);
|
|
362
|
-
releaseType = calculateReleaseType(unreleasedCommits);
|
|
363
|
-
if (config.zeroMajorBreakingIsMinor && semver.major(currentVersion) === 0 && releaseType === "major") releaseType = "minor";
|
|
364
|
-
}
|
|
365
|
-
const newVersion = increaseVersion(currentVersion, releaseType);
|
|
366
|
-
console.log(`Determined new version: '${newVersion}' (release type: '${releaseType}')`);
|
|
367
|
-
return newVersion;
|
|
368
|
-
};
|
|
369
|
-
const calculateReleaseType = (commits) => {
|
|
370
|
-
const hasBreakingChange = commits.some((commit$1) => commit$1.breakingChanges);
|
|
371
|
-
if (hasBreakingChange) return "major";
|
|
372
|
-
const hasFeature = commits.some((commit$1) => commit$1.type === "feat");
|
|
373
|
-
if (hasFeature) return "minor";
|
|
374
|
-
return "patch";
|
|
375
|
-
};
|
|
376
|
-
const increaseVersion = (currentVersion, releaseType) => semver.inc(currentVersion, releaseType) ?? (() => {
|
|
377
|
-
throw new Error(`Failed to calculate new version from '${currentVersion}' with release type '${releaseType}'`);
|
|
378
|
-
})();
|
|
8
|
+
`),``]:[r.slice(0,d).join(`
|
|
379
9
|
|
|
380
|
-
|
|
381
|
-
//#region src/enums.ts
|
|
382
|
-
let GpgSigLabel = /* @__PURE__ */ function(GpgSigLabel$1) {
|
|
383
|
-
GpgSigLabel$1["G"] = "valid";
|
|
384
|
-
GpgSigLabel$1["B"] = "bad";
|
|
385
|
-
GpgSigLabel$1["U"] = "valid, unknown validity";
|
|
386
|
-
GpgSigLabel$1["X"] = "valid, expired";
|
|
387
|
-
GpgSigLabel$1["Y"] = "valid, made by expired key";
|
|
388
|
-
GpgSigLabel$1["R"] = "valid, made by revoked key";
|
|
389
|
-
GpgSigLabel$1["E"] = "cannot check (missing key)";
|
|
390
|
-
GpgSigLabel$1["N"] = "no signature";
|
|
391
|
-
return GpgSigLabel$1;
|
|
392
|
-
}({});
|
|
10
|
+
`),r.slice(d).join(`
|
|
393
11
|
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
const rawCommits = Array.isArray(arg1) ? arg1 : getRawCommits(arg1, prevReleaseTagPattern);
|
|
398
|
-
const parser = commitsParser;
|
|
399
|
-
return (await Promise.all(rawCommits.map(async (commit$1) => {
|
|
400
|
-
if (typeof commit$1 === "string") commit$1 = { message: commit$1 };
|
|
401
|
-
const { hash, tagRefs } = commit$1;
|
|
402
|
-
const message = commit$1.message.trim();
|
|
403
|
-
if (!message) throw new Error(`Message is missing for commit: ${JSON.stringify(commit$1)}`);
|
|
404
|
-
let parsedMessage;
|
|
405
|
-
try {
|
|
406
|
-
parsedMessage = parseCommitMessage(message, parser);
|
|
407
|
-
} catch (error) {
|
|
408
|
-
console.warn(`Error parsing commit '${hash ?? "<no hash>"}':`, error.message);
|
|
409
|
-
return;
|
|
410
|
-
}
|
|
411
|
-
const { type, scope, subject, body, breakingChanges, footer } = parsedMessage;
|
|
412
|
-
const tags = tagRefs ? [...tagRefs.matchAll(parser.tagPattern)].map((m) => m.groups?.tag ?? "") : [];
|
|
413
|
-
const signers = footer ? [...footer.matchAll(parser.signerPattern)].map((m) => m.groups) : [];
|
|
414
|
-
const authors = [];
|
|
415
|
-
const addAuthor = (contributor) => {
|
|
416
|
-
if (!authors.some((a) => a.email === contributor.email)) authors.push(contributor);
|
|
417
|
-
};
|
|
418
|
-
const author = commit$1.authorName && commit$1.authorEmail ? getContributorDetails({
|
|
419
|
-
name: commit$1.authorName,
|
|
420
|
-
email: commit$1.authorEmail
|
|
421
|
-
}, signers) : void 0;
|
|
422
|
-
if (author) addAuthor(author);
|
|
423
|
-
const committer = commit$1.committerName && commit$1.committerEmail ? getContributorDetails({
|
|
424
|
-
name: commit$1.committerName,
|
|
425
|
-
email: commit$1.committerEmail
|
|
426
|
-
}, signers) : void 0;
|
|
427
|
-
if (committer) addAuthor(committer);
|
|
428
|
-
const coAuthors = footer ? [...footer.matchAll(parser.coAuthorPattern)].map((m) => m.groups).map((coAuthor) => getContributorDetails(coAuthor, signers)) : [];
|
|
429
|
-
coAuthors.forEach((coAuthor) => addAuthor(coAuthor));
|
|
430
|
-
const refs = await parseRefs(footer ?? "", parser);
|
|
431
|
-
const gpgSig = commit$1.gpgSigCode ? {
|
|
432
|
-
code: commit$1.gpgSigCode,
|
|
433
|
-
label: GpgSigLabel[commit$1.gpgSigCode],
|
|
434
|
-
keyId: commit$1.gpgSigKeyId
|
|
435
|
-
} : void 0;
|
|
436
|
-
let date = commit$1[parser.dateSource === "committerDate" ? "committerTs" : "authorTs"];
|
|
437
|
-
if (typeof date === "string") date = formatDate(/* @__PURE__ */ new Date(+date * 1e3), parser.dateFormat);
|
|
438
|
-
const parsedCommit = {
|
|
439
|
-
hash,
|
|
440
|
-
type,
|
|
441
|
-
scope,
|
|
442
|
-
subject,
|
|
443
|
-
body,
|
|
444
|
-
breakingChanges,
|
|
445
|
-
footer,
|
|
446
|
-
committer,
|
|
447
|
-
gpgSig,
|
|
448
|
-
date,
|
|
449
|
-
tags: tags.length ? tags : void 0,
|
|
450
|
-
authors: authors.length ? authors : void 0,
|
|
451
|
-
refs: refs.length ? refs : void 0
|
|
452
|
-
};
|
|
453
|
-
return parsedCommit;
|
|
454
|
-
}))).filter((commit$1) => commit$1 !== void 0);
|
|
455
|
-
};
|
|
456
|
-
const parseCommitMessage = (message, parser) => {
|
|
457
|
-
const [header, ...details] = message.split("\n\n");
|
|
458
|
-
const headerMatch = parser.headerPattern.exec(header);
|
|
459
|
-
if (!headerMatch?.groups) throw new Error(`Commit header '${header}' doesn't match expected format`);
|
|
460
|
-
const { type, scope, bang, subject } = headerMatch.groups;
|
|
461
|
-
let breakingChanges;
|
|
462
|
-
const breakingChangesPart = details.find((detail) => parser.breakingChangesPattern.test(detail));
|
|
463
|
-
if (breakingChangesPart) {
|
|
464
|
-
breakingChanges = parseBreakingChanges(breakingChangesPart, parser);
|
|
465
|
-
details.splice(details.indexOf(breakingChangesPart), 1);
|
|
466
|
-
} else if (bang) breakingChanges = subject;
|
|
467
|
-
const footerStart = details.findIndex((detail) => detail.match(parser.refActionPattern) ?? detail.match(parser.coAuthorPattern) ?? detail.match(parser.signerPattern));
|
|
468
|
-
const [body, footer] = footerStart === -1 ? [details.join("\n\n"), ""] : [details.slice(0, footerStart).join("\n\n"), details.slice(footerStart).join("\n\n")];
|
|
469
|
-
return {
|
|
470
|
-
type,
|
|
471
|
-
scope: scope || void 0,
|
|
472
|
-
subject,
|
|
473
|
-
body: body || void 0,
|
|
474
|
-
breakingChanges,
|
|
475
|
-
footer: footer || void 0
|
|
476
|
-
};
|
|
477
|
-
};
|
|
478
|
-
const parseBreakingChanges = (value, parser) => {
|
|
479
|
-
const breakingChanges = parser.breakingChangesPattern.exec(value)?.groups?.content;
|
|
480
|
-
if (!breakingChanges) throw new Error(`Failed to extract breaking changes content from '${value}' using pattern "${parser.breakingChangesPattern}"`);
|
|
481
|
-
const breakingChangeList = [...breakingChanges.matchAll(parser.breakingChangeListPattern)];
|
|
482
|
-
return breakingChangeList.length ? breakingChangeList.map((m) => m[1]) : breakingChanges;
|
|
483
|
-
};
|
|
484
|
-
const getContributorDetails = (contributor, signers) => {
|
|
485
|
-
const hasSignedOff = signers.some((signer) => signer.email === contributor.email && signer.name === contributor.name);
|
|
486
|
-
return {
|
|
487
|
-
...contributor,
|
|
488
|
-
hasSignedOff,
|
|
489
|
-
ghLogin: contributor.name,
|
|
490
|
-
ghUrl: `https://github.com/${contributor.name}`
|
|
491
|
-
};
|
|
492
|
-
};
|
|
493
|
-
const parseRefs = async (value, parser) => await Promise.all([...value.matchAll(parser.refPattern)].map((m) => m.groups).filter((rawRef) => parser.refActionPattern.test(rawRef.action)).flatMap((rawRef) => [...rawRef.labels.matchAll(parser.refLabelPattern)].map((m) => m.groups).filter((label) => !!label.number).map((label) => ({
|
|
494
|
-
action: rawRef.action,
|
|
495
|
-
owner: label.owner,
|
|
496
|
-
repo: label.repo,
|
|
497
|
-
number: label.number
|
|
498
|
-
}))));
|
|
499
|
-
const formatDate = (date, format) => {
|
|
500
|
-
const pad = (num) => num.toString().padStart(2, "0");
|
|
501
|
-
const dateParts = {
|
|
502
|
-
YYYY: date.getUTCFullYear().toString(),
|
|
503
|
-
MM: pad(date.getUTCMonth() + 1),
|
|
504
|
-
DD: pad(date.getUTCDate()),
|
|
505
|
-
HH: pad(date.getUTCHours()),
|
|
506
|
-
mm: pad(date.getUTCMinutes()),
|
|
507
|
-
ss: pad(date.getUTCSeconds())
|
|
508
|
-
};
|
|
509
|
-
return format.replace(/YYYY|MM|DD|HH|mm|ss/g, (k) => dateParts[k]);
|
|
510
|
-
};
|
|
12
|
+
`)];return{type:a,scope:o||void 0,subject:c,body:f||void 0,breakingChanges:l,footer:p||void 0}},A=(e,t)=>{let n=t.breakingChangesPattern.exec(e)?.groups?.content;if(!n)throw Error(`Failed to extract breaking changes content from '${e}' using pattern "${t.breakingChangesPattern}"`);let r=[...n.matchAll(t.breakingChangeListPattern)];return r.length?r.map(e=>e[1]):n},j=(e,t)=>{let n=t.some(t=>t.email===e.email&&t.name===e.name);return{...e,hasSignedOff:n,ghLogin:e.name,ghUrl:`https://github.com/${e.name}`}},M=async(e,t)=>await Promise.all([...e.matchAll(t.refPattern)].map(e=>e.groups).filter(e=>t.refActionPattern.test(e.action)).flatMap(e=>[...e.labels.matchAll(t.refLabelPattern)].map(e=>e.groups).filter(e=>!!e.number).map(t=>({action:e.action,owner:t.owner,repo:t.repo,number:t.number})))),N=(e,t)=>{let n=e=>e.toString().padStart(2,`0`),r={YYYY:e.getUTCFullYear().toString(),MM:n(e.getUTCMonth()+1),DD:n(e.getUTCDate()),HH:n(e.getUTCHours()),mm:n(e.getUTCMinutes()),ss:n(e.getUTCSeconds())};return t.replace(/YYYY|MM|DD|HH|mm|ss/g,e=>r[e])},P=/##COMMIT##\n#HASH# (?<hash>.+)?\n#MSG# (?<message>[\s\S]*?)\n#REFS#\s+(?<tagRefs>.+)?\n#AUTHOR-NAME# (?<authorName>.+)?\n#AUTHOR-EMAIL# (?<authorEmail>.+)?\n#AUTHOR-DATE# (?<authorTs>.+)?\n#COMMITTER-NAME# (?<committerName>.+)?\n#COMMITTER-EMAIL# (?<committerEmail>.+)?\n#COMMITTER-DATE# (?<committerTs>.+)?\n#GPGSIG-CODE# (?<gpgSigCode>.+)?\n#GPGSIG-KEYID# (?<gpgSigKeyId>.+)?/g,F=e=>{let t=a(`git remote get-url origin`,{encoding:`utf8`}).trim(),n=e.exec(t);if(!n?.groups)throw Error(`Couldn't parse remote URL: `+t);let{host:r,owner:i,name:o}=n.groups,s=`https://${r}/${i}/${o}`;return{host:r,owner:i,name:o,homepage:s}},I=(e,t)=>{let n=L(),r=R(t),i,o;if(e===`all`)i=n,o=`HEAD`;else if(e===`unreleased`)i=r[0]??n,o=`HEAD`;else if(e===`latest-release`)i=r[1]??n,o=r[0]?r[0]+`^`:`HEAD`;else if(`from`in e||`to`in e){let t=`from`in e?e.from:`firstCommit`;i=t===`firstCommit`?n:t,o=`to`in e?e.to:`HEAD`}else if(`versionTag`in e){let t=r.indexOf(e.versionTag);if(t===-1)throw Error(`Version tag '${e.versionTag}' not found`);i=r[t],o=r[t+1]??`HEAD`}else throw Error(`Invalid commit range provided`);let s=a(`git log "${i}..${o}" --pretty="##COMMIT##%n#HASH# %h%n#MSG# %B%n#REFS# %d%n#AUTHOR-NAME# %an%n#AUTHOR-EMAIL# %ae%n#AUTHOR-DATE# %at%n#COMMITTER-NAME# %cn%n#COMMITTER-EMAIL# %ce%n#COMMITTER-DATE# %ct%n#GPGSIG-CODE# %G?%n#GPGSIG-KEYID# %GK%n"`,{encoding:`utf8`});return[...s.matchAll(P)].map(e=>e.groups)},L=()=>a(`git rev-list --max-parents=0 HEAD`,{encoding:`utf8`}).trim(),R=e=>{let t=a(`git tag --sort=-creatordate`,{encoding:`utf8`});return t.split(`
|
|
13
|
+
`).filter(t=>e.test(t))};var z=`{{#>header}}
|
|
14
|
+
##   [\` 📦 {{tag}} \`]({{repo.homepage}}/{{#if prevTag}}compare/{{prevTag}}...{{tag}}{{else}}commits/{{tag}}{{/if}})
|
|
511
15
|
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
const remoteUrlMatch = remoteUrlPattern.exec(remoteUrl);
|
|
519
|
-
if (!remoteUrlMatch?.groups) throw new Error(`Couldn't parse remote URL: ` + remoteUrl);
|
|
520
|
-
const { host, owner, name } = remoteUrlMatch.groups;
|
|
521
|
-
const homepage = `https://${host}/${owner}/${name}`;
|
|
522
|
-
return {
|
|
523
|
-
host,
|
|
524
|
-
owner,
|
|
525
|
-
name,
|
|
526
|
-
homepage
|
|
527
|
-
};
|
|
528
|
-
};
|
|
529
|
-
const getRawCommits = (commitRange, prevReleaseTagPattern) => {
|
|
530
|
-
const firstCommitHash = getFirstCommitHash();
|
|
531
|
-
const versionTags = getVersionTags(prevReleaseTagPattern);
|
|
532
|
-
let from, to;
|
|
533
|
-
if (typeof commitRange === "string") {
|
|
534
|
-
from = {
|
|
535
|
-
all: firstCommitHash,
|
|
536
|
-
unreleased: versionTags[0] ?? firstCommitHash
|
|
537
|
-
}[commitRange];
|
|
538
|
-
if (!from) throw new Error(`Invalid commit range: '${commitRange}'`);
|
|
539
|
-
to = "HEAD";
|
|
540
|
-
} else if ("from" in commitRange || "to" in commitRange) {
|
|
541
|
-
const fromValue = "from" in commitRange ? commitRange.from : "firstCommit";
|
|
542
|
-
from = fromValue === "firstCommit" ? firstCommitHash : fromValue;
|
|
543
|
-
to = "to" in commitRange ? commitRange.to : "HEAD";
|
|
544
|
-
} else if ("versionTag" in commitRange) {
|
|
545
|
-
const targetTagIndex = versionTags.indexOf(commitRange.versionTag);
|
|
546
|
-
if (targetTagIndex === -1) throw new Error(`Version tag '${commitRange.versionTag}' not found`);
|
|
547
|
-
from = versionTags[targetTagIndex];
|
|
548
|
-
to = versionTags[targetTagIndex + 1] ?? "HEAD";
|
|
549
|
-
} else throw new Error(`Invalid commit range provided`);
|
|
550
|
-
const gitLogCommits = execSync(`git log ${from}..${to} --pretty="${commitLogFormat}"`, { encoding: "utf8" });
|
|
551
|
-
return [...gitLogCommits.matchAll(rawCommitPattern)].map((m) => m.groups);
|
|
552
|
-
};
|
|
553
|
-
const getFirstCommitHash = () => execSync("git rev-list --max-parents=0 HEAD", { encoding: "utf8" }).trim();
|
|
554
|
-
const getVersionTags = (tagPattern) => {
|
|
555
|
-
const rawTags = execSync("git tag --sort=-creatordate", { encoding: "utf8" });
|
|
556
|
-
return rawTags.split("\n").filter((tag$1) => tagPattern.test(tag$1));
|
|
557
|
-
};
|
|
16
|
+
{{/header}}
|
|
17
|
+
{{#each commitGroups}}
|
|
18
|
+
### {{{repeat ' ' 5}}}{{title}}
|
|
19
|
+
{{#each commits}}
|
|
20
|
+
* {{#if scope}}\`{{scope}}\` {{/if}}{{{subject}}} {{#if ../../commitHyperlink}}[\`{{hash}}\`]({{../../repo.homepage}}/commit/{{hash}}){{else}}{{hash}}{{/if}}
|
|
21
|
+
{{/each}}
|
|
558
22
|
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
var release_default = "{{#>header}}\n##   [` 📦 {{tag}} `]({{repo.homepage}}/{{#if prevTag}}compare/{{prevTag}}...{{tag}}{{else}}commits/{{tag}}{{/if}})\n\n{{/header}}\n{{#each commitGroups}}\n### {{{repeat ' ' 5}}}{{title}}\n{{#each commits}}\n* {{#if scope}}`{{scope}}` {{/if}}{{{subject}}} {{#if ../../commitHyperlink}}[`{{hash}}`]({{../../repo.homepage}}/commit/{{hash}}){{else}}{{hash}}{{/if}}\n{{/each}}\n\n{{/each}}\n#####    🔗 [Full Commit History: {{#if prevTag}}`{{prevTag}}` → `{{tag}}`]({{repo.homepage}}/compare/{{prevTag}}...{{tag}}){{else}}`{{tag}}`]({{repo.homepage}}/commits/{{tag}}){{/if}}  /  _{{date}}_\n\n\n";
|
|
23
|
+
{{/each}}
|
|
24
|
+
#####    🔗 [Full Commit History: {{#if prevTag}}\`{{prevTag}}\` → \`{{tag}}\`]({{repo.homepage}}/compare/{{prevTag}}...{{tag}}){{else}}\`{{tag}}\`]({{repo.homepage}}/commits/{{tag}}){{/if}}  /  _{{date}}_
|
|
562
25
|
|
|
563
|
-
//#endregion
|
|
564
|
-
//#region src/lifecycles/changelog.ts
|
|
565
|
-
const changelog = (config) => {
|
|
566
|
-
if (!config.changelog) return;
|
|
567
|
-
const options = config.changelog;
|
|
568
|
-
const releases = config.context.releases;
|
|
569
|
-
if (!releases) return;
|
|
570
|
-
const versionTags = getVersionTags(config.prevReleaseTagPattern);
|
|
571
|
-
Handlebars.registerPartial(options.compiledPartials);
|
|
572
|
-
Handlebars.registerHelper(options.helpers);
|
|
573
|
-
let result = options.header;
|
|
574
|
-
releases.forEach((release, index) => {
|
|
575
|
-
const prevRelease = releases[index + 1];
|
|
576
|
-
let prevTag, prevVersion;
|
|
577
|
-
if (prevRelease) {
|
|
578
|
-
prevTag = prevRelease.tag;
|
|
579
|
-
prevVersion = getVersionFromTag(prevTag, config.prevReleaseTagPattern);
|
|
580
|
-
} else {
|
|
581
|
-
const targetTagIndex = versionTags.indexOf(release.tag);
|
|
582
|
-
if (targetTagIndex === -1) {
|
|
583
|
-
prevTag = config.context.currentTag;
|
|
584
|
-
prevVersion = getVersionFromTag(prevTag, config.prevReleaseTagPattern);
|
|
585
|
-
} else {
|
|
586
|
-
prevTag = versionTags[targetTagIndex + 1];
|
|
587
|
-
prevVersion = prevTag && getVersionFromTag(prevTag, config.prevReleaseTagPattern);
|
|
588
|
-
}
|
|
589
|
-
}
|
|
590
|
-
const releaseContext = {
|
|
591
|
-
...release,
|
|
592
|
-
...config.context,
|
|
593
|
-
prevTag,
|
|
594
|
-
prevVersion
|
|
595
|
-
};
|
|
596
|
-
const rendered = renderTemplate(release_default, releaseContext);
|
|
597
|
-
result += rendered;
|
|
598
|
-
});
|
|
599
|
-
if (options.stdout) console.log(`Generated changelog:\n${result}`);
|
|
600
|
-
if (options.outputFile) {
|
|
601
|
-
console.log(`Writing changelog to file '${options.outputFile}'`);
|
|
602
|
-
if (!config.dryRun) writeToChangelogFile(options.outputFile, result, options.prevReleaseHeaderPattern);
|
|
603
|
-
}
|
|
604
|
-
};
|
|
605
|
-
const writeToChangelogFile = (outputFile, content, prevReleaseHeaderPattern) => {
|
|
606
|
-
const changelogContent = existsSync(outputFile) ? readFileSync(outputFile, { encoding: "utf8" }) : "";
|
|
607
|
-
const prevReleaseStart = changelogContent.search(prevReleaseHeaderPattern);
|
|
608
|
-
const headlessChangelog = changelogContent.slice(prevReleaseStart);
|
|
609
|
-
const newChangelog = content + headlessChangelog;
|
|
610
|
-
writeFileSync(outputFile, newChangelog, { encoding: "utf8" });
|
|
611
|
-
};
|
|
612
|
-
const renderTemplate = (template, releaseContext) => {
|
|
613
|
-
const compile = Handlebars.compile(template);
|
|
614
|
-
return compile(releaseContext);
|
|
615
|
-
};
|
|
616
|
-
const getVersionFromTag = (tag$1, tagPattern) => {
|
|
617
|
-
return tagPattern.exec(tag$1)?.groups?.version;
|
|
618
|
-
};
|
|
619
26
|
|
|
620
|
-
|
|
621
|
-
//#region src/lifecycles/commit.ts
|
|
622
|
-
const commit = (config) => {
|
|
623
|
-
if (!config.commit) return;
|
|
624
|
-
const options = config.commit;
|
|
625
|
-
console.log("Committing with options:", options);
|
|
626
|
-
if (options.stageAll && !config.dryRun) execSync("git add -A", { stdio: "inherit" });
|
|
627
|
-
if (!config.dryRun) execSync(`git commit -m "${options.message}" ${options.signOff ? "-s" : ""} ${options.gpgSign ? "-S" : ""} ${options.extraArgs}`, { stdio: "inherit" });
|
|
628
|
-
};
|
|
629
|
-
|
|
630
|
-
//#endregion
|
|
631
|
-
//#region src/lifecycles/tag.ts
|
|
632
|
-
const tag = (config) => {
|
|
633
|
-
if (!config.tag) return;
|
|
634
|
-
const options = config.tag;
|
|
635
|
-
console.log("Tagging with options:", options);
|
|
636
|
-
if (!config.dryRun) execSync(`git tag -a ${options.name} -m "${options.message}" ${options.gpgSign ? "-s" : ""} ${options.force ? "-f" : ""} ${options.extraArgs}`, { stdio: "inherit" });
|
|
637
|
-
};
|
|
638
|
-
|
|
639
|
-
//#endregion
|
|
640
|
-
//#region src/relion.ts
|
|
641
|
-
async function relion(userConfig) {
|
|
642
|
-
const config = await resolveConfig(userConfig);
|
|
643
|
-
bump(config);
|
|
644
|
-
changelog(config);
|
|
645
|
-
commit(config);
|
|
646
|
-
tag(config);
|
|
647
|
-
}
|
|
648
|
-
const defineConfig = (config) => config;
|
|
649
|
-
|
|
650
|
-
//#endregion
|
|
651
|
-
export { relion as default, defineConfig };
|
|
652
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","names":["defaultConfig: MergedConfig","defaultChangelogSections: DefaultChangelogSections","commit","defaultChangelogOptions: CompleteChangelogOptions","defaultCommitOptions: CompleteCommitOptions","defaultTagOptions: CompleteTagOptions","defaultVersionedFiles: DefaultVersionedFile[]","result: Required<T>","bump","commit","releases: Record<string, ReleaseWithFlatCommits>","tag","commitGroups: Record<string, Commit[]>","commits","releaseType: ReleaseType","commit","commit","authors: Contributor[]","dateParts: Record<string, string>","from: string","to: string","tag","prevTag: string | undefined","prevVersion: string | undefined","releaseContext: ReleaseContext","releaseTemplate","tag"],"sources":["../src/lifecycles/bump.ts","../src/defaults.ts","../src/utils/config-resolver.ts","../src/utils/version-manager.ts","../src/enums.ts","../src/utils/commits-parser.ts","../src/utils/git-utils.ts","../src/templates/release.hbs","../src/lifecycles/changelog.ts","../src/lifecycles/commit.ts","../src/lifecycles/tag.ts","../src/relion.ts"],"sourcesContent":["import { readFileSync, writeFileSync } from 'node:fs'\nimport type { ResolvedConfig } from '@/types'\n\nexport const bump = (config: ResolvedConfig): void => {\n\tif (!config.bump) return\n\tconst bumpFiles = config.bump, newVersion = config.context.newVersion\n\tbumpFiles.forEach((versionedFile) => {\n\t\tconst fileContent = readFileSync(versionedFile.filePath, 'utf8')\n\t\tconst updatedContent = fileContent.replace(versionedFile.versionPattern, `$1${newVersion}$3`)\n\t\tif (!config.dryRun) writeFileSync(versionedFile.filePath, updatedContent, 'utf8')\n\t\tconsole.log(`Updated version in '${versionedFile.filePath}' to '${newVersion}'`)\n\t})\n}","import type { MergedConfig, DefaultVersionedFile, CompleteChangelogOptions, CompleteCommitOptions, CompleteTagOptions, DefaultChangelogSections } from '@/types'\n\nexport const defaultConfig: MergedConfig = {\n\tbump: false,\n\tchangelog: false,\n\tcommit: false,\n\ttag: false,\n\tversionSourceFile: './package.json',\n\tnewTagFormat: 'v{{newVersion}}',\n\tprevReleaseTagPattern: /^v(?<version>\\d+\\.\\d+\\.\\d+)/,\n\tzeroMajorBreakingIsMinor: true,\n\tdryRun: false,\n\tcontext: {\n\t\tcommitHyperlink: true,\n\t\trefHyperlink: true,\n\t},\n\tcommitsParser: {\n\t\theaderPattern: /^(?<type>\\w+)(?:\\((?<scope>.+)\\))?(?<bang>!)?: (?<subject>.+)/s,\n\t\tbreakingChangesPattern: /BREAKING CHANGES?:\\s*(?<content>.+)/s,\n\t\tbreakingChangeListPattern: /- (.+)/g,\n\t\ttagPattern: /tag: (?<tag>.*?)[,)]/g,\n\t\tcoAuthorPattern: /Co-authored-by: (?<name>.+?) <(?<email>.+)>/g,\n\t\tsignerPattern: /Signed-off-by: (?<name>.+?) <(?<email>.+)>/g,\n\t\tghEmailPattern: /^(?:\\d+\\+)?(?<username>.+)@users\\.noreply\\.github\\.com$/,\n\t\tremoteUrlPattern: /^(https:\\/\\/|git@)(?<host>[^/:]+)[/:](?<owner>.+?)\\/(?<name>.+?)(?:\\..*)?$/,\n\t\trefPattern: /^(?<action>.+?) (?<labels>.+)$/gm,\n\t\trefLabelPattern: /(?:(?<owner>\\S+?)\\/(?<repo>\\S+?))?#(?<number>\\d+)/g,\n\t\trefActionPattern: /Fixes|Closes|Refs/i,\n\t\tdateSource: 'authorDate',\n\t\tdateFormat: 'YYYY-MM-DD',\n\t},\n}\n\nexport const defaultChangelogSections: DefaultChangelogSections = {\n\tbreaking: { title: '⚠️ BREAKING CHANGES', commitType: 'breaking' },\n\tfeat: { title: '✨ Features', commitType: 'feat' },\n\tfix: { title: '🩹 Fixes', commitType: 'fix' },\n\tperf: { title: '⚡ Performance', commitType: 'perf' },\n\trefactor: { title: '🚜 Refactoring', commitType: 'refactor' },\n\tdocs: { title: '📚 Documentation', commitType: 'docs' },\n\tstyle: { title: '🎨 Formatting', commitType: 'style' },\n\tbuild: { title: '📦 Build', commitType: 'build' },\n\tci: { title: '🚀 CI', commitType: 'ci' },\n\trevert: { title: '♻️ Reverts', commitType: 'revert' },\n\tdeps: { title: '🧩 Dependencies', commitType: 'chore',\n\t\tfilter: commit => !!commit.scope?.includes('deps') },\n\tchore: { title: '🛠️ Chores', commitType: 'chore' },\n\ttest: { title: '🧪 Tests', commitType: 'test' },\n\tmisc: { title: '⚙️ Miscellaneous', commitType: '*' },\n\t[Symbol.iterator]() {\n\t\treturn Object.values(this)[Symbol.iterator]()\n\t},\n}\n\nexport const defaultChangelogOptions: CompleteChangelogOptions = {\n\tstdout: false,\n\toutputFile: './CHANGELOG.md',\n\tcommitRange: 'unreleased',\n\tsections: [...defaultChangelogSections],\n\theader: '# Changelog\\n\\n\\n',\n\tprevReleaseHeaderPattern: /^##.*?\\d+\\.\\d+\\.\\d+/m,\n\thelpers: {\n\t\trepeat: (string: string, n: number) => string.repeat(n),\n\t},\n\tpartials: {},\n}\n\nexport const defaultCommitOptions: CompleteCommitOptions = {\n\tmessage: 'release({{repo.name}}): {{newTag}}',\n\tsignOff: false,\n\tgpgSign: false,\n\tstageAll: true,\n\textraArgs: '',\n}\n\nexport const defaultTagOptions: CompleteTagOptions = {\n\tname: '{{newTag}}',\n\tmessage: 'release({{repo.name}}): {{newTag}}',\n\tgpgSign: false,\n\tforce: false,\n\textraArgs: '',\n}\n\nexport const defaultVersionedFiles: DefaultVersionedFile[] = [\n\t{\n\t\tfilePathRegex: /package\\.json$/,\n\t\tversionPattern: /(^.*?\"version\".*?\")(.*?)(\")/s,\n\t},\n\t{\n\t\tfilePathRegex: /package-lock\\.json$/,\n\t\tversionPattern: /(^.*?\"version\".*?\"|\"packages\".*?\"\".*\"version\".*?\")(.*?)(\")/gs,\n\t},\n]","import { parseVersion, determineNextVersion, getVersionTags, getRepoInfo, parseCommits } from '@/utils'\nimport type { UserConfig, ResolvedConfig, TransformedConfig, VersionedFile, MergedConfig, ResolvedContext, FalseOrComplete, ContextualConfig, ResolvedChangelogSection, ChangelogSectionDefinition, Commit, ReleaseWithFlatCommits, ReleaseWithGroupedCommits } from '@/types'\nimport { defaultConfig, defaultVersionedFiles, defaultChangelogOptions, defaultCommitOptions, defaultTagOptions } from '@/defaults'\nimport Handlebars from 'handlebars'\n\nexport const resolveConfig = async (userConfig: UserConfig): Promise<ResolvedConfig> => {\n\tconst profileMergedConfig = mergeProfileConfig(userConfig)\n\tconst mergedConfig = mergeWithDefaults(profileMergedConfig)\n\tconst transformedConfig = transformVersionedFiles(mergedConfig)\n\tconst contextualConfig = await fillContext(transformedConfig)\n\tconst finalConfig = resolveTemplates(contextualConfig)\n\treturn finalConfig\n}\n\nconst mergeProfileConfig = (baseConfig: UserConfig): UserConfig => {\n\tconst profileName = baseConfig.profile\n\tif (!profileName) return baseConfig\n\n\tconst profileConfig = baseConfig[`_${profileName}`]\n\tif (!profileConfig) throw new Error(`Profile \"${profileName}\" not found in configuration.`)\n\n\tconst mergeOption = <T extends keyof UserConfig>(propKey: T, ...nestedPropKeys: string[]): UserConfig[T] => {\n\t\ttype PlainObject = Record<string, unknown>\n\n\t\tconst isPlainObject = (value: unknown): value is PlainObject =>\n\t\t\tObject.prototype.toString.call(value) === '[object Object]'\n\n\t\tconst mergeObjects = (baseObj: unknown, overrideObject: unknown): unknown => {\n\t\t\tconst isBasePlainObject = isPlainObject(baseObj)\n\t\t\tconst isOverridePlainObject = isPlainObject(overrideObject)\n\t\t\tif (isBasePlainObject && isOverridePlainObject) {\n\t\t\t\treturn { ...baseObj, ...overrideObject }\n\t\t\t} else if (!isBasePlainObject && isOverridePlainObject) {\n\t\t\t\treturn overrideObject\n\t\t\t} else if (isBasePlainObject && !isOverridePlainObject) {\n\t\t\t\treturn baseObj\n\t\t\t}\n\t\t}\n\n\t\tconst baseConfigProp = baseConfig[propKey] as PlainObject | undefined\n\t\tconst profileConfigProp = profileConfig[propKey] as PlainObject | undefined\n\t\tconst result = mergeObjects(baseConfigProp, profileConfigProp) as PlainObject | undefined\n\t\tif (result === undefined) return undefined\n\n\t\tnestedPropKeys.forEach((key) => {\n\t\t\tresult[key] = mergeObjects(baseConfigProp?.[key], profileConfigProp?.[key])\n\t\t})\n\n\t\treturn result as UserConfig[T]\n\t}\n\n\treturn {\n\t\t...baseConfig, ...profileConfig,\n\t\tcommitsParser: mergeOption('commitsParser'),\n\t\tchangelog: mergeOption('changelog', 'partials', 'helpers'),\n\t\tcommit: mergeOption('commit'),\n\t\ttag: mergeOption('tag'),\n\t\tcontext: mergeOption('context'),\n\t}\n}\n\nconst mergeWithDefaults = (userConfig: UserConfig): MergedConfig => {\n\tconst resolveOptions = <T>(optionsName: string, options: boolean | T | undefined, defaults: Required<T>, ...subObjects: (keyof T)[]): FalseOrComplete<T> => {\n\t\tif (options == undefined || options === false) return false\n\t\tif (options === true) return defaults\n\t\tif (typeof options !== 'object') throw new Error(`Invalid value for ${optionsName}. It should be a boolean or an object.`)\n\t\tconst result: Required<T> = { ...defaults, ...options }\n\t\tsubObjects.forEach(subObjectKey => result[subObjectKey] = { ...defaults[subObjectKey], ...options[subObjectKey] })\n\t\treturn result\n\t}\n\n\treturn {\n\t\t...defaultConfig, ...userConfig,\n\t\tcommitsParser: { ...defaultConfig.commitsParser, ...userConfig.commitsParser },\n\t\tchangelog: resolveOptions('changelog', userConfig.changelog, defaultChangelogOptions, 'partials', 'helpers'),\n\t\tcommit: resolveOptions('commit', userConfig.commit, defaultCommitOptions),\n\t\ttag: resolveOptions('tag', userConfig.tag, defaultTagOptions),\n\t}\n}\n\nconst transformVersionedFiles = (config: MergedConfig): TransformedConfig => {\n\tconst resolveVersionedFile = (filePath: string): VersionedFile => {\n\t\tconst matchingDefaultVersionFile = defaultVersionedFiles.find(defaultFile =>\n\t\t\tdefaultFile.filePathRegex.test(filePath))\n\t\tif (matchingDefaultVersionFile) {\n\t\t\treturn { filePath, versionPattern: matchingDefaultVersionFile.versionPattern }\n\t\t} else {\n\t\t\tthrow new Error(\n\t\t\t\t`File ${filePath} doesn't match any default versioned files. `\n\t\t\t\t+ 'Please provide a custom version pattern for this file.',\n\t\t\t)\n\t\t}\n\t}\n\n\tconst resolveBump = (bump: MergedConfig['bump']): false | VersionedFile[] => {\n\t\tif (bump === false) return false\n\t\tif (bump === true) return [versionSourceFile]\n\t\tif (Array.isArray(bump)) return [\n\t\t\tversionSourceFile,\n\t\t\t...(bump.map(bumpFile =>\n\t\t\t\ttypeof bumpFile === 'string' ? resolveVersionedFile(bumpFile) : bumpFile,\n\t\t\t)),\n\t\t]\n\t\tthrow new Error('Invalid value for bump. It should be a boolean or an array.')\n\t}\n\n\tconst versionSourceFile = typeof config.versionSourceFile === 'string'\n\t\t? resolveVersionedFile(config.versionSourceFile)\n\t\t: config.versionSourceFile\n\n\treturn {\n\t\t...config,\n\t\tversionSourceFile,\n\t\tbump: resolveBump(config.bump),\n\t\tchangelog: config.changelog === false\n\t\t\t? false\n\t\t\t: { ...config.changelog,\n\t\t\t\tcompiledPartials: Object.fromEntries(Object.entries(config.changelog.partials).map(([key, template]) => [key, Handlebars.compile(template)])),\n\t\t\t},\n\t}\n}\n\nconst fillContext = async (config: TransformedConfig): Promise<ContextualConfig> => {\n\tconst resolvedContext = (config.context ?? {}) as Partial<ResolvedContext>\n\n\tconst repoInfo = getRepoInfo(config.commitsParser.remoteUrlPattern)\n\tresolvedContext.repo = { ...repoInfo, ...resolvedContext.repo }\n\n\tconst commitRange = config.changelog ? config.changelog.commitRange : 'unreleased'\n\n\tresolvedContext.commits = config.context?.commits\n\t\t? await Promise.all(config.context.commits.map(async (commit) => {\n\t\t\treturn ((typeof commit === 'object' && 'message' in commit) || typeof commit === 'string')\n\t\t\t\t? (await parseCommits([commit], config.commitsParser, config.prevReleaseTagPattern))[0]\n\t\t\t\t: commit\n\t\t}))\n\t\t: await parseCommits(commitRange, config.commitsParser, config.prevReleaseTagPattern)\n\n\tresolvedContext.currentVersion ??= parseVersion(config.versionSourceFile)\n\tresolvedContext.currentTag ??= getVersionTags(config.prevReleaseTagPattern)[0]\n\tresolvedContext.newVersion ??= await determineNextVersion(config, resolvedContext.currentVersion)\n\tresolvedContext.newTag ??= compileTemplate(config.newTagFormat, resolvedContext as ResolvedContext)\n\n\tconst contextualConfig = { ...config, context: resolvedContext as ResolvedContext }\n\n\tresolvedContext.releases = config.changelog\n\t\t? groupCommitsByReleases(resolvedContext.commits, config.changelog.sections, contextualConfig)\n\t\t: null\n\n\treturn contextualConfig\n}\n\nconst groupCommitsByReleases = (commits: Commit[], sections: ChangelogSectionDefinition[], config: ContextualConfig): ReleaseWithGroupedCommits[] => {\n\tconst releases: Record<string, ReleaseWithFlatCommits> = {}\n\n\tcommits.forEach((commit) => {\n\t\tconst releaseTag = commit.tags?.find(tag => config.prevReleaseTagPattern.test(tag))\n\t\tif (releaseTag) {\n\t\t\treleases[releaseTag] ??= {\n\t\t\t\ttag: releaseTag,\n\t\t\t\tversion: config.prevReleaseTagPattern.exec(releaseTag)?.groups?.version,\n\t\t\t\tdate: commit.date,\n\t\t\t\tcommits: [commit],\n\t\t\t}\n\t\t} else {\n\t\t\tconst latestReleaseTag = Object.keys(releases).at(-1)\n\t\t\tif (latestReleaseTag) {\n\t\t\t\treleases[latestReleaseTag].commits.push(commit)\n\t\t\t} else {\n\t\t\t\treleases[config.context.newTag] = {\n\t\t\t\t\ttag: config.context.newTag,\n\t\t\t\t\tversion: config.context.newVersion,\n\t\t\t\t\tdate: commit.date,\n\t\t\t\t\tcommits: [commit],\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t})\n\n\treturn Object.values(releases).map(release => groupReleaseCommitsBySections(release, sections))\n}\n\nconst groupReleaseCommitsBySections = (release: ReleaseWithFlatCommits, sections: ChangelogSectionDefinition[]): ReleaseWithGroupedCommits => {\n\tconst { commits, ...releaseWithoutCommits } = release\n\treturn {\n\t\t...releaseWithoutCommits,\n\t\tcommitGroups: groupCommitsBySections(commits, sections),\n\t}\n}\n\nconst groupCommitsBySections = (commits: Commit[], sections: ChangelogSectionDefinition[]): ResolvedChangelogSection[] => {\n\tconst commitGroups: Record<string, Commit[]> = Object.fromEntries(sections.map(section => [section.title, []]))\n\n\tcommits.forEach((commit) => {\n\t\tconst isBreaking = !!commit.breakingChanges\n\t\tlet isGrouped = false\n\t\tlet isBreakingGrouped = false\n\n\t\tfor (const section of sections) {\n\t\t\tif (section.filter && !section.filter(commit)) continue\n\n\t\t\tconst sectionTypes = [section.commitType].flat()\n\n\t\t\tif (isBreaking && !isBreakingGrouped && sectionTypes.includes('breaking')) {\n\t\t\t\tcommitGroups[section.title].push(commit)\n\t\t\t\tisBreakingGrouped = true\n\t\t\t\tcontinue\n\t\t\t}\n\n\t\t\tif (!isGrouped && (sectionTypes.includes(commit.type) || sectionTypes.includes('*'))) {\n\t\t\t\tcommitGroups[section.title].push(commit)\n\t\t\t\tisGrouped = true\n\t\t\t}\n\n\t\t\tif (isGrouped && (!isBreaking || isBreakingGrouped)) return\n\t\t}\n\t})\n\n\tObject.keys(commitGroups).forEach((key) => {\n\t\tif (!commitGroups[key].length) delete commitGroups[key]\n\t})\n\n\treturn Object.entries(commitGroups).map(([title, commits]) => ({ title, commits }))\n}\n\nconst resolveTemplates = (config: ContextualConfig): ResolvedConfig => {\n\treturn {\n\t\t...config,\n\t\tcommit: config.commit\n\t\t\t? { ...config.commit, message: compileTemplate(config.commit.message, config.context) }\n\t\t\t: config.commit,\n\t\ttag: config.tag\n\t\t\t? {\n\t\t\t\t...config.tag,\n\t\t\t\tname: compileTemplate(config.tag.name, config.context),\n\t\t\t\tmessage: compileTemplate(config.tag.message, config.context),\n\t\t\t}\n\t\t\t: config.tag,\n\t}\n}\n\nconst compileTemplate = (value: string, context: ResolvedContext): string => {\n\tconst compile = Handlebars.compile(value)\n\treturn compile(context)\n}","import { readFileSync } from 'node:fs'\nimport semver from 'semver'\nimport type { ReleaseType, Commit, VersionedFile, TransformedConfig } from '@/types'\nimport { parseCommits } from '@/utils'\n\nexport const parseVersion = (versionedFile: VersionedFile): string => {\n\tconst fileContent = readFileSync(versionedFile.filePath, 'utf8')\n\tconst version = versionedFile.versionPattern.exec(fileContent)?.[2]\n\tif (!version) throw new Error(`Version not found in '${versionedFile.filePath}' with pattern '${versionedFile.versionPattern}'`)\n\tif (!semver.valid(version)) throw new Error(`Invalid version format in '${versionedFile.filePath}': '${version}'`)\n\tconsole.log(`Current version from '${versionedFile.filePath}': '${version}'`)\n\treturn version\n}\n\nexport const determineNextVersion = async (config: TransformedConfig, currentVersion: string): Promise<string> => {\n\tif (config.releaseVersion) {\n\t\tif (!semver.valid(config.releaseVersion)) {\n\t\t\tthrow new Error(`Invalid release version format: '${config.releaseVersion}'`)\n\t\t}\n\t\treturn config.releaseVersion\n\t}\n\tlet releaseType: ReleaseType\n\tif (config.releaseType) {\n\t\treleaseType = config.releaseType\n\t} else {\n\t\tconst unreleasedCommits = await parseCommits('unreleased', config.commitsParser, config.prevReleaseTagPattern)\n\t\treleaseType = calculateReleaseType(unreleasedCommits)\n\t\tif (config.zeroMajorBreakingIsMinor && semver.major(currentVersion) === 0 && releaseType === 'major') releaseType = 'minor'\n\t}\n\tconst newVersion = increaseVersion(currentVersion, releaseType)\n\tconsole.log(`Determined new version: '${newVersion}' (release type: '${releaseType}')`)\n\treturn newVersion\n}\n\nconst calculateReleaseType = (commits: Commit[]): ReleaseType => {\n\tconst hasBreakingChange = commits.some(commit => commit.breakingChanges)\n\tif (hasBreakingChange) return 'major'\n\n\tconst hasFeature = commits.some(commit => commit.type === 'feat')\n\tif (hasFeature) return 'minor'\n\n\treturn 'patch'\n}\n\nconst increaseVersion = (currentVersion: string, releaseType: ReleaseType): string =>\n\tsemver.inc(currentVersion, releaseType) ?? (() => {\n\t\tthrow new Error(`Failed to calculate new version from '${currentVersion}' with release type '${releaseType}'`)\n\t})()","export enum GpgSigLabel {\n\tG = 'valid',\n\tB = 'bad',\n\tU = 'valid, unknown validity',\n\tX = 'valid, expired',\n\tY = 'valid, made by expired key',\n\tR = 'valid, made by revoked key',\n\tE = 'cannot check (missing key)',\n\tN = 'no signature',\n}\n\nexport enum RefType {\n\tissue = 'issue',\n\tpr = 'PR',\n}","import type { CompleteCommitsParser, Commit, RawCommit, RawReference, RefLabel, Reference, Contributor, CommitMessage, CommitRange } from '@/types'\nimport { GpgSigLabel } from '@/enums'\nimport { getRawCommits } from '@/utils'\n\n// const commitStore: Record<string, Commit> = {}\n\nexport const parseCommits = async (arg1: CommitRange | RawCommit[], commitsParser: CompleteCommitsParser, prevReleaseTagPattern: RegExp): Promise<Commit[]> => {\n\tconst rawCommits = Array.isArray(arg1) ? arg1 : getRawCommits(arg1, prevReleaseTagPattern)\n\tconst parser = commitsParser\n\n\treturn (await Promise.all(rawCommits.map(async (commit) => {\n\t\tif (typeof commit === 'string') commit = { message: commit }\n\n\t\tconst { hash, tagRefs } = commit\n\n\t\tconst message = commit.message.trim()\n\t\tif (!message) throw new Error(`Message is missing for commit: ${JSON.stringify(commit)}`)\n\n\t\tlet parsedMessage\n\t\ttry {\n\t\t\tparsedMessage = parseCommitMessage(message, parser)\n\t\t} catch (error) {\n\t\t\tconsole.warn(`Error parsing commit '${hash ?? '<no hash>'}':`, (error as Error).message)\n\t\t\treturn\n\t\t}\n\t\tconst { type, scope, subject, body, breakingChanges, footer } = parsedMessage\n\t\tconst tags = tagRefs ? [...tagRefs.matchAll(parser.tagPattern)].map(m => m.groups?.tag ?? '') : []\n\n\t\tconst signers = footer\n\t\t\t? [...footer.matchAll(parser.signerPattern)].map(m => m.groups as unknown as Contributor)\n\t\t\t: []\n\n\t\tconst authors: Contributor[] = []\n\t\tconst addAuthor = (contributor: Contributor): void => {\n\t\t\tif (!authors.some(a => a.email === contributor.email)) authors.push(contributor)\n\t\t}\n\n\t\tconst author = commit.authorName && commit.authorEmail\n\t\t\t? getContributorDetails({ name: commit.authorName, email: commit.authorEmail }, signers)\n\t\t\t: undefined\n\t\tif (author) addAuthor(author)\n\n\t\tconst committer = commit.committerName && commit.committerEmail\n\t\t\t? getContributorDetails({ name: commit.committerName, email: commit.committerEmail }, signers)\n\t\t\t: undefined\n\t\tif (committer) addAuthor(committer)\n\n\t\tconst coAuthors = footer\n\t\t\t? [...footer.matchAll(parser.coAuthorPattern)].map(m => m.groups as unknown as Contributor)\n\t\t\t\t.map(coAuthor => getContributorDetails(coAuthor, signers))\n\t\t\t: []\n\t\tcoAuthors.forEach(coAuthor => addAuthor(coAuthor))\n\n\t\tconst refs = await parseRefs((footer ?? ''), parser)\n\n\t\tconst gpgSig = commit.gpgSigCode\n\t\t\t? {\n\t\t\t\tcode: commit.gpgSigCode,\n\t\t\t\tlabel: GpgSigLabel[commit.gpgSigCode],\n\t\t\t\tkeyId: commit.gpgSigKeyId,\n\t\t\t}\n\t\t\t: undefined\n\n\t\tlet date = commit[parser.dateSource === 'committerDate' ? 'committerTs' : 'authorTs']\n\t\tif (typeof date === 'string') date = formatDate(new Date(+date * 1000), parser.dateFormat)\n\n\t\tconst parsedCommit = { hash, type, scope, subject, body, breakingChanges, footer, committer, gpgSig, date,\n\t\t\ttags: tags.length ? tags : undefined,\n\t\t\tauthors: authors.length ? authors : undefined,\n\t\t\trefs: refs.length ? refs : undefined,\n\t\t}\n\t\t// if (hash && !(hash in commitStore)) commitStore[hash] = parsedCommit\n\t\t// console.log(Object.keys(commitStore).length, 'commits in store')\n\t\treturn parsedCommit\n\t}))).filter(commit => commit !== undefined)\n}\n\nconst parseCommitMessage = (message: string, parser: CompleteCommitsParser): CommitMessage => {\n\tconst [header, ...details] = message.split('\\n\\n')\n\n\tconst headerMatch = parser.headerPattern.exec(header)\n\tif (!headerMatch?.groups) throw new Error(`Commit header '${header}' doesn't match expected format`)\n\tconst { type, scope, bang, subject } = headerMatch.groups\n\n\tlet breakingChanges\n\tconst breakingChangesPart = details.find(detail => parser.breakingChangesPattern.test(detail))\n\tif (breakingChangesPart) {\n\t\tbreakingChanges = parseBreakingChanges(breakingChangesPart, parser)\n\t\tdetails.splice(details.indexOf(breakingChangesPart), 1)\n\t} else if (bang) {\n\t\tbreakingChanges = subject\n\t}\n\n\tconst footerStart = details.findIndex(detail =>\n\t\tdetail.match(parser.refActionPattern)\n\t\t?? detail.match(parser.coAuthorPattern)\n\t\t?? detail.match(parser.signerPattern))\n\tconst [body, footer] = footerStart === -1\n\t\t? [details.join('\\n\\n'), '']\n\t\t: [details.slice(0, footerStart).join('\\n\\n'), details.slice(footerStart).join('\\n\\n')]\n\n\treturn {\n\t\ttype,\n\t\tscope: scope || undefined,\n\t\tsubject,\n\t\tbody: body || undefined,\n\t\tbreakingChanges,\n\t\tfooter: footer || undefined,\n\t}\n}\n\nconst parseBreakingChanges = (value: string, parser: CompleteCommitsParser): Commit['breakingChanges'] => {\n\tconst breakingChanges = parser.breakingChangesPattern.exec(value)?.groups?.content\n\tif (!breakingChanges) throw new Error(`Failed to extract breaking changes content from '${value}' using pattern \"${parser.breakingChangesPattern}\"`)\n\n\tconst breakingChangeList = [...breakingChanges.matchAll(parser.breakingChangeListPattern)]\n\n\treturn breakingChangeList.length\n\t\t? breakingChangeList.map(m => m[1])\n\t\t: breakingChanges\n}\n\nconst getContributorDetails = (contributor: Contributor, signers: Contributor[]): Contributor => {\n\tconst hasSignedOff = signers.some(signer => signer.email === contributor.email && signer.name === contributor.name)\n\treturn {\n\t\t...contributor,\n\t\thasSignedOff,\n\t\tghLogin: contributor.name,\n\t\tghUrl: `https://github.com/${contributor.name}`,\n\t}\n}\n\nconst parseRefs = async (value: string, parser: CompleteCommitsParser): Promise<Reference[]> =>\n\tawait Promise.all([...value.matchAll(parser.refPattern)].map(m => m.groups as unknown as RawReference)\n\t\t.filter(rawRef => parser.refActionPattern.test(rawRef.action))\n\t\t.flatMap(rawRef =>\n\t\t\t[...rawRef.labels.matchAll(parser.refLabelPattern)].map(m => m.groups as unknown as RefLabel)\n\t\t\t\t.filter(label => !!label.number)\n\t\t\t\t.map(label => ({\n\t\t\t\t\taction: rawRef.action,\n\t\t\t\t\towner: label.owner,\n\t\t\t\t\trepo: label.repo,\n\t\t\t\t\tnumber: label.number,\n\t\t\t\t})),\n\t\t),\n\t)\n\nconst formatDate = (date: Date, format: string): string => {\n\tconst pad = (num: number) => num.toString().padStart(2, '0')\n\n\tconst dateParts: Record<string, string> = {\n\t\tYYYY: date.getUTCFullYear().toString(),\n\t\tMM: pad(date.getUTCMonth() + 1),\n\t\tDD: pad(date.getUTCDate()),\n\t\tHH: pad(date.getUTCHours()),\n\t\tmm: pad(date.getUTCMinutes()),\n\t\tss: pad(date.getUTCSeconds()),\n\t}\n\n\treturn format.replace(/YYYY|MM|DD|HH|mm|ss/g, k => dateParts[k])\n}","import { execSync } from 'node:child_process'\nimport type { CommitRange, RawCommit, RepoInfo } from '@/types'\n\nconst commitLogFormat = `##COMMIT##%n#HASH# %h%n#MSG# %B%n#REFS# %d%n#AUTHOR-NAME# %an%n#AUTHOR-EMAIL# %ae%n#AUTHOR-DATE# %at%n#COMMITTER-NAME# %cn%n#COMMITTER-EMAIL# %ce%n#COMMITTER-DATE# %ct%n#GPGSIG-CODE# %G?%n#GPGSIG-KEYID# %GK%n`\nconst rawCommitPattern = /##COMMIT##\\n#HASH# (?<hash>.+)?\\n#MSG# (?<message>[\\s\\S]*?)\\n#REFS#\\s+(?<tagRefs>.+)?\\n#AUTHOR-NAME# (?<authorName>.+)?\\n#AUTHOR-EMAIL# (?<authorEmail>.+)?\\n#AUTHOR-DATE# (?<authorTs>.+)?\\n#COMMITTER-NAME# (?<committerName>.+)?\\n#COMMITTER-EMAIL# (?<committerEmail>.+)?\\n#COMMITTER-DATE# (?<committerTs>.+)?\\n#GPGSIG-CODE# (?<gpgSigCode>.+)?\\n#GPGSIG-KEYID# (?<gpgSigKeyId>.+)?/g\n\nexport const getRepoInfo = (remoteUrlPattern: RegExp): RepoInfo => {\n\tconst remoteUrl = execSync('git remote get-url origin', { encoding: 'utf8' }).trim()\n\tconst remoteUrlMatch = remoteUrlPattern.exec(remoteUrl)\n\tif (!remoteUrlMatch?.groups) throw new Error(`Couldn't parse remote URL: ` + remoteUrl)\n\n\tconst { host, owner, name } = remoteUrlMatch.groups\n\tconst homepage = `https://${host}/${owner}/${name}`\n\n\treturn { host, owner, name, homepage }\n}\n\nexport const getRawCommits = (commitRange: CommitRange, prevReleaseTagPattern: RegExp): RawCommit[] => {\n\tconst firstCommitHash = getFirstCommitHash()\n\tconst versionTags = getVersionTags(prevReleaseTagPattern)\n\n\tlet from: string, to: string\n\tif (typeof commitRange === 'string') {\n\t\tfrom = {\n\t\t\tall: firstCommitHash,\n\t\t\tunreleased: versionTags[0] ?? firstCommitHash,\n\t\t}[commitRange]\n\t\tif (!from) throw new Error(`Invalid commit range: '${commitRange}'`)\n\t\tto = 'HEAD'\n\t} else if ('from' in commitRange || 'to' in commitRange) {\n\t\tconst fromValue = 'from' in commitRange ? commitRange.from : 'firstCommit'\n\t\tfrom = fromValue === 'firstCommit' ? firstCommitHash : fromValue\n\t\tto = 'to' in commitRange ? commitRange.to : 'HEAD'\n\t} else if ('versionTag' in commitRange) {\n\t\tconst targetTagIndex = versionTags.indexOf(commitRange.versionTag)\n\t\tif (targetTagIndex === -1) throw new Error(`Version tag '${commitRange.versionTag}' not found`)\n\t\tfrom = versionTags[targetTagIndex]\n\t\tto = versionTags[targetTagIndex + 1] ?? 'HEAD'\n\t} else {\n\t\tthrow new Error(`Invalid commit range provided`)\n\t}\n\n\tconst gitLogCommits = execSync(`git log ${from}..${to} --pretty=\"${commitLogFormat}\"`, { encoding: 'utf8' })\n\treturn [...gitLogCommits.matchAll(rawCommitPattern)].map(m => m.groups as RawCommit)\n}\n\nconst getFirstCommitHash = (): string => execSync('git rev-list --max-parents=0 HEAD', { encoding: 'utf8' }).trim()\n\nexport const getVersionTags = (tagPattern: RegExp): string[] => {\n\tconst rawTags = execSync('git tag --sort=-creatordate', { encoding: 'utf8' })\n\treturn rawTags.split('\\n').filter(tag => tagPattern.test(tag))\n}","\"{{#>header}}\\n## &ensp; [` 📦 {{tag}}  `]({{repo.homepage}}/{{#if prevTag}}compare/{{prevTag}}...{{tag}}{{else}}commits/{{tag}}{{/if}})\\n\\n{{/header}}\\n{{#each commitGroups}}\\n### {{{repeat '&nbsp;' 5}}}{{title}}\\n{{#each commits}}\\n* {{#if scope}}`{{scope}}` {{/if}}{{{subject}}} {{#if ../../commitHyperlink}}[`{{hash}}`]({{../../repo.homepage}}/commit/{{hash}}){{else}}{{hash}}{{/if}}\\n{{/each}}\\n\\n{{/each}}\\n##### &emsp;&ensp;&nbsp; 🔗 [Full Commit History: {{#if prevTag}}`{{prevTag}}` → `{{tag}}`]({{repo.homepage}}/compare/{{prevTag}}...{{tag}}){{else}}`{{tag}}`]({{repo.homepage}}/commits/{{tag}}){{/if}} &ensp;/&ensp; _{{date}}_\\n\\n\\n\"","import { getVersionTags } from '@/utils'\nimport type { ReleaseContext, ReleaseWithGroupedCommits, ResolvedConfig } from '@/types'\nimport { existsSync, readFileSync, writeFileSync } from 'node:fs'\nimport Handlebars from 'handlebars'\nimport releaseTemplate from '@/templates/release.hbs'\n\nexport const changelog = (config: ResolvedConfig): void => {\n\tif (!config.changelog) return\n\n\tconst options = config.changelog\n\tconst releases = config.context.releases\n\tif (!releases) return\n\n\tconst versionTags = getVersionTags(config.prevReleaseTagPattern)\n\n\tHandlebars.registerPartial(options.compiledPartials)\n\tHandlebars.registerHelper(options.helpers)\n\n\tlet result = options.header\n\treleases.forEach((release: ReleaseWithGroupedCommits, index: number) => {\n\t\tconst prevRelease = releases[index + 1] as ReleaseWithGroupedCommits | undefined\n\t\tlet prevTag: string | undefined, prevVersion: string | undefined\n\t\tif (prevRelease) {\n\t\t\tprevTag = prevRelease.tag\n\t\t\tprevVersion = getVersionFromTag(prevTag, config.prevReleaseTagPattern)\n\t\t} else {\n\t\t\tconst targetTagIndex = versionTags.indexOf(release.tag)\n\t\t\tif (targetTagIndex === -1) {\n\t\t\t\tprevTag = config.context.currentTag\n\t\t\t\tprevVersion = getVersionFromTag(prevTag, config.prevReleaseTagPattern)\n\t\t\t} else {\n\t\t\t\tprevTag = versionTags[targetTagIndex + 1]\n\t\t\t\tprevVersion = prevTag && getVersionFromTag(prevTag, config.prevReleaseTagPattern)\n\t\t\t}\n\t\t}\n\t\tconst releaseContext: ReleaseContext = {\n\t\t\t...release,\n\t\t\t...config.context,\n\t\t\tprevTag,\n\t\t\tprevVersion,\n\t\t}\n\t\tconst rendered = renderTemplate(releaseTemplate, releaseContext)\n\t\tresult += rendered\n\t})\n\n\tif (options.stdout) {\n\t\tconsole.log(`Generated changelog:\\n${result}`)\n\t}\n\tif (options.outputFile) {\n\t\tconsole.log(`Writing changelog to file '${options.outputFile}'`)\n\t\tif (!config.dryRun) writeToChangelogFile(options.outputFile, result, options.prevReleaseHeaderPattern)\n\t}\n}\n\nconst writeToChangelogFile = (outputFile: string, content: string, prevReleaseHeaderPattern: RegExp): void => {\n\tconst changelogContent = existsSync(outputFile) ? readFileSync(outputFile, { encoding: 'utf8' }) : ''\n\tconst prevReleaseStart = changelogContent.search(prevReleaseHeaderPattern)\n\tconst headlessChangelog = changelogContent.slice(prevReleaseStart)\n\tconst newChangelog = content + headlessChangelog\n\twriteFileSync(outputFile, newChangelog, { encoding: 'utf8' })\n}\n\nconst renderTemplate = (template: string, releaseContext: ReleaseContext): string => {\n\tconst compile = Handlebars.compile(template)\n\treturn compile(releaseContext)\n}\n\nconst getVersionFromTag = (tag: string, tagPattern: RegExp): string | undefined => {\n\treturn tagPattern.exec(tag)?.groups?.version\n}","import { execSync } from 'node:child_process'\nimport type { ResolvedConfig } from '@/types'\n\nexport const commit = (config: ResolvedConfig): void => {\n\tif (!config.commit) return\n\tconst options = config.commit\n\tconsole.log('Committing with options:', options)\n\tif (options.stageAll && !config.dryRun) execSync('git add -A', { stdio: 'inherit' })\n\tif (!config.dryRun) execSync(`git commit -m \"${options.message}\" ${options.signOff ? '-s' : ''} ${options.gpgSign ? '-S' : ''} ${options.extraArgs}`, { stdio: 'inherit' })\n}","import { execSync } from 'node:child_process'\nimport type { ResolvedConfig } from '@/types'\n\nexport const tag = (config: ResolvedConfig): void => {\n\tif (!config.tag) return\n\tconst options = config.tag\n\tconsole.log('Tagging with options:', options)\n\tif (!config.dryRun) execSync(`git tag -a ${options.name} -m \"${options.message}\" ${options.gpgSign ? '-s' : ''} ${options.force ? '-f' : ''} ${options.extraArgs}`, { stdio: 'inherit' })\n}","import { bump, commit, tag, changelog } from '@/lifecycles'\nimport { resolveConfig } from '@/utils'\nimport type { UserConfig } from '@/types'\n\nexport default async function relion(userConfig: UserConfig) {\n\tconst config = await resolveConfig(userConfig)\n\tbump(config)\n\tchangelog(config)\n\tcommit(config)\n\ttag(config)\n}\n\nexport const defineConfig = (config: UserConfig): UserConfig => config"],"mappings":"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"}
|
|
27
|
+
`;const B=e=>{if(!e.changelog)return;let t=e.changelog,n=e.context.releases;if(!n)return;let i=R(e.prevReleaseTagPattern);r.registerPartial(t.compiledPartials),r.registerHelper(t.helpers);let a=t.header;n.forEach((t,r)=>{let o=n[r+1],s,c;if(o)s=o.tag,c=U(s,e.prevReleaseTagPattern);else{let n=i.indexOf(t.tag);n===-1?(s=e.context.currentTag,c=U(s,e.prevReleaseTagPattern)):(s=i[n+1],c=s&&U(s,e.prevReleaseTagPattern))}let l={...t,...e.context,prevTag:s,prevVersion:c},u=H(z,l);a+=u}),t.stdout&&console.log(`Generated changelog:\n${a}`),t.outputFile&&(console.log(`Writing changelog to file '${t.outputFile}'`),e.dryRun||V(t.outputFile,a,t.prevReleaseHeaderPattern))},V=(r,i,a)=>{let o=e(r)?t(r,{encoding:`utf8`}):``,s=o.search(a),c=o.slice(s),l=i+c;n(r,l,{encoding:`utf8`})},H=(e,t)=>{let n=r.compile(e);return n(t)},U=(e,t)=>t.exec(e)?.groups?.version,W=e=>{if(!e.commit)return;let t=e.commit;console.log(`Committing with options:`,t),t.stageAll&&!e.dryRun&&a(`git add -A`,{stdio:`inherit`}),e.dryRun||a(`git commit -m "${t.message}" ${t.signOff?`-s`:``} ${t.gpgSign?`-S`:``} ${t.extraArgs}`,{stdio:`inherit`})},G=e=>{if(!e.tag)return;let t=e.tag;console.log(`Tagging with options:`,t),e.dryRun||a(`git tag -a ${t.name} -m "${t.message}" ${t.gpgSign?`-s`:``} ${t.force?`-f`:``} ${t.extraArgs}`,{stdio:`inherit`})};async function K(e){let t=await p(e);o(t),B(t),W(t),G(t)}const q=e=>e;export{K as default,q as defineConfig};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "relion",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.4.0",
|
|
4
4
|
"description": "Release workflow helper for Node.js projects.",
|
|
5
5
|
"author": "Kh4f <kh4f.dev@gmail.com>",
|
|
6
6
|
"license": "MIT",
|
|
@@ -53,14 +53,14 @@
|
|
|
53
53
|
"*.{js,ts}": "pnpm lint:fix"
|
|
54
54
|
},
|
|
55
55
|
"scripts": {
|
|
56
|
+
"build": "tsdown",
|
|
56
57
|
"lint": "eslint",
|
|
57
|
-
"lint:fix": "eslint --fix",
|
|
58
58
|
"test": "vitest run",
|
|
59
|
-
"test:watch": "vitest",
|
|
60
|
-
"build": "tsdown",
|
|
61
59
|
"build:watch": "tsdown --watch",
|
|
62
60
|
"build:prod": "tsdown --production",
|
|
61
|
+
"test:watch": "vitest",
|
|
62
|
+
"lint:fix": "eslint --fix",
|
|
63
63
|
"release": "node dist/cli.js -blct",
|
|
64
|
-
"release:github": "node dist/cli.js -l -p github"
|
|
64
|
+
"release:github": "node dist/cli.js -l -p github -L"
|
|
65
65
|
}
|
|
66
66
|
}
|