Выпуск пакета android с поддержкой реакции не запускается на некоторых устройствах

Я работаю над мобильным приложением и разрабатываю его с помощью react-native. Я могу успешно запустить проект на своем эмуляторе через

react-native run-android 

и я могу успешно получить выпуск пакета с помощью команды

./gradlew bundleRelease

И я даже могу успешно загрузить свое приложение в Google Play Store. Проблема в том, что некоторые устройства не смогут установить приложение.

После поиска в Интернете я понял, что проблема заключалась в различных архитектурах процессора на разных устройствах (насколько я понял). И мой эмулятор использует 32-битный процессор (x86), и сборка успешно выполняется на моем эмуляторе.

У меня нет устройства Android, поэтому лучший способ, который я нашел для отладки ошибки, - это использовать «Test Lab» Firebase и наблюдать за результатом.

Я считаю, что есть еще одна проблема, отличная от архитектуры процессора, а именно Hermes. Я отключаю hermes в моем app / build.gradle, однако, когда я выпускаю пакет и тестирую его в Firebase Test Lab, приложение не сработает и выдаст мне этот результат ;

java.lang.UnsatisfiedLinkError: couldn't find DSO to load: libhermes.so
     FATAL EXCEPTION: create_react_context
Process: com.lockerapp, PID: 18180
java.lang.UnsatisfiedLinkError: couldn't find DSO to load: libhermes.so
    at com.facebook.soloader.SoLoader.doLoadLibraryBySoName(SoLoader.java:738)
    at com.facebook.soloader.SoLoader.loadLibraryBySoName(SoLoader.java:591)
    at com.facebook.soloader.SoLoader.loadLibrary(SoLoader.java:529)
    at com.facebook.soloader.SoLoader.loadLibrary(SoLoader.java:484)
    at com.facebook.hermes.reactexecutor.HermesExecutor.<clinit>(HermesExecutor.java:20)
    at com.facebook.hermes.reactexecutor.HermesExecutorFactory.create(HermesExecutorFactory.java:27)
    at com.facebook.react.ReactInstanceManager$5.run(ReactInstanceManager.java:949)
    at java.lang.Thread.run(Thread.java:764)

Вот мое окружение;

System:
    OS: macOS 10.14.4
    CPU: (4) x64 Intel(R) Core(TM) i5-4278U CPU @ 2.60GHz
    Memory: 242.76 MB / 8.00 GB
    Shell: 3.2.57 - /bin/bash
  Binaries:
    Node: 12.9.0 - /usr/local/bin/node
    Yarn: 1.17.3 - /usr/local/bin/yarn
    npm: 6.10.3 - /usr/local/bin/npm
    Watchman: 4.9.0 - /usr/local/bin/watchman
  SDKs:
    iOS SDK:
      Platforms: iOS 12.4, macOS 10.14, tvOS 12.4, watchOS 5.3
    Android SDK:
      API Levels: 25, 26, 27, 28, 29
      Build Tools: 26.0.2, 27.0.3, 28.0.3, 29.0.2
      System Images: android-28 | Intel x86 Atom_64, android-28 | Google Play Intel x86 Atom
  IDEs:
    Android Studio: 3.4 AI-183.6156.11.34.5692245
    Xcode: 10.3/10G8 - /usr/bin/xcodebuild
  npmPackages:
    react: 16.8.6 => 16.8.6 
    react-native: 0.60.5 => 0.60.5 
  npmGlobalPackages:
    create-react-native-app: 2.0.2
    react-native-cli: 2.0.1

А вот и мой app / build.gralde;

apply plugin: "com.android.application"

import com.android.build.OutputFile

project.ext.react = [
    entryFile: "index.js",
    enableHermes: false,  // clean and rebuild if changing
]

apply from: "../../node_modules/react-native/react.gradle"


def enableHermes = project.ext.react.get("enableHermes", false);

