Как использовать диспетчер работы с GoogleApiClient и модернизировать?

Поэтому я использую Worker Manger, чтобы открыть свое приложение в фоновом режиме, собрать данные из Google Fit и отправить их на сервер.

  • Я попытался вызвать API внутри моего работника, и это работает хорошо.
  • Затем я попытался подключиться к Google Fit и собрать данные, и это работало также хорошо.

но когда я попытался объединить 2 пункта вместе, это не сработало. это всегда бросает android.os.NetworkOnMainThreadException с нулевым сообщением

Кто-нибудь может дать мне причину, почему это происходит?

вот мой код

class TrackWorker(context: Context, params: WorkerParameters) : Worker(context, params), GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener {

@Inject
lateinit var useCase: TrackUseCase
val TAG: String = TrackWorker::class.java.simpleName
private var client: GoogleApiClient? = null
private var totalSteps: Int = 0
private var totalDistance: Float = 0f


override fun onConnectionFailed(p0: ConnectionResult) {
    Log.d(TAG, "Failed")
}

override fun onConnected(p0: Bundle?) {
    val ESTIMATED_STEP_DELTAS = DataSource.Builder()
            .setAppPackageName("com.google.android.gms")
            .setDataType(DataType.TYPE_STEP_COUNT_DELTA)
            .setType(DataSource.TYPE_DERIVED)
            .setStreamName("estimated_steps")
            .build()
    val readRequest = DataReadRequest.Builder()
            .aggregate(ESTIMATED_STEP_DELTAS, DataType.AGGREGATE_STEP_COUNT_DELTA)
            .aggregate(DataType.TYPE_DISTANCE_DELTA, DataType.AGGREGATE_DISTANCE_DELTA)
            .aggregate(DataType.TYPE_CALORIES_EXPENDED, DataType.AGGREGATE_CALORIES_EXPENDED)
            .aggregate(DataType.TYPE_ACTIVITY_SEGMENT, DataType.AGGREGATE_ACTIVITY_SUMMARY)
            .bucketByTime(1, TimeUnit.DAYS)
            .setTimeRange(CalendarUtil.getFirstDayOfWeek(), CalendarUtil.getNow(), TimeUnit.MILLISECONDS)
            .build()


    val result = Fitness.HistoryApi.readData(client, readRequest)
    val totalResult = result.setResultCallback { it ->

        if (it.status.isSuccess) {
            val buckets = it.buckets
            for (bucket in buckets) {
                val dataSets = bucket.dataSets

                for (dataSet in dataSets) {
                    if (dataSet.dataType.name == "com.google.step_count.delta") {
                        for (dp in dataSet.dataPoints) {
                            for (field in dp.dataType.fields) {
                                val steps = dp.getValue(field).asInt()
                                totalSteps += steps
                            }
                        }
                    }
                }

                for (dataSet in dataSets) {
                    if (dataSet.dataType.name == "com.google.distance.delta") {
                        for (dp in dataSet.dataPoints) {
                            for (field in dp.dataType.fields) {
                                val distance = dp.getValue(field).asFloat()
                                totalDistance += distance
                            }
                        }
                    }
                }
            }

        } else {
            Log.e(TAG, "There was a problem getting the step count")
        }

        if (totalSteps != 0 && totalDistance != 0f) {
            val jsonArray = JsonArray()
            val trackBodyJSON = JsonObject()


            trackBodyJSON.addProperty("trackDate", CalendarUtil.getCurrentDate())
            trackBodyJSON.addProperty("walksteps", totalSteps)
            trackBodyJSON.addProperty("walkkm", totalDistance)
            jsonArray.add(trackBodyJSON)


            useCase.execute(TrackUseCase.Request(jsonArray))
                    .subscribe({
                        Log.d(TAG, "it's here $it")
                    }, {
                        Log.d(TAG, "errorrrr ${it}")
                        when (it) {
                            is ApiException ->
                                Log.d(TAG, "API ERROR ? ${it.apiErrorResponse.message}")
                            is TimeoutConnectionException -> {
                                Log.d(TAG, "API ERROR ? Timeout")
                            }
                            else -> {
                                Log.d(TAG, "error ${it.localizedMessage}")
                            }
                        }
                    })


        }
    }


}

override fun onConnectionSuspended(p0: Int) {
    Log.d(TAG, "SUSSPENDED")
}


override fun doWork(): Result {
    (applicationContext as WalkAndTalkApp).component().inject(this)

    if (client == null) {
        client = GoogleApiClient.Builder(applicationContext)
                .addApi(Fitness.RECORDING_API)
                .addApi(Fitness.HISTORY_API)
                .addScope(Scope(Scopes.FITNESS_LOCATION_READ))
                .addScope(Scope(Scopes.FITNESS_ACTIVITY_READ_WRITE))
                .addConnectionCallbacks(this)
                .addOnConnectionFailedListener(this)
                .build()

        if (!client!!.isConnected)
            client!!.connect()
    }


    return Result.SUCCESS
}

}

Я пытался использовать цепочечные рабочие менеджеры, но второй рабочий исполняется раньше, чем первый подключается к Google Fit SDK и получает данные

