@xiboplayer/pwa 0.6.2 → 0.6.3

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.
Files changed (46) hide show
  1. package/README.md +1 -1
  2. package/dist/assets/{chunk-config-BjXWw-V8.js → chunk-config-BqCdX6OJ.js} +2 -2
  3. package/dist/assets/{chunk-config-BjXWw-V8.js.map → chunk-config-BqCdX6OJ.js.map} +1 -1
  4. package/dist/assets/{index-D4fqdfS-.js → index-BHPUGsQU.js} +2 -2
  5. package/dist/assets/{index-D4fqdfS-.js.map → index-BHPUGsQU.js.map} +1 -1
  6. package/dist/assets/{index-FG4fTI4i.js → index-BS70-4bE.js} +2 -2
  7. package/dist/assets/{index-FG4fTI4i.js.map → index-BS70-4bE.js.map} +1 -1
  8. package/dist/assets/{index-CwIMcVP6.js → index-BftPXCfG.js} +2 -2
  9. package/dist/assets/{index-CwIMcVP6.js.map → index-BftPXCfG.js.map} +1 -1
  10. package/dist/assets/{index-BglkPErZ.js → index-C122u38X.js} +2 -2
  11. package/dist/assets/{index-BglkPErZ.js.map → index-C122u38X.js.map} +1 -1
  12. package/dist/assets/{index-D9EQ-6Wd.js → index-C9mU5kWi.js} +2 -2
  13. package/dist/assets/{index-D9EQ-6Wd.js.map → index-C9mU5kWi.js.map} +1 -1
  14. package/dist/assets/index-CThL7HEa.js +2 -0
  15. package/dist/assets/{index-IEsktHaW.js.map → index-CThL7HEa.js.map} +1 -1
  16. package/dist/assets/{index-so_tPJrr.js → index-DTsYjcZZ.js} +2 -2
  17. package/dist/assets/{index-so_tPJrr.js.map → index-DTsYjcZZ.js.map} +1 -1
  18. package/dist/assets/{index-C0mqgU51.js → index-H79-vv8p.js} +2 -2
  19. package/dist/assets/{index-C0mqgU51.js.map → index-H79-vv8p.js.map} +1 -1
  20. package/dist/assets/index-gRI8sFgQ.js +2 -0
  21. package/dist/assets/{index-DnB9w-Dv.js.map → index-gRI8sFgQ.js.map} +1 -1
  22. package/dist/assets/index-xcL2piI3.js +2 -0
  23. package/dist/assets/index-xcL2piI3.js.map +1 -0
  24. package/dist/assets/main-DGfrcKkR.js +735 -0
  25. package/dist/assets/main-DGfrcKkR.js.map +1 -0
  26. package/dist/assets/protocol-detector-bDP-d2EL.js +16 -0
  27. package/dist/assets/protocol-detector-bDP-d2EL.js.map +1 -0
  28. package/dist/assets/setup-D3frvQ4W.js +2 -0
  29. package/dist/assets/{setup-BXLaipeI.js.map → setup-D3frvQ4W.js.map} +1 -1
  30. package/dist/assets/{sync-manager-DMpg8tED.js → sync-manager-ljAFKrR5.js} +2 -2
  31. package/dist/assets/{sync-manager-DMpg8tED.js.map → sync-manager-ljAFKrR5.js.map} +1 -1
  32. package/dist/assets/{widget-html-D6xqg8Sp.js → widget-html-tM9V3Qlc.js} +2 -2
  33. package/dist/assets/{widget-html-D6xqg8Sp.js.map → widget-html-tM9V3Qlc.js.map} +1 -1
  34. package/dist/index.html +3 -3
  35. package/dist/setup.html +3 -3
  36. package/dist/sw-pwa.js +1 -1
  37. package/package.json +13 -13
  38. package/dist/assets/index-BwkVsox3.js +0 -2
  39. package/dist/assets/index-BwkVsox3.js.map +0 -1
  40. package/dist/assets/index-DnB9w-Dv.js +0 -2
  41. package/dist/assets/index-IEsktHaW.js +0 -2
  42. package/dist/assets/main-DJYbit7N.js +0 -729
  43. package/dist/assets/main-DJYbit7N.js.map +0 -1
  44. package/dist/assets/setup-BXLaipeI.js +0 -2
  45. package/dist/assets/xmds-client-D6JI3K3_.js +0 -16
  46. package/dist/assets/xmds-client-D6JI3K3_.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"index-CwIMcVP6.js","sources":["../../../stats/src/stats-collector.js","../../../stats/src/log-reporter.js","../../../stats/src/index.js"],"sourcesContent":["/**\n * StatsCollector - Proof of play tracking for Xibo CMS\n *\n * Tracks layout and widget playback for reporting to CMS via XMDS.\n * Uses IndexedDB for persistent storage across sessions.\n *\n * @module @xiboplayer/stats/collector\n */\n\nimport { createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('@xiboplayer/stats');\n\n// IndexedDB configuration\nconst DB_NAME = 'xibo-player-stats';\nconst DB_VERSION = 1;\nconst STATS_STORE = 'stats';\n\n/**\n * Stats collector for proof of play tracking\n *\n * Stores layout and widget playback statistics in IndexedDB.\n * Stats are submitted to CMS via XMDS SubmitStats API.\n *\n * @example\n * const collector = new StatsCollector();\n * await collector.init();\n *\n * // Track layout\n * await collector.startLayout(123, 456);\n * // ... layout plays ...\n * await collector.endLayout(123, 456);\n *\n * // Get stats for submission\n * const stats = await collector.getStatsForSubmission(50);\n * const xml = formatStats(stats);\n * // ... submit to CMS ...\n * await collector.clearSubmittedStats(stats);\n */\nexport class StatsCollector {\n constructor() {\n this.db = null;\n this.inProgressStats = new Map(); // Track in-progress stats by key\n }\n\n /**\n * Initialize IndexedDB\n *\n * Creates stats store with index on 'submitted' field for fast queries.\n * Safe to call multiple times (idempotent).\n *\n * @returns {Promise<void>}\n * @throws {Error} If IndexedDB is not available or initialization fails\n */\n async init() {\n if (this.db) {\n log.debug('Stats collector already initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n // Check if IndexedDB is available\n if (typeof indexedDB === 'undefined') {\n const error = new Error('IndexedDB not available');\n log.error('IndexedDB not available - stats will not be persisted');\n reject(error);\n return;\n }\n\n const request = indexedDB.open(DB_NAME, DB_VERSION);\n\n request.onerror = () => {\n const error = new Error(`Failed to open IndexedDB: ${request.error}`);\n log.error('Failed to open stats database:', request.error);\n reject(error);\n };\n\n request.onsuccess = () => {\n this.db = request.result;\n log.info('Stats database initialized');\n resolve();\n };\n\n request.onupgradeneeded = (event) => {\n const db = event.target.result;\n\n // Create stats store if it doesn't exist\n if (!db.objectStoreNames.contains(STATS_STORE)) {\n const store = db.createObjectStore(STATS_STORE, {\n keyPath: 'id',\n autoIncrement: true\n });\n\n // Index on 'submitted' for fast queries\n store.createIndex('submitted', 'submitted', { unique: false });\n\n log.info('Stats store created');\n }\n };\n });\n }\n\n /**\n * Start tracking a layout\n *\n * Creates a new layout stat entry and tracks it as in-progress.\n * If a layout with the same ID is already in progress (replay),\n * silently ends the previous cycle and starts a new one.\n *\n * @param {number} layoutId - Layout ID from CMS\n * @param {number} scheduleId - Schedule ID that triggered this layout\n * @param {Object} [options] - Options\n * @param {boolean} [options.enableStat=true] - Whether stats are enabled for this layout\n * @returns {Promise<void>}\n */\n async startLayout(layoutId, scheduleId, options) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n // Respect enableStat flag from XLF (layout/widget level stat suppression)\n if (options?.enableStat === false) {\n log.debug(`Stats disabled for layout ${layoutId} (enableStat=false)`);\n return;\n }\n\n // Key excludes scheduleId: only one layout instance can be in-progress at a time,\n // and scheduleId may change mid-play when a collection cycle completes.\n const key = `layout-${layoutId}`;\n\n // Layout replay: end previous cycle silently before starting new one\n if (this.inProgressStats.has(key)) {\n const prev = this.inProgressStats.get(key);\n prev.end = new Date();\n prev.duration = Math.floor((prev.end - prev.start) / 1000);\n await this._saveStatSplit(prev);\n this.inProgressStats.delete(key);\n log.debug(`Layout ${layoutId} replay - ended previous cycle (${prev.duration}s)`);\n }\n\n const stat = {\n type: 'layout',\n layoutId,\n scheduleId,\n start: new Date(),\n end: null,\n duration: 0,\n count: 1,\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n this.inProgressStats.set(key, stat);\n log.debug(`Started tracking layout ${layoutId} (schedule ${scheduleId})`);\n }\n\n /**\n * End tracking a layout\n *\n * Finalizes the layout stat entry and saves it to IndexedDB.\n * Calculates duration in seconds.\n *\n * @param {number} layoutId - Layout ID from CMS\n * @param {number} scheduleId - Schedule ID that triggered this layout\n * @returns {Promise<void>}\n */\n async endLayout(layoutId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const key = `layout-${layoutId}`;\n const stat = this.inProgressStats.get(key);\n\n if (!stat) {\n log.debug(`Layout ${layoutId} not found in progress (may have been ended by replay)`);\n return;\n }\n\n // Calculate duration in seconds\n stat.end = new Date();\n stat.duration = Math.floor((stat.end - stat.start) / 1000);\n\n // Save to database (splitting at hour boundaries for CMS aggregation)\n try {\n await this._saveStatSplit(stat);\n this.inProgressStats.delete(key);\n log.debug(`Ended tracking layout ${layoutId} (${stat.duration}s)`);\n } catch (error) {\n log.error(`Failed to save layout stat ${layoutId}:`, error);\n throw error;\n }\n }\n\n /**\n * Start tracking a widget/media\n *\n * Creates a new media stat entry and tracks it as in-progress.\n * If a widget with the same key is already in progress (replay),\n * silently ends the previous cycle and starts a new one.\n *\n * @param {number} mediaId - Media ID from CMS\n * @param {number} layoutId - Parent layout ID\n * @param {number} scheduleId - Schedule ID\n * @param {string|number} [widgetId] - Widget ID (for non-library widgets with no mediaId)\n * @param {Object} [options] - Options\n * @param {boolean} [options.enableStat=true] - Whether stats are enabled for this widget\n * @returns {Promise<void>}\n */\n async startWidget(mediaId, layoutId, scheduleId, widgetId, options) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n // Respect enableStat flag from XLF (layout/widget level stat suppression)\n if (options?.enableStat === false) {\n log.debug(`Stats disabled for widget ${mediaId} (enableStat=false)`);\n return;\n }\n\n // Key excludes scheduleId: it may change mid-play during collection cycles.\n const key = `media-${mediaId}-${layoutId}`;\n\n // Widget replay: end previous cycle silently before starting new one\n if (this.inProgressStats.has(key)) {\n const prev = this.inProgressStats.get(key);\n prev.end = new Date();\n prev.duration = Math.floor((prev.end - prev.start) / 1000);\n await this._saveStatSplit(prev);\n this.inProgressStats.delete(key);\n log.debug(`Widget ${mediaId} replay - ended previous cycle (${prev.duration}s)`);\n }\n\n const stat = {\n type: 'media',\n mediaId,\n widgetId: widgetId || null,\n layoutId,\n scheduleId,\n start: new Date(),\n end: null,\n duration: 0,\n count: 1,\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n this.inProgressStats.set(key, stat);\n log.debug(`Started tracking widget ${mediaId} in layout ${layoutId}`);\n }\n\n /**\n * End tracking a widget/media\n *\n * Finalizes the media stat entry and saves it to IndexedDB.\n * Calculates duration in seconds.\n *\n * @param {number} mediaId - Media ID from CMS\n * @param {number} layoutId - Parent layout ID\n * @param {number} scheduleId - Schedule ID\n * @returns {Promise<void>}\n */\n async endWidget(mediaId, layoutId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const key = `media-${mediaId}-${layoutId}`;\n const stat = this.inProgressStats.get(key);\n\n if (!stat) {\n log.debug(`Widget ${mediaId} not found in progress (expected during layout transitions)`);\n return;\n }\n\n // Calculate duration in seconds\n stat.end = new Date();\n stat.duration = Math.floor((stat.end - stat.start) / 1000);\n\n // Save to database (splitting at hour boundaries for CMS aggregation)\n try {\n await this._saveStatSplit(stat);\n this.inProgressStats.delete(key);\n log.debug(`Ended tracking widget ${mediaId} (${stat.duration}s)`);\n } catch (error) {\n log.error(`Failed to save widget stat ${mediaId}:`, error);\n throw error;\n }\n }\n\n /**\n * Record an event stat (point-in-time engagement data)\n *\n * Creates an instant stat entry with no duration. Used for tracking\n * interactive touches, webhook triggers, and other engagement events.\n * Unlike layout/widget stats, events have no start/end cycle.\n *\n * @param {string} tag - Event tag describing the interaction (e.g. 'touch', 'webhook')\n * @param {number} layoutId - Layout ID where the event occurred\n * @param {number} widgetId - Widget ID that triggered the event\n * @param {number} scheduleId - Schedule ID for the current schedule\n * @returns {Promise<void>}\n */\n async recordEvent(tag, layoutId, widgetId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const now = new Date();\n const stat = {\n type: 'event',\n tag,\n layoutId,\n widgetId,\n scheduleId,\n start: now,\n end: now,\n duration: 0,\n count: 1,\n submitted: 0\n };\n\n try {\n await this._saveStat(stat);\n log.debug(`Recorded event '${tag}' for widget ${widgetId} in layout ${layoutId}`);\n } catch (error) {\n log.error(`Failed to record event '${tag}':`, error);\n throw error;\n }\n }\n\n /**\n * Query records from an IndexedDB index with a cursor, up to a limit.\n * @param {string} storeName - Object store name\n * @param {string} indexName - Index name\n * @param {any} keyValue - Key to query (passed to openCursor)\n * @param {number} limit - Maximum records to return\n * @returns {Promise<Array>}\n */\n _queryByIndex(storeName, indexName, keyValue, limit) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readonly');\n const index = tx.objectStore(storeName).index(indexName);\n const request = index.openCursor(keyValue);\n const results = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor && results.length < limit) {\n results.push(cursor.value);\n cursor.continue();\n } else {\n resolve(results);\n }\n };\n request.onerror = () => reject(new Error(`Index query failed: ${request.error}`));\n });\n }\n\n /**\n * Delete records by ID from an IndexedDB object store.\n * @param {string} storeName - Object store name\n * @param {Array} records - Records with .id property\n * @returns {Promise<number>} Number of deleted records\n */\n _deleteByIds(storeName, records) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readwrite');\n const store = tx.objectStore(storeName);\n let deleted = 0;\n\n for (const record of records) {\n if (record.id) {\n const req = store.delete(record.id);\n req.onsuccess = () => { deleted++; };\n req.onerror = () => { log.error(`Failed to delete ${record.id}:`, req.error); };\n }\n }\n\n tx.oncomplete = () => resolve(deleted);\n tx.onerror = () => reject(new Error(`Delete failed: ${tx.error}`));\n });\n }\n\n /**\n * Get stats ready for submission to CMS\n *\n * Returns unsubmitted stats up to the specified limit.\n * Stats are ordered by ID (oldest first).\n *\n * @param {number} limit - Maximum number of stats to return (default: 50)\n * @returns {Promise<Array>} Array of stat objects\n */\n async getStatsForSubmission(limit = 50) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return [];\n }\n\n const stats = await this._queryByIndex(STATS_STORE, 'submitted', IDBKeyRange.only(0), limit);\n log.debug(`Retrieved ${stats.length} unsubmitted stats`);\n return stats;\n }\n\n /**\n * Clear submitted stats from database\n *\n * Deletes stats that were successfully submitted to CMS.\n *\n * @param {Array} stats - Array of stat objects to delete\n * @returns {Promise<void>}\n */\n async clearSubmittedStats(stats) {\n if (!this.db || !stats?.length) return;\n const deleted = await this._deleteByIds(STATS_STORE, stats);\n log.debug(`Deleted ${deleted} submitted stats`);\n }\n\n /**\n * Get aggregated stats for submission\n *\n * Groups stats by (type, layoutId, mediaId, scheduleId, hour) and sums\n * durations/counts. Used when CMS aggregationLevel is 'Aggregate'.\n *\n * @param {number} limit - Maximum number of raw stats to read (default: 50)\n * @returns {Promise<Array>} Aggregated stat objects\n */\n async getAggregatedStatsForSubmission(limit = 50) {\n const rawStats = await this.getStatsForSubmission(limit);\n if (rawStats.length === 0) return [];\n\n // Group by (type, layoutId, mediaId, scheduleId, hour)\n const groups = new Map();\n for (const stat of rawStats) {\n const hour = stat.start instanceof Date\n ? stat.start.toISOString().slice(0, 13)\n : new Date(stat.start).toISOString().slice(0, 13);\n const key = `${stat.type}|${stat.layoutId}|${stat.mediaId || ''}|${stat.widgetId || ''}|${stat.tag || ''}|${stat.scheduleId}|${hour}`;\n\n if (groups.has(key)) {\n const group = groups.get(key);\n group.count += stat.count || 1;\n group.duration += stat.duration || 0;\n // Keep earliest start and latest end\n const statStart = stat.start instanceof Date ? stat.start : new Date(stat.start);\n const statEnd = stat.end instanceof Date ? stat.end : new Date(stat.end || stat.start);\n if (statStart < group.start) group.start = statStart;\n if (statEnd > group.end) group.end = statEnd;\n group._rawIds.push(stat.id);\n } else {\n groups.set(key, {\n ...stat,\n start: stat.start instanceof Date ? stat.start : new Date(stat.start),\n end: stat.end instanceof Date ? stat.end : new Date(stat.end || stat.start),\n count: stat.count || 1,\n _rawIds: [stat.id]\n });\n }\n }\n\n return Array.from(groups.values());\n }\n\n /**\n * Get all stats (for debugging)\n *\n * @returns {Promise<Array>} All stats in database\n */\n async getAllStats() {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return [];\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readonly');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.getAll();\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n log.error('Failed to get all stats:', request.error);\n reject(new Error(`Failed to get all stats: ${request.error}`));\n };\n });\n }\n\n /**\n * Clear all stats (for testing)\n *\n * @returns {Promise<void>}\n */\n async clearAllStats() {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.clear();\n\n request.onsuccess = () => {\n log.debug('Cleared all stats');\n this.inProgressStats.clear();\n resolve();\n };\n\n request.onerror = () => {\n log.error('Failed to clear all stats:', request.error);\n reject(new Error(`Failed to clear stats: ${request.error}`));\n };\n });\n }\n\n /**\n * Save a stat to IndexedDB\n * @private\n */\n async _saveStat(stat) {\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.add(stat);\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n // Check for quota exceeded error\n if (request.error.name === 'QuotaExceededError') {\n log.error('IndexedDB quota exceeded - cleaning old stats');\n this._cleanOldStats().then(() => {\n // Retry once after cleanup\n const retryRequest = store.add(stat);\n retryRequest.onsuccess = () => resolve(retryRequest.result);\n retryRequest.onerror = () => reject(retryRequest.error);\n }).catch(reject);\n } else {\n reject(request.error);\n }\n };\n });\n }\n\n /**\n * Split a stat record at hour boundaries.\n * If a stat spans multiple hours (e.g. 12:50→13:10), it is split into\n * separate records at each hour boundary for correct CMS aggregation.\n * Returns an array of one or more stat objects.\n * @param {Object} stat - Finalized stat with start, end, duration\n * @returns {Object[]}\n * @private\n */\n _splitAtHourBoundaries(stat) {\n const start = stat.start;\n const end = stat.end;\n\n // No split needed if start and end are in the same hour\n if (start.getFullYear() === end.getFullYear() &&\n start.getMonth() === end.getMonth() &&\n start.getDate() === end.getDate() &&\n start.getHours() === end.getHours()) {\n return [stat];\n }\n\n const results = [];\n let segStart = new Date(start.getTime());\n\n while (segStart < end) {\n // Next hour boundary: top of the next hour from segStart\n const nextHour = new Date(segStart.getTime());\n nextHour.setMinutes(0, 0, 0);\n nextHour.setHours(nextHour.getHours() + 1);\n\n const segEnd = nextHour < end ? nextHour : end;\n const duration = Math.floor((segEnd - segStart) / 1000);\n\n results.push({\n ...stat,\n start: new Date(segStart.getTime()),\n end: new Date(segEnd.getTime()),\n duration,\n count: 1\n });\n\n segStart = segEnd;\n }\n\n return results;\n }\n\n /**\n * Save a stat to IndexedDB, splitting at hour boundaries first.\n * @param {Object} stat - Finalized stat with start, end, duration\n * @private\n */\n async _saveStatSplit(stat) {\n const parts = this._splitAtHourBoundaries(stat);\n for (const part of parts) {\n await this._saveStat(part);\n }\n }\n\n /**\n * Clean old stats when quota is exceeded\n * Deletes oldest 100 submitted stats\n * @private\n */\n async _cleanOldStats() {\n if (!this.db) {\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const index = store.index('submitted');\n\n // Get oldest 100 submitted stats (use 1 for boolean true in IndexedDB)\n const request = index.openCursor(1);\n const toDelete = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && toDelete.length < 100) {\n toDelete.push(cursor.value.id);\n cursor.continue();\n } else {\n // Delete collected IDs\n toDelete.forEach((id) => {\n store.delete(id);\n });\n\n log.info(`Cleaned ${toDelete.length} old stats due to quota`);\n resolve();\n }\n };\n\n request.onerror = () => {\n log.error('Failed to clean old stats:', request.error);\n reject(request.error);\n };\n });\n }\n}\n\n/**\n * Format stats as XML for XMDS submission\n *\n * Converts array of stat objects to XML format expected by CMS.\n *\n * XML format:\n * ```xml\n * <stats>\n * <stat type=\"layout\" fromdt=\"2026-02-10 12:00:00\" todt=\"2026-02-10 12:05:00\"\n * scheduleid=\"123\" layoutid=\"456\" count=\"1\" duration=\"300\" />\n * <stat type=\"media\" fromdt=\"2026-02-10 12:00:00\" todt=\"2026-02-10 12:01:00\"\n * scheduleid=\"123\" layoutid=\"456\" mediaid=\"789\" count=\"1\" duration=\"60\" />\n * </stats>\n * ```\n *\n * @param {Array} stats - Array of stat objects from getStatsForSubmission()\n * @returns {string} XML string for XMDS SubmitStats\n *\n * @example\n * const stats = await collector.getStatsForSubmission(50);\n * const xml = formatStats(stats);\n * await xmds.submitStats(xml);\n */\nexport function formatStats(stats) {\n if (!stats || stats.length === 0) {\n return '<stats></stats>';\n }\n\n const statElements = stats.map((stat) => {\n // Format dates as \"YYYY-MM-DD HH:MM:SS\"\n const fromdt = formatDateTime(stat.start);\n const todt = formatDateTime(stat.end || stat.start);\n\n // Build attributes\n const attrs = [\n `type=\"${escapeXml(stat.type)}\"`,\n `fromdt=\"${escapeXml(fromdt)}\"`,\n `todt=\"${escapeXml(todt)}\"`,\n `scheduleid=\"${stat.scheduleId}\"`,\n `layoutid=\"${stat.layoutId}\"`,\n ];\n\n // Add mediaId and widgetId for media/widget stats\n if (stat.type === 'media') {\n if (stat.mediaId) {\n attrs.push(`mediaid=\"${stat.mediaId}\"`);\n }\n // Include widgetId for non-library widgets (native widgets have no mediaId)\n if (stat.widgetId) {\n attrs.push(`widgetid=\"${stat.widgetId}\"`);\n }\n }\n\n // Add tag and widgetId for event stats\n if (stat.type === 'event') {\n if (stat.tag) {\n attrs.push(`tag=\"${escapeXml(stat.tag)}\"`);\n }\n if (stat.widgetId) {\n attrs.push(`widgetid=\"${stat.widgetId}\"`);\n }\n }\n\n // Add count and duration\n attrs.push(`count=\"${stat.count}\"`);\n attrs.push(`duration=\"${stat.duration}\"`);\n\n return ` <stat ${attrs.join(' ')} />`;\n });\n\n return `<stats>\\n${statElements.join('\\n')}\\n</stats>`;\n}\n\n/**\n * Format Date object as \"YYYY-MM-DD HH:MM:SS\"\n * @private\n */\nfunction formatDateTime(date) {\n if (!(date instanceof Date)) {\n date = new Date(date);\n }\n\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, '0');\n const day = String(date.getDate()).padStart(2, '0');\n const hours = String(date.getHours()).padStart(2, '0');\n const minutes = String(date.getMinutes()).padStart(2, '0');\n const seconds = String(date.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * Escape XML special characters\n * @private\n */\nfunction escapeXml(str) {\n if (typeof str !== 'string') {\n return str;\n }\n\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n","/**\n * LogReporter - CMS logging for Xibo Players\n *\n * Collects and submits logs to CMS via XMDS.\n * Uses IndexedDB for persistent storage across sessions.\n *\n * @module @xiboplayer/stats/logger\n */\n\nimport { createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('@xiboplayer/stats');\n\n// IndexedDB configuration\nconst DB_NAME = 'xibo-player-logs';\nconst DB_VERSION = 1;\nconst LOGS_STORE = 'logs';\n\n/**\n * Log reporter for CMS logging\n *\n * Stores log entries in IndexedDB and submits to CMS via XMDS.\n * Supports multiple log levels: error, audit, info, debug.\n *\n * @example\n * const reporter = new LogReporter();\n * await reporter.init();\n *\n * // Log messages\n * await reporter.error('Failed to load layout', 'PLAYER');\n * await reporter.info('Layout loaded successfully', 'PLAYER');\n *\n * // Get logs for submission\n * const logs = await reporter.getLogsForSubmission(100);\n * const xml = formatLogs(logs);\n * // ... submit to CMS ...\n * await reporter.clearSubmittedLogs(logs);\n */\nexport class LogReporter {\n constructor() {\n this.db = null;\n this._reportedFaults = new Map(); // code -> timestamp (deduplication)\n }\n\n /**\n * Initialize IndexedDB\n *\n * Creates logs store with index on 'submitted' field for fast queries.\n * Safe to call multiple times (idempotent).\n *\n * @returns {Promise<void>}\n * @throws {Error} If IndexedDB is not available or initialization fails\n */\n async init() {\n if (this.db) {\n log.debug('Log reporter already initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n // Check if IndexedDB is available\n if (typeof indexedDB === 'undefined') {\n const error = new Error('IndexedDB not available');\n log.error('IndexedDB not available - logs will not be persisted');\n reject(error);\n return;\n }\n\n const request = indexedDB.open(DB_NAME, DB_VERSION);\n\n request.onerror = () => {\n const error = new Error(`Failed to open IndexedDB: ${request.error}`);\n log.error('Failed to open logs database:', request.error);\n reject(error);\n };\n\n request.onsuccess = () => {\n this.db = request.result;\n log.info('Logs database initialized');\n resolve();\n };\n\n request.onupgradeneeded = (event) => {\n const db = event.target.result;\n\n // Create logs store if it doesn't exist\n if (!db.objectStoreNames.contains(LOGS_STORE)) {\n const store = db.createObjectStore(LOGS_STORE, {\n keyPath: 'id',\n autoIncrement: true\n });\n\n // Index on 'submitted' for fast queries\n store.createIndex('submitted', 'submitted', { unique: false });\n\n log.info('Logs store created');\n }\n };\n });\n }\n\n /**\n * Log a message\n *\n * Stores a log entry in IndexedDB for later submission to CMS.\n *\n * @param {string} level - Log level: 'error', 'audit', 'info', or 'debug'\n * @param {string} message - Log message\n * @param {string} category - Log category (default: 'PLAYER')\n * @param {Object} [extra] - Optional extra fields (alertType, eventType)\n * @returns {Promise<void>}\n */\n async log(level, message, category = 'PLAYER', extra = null) {\n if (!this.db) {\n // Use console directly — NOT the logger — to avoid infinite feedback loop.\n // The logger dispatches to log sinks, and this method IS the sink target.\n console.warn('[LogReporter] Database not initialized, dropping log entry');\n return;\n }\n\n // Validate log level\n const validLevels = ['error', 'warning', 'audit', 'info', 'debug'];\n if (!validLevels.includes(level)) {\n level = 'info';\n }\n\n const logEntry = {\n level,\n message,\n category,\n timestamp: new Date(),\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n // Add alert fields for faults (triggers CMS dashboard alerts)\n if (extra) {\n if (extra.alertType) logEntry.alertType = extra.alertType;\n if (extra.eventType) logEntry.eventType = extra.eventType;\n }\n\n try {\n await this._saveLog(logEntry);\n // NOTE: Do NOT call log.debug() here — it dispatches to sinks, which call\n // logReporter.log() again, creating an infinite async loop.\n } catch (error) {\n // Use console directly to avoid feedback loop\n console.error('[LogReporter] Failed to save log entry:', error);\n throw error;\n }\n }\n\n /**\n * Report a fault to CMS (special log entry that triggers alerts)\n *\n * Faults are log entries with alertType/eventType fields that cause the\n * CMS to show alerts on the display dashboard and optionally send emails.\n * Deduplicates by code: same fault code won't be reported again within\n * the cooldown period (default 5 minutes).\n *\n * @param {string} code - Fault code (e.g., 'LAYOUT_LOAD_FAILED')\n * @param {string} reason - Human-readable description\n * @param {number} [cooldownMs=300000] - Dedup cooldown in ms (default 5 min)\n * @returns {Promise<void>}\n */\n async reportFault(code, reason, cooldownMs = 300000) {\n // Deduplication: skip if same code was reported recently\n const lastReported = this._reportedFaults.get(code);\n if (lastReported && (Date.now() - lastReported) < cooldownMs) {\n return;\n }\n\n this._reportedFaults.set(code, Date.now());\n\n await this.log('error', reason, 'event', {\n alertType: 'Player Fault',\n eventType: code\n });\n\n log.info(`Fault reported: ${code} - ${reason}`);\n }\n\n /**\n * Get unsubmitted fault entries for dedicated fault submission.\n * Returns log entries that have alertType='Player Fault' and submitted=0.\n * These are the high-priority entries that should be submitted faster\n * than the normal log collection cycle.\n *\n * @param {number} [limit=10] - Maximum faults to return per batch\n * @returns {Promise<Array>} Array of fault log objects\n */\n async getFaultsForSubmission(limit = 10) {\n if (!this.db) return [];\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n\n const request = index.openCursor(IDBKeyRange.only(0));\n const faults = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && faults.length < limit) {\n if (cursor.value.alertType === 'Player Fault') {\n faults.push(cursor.value);\n }\n cursor.continue();\n } else {\n resolve(faults);\n }\n };\n\n request.onerror = () => {\n log.error('Failed to retrieve faults:', request.error);\n reject(new Error(`Failed to retrieve faults: ${request.error}`));\n };\n });\n }\n\n /**\n * Log an error message\n *\n * Shorthand for log('error', message, category)\n *\n * @param {string} message - Error message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async error(message, category = 'PLAYER') {\n return this.log('error', message, category);\n }\n\n /**\n * Log an audit message\n *\n * Shorthand for log('audit', message, category)\n *\n * @param {string} message - Audit message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async audit(message, category = 'PLAYER') {\n return this.log('audit', message, category);\n }\n\n /**\n * Log an info message\n *\n * Shorthand for log('info', message, category)\n *\n * @param {string} message - Info message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async info(message, category = 'PLAYER') {\n return this.log('info', message, category);\n }\n\n /**\n * Log a debug message\n *\n * Shorthand for log('debug', message, category)\n *\n * @param {string} message - Debug message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async debug(message, category = 'PLAYER') {\n return this.log('debug', message, category);\n }\n\n /** Query records from an IndexedDB index with a cursor, up to a limit. */\n _queryByIndex(storeName, indexName, keyValue, limit) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readonly');\n const index = tx.objectStore(storeName).index(indexName);\n const request = index.openCursor(keyValue);\n const results = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor && results.length < limit) {\n results.push(cursor.value);\n cursor.continue();\n } else {\n resolve(results);\n }\n };\n request.onerror = () => reject(new Error(`Index query failed: ${request.error}`));\n });\n }\n\n /** Delete records by ID from an IndexedDB object store. */\n _deleteByIds(storeName, records) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readwrite');\n const store = tx.objectStore(storeName);\n let deleted = 0;\n\n for (const record of records) {\n if (record.id) {\n const req = store.delete(record.id);\n req.onsuccess = () => { deleted++; };\n req.onerror = () => { log.error(`Failed to delete ${record.id}:`, req.error); };\n }\n }\n\n tx.oncomplete = () => resolve(deleted);\n tx.onerror = () => reject(new Error(`Delete failed: ${tx.error}`));\n });\n }\n\n /**\n * Get logs ready for submission to CMS\n *\n * Returns unsubmitted logs up to the spec limit of 50 per batch.\n *\n * @param {number} [limit=50] - Maximum number of logs to return (spec max: 50)\n * @returns {Promise<Array>} Array of log objects\n */\n async getLogsForSubmission(limit = 50) {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return [];\n }\n\n const logs = await this._queryByIndex(LOGS_STORE, 'submitted', IDBKeyRange.only(0), limit);\n log.debug(`Retrieved ${logs.length} unsubmitted logs (limit: ${limit})`);\n return logs;\n }\n\n /**\n * Count unsubmitted logs in the database.\n * @returns {Promise<number>}\n */\n async _countUnsubmitted() {\n return new Promise((resolve) => {\n try {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n const request = index.count(IDBKeyRange.only(0));\n request.onsuccess = () => resolve(request.result);\n request.onerror = () => resolve(0);\n } catch (_) {\n resolve(0);\n }\n });\n }\n\n /**\n * Clear submitted logs from database\n *\n * Deletes logs that were successfully submitted to CMS.\n *\n * @param {Array} logs - Array of log objects to delete\n * @returns {Promise<void>}\n */\n async clearSubmittedLogs(logs) {\n if (!this.db || !logs?.length) return;\n const deleted = await this._deleteByIds(LOGS_STORE, logs);\n log.debug(`Deleted ${deleted} submitted logs`);\n }\n\n /**\n * Get all logs (for debugging)\n *\n * @returns {Promise<Array>} All logs in database\n */\n async getAllLogs() {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return [];\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.getAll();\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n log.error('Failed to get all logs:', request.error);\n reject(new Error(`Failed to get all logs: ${request.error}`));\n };\n });\n }\n\n /**\n * Clear all logs (for testing)\n *\n * @returns {Promise<void>}\n */\n async clearAllLogs() {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.clear();\n\n request.onsuccess = () => {\n log.debug('Cleared all logs');\n resolve();\n };\n\n request.onerror = () => {\n log.error('Failed to clear all logs:', request.error);\n reject(new Error(`Failed to clear logs: ${request.error}`));\n };\n });\n }\n\n /**\n * Save a log entry to IndexedDB\n * @private\n */\n async _saveLog(logEntry) {\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.add(logEntry);\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n // Check for quota exceeded error\n if (request.error.name === 'QuotaExceededError') {\n console.warn('[LogReporter] IndexedDB quota exceeded - cleaning old logs');\n this._cleanOldLogs().then(() => {\n // Retry once after cleanup\n const retryRequest = store.add(logEntry);\n retryRequest.onsuccess = () => resolve(retryRequest.result);\n retryRequest.onerror = () => reject(retryRequest.error);\n }).catch(reject);\n } else {\n reject(request.error);\n }\n };\n });\n }\n\n /**\n * Clean old logs when quota is exceeded\n * Deletes oldest 100 submitted logs\n * @private\n */\n async _cleanOldLogs() {\n if (!this.db) {\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n\n // Get oldest 100 submitted logs (use 1 for boolean true in IndexedDB)\n const request = index.openCursor(1);\n const toDelete = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && toDelete.length < 100) {\n toDelete.push(cursor.value.id);\n cursor.continue();\n } else {\n // Delete collected IDs\n toDelete.forEach((id) => {\n store.delete(id);\n });\n\n console.log(`[LogReporter] Cleaned ${toDelete.length} old logs due to quota`);\n resolve();\n }\n };\n\n request.onerror = () => {\n console.error('[LogReporter] Failed to clean old logs:', request.error);\n reject(request.error);\n };\n });\n }\n}\n\n/**\n * Format logs as XML for XMDS submission\n *\n * Converts array of log objects to XML format expected by CMS.\n *\n * XML format (spec-compliant):\n * ```xml\n * <logs>\n * <log date=\"2026-02-10 12:00:00\" category=\"error\">\n * <thread>main</thread>\n * <method>collect</method>\n * <message>Failed to load layout 123</message>\n * <scheduleID>0</scheduleID>\n * </log>\n * </logs>\n * ```\n *\n * @param {Array} logs - Array of log objects from getLogsForSubmission()\n * @returns {string} XML string for XMDS SubmitLog\n *\n * @example\n * const logs = await reporter.getLogsForSubmission(100);\n * const xml = formatLogs(logs);\n * await xmds.submitLog(xml);\n */\nexport function formatLogs(logs) {\n if (!logs || logs.length === 0) {\n return '<logs></logs>';\n }\n\n const logElements = logs.map((logEntry) => {\n // Format date as \"YYYY-MM-DD HH:MM:SS\"\n const date = formatDateTime(logEntry.timestamp);\n\n // Spec categories: only \"error\" and \"audit\" are valid\n const category = (logEntry.level === 'error' || logEntry.level === 'audit')\n ? logEntry.level : 'audit';\n\n // Build attributes on <log> element\n const attrs = [\n `date=\"${escapeXml(date)}\"`,\n `category=\"${escapeXml(category)}\"`\n ];\n\n // Fault alert fields (triggers CMS dashboard alerts)\n if (logEntry.alertType) {\n attrs.push(`alertType=\"${escapeXml(logEntry.alertType)}\"`);\n }\n if (logEntry.eventType) {\n attrs.push(`eventType=\"${escapeXml(logEntry.eventType)}\"`);\n }\n\n // Build child elements (spec format: thread, method, message, scheduleID)\n const thread = escapeXml(logEntry.thread || 'main');\n const method = escapeXml(logEntry.method || logEntry.category || 'PLAYER');\n const message = escapeXml(logEntry.message);\n const scheduleId = escapeXml(String(logEntry.scheduleId || '0'));\n\n return ` <log ${attrs.join(' ')}>\\n <thread>${thread}</thread>\\n <method>${method}</method>\\n <message>${message}</message>\\n <scheduleID>${scheduleId}</scheduleID>\\n </log>`;\n });\n\n return `<logs>\\n${logElements.join('\\n')}\\n</logs>`;\n}\n\n/**\n * Format fault log entries as JSON for XMDS ReportFaults submission.\n *\n * Converts fault log objects (from getFaultsForSubmission) into the JSON\n * string format expected by xmds.reportFaults().\n *\n * @param {Array} faults - Array of fault log objects from getFaultsForSubmission()\n * @returns {string} JSON string for XMDS ReportFaults\n *\n * @example\n * const faults = await reporter.getFaultsForSubmission();\n * if (faults.length > 0) {\n * const json = formatFaults(faults);\n * await xmds.reportFaults(json);\n * }\n */\nexport function formatFaults(faults) {\n if (!faults || faults.length === 0) return '[]';\n\n return JSON.stringify(faults.map(f => ({\n code: f.eventType || 'UNKNOWN',\n reason: f.message || '',\n date: formatDateTime(f.timestamp),\n layoutId: f.scheduleId || 0\n })));\n}\n\n/**\n * Format Date object as \"YYYY-MM-DD HH:MM:SS\"\n * @private\n */\nfunction formatDateTime(date) {\n if (!(date instanceof Date)) {\n date = new Date(date);\n }\n\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, '0');\n const day = String(date.getDate()).padStart(2, '0');\n const hours = String(date.getHours()).padStart(2, '0');\n const minutes = String(date.getMinutes()).padStart(2, '0');\n const seconds = String(date.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * Escape XML special characters\n * @private\n */\nfunction escapeXml(str) {\n if (typeof str !== 'string') {\n return str;\n }\n\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n","// @xiboplayer/stats - Proof of play and statistics reporting\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\n\n/**\n * Stats collector for proof of play tracking\n * @module @xiboplayer/stats/collector\n */\nexport { StatsCollector, formatStats } from './stats-collector.js';\n\n/**\n * Log reporter for CMS logging\n * @module @xiboplayer/stats/logger\n */\nexport { LogReporter, formatLogs, formatFaults } from './log-reporter.js';\n"],"names":["log","createLogger","DB_NAME","DB_VERSION","STATS_STORE","StatsCollector","resolve","reject","error","request","event","db","layoutId","scheduleId","options","key","prev","stat","mediaId","widgetId","tag","now","storeName","indexName","keyValue","limit","results","cursor","records","tx","store","deleted","record","req","stats","rawStats","groups","hour","group","statStart","statEnd","retryRequest","start","end","segStart","nextHour","segEnd","duration","parts","part","toDelete","id","formatStats","fromdt","formatDateTime","todt","attrs","escapeXml","date","year","month","day","hours","minutes","seconds","str","LOGS_STORE","LogReporter","level","message","category","extra","logEntry","code","reason","cooldownMs","lastReported","faults","logs","formatLogs","thread","method","formatFaults","f","VERSION","pkg"],"mappings":"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"}
