openspec-stat 1.3.4 → 1.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.
Files changed (57) hide show
  1. package/README.md +2 -1
  2. package/README.zh-CN.md +2 -1
  3. package/dist/esm/branch-selector.js +87 -214
  4. package/dist/esm/cli.js +14 -64
  5. package/dist/esm/commands/init.js +17 -44
  6. package/dist/esm/commands/multi.js +123 -180
  7. package/dist/esm/commands/single.js +143 -152
  8. package/dist/esm/config.js +22 -64
  9. package/dist/esm/formatters.js +285 -520
  10. package/dist/esm/git-analyzer.js +116 -354
  11. package/dist/esm/i18n/index.js +22 -10
  12. package/dist/esm/i18n/locales/en.json +43 -43
  13. package/dist/esm/i18n/locales/zh-CN.json +43 -43
  14. package/dist/esm/multi/config-validator.d.ts +1 -1
  15. package/dist/esm/multi/config-validator.js +25 -26
  16. package/dist/esm/multi/config-wizard.js +217 -459
  17. package/dist/esm/multi/multi-repo-analyzer.d.ts +11 -1
  18. package/dist/esm/multi/multi-repo-analyzer.js +228 -426
  19. package/dist/esm/stats-aggregator.js +115 -193
  20. package/dist/esm/time-utils.js +11 -17
  21. package/dist/esm/ui/spinner.d.ts +12 -0
  22. package/dist/esm/ui/spinner.js +38 -0
  23. package/package.json +21 -7
  24. package/dist/cjs/branch-selector.d.ts +0 -7
  25. package/dist/cjs/branch-selector.js +0 -128
  26. package/dist/cjs/cli.d.ts +0 -2
  27. package/dist/cjs/cli.js +0 -19
  28. package/dist/cjs/commands/init.d.ts +0 -7
  29. package/dist/cjs/commands/init.js +0 -58
  30. package/dist/cjs/commands/multi.d.ts +0 -16
  31. package/dist/cjs/commands/multi.js +0 -172
  32. package/dist/cjs/commands/single.d.ts +0 -2
  33. package/dist/cjs/commands/single.js +0 -148
  34. package/dist/cjs/config.d.ts +0 -3
  35. package/dist/cjs/config.js +0 -66
  36. package/dist/cjs/formatters.d.ts +0 -7
  37. package/dist/cjs/formatters.js +0 -482
  38. package/dist/cjs/git-analyzer.d.ts +0 -11
  39. package/dist/cjs/git-analyzer.js +0 -165
  40. package/dist/cjs/i18n/index.d.ts +0 -7
  41. package/dist/cjs/i18n/index.js +0 -84
  42. package/dist/cjs/i18n/locales/en.json +0 -154
  43. package/dist/cjs/i18n/locales/zh-CN.json +0 -154
  44. package/dist/cjs/index.d.ts +0 -6
  45. package/dist/cjs/index.js +0 -50
  46. package/dist/cjs/multi/config-validator.d.ts +0 -3
  47. package/dist/cjs/multi/config-validator.js +0 -130
  48. package/dist/cjs/multi/config-wizard.d.ts +0 -50
  49. package/dist/cjs/multi/config-wizard.js +0 -331
  50. package/dist/cjs/multi/multi-repo-analyzer.d.ts +0 -14
  51. package/dist/cjs/multi/multi-repo-analyzer.js +0 -210
  52. package/dist/cjs/stats-aggregator.d.ts +0 -7
  53. package/dist/cjs/stats-aggregator.js +0 -155
  54. package/dist/cjs/time-utils.d.ts +0 -6
  55. package/dist/cjs/time-utils.js +0 -55
  56. package/dist/cjs/types.d.ts +0 -136
  57. package/dist/cjs/types.js +0 -17
