A Gradle plugin that generates a BuildKonfig Kotlin object at build time — like Android's BuildConfig, but for any Kotlin project (JVM, Multiplatform, Android).
Fields can be constant, overridden per build type (debug/release), or scoped to named dimensions (e.g. environment, region) with their own variants.
Plugin ID: com.bitsycore.konfig
Group: com.bitsycore
Artifact: konfig-gradle-plugin
Version: 0.5.0
JVM target: 17
The plugin is published to GitHub Packages. Both a gpr.user (GitHub username) and gpr.key (Personal Access Token with read:packages scope) are required.
Store credentials in ~/.gradle/gradle.properties — never commit them:
gpr.user=YOUR_GITHUB_USERNAME
gpr.key=YOUR_GITHUB_PERSONAL_ACCESS_TOKENAlternatively export environment variables GPR_USER / GPR_KEY and the build script will pick them up automatically.
settings.gradle.kts:
pluginManagement {
repositories {
maven {
name = "GitHubPackages"
url = uri("https://maven.pkg.github.com/bitsycore/bitsykonfig-gradle-plugin")
credentials {
username = providers.gradleProperty("gpr.user").orNull ?: System.getenv("GPR_USER")
password = providers.gradleProperty("gpr.key").orNull ?: System.getenv("GPR_KEY")
}
}
gradlePluginPortal()
}
}Using a version catalog (libs.versions.toml):
[versions]
konfig = "0.5.0"
[plugins]
konfig = { id = "com.bitsycore.konfig", version.ref = "konfig" }build.gradle.kts:
plugins {
alias(libs.plugins.konfig)
}Or inline:
plugins {
id("com.bitsycore.konfig") version "0.5.0"
}The plugin auto-detects your project's group and name to set a default package. The generated file is placed in build/generated/konfig/ and wired into your source sets automatically.
konfig {
field("APP_NAME", "My App")
field("VERSION_CODE", 42)
field("ENABLE_LOGGING", false).debug(true)
}Generated output (BuildKonfig.kt):
public object BuildKonfig {
const val BUILD_TYPE: String = "release"
const val MODULE_NAME: String = "my-app"
const val IS_DEBUG: Boolean = false
const val APP_NAME: String = "My App"
const val VERSION_CODE: Int = 42
const val ENABLE_LOGGING: Boolean = false
}In debug builds (-Pkonfig.buildtype=DEBUG), ENABLE_LOGGING becomes inline val ENABLE_LOGGING: Boolean get() = true.
konfig {
objectPackage = "com.example.app" // default: derived from group + project name
objectName = "BuildKonfig" // default: "BuildKonfig"
objectVisibility = Visibility.INTERNAL // default: Visibility.PUBLIC
}| Kotlin type | Example |
|---|---|
String |
field("BASE_URL", "https://example.com") |
Boolean |
field("FEATURE_X", false) |
Int |
field("TIMEOUT", 30) |
Long |
field("MAX_SIZE", 1_000_000L) |
Float |
field("RATIO", 1.5f) |
Double |
field("PI", 3.14159) |
Three equivalent forms:
konfig {
// Fluent handle — default + one or both overrides
field("BASE_URL", "https://prod.example.com").debug("https://dev.example.com")
// Scope blocks — build type is fixed, field() returns Unit, no chaining
debug { field("MOCK_API", true) }
release { field("MOCK_API", false) }
}
field()insidedebug {}/release {}blocks intentionally returnsUnit— the build type is already fixed by the enclosing scope, so.debug()/.release()chaining is impossible by design.
Dimensions let you select a named variant at build time (e.g. env=prod, env=dev). Each active dimension generates a nested object inside BuildKonfig.
konfig {
dimension("env", defaultTo = "prod") {
common {
field("TIMEOUT", 30)
debug { field("TIMEOUT", 5) }
}
variant("prod") {
field("BASE_URL", "https://prod.example.com")
field("ANALYTICS", true)
}
variant("dev") {
field("BASE_URL", "https://dev.example.com")
field("ANALYTICS", false)
}
}
}Generated output (with env=dev, debug build):
public object BuildKonfig {
const val BUILD_TYPE: String = "debug"
// ...
public object Env /*env*/ {
const val VARIANT: String = "dev"
inline val TIMEOUT: Int get() = 5 // common {}, debug override
const val BASE_URL: String = "https://dev.example.com"
const val ANALYTICS: Boolean = false
}
}Fields declared in common {} act as fallbacks for all variants. A variant field with the same name takes precedence over the common field.
dimension("env", objectNameOverride = "Environment", defaultTo = "prod") { ... }
// generates: object Environment /*env*/ { ... }If no override is given, the object name is derived from the dimension name via CamelCase conversion (my-env → MyEnv).
Variants are resolved in priority order:
| Priority | Source | Example |
|---|---|---|
| 1 | Gradle property | -Pkonfig.dimension.env=dev |
| 2 | konfig.properties file |
konfig.dimension.env=dev |
| 3 | Task-name detection | Running assembleDevDebug matches dev |
| 4 | defaultTo in DSL |
dimension("env", defaultTo = "prod") |
| — | Omitted silently | No variant → no nested object generated |
Place a konfig.properties file in your project directory:
konfig.dimension.env=devThis file is tracked as a task input — changing it invalidates the build cache.
Build type is resolved in priority order:
| Priority | Source | Example |
|---|---|---|
| 1 | Explicit property | -Pkonfig.buildtype=DEBUG |
| 2 | Task-name detection | Running assembleDebug → DEBUG |
| — | Default | RELEASE |
| Property | Effect |
|---|---|
-Pkonfig.buildtype=DEBUG|RELEASE |
Forces build type |
-Pkonfig.dimension.<name>=<variant> |
Selects a dimension variant |
-Pkonfig.force |
Disables UP-TO-DATE checks — task always re-runs |
-Pkonfig.android.buildtypedetection=false |
Disables task-name build-type detection |
-Pkonfig.android.flavordetection=false |
Disables task-name dimension-variant detection |
Forces generateKonfig to re-run on every build, bypassing Gradle's UP-TO-DATE and build-cache checks.
./gradlew generateKonfig -Pkonfig.force
./gradlew assembleRelease -Pkonfig.forceThe flag is presence-based — any value (or no value) enables it.
import com.example.app.BuildKonfig
println(BuildKonfig.BUILD_TYPE) // "debug" or "release"
println(BuildKonfig.IS_DEBUG) // true (debug) or false (release)
println(BuildKonfig.Env.BASE_URL) // dimension field
println(BuildKonfig.Env.VARIANT) // "dev" or "prod"Lazy Provider<T> values are supported — useful for reading Gradle properties or environment variables:
konfig {
field("API_KEY", providers.gradleProperty("myapp.apiKey"))
field("CI_BUILD", providers.environmentVariable("CI").map { it.toBoolean() })
}Do not call
System.getenv()orproject.findProperty()directly insidefield()— these bypass the Provider API and break configuration cache.
The generated directory (build/generated/konfig/) is automatically added as a Kotlin source set for:
org.jetbrains.kotlin.multiplatform→commonMainorg.jetbrains.kotlin.jvm→mainorg.jetbrains.kotlin.android→maincom.android.application/com.android.library→main
The generateKonfig task is automatically wired as a dependency of all compileKotlin* and sourcesJar tasks.
# Publish to GitHub Packages (requires gpr.user + gpr.key)
./gradlew publish
# Publish only the plugin marker (fixes resolution without re-uploading the jar)
./gradlew publishKonfigPluginMarkerMavenPublicationToGitHubPackagesRepository
# Publish to local Maven for local testing
./gradlew publishToMavenLocal# Build and publish to local Maven (primary development loop)
./gradlew publishToMavenLocal
# Run unit tests only
./gradlew test
# Run functional tests (Gradle TestKit — starts real Gradle builds)
./gradlew functionalTest
# Run a specific functional test
./gradlew functionalTest --tests "*dimension with defaultTo*"
# Run all checks (test + functionalTest)
./gradlew check
# Force re-run (skip UP-TO-DATE / cache)
./gradlew functionalTest --rerun-tasks