Type-preserving serialization for Typst values. Converts Typst objects into an intermediate representation. This can be used for communication with WASM plugins.
What is this?
Unlike repr() or cbor.encode(), which produce ambiguous strings or lose type
information, sertyp:
- Enables roundtrips: Deserialize back to the original displayable value, not just a string representation. This means a content object will again be fully displayed as content.
- Works with WASM plugins: The intermediate representation can be further serialized to cbor and passed over the WASM boundary.
The Rust backend provides deserialization logic and typed data structures so plugins can work with actual Typst types instead of manual parsing efforts.
Usage
#import "@preview/sertyp:0.1.2"
Examples
Basic serialization
#import "@preview/sertyp:0.1.2"
// Serialize and deserialize complex content
#let value = [
Total displaced soil by glacial flow:
$ 7.32 beta + sum_(i=0)^nabla (Q_i (a_i - epsilon)) / 2 $
#metadata(title: "Glacial Flow Calculation")
]
#let serialized = sertyp.serialize(value)
#let deserialized = sertyp.deserialize(serialized)
#assert(repr(deserialized) == repr(value))
WASM Plugins using (rust sertyp crate)
use wasm_minimal_protocol::*;
use sertyp::{typst_func, Integer, String};
#[cfg(target_arch = "wasm32")]
initiate_protocol!();
// Result errors are automatically converted to typst panics.
#[typst_func]
pub fn fibonacci<'a>(n: Integer) -> Result<Integer, String<'a>> {
let n: i32 = n.try_into().map_err(|_| "Invalid integer range")?;
let (mut v0, mut v1) = (0, 1);
for _ in 0..n {
(v0, v1) = (v1, v0 + v1);
}
Ok(v1.into())
}
Type preservation examples
#import "@preview/sertyp:0.1.2"
#let color = rgb(255, 128, 0)
#let restored = sertyp.deserialize(sertyp.serialize(color))
// restored value is a real color object value, not a string
#let len = 2.5em + 10pt
#let restored = sertyp.deserialize(sertyp.serialize(len))
// Same with lengths and most other types
Overview
serialize(value) -> any
Converts a Typst value into an intermediate representation (nested dicts/arrays with type tags).
let serialized = sertyp.serialize(rgb(255, 0, 0))
// Returns: (type: "color", value: (components: ..., space: ...))
deserialize(serialized) -> any
Reconstructs a Typst value from its intermediate representation.
let value = sertyp.deserialize(serialized)
// Returns the original displayable value
Security note: Deserialization uses eval() internally. Deserializing
untrusted values may therefore lead to arbitrary code execution. Only
deserialize trusted data.
serialize-cbor(value) -> bytes
Serializes to CBOR binary format. Usefull when passing values to WASM plugins.
let bytes = sertyp.serialize-cbor(my_value)
// Returns CBOR-encoded bytes
deserialize-cbor(bytes) -> any
Deserializes from CBOR bytes back to a Typst value.
let value = sertyp.deserialize-cbor(plugin_output)
Security note: Deserialization uses eval() internally. Deserializing
untrusted values may therefore lead to arbitrary code execution. Only
deserialize trusted data.
call(function, arg) -> any
Shorthand for
sertyp.deserialize-cbor(function(sertyp.serialize-cbor(arg)))
Supported Types
Primitives: bool, int, float, decimal, string, bytes, none,
auto
Collections: array, dictionary
Numeric with units: length, angle, ratio, fraction, duration,
relative
Text & content: content, label, regex, symbol, version, datetime
Visual: color, stroke, gradient, alignment, direction, tiling
Advanced: function, module, selector, type, arguments, styles
Known Limitations
selector: Requires custom parsing for proper serialization (partially supported)- Dynamic types:
contextand other runtime-dependent elements cannot be fully serialized - Closures: Inline functions
(..) => ..serialize but lose their captured state
Plugin Development
See the Rust README for details on building WASM plugins that work with sertyp.