stormcloud-video-player 0.2.11 → 0.2.12

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.
@@ -176,10 +176,18 @@ async function getBrowserID(clientInfo) {
176
176
  if (typeof crypto !== "undefined" && crypto.subtle && crypto.subtle.digest) {
177
177
  try {
178
178
  await crypto.subtle.digest("SHA-256", new Uint8Array([1, 2, 3]));
179
- const hashBuffer = await crypto.subtle.digest(
180
- "SHA-256",
181
- new TextEncoder().encode(fingerprintString)
182
- );
179
+ let encodedData;
180
+ if (typeof TextEncoder !== "undefined") {
181
+ encodedData = new TextEncoder().encode(fingerprintString);
182
+ } else {
183
+ const utf8 = unescape(encodeURIComponent(fingerprintString));
184
+ const buffer = new Uint8Array(utf8.length);
185
+ for (let i = 0; i < utf8.length; i++) {
186
+ buffer[i] = utf8.charCodeAt(i);
187
+ }
188
+ encodedData = buffer;
189
+ }
190
+ const hashBuffer = await crypto.subtle.digest("SHA-256", encodedData);
183
191
  const hashArray = Array.from(new Uint8Array(hashBuffer));
184
192
  const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join("");
185
193
  cachedBrowserId = hashHex;
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/utils/tracking.ts"],"sourcesContent":["import type { ClientInfo, TrackingData, HeartbeatData } from \"../types\";\n\nlet cachedBrowserId: string | null = null;\n\nexport function getClientInfo(): ClientInfo {\n const ua = navigator.userAgent;\n const platform = navigator.platform;\n const vendor = navigator.vendor || \"\";\n const maxTouchPoints = navigator.maxTouchPoints || 0;\n const memory = (navigator as any).deviceMemory || null;\n const hardwareConcurrency = navigator.hardwareConcurrency || 1;\n\n const screenInfo = {\n width: screen?.width,\n height: screen?.height,\n availWidth: screen?.availWidth,\n availHeight: screen?.availHeight,\n orientation: (screen?.orientation as any)?.type || \"\",\n pixelDepth: screen?.pixelDepth,\n };\n\n let deviceType: \"tv\" | \"mobile\" | \"tablet\" | \"desktop\" = \"desktop\";\n let brand = \"Unknown\";\n let os = \"Unknown\";\n let model = \"\";\n let isSmartTV = false;\n let isAndroid = false;\n let isWebView = false;\n let isWebApp = false;\n\n if (ua.includes(\"Web0S\")) {\n brand = \"LG\";\n os = \"webOS\";\n isSmartTV = true;\n deviceType = \"tv\";\n const webosMatch = ua.match(/Web0S\\/([^\\s]+)/);\n model = webosMatch ? `webOS ${webosMatch[1]}` : \"webOS TV\";\n } else if (ua.includes(\"Tizen\")) {\n brand = \"Samsung\";\n os = \"Tizen\";\n isSmartTV = true;\n deviceType = \"tv\";\n const tizenMatch = ua.match(/Tizen\\/([^\\s]+)/);\n const tvMatch = ua.match(/(?:Smart-TV|SMART-TV|TV)/i) ? \"Smart TV\" : \"\";\n model = tizenMatch\n ? `Tizen ${tizenMatch[1]} ${tvMatch}`.trim()\n : \"Tizen TV\";\n } else if (ua.includes(\"Philips\")) {\n brand = \"Philips\";\n os = \"Saphi\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\"Sharp\") || ua.includes(\"AQUOS\")) {\n brand = \"Sharp\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (\n ua.includes(\"Android\") &&\n (ua.includes(\"Sony\") || vendor.includes(\"Sony\"))\n ) {\n brand = \"Sony\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (\n ua.includes(\"Android\") &&\n (ua.includes(\"NetCast\") || ua.includes(\"LG\"))\n ) {\n brand = \"LG\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\" Roku\") || ua.includes(\"Roku/\")) {\n brand = \"Roku\";\n os = \"Roku OS\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\"AppleTV\")) {\n brand = \"Apple\";\n os = \"tvOS\";\n isSmartTV = true;\n deviceType = \"tv\";\n }\n\n if (ua.includes(\"Android\")) {\n isAndroid = true;\n os = \"Android\";\n deviceType = /Mobile/.test(ua) ? \"mobile\" : \"tablet\";\n\n if (\n ua.includes(\"Android\") &&\n (maxTouchPoints === 0 ||\n ua.includes(\"Google TV\") ||\n ua.includes(\"XiaoMi\"))\n ) {\n deviceType = \"tv\";\n isSmartTV = true;\n brand = brand === \"Unknown\" ? \"Android TV\" : brand;\n }\n\n const androidModelMatch = ua.match(/\\(([^)]*Android[^)]*)\\)/);\n if (androidModelMatch && androidModelMatch[1]) {\n model = androidModelMatch[1];\n }\n }\n\n if (/iPad|iPhone|iPod/.test(ua)) {\n os = \"iOS\";\n deviceType = \"mobile\";\n brand = \"Apple\";\n if (navigator.maxTouchPoints > 1 && /iPad/.test(ua)) {\n deviceType = \"tablet\";\n }\n }\n\n if (!isAndroid && !isSmartTV && !/Mobile/.test(ua)) {\n if (ua.includes(\"Windows\")) {\n os = \"Windows\";\n deviceType = \"desktop\";\n } else if (ua.includes(\"Mac\") && !/iPhone/.test(ua)) {\n os = \"macOS\";\n deviceType = \"desktop\";\n if (maxTouchPoints > 1) deviceType = \"tablet\";\n } else if (ua.includes(\"Linux\")) {\n os = \"Linux\";\n deviceType = \"desktop\";\n }\n }\n\n if (brand === \"Unknown\") {\n if (vendor.includes(\"Google\") || ua.includes(\"Chrome\")) brand = \"Google\";\n if (vendor.includes(\"Apple\")) brand = \"Apple\";\n if (vendor.includes(\"Samsung\") || ua.includes(\"SM-\")) brand = \"Samsung\";\n }\n\n isWebView = /wv|WebView|Linux; U;/.test(ua);\n\n if (window?.outerHeight === 0 && window?.outerWidth === 0) {\n isWebView = true;\n }\n\n isWebApp =\n window.matchMedia(\"(display-mode: standalone)\").matches ||\n (window.navigator as any).standalone === true ||\n window.screen?.orientation?.angle !== undefined;\n\n return {\n brand,\n os,\n model: model || ua.substring(0, 50) + \"...\",\n deviceType,\n isSmartTV,\n isAndroid,\n isWebView,\n isWebApp,\n domain: window.location.hostname,\n origin: window.location.origin,\n path: window.location.pathname,\n userAgent: ua,\n vendor,\n platform,\n screen: screenInfo,\n hardwareConcurrency,\n deviceMemory: memory,\n maxTouchPoints,\n language: navigator.language,\n languages: navigator.languages?.join(\",\") || \"\",\n cookieEnabled: navigator.cookieEnabled,\n doNotTrack: navigator.doNotTrack || \"\",\n referrer: document.referrer,\n visibilityState: document.visibilityState,\n };\n}\n\nexport async function getBrowserID(clientInfo: ClientInfo): Promise<string> {\n if (cachedBrowserId) {\n return cachedBrowserId;\n }\n\n const fingerprintString = JSON.stringify(clientInfo);\n\n if (typeof crypto !== \"undefined\" && crypto.subtle && crypto.subtle.digest) {\n try {\n await crypto.subtle.digest(\"SHA-256\", new Uint8Array([1, 2, 3]));\n\n const hashBuffer = await crypto.subtle.digest(\n \"SHA-256\",\n new TextEncoder().encode(fingerprintString)\n );\n const hashArray = Array.from(new Uint8Array(hashBuffer));\n const hashHex = hashArray\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n cachedBrowserId = hashHex;\n return hashHex;\n } catch (error) {\n console.warn(\n \"[StormcloudVideoPlayer] crypto.subtle.digest not supported, using fallback hash\"\n );\n }\n }\n\n let hash = 0;\n for (let i = 0; i < fingerprintString.length; i++) {\n const char = fingerprintString.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash;\n }\n\n const fallbackHash = Math.abs(hash).toString(16).padStart(8, \"0\");\n const timestamp = Date.now().toString(16).padStart(12, \"0\");\n const random = Math.random().toString(16).substring(2, 14).padStart(12, \"0\");\n\n cachedBrowserId = (fallbackHash + timestamp + random).padEnd(64, \"0\");\n return cachedBrowserId;\n}\n\nexport async function sendInitialTracking(licenseKey?: string): Promise<void> {\n try {\n const clientInfo = getClientInfo();\n const browserId = await getBrowserID(clientInfo);\n\n const trackingData: TrackingData = {\n browserId,\n ...clientInfo,\n };\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (licenseKey) {\n headers[\"Authorization\"] = `Bearer ${licenseKey}`;\n }\n\n const response = await fetch(\n \"https://adstorm.co/api-adstorm-dev/adstorm/player-tracking/track\",\n {\n method: \"POST\",\n headers,\n body: JSON.stringify(trackingData),\n }\n );\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n await response.json();\n } catch (error) {\n console.error(\n \"[StormcloudVideoPlayer] Error sending initial tracking data:\",\n error\n );\n }\n}\n\nexport async function sendHeartbeat(licenseKey?: string): Promise<void> {\n try {\n const clientInfo = getClientInfo();\n const browserId = await getBrowserID(clientInfo);\n\n const heartbeatData: HeartbeatData = {\n browserId,\n timestamp: new Date().toISOString(),\n };\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (licenseKey) {\n headers[\"Authorization\"] = `Bearer ${licenseKey}`;\n }\n\n const response = await fetch(\n \"https://adstorm.co/api-adstorm-dev/adstorm/player-tracking/heartbeat\",\n {\n method: \"POST\",\n headers,\n body: JSON.stringify(heartbeatData),\n }\n );\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n await response.json();\n } catch (error) {\n console.error(\"[StormcloudVideoPlayer] Error sending heartbeat:\", error);\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../src/utils/tracking.ts"],"sourcesContent":["import type { ClientInfo, TrackingData, HeartbeatData } from \"../types\";\n\nlet cachedBrowserId: string | null = null;\n\nexport function getClientInfo(): ClientInfo {\n const ua = navigator.userAgent;\n const platform = navigator.platform;\n const vendor = navigator.vendor || \"\";\n const maxTouchPoints = navigator.maxTouchPoints || 0;\n const memory = (navigator as any).deviceMemory || null;\n const hardwareConcurrency = navigator.hardwareConcurrency || 1;\n\n const screenInfo = {\n width: screen?.width,\n height: screen?.height,\n availWidth: screen?.availWidth,\n availHeight: screen?.availHeight,\n orientation: (screen?.orientation as any)?.type || \"\",\n pixelDepth: screen?.pixelDepth,\n };\n\n let deviceType: \"tv\" | \"mobile\" | \"tablet\" | \"desktop\" = \"desktop\";\n let brand = \"Unknown\";\n let os = \"Unknown\";\n let model = \"\";\n let isSmartTV = false;\n let isAndroid = false;\n let isWebView = false;\n let isWebApp = false;\n\n if (ua.includes(\"Web0S\")) {\n brand = \"LG\";\n os = \"webOS\";\n isSmartTV = true;\n deviceType = \"tv\";\n const webosMatch = ua.match(/Web0S\\/([^\\s]+)/);\n model = webosMatch ? `webOS ${webosMatch[1]}` : \"webOS TV\";\n } else if (ua.includes(\"Tizen\")) {\n brand = \"Samsung\";\n os = \"Tizen\";\n isSmartTV = true;\n deviceType = \"tv\";\n const tizenMatch = ua.match(/Tizen\\/([^\\s]+)/);\n const tvMatch = ua.match(/(?:Smart-TV|SMART-TV|TV)/i) ? \"Smart TV\" : \"\";\n model = tizenMatch\n ? `Tizen ${tizenMatch[1]} ${tvMatch}`.trim()\n : \"Tizen TV\";\n } else if (ua.includes(\"Philips\")) {\n brand = \"Philips\";\n os = \"Saphi\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\"Sharp\") || ua.includes(\"AQUOS\")) {\n brand = \"Sharp\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (\n ua.includes(\"Android\") &&\n (ua.includes(\"Sony\") || vendor.includes(\"Sony\"))\n ) {\n brand = \"Sony\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (\n ua.includes(\"Android\") &&\n (ua.includes(\"NetCast\") || ua.includes(\"LG\"))\n ) {\n brand = \"LG\";\n os = \"Android TV\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\" Roku\") || ua.includes(\"Roku/\")) {\n brand = \"Roku\";\n os = \"Roku OS\";\n isSmartTV = true;\n deviceType = \"tv\";\n } else if (ua.includes(\"AppleTV\")) {\n brand = \"Apple\";\n os = \"tvOS\";\n isSmartTV = true;\n deviceType = \"tv\";\n }\n\n if (ua.includes(\"Android\")) {\n isAndroid = true;\n os = \"Android\";\n deviceType = /Mobile/.test(ua) ? \"mobile\" : \"tablet\";\n\n if (\n ua.includes(\"Android\") &&\n (maxTouchPoints === 0 ||\n ua.includes(\"Google TV\") ||\n ua.includes(\"XiaoMi\"))\n ) {\n deviceType = \"tv\";\n isSmartTV = true;\n brand = brand === \"Unknown\" ? \"Android TV\" : brand;\n }\n\n const androidModelMatch = ua.match(/\\(([^)]*Android[^)]*)\\)/);\n if (androidModelMatch && androidModelMatch[1]) {\n model = androidModelMatch[1];\n }\n }\n\n if (/iPad|iPhone|iPod/.test(ua)) {\n os = \"iOS\";\n deviceType = \"mobile\";\n brand = \"Apple\";\n if (navigator.maxTouchPoints > 1 && /iPad/.test(ua)) {\n deviceType = \"tablet\";\n }\n }\n\n if (!isAndroid && !isSmartTV && !/Mobile/.test(ua)) {\n if (ua.includes(\"Windows\")) {\n os = \"Windows\";\n deviceType = \"desktop\";\n } else if (ua.includes(\"Mac\") && !/iPhone/.test(ua)) {\n os = \"macOS\";\n deviceType = \"desktop\";\n if (maxTouchPoints > 1) deviceType = \"tablet\";\n } else if (ua.includes(\"Linux\")) {\n os = \"Linux\";\n deviceType = \"desktop\";\n }\n }\n\n if (brand === \"Unknown\") {\n if (vendor.includes(\"Google\") || ua.includes(\"Chrome\")) brand = \"Google\";\n if (vendor.includes(\"Apple\")) brand = \"Apple\";\n if (vendor.includes(\"Samsung\") || ua.includes(\"SM-\")) brand = \"Samsung\";\n }\n\n isWebView = /wv|WebView|Linux; U;/.test(ua);\n\n if (window?.outerHeight === 0 && window?.outerWidth === 0) {\n isWebView = true;\n }\n\n isWebApp =\n window.matchMedia(\"(display-mode: standalone)\").matches ||\n (window.navigator as any).standalone === true ||\n window.screen?.orientation?.angle !== undefined;\n\n return {\n brand,\n os,\n model: model || ua.substring(0, 50) + \"...\",\n deviceType,\n isSmartTV,\n isAndroid,\n isWebView,\n isWebApp,\n domain: window.location.hostname,\n origin: window.location.origin,\n path: window.location.pathname,\n userAgent: ua,\n vendor,\n platform,\n screen: screenInfo,\n hardwareConcurrency,\n deviceMemory: memory,\n maxTouchPoints,\n language: navigator.language,\n languages: navigator.languages?.join(\",\") || \"\",\n cookieEnabled: navigator.cookieEnabled,\n doNotTrack: navigator.doNotTrack || \"\",\n referrer: document.referrer,\n visibilityState: document.visibilityState,\n };\n}\n\nexport async function getBrowserID(clientInfo: ClientInfo): Promise<string> {\n if (cachedBrowserId) {\n return cachedBrowserId;\n }\n\n const fingerprintString = JSON.stringify(clientInfo);\n\n if (typeof crypto !== \"undefined\" && crypto.subtle && crypto.subtle.digest) {\n try {\n await crypto.subtle.digest(\"SHA-256\", new Uint8Array([1, 2, 3]));\n\n let encodedData: BufferSource;\n if (typeof TextEncoder !== \"undefined\") {\n encodedData = new TextEncoder().encode(fingerprintString);\n } else {\n const utf8 = unescape(encodeURIComponent(fingerprintString));\n const buffer = new Uint8Array(utf8.length);\n for (let i = 0; i < utf8.length; i++) {\n buffer[i] = utf8.charCodeAt(i);\n }\n encodedData = buffer;\n }\n\n const hashBuffer = await crypto.subtle.digest(\"SHA-256\", encodedData);\n const hashArray = Array.from(new Uint8Array(hashBuffer));\n const hashHex = hashArray\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n cachedBrowserId = hashHex;\n return hashHex;\n } catch (error) {\n console.warn(\n \"[StormcloudVideoPlayer] crypto.subtle.digest not supported, using fallback hash\"\n );\n }\n }\n\n let hash = 0;\n for (let i = 0; i < fingerprintString.length; i++) {\n const char = fingerprintString.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash;\n }\n\n const fallbackHash = Math.abs(hash).toString(16).padStart(8, \"0\");\n const timestamp = Date.now().toString(16).padStart(12, \"0\");\n const random = Math.random().toString(16).substring(2, 14).padStart(12, \"0\");\n\n cachedBrowserId = (fallbackHash + timestamp + random).padEnd(64, \"0\");\n return cachedBrowserId;\n}\n\nexport async function sendInitialTracking(licenseKey?: string): Promise<void> {\n try {\n const clientInfo = getClientInfo();\n const browserId = await getBrowserID(clientInfo);\n\n const trackingData: TrackingData = {\n browserId,\n ...clientInfo,\n };\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (licenseKey) {\n headers[\"Authorization\"] = `Bearer ${licenseKey}`;\n }\n\n const response = await fetch(\n \"https://adstorm.co/api-adstorm-dev/adstorm/player-tracking/track\",\n {\n method: \"POST\",\n headers,\n body: JSON.stringify(trackingData),\n }\n );\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n await response.json();\n } catch (error) {\n console.error(\n \"[StormcloudVideoPlayer] Error sending initial tracking data:\",\n error\n );\n }\n}\n\nexport async function sendHeartbeat(licenseKey?: string): Promise<void> {\n try {\n const clientInfo = getClientInfo();\n const browserId = await getBrowserID(clientInfo);\n\n const heartbeatData: HeartbeatData = {\n browserId,\n timestamp: new Date().toISOString(),\n };\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (licenseKey) {\n headers[\"Authorization\"] = `Bearer ${licenseKey}`;\n }\n\n const response = await fetch(\n \"https://adstorm.co/api-adstorm-dev/adstorm/player-tracking/heartbeat\",\n {\n method: \"POST\",\n headers,\n body: JSON.stringify(heartbeatData),\n }\n );\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n await response.json();\n } catch (error) {\n console.error(\"[StormcloudVideoPlayer] Error sending heartbeat:\", error);\n }\n}\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "stormcloud-video-player",
3
- "version": "0.2.11",
3
+ "version": "0.2.12",
4
4
  "main": "lib/index.js",
5
5
  "typings": "lib/index.d.ts",
6
6
  "scripts": {