WebAssembly Language Runtimes (or WLR for short) offers pre-built wasm32-wasi binaries for language runtimes and static builds of common open source libraries.
This repository contains the build scripts and patches that are used to do those builds, as well as examples on how to use them.
WLR is used in projects like mod_wasm for traditional deployments and Wasm Workers Server for the development of serverless apps. To get a glimpse of that you could:
- [5 min] Run WordPress with php.wasm, mod_wasm and Apache. (You'll need Docker for a container with Apache).
- [10 min] Create and run a Ruby worker on wws.
The released assets are also easy to use with various platforms and tools.
- [5 min] Take a look at how to run the python-wasm Docker container. (You'll need the latest Docker Desktop with enabled containerd).
- [10 min] Follow the steps in Fermyon's blog post on PHP, Spin and Fermyon cloud.
If you are into porting of C-based apps to wasm32-wasi you could play with the libs.
- [10 min] Build a C-app that uses the libuuid and zlib static libraries.
- [1 day – 2 weeks] Port a static library. Take a look at the build scripts for
libs/zlib
as an example. Pick a library you want to see ported. Try building it to wasm32-wasi. (Prior knowledge of building C apps with autotools, make, cmake, pkg-config will be an advantage).
Here is a reference to the latest releases of all built projects.
Language runtime | Latest release |
---|---|
php | 8.2.0 |
python | 3.11.1 |
ruby | 3.2.0 |
Library | Latest release |
---|---|
libs/bundle_wlr | 0.1.0 |
libs/libjpeg | 2.1.5.1 |
libs/libpng | 1.6.39 |
libs/libuuid | 1.0.3 |
libs/libxml2 | 2.10.3 |
libs/oniguruma | 6.9.8 |
libs/sqlite | 3.39.2 |
libs/zlib | 1.2.13 |
The rest of this document will help you if you want to build some of the assets on your own, or contribute with a patch, or add support for new release targets.
All you need in order to run these builds is to have docker
or podman
available in your system. You can execute the following Makefile
targets:
-
php/php-7.3.33
,php/php-7.4.32
,php/wasmedge-php-7.4.32
,php/php-8.1.11
,php/php-8.2.0
- Resulting binaries are placed in
build-output/php
.
- Resulting binaries are placed in
-
python/v3.11.1
- Resulting binaries are placed in
build-output/python
.
- Resulting binaries are placed in
-
ruby/v3_2_0
- Resulting binaries are placed in
build-output/ruby
.
- Resulting binaries are placed in
If you are interested in knowing more about the build system and how it produces the final binaries, keep reading.
All build orchestration scripts are written in bash in this initial version. They start with a wlr-
prefix (short for WebAssembly Language Runtimes). Review the build orchestration scripts section for more info.
All intermediary source code checkouts and build objects get created within the build-staging
folder. The final output gets written to the build-output
folder.
The patches and scripts to build different language runtimes are organized in a folder hierarchy that follows the tagged versions from the respective source code repositories. Several wlr-
scripts are added around that to facilitate setup of a local clone of the repository, application of respective patches and building with respective build configuration options.
For language runtimes we have something like this.
${LANGUAGE_RUNTIME_NAME} (e.g. 'php')
├── README.md (generic notes about what was patched to build this language)
├── ${VERSION_TAG_IN_REPO} (e.g. 'php-7.4.32' from the php repo)
│ ├── README.md (generic notes about what was patched to build this version)
│ ├── patches (consecutive patches on top of the tagged version, applied before building)
│ │ ├── (e.g. '0001-Initial-port-of-7.3.33-patch-to-7.4.32.patch')
│ │ ├── (e.g. '0002-Fix-mmap-issues.-Add-readme.patch')
│ │ └── Etc...
│ ├── wlr-build-deps.sh (script that builds dependencies)
│ └── wlr-build.sh (script that builds for this tag)
└── wlr-env-repo.sh (script that sets up the source code repository for given langauge and tag)
For common shared libraries we have something limilar.
libs (common libraries, needed by different modules)
└── ${LIBRARY_NAME} (e.g. 'sqlite')
├── README.md (generic notes about what was patched to build this language)
├──${VERSION_TAG_IN_REPO} (e.g. 'version-3.39.2' from the sqlite repo)
│ ├── patches (consecutive patches on top of the tagged version, applied before building)
│ │ ├── (e.g. '0001-Patch-to-build-sqlite-3.39.2-for-wasm32-wasi.patch')
│ │ ├── (e.g. '0002-Remove-build-script-from-patched-repo.patch')
│ │ └── Etc...
│ └── wlr-build.sh (script that builds for this tag)
└── wlr-env-repo.sh (script that sets up the source code repository for given langauge and tag)
-
The main script used to build something is
wlr-make.sh
in the root folder. It gets called with a path to the folder for a respective tag of what we want to build. -
It will first source the
scripts/wlr-env.sh
script. This one sets all environment variables necessary to checkout and build the desired target. It gets the same path fromwlr-make.sh
and is useful when you try to build locally. -
Then
wlr-make.sh
will callscripts/wlr-setup-repo.sh
to create a shallow clone of the necessary repository only for the specific tag that we want to build. On top of that it applies any relevant patches from thepatches
subfolder of the tagged version folder. -
As a final step
wlr-make.sh
will callscripts/wlr-build.sh
which will build from the code in the respective repository. -
Before building this will call a
$LANG/$TAG/wlr-build-deps.sh
if there is any to build required dependencies and setup CFLAGS or LDFLAGS for their artifacts. Then it will call the$LANG/$TAG/wlr-build.sh
script to build the actual target itself.
To add a build setup for a new version of something that is already configured:
- Add a subfolder with the respective tag, like this:
mkdir php/php-7.3.33
- Create a
wlr-build.sh
script in the target folder, like this:
touch php/php-7.3.33/wlr-build.sh
- Setup your build environment via
scripts/wlr-env.sh
with the target path then query the respective environment variables, like this:
source scripts/wlr-env.sh php/php-7.3.33
export | grep WLR_
- Create a local clone of the respective tag in build-staging, like this:
scripts/wlr-setup-repo.sh
- Open your favorite IDE in the said clone to iterate building from the tag until it works, like this:
code build-staging/php/php-7.3.33/checkout
- Patch the checked out code where necessary. Add flags and build commands to the
wlr-build.sh
script in the target folder and each time rebuild like this:
scripts/wlr-build.sh
- After you manage to get a working build, add proper lines at the end of your
wlr-build.sh
script to copy from thebuild-staging
folder to the respectivebuild-output
location, like this:
...
logStatus "Preparing artifacts... "
mkdir -p ${WLR_OUTPUT}/bin 2>/dev/null || exit 1
cp sapi/cgi/php-cgi ${WLR_OUTPUT}/bin/ || exit 1
logStatus "DONE. Artifacts in ${WLR_OUTPUT}"
- Commit the patch changes from 6. into the local shallow clone. If necessary, split them into commits. Then export them to the target folder (e.g.
php/php-7.3.33/patches
) like this:
scripts/wlr-update-patches.sh
- Now add and commit the new target description folder containing the build script and respective patches to the current repository, like this:
git add php/php-7.3.33
git commit -m "Add support to build php version 7.3.33"
In order to release a new version, you first have to tag the project you want to release. You can create a tag by using the scripts/wlr-tag.sh
script.
This script accepts the path to be released, and will create a local tag of the form <project>/<version>+YYYYMMDD-<short-sha>
. All parameters will be automatically filled by the script, so in order to create a valid tag for PHP 7.3.33, for example, you only have to execute:
scripts/wlr-tag.sh php/php-7.3.33
This will create a tag like the following in your local repository: php/7.3.33+20221123-d3d8901
.
When you push the tag to the remote repository, a GitHub release will be created automatically, and relevant artifacts will be automatically published to the release.