@@ -1,7 +1,3 @@
1
- function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
2
- function _regeneratorRuntime() { "use strict"; /*! regenerator-runtime -- Copyright (c) 2014-present, Facebook, Inc. -- license (MIT): https://github.com/facebook/regenerator/blob/main/LICENSE */ _regeneratorRuntime = function _regeneratorRuntime() { return e; }; var t, e = {}, r = Object.prototype, n = r.hasOwnProperty, o = Object.defineProperty || function (t, e, r) { t[e] = r.value; }, i = "function" == typeof Symbol ? Symbol : {}, a = i.iterator || "@@iterator", c = i.asyncIterator || "@@asyncIterator", u = i.toStringTag || "@@toStringTag"; function define(t, e, r) { return Object.defineProperty(t, e, { value: r, enumerable: !0, configurable: !0, writable: !0 }), t[e]; } try { define({}, ""); } catch (t) { define = function define(t, e, r) { return t[e] = r; }; } function wrap(t, e, r, n) { var i = e && e.prototype instanceof Generator ? e : Generator, a = Object.create(i.prototype), c = new Context(n || []); return o(a, "_invoke", { value: makeInvokeMethod(t, r, c) }), a; } function tryCatch(t, e, r) { try { return { type: "normal", arg: t.call(e, r) }; } catch (t) { return { type: "throw", arg: t }; } } e.wrap = wrap; var h = "suspendedStart", l = "suspendedYield", f = "executing", s = "completed", y = {}; function Generator() {} function GeneratorFunction() {} function GeneratorFunctionPrototype() {} var p = {}; define(p, a, function () { return this; }); var d = Object.getPrototypeOf, v = d && d(d(values([]))); v && v !== r && n.call(v, a) && (p = v); var g = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(p); function defineIteratorMethods(t) { ["next", "throw", "return"].forEach(function (e) { define(t, e, function (t) { return this._invoke(e, t); }); }); } function AsyncIterator(t, e) { function invoke(r, o, i, a) { var c = tryCatch(t[r], t, o); if ("throw" !== c.type) { var u = c.arg, h = u.value; return h && "object" == _typeof(h) && n.call(h, "__await") ? e.resolve(h.__await).then(function (t) { invoke("next", t, i, a); }, function (t) { invoke("throw", t, i, a); }) : e.resolve(h).then(function (t) { u.value = t, i(u); }, function (t) { return invoke("throw", t, i, a); }); } a(c.arg); } var r; o(this, "_invoke", { value: function value(t, n) { function callInvokeWithMethodAndArg() { return new e(function (e, r) { invoke(t, n, e, r); }); } return r = r ? r.then(callInvokeWithMethodAndArg, callInvokeWithMethodAndArg) : callInvokeWithMethodAndArg(); } }); } function makeInvokeMethod(e, r, n) { var o = h; return function (i, a) { if (o === f) throw new Error("Generator is already running"); if (o === s) { if ("throw" === i) throw a; return { value: t, done: !0 }; } for (n.method = i, n.arg = a;;) { var c = n.delegate; if (c) { var u = maybeInvokeDelegate(c, n); if (u) { if (u === y) continue; return u; } } if ("next" === n.method) n.sent = n._sent = n.arg;else if ("throw" === n.method) { if (o === h) throw o = s, n.arg; n.dispatchException(n.arg); } else "return" === n.method && n.abrupt("return", n.arg); o = f; var p = tryCatch(e, r, n); if ("normal" === p.type) { if (o = n.done ? s : l, p.arg === y) continue; return { value: p.arg, done: n.done }; } "throw" === p.type && (o = s, n.method = "throw", n.arg = p.arg); } }; } function maybeInvokeDelegate(e, r) { var n = r.method, o = e.iterator[n]; if (o === t) return r.delegate = null, "throw" === n && e.iterator.return && (r.method = "return", r.arg = t, maybeInvokeDelegate(e, r), "throw" === r.method) || "return" !== n && (r.method = "throw", r.arg = new TypeError("The iterator does not provide a '" + n + "' method")), y; var i = tryCatch(o, e.iterator, r.arg); if ("throw" === i.type) return r.method = "throw", r.arg = i.arg, r.delegate = null, y; var a = i.arg; return a ? a.done ? (r[e.resultName] = a.value, r.next = e.nextLoc, "return" !== r.method && (r.method = "next", r.arg = t), r.delegate = null, y) : a : (r.method = "throw", r.arg = new TypeError("iterator result is not an object"), r.delegate = null, y); } function pushTryEntry(t) { var e = { tryLoc: t[0] }; 1 in t && (e.catchLoc = t[1]), 2 in t && (e.finallyLoc = t[2], e.afterLoc = t[3]), this.tryEntries.push(e); } function resetTryEntry(t) { var e = t.completion || {}; e.type = "normal", delete e.arg, t.completion = e; } function Context(t) { this.tryEntries = [{ tryLoc: "root" }], t.forEach(pushTryEntry, this), this.reset(!0); } function values(e) { if (e || "" === e) { var r = e[a]; if (r) return r.call(e); if ("function" == typeof e.next) return e; if (!isNaN(e.length)) { var o = -1, i = function next() { for (; ++o < e.length;) if (n.call(e, o)) return next.value = e[o], next.done = !1, next; return next.value = t, next.done = !0, next; }; return i.next = i; } } throw new TypeError(_typeof(e) + " is not iterable"); } return GeneratorFunction.prototype = GeneratorFunctionPrototype, o(g, "constructor", { value: GeneratorFunctionPrototype, configurable: !0 }), o(GeneratorFunctionPrototype, "constructor", { value: GeneratorFunction, configurable: !0 }), GeneratorFunction.displayName = define(GeneratorFunctionPrototype, u, "GeneratorFunction"), e.isGeneratorFunction = function (t) { var e = "function" == typeof t && t.constructor; return !!e && (e === GeneratorFunction || "GeneratorFunction" === (e.displayName || e.name)); }, e.mark = function (t) { return Object.setPrototypeOf ? Object.setPrototypeOf(t, GeneratorFunctionPrototype) : (t.__proto__ = GeneratorFunctionPrototype, define(t, u, "GeneratorFunction")), t.prototype = Object.create(g), t; }, e.awrap = function (t) { return { __await: t }; }, defineIteratorMethods(AsyncIterator.prototype), define(AsyncIterator.prototype, c, function () { return this; }), e.AsyncIterator = AsyncIterator, e.async = function (t, r, n, o, i) { void 0 === i && (i = Promise); var a = new AsyncIterator(wrap(t, r, n, o), i); return e.isGeneratorFunction(r) ? a : a.next().then(function (t) { return t.done ? t.value : a.next(); }); }, defineIteratorMethods(g), define(g, u, "Generator"), define(g, a, function () { return this; }), define(g, "toString", function () { return "[object Generator]"; }), e.keys = function (t) { var e = Object(t), r = []; for (var n in e) r.push(n); return r.reverse(), function next() { for (; r.length;) { var t = r.pop(); if (t in e) return next.value = t, next.done = !1, next; } return next.done = !0, next; }; }, e.values = values, Context.prototype = { constructor: Context, reset: function reset(e) { if (this.prev = 0, this.next = 0, this.sent = this._sent = t, this.done = !1, this.delegate = null, this.method = "next", this.arg = t, this.tryEntries.forEach(resetTryEntry), !e) for (var r in this) "t" === r.charAt(0) && n.call(this, r) && !isNaN(+r.slice(1)) && (this[r] = t); }, stop: function stop() { this.done = !0; var t = this.tryEntries[0].completion; if ("throw" === t.type) throw t.arg; return this.rval; }, dispatchException: function dispatchException(e) { if (this.done) throw e; var r = this; function handle(n, o) { return a.type = "throw", a.arg = e, r.next = n, o && (r.method = "next", r.arg = t), !!o; } for (var o = this.tryEntries.length - 1; o >= 0; --o) { var i = this.tryEntries[o], a = i.completion; if ("root" === i.tryLoc) return handle("end"); if (i.tryLoc <= this.prev) { var c = n.call(i, "catchLoc"), u = n.call(i, "finallyLoc"); if (c && u) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } else if (c) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); } else { if (!u) throw new Error("try statement without catch or finally"); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } } } }, abrupt: function abrupt(t, e) { for (var r = this.tryEntries.length - 1; r >= 0; --r) { var o = this.tryEntries[r]; if (o.tryLoc <= this.prev && n.call(o, "finallyLoc") && this.prev < o.finallyLoc) { var i = o; break; } } i && ("break" === t || "continue" === t) && i.tryLoc <= e && e <= i.finallyLoc && (i = null); var a = i ? i.completion : {}; return a.type = t, a.arg = e, i ? (this.method = "next", this.next = i.finallyLoc, y) : this.complete(a); }, complete: function complete(t, e) { if ("throw" === t.type) throw t.arg; return "break" === t.type || "continue" === t.type ? this.next = t.arg : "return" === t.type ? (this.rval = this.arg = t.arg, this.method = "return", this.next = "end") : "normal" === t.type && e && (this.next = e), y; }, finish: function finish(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.finallyLoc === t) return this.complete(r.completion, r.afterLoc), resetTryEntry(r), y; } }, catch: function _catch(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.tryLoc === t) { var n = r.completion; if ("throw" === n.type) { var o = n.arg; resetTryEntry(r); } return o; } } throw new Error("illegal catch attempt"); }, delegateYield: function delegateYield(e, r, n) { return this.delegate = { iterator: values(e), resultName: r, nextLoc: n }, "next" === this.method && (this.arg = t), y; } }, e; }
3
- function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
4
- function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
5
1
  import chalk from 'chalk';
