Darryl Miles
03/06/2024, 4:50 PMimplementation(project(":web-wasm"), configuration = "webjar")Darryl Miles
03/06/2024, 10:57 PMimplementation(file("path/here.jar"))onlyIf<script src="/mount/webjars/my-app-thingy/1/wasmJs/productionExecutable/web-asm-alongside.js">// We setup an additional configuration just containing the webjar
//  in an ideal world you should be able to just consume this with:
//  implementation(project(":this-wasm-project"), configuration = "webjar")
// But some kind of output artifact capability colouring prevents this
//  so we have to copy the webjar an intermediate location outside of
//  the project buildDirectory and have the consuming use the raw path:
//  implementation(file("/location/to/raw/path/tmp/my-project.jar"))
val webjar by configuration.creating
// This is only needed due to not being able to consume artifacts directly by sibling
val fileCopyWebjarTask = tasks.register<Copy>("fileCopyWebjar") {
    group = "group"
    doNotTrackState("Outside projectDirectory")
    // finalizeBy is always run, attempt to skip this task if
    onlyIf { tasks.findByName("webjar")?.didWork == true }
    from(layout.buildDirectory.dir("libs"))
    into(layout.projectDirectory.dir("../tmp"))
    include("*-webjar.jar")
}
val WEBJAR_version = "1"
val WEBJAR_path = "my-app-thingy/$WEBJAR_version"
val WEBJAR_subpath = "wasmJs/productionExecutable"
val webjarTask = tasks.register<Jar>("webjar") {
    // Maybe this is the wrong task to depend on (maybe a better task to depend on exists upstream)
    dependsOn(tasks.findByName("wasmJsBrowserDistribution"))
    group = "build"
    // Have to use classifier, because some other JAR artifact is the default
    archiveClassifier = "webjar"
    // This doesn't conform to any standard, some metadata to help allow identification and bootstrap at runtime
    manifest {
        attributes["X-WebJar-Name"] = "my-app-thingy"
        attributes["X-WebJar-Path"] = WEBJAR_path
        attributes["X-WebJar-Subpath"] = WEBJAR_subpath
        attributes["X-WebJar-Version"] = WEBJAR_version
        attributes["X-WebJar-WASM-Loader"] = "$WEBJAR_path/$WEBJAR_subpath/web-wasm.js"
        attributes["X-WebJar-WASM-Binary"] = "$WEBJAR_path/$WEBJAR_subpath/my-app-thingy-web-wasm-wasm-js.wasm"
    }
    from(layout.buildDirectory.dir("dist")) {
        into("META-INF/resources/webjars/${WEBJAR_path}")
    }
    from(layout.buildDirectory.file("dist/wasmJs/productionExecutable/web-wasm.js")) {
        // By default, even if we load the JS from a webjar path, it will attempt to load the WASM binary
        //  from the consuming URL location.href/base and not from along-side the JS loader.
        // So this provides an additional copy of the bootstrap that is edited
        into("META-INF/resources/webjars/${WEBJAR_path}/${WEBJAR_subpath}")
        rename("web-wasm.js", "web-wasm-alongside.js")
        // TODO, investigate use of a script snippet like this below
        // if(document.currentScript) document.currentScript.getAttribute('src')
        // edited(document.currentScript) + "./my-app-thingy-web-wasm-wasm-js.wasm"
        // this might allow standard bootstrap code to exist which doesn't hardwire into the webjar the "/static/" part, instead it would resolve the alongside fully qualified URL based on the locate of the JS bootstrap
        //
        // TODO fixup/create web-wasm-alongside.js.map
        // const d = "./my-app-thingy-web-wasm-wasm-js.wasm", _ = {js_code: r};
        filter(Transformer { line -> line.replace(Regex("\"\\./(\\S+)\\.wasm\""), "\"/static/webjars/$WEBJAR_path/$WEBJAR_subpath/\$1.wasm\"") })
    }
    finalizedBy(fileCopyWebjarTask)
}
artifacts.add(webjar.name, layout.buildDirectory.dir("libs/")) {
    builtBy(webjarTask)
}Artem Kobzar
03/07/2024, 9:38 AMDarryl Miles
03/07/2024, 11:47 AM