@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.
@@ -0,0 +1,384 @@
1
+ "use strict";
2
+
3
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
+ var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
5
+ /**
6
+ * Weighted History Manager
7
+ * ========================
8
+ * Manages localStorage for user visit history with exponential decay.
9
+ * Tracks user visits with timestamps, UTM params, intents, and selected variants.
10
+ * Applies configurable exponential decay so recent visits carry more weight.
11
+ *
12
+ * Storage key: 'lua_personalize_history'
13
+ * Registers on window.LuaWeightedHistory
14
+ *
15
+ * No ES6 imports. Self-contained IIFE.
16
+ */
17
+ ;
18
+ (function (root) {
19
+ 'use strict';
20
+
21
+ // ===================================================================
22
+ // Constants
23
+ // ===================================================================
24
+ var STORAGE_KEY = 'lua_personalize_history';
25
+ var DEFAULT_DECAY_RATE = 0.9;
26
+ var DEFAULT_MAX_HISTORY = 10;
27
+ var DEFAULT_MAX_WEIGHTED = 5;
28
+ var MS_PER_DAY = 1000 * 60 * 60 * 24;
29
+
30
+ // ===================================================================
31
+ // UUID Generator (simple, no crypto dependency)
32
+ // ===================================================================
33
+
34
+ /**
35
+ * Generate a simple UUID v4
36
+ * Uses crypto.getRandomValues when available, Math.random fallback
37
+ * @returns {string} - UUID string
38
+ */
39
+ function generateUUID() {
40
+ try {
41
+ if (typeof crypto !== 'undefined' && crypto.getRandomValues) {
42
+ var buf = new Uint8Array(16);
43
+ crypto.getRandomValues(buf);
44
+ buf[6] = buf[6] & 0x0f | 0x40;
45
+ buf[8] = buf[8] & 0x3f | 0x80;
46
+ var hex = '';
47
+ for (var i = 0; i < 16; i++) {
48
+ var h = buf[i].toString(16);
49
+ hex += h.length === 1 ? '0' + h : h;
50
+ }
51
+ return hex.substring(0, 8) + '-' + hex.substring(8, 12) + '-' + hex.substring(12, 16) + '-' + hex.substring(16, 20) + '-' + hex.substring(20, 32);
52
+ }
53
+ } catch (e) {
54
+ // Fallback below
55
+ }
56
+
57
+ // Math.random fallback
58
+ return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
59
+ var r = Math.random() * 16 | 0;
60
+ var v = c === 'x' ? r : r & 0x3 | 0x8;
61
+ return v.toString(16);
62
+ });
63
+ }
64
+
65
+ // ===================================================================
66
+ // LocalStorage Helpers
67
+ // ===================================================================
68
+
69
+ /**
70
+ * Check if localStorage is available
71
+ * @returns {boolean}
72
+ */
73
+ function isLocalStorageAvailable() {
74
+ try {
75
+ if (typeof localStorage === 'undefined') return false;
76
+ var testKey = '__lua_test__';
77
+ localStorage.setItem(testKey, '1');
78
+ localStorage.removeItem(testKey);
79
+ return true;
80
+ } catch (e) {
81
+ return false;
82
+ }
83
+ }
84
+
85
+ /**
86
+ * Read history from localStorage
87
+ * @returns {Object|null} - Parsed history object or null
88
+ */
89
+ function readFromStorage() {
90
+ if (!isLocalStorageAvailable()) return null;
91
+ try {
92
+ var raw = localStorage.getItem(STORAGE_KEY);
93
+ if (!raw) return null;
94
+ var parsed = JSON.parse(raw);
95
+ if (parsed && (0, _typeof2.default)(parsed) === 'object' && Array.isArray(parsed.visits)) {
96
+ return parsed;
97
+ }
98
+ return null;
99
+ } catch (e) {
100
+ console.warn('[Lua History] Failed to read history:', e);
101
+ return null;
102
+ }
103
+ }
104
+
105
+ /**
106
+ * Write history to localStorage
107
+ * @param {Object} history - History object to persist
108
+ * @returns {boolean} - Whether the write was successful
109
+ */
110
+ function writeToStorage(history) {
111
+ if (!isLocalStorageAvailable()) return false;
112
+ try {
113
+ localStorage.setItem(STORAGE_KEY, JSON.stringify(history));
114
+ return true;
115
+ } catch (e) {
116
+ console.warn('[Lua History] Failed to write history:', e);
117
+ return false;
118
+ }
119
+ }
120
+
121
+ // ===================================================================
122
+ // Weighted Decay Functions
123
+ // ===================================================================
124
+
125
+ /**
126
+ * Calculate the weight of a visit based on its age using exponential decay
127
+ * Weight = decayRate ^ daysAgo
128
+ * Example: decayRate=0.9, 1 day ago = 0.9, 7 days ago = 0.478, 30 days ago = 0.042
129
+ *
130
+ * @param {number} timestamp - Visit timestamp in ms
131
+ * @param {number} [decayRate] - Decay rate per day (0-1, default: 0.9)
132
+ * @returns {number} - Weight between 0 and 1
133
+ */
134
+ function calculateWeight(timestamp, decayRate) {
135
+ decayRate = typeof decayRate === 'number' ? decayRate : DEFAULT_DECAY_RATE;
136
+ var now = Date.now();
137
+ var daysAgo = Math.max(0, (now - timestamp) / MS_PER_DAY);
138
+ return Math.pow(decayRate, daysAgo);
139
+ }
140
+
141
+ /**
142
+ * Build weighted context from visit history
143
+ * Returns the most relevant visits sorted by weight (highest first)
144
+ *
145
+ * @param {Object} history - History object with visits array
146
+ * @param {Object} [options] - Configuration
147
+ * @param {number} [options.decayRate] - Decay rate (default: 0.9)
148
+ * @param {number} [options.maxWeighted] - Max results to return (default: 5)
149
+ * @returns {Array} - Weighted visits sorted by relevance
150
+ */
151
+ function buildWeightedContext(history, options) {
152
+ options = options || {};
153
+ var decayRate = options.decayRate || DEFAULT_DECAY_RATE;
154
+ var maxWeighted = options.maxWeighted || DEFAULT_MAX_WEIGHTED;
155
+ if (!history || !Array.isArray(history.visits) || history.visits.length === 0) {
156
+ return [];
157
+ }
158
+ var weighted = [];
159
+ for (var i = 0; i < history.visits.length; i++) {
160
+ var visit = history.visits[i];
161
+ var weight = calculateWeight(visit.timestamp, decayRate);
162
+ weighted.push({
163
+ timestamp: visit.timestamp,
164
+ weight: Math.round(weight * 1000) / 1000,
165
+ intent: visit.intent || 'unknown',
166
+ selectedVariant: visit.selectedVariant || null,
167
+ source: visit.source || 'unknown',
168
+ aiDecision: visit.aiDecision || false,
169
+ context: visit.context || {}
170
+ });
171
+ }
172
+
173
+ // Sort by weight descending (most recent / relevant first)
174
+ weighted.sort(function (a, b) {
175
+ return b.weight - a.weight;
176
+ });
177
+
178
+ // Return top N results
179
+ return weighted.slice(0, maxWeighted);
180
+ }
181
+
182
+ /**
183
+ * Aggregate intent preferences from weighted history
184
+ * Produces a score map of intents weighted by recency
185
+ *
186
+ * @param {Array} weightedVisits - Output of buildWeightedContext
187
+ * @returns {Object} - Intent -> score mapping
188
+ */
189
+ function aggregatePreferences(weightedVisits) {
190
+ var scores = {};
191
+ for (var i = 0; i < weightedVisits.length; i++) {
192
+ var visit = weightedVisits[i];
193
+ var intent = visit.intent;
194
+ if (!intent || intent === 'unknown' || intent === 'default') continue;
195
+ if (!scores[intent]) {
196
+ scores[intent] = 0;
197
+ }
198
+ scores[intent] += visit.weight;
199
+ }
200
+ return scores;
201
+ }
202
+
203
+ // ===================================================================
204
+ // History Management
205
+ // ===================================================================
206
+
207
+ /**
208
+ * Initialize or retrieve existing history
209
+ * Creates a new history object with a UUID if none exists
210
+ *
211
+ * @returns {Object} - History object
212
+ */
213
+ function getHistory() {
214
+ var history = readFromStorage();
215
+ if (!history) {
216
+ history = {
217
+ userId: generateUUID(),
218
+ createdAt: Date.now(),
219
+ visits: [],
220
+ preferences: {}
221
+ };
222
+ writeToStorage(history);
223
+ }
224
+ return history;
225
+ }
226
+
227
+ /**
228
+ * Record a new visit to history
229
+ *
230
+ * @param {Object} visitData - Visit data to record
231
+ * @param {Object} visitData.context - UTM context (utm, referrer, userAgent)
232
+ * @param {string} visitData.intent - Inferred or AI-selected intent
233
+ * @param {string} visitData.selectedVariant - Selected template/variant key
234
+ * @param {string} visitData.source - Decision source ('ai', 'utm', 'referrer', 'random-ab', etc.)
235
+ * @param {boolean} [visitData.aiDecision] - Whether AI made this decision
236
+ * @param {Object} [options] - Configuration
237
+ * @param {number} [options.maxHistorySize] - Max visits to keep (default: 10)
238
+ * @returns {Object} - Updated history object
239
+ */
240
+ function recordVisit(visitData, options) {
241
+ options = options || {};
242
+ var maxHistorySize = options.maxHistorySize || DEFAULT_MAX_HISTORY;
243
+ var history = getHistory();
244
+
245
+ // Build a minimal context to store (avoid storing sensitive/large data)
246
+ var minimalContext = {};
247
+ if (visitData.context) {
248
+ if (visitData.context.utm) {
249
+ minimalContext.utm = visitData.context.utm;
250
+ }
251
+ if (visitData.context.referrer) {
252
+ minimalContext.referrer = {
253
+ source: visitData.context.referrer.source,
254
+ category: visitData.context.referrer.category
255
+ };
256
+ }
257
+ if (visitData.context.userAgent) {
258
+ minimalContext.device = visitData.context.userAgent.isMobile ? 'mobile' : visitData.context.userAgent.isTablet ? 'tablet' : 'desktop';
259
+ }
260
+ }
261
+ var visit = {
262
+ timestamp: Date.now(),
263
+ context: minimalContext,
264
+ intent: visitData.intent || 'unknown',
265
+ selectedVariant: visitData.selectedVariant || null,
266
+ source: visitData.source || 'unknown',
267
+ aiDecision: !!visitData.aiDecision
268
+ };
269
+ history.visits.push(visit);
270
+
271
+ // Trim to max size (keep most recent)
272
+ if (history.visits.length > maxHistorySize) {
273
+ history.visits = history.visits.slice(-maxHistorySize);
274
+ }
275
+
276
+ // Update aggregated preferences
277
+ var weighted = buildWeightedContext(history, options);
278
+ history.preferences = aggregatePreferences(weighted);
279
+ writeToStorage(history);
280
+ return history;
281
+ }
282
+
283
+ /**
284
+ * Get the user ID from history (creates one if needed)
285
+ * @returns {string} - User UUID
286
+ */
287
+ function getUserId() {
288
+ var history = getHistory();
289
+ return history.userId;
290
+ }
291
+
292
+ /**
293
+ * Check if this is a returning user (has previous visits)
294
+ * @returns {boolean}
295
+ */
296
+ function isReturningUser() {
297
+ var history = readFromStorage();
298
+ return history !== null && Array.isArray(history.visits) && history.visits.length > 0;
299
+ }
300
+
301
+ /**
302
+ * Get the most recent visit from history
303
+ * @returns {Object|null} - Most recent visit or null
304
+ */
305
+ function getLastVisit() {
306
+ var history = readFromStorage();
307
+ if (!history || !Array.isArray(history.visits) || history.visits.length === 0) {
308
+ return null;
309
+ }
310
+ return history.visits[history.visits.length - 1];
311
+ }
312
+
313
+ /**
314
+ * Format weighted history as a human-readable string for AI prompts
315
+ * @param {Array} weightedVisits - Output of buildWeightedContext
316
+ * @returns {string} - Formatted string
317
+ */
318
+ function formatForPrompt(weightedVisits) {
319
+ if (!weightedVisits || weightedVisits.length === 0) {
320
+ return 'No previous visit history available. This is a new visitor.';
321
+ }
322
+ var lines = [];
323
+ lines.push('Previous visits (' + weightedVisits.length + ' recorded, weighted by recency):');
324
+ for (var i = 0; i < weightedVisits.length; i++) {
325
+ var v = weightedVisits[i];
326
+ var date = new Date(v.timestamp);
327
+ var dateStr = date.toISOString().split('T')[0];
328
+ var line = ' - [' + dateStr + '] ';
329
+ line += 'Intent: ' + v.intent;
330
+ if (v.selectedVariant) line += ', Variant: ' + v.selectedVariant;
331
+ line += ', Source: ' + v.source;
332
+ line += ', Weight: ' + v.weight;
333
+ if (v.context && v.context.utm && v.context.utm.utm_source) {
334
+ line += ', UTM: ' + v.context.utm.utm_source;
335
+ }
336
+ lines.push(line);
337
+ }
338
+ return lines.join('\n');
339
+ }
340
+
341
+ /**
342
+ * Clear all history (useful for testing or user request)
343
+ * @returns {boolean} - Whether the clear was successful
344
+ */
345
+ function clearHistory() {
346
+ if (!isLocalStorageAvailable()) return false;
347
+ try {
348
+ localStorage.removeItem(STORAGE_KEY);
349
+ return true;
350
+ } catch (e) {
351
+ return false;
352
+ }
353
+ }
354
+
355
+ // ===================================================================
356
+ // Public API
357
+ // ===================================================================
358
+
359
+ var LuaWeightedHistory = {
360
+ // Core functions
361
+ getHistory: getHistory,
362
+ recordVisit: recordVisit,
363
+ getUserId: getUserId,
364
+ isReturningUser: isReturningUser,
365
+ getLastVisit: getLastVisit,
366
+ clearHistory: clearHistory,
367
+ // Weighted context
368
+ calculateWeight: calculateWeight,
369
+ buildWeightedContext: buildWeightedContext,
370
+ aggregatePreferences: aggregatePreferences,
371
+ formatForPrompt: formatForPrompt,
372
+ // Utilities
373
+ isLocalStorageAvailable: isLocalStorageAvailable,
374
+ generateUUID: generateUUID,
375
+ // Constants
376
+ STORAGE_KEY: STORAGE_KEY,
377
+ DEFAULT_DECAY_RATE: DEFAULT_DECAY_RATE,
378
+ DEFAULT_MAX_HISTORY: DEFAULT_MAX_HISTORY
379
+ };
380
+
381
+ // Register globally
382
+ root.LuaWeightedHistory = LuaWeightedHistory;
383
+ })(typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : void 0);
384
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["root","STORAGE_KEY","DEFAULT_DECAY_RATE","DEFAULT_MAX_HISTORY","DEFAULT_MAX_WEIGHTED","MS_PER_DAY","generateUUID","crypto","getRandomValues","buf","Uint8Array","hex","i","h","toString","length","substring","e","replace","c","r","Math","random","v","isLocalStorageAvailable","localStorage","testKey","setItem","removeItem","readFromStorage","raw","getItem","parsed","JSON","parse","_typeof2","default","Array","isArray","visits","console","warn","writeToStorage","history","stringify","calculateWeight","timestamp","decayRate","now","Date","daysAgo","max","pow","buildWeightedContext","options","maxWeighted","weighted","visit","weight","push","round","intent","selectedVariant","source","aiDecision","context","sort","a","b","slice","aggregatePreferences","weightedVisits","scores","getHistory","userId","createdAt","preferences","recordVisit","visitData","maxHistorySize","minimalContext","utm","referrer","category","userAgent","device","isMobile","isTablet","getUserId","isReturningUser","getLastVisit","formatForPrompt","lines","date","dateStr","toISOString","split","line","utm_source","join","clearHistory","LuaWeightedHistory","window","global"],"sources":["../../../src/storage/weighted-history.js"],"sourcesContent":["\n/**\n * Weighted History Manager\n * ========================\n * Manages localStorage for user visit history with exponential decay.\n * Tracks user visits with timestamps, UTM params, intents, and selected variants.\n * Applies configurable exponential decay so recent visits carry more weight.\n *\n * Storage key: 'lua_personalize_history'\n * Registers on window.LuaWeightedHistory\n *\n * No ES6 imports. Self-contained IIFE.\n */\n;(function (root) {\n    'use strict'\n\n    // ===================================================================\n    // Constants\n    // ===================================================================\n\n    var STORAGE_KEY = 'lua_personalize_history'\n    var DEFAULT_DECAY_RATE = 0.9\n    var DEFAULT_MAX_HISTORY = 10\n    var DEFAULT_MAX_WEIGHTED = 5\n    var MS_PER_DAY = 1000 * 60 * 60 * 24\n\n    // ===================================================================\n    // UUID Generator (simple, no crypto dependency)\n    // ===================================================================\n\n    /**\n     * Generate a simple UUID v4\n     * Uses crypto.getRandomValues when available, Math.random fallback\n     * @returns {string} - UUID string\n     */\n    function generateUUID() {\n        try {\n            if (typeof crypto !== 'undefined' && crypto.getRandomValues) {\n                var buf = new Uint8Array(16)\n                crypto.getRandomValues(buf)\n                buf[6] = (buf[6] & 0x0f) | 0x40\n                buf[8] = (buf[8] & 0x3f) | 0x80\n                var hex = ''\n                for (var i = 0; i < 16; i++) {\n                    var h = buf[i].toString(16)\n                    hex += h.length === 1 ? '0' + h : h\n                }\n                return (\n                    hex.substring(0, 8) + '-' +\n                    hex.substring(8, 12) + '-' +\n                    hex.substring(12, 16) + '-' +\n                    hex.substring(16, 20) + '-' +\n                    hex.substring(20, 32)\n                )\n            }\n        } catch (e) {\n            // Fallback below\n        }\n\n        // Math.random fallback\n        return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {\n            var r = (Math.random() * 16) | 0\n            var v = c === 'x' ? r : (r & 0x3) | 0x8\n            return v.toString(16)\n        })\n    }\n\n    // ===================================================================\n    // LocalStorage Helpers\n    // ===================================================================\n\n    /**\n     * Check if localStorage is available\n     * @returns {boolean}\n     */\n    function isLocalStorageAvailable() {\n        try {\n            if (typeof localStorage === 'undefined') return false\n            var testKey = '__lua_test__'\n            localStorage.setItem(testKey, '1')\n            localStorage.removeItem(testKey)\n            return true\n        } catch (e) {\n            return false\n        }\n    }\n\n    /**\n     * Read history from localStorage\n     * @returns {Object|null} - Parsed history object or null\n     */\n    function readFromStorage() {\n        if (!isLocalStorageAvailable()) return null\n        try {\n            var raw = localStorage.getItem(STORAGE_KEY)\n            if (!raw) return null\n            var parsed = JSON.parse(raw)\n            if (parsed && typeof parsed === 'object' && Array.isArray(parsed.visits)) {\n                return parsed\n            }\n            return null\n        } catch (e) {\n            console.warn('[Lua History] Failed to read history:', e)\n            return null\n        }\n    }\n\n    /**\n     * Write history to localStorage\n     * @param {Object} history - History object to persist\n     * @returns {boolean} - Whether the write was successful\n     */\n    function writeToStorage(history) {\n        if (!isLocalStorageAvailable()) return false\n        try {\n            localStorage.setItem(STORAGE_KEY, JSON.stringify(history))\n            return true\n        } catch (e) {\n            console.warn('[Lua History] Failed to write history:', e)\n            return false\n        }\n    }\n\n    // ===================================================================\n    // Weighted Decay Functions\n    // ===================================================================\n\n    /**\n     * Calculate the weight of a visit based on its age using exponential decay\n     * Weight = decayRate ^ daysAgo\n     * Example: decayRate=0.9, 1 day ago = 0.9, 7 days ago = 0.478, 30 days ago = 0.042\n     *\n     * @param {number} timestamp - Visit timestamp in ms\n     * @param {number} [decayRate] - Decay rate per day (0-1, default: 0.9)\n     * @returns {number} - Weight between 0 and 1\n     */\n    function calculateWeight(timestamp, decayRate) {\n        decayRate = typeof decayRate === 'number' ? decayRate : DEFAULT_DECAY_RATE\n        var now = Date.now()\n        var daysAgo = Math.max(0, (now - timestamp) / MS_PER_DAY)\n        return Math.pow(decayRate, daysAgo)\n    }\n\n    /**\n     * Build weighted context from visit history\n     * Returns the most relevant visits sorted by weight (highest first)\n     *\n     * @param {Object} history - History object with visits array\n     * @param {Object} [options] - Configuration\n     * @param {number} [options.decayRate] - Decay rate (default: 0.9)\n     * @param {number} [options.maxWeighted] - Max results to return (default: 5)\n     * @returns {Array} - Weighted visits sorted by relevance\n     */\n    function buildWeightedContext(history, options) {\n        options = options || {}\n        var decayRate = options.decayRate || DEFAULT_DECAY_RATE\n        var maxWeighted = options.maxWeighted || DEFAULT_MAX_WEIGHTED\n\n        if (!history || !Array.isArray(history.visits) || history.visits.length === 0) {\n            return []\n        }\n\n        var weighted = []\n\n        for (var i = 0; i < history.visits.length; i++) {\n            var visit = history.visits[i]\n            var weight = calculateWeight(visit.timestamp, decayRate)\n\n            weighted.push({\n                timestamp: visit.timestamp,\n                weight: Math.round(weight * 1000) / 1000,\n                intent: visit.intent || 'unknown',\n                selectedVariant: visit.selectedVariant || null,\n                source: visit.source || 'unknown',\n                aiDecision: visit.aiDecision || false,\n                context: visit.context || {}\n            })\n        }\n\n        // Sort by weight descending (most recent / relevant first)\n        weighted.sort(function (a, b) {\n            return b.weight - a.weight\n        })\n\n        // Return top N results\n        return weighted.slice(0, maxWeighted)\n    }\n\n    /**\n     * Aggregate intent preferences from weighted history\n     * Produces a score map of intents weighted by recency\n     *\n     * @param {Array} weightedVisits - Output of buildWeightedContext\n     * @returns {Object} - Intent -> score mapping\n     */\n    function aggregatePreferences(weightedVisits) {\n        var scores = {}\n\n        for (var i = 0; i < weightedVisits.length; i++) {\n            var visit = weightedVisits[i]\n            var intent = visit.intent\n            if (!intent || intent === 'unknown' || intent === 'default') continue\n\n            if (!scores[intent]) {\n                scores[intent] = 0\n            }\n            scores[intent] += visit.weight\n        }\n\n        return scores\n    }\n\n    // ===================================================================\n    // History Management\n    // ===================================================================\n\n    /**\n     * Initialize or retrieve existing history\n     * Creates a new history object with a UUID if none exists\n     *\n     * @returns {Object} - History object\n     */\n    function getHistory() {\n        var history = readFromStorage()\n\n        if (!history) {\n            history = {\n                userId: generateUUID(),\n                createdAt: Date.now(),\n                visits: [],\n                preferences: {}\n            }\n            writeToStorage(history)\n        }\n\n        return history\n    }\n\n    /**\n     * Record a new visit to history\n     *\n     * @param {Object} visitData - Visit data to record\n     * @param {Object} visitData.context - UTM context (utm, referrer, userAgent)\n     * @param {string} visitData.intent - Inferred or AI-selected intent\n     * @param {string} visitData.selectedVariant - Selected template/variant key\n     * @param {string} visitData.source - Decision source ('ai', 'utm', 'referrer', 'random-ab', etc.)\n     * @param {boolean} [visitData.aiDecision] - Whether AI made this decision\n     * @param {Object} [options] - Configuration\n     * @param {number} [options.maxHistorySize] - Max visits to keep (default: 10)\n     * @returns {Object} - Updated history object\n     */\n    function recordVisit(visitData, options) {\n        options = options || {}\n        var maxHistorySize = options.maxHistorySize || DEFAULT_MAX_HISTORY\n\n        var history = getHistory()\n\n        // Build a minimal context to store (avoid storing sensitive/large data)\n        var minimalContext = {}\n        if (visitData.context) {\n            if (visitData.context.utm) {\n                minimalContext.utm = visitData.context.utm\n            }\n            if (visitData.context.referrer) {\n                minimalContext.referrer = {\n                    source: visitData.context.referrer.source,\n                    category: visitData.context.referrer.category\n                }\n            }\n            if (visitData.context.userAgent) {\n                minimalContext.device = visitData.context.userAgent.isMobile\n                    ? 'mobile'\n                    : visitData.context.userAgent.isTablet\n                        ? 'tablet'\n                        : 'desktop'\n            }\n        }\n\n        var visit = {\n            timestamp: Date.now(),\n            context: minimalContext,\n            intent: visitData.intent || 'unknown',\n            selectedVariant: visitData.selectedVariant || null,\n            source: visitData.source || 'unknown',\n            aiDecision: !!visitData.aiDecision\n        }\n\n        history.visits.push(visit)\n\n        // Trim to max size (keep most recent)\n        if (history.visits.length > maxHistorySize) {\n            history.visits = history.visits.slice(-maxHistorySize)\n        }\n\n        // Update aggregated preferences\n        var weighted = buildWeightedContext(history, options)\n        history.preferences = aggregatePreferences(weighted)\n\n        writeToStorage(history)\n\n        return history\n    }\n\n    /**\n     * Get the user ID from history (creates one if needed)\n     * @returns {string} - User UUID\n     */\n    function getUserId() {\n        var history = getHistory()\n        return history.userId\n    }\n\n    /**\n     * Check if this is a returning user (has previous visits)\n     * @returns {boolean}\n     */\n    function isReturningUser() {\n        var history = readFromStorage()\n        return history !== null && Array.isArray(history.visits) && history.visits.length > 0\n    }\n\n    /**\n     * Get the most recent visit from history\n     * @returns {Object|null} - Most recent visit or null\n     */\n    function getLastVisit() {\n        var history = readFromStorage()\n        if (!history || !Array.isArray(history.visits) || history.visits.length === 0) {\n            return null\n        }\n        return history.visits[history.visits.length - 1]\n    }\n\n    /**\n     * Format weighted history as a human-readable string for AI prompts\n     * @param {Array} weightedVisits - Output of buildWeightedContext\n     * @returns {string} - Formatted string\n     */\n    function formatForPrompt(weightedVisits) {\n        if (!weightedVisits || weightedVisits.length === 0) {\n            return 'No previous visit history available. This is a new visitor.'\n        }\n\n        var lines = []\n        lines.push('Previous visits (' + weightedVisits.length + ' recorded, weighted by recency):')\n\n        for (var i = 0; i < weightedVisits.length; i++) {\n            var v = weightedVisits[i]\n            var date = new Date(v.timestamp)\n            var dateStr = date.toISOString().split('T')[0]\n            var line = '  - [' + dateStr + '] '\n            line += 'Intent: ' + v.intent\n            if (v.selectedVariant) line += ', Variant: ' + v.selectedVariant\n            line += ', Source: ' + v.source\n            line += ', Weight: ' + v.weight\n            if (v.context && v.context.utm && v.context.utm.utm_source) {\n                line += ', UTM: ' + v.context.utm.utm_source\n            }\n            lines.push(line)\n        }\n\n        return lines.join('\\n')\n    }\n\n    /**\n     * Clear all history (useful for testing or user request)\n     * @returns {boolean} - Whether the clear was successful\n     */\n    function clearHistory() {\n        if (!isLocalStorageAvailable()) return false\n        try {\n            localStorage.removeItem(STORAGE_KEY)\n            return true\n        } catch (e) {\n            return false\n        }\n    }\n\n    // ===================================================================\n    // Public API\n    // ===================================================================\n\n    var LuaWeightedHistory = {\n        // Core functions\n        getHistory: getHistory,\n        recordVisit: recordVisit,\n        getUserId: getUserId,\n        isReturningUser: isReturningUser,\n        getLastVisit: getLastVisit,\n        clearHistory: clearHistory,\n\n        // Weighted context\n        calculateWeight: calculateWeight,\n        buildWeightedContext: buildWeightedContext,\n        aggregatePreferences: aggregatePreferences,\n        formatForPrompt: formatForPrompt,\n\n        // Utilities\n        isLocalStorageAvailable: isLocalStorageAvailable,\n        generateUUID: generateUUID,\n\n        // Constants\n        STORAGE_KEY: STORAGE_KEY,\n        DEFAULT_DECAY_RATE: DEFAULT_DECAY_RATE,\n        DEFAULT_MAX_HISTORY: DEFAULT_MAX_HISTORY\n    }\n\n    // Register globally\n    root.LuaWeightedHistory = LuaWeightedHistory\n\n})(typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : this)\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ // NOTE: use a module
8
+ var _default = exports.default = function _default() {
9
+ return {
10
+ type: 'browserCookie',
11
+ /*eslint-disable */
12
+ get: function get(key) {
13
+ return decodeURIComponent(document.cookie.replace(new RegExp("(?:(?:^|.*;)\\s*" + encodeURIComponent(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=\\s*([^;]*).*$)|^.*$"), "$1")) || null;
14
+ },
15
+ set: function set(key, val) {
16
+ var expirationDate = new Date('12/31/9999').toUTCString();
17
+ document.cookie = "".concat(encodeURIComponent(key), "=").concat(encodeURIComponent(val), "; expires=").concat(expirationDate, "; path=/");
18
+ },
19
+ /* eslint-enable */
20
+ isSupported: function isSupported() {
21
+ return typeof document !== 'undefined';
22
+ }
23
+ };
24
+ };
25
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfZGVmYXVsdCIsImV4cG9ydHMiLCJkZWZhdWx0IiwidHlwZSIsImdldCIsImtleSIsImRlY29kZVVSSUNvbXBvbmVudCIsImRvY3VtZW50IiwiY29va2llIiwicmVwbGFjZSIsIlJlZ0V4cCIsImVuY29kZVVSSUNvbXBvbmVudCIsInNldCIsInZhbCIsImV4cGlyYXRpb25EYXRlIiwiRGF0ZSIsInRvVVRDU3RyaW5nIiwiY29uY2F0IiwiaXNTdXBwb3J0ZWQiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc3RvcmVzL2Jyb3dzZXItY29va2llLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIlxuLy8gTk9URTogdXNlIGEgbW9kdWxlXG5leHBvcnQgZGVmYXVsdCAoKSA9PiAoe1xuICB0eXBlOiAnYnJvd3NlckNvb2tpZScsXG4gIC8qZXNsaW50LWRpc2FibGUgKi9cbiAgZ2V0OiBrZXkgPT4gZGVjb2RlVVJJQ29tcG9uZW50KGRvY3VtZW50LmNvb2tpZS5yZXBsYWNlKG5ldyBSZWdFeHAoXCIoPzooPzpefC4qOylcXFxccypcIiArIGVuY29kZVVSSUNvbXBvbmVudChrZXkpLnJlcGxhY2UoL1tcXC1cXC5cXCtcXCpdL2csIFwiXFxcXCQmXCIpICsgXCJcXFxccypcXFxcPVxcXFxzKihbXjtdKikuKiQpfF4uKiRcIiksIFwiJDFcIikpIHx8IG51bGwsXG4gIHNldDogKGtleSwgdmFsKSA9PiB7XG4gICAgY29uc3QgZXhwaXJhdGlvbkRhdGUgPSBuZXcgRGF0ZSgnMTIvMzEvOTk5OScpLnRvVVRDU3RyaW5nKClcbiAgICBkb2N1bWVudC5jb29raWUgPSBgJHtlbmNvZGVVUklDb21wb25lbnQoa2V5KX09JHtlbmNvZGVVUklDb21wb25lbnQodmFsKX07IGV4cGlyZXM9JHtleHBpcmF0aW9uRGF0ZX07IHBhdGg9L2BcbiAgfSxcbiAgLyogZXNsaW50LWVuYWJsZSAqL1xuICBpc1N1cHBvcnRlZDogKCkgPT4gdHlwZW9mIGRvY3VtZW50ICE9PSAndW5kZWZpbmVkJyxcbn0pXG4iXSwibWFwcGluZ3MiOiI7Ozs7OztBQUNBO0FBQUEsSUFBQUEsUUFBQSxHQUFBQyxPQUFBLENBQUFDLE9BQUEsR0FDZSxTQUFBRixTQUFBO0VBQUEsT0FBTztJQUNwQkcsSUFBSSxFQUFFLGVBQWU7SUFDckI7SUFDQUMsR0FBRyxFQUFFLFNBQUxBLEdBQUdBLENBQUVDLEdBQUc7TUFBQSxPQUFJQyxrQkFBa0IsQ0FBQ0MsUUFBUSxDQUFDQyxNQUFNLENBQUNDLE9BQU8sQ0FBQyxJQUFJQyxNQUFNLENBQUMsa0JBQWtCLEdBQUdDLGtCQUFrQixDQUFDTixHQUFHLENBQUMsQ0FBQ0ksT0FBTyxDQUFDLGFBQWEsRUFBRSxNQUFNLENBQUMsR0FBRyw2QkFBNkIsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDLElBQUksSUFBSTtJQUFBO0lBQzlMRyxHQUFHLEVBQUUsU0FBTEEsR0FBR0EsQ0FBR1AsR0FBRyxFQUFFUSxHQUFHLEVBQUs7TUFDakIsSUFBTUMsY0FBYyxHQUFHLElBQUlDLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQ0MsV0FBVyxDQUFDLENBQUM7TUFDM0RULFFBQVEsQ0FBQ0MsTUFBTSxNQUFBUyxNQUFBLENBQU1OLGtCQUFrQixDQUFDTixHQUFHLENBQUMsT0FBQVksTUFBQSxDQUFJTixrQkFBa0IsQ0FBQ0UsR0FBRyxDQUFDLGdCQUFBSSxNQUFBLENBQWFILGNBQWMsYUFBVTtJQUM5RyxDQUFDO0lBQ0Q7SUFDQUksV0FBVyxFQUFFLFNBQWJBLFdBQVdBLENBQUE7TUFBQSxPQUFRLE9BQU9YLFFBQVEsS0FBSyxXQUFXO0lBQUE7RUFDcEQsQ0FBQztBQUFBLENBQUMiLCJpZ25vcmVMaXN0IjpbXX0=
@@ -0,0 +1,29 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _default = exports.default = function _default() {
8
+ return {
9
+ type: 'local',
10
+ get: function get(key) {
11
+ return localStorage.getItem(key);
12
+ },
13
+ set: function set(key, val) {
14
+ return localStorage.setItem(key, val);
15
+ },
16
+ isSupported: function isSupported() {
17
+ if (typeof localStorage !== 'undefined') return true;
18
+ var uid = new Date();
19
+ try {
20
+ localStorage.setItem(uid, uid);
21
+ localStorage.removeItem(uid);
22
+ return true;
23
+ } catch (e) {
24
+ return false;
25
+ }
26
+ }
27
+ };
28
+ };
29
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfZGVmYXVsdCIsInR5cGUiLCJnZXQiLCJrZXkiLCJsb2NhbFN0b3JhZ2UiLCJnZXRJdGVtIiwic2V0IiwidmFsIiwic2V0SXRlbSIsImlzU3VwcG9ydGVkIiwidWlkIiwiRGF0ZSIsInJlbW92ZUl0ZW0iLCJlIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0b3Jlcy9sb2NhbC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJcbmV4cG9ydCBkZWZhdWx0ICgpID0+ICh7XG4gIHR5cGU6ICdsb2NhbCcsXG4gIGdldDoga2V5ID0+IGxvY2FsU3RvcmFnZS5nZXRJdGVtKGtleSksXG4gIHNldDogKGtleSwgdmFsKSA9PiBsb2NhbFN0b3JhZ2Uuc2V0SXRlbShrZXksIHZhbCksXG4gIGlzU3VwcG9ydGVkOiAoKSA9PiB7XG4gICAgaWYgKHR5cGVvZiBsb2NhbFN0b3JhZ2UgIT09ICd1bmRlZmluZWQnKSByZXR1cm4gdHJ1ZVxuICAgIGNvbnN0IHVpZCA9IG5ldyBEYXRlKClcbiAgICB0cnkge1xuICAgICAgbG9jYWxTdG9yYWdlLnNldEl0ZW0odWlkLCB1aWQpXG4gICAgICBsb2NhbFN0b3JhZ2UucmVtb3ZlSXRlbSh1aWQpXG4gICAgICByZXR1cm4gdHJ1ZVxuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cbiAgfSxcbn0pXG4iXSwibWFwcGluZ3MiOiI7Ozs7OztpQ0FDZSxTQUFBQSxTQUFBO0VBQUEsT0FBTztJQUNwQkMsSUFBSSxFQUFFLE9BQU87SUFDYkMsR0FBRyxFQUFFLFNBQUxBLEdBQUdBLENBQUVDLEdBQUc7TUFBQSxPQUFJQyxZQUFZLENBQUNDLE9BQU8sQ0FBQ0YsR0FBRyxDQUFDO0lBQUE7SUFDckNHLEdBQUcsRUFBRSxTQUFMQSxHQUFHQSxDQUFHSCxHQUFHLEVBQUVJLEdBQUc7TUFBQSxPQUFLSCxZQUFZLENBQUNJLE9BQU8sQ0FBQ0wsR0FBRyxFQUFFSSxHQUFHLENBQUM7SUFBQTtJQUNqREUsV0FBVyxFQUFFLFNBQWJBLFdBQVdBLENBQUEsRUFBUTtNQUNqQixJQUFJLE9BQU9MLFlBQVksS0FBSyxXQUFXLEVBQUUsT0FBTyxJQUFJO01BQ3BELElBQU1NLEdBQUcsR0FBRyxJQUFJQyxJQUFJLENBQUMsQ0FBQztNQUN0QixJQUFJO1FBQ0ZQLFlBQVksQ0FBQ0ksT0FBTyxDQUFDRSxHQUFHLEVBQUVBLEdBQUcsQ0FBQztRQUM5Qk4sWUFBWSxDQUFDUSxVQUFVLENBQUNGLEdBQUcsQ0FBQztRQUM1QixPQUFPLElBQUk7TUFDYixDQUFDLENBQUMsT0FBT0csQ0FBQyxFQUFFO1FBQ1YsT0FBTyxLQUFLO01BQ2Q7SUFDRjtFQUNGLENBQUM7QUFBQSxDQUFDIiwiaWdub3JlTGlzdCI6W119
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _default = exports.default = function _default() {
8
+ var store = Object.create(null);
9
+ return {
10
+ type: 'memory',
11
+ get: function get(key) {
12
+ return store[key];
13
+ },
14
+ set: function set(key, val) {
15
+ store[key] = val;
16
+ },
17
+ isSupported: function isSupported() {
18
+ return true;
19
+ }
20
+ };
21
+ };
22
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfZGVmYXVsdCIsInN0b3JlIiwiT2JqZWN0IiwiY3JlYXRlIiwidHlwZSIsImdldCIsImtleSIsInNldCIsInZhbCIsImlzU3VwcG9ydGVkIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0b3Jlcy9tZW1vcnkuanMiXSwic291cmNlc0NvbnRlbnQiOlsiXG5leHBvcnQgZGVmYXVsdCAoKSA9PiB7XG4gIGNvbnN0IHN0b3JlID0gT2JqZWN0LmNyZWF0ZShudWxsKVxuXG4gIHJldHVybiB7XG4gICAgdHlwZTogJ21lbW9yeScsXG4gICAgZ2V0OiBrZXkgPT4gc3RvcmVba2V5XSxcbiAgICBzZXQ6IChrZXksIHZhbCkgPT4ge1xuICAgICAgc3RvcmVba2V5XSA9IHZhbFxuICAgIH0sXG4gICAgaXNTdXBwb3J0ZWQ6ICgpID0+IHRydWUsXG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O2lDQUNlLFNBQUFBLFNBQUEsRUFBTTtFQUNuQixJQUFNQyxLQUFLLEdBQUdDLE1BQU0sQ0FBQ0MsTUFBTSxDQUFDLElBQUksQ0FBQztFQUVqQyxPQUFPO0lBQ0xDLElBQUksRUFBRSxRQUFRO0lBQ2RDLEdBQUcsRUFBRSxTQUFMQSxHQUFHQSxDQUFFQyxHQUFHO01BQUEsT0FBSUwsS0FBSyxDQUFDSyxHQUFHLENBQUM7SUFBQTtJQUN0QkMsR0FBRyxFQUFFLFNBQUxBLEdBQUdBLENBQUdELEdBQUcsRUFBRUUsR0FBRyxFQUFLO01BQ2pCUCxLQUFLLENBQUNLLEdBQUcsQ0FBQyxHQUFHRSxHQUFHO0lBQ2xCLENBQUM7SUFDREMsV0FBVyxFQUFFLFNBQWJBLFdBQVdBLENBQUE7TUFBQSxPQUFRLElBQUk7SUFBQTtFQUN6QixDQUFDO0FBQ0gsQ0FBQyIsImlnbm9yZUxpc3QiOltdfQ==
@@ -0,0 +1,54 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.validateStore = exports.rand = exports.getRandomAssignment = exports.getDefaultBucket = exports.chooseWeightedItem = void 0;
7
+ var rand = exports.rand = function rand(min, max) {
8
+ return Math.random() * (max - min) + min;
9
+ };
10
+
11
+ // choose a random value with the specified weights
12
+ var chooseWeightedItem = exports.chooseWeightedItem = function chooseWeightedItem(names, weights) {
13
+ if (names.length !== weights.length) throw new Error('names and weights must have equal length!');
14
+ var sum = weights.reduce(function (a, b) {
15
+ return a + b;
16
+ }, 0);
17
+ var limit = 0;
18
+ var n = rand(0, sum);
19
+ for (var i = 0; i < names.length; i++) {
20
+ limit += weights[i];
21
+ if (n <= limit) return names[i];
22
+ }
23
+ // by default, return the last weight
24
+ return names[names.length - 1];
25
+ };
26
+
27
+ // get the default bucket,
28
+ // which is either the default/winner,
29
+ // otherwise whichever is returned first
30
+ var getDefaultBucket = exports.getDefaultBucket = function getDefaultBucket(buckets) {
31
+ var defaultBuckets = Object.keys(buckets).filter(function (name) {
32
+ var x = buckets[name];
33
+ return x.default || x.winner;
34
+ });
35
+ return defaultBuckets[0] || Object.keys(buckets)[0];
36
+ };
37
+ var validateStore = exports.validateStore = function validateStore(store) {
38
+ if (!store) throw new Error('You must supply a store!');
39
+ if (typeof store.get !== 'function') throw new Error('The store must implement .get()');
40
+ if (typeof store.set !== 'function') throw new Error('The store must implement .set()');
41
+ if (typeof store.isSupported !== 'function') throw new Error('The store must implement .isSupported()');
42
+ if (!store.isSupported()) throw new Error('The store is not supported.');
43
+ };
44
+ var getRandomAssignment = exports.getRandomAssignment = function getRandomAssignment(test) {
45
+ var names = Object.keys(test.buckets);
46
+ var weights = [];
47
+ names.forEach(function (innerBucketName) {
48
+ var weight = test.buckets[innerBucketName].weight;
49
+ if (weight == null) weight = 1;
50
+ weights.push(weight);
51
+ });
52
+ return chooseWeightedItem(names, weights);
53
+ };
54
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJyYW5kIiwiZXhwb3J0cyIsIm1pbiIsIm1heCIsIk1hdGgiLCJyYW5kb20iLCJjaG9vc2VXZWlnaHRlZEl0ZW0iLCJuYW1lcyIsIndlaWdodHMiLCJsZW5ndGgiLCJFcnJvciIsInN1bSIsInJlZHVjZSIsImEiLCJiIiwibGltaXQiLCJuIiwiaSIsImdldERlZmF1bHRCdWNrZXQiLCJidWNrZXRzIiwiZGVmYXVsdEJ1Y2tldHMiLCJPYmplY3QiLCJrZXlzIiwiZmlsdGVyIiwibmFtZSIsIngiLCJkZWZhdWx0Iiwid2lubmVyIiwidmFsaWRhdGVTdG9yZSIsInN0b3JlIiwiZ2V0Iiwic2V0IiwiaXNTdXBwb3J0ZWQiLCJnZXRSYW5kb21Bc3NpZ25tZW50IiwidGVzdCIsImZvckVhY2giLCJpbm5lckJ1Y2tldE5hbWUiLCJ3ZWlnaHQiLCJwdXNoIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3V0aWxzLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIlxuZXhwb3J0IGNvbnN0IHJhbmQgPSAobWluLCBtYXgpID0+IChNYXRoLnJhbmRvbSgpICogKG1heCAtIG1pbikpICsgbWluXG5cbi8vIGNob29zZSBhIHJhbmRvbSB2YWx1ZSB3aXRoIHRoZSBzcGVjaWZpZWQgd2VpZ2h0c1xuZXhwb3J0IGNvbnN0IGNob29zZVdlaWdodGVkSXRlbSA9IChuYW1lcywgd2VpZ2h0cykgPT4ge1xuICBpZiAobmFtZXMubGVuZ3RoICE9PSB3ZWlnaHRzLmxlbmd0aCkgdGhyb3cgbmV3IEVycm9yKCduYW1lcyBhbmQgd2VpZ2h0cyBtdXN0IGhhdmUgZXF1YWwgbGVuZ3RoIScpXG4gIGNvbnN0IHN1bSA9IHdlaWdodHMucmVkdWNlKChhLCBiKSA9PiBhICsgYiwgMClcbiAgbGV0IGxpbWl0ID0gMFxuICBjb25zdCBuID0gcmFuZCgwLCBzdW0pXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgbmFtZXMubGVuZ3RoOyBpKyspIHtcbiAgICBsaW1pdCArPSB3ZWlnaHRzW2ldXG4gICAgaWYgKG4gPD0gbGltaXQpIHJldHVybiBuYW1lc1tpXVxuICB9XG4gIC8vIGJ5IGRlZmF1bHQsIHJldHVybiB0aGUgbGFzdCB3ZWlnaHRcbiAgcmV0dXJuIG5hbWVzW25hbWVzLmxlbmd0aCAtIDFdXG59XG5cbi8vIGdldCB0aGUgZGVmYXVsdCBidWNrZXQsXG4vLyB3aGljaCBpcyBlaXRoZXIgdGhlIGRlZmF1bHQvd2lubmVyLFxuLy8gb3RoZXJ3aXNlIHdoaWNoZXZlciBpcyByZXR1cm5lZCBmaXJzdFxuZXhwb3J0IGNvbnN0IGdldERlZmF1bHRCdWNrZXQgPSAoYnVja2V0cykgPT4ge1xuICBjb25zdCBkZWZhdWx0QnVja2V0cyA9IE9iamVjdC5rZXlzKGJ1Y2tldHMpLmZpbHRlcigobmFtZSkgPT4ge1xuICAgIGNvbnN0IHggPSBidWNrZXRzW25hbWVdXG4gICAgcmV0dXJuIHguZGVmYXVsdCB8fCB4Lndpbm5lclxuICB9KVxuICByZXR1cm4gZGVmYXVsdEJ1Y2tldHNbMF0gfHwgT2JqZWN0LmtleXMoYnVja2V0cylbMF1cbn1cblxuZXhwb3J0IGNvbnN0IHZhbGlkYXRlU3RvcmUgPSAoc3RvcmUpID0+IHtcbiAgaWYgKCFzdG9yZSkgdGhyb3cgbmV3IEVycm9yKCdZb3UgbXVzdCBzdXBwbHkgYSBzdG9yZSEnKVxuICBpZiAodHlwZW9mIHN0b3JlLmdldCAhPT0gJ2Z1bmN0aW9uJykgdGhyb3cgbmV3IEVycm9yKCdUaGUgc3RvcmUgbXVzdCBpbXBsZW1lbnQgLmdldCgpJylcbiAgaWYgKHR5cGVvZiBzdG9yZS5zZXQgIT09ICdmdW5jdGlvbicpIHRocm93IG5ldyBFcnJvcignVGhlIHN0b3JlIG11c3QgaW1wbGVtZW50IC5zZXQoKScpXG4gIGlmICh0eXBlb2Ygc3RvcmUuaXNTdXBwb3J0ZWQgIT09ICdmdW5jdGlvbicpIHRocm93IG5ldyBFcnJvcignVGhlIHN0b3JlIG11c3QgaW1wbGVtZW50IC5pc1N1cHBvcnRlZCgpJylcbiAgaWYgKCFzdG9yZS5pc1N1cHBvcnRlZCgpKSB0aHJvdyBuZXcgRXJyb3IoJ1RoZSBzdG9yZSBpcyBub3Qgc3VwcG9ydGVkLicpXG59XG5cbmV4cG9ydCBjb25zdCBnZXRSYW5kb21Bc3NpZ25tZW50ID0gKHRlc3QpID0+IHtcbiAgY29uc3QgbmFtZXMgPSBPYmplY3Qua2V5cyh0ZXN0LmJ1Y2tldHMpXG4gIGNvbnN0IHdlaWdodHMgPSBbXVxuXG4gIG5hbWVzLmZvckVhY2goKGlubmVyQnVja2V0TmFtZSkgPT4ge1xuICAgIGxldCB3ZWlnaHQgPSB0ZXN0LmJ1Y2tldHNbaW5uZXJCdWNrZXROYW1lXS53ZWlnaHRcbiAgICBpZiAod2VpZ2h0ID09IG51bGwpIHdlaWdodCA9IDFcbiAgICB3ZWlnaHRzLnB1c2god2VpZ2h0KVxuICB9KVxuXG4gIHJldHVybiBjaG9vc2VXZWlnaHRlZEl0ZW0obmFtZXMsIHdlaWdodHMpXG59XG4iXSwibWFwcGluZ3MiOiI7Ozs7OztBQUNPLElBQU1BLElBQUksR0FBQUMsT0FBQSxDQUFBRCxJQUFBLEdBQUcsU0FBUEEsSUFBSUEsQ0FBSUUsR0FBRyxFQUFFQyxHQUFHO0VBQUEsT0FBTUMsSUFBSSxDQUFDQyxNQUFNLENBQUMsQ0FBQyxJQUFJRixHQUFHLEdBQUdELEdBQUcsQ0FBQyxHQUFJQSxHQUFHO0FBQUE7O0FBRXJFO0FBQ08sSUFBTUksa0JBQWtCLEdBQUFMLE9BQUEsQ0FBQUssa0JBQUEsR0FBRyxTQUFyQkEsa0JBQWtCQSxDQUFJQyxLQUFLLEVBQUVDLE9BQU8sRUFBSztFQUNwRCxJQUFJRCxLQUFLLENBQUNFLE1BQU0sS0FBS0QsT0FBTyxDQUFDQyxNQUFNLEVBQUUsTUFBTSxJQUFJQyxLQUFLLENBQUMsMkNBQTJDLENBQUM7RUFDakcsSUFBTUMsR0FBRyxHQUFHSCxPQUFPLENBQUNJLE1BQU0sQ0FBQyxVQUFDQyxDQUFDLEVBQUVDLENBQUM7SUFBQSxPQUFLRCxDQUFDLEdBQUdDLENBQUM7RUFBQSxHQUFFLENBQUMsQ0FBQztFQUM5QyxJQUFJQyxLQUFLLEdBQUcsQ0FBQztFQUNiLElBQU1DLENBQUMsR0FBR2hCLElBQUksQ0FBQyxDQUFD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