Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add example for embedding simple python scripts and debugging them using VSCode #13

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 34 additions & 0 deletions .github/workflows/graalpy-script-debug.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
name: Test GraalPy Scripts Guide
on:
push:
paths:
- 'graalpy/graalpy-scripts-debug/**'
- '.github/workflows/graalpy-scripts-debug.yml'
pull_request:
paths:
- 'graalpy/graalpy-scripts-debug/**'
- '.github/workflows/graalpy-scripts-debug.yml'
workflow_dispatch:
permissions:
contents: read
jobs:
run:
name: 'graalpy-scripts-debug'
runs-on: ubuntu-latest
timeout-minutes: 15
steps:
- uses: actions/checkout@v4
- uses: graalvm/setup-graalvm@v1
with:
java-version: '23.0.1'
distribution: 'graalvm'
github-token: ${{ secrets.GITHUB_TOKEN }}
cache: 'maven'
- name: Build, test, and run 'graalpy-scripts-debug' using Maven
run: |
cd graalpy/graalpy-scripts-debug
./mvnw --no-transfer-progress test
- name: Build, test, and run 'graalpy-scripts-debug' using Gradle
run: |
cd graalpy/graalpy-scripts-debug
./gradlew test
12 changes: 12 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/.gitattributes
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
#
# https://help.github.com/articles/dealing-with-line-endings/
#
# Linux start script should use lf
/gradlew text eol=lf

# These are Windows script files and should use crlf
*.bat text eol=crlf

# Binary files should be left untouched
*.jar binary

11 changes: 11 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# Ignore Gradle project-specific cache directory
.gradle

# Ignore Gradle build output directory
build

# Ignore maven build output directory
target

# Ignore JDTLS build directory
bin
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
wrapperVersion=3.3.2
distributionType=only-script
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.9.9/apache-maven-3.9.9-bin.zip
6 changes: 6 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/.vscode/extensions.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
{
"recommendations": [
"ms-python.python",
"vscjava.vscode-java-pack"
]
}
8 changes: 8 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/.vscode/launch.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"configurations": [{
"name": "GraalPy: Attach embedded",
"type": "debugpy",
"request": "attach",
"connect": { "host": "localhost", "port": 4711 },
}]
}
244 changes: 244 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,244 @@
# Using and Debugging Python Scripts in Java Applications using VSCode

