@okeyamy/lua 5.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +552 -0
- package/build/es5/__tests__/ai-personalize.test.js +811 -0
- package/build/es5/__tests__/lua.js +134 -0
- package/build/es5/__tests__/original-roughly.js +197 -0
- package/build/es5/__tests__/original.js +174 -0
- package/build/es5/__tests__/unit.js +72 -0
- package/build/es5/__tests__/weighted-history.test.js +376 -0
- package/build/es5/ai-personalize.js +641 -0
- package/build/es5/index.js +30 -0
- package/build/es5/lua.js +366 -0
- package/build/es5/personalization.js +811 -0
- package/build/es5/prompts/personalization-prompts.js +260 -0
- package/build/es5/storage/weighted-history.js +384 -0
- package/build/es5/stores/browser-cookie.js +25 -0
- package/build/es5/stores/local.js +29 -0
- package/build/es5/stores/memory.js +22 -0
- package/build/es5/utils.js +54 -0
- package/build/es5/utm-personalize.js +817 -0
- package/build/es5/utm.js +304 -0
- package/build/lua.dev.js +1574 -0
- package/build/lua.es.js +1566 -0
- package/build/lua.js +1574 -0
- package/build/lua.min.js +8 -0
- package/package.json +68 -0
|
@@ -0,0 +1,811 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
|
|
5
|
+
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
|
|
6
|
+
var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
|
|
7
|
+
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
|
|
8
|
+
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
|
|
9
|
+
/**
|
|
10
|
+
* Tests for AI Personalization Engine
|
|
11
|
+
* Tests config validation, API communication, response validation, and caching
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
// Mock localStorage
|
|
15
|
+
var localStorageMock = function () {
|
|
16
|
+
var store = {};
|
|
17
|
+
return {
|
|
18
|
+
getItem: jest.fn(function (key) {
|
|
19
|
+
return store[key] || null;
|
|
20
|
+
}),
|
|
21
|
+
setItem: jest.fn(function (key, val) {
|
|
22
|
+
store[key] = String(val);
|
|
23
|
+
}),
|
|
24
|
+
removeItem: jest.fn(function (key) {
|
|
25
|
+
delete store[key];
|
|
26
|
+
}),
|
|
27
|
+
clear: jest.fn(function () {
|
|
28
|
+
store = {};
|
|
29
|
+
}),
|
|
30
|
+
get length() {
|
|
31
|
+
return Object.keys(store).length;
|
|
32
|
+
},
|
|
33
|
+
key: jest.fn(function (i) {
|
|
34
|
+
return Object.keys(store)[i] || null;
|
|
35
|
+
})
|
|
36
|
+
};
|
|
37
|
+
}();
|
|
38
|
+
Object.defineProperty(global, 'localStorage', {
|
|
39
|
+
value: localStorageMock
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
// Mock crypto
|
|
43
|
+
Object.defineProperty(global, 'crypto', {
|
|
44
|
+
value: {
|
|
45
|
+
getRandomValues: function getRandomValues(buf) {
|
|
46
|
+
for (var i = 0; i < buf.length; i++) buf[i] = Math.floor(Math.random() * 256);
|
|
47
|
+
return buf;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
// Mock fetch
|
|
53
|
+
global.fetch = jest.fn();
|
|
54
|
+
global.AbortController = /*#__PURE__*/function () {
|
|
55
|
+
function _class() {
|
|
56
|
+
(0, _classCallCheck2.default)(this, _class);
|
|
57
|
+
this.signal = {};
|
|
58
|
+
}
|
|
59
|
+
return (0, _createClass2.default)(_class, [{
|
|
60
|
+
key: "abort",
|
|
61
|
+
value: function abort() {}
|
|
62
|
+
}]);
|
|
63
|
+
}();
|
|
64
|
+
|
|
65
|
+
// Load modules in order (dependencies first)
|
|
66
|
+
require('../storage/weighted-history');
|
|
67
|
+
require('../prompts/personalization-prompts');
|
|
68
|
+
require('../ai-personalize');
|
|
69
|
+
var LuaAIPersonalize = global.LuaAIPersonalize;
|
|
70
|
+
var LuaPrompts = global.LuaPrompts;
|
|
71
|
+
var LuaWeightedHistory = global.LuaWeightedHistory;
|
|
72
|
+
var mockTemplates = {
|
|
73
|
+
'gaming': {
|
|
74
|
+
headline: 'Level Up Your Setup',
|
|
75
|
+
subheadline: 'Pro gear for serious gamers.',
|
|
76
|
+
ctaLabel: 'Explore Gaming',
|
|
77
|
+
ctaLink: '/gaming',
|
|
78
|
+
image: 'gaming.jpg'
|
|
79
|
+
},
|
|
80
|
+
'professional': {
|
|
81
|
+
headline: 'Work Smarter',
|
|
82
|
+
subheadline: 'Premium productivity tools.',
|
|
83
|
+
ctaLabel: 'View Collection',
|
|
84
|
+
ctaLink: '/professional',
|
|
85
|
+
image: 'pro.jpg'
|
|
86
|
+
},
|
|
87
|
+
'default': {
|
|
88
|
+
headline: 'Welcome',
|
|
89
|
+
subheadline: 'Discover products.',
|
|
90
|
+
ctaLabel: 'Shop Now',
|
|
91
|
+
ctaLink: '/shop',
|
|
92
|
+
image: 'default.jpg'
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
var mockContext = {
|
|
96
|
+
utm: {
|
|
97
|
+
utm_source: 'reddit',
|
|
98
|
+
utm_campaign: 'gaming_console'
|
|
99
|
+
},
|
|
100
|
+
referrer: {
|
|
101
|
+
source: 'reddit',
|
|
102
|
+
category: 'social',
|
|
103
|
+
url: 'https://reddit.com'
|
|
104
|
+
},
|
|
105
|
+
userAgent: {
|
|
106
|
+
isMobile: false,
|
|
107
|
+
isTablet: false,
|
|
108
|
+
isDesktop: true,
|
|
109
|
+
raw: ''
|
|
110
|
+
},
|
|
111
|
+
timestamp: Date.now(),
|
|
112
|
+
hasUTM: true,
|
|
113
|
+
primaryIntent: 'gaming'
|
|
114
|
+
};
|
|
115
|
+
beforeEach(function () {
|
|
116
|
+
localStorageMock.clear();
|
|
117
|
+
jest.clearAllMocks();
|
|
118
|
+
global.fetch.mockReset();
|
|
119
|
+
});
|
|
120
|
+
describe('LuaAIPersonalize', function () {
|
|
121
|
+
it('should be registered on global', function () {
|
|
122
|
+
expect(LuaAIPersonalize).toBeDefined();
|
|
123
|
+
expect((0, _typeof2.default)(LuaAIPersonalize.decide)).toBe('function');
|
|
124
|
+
expect((0, _typeof2.default)(LuaAIPersonalize.normalizeConfig)).toBe('function');
|
|
125
|
+
});
|
|
126
|
+
describe('normalizeConfig', function () {
|
|
127
|
+
it('should reject null config', function () {
|
|
128
|
+
var result = LuaAIPersonalize.normalizeConfig(null);
|
|
129
|
+
expect(result.valid).toBe(false);
|
|
130
|
+
});
|
|
131
|
+
it('should reject config without apiKey or apiUrl', function () {
|
|
132
|
+
var result = LuaAIPersonalize.normalizeConfig({});
|
|
133
|
+
expect(result.valid).toBe(false);
|
|
134
|
+
expect(result.error).toContain('apiKey');
|
|
135
|
+
});
|
|
136
|
+
it('should accept config with apiKey', function () {
|
|
137
|
+
var result = LuaAIPersonalize.normalizeConfig({
|
|
138
|
+
apiKey: 'sk-test123'
|
|
139
|
+
});
|
|
140
|
+
expect(result.valid).toBe(true);
|
|
141
|
+
expect(result.useDirectApi).toBe(true);
|
|
142
|
+
expect(result.apiKey).toBe('sk-test123');
|
|
143
|
+
});
|
|
144
|
+
it('should accept config with apiUrl', function () {
|
|
145
|
+
var result = LuaAIPersonalize.normalizeConfig({
|
|
146
|
+
apiUrl: 'https://proxy.example.com/api'
|
|
147
|
+
});
|
|
148
|
+
expect(result.valid).toBe(true);
|
|
149
|
+
expect(result.useDirectApi).toBe(false);
|
|
150
|
+
expect(result.apiUrl).toBe('https://proxy.example.com/api');
|
|
151
|
+
});
|
|
152
|
+
it('should apply defaults for missing fields', function () {
|
|
153
|
+
var result = LuaAIPersonalize.normalizeConfig({
|
|
154
|
+
apiKey: 'sk-test'
|
|
155
|
+
});
|
|
156
|
+
expect(result.model).toBe('gpt-4o-mini');
|
|
157
|
+
expect(result.mode).toBe('select');
|
|
158
|
+
expect(result.timeout).toBe(5000);
|
|
159
|
+
expect(result.maxRetries).toBe(1);
|
|
160
|
+
expect(result.fallbackToStandard).toBe(true);
|
|
161
|
+
expect(result.cacheDecisions).toBe(true);
|
|
162
|
+
expect(result.historyEnabled).toBe(true);
|
|
163
|
+
expect(result.historyDecayRate).toBe(0.9);
|
|
164
|
+
});
|
|
165
|
+
it('should use custom values when provided', function () {
|
|
166
|
+
var result = LuaAIPersonalize.normalizeConfig({
|
|
167
|
+
apiKey: 'sk-test',
|
|
168
|
+
model: 'gpt-4o',
|
|
169
|
+
mode: 'generate',
|
|
170
|
+
timeout: 10000,
|
|
171
|
+
temperature: 0.5,
|
|
172
|
+
maxTokens: 1000
|
|
173
|
+
});
|
|
174
|
+
expect(result.model).toBe('gpt-4o');
|
|
175
|
+
expect(result.mode).toBe('generate');
|
|
176
|
+
expect(result.timeout).toBe(10000);
|
|
177
|
+
expect(result.temperature).toBe(0.5);
|
|
178
|
+
expect(result.maxTokens).toBe(1000);
|
|
179
|
+
});
|
|
180
|
+
it('should prefer apiUrl over direct when both provided', function () {
|
|
181
|
+
var result = LuaAIPersonalize.normalizeConfig({
|
|
182
|
+
apiKey: 'sk-test',
|
|
183
|
+
apiUrl: 'https://proxy.com/api'
|
|
184
|
+
});
|
|
185
|
+
expect(result.useDirectApi).toBe(false);
|
|
186
|
+
expect(result.apiUrl).toBe('https://proxy.com/api');
|
|
187
|
+
});
|
|
188
|
+
});
|
|
189
|
+
describe('validateSelectResponse', function () {
|
|
190
|
+
it('should accept valid selection response', function () {
|
|
191
|
+
var result = LuaAIPersonalize.validateSelectResponse({
|
|
192
|
+
selectedVariant: 'gaming',
|
|
193
|
+
confidence: 0.9,
|
|
194
|
+
reasoning: 'test'
|
|
195
|
+
}, mockTemplates);
|
|
196
|
+
expect(result.valid).toBe(true);
|
|
197
|
+
});
|
|
198
|
+
it('should reject missing selectedVariant', function () {
|
|
199
|
+
var result = LuaAIPersonalize.validateSelectResponse({
|
|
200
|
+
confidence: 0.9
|
|
201
|
+
}, mockTemplates);
|
|
202
|
+
expect(result.valid).toBe(false);
|
|
203
|
+
});
|
|
204
|
+
it('should reject non-existent variant', function () {
|
|
205
|
+
var result = LuaAIPersonalize.validateSelectResponse({
|
|
206
|
+
selectedVariant: 'nonexistent'
|
|
207
|
+
}, mockTemplates);
|
|
208
|
+
expect(result.valid).toBe(false);
|
|
209
|
+
expect(result.error).toContain('nonexistent');
|
|
210
|
+
});
|
|
211
|
+
it('should reject null response', function () {
|
|
212
|
+
var result = LuaAIPersonalize.validateSelectResponse(null, mockTemplates);
|
|
213
|
+
expect(result.valid).toBe(false);
|
|
214
|
+
});
|
|
215
|
+
});
|
|
216
|
+
describe('validateGenerateResponse', function () {
|
|
217
|
+
it('should accept valid generation response', function () {
|
|
218
|
+
var result = LuaAIPersonalize.validateGenerateResponse({
|
|
219
|
+
headline: 'Test Headline',
|
|
220
|
+
subheadline: 'Test subheadline here',
|
|
221
|
+
ctaLabel: 'Click Me'
|
|
222
|
+
});
|
|
223
|
+
expect(result.valid).toBe(true);
|
|
224
|
+
});
|
|
225
|
+
it('should reject missing headline', function () {
|
|
226
|
+
var result = LuaAIPersonalize.validateGenerateResponse({
|
|
227
|
+
subheadline: 'Test',
|
|
228
|
+
ctaLabel: 'Click'
|
|
229
|
+
});
|
|
230
|
+
expect(result.valid).toBe(false);
|
|
231
|
+
expect(result.error).toContain('headline');
|
|
232
|
+
});
|
|
233
|
+
it('should reject missing subheadline', function () {
|
|
234
|
+
var result = LuaAIPersonalize.validateGenerateResponse({
|
|
235
|
+
headline: 'Test',
|
|
236
|
+
ctaLabel: 'Click'
|
|
237
|
+
});
|
|
238
|
+
expect(result.valid).toBe(false);
|
|
239
|
+
});
|
|
240
|
+
it('should reject missing ctaLabel', function () {
|
|
241
|
+
var result = LuaAIPersonalize.validateGenerateResponse({
|
|
242
|
+
headline: 'Test',
|
|
243
|
+
subheadline: 'Sub'
|
|
244
|
+
});
|
|
245
|
+
expect(result.valid).toBe(false);
|
|
246
|
+
});
|
|
247
|
+
});
|
|
248
|
+
describe('decide (select mode)', function () {
|
|
249
|
+
it('should call OpenAI and return a valid decision', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee() {
|
|
250
|
+
var decision;
|
|
251
|
+
return _regenerator.default.wrap(function (_context) {
|
|
252
|
+
while (1) switch (_context.prev = _context.next) {
|
|
253
|
+
case 0:
|
|
254
|
+
global.fetch.mockResolvedValueOnce({
|
|
255
|
+
ok: true,
|
|
256
|
+
json: function json() {
|
|
257
|
+
return Promise.resolve({
|
|
258
|
+
choices: [{
|
|
259
|
+
message: {
|
|
260
|
+
content: JSON.stringify({
|
|
261
|
+
selectedVariant: 'gaming',
|
|
262
|
+
confidence: 0.92,
|
|
263
|
+
reasoning: 'User came from Reddit with gaming console campaign'
|
|
264
|
+
})
|
|
265
|
+
}
|
|
266
|
+
}]
|
|
267
|
+
});
|
|
268
|
+
}
|
|
269
|
+
});
|
|
270
|
+
_context.next = 1;
|
|
271
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
272
|
+
templates: mockTemplates,
|
|
273
|
+
aiConfig: {
|
|
274
|
+
apiKey: 'sk-test',
|
|
275
|
+
mode: 'select',
|
|
276
|
+
cacheDecisions: false,
|
|
277
|
+
historyEnabled: false
|
|
278
|
+
}
|
|
279
|
+
});
|
|
280
|
+
case 1:
|
|
281
|
+
decision = _context.sent;
|
|
282
|
+
expect(decision.intent).toBe('gaming');
|
|
283
|
+
expect(decision.source).toBe('ai');
|
|
284
|
+
expect(decision.template).toBe(mockTemplates['gaming']);
|
|
285
|
+
expect(decision.aiResponse.confidence).toBe(0.92);
|
|
286
|
+
expect(decision.aiResponse.model).toBe('gpt-4o-mini');
|
|
287
|
+
expect(decision.aiResponse.mode).toBe('select');
|
|
288
|
+
case 2:
|
|
289
|
+
case "end":
|
|
290
|
+
return _context.stop();
|
|
291
|
+
}
|
|
292
|
+
}, _callee);
|
|
293
|
+
})));
|
|
294
|
+
it('should include Authorization header for direct API', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee2() {
|
|
295
|
+
return _regenerator.default.wrap(function (_context2) {
|
|
296
|
+
while (1) switch (_context2.prev = _context2.next) {
|
|
297
|
+
case 0:
|
|
298
|
+
global.fetch.mockResolvedValueOnce({
|
|
299
|
+
ok: true,
|
|
300
|
+
json: function json() {
|
|
301
|
+
return Promise.resolve({
|
|
302
|
+
choices: [{
|
|
303
|
+
message: {
|
|
304
|
+
content: JSON.stringify({
|
|
305
|
+
selectedVariant: 'gaming',
|
|
306
|
+
confidence: 0.8,
|
|
307
|
+
reasoning: 'test'
|
|
308
|
+
})
|
|
309
|
+
}
|
|
310
|
+
}]
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
});
|
|
314
|
+
_context2.next = 1;
|
|
315
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
316
|
+
templates: mockTemplates,
|
|
317
|
+
aiConfig: {
|
|
318
|
+
apiKey: 'sk-mykey123',
|
|
319
|
+
mode: 'select',
|
|
320
|
+
cacheDecisions: false,
|
|
321
|
+
historyEnabled: false
|
|
322
|
+
}
|
|
323
|
+
});
|
|
324
|
+
case 1:
|
|
325
|
+
expect(global.fetch).toHaveBeenCalledWith('https://api.openai.com/v1/chat/completions', expect.objectContaining({
|
|
326
|
+
method: 'POST',
|
|
327
|
+
headers: expect.objectContaining({
|
|
328
|
+
'Authorization': 'Bearer sk-mykey123'
|
|
329
|
+
})
|
|
330
|
+
}));
|
|
331
|
+
case 2:
|
|
332
|
+
case "end":
|
|
333
|
+
return _context2.stop();
|
|
334
|
+
}
|
|
335
|
+
}, _callee2);
|
|
336
|
+
})));
|
|
337
|
+
it('should use proxy URL when apiUrl is provided', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee3() {
|
|
338
|
+
return _regenerator.default.wrap(function (_context3) {
|
|
339
|
+
while (1) switch (_context3.prev = _context3.next) {
|
|
340
|
+
case 0:
|
|
341
|
+
global.fetch.mockResolvedValueOnce({
|
|
342
|
+
ok: true,
|
|
343
|
+
json: function json() {
|
|
344
|
+
return Promise.resolve({
|
|
345
|
+
choices: [{
|
|
346
|
+
message: {
|
|
347
|
+
content: JSON.stringify({
|
|
348
|
+
selectedVariant: 'gaming',
|
|
349
|
+
confidence: 0.8,
|
|
350
|
+
reasoning: 'test'
|
|
351
|
+
})
|
|
352
|
+
}
|
|
353
|
+
}]
|
|
354
|
+
});
|
|
355
|
+
}
|
|
356
|
+
});
|
|
357
|
+
_context3.next = 1;
|
|
358
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
359
|
+
templates: mockTemplates,
|
|
360
|
+
aiConfig: {
|
|
361
|
+
apiUrl: 'https://myproxy.com/openai',
|
|
362
|
+
mode: 'select',
|
|
363
|
+
cacheDecisions: false,
|
|
364
|
+
historyEnabled: false
|
|
365
|
+
}
|
|
366
|
+
});
|
|
367
|
+
case 1:
|
|
368
|
+
expect(global.fetch).toHaveBeenCalledWith('https://myproxy.com/openai', expect.any(Object));
|
|
369
|
+
case 2:
|
|
370
|
+
case "end":
|
|
371
|
+
return _context3.stop();
|
|
372
|
+
}
|
|
373
|
+
}, _callee3);
|
|
374
|
+
})));
|
|
375
|
+
});
|
|
376
|
+
describe('decide (generate mode)', function () {
|
|
377
|
+
it('should call OpenAI and return generated content', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee4() {
|
|
378
|
+
var decision;
|
|
379
|
+
return _regenerator.default.wrap(function (_context4) {
|
|
380
|
+
while (1) switch (_context4.prev = _context4.next) {
|
|
381
|
+
case 0:
|
|
382
|
+
global.fetch.mockResolvedValueOnce({
|
|
383
|
+
ok: true,
|
|
384
|
+
json: function json() {
|
|
385
|
+
return Promise.resolve({
|
|
386
|
+
choices: [{
|
|
387
|
+
message: {
|
|
388
|
+
content: JSON.stringify({
|
|
389
|
+
headline: 'Game On, Gear Up',
|
|
390
|
+
subheadline: 'The ultimate gaming setup awaits you.',
|
|
391
|
+
ctaLabel: 'Level Up',
|
|
392
|
+
confidence: 0.88,
|
|
393
|
+
reasoning: 'Gamer from Reddit'
|
|
394
|
+
})
|
|
395
|
+
}
|
|
396
|
+
}]
|
|
397
|
+
});
|
|
398
|
+
}
|
|
399
|
+
});
|
|
400
|
+
_context4.next = 1;
|
|
401
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
402
|
+
templates: mockTemplates,
|
|
403
|
+
aiConfig: {
|
|
404
|
+
apiKey: 'sk-test',
|
|
405
|
+
mode: 'generate',
|
|
406
|
+
cacheDecisions: false,
|
|
407
|
+
historyEnabled: false
|
|
408
|
+
}
|
|
409
|
+
});
|
|
410
|
+
case 1:
|
|
411
|
+
decision = _context4.sent;
|
|
412
|
+
expect(decision.intent).toBe('ai-generated');
|
|
413
|
+
expect(decision.source).toBe('ai');
|
|
414
|
+
expect(decision.template.headline).toBe('Game On, Gear Up');
|
|
415
|
+
expect(decision.template.subheadline).toBe('The ultimate gaming setup awaits you.');
|
|
416
|
+
expect(decision.template.ctaLabel).toBe('Level Up');
|
|
417
|
+
expect(decision.aiResponse.mode).toBe('generate');
|
|
418
|
+
case 2:
|
|
419
|
+
case "end":
|
|
420
|
+
return _context4.stop();
|
|
421
|
+
}
|
|
422
|
+
}, _callee4);
|
|
423
|
+
})));
|
|
424
|
+
});
|
|
425
|
+
describe('error handling', function () {
|
|
426
|
+
it('should reject with error for invalid config', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee5() {
|
|
427
|
+
return _regenerator.default.wrap(function (_context5) {
|
|
428
|
+
while (1) switch (_context5.prev = _context5.next) {
|
|
429
|
+
case 0:
|
|
430
|
+
_context5.next = 1;
|
|
431
|
+
return expect(LuaAIPersonalize.decide(mockContext, {
|
|
432
|
+
templates: mockTemplates,
|
|
433
|
+
aiConfig: {}
|
|
434
|
+
})).rejects.toThrow('apiKey');
|
|
435
|
+
case 1:
|
|
436
|
+
case "end":
|
|
437
|
+
return _context5.stop();
|
|
438
|
+
}
|
|
439
|
+
}, _callee5);
|
|
440
|
+
})));
|
|
441
|
+
it('should reject on API error', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee6() {
|
|
442
|
+
return _regenerator.default.wrap(function (_context6) {
|
|
443
|
+
while (1) switch (_context6.prev = _context6.next) {
|
|
444
|
+
case 0:
|
|
445
|
+
global.fetch.mockResolvedValueOnce({
|
|
446
|
+
ok: false,
|
|
447
|
+
status: 401,
|
|
448
|
+
text: function text() {
|
|
449
|
+
return Promise.resolve('Unauthorized');
|
|
450
|
+
}
|
|
451
|
+
});
|
|
452
|
+
_context6.next = 1;
|
|
453
|
+
return expect(LuaAIPersonalize.decide(mockContext, {
|
|
454
|
+
templates: mockTemplates,
|
|
455
|
+
aiConfig: {
|
|
456
|
+
apiKey: 'sk-bad',
|
|
457
|
+
cacheDecisions: false,
|
|
458
|
+
historyEnabled: false,
|
|
459
|
+
maxRetries: 0
|
|
460
|
+
}
|
|
461
|
+
})).rejects.toThrow('401');
|
|
462
|
+
case 1:
|
|
463
|
+
case "end":
|
|
464
|
+
return _context6.stop();
|
|
465
|
+
}
|
|
466
|
+
}, _callee6);
|
|
467
|
+
})));
|
|
468
|
+
it('should reject on invalid JSON response', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee7() {
|
|
469
|
+
return _regenerator.default.wrap(function (_context7) {
|
|
470
|
+
while (1) switch (_context7.prev = _context7.next) {
|
|
471
|
+
case 0:
|
|
472
|
+
global.fetch.mockResolvedValueOnce({
|
|
473
|
+
ok: true,
|
|
474
|
+
json: function json() {
|
|
475
|
+
return Promise.resolve({
|
|
476
|
+
choices: [{
|
|
477
|
+
message: {
|
|
478
|
+
content: 'this is not json'
|
|
479
|
+
}
|
|
480
|
+
}]
|
|
481
|
+
});
|
|
482
|
+
}
|
|
483
|
+
});
|
|
484
|
+
_context7.next = 1;
|
|
485
|
+
return expect(LuaAIPersonalize.decide(mockContext, {
|
|
486
|
+
templates: mockTemplates,
|
|
487
|
+
aiConfig: {
|
|
488
|
+
apiKey: 'sk-test',
|
|
489
|
+
cacheDecisions: false,
|
|
490
|
+
historyEnabled: false,
|
|
491
|
+
maxRetries: 0
|
|
492
|
+
}
|
|
493
|
+
})).rejects.toThrow();
|
|
494
|
+
case 1:
|
|
495
|
+
case "end":
|
|
496
|
+
return _context7.stop();
|
|
497
|
+
}
|
|
498
|
+
}, _callee7);
|
|
499
|
+
})));
|
|
500
|
+
it('should reject on invalid selection (nonexistent variant)', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee8() {
|
|
501
|
+
return _regenerator.default.wrap(function (_context8) {
|
|
502
|
+
while (1) switch (_context8.prev = _context8.next) {
|
|
503
|
+
case 0:
|
|
504
|
+
global.fetch.mockResolvedValueOnce({
|
|
505
|
+
ok: true,
|
|
506
|
+
json: function json() {
|
|
507
|
+
return Promise.resolve({
|
|
508
|
+
choices: [{
|
|
509
|
+
message: {
|
|
510
|
+
content: JSON.stringify({
|
|
511
|
+
selectedVariant: 'fake',
|
|
512
|
+
confidence: 0.5,
|
|
513
|
+
reasoning: 'oops'
|
|
514
|
+
})
|
|
515
|
+
}
|
|
516
|
+
}]
|
|
517
|
+
});
|
|
518
|
+
}
|
|
519
|
+
});
|
|
520
|
+
_context8.next = 1;
|
|
521
|
+
return expect(LuaAIPersonalize.decide(mockContext, {
|
|
522
|
+
templates: mockTemplates,
|
|
523
|
+
aiConfig: {
|
|
524
|
+
apiKey: 'sk-test',
|
|
525
|
+
cacheDecisions: false,
|
|
526
|
+
historyEnabled: false,
|
|
527
|
+
maxRetries: 0
|
|
528
|
+
}
|
|
529
|
+
})).rejects.toThrow('does not exist');
|
|
530
|
+
case 1:
|
|
531
|
+
case "end":
|
|
532
|
+
return _context8.stop();
|
|
533
|
+
}
|
|
534
|
+
}, _callee8);
|
|
535
|
+
})));
|
|
536
|
+
});
|
|
537
|
+
describe('caching', function () {
|
|
538
|
+
it('should cache a decision and return it on subsequent calls', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee9() {
|
|
539
|
+
var first, second;
|
|
540
|
+
return _regenerator.default.wrap(function (_context9) {
|
|
541
|
+
while (1) switch (_context9.prev = _context9.next) {
|
|
542
|
+
case 0:
|
|
543
|
+
global.fetch.mockResolvedValueOnce({
|
|
544
|
+
ok: true,
|
|
545
|
+
json: function json() {
|
|
546
|
+
return Promise.resolve({
|
|
547
|
+
choices: [{
|
|
548
|
+
message: {
|
|
549
|
+
content: JSON.stringify({
|
|
550
|
+
selectedVariant: 'gaming',
|
|
551
|
+
confidence: 0.9,
|
|
552
|
+
reasoning: 'test'
|
|
553
|
+
})
|
|
554
|
+
}
|
|
555
|
+
}]
|
|
556
|
+
});
|
|
557
|
+
}
|
|
558
|
+
});
|
|
559
|
+
|
|
560
|
+
// First call - hits API
|
|
561
|
+
_context9.next = 1;
|
|
562
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
563
|
+
templates: mockTemplates,
|
|
564
|
+
aiConfig: {
|
|
565
|
+
apiKey: 'sk-test',
|
|
566
|
+
cacheDecisions: true,
|
|
567
|
+
historyEnabled: false
|
|
568
|
+
}
|
|
569
|
+
});
|
|
570
|
+
case 1:
|
|
571
|
+
first = _context9.sent;
|
|
572
|
+
expect(first.source).toBe('ai');
|
|
573
|
+
expect(global.fetch).toHaveBeenCalledTimes(1);
|
|
574
|
+
|
|
575
|
+
// Second call - should use cache
|
|
576
|
+
_context9.next = 2;
|
|
577
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
578
|
+
templates: mockTemplates,
|
|
579
|
+
aiConfig: {
|
|
580
|
+
apiKey: 'sk-test',
|
|
581
|
+
cacheDecisions: true,
|
|
582
|
+
historyEnabled: false
|
|
583
|
+
}
|
|
584
|
+
});
|
|
585
|
+
case 2:
|
|
586
|
+
second = _context9.sent;
|
|
587
|
+
expect(second.source).toBe('ai-cached');
|
|
588
|
+
expect(global.fetch).toHaveBeenCalledTimes(1); // No new fetch call
|
|
589
|
+
case 3:
|
|
590
|
+
case "end":
|
|
591
|
+
return _context9.stop();
|
|
592
|
+
}
|
|
593
|
+
}, _callee9);
|
|
594
|
+
})));
|
|
595
|
+
it('clearCache should remove all cached decisions', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee0() {
|
|
596
|
+
var result;
|
|
597
|
+
return _regenerator.default.wrap(function (_context0) {
|
|
598
|
+
while (1) switch (_context0.prev = _context0.next) {
|
|
599
|
+
case 0:
|
|
600
|
+
// Store something in cache
|
|
601
|
+
global.fetch.mockResolvedValueOnce({
|
|
602
|
+
ok: true,
|
|
603
|
+
json: function json() {
|
|
604
|
+
return Promise.resolve({
|
|
605
|
+
choices: [{
|
|
606
|
+
message: {
|
|
607
|
+
content: JSON.stringify({
|
|
608
|
+
selectedVariant: 'gaming',
|
|
609
|
+
confidence: 0.9,
|
|
610
|
+
reasoning: 'test'
|
|
611
|
+
})
|
|
612
|
+
}
|
|
613
|
+
}]
|
|
614
|
+
});
|
|
615
|
+
}
|
|
616
|
+
});
|
|
617
|
+
_context0.next = 1;
|
|
618
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
619
|
+
templates: mockTemplates,
|
|
620
|
+
aiConfig: {
|
|
621
|
+
apiKey: 'sk-test',
|
|
622
|
+
cacheDecisions: true,
|
|
623
|
+
historyEnabled: false
|
|
624
|
+
}
|
|
625
|
+
});
|
|
626
|
+
case 1:
|
|
627
|
+
LuaAIPersonalize.clearCache();
|
|
628
|
+
|
|
629
|
+
// Next call should hit API again
|
|
630
|
+
global.fetch.mockResolvedValueOnce({
|
|
631
|
+
ok: true,
|
|
632
|
+
json: function json() {
|
|
633
|
+
return Promise.resolve({
|
|
634
|
+
choices: [{
|
|
635
|
+
message: {
|
|
636
|
+
content: JSON.stringify({
|
|
637
|
+
selectedVariant: 'professional',
|
|
638
|
+
confidence: 0.8,
|
|
639
|
+
reasoning: 'cleared'
|
|
640
|
+
})
|
|
641
|
+
}
|
|
642
|
+
}]
|
|
643
|
+
});
|
|
644
|
+
}
|
|
645
|
+
});
|
|
646
|
+
_context0.next = 2;
|
|
647
|
+
return LuaAIPersonalize.decide(mockContext, {
|
|
648
|
+
templates: mockTemplates,
|
|
649
|
+
aiConfig: {
|
|
650
|
+
apiKey: 'sk-test',
|
|
651
|
+
cacheDecisions: true,
|
|
652
|
+
historyEnabled: false
|
|
653
|
+
}
|
|
654
|
+
});
|
|
655
|
+
case 2:
|
|
656
|
+
result = _context0.sent;
|
|
657
|
+
expect(result.source).toBe('ai');
|
|
658
|
+
expect(global.fetch).toHaveBeenCalledTimes(2);
|
|
659
|
+
case 3:
|
|
660
|
+
case "end":
|
|
661
|
+
return _context0.stop();
|
|
662
|
+
}
|
|
663
|
+
}, _callee0);
|
|
664
|
+
})));
|
|
665
|
+
});
|
|
666
|
+
describe('isReady', function () {
|
|
667
|
+
it('should return ready for valid config with modules loaded', function () {
|
|
668
|
+
var result = LuaAIPersonalize.isReady({
|
|
669
|
+
apiKey: 'sk-test'
|
|
670
|
+
});
|
|
671
|
+
expect(result.ready).toBe(true);
|
|
672
|
+
});
|
|
673
|
+
it('should return not ready for invalid config', function () {
|
|
674
|
+
var result = LuaAIPersonalize.isReady({});
|
|
675
|
+
expect(result.ready).toBe(false);
|
|
676
|
+
expect(result.error).toBeDefined();
|
|
677
|
+
});
|
|
678
|
+
});
|
|
679
|
+
});
|
|
680
|
+
describe('LuaPrompts', function () {
|
|
681
|
+
it('should be registered on global', function () {
|
|
682
|
+
expect(LuaPrompts).toBeDefined();
|
|
683
|
+
expect((0, _typeof2.default)(LuaPrompts.buildSelectPrompt)).toBe('function');
|
|
684
|
+
expect((0, _typeof2.default)(LuaPrompts.buildGeneratePrompt)).toBe('function');
|
|
685
|
+
expect((0, _typeof2.default)(LuaPrompts.buildMessages)).toBe('function');
|
|
686
|
+
});
|
|
687
|
+
describe('buildMessages', function () {
|
|
688
|
+
it('should build select mode messages', function () {
|
|
689
|
+
var messages = LuaPrompts.buildMessages('select', {
|
|
690
|
+
context: mockContext,
|
|
691
|
+
weightedHistory: 'No history',
|
|
692
|
+
variants: mockTemplates
|
|
693
|
+
});
|
|
694
|
+
expect(messages.length).toBe(2);
|
|
695
|
+
expect(messages[0].role).toBe('system');
|
|
696
|
+
expect(messages[1].role).toBe('user');
|
|
697
|
+
expect(messages[0].content).toContain('personalization');
|
|
698
|
+
expect(messages[1].content).toContain('reddit');
|
|
699
|
+
expect(messages[1].content).toContain('gaming');
|
|
700
|
+
expect(messages[1].content).toContain('AVAILABLE VARIANTS');
|
|
701
|
+
});
|
|
702
|
+
it('should build generate mode messages', function () {
|
|
703
|
+
var messages = LuaPrompts.buildMessages('generate', {
|
|
704
|
+
context: mockContext,
|
|
705
|
+
weightedHistory: 'No history',
|
|
706
|
+
brandContext: {
|
|
707
|
+
brandVoice: 'bold and energetic',
|
|
708
|
+
targetAudience: 'gamers'
|
|
709
|
+
}
|
|
710
|
+
});
|
|
711
|
+
expect(messages.length).toBe(2);
|
|
712
|
+
expect(messages[0].content).toContain('copywriter');
|
|
713
|
+
expect(messages[1].content).toContain('bold and energetic');
|
|
714
|
+
expect(messages[1].content).toContain('gamers');
|
|
715
|
+
});
|
|
716
|
+
it('should use custom system prompt when provided', function () {
|
|
717
|
+
var messages = LuaPrompts.buildMessages('select', {
|
|
718
|
+
context: mockContext,
|
|
719
|
+
variants: mockTemplates
|
|
720
|
+
}, {
|
|
721
|
+
system: 'You are a custom AI.'
|
|
722
|
+
});
|
|
723
|
+
expect(messages[0].content).toBe('You are a custom AI.');
|
|
724
|
+
});
|
|
725
|
+
});
|
|
726
|
+
describe('buildSelectPrompt', function () {
|
|
727
|
+
it('should include all UTM parameters', function () {
|
|
728
|
+
var prompt = LuaPrompts.buildSelectPrompt({
|
|
729
|
+
context: {
|
|
730
|
+
utm: {
|
|
731
|
+
utm_source: 'google',
|
|
732
|
+
utm_medium: 'cpc',
|
|
733
|
+
utm_campaign: 'sale'
|
|
734
|
+
},
|
|
735
|
+
referrer: {
|
|
736
|
+
source: 'google',
|
|
737
|
+
category: 'search'
|
|
738
|
+
},
|
|
739
|
+
userAgent: {
|
|
740
|
+
isMobile: true,
|
|
741
|
+
isTablet: false,
|
|
742
|
+
isDesktop: false
|
|
743
|
+
},
|
|
744
|
+
hasUTM: true,
|
|
745
|
+
primaryIntent: 'price-focused'
|
|
746
|
+
},
|
|
747
|
+
weightedHistory: 'New visitor',
|
|
748
|
+
variants: mockTemplates
|
|
749
|
+
});
|
|
750
|
+
expect(prompt).toContain('google');
|
|
751
|
+
expect(prompt).toContain('cpc');
|
|
752
|
+
expect(prompt).toContain('sale');
|
|
753
|
+
expect(prompt).toContain('mobile');
|
|
754
|
+
expect(prompt).toContain('price-focused');
|
|
755
|
+
});
|
|
756
|
+
it('should list all variant keys', function () {
|
|
757
|
+
var prompt = LuaPrompts.buildSelectPrompt({
|
|
758
|
+
context: mockContext,
|
|
759
|
+
variants: mockTemplates
|
|
760
|
+
});
|
|
761
|
+
expect(prompt).toContain('"gaming"');
|
|
762
|
+
expect(prompt).toContain('"professional"');
|
|
763
|
+
expect(prompt).toContain('"default"');
|
|
764
|
+
});
|
|
765
|
+
it('should include preference scores when provided', function () {
|
|
766
|
+
var prompt = LuaPrompts.buildSelectPrompt({
|
|
767
|
+
context: mockContext,
|
|
768
|
+
variants: mockTemplates,
|
|
769
|
+
preferences: {
|
|
770
|
+
gaming: 1.5,
|
|
771
|
+
professional: 0.7
|
|
772
|
+
}
|
|
773
|
+
});
|
|
774
|
+
expect(prompt).toContain('PREFERENCE SCORES');
|
|
775
|
+
expect(prompt).toContain('gaming');
|
|
776
|
+
expect(prompt).toContain('1.500');
|
|
777
|
+
});
|
|
778
|
+
});
|
|
779
|
+
describe('buildGeneratePrompt', function () {
|
|
780
|
+
it('should include brand context fields', function () {
|
|
781
|
+
var prompt = LuaPrompts.buildGeneratePrompt({
|
|
782
|
+
context: mockContext,
|
|
783
|
+
brandContext: {
|
|
784
|
+
brandVoice: 'friendly and modern',
|
|
785
|
+
targetAudience: 'young professionals',
|
|
786
|
+
productType: 'tech accessories',
|
|
787
|
+
industry: 'e-commerce',
|
|
788
|
+
customField: 'custom value'
|
|
789
|
+
}
|
|
790
|
+
});
|
|
791
|
+
expect(prompt).toContain('friendly and modern');
|
|
792
|
+
expect(prompt).toContain('young professionals');
|
|
793
|
+
expect(prompt).toContain('tech accessories');
|
|
794
|
+
expect(prompt).toContain('e-commerce');
|
|
795
|
+
expect(prompt).toContain('custom value');
|
|
796
|
+
});
|
|
797
|
+
it('should include reference template when provided', function () {
|
|
798
|
+
var prompt = LuaPrompts.buildGeneratePrompt({
|
|
799
|
+
context: mockContext,
|
|
800
|
+
fallbackTemplate: {
|
|
801
|
+
headline: 'Default Headline',
|
|
802
|
+
subheadline: 'Default sub',
|
|
803
|
+
ctaLabel: 'Default CTA'
|
|
804
|
+
}
|
|
805
|
+
});
|
|
806
|
+
expect(prompt).toContain('Default Headline');
|
|
807
|
+
expect(prompt).toContain('REFERENCE TEMPLATE');
|
|
808
|
+
});
|
|
809
|
+
});
|
|
810
|
+
});
|
|
811
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["localStorageMock","store","getItem","jest","fn","key","setItem","val","String","removeItem","clear","length","Object","keys","i","defineProperty","global","value","getRandomValues","buf","Math","floor","random","fetch","AbortController","_class","_classCallCheck2","default","signal","_createClass2","abort","require","LuaAIPersonalize","LuaPrompts","LuaWeightedHistory","mockTemplates","headline","subheadline","ctaLabel","ctaLink","image","mockContext","utm","utm_source","utm_campaign","referrer","source","category","url","userAgent","isMobile","isTablet","isDesktop","raw","timestamp","Date","now","hasUTM","primaryIntent","beforeEach","clearAllMocks","mockReset","describe","it","expect","toBeDefined","_typeof2","decide","toBe","normalizeConfig","result","valid","error","toContain","apiKey","useDirectApi","apiUrl","model","mode","timeout","maxRetries","fallbackToStandard","cacheDecisions","historyEnabled","historyDecayRate","temperature","maxTokens","validateSelectResponse","selectedVariant","confidence","reasoning","validateGenerateResponse","_asyncToGenerator2","_regenerator","mark","_callee","decision","wrap","_context","prev","next","mockResolvedValueOnce","ok","json","Promise","resolve","choices","message","content","JSON","stringify","templates","aiConfig","sent","intent","template","aiResponse","stop","_callee2","_context2","toHaveBeenCalledWith","objectContaining","method","headers","_callee3","_context3","any","_callee4","_context4","_callee5","_context5","rejects","toThrow","_callee6","_context6","status","text","_callee7","_context7","_callee8","_context8","_callee9","first","second","_context9","toHaveBeenCalledTimes","_callee0","_context0","clearCache","isReady","ready","buildSelectPrompt","buildGeneratePrompt","buildMessages","messages","context","weightedHistory","variants","role","brandContext","brandVoice","targetAudience","system","prompt","utm_medium","preferences","gaming","professional","productType","industry","customField","fallbackTemplate"],"sources":["../../../src/__tests__/ai-personalize.test.js"],"sourcesContent":["\n/**\n * Tests for AI Personalization Engine\n * Tests config validation, API communication, response validation, and caching\n */\n\n// Mock localStorage\nconst localStorageMock = (function () {\n    let store = {}\n    return {\n        getItem: jest.fn(function (key) { return store[key] || null }),\n        setItem: jest.fn(function (key, val) { store[key] = String(val) }),\n        removeItem: jest.fn(function (key) { delete store[key] }),\n        clear: jest.fn(function () { store = {} }),\n        get length() { return Object.keys(store).length },\n        key: jest.fn(function (i) { return Object.keys(store)[i] || null })\n    }\n})()\n\nObject.defineProperty(global, 'localStorage', { value: localStorageMock })\n\n// Mock crypto\nObject.defineProperty(global, 'crypto', {\n    value: {\n        getRandomValues: function (buf) {\n            for (let i = 0; i < buf.length; i++) buf[i] = Math.floor(Math.random() * 256)\n            return buf\n        }\n    }\n})\n\n// Mock fetch\nglobal.fetch = jest.fn()\nglobal.AbortController = class {\n    constructor() { this.signal = {} }\n    abort() {}\n}\n\n// Load modules in order (dependencies first)\nrequire('../storage/weighted-history')\nrequire('../prompts/personalization-prompts')\nrequire('../ai-personalize')\n\nconst LuaAIPersonalize = global.LuaAIPersonalize\nconst LuaPrompts = global.LuaPrompts\nconst LuaWeightedHistory = global.LuaWeightedHistory\n\nconst mockTemplates = {\n    'gaming': {\n        headline: 'Level Up Your Setup',\n        subheadline: 'Pro gear for serious gamers.',\n        ctaLabel: 'Explore Gaming',\n        ctaLink: '/gaming',\n        image: 'gaming.jpg'\n    },\n    'professional': {\n        headline: 'Work Smarter',\n        subheadline: 'Premium productivity tools.',\n        ctaLabel: 'View Collection',\n        ctaLink: '/professional',\n        image: 'pro.jpg'\n    },\n    'default': {\n        headline: 'Welcome',\n        subheadline: 'Discover products.',\n        ctaLabel: 'Shop Now',\n        ctaLink: '/shop',\n        image: 'default.jpg'\n    }\n}\n\nconst mockContext = {\n    utm: { utm_source: 'reddit', utm_campaign: 'gaming_console' },\n    referrer: { source: 'reddit', category: 'social', url: 'https://reddit.com' },\n    userAgent: { isMobile: false, isTablet: false, isDesktop: true, raw: '' },\n    timestamp: Date.now(),\n    hasUTM: true,\n    primaryIntent: 'gaming'\n}\n\nbeforeEach(() => {\n    localStorageMock.clear()\n    jest.clearAllMocks()\n    global.fetch.mockReset()\n})\n\ndescribe('LuaAIPersonalize', () => {\n    it('should be registered on global', () => {\n        expect(LuaAIPersonalize).toBeDefined()\n        expect(typeof LuaAIPersonalize.decide).toBe('function')\n        expect(typeof LuaAIPersonalize.normalizeConfig).toBe('function')\n    })\n\n    describe('normalizeConfig', () => {\n        it('should reject null config', () => {\n            const result = LuaAIPersonalize.normalizeConfig(null)\n            expect(result.valid).toBe(false)\n        })\n\n        it('should reject config without apiKey or apiUrl', () => {\n            const result = LuaAIPersonalize.normalizeConfig({})\n            expect(result.valid).toBe(false)\n            expect(result.error).toContain('apiKey')\n        })\n\n        it('should accept config with apiKey', () => {\n            const result = LuaAIPersonalize.normalizeConfig({ apiKey: 'sk-test123' })\n            expect(result.valid).toBe(true)\n            expect(result.useDirectApi).toBe(true)\n            expect(result.apiKey).toBe('sk-test123')\n        })\n\n        it('should accept config with apiUrl', () => {\n            const result = LuaAIPersonalize.normalizeConfig({ apiUrl: 'https://proxy.example.com/api' })\n            expect(result.valid).toBe(true)\n            expect(result.useDirectApi).toBe(false)\n            expect(result.apiUrl).toBe('https://proxy.example.com/api')\n        })\n\n        it('should apply defaults for missing fields', () => {\n            const result = LuaAIPersonalize.normalizeConfig({ apiKey: 'sk-test' })\n            expect(result.model).toBe('gpt-4o-mini')\n            expect(result.mode).toBe('select')\n            expect(result.timeout).toBe(5000)\n            expect(result.maxRetries).toBe(1)\n            expect(result.fallbackToStandard).toBe(true)\n            expect(result.cacheDecisions).toBe(true)\n            expect(result.historyEnabled).toBe(true)\n            expect(result.historyDecayRate).toBe(0.9)\n        })\n\n        it('should use custom values when provided', () => {\n            const result = LuaAIPersonalize.normalizeConfig({\n                apiKey: 'sk-test',\n                model: 'gpt-4o',\n                mode: 'generate',\n                timeout: 10000,\n                temperature: 0.5,\n                maxTokens: 1000\n            })\n            expect(result.model).toBe('gpt-4o')\n            expect(result.mode).toBe('generate')\n            expect(result.timeout).toBe(10000)\n            expect(result.temperature).toBe(0.5)\n            expect(result.maxTokens).toBe(1000)\n        })\n\n        it('should prefer apiUrl over direct when both provided', () => {\n            const result = LuaAIPersonalize.normalizeConfig({\n                apiKey: 'sk-test',\n                apiUrl: 'https://proxy.com/api'\n            })\n            expect(result.useDirectApi).toBe(false)\n            expect(result.apiUrl).toBe('https://proxy.com/api')\n        })\n    })\n\n    describe('validateSelectResponse', () => {\n        it('should accept valid selection response', () => {\n            const result = LuaAIPersonalize.validateSelectResponse(\n                { selectedVariant: 'gaming', confidence: 0.9, reasoning: 'test' },\n                mockTemplates\n            )\n            expect(result.valid).toBe(true)\n        })\n\n        it('should reject missing selectedVariant', () => {\n            const result = LuaAIPersonalize.validateSelectResponse(\n                { confidence: 0.9 },\n                mockTemplates\n            )\n            expect(result.valid).toBe(false)\n        })\n\n        it('should reject non-existent variant', () => {\n            const result = LuaAIPersonalize.validateSelectResponse(\n                { selectedVariant: 'nonexistent' },\n                mockTemplates\n            )\n            expect(result.valid).toBe(false)\n            expect(result.error).toContain('nonexistent')\n        })\n\n        it('should reject null response', () => {\n            const result = LuaAIPersonalize.validateSelectResponse(null, mockTemplates)\n            expect(result.valid).toBe(false)\n        })\n    })\n\n    describe('validateGenerateResponse', () => {\n        it('should accept valid generation response', () => {\n            const result = LuaAIPersonalize.validateGenerateResponse({\n                headline: 'Test Headline',\n                subheadline: 'Test subheadline here',\n                ctaLabel: 'Click Me'\n            })\n            expect(result.valid).toBe(true)\n        })\n\n        it('should reject missing headline', () => {\n            const result = LuaAIPersonalize.validateGenerateResponse({\n                subheadline: 'Test',\n                ctaLabel: 'Click'\n            })\n            expect(result.valid).toBe(false)\n            expect(result.error).toContain('headline')\n        })\n\n        it('should reject missing subheadline', () => {\n            const result = LuaAIPersonalize.validateGenerateResponse({\n                headline: 'Test',\n                ctaLabel: 'Click'\n            })\n            expect(result.valid).toBe(false)\n        })\n\n        it('should reject missing ctaLabel', () => {\n            const result = LuaAIPersonalize.validateGenerateResponse({\n                headline: 'Test',\n                subheadline: 'Sub'\n            })\n            expect(result.valid).toBe(false)\n        })\n    })\n\n    describe('decide (select mode)', () => {\n        it('should call OpenAI and return a valid decision', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: {\n                            content: JSON.stringify({\n                                selectedVariant: 'gaming',\n                                confidence: 0.92,\n                                reasoning: 'User came from Reddit with gaming console campaign'\n                            })\n                        }\n                    }]\n                })\n            })\n\n            const decision = await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', mode: 'select', cacheDecisions: false, historyEnabled: false }\n            })\n\n            expect(decision.intent).toBe('gaming')\n            expect(decision.source).toBe('ai')\n            expect(decision.template).toBe(mockTemplates['gaming'])\n            expect(decision.aiResponse.confidence).toBe(0.92)\n            expect(decision.aiResponse.model).toBe('gpt-4o-mini')\n            expect(decision.aiResponse.mode).toBe('select')\n        })\n\n        it('should include Authorization header for direct API', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'gaming', confidence: 0.8, reasoning: 'test' }) }\n                    }]\n                })\n            })\n\n            await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-mykey123', mode: 'select', cacheDecisions: false, historyEnabled: false }\n            })\n\n            expect(global.fetch).toHaveBeenCalledWith(\n                'https://api.openai.com/v1/chat/completions',\n                expect.objectContaining({\n                    method: 'POST',\n                    headers: expect.objectContaining({\n                        'Authorization': 'Bearer sk-mykey123'\n                    })\n                })\n            )\n        })\n\n        it('should use proxy URL when apiUrl is provided', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'gaming', confidence: 0.8, reasoning: 'test' }) }\n                    }]\n                })\n            })\n\n            await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiUrl: 'https://myproxy.com/openai', mode: 'select', cacheDecisions: false, historyEnabled: false }\n            })\n\n            expect(global.fetch).toHaveBeenCalledWith(\n                'https://myproxy.com/openai',\n                expect.any(Object)\n            )\n        })\n    })\n\n    describe('decide (generate mode)', () => {\n        it('should call OpenAI and return generated content', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: {\n                            content: JSON.stringify({\n                                headline: 'Game On, Gear Up',\n                                subheadline: 'The ultimate gaming setup awaits you.',\n                                ctaLabel: 'Level Up',\n                                confidence: 0.88,\n                                reasoning: 'Gamer from Reddit'\n                            })\n                        }\n                    }]\n                })\n            })\n\n            const decision = await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: {\n                    apiKey: 'sk-test',\n                    mode: 'generate',\n                    cacheDecisions: false,\n                    historyEnabled: false\n                }\n            })\n\n            expect(decision.intent).toBe('ai-generated')\n            expect(decision.source).toBe('ai')\n            expect(decision.template.headline).toBe('Game On, Gear Up')\n            expect(decision.template.subheadline).toBe('The ultimate gaming setup awaits you.')\n            expect(decision.template.ctaLabel).toBe('Level Up')\n            expect(decision.aiResponse.mode).toBe('generate')\n        })\n    })\n\n    describe('error handling', () => {\n        it('should reject with error for invalid config', async () => {\n            await expect(LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: {}\n            })).rejects.toThrow('apiKey')\n        })\n\n        it('should reject on API error', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: false,\n                status: 401,\n                text: () => Promise.resolve('Unauthorized')\n            })\n\n            await expect(LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-bad', cacheDecisions: false, historyEnabled: false, maxRetries: 0 }\n            })).rejects.toThrow('401')\n        })\n\n        it('should reject on invalid JSON response', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: 'this is not json' }\n                    }]\n                })\n            })\n\n            await expect(LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: false, historyEnabled: false, maxRetries: 0 }\n            })).rejects.toThrow()\n        })\n\n        it('should reject on invalid selection (nonexistent variant)', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'fake', confidence: 0.5, reasoning: 'oops' }) }\n                    }]\n                })\n            })\n\n            await expect(LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: false, historyEnabled: false, maxRetries: 0 }\n            })).rejects.toThrow('does not exist')\n        })\n    })\n\n    describe('caching', () => {\n        it('should cache a decision and return it on subsequent calls', async () => {\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'gaming', confidence: 0.9, reasoning: 'test' }) }\n                    }]\n                })\n            })\n\n            // First call - hits API\n            const first = await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: true, historyEnabled: false }\n            })\n            expect(first.source).toBe('ai')\n            expect(global.fetch).toHaveBeenCalledTimes(1)\n\n            // Second call - should use cache\n            const second = await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: true, historyEnabled: false }\n            })\n            expect(second.source).toBe('ai-cached')\n            expect(global.fetch).toHaveBeenCalledTimes(1) // No new fetch call\n        })\n\n        it('clearCache should remove all cached decisions', async () => {\n            // Store something in cache\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'gaming', confidence: 0.9, reasoning: 'test' }) }\n                    }]\n                })\n            })\n\n            await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: true, historyEnabled: false }\n            })\n\n            LuaAIPersonalize.clearCache()\n\n            // Next call should hit API again\n            global.fetch.mockResolvedValueOnce({\n                ok: true,\n                json: () => Promise.resolve({\n                    choices: [{\n                        message: { content: JSON.stringify({ selectedVariant: 'professional', confidence: 0.8, reasoning: 'cleared' }) }\n                    }]\n                })\n            })\n\n            const result = await LuaAIPersonalize.decide(mockContext, {\n                templates: mockTemplates,\n                aiConfig: { apiKey: 'sk-test', cacheDecisions: true, historyEnabled: false }\n            })\n            expect(result.source).toBe('ai')\n            expect(global.fetch).toHaveBeenCalledTimes(2)\n        })\n    })\n\n    describe('isReady', () => {\n        it('should return ready for valid config with modules loaded', () => {\n            const result = LuaAIPersonalize.isReady({ apiKey: 'sk-test' })\n            expect(result.ready).toBe(true)\n        })\n\n        it('should return not ready for invalid config', () => {\n            const result = LuaAIPersonalize.isReady({})\n            expect(result.ready).toBe(false)\n            expect(result.error).toBeDefined()\n        })\n    })\n})\n\ndescribe('LuaPrompts', () => {\n    it('should be registered on global', () => {\n        expect(LuaPrompts).toBeDefined()\n        expect(typeof LuaPrompts.buildSelectPrompt).toBe('function')\n        expect(typeof LuaPrompts.buildGeneratePrompt).toBe('function')\n        expect(typeof LuaPrompts.buildMessages).toBe('function')\n    })\n\n    describe('buildMessages', () => {\n        it('should build select mode messages', () => {\n            const messages = LuaPrompts.buildMessages('select', {\n                context: mockContext,\n                weightedHistory: 'No history',\n                variants: mockTemplates\n            })\n\n            expect(messages.length).toBe(2)\n            expect(messages[0].role).toBe('system')\n            expect(messages[1].role).toBe('user')\n            expect(messages[0].content).toContain('personalization')\n            expect(messages[1].content).toContain('reddit')\n            expect(messages[1].content).toContain('gaming')\n            expect(messages[1].content).toContain('AVAILABLE VARIANTS')\n        })\n\n        it('should build generate mode messages', () => {\n            const messages = LuaPrompts.buildMessages('generate', {\n                context: mockContext,\n                weightedHistory: 'No history',\n                brandContext: {\n                    brandVoice: 'bold and energetic',\n                    targetAudience: 'gamers'\n                }\n            })\n\n            expect(messages.length).toBe(2)\n            expect(messages[0].content).toContain('copywriter')\n            expect(messages[1].content).toContain('bold and energetic')\n            expect(messages[1].content).toContain('gamers')\n        })\n\n        it('should use custom system prompt when provided', () => {\n            const messages = LuaPrompts.buildMessages('select', {\n                context: mockContext,\n                variants: mockTemplates\n            }, {\n                system: 'You are a custom AI.'\n            })\n\n            expect(messages[0].content).toBe('You are a custom AI.')\n        })\n    })\n\n    describe('buildSelectPrompt', () => {\n        it('should include all UTM parameters', () => {\n            const prompt = LuaPrompts.buildSelectPrompt({\n                context: {\n                    utm: { utm_source: 'google', utm_medium: 'cpc', utm_campaign: 'sale' },\n                    referrer: { source: 'google', category: 'search' },\n                    userAgent: { isMobile: true, isTablet: false, isDesktop: false },\n                    hasUTM: true,\n                    primaryIntent: 'price-focused'\n                },\n                weightedHistory: 'New visitor',\n                variants: mockTemplates\n            })\n\n            expect(prompt).toContain('google')\n            expect(prompt).toContain('cpc')\n            expect(prompt).toContain('sale')\n            expect(prompt).toContain('mobile')\n            expect(prompt).toContain('price-focused')\n        })\n\n        it('should list all variant keys', () => {\n            const prompt = LuaPrompts.buildSelectPrompt({\n                context: mockContext,\n                variants: mockTemplates\n            })\n\n            expect(prompt).toContain('\"gaming\"')\n            expect(prompt).toContain('\"professional\"')\n            expect(prompt).toContain('\"default\"')\n        })\n\n        it('should include preference scores when provided', () => {\n            const prompt = LuaPrompts.buildSelectPrompt({\n                context: mockContext,\n                variants: mockTemplates,\n                preferences: { gaming: 1.5, professional: 0.7 }\n            })\n\n            expect(prompt).toContain('PREFERENCE SCORES')\n            expect(prompt).toContain('gaming')\n            expect(prompt).toContain('1.500')\n        })\n    })\n\n    describe('buildGeneratePrompt', () => {\n        it('should include brand context fields', () => {\n            const prompt = LuaPrompts.buildGeneratePrompt({\n                context: mockContext,\n                brandContext: {\n                    brandVoice: 'friendly and modern',\n                    targetAudience: 'young professionals',\n                    productType: 'tech accessories',\n                    industry: 'e-commerce',\n                    customField: 'custom value'\n                }\n            })\n\n            expect(prompt).toContain('friendly and modern')\n            expect(prompt).toContain('young professionals')\n            expect(prompt).toContain('tech accessories')\n            expect(prompt).toContain('e-commerce')\n            expect(prompt).toContain('custom value')\n        })\n\n        it('should include reference template when provided', () => {\n            const prompt = LuaPrompts.buildGeneratePrompt({\n                context: mockContext,\n                fallbackTemplate: {\n                    headline: 'Default Headline',\n                    subheadline: 'Default sub',\n                    ctaLabel: 'Default CTA'\n                }\n            })\n\n            expect(prompt).toContain('Default Headline')\n            expect(prompt).toContain('REFERENCE TEMPLATE')\n        })\n    })\n})\n"],"mappings":"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","ignoreList":[]}
|