Skip to content

Commit e7871a4

Browse files
chore: add webidl example (#544)
* feat(examples): add WebIDL example using webidl2wit This commit adds an example that showcases `webidl2wit` usage to generate `jco` components. Signed-off-by: Victor Adossi <[email protected]> * chore: add webidl-book-library to workspace Signed-off-by: Victor Adossi <[email protected]> * feat(ci): allow installing required rust crates for examples Signed-off-by: Victor Adossi <[email protected]> * refactor(examples): add demo to run all cmd for webidl example Signed-off-by: Victor Adossi <[email protected]> --------- Signed-off-by: Victor Adossi <[email protected]>
1 parent f618baf commit e7871a4

File tree

11 files changed

+380
-5
lines changed

11 files changed

+380
-5
lines changed

.github/workflows/main.yml

+19-5
Original file line numberDiff line numberDiff line change
@@ -279,6 +279,10 @@ jobs:
279279
- name: string-reverse-upper
280280
workspace: examples/components/string-reverse-upper
281281
is-composed: true
282+
- name: webidl-book-library
283+
workspace: examples/components/webidl-book-library
284+
requires-crates:
285+
- webidl2wit-cli
282286
steps:
283287
- uses: actions/checkout@v4
284288
- uses: actions/setup-node@v3
@@ -290,7 +294,18 @@ jobs:
290294
name: jco-build
291295
path: obj
292296

293-
- name: install wac
297+
298+
- name: Install Rust
299+
if: ${{ matrix.project.requires-crates != '[]' }}
300+
run: rustup update stable --no-self-update && rustup default stable
301+
302+
- name: Install required rust crates
303+
if: ${{ matrix.project.requires-crates != '[]' }}
304+
uses: taiki-e/install-action@v2
305+
with:
306+
tool: ${{ join(matrix.project.requires-crates, ',') }}
307+
308+
- name: Install wac
294309
if: ${{ matrix.project.is-composed }}
295310
uses: jaxxstorm/action-install-gh-release@v1
296311
with:
@@ -300,9 +315,8 @@ jobs:
300315
rename-to: wac
301316
chmod: 0755
302317

303-
- name: npm install
304-
run: npm install
318+
- run: npm install
305319

306-
- name: run all script for (${{ matrix.project.name }})
320+
- name: Run all script for (${{ matrix.project.name }})
307321
working-directory: ${{ matrix.project.dir }}
308-
run: npm run all --workspace ${{ matrix.project.workspace }}
322+
run: npm run all --workspace ${{ matrix.project.workspace }}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
node_modules
2+
dist
3+
*.wasm
4+
pnpm-lock.yaml
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
v22.5.1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
# `webidl-book-library`
2+
3+
This component showcases building WebAssembly components that line up with [WebIDL][webidl] interfaces by
4+
using the experimental [built-in support for WebIDL in `jco`][jco-experimental-webidl].
5+
6+
We can accomplish this by:
7+
8+
- Writing WebIDL specifications
9+
- Using [`webidl2wit`][webidl2wit] to turn WebIDL specifications into [WIT interfaces][wit]
10+
- Using `jco componentize` to build Javascript WebAssembly components that target the relevant WIT interfaces
11+
- Using `jco transpile` to compile that component to run in a JS context (like NodeJS)
12+
- Writing host bindings (`demo.js`) that WebAssembly cna use
13+
14+
A common use case for WebIDL is targeting browsers, as the WebIDL is primarily used there
15+
to document interfaces for the web platform. This example stops short of using the web platform
16+
WebIDL in favor of showing a simpler modeled interace with WebIDL: a book library (see [`./book-library.webidl`](./book-library.webidl)).
17+
18+
[jco-experimental-webidl]: https://github.com/bytecodealliance/jco/blob/main/docs/src/transpiling.md#experimental-webidl-imports
19+
[webidl]: https://en.wikipedia.org/wiki/Web_IDL
20+
[wit]: https://github.com/WebAssembly/component-model/blob/main/design/mvp/WIT.md
21+
22+
## Dependencies
23+
24+
To run this example, you'll need the following tools
25+
26+
| Tool | Description | Install instructions |
27+
|----------------------------|----------------------------------|-----------------------------------------|
28+
| [`jco`][jco] | Javascript WebAssembly toolcahin | `npm install -g @bytecodealliance/jco` |
29+
| [`webidl2wit`][webidl2wit] | Converts WebIDL to WIT | `cargo install --locked webidl2wit-cli` |
30+
31+
[jco]: https://github.com/bytecodealliance/jco
32+
[webidl2wit]: https://github.com/wasi-gfx/webidl2wit
33+
34+
## Quickstart
35+
36+
Once dependencies are installed, you can use your favorite NodeJS package manager to run the steps:
37+
38+
```console
39+
npm install
40+
npm run generate:wit
41+
npm run generate:types
42+
npm run build
43+
npm run transpile
44+
```
45+
46+
> [!NOTE]
47+
> To run all these steps at once, you can run `npm run all`
48+
49+
After running the component build, we can run the example code that uses our WebAssembly component,
50+
and the WebIDL interface & implementation we made:
51+
52+
```
53+
node demo.js
54+
```
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
// This WebIDL file will be used to automatically generate WIT,
2+
// using the `webidl2wit` tool.
3+
//
4+
// see: https://github.com/MendyBerger/webidl2wit/tree/main
5+
6+
// WebIDL enums are converted as standard WIT enums
7+
enum BookGenre {
8+
"fiction",
9+
"non-fiction",
10+
"mystery",
11+
"fantasy",
12+
"science-fiction",
13+
"biography"
14+
};
15+
16+
// WebIDL typedefs are converted into WIT type aliases
17+
typedef DOMString BookTitle;
18+
19+
// WebIDL dictionaries are turned into WIT structs
20+
dictionary Book {
21+
required BookTitle title;
22+
required DOMString author;
23+
BookGenre genre;
24+
unsigned short pages;
25+
};
26+
27+
// WebIDL interfaces become WIT resources
28+
interface Library {
29+
constructor();
30+
31+
readonly attribute unsigned long totalBooks;
32+
33+
// Add a Book
34+
boolean addBook(Book book);
35+
36+
// Remove a book
37+
boolean removeBook(DOMString title);
38+
39+
// Retrieve a book by title (if present)
40+
Book? getBookByTitle(DOMString title);
41+
42+
// List all the books
43+
sequence<Book>? listBooks();
44+
};
45+
46+
interface AdvancedLibrary : Library {
47+
FrozenArray<Book> filterBooks(DOMString name);
48+
};
49+
50+
partial interface Library {
51+
readonly attribute LibraryName libraryName;
52+
53+
// Rename this library
54+
undefined renameLibrary(LibraryName newName);
55+
};
56+
57+
typedef DOMString LibraryName;
58+
59+
// WebIDL interfaces become WIT resources
60+
interface BookManager {
61+
constructor();
62+
63+
readonly attribute Library library;
64+
65+
// Initialize a library
66+
undefined initLibrary(LibraryName name);
67+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
////////////////////////////////
2+
// WebIDL compliant interface //
3+
////////////////////////////////
4+
5+
class Book {
6+
constructor(title, author, genre, pages) {
7+
this.title = title;
8+
this.author = author;
9+
this.genre = genre || null;
10+
this.pages = pages || 0;
11+
}
12+
}
13+
14+
class Library {
15+
constructor(name) {
16+
this.libraryName = name;
17+
this.books = new Map();
18+
}
19+
20+
get totalBooks() {
21+
return this.books.size;
22+
}
23+
24+
addBook(book) {
25+
if (this.books.has(book.title)) {
26+
return false; // Book already exists
27+
}
28+
this.books.set(book.title, book);
29+
return true;
30+
}
31+
32+
removeBook(title) {
33+
return this.books.delete(title);
34+
}
35+
36+
getBookByTitle(title) {
37+
return this.books.get(title) || null;
38+
}
39+
40+
listBooks() {
41+
return Array.from(this.books.values());
42+
}
43+
44+
renameLibrary(newName) {
45+
this.libraryName = newName;
46+
}
47+
48+
asAdvancedLibrary() {
49+
const al = new AdvancedLibrary();
50+
al.libraryName = this.libraryName;
51+
al.books = this.books;
52+
return al;
53+
}
54+
}
55+
56+
class AdvancedLibrary extends Library {
57+
constructor() {
58+
super();
59+
}
60+
61+
asLibrary() {
62+
return new Library("advanced-library");
63+
}
64+
65+
filterBooks(name) {
66+
return Array.from(this.books.values()).filter((book) => {
67+
return (
68+
book.title.toLowerCase().includes(name.toLowerCase()) ||
69+
book.author.toLowerCase().includes(name.toLowerCase())
70+
);
71+
});
72+
}
73+
}
74+
75+
class BookManager {
76+
constructor() {
77+
this._library = null;
78+
}
79+
80+
initLibrary(name) {
81+
const library = new Library(name);
82+
this._library = library;
83+
}
84+
85+
library() {
86+
if (!this._library) {
87+
throw new Error("initLibrary must be called first!");
88+
}
89+
const library = this._library;
90+
this._library = null;
91+
return library;
92+
}
93+
}
94+
95+
// Wire up the implementations above to globalThis
96+
// which will be used by the component implicitly
97+
//
98+
// While somewhat awkward, current jco bindings
99+
// generate a member/namespace for every dash, so:
100+
//
101+
// "global-book-library" => globalThis.book.library
102+
// "global-console" => globalThis.console
103+
globalThis.book = {
104+
library: {
105+
AdvancedLibrary,
106+
Library,
107+
BookManager,
108+
},
109+
};
110+
111+
////////////////////////////////
112+
// Usage of transpiled module //
113+
////////////////////////////////
114+
115+
// NOTE: we use a dynamic import of our transpiled WebAssembly component
116+
// to ensure that globalThis is setup prior to the component logic running
117+
const { librarian } = await import("./dist/transpiled/librarian.js");
118+
119+
let library = librarian.createLocalLibrary();
120+
121+
library = librarian.addFavoriteBook(library);
122+
123+
const [_library, favorites] = librarian.getFavoriteBooks(library);
124+
125+
console.log("Librarian's favorite books:", favorites);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
import { BookManager } from "webidl:pkg/global-book-library";
2+
3+
export const librarian = {
4+
createLocalLibrary() {
5+
let bm = new BookManager();
6+
bm.initLibrary("great-library-of-wasmxandria");
7+
return bm.library();
8+
},
9+
10+
addFavoriteBook(library) {
11+
if (!library) {
12+
throw new Error("missing/invalid library");
13+
}
14+
library.addBook({
15+
title: "The Library Book",
16+
author: "Susan Orlean",
17+
genre: BookManager.NonFiction,
18+
pages: 317,
19+
});
20+
return library;
21+
},
22+
23+
getFavoriteBooks(library) {
24+
if (!library) {
25+
throw new Error("missing/invalid library");
26+
}
27+
let advanced = library.asAdvancedLibrary();
28+
let favorites = advanced.filterBooks("library");
29+
return [library, favorites];
30+
},
31+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "webidl-book-library-wasm",
3+
"description": "Example WebAssembly component showcasing WIT interfaces auto-generated from WebIDL",
4+
"type": "module",
5+
"scripts": {
6+
"generate:wit": "webidl2wit --webidl-interface global-book-library -i book-library.webidl -o wit/deps/book-library.wit",
7+
"generate:types": "jco types wit -o dist/generated/types",
8+
"build": "jco componentize librarian.js --wit wit/ --world-name component --out librarian.wasm --disable all",
9+
"transpile": "jco transpile librarian.wasm -o dist/transpiled",
10+
"all": "npm run generate:wit && npm run generate:types && npm run build && npm run transpile && node demo.js"
11+
},
12+
"devDependencies": {
13+
"@bytecodealliance/jco": "1.10.2"
14+
}
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
/// This package combines with of the auto-generated `book-library.wit` which
2+
/// produces the package 'webidl:pkg' which contains the `global-book-library`
3+
/// interface, which will be picked up and put on globalThis by jco.
4+
package example:component;
5+
6+
interface librarian {
7+
use webidl:pkg/global-book-library.{book, book-genre, library};
8+
9+
/// Create a local library
10+
create-local-library: func() -> library;
11+
12+
/// Add this librarian's favorite book to the library
13+
add-favorite-book: func(l: library) -> library;
14+
15+
/// Get a list of the favorite books
16+
get-favorite-books: func(l: library) -> tuple<library, list<book>>;
17+
}
18+
19+
world component {
20+
import webidl:pkg/global-book-library;
21+
22+
export librarian;
23+
}

0 commit comments

Comments
 (0)