@gpc-cli/cli 0.9.53 → 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.
- package/README.md +19 -19
- package/dist/{anomalies-V3AFS4LD.js → anomalies-I5P6QQAZ.js} +14 -5
- package/dist/anomalies-I5P6QQAZ.js.map +1 -0
- package/dist/{apps-4GP3FD7O.js → apps-WPUQOQLL.js} +2 -2
- package/dist/{audit-VTWXTXC6.js → audit-M7GYID74.js} +2 -2
- package/dist/{auth-BA4FE2PO.js → auth-4DRT7ZH2.js} +10 -4
- package/dist/auth-4DRT7ZH2.js.map +1 -0
- package/dist/bin.js +2 -2
- package/dist/{bundle-F7MUVC5J.js → bundle-PFTE7XMU.js} +4 -2
- package/dist/bundle-PFTE7XMU.js.map +1 -0
- package/dist/{cache-XKPLZYEB.js → cache-FGNP7Y37.js} +6 -2
- package/dist/cache-FGNP7Y37.js.map +1 -0
- package/dist/{chunk-SLNJEAMK.js → chunk-22XCOLZX.js} +4 -2
- package/dist/chunk-22XCOLZX.js.map +1 -0
- package/dist/{chunk-GQVZ2VCT.js → chunk-6HIY4IGM.js} +48 -45
- package/dist/chunk-6HIY4IGM.js.map +1 -0
- package/dist/{chunk-BCBXQC7J.js → chunk-E7SVZ7RF.js} +1 -1
- package/dist/chunk-E7SVZ7RF.js.map +1 -0
- package/dist/{chunk-NQH4G7BI.js → chunk-JDRY7HK5.js} +6 -9
- package/dist/chunk-JDRY7HK5.js.map +1 -0
- package/dist/{chunk-YFUBD2XB.js → chunk-RZQSEDKI.js} +6 -9
- package/dist/chunk-RZQSEDKI.js.map +1 -0
- package/dist/{chunk-A7VRCCNS.js → chunk-WSLFHX5X.js} +3 -3
- package/dist/chunk-WSLFHX5X.js.map +1 -0
- package/dist/{config-KT3PIL52.js → config-PUINDZON.js} +3 -3
- package/dist/{data-safety-AFMD6MYI.js → data-safety-46VY64OO.js} +12 -4
- package/dist/data-safety-46VY64OO.js.map +1 -0
- package/dist/{device-tiers-AQAMUQXI.js → device-tiers-MNZYMG3Y.js} +2 -2
- package/dist/device-tiers-MNZYMG3Y.js.map +1 -0
- package/dist/{diff-6EO4ID6W.js → diff-OBSHUSTL.js} +2 -2
- package/dist/diff-OBSHUSTL.js.map +1 -0
- package/dist/{docs-GMFN6V4K.js → docs-GP6AEX4N.js} +8 -4
- package/dist/docs-GP6AEX4N.js.map +1 -0
- package/dist/{doctor-GNI44SIP.js → doctor-T3QFYBRV.js} +5 -5
- package/dist/doctor-T3QFYBRV.js.map +1 -0
- package/dist/{external-transactions-LCZALS3V.js → external-transactions-JL3G4IG5.js} +11 -5
- package/dist/external-transactions-JL3G4IG5.js.map +1 -0
- package/dist/{feedback-XVTGMIBB.js → feedback-AULXQLJ7.js} +3 -3
- package/dist/feedback-AULXQLJ7.js.map +1 -0
- package/dist/{games-ZSNGEI7A.js → games-SVFN2YIS.js} +2 -2
- package/dist/games-SVFN2YIS.js.map +1 -0
- package/dist/{generated-apks-RX2IUWSF.js → generated-apks-TC33S2YN.js} +2 -2
- package/dist/generated-apks-TC33S2YN.js.map +1 -0
- package/dist/{grants-EBPECI26.js → grants-UHNBPIFD.js} +10 -3
- package/dist/grants-UHNBPIFD.js.map +1 -0
- package/dist/{iap-OUI5YYN4.js → iap-ETOL7OAC.js} +4 -4
- package/dist/iap-ETOL7OAC.js.map +1 -0
- package/dist/index.js +1 -1
- package/dist/{init-WSTQTJOD.js → init-6CCSVJC2.js} +7 -3
- package/dist/init-6CCSVJC2.js.map +1 -0
- package/dist/{install-skills-JKPYZHYS.js → install-skills-S342NJ7T.js} +7 -3
- package/dist/{install-skills-JKPYZHYS.js.map → install-skills-S342NJ7T.js.map} +1 -1
- package/dist/{internal-sharing-ONNIWIAT.js → internal-sharing-KW6YENDG.js} +2 -2
- package/dist/internal-sharing-KW6YENDG.js.map +1 -0
- package/dist/{listings-LNX6MQYN.js → listings-MOHHHNE6.js} +27 -7
- package/dist/listings-MOHHHNE6.js.map +1 -0
- package/dist/one-time-products-Y5RNIPV2.js +472 -0
- package/dist/one-time-products-Y5RNIPV2.js.map +1 -0
- package/dist/{preflight-W3JAJ4GO.js → preflight-KIWZPFTX.js} +4 -13
- package/dist/preflight-KIWZPFTX.js.map +1 -0
- package/dist/{pricing-JJZFICFL.js → pricing-HYQRXKNO.js} +3 -3
- package/dist/pricing-HYQRXKNO.js.map +1 -0
- package/dist/{prompt-GXC2JSLA.js → prompt-HJXNXXAR.js} +2 -2
- package/dist/{publish-P5KIGSLI.js → publish-26SSZ2W3.js} +11 -5
- package/dist/publish-26SSZ2W3.js.map +1 -0
- package/dist/{purchases-UBFLNYZC.js → purchases-AHWGLW6V.js} +35 -32
- package/dist/purchases-AHWGLW6V.js.map +1 -0
- package/dist/{quickstart-Z5Y3FYJU.js → quickstart-FOWM3OKT.js} +7 -2
- package/dist/quickstart-FOWM3OKT.js.map +1 -0
- package/dist/{recovery-YE3Z7NIN.js → recovery-B7DZQ6XG.js} +28 -12
- package/dist/recovery-B7DZQ6XG.js.map +1 -0
- package/dist/{releases-LUAHKIMY.js → releases-XY57V22L.js} +33 -8
- package/dist/releases-XY57V22L.js.map +1 -0
- package/dist/{reviews-YCBBM656.js → reviews-P4M6BEJI.js} +4 -6
- package/dist/reviews-P4M6BEJI.js.map +1 -0
- package/dist/{rtdn-LID2B7XZ.js → rtdn-YII4H6SD.js} +29 -21
- package/dist/rtdn-YII4H6SD.js.map +1 -0
- package/dist/{status-3HXBBXG6.js → status-32AJ7A6O.js} +13 -32
- package/dist/status-32AJ7A6O.js.map +1 -0
- package/dist/{subscriptions-LURZFPGJ.js → subscriptions-GTVJB4HX.js} +49 -12
- package/dist/subscriptions-GTVJB4HX.js.map +1 -0
- package/dist/system-apks-L6M7QYB3.js +111 -0
- package/dist/system-apks-L6M7QYB3.js.map +1 -0
- package/dist/{testers-6CQL4KQV.js → testers-QUWZHO6M.js} +25 -7
- package/dist/testers-QUWZHO6M.js.map +1 -0
- package/dist/{tracks-I4QZNZ3M.js → tracks-RH3RKVFB.js} +9 -3
- package/dist/{tracks-I4QZNZ3M.js.map → tracks-RH3RKVFB.js.map} +1 -1
- package/dist/{train-MDD2EBHS.js → train-D2NKUW3M.js} +3 -3
- package/dist/train-D2NKUW3M.js.map +1 -0
- package/dist/{update-QPPKXWI5.js → update-73YOR4GP.js} +8 -11
- package/dist/{update-QPPKXWI5.js.map → update-73YOR4GP.js.map} +1 -1
- package/dist/{users-UKG7VIQH.js → users-FOMAT7DY.js} +2 -2
- package/dist/{validate-QIYSA3N7.js → validate-RHWEZYD2.js} +6 -2
- package/dist/validate-RHWEZYD2.js.map +1 -0
- package/dist/{verify-UUQNQMPG.js → verify-H4ZUVHMZ.js} +5 -13
- package/dist/verify-H4ZUVHMZ.js.map +1 -0
- package/dist/{version-KUWLAZGT.js → version-RXLEX62V.js} +3 -3
- package/dist/{vitals-PJEQUUAK.js → vitals-PRBPNMVC.js} +26 -8
- package/dist/vitals-PRBPNMVC.js.map +1 -0
- package/package.json +5 -5
- package/dist/anomalies-V3AFS4LD.js.map +0 -1
- package/dist/auth-BA4FE2PO.js.map +0 -1
- package/dist/bundle-F7MUVC5J.js.map +0 -1
- package/dist/cache-XKPLZYEB.js.map +0 -1
- package/dist/chunk-A7VRCCNS.js.map +0 -1
- package/dist/chunk-BCBXQC7J.js.map +0 -1
- package/dist/chunk-GQVZ2VCT.js.map +0 -1
- package/dist/chunk-NQH4G7BI.js.map +0 -1
- package/dist/chunk-SLNJEAMK.js.map +0 -1
- package/dist/chunk-YFUBD2XB.js.map +0 -1
- package/dist/data-safety-AFMD6MYI.js.map +0 -1
- package/dist/device-tiers-AQAMUQXI.js.map +0 -1
- package/dist/diff-6EO4ID6W.js.map +0 -1
- package/dist/docs-GMFN6V4K.js.map +0 -1
- package/dist/doctor-GNI44SIP.js.map +0 -1
- package/dist/external-transactions-LCZALS3V.js.map +0 -1
- package/dist/feedback-XVTGMIBB.js.map +0 -1
- package/dist/games-ZSNGEI7A.js.map +0 -1
- package/dist/generated-apks-RX2IUWSF.js.map +0 -1
- package/dist/grants-EBPECI26.js.map +0 -1
- package/dist/iap-OUI5YYN4.js.map +0 -1
- package/dist/init-WSTQTJOD.js.map +0 -1
- package/dist/internal-sharing-ONNIWIAT.js.map +0 -1
- package/dist/listings-LNX6MQYN.js.map +0 -1
- package/dist/one-time-products-MGZTU7OM.js +0 -254
- package/dist/one-time-products-MGZTU7OM.js.map +0 -1
- package/dist/preflight-W3JAJ4GO.js.map +0 -1
- package/dist/pricing-JJZFICFL.js.map +0 -1
- package/dist/publish-P5KIGSLI.js.map +0 -1
- package/dist/purchases-UBFLNYZC.js.map +0 -1
- package/dist/quickstart-Z5Y3FYJU.js.map +0 -1
- package/dist/recovery-YE3Z7NIN.js.map +0 -1
- package/dist/releases-LUAHKIMY.js.map +0 -1
- package/dist/reviews-YCBBM656.js.map +0 -1
- package/dist/rtdn-LID2B7XZ.js.map +0 -1
- package/dist/status-3HXBBXG6.js.map +0 -1
- package/dist/subscriptions-LURZFPGJ.js.map +0 -1
- package/dist/testers-6CQL4KQV.js.map +0 -1
- package/dist/train-MDD2EBHS.js.map +0 -1
- package/dist/validate-QIYSA3N7.js.map +0 -1
- package/dist/verify-UUQNQMPG.js.map +0 -1
- package/dist/vitals-PJEQUUAK.js.map +0 -1
- /package/dist/{apps-4GP3FD7O.js.map → apps-WPUQOQLL.js.map} +0 -0
- /package/dist/{audit-VTWXTXC6.js.map → audit-M7GYID74.js.map} +0 -0
- /package/dist/{config-KT3PIL52.js.map → config-PUINDZON.js.map} +0 -0
- /package/dist/{prompt-GXC2JSLA.js.map → prompt-HJXNXXAR.js.map} +0 -0
- /package/dist/{users-UKG7VIQH.js.map → users-FOMAT7DY.js.map} +0 -0
- /package/dist/{version-KUWLAZGT.js.map → version-RXLEX62V.js.map} +0 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/subscriptions.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { Option } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\n\nimport type { Subscription } from \"@gpc-cli/api\";\nimport {\n listSubscriptions,\n getSubscription,\n createSubscription,\n updateSubscription,\n deleteSubscription,\n activateBasePlan,\n deactivateBasePlan,\n deleteBasePlan,\n migratePrices,\n listOffers,\n getOffer,\n createOffer,\n updateOffer,\n deleteOffer,\n activateOffer,\n deactivateOffer,\n diffSubscription,\n getSubscriptionAnalytics,\n formatOutput,\n sortResults,\n maybePaginate,\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\nexport function registerSubscriptionsCommands(program: Command): void {\n const subs = program.command(\"subscriptions\").description(\"Manage subscriptions and base plans\");\n\n subs\n .command(\"list\")\n .description(\"List subscriptions\")\n .addOption(new Option(\"--page-size <n>\", \"Results per page\").argParser(parseInt).hideHelp())\n .addOption(new Option(\"--page-token <token>\", \"Page token\").hideHelp())\n .option(\"--limit <n>\", \"Maximum total results\", parseInt)\n .option(\"--next-page <token>\", \"Resume from page token\")\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 listSubscriptions(client, packageName, {\n pageSize: options.pageSize,\n pageToken: options.pageToken,\n limit: options.limit,\n nextPage: options.nextPage,\n });\n if (options.sort) {\n result.subscriptions = sortResults(result.subscriptions, options.sort);\n }\n const subs = result.subscriptions || [];\n if (format !== \"json\") {\n if (subs.length === 0) {\n console.log(\"No subscriptions found.\");\n return;\n }\n const summary = subs.map((s: Subscription) => ({\n productId: s.productId,\n basePlans: s.basePlans?.length || 0,\n listings: s.listings ? Object.keys(s.listings).length : 0,\n firstBasePlanState: s.basePlans?.[0]?.state || \"-\",\n }));\n await maybePaginate(formatOutput(summary, format));\n } else {\n await maybePaginate(\n formatOutput(subs.length === 0 ? { subscriptions: [] } : result, format),\n );\n }\n });\n\n subs\n .command(\"get <product-id>\")\n .description(\"Get a subscription\")\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 getSubscription(client, packageName, productId);\n if (format !== \"json\") {\n const s = result as unknown as Record<string, unknown>;\n const basePlans = s[\"basePlans\"] as Array<Record<string, unknown>> | undefined;\n const listings = s[\"listings\"] as\n | Record<string, unknown>\n | Array<Record<string, unknown>>\n | undefined;\n const listingLanguages = listings\n ? Array.isArray(listings)\n ? listings.map((l) => l[\"languageCode\"] || l[\"language\"] || \"?\").join(\", \")\n : Object.keys(listings).join(\", \")\n : \"-\";\n const listingCount = listings\n ? Array.isArray(listings)\n ? listings.length\n : Object.keys(listings).length\n : 0;\n const summary = {\n productId: s[\"productId\"],\n basePlans: basePlans?.length || 0,\n basePlanIds: basePlans?.map((bp) => bp[\"basePlanId\"]).join(\", \") || \"-\",\n listings: listingCount,\n listingLanguages,\n taxCategory: (s[\"taxAndComplianceSettings\"] as Record<string, unknown>)?.[\n \"taxRateInfoByRegionCode\"\n ]\n ? \"configured\"\n : \"-\",\n };\n console.log(formatOutput(summary, format));\n } else {\n console.log(formatOutput(result, format));\n }\n });\n\n subs\n .command(\"create\")\n .description(\"Create a subscription from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with subscription data\")\n .option(\"--activate\", \"Activate all base plans after creation\")\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: \"subscriptions create\",\n action: \"create\",\n target: `subscription from ${options.file}`,\n details: options.activate ? { activate: true } : undefined,\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 createSubscription(client, packageName, data as any);\n\n if (options.activate && result.basePlans) {\n for (const bp of result.basePlans) {\n if (bp.state === \"DRAFT\") {\n await activateBasePlan(client, packageName, result.productId, bp.basePlanId);\n console.error(`Activated base plan: ${bp.basePlanId}`);\n }\n }\n // Re-fetch to get updated state\n const updated = await getSubscription(client, packageName, result.productId);\n console.log(formatOutput(updated, format));\n } else {\n console.log(formatOutput(result, format));\n }\n });\n\n subs\n .command(\"update <product-id>\")\n .description(\"Update a subscription from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with subscription 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: \"subscriptions update\",\n action: \"update\",\n target: productId,\n details: { file: options.file, updateMask: options.updateMask },\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 updateSubscription(\n client,\n packageName,\n productId,\n data as any,\n options.updateMask,\n );\n console.log(formatOutput(result, format));\n });\n\n subs\n .command(\"delete <product-id>\")\n .description(\"Delete a subscription\")\n .action(async (productId: string) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Delete subscription \"${productId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"subscriptions 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 deleteSubscription(client, packageName, productId);\n console.log(`Subscription ${productId} deleted.`);\n });\n\n // --- Base Plans ---\n const basePlans = subs.command(\"base-plans\").description(\"Manage base plans\");\n\n basePlans\n .command(\"activate <product-id> <base-plan-id>\")\n .description(\"Activate a base plan\")\n .action(async (productId: string, basePlanId: 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: \"subscriptions base-plans activate\",\n action: \"activate\",\n target: `${productId}/${basePlanId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await activateBasePlan(client, packageName, productId, basePlanId);\n console.log(formatOutput(result, format));\n });\n\n basePlans\n .command(\"deactivate <product-id> <base-plan-id>\")\n .description(\"Deactivate a base plan\")\n .action(async (productId: string, basePlanId: 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: \"subscriptions base-plans deactivate\",\n action: \"deactivate\",\n target: `${productId}/${basePlanId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await deactivateBasePlan(client, packageName, productId, basePlanId);\n console.log(formatOutput(result, format));\n });\n\n basePlans\n .command(\"delete <product-id> <base-plan-id>\")\n .description(\"Delete a base plan\")\n .action(async (productId: string, basePlanId: string) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(\n `Delete base plan \"${basePlanId}\" from subscription \"${productId}\"?`,\n program,\n );\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"subscriptions base-plans delete\",\n action: \"delete\",\n target: `${productId}/${basePlanId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n await deleteBasePlan(client, packageName, productId, basePlanId);\n console.log(`Base plan ${basePlanId} deleted.`);\n });\n\n basePlans\n .command(\"migrate-prices <product-id> <base-plan-id>\")\n .description(\"Migrate base plan prices\")\n .requiredOption(\"--file <path>\", \"JSON file with migration data\")\n .action(\n async (\n productId: string,\n basePlanId: string,\n options: { file: string; updateMask?: 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: \"subscriptions base-plans migrate-prices\",\n action: \"migrate prices for\",\n target: `${productId}/${basePlanId}`,\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 migratePrices(client, packageName, productId, basePlanId, data as any);\n console.log(formatOutput(result, format));\n },\n );\n\n // --- Offers ---\n const offers = subs.command(\"offers\").description(\"Manage subscription offers\");\n\n offers\n .command(\"list <product-id> <base-plan-id>\")\n .description(\"List offers for a base plan\")\n .action(async (productId: string, basePlanId: 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 listOffers(client, packageName, productId, basePlanId);\n const offers_list = (result as unknown as Record<string, unknown>)[\"subscriptionOffers\"] as\n | Array<Record<string, unknown>>\n | undefined;\n if (format !== \"json\") {\n if (!offers_list || offers_list.length === 0) {\n console.log(\"No offers found.\");\n return;\n }\n const summary = offers_list.map((o) => ({\n offerId: o[\"offerId\"],\n basePlanId: o[\"basePlanId\"],\n state: o[\"state\"] || \"-\",\n phases: (o[\"phases\"] as unknown[])?.length || 0,\n regionalConfigs: (o[\"regionalConfigs\"] as unknown[])?.length || 0,\n }));\n console.log(formatOutput(summary, format));\n } else {\n console.log(formatOutput(result, format));\n }\n });\n\n offers\n .command(\"get <product-id> <base-plan-id> <offer-id>\")\n .description(\"Get an offer\")\n .action(async (productId: string, basePlanId: string, offerId: 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 getOffer(client, packageName, productId, basePlanId, offerId);\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"create <product-id> <base-plan-id>\")\n .description(\"Create an offer from JSON file\")\n .requiredOption(\"--file <path>\", \"JSON file with offer data\")\n .action(\n async (\n productId: string,\n basePlanId: string,\n options: { file: string; updateMask?: 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: \"subscriptions offers create\",\n action: \"create offer for\",\n target: `${productId}/${basePlanId}`,\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 createOffer(client, packageName, productId, basePlanId, data as any);\n console.log(formatOutput(result, format));\n },\n );\n\n offers\n .command(\"update <product-id> <base-plan-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 .action(\n async (\n productId: string,\n basePlanId: string,\n offerId: string,\n options: { file: string; updateMask?: 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: \"subscriptions offers update\",\n action: \"update offer\",\n target: `${productId}/${basePlanId}/${offerId}`,\n details: { file: options.file, updateMask: options.updateMask },\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 updateOffer(\n client,\n packageName,\n productId,\n basePlanId,\n offerId,\n data as any,\n options.updateMask,\n );\n console.log(formatOutput(result, format));\n },\n );\n\n offers\n .command(\"delete <product-id> <base-plan-id> <offer-id>\")\n .description(\"Delete an offer\")\n .action(async (productId: string, basePlanId: string, offerId: string) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Delete offer \"${offerId}\"?`, program);\n\n if (isDryRun(program)) {\n const format = getOutputFormat(program, config);\n printDryRun(\n {\n command: \"subscriptions offers delete\",\n action: \"delete offer\",\n target: `${productId}/${basePlanId}/${offerId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n await deleteOffer(client, packageName, productId, basePlanId, offerId);\n console.log(`Offer ${offerId} deleted.`);\n });\n\n offers\n .command(\"activate <product-id> <base-plan-id> <offer-id>\")\n .description(\"Activate an offer\")\n .action(async (productId: string, basePlanId: string, offerId: 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: \"subscriptions offers activate\",\n action: \"activate offer\",\n target: `${productId}/${basePlanId}/${offerId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await activateOffer(client, packageName, productId, basePlanId, offerId);\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"deactivate <product-id> <base-plan-id> <offer-id>\")\n .description(\"Deactivate an offer\")\n .action(async (productId: string, basePlanId: string, offerId: 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: \"subscriptions offers deactivate\",\n action: \"deactivate offer\",\n target: `${productId}/${basePlanId}/${offerId}`,\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await deactivateOffer(client, packageName, productId, basePlanId, offerId);\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-get <product-id> <base-plan-id>\")\n .description(\"Batch get multiple offers (max 100)\")\n .requiredOption(\"--ids <offer-ids>\", \"Comma-separated offer IDs\")\n .action(async (productId: string, basePlanId: string, options: { ids: 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 offerIds = options.ids.split(\",\").map((id) => id.trim());\n const result = await client.subscriptions.batchGetOffers(\n packageName,\n productId,\n basePlanId,\n offerIds,\n );\n console.log(formatOutput(result, format));\n });\n\n offers\n .command(\"batch-update-states <product-id> <base-plan-id>\")\n .description(\"Batch activate or deactivate multiple offers (max 100)\")\n .requiredOption(\"--file <path>\", \"JSON file with state transition requests\")\n .action(async (productId: string, basePlanId: 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: \"subscriptions offers batch-update-states\",\n action: \"batch update offer states\",\n target: `${productId}/${basePlanId}`,\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 readJsonFile(options.file);\n const result = await client.subscriptions.batchUpdateOfferStates(\n packageName,\n productId,\n basePlanId,\n requests as any,\n );\n console.log(formatOutput(result, format));\n });\n\n // --- Diff ---\n subs\n .command(\"diff <product-id>\")\n .description(\"Compare local JSON file against remote subscription\")\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 Subscription;\n const diffs = await diffSubscription(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 subs\n .command(\"analytics\")\n .description(\"Subscription catalog analytics: active plans, offer counts, state breakdown\")\n .action(async () => {\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 getSubscriptionAnalytics(client, packageName);\n\n if (format === \"json\") {\n console.log(formatOutput(result, format));\n return;\n }\n\n console.log(`\\nSubscription Analytics — ${packageName}`);\n console.log(`${\"─\".repeat(50)}`);\n console.log(`Total subscriptions: ${result.totalSubscriptions}`);\n console.log(`Active subscriptions: ${result.activeCount}`);\n console.log(`Active base plans: ${result.activeBasePlans}`);\n console.log(`Draft base plans: ${result.trialBasePlans}`);\n console.log(`Inactive base plans: ${result.pausedBasePlans}`);\n console.log(`Total offers: ${result.offerCount}`);\n\n if (result.byProductId.length > 0) {\n console.log(`\\nBy product:`);\n console.log(formatOutput(result.byProductId, format));\n }\n });\n}\n"],"mappings":"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","names":["subs","basePlans"]}
|
|
@@ -0,0 +1,111 @@
|
|
|
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/system-apks.ts
|
|
21
|
+
import { loadConfig } from "@gpc-cli/config";
|
|
22
|
+
import { formatOutput } from "@gpc-cli/core";
|
|
23
|
+
import { writeFile, access } from "fs/promises";
|
|
24
|
+
import { resolve } from "path";
|
|
25
|
+
function parsePositiveInt(str, label) {
|
|
26
|
+
const n = parseInt(str, 10);
|
|
27
|
+
if (isNaN(n) || n <= 0) {
|
|
28
|
+
const err = new Error(`${label} must be a positive number`);
|
|
29
|
+
Object.assign(err, { code: "USAGE_ERROR", exitCode: 2 });
|
|
30
|
+
throw err;
|
|
31
|
+
}
|
|
32
|
+
return n;
|
|
33
|
+
}
|
|
34
|
+
function registerSystemApksCommands(program) {
|
|
35
|
+
const cmd = program.command("system-apks").description("Manage system APKs for OEM pre-installs");
|
|
36
|
+
cmd.command("list <version-code>").description("List system APK variants for a version code").action(async (versionCodeStr) => {
|
|
37
|
+
const config = await loadConfig();
|
|
38
|
+
const packageName = resolvePackageName(program.opts()["app"], config);
|
|
39
|
+
const client = await getClient(config);
|
|
40
|
+
const format = getOutputFormat(program, config);
|
|
41
|
+
const versionCode = parsePositiveInt(versionCodeStr, "version-code");
|
|
42
|
+
const result = await client.systemApks.list(packageName, versionCode);
|
|
43
|
+
const variants = result.variants || [];
|
|
44
|
+
if (variants.length === 0) {
|
|
45
|
+
console.log("No system APK variants found.");
|
|
46
|
+
} else {
|
|
47
|
+
console.log(formatOutput(variants, format));
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
cmd.command("get <version-code> <variant-id>").description("Get a system APK variant").action(async (versionCodeStr, variantIdStr) => {
|
|
51
|
+
const config = await loadConfig();
|
|
52
|
+
const packageName = resolvePackageName(program.opts()["app"], config);
|
|
53
|
+
const client = await getClient(config);
|
|
54
|
+
const format = getOutputFormat(program, config);
|
|
55
|
+
const versionCode = parsePositiveInt(versionCodeStr, "version-code");
|
|
56
|
+
const variantId = parsePositiveInt(variantIdStr, "variant-id");
|
|
57
|
+
const result = await client.systemApks.get(packageName, versionCode, variantId);
|
|
58
|
+
console.log(formatOutput(result, format));
|
|
59
|
+
});
|
|
60
|
+
cmd.command("create <version-code>").description("Create a system APK variant from a device spec JSON file").requiredOption(
|
|
61
|
+
"--file <path>",
|
|
62
|
+
"JSON file with device spec (supportedAbis, supportedLocales, screenDensity)"
|
|
63
|
+
).action(async (versionCodeStr, options) => {
|
|
64
|
+
const config = await loadConfig();
|
|
65
|
+
const packageName = resolvePackageName(program.opts()["app"], config);
|
|
66
|
+
const format = getOutputFormat(program, config);
|
|
67
|
+
const versionCode = parsePositiveInt(versionCodeStr, "version-code");
|
|
68
|
+
if (isDryRun(program)) {
|
|
69
|
+
printDryRun(
|
|
70
|
+
{
|
|
71
|
+
command: "system-apks create",
|
|
72
|
+
action: "create system APK variant",
|
|
73
|
+
target: `v${versionCode}`,
|
|
74
|
+
details: { file: options.file }
|
|
75
|
+
},
|
|
76
|
+
format,
|
|
77
|
+
formatOutput
|
|
78
|
+
);
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
const client = await getClient(config);
|
|
82
|
+
const spec = await readJsonFile(options.file);
|
|
83
|
+
const result = await client.systemApks.create(packageName, versionCode, spec);
|
|
84
|
+
console.log(formatOutput(result, format));
|
|
85
|
+
});
|
|
86
|
+
cmd.command("download <version-code> <variant-id>").description("Download a system APK variant").requiredOption("--output <path>", "Output file path").action(async (versionCodeStr, variantIdStr, options) => {
|
|
87
|
+
const config = await loadConfig();
|
|
88
|
+
const packageName = resolvePackageName(program.opts()["app"], config);
|
|
89
|
+
const versionCode = parsePositiveInt(versionCodeStr, "version-code");
|
|
90
|
+
const variantId = parsePositiveInt(variantIdStr, "variant-id");
|
|
91
|
+
const outputPath = resolve(options.output);
|
|
92
|
+
if (outputPath.includes("\0")) {
|
|
93
|
+
const err = new Error("Output path contains null bytes");
|
|
94
|
+
Object.assign(err, { code: "USAGE_ERROR", exitCode: 2 });
|
|
95
|
+
throw err;
|
|
96
|
+
}
|
|
97
|
+
try {
|
|
98
|
+
await access(outputPath);
|
|
99
|
+
await requireConfirm(`File "${outputPath}" already exists. Overwrite?`, program);
|
|
100
|
+
} catch {
|
|
101
|
+
}
|
|
102
|
+
const client = await getClient(config);
|
|
103
|
+
const data = await client.systemApks.download(packageName, versionCode, variantId);
|
|
104
|
+
await writeFile(outputPath, Buffer.from(data));
|
|
105
|
+
console.log(`Downloaded to ${outputPath}`);
|
|
106
|
+
});
|
|
107
|
+
}
|
|
108
|
+
export {
|
|
109
|
+
registerSystemApksCommands
|
|
110
|
+
};
|
|
111
|
+
//# sourceMappingURL=system-apks-L6M7QYB3.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/system-apks.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { formatOutput } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { readJsonFile } from \"../json.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { requireConfirm } from \"../prompt.js\";\nimport { writeFile, access } from \"node:fs/promises\";\nimport { resolve } from \"node:path\";\n\nfunction parsePositiveInt(str: string, label: string): number {\n const n = parseInt(str, 10);\n if (isNaN(n) || n <= 0) {\n const err = new Error(`${label} must be a positive number`);\n Object.assign(err, { code: \"USAGE_ERROR\", exitCode: 2 });\n throw err;\n }\n return n;\n}\n\nexport function registerSystemApksCommands(program: Command): void {\n const cmd = program.command(\"system-apks\").description(\"Manage system APKs for OEM pre-installs\");\n\n cmd\n .command(\"list <version-code>\")\n .description(\"List system APK variants for a version code\")\n .action(async (versionCodeStr: 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 const versionCode = parsePositiveInt(versionCodeStr, \"version-code\");\n\n const result = await client.systemApks.list(packageName, versionCode);\n const variants = result.variants || [];\n if (variants.length === 0) {\n console.log(\"No system APK variants found.\");\n } else {\n console.log(formatOutput(variants, format));\n }\n });\n\n cmd\n .command(\"get <version-code> <variant-id>\")\n .description(\"Get a system APK variant\")\n .action(async (versionCodeStr: string, variantIdStr: 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 const versionCode = parsePositiveInt(versionCodeStr, \"version-code\");\n const variantId = parsePositiveInt(variantIdStr, \"variant-id\");\n\n const result = await client.systemApks.get(packageName, versionCode, variantId);\n console.log(formatOutput(result, format));\n });\n\n cmd\n .command(\"create <version-code>\")\n .description(\"Create a system APK variant from a device spec JSON file\")\n .requiredOption(\n \"--file <path>\",\n \"JSON file with device spec (supportedAbis, supportedLocales, screenDensity)\",\n )\n .action(async (versionCodeStr: string, options: { file: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const versionCode = parsePositiveInt(versionCodeStr, \"version-code\");\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"system-apks create\",\n action: \"create system APK variant\",\n target: `v${versionCode}`,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n const spec = await readJsonFile(options.file);\n const result = await client.systemApks.create(packageName, versionCode, spec as any);\n console.log(formatOutput(result, format));\n });\n\n cmd\n .command(\"download <version-code> <variant-id>\")\n .description(\"Download a system APK variant\")\n .requiredOption(\"--output <path>\", \"Output file path\")\n .action(async (versionCodeStr: string, variantIdStr: string, options: { output: string }) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const versionCode = parsePositiveInt(versionCodeStr, \"version-code\");\n const variantId = parsePositiveInt(variantIdStr, \"variant-id\");\n\n const outputPath = resolve(options.output);\n if (outputPath.includes(\"\\0\")) {\n const err = new Error(\"Output path contains null bytes\");\n Object.assign(err, { code: \"USAGE_ERROR\", exitCode: 2 });\n throw err;\n }\n try {\n await access(outputPath);\n // File exists -- require confirmation\n await requireConfirm(`File \"${outputPath}\" already exists. Overwrite?`, program);\n } catch {\n // File does not exist -- safe to write\n }\n\n const client = await getClient(config);\n const data = await client.systemApks.download(packageName, versionCode, variantId);\n await writeFile(outputPath, Buffer.from(data));\n console.log(`Downloaded to ${outputPath}`);\n });\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
buildCommitOptions
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-WSLFHX5X.js";
|
|
5
5
|
import {
|
|
6
6
|
getClient,
|
|
7
7
|
resolvePackageName
|
|
8
|
-
} from "./chunk-
|
|
8
|
+
} from "./chunk-JDRY7HK5.js";
|
|
9
9
|
import {
|
|
10
10
|
isDryRun,
|
|
11
11
|
printDryRun
|
|
@@ -17,7 +17,7 @@ import {
|
|
|
17
17
|
isInteractive,
|
|
18
18
|
requireConfirm,
|
|
19
19
|
requireOption
|
|
20
|
-
} from "./chunk-
|
|
20
|
+
} from "./chunk-RZQSEDKI.js";
|
|
21
21
|
|
|
22
22
|
// src/commands/testers.ts
|
|
23
23
|
import { loadConfig } from "@gpc-cli/config";
|
|
@@ -92,7 +92,13 @@ function registerTestersCommands(program) {
|
|
|
92
92
|
return;
|
|
93
93
|
}
|
|
94
94
|
const client = await getClient(config);
|
|
95
|
-
const result = await addTesters(
|
|
95
|
+
const result = await addTesters(
|
|
96
|
+
client,
|
|
97
|
+
packageName,
|
|
98
|
+
options.track,
|
|
99
|
+
emails,
|
|
100
|
+
buildCommitOptions(options)
|
|
101
|
+
);
|
|
96
102
|
console.log(formatOutput(result, format));
|
|
97
103
|
});
|
|
98
104
|
testers.command("remove <emails...>").description("Remove testers (Google Group emails) from a track").option("--track <track>", "Track name (e.g., internal, alpha, beta)").option("--changes-not-sent-for-review", "Commit changes without sending for review").option("--error-if-in-review", "Fail if changes are already in review").action(async (emails, options) => {
|
|
@@ -124,7 +130,13 @@ function registerTestersCommands(program) {
|
|
|
124
130
|
return;
|
|
125
131
|
}
|
|
126
132
|
const client = await getClient(config);
|
|
127
|
-
const result = await removeTesters(
|
|
133
|
+
const result = await removeTesters(
|
|
134
|
+
client,
|
|
135
|
+
packageName,
|
|
136
|
+
options.track,
|
|
137
|
+
emails,
|
|
138
|
+
buildCommitOptions(options)
|
|
139
|
+
);
|
|
128
140
|
console.log(formatOutput(result, format));
|
|
129
141
|
});
|
|
130
142
|
testers.command("import").description("Import testers from a CSV file").option("--track <track>", "Track name (e.g., internal, alpha, beta)").option("--file <path>", "CSV file with email addresses").option("--changes-not-sent-for-review", "Commit changes without sending for review").option("--error-if-in-review", "Fail if changes are already in review").action(async (options) => {
|
|
@@ -163,11 +175,17 @@ function registerTestersCommands(program) {
|
|
|
163
175
|
return;
|
|
164
176
|
}
|
|
165
177
|
const client = await getClient(config);
|
|
166
|
-
const result = await importTestersFromCsv(
|
|
178
|
+
const result = await importTestersFromCsv(
|
|
179
|
+
client,
|
|
180
|
+
packageName,
|
|
181
|
+
options.track,
|
|
182
|
+
options.file,
|
|
183
|
+
buildCommitOptions(options)
|
|
184
|
+
);
|
|
167
185
|
console.log(formatOutput({ added: result.added, testers: result.testers }, format));
|
|
168
186
|
});
|
|
169
187
|
}
|
|
170
188
|
export {
|
|
171
189
|
registerTestersCommands
|
|
172
190
|
};
|
|
173
|
-
//# sourceMappingURL=testers-
|
|
191
|
+
//# sourceMappingURL=testers-QUWZHO6M.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/testers.ts"],"sourcesContent":["import { resolvePackageName, getClient } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport {\n listTesters,\n addTesters,\n removeTesters,\n importTestersFromCsv,\n formatOutput,\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 { buildCommitOptions } from \"../commit-options.js\";\n\nexport function registerTestersCommands(program: Command): void {\n const testers = program.command(\"testers\").description(\"Manage testers and tester groups\");\n\n testers\n .command(\"list\")\n .description(\"List testers for a track\")\n .option(\"--track <track>\", \"Track name (e.g., internal, alpha, beta)\")\n .option(\"--sort <field>\", \"Sort by field (prefix with - for descending, e.g., email or -email)\")\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.track = await requireOption(\n \"track\",\n options.track,\n {\n message: \"Track:\",\n choices: [\"internal\", \"alpha\", \"beta\"],\n },\n interactive,\n );\n\n const client = await getClient(config);\n const format = getOutputFormat(program, config);\n\n const result = await listTesters(client, packageName, options.track);\n if (options.sort && result.googleGroups) {\n const descending = options.sort.startsWith(\"-\");\n result.googleGroups = [...result.googleGroups].sort((a: string, b: string) =>\n descending ? b.localeCompare(a) : a.localeCompare(b),\n );\n }\n if (format !== \"json\") {\n const groups = (result.googleGroups || []) as string[];\n const rows = groups.map((g: string) => ({ googleGroup: g }));\n if (rows.length > 0) {\n console.log(formatOutput(rows, format));\n } else {\n console.log(\"No testers found.\");\n }\n } else {\n console.log(formatOutput(result, format));\n }\n });\n\n testers\n .command(\"add <emails...>\")\n .description(\"Add testers (Google Group emails) to a track\")\n .option(\"--track <track>\", \"Track name (e.g., internal, alpha, beta)\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (emails: string[], options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const interactive = isInteractive(program);\n\n options.track = await requireOption(\n \"track\",\n options.track,\n {\n message: \"Track:\",\n choices: [\"internal\", \"alpha\", \"beta\"],\n },\n interactive,\n );\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"testers add\",\n action: \"add testers to\",\n target: options.track,\n details: { emails },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await addTesters(\n client,\n packageName,\n options.track,\n emails,\n buildCommitOptions(options),\n );\n console.log(formatOutput(result, format));\n });\n\n testers\n .command(\"remove <emails...>\")\n .description(\"Remove testers (Google Group emails) from a track\")\n .option(\"--track <track>\", \"Track name (e.g., internal, alpha, beta)\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (emails: string[], options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const interactive = isInteractive(program);\n\n options.track = await requireOption(\n \"track\",\n options.track,\n {\n message: \"Track:\",\n choices: [\"internal\", \"alpha\", \"beta\"],\n },\n interactive,\n );\n\n await requireConfirm(`Remove ${emails.length} tester(s) from ${options.track}?`, program);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"testers remove\",\n action: \"remove testers from\",\n target: options.track,\n details: { emails },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await removeTesters(\n client,\n packageName,\n options.track,\n emails,\n buildCommitOptions(options),\n );\n console.log(formatOutput(result, format));\n });\n\n testers\n .command(\"import\")\n .description(\"Import testers from a CSV file\")\n .option(\"--track <track>\", \"Track name (e.g., internal, alpha, beta)\")\n .option(\"--file <path>\", \"CSV file with email addresses\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (options) => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const interactive = isInteractive(program);\n\n options.track = await requireOption(\n \"track\",\n options.track,\n {\n message: \"Track:\",\n choices: [\"internal\", \"alpha\", \"beta\"],\n },\n interactive,\n );\n\n options.file = await requireOption(\n \"file\",\n options.file,\n {\n message: \"CSV file path:\",\n },\n interactive,\n );\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"testers import\",\n action: \"import testers to\",\n target: options.track,\n details: { file: options.file },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const client = await getClient(config);\n\n const result = await importTestersFromCsv(\n client,\n packageName,\n options.track,\n options.file,\n buildCommitOptions(options),\n );\n console.log(formatOutput({ added: result.added, testers: result.testers }, format));\n });\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;AAEA,SAAS,kBAAkB;AAC3B;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAMA,SAAS,wBAAwB,SAAwB;AAC9D,QAAM,UAAU,QAAQ,QAAQ,SAAS,EAAE,YAAY,kCAAkC;AAEzF,UACG,QAAQ,MAAM,EACd,YAAY,0BAA0B,EACtC,OAAO,mBAAmB,0CAA0C,EACpE,OAAO,kBAAkB,qEAAqE,EAC9F,OAAO,eAAe,2BAA2B,EACjD,OAAO,uBAAuB,gCAAgC,EAC9D,OAAO,OAAO,YAAY;AACzB,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,cAAc,mBAAmB,QAAQ,KAAK,EAAE,KAAK,GAAG,MAAM;AACpE,UAAM,cAAc,cAAc,OAAO;AAEzC,YAAQ,QAAQ,MAAM;AAAA,MACpB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,QACT,SAAS,CAAC,YAAY,SAAS,MAAM;AAAA,MACvC;AAAA,MACA;AAAA,IACF;AAEA,UAAM,SAAS,MAAM,UAAU,MAAM;AACrC,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAE9C,UAAM,SAAS,MAAM,YAAY,QAAQ,aAAa,QAAQ,KAAK;AACnE,QAAI,QAAQ,QAAQ,OAAO,cAAc;AACvC,YAAM,aAAa,QAAQ,KAAK,WAAW,GAAG;AAC9C,aAAO,eAAe,CAAC,GAAG,OAAO,YAAY,EAAE;AAAA,QAAK,CAAC,GAAW,MAC9D,aAAa,EAAE,cAAc,CAAC,IAAI,EAAE,cAAc,CAAC;AAAA,MACrD;AAAA,IACF;AACA,QAAI,WAAW,QAAQ;AACrB,YAAM,SAAU,OAAO,gBAAgB,CAAC;AACxC,YAAM,OAAO,OAAO,IAAI,CAAC,OAAe,EAAE,aAAa,EAAE,EAAE;AAC3D,UAAI,KAAK,SAAS,GAAG;AACnB,gBAAQ,IAAI,aAAa,MAAM,MAAM,CAAC;AAAA,MACxC,OAAO;AACL,gBAAQ,IAAI,mBAAmB;AAAA,MACjC;AAAA,IACF,OAAO;AACL,cAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,IAC1C;AAAA,EACF,CAAC;AAEH,UACG,QAAQ,iBAAiB,EACzB,YAAY,8CAA8C,EAC1D,OAAO,mBAAmB,0CAA0C,EACpE,OAAO,iCAAiC,2CAA2C,EACnF,OAAO,wBAAwB,uCAAuC,EACtE,OAAO,OAAO,QAAkB,YAAY;AAC3C,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,cAAc,mBAAmB,QAAQ,KAAK,EAAE,KAAK,GAAG,MAAM;AACpE,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAC9C,UAAM,cAAc,cAAc,OAAO;AAEzC,YAAQ,QAAQ,MAAM;AAAA,MACpB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,QACT,SAAS,CAAC,YAAY,SAAS,MAAM;AAAA,MACvC;AAAA,MACA;AAAA,IACF;AAEA,QAAI,SAAS,OAAO,GAAG;AACrB;AAAA,QACE;AAAA,UACE,SAAS;AAAA,UACT,QAAQ;AAAA,UACR,QAAQ,QAAQ;AAAA,UAChB,SAAS,EAAE,OAAO;AAAA,QACpB;AAAA,QACA;AAAA,QACA;AAAA,MACF;AACA;AAAA,IACF;AAEA,UAAM,SAAS,MAAM,UAAU,MAAM;AAErC,UAAM,SAAS,MAAM;AAAA,MACnB;AAAA,MACA;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,MACA,mBAAmB,OAAO;AAAA,IAC5B;AACA,YAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,EAC1C,CAAC;AAEH,UACG,QAAQ,oBAAoB,EAC5B,YAAY,mDAAmD,EAC/D,OAAO,mBAAmB,0CAA0C,EACpE,OAAO,iCAAiC,2CAA2C,EACnF,OAAO,wBAAwB,uCAAuC,EACtE,OAAO,OAAO,QAAkB,YAAY;AAC3C,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,cAAc,mBAAmB,QAAQ,KAAK,EAAE,KAAK,GAAG,MAAM;AACpE,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAC9C,UAAM,cAAc,cAAc,OAAO;AAEzC,YAAQ,QAAQ,MAAM;AAAA,MACpB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,QACT,SAAS,CAAC,YAAY,SAAS,MAAM;AAAA,MACvC;AAAA,MACA;AAAA,IACF;AAEA,UAAM,eAAe,UAAU,OAAO,MAAM,mBAAmB,QAAQ,KAAK,KAAK,OAAO;AAExF,QAAI,SAAS,OAAO,GAAG;AACrB;AAAA,QACE;AAAA,UACE,SAAS;AAAA,UACT,QAAQ;AAAA,UACR,QAAQ,QAAQ;AAAA,UAChB,SAAS,EAAE,OAAO;AAAA,QACpB;AAAA,QACA;AAAA,QACA;AAAA,MACF;AACA;AAAA,IACF;AAEA,UAAM,SAAS,MAAM,UAAU,MAAM;AAErC,UAAM,SAAS,MAAM;AAAA,MACnB;AAAA,MACA;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,MACA,mBAAmB,OAAO;AAAA,IAC5B;AACA,YAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,EAC1C,CAAC;AAEH,UACG,QAAQ,QAAQ,EAChB,YAAY,gCAAgC,EAC5C,OAAO,mBAAmB,0CAA0C,EACpE,OAAO,iBAAiB,+BAA+B,EACvD,OAAO,iCAAiC,2CAA2C,EACnF,OAAO,wBAAwB,uCAAuC,EACtE,OAAO,OAAO,YAAY;AACzB,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,cAAc,mBAAmB,QAAQ,KAAK,EAAE,KAAK,GAAG,MAAM;AACpE,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAC9C,UAAM,cAAc,cAAc,OAAO;AAEzC,YAAQ,QAAQ,MAAM;AAAA,MACpB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,QACT,SAAS,CAAC,YAAY,SAAS,MAAM;AAAA,MACvC;AAAA,MACA;AAAA,IACF;AAEA,YAAQ,OAAO,MAAM;AAAA,MACnB;AAAA,MACA,QAAQ;AAAA,MACR;AAAA,QACE,SAAS;AAAA,MACX;AAAA,MACA;AAAA,IACF;AAEA,QAAI,SAAS,OAAO,GAAG;AACrB;AAAA,QACE;AAAA,UACE,SAAS;AAAA,UACT,QAAQ;AAAA,UACR,QAAQ,QAAQ;AAAA,UAChB,SAAS,EAAE,MAAM,QAAQ,KAAK;AAAA,QAChC;AAAA,QACA;AAAA,QACA;AAAA,MACF;AACA;AAAA,IACF;AAEA,UAAM,SAAS,MAAM,UAAU,MAAM;AAErC,UAAM,SAAS,MAAM;AAAA,MACnB;AAAA,MACA;AAAA,MACA,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,mBAAmB,OAAO;AAAA,IAC5B;AACA,YAAQ,IAAI,aAAa,EAAE,OAAO,OAAO,OAAO,SAAS,OAAO,QAAQ,GAAG,MAAM,CAAC;AAAA,EACpF,CAAC;AACL;","names":[]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
buildCommitOptions
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-WSLFHX5X.js";
|
|
5
5
|
import {
|
|
6
6
|
isDryRun,
|
|
7
7
|
printDryRun
|
|
@@ -91,11 +91,17 @@ function registerTracksCommands(program) {
|
|
|
91
91
|
const trackConfig = JSON.parse(raw);
|
|
92
92
|
const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });
|
|
93
93
|
const client = createApiClient({ auth });
|
|
94
|
-
const track = await updateTrackConfig(
|
|
94
|
+
const track = await updateTrackConfig(
|
|
95
|
+
client,
|
|
96
|
+
packageName,
|
|
97
|
+
name,
|
|
98
|
+
trackConfig,
|
|
99
|
+
buildCommitOptions(options)
|
|
100
|
+
);
|
|
95
101
|
console.log(formatOutput(track, format));
|
|
96
102
|
});
|
|
97
103
|
}
|
|
98
104
|
export {
|
|
99
105
|
registerTracksCommands
|
|
100
106
|
};
|
|
101
|
-
//# sourceMappingURL=tracks-
|
|
107
|
+
//# sourceMappingURL=tracks-RH3RKVFB.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/commands/tracks.ts"],"sourcesContent":["import { readFile } from \"node:fs/promises\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { resolveAuth } from \"@gpc-cli/auth\";\nimport { createApiClient } from \"@gpc-cli/api\";\nimport { listTracks, createTrack, updateTrackConfig, GpcError } from \"@gpc-cli/core\";\nimport { formatOutput, maybePaginate } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { buildCommitOptions } from \"../commit-options.js\";\n\nexport function registerTracksCommands(program: Command): void {\n const tracks = program.command(\"tracks\").description(\"Manage tracks\");\n\n tracks\n .command(\"list\")\n .description(\"List all tracks\")\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 = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const trackList = await listTracks(client, packageName);\n const format = getOutputFormat(program, config);\n\n const summary = trackList.map((t) => ({\n track: t.track,\n releases: t.releases?.length || 0,\n latestStatus: t.releases?.[0]?.status || \"none\",\n latestVersion: t.releases?.[0]?.versionCodes?.[0] || \"-\",\n }));\n\n await maybePaginate(formatOutput(summary, format));\n });\n\n tracks\n .command(\"create <name>\")\n .description(\"Create a custom track\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (name: string, options: Record<string, unknown>) => {\n const config = await loadConfig();\n const packageName = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n { command: \"tracks create\", action: \"create custom track\", target: name },\n format,\n formatOutput,\n );\n return;\n }\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const track = await createTrack(client, packageName, name, buildCommitOptions(options));\n console.log(formatOutput(track, format));\n });\n\n tracks\n .command(\"update <name>\")\n .description(\"Update track configuration from a JSON file\")\n .requiredOption(\"--file <path>\", \"Path to JSON config file\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (name: string, options: { file: string } & Record<string, unknown>) => {\n const config = await loadConfig();\n const packageName = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n { command: \"tracks update\", action: \"update track config from\", target: options.file },\n format,\n formatOutput,\n );\n return;\n }\n\n const raw = await readFile(options.file, \"utf-8\");\n const trackConfig = JSON.parse(raw) as Record<string, unknown>;\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const track = await updateTrackConfig(client
|
|
1
|
+
{"version":3,"sources":["../src/commands/tracks.ts"],"sourcesContent":["import { readFile } from \"node:fs/promises\";\nimport type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { resolveAuth } from \"@gpc-cli/auth\";\nimport { createApiClient } from \"@gpc-cli/api\";\nimport { listTracks, createTrack, updateTrackConfig, GpcError } from \"@gpc-cli/core\";\nimport { formatOutput, maybePaginate } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { buildCommitOptions } from \"../commit-options.js\";\n\nexport function registerTracksCommands(program: Command): void {\n const tracks = program.command(\"tracks\").description(\"Manage tracks\");\n\n tracks\n .command(\"list\")\n .description(\"List all tracks\")\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 = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const trackList = await listTracks(client, packageName);\n const format = getOutputFormat(program, config);\n\n const summary = trackList.map((t) => ({\n track: t.track,\n releases: t.releases?.length || 0,\n latestStatus: t.releases?.[0]?.status || \"none\",\n latestVersion: t.releases?.[0]?.versionCodes?.[0] || \"-\",\n }));\n\n await maybePaginate(formatOutput(summary, format));\n });\n\n tracks\n .command(\"create <name>\")\n .description(\"Create a custom track\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (name: string, options: Record<string, unknown>) => {\n const config = await loadConfig();\n const packageName = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n { command: \"tracks create\", action: \"create custom track\", target: name },\n format,\n formatOutput,\n );\n return;\n }\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const track = await createTrack(client, packageName, name, buildCommitOptions(options));\n console.log(formatOutput(track, format));\n });\n\n tracks\n .command(\"update <name>\")\n .description(\"Update track configuration from a JSON file\")\n .requiredOption(\"--file <path>\", \"Path to JSON config file\")\n .option(\"--changes-not-sent-for-review\", \"Commit changes without sending for review\")\n .option(\"--error-if-in-review\", \"Fail if changes are already in review\")\n .action(async (name: string, options: { file: string } & Record<string, unknown>) => {\n const config = await loadConfig();\n const packageName = program.opts()[\"app\"] || config.app;\n if (!packageName) {\n throw new GpcError(\n \"No package name. Use --app <package> or gpc config set app <package>\",\n \"MISSING_PACKAGE\",\n 2,\n \"gpc config set app com.example.app\",\n );\n }\n\n const format = getOutputFormat(program, config);\n\n if (isDryRun(program)) {\n printDryRun(\n { command: \"tracks update\", action: \"update track config from\", target: options.file },\n format,\n formatOutput,\n );\n return;\n }\n\n const raw = await readFile(options.file, \"utf-8\");\n const trackConfig = JSON.parse(raw) as Record<string, unknown>;\n\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n const client = createApiClient({ auth });\n const track = await updateTrackConfig(\n client,\n packageName,\n name,\n trackConfig,\n buildCommitOptions(options),\n );\n console.log(formatOutput(track, format));\n });\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
resolvePackageName
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-JDRY7HK5.js";
|
|
5
5
|
import {
|
|
6
6
|
isDryRun,
|
|
7
7
|
printDryRun
|
|
@@ -11,7 +11,7 @@ import {
|
|
|
11
11
|
} from "./chunk-ELXAK7GI.js";
|
|
12
12
|
import {
|
|
13
13
|
requireConfirm
|
|
14
|
-
} from "./chunk-
|
|
14
|
+
} from "./chunk-RZQSEDKI.js";
|
|
15
15
|
|
|
16
16
|
// src/commands/train.ts
|
|
17
17
|
import { loadConfig } from "@gpc-cli/config";
|
|
@@ -122,4 +122,4 @@ function registerTrainCommands(program) {
|
|
|
122
122
|
export {
|
|
123
123
|
registerTrainCommands
|
|
124
124
|
};
|
|
125
|
-
//# sourceMappingURL=train-
|
|
125
|
+
//# sourceMappingURL=train-D2NKUW3M.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/train.ts"],"sourcesContent":["import { resolvePackageName } from \"../resolve.js\";\nimport type { Command } from \"commander\";\nimport type { GpcConfig } from \"@gpc-cli/config\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { resolveAuth } from \"@gpc-cli/auth\";\nimport { createApiClient, createReportingClient } from \"@gpc-cli/api\";\nimport {\n startTrain,\n getTrainStatus,\n pauseTrain,\n abortTrain,\n advanceTrain,\n formatOutput,\n GpcError,\n} from \"@gpc-cli/core\";\nimport type { TrainConfig } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { isDryRun, printDryRun } from \"../dry-run.js\";\nimport { requireConfirm } from \"../prompt.js\";\n\nasync function getClients(config: GpcConfig) {\n const auth = await resolveAuth({ serviceAccountPath: config.auth?.serviceAccount });\n return {\n apiClient: createApiClient({ auth }),\n reportingClient: createReportingClient({ auth }),\n };\n}\n\nasync function loadTrainConfig(file: string): Promise<TrainConfig> {\n const { readFile } = await import(\"node:fs/promises\");\n try {\n const raw = await readFile(file, \"utf-8\");\n return JSON.parse(raw) as TrainConfig;\n } catch {\n throw new GpcError(\n `Could not read train config from \"${file}\"`,\n \"INVALID_CONFIG\",\n 2,\n `Ensure the file exists and contains valid JSON`,\n );\n }\n}\n\nexport function registerTrainCommands(program: Command): void {\n const train = program\n .command(\"train\")\n .description(\"Manage config-driven staged rollout release trains\");\n\n train\n .command(\"start\")\n .description(\"Start a release train from a config file\")\n .option(\"--config <file>\", \"Train config file (JSON)\", \".gpcrc-train.json\")\n .option(\"--force\", \"Restart even if a train is already running\")\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 const trainConfig = await loadTrainConfig(options.config as string);\n\n if (isDryRun(program)) {\n printDryRun(\n {\n command: \"train start\",\n action: \"start release train\",\n target: packageName,\n details: { stages: trainConfig.stages.length, gates: trainConfig.gates },\n },\n format,\n formatOutput,\n );\n return;\n }\n\n const { apiClient } = await getClients(config);\n\n const state = await startTrain(apiClient, packageName, trainConfig, {\n force: options.force as boolean,\n });\n console.log(formatOutput(state, format));\n });\n\n train\n .command(\"status\")\n .description(\"Show current release train state\")\n .action(async () => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n const state = await getTrainStatus(packageName);\n if (!state) {\n if (format === \"json\") {\n console.log(formatOutput(null, format));\n } else {\n console.log(`No active release train for ${packageName}.`);\n }\n return;\n }\n console.log(formatOutput(state, format));\n });\n\n train\n .command(\"advance\")\n .description(\"Advance the train to the next stage (after delay and gate checks pass)\")\n .action(async () => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n const { apiClient, reportingClient } = await getClients(config);\n\n const state = await advanceTrain(apiClient, reportingClient, packageName);\n if (!state) {\n console.log(`No active release train for ${packageName}.`);\n return;\n }\n console.log(formatOutput(state, format));\n });\n\n train\n .command(\"pause\")\n .description(\"Pause the release train\")\n .action(async () => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n const format = getOutputFormat(program, config);\n\n const state = await pauseTrain(packageName);\n if (!state) {\n console.log(`No active release train for ${packageName}.`);\n return;\n }\n console.log(formatOutput(state, format));\n });\n\n train\n .command(\"abort\")\n .description(\"Abort and clear the release train state\")\n .action(async () => {\n const config = await loadConfig();\n const packageName = resolvePackageName(program.opts()[\"app\"], config);\n\n await requireConfirm(`Abort release train for ${packageName}?`, program);\n\n await abortTrain(packageName);\n console.log(`Release train aborted for ${packageName}.`);\n });\n}\n"],"mappings":"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","names":[]}
|
|
@@ -8,7 +8,7 @@ import {
|
|
|
8
8
|
updateBinaryInPlace,
|
|
9
9
|
updateViaBrew,
|
|
10
10
|
updateViaNpm
|
|
11
|
-
} from "./chunk-
|
|
11
|
+
} from "./chunk-E7SVZ7RF.js";
|
|
12
12
|
import "./chunk-3SJ6OXCZ.js";
|
|
13
13
|
|
|
14
14
|
// src/commands/update.ts
|
|
@@ -17,7 +17,7 @@ function registerUpdateCommand(program) {
|
|
|
17
17
|
program.command("update").description("Update gpc to the latest version").option("--check", "Check for updates without installing (exits 0 always)").option("--force", "Update even if already on the latest version").action(async (opts, cmd) => {
|
|
18
18
|
const parentOpts = cmd.parent?.opts() ?? {};
|
|
19
19
|
const jsonMode = !!(parentOpts["json"] || parentOpts["output"] === "json");
|
|
20
|
-
const currentVersion = "0.9.
|
|
20
|
+
const currentVersion = "0.9.54";
|
|
21
21
|
if (currentVersion === "0.0.0") {
|
|
22
22
|
if (jsonMode) {
|
|
23
23
|
console.log(
|
|
@@ -148,14 +148,11 @@ Run: gpc update`);
|
|
|
148
148
|
}
|
|
149
149
|
case "unknown":
|
|
150
150
|
spinner.fail();
|
|
151
|
-
throw Object.assign(
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
suggestion: "Update manually:\n npm: npm install -g @gpc-cli/cli@latest\n Homebrew: brew upgrade yasserstudio/tap/gpc\n Binary: https://github.com/yasserstudio/gpc/releases/latest"
|
|
157
|
-
}
|
|
158
|
-
);
|
|
151
|
+
throw Object.assign(new Error("Could not detect install method"), {
|
|
152
|
+
code: "UPDATE_UNKNOWN_METHOD",
|
|
153
|
+
exitCode: 1,
|
|
154
|
+
suggestion: "Update manually:\n npm: npm install -g @gpc-cli/cli@latest\n Homebrew: brew upgrade yasserstudio/tap/gpc\n Binary: https://github.com/yasserstudio/gpc/releases/latest"
|
|
155
|
+
});
|
|
159
156
|
}
|
|
160
157
|
if (jsonMode) {
|
|
161
158
|
console.log(
|
|
@@ -175,4 +172,4 @@ Run: gpc update`);
|
|
|
175
172
|
export {
|
|
176
173
|
registerUpdateCommand
|
|
177
174
|
};
|
|
178
|
-
//# sourceMappingURL=update-
|
|
175
|
+
//# sourceMappingURL=update-73YOR4GP.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/commands/update.ts"],"sourcesContent":["import type { Command } from \"commander\";\nimport {\n checkForUpdate,\n fetchChecksums,\n updateViaNpm,\n updateViaBrew,\n updateBinaryInPlace,\n getPlatformAsset,\n getCurrentBinaryPath,\n cleanupStaleUpdateFiles,\n} from \"../updater.js\";\nimport { createSpinner } from \"@gpc-cli/core\";\n\nexport function registerUpdateCommand(program: Command): void {\n program\n .command(\"update\")\n .description(\"Update gpc to the latest version\")\n .option(\"--check\", \"Check for updates without installing (exits 0 always)\")\n .option(\"--force\", \"Update even if already on the latest version\")\n .action(async (opts: { check?: boolean; force?: boolean }, cmd) => {\n const parentOpts = (cmd.parent?.opts() ?? {}) as Record<string, unknown>;\n const jsonMode = !!(parentOpts[\"json\"] || parentOpts[\"output\"] === \"json\");\n const currentVersion = process.env[\"__GPC_VERSION\"] ?? \"0.0.0\";\n\n // Dev build guard\n if (currentVersion === \"0.0.0\") {\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: false,\n reason: \"Cannot update a development build\",\n current: currentVersion,\n }),\n );\n } else {\n console.log(\"Cannot update a development build (version: 0.0.0).\");\n }\n return;\n }\n\n const spinner = createSpinner(\"Checking for updates...\");\n spinner.start();\n\n let result;\n try {\n result = await checkForUpdate(currentVersion);\n } catch (err) {\n spinner.fail(\"Failed to check for updates\");\n throw err;\n }\n\n // --check mode: always exit 0, communicate via output\n if (opts.check) {\n spinner.stop();\n if (jsonMode) {\n console.log(\n JSON.stringify({\n current: result.current,\n latest: result.latest,\n updateAvailable: result.updateAvailable,\n installMethod: result.installMethod,\n releaseUrl: result.release.html_url,\n }),\n );\n } else if (result.updateAvailable) {\n console.log(`Update available: ${result.current} → ${result.latest}`);\n console.log(`Install method: ${result.installMethod}`);\n console.log(`Release: ${result.release.html_url}`);\n console.log(`\\nRun: gpc update`);\n } else {\n console.log(`Already on latest version: v${result.current}`);\n console.log(`Install method: ${result.installMethod}`);\n }\n return;\n }\n\n // Already up to date\n if (!result.updateAvailable && !opts.force) {\n spinner.stop(`Already on latest version: v${result.current}`);\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: true,\n current: result.current,\n latest: result.latest,\n updated: false,\n installMethod: result.installMethod,\n }),\n );\n }\n return;\n }\n\n spinner.update(\n `Updating v${result.current} → v${result.latest} (${result.installMethod})...`,\n );\n\n switch (result.installMethod) {\n case \"npm\":\n spinner.stop();\n await updateViaNpm({ silent: jsonMode });\n break;\n\n case \"homebrew\":\n spinner.stop();\n await updateViaBrew({ silent: jsonMode });\n break;\n\n case \"binary\": {\n const assetName = getPlatformAsset();\n if (!assetName) {\n spinner.fail();\n throw Object.assign(\n new Error(`Unsupported platform: ${process.platform}/${process.arch}`),\n {\n code: \"UPDATE_UNSUPPORTED_PLATFORM\",\n exitCode: 2,\n suggestion: `Download manually: ${result.release.html_url}`,\n },\n );\n }\n\n const assetObj = result.release.assets.find((a) => a.name === assetName);\n if (!assetObj) {\n spinner.fail();\n throw Object.assign(\n new Error(`No binary found for ${assetName} in release ${result.latestTag}`),\n {\n code: \"UPDATE_ASSET_NOT_FOUND\",\n exitCode: 4,\n suggestion: `Check: ${result.release.html_url}`,\n },\n );\n }\n\n const sizeMB = (assetObj.size / (1024 * 1024)).toFixed(1);\n spinner.stop();\n\n const checksums = await fetchChecksums(result.release);\n const expectedHash = checksums.get(assetName) ?? \"\";\n const binaryPath = getCurrentBinaryPath();\n\n // Clean up stale files from previous update attempts\n cleanupStaleUpdateFiles(binaryPath);\n\n if (jsonMode) {\n await updateBinaryInPlace(assetObj.browser_download_url, expectedHash, binaryPath);\n } else {\n // Show live download progress on a single overwriting line\n const label = `Downloading ${assetName} (${sizeMB} MB)`;\n process.stdout.write(`${label}...\\n`);\n let lastPct = -1;\n await updateBinaryInPlace(assetObj.browser_download_url, expectedHash, binaryPath, {\n onProgress(downloaded, total) {\n if (total <= 0) return;\n const pct = Math.min(100, Math.round((downloaded / total) * 100));\n if (pct === lastPct) return; // avoid redundant writes\n lastPct = pct;\n const dlMB = (downloaded / (1024 * 1024)).toFixed(1);\n process.stdout.write(\n `\\r${label} ${dlMB} / ${sizeMB} MB (${String(pct).padStart(3)}%) `,\n );\n },\n });\n process.stdout.write(\"\\n\");\n }\n break;\n }\n\n case \"unknown\":\n spinner.fail();\n throw Object.assign(\n new Error(\"Could not detect install method\"),\n {\n code: \"UPDATE_UNKNOWN_METHOD\",\n exitCode: 1,\n suggestion:\n \"Update manually:\\n\" +\n \" npm: npm install -g @gpc-cli/cli@latest\\n\" +\n \" Homebrew: brew upgrade yasserstudio/tap/gpc\\n\" +\n \" Binary: https://github.com/yasserstudio/gpc/releases/latest\",\n },\n );\n }\n\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: true,\n previous: result.current,\n current: result.latest,\n method: result.installMethod,\n }),\n );\n } else {\n console.log(`\\n✔ Updated to v${result.latest}`);\n }\n });\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/commands/update.ts"],"sourcesContent":["import type { Command } from \"commander\";\nimport {\n checkForUpdate,\n fetchChecksums,\n updateViaNpm,\n updateViaBrew,\n updateBinaryInPlace,\n getPlatformAsset,\n getCurrentBinaryPath,\n cleanupStaleUpdateFiles,\n} from \"../updater.js\";\nimport { createSpinner } from \"@gpc-cli/core\";\n\nexport function registerUpdateCommand(program: Command): void {\n program\n .command(\"update\")\n .description(\"Update gpc to the latest version\")\n .option(\"--check\", \"Check for updates without installing (exits 0 always)\")\n .option(\"--force\", \"Update even if already on the latest version\")\n .action(async (opts: { check?: boolean; force?: boolean }, cmd) => {\n const parentOpts = (cmd.parent?.opts() ?? {}) as Record<string, unknown>;\n const jsonMode = !!(parentOpts[\"json\"] || parentOpts[\"output\"] === \"json\");\n const currentVersion = process.env[\"__GPC_VERSION\"] ?? \"0.0.0\";\n\n // Dev build guard\n if (currentVersion === \"0.0.0\") {\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: false,\n reason: \"Cannot update a development build\",\n current: currentVersion,\n }),\n );\n } else {\n console.log(\"Cannot update a development build (version: 0.0.0).\");\n }\n return;\n }\n\n const spinner = createSpinner(\"Checking for updates...\");\n spinner.start();\n\n let result;\n try {\n result = await checkForUpdate(currentVersion);\n } catch (err) {\n spinner.fail(\"Failed to check for updates\");\n throw err;\n }\n\n // --check mode: always exit 0, communicate via output\n if (opts.check) {\n spinner.stop();\n if (jsonMode) {\n console.log(\n JSON.stringify({\n current: result.current,\n latest: result.latest,\n updateAvailable: result.updateAvailable,\n installMethod: result.installMethod,\n releaseUrl: result.release.html_url,\n }),\n );\n } else if (result.updateAvailable) {\n console.log(`Update available: ${result.current} → ${result.latest}`);\n console.log(`Install method: ${result.installMethod}`);\n console.log(`Release: ${result.release.html_url}`);\n console.log(`\\nRun: gpc update`);\n } else {\n console.log(`Already on latest version: v${result.current}`);\n console.log(`Install method: ${result.installMethod}`);\n }\n return;\n }\n\n // Already up to date\n if (!result.updateAvailable && !opts.force) {\n spinner.stop(`Already on latest version: v${result.current}`);\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: true,\n current: result.current,\n latest: result.latest,\n updated: false,\n installMethod: result.installMethod,\n }),\n );\n }\n return;\n }\n\n spinner.update(\n `Updating v${result.current} → v${result.latest} (${result.installMethod})...`,\n );\n\n switch (result.installMethod) {\n case \"npm\":\n spinner.stop();\n await updateViaNpm({ silent: jsonMode });\n break;\n\n case \"homebrew\":\n spinner.stop();\n await updateViaBrew({ silent: jsonMode });\n break;\n\n case \"binary\": {\n const assetName = getPlatformAsset();\n if (!assetName) {\n spinner.fail();\n throw Object.assign(\n new Error(`Unsupported platform: ${process.platform}/${process.arch}`),\n {\n code: \"UPDATE_UNSUPPORTED_PLATFORM\",\n exitCode: 2,\n suggestion: `Download manually: ${result.release.html_url}`,\n },\n );\n }\n\n const assetObj = result.release.assets.find((a) => a.name === assetName);\n if (!assetObj) {\n spinner.fail();\n throw Object.assign(\n new Error(`No binary found for ${assetName} in release ${result.latestTag}`),\n {\n code: \"UPDATE_ASSET_NOT_FOUND\",\n exitCode: 4,\n suggestion: `Check: ${result.release.html_url}`,\n },\n );\n }\n\n const sizeMB = (assetObj.size / (1024 * 1024)).toFixed(1);\n spinner.stop();\n\n const checksums = await fetchChecksums(result.release);\n const expectedHash = checksums.get(assetName) ?? \"\";\n const binaryPath = getCurrentBinaryPath();\n\n // Clean up stale files from previous update attempts\n cleanupStaleUpdateFiles(binaryPath);\n\n if (jsonMode) {\n await updateBinaryInPlace(assetObj.browser_download_url, expectedHash, binaryPath);\n } else {\n // Show live download progress on a single overwriting line\n const label = `Downloading ${assetName} (${sizeMB} MB)`;\n process.stdout.write(`${label}...\\n`);\n let lastPct = -1;\n await updateBinaryInPlace(assetObj.browser_download_url, expectedHash, binaryPath, {\n onProgress(downloaded, total) {\n if (total <= 0) return;\n const pct = Math.min(100, Math.round((downloaded / total) * 100));\n if (pct === lastPct) return; // avoid redundant writes\n lastPct = pct;\n const dlMB = (downloaded / (1024 * 1024)).toFixed(1);\n process.stdout.write(\n `\\r${label} ${dlMB} / ${sizeMB} MB (${String(pct).padStart(3)}%) `,\n );\n },\n });\n process.stdout.write(\"\\n\");\n }\n break;\n }\n\n case \"unknown\":\n spinner.fail();\n throw Object.assign(new Error(\"Could not detect install method\"), {\n code: \"UPDATE_UNKNOWN_METHOD\",\n exitCode: 1,\n suggestion:\n \"Update manually:\\n\" +\n \" npm: npm install -g @gpc-cli/cli@latest\\n\" +\n \" Homebrew: brew upgrade yasserstudio/tap/gpc\\n\" +\n \" Binary: https://github.com/yasserstudio/gpc/releases/latest\",\n });\n }\n\n if (jsonMode) {\n console.log(\n JSON.stringify({\n success: true,\n previous: result.current,\n current: result.latest,\n method: result.installMethod,\n }),\n );\n } else {\n console.log(`\\n✔ Updated to v${result.latest}`);\n }\n });\n}\n"],"mappings":"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","names":[]}
|
|
@@ -8,7 +8,7 @@ import {
|
|
|
8
8
|
} from "./chunk-ELXAK7GI.js";
|
|
9
9
|
import {
|
|
10
10
|
requireConfirm
|
|
11
|
-
} from "./chunk-
|
|
11
|
+
} from "./chunk-RZQSEDKI.js";
|
|
12
12
|
|
|
13
13
|
// src/commands/users.ts
|
|
14
14
|
import { loadConfig } from "@gpc-cli/config";
|
|
@@ -162,4 +162,4 @@ function registerUsersCommands(program) {
|
|
|
162
162
|
export {
|
|
163
163
|
registerUsersCommands
|
|
164
164
|
};
|
|
165
|
-
//# sourceMappingURL=users-
|
|
165
|
+
//# sourceMappingURL=users-FOMAT7DY.js.map
|
|
@@ -16,7 +16,11 @@ function registerValidateCommand(program) {
|
|
|
16
16
|
program.command("validate <file>").description("Pre-submission validation checks").option("--track <track>", "Target track to validate").option("--mapping <file>", "ProGuard/R8 mapping file").option("--notes <text>", "Release notes (en-US)").option("--notes-dir <dir>", "Read release notes from directory (<dir>/<lang>.txt)").action(async (file, options) => {
|
|
17
17
|
if (options.notes && options.notesDir) {
|
|
18
18
|
const err = new Error("Cannot use both --notes and --notes-dir");
|
|
19
|
-
Object.assign(err, {
|
|
19
|
+
Object.assign(err, {
|
|
20
|
+
code: "USAGE_ERROR",
|
|
21
|
+
exitCode: 2,
|
|
22
|
+
suggestion: "Use either --notes or --notes-dir, not both."
|
|
23
|
+
});
|
|
20
24
|
throw err;
|
|
21
25
|
}
|
|
22
26
|
const config = await loadConfig();
|
|
@@ -59,4 +63,4 @@ ${result.valid ? green("\u2713 Valid") : red("\u2717 Invalid")}`);
|
|
|
59
63
|
export {
|
|
60
64
|
registerValidateCommand
|
|
61
65
|
};
|
|
62
|
-
//# sourceMappingURL=validate-
|
|
66
|
+
//# sourceMappingURL=validate-RHWEZYD2.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/validate.ts"],"sourcesContent":["import type { Command } from \"commander\";\nimport { loadConfig } from \"@gpc-cli/config\";\nimport { validatePreSubmission, readReleaseNotesFromDir } from \"@gpc-cli/core\";\nimport { formatOutput } from \"@gpc-cli/core\";\nimport { getOutputFormat } from \"../format.js\";\nimport { green, red, yellow } from \"../colors.js\";\n\nexport function registerValidateCommand(program: Command): void {\n program\n .command(\"validate <file>\")\n .description(\"Pre-submission validation checks\")\n .option(\"--track <track>\", \"Target track to validate\")\n .option(\"--mapping <file>\", \"ProGuard/R8 mapping file\")\n .option(\"--notes <text>\", \"Release notes (en-US)\")\n .option(\"--notes-dir <dir>\", \"Read release notes from directory (<dir>/<lang>.txt)\")\n .action(async (file: string, options) => {\n if (options.notes && options.notesDir) {\n const err = new Error(\"Cannot use both --notes and --notes-dir\");\n Object.assign(err, {\n code: \"USAGE_ERROR\",\n exitCode: 2,\n suggestion: \"Use either --notes or --notes-dir, not both.\",\n });\n throw err;\n }\n\n const config = await loadConfig();\n const format = getOutputFormat(program, config);\n\n let notes: { language: string; text: string }[] | undefined;\n if (options.notesDir) {\n notes = await readReleaseNotesFromDir(options.notesDir);\n } else if (options.notes) {\n notes = [{ language: \"en-US\", text: options.notes }];\n }\n\n const result = await validatePreSubmission({\n filePath: file,\n mappingFile: options.mapping,\n track: options.track,\n notes,\n });\n\n if (format === \"json\") {\n console.log(formatOutput(result, format));\n } else {\n const checkRows = result.checks.map((c) => ({\n check: c.name,\n passed: c.passed ? green(\"✓ pass\") : red(\"✗ FAIL\"),\n message: c.message,\n }));\n console.log(formatOutput(checkRows, format));\n if (result.warnings.length > 0) {\n console.log(\"\\nWarnings:\");\n for (const w of result.warnings) {\n console.log(` ${yellow(\"⚠\")} ${w}`);\n }\n }\n console.log(`\\n${result.valid ? green(\"✓ Valid\") : red(\"✗ Invalid\")}`);\n }\n if (!result.valid) {\n process.exitCode = 1;\n }\n });\n}\n"],"mappings":";;;;;;;;;;;AACA,SAAS,kBAAkB;AAC3B,SAAS,uBAAuB,+BAA+B;AAC/D,SAAS,oBAAoB;AAItB,SAAS,wBAAwB,SAAwB;AAC9D,UACG,QAAQ,iBAAiB,EACzB,YAAY,kCAAkC,EAC9C,OAAO,mBAAmB,0BAA0B,EACpD,OAAO,oBAAoB,0BAA0B,EACrD,OAAO,kBAAkB,uBAAuB,EAChD,OAAO,qBAAqB,sDAAsD,EAClF,OAAO,OAAO,MAAc,YAAY;AACvC,QAAI,QAAQ,SAAS,QAAQ,UAAU;AACrC,YAAM,MAAM,IAAI,MAAM,yCAAyC;AAC/D,aAAO,OAAO,KAAK;AAAA,QACjB,MAAM;AAAA,QACN,UAAU;AAAA,QACV,YAAY;AAAA,MACd,CAAC;AACD,YAAM;AAAA,IACR;AAEA,UAAM,SAAS,MAAM,WAAW;AAChC,UAAM,SAAS,gBAAgB,SAAS,MAAM;AAE9C,QAAI;AACJ,QAAI,QAAQ,UAAU;AACpB,cAAQ,MAAM,wBAAwB,QAAQ,QAAQ;AAAA,IACxD,WAAW,QAAQ,OAAO;AACxB,cAAQ,CAAC,EAAE,UAAU,SAAS,MAAM,QAAQ,MAAM,CAAC;AAAA,IACrD;AAEA,UAAM,SAAS,MAAM,sBAAsB;AAAA,MACzC,UAAU;AAAA,MACV,aAAa,QAAQ;AAAA,MACrB,OAAO,QAAQ;AAAA,MACf;AAAA,IACF,CAAC;AAED,QAAI,WAAW,QAAQ;AACrB,cAAQ,IAAI,aAAa,QAAQ,MAAM,CAAC;AAAA,IAC1C,OAAO;AACL,YAAM,YAAY,OAAO,OAAO,IAAI,CAAC,OAAO;AAAA,QAC1C,OAAO,EAAE;AAAA,QACT,QAAQ,EAAE,SAAS,MAAM,aAAQ,IAAI,IAAI,aAAQ;AAAA,QACjD,SAAS,EAAE;AAAA,MACb,EAAE;AACF,cAAQ,IAAI,aAAa,WAAW,MAAM,CAAC;AAC3C,UAAI,OAAO,SAAS,SAAS,GAAG;AAC9B,gBAAQ,IAAI,aAAa;AACzB,mBAAW,KAAK,OAAO,UAAU;AAC/B,kBAAQ,IAAI,KAAK,OAAO,QAAG,CAAC,IAAI,CAAC,EAAE;AAAA,QACrC;AAAA,MACF;AACA,cAAQ,IAAI;AAAA,EAAK,OAAO,QAAQ,MAAM,cAAS,IAAI,IAAI,gBAAW,CAAC,EAAE;AAAA,IACvE;AACA,QAAI,CAAC,OAAO,OAAO;AACjB,cAAQ,WAAW;AAAA,IACrB;AAAA,EACF,CAAC;AACL;","names":[]}
|
|
@@ -73,17 +73,13 @@ function registerVerifyCommand(program) {
|
|
|
73
73
|
` Status: ${isBeforeEnforcement() ? yellow(formatDeadline()) : green(formatDeadline())}`
|
|
74
74
|
);
|
|
75
75
|
console.log(` Account: ${accountEmail}`);
|
|
76
|
-
console.log(
|
|
77
|
-
` Console: Play Console (auto-registered for Play apps)`
|
|
78
|
-
);
|
|
76
|
+
console.log(` Console: Play Console (auto-registered for Play apps)`);
|
|
79
77
|
console.log("");
|
|
80
78
|
console.log(" What you need to do:");
|
|
81
79
|
console.log(
|
|
82
80
|
` 1. Confirm identity verification in Play Console ${dim("\u2192 Settings \u2192 Developer Account")}`
|
|
83
81
|
);
|
|
84
|
-
console.log(
|
|
85
|
-
` 2. Check auto-registration results above your app list`
|
|
86
|
-
);
|
|
82
|
+
console.log(` 2. Check auto-registration results above your app list`);
|
|
87
83
|
console.log(` 3. Register any non-Play apps within Play Console`);
|
|
88
84
|
console.log("");
|
|
89
85
|
console.log(" Resources:");
|
|
@@ -91,16 +87,12 @@ function registerVerifyCommand(program) {
|
|
|
91
87
|
console.log(` ${dim("\u2192")} ${RESOURCES.playConsole}`);
|
|
92
88
|
console.log(` ${dim("\u2192")} ${RESOURCES.limitedDistribution}`);
|
|
93
89
|
console.log("");
|
|
94
|
-
console.log(
|
|
95
|
-
|
|
96
|
-
);
|
|
97
|
-
console.log(
|
|
98
|
-
dim(" Full guide: gpc docs developer-verification")
|
|
99
|
-
);
|
|
90
|
+
console.log(dim(" Run 'gpc verify --open' to open the verification page in your browser."));
|
|
91
|
+
console.log(dim(" Full guide: gpc docs developer-verification"));
|
|
100
92
|
console.log("");
|
|
101
93
|
});
|
|
102
94
|
}
|
|
103
95
|
export {
|
|
104
96
|
registerVerifyCommand
|
|
105
97
|
};
|
|
106
|
-
//# sourceMappingURL=verify-
|
|
98
|
+
//# sourceMappingURL=verify-H4ZUVHMZ.js.map
|