6
2
  import { loadConfig } from "../config.js";
7
3
  import { GitAnalyzer } from "../git-analyzer.js";
@@ -10,153 +6,148 @@ import { OutputFormatter } from "../formatters.js";
10
6
  import { getDefaultTimeRange, parseDateTime, parseBranches } from "../time-utils.js";
11
7
  import { selectBranches } from "../branch-selector.js";
12
8
  import { initI18n, t } from "../i18n/index.js";
13
- export function runSingleRepoCommand(_x) {
14
- return _runSingleRepoCommand.apply(this, arguments);
15
- }
16
- function _runSingleRepoCommand() {
17
- _runSingleRepoCommand = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime().mark(function _callee(options) {
18
- var config, since, until, defaultRange, branches, analyzer, activeAuthors, commits, analyses, i, commit, analysis, aggregator, result, formatter;
19
- return _regeneratorRuntime().wrap(function _callee$(_context) {
20
- while (1) switch (_context.prev = _context.next) {
21
- case 0:
22
- _context.prev = 0;
23
- initI18n(options.lang);
24
- console.log(chalk.blue(t('loading.config')));
25
- _context.next = 5;
26
- return loadConfig(options.config, options.repo);
27
- case 5:
28
- config = _context.sent;
29
- if (options.since || options.until) {
30
- since = options.since ? parseDateTime(options.since) : getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours).since;
31
- until = options.until ? parseDateTime(options.until) : getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours).until;
32
- } else {
33
- defaultRange = getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours);
34
- since = defaultRange.since;
35
- until = defaultRange.until;
36
- }
37
- if (!options.branches) {
38
- _context.next = 11;
39
- break;
40
- }
41
- branches = parseBranches(options.branches);
42
- _context.next = 18;
43
- break;
44
- case 11:
45
- if (!(options.interactive !== false)) {
46
- _context.next = 17;
47
- break;
48
- }
49
- _context.next = 14;
50
- return selectBranches(options.repo, config.defaultBranches);
51
- case 14:
52
- branches = _context.sent;
53
- _context.next = 18;
54
- break;
55
- case 17:
56
- branches = config.defaultBranches || [];
57
- case 18:
58
- console.log(chalk.blue(t('info.timeRange', {
59
- since: since.toLocaleString(),
60
- until: until.toLocaleString()
61
- })));
62
- console.log(chalk.blue(t('info.branches', {
63
- branches: branches.join(', ') || t('info.allBranches')
64
- })));
65
- analyzer = new GitAnalyzer(options.repo, config); // Fetch remote branches to ensure data is up-to-date
66
- if (!(!options.noFetch && config.autoFetch !== false)) {
67
- _context.next = 25;
68
- break;
69
- }
70
- console.log(chalk.blue(t('loading.fetching')));
71
- _context.next = 25;
72
- return analyzer.fetchRemote();
73
- case 25:
74
- console.log(chalk.blue(t('loading.activeUsers')));
75
- _context.next = 28;
76
- return analyzer.getActiveAuthors(config.activeUserWeeks || 2);
77
- case 28:
78
- activeAuthors = _context.sent;
79
- if (options.verbose) {
80
- console.log(chalk.gray(t('info.activeUsers', {
81
- weeks: String(config.activeUserWeeks || 2),
82
- users: Array.from(activeAuthors).join(', ')
83
- })));
84
- }
85
- console.log(chalk.blue(t('loading.analyzing')));
86
- _context.next = 33;
87
- return analyzer.getCommits(since, until, branches);
88
- case 33:
89
- commits = _context.sent;
90
- if (!(commits.length === 0)) {
91
- _context.next = 37;
92
- break;
93
- }
94
- console.log(chalk.yellow(t('warning.noCommits')));
95
- return _context.abrupt("return");
96
- case 37:
97
- console.log(chalk.blue(t('info.foundCommits', {
98
- count: String(commits.length)
99
- })));
100
- analyses = [];
101
- i = 0;
102
- case 40:
103
- if (!(i < commits.length)) {
104
- _context.next = 50;
105
- break;
106
- }
107
- commit = commits[i];
108
- if (options.verbose && i % 10 === 0) {
109
- console.log(chalk.gray(t('info.analysisProgress', {
110
- current: String(i + 1),
111
- total: String(commits.length)
112
- })));
113
- }
114
- _context.next = 45;
115
- return analyzer.analyzeCommit(commit);
116
- case 45:
117
- analysis = _context.sent;
118
- if (analysis) {
119
- analyses.push(analysis);
120
- }
121
- case 47:
122
- i++;
123
- _context.next = 40;
124
- break;
125
- case 50:
126
- if (!(analyses.length === 0)) {
127
- _context.next = 53;
128
- break;
129
- }
130
- console.log(chalk.yellow(t('warning.noQualifyingCommits')));
131
- return _context.abrupt("return");
132
- case 53:
133
- console.log(chalk.blue(t('info.qualifyingCommits', {
134
- count: String(analyses.length)
135
- })));
136
- aggregator = new StatsAggregator(config, activeAuthors);
137
- result = aggregator.aggregate(analyses, since, until, branches, options.author);
138
- formatter = new OutputFormatter();
139
- if (options.json) {
140
- console.log(formatter.formatJSON(result));
141
- } else if (options.csv) {
142
- console.log(formatter.formatCSV(result));
143
- } else if (options.markdown) {
144
- console.log(formatter.formatMarkdown(result));
145
- } else {
146
- console.log(formatter.formatTable(result, options.verbose));
147
- }
148
- _context.next = 64;
149
- break;
150
- case 60:
151
- _context.prev = 60;
152
- _context.t0 = _context["catch"](0);
153
- console.error(chalk.red(t('error.prefix')), _context.t0);
154
- process.exit(1);
155
- case 64:
156
- case "end":
157
- return _context.stop();
9
+ import { SpinnerManager } from "../ui/spinner.js";
10
+ export async function runSingleRepoCommand(options) {
11
+ try {
12
+ initI18n(options.lang);
13
+ const isQuiet = Boolean(options.json || options.csv || options.markdown);
14
+ const spinner = new SpinnerManager(isQuiet);
15
+ if (isQuiet) {
16
+ console.log(chalk.blue(t('loading.config')));
17
+ } else {
18
+ spinner.start(t('loading.config'));
19
+ }
20
+ const config = await loadConfig(options.config, options.repo);
21
+ if (!isQuiet) {
22
+ spinner.succeed();
23
+ }
24
+ let since;
25
+ let until;
26
+ if (options.since || options.until) {
27
+ since = options.since ? parseDateTime(options.since) : getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours).since;
28
+ until = options.until ? parseDateTime(options.until) : getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours).until;
29
+ } else {
30
+ const defaultRange = getDefaultTimeRange(config.defaultSinceHours, config.defaultUntilHours);
31
+ since = defaultRange.since;
32
+ until = defaultRange.until;
33
+ }
34
+ let branches;
35
+ if (options.branches) {
36
+ branches = parseBranches(options.branches);
37
+ } else if (options.interactive !== false) {
38
+ branches = await selectBranches(options.repo, config.defaultBranches);
39
+ } else {
40
+ branches = config.defaultBranches || [];
41
+ }
42
+ console.log(chalk.blue(t('info.timeRange', {
43
+ since: since.toLocaleString(),
44
+ until: until.toLocaleString()
45
+ })));
46
+ console.log(chalk.blue(t('info.branches', {
47
+ branches: branches.join(', ') || t('info.allBranches')
48
+ })));
49
+ const analyzer = new GitAnalyzer(options.repo, config);
50
+
51
+ // Fetch remote branches to ensure data is up-to-date
52
+ if (!options.noFetch && config.autoFetch !== false) {
53
+ if (isQuiet) {
54
+ console.log(chalk.blue(t('loading.fetching')));
55
+ } else {
56
+ spinner.start(t('loading.fetching'));
158
57
  }
159
- }, _callee, null, [[0, 60]]);
160
- }));
161
- return _runSingleRepoCommand.apply(this, arguments);
58
+ await analyzer.fetchRemote();
59
+ if (!isQuiet) {
60
+ spinner.succeed();
61
+ }
62
+ }
63
+ if (isQuiet) {
64
+ console.log(chalk.blue(t('loading.activeUsers')));
65
+ } else {
66
+ spinner.start(t('loading.activeUsers'));
67
+ }
68
+ const activeAuthors = await analyzer.getActiveAuthors(config.activeUserWeeks || 2);
69
+ if (!isQuiet) {
70
+ spinner.succeed();
71
+ }
72
+ if (options.verbose) {
73
+ console.log(chalk.gray(t('info.activeUsers', {
74
+ weeks: String(config.activeUserWeeks || 2),
75
+ users: Array.from(activeAuthors).join(', ')
76
+ })));
77
+ }
78
+ if (isQuiet) {
79
+ console.log(chalk.blue(t('loading.analyzing')));
80
+ } else {
81
+ spinner.start(t('loading.analyzing'));
82
+ }
83
+ const commits = await analyzer.getCommits(since, until, branches);
84
+ if (commits.length === 0) {
85
+ if (isQuiet) {
86
+ console.log(chalk.yellow(t('warning.noCommits')));
87
+ } else {
88
+ spinner.warn(t('warning.noCommits'));
89
+ }
90
+ return;
91
+ }
92
+ if (isQuiet) {
93
+ console.log(chalk.blue(t('info.foundCommits', {
94
+ count: String(commits.length)
95
+ })));
96
+ } else {
97
+ spinner.update(t('info.foundCommits', {
98
+ count: String(commits.length)
99
+ }));
100
+ }
101
+ const analyses = [];
102
+ for (let i = 0; i < commits.length; i++) {
103
+ const commit = commits[i];
104
+ if (options.verbose && i % 10 === 0) {
105
+ const progressText = t('info.analysisProgress', {
106
+ current: String(i + 1),
107
+ total: String(commits.length)
108
+ });
109
+ if (isQuiet) {
110
+ console.log(chalk.gray(progressText));
111
+ } else {
112
+ spinner.update(progressText);
113
+ }
114
+ }
115
+ const analysis = await analyzer.analyzeCommit(commit);
116
+ if (analysis) {
117
+ analyses.push(analysis);
118
+ }
119
+ }
120
+ if (analyses.length === 0) {
121
+ if (isQuiet) {
122
+ console.log(chalk.yellow(t('warning.noQualifyingCommits')));
123
+ } else {
124
+ spinner.warn(t('warning.noQualifyingCommits'));
125
+ }
126
+ return;
127
+ }
128
+ if (isQuiet) {
129
+ console.log(chalk.blue(t('info.qualifyingCommits', {
130
+ count: String(analyses.length)
131
+ })));
132
+ } else {
133
+ spinner.succeed(t('info.qualifyingCommits', {
134
+ count: String(analyses.length)
135
+ }));
136
+ }
137
+ const aggregator = new StatsAggregator(config, activeAuthors);
138
+ const result = aggregator.aggregate(analyses, since, until, branches, options.author);
139
+ const formatter = new OutputFormatter();
140
+ if (options.json) {
141
+ console.log(formatter.formatJSON(result));
142
+ } else if (options.csv) {
143
+ console.log(formatter.formatCSV(result));
144
+ } else if (options.markdown) {
145
+ console.log(formatter.formatMarkdown(result));
146
+ } else {
147
+ console.log(formatter.formatTable(result, options.verbose));
148
+ }
149
+ } catch (error) {
150
+ console.error(chalk.red(t('error.prefix')), error);
151
+ process.exit(1);
152
+ }
162
153
  }
