Claude-skill-registry flow-convert-handlebars-to-liquid
Convert Handlebars template syntax to Liquid.js in prompts. Use when migrating prompt templates from Flow SDK to Output SDK.
install
source · Clone the upstream repo
git clone https://github.com/majiayu000/claude-skill-registry
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/flow-convert-handlebars-to-liquid" ~/.claude/skills/majiayu000-claude-skill-registry-flow-convert-handlebars-to-liquid && rm -rf "$T"
manifest:
skills/data/flow-convert-handlebars-to-liquid/SKILL.mdsource content
Convert Handlebars to Liquid.js Templates
Overview
This skill guides the conversion of Handlebars template syntax (used in Flow SDK) to Liquid.js syntax (required by Output SDK). This is critical for prompt template migration.
When to Use This Skill
During Migration:
- Converting prompt templates from Flow SDK
- Fixing template syntax errors in migrated prompts
- Setting up new prompt files with correct syntax
Error Symptoms:
- Template conditionals not rendering
- Variables not being interpolated
- Syntax errors in prompt files
Syntax Conversion Reference
Variable Interpolation
| Handlebars | Liquid.js |
|---|---|
| |
| |
| |
Key Rule: Always include spaces inside braces:
{{ variable }} not {{variable}}
Conditionals
| Handlebars | Liquid.js |
|---|---|
| |
| |
| |
| |
| |
Comparison Operators
| Handlebars | Liquid.js |
|---|---|
| |
| |
| |
| |
| |
| |
Loops
| Handlebars | Liquid.js |
|---|---|
| |
| |
| |
| |
| |
| |
Default Values
| Handlebars | Liquid.js |
|---|---|
(with default helper) | |
Comments
| Handlebars | Liquid.js |
|---|---|
| |
| |
Common Conversion Examples
Example 1: Simple Variable
<!-- Handlebars --> Hello, {{userName}}! Your order {{orderId}} is ready.
<!-- Liquid.js --> Hello, {{ userName }}! Your order {{ orderId }} is ready.
Example 2: Conditional
<!-- Handlebars --> {{#if includeDetails}} Here are the details: {{details}} {{else}} No details available. {{/if}}
<!-- Liquid.js --> {% if includeDetails %} Here are the details: {{ details }} {% else %} No details available. {% endif %}
Example 3: Nested Conditionals
<!-- Handlebars --> {{#if isPremium}} Premium content: {{#if hasAccess}} {{premiumContent}} {{else}} Please upgrade to access. {{/if}} {{else}} Basic content: {{basicContent}} {{/if}}
<!-- Liquid.js --> {% if isPremium %} Premium content: {% if hasAccess %} {{ premiumContent }} {% else %} Please upgrade to access. {% endif %} {% else %} Basic content: {{ basicContent }} {% endif %}
Example 4: Loop
<!-- Handlebars --> Items to process: {{#each items}} - {{this.name}}: {{this.value}} {{/each}}
<!-- Liquid.js --> Items to process: {% for item in items %} - {{ item.name }}: {{ item.value }} {% endfor %}
Example 5: Loop with Index
<!-- Handlebars --> {{#each steps}} Step {{@index}}: {{this.description}} {{/each}}
<!-- Liquid.js --> {% for step in steps %} Step {{ forloop.index }}: {{ step.description }} {% endfor %}
Example 6: Comparison
<!-- Handlebars --> {{#if (eq status "active")}} Active user {{else if (eq status "pending")}} Pending approval {{else}} Inactive {{/if}}
<!-- Liquid.js --> {% if status == "active" %} Active user {% elsif status == "pending" %} Pending approval {% else %} Inactive {% endif %}
Example 7: Boolean Handling
Important: Booleans in Liquid.js templates can be tricky. Convert to strings for reliable comparisons.
// In step code, convert boolean to string variables: { hasData: data ? 'yes' : 'no', isEnabled: enabled ? 'true' : 'false' }
<!-- In prompt --> {% if hasData == 'yes' %} Data is available: {{ data }} {% endif %} {% if isEnabled == 'true' %} Feature is enabled. {% endif %}
Example 8: Default Values
<!-- Handlebars (with helper) --> Language: {{language}}
<!-- Liquid.js --> Language: {{ language | default: "English" }}
Complete Prompt Migration Example
Before: Handlebars Prompt
// prompts.ts export const analyzePrompt = ` You are an AI assistant. {{#if systemContext}} Context: {{systemContext}} {{/if}} User Query: {{query}} {{#if examples}} Examples: {{#each examples}} {{@index}}. Input: {{this.input}} Output: {{this.output}} {{/each}} {{/if}} {{#if (eq mode "detailed")}} Provide a detailed analysis with explanations. {{else if (eq mode "brief")}} Provide a brief summary. {{else}} Provide a standard response. {{/if}} `;
After: Liquid.js Prompt File
--- provider: openai model: gpt-4o temperature: 0.5 --- <system> You are an AI assistant. {% if systemContext %} Context: {{ systemContext }} {% endif %} </system> <user> User Query: {{ query }} {% if examples %} Examples: {% for example in examples %} {{ forloop.index }}. Input: {{ example.input }} Output: {{ example.output }} {% endfor %} {% endif %} {% if mode == "detailed" %} Provide a detailed analysis with explanations. {% elsif mode == "brief" %} Provide a brief summary. {% else %} Provide a standard response. {% endif %} </user>
Finding Handlebars Syntax
Search for patterns that need conversion:
# Find Handlebars conditionals grep -r "{{#if" src/workflows/ grep -r "{{/if}}" src/workflows/ grep -r "{{#each" src/workflows/ grep -r "{{#unless" src/workflows/ # Find variables without spaces grep -r "{{[^#/!]" src/workflows/ | grep -v "{{ "
Verification Checklist
- All
converted to{{#if ...}}{% if ... %} - All
converted to{{/if}}{% endif %} - All
converted to{{#each ...}}{% for ... in ... %} - All
converted to{{/each}}{% endfor %} - All
converted to{{else}}{% else %} - All variables have spaces:
not{{ var }}{{var}} - Boolean variables converted to strings for comparison
- Comparison operators use
,==
,!=
,>< - Loop variables use
instead offorloop.index@index
Related Skills
- Full prompt conversionflow-convert-prompts-to-files
- Identifying prompts to convertflow-analyze-prompts
- Migration validationflow-validation-checklist