Реализация AndroidX Worker с пользовательскими параметрами - PullRequest
2 голосов
/ 09 мая 2019

Я выполняю миграцию из приоритетной библиотеки Android JobQueue в AndroidX WorkManager в моем проекте, но она построена таким образом, что Worker должно выглядеть примерно так:

TaskSchedulerWorker.kt

class TaskSchedulerWorker(
    private val workManager: WorkManager = WorkManager.getInstance()
) : TaskScheduler {

    override fun execute(useCaseWrapper: UseCaseWrapper) {
        workManager.beginUniqueWork(
            useCaseWrapper.useCaseName,
            ExistingWorkPolicy.APPEND,
            OneTimeWorkRequest.from(UseCaseWrapperWorker::class.java)
        ).enqueue()
    }

    override fun onCancel(useCaseWrapper: UseCaseWrapper) {
        workManager.cancelUniqueWork(useCaseWrapper.useCaseName)
    }
}

UseCaseWrapperWorker.kt

class UseCaseWrapperWorker(
    context: Context, 
    params: WorkerParams,
    private val useCaseWrapper: UseCaseWrapper
) : Worker(context, params) {

    override fun doWork(): Result {
        return try {
            useCaseWrapper.execute()
            Result.success()
        } catch (ex: Exception) {
            ex.printStackTrace()
            Result.failure()
        }
    }
}

Дело в том, что если я пытаюсь запустить свое приложениес этими классами я получаю следующую ошибку:

2019-05-09 09:34:45.784 27260-27443/com.mobileapp.debug E/WM-WorkerFactory: Could not instantiate com.mobileapp.domain.jobqueue.UseCaseWrapperWorker
    java.lang.NoSuchMethodException: com.mobileapp.domain.jobqueue.UseCaseWrapperWorker.<init> [class android.content.Context, class androidx.work.WorkerParameters]
        at java.lang.Class.getConstructor0(Class.java:2328)
        at java.lang.Class.getDeclaredConstructor(Class.java:2167)
        at androidx.work.WorkerFactory.createWorkerWithDefaultFallback(WorkerFactory.java:92)
        at androidx.work.impl.WorkerWrapper.runWorker(WorkerWrapper.java:234)
        at androidx.work.impl.WorkerWrapper.run(WorkerWrapper.java:128)
        at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        at java.lang.Thread.run(Thread.java:764)
2019-05-09 09:34:45.784 27260-27443/com.mobileapp.debug E/WM-WorkerWrapper: Could not create Worker com.mobileapp.domain.jobqueue.UseCaseWrapperWorker

Поскольку конструктор Worker не может иметь больше двух параметров по умолчанию (Context и WorkerParameters).

Я не могу понять, как реализовать класс UseCaseWrapperWorker с помощью конструктора, который получает Context, WorkerParameters и UseCaseWrapper, и вызывать его из TaskSchedulerWorker с помощьюOneTimeWorkRequest.from(UseCaseWrapperWorker::class.java) инструкция.


РЕДАКТИРОВАНИЕ:

Я пытался реализовать CustomWorkerFactory и CustomWorker, чтобы передать им UseCaseWrapper.Проблема в том, что мне нужно запустить WorkManager в методе Application#onCreate ( Документация для разработчиков Android ), и у меня нет доступа к UseCaseWrapper, необходимому на данный момент.

TaskSchedulerWorker.kt