ниже код

 private var totalSteps: Int = 0
private var totalDistance: Float = 0f
val TAG: String = "DAILYTRAKWORKER"

class ConnectWorker(context: Context, params: WorkerParameters) : Worker(context, params), GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener {

    private var client: GoogleApiClient? = null



    override fun onConnectionFailed(p0: ConnectionResult) {
        Log.d(TAG, "Failed")
    }

    override fun onConnected(p0: Bundle?) {
        Log.d(TAG, "CONNNNECTED")

        val ESTIMATED_STEP_DELTAS = DataSource.Builder()
                .setAppPackageName("com.google.android.gms")
                .setDataType(DataType.TYPE_STEP_COUNT_DELTA)
                .setType(DataSource.TYPE_DERIVED)
                .setStreamName("estimated_steps")
                .build()
        val readRequest = DataReadRequest.Builder()
                .aggregate(ESTIMATED_STEP_DELTAS, DataType.AGGREGATE_STEP_COUNT_DELTA)
                .aggregate(DataType.TYPE_DISTANCE_DELTA, DataType.AGGREGATE_DISTANCE_DELTA)
                .aggregate(DataType.TYPE_CALORIES_EXPENDED, DataType.AGGREGATE_CALORIES_EXPENDED)
                .aggregate(DataType.TYPE_ACTIVITY_SEGMENT, DataType.AGGREGATE_ACTIVITY_SUMMARY)
                .bucketByTime(1, TimeUnit.DAYS)
                .setTimeRange(CalendarUtil.getFirstDayOfWeek(), CalendarUtil.getNow(), TimeUnit.MILLISECONDS)
                .build()


        val result = Fitness.HistoryApi.readData(client, readRequest)
        result.setResultCallback { it ->

            if (it.status.isSuccess) {
                val buckets = it.buckets
                for (bucket in buckets) {
                    val dataSets = bucket.dataSets
                    for (dataSet in dataSets) {
                        if (dataSet.dataType.name == "com.google.step_count.delta") {
                            for (dp in dataSet.dataPoints) {
                                for (field in dp.dataType.fields) {
                                    val steps = dp.getValue(field).asInt()
                                    totalSteps += steps
                                    Log.d(TAG, "steps: $steps")
                                }
                            }
                        }
                    }

                    for (dataSet in dataSets) {
                        if (dataSet.dataType.name == "com.google.distance.delta") {
                            for (dp in dataSet.dataPoints) {
                                for (field in dp.dataType.fields) {
                                    val distance = dp.getValue(field).asFloat()
                                    Log.d(TAG, "distance: " + distance / 1000)
                                    totalDistance += distance
                                }
                            }
                        }
                    }
                }

            } else {
                Log.e(TAG, "There was a problem getting the step count")
            }

        }


    }

    override fun onConnectionSuspended(p0: Int) {
        Log.d(TAG, "SUSSPENDED")
    }

    override fun doWork(): Result {

        if (client == null) {
            client = GoogleApiClient.Builder(applicationContext)
                    .addApi(Fitness.RECORDING_API)
                    .addApi(Fitness.HISTORY_API)
                    .addScope(Scope(Scopes.FITNESS_LOCATION_READ))
                    .addScope(Scope(Scopes.FITNESS_ACTIVITY_READ_WRITE))
                    .addConnectionCallbacks(this)
                    .addOnConnectionFailedListener(this)
                    .build()

            if (!client!!.isConnected)
                client!!.connect()
        }
        return Result.SUCCESS
    }


}

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


    @Inject
    lateinit var useCase: TrackUseCase


    override fun doWork(): Result {
        (applicationContext as WalkAndTalkApp).component().inject(this)
        Log.i(TAG, "trackworker:$totalSteps ")

        if (totalSteps != 0 && totalDistance != 0f) {
            val jsonArray = JsonArray()
            val trackBodyJSON = JsonObject()
            Log.i(TAG, "Total steps:$totalSteps ")
            Log.i(TAG, "Total Distance:$totalDistance ")
            Log.d(TAG, "CURRENT DATE ${CalendarUtil.getCurrentDate()}")

            trackBodyJSON.addProperty("trackDate", CalendarUtil.getCurrentDate())
            trackBodyJSON.addProperty("walksteps", totalSteps)
            trackBodyJSON.addProperty("walkkm", totalDistance)
            jsonArray.add(trackBodyJSON)


            Log.d(TAG, "LIST --->   $jsonArray")

            useCase.execute(TrackUseCase.Request(jsonArray))
                    .subscribe({
                        Log.d(TAG, "it's here $it")
                    }, {
                        Log.d(TAG, "errorrrr ${it}")
                        when (it) {
                            is ApiException ->
                                Log.d(TAG, "API ERROR ?       ${it.apiErrorResponse.message}")
                            is TimeoutConnectionException -> {
                                Log.d(TAG, "API ERROR ? Timeout")
                            }

                            else -> {
                                Log.d(TAG, "error ${it.localizedMessage}")
                            }
                        }
                    })
        }

        return Result.SUCCESS
    }


}

0 ответов