diff options
author | Linnea Gräf <nea@nea.moe> | 2025-01-17 13:40:43 +0100 |
---|---|---|
committer | Linnea Gräf <nea@nea.moe> | 2025-01-17 13:40:43 +0100 |
commit | e08c8778640967cc086a922f178b18e08b313a29 (patch) | |
tree | c328988177e5a061218bc0df69498264eea1ae46 | |
parent | ce6e008426c30cba493832d0866950c59f7c31c1 (diff) | |
download | LocalTransactionLedger-e08c8778640967cc086a922f178b18e08b313a29.tar.gz LocalTransactionLedger-e08c8778640967cc086a922f178b18e08b313a29.tar.bz2 LocalTransactionLedger-e08c8778640967cc086a922f178b18e08b313a29.zip |
feat(server): Bundle swagger ui
4 files changed, 118 insertions, 9 deletions
diff --git a/server/core/src/main/kotlin/moe/nea/ledger/server/core/Application.kt b/server/core/src/main/kotlin/moe/nea/ledger/server/core/Application.kt index f00f26d..0f13606 100644 --- a/server/core/src/main/kotlin/moe/nea/ledger/server/core/Application.kt +++ b/server/core/src/main/kotlin/moe/nea/ledger/server/core/Application.kt @@ -14,6 +14,7 @@ import moe.nea.ledger.server.core.api.Documentation import moe.nea.ledger.server.core.api.Info import moe.nea.ledger.server.core.api.apiRouting import moe.nea.ledger.server.core.api.openApiDocsJson +import moe.nea.ledger.server.core.api.openApiUi import java.io.File fun main(args: Array<String>) { @@ -46,6 +47,9 @@ fun Application.module() { route("/api.json") { openApiDocsJson() } + route("/openapi") { + openApiUi("/api.json") + } } } diff --git a/server/core/src/main/kotlin/moe/nea/ledger/server/core/api/BaseApi.kt b/server/core/src/main/kotlin/moe/nea/ledger/server/core/api/BaseApi.kt index ad05118..d407f0b 100644 --- a/server/core/src/main/kotlin/moe/nea/ledger/server/core/api/BaseApi.kt +++ b/server/core/src/main/kotlin/moe/nea/ledger/server/core/api/BaseApi.kt @@ -21,8 +21,21 @@ fun Route.apiRouting(database: Database) { } call.respond(profiles) }.docs { + summary = "List all profiles and players known to ledger" + operationId = "listProfiles" + tag(Tags.PROFILE) respondsOk { schema<List<Profile>>() } } } + +enum class Tags : IntoTag { + PROFILE, + MANAGEMENT, + ; + + override fun intoTag(): String { + return name + } +}
\ No newline at end of file diff --git a/server/swagger/build.gradle.kts b/server/swagger/build.gradle.kts index 9ea7cb2..1a578cc 100644 --- a/server/swagger/build.gradle.kts +++ b/server/swagger/build.gradle.kts @@ -8,7 +8,9 @@ plugins { dependencies { declareKtorVersion() api("io.ktor:ktor-server-core") + api("io.ktor:ktor-server-webjars") api("sh.ondr:kotlin-json-schema:0.1.1") + implementation("org.webjars:swagger-ui:5.18.2") } java { diff --git a/server/swagger/src/main/kotlin/moe/nea/ledger/server/core/api/docs.kt b/server/swagger/src/main/kotlin/moe/nea/ledger/server/core/api/docs.kt index 5680027..86ed99d 100644 --- a/server/swagger/src/main/kotlin/moe/nea/ledger/server/core/api/docs.kt +++ b/server/swagger/src/main/kotlin/moe/nea/ledger/server/core/api/docs.kt @@ -2,9 +2,12 @@ package moe.nea.ledger.server.core.api import io.ktor.http.ContentType import io.ktor.http.HttpStatusCode +import io.ktor.http.content.OutgoingContent +import io.ktor.http.defaultForFilePath import io.ktor.server.application.ApplicationCallPipeline import io.ktor.server.application.BaseApplicationPlugin import io.ktor.server.response.respond +import io.ktor.server.response.respondText import io.ktor.server.routing.HttpMethodRouteSelector import io.ktor.server.routing.PathSegmentConstantRouteSelector import io.ktor.server.routing.RootRouteSelector @@ -12,9 +15,16 @@ import io.ktor.server.routing.Route import io.ktor.server.routing.RoutingNode import io.ktor.server.routing.TrailingSlashRouteSelector import io.ktor.server.routing.get +import io.ktor.server.routing.route import io.ktor.util.AttributeKey +import io.ktor.util.cio.KtorDefaultPool +import io.ktor.utils.io.ByteReadChannel +import io.ktor.utils.io.jvm.javaio.toByteReadChannel +import kotlinx.serialization.json.JsonPrimitive import sh.ondr.jsonschema.JsonSchema import sh.ondr.jsonschema.jsonSchema +import java.io.File +import java.io.InputStream fun Route.openApiDocsJson() { @@ -25,6 +35,70 @@ fun Route.openApiDocsJson() { } } +fun Route.openApiUi(apiJsonUrl: String) { + get("swagger-initializer.js") { + call.respondText( + //language=JavaScript + """ + window.onload = function() { + //<editor-fold desc="Changeable Configuration Block"> + + // the following lines will be replaced by docker/configurator, when it runs in a docker-container + window.ui = SwaggerUIBundle({ + url: ${JsonPrimitive(apiJsonUrl)}, + dom_id: '#swagger-ui', + deepLinking: true, + presets: [ + SwaggerUIBundle.presets.apis, + SwaggerUIStandalonePreset + ], + plugins: [ + SwaggerUIBundle.plugins.DownloadUrl + ], + layout: "StandaloneLayout" + }); + + //</editor-fold> + }; + """.trimIndent()) + } +// val swaggerUiProperties = +// environment.classLoader.getResource("/META-INF/maven/org.webjars/swagger-ui/pom.properties") +// ?: error("Could not find swagger webjar") +// val swaggerUiZip = swaggerUiProperties.toString().substringBefore("!") + val pathParameterName = "static-content-path-parameter" + route("{$pathParameterName...}") { + get { + var requestedPath = call.parameters.getAll(pathParameterName)?.joinToString(File.separator) ?: "" + requestedPath = requestedPath.replace("\\", "/") + if (requestedPath.isEmpty()) requestedPath = "index.html" + if (requestedPath.contains("..")) { + call.respondText("Forbidden", status = HttpStatusCode.Forbidden) + return@get + } + //TODO: I mean i should read out the version properties but idc + val version = "5.18.2" + val resource = + environment.classLoader.getResourceAsStream("META-INF/resources/webjars/swagger-ui/$version/$requestedPath") + + if (resource == null) { + call.respondText("Not Found", status = HttpStatusCode.NotFound) + return@get + } + + call.respond(InputStreamContent(resource, ContentType.defaultForFilePath(requestedPath))) + } + } +} + +internal class InputStreamContent( + private val input: InputStream, + override val contentType: ContentType +) : OutgoingContent.ReadChannelContent() { + + override fun readFrom(): ByteReadChannel = input.toByteReadChannel(pool = KtorDefaultPool) +} + class DocumentationPath(val path: String) { companion object { @@ -45,12 +119,13 @@ class DocumentationPath(val path: String) { return "/" } var parentPath = createRoutePath(baseRoute, parent) - if (!parentPath.endsWith("/")) - parentPath += "/" + var parentPathAppendable = parentPath + if (!parentPathAppendable.endsWith("/")) + parentPathAppendable += "/" return when (val selector = route.selector) { - is TrailingSlashRouteSelector -> parentPath + is TrailingSlashRouteSelector -> parentPathAppendable is RootRouteSelector -> parentPath - is PathSegmentConstantRouteSelector -> parentPath + selector.value + is PathSegmentConstantRouteSelector -> parentPathAppendable + selector.value is HttpMethodRouteSelector -> parentPath // TODO: generate a separate path here else -> error("Could not comprehend $selector (${selector.javaClass})") } @@ -74,6 +149,10 @@ class Response { } } +interface IntoTag { + fun intoTag(): String +} + class DocumentationContext(val path: DocumentationPath) { val responses = mutableMapOf<HttpStatusCode, Response>() fun responds(statusCode: HttpStatusCode, block: Response.() -> Unit) { @@ -86,16 +165,27 @@ class DocumentationContext(val path: DocumentationPath) { var summary: String = "" var description: String = "" + var deprecated: Boolean = false + var operationId: String = "" + val tags: MutableList<String> = mutableListOf() + fun tag(vararg tag: String) { + tags.addAll(tag) + } + + fun tag(vararg tag: IntoTag) { + tag.mapTo(tags) { it.intoTag() } + } + fun intoJson(): OpenApiRoute { return OpenApiRoute( summary, description, get = OpenApiOperation( - tags = listOf(), // TODO: tags - summary = "", - description = "", - operationId = "", - deprecated = false, + tags = tags.map { Tag(it) }, + summary = summary, + description = description, + operationId = operationId, + deprecated = deprecated, responses = responses.mapValues { it.value.intoJson() } |