@gpc-cli/cli 0.9.45 → 0.9.47

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 (172) hide show
  1. package/README.md +36 -15
  2. package/dist/anomalies-V3AFS4LD.js +66 -0
  3. package/dist/anomalies-V3AFS4LD.js.map +1 -0
  4. package/dist/{apps-J2446UDA.js → apps-FKD3ZG5X.js} +31 -35
  5. package/dist/apps-FKD3ZG5X.js.map +1 -0
  6. package/dist/{audit-N2CRHWUN.js → audit-JASSHRWN.js} +47 -62
  7. package/dist/audit-JASSHRWN.js.map +1 -0
  8. package/dist/{auth-XGSTT5G5.js → auth-OTA3SV3J.js} +145 -103
  9. package/dist/auth-OTA3SV3J.js.map +1 -0
  10. package/dist/bin.js +6 -4
  11. package/dist/bin.js.map +1 -1
  12. package/dist/bundle-F7MUVC5J.js +204 -0
  13. package/dist/bundle-F7MUVC5J.js.map +1 -0
  14. package/dist/{cache-SLNFRTI2.js → cache-XKPLZYEB.js} +4 -5
  15. package/dist/cache-XKPLZYEB.js.map +1 -0
  16. package/dist/changelog-QLDFG5TV.js +48 -0
  17. package/dist/changelog-QLDFG5TV.js.map +1 -0
  18. package/dist/{chunk-4O4D5SGL.js → chunk-3SJ6OXCZ.js} +4 -5
  19. package/dist/chunk-3SJ6OXCZ.js.map +1 -0
  20. package/dist/{chunk-U6ZTQ34I.js → chunk-BCBXQC7J.js} +45 -11
  21. package/dist/chunk-BCBXQC7J.js.map +1 -0
  22. package/dist/{chunk-AA577WVQ.js → chunk-NQH4G7BI.js} +9 -3
  23. package/dist/chunk-NQH4G7BI.js.map +1 -0
  24. package/dist/chunk-SLNJEAMK.js +23 -0
  25. package/dist/chunk-SLNJEAMK.js.map +1 -0
  26. package/dist/{chunk-SEVX56VN.js → chunk-WWVURXVO.js} +56 -49
  27. package/dist/chunk-WWVURXVO.js.map +1 -0
  28. package/dist/{chunk-NV75I5VP.js → chunk-YFUBD2XB.js} +10 -8
  29. package/dist/chunk-YFUBD2XB.js.map +1 -0
  30. package/dist/{config-BUXPDN7N.js → config-NY3TZGVS.js} +8 -5
  31. package/dist/config-NY3TZGVS.js.map +1 -0
  32. package/dist/{data-safety-Q7FTCEWU.js → data-safety-AFMD6MYI.js} +12 -27
  33. package/dist/data-safety-AFMD6MYI.js.map +1 -0
  34. package/dist/{device-tiers-MIOQEXYY.js → device-tiers-AQAMUQXI.js} +23 -38
  35. package/dist/device-tiers-AQAMUQXI.js.map +1 -0
  36. package/dist/diff-6EO4ID6W.js +91 -0
  37. package/dist/diff-6EO4ID6W.js.map +1 -0
  38. package/dist/{docs-7DUXIKA3.js → docs-4D2SJ4LY.js} +4 -3
  39. package/dist/docs-4D2SJ4LY.js.map +1 -0
  40. package/dist/doctor-QCCWG6Y3.js +708 -0
  41. package/dist/doctor-QCCWG6Y3.js.map +1 -0
  42. package/dist/{enterprise-7THXNBTC.js → enterprise-7PWXMSUN.js} +11 -21
  43. package/dist/enterprise-7PWXMSUN.js.map +1 -0
  44. package/dist/{external-transactions-2GWIMUVM.js → external-transactions-LCZALS3V.js} +12 -28
  45. package/dist/external-transactions-LCZALS3V.js.map +1 -0
  46. package/dist/{feedback-DPTO6DUT.js → feedback-CET2X67K.js} +4 -4
  47. package/dist/{games-BT777WUO.js → games-ZSNGEI7A.js} +17 -32
  48. package/dist/games-ZSNGEI7A.js.map +1 -0
  49. package/dist/{generated-apks-RJWTIX7L.js → generated-apks-RX2IUWSF.js} +30 -38
  50. package/dist/generated-apks-RX2IUWSF.js.map +1 -0
  51. package/dist/{grants-TKQJ3IER.js → grants-EBPECI26.js} +22 -40
  52. package/dist/grants-EBPECI26.js.map +1 -0
  53. package/dist/{iap-ICAEQLK5.js → iap-OUI5YYN4.js} +30 -51
  54. package/dist/iap-OUI5YYN4.js.map +1 -0
  55. package/dist/index.js +1 -1
  56. package/dist/{init-JZ2THPMS.js → init-WSTQTJOD.js} +5 -4
  57. package/dist/init-WSTQTJOD.js.map +1 -0
  58. package/dist/{install-skills-OV4HVANW.js → install-skills-6QDUXI5F.js} +5 -6
  59. package/dist/{install-skills-OV4HVANW.js.map → install-skills-6QDUXI5F.js.map} +1 -1
  60. package/dist/{internal-sharing-3U2XFHA4.js → internal-sharing-ONNIWIAT.js} +3 -4
  61. package/dist/{internal-sharing-3U2XFHA4.js.map → internal-sharing-ONNIWIAT.js.map} +1 -1
  62. package/dist/{listings-77HZW4S5.js → listings-7SGQ4SRX.js} +118 -157
  63. package/dist/listings-7SGQ4SRX.js.map +1 -0
  64. package/dist/migrate-ZQCJGQQS.js +138 -0
  65. package/dist/migrate-ZQCJGQQS.js.map +1 -0
  66. package/dist/{one-time-products-LHZAXQES.js → one-time-products-MGZTU7OM.js} +65 -120
  67. package/dist/one-time-products-MGZTU7OM.js.map +1 -0
  68. package/dist/{preflight-H3HEBYQW.js → preflight-N7ZRG2JI.js} +58 -55
  69. package/dist/preflight-N7ZRG2JI.js.map +1 -0
  70. package/dist/{pricing-XQSDTTK5.js → pricing-JJZFICFL.js} +8 -8
  71. package/dist/{pricing-XQSDTTK5.js.map → pricing-JJZFICFL.js.map} +1 -1
  72. package/dist/{prompt-BSV22CQZ.js → prompt-GXC2JSLA.js} +2 -2
  73. package/dist/{publish-Q5ZKEKZ5.js → publish-JPTI4EBT.js} +34 -30
  74. package/dist/publish-JPTI4EBT.js.map +1 -0
  75. package/dist/{purchase-options-CKRN4VIW.js → purchase-options-KFWW4JW2.js} +16 -11
  76. package/dist/purchase-options-KFWW4JW2.js.map +1 -0
  77. package/dist/{purchases-43AKV6HG.js → purchases-Z3QBM3UO.js} +121 -194
  78. package/dist/purchases-Z3QBM3UO.js.map +1 -0
  79. package/dist/{quickstart-4HB62YEL.js → quickstart-Z5Y3FYJU.js} +5 -3
  80. package/dist/quickstart-Z5Y3FYJU.js.map +1 -0
  81. package/dist/{quota-UHIQQYOY.js → quota-MZRWYJGR.js} +5 -15
  82. package/dist/quota-MZRWYJGR.js.map +1 -0
  83. package/dist/{recovery-5EV2R476.js → recovery-YE3Z7NIN.js} +32 -61
  84. package/dist/recovery-YE3Z7NIN.js.map +1 -0
  85. package/dist/{releases-C2WC2K4E.js → releases-276W3BR7.js} +188 -187
  86. package/dist/releases-276W3BR7.js.map +1 -0
  87. package/dist/{reports-2YX3RDOS.js → reports-CIB2T3XT.js} +19 -21
  88. package/dist/reports-CIB2T3XT.js.map +1 -0
  89. package/dist/reviews-YCBBM656.js +199 -0
  90. package/dist/reviews-YCBBM656.js.map +1 -0
  91. package/dist/rtdn-LID2B7XZ.js +87 -0
  92. package/dist/rtdn-LID2B7XZ.js.map +1 -0
  93. package/dist/{status-WHGLODGV.js → status-6LH5W4FU.js} +105 -83
  94. package/dist/status-6LH5W4FU.js.map +1 -0
  95. package/dist/{subscriptions-CI3JH3VQ.js → subscriptions-DZP3Y7O7.js} +142 -232
  96. package/dist/subscriptions-DZP3Y7O7.js.map +1 -0
  97. package/dist/{testers-NZOFA3EF.js → testers-LSMBXCA2.js} +24 -44
  98. package/dist/testers-LSMBXCA2.js.map +1 -0
  99. package/dist/tracks-YHMO2A6B.js +98 -0
  100. package/dist/tracks-YHMO2A6B.js.map +1 -0
  101. package/dist/{train-CJJVLY4B.js → train-MDD2EBHS.js} +35 -55
  102. package/dist/train-MDD2EBHS.js.map +1 -0
  103. package/dist/{update-NAK6CMUX.js → update-XAO5EZHC.js} +30 -15
  104. package/dist/update-XAO5EZHC.js.map +1 -0
  105. package/dist/{users-2YTC4Q36.js → users-UKG7VIQH.js} +45 -67
  106. package/dist/users-UKG7VIQH.js.map +1 -0
  107. package/dist/{validate-UOVTM6L3.js → validate-QIYSA3N7.js} +8 -10
  108. package/dist/validate-QIYSA3N7.js.map +1 -0
  109. package/dist/{version-N64UBW7A.js → version-R3P4NHCF.js} +4 -4
  110. package/dist/{vitals-A4CS4MSS.js → vitals-PJEQUUAK.js} +174 -165
  111. package/dist/vitals-PJEQUUAK.js.map +1 -0
  112. package/package.json +6 -6
  113. package/dist/anomalies-NU2IN2GJ.js +0 -54
  114. package/dist/anomalies-NU2IN2GJ.js.map +0 -1
  115. package/dist/apps-J2446UDA.js.map +0 -1
  116. package/dist/audit-N2CRHWUN.js.map +0 -1
  117. package/dist/auth-XGSTT5G5.js.map +0 -1
  118. package/dist/bundle-F43TD2BQ.js +0 -218
  119. package/dist/bundle-F43TD2BQ.js.map +0 -1
  120. package/dist/cache-SLNFRTI2.js.map +0 -1
  121. package/dist/changelog-ZYD6W5IV.js +0 -53
  122. package/dist/changelog-ZYD6W5IV.js.map +0 -1
  123. package/dist/chunk-4O4D5SGL.js.map +0 -1
  124. package/dist/chunk-AA577WVQ.js.map +0 -1
  125. package/dist/chunk-FWKYRLKY.js +0 -19
  126. package/dist/chunk-FWKYRLKY.js.map +0 -1
  127. package/dist/chunk-NV75I5VP.js.map +0 -1
  128. package/dist/chunk-SEVX56VN.js.map +0 -1
  129. package/dist/chunk-U6ZTQ34I.js.map +0 -1
  130. package/dist/config-BUXPDN7N.js.map +0 -1
  131. package/dist/data-safety-Q7FTCEWU.js.map +0 -1
  132. package/dist/device-tiers-MIOQEXYY.js.map +0 -1
  133. package/dist/diff-V77SMKAQ.js +0 -96
  134. package/dist/diff-V77SMKAQ.js.map +0 -1
  135. package/dist/docs-7DUXIKA3.js.map +0 -1
  136. package/dist/doctor-3Z4ARPM2.js +0 -372
  137. package/dist/doctor-3Z4ARPM2.js.map +0 -1
  138. package/dist/enterprise-7THXNBTC.js.map +0 -1
  139. package/dist/external-transactions-2GWIMUVM.js.map +0 -1
  140. package/dist/games-BT777WUO.js.map +0 -1
  141. package/dist/generated-apks-RJWTIX7L.js.map +0 -1
  142. package/dist/grants-TKQJ3IER.js.map +0 -1
  143. package/dist/iap-ICAEQLK5.js.map +0 -1
  144. package/dist/init-JZ2THPMS.js.map +0 -1
  145. package/dist/listings-77HZW4S5.js.map +0 -1
  146. package/dist/migrate-SQT6RD6T.js +0 -143
  147. package/dist/migrate-SQT6RD6T.js.map +0 -1
  148. package/dist/one-time-products-LHZAXQES.js.map +0 -1
  149. package/dist/preflight-H3HEBYQW.js.map +0 -1
  150. package/dist/publish-Q5ZKEKZ5.js.map +0 -1
  151. package/dist/purchase-options-CKRN4VIW.js.map +0 -1
  152. package/dist/purchases-43AKV6HG.js.map +0 -1
  153. package/dist/quickstart-4HB62YEL.js.map +0 -1
  154. package/dist/quota-UHIQQYOY.js.map +0 -1
  155. package/dist/recovery-5EV2R476.js.map +0 -1
  156. package/dist/releases-C2WC2K4E.js.map +0 -1
  157. package/dist/reports-2YX3RDOS.js.map +0 -1
  158. package/dist/reviews-2CWOI5CV.js +0 -213
  159. package/dist/reviews-2CWOI5CV.js.map +0 -1
  160. package/dist/status-WHGLODGV.js.map +0 -1
  161. package/dist/subscriptions-CI3JH3VQ.js.map +0 -1
  162. package/dist/testers-NZOFA3EF.js.map +0 -1
  163. package/dist/tracks-NERFFEDT.js +0 -107
  164. package/dist/tracks-NERFFEDT.js.map +0 -1
  165. package/dist/train-CJJVLY4B.js.map +0 -1
  166. package/dist/update-NAK6CMUX.js.map +0 -1
  167. package/dist/users-2YTC4Q36.js.map +0 -1
  168. package/dist/validate-UOVTM6L3.js.map +0 -1
  169. package/dist/vitals-A4CS4MSS.js.map +0 -1
  170. /package/dist/{feedback-DPTO6DUT.js.map → feedback-CET2X67K.js.map} +0 -0
  171. /package/dist/{prompt-BSV22CQZ.js.map → prompt-GXC2JSLA.js.map} +0 -0
  172. /package/dist/{version-N64UBW7A.js.map → version-R3P4NHCF.js.map} +0 -0
