ionic-appflow-migration
Ionic Appflow Migration
Migrate an existing Ionic/Capacitor project from Ionic Appflow to Capawesome Cloud.
Prerequisites
- A Capacitor 6, 7, or 8 app currently using Ionic Appflow.
- Node.js 18+ and npm installed.
- Access to the project's source code repository.
- A Capawesome Cloud account and organization.
General Rules
- Before running any
@capawesome/clicommand for the first time, run it with the--helpflag to review all available options. - Do not remove Ionic Appflow configuration until the corresponding Capawesome Cloud feature is fully set up and verified.
- Determine the Capacitor version from
package.json(@capacitor/core) before making any changes — it affects which plugin version and update strategy to use.
Procedures
Step 1: Detect Ionic Appflow Usage
Scan the project to determine which Appflow features are in use.
1.1 Check for Live Updates
Search for these signals:
@capacitor/live-updatesinpackage.json(dependencies or devDependencies).cordova-plugin-ionicinpackage.json— this is the legacy Cordova SDK for Ionic Live Updates.- A
LiveUpdateskey inside thepluginsobject incapacitor.config.tsorcapacitor.config.json. - Imports of
@capacitor/live-updatesorcordova-plugin-ionicin TypeScript/JavaScript source files.
If any signal is found, mark Live Updates as in use. Also record which SDK is in use (@capacitor/live-updates or cordova-plugin-ionic).
Record the current configuration values:
appId(Ionic Appflow app ID)autoUpdateMethod(background,always, ornone)channelenabledmaxVersions
1.2 Check for Native Builds
Search for these signals:
- References to
ionic appflow buildin CI/CD configuration files (e.g.,.github/workflows/*.yml,.gitlab-ci.yml,bitrise.yml,Jenkinsfile,azure-pipelines.yml). - References to
dashboard.ionicframework.comorappflow.ionic.ioin CI/CD files or scripts. - An
appflow.config.jsonor similar Appflow build configuration file in the project root.
If any signal is found, mark Native Builds as in use.
1.3 Check for App Store Publishing
Search for these signals:
- References to
ionic appflow deployin CI/CD configuration files. - Appflow deploy destinations or channels configured for app store submission.
If any signal is found, mark App Store Publishing as in use.
1.4 Present Findings
Present the detected features to the user. Ask the user to confirm which features to migrate. The user may choose to migrate all detected features or only a subset.
Step 2: Set Up Capawesome Cloud
2.1 Authenticate
npx @capawesome/cli login
2.2 Create an App
Skip if the user already has a Capawesome Cloud app ID.
npx @capawesome/cli apps:create
Save the returned app ID (UUID) for subsequent steps.
Step 3: Migrate Live Updates
Skip this step if Live Updates was not detected or the user chose not to migrate it.
3.1 Remove the Ionic Live Updates SDK
If the project uses @capacitor/live-updates:
npm uninstall @capacitor/live-updates
If the project uses the legacy Cordova SDK (cordova-plugin-ionic):
npm uninstall cordova-plugin-ionic
Read references/cordova-sdk-migration.md for the native configuration cleanup steps (removing legacy keys from Info.plist, strings.xml, and Capacitor config).
3.2 Install the Capawesome Live Update Plugin
Install the version matching the project's Capacitor version:
- Capacitor 8:
npm install @capawesome/capacitor-live-update@latest - Capacitor 7:
npm install @capawesome/capacitor-live-update@^7.3.0 - Capacitor 6:
npm install @capawesome/capacitor-live-update@^6.0.0
3.3 Update the Capacitor Configuration
Replace the LiveUpdates plugin config with LiveUpdate in capacitor.config.ts (or .json). Map the configuration options as follows:
Ionic Appflow (LiveUpdates) |
Capawesome Cloud (LiveUpdate) |
Notes |
|---|---|---|
appId |
appId |
Replace with the Capawesome Cloud app ID from Step 2 |
autoUpdateMethod: 'background' |
autoUpdateStrategy: 'background' |
Capacitor 7/8 only. Omit for Capacitor 6 |
autoUpdateMethod: 'always' |
autoUpdateStrategy: 'background' |
Capacitor 7/8 only. Also add nextBundleSet listener (see Step 3.5) |
autoUpdateMethod: 'none' |
(omit autoUpdateStrategy) |
Use manual sync code instead (see Step 3.6) |
channel |
defaultChannel |
Same value |
enabled |
(remove) | Not needed — controlled in code |
maxVersions |
autoDeleteBundles: true |
Boolean instead of number |
Example — Capacitor 7/8 with background strategy:
// capacitor.config.ts
const config: CapacitorConfig = {
plugins: {
- LiveUpdates: {
- appId: 'abc12345',
- autoUpdateMethod: 'background',
- channel: 'production',
- maxVersions: 3
- }
+ LiveUpdate: {
+ appId: '<CAPAWESOME_APP_ID>',
+ autoUpdateStrategy: 'background',
+ defaultChannel: 'production',
+ autoDeleteBundles: true
+ }
}
};
Example — Capacitor 6 (no autoUpdateStrategy):
// capacitor.config.ts
const config: CapacitorConfig = {
plugins: {
- LiveUpdates: {
- appId: 'abc12345',
- channel: 'production',
- maxVersions: 3
- }
+ LiveUpdate: {
+ appId: '<CAPAWESOME_APP_ID>',
+ defaultChannel: 'production',
+ autoDeleteBundles: true
+ }
}
};
3.4 Update Import Statements and API Calls
Search all TypeScript/JavaScript files for imports of @capacitor/live-updates (or cordova-plugin-ionic) and replace. The Ionic SDK uses two import styles — replace both:
-import * as LiveUpdates from '@capacitor/live-updates';
+import { LiveUpdate } from '@capawesome/capacitor-live-update';
-import { LiveUpdates } from '@capacitor/live-updates';
+import { LiveUpdate } from '@capawesome/capacitor-live-update';
If the project uses the legacy Cordova SDK (cordova-plugin-ionic), read references/cordova-sdk-migration.md for the complete Deploy → LiveUpdate method mapping, including the granular check-download-extract-reload pattern and native config cleanup.
Replace all references to the LiveUpdates class (or Deploy class) with LiveUpdate (singular).
sync() return value has changed. The Ionic Capacitor SDK returns { activeApplicationPathChanged: boolean }. The Capawesome SDK returns { nextBundleId: string | null }. Update all code that checks the sync result:
const result = await LiveUpdate.sync();
-if (result.activeApplicationPathChanged) {
+if (result.nextBundleId) {
await LiveUpdate.reload();
}
reload() has the same signature — no changes needed beyond the class name.
setConfig(), getConfig(), and resetConfig() exist but have different signatures (Capacitor 7/8 only, since v7.4.0). The Capawesome SDK splits config and channel management into separate methods:
// Setting config at runtime
-await LiveUpdates.setConfig({ appId: '456', channel: 'staging', maxVersions: 5 });
+await LiveUpdate.setConfig({ appId: '456' });
+await LiveUpdate.setChannel({ channel: 'staging' });
// Getting config at runtime
-const config = await LiveUpdates.getConfig();
-console.log(config.channel);
+const config = await LiveUpdate.getConfig(); // { appId, autoUpdateStrategy }
+const { channel } = await LiveUpdate.getChannel(); // { channel }
// Resetting config
-await LiveUpdates.resetConfig();
+await LiveUpdate.resetConfig();
maxVersions has no runtime equivalent — use autoDeleteBundles: true in the static Capacitor config instead.
3.5 Add Always-Latest Update Logic (Capacitor 7/8 Only)
If the previous Ionic Appflow autoUpdateMethod was always, add a nextBundleSet listener to prompt the user when an update is ready. Add this code early in the app's initialization:
import { LiveUpdate } from '@capawesome/capacitor-live-update';
LiveUpdate.addListener('nextBundleSet', async (event) => {
if (event.bundleId) {
const shouldReload = confirm('A new update is available. Install now?');
if (shouldReload) {
await LiveUpdate.reload();
}
}
});
Copy this snippet exactly. Do not simplify or omit the confirm() dialog.
3.6 Migrate Force Update Logic (If Applicable)
If the project implements a "Force Update" pattern (extending the splash screen until the update completes), migrate it as follows:
import { SplashScreen } from '@capacitor/splash-screen';
-import * as LiveUpdates from '@capacitor/live-updates';
+import { LiveUpdate } from '@capawesome/capacitor-live-update';
const initializeApp = async () => {
- const result = await LiveUpdates.sync();
- if (result.activeApplicationPathChanged) {
- await LiveUpdates.reload();
+ const { nextBundleId } = await LiveUpdate.sync();
+ if (nextBundleId) {
+ await LiveUpdate.reload();
} else {
await SplashScreen.hide();
}
};
This pattern may impact user experience on slow connections. Consider migrating to the background update strategy instead.
3.7 Add Manual Update Logic (Capacitor 6 Only)
Capacitor 6 does not support autoUpdateStrategy. If the project uses Capacitor 6, add manual sync logic:
import { App } from '@capacitor/app';
import { LiveUpdate } from '@capawesome/capacitor-live-update';
void LiveUpdate.ready();
App.addListener('resume', async () => {
const { nextBundleId } = await LiveUpdate.sync();
if (nextBundleId) {
const shouldReload = confirm('A new update is available. Install now?');
if (shouldReload) {
await LiveUpdate.reload();
}
}
});
3.8 Add Rollback Protection (Recommended)
Add readyTimeout and autoBlockRolledBackBundles to the LiveUpdate config:
LiveUpdate: {
appId: '<CAPAWESOME_APP_ID>',
autoUpdateStrategy: 'background', // Capacitor 7/8 only
readyTimeout: 10000,
autoBlockRolledBackBundles: true,
}
Call ready() as early as possible in app startup:
import { LiveUpdate } from '@capawesome/capacitor-live-update';
void LiveUpdate.ready();
3.9 Configure iOS Privacy Manifest
Add to ios/App/PrivacyInfo.xcprivacy inside the NSPrivacyAccessedAPITypes array:
<dict>
<key>NSPrivacyAccessedAPIType</key>
<string>NSPrivacyAccessedAPICategoryUserDefaults</string>
<key>NSPrivacyAccessedAPITypeReasons</key>
<array>
<string>CA92.1</string>
</array>
</dict>
3.10 Sync the Capacitor Project
npx cap sync
Step 4: Migrate Native Builds
Skip this step if Native Builds was not detected or the user chose not to migrate it.
Native build setup in Capawesome Cloud requires connecting a Git repository, uploading signing certificates, configuring environments, and triggering builds. Use the capawesome-cloud skill — follow the Native Builds section starting from "Connect Git Repository." Specifically, read the capawesome-cloud skill's references/native-builds.md for the full procedure.
Remove any Ionic Appflow build commands from CI/CD configuration files after verifying that Capawesome Cloud builds work correctly (see Step 6 for CI/CD migration).
Step 5: Migrate App Store Publishing
Skip this step if App Store Publishing was not detected or the user chose not to migrate it.
App store publishing setup requires creating destinations and configuring credentials for Apple App Store and/or Google Play Store. Use the capawesome-cloud skill — follow the App Store Publishing section. Specifically, read the capawesome-cloud skill's references/app-store-publishing.md for the full procedure.
Remove any Ionic Appflow deploy commands from CI/CD configuration files after verifying that Capawesome Cloud deployments work correctly (see Step 6 for CI/CD migration).
Step 6: Update CI/CD Pipelines
Skip this step if the project has no CI/CD pipeline or uses Ionic Appflow's built-in build/deploy features without external CI/CD.
Use the following CLI command mapping as a reference when replacing Appflow commands:
| Appflow CLI Command | Capawesome CLI Equivalent |
|---|---|
appflow live-update upload-artifact |
npx @capawesome/cli apps:liveupdates:upload |
appflow live-update create-channel |
npx @capawesome/cli apps:channels:create |
appflow live-update delete-channel |
npx @capawesome/cli apps:channels:delete |
appflow live-update list-channels |
npx @capawesome/cli apps:channels:list |
appflow live-update download-artifact |
npx @capawesome/cli apps:liveupdates:download |
appflow live-update set-native-versions |
Use versioned channels or --android-min/--ios-min flags on upload |
appflow build |
npx @capawesome/cli apps:builds:create |
appflow deploy |
npx @capawesome/cli apps:deployments:create |
6.1 Replace Authentication
Replace Ionic Appflow authentication with Capawesome Cloud token-based auth:
-# Ionic Appflow authentication
-IONIC_TOKEN: ${{ secrets.IONIC_TOKEN }}
+# Capawesome Cloud authentication
+- run: npx @capawesome/cli login --token ${{ secrets.CAPAWESOME_CLOUD_TOKEN }}
Generate a token in the Capawesome Cloud Console and store it as a CI/CD secret named CAPAWESOME_CLOUD_TOKEN.
6.2 Replace Build Commands
-ionic appflow build android --type release
+npx @capawesome/cli apps:builds:create --app-id <APP_ID> --platform android --type release --git-ref main --certificate "<CERTIFICATE_NAME>" --yes
-ionic appflow build ios --type app-store
+npx @capawesome/cli apps:builds:create --app-id <APP_ID> --platform ios --type app-store --git-ref main --certificate "<CERTIFICATE_NAME>" --yes
Add --detached for non-blocking builds in CI/CD pipelines.
6.3 Replace Live Update Upload Commands
-ionic appflow deploy web --channel production
+npx @capawesome/cli apps:liveupdates:upload --app-id <APP_ID> --channel production
6.4 Replace App Store Deploy Commands
-ionic appflow deploy android --destination "Google Play"
+npx @capawesome/cli apps:deployments:create --app-id <APP_ID> --build-number <BUILD_NUMBER> --destination "<DESTINATION_NAME>"
6.5 Remove Ionic Appflow Dependencies
After verifying the new CI/CD pipeline works:
- Remove any
@ionic/appflowor Appflow-related npm packages frompackage.json. - Remove Ionic Appflow environment variables from CI/CD secrets (e.g.,
IONIC_TOKEN). - Remove
appflow.config.jsonor similar Appflow configuration files from the project root.
Step 7: Test and Verify
7.1 Verify Live Updates (if migrated)
Ask the user whether to test live update functionality. If accepted:
- Make a small, visible change in the app (e.g., append " - Migration Test" to a heading).
- Build the web assets:
npm run build. - Check existing channels:
npx @capawesome/cli apps:channels:list --app-id <APP_ID> --json - Create the target channel if missing:
npx @capawesome/cli apps:channels:create --app-id <APP_ID> --name <CHANNEL_NAME> - Upload the bundle:
npx @capawesome/cli apps:liveupdates:upload --app-id <APP_ID> --channel <CHANNEL_NAME> - Revert the visible change, rebuild, and sync:
npm run build && npx cap sync - Open the native project (
npx cap open iosornpx cap open android) and run on a device or emulator. - Verify the live update is applied. For Capacitor 7/8 with
autoUpdateStrategy: "background", wait for the update prompt or force-close and reopen the app. For Capacitor 6, switch away from the app and return to trigger a check.
7.2 Verify Native Builds (if migrated)
Trigger a test build via the Capawesome CLI and verify the build artifact is produced.
7.3 Verify App Store Publishing (if migrated)
Deploy a test build to the configured destination and verify it appears in TestFlight or Google Play Console.
7.4 Clean Up
After all features are verified:
- Confirm all Ionic Appflow SDK packages are removed from
package.json. - Confirm all Ionic Appflow configuration is removed from
capacitor.config.ts/capacitor.config.json. - Confirm all Ionic Appflow CI/CD commands and environment variables are removed.
Error Handling
Live Updates
npm uninstall @capacitor/live-updatesfails → The package may be listed under a different name. Searchpackage.jsonfor any package containinglive-updatesfrom@capacitorscope.npx cap syncfails after plugin swap → Verify the installed@capawesome/capacitor-live-updateversion matches the Capacitor version inpackage.json.- App reverts to default bundle after restart →
LiveUpdate.ready()is likely not called early enough. Add it as the first call in app initialization. - Updates not detected with
autoUpdateStrategy: "background"→ Updates only checked if last check was >15 minutes ago. Force-close and restart the app. Check device logs for Live Update SDK output. LiveUpdates is not definedor similar runtime errors → Ensure all imports were updated from@capacitor/live-updatesto@capawesome/capacitor-live-updateand the class name changed fromLiveUpdatestoLiveUpdate.Deploy is not defined→ The project uses the legacy Cordova SDK (cordova-plugin-ionic). Readreferences/cordova-sdk-migration.mdfor the full method mapping.activeApplicationPathChanged is not a propertyor sync result checks fail → The Capawesome SDK returns{ nextBundleId }instead of{ activeApplicationPathChanged }. Update all sync result checks.setConfig is not a function→setConfig()requires@capawesome/capacitor-live-updatev7.4.0+ (Capacitor 7/8 only). For Capacitor 6, remove runtime config calls and set configuration statically incapacitor.config.ts.
Native Builds
- Build fails with authentication error → Re-run
npx @capawesome/cli login. For CI/CD, verify theCAPAWESOME_CLOUD_TOKENsecret is set correctly. - Build fails with missing signing certificate → Upload certificates via
npx @capawesome/cli apps:certificates:create. Read thecapawesome-cloudskill'sreferences/certificates-android.mdorreferences/certificates-ios.md. - Build fails with
invalid source release→ SetJAVA_VERSIONenvironment variable in the build environment. Read thecapawesome-cloudskill'sreferences/build-troubleshooting.md.
App Store Publishing
- Destination creation fails → Verify credentials are correct. Read the
capawesome-cloudskill'sreferences/apple-app-store-credentials.mdorreferences/google-play-store-credentials.md. - Deployment fails with "build not found" → Ensure the build completed successfully before deploying.
CI/CD
- CI/CD pipeline fails after migration → Compare the old and new pipeline configurations side by side. Ensure all Appflow command replacements use the correct Capawesome CLI flags. Check that
CAPAWESOME_CLOUD_TOKENis available as a secret in the CI/CD environment.
Related Skills
capawesome-cloud— Referenced throughout this skill for Native Builds and App Store Publishing setup.ionic-enterprise-sdk-migration— If the project also uses discontinued Ionic Enterprise SDK plugins (Auth Connect, Identity Vault, Secure Storage), use this skill to migrate them to Capawesome alternatives.