Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/sureshg/kotlin-mpp-playground
Kotlin Multiplatform playground!
https://github.com/sureshg/kotlin-mpp-playground
compose-web coroutines java jvm kotlin kotlin-kmp kotlin-multiplatform openjdk structured-concurrency virtual-threads wasm
Last synced: 6 days ago
JSON representation
Kotlin Multiplatform playground!
- Host: GitHub
- URL: https://github.com/sureshg/kotlin-mpp-playground
- Owner: sureshg
- Created: 2023-04-13T09:11:28.000Z (almost 2 years ago)
- Default Branch: main
- Last Pushed: 2024-04-13T05:21:13.000Z (10 months ago)
- Last Synced: 2024-04-13T21:52:24.975Z (10 months ago)
- Topics: compose-web, coroutines, java, jvm, kotlin, kotlin-kmp, kotlin-multiplatform, openjdk, structured-concurrency, virtual-threads, wasm
- Language: Kotlin
- Homepage: https://suresh.dev/kotlin-mpp-playground/app
- Size: 76.6 MB
- Stars: 3
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
Awesome Lists containing this project
README
🎨 Kotlin Multiplatform Playground!
----------[![GitHub Workflow Status][gha_badge]][gha_url]
[![OpenJDK Version][java_img]][java_url]
[![Kotlin release][kt_img]][kt_url]
[![Maven Central Version][maven_img]][maven_url]
[![Ktor][ktor_img]][ktor_url]
[![Compose MP][cmp_img]][cmp_url]
[![Style guide][ktfmt_img]][ktfmt_url]This repo shows a Gradle multi-project build structure that uses the [Kotlin Multiplatform][Kotlin Multiplatform] to
build a [JVM][Kotlin-JVM], [JS][Kotlin-JS], [Desktop][Compose-Multiplatform]
and [Compose Web (wasm)][Compose-Multiplatform] applications.### Install OpenJDK EA Build
```bash
# Mac OS
$ curl -s "https://get.sdkman.io" | bash
$ sdk i java 25.ea-open
$ sdk u java 25.ea-open
```### Build & Run
```bash
$ ./gradlew build [-Pskip.test]# Run the app
$ ./gradlew :backend:jvm:run# Publish to local repo
$ ./gradlew buildAndPublish
```### Publishing
Push a new tag to trigger the release workflow and publish the artifacts. That's it 🎉.
The next version will be based on the semantic version scope (`major`, `minor`, `patch`)```bash
$ ./gradlew pushSemverTag "-Psemver.scope=patch"# To see the current version
# ./gradlew v# Print the new version
# ./gradlew printSemver "-Psemver.scope=patch"
```Multiplatform Targets
### JVM
* Build and Run
```bash
# Kotlin Multiplatform
$ ./gradlew :shared:runJvm
$ ./gradlew :shared:jvmDistZip
# Run task for target 'jvm' and compilation 'main' (it's confusing)
$ ./gradlew :shared:jvmRun# Kotlin JVM
$ ./gradlew :backend:jvm:run
$ ./gradlew :backend:jvm:build
$ ./gradlew :backend:jvm:jdeprscan
$ ./gradlew :backend:jvm:printModuleDeps# Benchmark
$ ./gradlew :benchmark:benchmark
```
* GraalVM Native Image```bash
$ sdk u java graalvm-ce-dev
$ ./gradlew :backend:jvm:nativeCompile
$ backend/jvm/build/native/nativeCompile/jvm# To generate the native image configurations
$ ./gradlew :backend:jvm:run -Pagent
$ curl http://localhost:8080/shutdown
$ ./gradlew :backend:jvm:metadataCopy```
* Containers
```bash
$ docker run \
-it \
--rm \
--pull always \
--workdir /app \
--publish 8080:8080 \
--publish 8081:8081 \
--name kotlin-mpp-playground \
--mount type=bind,source=$(pwd),destination=/app,readonly \
openjdk:25-slim /bin/bash -c "printenv && nohup jwebserver -b 0.0.0.0 -p 8081 -d / & backend/jvm/build/libs/jvm-app"$ ./gradlew :backend:jvm:jibDockerBuild
$ docker run -it --rm --name jvm-app -p 8080:8080 -p 9898:9898 sureshg/jvm
$ docker stats
```* OpenTelemetry
```bash
# Install otel-desktop-viewer or Jaeger
$ brew tap CtrlSpice/homebrew-otel-desktop-viewer
$ brew install otel-desktop-viewer
$ otel-desktop-viewer# or run the Jaeger collector
$ docker run -it --rm --pull=always \
-e COLLECTOR_OTLP_ENABLED=true \
-p 4317:4317 \
-p 16686:16686 \
jaegertracing/all-in-one:latest
$ open http://localhost:16686# Run otel tui
$ brew install ymtdzzz/tap/otel-tui
$ otel-tui# Run the app
$ docker run -it --rm \
--name jvm \
-p 8080:8080 \
-p 9898:9898 \
-e OTEL_JAVAAGENT_ENABLED=true \
-e OTEL_TRACES_EXPORTER="otlp" \
-e OTEL_EXPORTER_OTLP_PROTOCOL="grpc" \
-e OTEL_EXPORTER_OTLP_ENDPOINT="http://host.docker.internal:4317" \
-e OTEL_DROP_SPANS="/swagger" \
sureshg/jvm:latest
$ curl -v -X GET http://localhost:8080/trace# Change/Reset log level
$ curl -v -X POST http://localhost:8080/loglevel/dev.suresh.http/debug
$ curl -v -X POST http://localhost:8080/loglevel/reset
```* JVM Agents
```bash
# Normal agent with Launcher-Agent-Class
$ ./gradlew :backend:agent:jfr:build
$ backend/agent/jfr/build/libs/jfr-app# Custom OpenTelemetry agent
$ ./gradlew :backend:agent:otel:build
```* AOT Linking
```bash
# Training Run
$ java --enable-preview \
-XX:+UnlockExperimentalVMOptions \
-XX:+UseCompactObjectHeaders \
-XX:AOTMode=record -XX:AOTConfiguration=app.aotconf \
-jar backend/jvm/build/libs/jvm-all.jar# Create AOT archive
$ java --enable-preview \
-XX:+UnlockExperimentalVMOptions \
-XX:+UseCompactObjectHeaders \
-XX:AOTMode=create -XX:AOTConfiguration=app.aotconf -XX:AOTCache=app.aot \
-jar backend/jvm/build/libs/jvm-all.jar# Run with AOT
$ java --enable-preview \
-XX:+UnlockExperimentalVMOptions \
-XX:+UseCompactObjectHeaders \
-XX:AOTCache=app.aot \
-jar backend/jvm/build/libs/jvm-all.jar# Show native memory details
$ jcmd jvm-app System.map
```* Tests
```bash
$ ./gradlew :backend:jvm:test -PktorTest
$ ./gradlew :backend:jvm:test -Pk8sTest
$ ./gradlew :backend:jvm:jvmRun -DmainClass=dev.suresh.lang.SysCallKt --quiet
```* BinCompat & Missing Targets
```bash
$ ./gradlew :backend:security:apiDump
$ ./gradlew :backend:security:apiCheck# KMP missing targets report
$ ./gradlew :shared:kmpMissingTargets
$ open shared/build/reports/kmp-missing-targets.md
```### Wasm/JS
```bash
$ ./gradlew :web:jsBrowserProductionRun -t
$ ./gradlew :web:wasmJsBrowserProductionRun -t
$ ./gradlew kotlinUpgradePackageLock# Kobweb
$ kobweb run -p compose/web
$ ./gradlew :compose:html:kobwebStart -t
$ ./gradlew :compose:html:kobwebStop
```### Native
```bash
$ ./gradlew :backend:native:build
$ find backend/native/build/bin -type f -perm +111 -exec ls -lh {} \; | awk '{print $9 ": " $5}'# Arch specific binaries
$ ./gradlew :backend:native:macosArm64Binaries
$ ./gradlew :backend:native:macosX64Binaries
$ ./gradlew :backend:native:macOsUniversalBinary# Native container image
$ ./gradlew :backend:native:jibDockerBuild
$ docker run -it --rm --name native-app sureshg/native# Debug distroless image
# docker run -it --entrypoint=sh gcr.io/distroless/cc-debian12:debug# Test linux binary on ARM64 MacOS
$ ./gradlew :backend:native:linuxArm64Binaries
$ docker run \
-it \
--rm \
--publish 8080:80 \
--mount type=bind,source=$(pwd),destination=/app,readonly \
debian:stable-slim
# /app/backend/native/build/bin/linuxArm64/releaseExecutable/native.kexe
# libtree -v /app/backend/native/build/bin/linuxArm64/releaseExecutable/native.kexe# Build native binaries on container
$ docker run \
--platform=linux/amd64 \
-it \
--rm \
--pull always \
--workdir /app \
--name kotlin-native-build \
--mount type=bind,source=$(pwd),destination=/app \
--mount type=bind,source=${HOME}/.gradle,destination=/root/.gradle \
openjdk:25-slim /bin/bash
# apt update && apt install libtree tree
# ./gradlew --no-daemon :backend:native:build
# backend/native/build/bin/linuxX64/releaseExecutable/native.kexe
```### Compose
```bash
# Compose Desktop
$ ./gradlew :compose:cmp:runDistributable
$ ./gradlew :compose:cmp:packageDistributionForCurrentOS
$ ./gradlew :compose:cmp:packageReleaseUberJarForCurrentOS
$ ./gradlew :compose:cmp:suggestModules# Compose Web
$ ./gradlew :compose:cmp:wasmJsBrowserProductionRun -t# Compose multiplatform tests
$ ./gradlew :compose:cmp:allTests
$ ./gradlew :compose:cmp:jvmTest
```### Publishing
```bash
$ ./gradlew publishAllPublicationsToLocalRepository# Publishing to all repo except Maven Central
$ ./gradlew buildAndPublish# Maven Central Publishing
# https://central.sonatype.org/publish/publish-portal-gradle/#alternatives
# https://vanniktech.github.io/gradle-maven-publish-plugin/central/#in-memory-gpg-key
$ gpg --export-secret-keys --armor XXXXXXXX | grep -v '\-\-' | grep -v '^=.' | tr -d '\n'
# OR
$ gpg --export-secret-keys --armor XXXXXXXX | awk 'NR == 1 { print "SIGNING_KEY=" } 1' ORS='\\n'$ export ORG_GRADLE_PROJECT_mavenCentralUsername=
$ export ORG_GRADLE_PROJECT_mavenCentralPassword=
$ export ORG_GRADLE_PROJECT_signingInMemoryKeyId=
$ export ORG_GRADLE_PROJECT_signingInMemoryKeyPassword=
$ export ORG_GRADLE_PROJECT_signingInMemoryKey=$(gpg --export-secret-keys --armor ${ORG_GRADLE_PROJECT_signingInMemoryKeyId} | grep -v '\-\-' | grep -v '^=.' | tr -d '\n')# For aggregated publication (preferred) to Central
$ ./gradlew publishAggregatedPublicationToCentralPortal# For all publications (separate publications)
$ ./gradlew publishAllPublicationsToCentralPortal
```### Misc
```bash
# Dependency Insight
$ ./gradlew dependencies
$ ./gradlew :shared:dependencies --configuration testRuntimeClasspath$ ./gradlew -q :build-logic:dependencyInsight --dependency kotlin-compiler-embeddable --configuration RuntimeClasspath
$ ./gradlew -q :shared:dependencyInsight --dependency slf4j-api --configuration RuntimeClasspath$ ./gradlew :backend:jvm:listResolvedArtifacts
# KMP hierarchy and module graphs
$ ./gradlew :shared:printHierarchy
$ ./gradlew createModuleGraph
$ ./gradlew generateChangelog# Clean
$ ./gradlew cleanAll# Gradle Toolchains
$ ./gradlew buildEnvironment
$ ./gradlew updateDaemonJvm
$ ./gradlew javaToolchains
$ ./gradlew wrapper --gradle-version=x.x.x# Gradle Best Practices
$ ./gradlew -p gradle/build-logic :bestPracticesBaseline
$ ./gradlew checkBuildLogicBestPractices# GitHub Actions lint
$ actionlint
```### Deployed App and Docs
- [Web App](https://suresh.dev/kotlin-mpp-playground/app/)
- [Docs](https://suresh.dev/kotlin-mpp-playground/docs)
- [Coverage](https://suresh.dev/kotlin-mpp-playground/reports)
- [Tests](https://suresh.dev/kotlin-mpp-playground/tests)### Verifying Artifacts
The published artifacts are signed using this [key][signing_key]. The best way to verify artifacts
is [automatically with Gradle][gradle_verification].[gradle_verification]: https://docs.gradle.org/current/userguide/dependency_verification.html#sec:signature-verification
[signing_key]: https://keyserver.ubuntu.com/pks/lookup?op=get&search=0xc124db3a8ad1c13f7153decdf209c085c8b53ca1
### Resources
- [🔍 Kotlin Multiplatform Package Search](https://package-search.jetbrains.com/search?query=ktor&onlyMpp=true)
- [🎨 Kotlin Multiplatform Wizard](https://kmp.jetbrains.com/)
- [📏 Jetbrains Compose Rules](https://mrmans0n.github.io/compose-rules/rules/)[java_url]: https://jdk.java.net/25/
[java_img]: https://img.shields.io/badge/OpenJDK-25-e76f00?logo=openjdk&logoColor=e76f00
[kt_url]: https://github.com/JetBrains/kotlin/releases/latest
[kt_img]: https://img.shields.io/github/v/release/Jetbrains/kotlin?include_prereleases&color=7f53ff&label=Kotlin&logo=kotlin&logoColor=7f53ff
[maven_img]: https://img.shields.io/maven-central/v/dev.suresh.kmp/shared?logo=apachemaven&logoColor=cd2237&color=cd2237
[maven_url]: https://central.sonatype.com/search?q=dev.suresh.kmp&namespace=dev.suresh.kmp
[maven_dl]: https://search.maven.org/remote_content?g=dev.suresh.kmp&a=shared&v=LATEST
[gha_url]: https://github.com/sureshg/kotlin-mpp-playground/actions/workflows/build.yml
[gha_badge]: https://img.shields.io/github/actions/workflow/status/sureshg/kotlin-mpp-playground/build.yml?branch=main&color=green&label=Build&logo=Github-Actions&logoColor=green
[sty_url]: https://kotlinlang.org/docs/coding-conventions.html
[sty_img]: https://img.shields.io/badge/style-Kotlin--Official-40c4ff.svg?style=for-the-badge&logo=kotlin&logoColor=40c4ff
[ktfmt_url]: https://github.com/facebookincubator/ktfmt#ktfmt
[ktfmt_img]: https://img.shields.io/badge/code%20style-%E2%9D%A4-FF4081.svg?logo=kotlin&logoColor=FF4081
[cmp_url]: https://github.com/JetBrains/compose-multiplatform/releases
[cmp_img]: https://img.shields.io/github/v/release/JetBrains/compose-multiplatform?color=3cdc84&label=Compose%20MP&logo=JetpackCompose&logoColor=3cdc84
[ktor_url]: https://search.maven.org/artifact/io.ktor/ktor-bom
[ktor_download]: https://search.maven.org/remote_content?g=io.ktor&a=ktor-client&v=LATEST
[ktor_img]: https://img.shields.io/maven-central/v/io.ktor/ktor-bom?color=4a79fe&label=Ktor&logo=
[Kotlin-JVM]: https://kotlinlang.org/docs/jvm-get-started.html
[Kotlin-JS]: https://kotlinlang.org/docs/js-project-setup.html
[Kotlin Multiplatform]: https://kotlinlang.org/docs/multiplatform.html
[Compose-Multiplatform]: https://github.com/JetBrains/compose-multiplatform
[Kotlin Multiplatform DSL]: https://kotlinlang.org/docs/multiplatform-dsl-reference.html
[simple-icons-logo]: https://simpleicons.org/icons/kotlin.svg
Module Dependency Graph
### Module Dependency
```mermaid
%%{
init: {
'theme': 'neutral'
}
}%%
graph LR
subgraph :backend
:backend:native["native"]
:backend:data["data"]
:backend:profiling["profiling"]
:backend:jvm["jvm"]
:backend:security["security"]
end
subgraph :compose
:compose:desktop["desktop"]
:compose:html["html"]
end
subgraph :dep-mgmt
:dep-mgmt:bom["bom"]
:dep-mgmt:catalog["catalog"]
end
subgraph :meta
:meta:compiler["compiler"]
:meta:ksp["ksp"]
end
subgraph :meta:compiler
:meta:compiler:plugin["plugin"]
end
subgraph :meta:ksp
:meta:ksp:processor["processor"]
end
subgraph :web
:web:js["js"]
:web:wasm["wasm"]
end
:web:js --> :shared
:benchmark --> :shared
:backend:native --> :shared
:web:wasm --> :shared
:compose:desktop --> :shared
:meta:compiler:plugin --> :shared
:meta:ksp:processor --> :shared
:backend:data --> :shared
:backend:profiling --> :shared
:compose:html --> :shared
: --> :backend
: --> :benchmark
: --> :compose
: --> :meta
: --> :shared
: --> :web
: --> :backend:data
: --> :backend:jvm
: --> :backend:native
: --> :backend:profiling
: --> :backend:security
: --> :compose:desktop
: --> :compose:html
: --> :meta:compiler
: --> :meta:ksp
: --> :web:js
: --> :web:wasm
: --> :meta:compiler:plugin
: --> :meta:ksp:processor
: --> :dep-mgmt:bom
: --> :dep-mgmt:catalog
:backend:jvm --> :shared
:backend:jvm --> :backend:data
:backend:jvm --> :backend:profiling
:backend:jvm --> :web:js
:backend:jvm --> :web:wasm
:backend:security --> :shared
```