Skip to content

Commit

Permalink
Add gradle to graalpy-freeze-dependencies-guide
Browse files Browse the repository at this point in the history
  • Loading branch information
timfel committed Nov 4, 2024
1 parent 8240a96 commit e15dcde
Show file tree
Hide file tree
Showing 8 changed files with 477 additions and 14 deletions.
90 changes: 77 additions & 13 deletions graalpy/graalpy-freeze-dependencies-guide/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,8 +36,9 @@ However, you can go right to the [completed example](https://github.com/graalvm/

## 3. Writing the application

You can start with any Maven application that runs on JDK 17 or newer.
We will use a default Maven application [generated](https://maven.apache.org/archetypes/maven-archetype-quickstart/) from an archetype.
You can start with any Maven or Gradle application that runs on JDK 17 or newer.
We will demonstrate on both build systems.
A default Maven application [generated](https://maven.apache.org/archetypes/maven-archetype-quickstart/) from an archetype.

```shell
mvn archetype:generate -DarchetypeGroupId=org.apache.maven.archetypes \
Expand All @@ -46,15 +47,23 @@ mvn archetype:generate -DarchetypeGroupId=org.apache.maven.archetypes \
-Dversion=1.0-SNAPSHOT -DinteractiveMode=false
```

And a default Gradle Java application [generated](https://docs.gradle.org/current/samples/sample_building_java_applications.html#run_the_init_task) with the init task.

```shell
gradle init --type java-application --dsl kotlin --test-framework junit-jupiter \
--package org.example --project-name example --java-version 17 \
--no-split-project --no-incubating
```

## 4. Adding packages

Most Python packages are hosted on [PyPI](https://pypi.org) and can be installed via the `pip` tool.
The Python ecosystem has conventions about the filesystem layout of installed packages that need to be kept in mind when embedding into Java.
You can use the GraalPy plugin to manage Python packages for you.
You can use the GraalPy plugins for Maven or Gradle to manage Python packages for you.

`pom.xml`
```xml
<build>
<build>
<plugins>
<plugin>
<groupId>org.graalvm.python</groupId>
Expand All @@ -77,25 +86,66 @@ You can use the GraalPy plugin to manage Python packages for you.
</build>
```

`build.gradle.kts`
```kotlin
plugins {
application
id("org.graalvm.python") version "24.1.1"
}
```

`build.gradle.kts`
```kotlin
graalPy {
packages = setOf("vaderSentiment==3.3.2") //
}
```

❶ The `packages` section lists all Python packages optionally with [requirement specifiers](https://pip.pypa.io/en/stable/reference/requirement-specifiers/).
In this case, we install the `vaderSentiment` package and pin it to version `3.3.2`. Because we are not
specifying `<pythonResourcesDirectory>` the Maven plugin will embed the packages into the
specifying `<pythonResourcesDirectory>` the plugins will embed the packages into the
resulting JAR as a standard Java resource.

## 5. Determining and Pinning Transitive Dependencies

Package the application using Maven, this should install all the transitive dependencies
When you package the application, it installs all the transitive dependencies
in a newly created [virtual environment](https://docs.python.org/3/library/venv.html):

```shell
mvn package
./mvnw package
```

If the compilation is successful, one can run the following command to get
versions of all the installed Python packages:
```shell
./gradlew build
```

If the compilation is successful, one can run the following command to get versions of all the installed Python packages if you use Maven:

On macOS and Linux:
```shell
./target/classes/org.graalvm.python.vfs/venv/bin/pip3 freeze -l
```

On Windows:
```shell
.\target\classes\org.graalvm.python.vfs\venv\Scripts\pip3.exe freeze -l
```

If you are using Gradle, run the following command:

On macOS and Linux:
```shell
./build/generated/graalpy/resources/org.graalvm.python.vfs/venv/bin/pip3 freeze -l
```

On Windows:
```shell
.\build\generated\graalpy\resources\org.graalvm.python.vfs\venv\Scripts\pip3.exe freeze -l
```

The output will look something like this:

```
certifi==2024.8.30
charset-normalizer==3.1.0
idna==3.8
Expand All @@ -104,8 +154,8 @@ urllib3==2.2.2
vaderSentiment==3.3.2
```

Copy and paste the package names and versions to the `pom.xml`, and wrap
them in `<package>` xml tag:
Copy and paste the package names and versions.
If you use Maven, paste them in the `pom.xml` section of the packages and wrap them in `<package>` xml tag:

`pom.xml`
```xml
Expand Down Expand Up @@ -137,6 +187,20 @@ them in `<package>` xml tag:
</build>
```

For Gradle, paste them into the packages block:

`build.gradle.kts`
```kotlin
packages = setOf(
"vaderSentiment==3.3.2",
"certifi==2024.8.30",
"charset-normalizer==3.1.0",
"idna==3.8",
"requests==2.32.3",
"urllib3==2.2.2"
)
```

Note: one can use other Python tools, such as `pipdeptree` to generate the following
dependency tree, where we can also see the version ranges.

Expand All @@ -151,7 +215,7 @@ vaderSentiment==3.3.2

*Warning:
Is it not recommended to manually alter the virtual environment.
Any changes will be overridden by the GraalPy Maven plugin.*
Any changes will be overridden by the GraalPy build plugins.*

## 8. Next steps

Expand All @@ -165,4 +229,4 @@ Any changes will be overridden by the GraalPy Maven plugin.*

- Learn more about the GraalPy [Maven plugin](https://www.graalvm.org/latest/reference-manual/python/Embedding-Build-Tools/)
- Learn more about the Polyglot API for [embedding languages](https://www.graalvm.org/latest/reference-manual/embed-languages/)
- Explore in depth with GraalPy [reference manual](https://www.graalvm.org/latest/reference-manual/python/)
- Explore in depth with GraalPy [reference manual](https://www.graalvm.org/latest/reference-manual/python/)
45 changes: 45 additions & 0 deletions graalpy/graalpy-freeze-dependencies-guide/build.gradle.kts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
plugins {
application
id("org.graalvm.python") version "24.1.1"
}

if ("true".equals(System.getProperty("no.transitive.dependencies"))) {
graalPy {
packages = setOf("vaderSentiment==3.3.2") //
}
} else {
// The default profile shows the end result: all our transitive
// dependencies are explicitly pinned to a specific version.
graalPy {
packages = setOf(
"vaderSentiment==3.3.2",
"certifi==2024.8.30",
"charset-normalizer==3.1.0",
"idna==3.8",
"requests==2.32.3",
"urllib3==2.2.2"
)
}
}

repositories {
mavenLocal()
maven {
url = uri("https://repo.maven.apache.org/maven2/")
}
}

dependencies {
implementation("org.graalvm.python:python:24.1.1") //
implementation("org.graalvm.python:python-embedding:24.1.1") //
}

group = "org.example"
version = "1.0-SNAPSHOT"
description = "javase"
java.sourceCompatibility = JavaVersion.VERSION_17

application {
mainClass = "org.example.App"
applicationDefaultJvmArgs = listOf("-Dgraalpy.resources=" + System.getProperty("graalpy.resources"))
}
Binary file not shown.
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.10.2-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
Loading

0 comments on commit e15dcde

Please sign in to comment.