@gpc-cli/cli 0.9.52 → 0.9.54

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 (160) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +19 -19
  3. package/dist/{anomalies-V3AFS4LD.js → anomalies-I5P6QQAZ.js} +14 -5
  4. package/dist/anomalies-I5P6QQAZ.js.map +1 -0
  5. package/dist/{apps-4GP3FD7O.js → apps-WPUQOQLL.js} +2 -2
  6. package/dist/{audit-VTWXTXC6.js → audit-M7GYID74.js} +2 -2
  7. package/dist/{auth-BA4FE2PO.js → auth-4DRT7ZH2.js} +10 -4
  8. package/dist/auth-4DRT7ZH2.js.map +1 -0
  9. package/dist/bin.js +2 -2
  10. package/dist/{bundle-F7MUVC5J.js → bundle-PFTE7XMU.js} +4 -2
  11. package/dist/bundle-PFTE7XMU.js.map +1 -0
  12. package/dist/{cache-XKPLZYEB.js → cache-FGNP7Y37.js} +6 -2
  13. package/dist/cache-FGNP7Y37.js.map +1 -0
  14. package/dist/changelog-QLDFG5TV.js +0 -0
  15. package/dist/{chunk-SLNJEAMK.js → chunk-22XCOLZX.js} +4 -2
  16. package/dist/chunk-22XCOLZX.js.map +1 -0
  17. package/dist/chunk-3SJ6OXCZ.js +0 -0
  18. package/dist/{chunk-FXOWADQD.js → chunk-6HIY4IGM.js} +48 -45
  19. package/dist/chunk-6HIY4IGM.js.map +1 -0
  20. package/dist/{chunk-BCBXQC7J.js → chunk-E7SVZ7RF.js} +1 -1
  21. package/dist/chunk-E7SVZ7RF.js.map +1 -0
  22. package/dist/chunk-ELXAK7GI.js +0 -0
  23. package/dist/chunk-FAN4ZITI.js +0 -0
  24. package/dist/{chunk-NQH4G7BI.js → chunk-JDRY7HK5.js} +6 -9
  25. package/dist/chunk-JDRY7HK5.js.map +1 -0
  26. package/dist/{chunk-YFUBD2XB.js → chunk-RZQSEDKI.js} +6 -9
  27. package/dist/chunk-RZQSEDKI.js.map +1 -0
  28. package/dist/{chunk-A7VRCCNS.js → chunk-WSLFHX5X.js} +3 -3
  29. package/dist/chunk-WSLFHX5X.js.map +1 -0
  30. package/dist/chunk-Y3QZDAKS.js +0 -0
  31. package/dist/completion-BCHRJSAT.js +0 -0
  32. package/dist/{config-BLMJ35J2.js → config-PUINDZON.js} +3 -3
  33. package/dist/{data-safety-AFMD6MYI.js → data-safety-46VY64OO.js} +12 -4
  34. package/dist/data-safety-46VY64OO.js.map +1 -0
  35. package/dist/{device-tiers-AQAMUQXI.js → device-tiers-MNZYMG3Y.js} +2 -2
  36. package/dist/device-tiers-MNZYMG3Y.js.map +1 -0
  37. package/dist/{diff-6EO4ID6W.js → diff-OBSHUSTL.js} +2 -2
  38. package/dist/diff-OBSHUSTL.js.map +1 -0
  39. package/dist/{docs-GMFN6V4K.js → docs-GP6AEX4N.js} +8 -4
  40. package/dist/docs-GP6AEX4N.js.map +1 -0
  41. package/dist/{doctor-7LQWPY5P.js → doctor-T3QFYBRV.js} +5 -5
  42. package/dist/doctor-T3QFYBRV.js.map +1 -0
  43. package/dist/enterprise-7PWXMSUN.js +0 -0
  44. package/dist/{external-transactions-LCZALS3V.js → external-transactions-JL3G4IG5.js} +11 -5
  45. package/dist/external-transactions-JL3G4IG5.js.map +1 -0
  46. package/dist/{feedback-7ADYSGRD.js → feedback-AULXQLJ7.js} +3 -3
  47. package/dist/feedback-AULXQLJ7.js.map +1 -0
  48. package/dist/{games-ZSNGEI7A.js → games-SVFN2YIS.js} +2 -2
  49. package/dist/games-SVFN2YIS.js.map +1 -0
  50. package/dist/{generated-apks-RX2IUWSF.js → generated-apks-TC33S2YN.js} +2 -2
  51. package/dist/generated-apks-TC33S2YN.js.map +1 -0
  52. package/dist/{grants-EBPECI26.js → grants-UHNBPIFD.js} +10 -3
  53. package/dist/grants-UHNBPIFD.js.map +1 -0
  54. package/dist/{iap-OUI5YYN4.js → iap-ETOL7OAC.js} +4 -4
  55. package/dist/iap-ETOL7OAC.js.map +1 -0
  56. package/dist/index.js +1 -1
  57. package/dist/{init-WSTQTJOD.js → init-6CCSVJC2.js} +7 -3
  58. package/dist/init-6CCSVJC2.js.map +1 -0
  59. package/dist/{install-skills-JKPYZHYS.js → install-skills-S342NJ7T.js} +7 -3
  60. package/dist/{install-skills-JKPYZHYS.js.map → install-skills-S342NJ7T.js.map} +1 -1
  61. package/dist/{internal-sharing-ONNIWIAT.js → internal-sharing-KW6YENDG.js} +2 -2
  62. package/dist/internal-sharing-KW6YENDG.js.map +1 -0
  63. package/dist/{listings-LNX6MQYN.js → listings-MOHHHNE6.js} +27 -7
  64. package/dist/listings-MOHHHNE6.js.map +1 -0
  65. package/dist/migrate-ZQCJGQQS.js +0 -0
  66. package/dist/one-time-products-Y5RNIPV2.js +472 -0
  67. package/dist/one-time-products-Y5RNIPV2.js.map +1 -0
  68. package/dist/{preflight-W3JAJ4GO.js → preflight-KIWZPFTX.js} +4 -13
  69. package/dist/preflight-KIWZPFTX.js.map +1 -0
  70. package/dist/{pricing-JJZFICFL.js → pricing-HYQRXKNO.js} +3 -3
  71. package/dist/pricing-HYQRXKNO.js.map +1 -0
  72. package/dist/{prompt-GXC2JSLA.js → prompt-HJXNXXAR.js} +2 -2
  73. package/dist/{publish-P5KIGSLI.js → publish-26SSZ2W3.js} +11 -5
  74. package/dist/publish-26SSZ2W3.js.map +1 -0
  75. package/dist/purchase-options-KFWW4JW2.js +0 -0
  76. package/dist/{purchases-UBFLNYZC.js → purchases-AHWGLW6V.js} +35 -32
  77. package/dist/purchases-AHWGLW6V.js.map +1 -0
  78. package/dist/{quickstart-Z5Y3FYJU.js → quickstart-FOWM3OKT.js} +7 -2
  79. package/dist/quickstart-FOWM3OKT.js.map +1 -0
  80. package/dist/quota-MZRWYJGR.js +0 -0
  81. package/dist/{recovery-YE3Z7NIN.js → recovery-B7DZQ6XG.js} +28 -12
  82. package/dist/recovery-B7DZQ6XG.js.map +1 -0
  83. package/dist/{releases-LUAHKIMY.js → releases-XY57V22L.js} +33 -8
  84. package/dist/releases-XY57V22L.js.map +1 -0
  85. package/dist/reports-CIB2T3XT.js +0 -0
  86. package/dist/{reviews-YCBBM656.js → reviews-P4M6BEJI.js} +4 -6
  87. package/dist/reviews-P4M6BEJI.js.map +1 -0
  88. package/dist/{rtdn-LID2B7XZ.js → rtdn-YII4H6SD.js} +29 -21
  89. package/dist/rtdn-YII4H6SD.js.map +1 -0
  90. package/dist/{status-3HXBBXG6.js → status-32AJ7A6O.js} +13 -32
  91. package/dist/status-32AJ7A6O.js.map +1 -0
  92. package/dist/{subscriptions-LURZFPGJ.js → subscriptions-GTVJB4HX.js} +49 -12
  93. package/dist/subscriptions-GTVJB4HX.js.map +1 -0
  94. package/dist/system-apks-L6M7QYB3.js +111 -0
  95. package/dist/system-apks-L6M7QYB3.js.map +1 -0
  96. package/dist/{testers-6CQL4KQV.js → testers-QUWZHO6M.js} +25 -7
  97. package/dist/testers-QUWZHO6M.js.map +1 -0
  98. package/dist/{tracks-I4QZNZ3M.js → tracks-RH3RKVFB.js} +9 -3
  99. package/dist/{tracks-I4QZNZ3M.js.map → tracks-RH3RKVFB.js.map} +1 -1
  100. package/dist/{train-MDD2EBHS.js → train-D2NKUW3M.js} +3 -3
  101. package/dist/train-D2NKUW3M.js.map +1 -0
  102. package/dist/{update-FZ3MNLOH.js → update-73YOR4GP.js} +8 -11
  103. package/dist/{update-FZ3MNLOH.js.map → update-73YOR4GP.js.map} +1 -1
  104. package/dist/{users-UKG7VIQH.js → users-FOMAT7DY.js} +2 -2
  105. package/dist/{validate-QIYSA3N7.js → validate-RHWEZYD2.js} +6 -2
  106. package/dist/validate-RHWEZYD2.js.map +1 -0
  107. package/dist/{verify-UUQNQMPG.js → verify-H4ZUVHMZ.js} +5 -13
  108. package/dist/verify-H4ZUVHMZ.js.map +1 -0
  109. package/dist/{version-VHQBXU2I.js → version-RXLEX62V.js} +3 -3
  110. package/dist/{vitals-PJEQUUAK.js → vitals-PRBPNMVC.js} +26 -8
  111. package/dist/vitals-PRBPNMVC.js.map +1 -0
  112. package/package.json +18 -18
  113. package/dist/anomalies-V3AFS4LD.js.map +0 -1
  114. package/dist/auth-BA4FE2PO.js.map +0 -1
  115. package/dist/bundle-F7MUVC5J.js.map +0 -1
  116. package/dist/cache-XKPLZYEB.js.map +0 -1
  117. package/dist/chunk-A7VRCCNS.js.map +0 -1
  118. package/dist/chunk-BCBXQC7J.js.map +0 -1
  119. package/dist/chunk-FXOWADQD.js.map +0 -1
  120. package/dist/chunk-NQH4G7BI.js.map +0 -1
  121. package/dist/chunk-SLNJEAMK.js.map +0 -1
  122. package/dist/chunk-YFUBD2XB.js.map +0 -1
  123. package/dist/data-safety-AFMD6MYI.js.map +0 -1
  124. package/dist/device-tiers-AQAMUQXI.js.map +0 -1
  125. package/dist/diff-6EO4ID6W.js.map +0 -1
  126. package/dist/docs-GMFN6V4K.js.map +0 -1
  127. package/dist/doctor-7LQWPY5P.js.map +0 -1
  128. package/dist/external-transactions-LCZALS3V.js.map +0 -1
  129. package/dist/feedback-7ADYSGRD.js.map +0 -1
  130. package/dist/games-ZSNGEI7A.js.map +0 -1
  131. package/dist/generated-apks-RX2IUWSF.js.map +0 -1
  132. package/dist/grants-EBPECI26.js.map +0 -1
  133. package/dist/iap-OUI5YYN4.js.map +0 -1
  134. package/dist/init-WSTQTJOD.js.map +0 -1
  135. package/dist/internal-sharing-ONNIWIAT.js.map +0 -1
  136. package/dist/listings-LNX6MQYN.js.map +0 -1
  137. package/dist/one-time-products-MGZTU7OM.js +0 -254
  138. package/dist/one-time-products-MGZTU7OM.js.map +0 -1
  139. package/dist/preflight-W3JAJ4GO.js.map +0 -1
  140. package/dist/pricing-JJZFICFL.js.map +0 -1
  141. package/dist/publish-P5KIGSLI.js.map +0 -1
  142. package/dist/purchases-UBFLNYZC.js.map +0 -1
  143. package/dist/quickstart-Z5Y3FYJU.js.map +0 -1
  144. package/dist/recovery-YE3Z7NIN.js.map +0 -1
  145. package/dist/releases-LUAHKIMY.js.map +0 -1
  146. package/dist/reviews-YCBBM656.js.map +0 -1
  147. package/dist/rtdn-LID2B7XZ.js.map +0 -1
  148. package/dist/status-3HXBBXG6.js.map +0 -1
  149. package/dist/subscriptions-LURZFPGJ.js.map +0 -1
  150. package/dist/testers-6CQL4KQV.js.map +0 -1
  151. package/dist/train-MDD2EBHS.js.map +0 -1
  152. package/dist/validate-QIYSA3N7.js.map +0 -1
  153. package/dist/verify-UUQNQMPG.js.map +0 -1
  154. package/dist/vitals-PJEQUUAK.js.map +0 -1
  155. /package/dist/{apps-4GP3FD7O.js.map → apps-WPUQOQLL.js.map} +0 -0
  156. /package/dist/{audit-VTWXTXC6.js.map → audit-M7GYID74.js.map} +0 -0
  157. /package/dist/{config-BLMJ35J2.js.map → config-PUINDZON.js.map} +0 -0
  158. /package/dist/{prompt-GXC2JSLA.js.map → prompt-HJXNXXAR.js.map} +0 -0
  159. /package/dist/{users-UKG7VIQH.js.map → users-FOMAT7DY.js.map} +0 -0
  160. /package/dist/{version-VHQBXU2I.js.map → version-RXLEX62V.js.map} +0 -0
