Fix using loro in cloudflare worker (#441)

* fix: try fix generated wasm bundler

* fix: loro-quill example

 and fix bundler patch issue

* chore: add release info
This commit is contained in:
Zixuan Chen 2024-09-05 13:10:27 +08:00 committed by GitHub
parent 94252102b3
commit dce00abd8a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 46 additions and 14 deletions

View file

@ -0,0 +1,6 @@
---
"loro-wasm": patch
"loro-crdt": patch
---
Make loro-wasm work in cloudflare worker

View file

@ -41,8 +41,7 @@ async function build() {
// const snip = `wasm-snip ./${target}/loro_wasm_bg.wasm -o ./${target}/loro_wasm_bg.wasm`; // const snip = `wasm-snip ./${target}/loro_wasm_bg.wasm -o ./${target}/loro_wasm_bg.wasm`;
// console.log(">", snip); // console.log(">", snip);
// await Deno.run({ cmd: snip.split(" "), cwd: LoroWasmDir }).status(); // await Deno.run({ cmd: snip.split(" "), cwd: LoroWasmDir }).status();
const cmd = const cmd = `wasm-opt -Os ./${target}/loro_wasm_bg.wasm -o ./${target}/loro_wasm_bg.wasm`;
`wasm-opt -Os ./${target}/loro_wasm_bg.wasm -o ./${target}/loro_wasm_bg.wasm`;
console.log(">", cmd); console.log(">", cmd);
await Deno.run({ cmd: cmd.split(" "), cwd: LoroWasmDir }).status(); await Deno.run({ cmd: cmd.split(" "), cwd: LoroWasmDir }).status();
}), }),
@ -58,8 +57,7 @@ async function build() {
} }
async function cargoBuild() { async function cargoBuild() {
const cmd = const cmd = `cargo build --target wasm32-unknown-unknown --profile ${profile}`;
`cargo build --target wasm32-unknown-unknown --profile ${profile}`;
console.log(cmd); console.log(cmd);
const status = await Deno.run({ const status = await Deno.run({
cmd: cmd.split(" "), cmd: cmd.split(" "),
@ -87,8 +85,7 @@ async function buildTarget(target: string) {
} }
// TODO: polyfill FinalizationRegistry // TODO: polyfill FinalizationRegistry
const cmd = const cmd = `wasm-bindgen --weak-refs --target ${target} --out-dir ${target} ../../target/wasm32-unknown-unknown/${profileDir}/loro_wasm.wasm`;
`wasm-bindgen --weak-refs --target ${target} --out-dir ${target} ../../target/wasm32-unknown-unknown/${profileDir}/loro_wasm.wasm`;
console.log(">", cmd); console.log(">", cmd);
await Deno.run({ cmd: cmd.split(" "), cwd: LoroWasmDir }).status(); await Deno.run({ cmd: cmd.split(" "), cwd: LoroWasmDir }).status();
console.log(); console.log();
@ -106,6 +103,16 @@ async function buildTarget(target: string) {
wasm + "\n" + patch, wasm + "\n" + patch,
); );
} }
if (target === "bundler") {
console.log("🔨 Patching bundler target");
const patch = await Deno.readTextFile(
path.resolve(__dirname, "./bundler_patch.js"),
);
await Deno.writeTextFile(
path.resolve(targetDirPath, "loro_wasm.js"),
patch,
);
}
} }
build(); build();

View file

@ -0,0 +1,16 @@
// See https://github.com/loro-dev/loro/issues/440
// Without this patch, Cloudflare Worker would raise issue like: "Uncaught TypeError: wasm2.__wbindgen_start is not a function"
import * as wasm from "./loro_wasm_bg.wasm";
import * as imports from "./loro_wasm_bg.js";
if (wasm.__wbindgen_start) {
imports.__wbg_set_wasm(wasm);
wasm.__wbindgen_start();
} else {
const wkmod = await import("./loro_wasm_bg.wasm");
const instance = new WebAssembly.Instance(wkmod.default, {
"./loro_wasm_bg.js": imports,
});
imports.__wbg_set_wasm(instance.exports);
}
export * from "./loro_wasm_bg.js";

View file

@ -57,7 +57,7 @@
}; };
text.subscribe((e) => { text.subscribe((e) => {
if (e.local) { if (e.by === "local") {
Promise.resolve().then(sync); Promise.resolve().then(sync);
} }
Promise.resolve().then(() => { Promise.resolve().then(() => {

View file

@ -2,7 +2,7 @@
* The skeleton of this binding is learned from https://github.com/yjs/y-quill * The skeleton of this binding is learned from https://github.com/yjs/y-quill
*/ */
import { Delta, Loro, LoroText, setDebug } from "loro-crdt"; import { Delta, Loro, LoroText } from "loro-crdt";
import Quill, { DeltaOperation, DeltaStatic, Sources } from "quill"; import Quill, { DeltaOperation, DeltaStatic, Sources } from "quill";
// @ts-ignore // @ts-ignore
import isEqual from "is-equal"; import isEqual from "is-equal";
@ -35,11 +35,14 @@ export class QuillBinding {
}); });
this.quill = quill; this.quill = quill;
this.richtext = doc.getText("text"); this.richtext = doc.getText("text");
this.richtext.subscribe(doc, (event) => { this.richtext.subscribe((event) => {
Promise.resolve().then(() => { if (event.by !== "import") {
if (!event.local && event.diff.type == "text") { return;
console.log(doc.peerId, "CRDT_EVENT", event); }
const eventDelta = event.diff.diff;
for (const e of event.events) {
if (e.diff.type == "text") {
const eventDelta = e.diff.diff;
const delta: Delta<string>[] = []; const delta: Delta<string>[] = [];
let index = 0; let index = 0;
for (let i = 0; i < eventDelta.length; i++) { for (let i = 0; i < eventDelta.length; i++) {
@ -74,7 +77,7 @@ export class QuillBinding {
quill.setContents(new Delta(a), "this" as any); quill.setContents(new Delta(a), "this" as any);
} }
} }
}); }
}); });
quill.setContents( quill.setContents(
new Delta( new Delta(