kappmaker
v1.13.16
Published
CLI tool for bootstrapping KAppMaker mobile apps
Readme
KAppMaker CLI
Full documentation at cli.kappmaker.com
CLI tool that automates the entire mobile app launch process — from project scaffolding to store-ready builds.
A single kappmaker create command can:
- Clone a template repository and set up a new project
- Create a Firebase project, register Android + iOS apps, enable authentication, and download SDK configs
- Generate an AI-powered app logo with automatic background removal
- Create an App Store Connect listing with metadata, categories, age rating, subscriptions, privacy declarations, and review contact info
- Configure an existing Google Play Console app — store listings, subscriptions, one-time in-app products, and the data safety declaration — via a built-in wrapper around the Play Publisher API (no external CLI, no extra dependencies)
- Set up Adapty subscription products, paywalls, and placements for both iOS and Android
- Refactor Gradle package names and application IDs
- Set up the build environment (Android SDK)
- Produce a signed Android release build (AAB) via Fastlane, ready to upload to Google Play
On top of that, standalone commands let you generate marketing screenshots from a text description, translate screenshots to 48+ locales in parallel, generate arbitrary images with AI, remove image backgrounds, enhance image quality, and split grid images — all powered by AI.
By default it uses the KAppMaker boilerplate (Kotlin Multiplatform), but you can bring your own template repository via --template-repo or kappmaker config set templateRepo <your-repo-url>. Boilerplate-specific steps (Gradle refactor, Fastlane build, CocoaPods) are automatically detected and skipped with a warning when using a custom template.
Installation
npm install -g kappmakerThen use it anywhere:
kappmaker create <AppName>npm install
npx tsx src/index.ts create <AppName>Configuration
Run interactive setup to configure API keys and preferences:
kappmaker config initOr set keys individually:
kappmaker config set falApiKey <your-key> # For AI features (logo, screenshots) — or prompted on first use
kappmaker config set imgbbApiKey <your-key> # For screenshot translation/generation — or prompted on first use
kappmaker config set openaiApiKey <your-key> # For generate-screenshots — or prompted on first use
kappmaker config set templateRepo <your-repo> # Use your own template (default: KAppMaker)See all config keys and external services setup for details.
Claude Code Skill
If you use Claude Code, you can install the /kappmaker skill to run any CLI command through natural language — with automatic prerequisite checks, guided setup, and inline error recovery.
Install:
npx skills add KAppMaker/KAppMaker-CLI --skill kappmakerOr via the Claude Code plugin system:
/plugin marketplace add KAppMaker/KAppMaker-CLI
/plugin install kappmaker@KAppMaker-CLIUse:
/kappmaker create MyApp
/kappmaker generate screenshots for my fitness app
/kappmaker set up App Store ConnectClaude will check your config, verify API keys are set, and walk you through any missing prerequisites before running the command.
Context-aware: when a command needs inputs you didn't pass (app description, app name, brand color, tagline, keywords, etc.), the skill reads your project's AiGuidelines/ folder first (app-idea.md, prd.md, keywords.md, brand.md), then README.md, then existing ASO metadata, and fills in what it finds — you're only prompted for things that aren't already written down somewhere in the project.
Table of Contents
- Claude Code Skill
- Configuration
- Commands Overview
- Features
- Prerequisites
- External Services & API Keys
- Commands
create <app-name>clone <app-name>git setup-upstreamfirebase— Firebase setup steps (login, project, apps, auth, configs)create-logogenerate-imagecreate-appstore-appappstore-monetization-push— Push subs + IAPs from config to ASC (standalone)gpc— Google Play Console managementgpc monetization push— Push subs + IAPs from config to Play (standalone)subscription add— Quick-add a subscription to Play + ASCiap add— Quick-add a credit-pack IAP to Play + ASC + Adaptyadapty setup- Image Tools
convert-webptranslate-screenshotsgenerate-screenshotsgenerate-feature-imagegenerate-ios-iconsgenerate-android-icons- ASO Keyword Research (skill-driven)
- ASO Metadata Localization (skill-driven)
fastlane configurepublishgenerate-keystoreandroid-release-buildrefactorupdate-version
- Config Reference
- Project Structure
Commands Overview
| Command | Description |
|---------|-------------|
| kappmaker create <app-name> | Full end-to-end app setup (Firebase, logo, App Store Connect, Google Play Console, Adapty, release build) |
| kappmaker clone <app-name> | Clone the template into <AppName>-All (step 1 of create as a standalone) |
| kappmaker git setup-upstream | Rename origin to upstream (step 10 of create as a standalone) |
| kappmaker firebase login | firebase login — authenticate the Firebase CLI |
| kappmaker firebase project | Create a Firebase project (idempotent) |
| kappmaker firebase apps | Create Android + iOS apps in a Firebase project (idempotent) |
| kappmaker firebase auth-anonymous | Enable anonymous authentication |
| kappmaker firebase configs | Download google-services.json + GoogleService-Info.plist |
| kappmaker create-logo | Generate an app logo with AI (fal.ai) |
| kappmaker generate-image | Generate an arbitrary image with AI — generic wrapper around fal.ai nano-banana-2 |
| kappmaker create-appstore-app | Set up an app on App Store Connect (metadata, subscriptions, privacy) |
| kappmaker appstore-monetization-push | Push subscriptions + IAPs from appstore-config.json to App Store Connect (monetization step of create-appstore-app as a standalone; supports --subscriptions-only, --iap-only, --config) |
| kappmaker gpc setup | Set up an existing app on Google Play Console (listings, subscriptions, IAPs, data safety) |
| kappmaker gpc listings push | Push store listings from the Google Play config file |
| kappmaker gpc subscriptions list/push | List or push subscriptions on Google Play Console |
| kappmaker gpc iap list/push | List or push one-time in-app products on Google Play Console |
| kappmaker gpc data-safety push | Push data safety declaration on Google Play Console |
| kappmaker gpc app-check --package <pkg> | Check if a package exists on Google Play Console |
| kappmaker gpc monetization push | Push subscriptions + IAPs from googleplay-config.json to Google Play Console (monetization steps of gpc setup as a standalone; supports --subscriptions-only, --iap-only) |
| kappmaker subscription add | Quick-add one new subscription to Google Play + App Store Connect (auto-aligned IDs, full PPP fan-out, auto-creates ASC subscription group if missing) |
| kappmaker iap add | Quick-add one new credit-pack IAP to Google Play + App Store Connect + Adapty (auto-aligned IDs, full PPP fan-out) |
| kappmaker adapty setup | Set up Adapty products, paywalls, and placements |
| kappmaker image-split <image> | Split a grid image into individual tiles |
| kappmaker image-remove-bg <image> | Remove background from an image (fal.ai) |
| kappmaker image-enhance <image> | Upscale and enhance image quality (fal.ai) |
| kappmaker convert-webp <source> | Convert images (PNG, JPG, BMP, TIFF, GIF) to WebP |
| kappmaker translate-screenshots [dir] | Translate screenshots to multiple locales (fal.ai) |
| kappmaker generate-screenshots | Generate marketing screenshots with AI (OpenAI + fal.ai) |
| kappmaker generate-feature-image | Generate a Google Play feature graphic (1024×500) with AI (OpenAI + fal.ai) |
| kappmaker generate-ios-icons | Generate all iOS AppIcon.appiconset PNG sizes + Contents.json from one logo (sharp, no AI) |
| kappmaker generate-android-icons | Generate all Android mipmap-* launcher icons (5 densities × 3 files) + adaptive icon XML + colors.xml entry (sharp, no AI) |
| ASO Keyword Research | (skill-driven, no binary) — Find high-value keywords with popularity + difficulty scores via Astro MCP, cluster by sub-niche, write AiGuidelines/keywords.md |
| ASO Metadata Localization | (skill-driven, no binary) — Write localized name/subtitle/keywords/description and title/short/full_description files for App Store + Google Play |
| kappmaker fastlane configure | Set up Fastlane in the mobile app directory |
| kappmaker publish | Build and upload to Google Play and/or App Store via Fastlane |
| kappmaker generate-keystore | Generate an Android signing keystore for Play Store releases |
| kappmaker android-release-build | Build a signed Android release AAB |
| kappmaker refactor | Refactor package names, application ID, bundle ID, and app name |
| kappmaker update-version | Bump Android and iOS version codes and version name |
| kappmaker config | Manage CLI settings, API keys, and global defaults |
Features
Works with any project
These commands are standalone and don't depend on any specific boilerplate:
- AI logo generation — Generate logo variations with fal.ai, pick your favorite, auto-remove background
- AI screenshot generation — Generate marketing screenshots from a text description (8 style presets)
- Screenshot translation — Translate app screenshots to 48+ locales in parallel
- ASO keyword research — Skill-driven workflow that discovers high-value sub-niche keywords for a base topic via the Astro MCP tools (real-time App Store popularity + difficulty scores from competitor analysis and AI suggestions), filters them by configurable thresholds, clusters them by sub-niche, and writes
AiGuidelines/keywords.md. Falls back to a manual brainstorm when Astro MCP isn't connected. Output feeds directly into the metadata localization workflow below. - ASO metadata localization — Skill-driven workflow that writes per-locale
name/subtitle/keywords/description(iOS) andtitle/short_description/full_description(Android) text files with two strategies: US-indexed keyword expansion (English copy with different keywords per locale to multiply indexed surface) or native per-market localization. Char-limits, keyword-field rules, and no-duplicate-word-across-fields are enforced automatically. - App Store Connect setup — Register bundle ID (with Sign in with Apple, In-App Purchases, and Push Notifications capabilities enabled automatically), create app, set metadata, categories, age rating, subscriptions, privacy, and review info — fully automated, no manual App Store Connect steps needed
- Google Play Console setup — Push store listings, subscriptions (new monetization API), one-time in-app products, and the data safety declaration via a built-in wrapper around the Play Publisher API — no external CLI, no extra dependencies
- Adapty subscription setup — Create products, paywalls, and placements for iOS and Android
- Version bumping — Increment Android and iOS version codes and names in one command
- Image tools — Split grids, remove backgrounds, enhance quality, convert to WebP
KAppMaker boilerplate-specific
The create command runs the full end-to-end setup. Some steps assume the KAppMaker project structure and will be skipped with a warning if you use a custom template:
- Package refactor — Renames package name, app ID, and display name using the TypeScript refactor service (also available standalone via
kappmaker refactor) - Firebase SDK config placement — Downloads
google-services.jsonandGoogleService-Info.plistto KAppMaker-specific paths (falls back toAssets/for custom templates) - Build environment — Creates
local.properties; iOS dependencies resolve via Swift Package Manager at build time (CocoaPods runs only if a custom template still has aPodfile) - Android release build — Generates keystore and builds signed AAB (also available standalone via
kappmaker android-release-build) - Git remotes — Renames origin to upstream (designed for the "fork from template" workflow)
- Screenshot translation default path — Defaults to
MobileApp/distribution/ios/appstore_metadata/screenshots/en-US(falls back to parent of source directory)
Prerequisites
- Node.js >= 20
- Git
- Firebase CLI —
npm install -g firebase-tools - Fastlane — via Bundler in the template repo
- Android SDK — installed at
~/Library/Android/sdk(configurable) - asc CLI (optional, for App Store Connect) —
brew install asc(requires ≥ 1.4.0 as of KAppMaker 1.7.0 for the bulk CSV subscription-pricing path) - Adapty CLI (optional, for Adapty setup) —
npm install -g adapty - No extra CLI for Google Play Console —
kappmaker gpctalks to the Play Publisher API directly via Node's built-infetchandcrypto; all it needs is the service-account JSON path set ingoogleServiceAccountPath
External Services & API Keys
The CLI integrates with several external services for AI image generation, app store management, and subscription setup. All keys are stored locally at ~/.config/kappmaker/config.json.
fal.ai — AI Image Generation
Used for: Logo generation, background removal, image enhancement, screenshot translation, and screenshot generation.
How to get your key:
- Sign up at fal.ai
- Go to Dashboard > Keys and create an API key
kappmaker config set falApiKey <your-key>— or skip this and the CLI will prompt you the first time you run a command that needs it
ImgBB — Image Hosting
Used for: Temporarily hosting reference images when generating or translating screenshots (fal.ai needs a public URL to process images).
How to get your key:
- Sign up at imgbb.com
- Go to api.imgbb.com and get your free API key
kappmaker config set imgbbApiKey <your-key>— or prompted on first use
OpenAI — Prompt Generation
Used for: Generating detailed screenshot specifications from a short app description (uses GPT-4.1). Only needed for the generate-screenshots command.
How to get your key:
- Sign up at platform.openai.com
- Go to API Keys and create a new key
kappmaker config set openaiApiKey <your-key>— or prompted on first use
App Store Connect CLI (asc) — iOS App Management & Publishing
Used for: Creating apps, setting metadata, categories, subscriptions, privacy declarations, and review info on App Store Connect (create-appstore-app). The same API key credentials are also used by publish --platform ios to build and upload IPAs via Fastlane.
How to set up:
- Install:
brew install asc - Generate an API key at App Store Connect > Users and Access > Integrations > API (Admin role, download the
.p8file immediately) - Configure:
Or runkappmaker config set ascKeyId <your-key-id> kappmaker config set ascIssuerId <your-issuer-id> kappmaker config set ascPrivateKeyPath /path/to/AuthKey.p8 kappmaker config set appleId [email protected]kappmaker config appstore-defaults --initfor interactive setup.
Note:
kappmaker publish --platform iosusesascKeyId,ascIssuerId, andascPrivateKeyPathto automatically generate the Fastlane-format publisher JSON — no separate credentials needed.
Adapty CLI — Subscription Management
Used for: Setting up in-app subscription products, paywalls, and placements across iOS and Android via Adapty's backend.
How to set up:
- Install:
npm install -g adapty - Log in:
adapty auth login(opens browser for authentication) - Run:
kappmaker adapty setup
Firebase CLI — Backend Setup
Used for: Creating Firebase projects, registering Android/iOS apps, downloading SDK config files (google-services.json, GoogleService-Info.plist), and enabling anonymous authentication.
How to set up:
- Install:
npm install -g firebase-tools - The
createcommand handles login and project creation interactively.
Google Play Publisher — Android Store Uploads + Play Console Management
Used for:
- Building and uploading Android AABs to Google Play Store via
kappmaker publish --platform android(Fastlane) - Configuring store listings, subscriptions, in-app products, and the data safety declaration via
kappmaker gpc ...(direct Publisher API call, no external CLI)
Both flows share the same service account JSON key — set it once, use it everywhere.
How to set up:
- Go to Google Cloud Console and create a new project (or select existing)
- Open APIs & Services > Library, search for Google Play Android Developer API, and enable it
- Go to IAM & Admin > Service Accounts, create a new service account (skip role assignment)
- Open the service account, go to Keys, click Add key > Create new key > JSON, and download it
- Open Google Play Console, go to Settings > Users and permissions
- Click Invite new user with the service account email (
[email protected]) and grant permissions for your app(s) - Save the JSON key file and configure:
kappmaker config set googleServiceAccountPath /path/to/google-service-app-publisher.json
Note: Google Play does not allow creating new apps via any public API — you must create the app manually once in Play Console before
kappmaker gpccan configure it.
App Store Publisher — iOS Store Uploads
Used for: Building and uploading iOS IPAs to App Store Connect, managing App Store metadata and screenshots via kappmaker publish --platform ios.
The publish command reuses the same App Store Connect API key credentials used by create-appstore-app (ascKeyId, ascIssuerId, ascPrivateKeyPath) and automatically generates the Fastlane-format publisher JSON.
How to set up (if not already configured for create-appstore-app):
- Open App Store Connect > Users and Access > Integrations
- Create an API key with App Manager access and download the
.p8file - Note the Key ID and Issuer ID
- Configure:
kappmaker config set ascKeyId <your-key-id> kappmaker config set ascIssuerId <your-issuer-id> kappmaker config set ascPrivateKeyPath /path/to/AuthKey.p8
create <app-name>
Full end-to-end app bootstrapping. Creates a new KAppMaker app from the template and optionally sets up everything needed to publish.
kappmaker create RemimiWhat it does (13 steps):
| Step | Action | Details |
|------|--------|---------|
| 1 | Clone template | Clones into <AppName>-All/ (prompts to overwrite if exists) |
| 2 | Firebase login | Opens browser for authentication |
| 3 | Create Firebase project | <appname>-app (skips if exists) |
| 4 | Create Firebase apps | Android + iOS apps (reuses existing if found) |
| 5 | Enable anonymous auth | If brand-new project, prompts user to click "Get started" in Firebase Console, then enables via API |
| 6 | Download SDK configs | google-services.json + GoogleService-Info.plist (verifies package match, falls back to Assets/) |
| 7 | Logo generation | Optional — AI logo + automatic background removal |
| 8 | Package refactor | Renames packages, IDs, app name across all modules (shared, androidApp, desktopApp, webApp, designsystem, libs; also walks legacy composeApp/ for pre-rename projects) |
| 9 | Build environment | local.properties, generates signing keystore if missing (iOS deps resolve via SwiftPM) |
| 10 | Git remotes | Renames origin to upstream |
| | Pre-store reminder | Prompts user to create Google Play Console app; App Store Connect is created automatically |
| 11 | App Store Connect | Optional — full app setup (metadata, subs, privacy); app created automatically via asc web apps create |
| 12 | Google Play Console | Optional — Fastlane builds + uploads AAB to internal track, then runs full gpc setup |
| 13 | Adapty setup | Optional — products, paywalls, placements (links to ASC + Play products created in 11-12) |
Options:
| Flag | Description | Default |
|------|-------------|---------|
| --template-repo <url> | Template repository URL | KAppMaker template |
| --organization <org> | Organization for Fastlane signing | App name (configurable) |
clone <app-name>
Clones the template repository into <AppName>-All. This is step 1 of create exposed as a standalone command — useful when you only want to scaffold a project and apply your own changes (e.g. clone + refactor without Firebase, ASC, etc.).
kappmaker clone Remimi
kappmaker clone Remimi --template-repo [email protected]:my-org/my-template.git
kappmaker clone Remimi --target-dir ./projects/RemimiWhat it does:
- Validates the app name (PascalCase)
- If no config exists at
~/.config/kappmaker/config.json, runskappmaker config initfirst - If the target directory already exists, prompts to delete and start fresh
- Runs
git clone <templateRepo> <targetDir>
Minimal flow — clone + refactor without anything else:
kappmaker clone MyApp
cd MyApp-All/MobileApp
kappmaker refactor --app-id com.example.myapp --app-name MyApp| Flag | Description | Default |
|------|-------------|---------|
| --template-repo <url> | Git URL of the template repository | templateRepo from config |
| --target-dir <path> | Target directory for the clone | <AppName>-All |
git setup-upstream
Renames the origin remote to upstream. Step 10 of create as a standalone command — designed for the "fork from template" workflow where the template repo becomes upstream and you add your own origin later.
kappmaker git setup-upstream # Run from inside the cloned repo
kappmaker git setup-upstream ./MyApp-All # Or pass the path explicitlyExits non-zero if the path is not a git repository.
| Argument | Description | Default |
|----------|-------------|---------|
| [path] | Path to the repo root | Current directory |
firebase
Five standalone subcommands matching steps 2–6 of create. Run them individually for partial flows (e.g. set up Firebase for an existing project), or together to replicate what create does.
kappmaker firebase login
kappmaker firebase project --app-name MyApp # derives project-id = myapp-app, display = MyApp
kappmaker firebase project --project-id myapp-prod --display-name MyApp # explicit
kappmaker firebase apps --project myapp-app --app-name MyApp --package-name com.example.myapp
kappmaker firebase auth-anonymous --project myapp-app
kappmaker firebase configs --project myapp-app --app-name MyApp --package-name com.example.myappfirebase login
Runs firebase login (interactive). No args.
firebase project
Creates the Firebase project. Idempotent — if the project already exists it skips creation. Returns success/failure to the orchestrator (controls whether create proceeds with steps 4–6).
| Flag | Description | Required |
|------|-------------|----------|
| --project-id <id> | Firebase project ID (e.g. myapp-app) | Yes, unless --app-name is set |
| --display-name <name> | Project display name | Defaults to --project-id or --app-name |
| --app-name <name> | PascalCase app name; derives project-id = <lowercase>-app and display-name = <name> | Yes, unless explicit IDs are passed |
firebase apps
Creates an Android + iOS app under the project. Idempotent — reuses existing apps that match the display name ${appName} (Android App) / ${appName} (iOS App).
| Flag | Description | Required |
|------|-------------|----------|
| --project <id> | Firebase project ID | Yes |
| --app-name <name> | PascalCase display name (used to label the apps) | Yes |
| --package-name <pkg> | Android applicationId and iOS bundle ID (e.g. com.example.myapp) | Yes |
firebase auth-anonymous
Enables anonymous authentication via the Identity Toolkit Admin API. If Firebase Auth has never been initialized for the project, the command pauses and asks you to click "Get started" in the Firebase Console, then retries automatically.
| Flag | Description | Required |
|------|-------------|----------|
| --project <id> | Firebase project ID | Yes |
firebase configs
Downloads google-services.json and GoogleService-Info.plist for the Android + iOS apps and writes them to the right place. Apps are looked up via --app-name (display name match) unless you pass --android-app-id / --ios-app-id directly.
Output auto-detection (when --android-output / --ios-output aren't given): probes MobileApp/androidApp/google-services.json, then MobileApp/composeApp/google-services.json, then falls back to Assets/google-services.json. Same for iOS — MobileApp/iosApp/iosApp/GoogleService-Info.plist if it exists, otherwise Assets/GoogleService-Info.plist.
When --package-name is provided, the downloaded google-services.json is verified against the expected package name and patched in-place if it doesn't match (handles cases where the Firebase app was registered with an older bundleIdPrefix).
| Flag | Description | Required |
|------|-------------|----------|
| --project <id> | Firebase project ID | Yes |
| --app-name <name> | PascalCase display name (used to find the apps if app IDs aren't given) | Yes |
| --package-name <pkg> | Verify and fix the Android google-services.json package name | No |
| --android-app-id <id> | Skip lookup and use this Firebase App ID directly | No |
| --ios-app-id <id> | Skip lookup and use this Firebase App ID directly | No |
| --android-output <path> | Output path for google-services.json | Auto-detect |
| --ios-output <path> | Output path for GoogleService-Info.plist | Auto-detect |
create-logo
Generates an app logo using fal.ai's nano-banana-2 model.
kappmaker create-logo
kappmaker create-logo --prompt "A minimalist fitness tracker for runners"
kappmaker create-logo --output ./custom/path/logo.pngFlow:
- Reads app idea from
--prompt(or prompts interactively if omitted) - Generates a 4x4 grid of 16 logo variations (2K, 1:1)
- Opens grid in Preview.app for review
- Pick a logo (1-16) or R to regenerate — optional:
5 --zoom 1.1 --gap 3 - Extracts chosen logo at 512x512
Output: Assets/app_logo.png + Assets/logo_variations.png
Requires a fal.ai API key (prompted on first use if not set, or set manually: kappmaker config set falApiKey <your-key>)
| Flag | Description | Default |
|------|-------------|---------|
| --prompt <text> | App idea / concept (skips the interactive prompt) | — |
| --output <path> | Custom output path | Assets/app_logo.png |
generate-image
Generic AI image generator — a thin wrapper around fal.ai's nano-banana-2 (text-to-image) and nano-banana-2/edit (with reference images). Use this any time you need a one-off image without the logo-grid selection flow.
kappmaker generate-image --prompt "A minimalist mountain landscape at sunset"
kappmaker generate-image --prompt "Hero banner for a meditation app" --aspect-ratio 16:9 --resolution 4K
kappmaker generate-image --prompt "Product render" --num-images 4 --output Assets/hero
kappmaker generate-image --prompt "Put this logo on a black t-shirt" --reference Assets/app_logo.png
kappmaker generate-image --prompt "Blend these into a single composition" --reference Assets/moodboard
kappmaker generate-image --prompt "Match this product shot" --reference https://example.com/ref.jpgReference images (edit mode): Passing --reference switches the endpoint from nano-banana-2 (text-to-image) to nano-banana-2/edit (reference-guided). Each entry can be a file path, a directory (all png/jpg/jpeg/webp inside are picked up, sorted, non-recursive), or an HTTP(S) URL. Max 10 references total. If imgbbApiKey is configured, local files are uploaded to imgbb for reliable URLs; otherwise they are sent inline as data URIs (fine for small images, can fail on very large ones).
Output path behavior:
- No
--output→Assets/generated.png(single) orAssets/generated_1.png,_2.png… (multi) --output <dir>(no extension) → saves into that directory--output <file.png>→ single image uses path verbatim; multi-image appends_1,_2before extension
Requires a fal.ai API key (prompted on first use if not set).
| Flag | Description | Default |
|------|-------------|---------|
| --prompt <text> | Text description of the image (required) | — |
| --output <path> | Output file or directory | Assets/generated.png |
| --num-images <n> | Number of images to generate (1–8) | 1 |
| --aspect-ratio <ratio> | 1:1, 16:9, 9:16, 4:3, 3:4, 3:2, 2:3, 21:9, 9:21, auto | 1:1 |
| --resolution <res> | 1K, 2K, 4K | 2K |
| --output-format <fmt> | png, jpg, webp | png |
| --reference <paths...> | Reference images — file paths, directories, or HTTP URLs (edit mode, max 10) | — |
create-appstore-app
Creates and fully configures an app on App Store Connect using the asc CLI. Requires asc ≥ 1.4.0.
kappmaker create-appstore-app
kappmaker create-appstore-app --config ./my-config.jsonFirst-time setup
- Generate an API key at App Store Connect > Users and Access > Integrations > API — Admin access, download the
.p8file immediately - Run one-time setup:
kappmaker config appstore-defaults --init
What it does (13 steps)
- Validate asc CLI and authentication
- Load config (from file or interactive prompts)
- Register Bundle ID + enable capabilities (Sign in with Apple, In-App Purchases, Push Notifications). Name follows Apple convention:
XC com measify myappforcom.measify.myapp - Find or create app (fully automated via
asc web apps create— no manual App Store Connect step needed) - Set content rights
- Create app version (1.0.0)
- Set categories
- Set age rating
- Update localizations
- Set pricing, availability, and subscriptions
- Set privacy data usages
- Set encryption declarations
- Set review contact details
Config resolution
Layers are deep-merged (later overrides earlier):
- Built-in template — age rating, privacy, encryption, subscriptions
- Global defaults (
~/.config/kappmaker/appstore-defaults.json) — review contact, copyright - Local config (
./Assets/appstore-config.jsonor--config) - Interactive prompts — only for fields still empty
Default subscriptions
| Subscription | Period | Price | Product ID |
|-------------|--------|-------|------------|
| Weekly Premium | ONE_WEEK | $6.99 | {appname}.premium.weekly.v1.699.v1 |
| Yearly Premium | ONE_YEAR | $29.99 | {appname}.premium.yearly.v1.2999.v1 |
Auto-generated naming: group {appname}.premium.v1, ref name {AppName} Premium Weekly v1 (6.99).
Per-territory PPP pricing (1.7.0+ — bulk CSV import)
Subscriptions and IAPs are fanned out to every ASC territory (~175) with purchasing-power-parity-adjusted prices in a single API call per product. Requires asc CLI ≥ 1.4.0 (brew upgrade asc).
- Subscriptions (1.7.0+):
asc subscriptions pricing prices import --input <csv>. KAppMaker writes a temp CSV withterritory,price,price_point_idrows for all ~175 territories and pipes it in. Replaces the 155-call loop from 1.6.x which hit Apple's rate limits. - IAPs: a single
asc iap pricing schedules create --prices "PP_ID:DATE,…"call covers all territories (unchanged).
Set "ppp_enabled": false on any subscription or IAP to opt out and keep only your explicitly-listed prices. User-listed territory prices are always preserved (they win the merge).
Tier resolution: Apple's price-point catalog uses globally-stable tier numbers (1..800 — tier N = the same USD-equivalent across every territory). KAppMaker resolves each unique PPP USD target → tier ONCE via USA's catalog (where customerPrice is in USD), then synthesises per-territory price-point IDs locally using Apple's base64 {s, t, p} format. Critical fix vs 1.6.x: the old code compared a USD target against local-currency price-points (e.g. ¥, ₩, ₹) and ended up picking the FREE tier — products silently landed at $0 in JPN/IDR/INR/KRW/etc. The new tier-based resolution avoids that entirely.
Distinct catalogs: subscriptions use subscriptionPricePoints (CLI: asc subscriptions pricing price-points list) while IAPs use appPricePoints (CLI: asc pricing price-points). Their s (catalog identifier) fields differ — mixing IDs across catalogs returns 400 The provided entity is invalid.
Idempotent re-runs: when setup reports "already been used", KAppMaker now logs existing — refreshing pricing and continues into the PPP fan-out so re-runs actually update territory pricing (1.6.x silently skipped).
App Review screenshots (1.7.1+)
Apple requires a review screenshot on every subscription and IAP — without one, products remain in MISSING_METADATA state and per-territory pricing won't "resolve". KAppMaker uploads it automatically when you set review_screenshot in the config:
{
"review_screenshot": "Assets/appstore/review-screenshot.png",
"subscriptions": {
"groups": [{
"subscriptions": [{
"ref_name": "Premium Weekly",
"review_screenshot": "Assets/appstore/weekly-review.png"
}]
}]
},
"in_app_purchases": [{
"product_id": "credit_pack_10_499_myapp",
"review_screenshot": "Assets/appstore/iap-review.png"
}]
}- Global default:
review_screenshotat the top level applies to every sub + IAP. - Per-product override: each subscription / IAP can specify its own path.
- Idempotent: re-runs check
view/images listfirst and skip if a screenshot is already attached. Delete via Apple's UI to force a re-upload. - Silent skip on missing file: if the path doesn't resolve to an existing file, KAppMaker logs and moves on — no error.
Under the hood: asc subscriptions review screenshots create --file <path> for subs and asc iap images create --file <path> for IAPs.
Required size + auto-resize prompt
Apple's recommended size for App Review screenshots is 1290 × 2796 px (iPhone 6.7" Display, portrait — matches App Store listing screenshot dimensions). Minimum: 640 × 920 px. Format: PNG or JPG.
If the file's dimensions don't match 1290 × 2796, KAppMaker prompts:
WARN Review screenshot wrong-size.png is 1920×1080.
-- Apple's App Store recommended size for review screenshots: 1290×2796 (iPhone 6.7" Display, portrait).
Resize to 1290×2796 keeping aspect ratio? (Y/n)- Y (default) — sharp resizes the file with
fit: 'inside'(preserves aspect ratio, may end up smaller than the target on the constraining dimension; e.g. a 1920 × 1080 input becomes 1290 × 726). Writes to a temp file. Uploads the resized copy. - N — uploads the file as-is. Apple may still accept it (the rule is just
min 640 × 920) but the aspect ratio won't match typical iPhone screenshots.
Files that are already 1290 × 2796 skip the prompt entirely.
Standalone replace commands
Two top-level appstore- prefixed commands replace existing screenshots without running the full create-appstore-app flow:
# Replace the review screenshot on EVERY subscription in the config
kappmaker appstore-update-subscription-review-screenshot --file ./Assets/appstore/new-review.png
# Replace the review image on EVERY IAP
kappmaker appstore-update-iap-review-screenshot --file ./Assets/appstore/iap-review.png
# Without --file, the commands use the per-product `review_screenshot` from the config
kappmaker appstore-update-subscription-review-screenshot
kappmaker appstore-update-iap-review-screenshot
# Target a single product
kappmaker appstore-update-iap-review-screenshot \
--file ./Assets/appstore/credit-pack-30.png \
--product-id credit_pack_30_999_myapp| Flag | Description |
|---|---|
| --file <path> | Single screenshot applied to all matched products. Overrides per-product review_screenshot from the config. |
| --config <path> | Override default ./Assets/appstore-config.json. |
| --product-id <id> | Target ONE product (matches by product_id or ref_name). |
These commands force-replace existing screenshots (delete + create under the hood — empirically asc … update doesn't actually swap the binary). The setup-flow upload in create-appstore-app remains idempotent (skips when one is already attached).
Default in-app purchases (credit packs)
The template also ships three CONSUMABLE in-app purchases shaped as credit packs. Auto-fill triggers on any in_app_purchases[] entry with a credits numeric field.
| Pack | Credits | Price | Product ID |
|------|---------|-------|------------|
| Basic Credit Pack | 10 | $4.99 | credit_pack_10_499_{appname} |
| Pro Credit Pack | 30 | $9.99 | credit_pack_30_999_{appname} |
| Ultimate Credit Pack | 80 | $19.99 | credit_pack_80_1999_{appname} |
Format: credit_pack_{credits}_{priceDigits}_{appname}. The same ID is used on Google Play and Adapty so app code can reference one constant per platform pair.
Created via the same one-shot asc iap setup workflow that drives subscription setup. Reruns are idempotent — already-existing product IDs are skipped with an info log.
Default privacy
| Data Category | Purpose | Protection | |--------------|---------|------------| | User ID | App Functionality | Linked to You | | Device ID | App Functionality | Linked to You | | Crash Data | Analytics | Not Linked to You | | Performance Data | Analytics | Not Linked to You | | Other Diagnostic Data | Analytics | Not Linked to You | | Other Usage Data | Analytics | Not Linked to You | | Product Interaction | Analytics | Not Linked to You |
During interactive setup, the CLI asks if the app accesses user content (AI image/video wrapper). If yes, adds Photos or Videos + Other User Content (both App Functionality / Not Linked to You).
| Flag | Description | Default |
|------|-------------|---------|
| --config <path> | Path to JSON config file | ./Assets/appstore-config.json |
appstore-monetization-push
Standalone command that re-runs only the subscription + IAP push step from create-appstore-app. Useful when you want to sync pricing (including PPP fan-out) without repeating the full 13-step setup — e.g. after adding a new subscription period, adjusting prices, or upgrading to a newer KAppMaker version with PPP fixes.
# Push everything (subscriptions + IAPs)
kappmaker appstore-monetization-push
# Subscriptions only
kappmaker appstore-monetization-push --subscriptions-only
# IAPs only
kappmaker appstore-monetization-push --iap-only
# Custom config path
kappmaker appstore-monetization-push --config ./path/to/appstore-config.jsonReads Assets/appstore-config.json (or --config <path>). Resolves the ASC app ID from app.id or by looking up app.bundle_id via asc. Calls the same setupSubscriptions / setupInAppPurchases functions as create-appstore-app — fully idempotent: existing products get pricing refreshed with the full PPP regional fan-out.
| Flag | Description | Default |
|------|-------------|---------|
| --config <path> | Path to JSON config file | ./Assets/appstore-config.json |
| --subscriptions-only | Push subscription groups only, skip IAPs | — |
| --iap-only | Push IAPs only, skip subscription groups | — |
gpc
Google Play Console management — a tight wrapper around the official Google Play Android Publisher API. All subcommands authenticate via the service account JSON at googleServiceAccountPath (same one used by publish --platform android). No external CLI and no extra npm dependencies — the JWT flow and HTTPS calls are implemented with Node's built-in crypto and fetch.
Note: Google Play does not allow creating new apps via any public API. Create the app manually once in Play Console, then use these commands to configure it.
gpc setup
Full end-to-end configuration (11 steps) — the Google Play parallel to create-appstore-app.
kappmaker gpc setup
kappmaker gpc setup --config ./my-config.jsonWhat it does:
- Validate service account + obtain access token
- Load config (
./Assets/googleplay-config.jsonor interactive prompts). Auto-detects Play's actual default language via a throwaway edit probe and rewritesconfig.app.default_language+ the matching listing locale if your config drifted from reality. Also migrates legacy product IDs and legacy data-safety placeholders. Also reads tracks state to verify an APK/AAB is uploaded. - Review summary and confirm
- Verify app exists on Play Console (fails fast with a deep link if not — Google does not allow app creation via the public API)
- Start a Play Console edit (only if the default-language listing has a title — otherwise steps 5-7 are skipped cleanly)
- Update app details (default language + contact website/email/phone)
- Update store listings per locale (title, short/full description, video); commits the edit. Empty listing titles are auto-filled from the app name
- Pre-flight build check, then create subscriptions via the new monetization API (
POST /subscriptionswithproductId+regionsVersion.versionas query params, base plans in body in DRAFT state, thenbasePlans:activateeach one). Base plans are available to all ~175 Play regions — regions without explicit prices get auto-converted pricing from Google. Idempotent — existing product IDs are skipped. Subscription listings are auto-cloned into Play's default language if missing. Skipped with a clear message if no build is uploaded to any track (Google rejects monetization writes in that case). - Create one-time in-app products via the new monetization API (
PATCH /onetimeproducts/{id}?allowMissing=true+purchaseOptions:batchUpdateStatesto activate). Idempotent viaGET /oneTimeProducts. Replaces the legacy/inappproductsendpoint that returns 403 on migrated apps. - Update the data safety declaration: converts
data_safetyJSON → Google's CSV format using a bundled canonical template + KAppMaker defaults (USER_ID for app functionality + account management; DEVICE_ID for app functionality + advertising + account management; crash/performance/diagnostics/interactions for analytics), then posts it viaPOST /dataSafetywith{ safetyLabels: "<csv>" }. Respectsdata_safety_csv_pathas an escape hatch for pre-exported CSVs. - Print a checklist of policy declarations that the Publisher API does NOT expose (content rating / IARC, target audience, ads, health apps, financial features, government apps, news apps, gambling, COVID-19 tracing, app access, advertising ID, families compliance, app pricing tier) with a deep link to Play Console's App content page. All verified against Google's v3 discovery document — none have REST endpoints.
gpc listings push
Push just the store listings section from the config file (useful after editing copy).
kappmaker gpc listings push
kappmaker gpc listings push --config ./my-config.jsonRuns a single edit transaction: updates app details → updates every listing locale → commits.
gpc subscriptions list
Read-only — lists existing subscription product IDs on Play Console.
kappmaker gpc subscriptions list --package com.example.myapp
kappmaker gpc subscriptions list # uses app.package_name from the config filegpc subscriptions push
Create or reuse subscriptions from the config file. Idempotent — already-existing product IDs are skipped, and base plans are activated even for reused subscriptions.
kappmaker gpc subscriptions pushUses the new monetization API (applications/{pkg}/subscriptions + basePlans:activate). Auto-generated IDs:
| Field | Format | Example ($6.99 weekly) |
|---|---|---|
| productId (subscription) | {appname}.premium.{period}.v1 | myapp.premium.weekly.v1 |
| basePlanId | autorenew-{period}-{priceDigits}-v1 | autorenew-weekly-699-v1 |
| Subscription title | {AppName} Premium {PeriodLabel} | MyApp Premium Weekly |
These align 1-to-1 with what adapty setup writes to android_product_id and android_base_plan_id, so Adapty links the products automatically without any extra configuration.
gpc iap list
Read-only — lists existing one-time in-app product SKUs on Play Console.
kappmaker gpc iap list --package com.example.myapp
kappmaker gpc iap listgpc iap push
Create or reuse one-time in-app products from the config file. Idempotent — existing products are PATCHed with fresh PPP regional pricing.
kappmaker gpc iap push
kappmaker gpc iap push --recreate-stuck # DELETE + recreate products stuck at regionsVersion 2022/02| Flag | Description |
|---|---|
| --config <path> | Override the default config path (./Assets/googleplay-config.json) |
| --recreate-stuck | DELETE products whose stored regions can't coexist with regionsVersion=2022/02 (e.g. legacy products with MN) so they recreate fresh. WARNING: Google holds the productId in a soft-delete reservation window for a few minutes to a few hours after deletion — CREATE during that window fails with "Product ID already in use". Use only when you can tolerate downtime, or prefer bumping the product_id in config (e.g. v1 → v2) for zero downtime. |
Uses the new monetization API: PATCH /applications/{pkg}/onetimeproducts/{id}?allowMissing=true to create/update the product, then purchaseOptions:batchUpdateStates with an activatePurchaseOptionRequest to activate the default purchase option so it's available to buyers. Replaces the legacy /inappproducts endpoint, which Google now rejects with "Please migrate to the new publishing API" on migrated apps.
Default credit packs ship in Assets/googleplay-config.json. Auto-fill triggers on any in_app_products[] entry with a credits numeric field — SKU is set to credit_pack_{credits}_{priceDigits}_{appname} (matches the ASC and Adapty IDs):
| Pack | Credits | Price | SKU |
|------|---------|-------|-----|
| Basic Credit Pack | 10 | $4.99 | credit_pack_10_499_{appname} |
| Pro Credit Pack | 30 | $9.99 | credit_pack_30_999_{appname} |
| Ultimate Credit Pack | 80 | $19.99 | credit_pack_80_1999_{appname} |
Global region availability with PPP pricing (1.6.0+): subscriptions and one-time products are both created with explicit per-region prices in every Play-supported region (~175), with purchasing-power-parity multipliers applied — IN gets ~0.35×, AR/PK/EG get 0.30×, US/CA/EU base at 1.00×, CH/NO at 1.10×. The multipliers come from a Steam/Spotify-inspired tier table (iosdevmax/ppp-pricing, MIT). Prices round to .99 endings.
| Override | Result |
|---|---|
| You list { region_code: "DE", price: "5.99", currency_code: "EUR" } | Used as-is for DE; PPP fills the rest |
| You list only US/USD | PPP fans out from your USD anchor to ~175 regions |
| Set "ppp_enabled": false on a base plan or IAP | That entry stays restricted to your listed regions |
Each region's price is in its native currency (AE → AED, JP → JPY, IN → INR, etc.) — Google Play rejects mismatched currencies. The CLI calls Google's convertRegionPrices endpoint with your USD base price to get fair FX-converted native prices for every billable region, then multiplies by the region's PPP multiplier and charm-rounds (.99 for decimal currencies, X99 / X9 for zero-decimal currencies like JPY / KRW). Sanctioned countries (AF, IR, KP, SY, etc.) are auto-excluded — they don't appear in convertRegionPrices output.
regionsVersion=2022/02 drift override (1.6.11+): Google's monetization API requires regionsVersion.version and "2022/02" is the only documented value. Several regions have drifted since that snapshot was taken — Google's live API returns one currency but 2022/02 expects another. The CLI handles this with an explicit override table:
| Region | Live API | 2022/02 expects | What CLI does | |---|---|---|---| | BG | EUR | BGN | Converts EUR → BGN via the 1 EUR = 1.95583 BGN peg | | HR | EUR | EUR ✓ | No override (Google updated 2022/02 retroactively) | | CI / CM / SN | XOF / XAF | USD | Uses the USD anchor (PPP multiplier still applied) | | MN | billable | NOT BILLABLE | Skipped entirely (no currency works) |
Net effect: typical products land with all ~173 billable regions AVAILABLE (only MN drops out). Google's storage layer auto-converts each region's submitted currency to the user-facing display currency (we send BG/BGN → end users see EUR; we send CI/USD → end users see XOF).
Existing products with regions stuck at 2022/02: if Google's old catalog has stored MN (or another NEVER_BILLABLE region) on an existing product, that PATCH cannot be completed — Google won't accept MN's currency but also refuses to remove the region. The CLI surfaces these "stuck" products at the end of the run and offers three fix paths:
- Recommended — bump the
product_idin your config (e.g.credit_pack_10_499_myapp→credit_pack_10_499_myapp_v2). Fresh product, no legacy baggage, no waiting period. --recreate-stuckflag — DELETEs the stuck product(s) and recreates them fresh. WARNING: Google soft-deletes for a few minutes to a few hours; CREATE during that window returns "Product ID already in use". Plan downtime.- Manually delete on Play Console UI, wait for the reservation window to clear, then re-run.
"New countries and regions" set to AVAILABLE (1.6.11+): both subscriptions (otherRegionsConfig) and one-time products (newRegionsConfig) are now always set to availability: AVAILABLE with USD + EUR anchors when a USD anchor is present. Means any region Google adds to its billable catalog in the future is auto-priced from those anchors — no future-proofing maintenance needed.
Want simpler pricing without PPP? Set "ppp_enabled": false on a base plan or one-time product to skip the PPP fan-out. The CLI then submits only your user-listed regions in regional_configs (typically just US) plus otherRegionsConfig (subscriptions) / newRegionsConfig (one-time products) with USD + EUR anchors. Google handles the fan-out via its own pricing-template FX algorithm — every billable region gets a price, but without PPP discounting in lower-income markets. Trade-off: smaller payload, less control.
gpc data-safety push
Push only the data safety declaration. Faster than running the full setup when you're iterating on the privacy answers.
kappmaker gpc data-safety pushHow it works under the hood: Google's POST /applications/{pkg}/dataSafety does not take structured JSON — it takes a CSV file (the same one Play Console exports from Policy → App content → Data safety → Export to CSV), wrapped in { safetyLabels: "<csv>" }. KAppMaker lets you stay in JSON and converts internally.
Two ways to configure it in Assets/googleplay-config.json:
Structured JSON (recommended) —
data_safety.answersoverlay on top of KAppMaker defaults:Account creation: "My app does not allow users to create an account" (
PSL_ACM_NONE).Data deletion: skipped entirely (the question is optional).
Data types collected:
| Data type | Play Question / Response | |---|---| | User IDs |
PSL_DATA_TYPES_PERSONAL/PSL_USER_ACCOUNT| | Device ID |PSL_DATA_TYPES_IDENTIFIERS/PSL_DEVICE_ID| | Crash logs |PSL_DATA_TYPES_APP_PERFORMANCE/PSL_CRASH_LOGS| | Diagnostics |PSL_DATA_TYPES_APP_PERFORMANCE/PSL_PERFORMANCE_DIAGNOSTICS| | Other performance |PSL_DATA_TYPES_APP_PERFORMANCE/PSL_OTHER_PERFORMANCE| | App interactions |PSL_DATA_TYPES_APP_ACTIVITY/PSL_USER_INTERACTION|Data handling (same for ALL data types above):
- Collected only, not shared with third parties
- Processed ephemerally (YES)
- Collection is required — users can't turn it off
- Purposes: Analytics (+ App functionality for Device ID)
Security: encrypted in transit = YES.
Example config (just use the default — override if you need to add data types or change handling):
"data_safety": { "apply_defaults": true, "answers": {} }To override a specific answer, add keys to
answers. Keys are eitherQuestionID(for single-answer rows) orQuestionID/ResponseID(for multi-choice rows). Values aretrue/false/ URL string /null:"data_safety": { "apply_defaults": true, "answers": { "PSL_DATA_TYPES_PERSONAL/PSL_USER_ACCOUNT": true, "PSL_DATA_TYPES_LOCATION/PSL_APPROX_LOCATION": true } }Internally, KAppMaker loads a canonical 783-row template (extracted from the well-maintained fastlane-plugin-google_data_safety), applies defaults + your overrides, emits a filled CSV, and uploads it via
POST /dataSafety.Escape hatch: CSV file — if you already exported a real CSV from Play Console and filled it in:
"data_safety_csv_path": "Assets/data-safety.csv"When this field is set and the file exists, it takes priority over the JSON block and is uploaded verbatim.
Important: Review the summary Play Console shows after the push before publishing. Google's conditional question logic may reject some answers — paste the error and I'll adjust the defaults.
gpc app-check
Quick read-only probe to verify that an app exists on Play Console. Useful for CI scripts and before running destructive operations.
kappmaker gpc app-check --package com.example.myappExits 0 if found, 2 if missing (prints the Play Console deep link).
Config
All gpc subcommands except app-check load ./Assets/googleplay-config.json by default. The first run of kappmaker gpc setup creates it interactively.
| Flag | Description | Default |
|------|-------------|---------|
| --config <path> | Path to JSON config file | ./Assets/googleplay-config.json |
| --package <name> | (list commands) Override the package name from config | — |
Alias: kappmaker create-play-app is kept as a shortcut for kappmaker gpc setup (the entire 11-step flow is also what the top-level kappmaker create calls under step 8 when you answer yes to the Google Play Console prompt).
gpc monetization push
Standalone command that re-runs only the subscription + IAP push steps from gpc setup. Useful when you want to sync pricing (including full PPP fan-out across ~175 regions) without repeating listings, data safety, or the manual-checklist prompt.
# Push everything (subscriptions + IAPs)
kappmaker gpc monetization push
# Subscriptions only
kappmaker gpc monetization push --subscriptions-only
# IAPs only
kappmaker gpc monetization push --iap-only
# Custom config path
kappmaker gpc monetization push --config ./path/to/googleplay-config.json
# Re-create products stuck due to regionsVersion 2022/02 drift
kappmaker gpc monetization push --recreate-stuckReads Assets/googleplay-config.json (or --config <path>). Validates the service account and probes the app state (requires at least one uploaded build — internal testing track is enough). Calls the same setupSubscriptions / setupInAppProducts functions as gpc setup — fully idempotent: existing products are PATCHed with refreshed PPP regional fan-out.
| Flag | Description | Default |
|------|-------------|---------|
| --config <path> | Path to JSON config file | ./Assets/googleplay-config.json |
| --subscriptions-only | Push subscriptions only, skip IAPs | — |
| --iap-only | Push IAPs only, skip subscriptions | — |
| --recreate-stuck | Delete + recreate products stuck due to regionsVersion 2022/02 incompatibility | — |
subscription add
Quick-add a single new subscription to Google Play and App Store Connect without editing config files. Auto-generates aligned product IDs across both stores, fans out PPP pricing across ~175 Play regions and ~155 ASC territories, and auto-creates the App Store Connect subscription group if it doesn't exist yet.
# Minimum invocation — defaults to --platform all (Play + ASC)
kappmaker subscription add --period weekly --price 9.99
# v2 product line alongside an existing v1
kappmaker subscription add --period weekly --price 9.99 --product-version 2
# Single store
kappmaker subscription add --period monthly --price 19.99 --platform android
kappmaker subscription add --period yearly --price 29.99 --platform ios
# Full control — fully spelled out
kappmaker subscription add \
--period weekly --price 6.99 --product-version 2 \
--name "Weekly Premium" \
--description "Full access for one week." \
--review-screenshot "Assets/appstore/review-screenshot_subscription.jpg" \
--group "myapp.premium.v2" --group-name "Premium Access" \
--app-name "MyApp"What it creates (for --period weekly --price 9.99 --product-version 1):
| Where | Field | Value |
|---|---|---|
| ASC | product_id | myapp.premium.weekly.v1.999.v1 |
| ASC | ref_name | MyApp Premium Weekly v1 (9.99) |
| ASC | subscription period | ONE_WEEK |
| ASC | localization (en-US) | name = MyApp Premium Weekly, description = Full access for one week. |
| ASC | pricing | $9.99 USD anchor → PPP fan-out across ~155 territories in one prices import CSV call |
| ASC | review screenshot | top-level config.review_screenshot (or --review-screenshot) |
| ASC | subscription group | first group in config, OR --group <ref> (auto-created with en-US name if new) |
| Play | product_id | myapp.premium.weekly.v1 |
| Play | base plan | autorenew-weekly-999-v1, period P1W |
| Play | listing (en-US) | title = MyApp Premium Weekly, description = Full access for one week. |
| Play | pricing | $9.99 USD anchor → PPP fan-out across ~173 billable regions via convertRegionPrices |
Flags:
| Flag | Required | Default |
|---|---|---|
| --period <slug> | yes | — — one of weekly / monthly / twomonths / quarterly / semiannual / yearly |
| --price <number> | yes | — — USD anchor; PPP fans the rest |
| --platform <target> | no | all (= Play + ASC) — ios = ASC only, android = Play only |
| --product-version <n> | no | 1 — bumps every v marker, e.g. --product-version 2 → myapp.premium.weekly.v2.999.v2 + myapp.premium.weekly.v2 + autorenew-weekly-999-v2 |
| --name <text> | no | "<AppName> Premium <Period>" — localized display name |
| --description <text> | no | period-derived sentence: weekly → "Full access for one week.", monthly → "Full access for one month.", etc. |
| --review-screenshot <path> | no | top-level config.review_screenshot from Assets/appstore-config.json |
| --group <ref> | no | first group in Assets/appstore-config.json — if the ref doesn't exist on ASC yet, it's auto-created |
| --group-name <text> | no | inherits from matching config group's localizations[0].name, else "Premium Access" — used when auto-creating a new group |
| --app-name <name> | no | read from existing configs |
| --bundle-id <id> | no | iOS bundle ID override (e.g. com.example.myapp) — use when Assets/appstore-config.json doesn't exist yet |
| --package-name <pkg> | no | Android package name override — use when Assets/googleplay-config.json doesn't exist yet |
Idempotency: safe to re-run. Existing products are PATCHed with refreshed PPP regional fan-out (same code path as gpc subscriptions push and create-appstore-app). Existing ASC subscriptions log "already exists — refreshing pricing" and continue.
Adapty is intentionally NOT included — Adapty pulls live store prices at runtime via its store integrations, so creating an extra Adapty product entry adds noise without unlocking anything the SDK can't already fetch. Adapty product entries remain managed via kappmaker adapty setup for the canonical product set. (For credit packs iap add DOES include Adapty, see below.)
Notes:
- No
--free-trialflag yet. For intro offers, editAssets/{googleplay,appstore}-config.jsonand rungpc subscriptions push/create-appstore-app. - Only en-US localization is created. For multi-locale, edit the config files.
- Only first regional price (
US/USD) is set explicitly. Other regions come from PPP fan-out.
iap add
Quick-add a single credit-pack consumable IAP to Google Play, App Store Connect, AND Adapty (Adapty is included here because credit packs use the credit_pack_access access level to gate consumable entitlements that have no store-side equivalent). Auto-generates aligned product IDs across all three.
# Minimum invocation
kappmaker iap add --credits 50 --price 14.99
# v2 product line (appends "_v2" suffix to the ID)
kappmaker iap add --credits 50 --price 14.99 --product-version 2
# Single store
kappmaker iap add --credits 100 --price 24.99 --platform ios
# Full control
kappmaker iap add \
--credits 10 --price 4.99 --product-version 2 \
--name "Basic Credit Pack" \
--description "10 credits to use in the app." \
--review-screenshot "Assets/appstore/review-screenshot_credits.jpg"What it creates (for --credits 10 --price 4.99 --product-version 1 --name "Basic Credit Pack"):
| Where | Field | Value |
|---|---|---|
| ASC | product_id | credit_pack_10_499_myapp (v2+ appends _v2) |
| ASC | type | CONSUMABLE |
| ASC | ref_name | MyApp Basic Credit Pack v1 (4.99) |
| ASC | localization (en-US) | name = Basic Credit Pack, description = 10 credits to use in the app. |
| ASC | pricing | $4.99 USD anchor → PPP across ~155 territories |
| Play | sku | credit_pack_10_499_myapp |
| Play | purchase_type | managed |
| Play | listing (en-US) | title = Basic Credit Pack, description = 10 credits to use in the app. |
| Play | pricing | $4.99 USD anchor → PPP across ~173 billable regions |
| Adapty | title | MyApp Basic Credit Pack v1 (4.99) |
| Adapty | period | consumable (routed via Adapty REST API since CLI rejects this period value) |
| Adapty | ios_product_id / android_product_id | same ID on both — credit_pack_10_499_myapp |
| Adapty | access level | credit_pack_access (falls back to first level if missing) |
Flags:
| Flag | Required | Default |
|---|---|---|
| --credits <number> | yes | — |
| --price <number> | yes | — — USD anchor; PPP fans the rest |
| `--pla
