|
| 1 | +import io.opentelemetry.gradle.WeaverTasks |
| 2 | +import org.gradle.api.GradleException |
| 3 | +import org.gradle.api.provider.Property |
| 4 | + |
| 5 | +// Weaver code generation convention plugin for OpenTelemetry |
| 6 | +// Apply this plugin to modules that have a model/ directory with weaver model files |
| 7 | +// It will generate Java code, documentation, and YAML configs using the OpenTelemetry Weaver tool |
| 8 | + |
| 9 | +val weaverContainer = |
| 10 | + "otel/weaver:v0.18.0@sha256:5425ade81dc22ddd840902b0638b4b6a9186fb654c5b50c1d1ccd31299437390" |
| 11 | + |
| 12 | +// Auto-detect platform for Docker, with fallback to x86_64 |
| 13 | +val dockerPlatform = System.getProperty("os.arch").let { arch -> |
| 14 | + when { |
| 15 | + arch == "aarch64" || arch == "arm64" -> "linux/arm64" |
| 16 | + else -> "linux/x86_64" |
| 17 | + } |
| 18 | +} |
| 19 | + |
| 20 | +// Docker executable path: use env var DOCKER_EXECUTABLE or usual platform defaults |
| 21 | +val dockerExecutablePath = System.getenv("DOCKER_EXECUTABLE") ?: run { |
| 22 | + val os = System.getProperty("os.name").lowercase() |
| 23 | + when { |
| 24 | + os.contains("mac") || os.contains("darwin") -> "/usr/local/bin/docker" |
| 25 | + os.contains("windows") -> "docker" |
| 26 | + else -> "docker" // Linux typically has docker in PATH |
| 27 | + } |
| 28 | +} |
| 29 | + |
| 30 | +interface OtelWeaverExtension { |
| 31 | + /** |
| 32 | + * REQUIRED: The Java package path where generated code will be placed. Path should use forward |
| 33 | + * slashes (e.g., "io/opentelemetry/ibm/mq/metrics"). |
| 34 | + * |
| 35 | + * Example configuration in build.gradle.kts: |
| 36 | + * ```kotlin |
| 37 | + * otelWeaver { |
| 38 | + * javaOutputPackage.set("io/opentelemetry/ibm/mq/metrics") |
| 39 | + * } |
| 40 | + * ``` |
| 41 | + */ |
| 42 | + val javaOutputPackage: Property<String> |
| 43 | +} |
| 44 | + |
| 45 | +val weaverExtension = extensions.create("otelWeaver", OtelWeaverExtension::class.java) |
| 46 | + |
| 47 | +val projectModelDir = layout.projectDirectory.dir("model") |
| 48 | +val hasWeaverModel = projectModelDir.asFile.exists() && projectModelDir.asFile.isDirectory |
| 49 | + |
| 50 | +if (hasWeaverModel) { |
| 51 | + val projectTemplatesDir = layout.projectDirectory.dir("templates") |
| 52 | + val projectDocsDir = layout.projectDirectory.dir("docs") |
| 53 | + |
| 54 | + logger.lifecycle("Weaver model found in ${project.name}") |
| 55 | + logger.lifecycle(" Model directory: ${projectModelDir.asFile.absolutePath}") |
| 56 | + logger.lifecycle(" Templates directory: ${projectTemplatesDir.asFile.absolutePath}") |
| 57 | + logger.lifecycle(" Container: $weaverContainer") |
| 58 | + |
| 59 | + tasks.register<WeaverTasks>("weaverCheck") { |
| 60 | + group = "weaver" |
| 61 | + description = "Check the weaver model for errors" |
| 62 | + |
| 63 | + dockerExecutable.set(dockerExecutablePath) |
| 64 | + platform.set(dockerPlatform) |
| 65 | + image.set(weaverContainer) |
| 66 | + |
| 67 | + modelDir.set(projectModelDir) |
| 68 | + templatesDir.set(projectTemplatesDir) |
| 69 | + outputDir.set(layout.buildDirectory.dir("weaver-check")) |
| 70 | + |
| 71 | + toolArgs.set(listOf("registry", "check", "--registry=/home/weaver/model")) |
| 72 | + |
| 73 | + // Always run check task to ensure model validity, even if inputs haven't changed. |
| 74 | + // This is intentional as validation should always run when explicitly requested. |
| 75 | + outputs.upToDateWhen { false } |
| 76 | + } |
| 77 | + |
| 78 | + tasks.register<WeaverTasks>("weaverGenerateDocs") { |
| 79 | + group = "weaver" |
| 80 | + description = "Generate markdown documentation from weaver model" |
| 81 | + |
| 82 | + dockerExecutable.set(dockerExecutablePath) |
| 83 | + platform.set(dockerPlatform) |
| 84 | + image.set(weaverContainer) |
| 85 | + |
| 86 | + modelDir.set(projectModelDir) |
| 87 | + templatesDir.set(projectTemplatesDir) |
| 88 | + outputDir.set(projectDocsDir) |
| 89 | + |
| 90 | + toolArgs.set( |
| 91 | + listOf( |
| 92 | + "registry", |
| 93 | + "generate", |
| 94 | + "--registry=/home/weaver/model", |
| 95 | + "markdown", |
| 96 | + "--future", |
| 97 | + "/home/weaver/target" |
| 98 | + ) |
| 99 | + ) |
| 100 | + } |
| 101 | + |
| 102 | + val weaverGenerateJavaTask = |
| 103 | + tasks.register<WeaverTasks>("weaverGenerateJava") { |
| 104 | + group = "weaver" |
| 105 | + description = "Generate Java code from weaver model" |
| 106 | + |
| 107 | + dockerExecutable.set(dockerExecutablePath) |
| 108 | + platform.set(dockerPlatform) |
| 109 | + image.set(weaverContainer) |
| 110 | + |
| 111 | + modelDir.set(projectModelDir) |
| 112 | + templatesDir.set(projectTemplatesDir) |
| 113 | + |
| 114 | + // Map the javaOutputPackage to the output directory |
| 115 | + // Finalize the value to ensure it's set at configuration time and avoid capturing the extension |
| 116 | + val javaPackage = weaverExtension.javaOutputPackage |
| 117 | + javaPackage.finalizeValueOnRead() |
| 118 | + outputDir.set(javaPackage.map { layout.projectDirectory.dir("src/main/java/$it") }) |
| 119 | + |
| 120 | + toolArgs.set( |
| 121 | + listOf( |
| 122 | + "registry", |
| 123 | + "generate", |
| 124 | + "--registry=/home/weaver/model", |
| 125 | + "java", |
| 126 | + "--future", |
| 127 | + "/home/weaver/target" |
| 128 | + ) |
| 129 | + ) |
| 130 | + |
| 131 | + doFirst { logger.lifecycle(" Java output: ${outputDir.get().asFile.absolutePath}") } |
| 132 | + } |
| 133 | + |
| 134 | + // Validate the required configuration at configuration time (not execution time) |
| 135 | + afterEvaluate { |
| 136 | + if (weaverExtension.javaOutputPackage.orNull == null) { |
| 137 | + throw GradleException( |
| 138 | + """ |
| 139 | + otelWeaver.javaOutputPackage must be configured in project '${project.name}'. |
| 140 | +
|
| 141 | + Add this to your build.gradle.kts: |
| 142 | + otelWeaver { |
| 143 | + javaOutputPackage.set("io/opentelemetry/your/package") |
| 144 | + } |
| 145 | + """.trimIndent() |
| 146 | + ) |
| 147 | + } |
| 148 | + } |
| 149 | + |
| 150 | + // Make spotless tasks always run after the generate task |
| 151 | + tasks |
| 152 | + .matching { |
| 153 | + it.name == "spotlessJava" || it.name == "spotlessJavaApply" || it.name == "spotlessApply" |
| 154 | + } |
| 155 | + .configureEach { mustRunAfter(weaverGenerateJavaTask) } |
| 156 | + |
| 157 | + // Make weaverGenerateJava automatically format generated code |
| 158 | + weaverGenerateJavaTask.configure { finalizedBy("spotlessJavaApply") } |
| 159 | + |
| 160 | + tasks.register<WeaverTasks>("weaverGenerateYaml") { |
| 161 | + group = "weaver" |
| 162 | + description = "Generate YAML configuration from weaver model" |
| 163 | + |
| 164 | + dockerExecutable.set(dockerExecutablePath) |
| 165 | + platform.set(dockerPlatform) |
| 166 | + image.set(weaverContainer) |
| 167 | + |
| 168 | + modelDir.set(projectModelDir) |
| 169 | + templatesDir.set(projectTemplatesDir) |
| 170 | + outputFile.set(layout.projectDirectory.file("config.yml")) |
| 171 | + |
| 172 | + toolArgs.set( |
| 173 | + listOf( |
| 174 | + "registry", |
| 175 | + "generate", |
| 176 | + "--registry=/home/weaver/model", |
| 177 | + "yaml", |
| 178 | + "--future", |
| 179 | + "/home/weaver/target" |
| 180 | + ) |
| 181 | + ) |
| 182 | + } |
| 183 | + |
| 184 | + tasks.register("weaverGenerate") { |
| 185 | + description = "Generate all outputs (Java, docs, YAML) from weaver model" |
| 186 | + group = "weaver" |
| 187 | + dependsOn("weaverGenerateJava", "weaverGenerateDocs", "weaverGenerateYaml") |
| 188 | + } |
| 189 | + |
| 190 | + // Ensure proper task ordering without forcing automatic execution |
| 191 | + // Use mustRunAfter so weaver generation only runs when explicitly invoked |
| 192 | + tasks.named("compileJava") { mustRunAfter(weaverGenerateJavaTask) } |
| 193 | + tasks.named("sourcesJar") { mustRunAfter(weaverGenerateJavaTask) } |
| 194 | +} else { |
| 195 | + logger.debug( |
| 196 | + "No weaver model directory found in ${project.name}, skipping weaver task registration" |
| 197 | + ) |
| 198 | +} |
0 commit comments