If you haven't already done so, please read this document to understand the build requirements for your operating system. If you are specifically interested in building libraries for WebAssembly, read Libraries WebAssembly. Emscripten that is needed to build the project will be provisioned automatically, unless EMSDK_PATH
variable is set or emscripten is already present in src\mono\browser\emsdk
directory.
Windows build requirements
Note: The EMSDK has an implicit dependency on Python for it to be initialized. A consequence of this is that if the system doesn't have Python installed prior to attempting a build, the automatic provisioning will fail and be in an invalid state. Therefore, if Python needs to be installed after a build attempt the $reporoot/src/mono/browser/emsdk
directory should be manually deleted and then a rebuild attempted.
At this time no other build dependencies are necessary to start building for WebAssembly. If you haven't already done so, please read this document to understand configurations. Artifacts will be placed in artifacts/bin/microsoft.netcore.app.runtime.browser-wasm/Release/
.
- To build the whole repository, including libraries:
make build-all
- To build just the runtime (useful when doing incremental builds with only runtime changes):
make runtime
Note: Additional msbuild arguments can be passed with: make build-all MSBUILD_ARGS="/p:a=b"
build.cmd -os browser -subset mono+libs
in the repo top level directory.
- To build the whole repository, including libraries:
build.sh -os browser -subset mono+libs
in the repo top level directory.
The latest engines can be installed with jsvu (JavaScript engine Version Updater /~https://github.com/GoogleChromeLabs/jsvu)
- Install npm with brew:
brew install npm
- Install jsvu with npm:
npm install jsvu -g
- Run jsvu and install
v8
,SpiderMonkey
, orJavaScriptCore
engines:
jsvu
Add ~/.jsvu
to your PATH
:
export PATH="${HOME}/.jsvu:${PATH}"
-
Install node/npm from https://nodejs.org/en/ and add its npm and nodejs directories to the
PATH
environment variable -
Install jsvu with npm:
npm install jsvu -g
- Run jsvu and install
v8
,SpiderMonkey
, orJavaScriptCore
engines:
jsvu
- Add
~/.jsvu
to thePATH
environment variable
Library tests can be run with js engines: v8
, SpiderMonkey
,or JavaScriptCore
:
v8
:make run-tests-v8-$(lib_name)
- SpiderMonkey:
make run-tests-sm-$(lib_name)
- JavaScriptCore:
make run-tests-jsc-$(lib_name)
- Or default:
make run-tests-$(lib_name)
. This runs the tests withv8
.
For example, for System.Collections.Concurrent
: make run-tests-v8-System.Collections.Concurrent
Library tests on windows can be run as described in testing-libraries documentation. Without setting additional properties, it will run tests for all libraries on v8
engine:
.\build.cmd libs.tests -test -os browser
JSEngine
property can be used to specify which engine to use. Right nowv8
andSpiderMonkey
engines can be used.
Examples of running tests for individual libraries:
.\dotnet.cmd build /t:Test /p:TargetOS=browser src\libraries\System.Collections.Concurrent\tests
.\dotnet.cmd build /t:Test /p:TargetOS=browser /p:JSEngine="SpiderMonkey" src\libraries\System.Text.Json\tests
Or they can be run with a browser (Chrome):
make run-browser-tests-$(lib_name)
Note: this needs chromedriver
, and Google Chrome
to be installed.
For example, for System.Collections.Concurrent
: make run-browser-tests-System.Collections.Concurrent
These tests are run with xharness wasm test-browser
, for running on the browser. And xharness wasm test
for others.
The wrapper script used to actually run these tests, accepts:
$XHARNESS_COMMAND
, which defaults to test
.
$XHARNESS_CLI_PATH
(see next section)
XHarness consists of two pieces for WASM
- set
XHARNESS_CLI_PATH=/path/to/xharness/artifacts/bin/Microsoft.DotNet.XHarness.CLI/Debug/net9.0/Microsoft.DotNet.XHarness.CLI.dll
Note: Additional msbuild arguments can be passed with: make .. MSBUILD_ARGS="/p:a=b"
All library tests are hosted by WasmTestRunner.csproj
. The project references XHarness nuget for running tests using Xunit. To make changes and iterate quickly
- Add property
<RestoreAdditionalProjectSources>$(RestoreAdditionalProjectSources);LOCAL_CLONE_OF_XHARNESS\artifacts\packages\Debug\Shipping</RestoreAdditionalProjectSources>
inWasmTestRunner.csproj
. - Set environment variable in your terminal
$env:NUGET_PACKAGES="$pwd\.nuget"
(so that nuget packages are restored to local folder.nuget
) - Run "Pack" in the XHarness solution in Visual Studio on
Microsoft.DotNet.XHarness.TestRunners.Common
orMicrosoft.DotNet.XHarness.TestRunners.Xunit
based on your changes (it will generate a nuget package inLOCAL_CLONE_OF_XHARNESS\artifacts\packages\Debug\Shipping
). - Build WasmTestRunner
.\dotnet.cmd build -c Debug .\src\libraries\Common\tests\WasmTestRunner\WasmTestRunner.csproj
. - If you need to iterate, delete Xunit or Common nuget cache
rm -r .\.nuget\microsoft.dotnet.xharness.testrunners.xunit\
orrm -r .\.nuget\microsoft.dotnet.xharness.testrunners.common\
.
Exceptions thrown after the runtime starts get symbolicating from js itself. Exceptions before that, like asserts containing native traces get symbolicated by xharness using src/mono/wasm/symbolicator
.
If you need to symbolicate some traces manually, then you need the corresponding dotnet.native.js.symbols
file. Then:
src/mono/wasm/symbolicator$ dotnet run /path/to/dotnet.native.js.symbols /path/to/file/with/traces
When not relinking, or not building with AOT, you can find dotnet.native.js.symbols
in the runtime pack.
Debugger tests need Google Chrome
to be installed.
make run-debugger-tests
To run a test with FooBar
in the name:
make run-debugger-tests TEST_FILTER=FooBar
(See https://learn.microsoft.com/dotnet/core/testing/selective-unit-tests?pivots=xunit for filter options)
Additional arguments for dotnet test
can be passed via MSBUILD_ARGS
or TEST_ARGS
. For example MSBUILD_ARGS="/p:WasmDebugLevel=5"
. Though only one of TEST_ARGS
, or TEST_FILTER
can be used at a time.
Chrome can be installed for testing by setting InstallChromeForDebuggerTests=true
when building the tests.
The samples in src/mono/sample/wasm
can be build and run like this:
- console Hello world sample
dotnet build /t:RunSample console-v8/Wasm.Console.V8.Sample.csproj
- browser TestMeaning sample
dotnet build /t:RunSample browser/Wasm.Browser.Sample.csproj
To build and run the samples with AOT, add /p:RunAOTCompilation=true
to the above command lines.
- bench sample
Also check bench sample to measure mono/wasm runtime performance.
Use dotnet run to run wasm applications
The wasm templates, located in the templates
directory, are templates for dotnet new
, VS and VS for Mac. They are packaged and distributed as part of the wasm-experimental
workload. We have 2 templates, wasmbrowser
and wasmconsole
, for browser and console WebAssembly applications.
For details about using dotnet new
see the dotnet tool documentation.
To test changes in the templates, use dotnet new install --force src/mono/wasm/templates/templates/browser
.
Example use of the wasmconsole
template:
> dotnet new wasmconsole
> dotnet publish
> cd bin/Debug/net9.0/browser-wasm/AppBundle
> node main.mjs
Hello World!
Args:
JavaScript part of the .NET runtime for WebAssembly is composed of several ES6 modules.
dotnet.js
is entry point ("loader") containing public API for interacting with .NET runtime. This is the file that you import into your JavaScript code. This file don't change during app development. Adding fingerprint on this file can be turned on by setting msbuild property WasmFingerprintDotnetJs=true
.
dotnet.native.js
contains emscripten API and is loaded by the loader. This file changes when native build (relink) is invoked. The file name always contains fingerprint to avoid stale cache during app development.
dotnet.runtime.js
contains the rest JavaScript part of the .NET runtime. The file name also always contains fingerprint.
We have few tools to analyze binary wasm files. The wa-info and wa-diff to analyze dotnet.native.wasm
in the AppBundle
directory, once you build your app. These can be easily installed as dotnet tools.
They are handy to quickly disassemble functions and inspect webassembly module sections. The wa-diff is able to compare 2 wasm files, so you can for example check the effect of changes in your source code. You can see changes in the functions code as well as changes in sizes of sections and of code.
There is also the wa-edit tool, which is now used to prototype improved warm startup.
Bumping Emscripten version involves these steps:
- update /~https://github.com/dotnet/runtime/blob/main/src/mono/browser/emscripten-version.txt
- bump emscripten versions in docker images in /~https://github.com/dotnet/dotnet-buildtools-prereqs-docker
- bump emscripten in /~https://github.com/dotnet/emsdk
- bump docker images in /~https://github.com/dotnet/icu, update emscripten files in eng/patches/
- update version number in docs
- update
Microsoft.NET.Runtime.Emscripten.<emscripten version>.Node.win-x64
package name, version and sha hash in /~https://github.com/dotnet/runtime/blob/main/eng/Version.Details.xml and in /~https://github.com/dotnet/runtime/blob/main/eng/Versions.props. the sha is the commit hash in /~https://github.com/dotnet/emsdk and the package version can be found at https://dev.azure.com/dnceng/public/_packaging?_a=feed&feed=dotnet6 - update packages in the workload manifest /~https://github.com/dotnet/runtime/blob/main/src/mono/nuget/Microsoft.NET.Workload.Mono.Toolchain.Current.Manifest/WorkloadManifest.json.in
Two things to keep in mind:
-
We use the Azure DevOps NPM registry (configured in
src/mono/browser/runtime/.npmrc
). When updatingpackage.json
, you will need to be logged in (see instructions for Windows and mac/Linux, below) in order for the registry to populate with the correct package versions. Otherwise, CI builds will fail. -
Currently the Emscripten SDK uses NPM version 6 which creates
package-lock.json
files in the "v1" format. When updating NPM packages, it is important to use this older version of NPM (for example by using theemsdk_env.sh
script to set the right environment variables) or by using the--lockfile-format=1
option with more recent versions of NPM.
The steps below will download the vsts-npm-auth
tool from https://dev.azure.com/dnceng/public/_artifacts/feed/dotnet-public-npm/connect/npm
In folder src\mono\browser\runtime\
rm -rf node_modules
rm package-lock.json
npm install -g vsts-npm-auth`
vsts-npm-auth -config .npmrc
npm cache clean --force
npm outdated
npm update --lockfile-version=1
Go to https://dev.azure.com/dnceng/public/_artifacts/feed/dotnet-public-npm/connect/npm and log in and click on the "Other" tab.
Follow the instructions to set up your ~/.npmrc
with a personal authentication token.
In folder src/mono/browser/runtime/
rm -rf node_modules
rm package-lock.json
npm cache clean --force
npm outdated
npm update --lockfile-version=1
- Is enforced via eslint and rules are in
./.eslintrc.js
- You could check the style by running
npm run lint
insrc/mono/browser/runtime
directory - You can install plugin into your VS Code to show you the errors as you type
- For PRs, tests are generally triggered based on path changes. But if you have a change which would not trigger the relevant builds, then you can run
runtime-wasm
pipeline manually to run all of them. Comment/azp run runtime-wasm
on the PR.
only-pc
meansonly on relevant path changes
runtime
runs jobs only when relevant paths change. And forAOT
, only smoke tests are run.
. | runtime |
---|---|
libtests | linux+windows: all, only-pc |
libtests eat | linux+windows: smoke, only-pc |
libtests aot | linux+windows: smoke, only-pc |
high resource aot | none |
Wasm.Build.Tests | linux+windows: only-pc |
Debugger tests | linux+windows: only-pc |
Runtime tests | linux+windows: only-pc |
runtime-wasm*
pipelines are triggered manually, and they only run the jobs that would not run on any default pipelines based on path changes.- The
AOT
jobs run only smoke tests onruntime
, and onruntime-wasm*
pipelines all theAOT
tests are run.
. | runtime-wasm | runtime-wasm-libtests | runtime-wasm-non-libtests |
---|---|---|---|
libtests | linux+windows: all | linux+windows: all | none |
libtests eat | linux: all | linux: all | none |
libtests aot | linux+windows: all | linux+windows: all | none |
high resource aot | linux+windows: all | linux+windows: all | none |
Wasm.Build.Tests | linux+windows | none | linux+windows |
Debugger tests | linux+windows | none | linux+windows |
Runtime tests | linux | none | linux |
Multi-thread | linux: all tests | linux: all tests | none |
-
runtime-extra-platforms
does not run any wasm jobs on PRs -
high resource aot
runs a few specific library tests with AOT, that require more memory to AOT. -
runtime-wasm-dbgtests
runs all the debugger test jobs
runtime
runs all the wasm jobs, butAOT
still only runs smoke tests.runtime-extra-platforms
also runs by default. And it runs only the cases not covered byruntime
.
. | runtime | runtime-extra-platforms (always run) |
---|---|---|
libtests | linux+windows: all | none |
libtests eat | linux: all | none |
libtests aot | linux+windows: smoke | linux+windows: all |
high resource aot | none | linux+windows: all |
Wasm.Build.Tests | linux+windows | none |
Debugger tests | linux+windows | none |
Runtime tests | linux | none |
Multi-thread | linux: build only | none |
high resource aot
runs a few specific library tests with AOT, that require more memory to AOT.
Tests are run with V8, Chrome, node, and wasmtime for the various jobs.
- V8: the version used is from
eng/testing/BrowserVersions.props
. This is used for all the library tests, and WBT, but not runtime tests. - Chrome: Same as V8.
- Node: fixed version from emsdk
- wasmtime - fixed version in
src/mono/wasi/wasi-sdk-version.txt
.
This file is updated once a week by a github action .github/workflows/bump-chrome-version.yml
, and the version is obtained by src/tasks/WasmBuildTasks/GetChromeVersions.cs
task.
- V8 version used to run the microbenchmarks is from
eng/testing/BrowserVersions.props
TBD
- when the base OS is upgraded, check if the version of node installed in the
eng/pipelines/coreclr/templates/run-performance-job.yml
needs an upgrade too.