diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4f074fd --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.DS_Store +**/.DS_Store diff --git a/apk-direct-release-guide.md b/apk-direct-release-guide.md new file mode 100644 index 0000000..ae80b5e --- /dev/null +++ b/apk-direct-release-guide.md @@ -0,0 +1,106 @@ +# APK Direct Release Guide (Local Build + Cloudflare R2) + +This is the Dewemoji direct APK release flow. + +## 1) One-time setup + +### Required tools (local machine) + +```bash +brew install awscli +brew install --cask android-platform-tools +``` + +### Required environment variables + +```bash +export R2_ACCOUNT_ID="..." +export R2_ACCESS_KEY_ID="..." +export R2_SECRET_ACCESS_KEY="..." +export R2_BUCKET="dewemoji-downloads" +export R2_PUBLIC_BASE_URL="https://downloads.dewemoji.com" +``` + +Optional: + +```bash +export DEWEMOJI_APK_URL="https://dewemoji.com/downloads/dewemoji-latest.apk" +``` + +### Optional signing environment (recommended) + +```bash +export ANDROID_KEYSTORE_PATH="/absolute/path/release.jks" +export ANDROID_KEYSTORE_PASSWORD="..." +export ANDROID_KEY_ALIAS="..." +export ANDROID_KEY_PASSWORD="..." +``` + +--- + +## 2) Canonical URLs used by app updater + +- `https://dewemoji.com/downloads/version.json` +- `https://dewemoji.com/downloads/dewemoji-latest.apk` + +These endpoints redirect to R2 objects. + +--- + +## 3) Release steps + +Run from repo root. + +### A. Build APK + +```bash +./scripts/apk/build-release.sh +``` + +Output APK: + +- `dewemoji-capacitor/dist/apk/dewemoji-v{versionName}-{versionCode}.apk` + +### B. Publish APK + metadata to R2 + +```bash +./scripts/apk/publish-r2.sh \ + --apk dewemoji-capacitor/dist/apk/dewemoji-v1.1.2-112.apk \ + --version-name 1.1.2 \ + --version-code 112 \ + --min-supported-version-code 100 \ + --notes "Bug fixes and update UX improvements" \ + --force false +``` + +### C. Verify published release + +```bash +./scripts/apk/verify-release.sh --base-url https://dewemoji.com/downloads +``` + +--- + +## 4) Versioning rules + +1. Site-only deploy: do not bump APK version and do not publish new `version.json`. +2. Runtime/app-shell change: bump `versionCode` + `versionName`, then publish. +3. `versionCode` must always increase. +4. App update prompt appears only when remote `versionCode` is higher. + +--- + +## 5) Rollback + +1. Keep all versioned APK objects immutable (never overwrite). +2. Re-upload previous good APK to `apk/dewemoji-latest.apk`. +3. Re-publish `apk/version.json` with matching checksum/version fields. +4. Re-run verify script. + +--- + +## 6) Notes + +- Direct APK update is user-confirmed install (Android policy), not silent. +- Never embed R2 credentials in app. +- Keep app update payload over HTTPS only. diff --git a/app/app/Http/Controllers/Web/SiteController.php b/app/app/Http/Controllers/Web/SiteController.php index bf6736d..722491d 100644 --- a/app/app/Http/Controllers/Web/SiteController.php +++ b/app/app/Http/Controllers/Web/SiteController.php @@ -9,6 +9,7 @@ use App\Models\Subscription; use App\Models\UserKeyword; use App\Services\System\SettingsService; use Illuminate\Contracts\View\View; +use Illuminate\Http\JsonResponse; use Illuminate\Http\RedirectResponse; use Illuminate\Http\Request; use Illuminate\Http\Response; @@ -264,7 +265,40 @@ class SiteController extends Controller public function download(): View { - return view('site.download'); + $downloadBaseUrl = rtrim((string) config('dewemoji.apk_release.public_base_url', ''), '/'); + $androidEnabled = (bool) config('dewemoji.apk_release.enabled', false) && $downloadBaseUrl !== ''; + + return view('site.download', [ + 'androidEnabled' => $androidEnabled, + 'androidVersionJsonUrl' => $androidEnabled ? $downloadBaseUrl.'/version.json' : '', + 'androidLatestApkUrl' => $androidEnabled ? $downloadBaseUrl.'/dewemoji-latest.apk' : '', + ]); + } + + public function downloadVersionJson(Request $request): RedirectResponse|JsonResponse + { + $target = $this->apkReleaseTargetUrl('version_json'); + if ($target === '') { + return response()->json(['ok' => false, 'error' => 'apk_release_not_configured'], 404); + } + + return redirect()->away($target, 302, [ + 'Cache-Control' => 'no-store, no-cache, must-revalidate', + 'Pragma' => 'no-cache', + ]); + } + + public function downloadLatestApk(Request $request): RedirectResponse|JsonResponse + { + $target = $this->apkReleaseTargetUrl('latest_apk'); + if ($target === '') { + return response()->json(['ok' => false, 'error' => 'apk_release_not_configured'], 404); + } + + return redirect()->away($target, 302, [ + 'Cache-Control' => 'no-store, no-cache, must-revalidate', + 'Pragma' => 'no-cache', + ]); } public function privacy(): View @@ -465,6 +499,21 @@ class SiteController extends Controller return (string) config('dewemoji.data_path'); } + private function apkReleaseTargetUrl(string $key): string + { + if (!(bool) config('dewemoji.apk_release.enabled', false)) { + return ''; + } + + $base = trim((string) config('dewemoji.apk_release.r2_public_base_url', '')); + $objectKey = trim((string) config("dewemoji.apk_release.r2_keys.{$key}", '')); + if ($base === '' || $objectKey === '') { + return ''; + } + + return rtrim($base, '/').'/'.ltrim($objectKey, '/'); + } + /** * @param array $emoji */ diff --git a/app/config/dewemoji.php b/app/config/dewemoji.php index 913e63f..b95e5cd 100644 --- a/app/config/dewemoji.php +++ b/app/config/dewemoji.php @@ -123,4 +123,17 @@ return [ 'token' => (string) env('DEWEMOJI_METRICS_TOKEN', ''), 'allow_ips' => array_values(array_filter(array_map('trim', explode(',', (string) env('DEWEMOJI_METRICS_ALLOW_IPS', '127.0.0.1,::1'))))), ], + + 'apk_release' => [ + 'enabled' => filter_var(env('DEWEMOJI_APK_RELEASE_ENABLED', false), FILTER_VALIDATE_BOOL), + 'app_id' => (string) env('DEWEMOJI_APK_APP_ID', 'com.dewemoji.app'), + 'channel' => (string) env('DEWEMOJI_APK_CHANNEL', 'stable'), + 'min_supported_version_code' => (int) env('DEWEMOJI_APK_MIN_SUPPORTED_VERSION_CODE', 1), + 'public_base_url' => (string) env('DEWEMOJI_APK_PUBLIC_BASE_URL', 'https://dewemoji.com/downloads'), + 'r2_public_base_url' => (string) env('DEWEMOJI_R2_PUBLIC_BASE_URL', ''), + 'r2_keys' => [ + 'latest_apk' => (string) env('DEWEMOJI_R2_APK_LATEST_KEY', 'apk/dewemoji-latest.apk'), + 'version_json' => (string) env('DEWEMOJI_R2_APK_VERSION_KEY', 'apk/version.json'), + ], + ], ]; diff --git a/app/resources/views/dashboard/app.blade.php b/app/resources/views/dashboard/app.blade.php index d42a966..8bcaa01 100644 --- a/app/resources/views/dashboard/app.blade.php +++ b/app/resources/views/dashboard/app.blade.php @@ -105,47 +105,47 @@ -
Available
-
-
+
+
Android
-
In progress
+
+ {{ $androidEnabled ? 'Available' : 'In progress' }} +
diff --git a/app/routes/web.php b/app/routes/web.php index 30a1da0..9bfb6f9 100644 --- a/app/routes/web.php +++ b/app/routes/web.php @@ -17,6 +17,8 @@ Route::get('/emoji/{slug}', [SiteController::class, 'emojiDetail'])->name('emoji Route::get('/pricing', [SiteController::class, 'pricing'])->name('pricing'); Route::post('/pricing/currency', [SiteController::class, 'setPricingCurrency'])->name('pricing.currency'); Route::get('/download', [SiteController::class, 'download'])->name('download'); +Route::get('/downloads/version.json', [SiteController::class, 'downloadVersionJson'])->name('downloads.version'); +Route::get('/downloads/dewemoji-latest.apk', [SiteController::class, 'downloadLatestApk'])->name('downloads.latest-apk'); Route::get('/support', [SiteController::class, 'support'])->name('support'); Route::get('/privacy', [SiteController::class, 'privacy'])->name('privacy'); Route::get('/terms', [SiteController::class, 'terms'])->name('terms'); diff --git a/app/tests/Feature/SitePagesTest.php b/app/tests/Feature/SitePagesTest.php index 87eecc1..275cf56 100644 --- a/app/tests/Feature/SitePagesTest.php +++ b/app/tests/Feature/SitePagesTest.php @@ -11,6 +11,10 @@ class SitePagesTest extends TestCase parent::setUp(); config()->set('dewemoji.data_path', base_path('tests/Fixtures/emojis.fixture.json')); + config()->set('dewemoji.apk_release.enabled', true); + config()->set('dewemoji.apk_release.r2_public_base_url', 'https://downloads.example.com'); + config()->set('dewemoji.apk_release.r2_keys.latest_apk', 'apk/dewemoji-latest.apk'); + config()->set('dewemoji.apk_release.r2_keys.version_json', 'apk/version.json'); } public function test_core_pages_are_available(): void @@ -39,4 +43,15 @@ class SitePagesTest extends TestCase { $this->get('/emoji/unknown-slug')->assertNotFound(); } + + public function test_download_redirect_endpoints_are_available(): void + { + $this->get('/downloads/version.json') + ->assertStatus(302) + ->assertRedirect('https://downloads.example.com/apk/version.json'); + + $this->get('/downloads/dewemoji-latest.apk') + ->assertStatus(302) + ->assertRedirect('https://downloads.example.com/apk/dewemoji-latest.apk'); + } } diff --git a/deployment-live-walkthrough.md b/deployment-live-walkthrough.md index 6367af0..5fe2b8f 100644 --- a/deployment-live-walkthrough.md +++ b/deployment-live-walkthrough.md @@ -185,7 +185,34 @@ This avoids extension users hitting endpoints that are not ready. --- -## 8) Rollback Strategy +## 8) APK Release (Direct Download) + +APK release is independent from site redeploy. + +Canonical URLs used by the app updater: +1. `https://dewemoji.com/downloads/version.json` +2. `https://dewemoji.com/downloads/dewemoji-latest.apk` + +Set these env vars on app server: + +```env +DEWEMOJI_APK_RELEASE_ENABLED=true +DEWEMOJI_APK_PUBLIC_BASE_URL=https://dewemoji.com/downloads +DEWEMOJI_R2_PUBLIC_BASE_URL=https://downloads.your-r2-domain.com +DEWEMOJI_R2_APK_VERSION_KEY=apk/version.json +DEWEMOJI_R2_APK_LATEST_KEY=apk/dewemoji-latest.apk +``` + +Validate redirects: + +```bash +curl -I https://dewemoji.com/downloads/version.json +curl -I https://dewemoji.com/downloads/dewemoji-latest.apk +``` + +--- + +## 9) Rollback Strategy If release is broken: 1. Re-deploy previous known-good git commit. @@ -198,4 +225,3 @@ php artisan queue:restart ``` 3. If issue is emoji dataset, use snapshot activation in admin catalog. - diff --git a/dewemoji-capacitor/.gitignore b/dewemoji-capacitor/.gitignore new file mode 100644 index 0000000..0a687d7 --- /dev/null +++ b/dewemoji-capacitor/.gitignore @@ -0,0 +1,15 @@ +node_modules/ +.DS_Store + +# Capacitor / Android generated files +android/.gradle/ +android/.idea/ +android/local.properties +android/app/build/ +android/build/ +dist/ + +# logs +npm-debug.log* +yarn-debug.log* +yarn-error.log* diff --git a/dewemoji-capacitor/android/.gitignore b/dewemoji-capacitor/android/.gitignore new file mode 100644 index 0000000..48354a3 --- /dev/null +++ b/dewemoji-capacitor/android/.gitignore @@ -0,0 +1,101 @@ +# Using Android gitignore template: https://github.com/github/gitignore/blob/HEAD/Android.gitignore + +# Built application files +*.apk +*.aar +*.ap_ +*.aab + +# Files for the ART/Dalvik VM +*.dex + +# Java class files +*.class + +# Generated files +bin/ +gen/ +out/ +# Uncomment the following line in case you need and you don't have the release build type files in your app +# release/ + +# Gradle files +.gradle/ +build/ + +# Local configuration file (sdk path, etc) +local.properties + +# Proguard folder generated by Eclipse +proguard/ + +# Log Files +*.log + +# Android Studio Navigation editor temp files +.navigation/ + +# Android Studio captures folder +captures/ + +# IntelliJ +*.iml +.idea/workspace.xml +.idea/tasks.xml +.idea/gradle.xml +.idea/assetWizardSettings.xml +.idea/dictionaries +.idea/libraries +# Android Studio 3 in .gitignore file. +.idea/caches +.idea/modules.xml +# Comment next line if keeping position of elements in Navigation Editor is relevant for you +.idea/navEditor.xml + +# Keystore files +# Uncomment the following lines if you do not want to check your keystore files in. +#*.jks +#*.keystore + +# External native build folder generated in Android Studio 2.2 and later +.externalNativeBuild +.cxx/ + +# Google Services (e.g. APIs or Firebase) +# google-services.json + +# Freeline +freeline.py +freeline/ +freeline_project_description.json + +# fastlane +fastlane/report.xml +fastlane/Preview.html +fastlane/screenshots +fastlane/test_output +fastlane/readme.md + +# Version control +vcs.xml + +# lint +lint/intermediates/ +lint/generated/ +lint/outputs/ +lint/tmp/ +# lint/reports/ + +# Android Profiling +*.hprof + +# Cordova plugins for Capacitor +capacitor-cordova-android-plugins + +# Copied web assets +app/src/main/assets/public + +# Generated Config files +app/src/main/assets/capacitor.config.json +app/src/main/assets/capacitor.plugins.json +app/src/main/res/xml/config.xml diff --git a/dewemoji-capacitor/android/app/.gitignore b/dewemoji-capacitor/android/app/.gitignore new file mode 100644 index 0000000..043df80 --- /dev/null +++ b/dewemoji-capacitor/android/app/.gitignore @@ -0,0 +1,2 @@ +/build/* +!/build/.npmkeep diff --git a/dewemoji-capacitor/android/app/build.gradle b/dewemoji-capacitor/android/app/build.gradle new file mode 100644 index 0000000..cc21dd0 --- /dev/null +++ b/dewemoji-capacitor/android/app/build.gradle @@ -0,0 +1,59 @@ +apply plugin: 'com.android.application' + +android { + namespace "com.dewemoji.app" + compileSdk rootProject.ext.compileSdkVersion + defaultConfig { + applicationId "com.dewemoji.app" + minSdkVersion rootProject.ext.minSdkVersion + targetSdkVersion rootProject.ext.targetSdkVersion + versionCode 1 + versionName "1.0" + testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" + aaptOptions { + // Files and dirs to omit from the packaged assets dir, modified to accommodate modern web apps. + // Default: https://android.googlesource.com/platform/frameworks/base/+/282e181b58cf72b6ca770dc7ca5f91f135444502/tools/aapt/AaptAssets.cpp#61 + ignoreAssetsPattern '!.svn:!.git:!.ds_store:!*.scc:.*:!CVS:!thumbs.db:!picasa.ini:!*~' + } + } + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } +} + +repositories { + flatDir{ + dirs '../capacitor-cordova-android-plugins/src/main/libs', 'libs' + } +} + +configurations.all { + exclude group: 'org.jetbrains.kotlin', module: 'kotlin-stdlib-jdk7' + exclude group: 'org.jetbrains.kotlin', module: 'kotlin-stdlib-jdk8' +} + +dependencies { + implementation fileTree(include: ['*.jar'], dir: 'libs') + implementation "androidx.appcompat:appcompat:$androidxAppCompatVersion" + implementation "androidx.coordinatorlayout:coordinatorlayout:$androidxCoordinatorLayoutVersion" + implementation "androidx.core:core-splashscreen:$coreSplashScreenVersion" + implementation project(':capacitor-android') + testImplementation "junit:junit:$junitVersion" + androidTestImplementation "androidx.test.ext:junit:$androidxJunitVersion" + androidTestImplementation "androidx.test.espresso:espresso-core:$androidxEspressoCoreVersion" + implementation project(':capacitor-cordova-android-plugins') +} + +apply from: 'capacitor.build.gradle' + +try { + def servicesJSON = file('google-services.json') + if (servicesJSON.text) { + apply plugin: 'com.google.gms.google-services' + } +} catch(Exception e) { + logger.info("google-services.json not found, google-services plugin not applied. Push Notifications won't work") +} diff --git a/dewemoji-capacitor/android/app/capacitor.build.gradle b/dewemoji-capacitor/android/app/capacitor.build.gradle new file mode 100644 index 0000000..bbfb44f --- /dev/null +++ b/dewemoji-capacitor/android/app/capacitor.build.gradle @@ -0,0 +1,19 @@ +// DO NOT EDIT THIS FILE! IT IS GENERATED EACH TIME "capacitor update" IS RUN + +android { + compileOptions { + sourceCompatibility JavaVersion.VERSION_21 + targetCompatibility JavaVersion.VERSION_21 + } +} + +apply from: "../capacitor-cordova-android-plugins/cordova.variables.gradle" +dependencies { + + +} + + +if (hasProperty('postBuildExtras')) { + postBuildExtras() +} diff --git a/dewemoji-capacitor/android/app/proguard-rules.pro b/dewemoji-capacitor/android/app/proguard-rules.pro new file mode 100644 index 0000000..f1b4245 --- /dev/null +++ b/dewemoji-capacitor/android/app/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile diff --git a/dewemoji-capacitor/android/app/src/androidTest/java/com/getcapacitor/myapp/ExampleInstrumentedTest.java b/dewemoji-capacitor/android/app/src/androidTest/java/com/getcapacitor/myapp/ExampleInstrumentedTest.java new file mode 100644 index 0000000..f2c2217 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/androidTest/java/com/getcapacitor/myapp/ExampleInstrumentedTest.java @@ -0,0 +1,26 @@ +package com.getcapacitor.myapp; + +import static org.junit.Assert.*; + +import android.content.Context; +import androidx.test.ext.junit.runners.AndroidJUnit4; +import androidx.test.platform.app.InstrumentationRegistry; +import org.junit.Test; +import org.junit.runner.RunWith; + +/** + * Instrumented test, which will execute on an Android device. + * + * @see Testing documentation + */ +@RunWith(AndroidJUnit4.class) +public class ExampleInstrumentedTest { + + @Test + public void useAppContext() throws Exception { + // Context of the app under test. + Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext(); + + assertEquals("com.getcapacitor.app", appContext.getPackageName()); + } +} diff --git a/dewemoji-capacitor/android/app/src/main/AndroidManifest.xml b/dewemoji-capacitor/android/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..e8f2973 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/AndroidManifest.xml @@ -0,0 +1,42 @@ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dewemoji-capacitor/android/app/src/main/java/com/dewemoji/app/MainActivity.java b/dewemoji-capacitor/android/app/src/main/java/com/dewemoji/app/MainActivity.java new file mode 100644 index 0000000..ddcb914 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/java/com/dewemoji/app/MainActivity.java @@ -0,0 +1,357 @@ +package com.dewemoji.app; + +import android.app.AlertDialog; +import android.app.DownloadManager; +import android.content.ActivityNotFoundException; +import android.content.BroadcastReceiver; +import android.content.Context; +import android.content.DialogInterface; +import android.content.Intent; +import android.content.IntentFilter; +import android.content.pm.PackageInfo; +import android.content.pm.PackageManager; +import android.database.Cursor; +import android.net.Uri; +import android.os.Build; +import android.os.Bundle; +import android.os.Environment; +import android.util.Log; +import android.widget.Toast; + +import androidx.annotation.Nullable; +import androidx.core.content.FileProvider; +import androidx.core.content.ContextCompat; +import androidx.core.view.WindowCompat; +import androidx.core.view.WindowInsetsCompat; +import androidx.core.view.WindowInsetsControllerCompat; + +import com.getcapacitor.BridgeActivity; + +import org.json.JSONObject; + +import java.io.BufferedInputStream; +import java.io.ByteArrayOutputStream; +import java.io.InputStream; +import java.net.HttpURLConnection; +import java.net.URL; +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import java.util.Locale; + +public class MainActivity extends BridgeActivity { + private static final String TAG = "DewemojiUpdater"; + private static final String VERSION_URL = "https://dewemoji.com/downloads/version.json"; + private static final int CONNECT_TIMEOUT_MS = 10_000; + private static final int READ_TIMEOUT_MS = 15_000; + + @Nullable + private DownloadManager downloadManager; + private long activeDownloadId = -1L; + @Nullable + private String activeExpectedSha = null; + @Nullable + private BroadcastReceiver downloadReceiver = null; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + hideSystemBars(); + downloadManager = (DownloadManager) getSystemService(Context.DOWNLOAD_SERVICE); + registerDownloadReceiver(); + checkForUpdates(false); + } + + @Override + public void onDestroy() { + super.onDestroy(); + if (downloadReceiver != null) { + unregisterReceiver(downloadReceiver); + downloadReceiver = null; + } + } + + @Override + public void onWindowFocusChanged(boolean hasFocus) { + super.onWindowFocusChanged(hasFocus); + if (hasFocus) { + hideSystemBars(); + } + } + + private void hideSystemBars() { + WindowCompat.setDecorFitsSystemWindows(getWindow(), false); + WindowInsetsControllerCompat controller = + new WindowInsetsControllerCompat(getWindow(), getWindow().getDecorView()); + controller.hide(WindowInsetsCompat.Type.statusBars() | WindowInsetsCompat.Type.navigationBars()); + controller.setSystemBarsBehavior( + WindowInsetsControllerCompat.BEHAVIOR_SHOW_TRANSIENT_BARS_BY_SWIPE + ); + } + + private void checkForUpdates(boolean manual) { + new Thread(() -> { + try { + UpdateMetadata metadata = fetchVersionMetadata(); + if (metadata == null) { + return; + } + long installedVersion = getInstalledVersionCode(); + if (metadata.versionCode <= installedVersion) { + if (manual) { + runOnUiThread(() -> + Toast.makeText(this, "Dewemoji is up to date", Toast.LENGTH_SHORT).show() + ); + } + return; + } + runOnUiThread(() -> showUpdateDialog(metadata)); + } catch (Exception ex) { + Log.w(TAG, "Update check failed", ex); + if (manual) { + runOnUiThread(() -> + Toast.makeText(this, "Update check failed", Toast.LENGTH_SHORT).show() + ); + } + } + }).start(); + } + + @Nullable + private UpdateMetadata fetchVersionMetadata() throws Exception { + HttpURLConnection conn = (HttpURLConnection) new URL(VERSION_URL).openConnection(); + conn.setRequestMethod("GET"); + conn.setConnectTimeout(CONNECT_TIMEOUT_MS); + conn.setReadTimeout(READ_TIMEOUT_MS); + conn.setRequestProperty("Accept", "application/json"); + + int code = conn.getResponseCode(); + if (code < 200 || code >= 300) { + throw new IllegalStateException("Unexpected status " + code); + } + + try (InputStream in = new BufferedInputStream(conn.getInputStream()); + ByteArrayOutputStream out = new ByteArrayOutputStream()) { + byte[] buffer = new byte[8192]; + int read; + while ((read = in.read(buffer)) != -1) { + out.write(buffer, 0, read); + } + String json = out.toString(StandardCharsets.UTF_8.name()); + JSONObject obj = new JSONObject(json); + return UpdateMetadata.fromJson(obj); + } finally { + conn.disconnect(); + } + } + + private long getInstalledVersionCode() throws Exception { + PackageManager packageManager = getPackageManager(); + PackageInfo info = packageManager.getPackageInfo(getPackageName(), 0); + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) { + return info.getLongVersionCode(); + } + return info.versionCode; + } + + private void showUpdateDialog(UpdateMetadata metadata) { + StringBuilder message = new StringBuilder(); + message.append("New version ").append(metadata.versionName).append(" is available."); + if (!metadata.notes.isEmpty()) { + message.append("\n\n").append(metadata.notes); + } + if (metadata.force) { + message.append("\n\nThis update is required."); + } + + AlertDialog.Builder builder = new AlertDialog.Builder(this) + .setTitle("Update Dewemoji") + .setMessage(message.toString()) + .setPositiveButton("Update", (dialog, which) -> startApkDownload(metadata)) + .setCancelable(!metadata.force); + + if (!metadata.force) { + builder.setNegativeButton("Later", null); + } + + builder.show(); + } + + private void startApkDownload(UpdateMetadata metadata) { + if (downloadManager == null) { + Toast.makeText(this, "Download manager unavailable", Toast.LENGTH_SHORT).show(); + return; + } + + Uri uri = Uri.parse(metadata.apkUrl); + DownloadManager.Request request = new DownloadManager.Request(uri); + request.setTitle("Dewemoji update"); + request.setDescription("Downloading version " + metadata.versionName); + request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); + request.setDestinationInExternalPublicDir( + Environment.DIRECTORY_DOWNLOADS, + "dewemoji-latest.apk" + ); + request.setMimeType("application/vnd.android.package-archive"); + + activeExpectedSha = metadata.sha256.toLowerCase(Locale.US); + activeDownloadId = downloadManager.enqueue(request); + Toast.makeText(this, "Downloading update...", Toast.LENGTH_SHORT).show(); + } + + private void registerDownloadReceiver() { + downloadReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + if (!DownloadManager.ACTION_DOWNLOAD_COMPLETE.equals(intent.getAction())) { + return; + } + long downloadId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, -1L); + if (downloadId <= 0 || downloadId != activeDownloadId) { + return; + } + verifyAndInstallDownloadedApk(downloadId); + } + }; + + ContextCompat.registerReceiver( + this, + downloadReceiver, + new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE), + ContextCompat.RECEIVER_NOT_EXPORTED + ); + } + + private void verifyAndInstallDownloadedApk(long downloadId) { + if (downloadManager == null) { + return; + } + + DownloadManager.Query query = new DownloadManager.Query().setFilterById(downloadId); + try (Cursor cursor = downloadManager.query(query)) { + if (cursor == null || !cursor.moveToFirst()) { + showUpdateError("Download record not found"); + return; + } + + int statusCol = cursor.getColumnIndex(DownloadManager.COLUMN_STATUS); + int uriCol = cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI); + int reasonCol = cursor.getColumnIndex(DownloadManager.COLUMN_REASON); + int status = statusCol >= 0 ? cursor.getInt(statusCol) : DownloadManager.STATUS_FAILED; + String localUri = uriCol >= 0 ? cursor.getString(uriCol) : null; + int reason = reasonCol >= 0 ? cursor.getInt(reasonCol) : -1; + + if (status != DownloadManager.STATUS_SUCCESSFUL || localUri == null || localUri.isEmpty()) { + showUpdateError("Download failed (" + reason + ")"); + return; + } + + Uri apkUri = Uri.parse(localUri); + String localSha = computeSha256(apkUri); + if (localSha == null || activeExpectedSha == null || !localSha.equalsIgnoreCase(activeExpectedSha)) { + showUpdateError("Checksum mismatch"); + return; + } + + installApk(apkUri); + } catch (Exception ex) { + Log.e(TAG, "Failed to verify update APK", ex); + showUpdateError("Update verification failed"); + } + } + + @Nullable + private String computeSha256(Uri uri) { + try (InputStream input = getContentResolver().openInputStream(uri)) { + if (input == null) { + return null; + } + MessageDigest digest = MessageDigest.getInstance("SHA-256"); + byte[] buffer = new byte[8192]; + int read; + while ((read = input.read(buffer)) > 0) { + digest.update(buffer, 0, read); + } + byte[] bytes = digest.digest(); + StringBuilder out = new StringBuilder(bytes.length * 2); + for (byte b : bytes) { + out.append(String.format(Locale.US, "%02x", b)); + } + return out.toString(); + } catch (Exception ex) { + Log.e(TAG, "Failed to compute checksum", ex); + return null; + } + } + + private void installApk(Uri downloadUri) { + try { + Uri installUri = downloadUri; + if ("file".equals(downloadUri.getScheme())) { + installUri = FileProvider.getUriForFile( + this, + getPackageName() + ".fileprovider", + new java.io.File(downloadUri.getPath()) + ); + } + + Intent installIntent = new Intent(Intent.ACTION_VIEW); + installIntent.setDataAndType(installUri, "application/vnd.android.package-archive"); + installIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); + installIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); + startActivity(installIntent); + } catch (ActivityNotFoundException ex) { + showUpdateError("No installer found"); + } catch (Exception ex) { + Log.e(TAG, "Failed to launch APK installer", ex); + showUpdateError("Cannot open installer"); + } + } + + private void showUpdateError(String message) { + runOnUiThread(() -> new AlertDialog.Builder(this) + .setTitle("Update failed") + .setMessage(message) + .setPositiveButton("OK", (DialogInterface dialog, int which) -> dialog.dismiss()) + .show()); + } + + private static class UpdateMetadata { + final String versionName; + final long versionCode; + final String apkUrl; + final String sha256; + final String notes; + final boolean force; + + private UpdateMetadata( + String versionName, + long versionCode, + String apkUrl, + String sha256, + String notes, + boolean force + ) { + this.versionName = versionName; + this.versionCode = versionCode; + this.apkUrl = apkUrl; + this.sha256 = sha256; + this.notes = notes; + this.force = force; + } + + static UpdateMetadata fromJson(JSONObject obj) { + String versionName = obj.optString("versionName", ""); + long versionCode = obj.optLong("versionCode", 0); + String apkUrl = obj.optString("apkUrl", ""); + String sha256 = obj.optString("sha256", ""); + String notes = obj.optString("notes", ""); + boolean force = obj.optBoolean("force", false); + + if (versionName.isEmpty() || versionCode <= 0 || apkUrl.isEmpty() || sha256.isEmpty()) { + throw new IllegalStateException("Invalid version metadata payload"); + } + + return new UpdateMetadata(versionName, versionCode, apkUrl, sha256, notes, force); + } + } +} diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-land-hdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-land-hdpi/splash.png new file mode 100644 index 0000000..e31573b Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-land-hdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-land-mdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-land-mdpi/splash.png new file mode 100644 index 0000000..f7a6492 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-land-mdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-land-xhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xhdpi/splash.png new file mode 100644 index 0000000..8077255 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxhdpi/splash.png new file mode 100644 index 0000000..14c6c8f Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxxhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxxhdpi/splash.png new file mode 100644 index 0000000..244ca25 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-land-xxxhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-port-hdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-port-hdpi/splash.png new file mode 100644 index 0000000..74faaa5 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-port-hdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-port-mdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-port-mdpi/splash.png new file mode 100644 index 0000000..e944f4a Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-port-mdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-port-xhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xhdpi/splash.png new file mode 100644 index 0000000..564a82f Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxhdpi/splash.png new file mode 100644 index 0000000..bfabe68 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxxhdpi/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxxhdpi/splash.png new file mode 100644 index 0000000..6929071 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable-port-xxxhdpi/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable-v24/ic_launcher_foreground.xml b/dewemoji-capacitor/android/app/src/main/res/drawable-v24/ic_launcher_foreground.xml new file mode 100644 index 0000000..c7bd21d --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/drawable-v24/ic_launcher_foreground.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable/ic_launcher_background.xml b/dewemoji-capacitor/android/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 0000000..d5fccc5 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,170 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dewemoji-capacitor/android/app/src/main/res/drawable/splash.png b/dewemoji-capacitor/android/app/src/main/res/drawable/splash.png new file mode 100644 index 0000000..f7a6492 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/drawable/splash.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/layout/activity_main.xml b/dewemoji-capacitor/android/app/src/main/res/layout/activity_main.xml new file mode 100644 index 0000000..b5ad138 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,12 @@ + + + + + diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 0000000..036d09b --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 0000000..036d09b --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000..13aab44 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..af87109 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png new file mode 100644 index 0000000..13aab44 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 0000000..9a57ad0 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..8121162 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png new file mode 100644 index 0000000..9a57ad0 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 0000000..95142ae Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..4e78bc4 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png new file mode 100644 index 0000000..95142ae Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 0000000..ce7c804 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..c27abda Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png new file mode 100644 index 0000000..ce7c804 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000..e916872 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..8fb6665 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png new file mode 100644 index 0000000..e916872 Binary files /dev/null and b/dewemoji-capacitor/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png differ diff --git a/dewemoji-capacitor/android/app/src/main/res/values/ic_launcher_background.xml b/dewemoji-capacitor/android/app/src/main/res/values/ic_launcher_background.xml new file mode 100644 index 0000000..c5d5899 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/values/ic_launcher_background.xml @@ -0,0 +1,4 @@ + + + #FFFFFF + \ No newline at end of file diff --git a/dewemoji-capacitor/android/app/src/main/res/values/strings.xml b/dewemoji-capacitor/android/app/src/main/res/values/strings.xml new file mode 100644 index 0000000..b01331a --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/values/strings.xml @@ -0,0 +1,7 @@ + + + Dewemoji + Dewemoji + com.dewemoji.app + com.dewemoji.app + diff --git a/dewemoji-capacitor/android/app/src/main/res/values/styles.xml b/dewemoji-capacitor/android/app/src/main/res/values/styles.xml new file mode 100644 index 0000000..be874e5 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/values/styles.xml @@ -0,0 +1,22 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/dewemoji-capacitor/android/app/src/main/res/xml/file_paths.xml b/dewemoji-capacitor/android/app/src/main/res/xml/file_paths.xml new file mode 100644 index 0000000..bd0c4d8 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/main/res/xml/file_paths.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/dewemoji-capacitor/android/app/src/test/java/com/getcapacitor/myapp/ExampleUnitTest.java b/dewemoji-capacitor/android/app/src/test/java/com/getcapacitor/myapp/ExampleUnitTest.java new file mode 100644 index 0000000..0297327 --- /dev/null +++ b/dewemoji-capacitor/android/app/src/test/java/com/getcapacitor/myapp/ExampleUnitTest.java @@ -0,0 +1,18 @@ +package com.getcapacitor.myapp; + +import static org.junit.Assert.*; + +import org.junit.Test; + +/** + * Example local unit test, which will execute on the development machine (host). + * + * @see Testing documentation + */ +public class ExampleUnitTest { + + @Test + public void addition_isCorrect() throws Exception { + assertEquals(4, 2 + 2); + } +} diff --git a/dewemoji-capacitor/android/build.gradle b/dewemoji-capacitor/android/build.gradle new file mode 100644 index 0000000..f1b3b0e --- /dev/null +++ b/dewemoji-capacitor/android/build.gradle @@ -0,0 +1,29 @@ +// Top-level build file where you can add configuration options common to all sub-projects/modules. + +buildscript { + + repositories { + google() + mavenCentral() + } + dependencies { + classpath 'com.android.tools.build:gradle:8.7.2' + classpath 'com.google.gms:google-services:4.4.2' + + // NOTE: Do not place your application dependencies here; they belong + // in the individual module build.gradle files + } +} + +apply from: "variables.gradle" + +allprojects { + repositories { + google() + mavenCentral() + } +} + +task clean(type: Delete) { + delete rootProject.buildDir +} diff --git a/dewemoji-capacitor/android/capacitor.settings.gradle b/dewemoji-capacitor/android/capacitor.settings.gradle new file mode 100644 index 0000000..9a5fa87 --- /dev/null +++ b/dewemoji-capacitor/android/capacitor.settings.gradle @@ -0,0 +1,3 @@ +// DO NOT EDIT THIS FILE! IT IS GENERATED EACH TIME "capacitor update" IS RUN +include ':capacitor-android' +project(':capacitor-android').projectDir = new File('../node_modules/@capacitor/android/capacitor') diff --git a/dewemoji-capacitor/android/gradle.properties b/dewemoji-capacitor/android/gradle.properties new file mode 100644 index 0000000..2e87c52 --- /dev/null +++ b/dewemoji-capacitor/android/gradle.properties @@ -0,0 +1,22 @@ +# Project-wide Gradle settings. + +# IDE (e.g. Android Studio) users: +# Gradle settings configured through the IDE *will override* +# any settings specified in this file. + +# For more details on how to configure your build environment visit +# http://www.gradle.org/docs/current/userguide/build_environment.html + +# Specifies the JVM arguments used for the daemon process. +# The setting is particularly useful for tweaking memory settings. +org.gradle.jvmargs=-Xmx1536m + +# When configured, Gradle will run in incubating parallel mode. +# This option should only be used with decoupled projects. More details, visit +# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects +# org.gradle.parallel=true + +# AndroidX package structure to make it clearer which packages are bundled with the +# Android operating system, and which are packaged with your app's APK +# https://developer.android.com/topic/libraries/support-library/androidx-rn +android.useAndroidX=true diff --git a/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.jar b/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000..a4b76b9 Binary files /dev/null and b/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.jar differ diff --git a/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.properties b/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..c1d5e01 --- /dev/null +++ b/dewemoji-capacitor/android/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,7 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.11.1-all.zip +networkTimeout=10000 +validateDistributionUrl=true +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/dewemoji-capacitor/android/gradlew b/dewemoji-capacitor/android/gradlew new file mode 100755 index 0000000..f5feea6 --- /dev/null +++ b/dewemoji-capacitor/android/gradlew @@ -0,0 +1,252 @@ +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 +# + +############################################################################## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# +############################################################################## + +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done + +# This is normally unused +# shellcheck disable=SC2034 +APP_BASE_NAME=${0##*/} +# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) +APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s +' "$PWD" ) || exit + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD=maximum + +warn () { + echo "$*" +} >&2 + +die () { + echo + echo "$*" + echo + exit 1 +} >&2 + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD=$JAVA_HOME/jre/sh/java + else + JAVACMD=$JAVA_HOME/bin/java + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD=java + if ! command -v java >/dev/null 2>&1 + then + die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +fi + +# Increase the maximum file descriptors if we can. +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac +fi + +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + + # Now convert the arguments - kludge to limit ourselves to /bin/sh + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) + fi + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg + done +fi + + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command: +# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, +# and any embedded shellness will be escaped. +# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# treated as '${Hostname}' itself on the command line. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/dewemoji-capacitor/android/gradlew.bat b/dewemoji-capacitor/android/gradlew.bat new file mode 100644 index 0000000..9b42019 --- /dev/null +++ b/dewemoji-capacitor/android/gradlew.bat @@ -0,0 +1,94 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem +@rem SPDX-License-Identifier: Apache-2.0 +@rem + +@if "%DEBUG%"=="" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if %ERRORLEVEL% equ 0 goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if %ERRORLEVEL% equ 0 goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/dewemoji-capacitor/android/settings.gradle b/dewemoji-capacitor/android/settings.gradle new file mode 100644 index 0000000..3b4431d --- /dev/null +++ b/dewemoji-capacitor/android/settings.gradle @@ -0,0 +1,5 @@ +include ':app' +include ':capacitor-cordova-android-plugins' +project(':capacitor-cordova-android-plugins').projectDir = new File('./capacitor-cordova-android-plugins/') + +apply from: 'capacitor.settings.gradle' \ No newline at end of file diff --git a/dewemoji-capacitor/android/variables.gradle b/dewemoji-capacitor/android/variables.gradle new file mode 100644 index 0000000..c9278b5 --- /dev/null +++ b/dewemoji-capacitor/android/variables.gradle @@ -0,0 +1,16 @@ +ext { + minSdkVersion = 24 + compileSdkVersion = 35 + targetSdkVersion = 35 + androidxActivityVersion = '1.9.2' + androidxAppCompatVersion = '1.7.0' + androidxCoordinatorLayoutVersion = '1.2.0' + androidxCoreVersion = '1.15.0' + androidxFragmentVersion = '1.8.4' + coreSplashScreenVersion = '1.0.1' + androidxWebkitVersion = '1.12.1' + junitVersion = '4.13.2' + androidxJunitVersion = '1.2.1' + androidxEspressoCoreVersion = '3.6.1' + cordovaAndroidVersion = '13.0.0' +} diff --git a/dewemoji-capacitor/capacitor.config.json b/dewemoji-capacitor/capacitor.config.json new file mode 100644 index 0000000..57cdfc2 --- /dev/null +++ b/dewemoji-capacitor/capacitor.config.json @@ -0,0 +1,9 @@ +{ + "appId": "com.dewemoji.app", + "appName": "Dewemoji", + "webDir": "www", + "server": { + "url": "https://dewemoji.com", + "cleartext": false + } +} diff --git a/dewemoji-capacitor/package-lock.json b/dewemoji-capacitor/package-lock.json new file mode 100644 index 0000000..0b54fe5 --- /dev/null +++ b/dewemoji-capacitor/package-lock.json @@ -0,0 +1,1075 @@ +{ + "name": "dewemoji-capacitor", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "dewemoji-capacitor", + "version": "1.0.0", + "license": "ISC", + "dependencies": { + "@capacitor/android": "^8.1.0", + "@capacitor/cli": "^7.5.0", + "@capacitor/core": "^8.1.0" + } + }, + "node_modules/@capacitor/android": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/@capacitor/android/-/android-8.1.0.tgz", + "integrity": "sha512-z0acTPxj5DCy/U2FU7w+GA93oC+wdyKnsOcRg5rutDmSYa8Do1tzYqApKgf+hnuTNPbtrCTHp0Zy1cLiK/4MEw==", + "license": "MIT", + "peerDependencies": { + "@capacitor/core": "^8.1.0" + } + }, + "node_modules/@capacitor/cli": { + "version": "7.5.0", + "resolved": "https://registry.npmjs.org/@capacitor/cli/-/cli-7.5.0.tgz", + "integrity": "sha512-mlohsvLZjWrO5eAVTn1+dABNQwQawcphVp6NQVJZ3I4x2BAoNmJj53QflX7PYGUipL9gF9EM9Yiku3m1McxFZg==", + "license": "MIT", + "dependencies": { + "@ionic/cli-framework-output": "^2.2.8", + "@ionic/utils-subprocess": "^3.0.1", + "@ionic/utils-terminal": "^2.3.5", + "commander": "^12.1.0", + "debug": "^4.4.0", + "env-paths": "^2.2.0", + "fs-extra": "^11.2.0", + "kleur": "^4.1.5", + "native-run": "^2.0.3", + "open": "^8.4.0", + "plist": "^3.1.0", + "prompts": "^2.4.2", + "rimraf": "^6.0.1", + "semver": "^7.6.3", + "tar": "^7.5.3", + "tslib": "^2.8.1", + "xml2js": "^0.6.2" + }, + "bin": { + "cap": "bin/capacitor", + "capacitor": "bin/capacitor" + }, + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/@capacitor/core": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/@capacitor/core/-/core-8.1.0.tgz", + "integrity": "sha512-UfMBMWc1v7J+14AhH03QmeNwV3HZx3qnOWhpwnHfzALEwAwlV/itQOQqcasMQYhOHWL0tiymc5ByaLTn7KKQxw==", + "license": "MIT", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/@ionic/cli-framework-output": { + "version": "2.2.8", + "resolved": "https://registry.npmjs.org/@ionic/cli-framework-output/-/cli-framework-output-2.2.8.tgz", + "integrity": "sha512-TshtaFQsovB4NWRBydbNFawql6yul7d5bMiW1WYYf17hd99V6xdDdk3vtF51bw6sLkxON3bDQpWsnUc9/hVo3g==", + "license": "MIT", + "dependencies": { + "@ionic/utils-terminal": "2.3.5", + "debug": "^4.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-array": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/@ionic/utils-array/-/utils-array-2.1.6.tgz", + "integrity": "sha512-0JZ1Zkp3wURnv8oq6Qt7fMPo5MpjbLoUoa9Bu2Q4PJuSDWM8H8gwF3dQO7VTeUj3/0o1IB1wGkFWZZYgUXZMUg==", + "license": "MIT", + "dependencies": { + "debug": "^4.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-fs": { + "version": "3.1.7", + "resolved": "https://registry.npmjs.org/@ionic/utils-fs/-/utils-fs-3.1.7.tgz", + "integrity": "sha512-2EknRvMVfhnyhL1VhFkSLa5gOcycK91VnjfrTB0kbqkTFCOXyXgVLI5whzq7SLrgD9t1aqos3lMMQyVzaQ5gVA==", + "license": "MIT", + "dependencies": { + "@types/fs-extra": "^8.0.0", + "debug": "^4.0.0", + "fs-extra": "^9.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-fs/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "license": "MIT", + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@ionic/utils-object": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/@ionic/utils-object/-/utils-object-2.1.6.tgz", + "integrity": "sha512-vCl7sl6JjBHFw99CuAqHljYJpcE88YaH2ZW4ELiC/Zwxl5tiwn4kbdP/gxi2OT3MQb1vOtgAmSNRtusvgxI8ww==", + "license": "MIT", + "dependencies": { + "debug": "^4.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-process": { + "version": "2.1.12", + "resolved": "https://registry.npmjs.org/@ionic/utils-process/-/utils-process-2.1.12.tgz", + "integrity": "sha512-Jqkgyq7zBs/v/J3YvKtQQiIcxfJyplPgECMWgdO0E1fKrrH8EF0QGHNJ9mJCn6PYe2UtHNS8JJf5G21e09DfYg==", + "license": "MIT", + "dependencies": { + "@ionic/utils-object": "2.1.6", + "@ionic/utils-terminal": "2.3.5", + "debug": "^4.0.0", + "signal-exit": "^3.0.3", + "tree-kill": "^1.2.2", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-stream": { + "version": "3.1.7", + "resolved": "https://registry.npmjs.org/@ionic/utils-stream/-/utils-stream-3.1.7.tgz", + "integrity": "sha512-eSELBE7NWNFIHTbTC2jiMvh1ABKGIpGdUIvARsNPMNQhxJB3wpwdiVnoBoTYp+5a6UUIww4Kpg7v6S7iTctH1w==", + "license": "MIT", + "dependencies": { + "debug": "^4.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-subprocess": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@ionic/utils-subprocess/-/utils-subprocess-3.0.1.tgz", + "integrity": "sha512-cT4te3AQQPeIM9WCwIg8ohroJ8TjsYaMb2G4ZEgv9YzeDqHZ4JpeIKqG2SoaA3GmVQ3sOfhPM6Ox9sxphV/d1A==", + "license": "MIT", + "dependencies": { + "@ionic/utils-array": "2.1.6", + "@ionic/utils-fs": "3.1.7", + "@ionic/utils-process": "2.1.12", + "@ionic/utils-stream": "3.1.7", + "@ionic/utils-terminal": "2.3.5", + "cross-spawn": "^7.0.3", + "debug": "^4.0.0", + "tslib": "^2.0.1" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@ionic/utils-terminal": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@ionic/utils-terminal/-/utils-terminal-2.3.5.tgz", + "integrity": "sha512-3cKScz9Jx2/Pr9ijj1OzGlBDfcmx7OMVBt4+P1uRR0SSW4cm1/y3Mo4OY3lfkuaYifMNBW8Wz6lQHbs1bihr7A==", + "license": "MIT", + "dependencies": { + "@types/slice-ansi": "^4.0.0", + "debug": "^4.0.0", + "signal-exit": "^3.0.3", + "slice-ansi": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "tslib": "^2.0.1", + "untildify": "^4.0.0", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/@isaacs/fs-minipass": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@isaacs/fs-minipass/-/fs-minipass-4.0.1.tgz", + "integrity": "sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w==", + "license": "ISC", + "dependencies": { + "minipass": "^7.0.4" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@types/fs-extra": { + "version": "8.1.5", + "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-8.1.5.tgz", + "integrity": "sha512-0dzKcwO+S8s2kuF5Z9oUWatQJj5Uq/iqphEtE3GQJVRRYm/tD1LglU2UnXi2A8jLq5umkGouOXOR9y0n613ZwQ==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/node": { + "version": "25.3.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.0.tgz", + "integrity": "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A==", + "license": "MIT", + "dependencies": { + "undici-types": "~7.18.0" + } + }, + "node_modules/@types/slice-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/slice-ansi/-/slice-ansi-4.0.0.tgz", + "integrity": "sha512-+OpjSaq85gvlZAYINyzKpLeiFkSC4EsC6IIiT6v6TLSU5k5U83fHGj9Lel8oKEXM0HqgrMVCjXPDPVICtxF7EQ==", + "license": "MIT" + }, + "node_modules/@xmldom/xmldom": { + "version": "0.8.11", + "resolved": "https://registry.npmjs.org/@xmldom/xmldom/-/xmldom-0.8.11.tgz", + "integrity": "sha512-cQzWCtO6C8TQiYl1ruKNn2U6Ao4o4WBBcbL61yJl84x+j5sOWWFU9X7DpND8XZG3daDppSsigMdfAIl2upQBRw==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/astral-regex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", + "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/at-least-node": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/at-least-node/-/at-least-node-1.0.0.tgz", + "integrity": "sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg==", + "license": "ISC", + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/balanced-match": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.3.tgz", + "integrity": "sha512-1pHv8LX9CpKut1Zp4EXey7Z8OfH11ONNH6Dhi2WDUt31VVZFXZzKwXcysBgqSumFCmR+0dqjMK5v5JiFHzi0+g==", + "license": "MIT", + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/big-integer": { + "version": "1.6.52", + "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.52.tgz", + "integrity": "sha512-QxD8cf2eVqJOOz63z6JIN9BzvVs/dlySa5HGSBH5xtR8dPteIRQnBxxKqkNTiT6jbDTF6jAfrd4oMcND9RGbQg==", + "license": "Unlicense", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/bplist-parser": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/bplist-parser/-/bplist-parser-0.3.2.tgz", + "integrity": "sha512-apC2+fspHGI3mMKj+dGevkGo/tCqVB8jMb6i+OX+E29p0Iposz07fABkRIfVUPNd5A5VbuOz1bZbnmkKLYF+wQ==", + "license": "MIT", + "dependencies": { + "big-integer": "1.6.x" + }, + "engines": { + "node": ">= 5.10.0" + } + }, + "node_modules/brace-expansion": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.2.tgz", + "integrity": "sha512-Pdk8c9poy+YhOgVWw1JNN22/HcivgKWwpxKq04M/jTmHyCZn12WPJebZxdjSa5TmBqISrUSgNYU3eRORljfCCw==", + "license": "MIT", + "dependencies": { + "balanced-match": "^4.0.2" + }, + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/buffer-crc32": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", + "integrity": "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ==", + "license": "MIT", + "engines": { + "node": "*" + } + }, + "node_modules/chownr": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-3.0.0.tgz", + "integrity": "sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==", + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=18" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "license": "MIT" + }, + "node_modules/commander": { + "version": "12.1.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-12.1.0.tgz", + "integrity": "sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==", + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/define-lazy-prop": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz", + "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/elementtree": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/elementtree/-/elementtree-0.1.7.tgz", + "integrity": "sha512-wkgGT6kugeQk/P6VZ/f4T+4HB41BVgNBq5CDIZVbQ02nvTVqAiVTbskxxu3eA/X96lMlfYOwnLQpN2v5E1zDEg==", + "license": "Apache-2.0", + "dependencies": { + "sax": "1.1.4" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "license": "MIT" + }, + "node_modules/env-paths": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", + "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/fd-slicer": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.1.0.tgz", + "integrity": "sha512-cE1qsB/VwyQozZ+q1dGxR8LBYNZeofhEdUNGSMbQD3Gw2lAzX9Zb3uIU6Ebc/Fmyjo9AWWfnn0AUCHqtevs/8g==", + "license": "MIT", + "dependencies": { + "pend": "~1.2.0" + } + }, + "node_modules/fs-extra": { + "version": "11.3.3", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.3.3.tgz", + "integrity": "sha512-VWSRii4t0AFm6ixFFmLLx1t7wS1gh+ckoa84aOeapGum0h+EZd1EhEumSB+ZdDLnEPuucsVB9oB7cxJHap6Afg==", + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=14.14" + } + }, + "node_modules/glob": { + "version": "13.0.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-13.0.6.tgz", + "integrity": "sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw==", + "license": "BlueOak-1.0.0", + "dependencies": { + "minimatch": "^10.2.2", + "minipass": "^7.1.3", + "path-scurry": "^2.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "license": "ISC" + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/ini": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/ini/-/ini-4.1.3.tgz", + "integrity": "sha512-X7rqawQBvfdjS10YU1y1YVreA3SsLrW9dX2CewP2EbBJM4ypVNLDkO5y04gejPwKIY9lR+7r9gn3rFPt/kmWFg==", + "license": "ISC", + "engines": { + "node": "^14.17.0 || ^16.13.0 || >=18.0.0" + } + }, + "node_modules/is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "license": "MIT", + "bin": { + "is-docker": "cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "license": "MIT", + "dependencies": { + "is-docker": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "license": "ISC" + }, + "node_modules/jsonfile": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.2.0.tgz", + "integrity": "sha512-FGuPw30AdOIUTRMC2OMRtQV+jkVj2cfPqSeWXv1NEAJ1qZ5zb1X6z1mFhbfOB/iy3ssJCD+3KuZ8r8C3uVFlAg==", + "license": "MIT", + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/kleur": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", + "integrity": "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/lru-cache": { + "version": "11.2.6", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.2.6.tgz", + "integrity": "sha512-ESL2CrkS/2wTPfuend7Zhkzo2u0daGJ/A2VucJOgQ/C48S/zB8MMeMHSGKYpXhIjbPxfuezITkaBH1wqv00DDQ==", + "license": "BlueOak-1.0.0", + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/minimatch": { + "version": "10.2.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.2.tgz", + "integrity": "sha512-+G4CpNBxa5MprY+04MbgOw1v7So6n5JY166pFi9KfYwT78fxScCeSNQSNzp6dpPSW2rONOps6Ocam1wFhCgoVw==", + "license": "BlueOak-1.0.0", + "dependencies": { + "brace-expansion": "^5.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minipass": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.3.tgz", + "integrity": "sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==", + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/minizlib": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-3.1.0.tgz", + "integrity": "sha512-KZxYo1BUkWD2TVFLr0MQoM8vUUigWD3LlD83a/75BqC+4qE0Hb1Vo5v1FgcfaNXvfXzr+5EhQ6ing/CaBijTlw==", + "license": "MIT", + "dependencies": { + "minipass": "^7.1.2" + }, + "engines": { + "node": ">= 18" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/native-run": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/native-run/-/native-run-2.0.3.tgz", + "integrity": "sha512-U1PllBuzW5d1gfan+88L+Hky2eZx+9gv3Pf6rNBxKbORxi7boHzqiA6QFGSnqMem4j0A9tZ08NMIs5+0m/VS1Q==", + "license": "MIT", + "dependencies": { + "@ionic/utils-fs": "^3.1.7", + "@ionic/utils-terminal": "^2.3.4", + "bplist-parser": "^0.3.2", + "debug": "^4.3.4", + "elementtree": "^0.1.7", + "ini": "^4.1.1", + "plist": "^3.1.0", + "split2": "^4.2.0", + "through2": "^4.0.2", + "tslib": "^2.6.2", + "yauzl": "^2.10.0" + }, + "bin": { + "native-run": "bin/native-run" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/open": { + "version": "8.4.2", + "resolved": "https://registry.npmjs.org/open/-/open-8.4.2.tgz", + "integrity": "sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==", + "license": "MIT", + "dependencies": { + "define-lazy-prop": "^2.0.0", + "is-docker": "^2.1.1", + "is-wsl": "^2.2.0" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", + "license": "BlueOak-1.0.0" + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-scurry": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.2.tgz", + "integrity": "sha512-3O/iVVsJAPsOnpwWIeD+d6z/7PmqApyQePUtCndjatj/9I5LylHvt5qluFaBT3I5h3r1ejfR056c+FCv+NnNXg==", + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^11.0.0", + "minipass": "^7.1.2" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/pend": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", + "integrity": "sha512-F3asv42UuXchdzt+xXqfW1OGlVBe+mxa2mqI0pg5yAHZPvFmY3Y6drSf/GQ1A86WgWEN9Kzh/WrgKa6iGcHXLg==", + "license": "MIT" + }, + "node_modules/plist": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/plist/-/plist-3.1.0.tgz", + "integrity": "sha512-uysumyrvkUX0rX/dEVqt8gC3sTBzd4zoWfLeS29nb53imdaXVvLINYXTI2GNqzaMuvacNx4uJQ8+b3zXR0pkgQ==", + "license": "MIT", + "dependencies": { + "@xmldom/xmldom": "^0.8.8", + "base64-js": "^1.5.1", + "xmlbuilder": "^15.1.1" + }, + "engines": { + "node": ">=10.4.0" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "license": "MIT", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/prompts/node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "license": "MIT", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/rimraf": { + "version": "6.1.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.1.3.tgz", + "integrity": "sha512-LKg+Cr2ZF61fkcaK1UdkH2yEBBKnYjTyWzTJT6KNPcSPaiT7HSdhtMXQuN5wkTX0Xu72KQ1l8S42rlmexS2hSA==", + "license": "BlueOak-1.0.0", + "dependencies": { + "glob": "^13.0.3", + "package-json-from-dist": "^1.0.1" + }, + "bin": { + "rimraf": "dist/esm/bin.mjs" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/sax": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.1.4.tgz", + "integrity": "sha512-5f3k2PbGGp+YtKJjOItpg3P99IMD84E4HOvcfleTb5joCHNXYLsR9yWFPOYGgaeMPDubQILTCMdsFb2OMeOjtg==", + "license": "ISC" + }, + "node_modules/semver": { + "version": "7.7.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz", + "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "license": "ISC" + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "license": "MIT" + }, + "node_modules/slice-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", + "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "astral-regex": "^2.0.0", + "is-fullwidth-code-point": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/slice-ansi?sponsor=1" + } + }, + "node_modules/split2": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz", + "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==", + "license": "ISC", + "engines": { + "node": ">= 10.x" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tar": { + "version": "7.5.9", + "resolved": "https://registry.npmjs.org/tar/-/tar-7.5.9.tgz", + "integrity": "sha512-BTLcK0xsDh2+PUe9F6c2TlRp4zOOBMTkoQHQIWSIzI0R7KG46uEwq4OPk2W7bZcprBMsuaeFsqwYr7pjh6CuHg==", + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/fs-minipass": "^4.0.0", + "chownr": "^3.0.0", + "minipass": "^7.1.2", + "minizlib": "^3.1.0", + "yallist": "^5.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/through2": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-4.0.2.tgz", + "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", + "license": "MIT", + "dependencies": { + "readable-stream": "3" + } + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "license": "MIT", + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "license": "0BSD" + }, + "node_modules/undici-types": { + "version": "7.18.2", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.18.2.tgz", + "integrity": "sha512-AsuCzffGHJybSaRrmr5eHr81mwJU3kjw6M+uprWvCXiNeN9SOGwQ3Jn8jb8m3Z6izVgknn1R0FTCEAP2QrLY/w==", + "license": "MIT" + }, + "node_modules/universalify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", + "license": "MIT", + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/untildify": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/untildify/-/untildify-4.0.0.tgz", + "integrity": "sha512-KK8xQ1mkzZeg9inewmFVDNkg3l5LUhoq9kN6iWYB/CC9YMG8HA+c1Q8HwDe6dEX7kErrEVNVBO3fWsVq5iDgtw==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "license": "MIT" + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/xml2js": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.6.2.tgz", + "integrity": "sha512-T4rieHaC1EXcES0Kxxj4JWgaUQHDk+qwHcYOCFHfiwKz7tOVPLq7Hjq9dM1WCMhylqMEfP7hMcOIChvotiZegA==", + "license": "MIT", + "dependencies": { + "sax": ">=0.6.0", + "xmlbuilder": "~11.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/xml2js/node_modules/xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", + "license": "MIT", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/xmlbuilder": { + "version": "15.1.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-15.1.1.tgz", + "integrity": "sha512-yMqGBqtXyeN1e3TGYvgNgDVZ3j84W4cwkOXQswghol6APgZWaff9lnbvN7MHYJOiXsvGPXtjTYJEiC9J2wv9Eg==", + "license": "MIT", + "engines": { + "node": ">=8.0" + } + }, + "node_modules/yallist": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-5.0.0.tgz", + "integrity": "sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==", + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=18" + } + }, + "node_modules/yauzl": { + "version": "2.10.0", + "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.10.0.tgz", + "integrity": "sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g==", + "license": "MIT", + "dependencies": { + "buffer-crc32": "~0.2.3", + "fd-slicer": "~1.1.0" + } + } + } +} diff --git a/dewemoji-capacitor/package.json b/dewemoji-capacitor/package.json new file mode 100644 index 0000000..8fafcbe --- /dev/null +++ b/dewemoji-capacitor/package.json @@ -0,0 +1,17 @@ +{ + "name": "dewemoji-capacitor", + "version": "1.0.0", + "main": "index.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "keywords": [], + "author": "", + "license": "ISC", + "description": "", + "dependencies": { + "@capacitor/android": "^8.1.0", + "@capacitor/cli": "^7.5.0", + "@capacitor/core": "^8.1.0" + } +} diff --git a/dewemoji-capacitor/www/index.html b/dewemoji-capacitor/www/index.html new file mode 100644 index 0000000..17b413b --- /dev/null +++ b/dewemoji-capacitor/www/index.html @@ -0,0 +1,23 @@ + + + + + + Dewemoji + + + +
+
+

Dewemoji

+

If this screen appears, the app could not load https://dewemoji.com. Check internet connection and try again.

+
+
+ + diff --git a/scripts/apk/build-release.sh b/scripts/apk/build-release.sh new file mode 100755 index 0000000..850bc77 --- /dev/null +++ b/scripts/apk/build-release.sh @@ -0,0 +1,69 @@ +#!/usr/bin/env bash +set -euo pipefail + +ROOT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd)" +ANDROID_DIR="${ROOT_DIR}/dewemoji-capacitor/android" +APP_GRADLE="${ANDROID_DIR}/app/build.gradle" +DIST_DIR="${ROOT_DIR}/dewemoji-capacitor/dist/apk" + +if [[ ! -f "${APP_GRADLE}" ]]; then + echo "error: missing ${APP_GRADLE}" >&2 + exit 1 +fi + +version_name="$(awk '/versionName /{gsub(/"/, "", $2); print $2; exit}' "${APP_GRADLE}")" +version_code="$(awk '/versionCode /{print $2; exit}' "${APP_GRADLE}")" +if [[ -z "${version_name}" || -z "${version_code}" ]]; then + echo "error: failed to read versionName/versionCode from ${APP_GRADLE}" >&2 + exit 1 +fi + +mkdir -p "${DIST_DIR}" + +echo "== Build release APK ==" +( + cd "${ANDROID_DIR}" + ./gradlew clean assembleRelease +) + +unsigned_apk="${ANDROID_DIR}/app/build/outputs/apk/release/app-release-unsigned.apk" +signed_apk_default="${ANDROID_DIR}/app/build/outputs/apk/release/app-release.apk" +input_apk="" + +if [[ -f "${signed_apk_default}" ]]; then + input_apk="${signed_apk_default}" +elif [[ -f "${unsigned_apk}" ]]; then + input_apk="${unsigned_apk}" +else + echo "error: release APK not found under app/build/outputs/apk/release" >&2 + exit 1 +fi + +output_apk="${DIST_DIR}/dewemoji-v${version_name}-${version_code}.apk" + +if [[ -n "${ANDROID_KEYSTORE_PATH:-}" && -n "${ANDROID_KEYSTORE_PASSWORD:-}" && -n "${ANDROID_KEY_ALIAS:-}" && -n "${ANDROID_KEY_PASSWORD:-}" ]]; then + if ! command -v apksigner >/dev/null 2>&1; then + echo "error: apksigner is required for signing but not found" >&2 + exit 1 + fi + + echo "== Sign APK ==" + apksigner sign \ + --ks "${ANDROID_KEYSTORE_PATH}" \ + --ks-pass "pass:${ANDROID_KEYSTORE_PASSWORD}" \ + --ks-key-alias "${ANDROID_KEY_ALIAS}" \ + --key-pass "pass:${ANDROID_KEY_PASSWORD}" \ + --out "${output_apk}" \ + "${input_apk}" + + apksigner verify --verbose "${output_apk}" >/dev/null +else + echo "warning: signing env vars are not fully set; copying unsigned/gradle output as-is" + cp "${input_apk}" "${output_apk}" +fi + +sha256="$(shasum -a 256 "${output_apk}" | awk '{print $1}')" + +echo "Built APK: ${output_apk}" +echo "Version: ${version_name} (${version_code})" +echo "SHA256: ${sha256}" diff --git a/scripts/apk/make-version-json.sh b/scripts/apk/make-version-json.sh new file mode 100755 index 0000000..0c9af7b --- /dev/null +++ b/scripts/apk/make-version-json.sh @@ -0,0 +1,78 @@ +#!/usr/bin/env bash +set -euo pipefail + +usage() { + cat < \ + --notes "Release notes" \ + [--out ./version.json] \ + [--apk-url https://dewemoji.com/downloads/dewemoji-latest.apk] \ + [--app-id com.dewemoji.app] \ + [--channel stable] \ + [--min-supported-version-code 100] \ + [--force false] +USAGE +} + +out="./version.json" +apk_url="https://dewemoji.com/downloads/dewemoji-latest.apk" +app_id="com.dewemoji.app" +channel="stable" +min_supported_version_code="100" +force="false" +version_name="" +version_code="" +sha256="" +notes="" +published_at="$(date -u +"%Y-%m-%dT%H:%M:%SZ")" + +while [[ $# -gt 0 ]]; do + case "$1" in + --version-name) version_name="$2"; shift 2 ;; + --version-code) version_code="$2"; shift 2 ;; + --sha256) sha256="$2"; shift 2 ;; + --notes) notes="$2"; shift 2 ;; + --out) out="$2"; shift 2 ;; + --apk-url) apk_url="$2"; shift 2 ;; + --app-id) app_id="$2"; shift 2 ;; + --channel) channel="$2"; shift 2 ;; + --min-supported-version-code) min_supported_version_code="$2"; shift 2 ;; + --force) force="$2"; shift 2 ;; + --published-at) published_at="$2"; shift 2 ;; + -h|--help) usage; exit 0 ;; + *) echo "error: unknown argument '$1'" >&2; usage; exit 1 ;; + esac +done + +if [[ -z "${version_name}" || -z "${version_code}" || -z "${sha256}" ]]; then + echo "error: --version-name, --version-code, and --sha256 are required" >&2 + usage + exit 1 +fi + +python3 - <&2 + exit 1 + fi +done + +if ! command -v aws >/dev/null 2>&1; then + echo "error: aws cli is required" >&2 + exit 1 +fi + +ROOT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd)" +MAKE_VERSION_SCRIPT="${ROOT_DIR}/scripts/apk/make-version-json.sh" + +apk="" +version_name="" +version_code="" +notes="" +min_supported_version_code="100" +force="false" + +while [[ $# -gt 0 ]]; do + case "$1" in + --apk) apk="$2"; shift 2 ;; + --version-name) version_name="$2"; shift 2 ;; + --version-code) version_code="$2"; shift 2 ;; + --notes) notes="$2"; shift 2 ;; + --min-supported-version-code) min_supported_version_code="$2"; shift 2 ;; + --force) force="$2"; shift 2 ;; + -h|--help) usage; exit 0 ;; + *) echo "error: unknown argument '$1'" >&2; usage; exit 1 ;; + esac +done + +if [[ -z "${apk}" || -z "${version_name}" || -z "${version_code}" ]]; then + echo "error: --apk, --version-name, and --version-code are required" >&2 + usage + exit 1 +fi + +if [[ ! -f "${apk}" ]]; then + echo "error: apk file not found: ${apk}" >&2 + exit 1 +fi + +endpoint="https://${R2_ACCOUNT_ID}.r2.cloudflarestorage.com" +export AWS_ACCESS_KEY_ID="${R2_ACCESS_KEY_ID}" +export AWS_SECRET_ACCESS_KEY="${R2_SECRET_ACCESS_KEY}" + +tmp_dir="$(mktemp -d)" +trap 'rm -rf "${tmp_dir}"' EXIT + +sha256="$(shasum -a 256 "${apk}" | awk '{print $1}')" +versioned_key="apk/dewemoji-v${version_name}-${version_code}.apk" +latest_key="apk/dewemoji-latest.apk" +version_json_key="apk/version.json" + +apk_url="${DEWEMOJI_APK_URL:-https://dewemoji.com/downloads/dewemoji-latest.apk}" +version_json_path="${tmp_dir}/version.json" +"${MAKE_VERSION_SCRIPT}" \ + --version-name "${version_name}" \ + --version-code "${version_code}" \ + --sha256 "${sha256}" \ + --notes "${notes}" \ + --apk-url "${apk_url}" \ + --min-supported-version-code "${min_supported_version_code}" \ + --force "${force}" \ + --out "${version_json_path}" + +echo "== Upload versioned APK ==" +aws --endpoint-url "${endpoint}" s3 cp "${apk}" "s3://${R2_BUCKET}/${versioned_key}" --content-type application/vnd.android.package-archive + +echo "== Upload latest APK alias ==" +aws --endpoint-url "${endpoint}" s3 cp "${apk}" "s3://${R2_BUCKET}/${latest_key}" --content-type application/vnd.android.package-archive + +echo "== Upload version metadata ==" +aws --endpoint-url "${endpoint}" s3 cp "${version_json_path}" "s3://${R2_BUCKET}/${version_json_key}" --content-type application/json --cache-control no-store + +echo "Published to R2 bucket: ${R2_BUCKET}" +echo "Versioned APK key: ${versioned_key}" +echo "Latest APK key: ${latest_key}" +echo "Version JSON key: ${version_json_key}" + +if [[ -n "${R2_PUBLIC_BASE_URL:-}" ]]; then + base="${R2_PUBLIC_BASE_URL%/}" + echo "Public versioned APK URL: ${base}/${versioned_key}" + echo "Public latest APK URL: ${base}/${latest_key}" + echo "Public version JSON URL: ${base}/${version_json_key}" +fi diff --git a/scripts/apk/verify-release.sh b/scripts/apk/verify-release.sh new file mode 100755 index 0000000..6f68ebd --- /dev/null +++ b/scripts/apk/verify-release.sh @@ -0,0 +1,67 @@ +#!/usr/bin/env bash +set -euo pipefail + +usage() { + cat <&2; usage; exit 1 ;; + esac +done + +version_url="${base_url%/}/version.json" +apk_url="${base_url%/}/dewemoji-latest.apk" + +tmp_dir="$(mktemp -d)" +trap 'rm -rf "${tmp_dir}"' EXIT + +version_file="${tmp_dir}/version.json" +apk_file="${tmp_dir}/dewemoji-latest.apk" + +echo "== Fetch version metadata ==" +curl -fsSL "${version_url}" -o "${version_file}" +python3 - <&2 + exit 1 +fi + +echo "OK: release metadata and APK checksum are consistent"