diff --git a/.github/workflows/build_publish.yaml b/.github/workflows/build_publish.yaml new file mode 100644 index 0000000..db2f1d0 --- /dev/null +++ b/.github/workflows/build_publish.yaml @@ -0,0 +1,134 @@ +# This workflow compiles the native wasmer_jni lib into various OS and architectures using dynamic_libs_reusable.yaml. +# Then it builds and publishes the JAR based on the GITHUB_REF that calls the workflow. There are 2 options: +# 1. The workflow is executed from a branch. In that case the JAR is available in the GitHub action as an artifact. +# 2. The workflow is executed from a tag. In that case the JAR is published in the corresponding GitHub release. +name: Build and publish + +on: + push: + branches: + - '*' # Trigger on push to any branch + tags: + - '*' # Trigger on push to any tag + +jobs: + upload_dynamic_libs: + strategy: + matrix: + platform: + - os: 'macos-latest' + target: 'aarch64-apple-darwin' + artifact: 'darwin-arm64' + lib_name: 'libwasmer_jni.dylib' + + - os: 'macos-latest' + target: 'x86_64-apple-darwin' + artifact: 'darwin-amd64' + lib_name: 'libwasmer_jni.dylib' + + - os: 'ubuntu-latest' + target: 'x86_64-unknown-linux-gnu' + artifact: 'linux-amd64' + lib_name: 'libwasmer_jni.so' + + - os: 'windows-latest' + target: 'x86_64-pc-windows-msvc' + artifact: 'windows-amd64' + lib_name: 'wasmer_jni.dll' + uses: ./.github/workflows/dynamic_libs_reusable.yaml + with: + platform_os: ${{ matrix.platform.os }} + platform_target: ${{ matrix.platform.target }} + platform_artifact: ${{ matrix.platform.artifact }} + platform_lib_name: ${{ matrix.platform.lib_name }} + + publish_jar: + name: Publish the JARs + runs-on: ubuntu-latest + needs: [upload_dynamic_libs] + + steps: + - name: Check out code + uses: actions/checkout@v2 + + - name: Set up Java 21 + uses: actions/setup-java@v1 + with: + java-version: 21 + + - name: Set up Rust + uses: actions-rs/toolchain@v1 + with: + toolchain: stable + default: true + override: true + + - name: Cache Cargo registry + uses: actions/cache@v1 + with: + path: ~/.cargo/registry + key: cargo-registry-${{ hashFiles('**/Cargo.lock') }} + + - name: Cache Cargo bin + uses: actions/cache@v1 + with: + path: ~/.cargo/bin + key: cargo-bin-${{ hashFiles('**/Cargo.lock') }} + + - name: Cache Cargo build + uses: actions/cache@v1 + with: + path: target + key: cargo-build-target-${{ hashFiles('**/Cargo.lock') }} + + - name: Setup wasmer-jni artifacts dir + shell: bash + run: | + echo "EXT_ARTIFACTS_DIR=$(mktemp -d)" >> $GITHUB_ENV + + - name: Download wasmer_jni artifacts + uses: actions/download-artifact@v4 + with: + path: ${{ env.EXT_ARTIFACTS_DIR }} + + - name: Display structure of downloaded files + run: ls -R $EXT_ARTIFACTS_DIR + + - name: Run all the tests + shell: bash + run: | + export PATH="$HOME/.cargo/bin:$PATH" + make test + + - name: Create the JAR + id: create_jar + shell: bash + run: | + export PATH="$HOME/.cargo/bin:$PATH" + make package + + - name: Upload JAR as workflow artifact + if: startsWith(github.ref, 'refs/heads/') + uses: actions/upload-artifact@v4 + with: + name: ${{ steps.create_jar.outputs.name }} + path: ${{ steps.create_jar.outputs.path }} + retention-days: 1 + + - name: Get release info + id: get_release_info + if: startsWith(github.ref, 'refs/tags/') + uses: bruceadams/get-release@v1.3.2 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + + - name: Upload JAR as Github pre-release asset + if: startsWith(github.ref, 'refs/tags/') + uses: actions/upload-release-asset@v1 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + with: + upload_url: ${{ steps.get_release_info.outputs.upload_url }} + asset_path: ${{ steps.create_jar.outputs.path }} + asset_name: ${{ steps.create_jar.outputs.name }} + asset_content_type: application/java-archive \ No newline at end of file diff --git a/.github/workflows/dynamic_libs.yaml b/.github/workflows/dynamic_libs.yaml deleted file mode 100644 index b1f50a3..0000000 --- a/.github/workflows/dynamic_libs.yaml +++ /dev/null @@ -1,67 +0,0 @@ -name: Dynamic libs - -on: - push: - tags: - - '**' - -jobs: - publish_dlibs: - name: Publish the dynamic libraries - strategy: - matrix: - platform: - - os_name: macos-aarch64 - os: macos-latest - target: aarch64-apple-darwin - lib_name: libwasmer_jni.dylib - zip_name: macos-aarch64.zip - - - os_name: macos-x86_64 - os: macos-latest - target: x86_64-apple-darwin - lib_name: libwasmer_jni.dylib - zip_name: macos-x86_64.zip - - - os_name: linux-x86_64 - os: ubuntu-latest - target: x86_64-unknown-linux-gnu - lib_name: libwasmer_jni.so - zip_name: linux-x86_64.zip - - - os_name: windows - os: windows-latest - target: x86_64-pc-windows-msvc - lib_name: wasmer_jni.dll - zip_name: windows.zip - - runs-on: ${{ matrix.platform.os }} - steps: - - name: Check out code - uses: actions/checkout@v3 - - - name: Set up Rust - uses: actions-rs/toolchain@v1 - with: - toolchain: stable - default: true - override: true - - - name: Create the dynamic libs - shell: bash - run: | - export PATH="$HOME/.cargo/bin:$PATH" - rustup target add ${{ matrix.platform.target }} - cargo build --release --target=${{ matrix.platform.target }} - - - name: Archive dynamic lib - uses: thedoctor0/zip-release@0.7.5 - with: - type: 'zip' - filename: ${{ matrix.platform.zip_name }} - path: ./target/${{ matrix.platform.target }}/release/${{ matrix.platform.lib_name }} - - - name: Release - uses: softprops/action-gh-release@v2 - with: - files: ${{ matrix.platform.zip_name }} diff --git a/.github/workflows/dynamic_libs_reusable.yaml b/.github/workflows/dynamic_libs_reusable.yaml new file mode 100644 index 0000000..fe5b6f9 --- /dev/null +++ b/.github/workflows/dynamic_libs_reusable.yaml @@ -0,0 +1,53 @@ +# This workflow cross compiles a rust library into a native based on a provided target. +# The compiled library is then uploaded as a GitHub artifact. +name: upload_dynamic_libs + +on: + workflow_call: + inputs: + platform_os: + description: Operating system that the runner will use. + type: string + required: true + platform_target: + description: The architecture that rust will compile the libs for. + type: string + required: true + platform_artifact: + description: The name of the artifact that will be uploaded. + type: string + required: true + platform_lib_name: + description: Name of the native library to be uploaded in the artifact. + type: string + required: true + + +jobs: + publish_dlibs: + name: Publish the dynamic libraries + runs-on: ${{ inputs.platform_os }} + steps: + - name: Check out code + uses: actions/checkout@v3 + + - name: Set up Rust + uses: actions-rs/toolchain@v1 + with: + toolchain: stable + default: true + override: true + + - name: Create the dynamic libs + shell: bash + run: | + export PATH="$HOME/.cargo/bin:$PATH" + rustup target add ${{ inputs.platform_target }} + cargo build --release --target=${{ inputs.platform_target }} + + - name: Upload the dynamic libs + uses: actions/upload-artifact@v4 + with: + name: ${{ inputs.platform_artifact }} + path: ./target/${{ inputs.platform_target }}/release/${{ inputs.platform_lib_name }} + retention-days: 1 \ No newline at end of file diff --git a/.github/workflows/mac_release.yaml b/.github/workflows/mac_release.yaml deleted file mode 100644 index cc49b9d..0000000 --- a/.github/workflows/mac_release.yaml +++ /dev/null @@ -1,85 +0,0 @@ -name: Release - -on: - push: - tags: - - '**' - -jobs: - publish_jar: - name: Publish the mac JAR - - strategy: - matrix: - os: [macos-latest] - # The job runs on different Java versions (LTS). - java: [21] - architecture: [aarch64] - distribution: [temurin] - - runs-on: ${{ matrix.os }} - - steps: - - name: Check out code - uses: actions/checkout@v2 - - - name: Set up Java ${{ matrix.version }} - uses: actions/setup-java@v2 - with: - java-version: ${{ matrix.java }} - architecture: ${{ matrix.architecture }} - distribution: ${{ matrix.distribution }} - - - name: Set up Rust - uses: actions-rs/toolchain@v1 - with: - toolchain: stable - default: true - override: true - - - name: Cache Cargo registry - uses: actions/cache@v1 - with: - path: ~/.cargo/registry - key: ${{ runner.os }}-cargo-registry-${{ hashFiles('**/Cargo.lock') }} - - - name: Cache Cargo bin - uses: actions/cache@v1 - with: - path: ~/.cargo/bin - key: ${{ runner.os }}-cargo-bin-${{ hashFiles('**/Cargo.lock') }} - - - name: Cache Cargo build - uses: actions/cache@v1 - with: - path: target - key: ${{ runner.os }}-cargo-build-target-${{ hashFiles('**/Cargo.lock') }} - - - name: Run all the tests - shell: bash - run: | - export PATH="$HOME/.cargo/bin:$PATH" - make test - - - name: Create the JAR - id: create_jar - shell: bash - run: | - export PATH="$HOME/.cargo/bin:$PATH" - make package - - - name: Get release info - id: get_release_info - uses: bruceadams/get-release@v1.3.2 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - - - name: Upload JAR as Github pre-release asset - uses: actions/upload-release-asset@v1 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - with: - upload_url: ${{ steps.get_release_info.outputs.upload_url }} - asset_path: ${{ steps.create_jar.outputs.path }} - asset_name: ${{ steps.create_jar.outputs.name }} - asset_content_type: application/java-archive \ No newline at end of file diff --git a/.github/workflows/release.yaml b/.github/workflows/release.yaml deleted file mode 100644 index a601cf9..0000000 --- a/.github/workflows/release.yaml +++ /dev/null @@ -1,85 +0,0 @@ -name: Release - -on: - push: - tags: - - '**' - -jobs: - publish_jar: - name: Publish the JARs - - strategy: - matrix: - # The job runs on 2 different OS. - os: [ubuntu-latest, windows-latest] - # The job runs on different Java versions (LTS). - java: [21] - # As soon as one job fails in the matrix, all the other - # in-progress jobs are canceled. - fail-fast: true - - runs-on: ${{ matrix.os }} - - steps: - - name: Check out code - uses: actions/checkout@v2 - - - name: Set up Java ${{ matrix.version }} - uses: actions/setup-java@v1 - with: - java-version: ${{ matrix.java }} - - - name: Set up Rust - uses: actions-rs/toolchain@v1 - with: - toolchain: stable - default: true - override: true - - - name: Cache Cargo registry - uses: actions/cache@v1 - with: - path: ~/.cargo/registry - key: ${{ runner.os }}-cargo-registry-${{ hashFiles('**/Cargo.lock') }} - - - name: Cache Cargo bin - uses: actions/cache@v1 - with: - path: ~/.cargo/bin - key: ${{ runner.os }}-cargo-bin-${{ hashFiles('**/Cargo.lock') }} - - - name: Cache Cargo build - uses: actions/cache@v1 - with: - path: target - key: ${{ runner.os }}-cargo-build-target-${{ hashFiles('**/Cargo.lock') }} - - - name: Run all the tests - shell: bash - run: | - export PATH="$HOME/.cargo/bin:$PATH" - make test - - - name: Create the JAR - id: create_jar - shell: bash - run: | - export PATH="$HOME/.cargo/bin:$PATH" - make package - - - name: Get release info - id: get_release_info - uses: bruceadams/get-release@v1.3.2 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - - - name: Upload JAR as Github pre-release asset - uses: actions/upload-release-asset@v1 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - with: - upload_url: ${{ steps.get_release_info.outputs.upload_url }} - asset_path: ${{ steps.create_jar.outputs.path }} - asset_name: ${{ steps.create_jar.outputs.name }} - asset_content_type: application/java-archive \ No newline at end of file diff --git a/README.md b/README.md index 385a906..b543fd7 100644 --- a/README.md +++ b/README.md @@ -44,12 +44,12 @@ Features: # Install The JAR files are named as follows: -`wasmer-jni-$(architecture)-$(os)-$(version).jar`. Thus, to include +`wasmer-jni-$(version).jar`. Thus, to include Wasmer JNI as a dependency, write for instance: ```gradle dependencies { - implementation "org.wasmer:wasmer-jni-amd64-linux:0.3.0" + implementation "org.wasmer:wasmer-jni:1.1.0" } ``` @@ -302,61 +302,13 @@ To build the JAR package: $ make package ``` -This will generate the file `build/libs/wasmer-jni-$(architecture)-$(os)-0.3.0.jar`. +This will generate the file `build/libs/wasmer-jni-1.1.0.jar`. -#### Automatic dependencies per architecture and platform +### Native `wasmer-jni` inclusion flow. -
-It is possible to infer the archive appendix automatically, see how. - -According the [Gradle Jar -API](https://docs.gradle.org/current/dsl/org.gradle.api.tasks.bundling.Jar.html#org.gradle.api.tasks.bundling.Jar:appendix), -the `$(architecture)-$(os)` part is called the _archive prefix_. To -infer that appendix automatically to configure your dependencies, you -can use the following `inferWasmerJarAppendix` function: - -```gradle -String inferWasmerJarAppendix() { - def nativePlatform = new org.gradle.nativeplatform.platform.internal.DefaultNativePlatform("current") - def arch = nativePlatform.architecture - def os = nativePlatform.operatingSystem - - def arch_name - - switch (arch.getName()) { - case ["x86_64", "x64", "x86-64"]: - arch_name = "amd64" - break; - - default: - throw new RuntimeException("`wasmer-jni` has no pre-compiled archive for the architecture " + arch.getName()) - } - - def os_name - - if (os.isMacOsX()) { - os_name = "darwin" - } else if (os.isLinux()) { - os_name = "linux" - } else if (os.isWindows()) { - os_name = "windows" - } else { - throw new RuntimeException("`wasmer-jni` has no pre-compiled archive for the platform " + os.getName()) - } - - return arch_name + "-" + os_name -} -``` - -Finally, you can configure your dependencies such as: - -```gradle -dependencies { - implementation "org.wasmer:wasmer-jni-" + inferWasmerJarAppendix() + ":0.3.0" -} -``` - -
+You can export the environment variable `EXT_ARTIFACTS_DIR` in the terminal that you are building the project from. +It should point to the directory where you've put your `wasmer-jni` native libs. If the environment variable is missing +gradle will copy the contents of the `artifacts` folder, which is generated by the `make build-rust` command. ### Testing diff --git a/build.gradle b/build.gradle index e9793a4..1f4f40e 100644 --- a/build.gradle +++ b/build.gradle @@ -6,7 +6,7 @@ plugins { allprojects { group "org.wasmer" - version "0.3.0" + version "1.1.0" } // This is needed for the Java plugin to make sure @@ -65,43 +65,6 @@ jar { attributes("Implementation-Title": project.name, "Implementation-Version": project.version) } - - // The JAR name is defined as - // `${baseName}-${appendix}-${version}-${classifier}.${extension}`. - archiveAppendix = inferWasmerJarAppendix() -} - -String inferWasmerJarAppendix() { - def nativePlatform = new org.gradle.nativeplatform.platform.internal.DefaultNativePlatform("current") - def arch = nativePlatform.architecture - def os = nativePlatform.operatingSystem - - def arch_name - - switch (arch.getName()) { - case ["x86_64", "x64", "x86-64"]: - arch_name = "amd64" - break; - case ["arm64", "aarch64"]: - arch_name = "arm64" - break; - default: - throw new RuntimeException("Architecture " + arch.getName() + " is not supported.") - } - - def os_name - - if (os.isMacOsX()) { - os_name = "darwin" - } else if (os.isLinux()) { - os_name = "linux" - } else if (os.isWindows()) { - os_name = "windows" - } else { - throw new RuntimeException("Platform " + os.getName() + " is not supported.") - } - - return arch_name + "-" + os_name } task generateJniHeaders(type: JavaCompile) { @@ -127,10 +90,17 @@ task buildRust(type: Exec) { task copyAllArtifacts(type: Copy) { dependsOn buildRust - description "Copy build artifacts to the `build/` directory." + description "Copy artifacts to the `build/` directory. If a non-empty directory exists at the value of " + + "'EXT_ARTIFACTS_DIR' env variable it copies its contents, otherwise it copies from the artifacts folder." + + def extArtifactsDir = System.getenv('EXT_ARTIFACTS_DIR') + def sourceDir = extArtifactsDir != null && !extArtifactsDir.isBlank() + ? extArtifactsDir + : 'artifacts' + + from(sourceDir) + include '**/*' - from "artifacts" - include "**/*" into "$buildDir/toArtifact/org/wasmer/native/" } @@ -149,7 +119,7 @@ tasks.withType(Test) { jar.doLast() { // Display specific “action outputs” for Github Actions. - def jar_name = project.archivesBaseName + "-" + inferWasmerJarAppendix() + "-" + project.version + ".jar" + def jar_name = project.archivesBaseName + "-" + project.version + ".jar" println(jar_name) println("::set-output name=path::./build/libs/" + jar_name) println("::set-output name=name::" + jar_name) diff --git a/src/java/org/wasmer/Native.java b/src/java/org/wasmer/Native.java index 9969073..c4a9397 100644 --- a/src/java/org/wasmer/Native.java +++ b/src/java/org/wasmer/Native.java @@ -5,7 +5,13 @@ package org.wasmer; -import java.io.*; +import java.io.BufferedOutputStream; +import java.io.BufferedReader; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.io.OutputStream; import java.net.URL; import java.nio.file.Files; import java.util.logging.Level; @@ -19,7 +25,8 @@ public class Native { LOADED_EMBEDDED_LIBRARY = loadEmbeddedLibrary(); } - private Native() {} + private Native() { + } public static String getCurrentPlatformIdentifier() { String osName = System.getProperty("os.name").toLowerCase(); @@ -28,13 +35,15 @@ public static String getCurrentPlatformIdentifier() { osName = "windows"; } else if (osName.contains("mac os x")) { osName = "darwin"; - String[] args = new String[] {"/bin/bash", "-c", "uname", "-p"}; + String[] args = new String[]{"/bin/bash", "-c", "uname -m"}; try { Process proc = new ProcessBuilder(args).start(); - BufferedReader reader = new BufferedReader(new InputStreamReader(proc.getInputStream())); - if (reader.readLine().equals("Darwin")) { - return osName + "-arm64"; - } + String arch = new BufferedReader(new InputStreamReader(proc.getInputStream())) + .readLine().equals("x86_64") + ? "amd64" + : "arm64"; + + return osName + "-" + arch; } catch (IOException e) { throw new RuntimeException(e); } @@ -64,14 +73,14 @@ private static boolean loadEmbeddedLibrary() { URL nativeLibraryUrl = null; // loop through extensions, stopping after finding first one - for (String lib: libs) { + for (String lib : libs) { nativeLibraryUrl = Module.class.getResource(url + lib); if (nativeLibraryUrl != null) { break; } } - + if (nativeLibraryUrl != null) { // native library found within JAR, extract and load try {