この記事は、Zennにも投稿しています。
最近話題の組版システムのTypstですが、プラグインシステムを備えておりWASMを使って拡張することが可能です。
Plugin Function – Typst Documentation
Documentation for the `plugin` function.
 typst.app
typst.app
 
プラグインを使うことで従来のTypst言語のみでは難しかった様々な処理を行うことができます。
公式のパッケージリストに掲載されているパッケージの中にも内部でWASMプラグインを使用しているものがあります。例えばQuickJSを利用してJavaScriptを実行する「Jogs」やMarkdownをTypstに変換する「cmarker」、さらにはLaTeXをTypst構文に変換して表示する(!)「mitex」というものもあります。
この記事ではRustを使ってTypstのプラグインを作成することによりTypstプラグインの基本的な仕組みについて解説します。Typstはそれ自体Rustで作られているためRustでプラグインを書くための環境がよく整備されていますが、WASMにコンパイルできる言語であればどのような言語も使用可能です。ただし、WASIはサポートされていないため、WASIが必須な言語やライブラリを使用する際にはwasi-stubを使用する必要があります。
ZigとCの例がここにある他、その他の言語でもTypstのwasm protocolに従って関数をエクスポートすることでTypstプラグインを作成できます。
プラグインとパッケージ
Typstでのプラグインというのは
wasmファイルのことを指します。一方、パッケージはプラグインをロードする処理などが記述されたtypファイルなどを含んだ一連のファイル群のことを指します。パッケージはwasmプラグインを含まずに.typファイルのみで構成することもできます。挨拶プラグイン
まずは引数
nameを受け取りHello {name}という文字列を出力するだけのプラグインを作ってみましょう。を実行してRustのプロジェクトを作り、wasmにビルドできるようにしておいたら次に
Cargo.tomlに以下を追記します。crate-type = ["cdylib"]はwasmにビルドするのに必要です。また、wasm-minimal-protocolクレートでは関数をTypstから呼び出すのに必要な諸々をやってくれます。また、
.cargo/config.tomlファイルを作成し、デフォルトでwasmがコンパイルされるようにしておきます。次に
lib.rsを以下のように書き換えます。initiate_protocol!()を実行した後、wasm_funcアトリビュートを関数に付与することで関数をTypstにエクスポートすることができます。エクスポートされたgreet関数では、バイト列として受け取った引数にHello,を付加してそれをやはりバイト列として返しています。Rustには文字列を表す
String型がありますが、それは使わずにVec<u8>を関数から返しているのに疑問を持ったかもしれません。これは現状Typstのプラグインは「バイト列を受け取ってバイト列を返すこと」しかできないからです。では実行するために以下のコマンドでビルドします。
target/wasm32-unknown-unknown/releaseディレクトリ内にwasmが生成されたはずです。では実際にTypstで読み込んでみましょう。作成プロジェクトのルートに以下のようなTypstファイルを作成します。wasmファイルは
plugin関数を用いて読み込み、後は通常のメソッドのように使うことができます。ただし、先程記したようにプラグインとのデータのやり取りはバイト列であるため、bytes関数を使って引数をバイト列とし、str関数を用いて返り値を文字列に戻していることに注意してください。これを
でコンパイルすれば… 
このようなpdfファイルが生成されているはずです。非常に簡単ですね。

このようなpdfファイルが生成されているはずです。非常に簡単ですね。
Excel読み込みプラグイン
これだけでは面白くないのでもう少し実用的なものを作りましょう。
自分は表を作る時に雑にExcelで作ることが多いのですが、Typstでxlsxファイルは読み込めないのでCSVにいちいち変換しなければならず面倒です。これを簡略化するためにxlsxファイルを直接読み込むTypstプラグインを作ってみましょう。
一からxlsxファイルを読み込む処理を実装するとなると非常に大変ですが、プラグインを作るのにRustが使えるということは当然Rustのエコシステムを使えるということです。Rustのエコシステムは結構豊富で、今回の目的にドンピシャなcalamineというクレートを見つけました。Pure
Rust製なのでビルドも難しい所はありません(Cライブラリが混ざったRustプロジェクトをwasmにコンパイルするのはまあまあ面倒です)。
Rust製なのでビルドも難しい所はありません(Cライブラリが混ざったRustプロジェクトをwasmにコンパイルするのはまあまあ面倒です)。
では実際にプラグインを作っていきましょう。まずは前節と同様に
cargo newでRustプロジェクトを作成してからで依存関係を追加し、以下のコードを
lib.rsに書きます。get_table関数が実際の処理内容で、calamineクレートでバイト列として受け取ったxlsxファイルの内容を解析し、引数として指定された範囲の内容を読み取っています。読み取ったデータはTypstで解析できるようにtsvの文字列として返しています。Typstの表データとして直接返せれば良いのですがそのような方法は今は無いようです。
そしてこちらが上のRustコードから生成されたプラグインWASMを実行するためのファイルです。Rust内の
get_table関数にxlsxファイルの内容と引数を渡して実行し、tsvとしてパースすることでxlsxファイル内の値を表示しています。では試しに以下のような

するとこのように期待通りの表が得られました!

今回始めて知ったのですがxlsxでは計算結果もファイルの中に保持してあるようです。数式を取りたければコード内の
Book1.xlsxを作ってtypst compileを実行してみましょう。
するとこのように期待通りの表が得られました!

今回始めて知ったのですがxlsxでは計算結果もファイルの中に保持してあるようです。数式を取りたければコード内の
worksheet_rangeをworksheet_formulaに変えるといいはずです。ファイルをプラグインから読み込めない理由
先程のコードでは、わざわざTypstからファイルをバイト列として渡していましたが、Rustから直接IOできたりすれば便利なのではないでしょうか?
そのようなことができない直接的な理由は「Typstがサポートしているwasm環境向けターゲットの
wasm32-unknown-unknownではファイルを読み込めないから」です。しかしこれは意図的な物であると考えられます。というのもtypstの関数は純粋でなければならないからです。これによりドキュメントの再現性が確保され、高速な差分コンパイルが実現されています。ここでもしプラグインから外部の環境にアクセスできてしまうと全く純粋ではなくなってしまうわけですね。
なので、プラグイン内で状態を保持することもできません。例えば以下のようなコードを考えてみましょう。
このコードは、
countが呼び出される度にCOUNTERに1加算するコードなのですが、これをTypstから複数回呼び出しても結果は全て0になります。これがTypstのプラグインが純粋であるということです。最後に
いかがでしたか?とても簡単にTypstのプラグインを作成できることがお分かり頂けたかと思います。既存の言語のエコシステムを使って比較的容易に複雑なプラグインを開発することができることはTypstの大きな利点だと思います。
ちなみに、パッケージを作ったら
typst/packagesリポジトリにPRを送ることで公式のプラグインリストに載り、import "@preview/..."でインポートできるようになります。参考記事:
Typstの日本語Lipsumパッケージを作ってみた件
 zenn.dev
zenn.dev
 
また、今回使用したソースコードは
playground/other/typst-plugin at c0fb192f71e71fbbaafcc57673bdc4e931f3dd39 · nazo6/playground
Contribute to nazo6/playground development by creating an account on GitHub.
 github.com
github.com
で公開しています。
是非みなさんもTypstプラグインを作ってみてください。