mirror of
https://github.com/typst/typst
synced 2025-05-13 20:46:23 +08:00
423 lines
14 KiB
Rust
423 lines
14 KiB
Rust
use std::fmt::{self, Debug, Formatter};
|
|
use std::hash::{Hash, Hasher};
|
|
use std::sync::Arc;
|
|
|
|
use comemo::{Track, Tracked};
|
|
|
|
use super::{
|
|
Args, Eval, Flow, Node, NodeId, Route, Scope, Scopes, Selector, StyleMap, Value, Vm,
|
|
};
|
|
use crate::diag::{bail, SourceResult, StrResult};
|
|
use crate::syntax::ast::{self, Expr, TypedNode};
|
|
use crate::syntax::{SourceId, Span, SyntaxNode};
|
|
use crate::util::EcoString;
|
|
use crate::World;
|
|
|
|
/// An evaluatable function.
|
|
#[derive(Clone, Hash)]
|
|
pub struct Func(Arc<Repr>);
|
|
|
|
/// The different kinds of function representations.
|
|
#[derive(Hash)]
|
|
enum Repr {
|
|
/// A native rust function.
|
|
Native(Native),
|
|
/// A user-defined closure.
|
|
Closure(Closure),
|
|
/// A nested function with pre-applied arguments.
|
|
With(Func, Args),
|
|
}
|
|
|
|
impl Func {
|
|
/// Create a new function from a native rust function.
|
|
pub fn from_fn(
|
|
name: &'static str,
|
|
func: fn(&mut Vm, &mut Args) -> SourceResult<Value>,
|
|
) -> Self {
|
|
Self(Arc::new(Repr::Native(Native { name, func, set: None, node: None })))
|
|
}
|
|
|
|
/// Create a new function from a native rust node.
|
|
pub fn from_node<T: Node>(name: &'static str) -> Self {
|
|
Self(Arc::new(Repr::Native(Native {
|
|
name,
|
|
func: |ctx, args| {
|
|
let styles = T::set(args, true)?;
|
|
let content = T::construct(ctx, args)?;
|
|
Ok(Value::Content(content.styled_with_map(styles.scoped())))
|
|
},
|
|
set: Some(|args| T::set(args, false)),
|
|
node: Some(NodeId::of::<T>()),
|
|
})))
|
|
}
|
|
|
|
/// Create a new function from a closure.
|
|
pub fn from_closure(closure: Closure) -> Self {
|
|
Self(Arc::new(Repr::Closure(closure)))
|
|
}
|
|
|
|
/// The name of the function.
|
|
pub fn name(&self) -> Option<&str> {
|
|
match self.0.as_ref() {
|
|
Repr::Native(native) => Some(native.name),
|
|
Repr::Closure(closure) => closure.name.as_deref(),
|
|
Repr::With(func, _) => func.name(),
|
|
}
|
|
}
|
|
|
|
/// The number of positional arguments this function takes, if known.
|
|
pub fn argc(&self) -> Option<usize> {
|
|
match self.0.as_ref() {
|
|
Repr::Closure(closure) => closure.argc(),
|
|
Repr::With(wrapped, applied) => Some(wrapped.argc()?.saturating_sub(
|
|
applied.items.iter().filter(|arg| arg.name.is_none()).count(),
|
|
)),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
/// Call the function with the given arguments.
|
|
pub fn call(&self, vm: &mut Vm, mut args: Args) -> SourceResult<Value> {
|
|
let value = match self.0.as_ref() {
|
|
Repr::Native(native) => (native.func)(vm, &mut args)?,
|
|
Repr::Closure(closure) => closure.call(vm, &mut args)?,
|
|
Repr::With(wrapped, applied) => {
|
|
args.items.splice(..0, applied.items.iter().cloned());
|
|
return wrapped.call(vm, args);
|
|
}
|
|
};
|
|
args.finish()?;
|
|
Ok(value)
|
|
}
|
|
|
|
/// Call the function without an existing virtual machine.
|
|
pub fn call_detached(
|
|
&self,
|
|
world: Tracked<dyn World>,
|
|
args: Args,
|
|
) -> SourceResult<Value> {
|
|
let route = Route::default();
|
|
let mut vm = Vm::new(world, route.track(), None, Scopes::new(None));
|
|
self.call(&mut vm, args)
|
|
}
|
|
|
|
/// Apply the given arguments to the function.
|
|
pub fn with(self, args: Args) -> Self {
|
|
Self(Arc::new(Repr::With(self, args)))
|
|
}
|
|
|
|
/// Execute the function's set rule and return the resulting style map.
|
|
pub fn set(&self, mut args: Args, span: Span) -> SourceResult<StyleMap> {
|
|
let Repr::Native(Native { set: Some(set), .. }) = self.0.as_ref() else {
|
|
bail!(span, "this function cannot be customized with set");
|
|
};
|
|
|
|
let styles = set(&mut args)?;
|
|
args.finish()?;
|
|
Ok(styles)
|
|
}
|
|
|
|
/// The id of the node to customize with this function's show rule.
|
|
pub fn node(&self) -> StrResult<NodeId> {
|
|
match self.0.as_ref() {
|
|
Repr::Native(Native { node: Some(id), .. }) => Ok(*id),
|
|
_ => Err("this function cannot be customized with show")?,
|
|
}
|
|
}
|
|
|
|
/// Create a selector from this node and the given arguments.
|
|
pub fn where_(self, args: &mut Args) -> StrResult<Selector> {
|
|
match self.0.as_ref() {
|
|
Repr::Native(Native { node: Some(id), .. }) => {
|
|
let named = args.to_named();
|
|
args.items.retain(|arg| arg.name.is_none());
|
|
Ok(Selector::Node(*id, Some(named)))
|
|
}
|
|
_ => Err("this function is not selectable")?,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Debug for Func {
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
|
match self.name() {
|
|
Some(name) => f.write_str(name),
|
|
None => f.write_str("(..) => {..}"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PartialEq for Func {
|
|
fn eq(&self, other: &Self) -> bool {
|
|
Arc::ptr_eq(&self.0, &other.0)
|
|
}
|
|
}
|
|
|
|
/// A function defined by a native rust function or node.
|
|
struct Native {
|
|
/// The name of the function.
|
|
pub name: &'static str,
|
|
/// The function pointer.
|
|
pub func: fn(&mut Vm, &mut Args) -> SourceResult<Value>,
|
|
/// The set rule.
|
|
pub set: Option<fn(&mut Args) -> SourceResult<StyleMap>>,
|
|
/// The id of the node to customize with this function's show rule.
|
|
pub node: Option<NodeId>,
|
|
}
|
|
|
|
impl Hash for Native {
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
self.name.hash(state);
|
|
(self.func as usize).hash(state);
|
|
self.set.map(|set| set as usize).hash(state);
|
|
self.node.hash(state);
|
|
}
|
|
}
|
|
|
|
/// A user-defined closure.
|
|
#[derive(Hash)]
|
|
pub struct Closure {
|
|
/// The source file where the closure was defined.
|
|
pub location: Option<SourceId>,
|
|
/// The name of the closure.
|
|
pub name: Option<EcoString>,
|
|
/// Captured values from outer scopes.
|
|
pub captured: Scope,
|
|
/// The parameter names and default values. Parameters with default value
|
|
/// are named parameters.
|
|
pub params: Vec<(EcoString, Option<Value>)>,
|
|
/// The name of an argument sink where remaining arguments are placed.
|
|
pub sink: Option<EcoString>,
|
|
/// The expression the closure should evaluate to.
|
|
pub body: Expr,
|
|
}
|
|
|
|
impl Closure {
|
|
/// Call the function in the context with the arguments.
|
|
pub fn call(&self, vm: &mut Vm, args: &mut Args) -> SourceResult<Value> {
|
|
// Don't leak the scopes from the call site. Instead, we use the scope
|
|
// of captured variables we collected earlier.
|
|
let mut scopes = Scopes::new(None);
|
|
scopes.top = self.captured.clone();
|
|
|
|
// Parse the arguments according to the parameter list.
|
|
for (param, default) in &self.params {
|
|
scopes.top.define(
|
|
param.clone(),
|
|
match default {
|
|
Some(default) => {
|
|
args.named::<Value>(param)?.unwrap_or_else(|| default.clone())
|
|
}
|
|
None => args.expect::<Value>(param)?,
|
|
},
|
|
);
|
|
}
|
|
|
|
// Put the remaining arguments into the sink.
|
|
if let Some(sink) = &self.sink {
|
|
scopes.top.define(sink.clone(), args.take());
|
|
}
|
|
|
|
// Determine the route inside the closure.
|
|
let detached = vm.location.is_none();
|
|
let fresh = Route::new(self.location);
|
|
let route = if detached { fresh.track() } else { vm.route };
|
|
|
|
// Evaluate the body.
|
|
let mut sub = Vm::new(vm.world, route, self.location, scopes);
|
|
let result = self.body.eval(&mut sub);
|
|
|
|
// Handle control flow.
|
|
match sub.flow {
|
|
Some(Flow::Return(_, Some(explicit))) => return Ok(explicit),
|
|
Some(Flow::Return(_, None)) => {}
|
|
Some(flow) => bail!(flow.forbidden()),
|
|
None => {}
|
|
}
|
|
|
|
result
|
|
}
|
|
|
|
/// The number of positional arguments this function takes, if known.
|
|
pub fn argc(&self) -> Option<usize> {
|
|
if self.sink.is_some() {
|
|
return None;
|
|
}
|
|
|
|
Some(self.params.iter().filter(|(_, default)| default.is_none()).count())
|
|
}
|
|
}
|
|
|
|
/// A visitor that determines which variables to capture for a closure.
|
|
pub(super) struct CapturesVisitor<'a> {
|
|
external: &'a Scopes<'a>,
|
|
internal: Scopes<'a>,
|
|
captures: Scope,
|
|
}
|
|
|
|
impl<'a> CapturesVisitor<'a> {
|
|
/// Create a new visitor for the given external scopes.
|
|
pub fn new(external: &'a Scopes) -> Self {
|
|
Self {
|
|
external,
|
|
internal: Scopes::new(None),
|
|
captures: Scope::new(),
|
|
}
|
|
}
|
|
|
|
/// Return the scope of captured variables.
|
|
pub fn finish(self) -> Scope {
|
|
self.captures
|
|
}
|
|
|
|
/// Bind a new internal variable.
|
|
pub fn bind(&mut self, ident: ast::Ident) {
|
|
self.internal.top.define(ident.take(), Value::None);
|
|
}
|
|
|
|
/// Capture a variable if it isn't internal.
|
|
pub fn capture(&mut self, ident: ast::Ident) {
|
|
if self.internal.get(&ident).is_err() {
|
|
if let Ok(value) = self.external.get(&ident) {
|
|
self.captures.define_captured(ident.take(), value.clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Visit any node and collect all captured variables.
|
|
pub fn visit(&mut self, node: &SyntaxNode) {
|
|
match node.cast() {
|
|
// Every identifier is a potential variable that we need to capture.
|
|
// Identifiers that shouldn't count as captures because they
|
|
// actually bind a new name are handled below (individually through
|
|
// the expressions that contain them).
|
|
Some(ast::Expr::Ident(ident)) => self.capture(ident),
|
|
|
|
// Code and content blocks create a scope.
|
|
Some(ast::Expr::Code(_) | ast::Expr::Content(_)) => {
|
|
self.internal.enter();
|
|
for child in node.children() {
|
|
self.visit(child);
|
|
}
|
|
self.internal.exit();
|
|
}
|
|
|
|
// A closure contains parameter bindings, which are bound before the
|
|
// body is evaluated. Care must be taken so that the default values
|
|
// of named parameters cannot access previous parameter bindings.
|
|
Some(ast::Expr::Closure(expr)) => {
|
|
for param in expr.params() {
|
|
if let ast::Param::Named(named) = param {
|
|
self.visit(named.expr().as_untyped());
|
|
}
|
|
}
|
|
|
|
for param in expr.params() {
|
|
match param {
|
|
ast::Param::Pos(ident) => self.bind(ident),
|
|
ast::Param::Named(named) => self.bind(named.name()),
|
|
ast::Param::Sink(ident) => self.bind(ident),
|
|
}
|
|
}
|
|
|
|
self.visit(expr.body().as_untyped());
|
|
}
|
|
|
|
// A let expression contains a binding, but that binding is only
|
|
// active after the body is evaluated.
|
|
Some(ast::Expr::Let(expr)) => {
|
|
if let Some(init) = expr.init() {
|
|
self.visit(init.as_untyped());
|
|
}
|
|
self.bind(expr.binding());
|
|
}
|
|
|
|
// A for loop contains one or two bindings in its pattern. These are
|
|
// active after the iterable is evaluated but before the body is
|
|
// evaluated.
|
|
Some(ast::Expr::For(expr)) => {
|
|
self.visit(expr.iter().as_untyped());
|
|
let pattern = expr.pattern();
|
|
if let Some(key) = pattern.key() {
|
|
self.bind(key);
|
|
}
|
|
self.bind(pattern.value());
|
|
self.visit(expr.body().as_untyped());
|
|
}
|
|
|
|
// An import contains items, but these are active only after the
|
|
// path is evaluated.
|
|
Some(ast::Expr::Import(expr)) => {
|
|
self.visit(expr.path().as_untyped());
|
|
if let ast::Imports::Items(items) = expr.imports() {
|
|
for item in items {
|
|
self.bind(item);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Everything else is traversed from left to right.
|
|
_ => {
|
|
for child in node.children() {
|
|
self.visit(child);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use crate::syntax::parse;
|
|
|
|
#[track_caller]
|
|
fn test(text: &str, result: &[&str]) {
|
|
let mut scopes = Scopes::new(None);
|
|
scopes.top.define("x", 0);
|
|
scopes.top.define("y", 0);
|
|
scopes.top.define("z", 0);
|
|
|
|
let mut visitor = CapturesVisitor::new(&scopes);
|
|
let root = parse(text);
|
|
visitor.visit(&root);
|
|
|
|
let captures = visitor.finish();
|
|
let mut names: Vec<_> = captures.iter().map(|(k, _)| k).collect();
|
|
names.sort();
|
|
|
|
assert_eq!(names, result);
|
|
}
|
|
|
|
#[test]
|
|
fn test_captures() {
|
|
// Let binding and function definition.
|
|
test("#let x = x", &["x"]);
|
|
test("#let x; {x + y}", &["y"]);
|
|
test("#let f(x, y) = x + y", &[]);
|
|
|
|
// Closure with different kinds of params.
|
|
test("{(x, y) => x + z}", &["z"]);
|
|
test("{(x: y, z) => x + z}", &["y"]);
|
|
test("{(..x) => x + y}", &["y"]);
|
|
test("{(x, y: x + z) => x + y}", &["x", "z"]);
|
|
|
|
// Show rule.
|
|
test("#show y: x => x", &["y"]);
|
|
test("#show y: x => x + z", &["y", "z"]);
|
|
test("#show x: x => x", &["x"]);
|
|
|
|
// For loop.
|
|
test("#for x in y { x + z }", &["y", "z"]);
|
|
test("#for x, y in y { x + y }", &["y"]);
|
|
|
|
// Import.
|
|
test("#import x, y from z", &["z"]);
|
|
test("#import x, y, z from x + y", &["x", "y"]);
|
|
|
|
// Blocks.
|
|
test("{ let x = 1; { let y = 2; y }; x + y }", &["y"]);
|
|
test("[#let x = 1]#x", &["x"]);
|
|
}
|
|
}
|