android {
    compileSdkVersion rootProject.ext.compileSdkVersion

    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }

    defaultConfig {
        applicationId "com.lockerapp"
        minSdkVersion rootProject.ext.minSdkVersion
        targetSdkVersion rootProject.ext.targetSdkVersion
        versionCode 11
        versionName "1.1.5"
    }
    splits {
        abi {
            reset()
            enable enableSeparateBuildPerCPUArchitecture
            universalApk false  // If true, also generate a universal APK
            include "armeabi-v7a", "x86", "arm64-v8a", "x86_64"
        }
    }
    signingConfigs {
        debug {
            storeFile file('debug.keystore')
            storePassword 'android'
            keyAlias 'androiddebugkey'
            keyPassword 'android'
        }
    release {
            if (project.hasProperty('MYAPP_UPLOAD_STORE_FILE')) {
                storeFile file(MYAPP_UPLOAD_STORE_FILE)
                storePassword MYAPP_UPLOAD_STORE_PASSWORD
                keyAlias MYAPP_UPLOAD_KEY_ALIAS
                keyPassword MYAPP_UPLOAD_KEY_PASSWORD
            }
        }
    }
    buildTypes {
        debug {
            signingConfig signingConfigs.debug
        }
        release {
            // Caution! In production, you need to generate your own keystore file.
            // see https://facebook.github.io/react-native/docs/signed-apk-android.
            signingConfig signingConfigs.release
        minifyEnabled enableProguardInReleaseBuilds
            proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
        }
    }
    // applicationVariants are e.g. debug, release
    applicationVariants.all { variant ->
        variant.outputs.each { output ->
            // For each separate APK per architecture, set a unique version code as described here:
            // https://developer.android.com/studio/build/configure-apk-splits.html
            def versionCodes = ["armeabi-v7a": 1, "x86": 2, "arm64-v8a": 3, "x86_64": 4]
            def abi = output.getFilter(OutputFile.ABI)
            if (abi != null) {  // null for the universal-debug, universal-release variants
                output.versionCodeOverride =
                        versionCodes.get(abi) * 1048576 + defaultConfig.versionCode
            }

        }
    }

    packagingOptions {
        pickFirst '**/armeabi-v7a/libc++_shared.so'
        pickFirst '**/x86/libc++_shared.so'
        pickFirst '**/arm64-v8a/libc++_shared.so'
        pickFirst '**/x86_64/libc++_shared.so'
        pickFirst '**/x86/libjsc.so'
        pickFirst '**/armeabi-v7a/libjsc.so'
    }
}

dependencies {
    implementation fileTree(dir: "libs", include: ["*.jar"])
    implementation "com.facebook.react:react-native:+"  // From node_modules

    if (enableHermes) {
      def hermesPath = "../../node_modules/hermesvm/android/";
      debugİmplementation files(hermesPath + "hermes-debug.aar")
      releaseİmplementation files(hermesPath + "hermes-release.aar")
    } else {
      implementation jscFlavor
    }
}

// Run this once to be able to run the application with BUCK
// puts all compile dependencies into folder libs for BUCK to use
task copyDownloadableDepsToLibs(type: Copy) {
    from configurations.compile
    into 'libs'
}

