From 6eca49df30836633cb584e2a62301825fab6f883 Mon Sep 17 00:00:00 2001 From: Robert Jaros Date: Wed, 15 Apr 2020 20:16:39 +0200 Subject: Increase the number of parameters for server side interface methods --- .../pl/treksoft/kvision/remote/KVServiceManager.kt | 12 ++++++ .../pl/treksoft/kvision/remote/KVServiceManager.kt | 19 +++++++- .../pl/treksoft/kvision/remote/KVServiceManager.kt | 50 ++++++++++++++++++++++ 3 files changed, 80 insertions(+), 1 deletion(-) (limited to 'kvision-modules/kvision-server-javalin/src') diff --git a/kvision-modules/kvision-server-javalin/src/commonMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt b/kvision-modules/kvision-server-javalin/src/commonMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt index 19da13c0..3ba4998d 100644 --- a/kvision-modules/kvision-server-javalin/src/commonMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt +++ b/kvision-modules/kvision-server-javalin/src/commonMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt @@ -103,6 +103,18 @@ expect open class KVServiceManager(serviceClass: KClass) { route: String? = null ) + /** + * Binds a given route with a function of the receiver. + * @param function a function of the receiver + * @param method a HTTP method + * @param route a route + */ + protected inline fun bind( + noinline function: suspend T.(PAR1, PAR2, PAR3, PAR4, PAR5, PAR6) -> RET, + method: HttpMethod = HttpMethod.POST, + route: String? = null + ) + /** * Binds a given function of the receiver as a tabulator component source * @param function a function of the receiver diff --git a/kvision-modules/kvision-server-javalin/src/jsMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt b/kvision-modules/kvision-server-javalin/src/jsMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt index 23da6ac5..c7a35150 100644 --- a/kvision-modules/kvision-server-javalin/src/jsMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt +++ b/kvision-modules/kvision-server-javalin/src/jsMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt @@ -28,7 +28,7 @@ import kotlin.reflect.KClass /** * Multiplatform service manager. */ -actual open class KVServiceManager actual constructor(serviceClass: KClass): KVServiceMgr { +actual open class KVServiceManager actual constructor(serviceClass: KClass) : KVServiceMgr { protected val calls: MutableMap> = mutableMapOf() var counter: Int = 0 @@ -128,6 +128,23 @@ actual open class KVServiceManager actual constructor(serviceClass: KCl calls[function.toString().replace("\\s".toRegex(), "")] = Pair("/kv/$routeDef", method) } + /** + * Binds a given route with a function of the receiver. + * @param function a function of the receiver + * @param method a HTTP method + * @param route a route + */ + protected actual inline fun bind( + noinline function: suspend T.(PAR1, PAR2, PAR3, PAR4, PAR5, PAR6) -> RET, + method: HttpMethod, route: String? + ) { + if (method == HttpMethod.GET) + throw UnsupportedOperationException("GET method is only supported for methods without parameters") + val routeDef = route ?: "route${this::class.simpleName}${counter++}" + calls[function.toString().replace("\\s".toRegex(), "")] = Pair("/kv/$routeDef", method) + } + /** * Binds a given function of the receiver as a tabulator component source * @param function a function of the receiver diff --git a/kvision-modules/kvision-server-javalin/src/jvmMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt b/kvision-modules/kvision-server-javalin/src/jvmMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt index e3189a74..33c858f4 100644 --- a/kvision-modules/kvision-server-javalin/src/jvmMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt +++ b/kvision-modules/kvision-server-javalin/src/jvmMain/kotlin/pl/treksoft/kvision/remote/KVServiceManager.kt @@ -383,6 +383,56 @@ actual open class KVServiceManager actual constructor(val serviceClass: } } + /** + * Binds a given route with a function of the receiver. + * @param function a function of the receiver + * @param method a HTTP method + * @param route a route + */ + @Suppress("TooGenericExceptionCaught") + protected actual inline fun bind( + noinline function: suspend T.(PAR1, PAR2, PAR3, PAR4, PAR5, PAR6) -> RET, + method: HttpMethod, route: String? + ) { + if (method == HttpMethod.GET) + throw UnsupportedOperationException("GET method is only supported for methods without parameters") + val routeDef = route ?: "route${this::class.simpleName}${counter++}" + addRoute(method, "/kv/$routeDef") { ctx -> + val jsonRpcRequest = ctx.body() + @Suppress("MagicNumber") + if (jsonRpcRequest.params.size == 6) { + val param1 = getParameter(jsonRpcRequest.params[0]) + val param2 = getParameter(jsonRpcRequest.params[1]) + val param3 = getParameter(jsonRpcRequest.params[2]) + val param4 = getParameter(jsonRpcRequest.params[3]) + val param5 = getParameter(jsonRpcRequest.params[4]) + val param6 = getParameter(jsonRpcRequest.params[5]) + val injector = ctx.attribute(KV_INJECTOR_KEY)!! + val service = injector.getInstance(serviceClass.java) + initializeService(service, ctx) + val future = GlobalScope.future { + try { + val result = function.invoke(service, param1, param2, param3, param4, param5, param6) + JsonRpcResponse( + id = jsonRpcRequest.id, + result = mapper.writeValueAsString(result) + ) + } catch (e: Exception) { + if (e !is ServiceException) LOG.error(e.message, e) + JsonRpcResponse( + id = jsonRpcRequest.id, error = e.message ?: "Error", + exceptionType = e.javaClass.canonicalName + ) + } + } + ctx.json(future) + } else { + ctx.json(JsonRpcResponse(id = jsonRpcRequest.id, error = "Invalid parameters")) + } + } + } + /** * Binds a given web socket connection with a function of the receiver. * @param function a function of the receiver -- cgit