machineuser
Sync widgets demo
94753b6
raw
history blame
1.88 kB
/**
* @file Jinja templating engine
*
* A minimalistic JavaScript reimplementation of the [Jinja](https://github.com/pallets/jinja) templating engine,
* to support the chat templates. Special thanks to [Tyler Laceby](https://github.com/tlaceby) for his amazing
* ["Guide to Interpreters"](https://github.com/tlaceby/guide-to-interpreters-series) tutorial series,
* which provided the basis for this implementation.
*
* See the [Transformers documentation](https://huggingface.co/docs/transformers/main/en/chat_templating) for more information.
*
* @module index
*/
import type { Program } from "./ast";
import { tokenize } from "./lexer";
import { parse } from "./parser";
import type { StringValue } from "./runtime";
import { Environment, Interpreter } from "./runtime";
export class Template {
parsed: Program;
/**
* @param {string} template The template string
*/
constructor(template: string) {
// Since `lstrip_blocks` is enabled, we strip tabs and spaces from the beginning of a line to the start of a block.
// We can achieve the same effect by replacing all instances of `%}\s+{%` with `%}{%` (as a pre-processing step).
template = template.replace(/%}\s+{%/g, "%}{%");
const tokens = tokenize(template);
this.parsed = parse(tokens);
}
render(items: Record<string, unknown>): string {
// Create a new environment for this template
const env = new Environment();
// Declare global variables
env.set("false", false);
env.set("true", true);
env.set("raise_exception", (args: string) => {
throw new Error(args);
});
// Add user-defined variables
for (const [key, value] of Object.entries(items)) {
env.set(key, value);
}
const interpreter = new Interpreter(env);
const result = interpreter.run(this.parsed) as StringValue;
return result.value;
}
}
export { Environment, Interpreter, tokenize, parse };