@gpc-cli/cli 0.9.44 → 0.9.46

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 (168) hide show
  1. package/dist/{anomalies-NU2IN2GJ.js → anomalies-UDE4NGHJ.js} +19 -24
  2. package/dist/anomalies-UDE4NGHJ.js.map +1 -0
  3. package/dist/{apps-J2446UDA.js → apps-FKD3ZG5X.js} +31 -35
  4. package/dist/apps-FKD3ZG5X.js.map +1 -0
  5. package/dist/{audit-N2CRHWUN.js → audit-JASSHRWN.js} +47 -62
  6. package/dist/audit-JASSHRWN.js.map +1 -0
  7. package/dist/{auth-XGSTT5G5.js → auth-OTA3SV3J.js} +145 -103
  8. package/dist/auth-OTA3SV3J.js.map +1 -0
  9. package/dist/bin.js +6 -4
  10. package/dist/bin.js.map +1 -1
  11. package/dist/bundle-F7MUVC5J.js +204 -0
  12. package/dist/bundle-F7MUVC5J.js.map +1 -0
  13. package/dist/{cache-SLNFRTI2.js → cache-XKPLZYEB.js} +4 -5
  14. package/dist/cache-XKPLZYEB.js.map +1 -0
  15. package/dist/changelog-7COFZO7Q.js +48 -0
  16. package/dist/changelog-7COFZO7Q.js.map +1 -0
  17. package/dist/{chunk-4O4D5SGL.js → chunk-3SJ6OXCZ.js} +4 -5
  18. package/dist/chunk-3SJ6OXCZ.js.map +1 -0
  19. package/dist/{chunk-7LURVNQV.js → chunk-6OWN6S6X.js} +53 -49
  20. package/dist/{chunk-7LURVNQV.js.map → chunk-6OWN6S6X.js.map} +1 -1
  21. package/dist/{chunk-U6ZTQ34I.js → chunk-BCBXQC7J.js} +45 -11
  22. package/dist/chunk-BCBXQC7J.js.map +1 -0
  23. package/dist/{chunk-AA577WVQ.js → chunk-NQH4G7BI.js} +9 -3
  24. package/dist/chunk-NQH4G7BI.js.map +1 -0
  25. package/dist/chunk-SLNJEAMK.js +23 -0
  26. package/dist/chunk-SLNJEAMK.js.map +1 -0
  27. package/dist/{chunk-NV75I5VP.js → chunk-YFUBD2XB.js} +10 -8
  28. package/dist/chunk-YFUBD2XB.js.map +1 -0
  29. package/dist/{config-222P3MKK.js → config-2FTCYEGD.js} +8 -5
  30. package/dist/config-2FTCYEGD.js.map +1 -0
  31. package/dist/{data-safety-Q7FTCEWU.js → data-safety-AFMD6MYI.js} +12 -27
  32. package/dist/data-safety-AFMD6MYI.js.map +1 -0
  33. package/dist/{device-tiers-MIOQEXYY.js → device-tiers-AQAMUQXI.js} +23 -38
  34. package/dist/device-tiers-AQAMUQXI.js.map +1 -0
  35. package/dist/diff-6EO4ID6W.js +91 -0
  36. package/dist/diff-6EO4ID6W.js.map +1 -0
  37. package/dist/{docs-7DUXIKA3.js → docs-4D2SJ4LY.js} +4 -3
  38. package/dist/docs-4D2SJ4LY.js.map +1 -0
  39. package/dist/doctor-H4X7Q57B.js +691 -0
  40. package/dist/doctor-H4X7Q57B.js.map +1 -0
  41. package/dist/{enterprise-7THXNBTC.js → enterprise-7PWXMSUN.js} +11 -21
  42. package/dist/enterprise-7PWXMSUN.js.map +1 -0
  43. package/dist/{external-transactions-2GWIMUVM.js → external-transactions-LCZALS3V.js} +12 -28
  44. package/dist/external-transactions-LCZALS3V.js.map +1 -0
  45. package/dist/{feedback-2W2XJGZX.js → feedback-XP765TOO.js} +4 -4
  46. package/dist/{games-BT777WUO.js → games-ZSNGEI7A.js} +17 -32
  47. package/dist/games-ZSNGEI7A.js.map +1 -0
  48. package/dist/{generated-apks-RJWTIX7L.js → generated-apks-RX2IUWSF.js} +30 -38
  49. package/dist/generated-apks-RX2IUWSF.js.map +1 -0
  50. package/dist/{grants-TKQJ3IER.js → grants-EBPECI26.js} +22 -40
  51. package/dist/grants-EBPECI26.js.map +1 -0
  52. package/dist/{iap-ICAEQLK5.js → iap-OUI5YYN4.js} +30 -51
  53. package/dist/iap-OUI5YYN4.js.map +1 -0
  54. package/dist/index.js +1 -1
  55. package/dist/{init-JZ2THPMS.js → init-WSTQTJOD.js} +5 -4
  56. package/dist/init-WSTQTJOD.js.map +1 -0
  57. package/dist/{install-skills-OV4HVANW.js → install-skills-6QDUXI5F.js} +5 -6
  58. package/dist/{install-skills-OV4HVANW.js.map → install-skills-6QDUXI5F.js.map} +1 -1
  59. package/dist/{internal-sharing-3U2XFHA4.js → internal-sharing-ONNIWIAT.js} +3 -4
  60. package/dist/{internal-sharing-3U2XFHA4.js.map → internal-sharing-ONNIWIAT.js.map} +1 -1
  61. package/dist/{listings-77HZW4S5.js → listings-7SGQ4SRX.js} +118 -157
  62. package/dist/listings-7SGQ4SRX.js.map +1 -0
  63. package/dist/migrate-ZQCJGQQS.js +138 -0
  64. package/dist/migrate-ZQCJGQQS.js.map +1 -0
  65. package/dist/{one-time-products-LHZAXQES.js → one-time-products-MGZTU7OM.js} +65 -120
  66. package/dist/one-time-products-MGZTU7OM.js.map +1 -0
  67. package/dist/{preflight-H3HEBYQW.js → preflight-N7ZRG2JI.js} +58 -55
  68. package/dist/preflight-N7ZRG2JI.js.map +1 -0
  69. package/dist/{pricing-XQSDTTK5.js → pricing-JJZFICFL.js} +8 -8
  70. package/dist/{pricing-XQSDTTK5.js.map → pricing-JJZFICFL.js.map} +1 -1
  71. package/dist/{prompt-BSV22CQZ.js → prompt-GXC2JSLA.js} +2 -2
  72. package/dist/{publish-Q5ZKEKZ5.js → publish-JPTI4EBT.js} +34 -30
  73. package/dist/publish-JPTI4EBT.js.map +1 -0
  74. package/dist/{purchase-options-CKRN4VIW.js → purchase-options-KFWW4JW2.js} +16 -11
  75. package/dist/purchase-options-KFWW4JW2.js.map +1 -0
  76. package/dist/purchases-DAWTMXP6.js +383 -0
  77. package/dist/purchases-DAWTMXP6.js.map +1 -0
  78. package/dist/{quickstart-4HB62YEL.js → quickstart-Z5Y3FYJU.js} +5 -3
  79. package/dist/quickstart-Z5Y3FYJU.js.map +1 -0
  80. package/dist/{quota-UHIQQYOY.js → quota-MZRWYJGR.js} +5 -15
  81. package/dist/quota-MZRWYJGR.js.map +1 -0
  82. package/dist/{recovery-5EV2R476.js → recovery-YE3Z7NIN.js} +32 -61
  83. package/dist/recovery-YE3Z7NIN.js.map +1 -0
  84. package/dist/{releases-C2WC2K4E.js → releases-2I3WBULC.js} +184 -185
  85. package/dist/releases-2I3WBULC.js.map +1 -0
  86. package/dist/{reports-2YX3RDOS.js → reports-CIB2T3XT.js} +19 -21
  87. package/dist/reports-CIB2T3XT.js.map +1 -0
  88. package/dist/reviews-BCCXIQ6C.js +188 -0
  89. package/dist/reviews-BCCXIQ6C.js.map +1 -0
  90. package/dist/{status-WHGLODGV.js → status-6LH5W4FU.js} +105 -83
  91. package/dist/status-6LH5W4FU.js.map +1 -0
  92. package/dist/{subscriptions-CI3JH3VQ.js → subscriptions-DZP3Y7O7.js} +142 -232
  93. package/dist/subscriptions-DZP3Y7O7.js.map +1 -0
  94. package/dist/{testers-NZOFA3EF.js → testers-LSMBXCA2.js} +24 -44
  95. package/dist/testers-LSMBXCA2.js.map +1 -0
  96. package/dist/tracks-YHMO2A6B.js +98 -0
  97. package/dist/tracks-YHMO2A6B.js.map +1 -0
  98. package/dist/{train-XKE4JN3Y.js → train-MDD2EBHS.js} +35 -55
  99. package/dist/train-MDD2EBHS.js.map +1 -0
  100. package/dist/{update-QMPRL5Y6.js → update-OMALGIBR.js} +30 -15
  101. package/dist/update-OMALGIBR.js.map +1 -0
  102. package/dist/{users-2YTC4Q36.js → users-UKG7VIQH.js} +45 -67
  103. package/dist/users-UKG7VIQH.js.map +1 -0
  104. package/dist/{validate-UOVTM6L3.js → validate-QIYSA3N7.js} +8 -10
  105. package/dist/validate-QIYSA3N7.js.map +1 -0
  106. package/dist/{version-NK5SJLHJ.js → version-NCSNXNVN.js} +4 -4
  107. package/dist/{vitals-A4CS4MSS.js → vitals-C23L2Y2E.js} +153 -172
  108. package/dist/vitals-C23L2Y2E.js.map +1 -0
  109. package/package.json +6 -6
  110. package/dist/anomalies-NU2IN2GJ.js.map +0 -1
  111. package/dist/apps-J2446UDA.js.map +0 -1
  112. package/dist/audit-N2CRHWUN.js.map +0 -1
  113. package/dist/auth-XGSTT5G5.js.map +0 -1
  114. package/dist/bundle-F43TD2BQ.js +0 -218
  115. package/dist/bundle-F43TD2BQ.js.map +0 -1
  116. package/dist/cache-SLNFRTI2.js.map +0 -1
  117. package/dist/changelog-OYUZOCOL.js +0 -53
  118. package/dist/changelog-OYUZOCOL.js.map +0 -1
  119. package/dist/chunk-4O4D5SGL.js.map +0 -1
  120. package/dist/chunk-AA577WVQ.js.map +0 -1
  121. package/dist/chunk-FWKYRLKY.js +0 -19
  122. package/dist/chunk-FWKYRLKY.js.map +0 -1
  123. package/dist/chunk-NV75I5VP.js.map +0 -1
  124. package/dist/chunk-U6ZTQ34I.js.map +0 -1
  125. package/dist/config-222P3MKK.js.map +0 -1
  126. package/dist/data-safety-Q7FTCEWU.js.map +0 -1
  127. package/dist/device-tiers-MIOQEXYY.js.map +0 -1
  128. package/dist/diff-V77SMKAQ.js +0 -96
  129. package/dist/diff-V77SMKAQ.js.map +0 -1
  130. package/dist/docs-7DUXIKA3.js.map +0 -1
  131. package/dist/doctor-3Z4ARPM2.js +0 -372
  132. package/dist/doctor-3Z4ARPM2.js.map +0 -1
  133. package/dist/enterprise-7THXNBTC.js.map +0 -1
  134. package/dist/external-transactions-2GWIMUVM.js.map +0 -1
  135. package/dist/games-BT777WUO.js.map +0 -1
  136. package/dist/generated-apks-RJWTIX7L.js.map +0 -1
  137. package/dist/grants-TKQJ3IER.js.map +0 -1
  138. package/dist/iap-ICAEQLK5.js.map +0 -1
  139. package/dist/init-JZ2THPMS.js.map +0 -1
  140. package/dist/listings-77HZW4S5.js.map +0 -1
  141. package/dist/migrate-SQT6RD6T.js +0 -143
  142. package/dist/migrate-SQT6RD6T.js.map +0 -1
  143. package/dist/one-time-products-LHZAXQES.js.map +0 -1
  144. package/dist/preflight-H3HEBYQW.js.map +0 -1
  145. package/dist/publish-Q5ZKEKZ5.js.map +0 -1
  146. package/dist/purchase-options-CKRN4VIW.js.map +0 -1
  147. package/dist/purchases-HSMCOG4A.js +0 -330
  148. package/dist/purchases-HSMCOG4A.js.map +0 -1
  149. package/dist/quickstart-4HB62YEL.js.map +0 -1
  150. package/dist/quota-UHIQQYOY.js.map +0 -1
  151. package/dist/recovery-5EV2R476.js.map +0 -1
  152. package/dist/releases-C2WC2K4E.js.map +0 -1
  153. package/dist/reports-2YX3RDOS.js.map +0 -1
  154. package/dist/reviews-2CWOI5CV.js +0 -213
  155. package/dist/reviews-2CWOI5CV.js.map +0 -1
  156. package/dist/status-WHGLODGV.js.map +0 -1
  157. package/dist/subscriptions-CI3JH3VQ.js.map +0 -1
  158. package/dist/testers-NZOFA3EF.js.map +0 -1
  159. package/dist/tracks-NERFFEDT.js +0 -107
  160. package/dist/tracks-NERFFEDT.js.map +0 -1
  161. package/dist/train-XKE4JN3Y.js.map +0 -1
  162. package/dist/update-QMPRL5Y6.js.map +0 -1
  163. package/dist/users-2YTC4Q36.js.map +0 -1
  164. package/dist/validate-UOVTM6L3.js.map +0 -1
  165. package/dist/vitals-A4CS4MSS.js.map +0 -1
  166. /package/dist/{feedback-2W2XJGZX.js.map → feedback-XP765TOO.js.map} +0 -0
  167. /package/dist/{prompt-BSV22CQZ.js.map → prompt-GXC2JSLA.js.map} +0 -0
  168. /package/dist/{version-NK5SJLHJ.js.map → version-NCSNXNVN.js.map} +0 -0