(WorkManager init должен быть в Application#onCreate)

class TaskSchedulerWorker(private val appContext: Context) : TaskScheduler {

    override fun execute(useCaseWrapper: UseCaseWrapper) {
        val workerFactory = CustomWorkerFactory(useCaseWrapper)
        val configuration = Configuration.Builder()
            .setWorkerFactory(workerFactory)
            .build()
        WorkManager.initialize(context, configuration)

        WorkManager.getInstance().beginWork(
            "uniqueWorkName",
            ExistingWorkPolicy.APPEND,
            OneTimeWorkRequest.from(CustomWorker::class.java)
        )
    }
}

CustomWorkerFactory.kt

class CustomWorkerFactory(private val useCaseWrapper: UseCaseWrapper) : WorkerFactory() {

    override fun createWorker(
        appContext: Context,
        workerClassName: String,
        workerParameters: WorkerParameters
    ): ListenableWorker? {
        return CustomWorker(appContext, workerParameters, useCaseWrapper)
    }
}

CustomWorker.kt

class CustomWorker(
    context: Context, 
    params: WorkerParameters
) : Worker(context, params) {

    private var useCaseWrapper: UseCaseWrapper? = null

    constructor(
        context: Context,
        params: WorkerParameters,
        useCaseWrapper: UseCaseWrapper
    ) : this(context, params) {
        this.useCaseWrapper = useCaseWrapper
    }

    override fun doWork(): Result {
        return useCaseWrapper?.let { useCaseWrapper ->
            try {
                useCaseWrapper.execute()
                Result.success()
            } catch (ex: Exception) {
                Result.failure()
            }
        } ?: Result.failure()
    }
}

Выдается RuntimeException:

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.mobileapp.debug/com.mobileapp.home.view.activity.HomeActivity}: java.lang.IllegalStateException: WorkManager is already initialized.  
Did you try to initialize it manually without disabling WorkManagerInitializer? See WorkManager#initialize(Context, Configuration) or the class levelJavadoc for more information.
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3086)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3229)
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78)
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108)
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1926)
    at android.os.Handler.dispatchMessage(Handler.java:106)
    at android.os.Looper.loop(Looper.java:214)
    at android.app.ActivityThread.main(ActivityThread.java:6981)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1445)
Caused by: java.lang.IllegalStateException: WorkManager is already initialized.  Did you try to initialize it manually without disabling WorkManagerInitializer? 
See WorkManager#initialize(Context, Configuration) or the class levelJavadoc for more information.
    at androidx.work.impl.WorkManagerImpl.initialize(WorkManagerImpl.java:137)
    at androidx.work.WorkManager.initialize(WorkManager.java:169)
    at com.mobileapp.domain.worker.TaskSchedulerWorker.execute(TaskSchedulerWorker.kt:20)
    at com.mobileapp.domain.usecase.UseCaseHandler.execute(UseCaseHandler.kt:41)
    at com.mobileapp.domain.usecase.UseCaseCall.execute(UseCaseCall.kt:50)
    at com.mobileapp.home.view.presenter.HomePresenter.getConfigurationJSON(HomePresenter.kt:107)
    at com.mobileapp.home.view.activity.HomeActivity.initComponents(HomeActivity.kt:107)
    at com.mobileapp.home.view.activity.HomeActivity.onCreate(HomeActivity.kt:67)
    at android.app.Activity.performCreate(Activity.java:7326)
    at android.app.Activity.performCreate(Activity.java:7317)
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1271)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3066)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3229) 
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78) 
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108) 
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1926) 
    at android.os.Handler.dispatchMessage(Handler.java:106) 
    at android.os.Looper.loop(Looper.java:214) 
    at android.app.ActivityThread.main(ActivityThread.java:6981) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1445) 

Есть предложения?

Ответы [ 2 ]

1 голос
/ 09 мая 2019

Вы должны использовать пользовательский WorkerFactory для инициализации ваших работников.Смотри https://developer.android.com/reference/androidx/work/WorkerFactory

0 голосов
/ 04 июня 2019

вам нужно отключить инициализатор по умолчанию

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

Отключение инициализатора по умолчанию

Чтобы предоставить собственную конфигурацию, сначала необходимо удалить инициализатор по умолчанию.Для этого обновите AndroidManifest.xml с помощью инструментов правила слияния: node = "remove":

<provider
        android:name="androidx.work.impl.WorkManagerInitializer"
        android:authorities="${applicationId}.workmanager-init"
        tools:node="remove" />

https://developer.android.com/topic/libraries/architecture/workmanager/advanced/custom-configuration

...