A Yarn parser written in Lua to convert Yarn Spinner dialogues into Lua structures. There is also a interpreter to demonstrate how to interpret the parsed node structures.
This Lua module provides a parser for Yarn scripts, which are commonly used in interactive narrative games. The parser can handle various elements of Yarn syntax, including dialogue, choices, conditional statements, variable assignments, and commands.
The supplied interpreter is only an example of how the parsed structure can be interpreted. You must adapt it or implement your own interpreter that meets your requirements.
- Parse Yarn scripts into structured node objects
- Support for:
- Dialogue lines
- Choices (including nested choices)
- Conditional statements (if/else)
- Variable assignments, declarations, and interpolation
- Commands (including jump, set, declare)
- Comments (single-line and multi-line)
- Grouping of related content (e.g., choices and their responses)
- Ability to find dialogue preceding choice groups
- Sample interpreter with callbacks to run the dialogues.
main.lua
local YarnParser = require("yarn_parser")
local script = [[
title: Start
---
Player: Hello, world!
<<declare $goldAmount = 100>>
<<set $health to 100>>
Your health is {$health}.
-> Choice 1: gold
NPC: You chose {$goldAmount} gold.
<<set $health to 50>>
-> Choice 2: health
NPC: Your health is {$health}.
NPC: You have {$health} health.
<<jump TEST>>
Jump does not work.
===
title: TEST
---
Jump: Test.
===
]]
-- Function to print the content of a node
local function print_content(content, indent)
indent = indent or ""
for _, item in ipairs(content) do
if item.type == "dialogue" then
print(indent .. item.text)
elseif item.type == "choice" then
print(indent .. " -> " .. item.text)
print_content(item.response, indent .. " ")
elseif item.type == "set" then
print(indent .. "Set: " .. item.variable .. " to " .. item.value)
elseif item.type == "conditional" then
print(indent .. "If: " .. item.condition)
print(indent .. "Then:")
print_content(item.if_block, indent .. " ")
if #item.else_block > 0 then
print(indent .. "Else:")
print_content(item.else_block, indent .. " ")
end
elseif item.type == "jump" then
print(indent .. "Jump to: " .. item.target)
elseif item.type == "declare" then
print(indent .. "Declare: " .. item.variable .. " = " .. item.value)
elseif item.type == "comment" then
print(indent .. "Comment: " .. item.text)
else
print(indent .. "Unknown type: " .. tostring(item.type))
end
end
end
local parsed_nodes = YarnParser:parse(script)
-- Print the parsed structure
if parsed_nodes then
for i, node in ipairs(parsed_nodes) do
print("Node: " .. node.title)
print_content(node.content, " ")
print("") -- Empty line between nodes for readability
end
end
Parses a Yarn script and returns an array of node objects.
script
: A string containing the entire Yarn script.- Returns: An array of parsed node objects.
Each parsed node has the following structure (exported json table):
[
{
"title": "Start",
"content": [
{
"text": "Player: Hello, world!",
"indent": 0,
"type": "dialogue"
},
{
"variable": "goldAmount",
"value": "100",
"indent": 0,
"type": "declare"
},
{
"variable": "health",
"value": "100",
"indent": 0,
"type": "set"
},
{
"text": "Your health is {$health}.",
"indent": 0,
"type": "dialogue"
},
{
"response": [
{
"text": "NPC: You chose {$goldAmount} gold.",
"indent": 4,
"type": "dialogue"
},
{
"variable": "health",
"value": "50",
"indent": 4,
"type": "set"
}
],
"text": "Choice 1: gold",
"indent": 0,
"type": "choice"
},
{
"response": [
{
"text": "NPC: Your health is {$health}.",
"indent": 4,
"type": "dialogue"
}
],
"text": "Choice 2: health",
"indent": 0,
"type": "choice"
},
{
"text": "NPC: You have {$health} health.",
"indent": 0,
"type": "dialogue"
},
{
"target": "TEST",
"indent": 0,
"type": "jump"
},
{
"text": "Jump does not work.",
"indent": 0,
"type": "dialogue"
}
]
},
{
"title": "TEST",
"content": [
{
"text": "Jump: Test.",
"indent": 0,
"type": "dialogue"
}
]
}
]
Content objects can be of various types, including "dialogue", "choice", "conditional", "set", "declare", "jump", and "comment".
For a detailed description of the Yarn syntax, please refer to Yarn syntax description.
The included interpreter demonstrates how to run parsed Yarn scripts with callback support for various events.
local YarnParser = require("yarn_parser")
local YarnInterpreter = require("yarn_interpreter")
-- Parse your Yarn script
local parsed_nodes = YarnParser:parse(script)
-- Create interpreter instance
local interpreter = YarnInterpreter.new(parsed_nodes)
-- OPTIONAL: Set up callbacks
interpreter:set_callbacks({
on_dialogue = function(text)
-- Custom dialogue display
print("[DIALOGUE] " .. text)
end,
on_choice = function(choices)
-- Custom choice handling
print("[CHOICE]")
for i, choice in ipairs(choices) do
print(i .. ": " .. choice)
end
return tonumber(io.read())
end,
on_variable = function(name, value)
-- Variable change notification
print("[VARIABLE] " .. name .. " = " .. tostring(value))
end,
on_node_enter = function(title)
print("[ENTER NODE] " .. title)
end,
on_node_exit = function(title)
print("[EXIT NODE] " .. title)
end
})
-- Run the interpreter
interpreter:run()
on_dialogue(text)
: Called when dialogue text is encounteredon_choice(choices, path)
: Called when choices are presented. Must return the selected choice indexon_variable(name, value)
: Called when a variable is set or declaredon_node_enter(title)
: Called when entering a new nodeon_node_exit(title)
: Called when exiting a node
The interpreter maintains its own variable state, but you can interact with it:
-- Get a variable value
local value = interpreter:get_variable("health")
-- Set a variable value
interpreter:set_variable("health", 100)
The on_choice
callback receives:
- An array of choice texts with variables already interpolated
- A path string showing the hierarchy of nested choices (e.g., "Choice 1 > Nested Choice 2")
The callback must return a number indicating the selected choice (1-based index).
The interpreter will emit warnings (via print) when:
- Invalid choices are selected
- Jump targets are not found
- Variables are undefined
- Conditions cannot be evaluated
- The parser treats choices at level 2 and above as being on the same level. In the structure, levels 2 and 3 are placed together. However, using indentation, you can still assign them different levels in your interpreter.
Test Sample:
-> Choice 2: health
NPC: Your health is {$health}.
-> Level 2 Test
Level 2 works?
-> Level 3 Test
Level 3 works?
- The parser assumes well-formed Yarn syntax. Malformed scripts may lead to unexpected results.
- Complex nested structures (e.g., conditionals within choices within conditionals) may not be handled perfectly and might require additional processing.
Florian Fischer ( https://github.com/SiENcE )
This project is licensed under the MIT License - see the LICENSE file for details.
Contributions are welcome! Please feel free to submit a Pull Request.