@@ -2,7 +2,7 @@
2
2
  import {
3
3
  getClient,
4
4
  resolvePackageName
5
- } from "./chunk-AA577WVQ.js";
5
+ } from "./chunk-NQH4G7BI.js";
6
6
  import {
7
7
  isDryRun,
8
8
  printDryRun
@@ -18,7 +18,7 @@ import {
18
18
  isInteractive,
19
19
  requireConfirm,
20
20
  requireOption
21
- } from "./chunk-NV75I5VP.js";
21
+ } from "./chunk-YFUBD2XB.js";
22
22
 
23
23
  // src/commands/listings.ts
24
24
  import { loadConfig } from "@gpc-cli/config";
@@ -37,7 +37,8 @@ import {
37
37
  exportImages,
38
38
  getCountryAvailability,
39
39
  formatOutput,
40
- createSpinner
40
+ createSpinner,
41
+ GpcError
41
42
  } from "@gpc-cli/core";
42
43
  var VALID_IMAGE_TYPES = [
43
44
  "phoneScreenshots",
@@ -51,9 +52,12 @@ var VALID_IMAGE_TYPES = [
51
52
  ];
52
53
  function validateImageType(type) {
53
54
  if (!VALID_IMAGE_TYPES.includes(type)) {
54
- console.error(`Error: Invalid image type "${type}".`);
55
- console.error(`Valid types: ${VALID_IMAGE_TYPES.join(", ")}`);
56
- process.exit(2);
55
+ throw new GpcError(
56
+ `Invalid image type "${type}". Valid types: ${VALID_IMAGE_TYPES.join(", ")}`,
57
+ "LISTINGS_USAGE_ERROR",
58
+ 2,
59
+ `Use one of: ${VALID_IMAGE_TYPES.join(", ")}`
60
+ );
57
61
  }
58
62
  return type;
59
63
  }
@@ -64,13 +68,8 @@ function registerListingsCommands(program) {
64
68
  const packageName = resolvePackageName(program.opts()["app"], config);
65
69
  const client = await getClient(config);
66
70
  const format = getOutputFormat(program, config);
67
- try {
68
- const result = await getListings(client, packageName, options.lang);
69
- console.log(formatOutput(result, format));
70
- } catch (error) {
71
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
72
- process.exit(4);
73
- }
71
+ const result = await getListings(client, packageName, options.lang);
72
+ console.log(formatOutput(result, format));
74
73
  });
