package {{packageName}}.infrastructure import okhttp3.* import java.io.File open class ApiClient(val baseUrl: String) { companion object { protected val ContentType = "Content-Type" protected val Accept = "Accept" protected val JsonMediaType = "application/json" protected val FormDataMediaType = "multipart/form-data" protected val XmlMediaType = "application/xml" @JvmStatic val client : OkHttpClient = OkHttpClient() @JvmStatic var defaultHeaders: Map by ApplicationDelegates.setOnce(mapOf(ContentType to JsonMediaType, Accept to JsonMediaType)) @JvmStatic val jsonHeaders: Map = mapOf(ContentType to JsonMediaType, Accept to JsonMediaType) } inline protected fun requestBody(content: T, mediaType: String = JsonMediaType): RequestBody { if(content is File) { return RequestBody.create( MediaType.parse(mediaType), content ) } else if(mediaType == FormDataMediaType) { var builder = FormBody.Builder() // content's type *must* be Map @Suppress("UNCHECKED_CAST") (content as Map).forEach { key, value -> builder = builder.add(key, value) } return builder.build() } else if(mediaType == JsonMediaType) { return RequestBody.create( MediaType.parse(mediaType), Serializer.moshi.adapter(T::class.java).toJson(content) ) } else if (mediaType == XmlMediaType) { TODO("xml not currently supported.") } // TODO: this should be extended with other serializers TODO("requestBody currently only supports JSON body and File body.") } inline protected fun responseBody(body: ResponseBody?, mediaType: String = JsonMediaType): T? { if(body == null) return null return when(mediaType) { JsonMediaType -> Serializer.moshi.adapter(T::class.java).fromJson(body.source()) else -> TODO() } } inline protected fun request(requestConfig: RequestConfig, body : Any? = null): ApiInfrastructureResponse { val httpUrl = HttpUrl.parse(baseUrl) ?: throw IllegalStateException("baseUrl is invalid.") var urlBuilder = httpUrl.newBuilder() .addPathSegments(requestConfig.path.trimStart('/')) requestConfig.query.forEach { query -> query.value.forEach { queryValue -> urlBuilder = urlBuilder.addQueryParameter(query.key, queryValue) } } val url = urlBuilder.build() val headers = requestConfig.headers + defaultHeaders if(headers[ContentType] ?: "" == "") { throw kotlin.IllegalStateException("Missing Content-Type header. This is required.") } if(headers[Accept] ?: "" == "") { throw kotlin.IllegalStateException("Missing Accept header. This is required.") } // TODO: support multiple contentType,accept options here. val contentType = (headers[ContentType] as String).substringBefore(";").toLowerCase() val accept = (headers[Accept] as String).substringBefore(";").toLowerCase() var request : Request.Builder = when (requestConfig.method) { RequestMethod.DELETE -> Request.Builder().url(url).delete() RequestMethod.GET -> Request.Builder().url(url) RequestMethod.HEAD -> Request.Builder().url(url).head() RequestMethod.PATCH -> Request.Builder().url(url).patch(requestBody(body, contentType)) RequestMethod.PUT -> Request.Builder().url(url).put(requestBody(body, contentType)) RequestMethod.POST -> Request.Builder().url(url).post(requestBody(body, contentType)) RequestMethod.OPTIONS -> Request.Builder().url(url).method("OPTIONS", null) } headers.forEach { header -> request = request.addHeader(header.key, header.value) } val realRequest = request.build() val response = client.newCall(realRequest).execute() // TODO: handle specific mapping types. e.g. Map> when { response.isRedirect -> return Redirection( response.code(), response.headers().toMultimap() ) response.isInformational -> return Informational( response.message(), response.code(), response.headers().toMultimap() ) response.isSuccessful -> return Success( responseBody(response.body(), accept), response.code(), response.headers().toMultimap() ) response.isClientError -> return ClientError( response.body()?.string(), response.code(), response.headers().toMultimap() ) else -> return ServerError( null, response.body()?.string(), response.code(), response.headers().toMultimap() ) } } }