Simple, unpackaged Python scripts can be run and shipped with Java applications.
The [GraalVM Polyglot APIs](https://www.graalvm.org/latest/reference-manual/embed-languages/) make it easy to run scripts that are simply included in the Java resources.

## 1. Getting Started

In this guide, we will add a small Python script to calculate the similarity of two files to a JavaFX application:
![Screenshot of the app](screenshot.png)

## 2. What you will need

To complete this guide, you will need the following:

* Some time on your hands
* A decent text editor or IDE
* A supported JDK[^1], preferably the latest [GraalVM JDK](https://graalvm.org/downloads/)

[^1]: Oracle JDK 17 and OpenJDK 17 are supported with interpreter only for GraalPy, but JavaFX requires JDK 21 or newer.
GraalVM JDK 21, Oracle JDK 21, OpenJDK 21 and offer GraalPy [JIT compilation](https://www.graalvm.org/latest/reference-manual/embed-languages/#runtime-optimization-support).
Note: GraalVM for JDK 17 is **not supported** for GraalPy.

## 3. Solution

We encourage you to check out the [completed example](./) and follow with this guide step by step.

## 4. Writing the application

You can use either [Maven](https://openjfx.io/openjfx-docs/#maven) or [Gradle](https://openjfx.io/openjfx-docs/#gradle) to run the JavaFX example application.
We will demonstrate on both build systems.

## 4.1 Dependency configuration

We have added the required dependencies for GraalPy in the `<dependencies>` section of the POM or to the `dependencies` block in the `build.gradle.kts` file.

`pom.xml`
```xml
<dependency>
<groupId>org.graalvm.polyglot</groupId>
<artifactId>python</artifactId> <!-- ① -->
<version>24.1.1</version>
<type>pom</type> <!-- ② -->
</dependency>
<dependency>
<groupId>org.graalvm.polyglot</groupId>
<artifactId>polyglot</artifactId> <!-- ③ -->
<version>24.1.1</version>
</dependency>
<dependency>
<groupId>org.graalvm.tools</groupId>
<artifactId>dap-tool</artifactId> <!-- ④ -->
<version>24.1.1</version>
</dependency>
```

`build.gradle.kts`
```kotlin
implementation("org.graalvm.polyglot:python:24.1.1") // ①
implementation("org.graalvm.polyglot:polyglot:24.1.1") // ③
implementation("org.graalvm.tools:dap-tool:24.1.1") // ④
```

❶ The `python` dependency is a meta-package that transitively depends on all resources and libraries to run GraalPy.

❷ Note that the `python` package is not a JAR - it is simply a `pom` that declares more dependencies.

❸ The `polyglot` dependency provides the APIs to manage and use GraalPy from Java.

❹ The `dap` dependency provides a remote debugger for GraalPy that we can use when Python code is embedded in a Java application.

## 4.2 Adding a Python script

We can just include simple Python scripts in our resources source folder.
In this example, the script contains a function that uses the Python standard library to compute the similarity between two files.

`src/main/resources/compare_files.py`
```python
import polyglot # pyright: ignore

from difflib import SequenceMatcher
from os import PathLike


@polyglot.export_value # ①
def compare_files(a: PathLike, b: PathLike) -> float:
with open(a) as file_1, open(b) as file_2:
file1_data = file_1.read()
file2_data = file_2.read()
similarity_ratio = SequenceMatcher(None, file1_data, file2_data).ratio()
return similarity_ratio
```

❶ The only GraalPy-specific code here is this `polyglot.export_value` annotation, which makes the function accessible by name to the Java world.

## 4.2.1 Working with GraalPy in VSCode

You can use [pyenv](https://github.com/pyenv/pyenv) or [pyenv-win](https://github.com/pyenv-win/pyenv-win) with the [Python extensions](https://marketplace.visualstudio.com/items?itemName=ms-python.python) in VSCode to setup and use GraalPy during development.
You can than edit and debug your Python files using the standard Python tooling.

![Gif animation of installing GraalPy with pyenv](./graalpy-vscode-pyenv.gif)
![Gif animation of using GraalPy in VSCode](./graalpy-vscode-select.gif)
![Gif animation of debugging with GraalPy in VSCode](./graalpy-vscode-debug.gif)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am getting this:
image
have you seen anything like that before?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Argh. They haven't pushed the new version out, yet, and I still had my local changes that I pushed upstream. They updated the debugpy version in the extension to one with the fix (microsoft/vscode-python-debugger#505) but haven't pushed a new extension release


## 4.3 Creating a Python context

GraalVM provides Polyglot APIs to make starting a Python context easy.
We create the Python context in the JavaFX `start` method.
We also override the `stop` method to close the context and free any associated resources.

`App.java`
```java
public class App extends Application {
private Context context;

@Override
public void stop() throws Exception {
context.close();
super.stop();
}

@Override
public void start(Stage stage) {
context = Context.newBuilder("python")
.allowIO(IOAccess.newBuilder() // ①
.fileSystem(FileSystem.newReadOnlyFileSystem(FileSystem.newDefaultFileSystem()))
.build())
.allowPolyglotAccess(PolyglotAccess.newBuilder() // ②
.allowBindingsAccess("python")
.build())
// These are all the options we need to run the app
```

❶ By default, GraalPy will be sandboxed completely, but our script wants to access files.
Read-only access is enough for this case, so we grant no more.

❷ Our script exposes the `compare_files` function by name to the Java world.
We explicitly allow this as well.

## 4.3 Calling the Python script from Java

`App.java`
```java
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I want to follow just the guide without looking at the reference solution, then this snippet is not enough. In other guides like the Micronaut/Spring guide we included even the large html listing just so that the guide itself is self-contained. We should probably be consistent in this?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point, I'll expand the readme

try {
context.eval(Source.newBuilder("python", App.class.getResource("/compare_files.py")).build()); // ①
} catch (IOException e) {
throw new RuntimeException(e);
}
final Value compareFiles = context.getBindings("python").getMember("compare_files"); // ②

target.setOnDragDropped((event) -> {
var success = false;
List<File> files;
if ((files = event.getDragboard().getFiles()) != null && files.size() == 2) {
try {
File file0 = files.get(0), file1 = files.get(1);
var result = compareFiles.execute(file0.getAbsolutePath(), file1.getAbsolutePath()).asDouble(); // ③
target.setText(String.format("%s = %f x %s", file0.getName(), result, file1.getName()));
success = true;
} catch (RuntimeException e) {
target.setText(e.getMessage());
}
}
resetTargetColor(target);
event.setDropCompleted(success);
event.consume();
});
```

❶ We can pass a resource URL to the GraalVM Polyglot [`Source`](https://docs.oracle.com/en/graalvm/enterprise/20/sdk/org/graalvm/polyglot/Source.html) API.
The content is read by the `Source` object, GraalPy and the Python code do not gain access to Java resources this way.

❷ Python objects are returned using a generic [`Value`](https://docs.oracle.com/en/graalvm/enterprise/20/sdk/org/graalvm/polyglot/Value.html) type.

❸ As a Python function, `compare_files` can be executed.
GraalPy accepts Java objects and tries to match them to the appropriate Python types.
Return values are again represented as `Value`.
In this case we know the result will be a Python `float`, which can be converted to a Java `double`.

## 5. Running the application

If you followed along with the example, you can now compile and run your application from the commandline:

With Maven:

```shell
./mvnw compile
./mvnw javafx:run
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When I followed the JavaFX quickstart from https://openjfx.io/openjfx-docs/#maven it did not generate the wrapper. Maybe this should be mvn compiler etc. or mention something like "if you don't use Maven wrapper, the commands would be ...".

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ouch, the project generated by javafx-archetype-simple seem to be broken on many fronts...

  • it sets maven-compiler-plugin to version 3.8.0, but that seems to be broken so one must manually update to at least 3.8.1
  • it generates module-info.java, so javac does module stuff and since polyglot is a proper module, one has to add requires org.graalvm.polyglot; to the generated module-info.java.
  • after fixing all that it fails in runtime on some Java module permission issue

All in all, maybe it's easier to not even mention this Java FX quickstart guide..?

```

With Gradle:

```shell
./gradlew assemble
./gradlwe run
```

## 5.1 Debugging embedded Python code

Your Python code may behave differently when run in a Java embedding.
This can have many reasons, from different types passed in from Java, permissions of the GraalVM Polyglot sandbox, to Python libraries assuming OS-specific process properties that Java applications do not expose.

To debug Python scripts, we recommend you use VSCode.
Make sure you have installed the [Python extensions](https://marketplace.visualstudio.com/items?itemName=ms-python.python).
Where we build the Python context, we can add the following options to enable remote debugging:

`App.java`
```java
.option("dap", "localhost:4711")
.option("dap.Suspend", "false")
```

This instructs the runtime to accept [DAP]() connections on port 4711 and continue execution.
We add a debug configuration to VSCode to match:

`.vscode/launch.json`
```json
{
"configurations": [{
"name": "GraalPy: Attach embedded",
"type": "debugpy",
"request": "attach",
"connect": { "host": "localhost", "port": 4711 },
}]
}
```

When we run the application now, we will see the following output:

```
[Graal DAP] Starting server and listening on localhost/127.0.0.1:4711
```

We can connect using VSCode or any other DAP client.
The loaded sources can be opened to view the Python code as loaded from the Java resources.
We can set breakpoints and inspect runtime state as we would expect.

![Gif animation debugging GraalPy in Java in VSCode](./graalpy-vscode-dap-debug.gif)

## 6. Next steps

- Use GraalPy with popular Java frameworks, such as [Spring Boot](../graalpy-spring-boot-guide/README.md) or [Micronaut](../graalpy-micronaut-guide/README.md)
- [Migrate from Jython](../graalpy-jython-guide/README.md) to GraalPy
- 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/)
35 changes: 35 additions & 0 deletions graalpy/graalpy-scripts-debug-guide/build.gradle.kts
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
plugins {
application;
id("org.openjfx.javafxplugin") version "0.1.0"
}

javafx {
version = "23.0.1"
modules = listOf("javafx.controls")
}

repositories {
// Use Maven Central for resolving dependencies.
mavenCentral()
}

dependencies {
implementation("org.graalvm.polyglot:python:24.1.1") // ①
implementation("org.graalvm.polyglot:polyglot:24.1.1") // ③
implementation("org.graalvm.tools:dap-tool:24.1.1") // ④

// Use JUnit Jupiter for testing.
testImplementation("org.junit.jupiter:junit-jupiter:5.11.0")

testRuntimeOnly("org.junit.platform:junit-platform-launcher")
}

application {
// Define the main class for the application.
mainClass = "com.example.App"
}

tasks.named<Test>("test") {
// Use JUnit Platform for unit tests.
useJUnitPlatform()
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
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.1-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
Loading