75
74
  listings.command("update").description("Update a store listing").option("--lang <language>", "Language code (BCP 47)").option("--title <text>", "App title").option("--short <text>", "Short description").option("--full <text>", "Full description").option("--full-file <path>", "Read full description from file").option("--video <url>", "Video URL").action(async (options) => {
76
75
  const config = await loadConfig();
@@ -86,42 +85,39 @@ function registerListingsCommands(program) {
86
85
  interactive
87
86
  );
88
87
  const format = getOutputFormat(program, config);
89
- try {
90
- const data = {};
91
- if (options["title"]) data["title"] = options["title"];
92
- if (options["short"]) data["shortDescription"] = options["short"];
93
- if (options["full"]) data["fullDescription"] = options["full"];
94
- if (options["fullFile"]) {
95
- const { readFile } = await import("fs/promises");
96
- data["fullDescription"] = (await readFile(options["fullFile"], "utf-8")).trimEnd();
97
- }
98
- if (options["video"]) data["video"] = options["video"];
99
- if (Object.keys(data).length === 0) {
100
- console.error(
101
- "Error: Provide at least one field to update (--title, --short, --full, --full-file, --video)."
102
- );
103
- process.exit(2);
104
- }
105
- if (isDryRun(program)) {
106
- printDryRun(
107
- {
108
- command: "listings update",
109
- action: "update listing for",
110
- target: options.lang,
111
- details: data
112
- },
113
- format,
114
- formatOutput
115
- );
116
- return;
117
- }
118
- const client = await getClient(config);
119
- const result = await updateListing(client, packageName, options.lang, data);
120
- console.log(formatOutput(result, format));
121
- } catch (error) {
122
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
123
- process.exit(4);
88
+ const data = {};
89
+ if (options["title"]) data["title"] = options["title"];
90
+ if (options["short"]) data["shortDescription"] = options["short"];
91
+ if (options["full"]) data["fullDescription"] = options["full"];
92
+ if (options["fullFile"]) {
93
+ const { readFile } = await import("fs/promises");
94
+ data["fullDescription"] = (await readFile(options["fullFile"], "utf-8")).trimEnd();
124
95
  }
96
+ if (options["video"]) data["video"] = options["video"];
97
+ if (Object.keys(data).length === 0) {
98
+ throw new GpcError(
99
+ "Provide at least one field to update (--title, --short, --full, --full-file, --video).",
100
+ "LISTINGS_USAGE_ERROR",
101
+ 2,
102
+ "Pass at least one of: --title, --short, --full, --full-file, --video"
103
+ );
104
+ }
105
+ if (isDryRun(program)) {
106
+ printDryRun(
107
+ {
108
+ command: "listings update",
109
+ action: "update listing for",
110
+ target: options.lang,
111
+ details: data
112
+ },
113
+ format,
114
+ formatOutput
115
+ );
116
+ return;
117
+ }
118
+ const client = await getClient(config);
119
+ const result = await updateListing(client, packageName, options.lang, data);
120
+ console.log(formatOutput(result, format));
125
121
  });
126
122
  listings.command("delete").description("Delete a store listing for a language").option("--lang <language>", "Language code (BCP 47)").action(async (options) => {
127
123
  const config = await loadConfig();
@@ -150,35 +146,25 @@ function registerListingsCommands(program) {
150
146
  return;
151
147
  }
152
148
  const client = await getClient(config);
153
- try {
154
- await deleteListing(client, packageName, options.lang);
155
- console.log(`Listing for "${options.lang}" deleted.`);
156
- } catch (error) {
157
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
158
- process.exit(4);
159
- }
149
+ await deleteListing(client, packageName, options.lang);
150
+ console.log(`Listing for "${options.lang}" deleted.`);
160
151
  });
161
152
  listings.command("pull").description("Download listings to Fastlane-format directory").option("--dir <path>", "Target directory (default: metadata)", "metadata").action(async (options) => {
162
153
  const config = await loadConfig();
163
154
  const packageName = resolvePackageName(program.opts()["app"], config);
164
155
  const client = await getClient(config);
165
156
  const format = getOutputFormat(program, config);
166
- try {
167
- const result = await pullListings(client, packageName, options.dir);
168
- console.log(
169
- formatOutput(
170
- {
171
- directory: options.dir,
172
- languages: result.listings.map((l) => l.language),
173
- count: result.listings.length
174
- },
175
- format
176
- )
177
- );
178
- } catch (error) {
179
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
180
- process.exit(4);
181
- }
157
+ const result = await pullListings(client, packageName, options.dir);
158
+ console.log(
159
+ formatOutput(
160
+ {
161
+ directory: options.dir,
162
+ languages: result.listings.map((l) => l.language),
163
+ count: result.listings.length
164
+ },
165
+ format
166
+ )
167
+ );
182
168
  });
183
169
  listings.command("push").description("Upload listings from Fastlane-format directory").option("--dir <path>", "Source directory (default: metadata)", "metadata").option("--force", "Push even if fields exceed character limits").action(async (options) => {
184
170
  const config = await loadConfig();
@@ -197,8 +183,7 @@ function registerListingsCommands(program) {
197
183
  console.log(formatOutput(result, format));
198
184
  } catch (error) {
199
185
  spinner.fail("Push failed");
200
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
201
- process.exit(4);
186
+ throw error;
202
187
  }
203
188
  });
204
189
  listings.command("diff").description("Compare local Fastlane-format metadata against remote listings").option("--dir <path>", "Local metadata directory", "metadata").option("--lang <language>", "Filter diff to a specific language").option("--word-diff", "Show word-level inline diff for fullDescription").action(async (options) => {
@@ -206,64 +191,56 @@ function registerListingsCommands(program) {
206
191
  const packageName = resolvePackageName(program.opts()["app"], config);
207
192
  const client = await getClient(config);
208
193
  const format = getOutputFormat(program, config);
209
- try {
210
- const diffs = await diffListingsEnhanced(client, packageName, options.dir, {
211
- lang: options.lang,
212
- wordLevel: options.wordDiff
213
- });
214
- if (diffs.length === 0) {
215
- if (format === "json") {
216
- console.log(formatOutput([], format));
217
- } else {
218
- console.log("No differences found.");
219
- }
220
- return;
221
- }
194
+ const diffs = await diffListingsEnhanced(client, packageName, options.dir, {
195
+ lang: options.lang,
196
+ wordLevel: options.wordDiff
197
+ });
198
+ if (diffs.length === 0) {
222
199
  if (format === "json") {
223
- console.log(formatOutput(diffs, format));
200
+ console.log(formatOutput([], format));
224
201
  } else {
225
- for (const diff of diffs) {
226
- const charInfo = diff["chars"] ? ` (${diff["chars"]} chars)` : "";
227
- console.log(`[${diff.language}] ${diff.field}:${charInfo}`);
228
- if (diff["diffSummary"]) {
229
- console.log(` ${diff["diffSummary"]}`);
230
- } else {
231
- console.log(green(` + local: ${diff.local || "(empty)"}`));
232
- console.log(red(` - remote: ${diff.remote || "(empty)"}`));
233
- }
202
+ console.log("No differences found.");
203
+ }
204
+ return;
205
+ }
206
+ if (format === "json") {
207
+ console.log(formatOutput(diffs, format));
208
+ } else {
209
+ for (const diff of diffs) {
210
+ const charInfo = diff["chars"] ? ` (${diff["chars"]} chars)` : "";
211
+ console.log(`[${diff.language}] ${diff.field}:${charInfo}`);
212
+ if (diff["diffSummary"]) {
213
+ console.log(` ${diff["diffSummary"]}`);
214
+ } else {
215
+ console.log(green(` + local: ${diff.local || "(empty)"}`));
216
+ console.log(red(` - remote: ${diff.remote || "(empty)"}`));
234
217
  }
235
218
  }
236
- } catch (error) {
237
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
238
- process.exit(4);
239
219
  }
240
220
  });
241
221
  listings.command("lint").description("Lint local listing metadata for Play Store character limits (no API)").option("--dir <path>", "Metadata directory", "metadata").action(async (options) => {
242
222
  const format = getOutputFormat(program, await loadConfig());
243
- try {
244
- const results = await lintLocalListings(options.dir);
245
- if (format === "json") {
246
- console.log(formatOutput(results, format));
247
- return;
248
- }
249
- let exitCode = 0;
250
- for (const r of results) {
251
- console.log(`
223
+ const results = await lintLocalListings(options.dir);
224
+ if (format === "json") {
225
+ console.log(formatOutput(results, format));
226
+ return;
227
+ }
228
+ let hasErrors = false;
229
+ for (const r of results) {
230
+ console.log(`
252
231
  [${r.language}] ${r.valid ? green("\u2713 valid") : red("\u2717 over limit")}`);
253
- const rows = r.fields.map((f) => ({
254
- field: f.field,
255
- chars: f.chars,
256
- limit: f.limit,
257
- pct: `${f.pct}%`,
258
- status: f.status === "ok" ? green("\u2713") : f.status === "warn" ? "\u26A0" : red("\u2717")
259
- }));
260
- console.log(formatOutput(rows, "table"));
261
- if (!r.valid) exitCode = 1;
262
- }
263
- if (exitCode) process.exit(exitCode);
264
- } catch (error) {
265
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
266
- process.exit(1);
232
+ const rows = r.fields.map((f) => ({
233
+ field: f.field,
234
+ chars: f.chars,
235
+ limit: f.limit,
236
+ pct: `${f.pct}%`,
237
+ status: f.status === "ok" ? green("\u2713") : f.status === "warn" ? "\u26A0" : red("\u2717")
238
+ }));
239
+ console.log(formatOutput(rows, "table"));
240
+ if (!r.valid) hasErrors = true;
241
+ }
242
+ if (hasErrors) {
243
+ process.exitCode = 1;
267
244
  }
268
245
  });
269
246
  listings.command("analyze").description("Analyze live Play Store listings for character limit compliance").option("--expected <locales>", "Comma-separated list of expected locale codes").action(async (options) => {
@@ -283,7 +260,7 @@ function registerListingsCommands(program) {
283
260
  console.log(formatOutput({ results, missingLocales }, format));
284
261
  return;
285
262
  }
286
- let exitCode = 0;
263
+ let hasErrors = false;
287
264
  for (const r of results) {
288
265
  console.log(`
289
266
  [${r.language}] ${r.valid ? green("\u2713 valid") : red("\u2717 over limit")}`);
@@ -295,17 +272,18 @@ function registerListingsCommands(program) {
295
272
  status: f.status === "ok" ? green("\u2713") : f.status === "warn" ? "\u26A0" : red("\u2717")
296
273
  }));
297
274
  console.log(formatOutput(rows, "table"));
298
- if (!r.valid) exitCode = 1;
275
+ if (!r.valid) hasErrors = true;
299
276
  }
300
277
  if (missingLocales && missingLocales.length > 0) {
301
278
  console.log(`
302
279
  Missing locales: ${missingLocales.join(", ")}`);
303
280
  }
304
- if (exitCode) process.exit(exitCode);
281
+ if (hasErrors) {
282
+ process.exitCode = 1;
283
+ }
305
284
  } catch (error) {
306
285
  spinner.fail("Analysis failed");
307
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
308
- process.exit(4);
286
+ throw error;
309
287
  }
310
288
  });
311
289
  const images = listings.command("images").description("Manage listing images");
@@ -334,13 +312,8 @@ Missing locales: ${missingLocales.join(", ")}`);
334
312
  const client = await getClient(config);
335
313
  const format = getOutputFormat(program, config);
336
314
  const imageType = validateImageType(options.type);
337
- try {
338
- const result = await listImages(client, packageName, options.lang, imageType);
339
- console.log(formatOutput(result, format));
340
- } catch (error) {
341
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
342
- process.exit(4);
343
- }
315
+ const result = await listImages(client, packageName, options.lang, imageType);
316
+ console.log(formatOutput(result, format));
344
317
  });
345
318
  images.command("upload <file>").description("Upload an image").option("--lang <language>", "Language code (BCP 47)").option("--type <type>", "Image type").action(async (file, options) => {
346
319
  const config = await loadConfig();
@@ -375,8 +348,7 @@ Missing locales: ${missingLocales.join(", ")}`);
375
348
  console.log(formatOutput(result, format));
376
349
  } catch (error) {
377
350
  spinner.fail("Image upload failed");
378
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
379
- process.exit(4);
351
+ throw error;
380
352
  }
381
353
  });
382
354
  images.command("delete").description("Delete an image").option("--lang <language>", "Language code (BCP 47)").option("--type <type>", "Image type").option("--id <imageId>", "Image ID to delete").action(async (options) => {
@@ -411,13 +383,8 @@ Missing locales: ${missingLocales.join(", ")}`);
411
383
  await requireConfirm(`Delete image "${options.id}"?`, program);
412
384
  const client = await getClient(config);
413
385
  const imageType = validateImageType(options.type);
414
- try {
415
- await deleteImage(client, packageName, options.lang, imageType, options.id);
416
- console.log(`Image "${options.id}" deleted.`);
417
- } catch (error) {
418
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
419
- process.exit(4);
420
- }
386
+ await deleteImage(client, packageName, options.lang, imageType, options.id);
387
+ console.log(`Image "${options.id}" deleted.`);
421
388
  });
422
389
  images.command("export").description("Export all images to a local directory").option("--dir <path>", "Output directory", "images").option("--lang <language>", "Language code (BCP 47) \u2014 export only this language").option("--type <type>", "Image type \u2014 export only this type").action(async (options) => {
423
390
  const config = await loadConfig();
@@ -437,8 +404,7 @@ Missing locales: ${missingLocales.join(", ")}`);
437
404
  console.log(formatOutput(result, format));
438
405
  } catch (error) {
439
406
  spinner.fail("Image export failed");
440
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
441
- process.exit(4);
407
+ throw error;
442
408
  }
443
409
  });
444
410
  listings.command("availability").description("Get country availability for a track").option("--track <track>", "Track name", "production").action(async (options) => {
@@ -446,21 +412,16 @@ Missing locales: ${missingLocales.join(", ")}`);
446
412
  const packageName = resolvePackageName(program.opts()["app"], config);
447
413
  const client = await getClient(config);
448
414
  const format = getOutputFormat(program, config);
449
- try {
450
- const result = await getCountryAvailability(client, packageName, options.track);
451
- const countries = result["countryTargeting"];
452
- if (format !== "json" && (!countries || Array.isArray(countries) && countries.length === 0) && Object.keys(result).length === 0) {
453
- console.log("No availability data.");
454
- return;
455
- }
456
- console.log(formatOutput(result, format));
457
- } catch (error) {
458
- console.error(`Error: ${error instanceof Error ? error.message : String(error)}`);
459
- process.exit(4);
415
+ const result = await getCountryAvailability(client, packageName, options.track);
416
+ const countries = result["countryTargeting"];
417
+ if (format !== "json" && (!countries || Array.isArray(countries) && countries.length === 0) && Object.keys(result).length === 0) {
418
+ console.log("No availability data.");
419
+ return;
460
420
  }
421
+ console.log(formatOutput(result, format));
461
422
  });
462
423
  }
463
424
  export {
464
425
  registerListingsCommands
465
426
  };
466
- //# sourceMappingURL=listings-77HZW4S5.js.map
427
+ //# sourceMappingURL=listings-7SGQ4SRX.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/listings.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\n\nimport type { ImageType } from \"@gpc-cli/api\";\nimport {\n getListings,\n updateListing,\n deleteListing,\n pullListings,\n pushListings,\n diffListingsEnhanced,\n lintLocalListings,\n analyzeRemoteListings,\n listImages,\n uploadImage,\n deleteImage,\n exportImages,\n getCountryAvailability,\n formatOutput,\n createSpinner,\n GpcError,\n} from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { isInteractive, requireOption, requireConfirm } from \"../prompt.js\";\nimport { green, red } from \"../colors.js\";\n\n\n\nconst VALID_IMAGE_TYPES: ImageType[] = [\n \"phoneScreenshots\",\n \"sevenInchScreenshots\",\n \"tenInchScreenshots\",\n \"tvScreenshots\",\n \"wearScreenshots\",\n \"icon\",\n \"featureGraphic\",\n \"tvBanner\",\n];\n\nfunction validateImageType(type: string): ImageType {\n if (!VALID_IMAGE_TYPES.includes(type as ImageType)) {\n throw new GpcError(\n `Invalid image type \"${type}\". Valid types: ${VALID_IMAGE_TYPES.join(\", \")}`,\n \"LISTINGS_USAGE_ERROR\",\n 2,\n `Use one of: ${VALID_IMAGE_TYPES.join(\", \")}`,\n );\n }\n return type as ImageType;\n}\n\nexport function registerListingsCommands(program: Command): void {\n const listings = program.command(\"listings\").description(\"Manage store listings and metadata\");\n\n // Get\n listings\n .command(\"get\")\n .description(\"Get store listing(s)\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\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 getListings(client, packageName, options.lang);\n console.log(formatOutput(result, format));\n });\n\n // Update\n listings\n .command(\"update\")\n .description(\"Update a store listing\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\n .option(\"--title <text>\", \"App title\")\n .option(\"--short <text>\", \"Short description\")\n .option(\"--full <text>\", \"Full description\")\n .option(\"--full-file <path>\", \"Read full description from file\")\n .option(\"--video <url>\", \"Video URL\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const interactive = isInteractive(program);\n\n options.lang = await requireOption(\n \"lang\",\n options.lang,\n {\n message: \"Language code (BCP 47):\",\n default: \"en-US\",\n },\n interactive,\n );\n const format = getOutputFormat(program, config);\n\n const data: Record<string, string> = {};\n if (options[\"title\"]) data[\"title\"] = options[\"title\"];\n if (options[\"short\"]) data[\"shortDescription\"] = options[\"short\"];\n if (options[\"full\"]) data[\"fullDescription\"] = options[\"full\"];\n if (options[\"fullFile\"]) {\n const { readFile } = await import(\"node:fs/promises\");\n data[\"fullDescription\"] = (await readFile(options[\"fullFile\"], \"utf-8\")).trimEnd();\n }\n if (options[\"video\"]) data[\"video\"] = options[\"video\"];\n\n if (Object.keys(data).length === 0) {\n throw new GpcError(\n \"Provide at least one field to update (--title, --short, --full, --full-file, --video).\",\n \"LISTINGS_USAGE_ERROR\",\n 2,\n \"Pass at least one of: --title, --short, --full, --full-file, --video\",\n );\n }\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"listings update\",\n action: \"update listing for\",\n target: options.lang,\n details: data,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const result = await updateListing(client, packageName, options.lang, data);\n console.log(formatOutput(result, format));\n });\n\n // Delete\n listings\n .command(\"delete\")\n .description(\"Delete a store listing for a language\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const interactive = isInteractive(program);\n\n options.lang = await requireOption(\n \"lang\",\n options.lang,\n {\n message: \"Language code (BCP 47):\",\n },\n interactive,\n );\n\n await requireConfirm(`Delete listing for \"${options.lang}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"listings delete\",\n action: \"delete listing for\",\n target: options.lang,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n await deleteListing(client, packageName, options.lang);\n console.log(`Listing for \"${options.lang}\" deleted.`);\n });\n\n // Pull\n listings\n .command(\"pull\")\n .description(\"Download listings to Fastlane-format directory\")\n .option(\"--dir <path>\", \"Target directory (default: metadata)\", \"metadata\")\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 pullListings(client, packageName, options.dir);\n console.log(\n formatOutput(\n {\n directory: options.dir,\n languages: result.listings.map((l) => l.language),\n count: result.listings.length,\n },\n format,\n ),\n );\n });\n\n // Push\n listings\n .command(\"push\")\n .description(\"Upload listings from Fastlane-format directory\")\n .option(\"--dir <path>\", \"Source directory (default: metadata)\", \"metadata\")\n .option(\"--force\", \"Push even if fields exceed character limits\")\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 spinner = createSpinner(\"Pushing listings...\");\n if (!program.opts()[\"quiet\"] && process.stderr.isTTY) spinner.start();\n\n try {\n const dryRun = isDryRun(program);\n const result = await pushListings(client, packageName, options.dir, {\n dryRun,\n force: options.force,\n });\n spinner.stop(dryRun ? \"Dry-run complete (no changes made)\" : \"Listings pushed\");\n console.log(formatOutput(result, format));\n } catch (error) {\n spinner.fail(\"Push failed\");\n throw error;\n }\n });\n\n // Diff (enhanced: --lang filter, word-level inline diff)\n listings\n .command(\"diff\")\n .description(\"Compare local Fastlane-format metadata against remote listings\")\n .option(\"--dir <path>\", \"Local metadata directory\", \"metadata\")\n .option(\"--lang <language>\", \"Filter diff to a specific language\")\n .option(\"--word-diff\", \"Show word-level inline diff for fullDescription\")\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 diffs = await diffListingsEnhanced(client, packageName, options.dir, {\n lang: options.lang,\n wordLevel: options.wordDiff,\n });\n\n if (diffs.length === 0) {\n if (format === \"json\") {\n console.log(formatOutput([], format));\n } else {\n console.log(\"No differences found.\");\n }\n return;\n }\n\n if (format === \"json\") {\n console.log(formatOutput(diffs, format));\n } else {\n for (const diff of diffs) {\n const charInfo = (diff as unknown as Record<string, unknown>)[\"chars\"]\n ? ` (${(diff as unknown as Record<string, unknown>)[\"chars\"]} chars)`\n : \"\";\n console.log(`[${diff.language}] ${diff.field}:${charInfo}`);\n if ((diff as unknown as Record<string, unknown>)[\"diffSummary\"]) {\n console.log(` ${(diff as unknown as Record<string, unknown>)[\"diffSummary\"]}`);\n } else {\n console.log(green(` + local: ${diff.local || \"(empty)\"}`));\n console.log(red(` - remote: ${diff.remote || \"(empty)\"}`));\n }\n }\n }\n });\n\n // Lint (local, no API)\n listings\n .command(\"lint\")\n .description(\"Lint local listing metadata for Play Store character limits (no API)\")\n .option(\"--dir <path>\", \"Metadata directory\", \"metadata\")\n .action(async (options) => {\n const format = getOutputFormat(program, await loadConfig());\n const results = await lintLocalListings(options.dir);\n if (format === \"json\") {\n console.log(formatOutput(results, format));\n return;\n }\n let hasErrors = false;\n for (const r of results) {\n console.log(`\\n[${r.language}] ${r.valid ? green(\"✓ valid\") : red(\"✗ over limit\")}`);\n const rows = r.fields.map((f) => ({\n field: f.field,\n chars: f.chars,\n limit: f.limit,\n pct: `${f.pct}%`,\n status: f.status === \"ok\" ? green(\"✓\") : f.status === \"warn\" ? \"⚠\" : red(\"✗\"),\n }));\n console.log(formatOutput(rows, \"table\"));\n if (!r.valid) hasErrors = true;\n }\n if (hasErrors) {\n process.exitCode = 1;\n }\n });\n\n // Analyze (live, fetches remote)\n listings\n .command(\"analyze\")\n .description(\"Analyze live Play Store listings for character limit compliance\")\n .option(\"--expected <locales>\", \"Comma-separated list of expected locale codes\")\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 spinner = createSpinner(\"Fetching remote listings...\");\n if (!program.opts()[\"quiet\"] && process.stderr.isTTY) spinner.start();\n\n try {\n const expectedLocales = options.expected\n ? (options.expected as string).split(\",\").map((s: string) => s.trim())\n : undefined;\n const { results, missingLocales } = await analyzeRemoteListings(client, packageName, {\n expectedLocales,\n });\n spinner.stop(\"Done\");\n\n if (format === \"json\") {\n console.log(formatOutput({ results, missingLocales }, format));\n return;\n }\n\n let hasErrors = false;\n for (const r of results) {\n console.log(`\\n[${r.language}] ${r.valid ? green(\"✓ valid\") : red(\"✗ over limit\")}`);\n const rows = r.fields.map((f) => ({\n field: f.field,\n chars: f.chars,\n limit: f.limit,\n pct: `${f.pct}%`,\n status: f.status === \"ok\" ? green(\"✓\") : f.status === \"warn\" ? \"⚠\" : red(\"✗\"),\n }));\n console.log(formatOutput(rows, \"table\"));\n if (!r.valid) hasErrors = true;\n }\n\n if (missingLocales && missingLocales.length > 0) {\n console.log(`\\nMissing locales: ${missingLocales.join(\", \")}`);\n }\n\n if (hasErrors) {\n process.exitCode = 1;\n }\n } catch (error) {\n spinner.fail(\"Analysis failed\");\n throw error;\n }\n });\n\n // Images subcommand\n const images = listings.command(\"images\").description(\"Manage listing images\");\n\n // Images list\n images\n .command(\"list\")\n .description(\"List images for a language and type\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\n .option(\"--type <type>\", \"Image type\")\n .option(\"--limit <n>\", \"Maximum results to return\")\n .option(\"--next-page <token>\", \"Pagination token for next page\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const interactive = isInteractive(program);\n\n options.lang = await requireOption(\n \"lang\",\n options.lang,\n {\n message: \"Language code (BCP 47):\",\n default: \"en-US\",\n },\n interactive,\n );\n\n options.type = await requireOption(\n \"type\",\n options.type,\n {\n message: \"Image type:\",\n choices: VALID_IMAGE_TYPES as unknown as string[],\n },\n interactive,\n );\n\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n const imageType = validateImageType(options.type);\n\n const result = await listImages(client, packageName, options.lang, imageType);\n console.log(formatOutput(result, format));\n });\n\n // Images upload\n images\n .command(\"upload <file>\")\n .description(\"Upload an image\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\n .option(\"--type <type>\", \"Image type\")\n .action(async (file: string, options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const interactive = isInteractive(program);\n\n options.lang = await requireOption(\n \"lang\",\n options.lang,\n {\n message: \"Language code (BCP 47):\",\n default: \"en-US\",\n },\n interactive,\n );\n\n options.type = await requireOption(\n \"type\",\n options.type,\n {\n message: \"Image type:\",\n choices: VALID_IMAGE_TYPES as unknown as string[],\n },\n interactive,\n );\n\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n const imageType = validateImageType(options.type);\n\n const spinner = createSpinner(\"Uploading image...\");\n if (!program.opts()[\"quiet\"] && process.stderr.isTTY) spinner.start();\n\n try {\n const result = await uploadImage(client, packageName, options.lang, imageType, file);\n spinner.stop(\"Image uploaded\");\n console.log(formatOutput(result, format));\n } catch (error) {\n spinner.fail(\"Image upload failed\");\n throw error;\n }\n });\n\n // Images delete\n images\n .command(\"delete\")\n .description(\"Delete an image\")\n .option(\"--lang <language>\", \"Language code (BCP 47)\")\n .option(\"--type <type>\", \"Image type\")\n .option(\"--id <imageId>\", \"Image ID to delete\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const interactive = isInteractive(program);\n\n options.lang = await requireOption(\n \"lang\",\n options.lang,\n {\n message: \"Language code (BCP 47):\",\n },\n interactive,\n );\n\n options.type = await requireOption(\n \"type\",\n options.type,\n {\n message: \"Image type:\",\n choices: VALID_IMAGE_TYPES as unknown as string[],\n },\n interactive,\n );\n\n options.id = await requireOption(\n \"id\",\n options.id,\n {\n message: \"Image ID to delete:\",\n },\n interactive,\n );\n\n await requireConfirm(`Delete image \"${options.id}\"?`, program);\n\n const client = await getClient(config);\n const imageType = validateImageType(options.type);\n\n await deleteImage(client, packageName, options.lang, imageType, options.id);\n console.log(`Image \"${options.id}\" deleted.`);\n });\n\n // Images export\n images\n .command(\"export\")\n .description(\"Export all images to a local directory\")\n .option(\"--dir <path>\", \"Output directory\", \"images\")\n .option(\"--lang <language>\", \"Language code (BCP 47) — export only this language\")\n .option(\"--type <type>\", \"Image type — export only this type\")\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 exportOpts: { lang?: string; type?: ImageType } = {};\n if (options.lang) exportOpts.lang = options.lang;\n if (options.type) {\n exportOpts.type = validateImageType(options.type);\n }\n\n const spinner = createSpinner(\"Exporting images...\");\n if (!program.opts()[\"quiet\"] && process.stderr.isTTY) spinner.start();\n\n try {\n const result = await exportImages(client, packageName, options.dir, exportOpts);\n spinner.stop(\"Images exported\");\n console.log(formatOutput(result, format));\n } catch (error) {\n spinner.fail(\"Image export failed\");\n throw error;\n }\n });\n\n // Availability\n listings\n .command(\"availability\")\n .description(\"Get country availability for a track\")\n .option(\"--track <track>\", \"Track name\", \"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 getCountryAvailability(client, packageName, options.track);\n const countries = (result as unknown as Record<string, unknown>)[\"countryTargeting\"] as\n | unknown[]\n | undefined;\n if (\n format !== \"json\" &&\n (!countries || (Array.isArray(countries) && countries.length === 0)) &&\n Object.keys(result as object).length === 0\n ) {\n console.log(\"No availability data.\");\n return;\n }\n console.log(formatOutput(result, format));\n });\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,138 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ getOutputFormat
4
+ } from "./chunk-ELXAK7GI.js";
5
+
6
+ // src/commands/migrate.ts
7
+ import { access } from "fs/promises";
8
+ import { join } from "path";
9
+ import { loadConfig } from "@gpc-cli/config";
10
+ import {
11
+ detectFastlane,
12
+ generateMigrationPlan,
13
+ writeMigrationOutput,
14
+ formatOutput
15
+ } from "@gpc-cli/core";
16
+ var WARN = "\u26A0";
17
+ async function fileExists(path) {
18
+ try {
19
+ await access(path);
20
+ return true;
21
+ } catch {
22
+ return false;
23
+ }
24
+ }
25
+ function registerMigrateCommands(program) {
26
+ const migrate = program.command("migrate").description("Migrate from other tools to GPC");
27
+ migrate.command("fastlane").description("Migrate from Fastlane to GPC").option("--dir <path>", "Directory containing Fastlane files", ".").option("--output <path>", "Output directory for migration files", ".").option("--dry-run", "Preview migration plan without writing any files").action(async (options) => {
28
+ const config = await loadConfig();
29
+ const format = getOutputFormat(program, config);
30
+ const dryRun = options.dryRun ?? false;
31
+ const detection = await detectFastlane(options.dir);
32
+ if (format === "json") {
33
+ if (!detection.hasFastfile && !detection.hasAppfile && !detection.hasMetadata) {
34
+ console.log(formatOutput({ detection, plan: null, files: [] }, format));
35
+ return;
36
+ }
37
+ const plan2 = generateMigrationPlan(detection);
38
+ if (!dryRun) {
39
+ const files2 = await writeMigrationOutput(plan2, options.output);
40
+ console.log(formatOutput({ detection, plan: plan2, files: files2 }, format));
41
+ } else {
42
+ console.log(formatOutput({ detection, plan: plan2, files: [] }, format));
43
+ }
44
+ return;
45
+ }
46
+ console.log("Fastlane Detection Results:\n");
47
+ console.log(` Fastfile: ${detection.hasFastfile ? "found" : "not found"}`);
48
+ console.log(` Appfile: ${detection.hasAppfile ? "found" : "not found"}`);
49
+ console.log(` Metadata: ${detection.hasMetadata ? "found" : "not found"}`);
50
+ console.log(` Gemfile: ${detection.hasGemfile ? "found" : "not found"}`);
51
+ if (detection.packageName) {
52
+ console.log(` Package: ${detection.packageName}`);
53
+ }
54
+ if (detection.lanes.length > 0) {
55
+ console.log(`
56
+ Lanes (${detection.lanes.length}):`);
57
+ for (const lane of detection.lanes) {
58
+ const equiv = lane.gpcEquivalent ? ` \u2192 ${lane.gpcEquivalent}` : " (no equivalent)";
59
+ console.log(` ${lane.name}${equiv}`);
60
+ }
61
+ }
62
+ if (detection.metadataLanguages.length > 0) {
63
+ console.log(`
64
+ Metadata languages: ${detection.metadataLanguages.join(", ")}`);
65
+ }
66
+ if (detection.parseWarnings.length > 0) {
67
+ console.log("");
68
+ for (const w of detection.parseWarnings) {
69
+ console.log(` ${WARN} ${w}`);
70
+ }
71
+ }
72
+ if (!detection.hasFastfile && !detection.hasAppfile && !detection.hasMetadata) {
73
+ console.log("\nNo Fastlane files detected in this directory. Nothing to migrate.");
74
+ console.log(" Try: gpc migrate fastlane --dir <path-to-your-android-project>");
75
+ return;
76
+ }
77
+ const plan = generateMigrationPlan(detection);
78
+ if (dryRun) {
79
+ console.log("\nMigration Plan (dry run \u2014 nothing written):\n");
80
+ if (plan.warnings.length > 0) {
81
+ console.log("Warnings:");
82
+ for (const w of plan.warnings) {
83
+ console.log(` ${WARN} ${w}`);
84
+ }
85
+ console.log("");
86
+ }
87
+ console.log("Checklist:");
88
+ for (const item of plan.checklist) {
89
+ console.log(` [ ] ${item}`);
90
+ }
91
+ if (Object.keys(plan.config).length > 0) {
92
+ console.log("\n.gpcrc.json (would be written):");
93
+ console.log(JSON.stringify(plan.config, null, 2).replace(/^/gm, " "));
94
+ }
95
+ console.log(
96
+ "\nRun without --dry-run to write MIGRATION.md" + (Object.keys(plan.config).length > 0 ? " and .gpcrc.json" : "") + "."
97
+ );
98
+ return;
99
+ }
100
+ if (Object.keys(plan.config).length > 0 && await fileExists(join(options.output, ".gpcrc.json"))) {
101
+ const hasYes = process.argv.includes("--yes") || process.argv.includes("-y");
102
+ if (!hasYes) {
103
+ console.log(
104
+ `
105
+ ${WARN} .gpcrc.json already exists and will be overwritten. Use --dry-run to preview first.`
106
+ );
107
+ console.log("Aborting. Pass --yes to overwrite without confirmation.");
108
+ return;
109
+ }
110
+ console.log(
111
+ `
112
+ ${WARN} .gpcrc.json already exists in ${options.output} \u2014 overwriting (--yes passed).`
113
+ );
114
+ }
115
+ const files = await writeMigrationOutput(plan, options.output);
116
+ console.log("\nMigration files written:");
117
+ for (const file of files) {
118
+ console.log(` ${file}`);
119
+ }
120
+ if (plan.warnings.length > 0) {
121
+ console.log("\nWarnings:");
122
+ for (const w of plan.warnings) {
123
+ console.log(` ${WARN} ${w}`);
124
+ }
125
+ }
126
+ console.log("\nMigration Checklist:");
127
+ for (const item of plan.checklist) {
128
+ console.log(` [ ] ${item}`);
129
+ }
130
+ console.log(
131
+ "\nNext step: open MIGRATION.md and work through the checklist. Run `gpc doctor` when done."
132
+ );
133
+ });
134
+ }
135
+ export {
136
+ registerMigrateCommands
137
+ };
138
+ //# sourceMappingURL=migrate-ZQCJGQQS.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/migrate.ts"],"sourcesContent":["import { access } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport {\n detectFastlane,\n generateMigrationPlan,\n writeMigrationOutput,\n formatOutput,\n} from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\n\nconst WARN = \"\\u26A0\";\n\nasync function fileExists(path: string): Promise<boolean> {\n try {\n await access(path);\n return true;\n } catch {\n return false;\n }\n}\n\nexport function registerMigrateCommands(program: Command): void {\n const migrate = program.command(\"migrate\").description(\"Migrate from other tools to GPC\");\n\n migrate\n .command(\"fastlane\")\n .description(\"Migrate from Fastlane to GPC\")\n .option(\"--dir <path>\", \"Directory containing Fastlane files\", \".\")\n .option(\"--output <path>\", \"Output directory for migration files\", \".\")\n .option(\"--dry-run\", \"Preview migration plan without writing any files\")\n .action(async (options: { dir: string; output: string; dryRun?: boolean }) => {\n const config = await loadConfig();\n const format = getOutputFormat(program, config);\n const dryRun = options.dryRun ?? false;\n\n const detection = await detectFastlane(options.dir);\n\n if (format === \"json\") {\n if (!detection.hasFastfile && !detection.hasAppfile && !detection.hasMetadata) {\n console.log(formatOutput({ detection, plan: null, files: [] }, format));\n return;\n }\n const plan = generateMigrationPlan(detection);\n if (!dryRun) {\n const files = await writeMigrationOutput(plan, options.output);\n console.log(formatOutput({ detection, plan, files }, format));\n } else {\n console.log(formatOutput({ detection, plan, files: [] }, format));\n }\n return;\n }\n\n // Human-readable output\n console.log(\"Fastlane Detection Results:\\n\");\n console.log(` Fastfile: ${detection.hasFastfile ? \"found\" : \"not found\"}`);\n console.log(` Appfile: ${detection.hasAppfile ? \"found\" : \"not found\"}`);\n console.log(` Metadata: ${detection.hasMetadata ? \"found\" : \"not found\"}`);\n console.log(` Gemfile: ${detection.hasGemfile ? \"found\" : \"not found\"}`);\n\n if (detection.packageName) {\n console.log(` Package: ${detection.packageName}`);\n }\n\n if (detection.lanes.length > 0) {\n console.log(`\\n Lanes (${detection.lanes.length}):`);\n for (const lane of detection.lanes) {\n const equiv = lane.gpcEquivalent ? ` → ${lane.gpcEquivalent}` : \" (no equivalent)\";\n console.log(` ${lane.name}${equiv}`);\n }\n }\n\n if (detection.metadataLanguages.length > 0) {\n console.log(`\\n Metadata languages: ${detection.metadataLanguages.join(\", \")}`);\n }\n\n if (detection.parseWarnings.length > 0) {\n console.log(\"\");\n for (const w of detection.parseWarnings) {\n console.log(` ${WARN} ${w}`);\n }\n }\n\n if (!detection.hasFastfile && !detection.hasAppfile && !detection.hasMetadata) {\n console.log(\"\\nNo Fastlane files detected in this directory. Nothing to migrate.\");\n console.log(\" Try: gpc migrate fastlane --dir <path-to-your-android-project>\");\n return;\n }\n\n const plan = generateMigrationPlan(detection);\n\n if (dryRun) {\n console.log(\"\\nMigration Plan (dry run — nothing written):\\n\");\n\n if (plan.warnings.length > 0) {\n console.log(\"Warnings:\");\n for (const w of plan.warnings) {\n console.log(` ${WARN} ${w}`);\n }\n console.log(\"\");\n }\n\n console.log(\"Checklist:\");\n for (const item of plan.checklist) {\n console.log(` [ ] ${item}`);\n }\n\n if (Object.keys(plan.config).length > 0) {\n console.log(\"\\n.gpcrc.json (would be written):\");\n console.log(JSON.stringify(plan.config, null, 2).replace(/^/gm, \" \"));\n }\n\n console.log(\n \"\\nRun without --dry-run to write MIGRATION.md\" +\n (Object.keys(plan.config).length > 0 ? \" and .gpcrc.json\" : \"\") +\n \".\",\n );\n return;\n }\n\n // Conflict check — abort before clobbering existing .gpcrc.json unless --yes\n if (\n Object.keys(plan.config).length > 0 &&\n (await fileExists(join(options.output, \".gpcrc.json\")))\n ) {\n const hasYes = process.argv.includes(\"--yes\") || process.argv.includes(\"-y\");\n\n if (!hasYes) {\n console.log(\n `\\n${WARN} .gpcrc.json already exists and will be overwritten. Use --dry-run to preview first.`,\n );\n console.log(\"Aborting. Pass --yes to overwrite without confirmation.\");\n return;\n }\n\n console.log(\n `\\n${WARN} .gpcrc.json already exists in ${options.output} — overwriting (--yes passed).`,\n );\n }\n\n const files = await writeMigrationOutput(plan, options.output);\n\n console.log(\"\\nMigration files written:\");\n for (const file of files) {\n console.log(` ${file}`);\n }\n\n if (plan.warnings.length > 0) {\n console.log(\"\\nWarnings:\");\n for (const w of plan.warnings) {\n console.log(` ${WARN} ${w}`);\n }\n }\n\n console.log(\"\\nMigration Checklist:\");\n for (const item of plan.checklist) {\n console.log(` [ ] ${item}`);\n }\n\n console.log(\n \"\\nNext step: open MIGRATION.md and work through the checklist. Run `gpc doctor` when done.\",\n );\n });\n}\n"],"mappings":"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","names":["plan","files"]}