@@ -1,15 +1,6 @@
1
- function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
2
- function _regeneratorRuntime() { "use strict"; /*! regenerator-runtime -- Copyright (c) 2014-present, Facebook, Inc. -- license (MIT): https://github.com/facebook/regenerator/blob/main/LICENSE */ _regeneratorRuntime = function _regeneratorRuntime() { return e; }; var t, e = {}, r = Object.prototype, n = r.hasOwnProperty, o = Object.defineProperty || function (t, e, r) { t[e] = r.value; }, i = "function" == typeof Symbol ? Symbol : {}, a = i.iterator || "@@iterator", c = i.asyncIterator || "@@asyncIterator", u = i.toStringTag || "@@toStringTag"; function define(t, e, r) { return Object.defineProperty(t, e, { value: r, enumerable: !0, configurable: !0, writable: !0 }), t[e]; } try { define({}, ""); } catch (t) { define = function define(t, e, r) { return t[e] = r; }; } function wrap(t, e, r, n) { var i = e && e.prototype instanceof Generator ? e : Generator, a = Object.create(i.prototype), c = new Context(n || []); return o(a, "_invoke", { value: makeInvokeMethod(t, r, c) }), a; } function tryCatch(t, e, r) { try { return { type: "normal", arg: t.call(e, r) }; } catch (t) { return { type: "throw", arg: t }; } } e.wrap = wrap; var h = "suspendedStart", l = "suspendedYield", f = "executing", s = "completed", y = {}; function Generator() {} function GeneratorFunction() {} function GeneratorFunctionPrototype() {} var p = {}; define(p, a, function () { return this; }); var d = Object.getPrototypeOf, v = d && d(d(values([]))); v && v !== r && n.call(v, a) && (p = v); var g = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(p); function defineIteratorMethods(t) { ["next", "throw", "return"].forEach(function (e) { define(t, e, function (t) { return this._invoke(e, t); }); }); } function AsyncIterator(t, e) { function invoke(r, o, i, a) { var c = tryCatch(t[r], t, o); if ("throw" !== c.type) { var u = c.arg, h = u.value; return h && "object" == _typeof(h) && n.call(h, "__await") ? e.resolve(h.__await).then(function (t) { invoke("next", t, i, a); }, function (t) { invoke("throw", t, i, a); }) : e.resolve(h).then(function (t) { u.value = t, i(u); }, function (t) { return invoke("throw", t, i, a); }); } a(c.arg); } var r; o(this, "_invoke", { value: function value(t, n) { function callInvokeWithMethodAndArg() { return new e(function (e, r) { invoke(t, n, e, r); }); } return r = r ? r.then(callInvokeWithMethodAndArg, callInvokeWithMethodAndArg) : callInvokeWithMethodAndArg(); } }); } function makeInvokeMethod(e, r, n) { var o = h; return function (i, a) { if (o === f) throw new Error("Generator is already running"); if (o === s) { if ("throw" === i) throw a; return { value: t, done: !0 }; } for (n.method = i, n.arg = a;;) { var c = n.delegate; if (c) { var u = maybeInvokeDelegate(c, n); if (u) { if (u === y) continue; return u; } } if ("next" === n.method) n.sent = n._sent = n.arg;else if ("throw" === n.method) { if (o === h) throw o = s, n.arg; n.dispatchException(n.arg); } else "return" === n.method && n.abrupt("return", n.arg); o = f; var p = tryCatch(e, r, n); if ("normal" === p.type) { if (o = n.done ? s : l, p.arg === y) continue; return { value: p.arg, done: n.done }; } "throw" === p.type && (o = s, n.method = "throw", n.arg = p.arg); } }; } function maybeInvokeDelegate(e, r) { var n = r.method, o = e.iterator[n]; if (o === t) return r.delegate = null, "throw" === n && e.iterator.return && (r.method = "return", r.arg = t, maybeInvokeDelegate(e, r), "throw" === r.method) || "return" !== n && (r.method = "throw", r.arg = new TypeError("The iterator does not provide a '" + n + "' method")), y; var i = tryCatch(o, e.iterator, r.arg); if ("throw" === i.type) return r.method = "throw", r.arg = i.arg, r.delegate = null, y; var a = i.arg; return a ? a.done ? (r[e.resultName] = a.value, r.next = e.nextLoc, "return" !== r.method && (r.method = "next", r.arg = t), r.delegate = null, y) : a : (r.method = "throw", r.arg = new TypeError("iterator result is not an object"), r.delegate = null, y); } function pushTryEntry(t) { var e = { tryLoc: t[0] }; 1 in t && (e.catchLoc = t[1]), 2 in t && (e.finallyLoc = t[2], e.afterLoc = t[3]), this.tryEntries.push(e); } function resetTryEntry(t) { var e = t.completion || {}; e.type = "normal", delete e.arg, t.completion = e; } function Context(t) { this.tryEntries = [{ tryLoc: "root" }], t.forEach(pushTryEntry, this), this.reset(!0); } function values(e) { if (e || "" === e) { var r = e[a]; if (r) return r.call(e); if ("function" == typeof e.next) return e; if (!isNaN(e.length)) { var o = -1, i = function next() { for (; ++o < e.length;) if (n.call(e, o)) return next.value = e[o], next.done = !1, next; return next.value = t, next.done = !0, next; }; return i.next = i; } } throw new TypeError(_typeof(e) + " is not iterable"); } return GeneratorFunction.prototype = GeneratorFunctionPrototype, o(g, "constructor", { value: GeneratorFunctionPrototype, configurable: !0 }), o(GeneratorFunctionPrototype, "constructor", { value: GeneratorFunction, configurable: !0 }), GeneratorFunction.displayName = define(GeneratorFunctionPrototype, u, "GeneratorFunction"), e.isGeneratorFunction = function (t) { var e = "function" == typeof t && t.constructor; return !!e && (e === GeneratorFunction || "GeneratorFunction" === (e.displayName || e.name)); }, e.mark = function (t) { return Object.setPrototypeOf ? Object.setPrototypeOf(t, GeneratorFunctionPrototype) : (t.__proto__ = GeneratorFunctionPrototype, define(t, u, "GeneratorFunction")), t.prototype = Object.create(g), t; }, e.awrap = function (t) { return { __await: t }; }, defineIteratorMethods(AsyncIterator.prototype), define(AsyncIterator.prototype, c, function () { return this; }), e.AsyncIterator = AsyncIterator, e.async = function (t, r, n, o, i) { void 0 === i && (i = Promise); var a = new AsyncIterator(wrap(t, r, n, o), i); return e.isGeneratorFunction(r) ? a : a.next().then(function (t) { return t.done ? t.value : a.next(); }); }, defineIteratorMethods(g), define(g, u, "Generator"), define(g, a, function () { return this; }), define(g, "toString", function () { return "[object Generator]"; }), e.keys = function (t) { var e = Object(t), r = []; for (var n in e) r.push(n); return r.reverse(), function next() { for (; r.length;) { var t = r.pop(); if (t in e) return next.value = t, next.done = !1, next; } return next.done = !0, next; }; }, e.values = values, Context.prototype = { constructor: Context, reset: function reset(e) { if (this.prev = 0, this.next = 0, this.sent = this._sent = t, this.done = !1, this.delegate = null, this.method = "next", this.arg = t, this.tryEntries.forEach(resetTryEntry), !e) for (var r in this) "t" === r.charAt(0) && n.call(this, r) && !isNaN(+r.slice(1)) && (this[r] = t); }, stop: function stop() { this.done = !0; var t = this.tryEntries[0].completion; if ("throw" === t.type) throw t.arg; return this.rval; }, dispatchException: function dispatchException(e) { if (this.done) throw e; var r = this; function handle(n, o) { return a.type = "throw", a.arg = e, r.next = n, o && (r.method = "next", r.arg = t), !!o; } for (var o = this.tryEntries.length - 1; o >= 0; --o) { var i = this.tryEntries[o], a = i.completion; if ("root" === i.tryLoc) return handle("end"); if (i.tryLoc <= this.prev) { var c = n.call(i, "catchLoc"), u = n.call(i, "finallyLoc"); if (c && u) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } else if (c) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); } else { if (!u) throw new Error("try statement without catch or finally"); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } } } }, abrupt: function abrupt(t, e) { for (var r = this.tryEntries.length - 1; r >= 0; --r) { var o = this.tryEntries[r]; if (o.tryLoc <= this.prev && n.call(o, "finallyLoc") && this.prev < o.finallyLoc) { var i = o; break; } } i && ("break" === t || "continue" === t) && i.tryLoc <= e && e <= i.finallyLoc && (i = null); var a = i ? i.completion : {}; return a.type = t, a.arg = e, i ? (this.method = "next", this.next = i.finallyLoc, y) : this.complete(a); }, complete: function complete(t, e) { if ("throw" === t.type) throw t.arg; return "break" === t.type || "continue" === t.type ? this.next = t.arg : "return" === t.type ? (this.rval = this.arg = t.arg, this.method = "return", this.next = "end") : "normal" === t.type && e && (this.next = e), y; }, finish: function finish(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.finallyLoc === t) return this.complete(r.completion, r.afterLoc), resetTryEntry(r), y; } }, catch: function _catch(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.tryLoc === t) { var n = r.completion; if ("throw" === n.type) { var o = n.arg; resetTryEntry(r); } return o; } } throw new Error("illegal catch attempt"); }, delegateYield: function delegateYield(e, r, n) { return this.delegate = { iterator: values(e), resultName: r, nextLoc: n }, "next" === this.method && (this.arg = t), y; } }, e; }
3
- function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
4
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
5
- function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
6
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : String(i); }
7
- function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
8
- function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
9
- function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
10
1
  import { existsSync, readFileSync } from 'fs';
