1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
|
- import nunjucks from 'nunjucks';
- import {
- extractVariablesFromTemplate,
- extractVariablesFromTemplates,
- getNunjucksEngine,
- } from '../src/util/templates';
- describe('extractVariablesFromTemplate', () => {
- it('should extract simple variables', () => {
- const template = 'Hello {{ name }}, welcome to {{ place }}!';
- expect(extractVariablesFromTemplate(template)).toEqual(['name', 'place']);
- });
- it('should extract variables without spaces', () => {
- const template = 'Hello {{name}}, welcome to {{place}}!';
- expect(extractVariablesFromTemplate(template)).toEqual(['name', 'place']);
- });
- it('should extract variables with dot notation', () => {
- const template = 'Hello {{ user.name }}, your score is {{ game.score }}!';
- expect(extractVariablesFromTemplate(template)).toEqual(['user.name', 'game.score']);
- });
- it('should extract variables with underscores', () => {
- const template = 'Your order {{ order_id }} will arrive on {{ delivery_date }}.';
- expect(extractVariablesFromTemplate(template)).toEqual(['order_id', 'delivery_date']);
- });
- it('should extract variables with numbers', () => {
- const template = 'Player1: {{ player1 }}, Player2: {{ player2 }}';
- expect(extractVariablesFromTemplate(template)).toEqual(['player1', 'player2']);
- });
- it('should extract variables used in filters', () => {
- const template = '{{ name | capitalize }} - {{ date | date("yyyy-MM-dd") }}';
- expect(extractVariablesFromTemplate(template)).toEqual(['name', 'date']);
- });
- it('should extract variables used in complex expressions', () => {
- const template = '{% if user.age > 18 %}Welcome, {{ user.name }}!{% endif %}';
- expect(extractVariablesFromTemplate(template)).toEqual(['user.age', 'user.name']);
- });
- it('should extract variables from for loops', () => {
- const template = '{% for item in items %}{{ item.name }}{% endfor %}';
- expect(extractVariablesFromTemplate(template)).toEqual(['item.name', 'items']);
- });
- it('should extract variables with multiple occurrences', () => {
- const template = '{{ name }} {{ age }} {{ name }}';
- expect(extractVariablesFromTemplate(template)).toEqual(['name', 'age']);
- });
- it('should not extract variables from comments', () => {
- const template = '{# This is a comment with {{ variable }} #}{{ actual_variable }}';
- expect(extractVariablesFromTemplate(template)).toEqual(['actual_variable']);
- });
- it('should handle empty templates', () => {
- const template = '';
- expect(extractVariablesFromTemplate(template)).toEqual([]);
- });
- it('should handle templates without variables', () => {
- const template = 'This is a static template without variables.';
- expect(extractVariablesFromTemplate(template)).toEqual([]);
- });
- });
- describe('extractVariablesFromTemplates', () => {
- it('should extract variables from multiple templates', () => {
- const templates = [
- 'Hello {{ name }}, welcome to {{ place }}!',
- '{{ user.age }} years old',
- '{% for item in items %}{{ item.name }}{% endfor %}',
- ];
- const result = extractVariablesFromTemplates(templates);
- expect(result).toEqual(['name', 'place', 'user.age', 'item.name', 'items']);
- });
- it('should handle empty array of templates', () => {
- const templates: string[] = [];
- const result = extractVariablesFromTemplates(templates);
- expect(result).toEqual([]);
- });
- it('should deduplicate variables across templates', () => {
- const templates = ['Hello {{ name }}!', 'Welcome, {{ name }}!', '{{ age }} years old'];
- const result = extractVariablesFromTemplates(templates);
- expect(result).toEqual(['name', 'age']);
- });
- });
- describe('getNunjucksEngine', () => {
- const originalEnv = process.env;
- beforeEach(() => {
- jest.resetModules();
- process.env = { ...originalEnv };
- });
- afterEach(() => {
- process.env = originalEnv;
- });
- it('should return a nunjucks environment by default', () => {
- const engine = getNunjucksEngine();
- expect(engine).toBeInstanceOf(nunjucks.Environment);
- expect(engine.renderString('Hello {{ name }}', { name: 'World' })).toBe('Hello World');
- });
- it('should return a simple render function when PROMPTFOO_DISABLE_TEMPLATING is set', () => {
- process.env.PROMPTFOO_DISABLE_TEMPLATING = 'true';
- const engine = getNunjucksEngine();
- expect(engine.renderString('Hello {{ name }}', { name: 'World' })).toBe('Hello {{ name }}');
- });
- it('should return a nunjucks environment when isGrader is true, regardless of PROMPTFOO_DISABLE_TEMPLATING', () => {
- process.env.PROMPTFOO_DISABLE_TEMPLATING = 'true';
- const engine = getNunjucksEngine({}, false, true);
- expect(engine).toBeInstanceOf(nunjucks.Environment);
- expect(engine.renderString('Hello {{ name }}', { name: 'Grader' })).toBe('Hello Grader');
- });
- it('should use nunjucks when isGrader is true, even if PROMPTFOO_DISABLE_TEMPLATING is set', () => {
- process.env.PROMPTFOO_DISABLE_TEMPLATING = 'true';
- const engine = getNunjucksEngine({}, false, true);
- expect(engine).toBeInstanceOf(nunjucks.Environment);
- expect(engine.renderString('Hello {{ name }}', { name: 'Grader' })).toBe('Hello Grader');
- });
- it('should add custom filters when provided', () => {
- const customFilters = {
- uppercase: (str: string) => str.toUpperCase(),
- add: (a: number, b: number) => (a + b).toString(),
- };
- const engine = getNunjucksEngine(customFilters);
- expect(engine.renderString('{{ "hello" | uppercase }}', {})).toBe('HELLO');
- expect(engine.renderString('{{ 5 | add(3) }}', {})).toBe('8');
- });
- it('should add environment variables as globals under "env"', () => {
- process.env.TEST_VAR = 'test_value';
- const engine = getNunjucksEngine();
- expect(engine.renderString('{{ env.TEST_VAR }}', {})).toBe('test_value');
- });
- it('should throw an error when throwOnUndefined is true and a variable is undefined', () => {
- const engine = getNunjucksEngine({}, true);
- expect(() => {
- engine.renderString('{{ undefined_var }}', {});
- }).toThrow(/attempted to output null or undefined value/);
- });
- it('should not throw an error when throwOnUndefined is false and a variable is undefined', () => {
- const engine = getNunjucksEngine({}, false);
- expect(() => {
- engine.renderString('{{ undefined_var }}', {});
- }).not.toThrow();
- });
- it('should respect all parameters when provided', () => {
- process.env.PROMPTFOO_DISABLE_TEMPLATING = 'true';
- const customFilters = {
- double: (n: number) => (n * 2).toString(),
- };
- const engine = getNunjucksEngine(customFilters, true, true);
- expect(engine).toBeInstanceOf(nunjucks.Environment);
- expect(engine.renderString('{{ 5 | double }}', {})).toBe('10');
- expect(() => {
- engine.renderString('{{ undefined_var }}', {});
- }).toThrow(/attempted to output null or undefined value/);
- });
- });
|