apply from: file("../../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesAppBuildGradle(project)

Я изо всех сил пытаюсь выпустить пакет для как можно большего количества устройств и опубликовать его в Google Play Store. Меня не волнует использование Hermes или нет, поскольку моя сборка успешна и работает на всех возможных устройствах.

Есть ли известное руководство, которое я мог пропустить, чтобы выпустить пакет для всех архитектур процессоров и избежать ошибок с Hermes? Любой ответ приветствуется, спасибо.

ОБНОВЛЕНИЕ:

После долгих исследований я все еще не мог найти решение для этого, однако я нашел решение проблемы вместо создания файла .aab с

./gradlew bundleRelase 

я создаю универсальный файл APK с помощью команды;

./gradlew assembleRelease

Сгенерированный файл .apk принимается магазином Google Play, и я прошел тесты в Firebase Test Lab, также работает на 32-битных и 64-битных локальных эмуляторах Android, с обратной стороны, конечно, размер моей загрузки увеличился (с 8 МБ до 15 МБ), но тем не менее, это решение на какое-то время.


person Emre Bicer    schedule 07.09.2019    source источник
comment
Вы определили jscFlavor в своем файле build.gradle?   -  person Anus Kaleem    schedule 07.09.2019
comment
@AnusKaleem определяется как def jscFlavor = 'org.webkit: android-jsc: +', возможно, пропустил при написании моего файла build.gradle здесь, я старался по возможности избегать комментариев ...   -  person Emre Bicer    schedule 07.09.2019
comment
Можете ли вы проанализировать свой apk с помощью Android Studio, чтобы увидеть, есть ли библиотеки libhermes.so в папке libs для каждой архитектуры?   -  person Anus Kaleem    schedule 07.09.2019
comment
@AnusKaleem Изначально у меня не было APK, так как моя цель - создать aab, получить APK. Я выполнил шаги в этой ссылке; facebook.github.io/react-native/docs/signed-apk- android, и после запуска ./gradlew assemblyRelease я получил свой app-universal-release.apk, и когда я проверил его, как вы сказали, я обнаружил x86_64, arm64-v8a, x86, armeabi-v7a, armeabi под lib и ни под одним из них libhermes.so нет.   -  person Emre Bicer    schedule 07.09.2019
comment
Это означает, что библиотеки hermes не были установлены в первую очередь ... следуйте этому ответу и ответу, следующему за ним, проверьте, не упускаете ли вы что-то здесь stackoverflow .com / a / 57202275/3652368   -  person Anus Kaleem    schedule 07.09.2019
comment
@AnusKaleem Я видел этот вопрос в stackoverflow раньше, и, к сожалению, мое приложение / build.gradle ничего не упускает из этого ответа ...   -  person Emre Bicer    schedule 07.09.2019


Ответы (1)


Мне удалось включить hermes, а также создать пакет и отправить его в игровой магазин с этим исправлением.

if (getGradle().getStartParameter().getTaskRequests().toString().toLowerCase().contains("bundle")) {
    packagingOptions {
        exclude '**/libhermes-inspector.so'
        exclude '**/libhermes-executor-debug.so'
        exclude '**/libjscexecutor.so'
    }
}

https://github.com/facebook/react-native/issues/25927#issuecomment-535695869

если вы хотите протестировать окончательный bundle файл, я рекомендую пару вещей:

  1. Загрузите инструмент для создания пакетов Android (BundleTool GitHub)
  2. Создайте профиль на одном устройстве Android

    java -jar bundletool.jar get-device-spec --output=device-spec.json

  3. Создайте файлы apks для этого конкретного профиля

    java -jar bundletool.jar build-apks --device-spec=device-spec.json --bundle=app-release.aab --output=device.apks

  4. Установите сгенерированный APK на устройство

    java -jar bundletool.jar install-apks --apks=device.apks

  5. Кроме того, вы можете использовать анализатор apk android, изучить этот APK и убедиться, что двоичный файл hermes присутствует, а двоичный файл, связанный с libjs, отсутствует.

person BlaShadow    schedule 31.10.2019
comment
Примечание: adb не устанавливает файлы .apks, в отличие от bundletool. - person Pierre; 01.11.2019
comment
@Pierre используйте adb install -r -t app-release.apk для установки apks;) - person Alex Lévy; 21.01.2020
comment
Нет, ADB устанавливает файлы .apk, а не файлы .apks. Обратите внимание на дополнительные s. - person Pierre; 22.01.2020
comment
Спасибо! Мне было труднее всего понять, почему Google Play показывал следующую ошибку после загрузки моего пакета приложений: следующие APK или пакеты приложений доступны для 64-разрядных устройств, но у них есть только 32-разрядный собственный код. Я трижды проверил, что все 64-битные библиотеки присутствуют внутри aab файла. Запуск bundletool показывает: ПРЕДУПРЕЖДЕНИЕ. Пакет приложений содержит 32-разрядный файл битового кода RenderScript (.bc), который отключает поддержку 64-разрядной версии в Android. 64-битные собственные библиотеки не будут включены в сгенерированные APK. - person Daniel Chesher; 15.02.2020