11
2
  import { resolve } from 'path';
12
- var DEFAULT_CONFIG = {
3
+ const DEFAULT_CONFIG = {
13
4
  defaultBranches: ['origin/master'],
14
5
  defaultSinceHours: -30,
15
6
  defaultUntilHours: 20,
@@ -18,61 +9,28 @@ var DEFAULT_CONFIG = {
18
9
  excludeExtensions: ['.md', '.txt', '.png', '.jpg', '.jpeg', '.gif', '.svg', '.ico', '.webp'],
19
10
  activeUserWeeks: 2
20
11
  };
21
- export function loadConfig(_x) {
22
- return _loadConfig.apply(this, arguments);
23
- }
24
- function _loadConfig() {
25
- _loadConfig = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime().mark(function _callee(configPath) {
26
- var repoPath,
27
- config,
28
- searchPaths,
29
- _i,
30
- _searchPaths,
31
- path,
32
- fileContent,
33
- userConfig,
34
- _args = arguments;
35
- return _regeneratorRuntime().wrap(function _callee$(_context) {
36
- while (1) switch (_context.prev = _context.next) {
37
- case 0:
38
- repoPath = _args.length > 1 && _args[1] !== undefined ? _args[1] : '.';
39
- config = _objectSpread({}, DEFAULT_CONFIG);
40
- searchPaths = configPath ? [configPath] : [resolve(repoPath, '.openspec-stats.json'), resolve(repoPath, 'openspec-stats.config.json'), resolve(process.cwd(), '.openspec-stats.json'), resolve(process.cwd(), 'openspec-stats.config.json')];
41
- _i = 0, _searchPaths = searchPaths;
42
- case 4:
43
- if (!(_i < _searchPaths.length)) {
44
- _context.next = 20;
45
- break;
46
- }
47
- path = _searchPaths[_i];
48
- if (!existsSync(path)) {
49
- _context.next = 17;
50
- break;
51
- }
52
- _context.prev = 7;
53
- fileContent = readFileSync(path, 'utf-8');
54
- userConfig = JSON.parse(fileContent);
55
- config = _objectSpread(_objectSpread({}, config), userConfig);
56
- return _context.abrupt("break", 20);
57
- case 14:
58
- _context.prev = 14;
59
- _context.t0 = _context["catch"](7);
60
- console.warn("Failed to load config from ".concat(path, ":"), _context.t0);
61
- case 17:
62
- _i++;
63
- _context.next = 4;
64
- break;
65
- case 20:
66
- return _context.abrupt("return", config);
67
- case 21:
68
- case "end":
69
- return _context.stop();
12
+ export async function loadConfig(configPath, repoPath = '.') {
13
+ let config = {
14
+ ...DEFAULT_CONFIG
15
+ };
16
+ const searchPaths = configPath ? [configPath] : [resolve(repoPath, '.openspec-stats.json'), resolve(repoPath, 'openspec-stats.config.json'), resolve(process.cwd(), '.openspec-stats.json'), resolve(process.cwd(), 'openspec-stats.config.json')];
17
+ for (const path of searchPaths) {
18
+ if (existsSync(path)) {
19
+ try {
20
+ const fileContent = readFileSync(path, 'utf-8');
21
+ const userConfig = JSON.parse(fileContent);
22
+ config = {
23
+ ...config,
24
+ ...userConfig
25
+ };
26
+ break;
27
+ } catch (error) {
28
+ console.warn(`Failed to load config from ${path}:`, error);
70
29
  }
71
- }, _callee, null, [[7, 14]]);
72
- }));
73
- return _loadConfig.apply(this, arguments);
30
+ }
31
+ }
32
+ return config;
74
33
  }
75
- export function normalizeAuthor(author) {
76
- var mapping = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
34
+ export function normalizeAuthor(author, mapping = {}) {
77
35
  return mapping[author] || author;
78
36
  }