All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.lemfi.kest.cadence.executor.ActivityExecution.kt Maven / Gradle / Ivy

There is a newer version: 0.8.1
Show newest version
package com.github.lemfi.kest.cadence.executor

import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
import com.github.lemfi.kest.core.model.Execution
import com.google.gson.Gson
import com.google.gson.internal.LinkedTreeMap
import com.uber.cadence.activity.ActivityOptions
import com.uber.cadence.client.WorkflowClient
import com.uber.cadence.client.WorkflowOptions
import com.uber.cadence.common.RetryOptions
import com.uber.cadence.context.ContextPropagator
import com.uber.cadence.converter.DataConverter
import com.uber.cadence.worker.Worker
import com.uber.cadence.worker.WorkerOptions
import com.uber.cadence.workflow.Workflow
import com.uber.cadence.workflow.WorkflowMethod
import org.opentest4j.AssertionFailedError
import java.lang.reflect.Type
import java.time.Duration
import kotlin.reflect.KFunction
import kotlin.reflect.javaType

class ActivityExecution(
        private val cadenceHost: String,
        private val cadencePort: Int,
        private val cadenceDomain: String,
        private val tasklist: String,

        private val cls: Class,

        private val activity: KFunction,
        private val params: Array?,
        private val contextPropagators: List?,

        override val withResult: RESULT.()->Unit = {},

        ): Execution() {

    @ExperimentalStdlibApi
    @Suppress("unchecked_cast")
    override fun execute(): RESULT {

        Worker.Factory(cadenceHost, cadencePort, cadenceDomain).apply {
            val worker = newWorker("KEST_TL", WorkerOptions
                    .Builder()
                    .setReportActivityFailureRetryOptions(RetryOptions.Builder()
                            .setInitialInterval(Duration.ofSeconds(5))
                            .setExpiration(Duration.ofMinutes(30))
                            .setMaximumInterval(Duration.ofMinutes(1))
                            .setMaximumAttempts(10)
                            .build())
                    .build())

            worker.registerWorkflowImplementationTypes(com.github.lemfi.kest.cadence.executor.Workflow::class.java)

        }.start()

        val parameterTypes = activity.parameters.subList(1, activity.parameters.size).also {
            if ((params?.size ?: 0) != it.size) throw AssertionFailedError("Wrong number of parameter for activity, expected [${it.map { it.type }.joinToString(", ")}] got ${params?.toList()}")
        }

        return WorkflowClient.newInstance(cadenceHost, cadencePort, cadenceDomain)
                .newWorkflowStub(IWorkflow::class.java, WorkflowOptions.Builder()
                        .setExecutionStartToCloseTimeout(Duration.ofSeconds(30))
                        .setTaskList("KEST_TL")
                        .apply {
                            contextPropagators?.let {
                                setContextPropagators(it)
                            }
                        }
                        .build())

                .run(WorkflowParameter(activity.parameters[0].type::javaType.get().typeName, tasklist, activity.name, params?.mapIndexed { index, it ->
                    Parameter(jacksonObjectMapper().writeValueAsString(it), parameterTypes[index].type.javaType.typeName)
                })).let {
                    when {
                        it == null -> null as RESULT
                        cls.isAssignableFrom(it::class.java) -> it as RESULT
                        it is LinkedTreeMap<*, *> -> it.toString().let {
                            Gson().fromJson(it, cls)
                        }
                        else -> throw IllegalArgumentException("blah")
                    }
                }


    }
}

class WorkflowParameter(
        val className: String,
        val tasklist: String,
        val function: String,
        val parameters: List?
)

interface IWorkflow {

    @WorkflowMethod
    fun run(parameters: WorkflowParameter): Any?
}

class Workflow: IWorkflow {

    override fun run(parameters: WorkflowParameter): Any? {

        val cls = Class.forName(parameters.className)

        val params = parameters.parameters?.map { jacksonObjectMapper().readValue(it.value, Class.forName(it.cls)) }
        val types = parameters.parameters?.map { Class.forName(it.cls) }

        val method = types?.let { cls.getMethod(parameters.function, *types.toTypedArray()) } ?: cls.getMethod(parameters.function)

        return if (params != null) {
            method.invoke(getActivity(cls, parameters.tasklist), *params.toTypedArray())
        } else {
            method.invoke(getActivity(cls, parameters.tasklist))
        }
    }

    fun  getActivity(cls: Class, tasklist: String): T {
        return Workflow.newActivityStub(cls,
                ActivityOptions.Builder()
                        .setTaskList(tasklist)
                        .setScheduleToCloseTimeout(Duration.ofSeconds(600))
                        .setRetryOptions(
                                RetryOptions.Builder()
                                        .setInitialInterval(Duration.ofSeconds(1))
                                        .setExpiration(Duration.ofMinutes(1))
                                        .setMaximumAttempts(5)
                                        .build()
                        )
                        .build()
        )
    }
}

data class Parameter(
        val value: String?,
        val cls: String
)




© 2015 - 2025 Weber Informatics LLC | Privacy Policy