@@ -12,7 +12,7 @@ import {
12
12
  promptSelect,
13
13
  requireConfirm,
14
14
  requireOption
15
- } from "./chunk-NV75I5VP.js";
15
+ } from "./chunk-YFUBD2XB.js";
16
16
 
17
17
  // src/commands/releases.ts
18
18
  import { appendFile, stat } from "fs/promises";
@@ -33,14 +33,19 @@ import {
33
33
  diffReleases,
34
34
  fetchReleaseNotes,
35
35
  getVitalsCrashes,
36
- checkThreshold
36
+ checkThreshold,
37
+ GpcError
37
38
  } from "@gpc-cli/core";
38
39
  import { formatOutput, sortResults, createSpinner } from "@gpc-cli/core";
39
40
  function resolvePackageName(packageArg, config) {
40
41
  const name = packageArg || config.app;
41
42
  if (!name) {
42
- console.error("Error: No package name. Use --app <package> or gpc config set app <package>");
43
- process.exit(2);
43
+ throw new GpcError(
44
+ "No package name. Use --app <package> or gpc config set app <package>",
45
+ "RELEASES_USAGE_ERROR",
46
+ 2,
47
+ "Set a default app: gpc config set app <package>"
48
+ );
44
49
  }
45
50
  return name;
46
51
  }