@@ -0,0 +1,472 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ readJsonFile
4
+ } from "./chunk-22XCOLZX.js";
5
+ import {
6
+ getClient,
7
+ resolvePackageName
8
+ } from "./chunk-JDRY7HK5.js";
9
+ import {
10
+ isDryRun,
11
+ printDryRun
12
+ } from "./chunk-Y3QZDAKS.js";
13
+ import {
14
+ getOutputFormat
15
+ } from "./chunk-ELXAK7GI.js";
16
+ import {
17
+ requireConfirm
18
+ } from "./chunk-RZQSEDKI.js";
19
+
20
+ // src/commands/one-time-products.ts
21
+ import { loadConfig } from "@gpc-cli/config";
22
+ import {
23
+ listOneTimeProducts,
24
+ getOneTimeProduct,
25
+ createOneTimeProduct,
26
+ updateOneTimeProduct,
27
+ deleteOneTimeProduct,
28
+ listOneTimeOffers,
29
+ getOneTimeOffer,
30
+ createOneTimeOffer,
31
+ updateOneTimeOffer,
32
+ deleteOneTimeOffer,
33
+ diffOneTimeProduct,
34
+ formatOutput,
35
+ sortResults
36
+ } from "@gpc-cli/core";
37
+ async function readJsonArray(filePath) {
38
+ const data = await readJsonFile(filePath);
39
+ if (!Array.isArray(data)) {
40
+ const err = new Error(`Expected a JSON array in ${filePath}, got ${typeof data}`);
41
+ Object.assign(err, { code: "USAGE_ERROR", exitCode: 2 });
42
+ throw err;
43
+ }
44
+ return data;
45
+ }
46
+ function registerOneTimeProductsCommands(program) {
47
+ const otp = program.command("one-time-products").alias("otp").description("Manage one-time products and offers (modern OTP API)");
48
+ otp.command("list").description("List one-time products").option("--sort <field>", "Sort by field (prefix with - for descending)").action(async (options) => {
49
+ const config = await loadConfig();
50
+ const packageName = resolvePackageName(program.opts()["app"], config);
51
+ const client = await getClient(config);
52
+ const format = getOutputFormat(program, config);
53
+ const result = await listOneTimeProducts(client, packageName);
54
+ if (options.sort) {
55
+ result.oneTimeProducts = sortResults(result.oneTimeProducts, options.sort);
56
+ }
57
+ const products = result.oneTimeProducts || [];
58
+ if (format !== "json") {
59
+ if (products.length === 0) {
60
+ console.log("No one-time products found.");
61
+ return;
62
+ }
63
+ const summary = products.map((p) => ({
64
+ productId: p.productId,
65
+ purchaseType: p["purchaseType"] || "-",
66
+ listings: p.listings ? Object.keys(p.listings).length : 0,
67
+ firstTitle: p.listings ? Object.values(p.listings)[0]?.title || "-" : "-"
68
+ }));
69
+ console.log(formatOutput(summary, format));
70
+ } else {
71
+ console.log(formatOutput(result, format));
72
+ }
73
+ });
74
+ otp.command("get <product-id>").description("Get a one-time product").action(async (productId) => {
75
+ const config = await loadConfig();
76
+ const packageName = resolvePackageName(program.opts()["app"], config);
77
+ const client = await getClient(config);
78
+ const format = getOutputFormat(program, config);
79
+ const result = await getOneTimeProduct(client, packageName, productId);
80
+ console.log(formatOutput(result, format));
81
+ });
82
+ otp.command("create").description("Create a one-time product from JSON file").requiredOption("--file <path>", "JSON file with product data").action(async (options) => {
83
+ const config = await loadConfig();
84
+ const packageName = resolvePackageName(program.opts()["app"], config);
85
+ const format = getOutputFormat(program, config);
86
+ if (isDryRun(program)) {
87
+ printDryRun(
88
+ {
89
+ command: "one-time-products create",
90
+ action: "create",
91
+ target: `one-time product from ${options.file}`
92
+ },
93
+ format,
94
+ formatOutput
95
+ );
96
+ return;
97
+ }
98
+ const client = await getClient(config);
99
+ const data = await readJsonFile(options.file);
100
+ const result = await createOneTimeProduct(client, packageName, data);
101
+ console.log(formatOutput(result, format));
102
+ });
103
+ otp.command("update <product-id>").description("Update a one-time product from JSON file").requiredOption("--file <path>", "JSON file with product data").option("--update-mask <fields>", "Comma-separated field mask").action(async (productId, options) => {
104
+ const config = await loadConfig();
105
+ const packageName = resolvePackageName(program.opts()["app"], config);
106
+ const format = getOutputFormat(program, config);
107
+ if (isDryRun(program)) {
108
+ printDryRun(
109
+ {
110
+ command: "one-time-products update",
111
+ action: "update",
112
+ target: productId,
113
+ details: { file: options.file }
114
+ },
115
+ format,
116
+ formatOutput
117
+ );
118
+ return;
119
+ }
120
+ const client = await getClient(config);
121
+ const data = await readJsonFile(options.file);
122
+ const result = await updateOneTimeProduct(
123
+ client,
124
+ packageName,
125
+ productId,
126
+ data,
127
+ options.updateMask
128
+ );
129
+ console.log(formatOutput(result, format));
130
+ });
131
+ otp.command("delete <product-id>").description("Delete a one-time product").action(async (productId) => {
132
+ const config = await loadConfig();
133
+ const packageName = resolvePackageName(program.opts()["app"], config);
134
+ await requireConfirm(`Delete one-time product "${productId}"?`, program);
135
+ if (isDryRun(program)) {
136
+ const format = getOutputFormat(program, config);
137
+ printDryRun(
138
+ {
139
+ command: "one-time-products delete",
140
+ action: "delete",
141
+ target: productId
142
+ },
143
+ format,
144
+ formatOutput
145
+ );
146
+ return;
147
+ }
148
+ const client = await getClient(config);
149
+ await deleteOneTimeProduct(client, packageName, productId);
150
+ console.log(`One-time product ${productId} deleted.`);
151
+ });
152
+ const offers = otp.command("offers").description("Manage one-time product offers");
153
+ offers.command("list <product-id>").description("List offers for a one-time product").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").option("--sort <field>", "Sort by field (prefix with - for descending)").action(async (productId, options) => {
154
+ const config = await loadConfig();
155
+ const packageName = resolvePackageName(program.opts()["app"], config);
156
+ const client = await getClient(config);
157
+ const format = getOutputFormat(program, config);
158
+ const result = await listOneTimeOffers(
159
+ client,
160
+ packageName,
161
+ productId,
162
+ options.purchaseOption
163
+ );
164
+ const items = result.oneTimeProductOffers ?? result.oneTimeOffers ?? [];
165
+ if (options.sort) {
166
+ console.log(formatOutput(sortResults(items, options.sort), format));
167
+ } else {
168
+ console.log(
169
+ formatOutput(items.length > 0 ? result : { oneTimeProductOffers: items }, format)
170
+ );
171
+ }
172
+ });
173
+ offers.command("get <product-id> <offer-id>").description("Get an offer for a one-time product").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").action(async (productId, offerId, options) => {
174
+ const config = await loadConfig();
175
+ const packageName = resolvePackageName(program.opts()["app"], config);
176
+ const client = await getClient(config);
177
+ const format = getOutputFormat(program, config);
178
+ const result = await getOneTimeOffer(
179
+ client,
180
+ packageName,
181
+ productId,
182
+ offerId,
183
+ options.purchaseOption
184
+ );
185
+ console.log(formatOutput(result, format));
186
+ });
187
+ offers.command("create <product-id>").description("Create an offer from JSON file").requiredOption("--file <path>", "JSON file with offer data").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").action(async (productId, options) => {
188
+ const config = await loadConfig();
189
+ const packageName = resolvePackageName(program.opts()["app"], config);
190
+ const format = getOutputFormat(program, config);
191
+ if (isDryRun(program)) {
192
+ printDryRun(
193
+ {
194
+ command: "one-time-products offers create",
195
+ action: "create offer for",
196
+ target: productId,
197
+ details: { file: options.file, purchaseOption: options.purchaseOption }
198
+ },
199
+ format,
200
+ formatOutput
201
+ );
202
+ return;
203
+ }
204
+ const client = await getClient(config);
205
+ const data = await readJsonFile(options.file);
206
+ const result = await createOneTimeOffer(
207
+ client,
208
+ packageName,
209
+ productId,
210
+ data,
211
+ options.purchaseOption
212
+ );
213
+ console.log(formatOutput(result, format));
214
+ });
215
+ offers.command("update <product-id> <offer-id>").description("Update an offer from JSON file").requiredOption("--file <path>", "JSON file with offer data").option("--update-mask <fields>", "Comma-separated field mask").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").action(
216
+ async (productId, offerId, options) => {
217
+ const config = await loadConfig();
218
+ const packageName = resolvePackageName(program.opts()["app"], config);
219
+ const format = getOutputFormat(program, config);
220
+ if (isDryRun(program)) {
221
+ printDryRun(
222
+ {
223
+ command: "one-time-products offers update",
224
+ action: "update offer",
225
+ target: `${productId}/${offerId}`,
226
+ details: { file: options.file, purchaseOption: options.purchaseOption }
227
+ },
228
+ format,
229
+ formatOutput
230
+ );
231
+ return;
232
+ }
233
+ const client = await getClient(config);
234
+ const data = await readJsonFile(options.file);
235
+ const result = await updateOneTimeOffer(
236
+ client,
237
+ packageName,
238
+ productId,
239
+ offerId,
240
+ data,
241
+ options.updateMask,
242
+ options.purchaseOption
243
+ );
244
+ console.log(formatOutput(result, format));
245
+ }
246
+ );
247
+ offers.command("delete <product-id> <offer-id>").description("Delete an offer").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").action(async (productId, offerId, options) => {
248
+ const config = await loadConfig();
249
+ const packageName = resolvePackageName(program.opts()["app"], config);
250
+ await requireConfirm(`Delete offer "${offerId}" for product "${productId}"?`, program);
251
+ if (isDryRun(program)) {
252
+ const format = getOutputFormat(program, config);
253
+ printDryRun(
254
+ {
255
+ command: "one-time-products offers delete",
256
+ action: "delete offer",
257
+ target: `${productId}/${offerId}`
258
+ },
259
+ format,
260
+ formatOutput
261
+ );
262
+ return;
263
+ }
264
+ const client = await getClient(config);
265
+ await deleteOneTimeOffer(client, packageName, productId, offerId, options.purchaseOption);
266
+ console.log(`Offer ${offerId} deleted.`);
267
+ });
268
+ offers.command("cancel <product-id> <offer-id>").description("Cancel an offer (permanent, cannot be re-activated)").option("--purchase-option <id>", 'Purchase option ID (default: "-" for all)', "-").action(async (productId, offerId, options) => {
269
+ const config = await loadConfig();
270
+ const packageName = resolvePackageName(program.opts()["app"], config);
271
+ await requireConfirm(
272
+ `Cancel offer "${offerId}"? This is permanent and cannot be undone.`,
273
+ program
274
+ );
275
+ if (isDryRun(program)) {
276
+ const format2 = getOutputFormat(program, config);
277
+ printDryRun(
278
+ {
279
+ command: "otp offers cancel",
280
+ action: "cancel offer",
281
+ target: `${productId}/${offerId}`
282
+ },
283
+ format2,
284
+ formatOutput
285
+ );
286
+ return;
287
+ }
288
+ const client = await getClient(config);
289
+ const format = getOutputFormat(program, config);
290
+ const result = await client.oneTimeProducts.cancelOffer(
291
+ packageName,
292
+ productId,
293
+ options.purchaseOption,
294
+ offerId
295
+ );
296
+ console.log(formatOutput(result, format));
297
+ });
298
+ offers.command("batch-get <product-id>").description("Batch get multiple offers").requiredOption(
299
+ "--file <path>",
300
+ "JSON file with array of {productId, purchaseOptionId, offerId}"
301
+ ).option("--purchase-option <id>", 'Purchase option ID in URL path (default: "-")', "-").action(async (productId, options) => {
302
+ const config = await loadConfig();
303
+ const packageName = resolvePackageName(program.opts()["app"], config);
304
+ const client = await getClient(config);
305
+ const format = getOutputFormat(program, config);
306
+ const requests = await readJsonArray(options.file);
307
+ const result = await client.oneTimeProducts.batchGetOffers(
308
+ packageName,
309
+ productId,
310
+ options.purchaseOption,
311
+ requests.map((r) => ({
312
+ packageName,
313
+ productId: r.productId || productId,
314
+ purchaseOptionId: r.purchaseOptionId || "-",
315
+ offerId: r.offerId
316
+ }))
317
+ );
318
+ console.log(formatOutput(result, format));
319
+ });
320
+ offers.command("batch-update <product-id>").description("Batch create or update multiple offers (max 100)").requiredOption("--file <path>", "JSON file with update requests").option("--purchase-option <id>", 'Purchase option ID in URL path (default: "-")', "-").action(async (productId, options) => {
321
+ const config = await loadConfig();
322
+ const packageName = resolvePackageName(program.opts()["app"], config);
323
+ const format = getOutputFormat(program, config);
324
+ if (isDryRun(program)) {
325
+ printDryRun(
326
+ {
327
+ command: "otp offers batch-update",
328
+ action: "batch update offers for",
329
+ target: productId,
330
+ details: { file: options.file }
331
+ },
332
+ format,
333
+ formatOutput
334
+ );
335
+ return;
336
+ }
337
+ const client = await getClient(config);
338
+ const requests = await readJsonArray(options.file);
339
+ const result = await client.oneTimeProducts.batchUpdateOffers(
340
+ packageName,
341
+ productId,
342
+ options.purchaseOption,
343
+ requests
344
+ );
345
+ console.log(formatOutput(result, format));
346
+ });
347
+ offers.command("batch-update-states <product-id>").description("Batch activate, deactivate, or cancel multiple offers (max 100)").requiredOption("--file <path>", "JSON file with state transition requests").option("--purchase-option <id>", 'Purchase option ID in URL path (default: "-")', "-").action(async (productId, options) => {
348
+ const config = await loadConfig();
349
+ const packageName = resolvePackageName(program.opts()["app"], config);
350
+ const format = getOutputFormat(program, config);
351
+ if (isDryRun(program)) {
352
+ printDryRun(
353
+ {
354
+ command: "otp offers batch-update-states",
355
+ action: "batch update offer states for",
356
+ target: productId,
357
+ details: { file: options.file }
358
+ },
359
+ format,
360
+ formatOutput
361
+ );
362
+ return;
363
+ }
364
+ const client = await getClient(config);
365
+ const requests = await readJsonArray(options.file);
366
+ const result = await client.oneTimeProducts.batchUpdateOfferStates(
367
+ packageName,
368
+ productId,
369
+ options.purchaseOption,
370
+ requests
371
+ );
372
+ console.log(formatOutput(result, format));
373
+ });
374
+ offers.command("batch-delete <product-id>").description("Batch delete multiple offers (max 100)").requiredOption(
375
+ "--file <path>",
376
+ "JSON file with array of {productId, purchaseOptionId, offerId}"
377
+ ).option("--purchase-option <id>", 'Purchase option ID in URL path (default: "-")', "-").action(async (productId, options) => {
378
+ const config = await loadConfig();
379
+ const packageName = resolvePackageName(program.opts()["app"], config);
380
+ await requireConfirm(`Batch delete offers for product "${productId}"?`, program);
381
+ if (isDryRun(program)) {
382
+ const format = getOutputFormat(program, config);
383
+ printDryRun(
384
+ {
385
+ command: "otp offers batch-delete",
386
+ action: "batch delete offers for",
387
+ target: productId,
388
+ details: { file: options.file }
389
+ },
390
+ format,
391
+ formatOutput
392
+ );
393
+ return;
394
+ }
395
+ const client = await getClient(config);
396
+ const requests = await readJsonArray(options.file);
397
+ await client.oneTimeProducts.batchDeleteOffers(
398
+ packageName,
399
+ productId,
400
+ options.purchaseOption,
401
+ requests
402
+ );
403
+ console.log(`Batch delete completed.`);
404
+ });
405
+ const po = otp.command("purchase-options").description("Manage purchase option batch operations");
406
+ po.command("batch-delete <product-id>").description("Batch delete purchase options (max 100)").requiredOption("--file <path>", "JSON file with delete requests").action(async (productId, options) => {
407
+ const config = await loadConfig();
408
+ const packageName = resolvePackageName(program.opts()["app"], config);
409
+ await requireConfirm(`Batch delete purchase options for product "${productId}"?`, program);
410
+ if (isDryRun(program)) {
411
+ const format = getOutputFormat(program, config);
412
+ printDryRun(
413
+ {
414
+ command: "otp purchase-options batch-delete",
415
+ action: "batch delete purchase options for",
416
+ target: productId,
417
+ details: { file: options.file }
418
+ },
419
+ format,
420
+ formatOutput
421
+ );
422
+ return;
423
+ }
424
+ const client = await getClient(config);
425
+ const requests = await readJsonArray(options.file);
426
+ await client.oneTimeProducts.batchDeletePurchaseOptions(packageName, productId, requests);
427
+ console.log(`Batch delete completed.`);
428
+ });
429
+ po.command("batch-update-states <product-id>").description("Batch activate or deactivate purchase options (max 100)").requiredOption("--file <path>", "JSON file with state transition requests").action(async (productId, options) => {
430
+ const config = await loadConfig();
431
+ const packageName = resolvePackageName(program.opts()["app"], config);
432
+ const format = getOutputFormat(program, config);
433
+ if (isDryRun(program)) {
434
+ printDryRun(
435
+ {
436
+ command: "otp purchase-options batch-update-states",
437
+ action: "batch update purchase option states for",
438
+ target: productId,
439
+ details: { file: options.file }
440
+ },
441
+ format,
442
+ formatOutput
443
+ );
444
+ return;
445
+ }
446
+ const client = await getClient(config);
447
+ const requests = await readJsonArray(options.file);
448
+ const result = await client.oneTimeProducts.batchUpdatePurchaseOptionStates(
449
+ packageName,
450
+ productId,
451
+ requests
452
+ );
453
+ console.log(formatOutput(result, format));
454
+ });
455
+ otp.command("diff <product-id>").description("Compare local JSON file against remote one-time product").requiredOption("--file <path>", "Local JSON file to compare against remote").action(async (productId, options) => {
456
+ const config = await loadConfig();
457
+ const packageName = resolvePackageName(program.opts()["app"], config);
458
+ const client = await getClient(config);
459
+ const format = getOutputFormat(program, config);
460
+ const localData = await readJsonFile(options.file);
461
+ const diffs = await diffOneTimeProduct(client, packageName, productId, localData);
462
+ if (diffs.length === 0) {
463
+ console.log("No differences found.");
464
+ } else {
465
+ console.log(formatOutput(diffs, format));
466
+ }
467
+ });
468
+ }
469
+ export {
470
+ registerOneTimeProductsCommands
471
+ };
472
+ //# sourceMappingURL=one-time-products-Y5RNIPV2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/one-time-products.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\n\nimport type { OneTimeProduct } from \"@gpc-cli/api\";\nimport {\n listOneTimeProducts,\n getOneTimeProduct,\n createOneTimeProduct,\n updateOneTimeProduct,\n deleteOneTimeProduct,\n listOneTimeOffers,\n getOneTimeOffer,\n createOneTimeOffer,\n updateOneTimeOffer,\n deleteOneTimeOffer,\n diffOneTimeProduct,\n formatOutput,\n sortResults,\n} from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { requireConfirm } from \"../prompt.js\";\nimport { readJsonFile } from \"../json.js\";\n\nasync function readJsonArray<T = any>(filePath: string): Promise<T[]> {\n const data = await readJsonFile(filePath);\n if (!Array.isArray(data)) {\n const err = new Error(`Expected a JSON array in ${filePath}, got ${typeof data}`);\n Object.assign(err, { code: \"USAGE_ERROR\", exitCode: 2 });\n throw err;\n }\n return data as T[];\n}\n\nexport function registerOneTimeProductsCommands(program: Command): void {\n const otp = program\n .command(\"one-time-products\")\n .alias(\"otp\")\n .description(\"Manage one-time products and offers (modern OTP API)\");\n\n otp\n .command(\"list\")\n .description(\"List one-time products\")\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 result = await listOneTimeProducts(client, packageName);\n if (options.sort) {\n result.oneTimeProducts = sortResults(result.oneTimeProducts, options.sort);\n }\n const products = result.oneTimeProducts || [];\n if (format !== \"json\") {\n if (products.length === 0) {\n console.log(\"No one-time products found.\");\n return;\n }\n const summary = products.map((p: OneTimeProduct) => ({\n productId: p.productId,\n purchaseType: (p as unknown as Record<string, unknown>)[\"purchaseType\"] || \"-\",\n listings: p.listings ? Object.keys(p.listings).length : 0,\n firstTitle: p.listings ? Object.values(p.listings)[0]?.title || \"-\" : \"-\",\n }));\n console.log(formatOutput(summary, format));\n } else {\n console.log(formatOutput(result, format));\n }\n });\n\n otp\n .command(\"get <product-id>\")\n .description(\"Get a one-time product\")\n .action(async (productId: string) => {\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 getOneTimeProduct(client, packageName, productId);\n console.log(formatOutput(result, format));\n });\n\n otp\n .command(\"create\")\n .description(\"Create a one-time product from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with product data\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"one-time-products create\",\n action: \"create\",\n target: `one-time product from ${options.file}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const data = await readJsonFile(options.file);\n const result = await createOneTimeProduct(client, packageName, data as any);\n console.log(formatOutput(result, format));\n });\n\n otp\n .command(\"update <product-id>\")\n .description(\"Update a one-time product from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with product data\")\n .option(\"--update-mask <fields>\", \"Comma-separated field mask\")\n .action(async (productId: string, options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"one-time-products update\",\n action: \"update\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const data = await readJsonFile(options.file);\n const result = await updateOneTimeProduct(\n client,\n packageName,\n productId,\n data as any,\n options.updateMask,\n );\n console.log(formatOutput(result, format));\n });\n\n otp\n .command(\"delete <product-id>\")\n .description(\"Delete a one-time product\")\n .action(async (productId: string) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Delete one-time product \"${productId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"one-time-products delete\",\n action: \"delete\",\n target: productId,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n await deleteOneTimeProduct(client, packageName, productId);\n console.log(`One-time product ${productId} deleted.`);\n });\n\n // --- Offers ---\n const offers = otp.command(\"offers\").description(\"Manage one-time product offers\");\n\n offers\n .command(\"list <product-id>\")\n .description(\"List offers for a one-time product\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .option(\"--sort <field>\", \"Sort by field (prefix with - for descending)\")\n .action(async (productId: string, options: { purchaseOption: string; sort?: string }) => {\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 listOneTimeOffers(\n client,\n packageName,\n productId,\n options.purchaseOption,\n );\n const items = result.oneTimeProductOffers ?? result.oneTimeOffers ?? [];\n if (options.sort) {\n console.log(formatOutput(sortResults(items, options.sort), format));\n } else {\n console.log(\n formatOutput(items.length > 0 ? result : { oneTimeProductOffers: items }, format),\n );\n }\n });\n\n offers\n .command(\"get <product-id> <offer-id>\")\n .description(\"Get an offer for a one-time product\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .action(async (productId: string, offerId: string, options: { purchaseOption: string }) => {\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 getOneTimeOffer(\n client,\n packageName,\n productId,\n offerId,\n options.purchaseOption,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"create <product-id>\")\n .description(\"Create an offer from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with offer data\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .action(async (productId: string, options: { file: string; purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"one-time-products offers create\",\n action: \"create offer for\",\n target: productId,\n details: { file: options.file, purchaseOption: options.purchaseOption },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const data = await readJsonFile(options.file);\n const result = await createOneTimeOffer(\n client,\n packageName,\n productId,\n data as any,\n options.purchaseOption,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"update <product-id> <offer-id>\")\n .description(\"Update an offer from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with offer data\")\n .option(\"--update-mask <fields>\", \"Comma-separated field mask\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .action(\n async (\n productId: string,\n offerId: string,\n options: { file: string; updateMask?: string; purchaseOption: string },\n ) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"one-time-products offers update\",\n action: \"update offer\",\n target: `${productId}/${offerId}`,\n details: { file: options.file, purchaseOption: options.purchaseOption },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const data = await readJsonFile(options.file);\n const result = await updateOneTimeOffer(\n client,\n packageName,\n productId,\n offerId,\n data as any,\n options.updateMask,\n options.purchaseOption,\n );\n console.log(formatOutput(result, format));\n },\n );\n\n offers\n .command(\"delete <product-id> <offer-id>\")\n .description(\"Delete an offer\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .action(async (productId: string, offerId: string, options: { purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Delete offer \"${offerId}\" for product \"${productId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"one-time-products offers delete\",\n action: \"delete offer\",\n target: `${productId}/${offerId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n await deleteOneTimeOffer(client, packageName, productId, offerId, options.purchaseOption);\n console.log(`Offer ${offerId} deleted.`);\n });\n\n offers\n .command(\"cancel <product-id> <offer-id>\")\n .description(\"Cancel an offer (permanent, cannot be re-activated)\")\n .option(\"--purchase-option <id>\", 'Purchase option ID (default: \"-\" for all)', \"-\")\n .action(async (productId: string, offerId: string, options: { purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(\n `Cancel offer \"${offerId}\"? This is permanent and cannot be undone.`,\n program,\n );\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"otp offers cancel\",\n action: \"cancel offer\",\n target: `${productId}/${offerId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n const result = await client.oneTimeProducts.cancelOffer(\n packageName,\n productId,\n options.purchaseOption,\n offerId,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-get <product-id>\")\n .description(\"Batch get multiple offers\")\n .requiredOption(\n \"--file <path>\",\n \"JSON file with array of {productId, purchaseOptionId, offerId}\",\n )\n .option(\"--purchase-option <id>\", 'Purchase option ID in URL path (default: \"-\")', \"-\")\n .action(async (productId: string, options: { file: string; purchaseOption: string }) => {\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 requests = await readJsonArray(options.file);\n const result = await client.oneTimeProducts.batchGetOffers(\n packageName,\n productId,\n options.purchaseOption,\n requests.map((r: any) => ({\n packageName,\n productId: r.productId || productId,\n purchaseOptionId: r.purchaseOptionId || \"-\",\n offerId: r.offerId,\n })),\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-update <product-id>\")\n .description(\"Batch create or update multiple offers (max 100)\")\n .requiredOption(\"--file <path>\", \"JSON file with update requests\")\n .option(\"--purchase-option <id>\", 'Purchase option ID in URL path (default: \"-\")', \"-\")\n .action(async (productId: string, options: { file: string; purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"otp offers batch-update\",\n action: \"batch update offers for\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const requests = await readJsonArray(options.file);\n const result = await client.oneTimeProducts.batchUpdateOffers(\n packageName,\n productId,\n options.purchaseOption,\n requests,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-update-states <product-id>\")\n .description(\"Batch activate, deactivate, or cancel multiple offers (max 100)\")\n .requiredOption(\"--file <path>\", \"JSON file with state transition requests\")\n .option(\"--purchase-option <id>\", 'Purchase option ID in URL path (default: \"-\")', \"-\")\n .action(async (productId: string, options: { file: string; purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"otp offers batch-update-states\",\n action: \"batch update offer states for\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const requests = await readJsonArray(options.file);\n const result = await client.oneTimeProducts.batchUpdateOfferStates(\n packageName,\n productId,\n options.purchaseOption,\n requests,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-delete <product-id>\")\n .description(\"Batch delete multiple offers (max 100)\")\n .requiredOption(\n \"--file <path>\",\n \"JSON file with array of {productId, purchaseOptionId, offerId}\",\n )\n .option(\"--purchase-option <id>\", 'Purchase option ID in URL path (default: \"-\")', \"-\")\n .action(async (productId: string, options: { file: string; purchaseOption: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Batch delete offers for product \"${productId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"otp offers batch-delete\",\n action: \"batch delete offers for\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const requests = await readJsonArray(options.file);\n await client.oneTimeProducts.batchDeleteOffers(\n packageName,\n productId,\n options.purchaseOption,\n requests,\n );\n console.log(`Batch delete completed.`);\n });\n\n // --- Purchase Options batch ---\n const po = otp.command(\"purchase-options\").description(\"Manage purchase option batch operations\");\n\n po.command(\"batch-delete <product-id>\")\n .description(\"Batch delete purchase options (max 100)\")\n .requiredOption(\"--file <path>\", \"JSON file with delete requests\")\n .action(async (productId: string, options: { file: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Batch delete purchase options for product \"${productId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"otp purchase-options batch-delete\",\n action: \"batch delete purchase options for\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const requests = await readJsonArray(options.file);\n await client.oneTimeProducts.batchDeletePurchaseOptions(packageName, productId, requests);\n console.log(`Batch delete completed.`);\n });\n\n po.command(\"batch-update-states <product-id>\")\n .description(\"Batch activate or deactivate purchase options (max 100)\")\n .requiredOption(\"--file <path>\", \"JSON file with state transition requests\")\n .action(async (productId: string, options: { file: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"otp purchase-options batch-update-states\",\n action: \"batch update purchase option states for\",\n target: productId,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const requests = await readJsonArray(options.file);\n const result = await client.oneTimeProducts.batchUpdatePurchaseOptionStates(\n packageName,\n productId,\n requests,\n );\n console.log(formatOutput(result, format));\n });\n\n // --- Diff ---\n otp\n .command(\"diff <product-id>\")\n .description(\"Compare local JSON file against remote one-time product\")\n .requiredOption(\"--file <path>\", \"Local JSON file to compare against remote\")\n .action(async (productId: string, options: { file: string }) => {\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 localData = (await readJsonFile(options.file)) as OneTimeProduct;\n const diffs = await diffOneTimeProduct(client, packageName, productId, localData);\n if (diffs.length === 0) {\n console.log(\"No differences found.\");\n } else {\n console.log(formatOutput(diffs, format));\n }\n });\n}\n"],"mappings":"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","names":["format"]}
@@ -112,10 +112,7 @@ async function runPreflightAction(program, file, options) {
112
112
  } else {
113
113
  for (const finding of result.findings) {
114
114
  const icon = SEVERITY_ICONS[finding.severity];
115
- const label = severityColor(
116
- finding.severity,
117
- `${icon} ${finding.severity.toUpperCase()}`
118
- );
115
+ const label = severityColor(finding.severity, `${icon} ${finding.severity.toUpperCase()}`);
119
116
  console.log(`${label} ${finding.title}`);
120
117
  console.log(` ${dim(finding.message)}`);
121
118
  if (finding.suggestion) {
@@ -138,15 +135,9 @@ async function runPreflightAction(program, file, options) {
138
135
  if (Date.now() < (/* @__PURE__ */ new Date("2026-09-01")).getTime()) {
139
136
  console.log("");
140
137
  console.log(
141
- dim(
142
- "Note: Apps must be registered by a verified developer to be installable on"
143
- )
144
- );
145
- console.log(
146
- dim(
147
- "certified Android devices after Sep 2026. Run 'gpc verify' for details."
148
- )
138
+ dim("Note: Apps must be registered by a verified developer to be installable on")
149
139
  );
140
+ console.log(dim("certified Android devices after Sep 2026. Run 'gpc verify' for details."));
150
141
  }
151
142
  }
152
143
  if (!result.passed) {
@@ -156,4 +147,4 @@ async function runPreflightAction(program, file, options) {
156
147
  export {
157
148
  registerPreflightCommand
158
149
  };
159
- //# sourceMappingURL=preflight-W3JAJ4GO.js.map
150
+ //# sourceMappingURL=preflight-KIWZPFTX.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/preflight.ts"],"sourcesContent":["// Named exports only. No default export.\n\nimport type { Command } from \"commander\";\nimport { runPreflight, getAllScannerNames, formatOutput, GpcError } from \"@gpc-cli/core\";\nimport type { FindingSeverity } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { green, red, yellow, dim, bold } from \"../colors.js\";\n\nconst SEVERITY_ICONS: Record<FindingSeverity, string> = {\n critical: \"✗\",\n error: \"✗\",\n warning: \"⚠\",\n info: \"ℹ\",\n};\n\nfunction severityColor(severity: FindingSeverity, text: string): string {\n switch (severity) {\n case \"critical\":\n return bold(red(text));\n case \"error\":\n return red(text);\n case \"warning\":\n return yellow(text);\n case \"info\":\n return dim(text);\n }\n}\n\nexport function registerPreflightCommand(program: Command): void {\n const cmd = program\n .command(\"preflight [file]\")\n .description(\"Pre-submission compliance scanner for AAB files (offline)\")\n .option(\n \"--fail-on <severity>\",\n \"Fail if any finding meets or exceeds severity: critical, error, warning, info\",\n \"error\",\n )\n .option(\"--scanners <names>\", \"Comma-separated scanner names to run (default: all)\")\n .option(\"--metadata <dir>\", \"Path to metadata directory (Fastlane format) for listing checks\")\n .option(\"--source <dir>\", \"Path to source directory for code scanning\")\n .option(\"--config <path>\", \"Path to .preflightrc.json config file\")\n .action(async (file: string | undefined, options) => {\n await runPreflightAction(program, file, options);\n });\n\n // Subcommand: preflight manifest\n cmd\n .command(\"manifest <file>\")\n .description(\"Run manifest scanner only\")\n .option(\"--fail-on <severity>\", \"Fail threshold\", \"error\")\n .action(async (file: string, options) => {\n await runPreflightAction(program, file, { ...options, scanners: \"manifest\" });\n });\n\n // Subcommand: preflight permissions\n cmd\n .command(\"permissions <file>\")\n .description(\"Run permissions scanner only\")\n .option(\"--fail-on <severity>\", \"Fail threshold\", \"error\")\n .action(async (file: string, options) => {\n await runPreflightAction(program, file, { ...options, scanners: \"permissions\" });\n });\n\n // Subcommand: preflight metadata\n cmd\n .command(\"metadata <dir>\")\n .description(\"Run metadata scanner on a listings directory\")\n .option(\"--fail-on <severity>\", \"Fail threshold\", \"error\")\n .action(async (dir: string, options) => {\n await runPreflightAction(program, undefined, {\n ...options,\n metadata: dir,\n scanners: \"metadata\",\n });\n });\n\n // Subcommand: preflight codescan\n cmd\n .command(\"codescan <dir>\")\n .description(\"Run code scanners (secrets, billing, privacy) on source directory\")\n .option(\"--fail-on <severity>\", \"Fail threshold\", \"error\")\n .action(async (dir: string, options) => {\n await runPreflightAction(program, undefined, {\n ...options,\n source: dir,\n scanners: \"secrets,billing,privacy\",\n });\n });\n}\n\nasync function runPreflightAction(\n program: Command,\n file: string | undefined,\n options: Record<string, string | undefined>,\n): Promise<void> {\n if (!file && !options[\"metadata\"] && !options[\"source\"]) {\n throw new GpcError(\n \"Provide an AAB file, --metadata <dir>, or --source <dir>\",\n \"MISSING_INPUT\",\n 2,\n \"gpc preflight app.aab\",\n );\n }\n\n const failOn = options[\"failOn\"] as FindingSeverity | undefined;\n const validSeverities = new Set([\"critical\", \"error\", \"warning\", \"info\"]);\n if (failOn && !validSeverities.has(failOn)) {\n throw new GpcError(\n `Invalid --fail-on value \"${failOn}\". Use: critical, error, warning, info`,\n \"INVALID_OPTION\",\n 2,\n );\n }\n\n const scannerNames = options[\"scanners\"]?.split(\",\").map((s) => s.trim());\n if (scannerNames) {\n const known = new Set(getAllScannerNames());\n const unknown = scannerNames.filter((s) => !known.has(s));\n if (unknown.length > 0) {\n throw new GpcError(\n `Unknown scanner(s): ${unknown.join(\", \")}. Available: ${getAllScannerNames().join(\", \")}`,\n \"UNKNOWN_SCANNER\",\n 2,\n );\n }\n }\n\n const config = await loadConfig();\n const format = getOutputFormat(program, config);\n\n const result = await runPreflight({\n aabPath: file,\n metadataDir: options[\"metadata\"],\n sourceDir: options[\"source\"],\n scanners: scannerNames,\n failOn,\n configPath: options[\"config\"],\n });\n\n if (format === \"json\") {\n console.log(formatOutput(result, format));\n } else {\n // Header\n console.log(bold(\"GPC Preflight Scanner\"));\n if (file) console.log(dim(`File: ${file}`));\n console.log(dim(`Scanners: ${result.scanners.join(\", \")}`));\n console.log(\"\");\n\n if (result.findings.length === 0) {\n console.log(green(\"✓ No issues found\"));\n } else {\n // Group by severity\n for (const finding of result.findings) {\n const icon = SEVERITY_ICONS[finding.severity];\n const label = severityColor(finding.severity, `${icon} ${finding.severity.toUpperCase()}`);\n console.log(`${label} ${finding.title}`);\n console.log(` ${dim(finding.message)}`);\n if (finding.suggestion) {\n console.log(` ${dim(\"→\")} ${finding.suggestion}`);\n }\n if (finding.policyUrl) {\n console.log(` ${dim(finding.policyUrl)}`);\n }\n console.log(\"\");\n }\n }\n\n // Summary line\n const parts: string[] = [];\n if (result.summary.critical > 0) parts.push(bold(red(`${result.summary.critical} critical`)));\n if (result.summary.error > 0) parts.push(red(`${result.summary.error} error`));\n if (result.summary.warning > 0) parts.push(yellow(`${result.summary.warning} warning`));\n if (result.summary.info > 0) parts.push(dim(`${result.summary.info} info`));\n\n const summaryLine = parts.length > 0 ? parts.join(\", \") : green(\"0 issues\");\n const passedLabel = result.passed ? green(\"✓ PASSED\") : red(\"✗ FAILED\");\n console.log(`${passedLabel} ${summaryLine} ${dim(`(${result.durationMs}ms)`)}`);\n\n // Verification deadline awareness (auto-expires Sep 2026)\n if (Date.now() < new Date(\"2026-09-01\").getTime()) {\n console.log(\"\");\n console.log(\n dim(\"Note: Apps must be registered by a verified developer to be installable on\"),\n );\n console.log(dim(\"certified Android devices after Sep 2026. Run 'gpc verify' for details.\"));\n }\n }\n\n if (!result.passed) {\n process.exitCode = 6;\n }\n}\n"],"mappings":"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","names":[]}
@@ -2,7 +2,7 @@
2
2
  import {
3
3
  getClient,
4
4
  resolvePackageName
5
- } from "./chunk-NQH4G7BI.js";
5
+ } from "./chunk-JDRY7HK5.js";
6
6
  import {
7
7
  getOutputFormat
8
8
  } from "./chunk-ELXAK7GI.js";
@@ -15,7 +15,7 @@ function registerPricingCommands(program) {
15
15
  pricing.command("convert").description("Convert a price to all regional prices").option("--from <currency>", "Source currency code (e.g. USD)").option("--amount <number>", "Price amount (e.g. 4.99)").action(async (options) => {
16
16
  const config = await loadConfig();
17
17
  const packageName = resolvePackageName(program.opts()["app"], config);
18
- const { isInteractive, requireOption } = await import("./prompt-GXC2JSLA.js");
18
+ const { isInteractive, requireOption } = await import("./prompt-HJXNXXAR.js");
19
19
  const interactive = isInteractive(program);
20
20
  options.from = await requireOption(
21
21
  "from",
@@ -74,4 +74,4 @@ function registerPricingCommands(program) {
74
74
  export {
75
75
  registerPricingCommands
76
76
  };
77
- //# sourceMappingURL=pricing-JJZFICFL.js.map
77
+ //# sourceMappingURL=pricing-HYQRXKNO.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/pricing.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\n\nimport { convertRegionPrices, formatOutput } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\n\nexport function registerPricingCommands(program: Command): void {\n const pricing = program.command(\"pricing\").description(\"Pricing and regional price conversion\");\n\n pricing\n .command(\"convert\")\n .description(\"Convert a price to all regional prices\")\n .option(\"--from <currency>\", \"Source currency code (e.g. USD)\")\n .option(\"--amount <number>\", \"Price amount (e.g. 4.99)\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const { isInteractive, requireOption } = await import(\"../prompt.js\");\n const interactive = isInteractive(program);\n\n options.from = await requireOption(\n \"from\",\n options.from,\n {\n message: \"Source currency code (e.g. USD):\",\n default: \"USD\",\n },\n interactive,\n );\n\n options.amount = await requireOption(\n \"amount\",\n options.amount,\n {\n message: \"Price amount (e.g. 4.99):\",\n },\n interactive,\n );\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n\n try {\n const result = await convertRegionPrices(client, packageName, options.from, options.amount);\n if (format !== \"json\") {\n const prices = (result as unknown as Record<string, unknown>)[\"convertedRegionPrices\"] as\n | Record<string, Record<string, unknown>>\n | undefined;\n if (prices) {\n const rows = Object.entries(prices).map(([region, data]) => {\n const money = data[\"price\"] as Record<string, unknown> | undefined;\n const units = money?.[\"units\"] || \"0\";\n const nanos = String(money?.[\"nanos\"] || 0)\n .padStart(9, \"0\")\n .slice(0, 2);\n return {\n region,\n price: money\n ? `${units}.${nanos}`\n : data[\"priceMicros\"]\n ? String(Number(data[\"priceMicros\"]) / 1_000_000)\n : \"-\",\n currencyCode: (money?.[\"currencyCode\"] || data[\"currencyCode\"] || \"-\") as string,\n };\n });\n console.log(formatOutput(rows, format));\n } else {\n console.log(formatOutput(result, format));\n }\n } else {\n console.log(formatOutput(result, format));\n }\n } catch (error) {\n const { PlayApiError } = await import(\"@gpc-cli/api\");\n if (\n error instanceof PlayApiError &&\n (error.code === \"API_HTTP_400\" || error.code === \"API_EDIT_EXPIRED\")\n ) {\n const err = new Error(\n \"Price conversion is not available for this app. \" +\n \"Ensure the app has monetization configured in Google Play Console (at least one paid product or subscription).\",\n );\n Object.assign(err, { code: \"API_PRICING_UNAVAILABLE\", exitCode: 4 });\n throw err;\n }\n throw error;\n }\n });\n}\n"],"mappings":";;;;;;;;;;AAEA,SAAS,kBAAkB;AAE3B,SAAS,qBAAqB,oBAAoB;AAG3C,SAAS,wBAAwB,SAAwB;AAC9D,QAAM,UAAU,QAAQ,QAAQ,SAAS,EAAE,YAAY,uCAAuC;AAE9F,UACG,QAAQ,SAAS,EACjB,YAAY,wCAAwC,EACpD,OAAO,qBAAqB,iCAAiC,EAC7D,OAAO,qBAAqB,0BAA0B,EACtD,OAAO,OAAO,YAAY;AACzB,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,cAAc,mBAAmB,QAAQ,KAAK,EAAE,KAAK,GAAG,MAAM;AACpE,UAAM,EAAE,eAAe,cAAc,IAAI,MAAM,OAAO,sBAAc;AACpE,UAAM,cAAc,cAAc,OAAO;AAEzC,YAAQ,OAAO,MAAM;AAAA,MACnB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,QACT,SAAS;AAAA,MACX;AAAA,MACA;AAAA,IACF;AAEA,YAAQ,SAAS,MAAM;AAAA,MACrB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,MACX;AAAA,MACA;AAAA,IACF;AACA,UAAM,SAAS,MAAM,UAAU,MAAM;AACrC,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAE9C,QAAI;AACF,YAAM,SAAS,MAAM,oBAAoB,QAAQ,aAAa,QAAQ,MAAM,QAAQ,MAAM;AAC1F,UAAI,WAAW,QAAQ;AACrB,cAAM,SAAU,OAA8C,uBAAuB;AAGrF,YAAI,QAAQ;AACV,gBAAM,OAAO,OAAO,QAAQ,MAAM,EAAE,IAAI,CAAC,CAAC,QAAQ,IAAI,MAAM;AAC1D,kBAAM,QAAQ,KAAK,OAAO;AAC1B,kBAAM,QAAQ,QAAQ,OAAO,KAAK;AAClC,kBAAM,QAAQ,OAAO,QAAQ,OAAO,KAAK,CAAC,EACvC,SAAS,GAAG,GAAG,EACf,MAAM,GAAG,CAAC;AACb,mBAAO;AAAA,cACL;AAAA,cACA,OAAO,QACH,GAAG,KAAK,IAAI,KAAK,KACjB,KAAK,aAAa,IAChB,OAAO,OAAO,KAAK,aAAa,CAAC,IAAI,GAAS,IAC9C;AAAA,cACN,cAAe,QAAQ,cAAc,KAAK,KAAK,cAAc,KAAK;AAAA,YACpE;AAAA,UACF,CAAC;AACD,kBAAQ,IAAI,aAAa,MAAM,MAAM,CAAC;AAAA,QACxC,OAAO;AACL,kBAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,QAC1C;AAAA,MACF,OAAO;AACL,gBAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,MAC1C;AAAA,IACF,SAAS,OAAO;AACd,YAAM,EAAE,aAAa,IAAI,MAAM,OAAO,cAAc;AACpD,UACE,iBAAiB,iBAChB,MAAM,SAAS,kBAAkB,MAAM,SAAS,qBACjD;AACA,cAAM,MAAM,IAAI;AAAA,UACd;AAAA,QAEF;AACA,eAAO,OAAO,KAAK,EAAE,MAAM,2BAA2B,UAAU,EAAE,CAAC;AACnE,cAAM;AAAA,MACR;AACA,YAAM;AAAA,IACR;AAAA,EACF,CAAC;AACL;","names":[]}