1
+ {"version":3,"file":"index-BftPXCfG.js","sources":["../../../stats/src/stats-collector.js","../../../stats/src/log-reporter.js","../../../stats/src/index.js"],"sourcesContent":["/**\n * StatsCollector - Proof of play tracking for Xibo CMS\n *\n * Tracks layout and widget playback for reporting to CMS via XMDS.\n * Uses IndexedDB for persistent storage across sessions.\n *\n * @module @xiboplayer/stats/collector\n */\n\nimport { createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('@xiboplayer/stats');\n\n// IndexedDB configuration\nconst DB_NAME = 'xibo-player-stats';\nconst DB_VERSION = 1;\nconst STATS_STORE = 'stats';\n\n/**\n * Stats collector for proof of play tracking\n *\n * Stores layout and widget playback statistics in IndexedDB.\n * Stats are submitted to CMS via XMDS SubmitStats API.\n *\n * @example\n * const collector = new StatsCollector();\n * await collector.init();\n *\n * // Track layout\n * await collector.startLayout(123, 456);\n * // ... layout plays ...\n * await collector.endLayout(123, 456);\n *\n * // Get stats for submission\n * const stats = await collector.getStatsForSubmission(50);\n * const xml = formatStats(stats);\n * // ... submit to CMS ...\n * await collector.clearSubmittedStats(stats);\n */\nexport class StatsCollector {\n constructor() {\n this.db = null;\n this.inProgressStats = new Map(); // Track in-progress stats by key\n }\n\n /**\n * Initialize IndexedDB\n *\n * Creates stats store with index on 'submitted' field for fast queries.\n * Safe to call multiple times (idempotent).\n *\n * @returns {Promise<void>}\n * @throws {Error} If IndexedDB is not available or initialization fails\n */\n async init() {\n if (this.db) {\n log.debug('Stats collector already initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n // Check if IndexedDB is available\n if (typeof indexedDB === 'undefined') {\n const error = new Error('IndexedDB not available');\n log.error('IndexedDB not available - stats will not be persisted');\n reject(error);\n return;\n }\n\n const request = indexedDB.open(DB_NAME, DB_VERSION);\n\n request.onerror = () => {\n const error = new Error(`Failed to open IndexedDB: ${request.error}`);\n log.error('Failed to open stats database:', request.error);\n reject(error);\n };\n\n request.onsuccess = () => {\n this.db = request.result;\n log.info('Stats database initialized');\n resolve();\n };\n\n request.onupgradeneeded = (event) => {\n const db = event.target.result;\n\n // Create stats store if it doesn't exist\n if (!db.objectStoreNames.contains(STATS_STORE)) {\n const store = db.createObjectStore(STATS_STORE, {\n keyPath: 'id',\n autoIncrement: true\n });\n\n // Index on 'submitted' for fast queries\n store.createIndex('submitted', 'submitted', { unique: false });\n\n log.info('Stats store created');\n }\n };\n });\n }\n\n /**\n * Start tracking a layout\n *\n * Creates a new layout stat entry and tracks it as in-progress.\n * If a layout with the same ID is already in progress (replay),\n * silently ends the previous cycle and starts a new one.\n *\n * @param {number} layoutId - Layout ID from CMS\n * @param {number} scheduleId - Schedule ID that triggered this layout\n * @param {Object} [options] - Options\n * @param {boolean} [options.enableStat=true] - Whether stats are enabled for this layout\n * @returns {Promise<void>}\n */\n async startLayout(layoutId, scheduleId, options) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n // Respect enableStat flag from XLF (layout/widget level stat suppression)\n if (options?.enableStat === false) {\n log.debug(`Stats disabled for layout ${layoutId} (enableStat=false)`);\n return;\n }\n\n // Key excludes scheduleId: only one layout instance can be in-progress at a time,\n // and scheduleId may change mid-play when a collection cycle completes.\n const key = `layout-${layoutId}`;\n\n // Layout replay: end previous cycle silently before starting new one\n if (this.inProgressStats.has(key)) {\n const prev = this.inProgressStats.get(key);\n prev.end = new Date();\n prev.duration = Math.floor((prev.end - prev.start) / 1000);\n await this._saveStatSplit(prev);\n this.inProgressStats.delete(key);\n log.debug(`Layout ${layoutId} replay - ended previous cycle (${prev.duration}s)`);\n }\n\n const stat = {\n type: 'layout',\n layoutId,\n scheduleId,\n start: new Date(),\n end: null,\n duration: 0,\n count: 1,\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n this.inProgressStats.set(key, stat);\n log.debug(`Started tracking layout ${layoutId} (schedule ${scheduleId})`);\n }\n\n /**\n * End tracking a layout\n *\n * Finalizes the layout stat entry and saves it to IndexedDB.\n * Calculates duration in seconds.\n *\n * @param {number} layoutId - Layout ID from CMS\n * @param {number} scheduleId - Schedule ID that triggered this layout\n * @returns {Promise<void>}\n */\n async endLayout(layoutId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const key = `layout-${layoutId}`;\n const stat = this.inProgressStats.get(key);\n\n if (!stat) {\n log.debug(`Layout ${layoutId} not found in progress (may have been ended by replay)`);\n return;\n }\n\n // Calculate duration in seconds\n stat.end = new Date();\n stat.duration = Math.floor((stat.end - stat.start) / 1000);\n\n // Save to database (splitting at hour boundaries for CMS aggregation)\n try {\n await this._saveStatSplit(stat);\n this.inProgressStats.delete(key);\n log.debug(`Ended tracking layout ${layoutId} (${stat.duration}s)`);\n } catch (error) {\n log.error(`Failed to save layout stat ${layoutId}:`, error);\n throw error;\n }\n }\n\n /**\n * Start tracking a widget/media\n *\n * Creates a new media stat entry and tracks it as in-progress.\n * If a widget with the same key is already in progress (replay),\n * silently ends the previous cycle and starts a new one.\n *\n * @param {number} mediaId - Media ID from CMS\n * @param {number} layoutId - Parent layout ID\n * @param {number} scheduleId - Schedule ID\n * @param {string|number} [widgetId] - Widget ID (for non-library widgets with no mediaId)\n * @param {Object} [options] - Options\n * @param {boolean} [options.enableStat=true] - Whether stats are enabled for this widget\n * @returns {Promise<void>}\n */\n async startWidget(mediaId, layoutId, scheduleId, widgetId, options) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n // Respect enableStat flag from XLF (layout/widget level stat suppression)\n if (options?.enableStat === false) {\n log.debug(`Stats disabled for widget ${mediaId} (enableStat=false)`);\n return;\n }\n\n // Key excludes scheduleId: it may change mid-play during collection cycles.\n const key = `media-${mediaId}-${layoutId}`;\n\n // Widget replay: end previous cycle silently before starting new one\n if (this.inProgressStats.has(key)) {\n const prev = this.inProgressStats.get(key);\n prev.end = new Date();\n prev.duration = Math.floor((prev.end - prev.start) / 1000);\n await this._saveStatSplit(prev);\n this.inProgressStats.delete(key);\n log.debug(`Widget ${mediaId} replay - ended previous cycle (${prev.duration}s)`);\n }\n\n const stat = {\n type: 'media',\n mediaId,\n widgetId: widgetId || null,\n layoutId,\n scheduleId,\n start: new Date(),\n end: null,\n duration: 0,\n count: 1,\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n this.inProgressStats.set(key, stat);\n log.debug(`Started tracking widget ${mediaId} in layout ${layoutId}`);\n }\n\n /**\n * End tracking a widget/media\n *\n * Finalizes the media stat entry and saves it to IndexedDB.\n * Calculates duration in seconds.\n *\n * @param {number} mediaId - Media ID from CMS\n * @param {number} layoutId - Parent layout ID\n * @param {number} scheduleId - Schedule ID\n * @returns {Promise<void>}\n */\n async endWidget(mediaId, layoutId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const key = `media-${mediaId}-${layoutId}`;\n const stat = this.inProgressStats.get(key);\n\n if (!stat) {\n log.debug(`Widget ${mediaId} not found in progress (expected during layout transitions)`);\n return;\n }\n\n // Calculate duration in seconds\n stat.end = new Date();\n stat.duration = Math.floor((stat.end - stat.start) / 1000);\n\n // Save to database (splitting at hour boundaries for CMS aggregation)\n try {\n await this._saveStatSplit(stat);\n this.inProgressStats.delete(key);\n log.debug(`Ended tracking widget ${mediaId} (${stat.duration}s)`);\n } catch (error) {\n log.error(`Failed to save widget stat ${mediaId}:`, error);\n throw error;\n }\n }\n\n /**\n * Record an event stat (point-in-time engagement data)\n *\n * Creates an instant stat entry with no duration. Used for tracking\n * interactive touches, webhook triggers, and other engagement events.\n * Unlike layout/widget stats, events have no start/end cycle.\n *\n * @param {string} tag - Event tag describing the interaction (e.g. 'touch', 'webhook')\n * @param {number} layoutId - Layout ID where the event occurred\n * @param {number} widgetId - Widget ID that triggered the event\n * @param {number} scheduleId - Schedule ID for the current schedule\n * @returns {Promise<void>}\n */\n async recordEvent(tag, layoutId, widgetId, scheduleId) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n const now = new Date();\n const stat = {\n type: 'event',\n tag,\n layoutId,\n widgetId,\n scheduleId,\n start: now,\n end: now,\n duration: 0,\n count: 1,\n submitted: 0\n };\n\n try {\n await this._saveStat(stat);\n log.debug(`Recorded event '${tag}' for widget ${widgetId} in layout ${layoutId}`);\n } catch (error) {\n log.error(`Failed to record event '${tag}':`, error);\n throw error;\n }\n }\n\n /**\n * Query records from an IndexedDB index with a cursor, up to a limit.\n * @param {string} storeName - Object store name\n * @param {string} indexName - Index name\n * @param {any} keyValue - Key to query (passed to openCursor)\n * @param {number} limit - Maximum records to return\n * @returns {Promise<Array>}\n */\n _queryByIndex(storeName, indexName, keyValue, limit) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readonly');\n const index = tx.objectStore(storeName).index(indexName);\n const request = index.openCursor(keyValue);\n const results = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor && results.length < limit) {\n results.push(cursor.value);\n cursor.continue();\n } else {\n resolve(results);\n }\n };\n request.onerror = () => reject(new Error(`Index query failed: ${request.error}`));\n });\n }\n\n /**\n * Delete records by ID from an IndexedDB object store.\n * @param {string} storeName - Object store name\n * @param {Array} records - Records with .id property\n * @returns {Promise<number>} Number of deleted records\n */\n _deleteByIds(storeName, records) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readwrite');\n const store = tx.objectStore(storeName);\n let deleted = 0;\n\n for (const record of records) {\n if (record.id) {\n const req = store.delete(record.id);\n req.onsuccess = () => { deleted++; };\n req.onerror = () => { log.error(`Failed to delete ${record.id}:`, req.error); };\n }\n }\n\n tx.oncomplete = () => resolve(deleted);\n tx.onerror = () => reject(new Error(`Delete failed: ${tx.error}`));\n });\n }\n\n /**\n * Get stats ready for submission to CMS\n *\n * Returns unsubmitted stats up to the specified limit.\n * Stats are ordered by ID (oldest first).\n *\n * @param {number} limit - Maximum number of stats to return (default: 50)\n * @returns {Promise<Array>} Array of stat objects\n */\n async getStatsForSubmission(limit = 50) {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return [];\n }\n\n const stats = await this._queryByIndex(STATS_STORE, 'submitted', IDBKeyRange.only(0), limit);\n log.debug(`Retrieved ${stats.length} unsubmitted stats`);\n return stats;\n }\n\n /**\n * Clear submitted stats from database\n *\n * Deletes stats that were successfully submitted to CMS.\n *\n * @param {Array} stats - Array of stat objects to delete\n * @returns {Promise<void>}\n */\n async clearSubmittedStats(stats) {\n if (!this.db || !stats?.length) return;\n const deleted = await this._deleteByIds(STATS_STORE, stats);\n log.debug(`Deleted ${deleted} submitted stats`);\n }\n\n /**\n * Get aggregated stats for submission\n *\n * Groups stats by (type, layoutId, mediaId, scheduleId, hour) and sums\n * durations/counts. Used when CMS aggregationLevel is 'Aggregate'.\n *\n * @param {number} limit - Maximum number of raw stats to read (default: 50)\n * @returns {Promise<Array>} Aggregated stat objects\n */\n async getAggregatedStatsForSubmission(limit = 50) {\n const rawStats = await this.getStatsForSubmission(limit);\n if (rawStats.length === 0) return [];\n\n // Group by (type, layoutId, mediaId, scheduleId, hour)\n const groups = new Map();\n for (const stat of rawStats) {\n const hour = stat.start instanceof Date\n ? stat.start.toISOString().slice(0, 13)\n : new Date(stat.start).toISOString().slice(0, 13);\n const key = `${stat.type}|${stat.layoutId}|${stat.mediaId || ''}|${stat.widgetId || ''}|${stat.tag || ''}|${stat.scheduleId}|${hour}`;\n\n if (groups.has(key)) {\n const group = groups.get(key);\n group.count += stat.count || 1;\n group.duration += stat.duration || 0;\n // Keep earliest start and latest end\n const statStart = stat.start instanceof Date ? stat.start : new Date(stat.start);\n const statEnd = stat.end instanceof Date ? stat.end : new Date(stat.end || stat.start);\n if (statStart < group.start) group.start = statStart;\n if (statEnd > group.end) group.end = statEnd;\n group._rawIds.push(stat.id);\n } else {\n groups.set(key, {\n ...stat,\n start: stat.start instanceof Date ? stat.start : new Date(stat.start),\n end: stat.end instanceof Date ? stat.end : new Date(stat.end || stat.start),\n count: stat.count || 1,\n _rawIds: [stat.id]\n });\n }\n }\n\n return Array.from(groups.values());\n }\n\n /**\n * Get all stats (for debugging)\n *\n * @returns {Promise<Array>} All stats in database\n */\n async getAllStats() {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return [];\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readonly');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.getAll();\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n log.error('Failed to get all stats:', request.error);\n reject(new Error(`Failed to get all stats: ${request.error}`));\n };\n });\n }\n\n /**\n * Clear all stats (for testing)\n *\n * @returns {Promise<void>}\n */\n async clearAllStats() {\n if (!this.db) {\n log.warn('Stats database not initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.clear();\n\n request.onsuccess = () => {\n log.debug('Cleared all stats');\n this.inProgressStats.clear();\n resolve();\n };\n\n request.onerror = () => {\n log.error('Failed to clear all stats:', request.error);\n reject(new Error(`Failed to clear stats: ${request.error}`));\n };\n });\n }\n\n /**\n * Save a stat to IndexedDB\n * @private\n */\n async _saveStat(stat) {\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const request = store.add(stat);\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n // Check for quota exceeded error\n if (request.error.name === 'QuotaExceededError') {\n log.error('IndexedDB quota exceeded - cleaning old stats');\n this._cleanOldStats().then(() => {\n // Retry once after cleanup\n const retryRequest = store.add(stat);\n retryRequest.onsuccess = () => resolve(retryRequest.result);\n retryRequest.onerror = () => reject(retryRequest.error);\n }).catch(reject);\n } else {\n reject(request.error);\n }\n };\n });\n }\n\n /**\n * Split a stat record at hour boundaries.\n * If a stat spans multiple hours (e.g. 12:50→13:10), it is split into\n * separate records at each hour boundary for correct CMS aggregation.\n * Returns an array of one or more stat objects.\n * @param {Object} stat - Finalized stat with start, end, duration\n * @returns {Object[]}\n * @private\n */\n _splitAtHourBoundaries(stat) {\n const start = stat.start;\n const end = stat.end;\n\n // No split needed if start and end are in the same hour\n if (start.getFullYear() === end.getFullYear() &&\n start.getMonth() === end.getMonth() &&\n start.getDate() === end.getDate() &&\n start.getHours() === end.getHours()) {\n return [stat];\n }\n\n const results = [];\n let segStart = new Date(start.getTime());\n\n while (segStart < end) {\n // Next hour boundary: top of the next hour from segStart\n const nextHour = new Date(segStart.getTime());\n nextHour.setMinutes(0, 0, 0);\n nextHour.setHours(nextHour.getHours() + 1);\n\n const segEnd = nextHour < end ? nextHour : end;\n const duration = Math.floor((segEnd - segStart) / 1000);\n\n results.push({\n ...stat,\n start: new Date(segStart.getTime()),\n end: new Date(segEnd.getTime()),\n duration,\n count: 1\n });\n\n segStart = segEnd;\n }\n\n return results;\n }\n\n /**\n * Save a stat to IndexedDB, splitting at hour boundaries first.\n * @param {Object} stat - Finalized stat with start, end, duration\n * @private\n */\n async _saveStatSplit(stat) {\n const parts = this._splitAtHourBoundaries(stat);\n for (const part of parts) {\n await this._saveStat(part);\n }\n }\n\n /**\n * Clean old stats when quota is exceeded\n * Deletes oldest 100 submitted stats\n * @private\n */\n async _cleanOldStats() {\n if (!this.db) {\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([STATS_STORE], 'readwrite');\n const store = transaction.objectStore(STATS_STORE);\n const index = store.index('submitted');\n\n // Get oldest 100 submitted stats (use 1 for boolean true in IndexedDB)\n const request = index.openCursor(1);\n const toDelete = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && toDelete.length < 100) {\n toDelete.push(cursor.value.id);\n cursor.continue();\n } else {\n // Delete collected IDs\n toDelete.forEach((id) => {\n store.delete(id);\n });\n\n log.info(`Cleaned ${toDelete.length} old stats due to quota`);\n resolve();\n }\n };\n\n request.onerror = () => {\n log.error('Failed to clean old stats:', request.error);\n reject(request.error);\n };\n });\n }\n}\n\n/**\n * Format stats as XML for XMDS submission\n *\n * Converts array of stat objects to XML format expected by CMS.\n *\n * XML format:\n * ```xml\n * <stats>\n * <stat type=\"layout\" fromdt=\"2026-02-10 12:00:00\" todt=\"2026-02-10 12:05:00\"\n * scheduleid=\"123\" layoutid=\"456\" count=\"1\" duration=\"300\" />\n * <stat type=\"media\" fromdt=\"2026-02-10 12:00:00\" todt=\"2026-02-10 12:01:00\"\n * scheduleid=\"123\" layoutid=\"456\" mediaid=\"789\" count=\"1\" duration=\"60\" />\n * </stats>\n * ```\n *\n * @param {Array} stats - Array of stat objects from getStatsForSubmission()\n * @returns {string} XML string for XMDS SubmitStats\n *\n * @example\n * const stats = await collector.getStatsForSubmission(50);\n * const xml = formatStats(stats);\n * await xmds.submitStats(xml);\n */\nexport function formatStats(stats) {\n if (!stats || stats.length === 0) {\n return '<stats></stats>';\n }\n\n const statElements = stats.map((stat) => {\n // Format dates as \"YYYY-MM-DD HH:MM:SS\"\n const fromdt = formatDateTime(stat.start);\n const todt = formatDateTime(stat.end || stat.start);\n\n // Build attributes\n const attrs = [\n `type=\"${escapeXml(stat.type)}\"`,\n `fromdt=\"${escapeXml(fromdt)}\"`,\n `todt=\"${escapeXml(todt)}\"`,\n `scheduleid=\"${stat.scheduleId}\"`,\n `layoutid=\"${stat.layoutId}\"`,\n ];\n\n // Add mediaId and widgetId for media/widget stats\n if (stat.type === 'media') {\n if (stat.mediaId) {\n attrs.push(`mediaid=\"${stat.mediaId}\"`);\n }\n // Include widgetId for non-library widgets (native widgets have no mediaId)\n if (stat.widgetId) {\n attrs.push(`widgetid=\"${stat.widgetId}\"`);\n }\n }\n\n // Add tag and widgetId for event stats\n if (stat.type === 'event') {\n if (stat.tag) {\n attrs.push(`tag=\"${escapeXml(stat.tag)}\"`);\n }\n if (stat.widgetId) {\n attrs.push(`widgetid=\"${stat.widgetId}\"`);\n }\n }\n\n // Add count and duration\n attrs.push(`count=\"${stat.count}\"`);\n attrs.push(`duration=\"${stat.duration}\"`);\n\n return ` <stat ${attrs.join(' ')} />`;\n });\n\n return `<stats>\\n${statElements.join('\\n')}\\n</stats>`;\n}\n\n/**\n * Format Date object as \"YYYY-MM-DD HH:MM:SS\"\n * @private\n */\nfunction formatDateTime(date) {\n if (!(date instanceof Date)) {\n date = new Date(date);\n }\n\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, '0');\n const day = String(date.getDate()).padStart(2, '0');\n const hours = String(date.getHours()).padStart(2, '0');\n const minutes = String(date.getMinutes()).padStart(2, '0');\n const seconds = String(date.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * Escape XML special characters\n * @private\n */\nfunction escapeXml(str) {\n if (typeof str !== 'string') {\n return str;\n }\n\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n","/**\n * LogReporter - CMS logging for Xibo Players\n *\n * Collects and submits logs to CMS via XMDS.\n * Uses IndexedDB for persistent storage across sessions.\n *\n * @module @xiboplayer/stats/logger\n */\n\nimport { createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('@xiboplayer/stats');\n\n// IndexedDB configuration\nconst DB_NAME = 'xibo-player-logs';\nconst DB_VERSION = 1;\nconst LOGS_STORE = 'logs';\n\n/**\n * Log reporter for CMS logging\n *\n * Stores log entries in IndexedDB and submits to CMS via XMDS.\n * Supports multiple log levels: error, audit, info, debug.\n *\n * @example\n * const reporter = new LogReporter();\n * await reporter.init();\n *\n * // Log messages\n * await reporter.error('Failed to load layout', 'PLAYER');\n * await reporter.info('Layout loaded successfully', 'PLAYER');\n *\n * // Get logs for submission\n * const logs = await reporter.getLogsForSubmission(100);\n * const xml = formatLogs(logs);\n * // ... submit to CMS ...\n * await reporter.clearSubmittedLogs(logs);\n */\nexport class LogReporter {\n constructor() {\n this.db = null;\n this._reportedFaults = new Map(); // code -> timestamp (deduplication)\n }\n\n /**\n * Initialize IndexedDB\n *\n * Creates logs store with index on 'submitted' field for fast queries.\n * Safe to call multiple times (idempotent).\n *\n * @returns {Promise<void>}\n * @throws {Error} If IndexedDB is not available or initialization fails\n */\n async init() {\n if (this.db) {\n log.debug('Log reporter already initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n // Check if IndexedDB is available\n if (typeof indexedDB === 'undefined') {\n const error = new Error('IndexedDB not available');\n log.error('IndexedDB not available - logs will not be persisted');\n reject(error);\n return;\n }\n\n const request = indexedDB.open(DB_NAME, DB_VERSION);\n\n request.onerror = () => {\n const error = new Error(`Failed to open IndexedDB: ${request.error}`);\n log.error('Failed to open logs database:', request.error);\n reject(error);\n };\n\n request.onsuccess = () => {\n this.db = request.result;\n log.info('Logs database initialized');\n resolve();\n };\n\n request.onupgradeneeded = (event) => {\n const db = event.target.result;\n\n // Create logs store if it doesn't exist\n if (!db.objectStoreNames.contains(LOGS_STORE)) {\n const store = db.createObjectStore(LOGS_STORE, {\n keyPath: 'id',\n autoIncrement: true\n });\n\n // Index on 'submitted' for fast queries\n store.createIndex('submitted', 'submitted', { unique: false });\n\n log.info('Logs store created');\n }\n };\n });\n }\n\n /**\n * Log a message\n *\n * Stores a log entry in IndexedDB for later submission to CMS.\n *\n * @param {string} level - Log level: 'error', 'audit', 'info', or 'debug'\n * @param {string} message - Log message\n * @param {string} category - Log category (default: 'PLAYER')\n * @param {Object} [extra] - Optional extra fields (alertType, eventType)\n * @returns {Promise<void>}\n */\n async log(level, message, category = 'PLAYER', extra = null) {\n if (!this.db) {\n // Use console directly — NOT the logger — to avoid infinite feedback loop.\n // The logger dispatches to log sinks, and this method IS the sink target.\n console.warn('[LogReporter] Database not initialized, dropping log entry');\n return;\n }\n\n // Validate log level\n const validLevels = ['error', 'warning', 'audit', 'info', 'debug'];\n if (!validLevels.includes(level)) {\n level = 'info';\n }\n\n const logEntry = {\n level,\n message,\n category,\n timestamp: new Date(),\n submitted: 0 // Use 0/1 instead of boolean for IndexedDB compatibility\n };\n\n // Add alert fields for faults (triggers CMS dashboard alerts)\n if (extra) {\n if (extra.alertType) logEntry.alertType = extra.alertType;\n if (extra.eventType) logEntry.eventType = extra.eventType;\n }\n\n try {\n await this._saveLog(logEntry);\n // NOTE: Do NOT call log.debug() here — it dispatches to sinks, which call\n // logReporter.log() again, creating an infinite async loop.\n } catch (error) {\n // Use console directly to avoid feedback loop\n console.error('[LogReporter] Failed to save log entry:', error);\n throw error;\n }\n }\n\n /**\n * Report a fault to CMS (special log entry that triggers alerts)\n *\n * Faults are log entries with alertType/eventType fields that cause the\n * CMS to show alerts on the display dashboard and optionally send emails.\n * Deduplicates by code: same fault code won't be reported again within\n * the cooldown period (default 5 minutes).\n *\n * @param {string} code - Fault code (e.g., 'LAYOUT_LOAD_FAILED')\n * @param {string} reason - Human-readable description\n * @param {number} [cooldownMs=300000] - Dedup cooldown in ms (default 5 min)\n * @returns {Promise<void>}\n */\n async reportFault(code, reason, cooldownMs = 300000) {\n // Deduplication: skip if same code was reported recently\n const lastReported = this._reportedFaults.get(code);\n if (lastReported && (Date.now() - lastReported) < cooldownMs) {\n return;\n }\n\n this._reportedFaults.set(code, Date.now());\n\n await this.log('error', reason, 'event', {\n alertType: 'Player Fault',\n eventType: code\n });\n\n log.info(`Fault reported: ${code} - ${reason}`);\n }\n\n /**\n * Get unsubmitted fault entries for dedicated fault submission.\n * Returns log entries that have alertType='Player Fault' and submitted=0.\n * These are the high-priority entries that should be submitted faster\n * than the normal log collection cycle.\n *\n * @param {number} [limit=10] - Maximum faults to return per batch\n * @returns {Promise<Array>} Array of fault log objects\n */\n async getFaultsForSubmission(limit = 10) {\n if (!this.db) return [];\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n\n const request = index.openCursor(IDBKeyRange.only(0));\n const faults = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && faults.length < limit) {\n if (cursor.value.alertType === 'Player Fault') {\n faults.push(cursor.value);\n }\n cursor.continue();\n } else {\n resolve(faults);\n }\n };\n\n request.onerror = () => {\n log.error('Failed to retrieve faults:', request.error);\n reject(new Error(`Failed to retrieve faults: ${request.error}`));\n };\n });\n }\n\n /**\n * Log an error message\n *\n * Shorthand for log('error', message, category)\n *\n * @param {string} message - Error message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async error(message, category = 'PLAYER') {\n return this.log('error', message, category);\n }\n\n /**\n * Log an audit message\n *\n * Shorthand for log('audit', message, category)\n *\n * @param {string} message - Audit message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async audit(message, category = 'PLAYER') {\n return this.log('audit', message, category);\n }\n\n /**\n * Log an info message\n *\n * Shorthand for log('info', message, category)\n *\n * @param {string} message - Info message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async info(message, category = 'PLAYER') {\n return this.log('info', message, category);\n }\n\n /**\n * Log a debug message\n *\n * Shorthand for log('debug', message, category)\n *\n * @param {string} message - Debug message\n * @param {string} category - Log category (default: 'PLAYER')\n * @returns {Promise<void>}\n */\n async debug(message, category = 'PLAYER') {\n return this.log('debug', message, category);\n }\n\n /** Query records from an IndexedDB index with a cursor, up to a limit. */\n _queryByIndex(storeName, indexName, keyValue, limit) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readonly');\n const index = tx.objectStore(storeName).index(indexName);\n const request = index.openCursor(keyValue);\n const results = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor && results.length < limit) {\n results.push(cursor.value);\n cursor.continue();\n } else {\n resolve(results);\n }\n };\n request.onerror = () => reject(new Error(`Index query failed: ${request.error}`));\n });\n }\n\n /** Delete records by ID from an IndexedDB object store. */\n _deleteByIds(storeName, records) {\n return new Promise((resolve, reject) => {\n const tx = this.db.transaction([storeName], 'readwrite');\n const store = tx.objectStore(storeName);\n let deleted = 0;\n\n for (const record of records) {\n if (record.id) {\n const req = store.delete(record.id);\n req.onsuccess = () => { deleted++; };\n req.onerror = () => { log.error(`Failed to delete ${record.id}:`, req.error); };\n }\n }\n\n tx.oncomplete = () => resolve(deleted);\n tx.onerror = () => reject(new Error(`Delete failed: ${tx.error}`));\n });\n }\n\n /**\n * Get logs ready for submission to CMS\n *\n * Returns unsubmitted logs up to the spec limit of 50 per batch.\n *\n * @param {number} [limit=50] - Maximum number of logs to return (spec max: 50)\n * @returns {Promise<Array>} Array of log objects\n */\n async getLogsForSubmission(limit = 50) {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return [];\n }\n\n const logs = await this._queryByIndex(LOGS_STORE, 'submitted', IDBKeyRange.only(0), limit);\n log.debug(`Retrieved ${logs.length} unsubmitted logs (limit: ${limit})`);\n return logs;\n }\n\n /**\n * Count unsubmitted logs in the database.\n * @returns {Promise<number>}\n */\n async _countUnsubmitted() {\n return new Promise((resolve) => {\n try {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n const request = index.count(IDBKeyRange.only(0));\n request.onsuccess = () => resolve(request.result);\n request.onerror = () => resolve(0);\n } catch (_) {\n resolve(0);\n }\n });\n }\n\n /**\n * Clear submitted logs from database\n *\n * Deletes logs that were successfully submitted to CMS.\n *\n * @param {Array} logs - Array of log objects to delete\n * @returns {Promise<void>}\n */\n async clearSubmittedLogs(logs) {\n if (!this.db || !logs?.length) return;\n const deleted = await this._deleteByIds(LOGS_STORE, logs);\n log.debug(`Deleted ${deleted} submitted logs`);\n }\n\n /**\n * Get all logs (for debugging)\n *\n * @returns {Promise<Array>} All logs in database\n */\n async getAllLogs() {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return [];\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readonly');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.getAll();\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n log.error('Failed to get all logs:', request.error);\n reject(new Error(`Failed to get all logs: ${request.error}`));\n };\n });\n }\n\n /**\n * Clear all logs (for testing)\n *\n * @returns {Promise<void>}\n */\n async clearAllLogs() {\n if (!this.db) {\n log.warn('Logs database not initialized');\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.clear();\n\n request.onsuccess = () => {\n log.debug('Cleared all logs');\n resolve();\n };\n\n request.onerror = () => {\n log.error('Failed to clear all logs:', request.error);\n reject(new Error(`Failed to clear logs: ${request.error}`));\n };\n });\n }\n\n /**\n * Save a log entry to IndexedDB\n * @private\n */\n async _saveLog(logEntry) {\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const request = store.add(logEntry);\n\n request.onsuccess = () => {\n resolve(request.result);\n };\n\n request.onerror = () => {\n // Check for quota exceeded error\n if (request.error.name === 'QuotaExceededError') {\n console.warn('[LogReporter] IndexedDB quota exceeded - cleaning old logs');\n this._cleanOldLogs().then(() => {\n // Retry once after cleanup\n const retryRequest = store.add(logEntry);\n retryRequest.onsuccess = () => resolve(retryRequest.result);\n retryRequest.onerror = () => reject(retryRequest.error);\n }).catch(reject);\n } else {\n reject(request.error);\n }\n };\n });\n }\n\n /**\n * Clean old logs when quota is exceeded\n * Deletes oldest 100 submitted logs\n * @private\n */\n async _cleanOldLogs() {\n if (!this.db) {\n return;\n }\n\n return new Promise((resolve, reject) => {\n const transaction = this.db.transaction([LOGS_STORE], 'readwrite');\n const store = transaction.objectStore(LOGS_STORE);\n const index = store.index('submitted');\n\n // Get oldest 100 submitted logs (use 1 for boolean true in IndexedDB)\n const request = index.openCursor(1);\n const toDelete = [];\n\n request.onsuccess = (event) => {\n const cursor = event.target.result;\n\n if (cursor && toDelete.length < 100) {\n toDelete.push(cursor.value.id);\n cursor.continue();\n } else {\n // Delete collected IDs\n toDelete.forEach((id) => {\n store.delete(id);\n });\n\n console.log(`[LogReporter] Cleaned ${toDelete.length} old logs due to quota`);\n resolve();\n }\n };\n\n request.onerror = () => {\n console.error('[LogReporter] Failed to clean old logs:', request.error);\n reject(request.error);\n };\n });\n }\n}\n\n/**\n * Format logs as XML for XMDS submission\n *\n * Converts array of log objects to XML format expected by CMS.\n *\n * XML format (spec-compliant):\n * ```xml\n * <logs>\n * <log date=\"2026-02-10 12:00:00\" category=\"error\">\n * <thread>main</thread>\n * <method>collect</method>\n * <message>Failed to load layout 123</message>\n * <scheduleID>0</scheduleID>\n * </log>\n * </logs>\n * ```\n *\n * @param {Array} logs - Array of log objects from getLogsForSubmission()\n * @returns {string} XML string for XMDS SubmitLog\n *\n * @example\n * const logs = await reporter.getLogsForSubmission(100);\n * const xml = formatLogs(logs);\n * await xmds.submitLog(xml);\n */\nexport function formatLogs(logs) {\n if (!logs || logs.length === 0) {\n return '<logs></logs>';\n }\n\n const logElements = logs.map((logEntry) => {\n // Format date as \"YYYY-MM-DD HH:MM:SS\"\n const date = formatDateTime(logEntry.timestamp);\n\n // Spec categories: only \"error\" and \"audit\" are valid\n const category = (logEntry.level === 'error' || logEntry.level === 'audit')\n ? logEntry.level : 'audit';\n\n // Build attributes on <log> element\n const attrs = [\n `date=\"${escapeXml(date)}\"`,\n `category=\"${escapeXml(category)}\"`\n ];\n\n // Fault alert fields (triggers CMS dashboard alerts)\n if (logEntry.alertType) {\n attrs.push(`alertType=\"${escapeXml(logEntry.alertType)}\"`);\n }\n if (logEntry.eventType) {\n attrs.push(`eventType=\"${escapeXml(logEntry.eventType)}\"`);\n }\n\n // Build child elements (spec format: thread, method, message, scheduleID)\n const thread = escapeXml(logEntry.thread || 'main');\n const method = escapeXml(logEntry.method || logEntry.category || 'PLAYER');\n const message = escapeXml(logEntry.message);\n const scheduleId = escapeXml(String(logEntry.scheduleId || '0'));\n\n return ` <log ${attrs.join(' ')}>\\n <thread>${thread}</thread>\\n <method>${method}</method>\\n <message>${message}</message>\\n <scheduleID>${scheduleId}</scheduleID>\\n </log>`;\n });\n\n return `<logs>\\n${logElements.join('\\n')}\\n</logs>`;\n}\n\n/**\n * Format fault log entries as JSON for XMDS ReportFaults submission.\n *\n * Converts fault log objects (from getFaultsForSubmission) into the JSON\n * string format expected by xmds.reportFaults().\n *\n * @param {Array} faults - Array of fault log objects from getFaultsForSubmission()\n * @returns {string} JSON string for XMDS ReportFaults\n *\n * @example\n * const faults = await reporter.getFaultsForSubmission();\n * if (faults.length > 0) {\n * const json = formatFaults(faults);\n * await xmds.reportFaults(json);\n * }\n */\nexport function formatFaults(faults) {\n if (!faults || faults.length === 0) return '[]';\n\n return JSON.stringify(faults.map(f => ({\n code: f.eventType || 'UNKNOWN',\n reason: f.message || '',\n date: formatDateTime(f.timestamp),\n layoutId: f.scheduleId || 0\n })));\n}\n\n/**\n * Format Date object as \"YYYY-MM-DD HH:MM:SS\"\n * @private\n */\nfunction formatDateTime(date) {\n if (!(date instanceof Date)) {\n date = new Date(date);\n }\n\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, '0');\n const day = String(date.getDate()).padStart(2, '0');\n const hours = String(date.getHours()).padStart(2, '0');\n const minutes = String(date.getMinutes()).padStart(2, '0');\n const seconds = String(date.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * Escape XML special characters\n * @private\n */\nfunction escapeXml(str) {\n if (typeof str !== 'string') {\n return str;\n }\n\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n","// @xiboplayer/stats - Proof of play and statistics reporting\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\n\n/**\n * Stats collector for proof of play tracking\n * @module @xiboplayer/stats/collector\n */\nexport { StatsCollector, formatStats } from './stats-collector.js';\n\n/**\n * Log reporter for CMS logging\n * @module @xiboplayer/stats/logger\n */\nexport { LogReporter, formatLogs, formatFaults } from './log-reporter.js';\n"],"names":["log","createLogger","DB_NAME","DB_VERSION","STATS_STORE","StatsCollector","resolve","reject","error","request","event","db","layoutId","scheduleId","options","key","prev","stat","mediaId","widgetId","tag","now","storeName","indexName","keyValue","limit","results","cursor","records","tx","store","deleted","record","req","stats","rawStats","groups","hour","group","statStart","statEnd","retryRequest","start","end","segStart","nextHour","segEnd","duration","parts","part","toDelete","id","formatStats","fromdt","formatDateTime","todt","attrs","escapeXml","date","year","month","day","hours","minutes","seconds","str","LOGS_STORE","LogReporter","level","message","category","extra","logEntry","code","reason","cooldownMs","lastReported","faults","logs","formatLogs","thread","method","formatFaults","f","VERSION","pkg"],"mappings":"iFAWMA,EAAMC,EAAa,mBAAmB,EAGtCC,EAAU,oBACVC,EAAa,EACbC,EAAc,QAuBb,MAAMC,CAAe,CAC1B,aAAc,CACZ,KAAK,GAAK,KACV,KAAK,gBAAkB,IAAI,GAC7B,CAWA,MAAM,MAAO,CACX,GAAI,KAAK,GAAI,CACXL,EAAI,MAAM,qCAAqC,EAC/C,MACF,CAEA,OAAO,IAAI,QAAQ,CAACM,EAASC,IAAW,CAEtC,GAAI,OAAO,UAAc,IAAa,CACpC,MAAMC,EAAQ,IAAI,MAAM,yBAAyB,EACjDR,EAAI,MAAM,uDAAuD,EACjEO,EAAOC,CAAK,EACZ,MACF,CAEA,MAAMC,EAAU,UAAU,KAAKP,EAASC,CAAU,EAElDM,EAAQ,QAAU,IAAM,CACtB,MAAMD,EAAQ,IAAI,MAAM,6BAA6BC,EAAQ,KAAK,EAAE,EACpET,EAAI,MAAM,iCAAkCS,EAAQ,KAAK,EACzDF,EAAOC,CAAK,CACd,EAEAC,EAAQ,UAAY,IAAM,CACxB,KAAK,GAAKA,EAAQ,OAClBT,EAAI,KAAK,4BAA4B,EACrCM,EAAO,CACT,EAEAG,EAAQ,gBAAmBC,GAAU,CACnC,MAAMC,EAAKD,EAAM,OAAO,OAGnBC,EAAG,iBAAiB,SAASP,CAAW,IAC7BO,EAAG,kBAAkBP,EAAa,CAC9C,QAAS,KACT,cAAe,EAC3B,CAAW,EAGK,YAAY,YAAa,YAAa,CAAE,OAAQ,GAAO,EAE7DJ,EAAI,KAAK,qBAAqB,EAElC,CACF,CAAC,CACH,CAeA,MAAM,YAAYY,EAAUC,EAAYC,EAAS,CAC/C,GAAI,CAAC,KAAK,GAAI,CACZd,EAAI,KAAK,gCAAgC,EACzC,MACF,CAGA,IAAIc,GAAA,YAAAA,EAAS,cAAe,GAAO,CACjCd,EAAI,MAAM,6BAA6BY,CAAQ,qBAAqB,EACpE,MACF,CAIA,MAAMG,EAAM,UAAUH,CAAQ,GAG9B,GAAI,KAAK,gBAAgB,IAAIG,CAAG,EAAG,CACjC,MAAMC,EAAO,KAAK,gBAAgB,IAAID,CAAG,EACzCC,EAAK,IAAM,IAAI,KACfA,EAAK,SAAW,KAAK,OAAOA,EAAK,IAAMA,EAAK,OAAS,GAAI,EACzD,MAAM,KAAK,eAAeA,CAAI,EAC9B,KAAK,gBAAgB,OAAOD,CAAG,EAC/Bf,EAAI,MAAM,UAAUY,CAAQ,mCAAmCI,EAAK,QAAQ,IAAI,CAClF,CAEA,MAAMC,EAAO,CACX,KAAM,SACN,SAAAL,EACA,WAAAC,EACA,MAAO,IAAI,KACX,IAAK,KACL,SAAU,EACV,MAAO,EACP,UAAW,CACjB,EAEI,KAAK,gBAAgB,IAAIE,EAAKE,CAAI,EAClCjB,EAAI,MAAM,2BAA2BY,CAAQ,cAAcC,CAAU,GAAG,CAC1E,CAYA,MAAM,UAAUD,EAAUC,EAAY,CACpC,GAAI,CAAC,KAAK,GAAI,CACZb,EAAI,KAAK,gCAAgC,EACzC,MACF,CAEA,MAAMe,EAAM,UAAUH,CAAQ,GACxBK,EAAO,KAAK,gBAAgB,IAAIF,CAAG,EAEzC,GAAI,CAACE,EAAM,CACTjB,EAAI,MAAM,UAAUY,CAAQ,wDAAwD,EACpF,MACF,CAGAK,EAAK,IAAM,IAAI,KACfA,EAAK,SAAW,KAAK,OAAOA,EAAK,IAAMA,EAAK,OAAS,GAAI,EAGzD,GAAI,CACF,MAAM,KAAK,eAAeA,CAAI,EAC9B,KAAK,gBAAgB,OAAOF,CAAG,EAC/Bf,EAAI,MAAM,yBAAyBY,CAAQ,KAAKK,EAAK,QAAQ,IAAI,CACnE,OAAST,EAAO,CACdR,MAAAA,EAAI,MAAM,8BAA8BY,CAAQ,IAAKJ,CAAK,EACpDA,CACR,CACF,CAiBA,MAAM,YAAYU,EAASN,EAAUC,EAAYM,EAAUL,EAAS,CAClE,GAAI,CAAC,KAAK,GAAI,CACZd,EAAI,KAAK,gCAAgC,EACzC,MACF,CAGA,IAAIc,GAAA,YAAAA,EAAS,cAAe,GAAO,CACjCd,EAAI,MAAM,6BAA6BkB,CAAO,qBAAqB,EACnE,MACF,CAGA,MAAMH,EAAM,SAASG,CAAO,IAAIN,CAAQ,GAGxC,GAAI,KAAK,gBAAgB,IAAIG,CAAG,EAAG,CACjC,MAAMC,EAAO,KAAK,gBAAgB,IAAID,CAAG,EACzCC,EAAK,IAAM,IAAI,KACfA,EAAK,SAAW,KAAK,OAAOA,EAAK,IAAMA,EAAK,OAAS,GAAI,EACzD,MAAM,KAAK,eAAeA,CAAI,EAC9B,KAAK,gBAAgB,OAAOD,CAAG,EAC/Bf,EAAI,MAAM,UAAUkB,CAAO,mCAAmCF,EAAK,QAAQ,IAAI,CACjF,CAEA,MAAMC,EAAO,CACX,KAAM,QACN,QAAAC,EACA,SAAUC,GAAY,KACtB,SAAAP,EACA,WAAAC,EACA,MAAO,IAAI,KACX,IAAK,KACL,SAAU,EACV,MAAO,EACP,UAAW,CACjB,EAEI,KAAK,gBAAgB,IAAIE,EAAKE,CAAI,EAClCjB,EAAI,MAAM,2BAA2BkB,CAAO,cAAcN,CAAQ,EAAE,CACtE,CAaA,MAAM,UAAUM,EAASN,EAAUC,EAAY,CAC7C,GAAI,CAAC,KAAK,GAAI,CACZb,EAAI,KAAK,gCAAgC,EACzC,MACF,CAEA,MAAMe,EAAM,SAASG,CAAO,IAAIN,CAAQ,GAClCK,EAAO,KAAK,gBAAgB,IAAIF,CAAG,EAEzC,GAAI,CAACE,EAAM,CACTjB,EAAI,MAAM,UAAUkB,CAAO,6DAA6D,EACxF,MACF,CAGAD,EAAK,IAAM,IAAI,KACfA,EAAK,SAAW,KAAK,OAAOA,EAAK,IAAMA,EAAK,OAAS,GAAI,EAGzD,GAAI,CACF,MAAM,KAAK,eAAeA,CAAI,EAC9B,KAAK,gBAAgB,OAAOF,CAAG,EAC/Bf,EAAI,MAAM,yBAAyBkB,CAAO,KAAKD,EAAK,QAAQ,IAAI,CAClE,OAAST,EAAO,CACdR,MAAAA,EAAI,MAAM,8BAA8BkB,CAAO,IAAKV,CAAK,EACnDA,CACR,CACF,CAeA,MAAM,YAAYY,EAAKR,EAAUO,EAAUN,EAAY,CACrD,GAAI,CAAC,KAAK,GAAI,CACZb,EAAI,KAAK,gCAAgC,EACzC,MACF,CAEA,MAAMqB,EAAM,IAAI,KACVJ,EAAO,CACX,KAAM,QACN,IAAAG,EACA,SAAAR,EACA,SAAAO,EACA,WAAAN,EACA,MAAOQ,EACP,IAAKA,EACL,SAAU,EACV,MAAO,EACP,UAAW,CACjB,EAEI,GAAI,CACF,MAAM,KAAK,UAAUJ,CAAI,EACzBjB,EAAI,MAAM,mBAAmBoB,CAAG,gBAAgBD,CAAQ,cAAcP,CAAQ,EAAE,CAClF,OAASJ,EAAO,CACdR,MAAAA,EAAI,MAAM,2BAA2BoB,CAAG,KAAMZ,CAAK,EAC7CA,CACR,CACF,CAUA,cAAcc,EAAWC,EAAWC,EAAUC,EAAO,CACnD,OAAO,IAAI,QAAQ,CAACnB,EAASC,IAAW,CAGtC,MAAME,EAFK,KAAK,GAAG,YAAY,CAACa,CAAS,EAAG,UAAU,EACrC,YAAYA,CAAS,EAAE,MAAMC,CAAS,EACjC,WAAWC,CAAQ,EACnCE,EAAU,CAAA,EAEhBjB,EAAQ,UAAaC,GAAU,CAC7B,MAAMiB,EAASjB,EAAM,OAAO,OACxBiB,GAAUD,EAAQ,OAASD,GAC7BC,EAAQ,KAAKC,EAAO,KAAK,EACzBA,EAAO,SAAQ,GAEfrB,EAAQoB,CAAO,CAEnB,EACAjB,EAAQ,QAAU,IAAMF,EAAO,IAAI,MAAM,uBAAuBE,EAAQ,KAAK,EAAE,CAAC,CAClF,CAAC,CACH,CAQA,aAAaa,EAAWM,EAAS,CAC/B,OAAO,IAAI,QAAQ,CAACtB,EAASC,IAAW,CACtC,MAAMsB,EAAK,KAAK,GAAG,YAAY,CAACP,CAAS,EAAG,WAAW,EACjDQ,EAAQD,EAAG,YAAYP,CAAS,EACtC,IAAIS,EAAU,EAEd,UAAWC,KAAUJ,EACnB,GAAII,EAAO,GAAI,CACb,MAAMC,EAAMH,EAAM,OAAOE,EAAO,EAAE,EAClCC,EAAI,UAAY,IAAM,CAAEF,GAAW,EACnCE,EAAI,QAAU,IAAM,CAAEjC,EAAI,MAAM,oBAAoBgC,EAAO,EAAE,IAAKC,EAAI,KAAK,CAAG,CAChF,CAGFJ,EAAG,WAAa,IAAMvB,EAAQyB,CAAO,EACrCF,EAAG,QAAU,IAAMtB,EAAO,IAAI,MAAM,kBAAkBsB,EAAG,KAAK,EAAE,CAAC,CACnE,CAAC,CACH,CAWA,MAAM,sBAAsBJ,EAAQ,GAAI,CACtC,GAAI,CAAC,KAAK,GACRzB,OAAAA,EAAI,KAAK,gCAAgC,EAClC,CAAA,EAGT,MAAMkC,EAAQ,MAAM,KAAK,cAAc9B,EAAa,YAAa,YAAY,KAAK,CAAC,EAAGqB,CAAK,EAC3FzB,OAAAA,EAAI,MAAM,aAAakC,EAAM,MAAM,oBAAoB,EAChDA,CACT,CAUA,MAAM,oBAAoBA,EAAO,CAC/B,GAAI,CAAC,KAAK,IAAM,EAACA,GAAA,MAAAA,EAAO,QAAQ,OAChC,MAAMH,EAAU,MAAM,KAAK,aAAa3B,EAAa8B,CAAK,EAC1DlC,EAAI,MAAM,WAAW+B,CAAO,kBAAkB,CAChD,CAWA,MAAM,gCAAgCN,EAAQ,GAAI,CAChD,MAAMU,EAAW,MAAM,KAAK,sBAAsBV,CAAK,EACvD,GAAIU,EAAS,SAAW,EAAG,MAAO,CAAA,EAGlC,MAAMC,EAAS,IAAI,IACnB,UAAWnB,KAAQkB,EAAU,CAC3B,MAAME,EAAOpB,EAAK,iBAAiB,KAC/BA,EAAK,MAAM,YAAW,EAAG,MAAM,EAAG,EAAE,EACpC,IAAI,KAAKA,EAAK,KAAK,EAAE,YAAW,EAAG,MAAM,EAAG,EAAE,EAC5CF,EAAM,GAAGE,EAAK,IAAI,IAAIA,EAAK,QAAQ,IAAIA,EAAK,SAAW,EAAE,IAAIA,EAAK,UAAY,EAAE,IAAIA,EAAK,KAAO,EAAE,IAAIA,EAAK,UAAU,IAAIoB,CAAI,GAEnI,GAAID,EAAO,IAAIrB,CAAG,EAAG,CACnB,MAAMuB,EAAQF,EAAO,IAAIrB,CAAG,EAC5BuB,EAAM,OAASrB,EAAK,OAAS,EAC7BqB,EAAM,UAAYrB,EAAK,UAAY,EAEnC,MAAMsB,EAAYtB,EAAK,iBAAiB,KAAOA,EAAK,MAAQ,IAAI,KAAKA,EAAK,KAAK,EACzEuB,EAAUvB,EAAK,eAAe,KAAOA,EAAK,IAAM,IAAI,KAAKA,EAAK,KAAOA,EAAK,KAAK,EACjFsB,EAAYD,EAAM,QAAOA,EAAM,MAAQC,GACvCC,EAAUF,EAAM,MAAKA,EAAM,IAAME,GACrCF,EAAM,QAAQ,KAAKrB,EAAK,EAAE,CAC5B,MACEmB,EAAO,IAAIrB,EAAK,CACd,GAAGE,EACH,MAAOA,EAAK,iBAAiB,KAAOA,EAAK,MAAQ,IAAI,KAAKA,EAAK,KAAK,EACpE,IAAKA,EAAK,eAAe,KAAOA,EAAK,IAAM,IAAI,KAAKA,EAAK,KAAOA,EAAK,KAAK,EAC1E,MAAOA,EAAK,OAAS,EACrB,QAAS,CAACA,EAAK,EAAE,CAC3B,CAAS,CAEL,CAEA,OAAO,MAAM,KAAKmB,EAAO,OAAM,CAAE,CACnC,CAOA,MAAM,aAAc,CAClB,OAAK,KAAK,GAKH,IAAI,QAAQ,CAAC9B,EAASC,IAAW,CAGtC,MAAME,EAFc,KAAK,GAAG,YAAY,CAACL,CAAW,EAAG,UAAU,EACvC,YAAYA,CAAW,EAC3B,OAAM,EAE5BK,EAAQ,UAAY,IAAM,CACxBH,EAAQG,EAAQ,MAAM,CACxB,EAEAA,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,2BAA4BS,EAAQ,KAAK,EACnDF,EAAO,IAAI,MAAM,4BAA4BE,EAAQ,KAAK,EAAE,CAAC,CAC/D,CACF,CAAC,GAjBCT,EAAI,KAAK,gCAAgC,EAClC,CAAA,EAiBX,CAOA,MAAM,eAAgB,CACpB,GAAI,CAAC,KAAK,GAAI,CACZA,EAAI,KAAK,gCAAgC,EACzC,MACF,CAEA,OAAO,IAAI,QAAQ,CAACM,EAASC,IAAW,CAGtC,MAAME,EAFc,KAAK,GAAG,YAAY,CAACL,CAAW,EAAG,WAAW,EACxC,YAAYA,CAAW,EAC3B,MAAK,EAE3BK,EAAQ,UAAY,IAAM,CACxBT,EAAI,MAAM,mBAAmB,EAC7B,KAAK,gBAAgB,MAAK,EAC1BM,EAAO,CACT,EAEAG,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,6BAA8BS,EAAQ,KAAK,EACrDF,EAAO,IAAI,MAAM,0BAA0BE,EAAQ,KAAK,EAAE,CAAC,CAC7D,CACF,CAAC,CACH,CAMA,MAAM,UAAUQ,EAAM,CACpB,OAAO,IAAI,QAAQ,CAACX,EAASC,IAAW,CAEtC,MAAMuB,EADc,KAAK,GAAG,YAAY,CAAC1B,CAAW,EAAG,WAAW,EACxC,YAAYA,CAAW,EAC3CK,EAAUqB,EAAM,IAAIb,CAAI,EAE9BR,EAAQ,UAAY,IAAM,CACxBH,EAAQG,EAAQ,MAAM,CACxB,EAEAA,EAAQ,QAAU,IAAM,CAElBA,EAAQ,MAAM,OAAS,sBACzBT,EAAI,MAAM,+CAA+C,EACzD,KAAK,iBAAiB,KAAK,IAAM,CAE/B,MAAMyC,EAAeX,EAAM,IAAIb,CAAI,EACnCwB,EAAa,UAAY,IAAMnC,EAAQmC,EAAa,MAAM,EAC1DA,EAAa,QAAU,IAAMlC,EAAOkC,EAAa,KAAK,CACxD,CAAC,EAAE,MAAMlC,CAAM,GAEfA,EAAOE,EAAQ,KAAK,CAExB,CACF,CAAC,CACH,CAWA,uBAAuBQ,EAAM,CAC3B,MAAMyB,EAAQzB,EAAK,MACb0B,EAAM1B,EAAK,IAGjB,GAAIyB,EAAM,gBAAkBC,EAAI,YAAW,GACvCD,EAAM,SAAQ,IAAOC,EAAI,SAAQ,GACjCD,EAAM,QAAO,IAAOC,EAAI,QAAO,GAC/BD,EAAM,SAAQ,IAAOC,EAAI,SAAQ,EACnC,MAAO,CAAC1B,CAAI,EAGd,MAAMS,EAAU,CAAA,EAChB,IAAIkB,EAAW,IAAI,KAAKF,EAAM,QAAO,CAAE,EAEvC,KAAOE,EAAWD,GAAK,CAErB,MAAME,EAAW,IAAI,KAAKD,EAAS,QAAO,CAAE,EAC5CC,EAAS,WAAW,EAAG,EAAG,CAAC,EAC3BA,EAAS,SAASA,EAAS,SAAQ,EAAK,CAAC,EAEzC,MAAMC,EAASD,EAAWF,EAAME,EAAWF,EACrCI,EAAW,KAAK,OAAOD,EAASF,GAAY,GAAI,EAEtDlB,EAAQ,KAAK,CACX,GAAGT,EACH,MAAO,IAAI,KAAK2B,EAAS,QAAO,CAAE,EAClC,IAAK,IAAI,KAAKE,EAAO,QAAO,CAAE,EAC9B,SAAAC,EACA,MAAO,CACf,CAAO,EAEDH,EAAWE,CACb,CAEA,OAAOpB,CACT,CAOA,MAAM,eAAeT,EAAM,CACzB,MAAM+B,EAAQ,KAAK,uBAAuB/B,CAAI,EAC9C,UAAWgC,KAAQD,EACjB,MAAM,KAAK,UAAUC,CAAI,CAE7B,CAOA,MAAM,gBAAiB,CACrB,GAAK,KAAK,GAIV,OAAO,IAAI,QAAQ,CAAC3C,EAASC,IAAW,CAEtC,MAAMuB,EADc,KAAK,GAAG,YAAY,CAAC1B,CAAW,EAAG,WAAW,EACxC,YAAYA,CAAW,EAI3CK,EAHQqB,EAAM,MAAM,WAAW,EAGf,WAAW,CAAC,EAC5BoB,EAAW,CAAA,EAEjBzC,EAAQ,UAAaC,GAAU,CAC7B,MAAMiB,EAASjB,EAAM,OAAO,OAExBiB,GAAUuB,EAAS,OAAS,KAC9BA,EAAS,KAAKvB,EAAO,MAAM,EAAE,EAC7BA,EAAO,SAAQ,IAGfuB,EAAS,QAASC,GAAO,CACvBrB,EAAM,OAAOqB,CAAE,CACjB,CAAC,EAEDnD,EAAI,KAAK,WAAWkD,EAAS,MAAM,yBAAyB,EAC5D5C,EAAO,EAEX,EAEAG,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,6BAA8BS,EAAQ,KAAK,EACrDF,EAAOE,EAAQ,KAAK,CACtB,CACF,CAAC,CACH,CACF,CAyBO,SAAS2C,EAAYlB,EAAO,CACjC,MAAI,CAACA,GAASA,EAAM,SAAW,EACtB,kBA6CF;AAAA,EA1CcA,EAAM,IAAKjB,GAAS,CAEvC,MAAMoC,EAASC,EAAerC,EAAK,KAAK,EAClCsC,EAAOD,EAAerC,EAAK,KAAOA,EAAK,KAAK,EAG5CuC,EAAQ,CACZ,SAASC,EAAUxC,EAAK,IAAI,CAAC,IAC7B,WAAWwC,EAAUJ,CAAM,CAAC,IAC5B,SAASI,EAAUF,CAAI,CAAC,IACxB,eAAetC,EAAK,UAAU,IAC9B,aAAaA,EAAK,QAAQ,GAChC,EAGI,OAAIA,EAAK,OAAS,UACZA,EAAK,SACPuC,EAAM,KAAK,YAAYvC,EAAK,OAAO,GAAG,EAGpCA,EAAK,UACPuC,EAAM,KAAK,aAAavC,EAAK,QAAQ,GAAG,GAKxCA,EAAK,OAAS,UACZA,EAAK,KACPuC,EAAM,KAAK,QAAQC,EAAUxC,EAAK,GAAG,CAAC,GAAG,EAEvCA,EAAK,UACPuC,EAAM,KAAK,aAAavC,EAAK,QAAQ,GAAG,GAK5CuC,EAAM,KAAK,UAAUvC,EAAK,KAAK,GAAG,EAClCuC,EAAM,KAAK,aAAavC,EAAK,QAAQ,GAAG,EAEjC,WAAWuC,EAAM,KAAK,GAAG,CAAC,KACnC,CAAC,EAE+B,KAAK;AAAA,CAAI,CAAC;AAAA,SAC5C,CAMA,SAASF,EAAeI,EAAM,CACtBA,aAAgB,OACpBA,EAAO,IAAI,KAAKA,CAAI,GAGtB,MAAMC,EAAOD,EAAK,YAAW,EACvBE,EAAQ,OAAOF,EAAK,SAAQ,EAAK,CAAC,EAAE,SAAS,EAAG,GAAG,EACnDG,EAAM,OAAOH,EAAK,QAAO,CAAE,EAAE,SAAS,EAAG,GAAG,EAC5CI,EAAQ,OAAOJ,EAAK,SAAQ,CAAE,EAAE,SAAS,EAAG,GAAG,EAC/CK,EAAU,OAAOL,EAAK,WAAU,CAAE,EAAE,SAAS,EAAG,GAAG,EACnDM,EAAU,OAAON,EAAK,WAAU,CAAE,EAAE,SAAS,EAAG,GAAG,EAEzD,MAAO,GAAGC,CAAI,IAAIC,CAAK,IAAIC,CAAG,IAAIC,CAAK,IAAIC,CAAO,IAAIC,CAAO,EAC/D,CAMA,SAASP,EAAUQ,EAAK,CACtB,OAAI,OAAOA,GAAQ,SACVA,EAGFA,EACJ,QAAQ,KAAM,OAAO,EACrB,QAAQ,KAAM,MAAM,EACpB,QAAQ,KAAM,MAAM,EACpB,QAAQ,KAAM,QAAQ,EACtB,QAAQ,KAAM,QAAQ,CAC3B,CChvBA,MAAMjE,EAAMC,EAAa,mBAAmB,EAGtCC,EAAU,mBACVC,EAAa,EACb+D,EAAa,OAsBZ,MAAMC,CAAY,CACvB,aAAc,CACZ,KAAK,GAAK,KACV,KAAK,gBAAkB,IAAI,GAC7B,CAWA,MAAM,MAAO,CACX,GAAI,KAAK,GAAI,CACXnE,EAAI,MAAM,kCAAkC,EAC5C,MACF,CAEA,OAAO,IAAI,QAAQ,CAACM,EAASC,IAAW,CAEtC,GAAI,OAAO,UAAc,IAAa,CACpC,MAAMC,EAAQ,IAAI,MAAM,yBAAyB,EACjDR,EAAI,MAAM,sDAAsD,EAChEO,EAAOC,CAAK,EACZ,MACF,CAEA,MAAMC,EAAU,UAAU,KAAKP,EAASC,CAAU,EAElDM,EAAQ,QAAU,IAAM,CACtB,MAAMD,EAAQ,IAAI,MAAM,6BAA6BC,EAAQ,KAAK,EAAE,EACpET,EAAI,MAAM,gCAAiCS,EAAQ,KAAK,EACxDF,EAAOC,CAAK,CACd,EAEAC,EAAQ,UAAY,IAAM,CACxB,KAAK,GAAKA,EAAQ,OAClBT,EAAI,KAAK,2BAA2B,EACpCM,EAAO,CACT,EAEAG,EAAQ,gBAAmBC,GAAU,CACnC,MAAMC,EAAKD,EAAM,OAAO,OAGnBC,EAAG,iBAAiB,SAASuD,CAAU,IAC5BvD,EAAG,kBAAkBuD,EAAY,CAC7C,QAAS,KACT,cAAe,EAC3B,CAAW,EAGK,YAAY,YAAa,YAAa,CAAE,OAAQ,GAAO,EAE7DlE,EAAI,KAAK,oBAAoB,EAEjC,CACF,CAAC,CACH,CAaA,MAAM,IAAIoE,EAAOC,EAASC,EAAW,SAAUC,EAAQ,KAAM,CAC3D,GAAI,CAAC,KAAK,GAAI,CAGZ,QAAQ,KAAK,4DAA4D,EACzE,MACF,CAGoB,CAAC,QAAS,UAAW,QAAS,OAAQ,OAAO,EAChD,SAASH,CAAK,IAC7BA,EAAQ,QAGV,MAAMI,EAAW,CACf,MAAAJ,EACA,QAAAC,EACA,SAAAC,EACA,UAAW,IAAI,KACf,UAAW,CACjB,EAGQC,IACEA,EAAM,YAAWC,EAAS,UAAYD,EAAM,WAC5CA,EAAM,YAAWC,EAAS,UAAYD,EAAM,YAGlD,GAAI,CACF,MAAM,KAAK,SAASC,CAAQ,CAG9B,OAAShE,EAAO,CAEd,cAAQ,MAAM,0CAA2CA,CAAK,EACxDA,CACR,CACF,CAeA,MAAM,YAAYiE,EAAMC,EAAQC,EAAa,IAAQ,CAEnD,MAAMC,EAAe,KAAK,gBAAgB,IAAIH,CAAI,EAC9CG,GAAiB,KAAK,IAAG,EAAKA,EAAgBD,IAIlD,KAAK,gBAAgB,IAAIF,EAAM,KAAK,IAAG,CAAE,EAEzC,MAAM,KAAK,IAAI,QAASC,EAAQ,QAAS,CACvC,UAAW,eACX,UAAWD,CACjB,CAAK,EAEDzE,EAAI,KAAK,mBAAmByE,CAAI,MAAMC,CAAM,EAAE,EAChD,CAWA,MAAM,uBAAuBjD,EAAQ,GAAI,CACvC,OAAK,KAAK,GAEH,IAAI,QAAQ,CAACnB,EAASC,IAAW,CAKtC,MAAME,EAJc,KAAK,GAAG,YAAY,CAACyD,CAAU,EAAG,UAAU,EACtC,YAAYA,CAAU,EAC5B,MAAM,WAAW,EAEf,WAAW,YAAY,KAAK,CAAC,CAAC,EAC9CW,EAAS,CAAA,EAEfpE,EAAQ,UAAaC,GAAU,CAC7B,MAAMiB,EAASjB,EAAM,OAAO,OAExBiB,GAAUkD,EAAO,OAASpD,GACxBE,EAAO,MAAM,YAAc,gBAC7BkD,EAAO,KAAKlD,EAAO,KAAK,EAE1BA,EAAO,SAAQ,GAEfrB,EAAQuE,CAAM,CAElB,EAEApE,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,6BAA8BS,EAAQ,KAAK,EACrDF,EAAO,IAAI,MAAM,8BAA8BE,EAAQ,KAAK,EAAE,CAAC,CACjE,CACF,CAAC,EA3BoB,CAAA,CA4BvB,CAWA,MAAM,MAAM4D,EAASC,EAAW,SAAU,CACxC,OAAO,KAAK,IAAI,QAASD,EAASC,CAAQ,CAC5C,CAWA,MAAM,MAAMD,EAASC,EAAW,SAAU,CACxC,OAAO,KAAK,IAAI,QAASD,EAASC,CAAQ,CAC5C,CAWA,MAAM,KAAKD,EAASC,EAAW,SAAU,CACvC,OAAO,KAAK,IAAI,OAAQD,EAASC,CAAQ,CAC3C,CAWA,MAAM,MAAMD,EAASC,EAAW,SAAU,CACxC,OAAO,KAAK,IAAI,QAASD,EAASC,CAAQ,CAC5C,CAGA,cAAchD,EAAWC,EAAWC,EAAUC,EAAO,CACnD,OAAO,IAAI,QAAQ,CAACnB,EAASC,IAAW,CAGtC,MAAME,EAFK,KAAK,GAAG,YAAY,CAACa,CAAS,EAAG,UAAU,EACrC,YAAYA,CAAS,EAAE,MAAMC,CAAS,EACjC,WAAWC,CAAQ,EACnCE,EAAU,CAAA,EAEhBjB,EAAQ,UAAaC,GAAU,CAC7B,MAAMiB,EAASjB,EAAM,OAAO,OACxBiB,GAAUD,EAAQ,OAASD,GAC7BC,EAAQ,KAAKC,EAAO,KAAK,EACzBA,EAAO,SAAQ,GAEfrB,EAAQoB,CAAO,CAEnB,EACAjB,EAAQ,QAAU,IAAMF,EAAO,IAAI,MAAM,uBAAuBE,EAAQ,KAAK,EAAE,CAAC,CAClF,CAAC,CACH,CAGA,aAAaa,EAAWM,EAAS,CAC/B,OAAO,IAAI,QAAQ,CAACtB,EAASC,IAAW,CACtC,MAAMsB,EAAK,KAAK,GAAG,YAAY,CAACP,CAAS,EAAG,WAAW,EACjDQ,EAAQD,EAAG,YAAYP,CAAS,EACtC,IAAIS,EAAU,EAEd,UAAWC,KAAUJ,EACnB,GAAII,EAAO,GAAI,CACb,MAAMC,EAAMH,EAAM,OAAOE,EAAO,EAAE,EAClCC,EAAI,UAAY,IAAM,CAAEF,GAAW,EACnCE,EAAI,QAAU,IAAM,CAAEjC,EAAI,MAAM,oBAAoBgC,EAAO,EAAE,IAAKC,EAAI,KAAK,CAAG,CAChF,CAGFJ,EAAG,WAAa,IAAMvB,EAAQyB,CAAO,EACrCF,EAAG,QAAU,IAAMtB,EAAO,IAAI,MAAM,kBAAkBsB,EAAG,KAAK,EAAE,CAAC,CACnE,CAAC,CACH,CAUA,MAAM,qBAAqBJ,EAAQ,GAAI,CACrC,GAAI,CAAC,KAAK,GACR,OAAAzB,EAAI,KAAK,+BAA+B,EACjC,CAAA,EAGT,MAAM8E,EAAO,MAAM,KAAK,cAAcZ,EAAY,YAAa,YAAY,KAAK,CAAC,EAAGzC,CAAK,EACzF,OAAAzB,EAAI,MAAM,aAAa8E,EAAK,MAAM,6BAA6BrD,CAAK,GAAG,EAChEqD,CACT,CAMA,MAAM,mBAAoB,CACxB,OAAO,IAAI,QAASxE,GAAY,CAC9B,GAAI,CAIF,MAAMG,EAHc,KAAK,GAAG,YAAY,CAACyD,CAAU,EAAG,UAAU,EACtC,YAAYA,CAAU,EAC5B,MAAM,WAAW,EACf,MAAM,YAAY,KAAK,CAAC,CAAC,EAC/CzD,EAAQ,UAAY,IAAMH,EAAQG,EAAQ,MAAM,EAChDA,EAAQ,QAAU,IAAMH,EAAQ,CAAC,CACnC,MAAY,CACVA,EAAQ,CAAC,CACX,CACF,CAAC,CACH,CAUA,MAAM,mBAAmBwE,EAAM,CAC7B,GAAI,CAAC,KAAK,IAAM,EAACA,GAAA,MAAAA,EAAM,QAAQ,OAC/B,MAAM/C,EAAU,MAAM,KAAK,aAAamC,EAAYY,CAAI,EACxD9E,EAAI,MAAM,WAAW+B,CAAO,iBAAiB,CAC/C,CAOA,MAAM,YAAa,CACjB,OAAK,KAAK,GAKH,IAAI,QAAQ,CAACzB,EAASC,IAAW,CAGtC,MAAME,EAFc,KAAK,GAAG,YAAY,CAACyD,CAAU,EAAG,UAAU,EACtC,YAAYA,CAAU,EAC1B,OAAM,EAE5BzD,EAAQ,UAAY,IAAM,CACxBH,EAAQG,EAAQ,MAAM,CACxB,EAEAA,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,0BAA2BS,EAAQ,KAAK,EAClDF,EAAO,IAAI,MAAM,2BAA2BE,EAAQ,KAAK,EAAE,CAAC,CAC9D,CACF,CAAC,GAjBCT,EAAI,KAAK,+BAA+B,EACjC,CAAA,EAiBX,CAOA,MAAM,cAAe,CACnB,GAAI,CAAC,KAAK,GAAI,CACZA,EAAI,KAAK,+BAA+B,EACxC,MACF,CAEA,OAAO,IAAI,QAAQ,CAACM,EAASC,IAAW,CAGtC,MAAME,EAFc,KAAK,GAAG,YAAY,CAACyD,CAAU,EAAG,WAAW,EACvC,YAAYA,CAAU,EAC1B,MAAK,EAE3BzD,EAAQ,UAAY,IAAM,CACxBT,EAAI,MAAM,kBAAkB,EAC5BM,EAAO,CACT,EAEAG,EAAQ,QAAU,IAAM,CACtBT,EAAI,MAAM,4BAA6BS,EAAQ,KAAK,EACpDF,EAAO,IAAI,MAAM,yBAAyBE,EAAQ,KAAK,EAAE,CAAC,CAC5D,CACF,CAAC,CACH,CAMA,MAAM,SAAS+D,EAAU,CACvB,OAAO,IAAI,QAAQ,CAAClE,EAASC,IAAW,CAEtC,MAAMuB,EADc,KAAK,GAAG,YAAY,CAACoC,CAAU,EAAG,WAAW,EACvC,YAAYA,CAAU,EAC1CzD,EAAUqB,EAAM,IAAI0C,CAAQ,EAElC/D,EAAQ,UAAY,IAAM,CACxBH,EAAQG,EAAQ,MAAM,CACxB,EAEAA,EAAQ,QAAU,IAAM,CAElBA,EAAQ,MAAM,OAAS,sBACzB,QAAQ,KAAK,4DAA4D,EACzE,KAAK,gBAAgB,KAAK,IAAM,CAE9B,MAAMgC,EAAeX,EAAM,IAAI0C,CAAQ,EACvC/B,EAAa,UAAY,IAAMnC,EAAQmC,EAAa,MAAM,EAC1DA,EAAa,QAAU,IAAMlC,EAAOkC,EAAa,KAAK,CACxD,CAAC,EAAE,MAAMlC,CAAM,GAEfA,EAAOE,EAAQ,KAAK,CAExB,CACF,CAAC,CACH,CAOA,MAAM,eAAgB,CACpB,GAAK,KAAK,GAIV,OAAO,IAAI,QAAQ,CAACH,EAASC,IAAW,CAEtC,MAAMuB,EADc,KAAK,GAAG,YAAY,CAACoC,CAAU,EAAG,WAAW,EACvC,YAAYA,CAAU,EAI1CzD,EAHQqB,EAAM,MAAM,WAAW,EAGf,WAAW,CAAC,EAC5BoB,EAAW,CAAA,EAEjBzC,EAAQ,UAAaC,GAAU,CAC7B,MAAMiB,EAASjB,EAAM,OAAO,OAExBiB,GAAUuB,EAAS,OAAS,KAC9BA,EAAS,KAAKvB,EAAO,MAAM,EAAE,EAC7BA,EAAO,SAAQ,IAGfuB,EAAS,QAASC,GAAO,CACvBrB,EAAM,OAAOqB,CAAE,CACjB,CAAC,EAED,QAAQ,IAAI,yBAAyBD,EAAS,MAAM,wBAAwB,EAC5E5C,EAAO,EAEX,EAEAG,EAAQ,QAAU,IAAM,CACtB,QAAQ,MAAM,0CAA2CA,EAAQ,KAAK,EACtEF,EAAOE,EAAQ,KAAK,CACtB,CACF,CAAC,CACH,CACF,CA2BO,SAASsE,EAAWD,EAAM,CAC/B,MAAI,CAACA,GAAQA,EAAK,SAAW,EACpB,gBAkCF;AAAA,EA/BaA,EAAK,IAAKN,GAAa,CAEzC,MAAMd,EAAOJ,EAAekB,EAAS,SAAS,EAGxCF,EAAYE,EAAS,QAAU,SAAWA,EAAS,QAAU,QAC/DA,EAAS,MAAQ,QAGfhB,EAAQ,CACZ,SAASC,EAAUC,CAAI,CAAC,IACxB,aAAaD,EAAUa,CAAQ,CAAC,GACtC,EAGQE,EAAS,WACXhB,EAAM,KAAK,cAAcC,EAAUe,EAAS,SAAS,CAAC,GAAG,EAEvDA,EAAS,WACXhB,EAAM,KAAK,cAAcC,EAAUe,EAAS,SAAS,CAAC,GAAG,EAI3D,MAAMQ,EAASvB,EAAUe,EAAS,QAAU,MAAM,EAC5CS,EAASxB,EAAUe,EAAS,QAAUA,EAAS,UAAY,QAAQ,EACnEH,EAAUZ,EAAUe,EAAS,OAAO,EACpC3D,EAAa4C,EAAU,OAAOe,EAAS,YAAc,GAAG,CAAC,EAE/D,MAAO,UAAUhB,EAAM,KAAK,GAAG,CAAC;AAAA,cAAkBwB,CAAM;AAAA,cAA0BC,CAAM;AAAA,eAA2BZ,CAAO;AAAA,kBAA+BxD,CAAU;AAAA,SACrK,CAAC,EAE6B,KAAK;AAAA,CAAI,CAAC;AAAA,QAC1C,CAkBO,SAASqE,EAAaL,EAAQ,CACnC,MAAI,CAACA,GAAUA,EAAO,SAAW,EAAU,KAEpC,KAAK,UAAUA,EAAO,IAAIM,IAAM,CACrC,KAAMA,EAAE,WAAa,UACrB,OAAQA,EAAE,SAAW,GACrB,KAAM7B,EAAe6B,EAAE,SAAS,EAChC,SAAUA,EAAE,YAAc,CAC9B,EAAI,CAAC,CACL,CAMA,SAAS7B,EAAeI,EAAM,CACtBA,aAAgB,OACpBA,EAAO,IAAI,KAAKA,CAAI,GAGtB,MAAMC,EAAOD,EAAK,YAAW,EACvBE,EAAQ,OAAOF,EAAK,SAAQ,EAAK,CAAC,EAAE,SAAS,EAAG,GAAG,EACnDG,EAAM,OAAOH,EAAK,QAAO,CAAE,EAAE,SAAS,EAAG,GAAG,EAC5CI,EAAQ,OAAOJ,EAAK,SAAQ,CAAE,EAAE,SAAS,EAAG,GAAG,EAC/CK,EAAU,OAAOL,EAAK,WAAU,CAAE,EAAE,SAAS,EAAG,GAAG,EACnDM,EAAU,OAAON,EAAK,WAAU,CAAE,EAAE,SAAS,EAAG,GAAG,EAEzD,MAAO,GAAGC,CAAI,IAAIC,CAAK,IAAIC,CAAG,IAAIC,CAAK,IAAIC,CAAO,IAAIC,CAAO,EAC/D,CAMA,SAASP,EAAUQ,EAAK,CACtB,OAAI,OAAOA,GAAQ,SACVA,EAGFA,EACJ,QAAQ,KAAM,OAAO,EACrB,QAAQ,KAAM,MAAM,EACpB,QAAQ,KAAM,MAAM,EACpB,QAAQ,KAAM,QAAQ,EACtB,QAAQ,KAAM,QAAQ,CAC3B,CC3mBY,MAACmB,EAAUC,EAAI"}
@@ -1,5 +1,5 @@
1
- const K="0.6.2",D={version:K},u={DEBUG:0,INFO:1,WARNING:2,ERROR:3,NONE:4},f=[];class _{constructor(e,t=null){this.name=e,this.useGlobal=t===null,this.useGlobal||this.setLevel(t)}_ts(){const e=new Date;return`${String(e.getHours()).padStart(2,"0")}:${String(e.getMinutes()).padStart(2,"0")}:${String(e.getSeconds()).padStart(2,"0")}.${String(e.getMilliseconds()).padStart(3,"0")}`}setLevel(e){this.useGlobal=!1,typeof e=="string"?this.level=u[e.toUpperCase()]??u.INFO:this.level=e}getEffectiveLevel(){return this.useGlobal?d.level:this.level}debug(...e){this.getEffectiveLevel()<=u.DEBUG&&console.log(`${this._ts()} [${this.name}] DEBUG:`,...e),w("debug",this.name,e)}info(...e){this.getEffectiveLevel()<=u.INFO&&console.log(`${this._ts()} [${this.name}]`,...e),w("info",this.name,e)}warn(...e){this.getEffectiveLevel()<=u.WARNING&&console.warn(`${this._ts()} [${this.name}]`,...e),w("warning",this.name,e)}error(...e){this.getEffectiveLevel()<=u.ERROR&&console.error(`${this._ts()} [${this.name}]`,...e),w("error",this.name,e)}log(e,...t){switch(e.toUpperCase()){case"DEBUG":return this.debug(...t);case"INFO":return this.info(...t);case"WARNING":case"WARN":return this.warn(...t);case"ERROR":return this.error(...t)}}}const d={level:u.WARNING,setGlobalLevel(i){typeof i=="string"?this.level=u[i.toUpperCase()]??u.INFO:this.level=i,console.log(`[Logger] Global log level set to: ${this.getLevelName(this.level)}`)},getLevelName(i){return Object.keys(u).find(e=>u[e]===i)||"UNKNOWN"}};let b=!1;if(typeof window<"u"){const e=new URLSearchParams(window.location.search).get("logLevel"),t=localStorage.getItem("xibo_log_level");e?(d.setGlobalLevel(e),b=!0):t?(d.setGlobalLevel(t),b=!0):d.setGlobalLevel("WARNING")}else typeof self<"u"&&self.swLogLevel&&d.setGlobalLevel(self.swLogLevel);function A(i,e=null){return new _(i,e)}function H(i){d.setGlobalLevel(i),typeof window<"u"&&localStorage.setItem("xibo_log_level",i.toUpperCase())}function F(){return d.getLevelName(d.level)}function W(){return d.level<=u.DEBUG}function Y(i){if(b||!i)return!1;const e=U(i);return d.setGlobalLevel(e),!0}function U(i){switch((i||"").toLowerCase()){case"debug":return"DEBUG";case"info":case"notice":case"audit":return"INFO";case"warning":return"WARNING";case"error":case"critical":case"alert":case"emergency":return"ERROR";default:return"INFO"}}function w(i,e,t){if(f.length!==0)for(const s of f)try{s({level:i,name:e,args:t})}catch{}}function z(i){f.push(i)}function X(i){const e=f.indexOf(i);e>=0&&f.splice(e,1)}class V{constructor(){this.events=new Map}on(e,t){this.events.has(e)||this.events.set(e,[]),this.events.get(e).push(t)}once(e,t){const s=(...a)=>{t(...a),this.off(e,s)};this.on(e,s)}off(e,t){if(!this.events.has(e))return;const s=this.events.get(e),a=s.indexOf(t);a!==-1&&s.splice(a,1)}emit(e,...t){if(!this.events.has(e))return;const s=this.events.get(e).slice();for(const a of s)a(...t)}removeAllListeners(e){e?this.events.delete(e):this.events.clear()}}function T(i,e){const t=new Uint8Array(i);let s="";for(let n=0;n<t.length;n++)s+=String.fromCharCode(t[n]);const a=btoa(s),r=[];for(let n=0;n<a.length;n+=64)r.push(a.substring(n,n+64));return`-----BEGIN ${e}-----
1
+ const K="0.6.3",D={version:K},u={DEBUG:0,INFO:1,WARNING:2,ERROR:3,NONE:4},f=[];class _{constructor(e,t=null){this.name=e,this.useGlobal=t===null,this.useGlobal||this.setLevel(t)}_ts(){const e=new Date;return`${String(e.getHours()).padStart(2,"0")}:${String(e.getMinutes()).padStart(2,"0")}:${String(e.getSeconds()).padStart(2,"0")}.${String(e.getMilliseconds()).padStart(3,"0")}`}setLevel(e){this.useGlobal=!1,typeof e=="string"?this.level=u[e.toUpperCase()]??u.INFO:this.level=e}getEffectiveLevel(){return this.useGlobal?d.level:this.level}debug(...e){this.getEffectiveLevel()<=u.DEBUG&&console.log(`${this._ts()} [${this.name}] DEBUG:`,...e),w("debug",this.name,e)}info(...e){this.getEffectiveLevel()<=u.INFO&&console.log(`${this._ts()} [${this.name}]`,...e),w("info",this.name,e)}warn(...e){this.getEffectiveLevel()<=u.WARNING&&console.warn(`${this._ts()} [${this.name}]`,...e),w("warning",this.name,e)}error(...e){this.getEffectiveLevel()<=u.ERROR&&console.error(`${this._ts()} [${this.name}]`,...e),w("error",this.name,e)}log(e,...t){switch(e.toUpperCase()){case"DEBUG":return this.debug(...t);case"INFO":return this.info(...t);case"WARNING":case"WARN":return this.warn(...t);case"ERROR":return this.error(...t)}}}const d={level:u.WARNING,setGlobalLevel(i){typeof i=="string"?this.level=u[i.toUpperCase()]??u.INFO:this.level=i,console.log(`[Logger] Global log level set to: ${this.getLevelName(this.level)}`)},getLevelName(i){return Object.keys(u).find(e=>u[e]===i)||"UNKNOWN"}};let b=!1;if(typeof window<"u"){const e=new URLSearchParams(window.location.search).get("logLevel"),t=localStorage.getItem("xibo_log_level");e?(d.setGlobalLevel(e),b=!0):t?(d.setGlobalLevel(t),b=!0):d.setGlobalLevel("WARNING")}else typeof self<"u"&&self.swLogLevel&&d.setGlobalLevel(self.swLogLevel);function A(i,e=null){return new _(i,e)}function H(i){d.setGlobalLevel(i),typeof window<"u"&&localStorage.setItem("xibo_log_level",i.toUpperCase())}function F(){return d.getLevelName(d.level)}function W(){return d.level<=u.DEBUG}function Y(i){if(b||!i)return!1;const e=U(i);return d.setGlobalLevel(e),!0}function U(i){switch((i||"").toLowerCase()){case"debug":return"DEBUG";case"info":case"notice":case"audit":return"INFO";case"warning":return"WARNING";case"error":case"critical":case"alert":case"emergency":return"ERROR";default:return"INFO"}}function w(i,e,t){if(f.length!==0)for(const s of f)try{s({level:i,name:e,args:t})}catch{}}function z(i){f.push(i)}function X(i){const e=f.indexOf(i);e>=0&&f.splice(e,1)}class V{constructor(){this.events=new Map}on(e,t){this.events.has(e)||this.events.set(e,[]),this.events.get(e).push(t)}once(e,t){const s=(...a)=>{t(...a),this.off(e,s)};this.on(e,s)}off(e,t){if(!this.events.has(e))return;const s=this.events.get(e),a=s.indexOf(t);a!==-1&&s.splice(a,1)}emit(e,...t){if(!this.events.has(e))return;const s=this.events.get(e).slice();for(const a of s)a(...t)}removeAllListeners(e){e?this.events.delete(e):this.events.clear()}}function T(i,e){const t=new Uint8Array(i);let s="";for(let n=0;n<t.length;n++)s+=String.fromCharCode(t[n]);const a=btoa(s),r=[];for(let n=0;n<a.length;n+=64)r.push(a.substring(n,n+64));return`-----BEGIN ${e}-----
2
2
  ${r.join(`
3
3
  `)}
4
4
  -----END ${e}-----`}async function N(){const i=await crypto.subtle.generateKey({name:"RSA-OAEP",modulusLength:1024,publicExponent:new Uint8Array([1,0,1]),hash:"SHA-256"},!0,["encrypt","decrypt"]),e=await crypto.subtle.exportKey("spki",i.publicKey),t=await crypto.subtle.exportKey("pkcs8",i.privateKey);return{publicKeyPem:T(e,"PUBLIC KEY"),privateKeyPem:T(t,"PRIVATE KEY")}}function O(i){return!i||typeof i!="string"?!1:/^-----BEGIN (PUBLIC KEY|PRIVATE KEY)-----\n[A-Za-z0-9+/=\n]+\n-----END \1-----$/.test(i.trim())}var S={};const m="xibo_config",E="xibo-hw-backup",L=1;function q(){const i=typeof process<"u"&&S?S:{},e={cmsUrl:i.CMS_URL||"",cmsKey:i.CMS_KEY||"",displayName:i.DISPLAY_NAME||"",hardwareKey:i.HARDWARE_KEY||"",xmrChannel:i.XMR_CHANNEL||"",googleGeoApiKey:i.GOOGLE_GEO_API_KEY||""};return Object.values(e).some(s=>s!=="")?e:null}class G{constructor(){this.data=this.load(),this._fromEnv||this._restoreHardwareKeyFromBackup()}load(){const e=q();if(e)return this._fromEnv=!0,e;if(typeof localStorage>"u")return{cmsUrl:"",cmsKey:"",displayName:"",hardwareKey:"",xmrChannel:""};const t=localStorage.getItem(m);let s={};if(t)try{s=JSON.parse(t)}catch(r){console.error("[Config] Failed to parse localStorage config:",r)}let a=!1;return!s.hardwareKey||s.hardwareKey.length<10?(console.warn("[Config] Missing/invalid hardwareKey — generating"),s.hardwareKey=this.generateStableHardwareKey(),this._backupHardwareKey(s.hardwareKey),a=!0):console.log("[Config] ✓ Loaded existing hardwareKey:",s.hardwareKey),s.xmrChannel||(console.warn("[Config] Missing xmrChannel — generating"),s.xmrChannel=this.generateXmrChannel(),a=!0),s.cmsUrl=s.cmsUrl||"",s.cmsKey=s.cmsKey||"",s.displayName=s.displayName||"",a&&localStorage.setItem(m,JSON.stringify(s)),s}_backupKeys(e){try{const t=indexedDB.open(E,L);t.onupgradeneeded=()=>{const s=t.result;s.objectStoreNames.contains("keys")||s.createObjectStore("keys")},t.onsuccess=()=>{const s=t.result,a=s.transaction("keys","readwrite"),r=a.objectStore("keys");for(const[n,o]of Object.entries(e))r.put(o,n);a.oncomplete=()=>{console.log("[Config] Keys backed up to IndexedDB:",Object.keys(e).join(", ")),s.close()}}}catch{}}_backupHardwareKey(e){this._backupKeys({hardwareKey:e})}async _restoreHardwareKeyFromBackup(){if(!(typeof indexedDB>"u"))try{const e=await new Promise((r,n)=>{const o=indexedDB.open(E,L);o.onupgradeneeded=()=>{const l=o.result;l.objectStoreNames.contains("keys")||l.createObjectStore("keys")},o.onsuccess=()=>r(o.result),o.onerror=()=>n(o.error)}),s=e.transaction("keys","readonly").objectStore("keys"),a=await new Promise(r=>{const n=s.get("hardwareKey");n.onsuccess=()=>r(n.result),n.onerror=()=>r(null)});e.close(),a&&a!==this.data.hardwareKey?(console.log("[Config] Restoring hardware key from IndexedDB backup:",a),console.log("[Config] (was:",this.data.hardwareKey,")"),this.data.hardwareKey=a,this.save()):!a&&this.data.hardwareKey&&this._backupHardwareKey(this.data.hardwareKey)}catch{}}save(){typeof localStorage<"u"&&localStorage.setItem(m,JSON.stringify(this.data))}isConfigured(){return!!(this.data.cmsUrl&&this.data.cmsKey&&this.data.displayName)}generateStableHardwareKey(){if(typeof crypto<"u"&&crypto.randomUUID){const a="pwa-"+crypto.randomUUID().replace(/-/g,"").substring(0,28);return console.log("[Config] Generated new UUID-based hardware key:",a),a}const t="pwa-"+Array.from({length:28},()=>Math.floor(Math.random()*16).toString(16)).join("");return console.log("[Config] Generated new random hardware key:",t),t}getCanvasFingerprint(){try{const e=document.createElement("canvas"),t=e.getContext("2d");return t?(t.textBaseline="top",t.font="14px Arial",t.fillStyle="#f60",t.fillRect(125,1,62,20),t.fillStyle="#069",t.fillText("Xibo Player",2,15),e.toDataURL()):"no-canvas"}catch{return"canvas-error"}}generateHardwareKey(){return this.generateStableHardwareKey()}generateXmrChannel(){return"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,e=>{const t=Math.random()*16|0;return(e==="x"?t:t&3|8).toString(16)})}async ensureXmrKeyPair(){if(this.data.xmrPubKey&&O(this.data.xmrPubKey))return;console.log("[Config] Generating RSA key pair for XMR registration...");const{publicKeyPem:e,privateKeyPem:t}=await N();this.data.xmrPubKey=e,this.data.xmrPrivKey=t,this.save(),typeof indexedDB<"u"&&this._backupKeys({xmrPubKey:e,xmrPrivKey:t}),console.log("[Config] RSA key pair generated and saved")}hash(e){let t=2166136261;for(let a=0;a<e.length;a++)t^=e.charCodeAt(a),t+=(t<<1)+(t<<4)+(t<<7)+(t<<8)+(t<<24);t=t>>>0;let s="";for(let a=0;a<4;a++){let r=t+a*1234567;for(let n=0;n<e.length;n++)r^=e.charCodeAt(n)+a,r+=(r<<1)+(r<<4)+(r<<7)+(r<<8)+(r<<24);r=r>>>0,s+=r.toString(16).padStart(8,"0")}return s.substring(0,32)}get cmsUrl(){return this.data.cmsUrl}set cmsUrl(e){this.data.cmsUrl=e,this.save()}get cmsKey(){return this.data.cmsKey}set cmsKey(e){this.data.cmsKey=e,this.save()}get displayName(){return this.data.displayName}set displayName(e){this.data.displayName=e,this.save()}get hardwareKey(){return this.data.hardwareKey||(console.error("[Config] CRITICAL: hardwareKey missing! Generating emergency key."),this.data.hardwareKey=this.generateStableHardwareKey(),this.save()),this.data.hardwareKey}get xmrChannel(){return this.data.xmrChannel||(console.warn("[Config] xmrChannel missing at access time — generating"),this.data.xmrChannel=this.generateXmrChannel(),this.save()),this.data.xmrChannel}get xmrPubKey(){return this.data.xmrPubKey||""}get xmrPrivKey(){return this.data.xmrPrivKey||""}get googleGeoApiKey(){return this.data.googleGeoApiKey||""}set googleGeoApiKey(e){this.data.googleGeoApiKey=e,this.save()}}const I=new G,M=new Set(["serverPort","kioskMode","fullscreen","hideMouseCursor","preventSleep","width","height"]);function J(i,e){const t=new Set([...M,...e||[]]),s={};for(const[a,r]of Object.entries(i))t.has(a)||(s[a]=r);return Object.keys(s).length>0?s:void 0}const $=A("FetchRetry"),P=3e4,v=12e4;function C(i){if(!i)return P;const e=Number(i);if(!isNaN(e)&&e>=0)return Math.min(e*1e3,v);const t=new Date(i);if(!isNaN(t.getTime())){const s=t.getTime()-Date.now();return Math.min(Math.max(s,0),v)}return P}async function Z(i,e={},t={}){const{maxRetries:s=3,baseDelayMs:a=1e3,maxDelayMs:r=3e4}=t;let n,o;for(let l=0;l<=s;l++){try{const c=await fetch(i,e);if(c.status===429){const y=C(c.headers.get("Retry-After"));if($.debug(`429 Rate limited, waiting ${y}ms (Retry-After: ${c.headers.get("Retry-After")})`),o=c,n=new Error("HTTP 429: Too Many Requests"),n.status=429,l<s){await new Promise(p=>setTimeout(p,y));continue}break}if(c.ok||c.status>=400&&c.status<500)return c;if(o=c,n=new Error(`HTTP ${c.status}: ${c.statusText}`),n.status=c.status,c.status===503&&l<s){const y=c.headers.get("Retry-After");if(y){const p=C(y);$.debug(`503 Service Unavailable, Retry-After: ${y} (${p}ms)`),await new Promise(k=>setTimeout(k,p));continue}}}catch(c){n=c,o=null}if(l<s){const y=Math.min(a*Math.pow(2,l),r)*(.5+Math.random()*.5);$.debug(`Retry ${l+1}/${s} in ${Math.round(y)}ms:`,String(i).slice(0,80)),await new Promise(p=>setTimeout(p,y))}}if(o)return o;throw n}const h=A("CmsApi");class Q{constructor({baseUrl:e,clientId:t,clientSecret:s,apiToken:a}={}){this.baseUrl=(e||"").replace(/\/+$/,""),this.clientId=t||null,this.clientSecret=s||null,this.accessToken=a||null,this.tokenExpiry=a?1/0:0}async authenticate(){h.info("Authenticating with CMS API...");const e=await fetch(`${this.baseUrl}/api/authorize/access_token`,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:new URLSearchParams({grant_type:"client_credentials",client_id:this.clientId,client_secret:this.clientSecret})});if(!e.ok){const s=await e.text();throw new Error(`OAuth2 authentication failed (${e.status}): ${s}`)}const t=await e.json();return this.accessToken=t.access_token,this.tokenExpiry=Date.now()+(t.expires_in||3600)*1e3,h.info("Authenticated successfully, token expires in",t.expires_in,"s"),this.accessToken}async ensureToken(){if(!(this.accessToken&&Date.now()<this.tokenExpiry-6e4)){if(!this.clientId||!this.clientSecret){if(this.accessToken)return;throw new g("AUTH","/authorize",0,"No valid token and no OAuth2 credentials")}await this.authenticate()}}async request(e,t,s={}){await this.ensureToken();const a=new URL(`${this.baseUrl}/api${t}`),r={method:e,headers:{Authorization:`Bearer ${this.accessToken}`}};if(e==="GET")for(const[l,c]of Object.entries(s))c!=null&&a.searchParams.set(l,String(c));else r.headers["Content-Type"]="application/x-www-form-urlencoded",r.body=new URLSearchParams(s);const n=await fetch(a,r);return n.ok||await this._handleErrorResponse(n,e,t),(n.headers.get("Content-Type")||"").includes("application/json")?n.json():null}get(e,t){return this.request("GET",e,t)}post(e,t){return this.request("POST",e,t)}put(e,t){return this.request("PUT",e,t)}del(e){return this.request("DELETE",e)}async _listRequest(e,t={}){const s=await this.request("GET",e,t);return Array.isArray(s)?s:[]}async _handleErrorResponse(e,t,s){var n;const a=await e.text();let r;try{const o=JSON.parse(a);r=((n=o.error)==null?void 0:n.message)||o.message||a}catch{r=a}throw new g(t,s,e.status,r)}async findDisplay(e){h.info("Looking up display by hardwareKey:",e);const t=await this._listRequest("/display",{hardwareKey:e});if(t.length===0)return h.info("No display found for hardwareKey:",e),null;const s=t[0];return h.info(`Found display: ${s.display} (ID: ${s.displayId}, licensed: ${s.licensed})`),s}async authorizeDisplay(e){h.info("Authorizing display:",e),await this.request("PUT",`/display/authorise/${e}`),h.info("Display authorized successfully")}async editDisplay(e,t){return h.info("Editing display:",e,t),this.request("PUT",`/display/${e}`,t)}async listDisplays(e={}){return this._listRequest("/display",e)}async requestScreenshot(e){await this.request("PUT",`/display/requestscreenshot/${e}`)}async getDisplayStatus(e){return this.request("GET",`/display/status/${e}`)}async requestMultipart(e,t,s){await this.ensureToken();const a=`${this.baseUrl}/api${t}`,r=await fetch(a,{method:e,headers:{Authorization:`Bearer ${this.accessToken}`},body:s});return r.ok||await this._handleErrorResponse(r,e,t),(r.headers.get("Content-Type")||"").includes("application/json")?r.json():null}async createLayout({name:e,resolutionId:t,description:s}){const a={name:e,resolutionId:t};return s&&(a.description=s),this.request("POST","/layout",a)}async listLayouts(e={}){return this._listRequest("/layout",e)}async getLayout(e){return this.request("GET",`/layout/${e}`)}async deleteLayout(e){await this.request("DELETE",`/layout/${e}`)}async publishLayout(e){await this.request("PUT",`/layout/publish/${e}`,{publishNow:1})}async checkoutLayout(e){return this.request("PUT",`/layout/checkout/${e}`)}async getDraftLayout(e){const t=await this.listLayouts({parentId:e});return t.length>0?t[0]:null}async editLayoutBackground(e,t){return this.request("PUT",`/layout/background/${e}`,t)}async addRegion(e,t){return this.request("POST",`/region/${e}`,t)}async editRegion(e,t){return this.request("PUT",`/region/${e}`,t)}async deleteRegion(e){await this.request("DELETE",`/region/${e}`)}async addWidget(e,t,s={}){const{templateId:a,displayOrder:r,...n}=s,o={};a!==void 0&&(o.templateId=a),r!==void 0&&(o.displayOrder=r);const l=await this.request("POST",`/playlist/widget/${e}/${t}`,o);return Object.keys(n).length>0?(n.duration!==void 0&&n.useDuration===void 0&&(n.useDuration=1),this.request("PUT",`/playlist/widget/${l.widgetId}`,n)):l}async editWidget(e,t){return this.request("PUT",`/playlist/widget/${e}`,t)}async deleteWidget(e){await this.request("DELETE",`/playlist/widget/${e}`)}async uploadMedia(e){return this.requestMultipart("POST","/library",e)}async listMedia(e={}){return this._listRequest("/library",e)}async getMedia(e){return this.request("GET",`/library/${e}`)}async deleteMedia(e){await this.request("DELETE",`/library/${e}`)}async createCampaign(e){return this.request("POST","/campaign",{name:e})}async listCampaigns(e={}){return this._listRequest("/campaign",e)}async deleteCampaign(e){await this.request("DELETE",`/campaign/${e}`)}async assignLayoutToCampaign(e,t,s){const a={layoutId:t};s!==void 0&&(a.displayOrder=s),await this.request("POST",`/campaign/layout/assign/${e}`,a)}async createSchedule(e){const t={...e};Array.isArray(t.displayGroupIds)&&delete t.displayGroupIds,await this.ensureToken();const s=`${this.baseUrl}/api/schedule`,a=new URLSearchParams;for(const[o,l]of Object.entries(t))l!=null&&a.set(o,String(l));if(Array.isArray(e.displayGroupIds))for(const o of e.displayGroupIds)a.append("displayGroupIds[]",String(o));const r=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:a});if(!r.ok){const o=await r.text();throw new Error(`CMS API POST /schedule failed (${r.status}): ${o}`)}return(r.headers.get("Content-Type")||"").includes("application/json")?r.json():null}async deleteSchedule(e){await this.request("DELETE",`/schedule/${e}`)}async listSchedules(e={}){const t=await this.request("GET","/schedule/data/events",e);return Array.isArray(t)?t:(t==null?void 0:t.events)||[]}async listDisplayGroups(e={}){return this._listRequest("/displaygroup",e)}async createDisplayGroup(e,t){const s={displayGroup:e};return t&&(s.description=t),this.request("POST","/displaygroup",s)}async deleteDisplayGroup(e){await this.request("DELETE",`/displaygroup/${e}`)}async assignDisplayToGroup(e,t){await this.ensureToken();const s=`${this.baseUrl}/api/displaygroup/${e}/display/assign`,a=new URLSearchParams;a.append("displayId[]",String(t));const r=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:a});if(!r.ok){const n=await r.text();throw new Error(`CMS API assign display to group failed (${r.status}): ${n}`)}}async unassignDisplayFromGroup(e,t){await this.ensureToken();const s=`${this.baseUrl}/api/displaygroup/${e}/display/unassign`,a=new URLSearchParams;a.append("displayId[]",String(t));const r=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:a});if(!r.ok){const n=await r.text();throw new Error(`CMS API unassign display from group failed (${r.status}): ${n}`)}}async listResolutions(){return this._listRequest("/resolution")}async listTemplates(e={}){return this._listRequest("/template",e)}async assignMediaToPlaylist(e,t){const s=Array.isArray(t)?t:[t];await this.ensureToken();const a=`${this.baseUrl}/api/playlist/library/assign/${e}`,r=new URLSearchParams;for(const l of s)r.append("media[]",String(l));const n=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:r});if(!n.ok){const l=await n.text();throw new g("POST",`/playlist/library/assign/${e}`,n.status,l)}return(n.headers.get("Content-Type")||"").includes("application/json")?n.json():null}async editLayout(e,t){return this.request("PUT",`/layout/${e}`,t)}async copyLayout(e,t={}){return this.post(`/layout/copy/${e}`,t)}async discardLayout(e){await this.put(`/layout/discard/${e}`)}async editCampaign(e,t){return this.put(`/campaign/${e}`,t)}async getCampaign(e){return this.get(`/campaign/${e}`)}async unassignLayoutFromCampaign(e,t){await this.post(`/campaign/layout/unassign/${e}`,{layoutId:t})}async editSchedule(e,t){return this.put(`/schedule/${e}`,t)}async retireLayout(e){await this.put(`/layout/retire/${e}`)}async unretireLayout(e){await this.put(`/layout/unretire/${e}`)}async getLayoutStatus(e){return this.get(`/layout/status/${e}`)}async tagLayout(e,t){await this.post(`/layout/${e}/tag`,{tag:t.join(",")})}async untagLayout(e,t){await this.post(`/layout/${e}/untag`,{tag:t.join(",")})}async listCommands(e={}){return this._listRequest("/command",e)}async createCommand(e){return this.post("/command",e)}async editCommand(e,t){return this.put(`/command/${e}`,t)}async deleteCommand(e){await this.del(`/command/${e}`)}async deleteDisplay(e){await this.del(`/display/${e}`)}async wolDisplay(e){await this.post(`/display/wol/${e}`)}async setDefaultLayout(e,t){return this.put(`/display/${e}`,{defaultLayoutId:t})}async purgeDisplay(e){await this.post(`/display/purge/${e}`)}async listDayParts(e={}){return this._listRequest("/daypart",e)}async createDayPart(e){return this.post("/daypart",e)}async editDayPart(e,t){return this.put(`/daypart/${e}`,t)}async deleteDayPart(e){await this.del(`/daypart/${e}`)}async uploadMediaUrl(e,t){return this.post("/library",{url:e,name:t,type:"uri"})}async copyMedia(e){return this.post(`/library/copy/${e}`)}async downloadMedia(e){await this.ensureToken();const t=`${this.baseUrl}/api/library/download/${e}`,s=await fetch(t,{headers:{Authorization:`Bearer ${this.accessToken}`}});if(!s.ok){const a=await s.text();throw new g("GET",`/library/download/${e}`,s.status,a)}return s}async editMedia(e,t){return this.put(`/library/${e}`,t)}async getMediaUsage(e){return this.get(`/library/usage/${e}`)}async tidyLibrary(){await this.post("/library/tidy")}async listPlaylists(e={}){return this._listRequest("/playlist",e)}async createPlaylist(e){return this.post("/playlist",{name:e})}async getPlaylist(e){return this.get(`/playlist/${e}`)}async editPlaylist(e,t){return this.put(`/playlist/${e}`,t)}async deletePlaylist(e){await this.del(`/playlist/${e}`)}async reorderPlaylist(e,t){await this.ensureToken();const s=`${this.baseUrl}/api/playlist/order/${e}`,a=new URLSearchParams;for(const n of t)a.append("widgets[]",String(n));const r=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:a});if(!r.ok){const n=await r.text();throw new g("POST",`/playlist/order/${e}`,r.status,n)}}async copyPlaylist(e){return this.post(`/playlist/copy/${e}`)}async setWidgetTransition(e,t,s={}){return this.put(`/playlist/widget/transition/${e}`,{type:t,...s})}async setWidgetAudio(e,t){return this.put(`/playlist/widget/${e}/audio`,t)}async removeWidgetAudio(e){await this.del(`/playlist/widget/${e}/audio`)}async setWidgetExpiry(e,t){return this.put(`/playlist/widget/${e}/expiry`,t)}async saveAsTemplate(e,t){return this.post(`/template/${e}`,t)}async getTemplate(e){return this.get(`/template/${e}`)}async deleteTemplate(e){await this.del(`/template/${e}`)}async listDatasets(e={}){return this._listRequest("/dataset",e)}async createDataset(e){return this.post("/dataset",e)}async editDataset(e,t){return this.put(`/dataset/${e}`,t)}async deleteDataset(e){await this.del(`/dataset/${e}`)}async listDatasetColumns(e){return this._listRequest(`/dataset/${e}/column`)}async createDatasetColumn(e,t){return this.post(`/dataset/${e}/column`,t)}async editDatasetColumn(e,t,s){return this.put(`/dataset/${e}/column/${t}`,s)}async deleteDatasetColumn(e,t){await this.del(`/dataset/${e}/column/${t}`)}async listDatasetData(e,t={}){return this._listRequest(`/dataset/data/${e}`,t)}async addDatasetRow(e,t){return this.post(`/dataset/data/${e}`,t)}async editDatasetRow(e,t,s){return this.put(`/dataset/data/${e}/${t}`,s)}async deleteDatasetRow(e,t){await this.del(`/dataset/data/${e}/${t}`)}async importDatasetCsv(e,t){return this.requestMultipart("POST",`/dataset/import/${e}`,t)}async clearDataset(e){await this.del(`/dataset/data/${e}`)}async listNotifications(e={}){return this._listRequest("/notification",e)}async createNotification(e){return this.post("/notification",e)}async editNotification(e,t){return this.put(`/notification/${e}`,t)}async deleteNotification(e){await this.del(`/notification/${e}`)}async listFolders(e={}){return this._listRequest("/folder",e)}async createFolder(e){return this.post("/folder",e)}async editFolder(e,t){return this.put(`/folder/${e}`,t)}async deleteFolder(e){await this.del(`/folder/${e}`)}async listTags(e={}){return this._listRequest("/tag",e)}async createTag(e){return this.post("/tag",e)}async editTag(e,t){return this.put(`/tag/${e}`,t)}async deleteTag(e){await this.del(`/tag/${e}`)}async tagEntity(e,t,s){await this.post(`/${e}/${t}/tag`,{tag:s.join(",")})}async untagEntity(e,t,s){await this.post(`/${e}/${t}/untag`,{tag:s.join(",")})}async dgChangeLayout(e,t){await this.post(`/displaygroup/${e}/action/changeLayout`,{layoutId:t})}async dgOverlayLayout(e,t){await this.post(`/displaygroup/${e}/action/overlayLayout`,{layoutId:t})}async dgRevertToSchedule(e){await this.post(`/displaygroup/${e}/action/revertToSchedule`)}async dgCollectNow(e){await this.post(`/displaygroup/${e}/action/collectNow`)}async dgSendCommand(e,t){await this.post(`/displaygroup/${e}/action/command`,{commandId:t})}async editDisplayGroup(e,t){return this.put(`/displaygroup/${e}`,t)}}class g extends Error{constructor(e,t,s,a){super(`CMS API ${e} ${t} → ${s}: ${a}`),this.name="CmsApiError",this.method=e,this.path=t,this.status=s,this.detail=a}}const ee=D.version,j="/api/v2/player";var R;let x=((R=I.data)==null?void 0:R.playerApiBase)||j,B=x;function te(i){x=i.replace(/\/+$/,""),B=x}export{Q as CmsApiClient,g as CmsApiError,V as EventEmitter,u as LOG_LEVELS,B as PLAYER_API,M as SHELL_ONLY_KEYS,ee as VERSION,Y as applyCmsLogLevel,I as config,A as createLogger,J as extractPwaConfig,Z as fetchWithRetry,F as getLogLevel,W as isDebug,U as mapCmsLogLevel,z as registerLogSink,H as setLogLevel,te as setPlayerApi,X as unregisterLogSink};
5
- //# sourceMappingURL=index-BglkPErZ.js.map
5
+ //# sourceMappingURL=index-C122u38X.js.map