@@ -62,17 +67,25 @@ function registerReleasesCommands(program) {
62
67
  "--timeout <ms>",
63
68
  "Upload timeout in milliseconds (auto-scales with file size by default)",
64
69
  parseInt
65
- ).action(async (file, options) => {
70
+ ).option("--status <status>", "Release status: completed, inProgress, draft, halted", "completed").action(async (file, options) => {
66
71
  try {
67
72
  await stat(file);
68
73
  } catch {
69
- console.error(`Error: File not found: ${file}`);
70
- process.exit(2);
74
+ throw new GpcError(
75
+ `File not found: ${file}`,
76
+ "RELEASES_USAGE_ERROR",
77
+ 2,
78
+ "Check the file path and try again."
79
+ );
71
80
  }
72
81
  const ext = extname(file).toLowerCase();
73
82
  if (ext !== ".aab" && ext !== ".apk") {
74
- console.error(`Error: Expected .aab or .apk file, got "${ext || "(no extension)"}"`);
75
- process.exit(2);
83
+ throw new GpcError(
84
+ `Expected .aab or .apk file, got "${ext || "(no extension)"}"`,
85
+ "RELEASES_USAGE_ERROR",
86
+ 2,
87
+ "Provide a .aab or .apk file."
88
+ );
76
89
  }
77
90
  const noteSources = [
78
91
  options.notes,
@@ -81,10 +94,12 @@ function registerReleasesCommands(program) {
81
94
  options.copyNotesFrom
82
95
  ].filter(Boolean);
83
96
  if (noteSources.length > 1) {
84
- console.error(
85
- "Error: Cannot combine --notes, --notes-dir, --notes-from-git, and --copy-notes-from. Use only one."
97
+ throw new GpcError(
98
+ "Cannot combine --notes, --notes-dir, --notes-from-git, and --copy-notes-from. Use only one.",
99
+ "RELEASES_USAGE_ERROR",
100
+ 2,
101
+ "Pick one release notes source."
86
102
  );
87
- process.exit(2);
88
103
  }
89
104
  const config = await loadConfig();
90
105
  const packageName = resolvePackageName(program.opts()["app"], config);
@@ -111,10 +126,12 @@ function registerReleasesCommands(program) {
111
126
  if (options.rollout !== void 0) {
112
127
  const rollout2 = Number(options.rollout);
113
128
  if (!Number.isFinite(rollout2) || rollout2 < 1 || rollout2 > 100) {
114
- console.error(
115
- `Error: --rollout must be a number between 1 and 100 (got: ${options.rollout})`
129
+ throw new GpcError(
130
+ `--rollout must be a number between 1 and 100 (got: ${options.rollout})`,
131
+ "RELEASES_USAGE_ERROR",
132
+ 2,
133
+ "Use a percentage between 1 and 100."
116
134
  );
117
- process.exit(2);
118
135
  }
119
136
  }
120
137
  const { size: fileSize } = await stat(file);
@@ -141,17 +158,12 @@ function registerReleasesCommands(program) {
141
158
  );
142
159
  } : void 0;
143
160
  if (isDryRun(program)) {
144
- try {
145
- const result = await uploadRelease(client, packageName, file, {
146
- track: options.track,
147
- userFraction: options.rollout ? Number(options.rollout) / 100 : void 0,
148
- dryRun: true
149
- });
150
- console.log(formatOutput(result, format));
151
- } catch (error) {
152
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
153
- process.exit(4);
154
- }
161
+ const result = await uploadRelease(client, packageName, file, {
162
+ track: options.track,
163
+ userFraction: options.rollout ? Number(options.rollout) / 100 : void 0,
164
+ dryRun: true
165
+ });
166
+ console.log(formatOutput(result, format));
155
167
  return;
156
168
  }
157
169
  const auditEntry = createAuditEntry(
@@ -179,6 +191,7 @@ function registerReleasesCommands(program) {
179
191
  }
180
192
  const result = await uploadRelease(client, packageName, file, {
181
193
  track: options.track,
194
+ status: options.status,
182
195
  userFraction: options.rollout ? Number(options.rollout) / 100 : void 0,
183
196
  releaseNotes,
184
197
  releaseName: options.name,
@@ -199,8 +212,7 @@ function registerReleasesCommands(program) {
199
212
  spinner.fail("Upload failed");
200
213
  auditEntry.success = false;
201
214
  auditEntry.error = error instanceof Error ? error.message : String(error);
202
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
203
- process.exit(4);
215
+ throw error;
204
216
  } finally {
205
217
  auditEntry.durationMs = Date.now() - new Date(auditEntry.timestamp).getTime();
206
218
  writeAuditLog(auditEntry).catch(() => {
@@ -212,43 +224,42 @@ function registerReleasesCommands(program) {
212
224
  const packageName = resolvePackageName(program.opts()["app"], config);
213
225
  const client = await getClient(config);
214
226
  const format = getOutputFormat(program, config);
215
- try {
216
- const TRACK_ORDER = ["production", "beta", "alpha", "internal"];
217
- const rawStatuses = await getReleasesStatus(client, packageName, options.track);
218
- const statuses = options.track ? Array.isArray(rawStatuses) ? rawStatuses.filter((s) => s.track === options.track) : rawStatuses : rawStatuses;
219
- const sorted = Array.isArray(statuses) ? options.sort ? sortResults(statuses, options.sort) : [...statuses].sort((a, b) => {
220
- const ai = TRACK_ORDER.indexOf(
221
- String(a["track"] ?? "")
222
- );
223
- const bi = TRACK_ORDER.indexOf(
224
- String(b["track"] ?? "")
225
- );
226
- return (ai === -1 ? 99 : ai) - (bi === -1 ? 99 : bi);
227
- }) : statuses;
228
- if (format !== "json" && Array.isArray(sorted)) {
229
- const rows = sorted.map((s) => {
230
- const sr = s;
231
- return {
232
- track: sr["track"] || "-",
233
- status: sr["status"] || "-",
234
- name: sr["name"] || "-",
235
- versionCodes: Array.isArray(sr["versionCodes"]) ? sr["versionCodes"].join(", ") : "-",
236
- userFraction: sr["userFraction"] !== void 0 ? `${Math.round(Number(sr["userFraction"]) * 100)}%` : "\u2014"
237
- };
238
- });
239
- console.log(formatOutput(rows, format));
240
- } else {
241
- console.log(formatOutput(sorted, format));
242
- }
243
- } catch (error) {
244
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
245
- process.exit(4);
227
+ const TRACK_ORDER = ["production", "beta", "alpha", "internal"];
228
+ const rawStatuses = await getReleasesStatus(client, packageName, options.track);
229
+ const statuses = options.track ? Array.isArray(rawStatuses) ? rawStatuses.filter((s) => s.track === options.track) : rawStatuses : rawStatuses;
230
+ const sorted = Array.isArray(statuses) ? options.sort ? sortResults(statuses, options.sort) : [...statuses].sort((a, b) => {
231
+ const ai = TRACK_ORDER.indexOf(
232
+ String(a["track"] ?? "")
233
+ );
234
+ const bi = TRACK_ORDER.indexOf(
235
+ String(b["track"] ?? "")
236
+ );
237
+ return (ai === -1 ? 99 : ai) - (bi === -1 ? 99 : bi);
238
+ }) : statuses;
239
+ if (format !== "json" && Array.isArray(sorted)) {
240
+ const rows = sorted.map((s) => {
241
+ const sr = s;
242
+ return {
243
+ track: sr["track"] || "-",
244
+ status: sr["status"] || "-",
245
+ name: sr["name"] || "-",
246
+ versionCodes: Array.isArray(sr["versionCodes"]) ? sr["versionCodes"].join(", ") : "-",
247
+ userFraction: sr["userFraction"] !== void 0 ? `${Math.round(Number(sr["userFraction"]) * 100)}%` : "\u2014"
248
+ };
249
+ });
250
+ console.log(formatOutput(rows, format));
251
+ } else {
252
+ console.log(formatOutput(sorted, format));
246
253
  }
247
254
  });
248
- releases.command("promote").description("Promote a release from one track to another").option("--from <track>", "Source track").option("--to <track>", "Target track").option("--rollout <percent>", "Staged rollout percentage").option("--notes <text>", "Release notes").option("--copy-notes-from <track>", "Copy release notes from another track").action(async (options) => {
255
+ releases.command("promote").description("Promote a release from one track to another").option("--from <track>", "Source track").option("--to <track>", "Target track").option("--rollout <percent>", "Staged rollout percentage").option("--notes <text>", "Release notes").option("--copy-notes-from <track>", "Copy release notes from another track").option("--status <status>", "Release status: completed, inProgress, draft, halted").action(async (options) => {
249
256
  if (options.notes && options.copyNotesFrom) {
250
- console.error("Error: Cannot combine --notes and --copy-notes-from. Use only one.");
251
- process.exit(2);
257
+ throw new GpcError(
258
+ "Cannot combine --notes and --copy-notes-from. Use only one.",
259
+ "RELEASES_USAGE_ERROR",
260
+ 2,
261
+ "Pick one release notes source."
262
+ );
252
263
  }
253
264
  const config = await loadConfig();
254
265
  const packageName = resolvePackageName(program.opts()["app"], config);
@@ -274,18 +285,22 @@ function registerReleasesCommands(program) {
274
285
  interactive
275
286
  );
276
287
  if (options.from === options.to) {
277
- console.error(
278
- `Error: --from and --to must be different tracks (both are "${options.from}")`
288
+ throw new GpcError(
289
+ `--from and --to must be different tracks (both are "${options.from}")`,
290
+ "RELEASES_USAGE_ERROR",
291
+ 2,
292
+ "Specify different source and target tracks."
279
293
  );
280
- process.exit(2);
281
294
  }
282
295
  if (options.rollout !== void 0) {
283
296
  const rollout2 = Number(options.rollout);
284
297
  if (!Number.isFinite(rollout2) || rollout2 < 1 || rollout2 > 100) {
285
- console.error(
286
- `Error: --rollout must be a number between 1 and 100 (got: ${options.rollout})`
298
+ throw new GpcError(
299
+ `--rollout must be a number between 1 and 100 (got: ${options.rollout})`,
300
+ "RELEASES_USAGE_ERROR",
301
+ 2,
302
+ "Use a percentage between 1 and 100."
287
303
  );
288
- process.exit(2);
289
304
  }
290
305
  }
291
306
  if (isDryRun(program)) {
@@ -302,22 +317,18 @@ function registerReleasesCommands(program) {
302
317
  return;
303
318
  }
304
319
  const client = await getClient(config);
305
- try {
306
- let releaseNotes;
307
- if (options.copyNotesFrom) {
308
- releaseNotes = await fetchReleaseNotes(client, packageName, options.copyNotesFrom);
309
- } else if (options.notes) {
310
- releaseNotes = [{ language: "en-US", text: options.notes }];
311
- }
312
- const result = await promoteRelease(client, packageName, options.from, options.to, {
313
- userFraction: options.rollout ? Number(options.rollout) / 100 : void 0,
314
- releaseNotes
315
- });
316
- console.log(formatOutput(result, format));
317
- } catch (error) {
318
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
319
- process.exit(4);
320
+ let releaseNotes;
321
+ if (options.copyNotesFrom) {
322
+ releaseNotes = await fetchReleaseNotes(client, packageName, options.copyNotesFrom);
323
+ } else if (options.notes) {
324
+ releaseNotes = [{ language: "en-US", text: options.notes }];
320
325
  }
326
+ const result = await promoteRelease(client, packageName, options.from, options.to, {
327
+ status: options.status,
328
+ userFraction: options.rollout ? Number(options.rollout) / 100 : void 0,
329
+ releaseNotes
330
+ });
331
+ console.log(formatOutput(result, format));
321
332
  });
322
333
  const rollout = releases.command("rollout").description("Manage staged rollouts");
323
334
  for (const action of ["increase", "halt", "resume", "complete"]) {
@@ -354,8 +365,12 @@ function registerReleasesCommands(program) {
354
365
  if (action === "increase" && options.to !== void 0) {
355
366
  const to = Number(options.to);
356
367
  if (!Number.isFinite(to) || to < 1 || to > 100) {
357
- console.error(`Error: --to must be a number between 1 and 100 (got: ${options.to})`);
358
- process.exit(2);
368
+ throw new GpcError(
369
+ `--to must be a number between 1 and 100 (got: ${options.to})`,
370
+ "RELEASES_USAGE_ERROR",
371
+ 2,
372
+ "Use a percentage between 1 and 100."
373
+ );
359
374
  }
360
375
  }
361
376
  if (action === "halt") {
@@ -383,59 +398,56 @@ function registerReleasesCommands(program) {
383
398
  return;
384
399
  }
385
400
  const client = await getClient(config);
386
- try {
387
- const result = await updateRollout(
388
- client,
389
- packageName,
390
- options.track,
391
- action,
392
- options.to ? Number(options.to) / 100 : void 0
393
- );
394
- if (action === "increase" && options.vitalsGate) {
395
- const threshold = config.vitals?.thresholds?.crashRate;
396
- if (!threshold) {
397
- console.error(
398
- "Warning: --vitals-gate requires vitals.thresholds.crashRate in config. Skipping gate."
399
- );
400
- } else {
401
- try {
402
- const { auth: authConfig } = config;
403
- const vitalsAuth = await resolveAuth({
404
- serviceAccountPath: authConfig?.serviceAccount
405
- });
406
- const reportingClient = createReportingClient({ auth: vitalsAuth });
407
- const vitalsResult = await getVitalsCrashes(reportingClient, packageName, {
408
- days: 1
409
- });
410
- const latest = vitalsResult.data?.[0]?.crashRate;
411
- const check = checkThreshold(latest, threshold);
412
- if (check.breached) {
413
- await updateRollout(client, packageName, options.track, "halt");
414
- console.error(
415
- `Vitals gate: crash rate ${String(latest)}% > threshold ${String(threshold)}%. Rollout halted.`
416
- );
417
- process.exit(6);
418
- }
419
- } catch (vitalsErr) {
401
+ const result = await updateRollout(
402
+ client,
403
+ packageName,
404
+ options.track,
405
+ action,
406
+ options.to ? Number(options.to) / 100 : void 0
407
+ );
408
+ if (action === "increase" && options.vitalsGate) {
409
+ const threshold = config.vitals?.thresholds?.crashRate;
410
+ if (!threshold) {
411
+ console.error(
412
+ "Warning: --vitals-gate requires vitals.thresholds.crashRate in config. Skipping gate."
413
+ );
414
+ } else {
415
+ try {
416
+ const { auth: authConfig } = config;
417
+ const vitalsAuth = await resolveAuth({
418
+ serviceAccountPath: authConfig?.serviceAccount
419
+ });
420
+ const reportingClient = createReportingClient({ auth: vitalsAuth });
421
+ const vitalsResult = await getVitalsCrashes(reportingClient, packageName, {
422
+ days: 1
423
+ });
424
+ const latest = vitalsResult.data?.[0]?.crashRate;
425
+ const check = checkThreshold(latest, threshold);
426
+ if (check.breached) {
427
+ await updateRollout(client, packageName, options.track, "halt");
420
428
  console.error(
421
- `Warning: Vitals gate check failed: ${vitalsErr instanceof Error ? vitalsErr.message : String(vitalsErr)}`
429
+ `Vitals gate: crash rate ${String(latest)}% > threshold ${String(threshold)}%. Rollout halted.`
422
430
  );
431
+ process.exitCode = 6;
423
432
  }
433
+ } catch (vitalsErr) {
434
+ console.error(
435
+ `Warning: Vitals gate check failed: ${vitalsErr instanceof Error ? vitalsErr.message : String(vitalsErr)}`
436
+ );
424
437
  }
425
438
  }
426
- console.log(formatOutput(result, format));
427
- } catch (error) {
428
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
429
- process.exit(4);
430
439
  }
440
+ console.log(formatOutput(result, format));
431
441
  });
432
442
  }
433
443
  releases.command("notes").description("Get or set release notes").argument("<action>", "Action: get|set").option("--track <track>", "Track name").option("--lang <language>", "Language code", "en-US").option("--notes <text>", "Release notes text").option("--file <path>", "Read notes from file").action(async (action, options) => {
434
444
  if (action === "set") {
435
- console.error(
436
- "Error: gpc releases notes set is not implemented as a standalone command.\nUse --notes, --notes-dir, or --notes-from-git with:\n gpc releases upload\n gpc publish"
445
+ throw new GpcError(
446
+ "gpc releases notes set is not implemented as a standalone command.",
447
+ "RELEASES_USAGE_ERROR",
448
+ 1,
449
+ "Use --notes, --notes-dir, or --notes-from-git with: gpc releases upload or gpc publish"
437
450
  );
438
- process.exit(1);
439
451
  }
440
452
  if (action === "get") {
441
453
  const config = await loadConfig();
@@ -458,52 +470,46 @@ function registerReleasesCommands(program) {
458
470
  console.log(formatOutput(notes, format));
459
471
  return;
460
472
  }
461
- console.error("Usage: gpc releases notes <get|set> --track <track>");
462
- process.exit(2);
473
+ throw new GpcError(
474
+ "Unknown action. Usage: gpc releases notes <get|set> --track <track>",
475
+ "RELEASES_USAGE_ERROR",
476
+ 2,
477
+ "Use: gpc releases notes get --track <track>"
478
+ );
463
479
  });
464
480
  releases.command("upload-external").description("Upload an externally hosted APK configuration").requiredOption("--url <url>", "External URL where the APK is hosted").requiredOption("--file <config>", "Path to JSON config file with APK metadata").action(async (options) => {
465
481
  const config = await loadConfig();
466
482
  const packageName = resolvePackageName(program.opts()["app"], config);
467
483
  const format = getOutputFormat(program, config);
468
- try {
469
- const { readFile } = await import("fs/promises");
470
- const raw = await readFile(options.file, "utf-8");
471
- const apkConfig = JSON.parse(raw);
472
- apkConfig["externallyHostedUrl"] = options.url;
473
- const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });
474
- const client = createApiClient({ auth });
475
- const result = await uploadExternallyHosted(
476
- client,
477
- packageName,
478
- apkConfig
479
- );
480
- console.log(formatOutput(result, format));
481
- } catch (error) {
482
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
483
- process.exit(4);
484
- }
484
+ const { readFile } = await import("fs/promises");
485
+ const raw = await readFile(options.file, "utf-8");
486
+ const apkConfig = JSON.parse(raw);
487
+ apkConfig["externallyHostedUrl"] = options.url;
488
+ const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });
489
+ const client = createApiClient({ auth });
490
+ const result = await uploadExternallyHosted(
491
+ client,
492
+ packageName,
493
+ apkConfig
494
+ );
495
+ console.log(formatOutput(result, format));
485
496
  });
486
497
  releases.command("diff").description("Compare releases between two tracks").option("--from <track>", "Source track", "internal").option("--to <track>", "Target track", "production").action(async (options) => {
487
498
  const config = await loadConfig();
488
499
  const packageName = resolvePackageName(program.opts()["app"], config);
489
500
  const client = await getClient(config);
490
501
  const format = getOutputFormat(program, config);
491
- try {
492
- const result = await diffReleases(client, packageName, options.from, options.to);
493
- if (result.diffs.length === 0) {
494
- console.log(`No differences between ${result.fromTrack} and ${result.toTrack}.`);
502
+ const result = await diffReleases(client, packageName, options.from, options.to);
503
+ if (result.diffs.length === 0) {
504
+ console.log(`No differences between ${result.fromTrack} and ${result.toTrack}.`);
505
+ } else {
506
+ if (format === "json") {
507
+ console.log(formatOutput(result, format));
495
508
  } else {
496
- if (format === "json") {
497
- console.log(formatOutput(result, format));
498
- } else {
499
- console.log(`Differences: ${result.fromTrack} vs ${result.toTrack}
509
+ console.log(`Differences: ${result.fromTrack} vs ${result.toTrack}
500
510
  `);
501
- console.log(formatOutput(result.diffs, format));
502
- }
511
+ console.log(formatOutput(result.diffs, format));
503
512
  }
504
- } catch (error) {
505
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
506
- process.exit(4);
507
513
  }
508
514
  });
509
515
  releases.command("count").description("Count releases per track").option("--track <track>", "Filter to a specific track").action(async (options) => {
@@ -511,39 +517,34 @@ function registerReleasesCommands(program) {
511
517
  const packageName = resolvePackageName(program.opts()["app"], config);
512
518
  const client = await getClient(config);
513
519
  const format = getOutputFormat(program, config);
514
- try {
515
- const statuses = await getReleasesStatus(client, packageName, options.track);
516
- const trackCounts = /* @__PURE__ */ new Map();
517
- for (const r of statuses) {
518
- const entry = trackCounts.get(r.track) ?? { total: 0, statuses: {} };
519
- entry.total++;
520
- entry.statuses[r.status] = (entry.statuses[r.status] ?? 0) + 1;
521
- trackCounts.set(r.track, entry);
522
- }
523
- if (format === "json") {
524
- const data = Object.fromEntries(
525
- [...trackCounts.entries()].map(([track, info]) => [track, info])
526
- );
527
- console.log(formatOutput(data, format));
520
+ const statuses = await getReleasesStatus(client, packageName, options.track);
521
+ const trackCounts = /* @__PURE__ */ new Map();
522
+ for (const r of statuses) {
523
+ const entry = trackCounts.get(r.track) ?? { total: 0, statuses: {} };
524
+ entry.total++;
525
+ entry.statuses[r.status] = (entry.statuses[r.status] ?? 0) + 1;
526
+ trackCounts.set(r.track, entry);
527
+ }
528
+ if (format === "json") {
529
+ const data = Object.fromEntries(
530
+ [...trackCounts.entries()].map(([track, info]) => [track, info])
531
+ );
532
+ console.log(formatOutput(data, format));
533
+ } else {
534
+ const rows = [...trackCounts.entries()].map(([track, info]) => ({
535
+ track,
536
+ releases: info.total,
537
+ ...info.statuses
538
+ }));
539
+ if (rows.length === 0) {
540
+ console.log("No releases found.");
528
541
  } else {
529
- const rows = [...trackCounts.entries()].map(([track, info]) => ({
530
- track,
531
- releases: info.total,
532
- ...info.statuses
533
- }));
534
- if (rows.length === 0) {
535
- console.log("No releases found.");
536
- } else {
537
- console.log(formatOutput(rows, format));
538
- }
542
+ console.log(formatOutput(rows, format));
539
543
  }
540
- } catch (error) {
541
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
542
- process.exit(4);
543
544
  }
544
545
  });
545
546
  }
546
547
  export {
547
548
  registerReleasesCommands
548
549
  };
549
- //# sourceMappingURL=releases-C2WC2K4E.js.map
550
+ //# sourceMappingURL=releases-276W3BR7.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/releases.ts"],"sourcesContent":["import { appendFile, stat } from \"node:fs/promises\";\nimport { basename, extname } from \"node:path\";\nimport type { GpcConfig } from \"@gpc-cli/config\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { resolveAuth } from \"@gpc-cli/auth\";\nimport { createApiClient, createReportingClient } from \"@gpc-cli/api\";\nimport type { RetryLogEntry, ExternallyHostedApk, UploadProgressEvent } from \"@gpc-cli/api\";\nimport {\n uploadRelease,\n getReleasesStatus,\n promoteRelease,\n updateRollout,\n readReleaseNotesFromDir,\n generateNotesFromGit,\n writeAuditLog,\n createAuditEntry,\n uploadExternallyHosted,\n diffReleases,\n fetchReleaseNotes,\n getVitalsCrashes,\n checkThreshold,\n GpcError,\n} from \"@gpc-cli/core\";\nimport { formatOutput, sortResults, createSpinner } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport {\n isInteractive,\n promptSelect,\n promptInput,\n requireOption,\n requireConfirm,\n} from \"../prompt.js\";\n\nfunction resolvePackageName(packageArg: string | undefined, config: GpcConfig): string {\n const name = packageArg || config.app;\n if (!name) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Set a default app: gpc config set app <package>\",\n );\n }\n return name;\n}\n\nfunction createRetryLogger(retryLogPath?: string): ((entry: RetryLogEntry) => void) | undefined {\n if (!retryLogPath) return undefined;\n return (entry: RetryLogEntry) => {\n const line = JSON.stringify(entry) + \"\\n\";\n appendFile(retryLogPath, line).catch(() => {});\n };\n}\n\nasync function getClient(config: GpcConfig, retryLogPath?: string, uploadTimeout?: number) {\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n return createApiClient({ auth, onRetry: createRetryLogger(retryLogPath), uploadTimeout });\n}\n\nexport function registerReleasesCommands(program: Command): void {\n const releases = program.command(\"releases\").description(\"Manage releases and rollouts\");\n\n // Upload\n releases\n .command(\"upload <file>\")\n .description(\"Upload AAB/APK and assign to a track\")\n .option(\"--track <track>\", \"Target track\", \"internal\")\n .option(\"--rollout <percent>\", \"Staged rollout percentage (1-100)\")\n .option(\"--notes <text>\", \"Release notes (en-US)\")\n .option(\"--name <name>\", \"Release name\")\n .option(\"--mapping <file>\", \"ProGuard/R8 mapping file for deobfuscation\")\n .option(\"--notes-dir <dir>\", \"Read release notes from directory (<dir>/<lang>.txt)\")\n .option(\"--notes-from-git\", \"Generate release notes from git commit history\")\n .option(\"--copy-notes-from <track>\", \"Copy release notes from another track\")\n .option(\"--since <ref>\", \"Git ref to start from (tag, SHA) — used with --notes-from-git\")\n .option(\"--retry-log <path>\", \"Write retry log entries to file (JSONL)\")\n .option(\n \"--timeout <ms>\",\n \"Upload timeout in milliseconds (auto-scales with file size by default)\",\n parseInt,\n )\n .option(\"--status <status>\", \"Release status: completed, inProgress, draft, halted\", \"completed\")\n .action(async (file: string, options) => {\n try {\n await stat(file);\n } catch {\n throw new GpcError(\n `File not found: ${file}`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Check the file path and try again.\",\n );\n }\n\n const ext = extname(file).toLowerCase();\n if (ext !== \".aab\" && ext !== \".apk\") {\n throw new GpcError(\n `Expected .aab or .apk file, got \"${ext || \"(no extension)\"}\"`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Provide a .aab or .apk file.\",\n );\n }\n\n const noteSources = [\n options.notes,\n options.notesDir,\n options.notesFromGit,\n options.copyNotesFrom,\n ].filter(Boolean);\n if (noteSources.length > 1) {\n throw new GpcError(\n \"Cannot combine --notes, --notes-dir, --notes-from-git, and --copy-notes-from. Use only one.\",\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Pick one release notes source.\",\n );\n }\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n // Interactive mode: prompt for missing options\n if (isInteractive(program)) {\n if (!options.track || options.track === \"internal\") {\n const tracks = [\"internal\", \"alpha\", \"beta\", \"production\"];\n options.track = await promptSelect(\"Select track:\", tracks, \"internal\");\n }\n\n if (!options.rollout && options.track === \"production\") {\n const rolloutStr = await promptInput(\n \"Staged rollout percentage (1-100, blank for full)\",\n \"100\",\n );\n if (rolloutStr && rolloutStr !== \"100\") {\n options.rollout = rolloutStr;\n }\n }\n\n if (!options.notes && !options.notesDir) {\n const notes = await promptInput(\"Release notes (en-US, blank to skip)\");\n if (notes) options.notes = notes;\n }\n }\n\n if (options.rollout !== undefined) {\n const rollout = Number(options.rollout);\n if (!Number.isFinite(rollout) || rollout < 1 || rollout > 100) {\n throw new GpcError(\n `--rollout must be a number between 1 and 100 (got: ${options.rollout})`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Use a percentage between 1 and 100.\",\n );\n }\n }\n\n const { size: fileSize } = await stat(file);\n const jsonMode = format === \"json\";\n const client = await getClient(config, options.retryLog, options.timeout);\n\n const showProgress = !jsonMode && process.stderr.isTTY && !program.opts()[\"quiet\"];\n const sizeMB = (fileSize / (1024 * 1024)).toFixed(1);\n\n function formatBytes(bytes: number): string {\n if (bytes >= 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} GB`;\n if (bytes >= 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n if (bytes >= 1024) return `${(bytes / 1024).toFixed(1)} KB`;\n return `${bytes} B`;\n }\n\n const BAR_WIDTH = 20;\n const onUploadProgress = showProgress\n ? (event: UploadProgressEvent) => {\n const filled = Math.round((event.percent / 100) * BAR_WIDTH);\n const bar = \"█\".repeat(filled) + \"░\".repeat(BAR_WIDTH - filled);\n const uploaded = formatBytes(event.bytesUploaded);\n const total = formatBytes(event.totalBytes);\n const speed =\n event.bytesPerSecond > 0 ? `${formatBytes(event.bytesPerSecond)}/s` : \"...\";\n const eta = event.etaSeconds > 0 ? `ETA ${event.etaSeconds}s` : \"\";\n process.stderr.write(\n `\\r ${bar} ${event.percent}% ${uploaded}/${total} ${speed} ${eta}\\x1b[K`,\n );\n }\n : undefined;\n\n if (isDryRun(program)) {\n const result = await uploadRelease(client, packageName, file, {\n track: options.track,\n userFraction: options.rollout ? Number(options.rollout) / 100 : undefined,\n dryRun: true,\n });\n console.log(formatOutput(result, format));\n return;\n }\n\n const auditEntry = createAuditEntry(\n \"releases upload\",\n {\n file,\n track: options.track,\n rollout: options.rollout,\n },\n packageName,\n );\n\n const spinner = createSpinner(`Uploading ${basename(file)} (${sizeMB} MB)...`);\n if (!showProgress) spinner.start();\n\n try {\n let releaseNotes: { language: string; text: string }[] | undefined;\n if (options.copyNotesFrom) {\n releaseNotes = await fetchReleaseNotes(client, packageName, options.copyNotesFrom);\n } else if (options.notesFromGit) {\n const gitNotes = await generateNotesFromGit({ since: options.since });\n releaseNotes = [{ language: gitNotes.language, text: gitNotes.text }];\n } else if (options.notesDir) {\n releaseNotes = await readReleaseNotesFromDir(options.notesDir);\n } else if (options.notes) {\n releaseNotes = [{ language: \"en-US\", text: options.notes }];\n }\n\n const result = await uploadRelease(client, packageName, file, {\n track: options.track,\n status: options.status,\n userFraction: options.rollout ? Number(options.rollout) / 100 : undefined,\n releaseNotes,\n releaseName: options.name,\n mappingFile: options.mapping,\n onUploadProgress,\n });\n if (showProgress) {\n process.stderr.write(`\\r ✓ Uploaded ${basename(file)} ${sizeMB} MB\\x1b[K\\n`);\n }\n spinner.stop(\"Upload complete\");\n console.log(formatOutput(result, format));\n auditEntry.success = true;\n } catch (error) {\n if (showProgress) {\n process.stderr.write(\"\\n\");\n }\n spinner.fail(\"Upload failed\");\n auditEntry.success = false;\n auditEntry.error = error instanceof Error ? error.message : String(error);\n throw error;\n } finally {\n auditEntry.durationMs = Date.now() - new Date(auditEntry.timestamp).getTime();\n writeAuditLog(auditEntry).catch(() => {});\n }\n });\n\n // Status\n releases\n .command(\"status\")\n .description(\"Show current release status across tracks\")\n .option(\"--track <track>\", \"Filter by track\")\n .option(\"--sort <field>\", \"Sort by field (prefix with - for descending)\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n\n const TRACK_ORDER = [\"production\", \"beta\", \"alpha\", \"internal\"];\n const rawStatuses = await getReleasesStatus(client, packageName, options.track);\n const statuses = options.track\n ? Array.isArray(rawStatuses)\n ? rawStatuses.filter((s: any) => s.track === options.track)\n : rawStatuses\n : rawStatuses;\n const sorted = Array.isArray(statuses)\n ? options.sort\n ? sortResults(statuses, options.sort)\n : [...statuses].sort((a, b) => {\n const ai = TRACK_ORDER.indexOf(\n String((a as unknown as Record<string, unknown>)[\"track\"] ?? \"\"),\n );\n const bi = TRACK_ORDER.indexOf(\n String((b as unknown as Record<string, unknown>)[\"track\"] ?? \"\"),\n );\n return (ai === -1 ? 99 : ai) - (bi === -1 ? 99 : bi);\n })\n : statuses;\n if (format !== \"json\" && Array.isArray(sorted)) {\n const rows = sorted.map((s: unknown) => {\n const sr = s as Record<string, unknown>;\n return {\n track: sr[\"track\"] || \"-\",\n status: sr[\"status\"] || \"-\",\n name: sr[\"name\"] || \"-\",\n versionCodes: Array.isArray(sr[\"versionCodes\"])\n ? (sr[\"versionCodes\"] as unknown[]).join(\", \")\n : \"-\",\n userFraction:\n sr[\"userFraction\"] !== undefined\n ? `${Math.round(Number(sr[\"userFraction\"]) * 100)}%`\n : \"—\",\n };\n });\n console.log(formatOutput(rows, format));\n } else {\n console.log(formatOutput(sorted, format));\n }\n });\n\n // Promote\n releases\n .command(\"promote\")\n .description(\"Promote a release from one track to another\")\n .option(\"--from <track>\", \"Source track\")\n .option(\"--to <track>\", \"Target track\")\n .option(\"--rollout <percent>\", \"Staged rollout percentage\")\n .option(\"--notes <text>\", \"Release notes\")\n .option(\"--copy-notes-from <track>\", \"Copy release notes from another track\")\n .option(\"--status <status>\", \"Release status: completed, inProgress, draft, halted\")\n .action(async (options) => {\n if (options.notes && options.copyNotesFrom) {\n throw new GpcError(\n \"Cannot combine --notes and --copy-notes-from. Use only one.\",\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Pick one release notes source.\",\n );\n }\n\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const interactive = isInteractive(program);\n const tracks = [\"internal\", \"alpha\", \"beta\", \"production\"];\n\n options.from = await requireOption(\n \"from\",\n options.from,\n {\n message: \"Source track:\",\n choices: tracks,\n },\n interactive,\n );\n\n options.to = await requireOption(\n \"to\",\n options.to,\n {\n message: \"Target track:\",\n choices: tracks.filter((t: string) => t !== options.from),\n },\n interactive,\n );\n\n if (options.from === options.to) {\n throw new GpcError(\n `--from and --to must be different tracks (both are \"${options.from}\")`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Specify different source and target tracks.\",\n );\n }\n\n if (options.rollout !== undefined) {\n const rollout = Number(options.rollout);\n if (!Number.isFinite(rollout) || rollout < 1 || rollout > 100) {\n throw new GpcError(\n `--rollout must be a number between 1 and 100 (got: ${options.rollout})`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Use a percentage between 1 and 100.\",\n );\n }\n }\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"releases promote\",\n action: \"promote\",\n target: `${options.from} → ${options.to}`,\n details: { rollout: options.rollout },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n let releaseNotes: { language: string; text: string }[] | undefined;\n if (options.copyNotesFrom) {\n releaseNotes = await fetchReleaseNotes(client, packageName, options.copyNotesFrom);\n } else if (options.notes) {\n releaseNotes = [{ language: \"en-US\", text: options.notes }];\n }\n\n const result = await promoteRelease(client, packageName, options.from, options.to, {\n status: options.status,\n userFraction: options.rollout ? Number(options.rollout) / 100 : undefined,\n releaseNotes,\n });\n console.log(formatOutput(result, format));\n });\n\n // Rollout subcommands\n const rollout = releases.command(\"rollout\").description(\"Manage staged rollouts\");\n\n for (const action of [\"increase\", \"halt\", \"resume\", \"complete\"] as const) {\n const cmd = rollout\n .command(action)\n .description(`${action.charAt(0).toUpperCase() + action.slice(1)} a staged rollout`)\n .option(\"--track <track>\", \"Track name\");\n\n if (action === \"increase\") {\n cmd.option(\"--to <percent>\", \"New rollout percentage\");\n cmd.option(\"--vitals-gate\", \"Halt rollout if crash rate exceeds configured threshold\");\n }\n\n cmd.action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const interactive = isInteractive(program);\n const tracks = [\"internal\", \"alpha\", \"beta\", \"production\"];\n\n options.track = await requireOption(\n \"track\",\n options.track,\n {\n message: \"Track:\",\n choices: tracks,\n },\n interactive,\n );\n\n if (action === \"increase\") {\n options.to = await requireOption(\n \"to\",\n options.to,\n {\n message: \"New rollout percentage (1-100):\",\n },\n interactive,\n );\n }\n\n if (action === \"increase\" && options.to !== undefined) {\n const to = Number(options.to);\n if (!Number.isFinite(to) || to < 1 || to > 100) {\n throw new GpcError(\n `--to must be a number between 1 and 100 (got: ${options.to})`,\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Use a percentage between 1 and 100.\",\n );\n }\n }\n\n // Require confirmation for destructive rollout halt\n if (action === \"halt\") {\n await requireConfirm(\n `Halt rollout on track \"${options.track}\" for ${packageName}?`,\n program,\n );\n }\n\n if (isDryRun(program)) {\n if (action === \"increase\" && options.vitalsGate) {\n console.error(\n \"Warning: --vitals-gate is ignored in --dry-run mode. Gate will run on live execution.\",\n );\n }\n printDryRun(\n {\n command: `releases rollout ${action}`,\n action: action,\n target: options.track,\n details: { percentage: options.to !== undefined ? `${options.to}%` : undefined },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await updateRollout(\n client,\n packageName,\n options.track,\n action,\n options.to ? Number(options.to) / 100 : undefined,\n );\n\n // Vitals gate: check crash rate after rollout increase\n if (action === \"increase\" && options.vitalsGate) {\n const threshold = (config as any).vitals?.thresholds?.crashRate;\n if (!threshold) {\n console.error(\n \"Warning: --vitals-gate requires vitals.thresholds.crashRate in config. Skipping gate.\",\n );\n } else {\n try {\n const { auth: authConfig } = config;\n const vitalsAuth = await resolveAuth({\n serviceAccountPath: authConfig?.serviceAccount,\n });\n const reportingClient = createReportingClient({ auth: vitalsAuth });\n const vitalsResult = await getVitalsCrashes(reportingClient, packageName, {\n days: 1,\n });\n const latest = (vitalsResult as any).data?.[0]?.crashRate;\n const check = checkThreshold(latest, threshold);\n if (check.breached) {\n await updateRollout(client, packageName, options.track, \"halt\");\n console.error(\n `Vitals gate: crash rate ${String(latest)}% > threshold ${String(threshold)}%. Rollout halted.`,\n );\n process.exitCode = 6;\n }\n } catch (vitalsErr) {\n console.error(\n `Warning: Vitals gate check failed: ${vitalsErr instanceof Error ? vitalsErr.message : String(vitalsErr)}`,\n );\n }\n }\n }\n\n console.log(formatOutput(result, format));\n });\n }\n\n // Release notes\n releases\n .command(\"notes\")\n .description(\"Get or set release notes\")\n .argument(\"<action>\", \"Action: get|set\")\n .option(\"--track <track>\", \"Track name\")\n .option(\"--lang <language>\", \"Language code\", \"en-US\")\n .option(\"--notes <text>\", \"Release notes text\")\n .option(\"--file <path>\", \"Read notes from file\")\n .action(async (action: string, options) => {\n if (action === \"set\") {\n throw new GpcError(\n \"gpc releases notes set is not implemented as a standalone command.\",\n \"RELEASES_USAGE_ERROR\",\n 1,\n \"Use --notes, --notes-dir, or --notes-from-git with: gpc releases upload or gpc publish\",\n );\n }\n\n if (action === \"get\") {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n const track = options.track ?? \"internal\";\n\n // Try getReleasesStatus first (has all releases), then fallback to fetchReleaseNotes\n const statuses = await getReleasesStatus(client, packageName, track);\n let notes = Array.isArray(statuses)\n ? statuses.flatMap((s: any) => s.releaseNotes ?? [])\n : ((statuses as any).releaseNotes ?? []);\n\n // Fallback: fetchReleaseNotes reads the raw track data which may have notes\n // even when getReleasesStatus doesn't (e.g. completed releases)\n if (notes.length === 0) {\n try {\n notes = await fetchReleaseNotes(client, packageName, track);\n } catch {\n // No release found on track — fall through to empty message\n }\n }\n\n if (notes.length === 0) {\n console.log(`No release notes found on track \"${track}\".`);\n return;\n }\n console.log(formatOutput(notes, format));\n return;\n }\n\n throw new GpcError(\n \"Unknown action. Usage: gpc releases notes <get|set> --track <track>\",\n \"RELEASES_USAGE_ERROR\",\n 2,\n \"Use: gpc releases notes get --track <track>\",\n );\n });\n\n // Upload externally hosted APK\n releases\n .command(\"upload-external\")\n .description(\"Upload an externally hosted APK configuration\")\n .requiredOption(\"--url <url>\", \"External URL where the APK is hosted\")\n .requiredOption(\"--file <config>\", \"Path to JSON config file with APK metadata\")\n .action(async (options: { url: string; file: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n const { readFile } = await import(\"node:fs/promises\");\n const raw = await readFile(options.file, \"utf-8\");\n const apkConfig = JSON.parse(raw) as Record<string, unknown>;\n\n // Override with CLI-provided URL\n apkConfig[\"externallyHostedUrl\"] = options.url;\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const result = await uploadExternallyHosted(\n client,\n packageName,\n apkConfig as unknown as ExternallyHostedApk,\n );\n console.log(formatOutput(result, format));\n });\n\n // Diff\n releases\n .command(\"diff\")\n .description(\"Compare releases between two tracks\")\n .option(\"--from <track>\", \"Source track\", \"internal\")\n .option(\"--to <track>\", \"Target track\", \"production\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n\n const result = await diffReleases(client, packageName, options.from, options.to);\n if (result.diffs.length === 0) {\n console.log(`No differences between ${result.fromTrack} and ${result.toTrack}.`);\n } else {\n if (format === \"json\") {\n console.log(formatOutput(result, format));\n } else {\n console.log(`Differences: ${result.fromTrack} vs ${result.toTrack}\\n`);\n console.log(formatOutput(result.diffs, format));\n }\n }\n });\n\n // Count\n releases\n .command(\"count\")\n .description(\"Count releases per track\")\n .option(\"--track <track>\", \"Filter to a specific track\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n\n const statuses = await getReleasesStatus(client, packageName, options.track);\n\n // Group by track\n const trackCounts = new Map<string, { total: number; statuses: Record<string, number> }>();\n for (const r of statuses) {\n const entry = trackCounts.get(r.track) ?? { total: 0, statuses: {} };\n entry.total++;\n entry.statuses[r.status] = (entry.statuses[r.status] ?? 0) + 1;\n trackCounts.set(r.track, entry);\n }\n\n if (format === \"json\") {\n const data = Object.fromEntries(\n [...trackCounts.entries()].map(([track, info]) => [track, info]),\n );\n console.log(formatOutput(data, format));\n } else {\n const rows = [...trackCounts.entries()].map(([track, info]) => ({\n track,\n releases: info.total,\n ...info.statuses,\n }));\n if (rows.length === 0) {\n console.log(\"No releases found.\");\n } else {\n console.log(formatOutput(rows, format));\n }\n }\n });\n}\n"],"mappings":"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","names":["rollout"]}