@vkontakte/videoplayer-statistics 1.0.110-dev.0bbcbb01d.0 → 1.0.110-dev.92c807ec7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/esnext.cjs CHANGED
@@ -1,6 +1,6 @@
1
1
  /**
2
- * @vkontakte/videoplayer-statistics v1.0.110-dev.0bbcbb01d.0
3
- * Tue, 16 Jun 2026 11:51:52 GMT
2
+ * @vkontakte/videoplayer-statistics v1.0.110-dev.92c807ec7.0
3
+ * Fri, 19 Jun 2026 04:22:28 GMT
4
4
  * https://st.mycdn.me/static/vkontakte-videoplayer/1-0-110/doc/
5
5
  */
6
- var e=Object.defineProperty,t=Object.getOwnPropertyNames,n=Object.getOwnPropertyDescriptor,r=Object.prototype.hasOwnProperty;function i(e){return this[e]}var a=a=>{var s=(o??=new WeakMap).get(a),c;if(s)return s;if(s=e({},`__esModule`,{value:!0}),a&&typeof a==`object`||typeof a==`function`)for(var l of t(a))r.call(s,l)||e(s,l,{get:i.bind(a,l),enumerable:!(c=n(a,l))||c.enumerable});return o.set(a,s),s},o,s=e=>e;function c(e,t){this[e]=s.bind(null,t)}var l=(t,n)=>{for(var r in n)e(t,r,{get:n[r],enumerable:!0,configurable:!0,set:c.bind(n,r)})},u={};l(u,{VERSION:()=>O,ThinOneStat:()=>Te,SeekAction:()=>g,Quality:()=>p,OneStat:()=>we,MediascopePixel:()=>_t,InteractiveInterfaceClick:()=>_,ContentType:()=>m,ConnectionType:()=>h,ApiEnv:()=>f}),module.exports=a(u);var d=require(`@vkontakte/videoplayer-shared`),f;(e=>{e.PROD=`prod`,e.VK_ALIAS=`vk_alias`,e.VIDEOTEST=`videotest`,e.TEST=`test`,e.OKCDN=`okcdn`,e.AUTO=`auto`})(f||={});var p;(e=>{e.Q144P=`mobile`,e.Q240P=`lowest`,e.Q360P=`low`,e.Q480P=`medium`,e.Q720P=`high`,e.Q1080P=`fullhd`,e.Q1440P=`quadhd`,e.Q2160P=`ultrahd`,e.UNKNOWN=`unknown`})(p||={});var m;(e=>{e.MP4=`mp4`,e.DASH=`dash`,e.DASH_SEP=`dash_sep`,e.ONDEMAND_DASH=`ondemand_dash`,e.HLS=`hls`,e.HLS_FMP4=`hls_fmp4`,e.ONDEMAND_HLS=`ondemand_hls`,e.WEBM=`webm`,e.AV1=`av1`,e.ONDEMAND_DASH_LIVE=`ondemand_dash_live`,e.ONDEMAND_HLS_LIVE=`ondemand_hls_live`,e.WEBRTC=`webrtc`,e.UNKNOWN=`unknown`,e.RTMP=`rtmp`})(m||={});var h;(e=>{e.HTTP1=`http1`,e.HTTP2=`http2`,e.HTTP3=`http3`})(h||={});var g;(e=>{e.SLIDER=`slider`,e.RICH=`rich`,e.DOUBLE_TAP=`double_tap`,e.TIME_CODE=`time_code`,e.EPISODE=`episode`,e.REWIND=`rewind`,e.LIVE=`live`,e.UNKNOWN=`unknown`})(g||={});var _;(e=>{e.GRAPH_SHOW=`iGraphShow`,e.GRAPH_HIDE=`iGraphHide`,e.NEXT_AREA=`iNextChapterArea`,e.NEXT_BUTTON=`iNextChapterBtn`,e.WATCH_AGAIN=`iWatchAgainBtn`})(_||={});var v={action_play:`ap`,action_play_interactive:`api`,first_bytes:`fb`,player_ready:`pr`,first_frame:`ff`,seek:`sk`,watch_coverage_record:`wr`,watch_coverage_live:`wl`,empty_buffer:`eb`,action_stop:`as`,close_at_empty_buffer:`cb`,content_error:`er`,player_interface_click:`pc`,next_movie:`nm`,track_switch:`ts`,failover:`fo`,quality:`qt`},y={vsid:`si`,isid:`is`,vid:`vi`,place:`pl`,quality:`qt`,cdn_host:`ch`,stat_type:`st`,param:`pm`,vk_app_id:`va`,track_code:`tc`,connection_type:`cnt`,connection_reused:`cr`,cached_data:`cd`,live:`lv`,muted:`mu`,mode:`mo`,subtitles:`sb`,failover:`fo`,download_speed:`ds`,manual_quality:`mq`,ref_domain:`rd`,direct_url:`du`},b={unknown:`un`,mobile:`m`,lowest:`ls`,low:`l`,medium:`md`,high:`h`,fullhd:`f`,quadhd:`q`,ultrahd:`u`},x={pip:`pi`,fullscreen:`fs`,external:`ex`,prefetch:`pr`,airplay:`ap`,chromecast:`cc`,invisible:`iv`,minimal:`minimal`},S=require(`@vkontakte/videoplayer-shared`),C=`CIOPGQJGDIHBABABA`,w={prod:`https://api.ok.ru`,vk_alias:`https://api.mycdn.me`,videotest:`https://videotestapi.ok.ru/api`,test:`https://apitest.ok.ru`,okcdn:`https://api.okcdn.ru`,auto:``},ee={apiEnv:`vk_alias`,apiKey:C,apiBaseUrl:null,apiUvRestBaseUrl:`https://uvapi.okcdn.ru`,apiCollector:`ok.mobile.apps.video`,apiForticom:!0,apiUvRest:!1,requestRetryCount:1,useVsid64:!1,firstFlushTime:5e3,flushDebounceTime:1e4,flushMaxWait:6e4,storageExpiration:2160*60*1e3,watchCoverageInterval:15e3,disabledOperations:[`quality`],disabledCustomFields:[],shorten:!0,maxLoops:100,embedUrlParams:[],useBeacon:!0,synchronizeTime:!0,debugLog:!1,clearStorageAtUnload:!0,useVKComIsMobileLogic:!1,backoff:{start:1e3,factor:1.5,max:300*1e3,random:.1},useKeepalive:!1,watchCoverageTimeoutFix:!0,watchCoverageExactTime:!1},te=e=>S.fillWithDefault(e,ee);function ne(){let e=`0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz`.split(``),t=Array(36),n=0,r,i;for(i=0;i<36;i++)i===8||i===13||i===18||i===23?t[i]=`-`:i===14?t[i]=`4`:(n<=2&&(n=33554432+Math.random()*16777216|0),r=n&15,n>>=4,t[i]=e[i===19?r&3|8:r]);return t.join(``)}var T=require(`@vkontakte/videoplayer-shared`),re=require(`@vkontakte/videoplayer-shared`),ie=()=>Math.floor(Math.random()*2**32).toString(36).padStart(13,`0`),E=()=>Math.floor(Math.random()*2**64).toString(36).padStart(13,`0`);class D{static vsid;static getVSID(){return D.vsid}static generateVSID(e){return D.vsid=e?E():ie(),D.vsid}}var ae=e=>{let{operation:t,custom:n}=e,r=Object.fromEntries(Object.entries(n).map(([e,t])=>{let n=y[e]??e,r=t;return t&&(e===`mode`?r=x[t]??t:e===`quality`&&(r=b[t]??t)),[n,r]}));return{...e,operation:v[t]??t,custom:r}},O=`1.0.110-dev.0bbcbb01d.0`,k=require(`@vkontakte/videoplayer-shared`),oe=/Mobile|mini|Fennec|Android|iP(ad|od|hone)|opera (mini|mobi)/i;class se{id=`forticom`;params;authToken;sessionKey;authorizePromise;refreshAuthTokenPromise;consequentAuthErrors=0;authorized$=new k.ValueSubject(!1);backoffTimeoutId;constructor(e){this.params=e}async authorize(e){return this.authToken=e??await this.refreshAuthToken(),this._authorizeWithBackoff()}logBeacon(e){let t=this.createLogParams(e),n=this.sessionKey;n&&(this.params.apiTransport.sendBeacon(`log.externalLog`,t,n)||this.logRequest(e).catch(()=>{}))}async logRequest(e){let t=`log.externalLog`,n=this.createLogParams(e),r=this.sessionKey??await this._authorizeWithBackoff();if(!r)return;let i=async(e,a=this.params.config.requestRetryCount)=>{try{return await this.params.apiTransport.sendRequest(t,n,r)}catch(n){if(!n||!(`error_code`in n)){this.params.error$.next({id:`logRequestUnknown`,category:k.ErrorCategory.NETWORK,message:`Unknown ${t} error`,thrown:n});return}let r=n?.error_code;switch(r){case 102:case 103:case 104:return this.authorized$.next(!1),this.sessionKey=await this._authorizeWithBackoff(),a>0?i(e,a-1):void 0;case 401:return this.authorized$.next(!1),this.authToken=await this.refreshAuthToken(),this.sessionKey=await this._authorizeWithBackoff(),a>0?i(e,a-1):void 0;default:this.params.error$.next({id:`LogRequest#${r}`,category:k.ErrorCategory.EXTERNAL_API,message:`${t} error`,data:n});return}}};return i(e)}destroy(){k.getWindow().clearTimeout(this.backoffTimeoutId),this.backoffTimeoutId=0}async refreshAuthToken(){if(this.params.refreshAuthToken)return this.refreshAuthTokenPromise||=this.params.refreshAuthToken().finally(()=>{this.refreshAuthTokenPromise=void 0}),this.refreshAuthTokenPromise}createLogParams(e){let t=`WEB`,n=!1,r=`@vkontakte/videoplayer-statistics:${O}`;if(this.params.config.useVKComIsMobileLogic){let{appVersion:e}=k.getWindow().navigator;n=oe.test(e)}else this.params.mobile&&(n=!0);return n&&(t=`M_WEB`),this.params.vktvVersion&&(t=`SMART_TV`,r=this.params.vktvVersion),{collector:this.params.config.apiCollector,data:JSON.stringify({application:r,platform:t,items:this.params.config.shorten?e.map(ae):e})}}_authorizeWithBackoff(){if(!this.consequentAuthErrors)return this._authorize();let e=k.getExponentialDelay(this.consequentAuthErrors,this.params.config.backoff);return new Promise(t=>{this.backoffTimeoutId||=k.getWindow().setTimeout(()=>{this._authorize().then(t).catch(e=>this.params.error$.next({id:`AuthorizeBackoff`,category:k.ErrorCategory.NETWORK,message:`Otherwise unhandled error in authorization`,thrown:e})).finally(()=>this.backoffTimeoutId=0)},e)})}async _authorize(){if(this.authorizePromise)return this.authorizePromise;this.sessionKey=void 0,this.authorized$.next(!1);let e={session_data:{version:2,device_id:this.params.deviceId,client_version:O.split(`-`)[0],client_type:`SDK_JS`}};return this.authToken!==void 0&&(e.session_data.auth_token=this.authToken,e.session_data.version=3),this.authorizePromise=this.params.apiTransport.sendRequest(`auth.anonymLogin`,e).then(e=>((!e||!e.session_key)&&this.params.error$.next({id:`AuthorizeFailed`,category:k.ErrorCategory.EXTERNAL_API,message:`No session key`,data:e}),this.sessionKey=e?.session_key??void 0,this.sessionKey)).catch(async e=>{this.sessionKey=void 0;let t=e?.error_code;switch(t){case 401:return this.authToken=await this.refreshAuthToken(),this._authorizeWithBackoff()}t?this.params.error$.next({id:`Authorize#${t}`,category:k.ErrorCategory.EXTERNAL_API,message:`authorize error`,data:e}):this.params.error$.next({id:`AuthorizeUnknown`,category:k.ErrorCategory.NETWORK,message:`authorize error`,thrown:e})}).finally(()=>{this.authorizePromise=void 0,this.consequentAuthErrors=this.sessionKey?0:this.consequentAuthErrors+1,this.authorized$.next(this.sessionKey!==void 0)}),this.authorizePromise}}class ce{id=`uvrest`;authorized$=new k.ValueSubject(!1);params;statToken;authorizePromise;isDestroyed=!1;constructor(e){this.params=e}async authorize(e){if(!this.isDestroyed)return e?(this.statToken=e,this.authorized$.next(!0),e):this.authorizePromise?this.authorizePromise:(this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize().finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.statToken!==void 0)}),this.authorizePromise)}async logRequest(e){if(!e.length)return;let t=this.buildBody(e);try{let e=await this.getValidToken();if(!e){this.params.error$.next({id:`OneStat:UvRestApi:LogRequestNoToken`,category:k.ErrorCategory.NETWORK,message:`No stat token available for log request`});return}await this.params.apiTransport.sendJsonRequest(`/uv/stat/externalLog`,t,{Authorization:`Bearer ${e}`})}catch(e){await this.handleLogRequestError(e)&&await this.retryLogRequestAfterRecovery(t)}}logBeacon(e){if(!e.length)return;let t=this.statToken;if(!t)return;let n=this.buildBody(e);this.params.apiTransport.sendJsonBeacon(`/uv/stat/externalLog`,n,{Authorization:`Bearer ${t}`})}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.statToken=void 0,this.authorized$.next(!1)}async getValidToken(){return this.statToken?this.statToken:this.authorize()}async executeAuthorize(){try{let e=await this.params.getAuthToken();if(this.isDestroyed)return;if(!e){this.params.error$.next({id:`OneStat:UvRestApi:NoToken`,category:k.ErrorCategory.EXTERNAL_API,message:`getAuthToken returned no token`});return}return this.statToken=e,this.statToken}catch(e){if(this.isDestroyed)return;this.params.error$.next({id:`OneStat:UvRestApi:AuthorizeFailed`,category:k.ErrorCategory.NETWORK,message:`Failed to get stat token`,thrown:e});return}}async handleLogRequestError(e){let t=e?.status;return t===401?(this.params.error$.next({id:`OneStat:UvRestApi:TokenExpired`,category:k.ErrorCategory.EXTERNAL_API,message:`Stat token expired (401), refreshing`,thrown:e}),this.authorized$.next(!1),this.statToken=void 0,await this.authorize()!==void 0):(this.params.error$.next({id:`OneStat:UvRestApi:LogRequestFailed#${t??`unknown`}`,category:k.ErrorCategory.EXTERNAL_API,message:`Log request failed with status ${t}`,thrown:e}),!1)}async retryLogRequestAfterRecovery(e){let t=this.statToken;if(!t){this.params.error$.next({id:`OneStat:UvRestApi:RetryNoToken`,category:k.ErrorCategory.NETWORK,message:`Cannot retry log request: no token after recovery`});return}try{await this.params.apiTransport.sendJsonRequest(`/uv/stat/externalLog`,e,{Authorization:`Bearer ${t}`})}catch(e){this.params.error$.next({id:`OneStat:UvRestApi:RetryFailed`,category:k.ErrorCategory.EXTERNAL_API,message:`Log request failed after token refresh`,thrown:e})}}buildBody(e){let t=`WEB`,n=!1,r=`@vkontakte/videoplayer-statistics:${O}`;if(this.params.config.useVKComIsMobileLogic){let{appVersion:e}=k.getWindow().navigator;n=oe.test(e)}else this.params.mobile&&(n=!0);return n&&(t=`M_WEB`),this.params.vktvVersion&&(t=`SMART_TV`,r=this.params.vktvVersion),[{application:r,platform:t,user_id:this.params.userId,items:this.params.config.shorten?e.map(ae):e}]}}var A=require(`@vkontakte/videoplayer-shared`),j=require(`@vkontakte/videoplayer-shared`);class le{apiKey;apiBaseUrl;apiUvRestBaseUrl;apiEnv;timeSynchronisation;isApiBaseUrlFetched;params;constructor(e){this.params=e,this.apiKey=e.config.apiKey,this.apiEnv=e.config.apiEnv,this.apiBaseUrl=e.config.apiBaseUrl??w[this.apiEnv],this.apiUvRestBaseUrl=e.config.apiUvRestBaseUrl,this.timeSynchronisation=e.timeSynchronisation,this.isApiBaseUrlFetched=!1}async resolveApiBaseUrl(){if(this.apiEnv!==`auto`||this.isApiBaseUrlFetched)return this.apiBaseUrl;try{let e=atob(`aHR0cHM6Ly9kbnMuZ29vZ2xlL3Jlc29sdmU/bmFtZT12aWRlby5fZW5kcG9pbnQub2sucnUmdHlwZT1UWFQ=`),t=(await(await A.fetchPolyfilled(e,{method:`GET`,mode:`cors`,cache:`no-cache`})).json())?.Answer[0]?.data;if(!t)throw Error(`Wrong DNS response`);return t}catch(e){return this.params.error$.next({id:`OneStat:ApiTransport:resolveApiBaseUrl`,category:j.ErrorCategory.NETWORK,message:`Unhandled resolve api base url error`,thrown:e}),w.vk_alias}finally{this.isApiBaseUrlFetched=!0}}sendBeacon(e,t,n){if(!j.getWindow().Blob||!j.getWindow().navigator.sendBeacon)return!1;let r=this._prepareQueryParams({method:e,queryParams:t,sessionKey:n}),i=j.getWindow().Blob,a=new i([r.toString()],{type:`application/x-www-form-urlencoded`});try{return j.getWindow().navigator.sendBeacon(`${this.apiBaseUrl}/fb.do`,a)}catch(n){this.params.error$.next({id:`OneStat:ApiTransport:sendBeacon`,category:j.ErrorCategory.NETWORK,message:`Unhandled beacon error`,thrown:n,data:{method:e,params:t}})}return!1}async sendRequest(e,t,n){let r=j.now(),i=i=>{if(i instanceof TypeError&&[`Failed to fetch`,`NetworkError when attempting to fetch resource.`].includes(i.message)){this.params.error$.next({id:`Network`,category:j.ErrorCategory.NETWORK,message:`Request failed`,thrown:i});return}this.params.error$.next({id:`OneStat:ApiTransport:sendRequest`,category:j.ErrorCategory.NETWORK,message:`Unhandled request error`,thrown:i,data:{method:e,params:t,sessionKey:n,time:j.now()-r}})};this.apiBaseUrl=await this.resolveApiBaseUrl();let a={method:`post`,headers:{"Content-type":`application/x-www-form-urlencoded`},body:this._prepareQueryParams({method:e,queryParams:t,sessionKey:n})};return this.params.config.useKeepalive&&(a.keepalive=!0),A.fetchPolyfilled(`${this.apiBaseUrl}/fb.do`,a).then(e=>{let t=Number(e.headers.get(`content-length`))===0,n=new Date(e.headers.get(`date`)??``).getTime(),a=j.now()-r;if(isFinite(n)&&this.timeSynchronisation?.addServerTime(n,a),!t)return e.json().then(e=>Object.prototype.hasOwnProperty.call(e,`error_msg`)?Promise.reject(e):e,i)},i)}sendJsonBeacon(e,t,n){let r=`${this.apiUvRestBaseUrl}${e}`;try{A.fetchPolyfilled(r,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t),keepalive:!0}).catch(t=>{this.params.error$.next({id:`OneStat:ApiTransport:sendJsonBeacon`,category:j.ErrorCategory.NETWORK,message:`Keepalive request failed`,thrown:t,data:{path:e}})})}catch(t){this.params.error$.next({id:`OneStat:ApiTransport:sendJsonBeacon`,category:j.ErrorCategory.NETWORK,message:`Failed to initiate keepalive request`,thrown:t,data:{path:e}})}}async sendJsonRequest(e,t,n){let r=j.now(),i=t=>{if(t instanceof TypeError&&[`Failed to fetch`,`NetworkError when attempting to fetch resource.`].includes(t.message)){this.params.error$.next({id:`Network`,category:j.ErrorCategory.NETWORK,message:`JSON request failed`,thrown:t});return}this.params.error$.next({id:`OneStat:ApiTransport:sendJsonRequest`,category:j.ErrorCategory.NETWORK,message:`Unhandled JSON request error`,thrown:t,data:{path:e,time:j.now()-r}})},a={method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t)};return this.params.config.useKeepalive&&(a.keepalive=!0),A.fetchPolyfilled(`${this.apiUvRestBaseUrl}${e}`,a).then(e=>{let t=Number(e.headers.get(`content-length`))===0,n=new Date(e.headers.get(`date`)??``).getTime(),a=j.now()-r;if(isFinite(n)&&this.timeSynchronisation?.addServerTime(n,a),!e.ok)return Promise.reject({status:e.status,statusText:e.statusText});if(!t)return e.json().catch(i)},i)}_prepareQueryParams(e){let t=new URLSearchParams({format:`JSON`,method:e.method,application_key:this.apiKey});return e.sessionKey!==void 0&&t.append(`session_key`,e.sessionKey),Object.entries(e.queryParams).forEach(([e,n])=>t.append(e,typeof n==`string`?n:JSON.stringify(n))),t}}var M=require(`@vkontakte/videoplayer-shared`),ue=e=>t=>t.timestamp+e>=M.now();class de{params;error$;userSalt;isPaused=!1;loggerDebugLog;queues=[];constructor(e){this.params=e,this.error$=e.error$,this.userSalt=e.userSalt,this.loggerDebugLog=e.debugLogger.createComponentLog(`stat logger`);for(let t of e.apis)this.queues.push(this.createQueueState(t));this.housekeepStorage()}createQueueState(e){let t=`onestat_events_${e.id}`,n=M.debounceFn(()=>this.safeFlushQueue(a),this.params.config.firstFlushTime,{maxWait:this.params.config.firstFlushTime}),r=M.safeStorage.isPersistent()?this.params.config.flushMaxWait:0,i=M.debounceFn(()=>this.safeFlushQueue(a),this.params.config.flushDebounceTime,{maxWait:r}),a={api:e,storageKey:t,unsaltedStorage:[],lastVsid:void 0,firstFlush:n,debouncedFlush:i,subscription:e.authorized$.subscribe(e=>{e&&i()})};return a}isUrgent=e=>{let{operation:t}=e;return[`action_play`,`action_play_interactive`,`watch_coverage_record`,`watch_coverage_live`,`action_stop`,`close_at_empty_buffer`].includes(t)};safeFlushQueue(e){try{this.flushQueue(e)}catch(t){this.error$.next({id:`LoggerError`,category:M.ErrorCategory.WTF,message:t?String(t):`Unknown logger error`,thrown:t,data:{storageKey:e.storageKey}})}}readFromStorage(e){let t=M.safeStorage.get(e);try{return t?JSON.parse(t):{}}catch{}return{}}addToStorage(e,t){if(!this.userSalt){e.unsaltedStorage.push(t);return}let n=this.readFromStorage(e.storageKey),r=(n[this.userSalt]??[]).filter(ue(this.params.config.storageExpiration));M.safeStorage.set(e.storageKey,JSON.stringify({...n,[this.userSalt]:[...r,t]}))}getFromStorage(e){return this.userSalt?(this.readFromStorage(e.storageKey)[this.userSalt]??[]).filter(ue(this.params.config.storageExpiration)):e.unsaltedStorage}markStorageSent(e){if(!this.userSalt){e.unsaltedStorage=[];return}let t=this.readFromStorage(e.storageKey);delete t[this.userSalt],M.safeStorage.set(e.storageKey,JSON.stringify(t))}housekeepStorage(){for(let e of this.queues){let t=this.readFromStorage(e.storageKey);for(let[n,r]of Object.entries(t)){let i=r.filter(ue(this.params.config.storageExpiration));i.length?t[n]=i:delete t[n],this.loggerDebugLog({message:`${r.length} retrieved from storage, ${i.length} of them actual (${e.storageKey})`})}M.safeStorage.set(e.storageKey,JSON.stringify(t))}}log(e){for(let t of this.queues)this.addToStorage(t,e),!this.isPaused&&(this.isUrgent(e)?this.flushQueue(t):t.lastVsid===e.custom.vsid?t.debouncedFlush():t.firstFlush(),t.lastVsid=e.custom.vsid)}flush(e={wantBeacon:!1,clearStorage:!0}){for(let t of this.queues)this.flushQueue(t,e)}flushQueue(e,{wantBeacon:t,clearStorage:n}={wantBeacon:!1,clearStorage:!0}){let r=this.getFromStorage(e);r.length!==0&&(e.api.authorized$.getValue()?(t&&this.params.config.useBeacon?(this.loggerDebugLog({message:`Flushing ${r.length} events through beacon (${e.storageKey})`}),e.api.logBeacon(r)):(this.loggerDebugLog({message:`Flushing ${r.length} events (${e.storageKey})`}),e.api.logRequest(r)),n&&this.markStorageSent(e)):e.api.authorize(),e.firstFlush.cancel(),e.debouncedFlush.cancel())}pause(){this.isPaused=!0;for(let e of this.queues)e.debouncedFlush.cancel()}resume(){this.isPaused=!1;for(let e of this.queues)e.debouncedFlush()}destroy(){for(let e of this.queues)e.subscription.unsubscribe(),e.firstFlush.cancel(),e.debouncedFlush.cancel();this.queues=[]}}var N=require(`@vkontakte/videoplayer-core`),fe=require(`@vkontakte/videoplayer-shared`),pe=e=>e&&{[N.VideoQuality.INVARIANT]:`unknown`,[N.VideoQuality.Q_144P]:`mobile`,[N.VideoQuality.Q_240P]:`lowest`,[N.VideoQuality.Q_360P]:`low`,[N.VideoQuality.Q_480P]:`medium`,[N.VideoQuality.Q_576P]:`unknown`,[N.VideoQuality.Q_720P]:`high`,[N.VideoQuality.Q_1080P]:`fullhd`,[N.VideoQuality.Q_1440P]:`quadhd`,[N.VideoQuality.Q_2160P]:`ultrahd`,[N.VideoQuality.Q_4320P]:`unknown`}[e],me=e=>e&&{[N.HttpConnectionType.HTTP1]:`http1`,[N.HttpConnectionType.HTTP2]:`http2`,[N.HttpConnectionType.QUIC]:`http3`}[e],he=e=>{if(e!==void 0)switch(e){case N.VideoFormat.MPEG:return`mp4`;case N.VideoFormat.DASH:case N.VideoFormat.DASH_LIVE:case N.VideoFormat.DASH_STREAMS:return`dash`;case N.VideoFormat.DASH_SEP:return`dash_sep`;case N.VideoFormat.DASH_ONDEMAND:return`ondemand_dash`;case N.VideoFormat.DASH_WEBM:case N.VideoFormat.DASH_LIVE_WEBM:return`webm`;case N.VideoFormat.DASH_WEBM_AV1:return`av1`;case N.VideoFormat.DASH_LIVE_CMAF:return`ondemand_dash_live`;case N.VideoFormat.HLS:case N.VideoFormat.HLS_LIVE:return`hls`;case N.VideoFormat.HLS_FMP4:return`hls_fmp4`;case N.VideoFormat.HLS_ONDEMAND:return`ondemand_hls`;case N.VideoFormat.HLS_LIVE_CMAF:return`ondemand_hls_live`;case N.VideoFormat.WEB_RTC_LIVE:return`webrtc`;default:return fe.assertNever(e)}},ge=(e,t)=>{if(t)return`minimal`;switch(e){case void 0:case N.Surface.NONE:case N.Surface.INLINE:return;case N.Surface.FULLSCREEN:return`fullscreen`;case N.Surface.SECOND_SCREEN:return`chromecast`;case N.Surface.PIP:return`pip`;case N.Surface.INVISIBLE:return`invisible`;default:return fe.assertNever(e)}},_e=e=>{switch(e){case`slow-2g`:return`poor`;case`2g`:return`poor`;case`3g`:return`good`;case`4g`:return`excellent`}};class ve{offset=void 0;constructor(e){this.offset=e}getOffset(){return this.offset??0}setOffset(e){this.offset=e}now(){return Date.now()+(this.offset??0)}date(e=new Date){return e.setTime(e.getTime()+(this.offset??0)),e}addServerTime(e,t){let n=e-Date.now()-t/2;if(Math.abs(n)<1e3){this.offset=0;return}if(this.offset===void 0)this.offset=Math.round(n);else{let e=.2;this.offset=Math.round(e*n+(1-e)*this.offset)}}}var ye=require(`@vkontakte/videoplayer-core`),P=require(`@vkontakte/videoplayer-shared`),be=(e,t)=>new P.Observable(n=>{let r=new P.Subscription,i=P.merge(P.fromEvent(window,`beforeunload`),e.events.willDestruct$),a=new P.ValueSubject(void 0),o;r.add(e.info.isLive$.pipe(P.filterChanged()).subscribe(t=>{o&&(o.unsubscribe(),a.next(void 0)),o=t?e.info.liveTime$.pipe(P.map(e=>e&&e/1e3)).subscribe(a):e.info.position$.subscribe(a),r.add(o)}));let{playing$:s,paused$:c,looped$:l}=e.events,u=e.events.willSeek$.pipe(P.filter(()=>e.info.playbackState$.getValue()===ye.PlaybackState.PLAYING)),d=e.events.seeked$.pipe(P.filter(()=>e.info.playbackState$.getValue()===ye.PlaybackState.PLAYING)),f=!1,p=new P.Subject;r.add(u.subscribe(()=>{f||p.next(),f=!0})),r.add(d.subscribe(()=>f=!1));let m=new P.Subject,h=new P.Subject,g=()=>t.watchCoverageExactTime?e.info.isLive$.getValue()?e.getExactLiveTime():e.getExactTime():a.getValue()??0,_=e.info.stallStartTime$.pipe(P.filterChanged(),P.filter(e=>e>0)),v=P.merge(s,d,m).pipe(P.map(()=>g())),y=P.merge(l,P.merge(c,p,i,h,_).pipe(P.map(()=>g()))),b,x=e=>{b=e},S=e=>{let t=e;P.isNullable(b)||b===t||P.isNullable(t)||(n.next({from:b,to:t}),b=void 0)},C=()=>{h.next(),m.next()};if(r.add(v.subscribe(x)),r.add(y.subscribe(S)),t.forceInterval&&isFinite(t.forceInterval))if(t.watchCoverageTimeoutFix){let e,n=()=>{e!==void 0&&(window.clearTimeout(e),e=void 0)};r.add(v.subscribe(()=>{n(),e=window.setTimeout(C,t.forceInterval)})),r.add(y.subscribe(()=>n())),r.add(()=>n())}else{let e=0;r.add(v.subscribe(()=>e=window.setTimeout(C,t.forceInterval))),r.add(y.subscribe(()=>window.clearTimeout(e)))}return r}),F=require(`@vkontakte/videoplayer-core`),I=require(`@vkontakte/videoplayer-shared`),xe=(e,t,n)=>{let r=new I.Subject,i=r.pipe(I.debounce(0),I.filterChanged()),a=!1,o=!1,s=!1;return t(e.inPiP$,e=>{a=e,r.next(e?F.Surface.PIP:o?F.Surface.FULLSCREEN:s?F.Surface.INLINE:void 0)}),t(e.inFullscreen$,e=>{o=e,r.next(e?F.Surface.FULLSCREEN:a?F.Surface.PIP:s?F.Surface.INLINE:void 0)}),n&&t(n.info.surface$,e=>{switch(e){case F.Surface.SECOND_SCREEN:case F.Surface.NONE:case F.Surface.INLINE:!o&&!a&&(e===F.Surface.INLINE&&(s=!0),r.next(e));break}}),{surface$:i}},Se=`_one-stat_deviceId`,Ce=(e,t)=>e&&t?`${e}_${t}`:e??t,L=()=>{let e=new T.Subscription;return{subscription:e,subscribe:(t,n)=>{t&&e.add(t.subscribe(n))}}};class we{apis;logger;config;subscription=new T.Subscription;beforeunloadSubscription;timeSynchronisation;debugLogger=new re.Logger;oneStatDebugLog=this.debugLogger.createComponentLog(`onestat`);loopCounter=0;disabled=!1;experimental;get vsid$(){return new T.ValueSubject(D.getVSID())}isid$=new T.ValueSubject(void 0);zenUid$=new T.ValueSubject(void 0);statContext;isEmbed;embedParent;uiEvents;player;seekAction$=new T.ValueSubject(`unknown`);deviceId;constructor(e,t){this.statContext=e,this.config=te(t.config??{}),T.isNonNullable(t.apiEnv)&&(this.config.apiEnv=t.apiEnv),this.config.synchronizeTime&&(this.timeSynchronisation=new ve);let n=new T.Subject;this.experimental={error$:n};let r=T.safeStorage.get(Se);this.statContext.deviceId?this.deviceId=this.statContext.deviceId:r?this.deviceId=r:(this.deviceId=ne(),T.safeStorage.set(Se,this.deviceId)),this.resetViewSession(),t.useIsid&&this.isid$.next(t.isid??E()),this.zenUid$.next(t.zenUid);let i=new le({config:this.config,error$:n,timeSynchronisation:this.timeSynchronisation}),a=[];this.config.apiForticom&&a.push(new se({config:this.config,apiTransport:i,refreshAuthToken:t.refreshAuthToken,mobile:this.statContext.mobile??!1,deviceId:this.deviceId,error$:n,vktvVersion:t.vktvVersion})),this.config.apiUvRest&&(t.getUvRestAuthToken?a.push(new ce({config:this.config,apiTransport:i,error$:n,userId:this.statContext.userId,getAuthToken:t.getUvRestAuthToken,mobile:this.statContext.mobile??!1,vktvVersion:t.vktvVersion})):n.next({id:`OneStat:Init:MissingGetUvRestAuthToken`,category:re.ErrorCategory.WTF,message:`oldStatToUvRest enabled but getUvRestAuthToken callback not provided`})),this.apis=a,this.logger=new de({config:this.config,debugLogger:this.debugLogger,apis:this.apis,error$:n,userSalt:t.userSalt});let{isEmbed:o,topOrigin:s}=re.detectEmbed();this.isEmbed=o,this.embedParent=s?new URL(s).hostname:void 0,this.subscribe()}updateContext(e){this.statContext={...this.statContext,...e}}attachTo(e){let t=new T.Subscription,n=(e,n)=>t.add(e.subscribe(n)),r,i,a,o=!1,s=!1,c=()=>{o||(o=!0,T.isNonNullable(r)&&(r={startedAt:T.now(),liveAtStall:!!this.statContext.liveEdge}))};n(e.events.willStart$,()=>{let t=e.info.position$.getValue();this.logActionPlay({position:t}),this.statContext.projectId&&this.logActionPlayInteractive()}),n(e.events.looped$,()=>{this.resetViewSession(),++this.loopCounter>this.config.maxLoops&&(this.disabled=!0),this.logActionPlay({position:e.info.position$.getValue()})}),n(e.events.willSeek$,()=>{T.isNonNullable(r)&&o&&this.logEmptyBuffer({duration:T.now()-r.startedAt,liveAtStall:r.liveAtStall}),r=void 0,s=!0}),n(e.events.seeked$,()=>{!s&&T.isNonNullable(r)&&o&&(this.logEmptyBuffer({duration:T.now()-r.startedAt,liveAtStall:r.liveAtStall}),r=void 0),s=!1;let t=this.seekAction$.getValue();this.logSeek({action:t===`rich`?`slider`:t,time:e.info.position$.getValue()}),this.seekAction$.next(`unknown`)}),n(e.events.paused$,()=>{this.logPause({position:e.info.position$.getValue()})}),n(e.events.willResume$,()=>{this.logPlay()}),n(e.events.started$,()=>{this.statContext.clip&&this.logWatchCoverageRecord({start:0,end:0})}),n(e.events.playing$,()=>{this.logPlaying()}),n(be(e,{forceInterval:this.config.watchCoverageInterval,watchCoverageTimeoutFix:this.config.watchCoverageTimeoutFix,watchCoverageExactTime:this.config.watchCoverageExactTime}),t=>{let n=e.info.isLive$.getValue(),r=e.info.atLiveEdge$.getValue(),i={start:parseFloat(t.from.toFixed(3)),end:parseFloat(t.to.toFixed(3))};if(r){let t=e.info.liveLatency$.getValue()??0,n=e.info.currentBuffer$.getValue()?.end??0,r=e.info.liveBufferTime$.getValue()??0,a=Math.round((t-(n-r))*1e3),o=Math.round(t*1e3);a&&o&&(i.latency=a,i.bufferLatency=o)}n?this.logWatchCoverageLive(i):this.logWatchCoverageRecord(i)}),n(e.info.stallStartTime$,e=>{e>0?r={startedAt:e,liveAtStall:!!this.statContext.liveEdge}:(T.isNonNullable(r)&&o&&this.logEmptyBuffer({duration:Math.round(T.now()-r.startedAt),liveAtStall:r.liveAtStall}),r=void 0)});let l=!1;t.add(e.events.fatalError$.pipe(T.once()).subscribe(()=>l=!0)),n(e.events.willStop$,()=>{if(e.info.isBuffering$.getValue()&&o){let e=T.isNonNullable(r)?T.now()-r.startedAt:void 0;this.logCloseAtEmptyBuffer({duration:e??0,liveAtStall:r?.liveAtStall}),r=void 0}else l||this.logActionStop()}),n(e.events.managedError$,({id:e})=>{this.logError({fatal:!1,errorType:e})}),n(e.events.fatalError$,({id:e})=>{this.logError({fatal:!0,errorType:e})}),n(e.events.firstBytes$,e=>{i=T.now(),this.logFirstBytes({time:e})}),n(e.events.willStart$,()=>a=T.now()),n(e.info.currentBuffer$,e=>{!o&&e&&e.end-e.start>0&&T.isNonNullable(i)&&(this.logPlayerReady({duration:T.now()-i}),c())}),n(e.events.firstFrame$,()=>{T.isNonNullable(i)&&!o&&(this.logPlayerReady({duration:T.now()-i}),c()),T.isNonNullable(a)&&this.logFirstFrame({time:T.now()-a})});let u;n(e.info.currentVideoStream$,e=>{e&&(this.updateContext({videoCodec:e.codec}),u&&e.id!==u&&this.logTrackSwitch(e),u=e.id)});let d;n(e.info.currentAudioStream$,e=>{e&&(this.updateContext({audioCodec:e.codec}),d&&e.id!==d&&this.logTrackSwitch(e),d=e.id)}),n(e.info.atLiveEdge$,e=>this.updateContext({liveEdge:e})),n(T.combine({muted:e.info.muted$,volume:e.info.volume$}),({muted:e,volume:t})=>this.updateContext({audible:!e&&t>0})),n(e.info.currentQuality$,e=>{let t=pe(e);this.updateContext({quality:t}),t&&this.logQuality(t)}),n(e.info.isAutoQualityEnabled$,e=>this.updateContext({autoQuality:e})),n(e.info.currentFormat$,e=>this.updateContext({contentType:he(e)})),n(e.info.currentPlaybackRate$,e=>this.updateContext({rate:e})),n(e.info.is3DVideo$,e=>this.updateContext({is3d:e}));let f;return n(e.info.hostname$,e=>{let t=f!==void 0&&f!==e;this.updateContext({cdnHostname:e,failover:t}),t&&this.logFailover(e),f=e}),n(e.info.throughputEstimation$,e=>this.updateContext({downloadSpeed:e})),n(e.info.httpConnectionType$,e=>{this.statContext.firstConnectionType||this.updateContext({firstConnectionType:me(e)}),this.updateContext({connectionType:me(e)})}),n(e.info.httpConnectionReused$,e=>{T.isNullable(this.statContext.firstConnectionReused)&&this.updateContext({firstConnectionReused:e}),this.updateContext({connectionReused:e})}),n(e.info.surface$,e=>{this.uiEvents||this.updateContext({mode:ge(e,this.statContext.isPreviewPlayerView)})}),n(T.combine({current:e.info.currentTextTrack$,available:e.info.availableTextTracks$}),({current:e,available:t})=>{let n=t.find(({id:t})=>t===e),r=n&&(n.isAuto?`${n.language}_auto`:n.language);this.updateContext({subtitles:r?.split(`.`)[0]})}),this.player=e,this.uiEvents&&this.attachToUi(this.uiEvents),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToUi(e){this.uiEvents=e;let{subscription:t,subscribe:n}=L();if(!this.player)return t;n(e.actionRewind$,()=>{this.player&&(this.resetViewSession(),this.logActionPlay({position:this.player.info.position$.getValue()}))}),n(e.actionSeek$,this.seekAction$);let{surface$:r}=xe(this.uiEvents,n,this.player);return n(r,e=>{this.updateContext({mode:ge(e,this.statContext.isPreviewPlayerView)})}),n(e.actionSetSubtitle$,e=>this.updateContext({subtitles:e?.split(`.`)[0]})),n(e.nextMovie$,e=>this.logNextMovie(e)),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToAds(e){let{subscription:t,subscribe:n}=L();return n(e.slotRequested$,()=>this.logAdSlotRequest()),n(e.started$,e=>this.logAdStarted(e)),n(e.paused$,()=>this.logAdPaused()),n(e.resumed$,()=>this.logAdResumed()),n(e.ended$,()=>this.logAdEnded()),n(e.skipped$,()=>this.logAdSkipped()),n(e.clicked$,()=>this.logAdClicked()),n(e.error$,e=>this.logError({errorType:e,fatal:!1})),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToInteractive(e){let{subscription:t,subscribe:n}=L();return n(e.click$,e=>this.logInterfaceClick(e)),n(e.nextMovie$,e=>this.logNextMovie(e)),this.resubscribeBeforeunload(),this.subscription.add(t),t}authorize(e){return Promise.all(this.apis.map(t=>t.authorize(e)))}pause(){this.logger.pause(),this.oneStatDebugLog({message:`paused`})}resume(){this.logger.resume(),this.oneStatDebugLog({message:`resumed`})}destroy(){this.logger.flush(),this.subscription.unsubscribe(),this.apis.forEach(e=>e.destroy()),this.logger.destroy()}resetViewSession(){this.oneStatDebugLog({message:`VSID reset`}),D.generateVSID(this.config.useVsid64)}getDeviceId(){return this.deviceId}setFieldBlacklist(){}logInited(){}logReady(e){this.log({operation:`player_ready`,param:String(e.time)})}logStarted(...e){this.logActionPlay(...e)}logPlay(){this.log({operation:`play_toggle`})}logPlaying(){this.log({operation:`playing`})}logPause(e){this.log({operation:`pause`,param:String(Math.round(e.position))})}logSeek(e){this.log({operation:`seek`,param:e.action,time:e.time})}logFirstBytes(e){this.log({operation:`first_bytes`,param:String(e.time)})}logFirstFrame(e){this.log({operation:`first_frame`,param:String(e.time)})}logError(e){this.log({operation:`content_error`,param:`${e.fatal?`fatal`:`recoverable`}_${e.errorType}`})}logWatchCoverageRecord(e){this.log({operation:`watch_coverage_record`,param:`${e.start}-${e.end}`},{in_history:this.statContext.inHistory?1:void 0})}logWatchCoverageLive(e){let t=this.timeSynchronisation?.getOffset()??0,n=e.start+t,r=e.end+t;this.log({operation:`watch_coverage_live`,param:`${n}-${r}`},{in_history:this.statContext.inHistory?1:void 0,latency:e.latency,buffer_latency:e.bufferLatency})}logEmptyBuffer(e){this.log({operation:`empty_buffer`,param:String(e.duration)},T.isNonNullable(e.liveAtStall)?{live:e.liveAtStall?1:void 0}:{})}logDownloadSpeed(){}logAdSlotRequest(){this.log({operation:`adv`,param:`slot_request`})}logAdStarted(e){this.log({operation:`adv`,param:e})}logAdPaused(){this.log({operation:`adv`,param:`pause`})}logAdResumed(){this.log({operation:`adv`,param:`resume`})}logAdEnded(){this.log({operation:`adv`,param:`ended`})}logAdSkipped(){this.log({operation:`adv`,param:`skip`})}logAdClicked(){this.log({operation:`adv`,param:`click`})}logInterfaceClick(e){this.log({operation:`player_interface_click`,param:e})}logNextMovie(e){this.log({operation:`next_movie`,param:String(e)})}subscribe(){this.resubscribeBeforeunload();let e=T.getWindow().navigator.connection;e&&`onchange`in e&&`effectiveType`in e&&this.subscription.add(T.merge(T.fromEvent(e,`change`),T.observableFrom([`init`])).subscribe(()=>this.updateContext({network:_e(e.effectiveType)}))),this.config.debugLog&&this.debugLogger.log$.subscribe(e=>{console.debug(`%c stat `,`background:#fa6470;`,`component: ${e.component}.`,e.message)})}resubscribeBeforeunload(){this.beforeunloadSubscription?.unsubscribe(),this.beforeunloadSubscription=T.fromEvent(window,`beforeunload`).subscribe(()=>this.logger.flush({wantBeacon:!0,clearStorage:this.config.clearStorageAtUnload})),this.subscription.add(this.beforeunloadSubscription)}logPlayerReady(e){this.log({operation:`player_ready`,param:String(Math.round(e.duration))})}logActionPlay(e){this.log({operation:`action_play`,param:String(Math.round(e.position))})}logActionPlayInteractive(){this.statContext.projectId&&this.log({operation:`action_play_interactive`,param:String(this.statContext.movieId)},{vid:this.statContext.projectId})}logFailover(e){this.log({operation:`failover`,param:e})}logCloseAtEmptyBuffer(e){this.log({operation:`close_at_empty_buffer`,param:String(Math.round(e.duration))},T.isNonNullable(e.liveAtStall)?{live:e.liveAtStall?1:void 0}:{})}logActionStop(){this.log({operation:`action_stop`})}logTrackSwitch(e){this.log({operation:`track_switch`,param:e.id})}logQuality(e){this.log({operation:`quality`,param:e})}log(e,t={}){if(this.disabled){this.oneStatDebugLog({message:`operation ${e.operation} but statistics is disabled`});return}if(this.config.disabledOperations.includes(e.operation)){this.oneStatDebugLog({message:`operation ${e.operation} but it is disabled`});return}this.oneStatDebugLog({message:`operation ${e.operation} ${e.param}`});let n=this.createLogItem(e,t);this.logger.log(n)}createLogItem({operation:e,param:t,time:n},r={}){let i=this.timeSynchronisation?.now()??T.now(),a=this.vsid$.getValue();T.assertNonNullable(a);let o=this.isid$.getValue(),s=e===`empty_buffer`||e===`close_at_empty_buffer`?this.statContext.connectionType:this.statContext.firstConnectionType,c=this.statContext.firstConnectionReused,l,u;if(this.isEmbed||this.statContext.place===`embed`){l=this.embedParent;let e=[...new URLSearchParams(location.search).entries()].filter(([e,t])=>this.config.embedUrlParams.includes(e));u=new URLSearchParams(e).toString()}else if(this.statContext.place===`direct`){let e=this.statContext.refDomain||document.referrer;l=e&&d.urlCanParsePolyfilled(URL,e)?new URL(e).hostname:e,u=location.href.substring(0,1024)}let f={vsid:a,isid:o,vid:this.statContext.movieId,uid:this.zenUid$.getValue(),ct:this.statContext.contentType,place:this.isEmbed?`embed`:this.statContext.place,quality:this.statContext.quality,cdn_host:this.statContext.cdnHostname,stat_type:this.statContext.autoplay===!0?`auto`:this.statContext.autoplay===!1?``:void 0,param:t,vk_app_id:this.statContext.vkAppId,track_code:this.statContext.trackCode,connection_type:s,connection_reused:c===!0?1:c===!1?0:void 0,cached_data:this.statContext.cached===!0?1:this.statContext.cached===!1?0:void 0,live:this.statContext.liveEdge?1:void 0,muted:this.statContext.audible===!1?1:void 0,mode:this.statContext.mode,subtitles:this.statContext.subtitles,download_speed:this.statContext.downloadSpeed,manual_quality:this.statContext.autoQuality?void 0:1,ref_domain:l,direct_url:u,rate:this.statContext.rate===1||T.isNullable(this.statContext.rate)?void 0:this.statContext.rate.toFixed(2),view_360:this.statContext.is3d?1:void 0,codec_info:Ce(this.statContext.videoCodec,this.statContext.audioCodec),aid:this.statContext.albumId,vk_playlist_id:this.statContext.vkPlaylistId,...r};for(let e of this.config.disabledCustomFields)delete f[e];return{operation:e,type:1,time:n,network:this.statContext.network,timestamp:i,custom:f}}}var Te={};l(Te,{VERSION:()=>O,ThinOneStat:()=>dt,ActionSeekType:()=>Oe});var R=require(`@vkontakte/videoplayer-shared`),z=require(`@vkontakte/videoplayer-core`),Ee=require(`@vkontakte/videoplayer-shared`),De;(e=>{e.AUTO=`auto`,e.AUTO_POOR=`auto_poor`,e.AUTO_RICH=`auto_rich`})(De||={});var Oe;(e=>{e.SLIDER=`slider`,e.RICH=`rich`,e.DOUBLE_TAP=`double_tap`,e.TIME_CODE=`time_code`,e.EPISODE=`episode`,e.REWIND=`rewind`,e.LIVE=`live`,e.UNKNOWN=`unknown`})(Oe||={});var ke=`CIOPGQJGDIHBABABA`,B={prod:`https://api.ok.ru`,vk_alias:`https://api.mycdn.me`,videotest:`https://videotestapi.ok.ru/api`,test:`https://apitest.ok.ru`,okcdn:`https://api.okcdn.ru`,auto:``},Ae={apiKey:ke,apiEnv:`okcdn`,apiBaseUrl:null,apiUvRestBaseUrl:`https://uvapi.okcdn.ru`,apiForticom:!0,apiUvRest:!1,requestRetryCount:3,useVsid64:!1,flushFirstTime:5e3,flushDebounceTime:1e4,flushMaxWait:6e4,flushRetryFirstDelay:3e3,flushRetryMaxDelay:18e5,storageExpiration:2160*60*1e3,clearStorageAtUnload:!1,disabledEvents:[],disabledParams:[],synchronizeTime:!0,useBeacon:!1,alwaysSendDesktop:!1,watchCoverageHeartbeatInterval:3e4,watchCoverageExactTime:!1,watchedNDefaultTargetDuration:1e3,telemetryInterval:3e4,downloadBytesMaxTime:3e4},je=e=>Ee.fillWithDefault(e,Ae),V=require(`@vkontakte/videoplayer-shared`);function Me(e){let t=Error(`HTTP ${e.status}: ${e.statusText}`);return t.status=e.status,t.statusText=e.statusText,t.response=e,t}function H(){let e=Error(`Request timeout`);return e.isTimeout=!0,e}function Ne(e){let t=Error(e.error_msg);return t.errorData=e,t}function U(e){if(!(!e||typeof e!=`object`)){if(`errorData`in e)return e.errorData?.error_code;if(`error_code`in e){let t=e.error_code;return typeof t==`number`?t:void 0}}}var Pe=({application:e,platform:t,product:n,...r})=>r;class Fe{id=`forticom`;authorized$;params;authToken;sessionKey;authorizePromise;refreshAuthTokenPromise;isDestroyed=!1;constructor(e){this.params=e,this.authorized$=new V.ValueSubject(!1)}async authorize(e){return this.authToken=e??await this.refreshAuthToken(),this.doAuthorize()}logBeacon(e){if(!e.length)return;let t=this.createLogParams(e),n=this.sessionKey;n&&(this.params.apiTransport.sendBeacon(`log.logUvStat`,t,n)||this.logRequest(e).catch(()=>{}))}async logRequest(e){if(!e.length)return;let t=`log.logUvStat`,n=this.createLogParams(e),r={};try{let e=await this.getValidSessionKey();if(!e){this.params.errors$.next({id:`ThinOneStat:Api:LogRequestNoSession`,category:V.ErrorCategory.NETWORK,message:`No session key available for log request`});return}await this.params.apiTransport.sendRequest(t,n,e,r)}catch(e){await this.handleLogRequestError(e,t,n)&&await this.retryLogRequestAfterSessionRecovery(t,n)}}async getValidSessionKey(){return this.sessionKey?this.sessionKey:this.doAuthorize()}async handleLogRequestError(e,t,n){let r=U(e);return r===void 0?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestUnknown`,category:V.ErrorCategory.NETWORK,message:`Unknown ${t} error`,thrown:e,data:{params:n}}),!1):this.isSessionExpiredError(r)?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestSessionError#${r}`,category:V.ErrorCategory.EXTERNAL_API,message:`Session error: ${r}`,data:{error:e,code:r}}),this.authorized$.next(!1),this.sessionKey=void 0,await this.doAuthorize()!==void 0):r===401?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestAuthLogin#${r}`,category:V.ErrorCategory.EXTERNAL_API,message:`Auth token expired`,data:{error:e,code:r}}),this.authorized$.next(!1),this.authToken=await this.refreshAuthToken(),this.sessionKey=void 0,await this.doAuthorize()!==void 0):(this.params.errors$.next({id:`ThinOneStat:Api:LogRequest#${r}`,category:V.ErrorCategory.EXTERNAL_API,message:`${t} error`,data:{error:e,code:r,params:n}}),!1)}async retryLogRequestAfterSessionRecovery(e,t){let n=this.sessionKey;if(!n){this.params.errors$.next({id:`ThinOneStat:Api:LogRequestRetryFailed`,category:V.ErrorCategory.NETWORK,message:`Cannot retry log request: no session key after recovery`});return}try{await this.params.apiTransport.sendRequest(e,t,n)}catch(e){let t=U(e);this.params.errors$.next({id:`ThinOneStat:Api:LogRequestRetryFailed#${t??`unknown`}`,category:V.ErrorCategory.EXTERNAL_API,message:`Log request failed after session recovery`,thrown:e,data:{originalError:e,retryCode:t}})}}isSessionExpiredError(e){return!!e&&[102,103,104].includes(e)}async doAuthorize(e=1){if(!this.isDestroyed)return this.authorizePromise?this.authorizePromise:(this.sessionKey=void 0,this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize(e).finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.sessionKey!==void 0)}),this.authorizePromise)}async executeAuthorize(e){let t=this.buildAuthParams();try{let e=await this.params.apiTransport.sendRequest(`auth.anonymLogin`,t);if(this.isDestroyed)return;if(!e?.session_key){this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeFailed`,category:V.ErrorCategory.EXTERNAL_API,message:`No session key in response`,data:e});return}return this.sessionKey=e.session_key,this.sessionKey}catch(t){if(this.isDestroyed)return;let n=U(t);if(n===401&&this.params.refreshAuthToken){if(this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeAuthLogin`,category:V.ErrorCategory.EXTERNAL_API,message:`Auth token expired during authorization`,data:{error:t,code:n,attempt:e}}),e>=2){this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeMaxAttemptsReached`,category:V.ErrorCategory.EXTERNAL_API,message:`Authorization failed after ${e} attempts`,data:{error:t,code:n}});return}return this.authToken=await this.refreshAuthToken(),this.doAuthorize(e+1)}n?this.params.errors$.next({id:`ThinOneStat:Api:Authorize#${n}`,category:V.ErrorCategory.EXTERNAL_API,message:`Authorize error`,data:{error:t,code:n}}):this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeUnknown`,category:V.ErrorCategory.NETWORK,message:`Authorize error`,thrown:t});return}}buildAuthParams(){let e={session_data:{version:2,device_id:this.params.deviceId,client_version:this.getClientVersion(),client_type:`SDK_JS`}};return this.authToken!==void 0&&(e.session_data.auth_token=this.authToken,e.session_data.version=3),e}getClientVersion(){return O}async refreshAuthToken(){if(this.params.refreshAuthToken)return this.refreshAuthTokenPromise||=this.params.refreshAuthToken().catch(e=>{throw this.params.errors$.next({id:`ThinOneStat:Api:RefreshAuthTokenFailed`,category:V.ErrorCategory.NETWORK,message:`Failed to refresh auth token`,thrown:e}),e}).finally(()=>{this.refreshAuthTokenPromise=void 0}),this.refreshAuthTokenPromise}createLogParams(e){let t=e[0],n=t.application??`@vkontakte/videoplayer-statistics:${this.getClientVersion()}`;return this.params.vktvVersion&&(n=this.params.vktvVersion),{collector:`ok.mobile.apps.video`,uv_stat_data:JSON.stringify({application:n,platform:t.platform,product:t.product,events:e.map(Pe)})}}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.refreshAuthTokenPromise=void 0,this.sessionKey=void 0,this.authorized$.next(!1)}}class Ie{id=`uvrest`;authorized$;params;statToken;authorizePromise;isDestroyed=!1;constructor(e){this.params=e,this.authorized$=new V.ValueSubject(!1)}async authorize(e){if(!this.isDestroyed)return e?(this.statToken=e,this.authorized$.next(!0),e):this.authorizePromise?this.authorizePromise:(this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize().finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.statToken!==void 0)}),this.authorizePromise)}async logRequest(e){if(!e.length)return;let t=this.buildBody(e);try{let e=await this.getValidToken();if(!e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:LogRequestNoToken`,category:V.ErrorCategory.NETWORK,message:`No stat token available for log request`});return}await this.params.apiTransport.sendJsonRequest(`/uv/stat/logUvStat`,t,{Authorization:`Bearer ${e}`})}catch(e){await this.handleLogRequestError(e)&&await this.retryLogRequestAfterRecovery(t)}}logBeacon(e){if(!e.length)return;let t=this.statToken;if(!t)return;let n=this.buildBody(e);this.params.apiTransport.sendJsonBeacon(`/uv/stat/logUvStat`,n,{Authorization:`Bearer ${t}`})}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.statToken=void 0,this.authorized$.next(!1)}async getValidToken(){return this.statToken?this.statToken:this.authorize()}async executeAuthorize(){try{let e=await this.params.getAuthToken();if(this.isDestroyed)return;if(!e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:NoToken`,category:V.ErrorCategory.EXTERNAL_API,message:`getAuthToken returned no token`});return}return this.statToken=e,this.statToken}catch(e){if(this.isDestroyed)return;this.params.errors$.next({id:`ThinOneStat:UvRestApi:AuthorizeFailed`,category:V.ErrorCategory.NETWORK,message:`Failed to get stat token`,thrown:e});return}}async handleLogRequestError(e){let t=e?.status;return t===401?(this.params.errors$.next({id:`ThinOneStat:UvRestApi:TokenExpired`,category:V.ErrorCategory.EXTERNAL_API,message:`Stat token expired (401), refreshing`,thrown:e}),this.authorized$.next(!1),this.statToken=void 0,await this.authorize()!==void 0):(this.params.errors$.next({id:`ThinOneStat:UvRestApi:LogRequestFailed#${t??`unknown`}`,category:V.ErrorCategory.EXTERNAL_API,message:`Log request failed with status ${t}`,thrown:e}),!1)}async retryLogRequestAfterRecovery(e){let t=this.statToken;if(!t){this.params.errors$.next({id:`ThinOneStat:UvRestApi:RetryNoToken`,category:V.ErrorCategory.NETWORK,message:`Cannot retry log request: no token after recovery`});return}try{await this.params.apiTransport.sendJsonRequest(`/uv/stat/logUvStat`,e,{Authorization:`Bearer ${t}`})}catch(e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:RetryFailed`,category:V.ErrorCategory.EXTERNAL_API,message:`Log request failed after token refresh`,thrown:e})}}buildBody(e){let t=e[0],n=t.application;return this.params.vktvVersion&&(n=this.params.vktvVersion),[{product:t.product,platform:t.platform,application:n,user_id:this.params.userId,events:e.map(Pe)}]}}var Le=require(`@vkontakte/videoplayer-shared`),Re=require(`@vkontakte/videoplayer-shared`),W=require(`@vkontakte/videoplayer-shared`);class ze{params;apiKey;apiBaseUrl;apiUvRestBaseUrl;apiEnv;resolvedApiBaseUrl=null;resolveApiBaseUrlPromise=null;timeSynchronisation;fetchTimeout=3e4;maxTotalRequestTime=6e4;retryConfig;constructor(e){this.params=e,this.apiKey=e.config.apiKey,this.apiEnv=e.config.apiEnv,this.apiBaseUrl=e.config.apiBaseUrl??B[this.apiEnv],this.apiUvRestBaseUrl=e.config.apiUvRestBaseUrl,this.timeSynchronisation=e.timeSynchronisation,this.retryConfig={maxAttempts:e.config.requestRetryCount??3,baseDelay:3e3,maxDelay:3e4,retryableErrors:[`Failed to fetch`,`NetworkError when attempting to fetch resource.`,`Request timeout`,`Network request failed`],retryableStatuses:[408,429,500,502,503,504]}}sendBeacon(e,t,n){if(!W.getWindow().Blob||!W.getWindow().navigator.sendBeacon)return!1;let r=this.resolvedApiBaseUrl??this.apiBaseUrl,i=this.prepareQueryParams({method:e,queryParams:t,sessionKey:n}),a=W.getWindow().Blob,o=new a([i.toString()],{type:`application/x-www-form-urlencoded`});try{return W.getWindow().navigator.sendBeacon(`${r}/fb.do`,o)}catch(n){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendBeacon`,category:W.ErrorCategory.NETWORK,message:`Unhandled beacon error`,thrown:n,data:{method:e,params:t}})}return!1}async sendRequest(e,t,n,r){return this.sendRequestWithRetry(e,t,n,r)}async sendRequestWithRetry(e,t,n,r,i=1,a){let o=W.now(),s=a??o+this.maxTotalRequestTime;if(W.now()>=s){let r=Error(`Request deadline exceeded`);throw this.handleFinalError(e,t,n,o,r,i),r}try{return await this.executeRequest(e,t,n,r,o,i,s)}catch(a){if(this.shouldRetry(a,i)&&W.now()<s){let c=this.getRetryDelay(a,i);if(W.now()+c>=s)throw this.handleFinalError(e,t,n,o,a,i),a;return this.logRetryAttempt(e,i,a,c),await this.sleep(c),this.sendRequestWithRetry(e,t,n,r,i+1,s)}throw this.handleFinalError(e,t,n,o,a,i),a}}async executeRequest(e,t,n,r,i,a,o){let s=i??W.now(),c=await this.resolveApiBaseUrl(),l=o===void 0?this.fetchTimeout:o-W.now();if(l<=0)throw H();try{let i=await this.fetchWithTimeout(`${c}/fb.do`,{method:`post`,headers:{"Content-type":`application/x-www-form-urlencoded`,...r},body:this.prepareQueryParams({method:e,queryParams:t,sessionKey:n})},Math.min(this.fetchTimeout,l));return await this.processResponse(i,s)}catch(t){throw t instanceof Error&&(t.context={method:e,attempt:a,url:`${c}/fb.do`}),t}}async sendJsonRequest(e,t,n){return this.sendJsonRequestWithRetry(e,t,n)}sendJsonBeacon(e,t,n){let r=`${this.apiUvRestBaseUrl}${e}`;try{Le.fetchPolyfilled(r,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t),keepalive:!0}).catch(t=>{this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonBeacon`,category:W.ErrorCategory.NETWORK,message:`Keepalive request failed`,thrown:t,data:{path:e}})})}catch(t){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonBeacon`,category:W.ErrorCategory.NETWORK,message:`Failed to initiate keepalive request`,thrown:t,data:{path:e}})}}async sendJsonRequestWithRetry(e,t,n,r=1,i){let a=W.now(),o=i??a+this.maxTotalRequestTime;if(W.now()>=o){let t=Error(`Request deadline exceeded`);throw this.handleJsonFinalError(e,a,t,r),t}try{return await this.executeJsonRequest(e,t,n,a,r,o)}catch(i){if(this.shouldRetry(i,r)&&W.now()<o){let s=this.getRetryDelay(i,r);if(W.now()+s>=o)throw this.handleJsonFinalError(e,a,i,r),i;return this.logRetryAttempt(e,r,i,s),await this.sleep(s),this.sendJsonRequestWithRetry(e,t,n,r+1,o)}throw this.handleJsonFinalError(e,a,i,r),i}}async executeJsonRequest(e,t,n,r,i,a){let o=r??W.now(),s=a===void 0?this.fetchTimeout:a-W.now();if(s<=0)throw H();let c=`${this.apiUvRestBaseUrl}${e}`;try{let e=await this.fetchWithTimeout(c,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t)},Math.min(this.fetchTimeout,s));return await this.processResponse(e,o)}catch(t){throw t instanceof Error&&(t.context={method:e,attempt:i,url:c}),t}}async fetchWithTimeout(e,t,n=this.fetchTimeout){let r=new Re.AbortControllerPolyfilled,i=W.getWindow().setTimeout(()=>r.abort(),n);try{let n=await Le.fetchPolyfilled(e,{...t,signal:r.signal});return W.getWindow().clearTimeout(i),n}catch(e){throw W.getWindow().clearTimeout(i),W.isAbortError(e)?H():e}}async processResponse(e,t){if(!e.ok){let t=Me(e),n=this.parseRetryAfter(e);throw n!==void 0&&(t.retryAfter=n),t}let n=new Date(e.headers.get(`date`)??``).getTime(),r=W.now()-t;isFinite(n)&&this.timeSynchronisation?.addServerTime(n,r);let i=e.headers.get(`content-length`);if(i!==null&&Number(i)===0)return{};let a=await e.json();if(a&&typeof a==`object`&&`error_msg`in a)throw Ne(a);return a}shouldRetry(e,t){if(t>=this.retryConfig.maxAttempts||!(e instanceof Error))return!1;let n=e;return!!(n.isTimeout||n.status&&this.retryConfig.retryableStatuses.includes(n.status)||this.retryConfig.retryableErrors.includes(e.message)||e.message.includes(`network`)||e.message.includes(`timeout`)||e.message.includes(`connection`))}logRetryAttempt(e,t,n,r){let i=n instanceof Error?n.message:String(n);this.params.errors$.next({id:`ThinOneStat:ApiTransport:retry`,category:W.ErrorCategory.NETWORK,message:`Retrying request (attempt ${t}/${this.retryConfig.maxAttempts}) after ${Math.round(r)}ms`,thrown:n,data:{method:e,attempt:t,delay:r,errorMessage:i}})}handleFinalError(e,t,n,r,i,a){let o=W.ErrorCategory.NETWORK;this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendRequest`,category:o,message:i instanceof Error?i.message:`Request failed after retries`,thrown:i,data:{method:e,params:t,sessionKey:n,time:W.now()-r,maxAttempts:this.retryConfig.maxAttempts,attempts:a}})}handleJsonFinalError(e,t,n,r){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonRequest`,category:W.ErrorCategory.NETWORK,message:n instanceof Error?n.message:`JSON request failed after retries`,thrown:n,data:{path:e,time:W.now()-t,maxAttempts:this.retryConfig.maxAttempts,attempts:r}})}sleep(e){return new Promise(t=>W.getWindow().setTimeout(t,e))}resolveApiBaseUrl(){return this.apiEnv===`auto`?(this.resolveApiBaseUrlPromise||=this.fetchApiBaseUrl().then(e=>(e===B.vk_alias&&(this.resolveApiBaseUrlPromise=null),e)),this.resolveApiBaseUrlPromise):Promise.resolve(this.apiBaseUrl)}async fetchApiBaseUrl(){try{let e=atob(`aHR0cHM6Ly9kbnMuZ29vZ2xlL3Jlc29sdmU/bmFtZT12aWRlby5fZW5kcG9pbnQub2sucnUmdHlwZT1UWFQ=`),t=(await(await this.fetchWithTimeout(e,{method:`GET`,mode:`cors`,cache:`no-cache`},5e3)).json())?.Answer[0]?.data;if(!t)throw Error(`Wrong DNS response`);return this.resolvedApiBaseUrl=t,t}catch(e){return this.params.errors$.next({id:`ThinOneStat:ApiTransport:resolveApiBaseUrl`,category:W.ErrorCategory.NETWORK,message:`Unhandled resolve api base url error`,thrown:e}),B.vk_alias}}prepareQueryParams(e){let t=new URLSearchParams({format:`JSON`,method:e.method,application_key:this.apiKey});return e.sessionKey!==void 0&&t.append(`session_key`,e.sessionKey),Object.entries(e.queryParams).forEach(([e,n])=>t.append(e,typeof n==`string`?n:JSON.stringify(n))),t}parseRetryAfter(e){let t=e.headers.get(`Retry-After`);if(!t)return;let n=Number(t);if(!isNaN(n)&&n>=0)return n*1e3;let r=new Date(t).getTime();if(!isNaN(r))return Math.max(0,r-Date.now())}getRetryDelay(e,t){let n=e;return n?.retryAfter!==void 0&&n.retryAfter>0?n.retryAfter:W.getExponentialDelay(t,{start:this.retryConfig.baseDelay,max:this.retryConfig.maxDelay,factor:2,random:.2})}}var Be=require(`@vkontakte/videoplayer-shared`),G=require(`@vkontakte/videoplayer-shared`),Ve=e=>t=>(t.client_time??G.now())+e>=G.now();class He{params;subscription;userSalt;unsaltedStorage=[];lastVsid;isPaused=!1;firstFlush;debouncedFlush;flushRetryCount=0;retryFlushTimerId;debugLog;flushPromise=null;flushSnapshotLength=0;onlineListener;urgentEvents=[`watch_coverage`,`watch_coverage_live`,`watched_n`,`playback_started`];STORAGE_MAX_ATTEMPTS=3;constructor(e){this.params=e,this.userSalt=e.userSalt,this.debugLog=e.debugLogger.createComponentLog(`batch-queue:${e.storageKey}`),this.firstFlush=G.debounceFn(()=>this.safeFlush(),this.params.config.flushFirstTime,{maxWait:this.params.config.flushFirstTime});let t=G.safeStorage.isPersistent()?this.params.config.flushMaxWait:0;this.debouncedFlush=G.debounceFn(()=>this.safeFlush(),this.params.config.flushDebounceTime,{maxWait:t}),this.subscription=this.params.canSend$.subscribe(e=>{e&&(this.debugLog({message:`Can send, triggering flush`}),this.debouncedFlush())}),this.onlineListener=()=>{this.debugLog({message:`Network online, triggering flush`}),this.cancelRetry(),this.safeFlush()},G.getWindow().addEventListener(`online`,this.onlineListener),this.housekeepStorage()}isUrgent=e=>this.urgentEvents.includes(e.event_name);scheduleRetry(){this.retryFlushTimerId!==void 0&&G.getWindow().clearTimeout(this.retryFlushTimerId);let e=this.params.config.flushRetryFirstDelay,t=this.params.config.flushRetryMaxDelay,n=Math.min(e*2**this.flushRetryCount,t);this.flushRetryCount++,this.debugLog({message:`Scheduling flush retry #${this.flushRetryCount} in ${n}ms`}),this.retryFlushTimerId=G.getWindow().setTimeout(()=>{this.retryFlushTimerId=void 0,this.safeFlush()},n)}cancelRetry(){this.retryFlushTimerId!==void 0&&(G.getWindow().clearTimeout(this.retryFlushTimerId),this.retryFlushTimerId=void 0),this.flushRetryCount=0}safeFlush(){if(this.flushPromise){this.debugLog({message:`Flush already in progress, skipping`});return}this.flushPromise=this.flush().finally(()=>{this.flushPromise=null})}isStorageAvailable(){let e=G.safeStorage.isPersistent?.()??!1;return e||this.debugLog({message:`Storage not available`}),e}readFromStorage(){if(!this.isStorageAvailable())return{};try{let e=G.safeStorage.get(this.params.storageKey),t=e?JSON.parse(e):{};return this.debugLog({message:`Read ${Object.keys(t).length} users from storage`}),t}catch(e){return this.params.errors$.next({id:`ThinOneStat:BatchQueue:ReadStorageError`,category:G.ErrorCategory.WTF,message:`Failed to read from storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to read from storage`}),{}}}writeToStorage(e){if(!this.isStorageAvailable())return!1;try{return G.safeStorage.set(this.params.storageKey,JSON.stringify(e)),this.debugLog({message:`Written ${Object.keys(e).length} users to storage`}),!0}catch(e){return this.params.errors$.next({id:`ThinOneStat:BatchQueue:WriteStorageError`,category:G.ErrorCategory.WTF,message:`Failed to write to storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to write to storage`}),!1}}addToStorage(e){if(!this.userSalt){this.unsaltedStorage.push(e),this.debugLog({message:`Added item to unsalted storage, total: ${this.unsaltedStorage.length}`});return}for(let t=0;t<this.STORAGE_MAX_ATTEMPTS;t++)try{let t=this.readFromStorage(),n=t[this.userSalt]??[];if(this.debugLog({message:`Adding item to storage for user ${this.userSalt}`}),this.writeToStorage({...t,[this.userSalt]:[...n,e]}))return}catch(n){t===this.STORAGE_MAX_ATTEMPTS-1&&(this.params.errors$.next({id:`ThinOneStat:BatchQueue:AddToStorageFailed`,category:G.ErrorCategory.WTF,message:`Failed to add item to storage after retries`,thrown:n,data:{item:e,attempt:t,storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to add item to storage after retries`}))}}getFromStorage(){if(!this.userSalt)return this.debugLog({message:`Retrieved ${this.unsaltedStorage.length} items from unsalted storage`}),this.flushSnapshotLength=this.unsaltedStorage.length,this.unsaltedStorage.slice();let e=this.readFromStorage(),t=e[this.userSalt]??[],n=t.filter(Ve(this.params.config.storageExpiration));if(this.debugLog({message:`Retrieved from storage for user ${this.userSalt}`}),n.length!==t.length){let r=t.length-n.length;this.params.errors$.next({id:`ThinOneStat:BatchQueue:DropExpiredEvents`,category:G.ErrorCategory.WTF,message:`Dropped ${r} expired events from storage`,data:{droppedCount:r,expiration:this.params.config.storageExpiration,storageKey:this.params.storageKey}}),this.debugLog({message:`Cleaning ${r} expired items from storage`});let i={...e};n.length?i[this.userSalt]=n:delete i[this.userSalt],this.writeToStorage(i)}return this.flushSnapshotLength=n.length,n}markStorageSent(){if(!this.userSalt){let e=this.unsaltedStorage.splice(0,this.flushSnapshotLength);this.debugLog({message:`Cleared unsalted storage, removed ${e.length} items, ${this.unsaltedStorage.length} new items kept`}),this.flushSnapshotLength=0;return}let e=this.flushSnapshotLength;this.flushSnapshotLength=0;for(let t=0;t<this.STORAGE_MAX_ATTEMPTS;t++)try{let t=this.readFromStorage(),n=(t[this.userSalt]??[]).slice(e),r={...t};if(n.length?r[this.userSalt]=n:delete r[this.userSalt],this.writeToStorage(r)){this.debugLog({message:`Marked storage as sent for user ${this.userSalt}, removed ${e} items, ${n.length} kept`});return}}catch(e){t===this.STORAGE_MAX_ATTEMPTS-1&&(this.params.errors$.next({id:`ThinOneStat:BatchQueue:MarkStorageSentFailed`,category:G.ErrorCategory.WTF,message:`Failed to mark storage as sent after retries`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to mark storage as sent after retries`}))}}housekeepStorage(){if(this.isStorageAvailable())try{let e=this.readFromStorage(),t=!1,n=0;for(let[r,i]of Object.entries(e)){let a=i.filter(Ve(this.params.config.storageExpiration)),o=i.length-a.length;n+=o,a.length?a.length!==i.length&&(e[r]=a,t=!0):(delete e[r],t=!0),this.debugLog({message:`${i.length} retrieved from storage, ${a.length} of them actual for ${r}`})}t?(this.debugLog({message:`Housekeeping completed, removed ${n} expired items total`}),this.writeToStorage(e),n>0&&this.params.errors$.next({id:`ThinOneStat:BatchQueue:DropExpiredEvents`,category:G.ErrorCategory.WTF,message:`Dropped ${n} expired events during housekeeping`,data:{droppedCount:n,expiration:this.params.config.storageExpiration,storageKey:this.params.storageKey}})):this.debugLog({message:`Housekeeping completed, no changes needed`})}catch(e){this.params.errors$.next({id:`ThinOneStat:BatchQueue:HousekeepStorageError`,category:G.ErrorCategory.WTF,message:`Failed to housekeep storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to housekeep storage`})}}log(e){try{if(this.debugLog({message:`Logging item: ${e.event_name}`}),this.addToStorage(e),this.isPaused){this.debugLog({message:`Queue paused, skipping flush`});return}this.isUrgent(e)?(this.debugLog({message:`Urgent item, flushing immediately`}),this.safeFlush()):this.lastVsid===e.vsid?(this.debugLog({message:`Same VSID, scheduling debounced flush`}),this.debouncedFlush()):(this.debugLog({message:`VSID changed, triggering first flush`}),this.firstFlush()),this.lastVsid=e.vsid}catch(t){this.params.errors$.next({id:`ThinOneStat:BatchQueue:LogError`,category:G.ErrorCategory.WTF,message:`Failed to log item`,thrown:t,data:{item:e,storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to log item`})}}flush(e={wantBeacon:!1,clearStorage:!0}){return this.flushPromise?(this.debugLog({message:`Flush already in progress, waiting`}),this.flushPromise):(this.flushPromise=this.flushInternal(e).finally(()=>{this.flushPromise=null}),this.flushPromise)}async flushInternal({wantBeacon:e,clearStorage:t}){this.debugLog({message:`Starting flush`});let n=this.getFromStorage();if(n.length===0){this.debugLog({message:`No items to flush`});return}let r=this.params.canSend$.getValue();if(this.debugLog({message:`Can send status: ${r}`}),!r){this.debugLog({message:`Cannot send, requesting ready state`}),this.params.requestReady();return}if(e&&this.params.config.useBeacon){this.debugLog({message:`Flushing ${n.length} events through beacon`}),this.params.sendBeacon(n),t&&this.markStorageSent(),this.firstFlush.cancel(),this.debouncedFlush.cancel();return}this.debugLog({message:`Flushing ${n.length} events`});try{await this.params.send(n),this.debugLog({message:`Flush completed successfully`}),this.cancelRetry(),t&&this.markStorageSent(),this.firstFlush.cancel(),this.debouncedFlush.cancel()}catch(e){throw this.params.errors$.next({id:`ThinOneStat:BatchQueue:FlushRequestError`,category:G.ErrorCategory.NETWORK,message:`Failed to send events, will retry later`,thrown:e,data:{itemsCount:n.length,storageKey:this.params.storageKey}}),this.debugLog({message:`Flush failed, scheduling retry`}),this.scheduleRetry(),e}}pause(){this.debugLog({message:`Queue paused`}),this.isPaused=!0,this.debouncedFlush.cancel(),this.firstFlush.cancel(),this.cancelRetry()}resume(){this.debugLog({message:`Queue resumed`}),this.isPaused=!1,this.debouncedFlush()}async destroy(){this.debugLog({message:`Destroying queue`}),this.flushPromise&&(this.debugLog({message:`Waiting for ongoing flush to complete`}),await this.flushPromise),this.cancelRetry(),this.onlineListener&&=(G.getWindow().removeEventListener(`online`,this.onlineListener),void 0),this.subscription.unsubscribe(),this.firstFlush.cancel(),this.debouncedFlush.cancel(),this.flushPromise=null,this.debugLog({message:`Queue destroyed`})}}class Ue{params;queues=[];constructor(e){this.params=e;for(let t of e.apis)this.queues.push(this.createQueueForApi(t))}createQueueForApi(e){let t=this.getStorageKeyForApi(e.id);return new He({config:this.params.config,storageKey:t,debugLogger:this.params.debugLogger,errors$:this.params.errors$,userSalt:this.params.userSalt,send:t=>e.logRequest(t),sendBeacon:t=>e.logBeacon(t),canSend$:e.authorized$,requestReady:()=>{e.authorize().catch(e=>{this.params.errors$.next({id:`ThinOneStat:Logger:RequestReadyFailed`,category:Be.ErrorCategory.NETWORK,message:`Failed to authorize on requestReady`,thrown:e,data:{storageKey:t}})})}})}getStorageKeyForApi(e){return`thinonestat_events_${e}`}log(e){this.queues.forEach(t=>t.log(e))}async flush(e){let t=await Promise.allSettled(this.queues.map(t=>t.flush(e)));if(t.length>0&&t.every(e=>e.status===`rejected`))throw t.find(e=>e.status===`rejected`).reason}pause(){this.queues.forEach(e=>e.pause())}resume(){this.queues.forEach(e=>e.resume())}async destroy(){await Promise.allSettled(this.queues.map(e=>e.destroy())),this.queues=[]}}function We(){let e=`0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz`.split(``),t=Array(36),n=0,r,i;for(i=0;i<36;i++)i===8||i===13||i===18||i===23?t[i]=`-`:i===14?t[i]=`4`:(n<=2&&(n=33554432+Math.random()*16777216|0),r=n&15,n>>=4,t[i]=e[i===19?r&3|8:r]);return t.join(``)}class Ge{offset=void 0;constructor(e){this.offset=e}getOffset(){return this.offset??0}setOffset(e){this.offset=e}now(){return Date.now()+(this.offset??0)}date(e=new Date){return e.setTime(e.getTime()+(this.offset??0)),e}addServerTime(e,t){let n=e-Date.now()-t/2;if(Math.abs(n)<1e3){this.offset=0;return}if(this.offset===void 0)this.offset=Math.round(n);else{let e=.2;this.offset=Math.round(e*n+(1-e)*this.offset)}}}var Ke=require(`@vkontakte/videoplayer-core`),K=require(`@vkontakte/videoplayer-shared`),qe=(e,t,n)=>new K.Observable(r=>{let i=new K.Subscription,a,o=new K.ValueSubject(void 0),s;i.add(e.info.isLive$.pipe(K.filterChanged()).subscribe(t=>{s&&(s.unsubscribe(),o.next(void 0)),a=void 0,s=t?e.info.liveTime$.pipe(K.map(e=>e&&e/1e3)).subscribe(o):e.info.position$.subscribe(o),i.add(s)}));let{playing$:c,paused$:l,ended$:u,looped$:d}=e.events,f=e.events.willSeek$.pipe(K.filter(()=>e.info.playbackState$.getValue()===Ke.PlaybackState.PLAYING)),p=e.events.seeked$.pipe(K.filter(()=>e.info.playbackState$.getValue()===Ke.PlaybackState.PLAYING)),m=e.info.isStalled$.pipe(K.filter(e=>e)),h=!1,g=new K.Subject;i.add(f.subscribe(()=>{h||g.next(),h=!0})).add(p.subscribe(()=>h=!1));let _=K.merge(K.fromEvent(K.getWindow(),`beforeunload`),e.events.willDestruct$),v=e=>{a=e},y=e=>{let t=a;if(a=void 0,!(K.isNullable(t)||K.isNullable(e)||!isFinite(t)||!isFinite(e))){if(Math.round(e*1e3)<=Math.round(t*1e3)){a=e;return}r.next({from:t,to:e})}},b=new K.Subject,x=new K.Subject,S=()=>n.exactTime?e.info.isLive$.getValue()?e.getExactLiveTime():e.getExactTime():o.getValue()??0,C=K.merge(c,p,b).pipe(K.map(()=>S())),w=K.merge(d,K.merge(l,g,_,u,x,m).pipe(K.map(()=>S()))),ee=t.started$&&t.ended$?K.merge(C,t.ended$):C,te=t.started$&&t.ended$?K.merge(w,t.started$):w;if(i.add(ee.subscribe(v)).add(te.subscribe(y)),n.heartbeatInterval&&isFinite(n.heartbeatInterval)){let e=[],t=()=>{x.next(),b.next()};i.add(ee.subscribe(()=>{let r=K.getWindow().setTimeout(t,n.heartbeatInterval);e.push(r)})).add(te.subscribe(()=>{e.forEach(e=>{K.getWindow().clearTimeout(e)}),e=[]}))}return i}),Je=require(`@vkontakte/videoplayer-core`),Ye=require(`@vkontakte/videoplayer-shared`),Xe=(e,t)=>new Ye.Observable(n=>{let r=new Je.Subscription;return r.add(Ye.merge(e.info.position$,e.info.liveTime$).subscribe(()=>{let e=t.getTotalViewTime();e>=t.targetDuration&&(n.next({target_duration:t.targetDuration,current_tvt:e}),r.unsubscribe())})),r}),Ze=require(`@vkontakte/videoplayer-core`),q=require(`@vkontakte/videoplayer-shared`),Qe=(e,{maxTimeWindow:t})=>new q.Observable(n=>{let r=new Ze.Subscription,i={maxBytesThreshold:100*1024*1024,minBytesThreshold:100*1024,maxTimeWindow:t,minTimeWindow:2e3},a={bytes:0,startTime:0,lastEmitTime:0,timeoutId:0},o=()=>{a.timeoutId&&=(q.getWindow().clearTimeout(a.timeoutId),null)},s=()=>{a.bytes=0,a.startTime=0,a.lastEmitTime=0,o()},c=e=>{if(a.bytes<i.minBytesThreshold||e-a.lastEmitTime<i.minTimeWindow&&a.lastEmitTime>0)return!1;let t=(a.startTime>0?e-a.startTime:0)>=i.maxTimeWindow,n=a.bytes>=i.maxBytesThreshold;return t||n},l=(e=!1)=>{let t=Date.now();if(!e&&!c(t))return;o();let r=a.startTime>0?t-a.startTime:Math.max(i.minTimeWindow,1),l=a.bytes/(r/1e3);n.next({download_bytes:Math.round(a.bytes),download_speed:Math.round(l)}),a.lastEmitTime=t,s()},u=()=>{if(o(),a.startTime>0&&a.bytes>=i.minBytesThreshold){let e=Date.now()-a.startTime,t=i.maxTimeWindow-e;t>100?a.timeoutId=q.getWindow().setTimeout(()=>{a.timeoutId=null,l()},t):t>0&&l()}};return r.add(e.info.httpDownloadMetrics$.pipe(q.filter(e=>e!==void 0)).subscribe(e=>{let t=Date.now();a.startTime===0&&(a.startTime=t),a.bytes+=e.bytes,c(t)?l():u()})),r.add(()=>{if(o(),a.bytes>0){let e=Date.now(),t=a.startTime>0?e-a.startTime:i.minTimeWindow,r=a.bytes/(t/1e3);n.next({download_bytes:Math.round(a.bytes),download_speed:Math.round(r)})}s()}),r}),J=require(`@vkontakte/videoplayer-shared`),$e=e=>new J.Observable(t=>{let n=new J.Subscription,r=0,i=1,a=0,o=!1,s=!1,c=()=>{if(!o)return;let e=Date.now();a+=(e-r)*i,r=e},l=()=>{(!o||s)&&(r=Date.now(),o=!0,s=!1)},u=()=>{o&&=(c(),!1)},d=()=>{o&&(c(),o=!1,s=!0)},f=e=>{o&&c(),i=e,r=Date.now()},{started$:p,playing$:m,paused$:h,willSeek$:g,ended$:_,willDestruct$:v}=e.events,{currentPlaybackRate$:y}=e.info;return n.add(g.subscribe(d)).add(J.merge(p,m).subscribe(l)).add(J.merge(h,_,v).subscribe(u)).add(y.subscribe(f)),t.next(()=>{let e=a;return o&&(e+=(Date.now()-r)*i),Math.round(e)}),n}),et=require(`@vkontakte/videoplayer-shared`),tt=(e,t)=>new et.Observable(n=>{let r=new et.Subscription,i=0;return r.add(et.merge(e.info.position$,e.info.liveTime$).subscribe(()=>{let e=t.getTotalViewTime();e>i&&Math.floor(e/t.interval)>Math.floor(i/t.interval)&&(i=e,n.next())})),r}),nt=require(`@vkontakte/videoplayer-shared`),Y=require(`@vkontakte/videoplayer-core`),rt=e=>{if(!nt.isNullable(e)){if(typeof e==`string`)return e;try{return JSON.stringify(e)}catch{return String(e)}}},it=e=>e&&{[Y.HttpConnectionType.HTTP1]:`http1`,[Y.HttpConnectionType.HTTP2]:`http2`,[Y.HttpConnectionType.QUIC]:`http3`}[e],X=e=>{if(e!==void 0)switch(e){case Y.VideoFormat.MPEG:return`MP4`;case Y.VideoFormat.DASH:case Y.VideoFormat.DASH_LIVE:return`DASH`;case Y.VideoFormat.DASH_SEP:return`DASH_SEP`;case Y.VideoFormat.DASH_ONDEMAND:return`ONDEMAND_DASH`;case Y.VideoFormat.HLS_ONDEMAND:return`ONDEMAND_HLS`;case Y.VideoFormat.HLS:case Y.VideoFormat.HLS_LIVE:return`HLS`;case Y.VideoFormat.HLS_FMP4:return`HLS_FMP4`;case Y.VideoFormat.DASH_WEBM:case Y.VideoFormat.DASH_LIVE_WEBM:return`WEBM`;case Y.VideoFormat.DASH_LIVE_CMAF:return`ONDEMAND_DASH_LIVE`;case Y.VideoFormat.HLS_LIVE_CMAF:return`ONDEMAND_HLS_LIVE`;case Y.VideoFormat.WEB_RTC_LIVE:return`WEBRTC`;case Y.VideoFormat.DASH_WEBM_AV1:return`AV1`;case Y.VideoFormat.DASH_STREAMS:return`MULTI_DASH`;default:return nt.assertNever(e)}},Z=e=>{if(e!==void 0)switch(e){case Y.VideoQuality.Q_144P:return`144p`;case Y.VideoQuality.Q_240P:return`240p`;case Y.VideoQuality.Q_360P:return`360p`;case Y.VideoQuality.Q_480P:return`480p`;case Y.VideoQuality.Q_720P:return`720p`;case Y.VideoQuality.Q_1080P:return`1080p`;case Y.VideoQuality.Q_1440P:return`1440p`;case Y.VideoQuality.Q_2160P:return`2160p`;case Y.VideoQuality.INVARIANT:case Y.VideoQuality.Q_576P:case Y.VideoQuality.Q_4320P:return`UNKNOWN`;default:return nt.assertNever(e)}},at=e=>{if(e!==void 0)switch(e){case`MP4`:case`ONDEMAND_DASH`:case`ONDEMAND_DASH_LIVE`:case`ONDEMAND_HLS`:case`ONDEMAND_HLS_LIVE`:case`MULTI_DASH`:case`WEBM`:case`AV1`:return`vod`;case`DASH`:case`HLS`:case`HLS_FMP4`:case`WEBRTC`:return`live`;default:return}},Q=(e,t,n)=>{if(e&&t)switch(e){case Y.VideoFormat.MPEG:return n&&t[e]&&t[e][n];default:return t[e]?.url}},ot=(e,t=!1)=>{if(t)return`minimal`;switch(e){case void 0:case Y.Surface.NONE:return;case Y.Surface.INLINE:return`inline`;case Y.Surface.FULLSCREEN:return`fullscreen`;case Y.Surface.SECOND_SCREEN:return`chromecast`;case Y.Surface.PIP:return`pip_external`}},st=e=>{switch(e){case void 0:return;case Y.Surface.INVISIBLE:return`background`;default:return`foreground`}},ct=`_thin-one-stat_deviceId`,lt=`2.2.0`,ut=()=>{let e=new R.Subscription;return{subscription:e,subscribe:(t,n)=>{t&&e.add(t.subscribe(n))}}};class dt{apis=[];logger;config;subscription;lifecycleFlushSubscription;timeSynchronisation;statContext;deviceId;targetDuration;debugLogger=new R.Logger;thinOneStatDebugLog=this.debugLogger.createComponentLog(`ThinOneStat`);eventNumber=1;isStarted;isSeeking;inBufferStarvation;firstByteManifestSent;firstMediaRequestSent;firstByteMediaSent;wasPaused;isLive;lastContentType;cdnHostname;isEmbed;embedHostname;connectionType;connectionReused;player;playerSize={width:0,height:0};currentSubtitle;userQuality;downloadQuality;downloadSpeed;networkType;uiEvents;playerSubscription;uiSubscription;adsSubscription;seekAction$=new R.ValueSubject(`unknown`);vsid$=new R.ValueSubject(void 0);isid$=new R.ValueSubject(void 0);errors$=new R.Subject;getTotalViewTime=()=>0;constructor(e,t){this.statContext=e,this.config=je(t.config??{}),this.subscription=new R.Subscription,this.config.synchronizeTime&&(this.timeSynchronisation=new Ge),this.subscription.add(this.errors$.subscribe(e=>{this.thinOneStatDebugLog({message:`[${e.category}] ${e.id}: ${e.message}`})}));let n=R.safeStorage.get(ct);this.statContext.deviceId?this.deviceId=this.statContext.deviceId:n?this.deviceId=n:(this.deviceId=We(),R.safeStorage.set(ct,this.deviceId)),this.statContext.targetDuration?this.targetDuration=this.statContext.targetDuration:this.targetDuration=this.config.watchedNDefaultTargetDuration,this.resetViewSession(),t.useIsid&&this.isid$.next(t.isid??E());let r=new ze({errors$:this.errors$,config:this.config,timeSynchronisation:this.timeSynchronisation});this.config.apiForticom&&this.apis.push(new Fe({config:this.config,apiTransport:r,errors$:this.errors$,deviceId:this.deviceId,refreshAuthToken:t.refreshAuthToken})),this.config.apiUvRest&&(t.getUvRestAuthToken?this.apis.push(new Ie({config:this.config,apiTransport:r,errors$:this.errors$,userId:this.statContext.userId,getAuthToken:t.getUvRestAuthToken})):this.errors$.next({id:`ThinOneStat:Init:MissingGetUvRestAuthToken`,category:R.ErrorCategory.WTF,message:`thinStatToUvRest enabled but getUvRestAuthToken callback not provided`})),this.logger=new Ue({config:this.config,debugLogger:this.debugLogger,apis:this.apis,errors$:this.errors$,userSalt:t.userSalt});let{isEmbed:i,topOrigin:a}=R.detectEmbed();this.isEmbed=i,this.embedHostname=a?new URL(a).hostname:void 0}authorize(e){this.apis.forEach(t=>{t.authorize(e).catch(()=>{})})}reportError(e){let t={error_severity:e.severity,error_category:this.mapErrorCategory(e.category),player_error_code:e.code,player_error_trace:e.trace,error_message:e.message,content_type:this.getCurrentContentType()};this.logError(t),e.severity===`critical`&&this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}updateContext(e){this.statContext={...this.statContext,...e}}attachTo(e,t){this.playerSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to player, unsubscribing from previous`}),this.playerSubscription.unsubscribe());let n=new R.Subscription,r=(e,t)=>n.add(e.subscribe(t));r($e(e),e=>{this.getTotalViewTime=e}),r(e.info.isLive$,e=>this.isLive=e),r(e.info.currentFormat$,e=>{let t=X(e);t&&(this.lastContentType=t)}),r(e.info.hostname$.pipe(R.filter(e=>e!==void 0)),e=>{this.cdnHostname=e});let i;r(R.combine({hostname:e.info.hostname$,connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$}).pipe(R.filter(({hostname:e})=>e!==void 0),R.filter(({connectionType:e})=>e!==void 0),R.filter(({connectionReused:e})=>e!==void 0)),({hostname:e,connectionType:t,connectionReused:n})=>{if(i!==void 0&&i!==e){let r={cdn_host:e,connection_type:it(t),connection_reused:n,content_type:this.getCurrentContentType()};this.logFailover(r),this.logTelemetry(this.collectTelemetryData())}this.cdnHostname=i=e}),r(e.info.httpConnectionType$,e=>{this.connectionType=it(e)}),r(e.info.httpConnectionReused$,e=>{this.connectionReused=e}),r(qe(e,t,{heartbeatInterval:this.config.watchCoverageHeartbeatInterval,exactTime:this.config.watchCoverageExactTime}),t=>{let n={watch_interval:`${Math.round(t.from*1e3)}-${Math.round(t.to*1e3)}`,in_history:!!this.statContext.inHistory,content_type:this.getCurrentContentType(),playback_quality:Z(e.info.currentQuality$.getValue()),recom_info:this.statContext.recomInfo};if(!e.info.isLive$.getValue())this.logWatchCoverage(n);else{let t=e.info.atLiveEdge$.getValue();this.logWatchCoverageLive({...n,live:!!t})}}),r(tt(e,{interval:this.statContext.telemetryInterval??this.config.telemetryInterval,getTotalViewTime:this.getTotalViewTime}),()=>{this.logTelemetry(this.collectTelemetryData())}),r(Xe(e,{targetDuration:this.targetDuration,getTotalViewTime:this.getTotalViewTime}),e=>{this.logWatchedN(e)}),r(e.events.willReady$,()=>{let t={isid:this.isid$.getValue(),stream_profile:at(this.getCurrentContentType()),dpi:Math.round(this.statContext.dpi??96*z.clientChecker.display.pixelRatio),web_layout:this.statContext.isMobile||z.clientChecker.device.isMobile?`mobile`:`desktop`,preloaded:this.statContext.preload,navigation:this.statContext.navigation,recom_info:this.statContext.recomInfo};if(this.isEmbed&&(t.iframe_host=this.embedHostname??`unknown`),this.logStartSession(t),this.statContext.preload){let t=e.info.currentFormat$.getValue(),n=e.info.availableSources$.getValue(),r=e.info.currentBuffer$.getValue(),i=e.info.currentQuality$.getValue(),a={target_buffer_time:this.calcBufferTime(r),playback_url:Q(t,n,i),playback_quality:Z(i)};this.logPreloadStarted(a)}}),r(R.combine({manifestRequested$:e.events.manifestRequested$,connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$}).pipe(R.filter(({connectionType:e})=>e!==void 0),R.filter(({connectionReused:e})=>e!==void 0),R.once()),()=>{let t={playback_url:Q(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logManifestRequest(t)}),r(R.combine({firstBytesManifest:e.events.firstBytesManifest$,connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$}).pipe(R.once()),()=>{if(this.firstByteManifestSent)return;this.firstByteManifestSent=!0;let t={playback_url:Q(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logFirstByteManifest(t)}),r(e.events.manifestReceived$,()=>{let t={playback_url:Q(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logManifestReceived(t)}),r(e.events.firstBytesRequested$,()=>{if(this.firstMediaRequestSent)return;this.firstMediaRequestSent=!0;let t={playback_url:Q(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logFirstMediaRequest(t)}),r(e.events.firstBytes$,()=>{if(this.firstByteMediaSent)return;this.firstByteMediaSent=!0;let t={playback_url:Q(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logFirstByteMedia(t)}),r(R.merge(e.events.willStart$,e.events.looped$),e=>{e&&this.resetViewSession(),this.isStarted=!0}),r(e.events.started$,()=>{let t=e.info.currentBuffer$.getValue(),n=e.info.currentQuality$.getValue(),r=e.info.isAutoQualityEnabled$.getValue(),i=e.info.muted$.getValue(),a=e.info.volume$.getValue(),o=e.info.surface$.getValue(),s=e.info.availableTextTracks$.getValue(),c=e.info.currentTextTrack$.getValue(),l=e.info.currentAudioStream$.getValue(),u=e.info.currentPlaybackRate$.getValue(),d={playback_quality:Z(n),user_quality:r?`auto`:Z(n),player_width:Math.round(this.statContext.playerSize?.width??this.playerSize.width),player_height:Math.round(this.statContext.playerSize?.height??this.playerSize.height),muted:i,sound_volume:Math.round(a*100),buffer_time:this.calcBufferTime(t),mode:ot(o,this.statContext.isPreviewPlayerView),visibility:st(o),subtitles_enabled:c?!!c:this.currentSubtitle?.enabled,auto_subtitles:s.find(({id:e})=>e===c)?.isAuto??this.currentSubtitle?.auto,audio_track_lang:l?.language,playback_rate:u*100,recom_info:this.statContext.recomInfo};this.logPlaybackStarted(d),this.logTelemetry(this.collectTelemetryData())}),r(e.events.willStart$,()=>{this.logPlay()}),r(e.events.playing$,()=>{let t=e.info.currentBuffer$.getValue(),n={buffer_time:this.calcBufferTime(t)};this.logPlaying(n),this.inBufferStarvation=!1,this.wasPaused&&this.logResume()}),r(e.events.paused$,()=>{this.wasPaused=!0,this.logPause()}),r(e.events.ended$,()=>{this.logContentEndReached()}),r(e.events.willStop$,()=>{this.isStarted=!1,this.isSeeking=!1,this.inBufferStarvation=!1,this.logStop()}),r(R.merge(e.events.fetcherRecoverableError$,e.events.fatalError$),()=>{this.statContext.preload&&this.logPreloadError()}),r(e.events.fetcherRecoverableError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:this.mapErrorSeverity(t),error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:rt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.mapErrorSeverity(t)===`critical`&&this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}),r(e.events.managedError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:this.mapErrorSeverity(t),error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:rt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.logTelemetry(this.collectTelemetryData())}),r(e.events.fatalError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:`critical`,error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:rt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}),r(R.combine({connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$,connectionMetrics:e.info.httpConnectionMetrics$}).pipe(R.filter(({connectionReused:e})=>e!==void 0),R.filter(({connectionMetrics:e})=>e!==void 0),R.filterChanged((e,t)=>e.connectionType===t.connectionType)),({connectionMetrics:t})=>{let n=e.info.currentFormat$.getValue(),r=e.info.availableSources$.getValue(),i=e.info.currentQuality$.getValue(),a=e.info.rttEstimation$.getValue();this.networkType=this.statContext.networkType??t?.networkType;let o=t?.dnsResolveTime,s=t?.tcpHandshakeTime,c=t?.tlsHandshakeTime,l={playback_url:Q(n,r,i),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused,rtt:a?Math.round(a):void 0,network_type:this.networkType,dns_resolve_time:o?Math.round(o):void 0,tcp_handshake_time:s?Math.round(s):void 0,tls_handshake_time:c?Math.round(c):void 0};this.logConnectionEstablished(l)}),r(Qe(e,{maxTimeWindow:this.config.downloadBytesMaxTime}),e=>{this.downloadSpeed=e.download_speed,this.logDownloadBytes(e)}),r(e.events.firstFrame$,()=>{this.logFirstVideoFrameDecoded()}),r(e.events.willSeek$,()=>{this.isSeeking=!0;let e={seek_type:this.seekAction$.getValue()};this.isStarted&&this.logSeeking(e)}),r(e.events.seeked$,()=>{this.isSeeking=!1;let e={seek_type:this.seekAction$.getValue()};this.isStarted&&this.logSeeked(e),this.seekAction$.next(`unknown`)}),r(e.info.stallStartTime$,t=>{if(t<=0){this.inBufferStarvation=!1;return}this.inBufferStarvation||(this.inBufferStarvation=!0,this.logBufferStarvation({buffer_time:this.calcBufferTime(e.info.currentBuffer$.getValue())}),this.logTelemetry(this.collectTelemetryData()))}),r(e.info.currentBuffer$.pipe(R.filter(e=>!!this.calcBufferTime(e)),R.once()),e=>{let t=this.calcBufferTime(e);if(this.statContext.preload){let e={buffer_time:t};this.logPreloadEnded(e)}this.logReady({buffer_time:t})});let a;r(R.combine({videoStream:e.info.currentVideoStream$,audioStream:e.info.currentAudioStream$,quality:e.info.currentQuality$}).pipe(R.filter(({videoStream:e,audioStream:t})=>e!==void 0&&t!==void 0),R.filter(({quality:e})=>e!==void 0)),({videoStream:t,audioStream:n,quality:r})=>{if(a!==r){if(this.userQuality=e.info.isAutoQualityEnabled$.getValue()?`auto`:Z(r),this.downloadQuality=Z(r),this.isStarted){let e=t?.codec,r=n?.codec,i={user_quality:this.userQuality,playback_quality:Z(a),download_quality:this.downloadQuality,codec_info:this.genCodecInfo(e,r)};this.logQualityChanged(i)}a=r}});let o,s;return r(e.info.surface$,e=>{let t=ot(e,this.statContext.isPreviewPlayerView);t&&!this.uiEvents&&this.isStarted&&this.logModeChanged({mode:t}),s=st(e),o!==s&&(this.isStarted&&this.logVisibilityChanged({visibility:s}),o=st(e))}),r(R.combine({muted:e.info.muted$,volume:e.info.volume$}).pipe(R.debounce(300)),({muted:e,volume:t})=>{if(this.isStarted){let n={muted:e,sound_volume:Math.round(t*100)};this.logSoundChanged(n)}}),r(e.info.currentFormat$.pipe(R.filter(e=>e!==void 0),R.skip(1)),e=>{let t={content_type:X(e),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logContentTypeChange(t)}),r(e.info.currentAudioStream$.pipe(R.filter(e=>e!==void 0),R.skip(1)),e=>{let t={audio_track_lang:e.language,codec_info:e.codec};this.logAudioTrackSwitched(t)}),r(e.info.currentVideoStream$.pipe(R.filter(e=>e!==void 0),R.skip(1)),e=>{let t={hdr:e.hdr,codec_info:e.codec};this.logVideoTrackSwitched(t)}),this.player=e,this.playerSubscription=n,this.resubscribeLifecycleFlush(),this.subscription.add(n),n}attachToUi(e){this.uiSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to UI, unsubscribing from previous`}),this.uiSubscription.unsubscribe()),this.uiEvents=e;let{subscribe:t,subscription:n}=ut();t(e.actionSeek$,this.seekAction$),t(e.playerSize$,({width:e,height:t})=>{e&&t&&(this.playerSize={width:e,height:t},this.isStarted&&this.logViewPortChanged({player_width:Math.round(e),player_height:Math.round(t)}))}),t(e.isLoaderVisible$,e=>{if(!e){this.inBufferStarvation=!1;return}this.isStarted&&(this.logShowLoader(),!(this.isSeeking||this.inBufferStarvation||!this.player)&&(this.inBufferStarvation=!0,this.logBufferStarvation({buffer_time:this.calcBufferTime(this.player.info.currentBuffer$.getValue())}),this.logTelemetry(this.collectTelemetryData())))}),t(e.actionSubtitlesSwitched$,e=>{if(this.currentSubtitle={...e},this.isStarted){let t={subtitles_enabled:e.enabled,subtitles_track_lang:e.lang,auto_subtitles:e.auto};this.logSubtitlesSwitched(t)}});let{surface$:r}=xe(this.uiEvents,t,this.player);return t(r,e=>{this.isStarted&&this.logModeChanged({mode:ot(e)})}),t(e.actionQuality$,e=>{let t=e=>Object.values(De).includes(e),n=this.player?.info.currentVideoStream$.getValue()?.codec,r=this.player?.info.currentAudioStream$.getValue()?.codec,i={user_quality:t(e)?`auto`:Z(e),playback_quality:Z(this.player?.info.currentQuality$.getValue()),download_quality:t(e)?Z(this.player?.info.currentQuality$?.getValue()):Z(e),codec_info:this.genCodecInfo(n,r)};this.logQualityChangeRequested(i)}),this.uiSubscription=n,this.resubscribeLifecycleFlush(),this.subscription.add(n),n}attachToAds(e){this.adsSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to ads, unsubscribing from previous`}),this.adsSubscription.unsubscribe());let{subscription:t,subscribe:n}=ut();return n(e.init$,e=>{let t={slot:e};this.logAdvConfiguration(t)}),n(e.slotRequested$,e=>{let t={adv_section:e};this.logAdvRequest(t)}),n(e.started$,e=>{let t={adv_section:e};this.logAdvBreakStarted(t)}),n(e.ended$,({section:e,ordIds:t})=>{let n={adv_section:e,erids:t.join(`,`)};this.logAdvBreakEnded(n)}),n(e.error$,e=>{let t={error_message:e};this.logAdvError(t)}),this.adsSubscription=t,this.resubscribeLifecycleFlush(),this.subscription.add(t),t}getDeviceId(){return this.deviceId}resetViewSession(){this.thinOneStatDebugLog({message:`VSID reset`}),D.generateVSID(this.config.useVsid64),this.vsid$.next(D.getVSID()),this.eventNumber=0,this.firstByteManifestSent=!1,this.firstMediaRequestSent=!1,this.firstByteMediaSent=!1}async destroy(){this.thinOneStatDebugLog({message:`Destroying ThinOneStat`}),this.subscription.unsubscribe(),this.lifecycleFlushSubscription?.unsubscribe(),await this.logger.destroy(),this.apis.forEach(e=>e.destroy()),this.thinOneStatDebugLog({message:`ThinOneStat destroyed`})}calcBufferTime(e){return e?Math.round(((e.end??0)-(e.start??0))*1e3):void 0}mapErrorCategory(e){switch(e){case R.ErrorCategory.NETWORK:return`network`;case R.ErrorCategory.VIDEO_PIPELINE:return`video_pipeline`;case R.ErrorCategory.EXTERNAL_API:return`external_api`;case R.ErrorCategory.PARSER:return`parser`;case R.ErrorCategory.DOM:return`dom`;default:return`wtf`}}mapErrorSeverity(e){return R.ErrorCategory.FATAL===e?`critical`:`informative`}getCurrentContentType(){let e=X(this.player?.info.currentFormat$.getValue());if(e)return e;if(this.lastContentType)return this.lastContentType;let t=this.player?.info.availableSources$.getValue();if(!t)return;let n=Object.keys(t)[0];return X(n)}genCodecInfo(e,t){return e&&t?`${e},${t}`:void 0}getLatencyInfo(){let e=this.player?.info.atLiveEdge$.getValue(),t={client_time:Date.now(),live:e};return JSON.stringify(t)}collectTelemetryData(){if(!this.player)return{};let e=this.player.info.currentBuffer$.getValue(),t=this.player.experimental.element$.getValue(),n=0,r=0;t&&(n=t.getVideoPlaybackQuality().droppedVideoFrames,r=t.getVideoPlaybackQuality().totalVideoFrames);let i=this.getCurrentContentType(),a=this.player.info.currentVideoStream$.getValue()?.codec,o=this.player.info.currentAudioStream$.getValue()?.codec,s=this.player.info.videoBitrate$.getValue(),c=this.player.info.rttEstimation$.getValue();return{buffer_time:this.calcBufferTime(e),dropped_frames_count:n,rendered_frames_count:r,content_type:i,user_quality:this.userQuality,playback_quality:Z(this.player.info.currentQuality$.getValue()),download_quality:this.downloadQuality,codec_info:this.genCodecInfo(a,o),bandwidth_from_manifest:s?Math.round(s):void 0,cdn_host:this.cdnHostname,download_speed:this.downloadSpeed?Math.round(this.downloadSpeed):void 0,network_type:this.networkType,rtt:c?Math.round(c):void 0,latency:this.getLatencyInfo()}}isValidPlatform(e=``){return[`web:desktop`,`web:mobile`,`smart_tv`].includes(e)}createRequiredParams(e){let t=this.vsid$.getValue();R.assertNonNullable(t);let n=this.eventNumber++,r=((this.isLive?this.player?.info.liveTime$.getValue():this.player?.info.position$.getValue())??0)*1e3,i={vsid:t,uv_movie_id:this.statContext.movieId,event_name:e,client_time:Date.now(),application:this.statContext.application??`@vkontakte/videoplayer-statistics:${O}`,platform:this.isValidPlatform(this.statContext.platform)?this.statContext.platform:`web:${this.statContext.isMobile||z.clientChecker.device.isMobile?`mobile`:`desktop`}`,product:this.statContext.product,event_number:n,playback_position:Math.round(r),current_tvt:this.getTotalViewTime(),cdn_host:this.cdnHostname};for(let e of this.config.disabledParams)delete i[e];return i}createRequiredFatParams(){let e={stats_version:lt,browser:this.statContext.browser??z.clientChecker.browser.current,browser_version:this.statContext.browserVersion??String(z.clientChecker.browser.currentVersion),os:this.statContext.os??z.clientChecker.device.os.name,os_version:this.statContext.osVersion??z.clientChecker.device.os.version,device_type:this.statContext.deviceType??(this.statContext.isMobile||z.clientChecker.device.isMobile?`mobile`:`desktop`),device_manufacturer:this.statContext.deviceManufacturer??z.clientChecker.device.details.vendor,device_model:this.statContext.deviceModel??z.clientChecker.device.details.model,navigation:this.statContext.navigation};for(let t of this.config.disabledParams)delete e[t];return e}log(e,t,n=!1){if(this.config.disabledEvents.includes(e))return;let r=this.createRequiredParams(e),i={};n&&(i=this.createRequiredFatParams()),this.logger.log({...r,...i,...t})}logWatchCoverage(e){this.log(`watch_coverage`,e,!0)}logWatchCoverageLive(e){this.log(`watch_coverage_live`,e,!0)}logWatchedN(e){this.log(`watched_n`,e,!0)}logStartSession(e){this.log(`start_session`,e,!0)}logPreloadStarted(e){this.log(`preload_started`,e)}logPreloadEnded(e){this.log(`preload_ended`,e)}logPreloadError(){this.log(`preload_error`)}logPlaybackStarted(e){this.log(`playback_started`,e,!0)}logPlay(){this.log(`play`)}logManifestRequest(e){this.log(`manifest_request`,e)}logFirstByteManifest(e){this.log(`first_byte_manifest`,e)}logFirstMediaRequest(e){this.log(`first_media_request`,e)}logFirstByteMedia(e){this.log(`first_byte_media`,e)}logManifestReceived(e){this.log(`manifest_received`,e)}logConnectionEstablished(e){this.log(`connection_established`,e)}logFirstVideoFrameDecoded(){this.log(`first_video_frame_decoded`)}logReady(e){this.log(`ready`,e)}logPlaying(e){this.log(`playing`,e)}logPause(){this.log(`pause`)}logResume(){this.log(`resume`)}logSeeking(e){this.log(`seeking`,e)}logSeeked(e){this.log(`seeked`,e)}logContentEndReached(){this.log(`content_end_reached`)}logStop(){this.log(`stop`)}logBufferStarvation(e){this.log(`buffer_starvation`,e)}logShowLoader(){this.log(`show_loader`)}logQualityChangeRequested(e){this.log(`quality_change_requested`,e)}logQualityChanged(e){this.log(`quality_changed`,e)}logCdnHostChanged(e){this.log(`cdn_host_changed`,e)}logViewPortChanged(e){this.log(`view_port_changed`,e)}logModeChanged(e){this.log(`mode_changed`,e)}logVisibilityChanged(e){this.log(`visibility_changed`,e)}logAudioTrackSwitched(e){this.log(`audio_track_switched`,e)}logVideoTrackSwitched(e){this.log(`video_track_switched`,e)}logSubtitlesSwitched(e){this.log(`subtitles_switched`,e)}logSoundChanged(e){this.log(`sound_changed`,e)}logFailover(e){this.log(`failover`,e)}logContentTypeChange(e){this.log(`content_type_change`,e)}logError(e){this.log(`error`,e,!0)}logCannotPlay(){this.log(`cannot_play`,{},!0)}logDownloadBytes(e){this.log(`download_bytes`,e)}logTelemetry(e){this.log(`telemetry`,e)}logAdvConfiguration(e){this.log(`adv_configuration`,e)}logAdvRequest(e){this.log(`adv_request`,e)}logAdvBreakStarted(e){this.log(`adv_break_started`,e)}logAdvBreakEnded(e){this.log(`adv_break_ended`,e)}logAdvError(e){this.log(`adv_error`,e)}resubscribeLifecycleFlush(){this.lifecycleFlushSubscription?.unsubscribe();let e=R.getWindow(),t=()=>{this.logger.flush({wantBeacon:!0,clearStorage:this.config.clearStorageAtUnload}).catch(()=>{})},n=new R.Subscription;n.add(R.fromEvent(e,`beforeunload`).subscribe(t)),n.add(R.fromEvent(e,`pagehide`).subscribe(t)),n.add(R.fromEvent(e.document,`visibilitychange`).subscribe(e=>{e.target?.visibilityState===`hidden`&&t()})),this.lifecycleFlushSubscription=n}}var ft=require(`@vkontakte/videoplayer-shared`),pt=require(`@vkontakte/videoplayer-shared`),$=require(`@vkontakte/videoplayer-core`),mt=require(`@vkontakte/videoplayer-shared`);class ht{startTime=null;stopTime=null;accumulatedTime=0;get time(){return Math.floor(this.timeMs/1e3)}get timeMs(){return this.startTime===null?0:(this.stopTime??this.now())-this.startTime-this.accumulatedTime}get isRunning(){return this.startTime!==null&&this.stopTime===null}start(){this.isRunning||(this.stopTime===null?this.startTime=this.now():(this.accumulatedTime+=this.now()-this.stopTime,this.stopTime=null))}stop(){this.isRunning&&(this.stopTime=this.now())}reset(){this.startTime=null,this.stopTime=null,this.accumulatedTime=0}now(){return Date.now()}}var gt=()=>window.Image?new Image:document.createElement(`img`);class _t{params;position=0;started=!1;isActiveLive;heartbeatPixels=[];heartbeatLastTimeSent={};heartbeatInterval;heartbeatFirstTimeoutId;idleCallbackIds=[];log;stopwatch=new ht;subscription=new $.Subscription;constructor(e){this.params=e,this.log=new mt.Logger().createComponentLog(`MediascopePixel`)}attachTo(e){this.subscription.add(e.info.playbackState$.subscribe(e=>this.onPlaybackState(e))).add(e.info.position$.subscribe(e=>this.onPosition(e))).add(e.info.atLiveEdge$.subscribe(e=>this.isActiveLive=e)),this.subscription.add(e.events.willSeek$.subscribe(()=>this.onWillSeek())).add(e.events.seeked$.subscribe(()=>this.onSeeked())).add(e.events.ended$.subscribe(()=>this.onEnded()))}destroy(){this.stopwatch.reset(),this.subscription.unsubscribe(),window.clearTimeout(this.heartbeatFirstTimeoutId),this.idleCallbackIds.forEach(e=>ft.cancelIdleCallbackPolyfilled(e)),this.send(`stop`)}onPlaybackState(e){e===$.PlaybackState.PLAYING?this.started?this.play():this.start():e===$.PlaybackState.PAUSED&&this.pause()}start(){this.started=!0,this.stopwatch.start(),this.send(`start`),this.heartbeatPixels=this.prepareHeartbeatPixels(),this.heartbeatInterval=this.heartbeatPixels[0]?.interval;let e=this.heartbeatPixels[0]?.url,t=this.heartbeatPixels[0]?.delay;if(t!==void 0&&e)try{this.heartbeatFirstTimeoutId=window.setTimeout(()=>{this.sendHeartbeat(e)},t*1e3)}catch(e){this.log({message:e.message})}}play(){this.stopwatch.start(),this.send(`resume`)}pause(){this.stopwatch.stop(),this.send(`pause`)}onPosition(e){if(this.position=e,this.heartbeatInterval&&this.stopwatch.time!==0&&this.stopwatch.time%this.heartbeatInterval===0)for(let[e,{url:t}]of this.heartbeatPixels.entries())this.heartbeatLastTimeSent[e]!==this.stopwatch.time&&(this.heartbeatLastTimeSent[e]=this.stopwatch.time,this.sendHeartbeat(t))}onWillSeek(){this.started&&this.send(`stop`)}onSeeked(){this.started&&this.send(`start`)}onEnded(){this.stopwatch.stop(),this.send(`stop`)}validatePixels(e,t){return e?.filter(e=>e.event===t.event&&t.keys.every(t=>!!e[t]))}getFrameTimestamp(){let e;if(this.isActiveLive){let t=Math.floor(Math.random()*25)+5;e=Date.now()/1e3-t}else e=this.position||0;return Math.floor(e)}getUTC(){return Math.floor(Date.now()/1e3)}prepareUrl(e){return e.replace(`{@fts_fake_sec}`,String(this.getFrameTimestamp())).replace(`{@utc_sec}`,String(this.getUTC())).split(`&`).filter(e=>!/={@[a-zA-Z_]+}/.test(e)).join(`&`)}preparePixels(e){return this.validatePixels(this.params.pixels,{event:e,keys:[`url`]})}prepareHeartbeatPixels(){return this.validatePixels(this.params.pixels,{event:`heartbeat`,keys:[`url`,`interval`]})}call(e){try{this.idleCallbackIds.push(pt.requestIdleCallbackPolyfilled(()=>gt().src=this.prepareUrl(e)))}catch(e){this.log({message:e.message})}}send(e){let t=this.preparePixels(e);for(let{url:e}of t)this.call(e)}sendHeartbeat(e){this.call(e)}}
6
+ var e=Object.defineProperty,t=Object.getOwnPropertyNames,n=Object.getOwnPropertyDescriptor,r=Object.prototype.hasOwnProperty;function i(e){return this[e]}var a=a=>{var s=(o??=new WeakMap).get(a),c;if(s)return s;if(s=e({},`__esModule`,{value:!0}),a&&typeof a==`object`||typeof a==`function`)for(var l of t(a))r.call(s,l)||e(s,l,{get:i.bind(a,l),enumerable:!(c=n(a,l))||c.enumerable});return o.set(a,s),s},o,s=e=>e;function c(e,t){this[e]=s.bind(null,t)}var l=(t,n)=>{for(var r in n)e(t,r,{get:n[r],enumerable:!0,configurable:!0,set:c.bind(n,r)})},u={};l(u,{VERSION:()=>A,ThinOneStat:()=>Ce,SeekAction:()=>g,Quality:()=>p,OneStat:()=>Se,MediascopePixel:()=>vt,InteractiveInterfaceClick:()=>_,ContentType:()=>m,ConnectionType:()=>h,ApiEnv:()=>f}),module.exports=a(u);var d=require(`@vkontakte/videoplayer-shared`),f;(e=>{e.PROD=`prod`,e.VK_ALIAS=`vk_alias`,e.VIDEOTEST=`videotest`,e.TEST=`test`,e.OKCDN=`okcdn`,e.AUTO=`auto`})(f||={});var p;(e=>{e.Q144P=`mobile`,e.Q240P=`lowest`,e.Q360P=`low`,e.Q480P=`medium`,e.Q720P=`high`,e.Q1080P=`fullhd`,e.Q1440P=`quadhd`,e.Q2160P=`ultrahd`,e.UNKNOWN=`unknown`})(p||={});var m;(e=>{e.MP4=`mp4`,e.DASH=`dash`,e.DASH_SEP=`dash_sep`,e.ONDEMAND_DASH=`ondemand_dash`,e.HLS=`hls`,e.HLS_FMP4=`hls_fmp4`,e.ONDEMAND_HLS=`ondemand_hls`,e.WEBM=`webm`,e.AV1=`av1`,e.ONDEMAND_DASH_LIVE=`ondemand_dash_live`,e.ONDEMAND_HLS_LIVE=`ondemand_hls_live`,e.WEBRTC=`webrtc`,e.UNKNOWN=`unknown`,e.RTMP=`rtmp`})(m||={});var h;(e=>{e.HTTP1=`http1`,e.HTTP2=`http2`,e.HTTP3=`http3`})(h||={});var g;(e=>{e.SLIDER=`slider`,e.RICH=`rich`,e.DOUBLE_TAP=`double_tap`,e.TIME_CODE=`time_code`,e.EPISODE=`episode`,e.REWIND=`rewind`,e.LIVE=`live`,e.UNKNOWN=`unknown`})(g||={});var _;(e=>{e.GRAPH_SHOW=`iGraphShow`,e.GRAPH_HIDE=`iGraphHide`,e.NEXT_AREA=`iNextChapterArea`,e.NEXT_BUTTON=`iNextChapterBtn`,e.WATCH_AGAIN=`iWatchAgainBtn`})(_||={});var v={action_play:`ap`,action_play_interactive:`api`,first_bytes:`fb`,player_ready:`pr`,first_frame:`ff`,seek:`sk`,watch_coverage_record:`wr`,watch_coverage_live:`wl`,empty_buffer:`eb`,action_stop:`as`,close_at_empty_buffer:`cb`,content_error:`er`,player_interface_click:`pc`,next_movie:`nm`,track_switch:`ts`,failover:`fo`,quality:`qt`},y={vsid:`si`,isid:`is`,vid:`vi`,place:`pl`,quality:`qt`,cdn_host:`ch`,stat_type:`st`,param:`pm`,vk_app_id:`va`,track_code:`tc`,connection_type:`cnt`,connection_reused:`cr`,cached_data:`cd`,live:`lv`,muted:`mu`,mode:`mo`,subtitles:`sb`,failover:`fo`,download_speed:`ds`,manual_quality:`mq`,ref_domain:`rd`,direct_url:`du`},b={unknown:`un`,mobile:`m`,lowest:`ls`,low:`l`,medium:`md`,high:`h`,fullhd:`f`,quadhd:`q`,ultrahd:`u`},x={pip:`pi`,fullscreen:`fs`,external:`ex`,prefetch:`pr`,airplay:`ap`,chromecast:`cc`,invisible:`iv`,minimal:`minimal`},S=require(`@vkontakte/videoplayer-shared`),C=`CIOPGQJGDIHBABABA`,w={prod:`https://api.ok.ru`,vk_alias:`https://api.mycdn.me`,videotest:`https://videotestapi.ok.ru/api`,test:`https://apitest.ok.ru`,okcdn:`https://api.okcdn.ru`,auto:``},T={apiEnv:`vk_alias`,apiKey:C,apiBaseUrl:null,apiUvRestBaseUrl:`https://uvapi.okcdn.ru`,apiCollector:`ok.mobile.apps.video`,apiForticom:!0,apiUvRest:!1,requestRetryCount:1,useVsid64:!1,firstFlushTime:5e3,flushDebounceTime:1e4,flushMaxWait:6e4,storageExpiration:2160*60*1e3,watchCoverageInterval:15e3,disabledOperations:[`quality`],disabledCustomFields:[],shorten:!0,maxLoops:100,embedUrlParams:[],useBeacon:!0,synchronizeTime:!0,debugLog:!1,clearStorageAtUnload:!0,useVKComIsMobileLogic:!1,backoff:{start:1e3,factor:1.5,max:300*1e3,random:.1},useKeepalive:!1,watchCoverageTimeoutFix:!0,watchCoverageExactTime:!1},E=e=>S.fillWithDefault(e,T);function ee(){let e=`0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz`.split(``),t=Array(36),n=0,r,i;for(i=0;i<36;i++)i===8||i===13||i===18||i===23?t[i]=`-`:i===14?t[i]=`4`:(n<=2&&(n=33554432+Math.random()*16777216|0),r=n&15,n>>=4,t[i]=e[i===19?r&3|8:r]);return t.join(``)}var D=require(`@vkontakte/videoplayer-shared`),O=require(`@vkontakte/videoplayer-shared`),te=()=>Math.floor(Math.random()*2**32).toString(36).padStart(13,`0`),ne=()=>Math.floor(Math.random()*2**64).toString(36).padStart(13,`0`);class k{static vsid;static getVSID(){return k.vsid}static generateVSID(e){return k.vsid=e?ne():te(),k.vsid}}var re=e=>{let{operation:t,custom:n}=e,r=Object.fromEntries(Object.entries(n).map(([e,t])=>{let n=y[e]??e,r=t;return t&&(e===`mode`?r=x[t]??t:e===`quality`&&(r=b[t]??t)),[n,r]}));return{...e,operation:v[t]??t,custom:r}},A=`1.0.110-dev.92c807ec7.0`,j=require(`@vkontakte/videoplayer-shared`),ie=/Mobile|mini|Fennec|Android|iP(ad|od|hone)|opera (mini|mobi)/i;class ae{id=`forticom`;params;authToken;sessionKey;authorizePromise;refreshAuthTokenPromise;consequentAuthErrors=0;authorized$=new j.ValueSubject(!1);backoffTimeoutId;constructor(e){this.params=e}async authorize(e){return this.authToken=e??await this.refreshAuthToken(),this._authorizeWithBackoff()}logBeacon(e){let t=this.createLogParams(e),n=this.sessionKey;n&&(this.params.apiTransport.sendBeacon(`log.externalLog`,t,n)||this.logRequest(e).catch(()=>{}))}async logRequest(e){let t=`log.externalLog`,n=this.createLogParams(e),r=this.sessionKey??await this._authorizeWithBackoff();if(!r)return;let i=async(e,a=this.params.config.requestRetryCount)=>{try{return await this.params.apiTransport.sendRequest(t,n,r)}catch(n){if(!n||!(`error_code`in n)){this.params.error$.next({id:`logRequestUnknown`,category:j.ErrorCategory.NETWORK,message:`Unknown ${t} error`,thrown:n});return}let r=n?.error_code;switch(r){case 102:case 103:case 104:return this.authorized$.next(!1),this.sessionKey=await this._authorizeWithBackoff(),a>0?i(e,a-1):void 0;case 401:return this.authorized$.next(!1),this.authToken=await this.refreshAuthToken(),this.sessionKey=await this._authorizeWithBackoff(),a>0?i(e,a-1):void 0;default:this.params.error$.next({id:`LogRequest#${r}`,category:j.ErrorCategory.EXTERNAL_API,message:`${t} error`,data:n});return}}};return i(e)}destroy(){j.getWindow().clearTimeout(this.backoffTimeoutId),this.backoffTimeoutId=0}async refreshAuthToken(){if(this.params.refreshAuthToken)return this.refreshAuthTokenPromise||=this.params.refreshAuthToken().finally(()=>{this.refreshAuthTokenPromise=void 0}),this.refreshAuthTokenPromise}createLogParams(e){let t=`WEB`,n=!1,r=`@vkontakte/videoplayer-statistics:${A}`;if(this.params.config.useVKComIsMobileLogic){let{appVersion:e}=j.getWindow().navigator;n=ie.test(e)}else this.params.mobile&&(n=!0);return n&&(t=`M_WEB`),this.params.vktvVersion&&(t=`SMART_TV`,r=this.params.vktvVersion),{collector:this.params.config.apiCollector,data:JSON.stringify({application:r,platform:t,items:this.params.config.shorten?e.map(re):e})}}_authorizeWithBackoff(){if(!this.consequentAuthErrors)return this._authorize();let e=j.getExponentialDelay(this.consequentAuthErrors,this.params.config.backoff);return new Promise(t=>{this.backoffTimeoutId||=j.getWindow().setTimeout(()=>{this._authorize().then(t).catch(e=>this.params.error$.next({id:`AuthorizeBackoff`,category:j.ErrorCategory.NETWORK,message:`Otherwise unhandled error in authorization`,thrown:e})).finally(()=>this.backoffTimeoutId=0)},e)})}async _authorize(){if(this.authorizePromise)return this.authorizePromise;this.sessionKey=void 0,this.authorized$.next(!1);let e={session_data:{version:2,device_id:this.params.deviceId,client_version:A.split(`-`)[0],client_type:`SDK_JS`}};return this.authToken!==void 0&&(e.session_data.auth_token=this.authToken,e.session_data.version=3),this.authorizePromise=this.params.apiTransport.sendRequest(`auth.anonymLogin`,e).then(e=>((!e||!e.session_key)&&this.params.error$.next({id:`AuthorizeFailed`,category:j.ErrorCategory.EXTERNAL_API,message:`No session key`,data:e}),this.sessionKey=e?.session_key??void 0,this.sessionKey)).catch(async e=>{this.sessionKey=void 0;let t=e?.error_code;switch(t){case 401:return this.authToken=await this.refreshAuthToken(),this._authorizeWithBackoff()}t?this.params.error$.next({id:`Authorize#${t}`,category:j.ErrorCategory.EXTERNAL_API,message:`authorize error`,data:e}):this.params.error$.next({id:`AuthorizeUnknown`,category:j.ErrorCategory.NETWORK,message:`authorize error`,thrown:e})}).finally(()=>{this.authorizePromise=void 0,this.consequentAuthErrors=this.sessionKey?0:this.consequentAuthErrors+1,this.authorized$.next(this.sessionKey!==void 0)}),this.authorizePromise}}class oe{id=`uvrest`;authorized$=new j.ValueSubject(!1);params;statToken;authorizePromise;isDestroyed=!1;constructor(e){this.params=e}async authorize(e){if(!this.isDestroyed)return e?(this.statToken=e,this.authorized$.next(!0),e):this.authorizePromise?this.authorizePromise:(this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize().finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.statToken!==void 0)}),this.authorizePromise)}async logRequest(e){if(!e.length)return;let t=this.buildBody(e);try{let e=await this.getValidToken();if(!e){this.params.error$.next({id:`OneStat:UvRestApi:LogRequestNoToken`,category:j.ErrorCategory.NETWORK,message:`No stat token available for log request`});return}await this.params.apiTransport.sendJsonRequest(`/uv/stat/externalLog`,t,{Authorization:`Bearer ${e}`})}catch(e){await this.handleLogRequestError(e)&&await this.retryLogRequestAfterRecovery(t)}}logBeacon(e){if(!e.length)return;let t=this.statToken;if(!t)return;let n=this.buildBody(e);this.params.apiTransport.sendJsonBeacon(`/uv/stat/externalLog`,n,{Authorization:`Bearer ${t}`})}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.statToken=void 0,this.authorized$.next(!1)}async getValidToken(){return this.statToken?this.statToken:this.authorize()}async executeAuthorize(){try{let e=await this.params.getAuthToken();if(this.isDestroyed)return;if(!e){this.params.error$.next({id:`OneStat:UvRestApi:NoToken`,category:j.ErrorCategory.EXTERNAL_API,message:`getAuthToken returned no token`});return}return this.statToken=e,this.statToken}catch(e){if(this.isDestroyed)return;this.params.error$.next({id:`OneStat:UvRestApi:AuthorizeFailed`,category:j.ErrorCategory.NETWORK,message:`Failed to get stat token`,thrown:e});return}}async handleLogRequestError(e){let t=e?.status;return t===401?(this.params.error$.next({id:`OneStat:UvRestApi:TokenExpired`,category:j.ErrorCategory.EXTERNAL_API,message:`Stat token expired (401), refreshing`,thrown:e}),this.authorized$.next(!1),this.statToken=void 0,await this.authorize()!==void 0):(this.params.error$.next({id:`OneStat:UvRestApi:LogRequestFailed#${t??`unknown`}`,category:j.ErrorCategory.EXTERNAL_API,message:`Log request failed with status ${t}`,thrown:e}),!1)}async retryLogRequestAfterRecovery(e){let t=this.statToken;if(!t){this.params.error$.next({id:`OneStat:UvRestApi:RetryNoToken`,category:j.ErrorCategory.NETWORK,message:`Cannot retry log request: no token after recovery`});return}try{await this.params.apiTransport.sendJsonRequest(`/uv/stat/externalLog`,e,{Authorization:`Bearer ${t}`})}catch(e){this.params.error$.next({id:`OneStat:UvRestApi:RetryFailed`,category:j.ErrorCategory.EXTERNAL_API,message:`Log request failed after token refresh`,thrown:e})}}buildBody(e){let t=`WEB`,n=!1,r=`@vkontakte/videoplayer-statistics:${A}`;if(this.params.config.useVKComIsMobileLogic){let{appVersion:e}=j.getWindow().navigator;n=ie.test(e)}else this.params.mobile&&(n=!0);return n&&(t=`M_WEB`),this.params.vktvVersion&&(t=`SMART_TV`,r=this.params.vktvVersion),[{application:r,platform:t,user_id:this.params.userId,items:this.params.config.shorten?e.map(re):e}]}}var M=require(`@vkontakte/videoplayer-shared`),N=require(`@vkontakte/videoplayer-shared`);class se{apiKey;apiBaseUrl;apiUvRestBaseUrl;apiEnv;timeSynchronisation;isApiBaseUrlFetched;params;constructor(e){this.params=e,this.apiKey=e.config.apiKey,this.apiEnv=e.config.apiEnv,this.apiBaseUrl=e.config.apiBaseUrl??w[this.apiEnv],this.apiUvRestBaseUrl=e.config.apiUvRestBaseUrl,this.timeSynchronisation=e.timeSynchronisation,this.isApiBaseUrlFetched=!1}async resolveApiBaseUrl(){if(this.apiEnv!==`auto`||this.isApiBaseUrlFetched)return this.apiBaseUrl;try{let e=atob(`aHR0cHM6Ly9kbnMuZ29vZ2xlL3Jlc29sdmU/bmFtZT12aWRlby5fZW5kcG9pbnQub2sucnUmdHlwZT1UWFQ=`),t=(await(await M.fetchPolyfilled(e,{method:`GET`,mode:`cors`,cache:`no-cache`})).json())?.Answer[0]?.data;if(!t)throw Error(`Wrong DNS response`);return t}catch(e){return this.params.error$.next({id:`OneStat:ApiTransport:resolveApiBaseUrl`,category:N.ErrorCategory.NETWORK,message:`Unhandled resolve api base url error`,thrown:e}),w.vk_alias}finally{this.isApiBaseUrlFetched=!0}}sendBeacon(e,t,n){if(!N.getWindow().Blob||!N.getWindow().navigator.sendBeacon)return!1;let r=this._prepareQueryParams({method:e,queryParams:t,sessionKey:n}),i=N.getWindow().Blob,a=new i([r.toString()],{type:`application/x-www-form-urlencoded`});try{return N.getWindow().navigator.sendBeacon(`${this.apiBaseUrl}/fb.do`,a)}catch(n){this.params.error$.next({id:`OneStat:ApiTransport:sendBeacon`,category:N.ErrorCategory.NETWORK,message:`Unhandled beacon error`,thrown:n,data:{method:e,params:t}})}return!1}async sendRequest(e,t,n){let r=N.now(),i=i=>{if(i instanceof TypeError&&[`Failed to fetch`,`NetworkError when attempting to fetch resource.`].includes(i.message)){this.params.error$.next({id:`Network`,category:N.ErrorCategory.NETWORK,message:`Request failed`,thrown:i});return}this.params.error$.next({id:`OneStat:ApiTransport:sendRequest`,category:N.ErrorCategory.NETWORK,message:`Unhandled request error`,thrown:i,data:{method:e,params:t,sessionKey:n,time:N.now()-r}})};this.apiBaseUrl=await this.resolveApiBaseUrl();let a={method:`post`,headers:{"Content-type":`application/x-www-form-urlencoded`},body:this._prepareQueryParams({method:e,queryParams:t,sessionKey:n})};return this.params.config.useKeepalive&&(a.keepalive=!0),M.fetchPolyfilled(`${this.apiBaseUrl}/fb.do`,a).then(e=>{let t=Number(e.headers.get(`content-length`))===0,n=new Date(e.headers.get(`date`)??``).getTime(),a=N.now()-r;if(isFinite(n)&&this.timeSynchronisation?.addServerTime(n,a),!t)return e.json().then(e=>Object.prototype.hasOwnProperty.call(e,`error_msg`)?Promise.reject(e):e,i)},i)}sendJsonBeacon(e,t,n){let r=`${this.apiUvRestBaseUrl}${e}`;try{M.fetchPolyfilled(r,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t),keepalive:!0}).catch(t=>{this.params.error$.next({id:`OneStat:ApiTransport:sendJsonBeacon`,category:N.ErrorCategory.NETWORK,message:`Keepalive request failed`,thrown:t,data:{path:e}})})}catch(t){this.params.error$.next({id:`OneStat:ApiTransport:sendJsonBeacon`,category:N.ErrorCategory.NETWORK,message:`Failed to initiate keepalive request`,thrown:t,data:{path:e}})}}async sendJsonRequest(e,t,n){let r=N.now(),i=t=>{if(t instanceof TypeError&&[`Failed to fetch`,`NetworkError when attempting to fetch resource.`].includes(t.message)){this.params.error$.next({id:`Network`,category:N.ErrorCategory.NETWORK,message:`JSON request failed`,thrown:t});return}this.params.error$.next({id:`OneStat:ApiTransport:sendJsonRequest`,category:N.ErrorCategory.NETWORK,message:`Unhandled JSON request error`,thrown:t,data:{path:e,time:N.now()-r}})},a={method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t)};return this.params.config.useKeepalive&&(a.keepalive=!0),M.fetchPolyfilled(`${this.apiUvRestBaseUrl}${e}`,a).then(e=>{let t=Number(e.headers.get(`content-length`))===0,n=new Date(e.headers.get(`date`)??``).getTime(),a=N.now()-r;if(isFinite(n)&&this.timeSynchronisation?.addServerTime(n,a),!e.ok)return Promise.reject({status:e.status,statusText:e.statusText});if(!t)return e.json().catch(i)},i)}_prepareQueryParams(e){let t=new URLSearchParams({format:`JSON`,method:e.method,application_key:this.apiKey});return e.sessionKey!==void 0&&t.append(`session_key`,e.sessionKey),Object.entries(e.queryParams).forEach(([e,n])=>t.append(e,typeof n==`string`?n:JSON.stringify(n))),t}}var P=require(`@vkontakte/videoplayer-shared`),F=e=>t=>t.timestamp+e>=P.now();class ce{params;error$;userSalt;isPaused=!1;loggerDebugLog;queues=[];constructor(e){this.params=e,this.error$=e.error$,this.userSalt=e.userSalt,this.loggerDebugLog=e.debugLogger.createComponentLog(`stat logger`);for(let t of e.apis)this.queues.push(this.createQueueState(t));this.housekeepStorage()}createQueueState(e){let t=`onestat_events_${e.id}`,n=P.debounceFn(()=>this.safeFlushQueue(a),this.params.config.firstFlushTime,{maxWait:this.params.config.firstFlushTime}),r=P.safeStorage.isPersistent()?this.params.config.flushMaxWait:0,i=P.debounceFn(()=>this.safeFlushQueue(a),this.params.config.flushDebounceTime,{maxWait:r}),a={api:e,storageKey:t,unsaltedStorage:[],lastVsid:void 0,firstFlush:n,debouncedFlush:i,subscription:e.authorized$.subscribe(e=>{e&&i()})};return a}isUrgent=e=>{let{operation:t}=e;return[`action_play`,`action_play_interactive`,`watch_coverage_record`,`watch_coverage_live`,`action_stop`,`close_at_empty_buffer`].includes(t)};safeFlushQueue(e){try{this.flushQueue(e)}catch(t){this.error$.next({id:`LoggerError`,category:P.ErrorCategory.WTF,message:t?String(t):`Unknown logger error`,thrown:t,data:{storageKey:e.storageKey}})}}readFromStorage(e){let t=P.safeStorage.get(e);try{return t?JSON.parse(t):{}}catch{}return{}}addToStorage(e,t){if(!this.userSalt){e.unsaltedStorage.push(t);return}let n=this.readFromStorage(e.storageKey),r=(n[this.userSalt]??[]).filter(F(this.params.config.storageExpiration));P.safeStorage.set(e.storageKey,JSON.stringify({...n,[this.userSalt]:[...r,t]}))}getFromStorage(e){return this.userSalt?(this.readFromStorage(e.storageKey)[this.userSalt]??[]).filter(F(this.params.config.storageExpiration)):e.unsaltedStorage}markStorageSent(e){if(!this.userSalt){e.unsaltedStorage=[];return}let t=this.readFromStorage(e.storageKey);delete t[this.userSalt],P.safeStorage.set(e.storageKey,JSON.stringify(t))}housekeepStorage(){for(let e of this.queues){let t=this.readFromStorage(e.storageKey);for(let[n,r]of Object.entries(t)){let i=r.filter(F(this.params.config.storageExpiration));i.length?t[n]=i:delete t[n],this.loggerDebugLog({message:`${r.length} retrieved from storage, ${i.length} of them actual (${e.storageKey})`})}P.safeStorage.set(e.storageKey,JSON.stringify(t))}}log(e){for(let t of this.queues)this.addToStorage(t,e),!this.isPaused&&(this.isUrgent(e)?this.flushQueue(t):t.lastVsid===e.custom.vsid?t.debouncedFlush():t.firstFlush(),t.lastVsid=e.custom.vsid)}flush(e={wantBeacon:!1,clearStorage:!0}){for(let t of this.queues)this.flushQueue(t,e)}flushQueue(e,{wantBeacon:t,clearStorage:n}={wantBeacon:!1,clearStorage:!0}){let r=this.getFromStorage(e);r.length!==0&&(e.api.authorized$.getValue()?(t&&this.params.config.useBeacon?(this.loggerDebugLog({message:`Flushing ${r.length} events through beacon (${e.storageKey})`}),e.api.logBeacon(r)):(this.loggerDebugLog({message:`Flushing ${r.length} events (${e.storageKey})`}),e.api.logRequest(r)),n&&this.markStorageSent(e)):e.api.authorize(),e.firstFlush.cancel(),e.debouncedFlush.cancel())}pause(){this.isPaused=!0;for(let e of this.queues)e.debouncedFlush.cancel()}resume(){this.isPaused=!1;for(let e of this.queues)e.debouncedFlush()}destroy(){for(let e of this.queues)e.subscription.unsubscribe(),e.firstFlush.cancel(),e.debouncedFlush.cancel();this.queues=[]}}var I=require(`@vkontakte/videoplayer-core`),le=require(`@vkontakte/videoplayer-shared`),ue=e=>e&&{[I.VideoQuality.INVARIANT]:`unknown`,[I.VideoQuality.Q_144P]:`mobile`,[I.VideoQuality.Q_240P]:`lowest`,[I.VideoQuality.Q_360P]:`low`,[I.VideoQuality.Q_480P]:`medium`,[I.VideoQuality.Q_576P]:`unknown`,[I.VideoQuality.Q_720P]:`high`,[I.VideoQuality.Q_1080P]:`fullhd`,[I.VideoQuality.Q_1440P]:`quadhd`,[I.VideoQuality.Q_2160P]:`ultrahd`,[I.VideoQuality.Q_4320P]:`unknown`}[e],de=e=>e&&{[I.HttpConnectionType.HTTP1]:`http1`,[I.HttpConnectionType.HTTP2]:`http2`,[I.HttpConnectionType.QUIC]:`http3`}[e],fe=e=>{if(e!==void 0)switch(e){case I.VideoFormat.MPEG:return`mp4`;case I.VideoFormat.DASH:case I.VideoFormat.DASH_LIVE:case I.VideoFormat.DASH_STREAMS:return`dash`;case I.VideoFormat.DASH_SEP:return`dash_sep`;case I.VideoFormat.DASH_ONDEMAND:return`ondemand_dash`;case I.VideoFormat.DASH_WEBM:case I.VideoFormat.DASH_LIVE_WEBM:return`webm`;case I.VideoFormat.DASH_WEBM_AV1:return`av1`;case I.VideoFormat.DASH_LIVE_CMAF:return`ondemand_dash_live`;case I.VideoFormat.HLS:case I.VideoFormat.HLS_LIVE:return`hls`;case I.VideoFormat.HLS_FMP4:return`hls_fmp4`;case I.VideoFormat.HLS_ONDEMAND:return`ondemand_hls`;case I.VideoFormat.HLS_LIVE_CMAF:return`ondemand_hls_live`;case I.VideoFormat.WEB_RTC_LIVE:return`webrtc`;default:return le.assertNever(e)}},pe=(e,t)=>{if(t)return`minimal`;switch(e){case void 0:case I.Surface.NONE:case I.Surface.INLINE:return;case I.Surface.FULLSCREEN:return`fullscreen`;case I.Surface.SECOND_SCREEN:return`chromecast`;case I.Surface.PIP:return`pip`;case I.Surface.INVISIBLE:return`invisible`;default:return le.assertNever(e)}},me=e=>{switch(e){case`slow-2g`:return`poor`;case`2g`:return`poor`;case`3g`:return`good`;case`4g`:return`excellent`}};class he{offset=void 0;constructor(e){this.offset=e}getOffset(){return this.offset??0}setOffset(e){this.offset=e}now(){return Date.now()+(this.offset??0)}date(e=new Date){return e.setTime(e.getTime()+(this.offset??0)),e}addServerTime(e,t){let n=e-Date.now()-t/2;if(Math.abs(n)<1e3){this.offset=0;return}if(this.offset===void 0)this.offset=Math.round(n);else{let e=.2;this.offset=Math.round(e*n+(1-e)*this.offset)}}}var ge=require(`@vkontakte/videoplayer-core`),L=require(`@vkontakte/videoplayer-shared`),_e=(e,t)=>new L.Observable(n=>{let r=new L.Subscription,i=L.merge(L.fromEvent(window,`beforeunload`),e.events.willDestruct$),a=new L.ValueSubject(void 0),o;r.add(e.info.isLive$.pipe(L.filterChanged()).subscribe(t=>{o&&(o.unsubscribe(),a.next(void 0)),o=t?e.info.liveTime$.pipe(L.map(e=>e&&e/1e3)).subscribe(a):e.info.position$.subscribe(a),r.add(o)}));let{playing$:s,paused$:c,looped$:l}=e.events,u=e.events.willSeek$.pipe(L.filter(()=>e.info.playbackState$.getValue()===ge.PlaybackState.PLAYING)),d=e.events.seeked$.pipe(L.filter(()=>e.info.playbackState$.getValue()===ge.PlaybackState.PLAYING)),f=!1,p=new L.Subject;r.add(u.subscribe(()=>{f||p.next(),f=!0})),r.add(d.subscribe(()=>f=!1));let m=new L.Subject,h=new L.Subject,g=()=>t.watchCoverageExactTime?e.info.isLive$.getValue()?e.getExactLiveTime():e.getExactTime():a.getValue()??0,_=e.info.stallStartTime$.pipe(L.filterChanged(),L.filter(e=>e>0)),v=L.merge(s,d,m).pipe(L.map(()=>g())),y=L.merge(l,L.merge(c,p,i,h,_).pipe(L.map(()=>g()))),b,x=e=>{b=e},S=e=>{let t=e;L.isNullable(b)||b===t||L.isNullable(t)||(n.next({from:b,to:t}),b=void 0)},C=()=>{h.next(),m.next()};if(r.add(v.subscribe(x)),r.add(y.subscribe(S)),t.forceInterval&&isFinite(t.forceInterval))if(t.watchCoverageTimeoutFix){let e,n=()=>{e!==void 0&&(window.clearTimeout(e),e=void 0)};r.add(v.subscribe(()=>{n(),e=window.setTimeout(C,t.forceInterval)})),r.add(y.subscribe(()=>n())),r.add(()=>n())}else{let e=0;r.add(v.subscribe(()=>e=window.setTimeout(C,t.forceInterval))),r.add(y.subscribe(()=>window.clearTimeout(e)))}return r}),R=require(`@vkontakte/videoplayer-core`),ve=require(`@vkontakte/videoplayer-shared`),ye=(e,t,n)=>{let r=new ve.Subject,i=r.pipe(ve.debounce(0),ve.filterChanged()),a=!1,o=!1,s=!1;return t(e.inPiP$,e=>{a=e,r.next(e?R.Surface.PIP:o?R.Surface.FULLSCREEN:s?R.Surface.INLINE:void 0)}),t(e.inFullscreen$,e=>{o=e,r.next(e?R.Surface.FULLSCREEN:a?R.Surface.PIP:s?R.Surface.INLINE:void 0)}),n&&t(n.info.surface$,e=>{switch(e){case R.Surface.SECOND_SCREEN:case R.Surface.NONE:case R.Surface.INLINE:!o&&!a&&(e===R.Surface.INLINE&&(s=!0),r.next(e));break}}),{surface$:i}},be=`_one-stat_deviceId`,xe=(e,t)=>e&&t?`${e}_${t}`:e??t,z=()=>{let e=new D.Subscription;return{subscription:e,subscribe:(t,n)=>{t&&e.add(t.subscribe(n))}}};class Se{apis;logger;config;subscription=new D.Subscription;beforeunloadSubscription;timeSynchronisation;debugLogger=new O.Logger;oneStatDebugLog=this.debugLogger.createComponentLog(`onestat`);loopCounter=0;disabled=!1;experimental;get vsid$(){return new D.ValueSubject(k.getVSID())}isid$=new D.ValueSubject(void 0);zenUid$=new D.ValueSubject(void 0);statContext;isEmbed;embedParent;uiEvents;player;seekAction$=new D.ValueSubject(`unknown`);deviceId;constructor(e,t){this.statContext=e,this.config=E(t.config??{}),D.isNonNullable(t.apiEnv)&&(this.config.apiEnv=t.apiEnv),this.config.synchronizeTime&&(this.timeSynchronisation=new he);let n=new D.Subject;this.experimental={error$:n};let r=D.safeStorage.get(be);this.statContext.deviceId?this.deviceId=this.statContext.deviceId:r?this.deviceId=r:(this.deviceId=ee(),D.safeStorage.set(be,this.deviceId)),this.resetViewSession(),t.useIsid&&this.isid$.next(t.isid??ne()),this.zenUid$.next(t.zenUid);let i=new se({config:this.config,error$:n,timeSynchronisation:this.timeSynchronisation}),a=[];this.config.apiForticom&&a.push(new ae({config:this.config,apiTransport:i,refreshAuthToken:t.refreshAuthToken,mobile:this.statContext.mobile??!1,deviceId:this.deviceId,error$:n,vktvVersion:t.vktvVersion})),this.config.apiUvRest&&(t.getUvRestAuthToken?a.push(new oe({config:this.config,apiTransport:i,error$:n,userId:this.statContext.userId,getAuthToken:t.getUvRestAuthToken,mobile:this.statContext.mobile??!1,vktvVersion:t.vktvVersion})):n.next({id:`OneStat:Init:MissingGetUvRestAuthToken`,category:O.ErrorCategory.WTF,message:`oldStatToUvRest enabled but getUvRestAuthToken callback not provided`})),this.apis=a,this.logger=new ce({config:this.config,debugLogger:this.debugLogger,apis:this.apis,error$:n,userSalt:t.userSalt});let{isEmbed:o,topOrigin:s}=O.detectEmbed();this.isEmbed=o,this.embedParent=s?new URL(s).hostname:void 0,this.subscribe()}updateContext(e){this.statContext={...this.statContext,...e}}attachTo(e){let t=new D.Subscription,n=(e,n)=>t.add(e.subscribe(n)),r,i,a,o=!1,s=!1,c=()=>{o||(o=!0,D.isNonNullable(r)&&(r={startedAt:D.now(),liveAtStall:!!this.statContext.liveEdge}))};n(e.events.willStart$,()=>{let t=e.info.position$.getValue();this.logActionPlay({position:t}),this.statContext.projectId&&this.logActionPlayInteractive()}),n(e.events.looped$,()=>{this.resetViewSession(),++this.loopCounter>this.config.maxLoops&&(this.disabled=!0),this.logActionPlay({position:e.info.position$.getValue()})}),n(e.events.willSeek$,()=>{D.isNonNullable(r)&&o&&this.logEmptyBuffer({duration:D.now()-r.startedAt,liveAtStall:r.liveAtStall}),r=void 0,s=!0}),n(e.events.seeked$,()=>{!s&&D.isNonNullable(r)&&o&&(this.logEmptyBuffer({duration:D.now()-r.startedAt,liveAtStall:r.liveAtStall}),r=void 0),s=!1;let t=this.seekAction$.getValue();this.logSeek({action:t===`rich`?`slider`:t,time:e.info.position$.getValue()}),this.seekAction$.next(`unknown`)}),n(e.events.paused$,()=>{this.logPause({position:e.info.position$.getValue()})}),n(e.events.willResume$,()=>{this.logPlay()}),n(e.events.started$,()=>{this.statContext.clip&&this.logWatchCoverageRecord({start:0,end:0})}),n(e.events.playing$,()=>{this.logPlaying()}),n(_e(e,{forceInterval:this.config.watchCoverageInterval,watchCoverageTimeoutFix:this.config.watchCoverageTimeoutFix,watchCoverageExactTime:this.config.watchCoverageExactTime}),t=>{let n=e.info.isLive$.getValue(),r=e.info.atLiveEdge$.getValue(),i={start:parseFloat(t.from.toFixed(3)),end:parseFloat(t.to.toFixed(3))};if(r){let t=e.info.liveLatency$.getValue()??0,n=e.info.currentBuffer$.getValue()?.end??0,r=e.info.liveBufferTime$.getValue()??0,a=Math.round((t-(n-r))*1e3),o=Math.round(t*1e3);a&&o&&(i.latency=a,i.bufferLatency=o)}n?this.logWatchCoverageLive(i):this.logWatchCoverageRecord(i)}),n(e.info.stallStartTime$,e=>{e>0?r={startedAt:e,liveAtStall:!!this.statContext.liveEdge}:(D.isNonNullable(r)&&o&&this.logEmptyBuffer({duration:Math.round(D.now()-r.startedAt),liveAtStall:r.liveAtStall}),r=void 0)});let l=!1;t.add(e.events.fatalError$.pipe(D.once()).subscribe(()=>l=!0)),n(e.events.willStop$,()=>{if(e.info.isBuffering$.getValue()&&o){let e=D.isNonNullable(r)?D.now()-r.startedAt:void 0;this.logCloseAtEmptyBuffer({duration:e??0,liveAtStall:r?.liveAtStall}),r=void 0}else l||this.logActionStop()}),n(e.events.managedError$,({id:e})=>{this.logError({fatal:!1,errorType:e})}),n(e.events.fatalError$,({id:e})=>{this.logError({fatal:!0,errorType:e})}),n(e.events.firstBytes$,e=>{i=D.now(),this.logFirstBytes({time:e})}),n(e.events.willStart$,()=>a=D.now()),n(e.info.currentBuffer$,e=>{!o&&e&&e.end-e.start>0&&D.isNonNullable(i)&&(this.logPlayerReady({duration:D.now()-i}),c())}),n(e.events.firstFrame$,()=>{D.isNonNullable(i)&&!o&&(this.logPlayerReady({duration:D.now()-i}),c()),D.isNonNullable(a)&&this.logFirstFrame({time:D.now()-a})});let u;n(e.info.currentVideoStream$,e=>{e&&(this.updateContext({videoCodec:e.codec}),u&&e.id!==u&&this.logTrackSwitch(e),u=e.id)});let d;n(e.info.currentAudioStream$,e=>{e&&(this.updateContext({audioCodec:e.codec}),d&&e.id!==d&&this.logTrackSwitch(e),d=e.id)}),n(e.info.atLiveEdge$,e=>this.updateContext({liveEdge:e})),n(D.combine({muted:e.info.muted$,volume:e.info.volume$}),({muted:e,volume:t})=>this.updateContext({audible:!e&&t>0})),n(e.info.currentQuality$,e=>{let t=ue(e);this.updateContext({quality:t}),t&&this.logQuality(t)}),n(e.info.isAutoQualityEnabled$,e=>this.updateContext({autoQuality:e})),n(e.info.currentFormat$,e=>this.updateContext({contentType:fe(e)})),n(e.info.currentPlaybackRate$,e=>this.updateContext({rate:e})),n(e.info.is3DVideo$,e=>this.updateContext({is3d:e}));let f;return n(e.info.hostname$,e=>{let t=f!==void 0&&f!==e;this.updateContext({cdnHostname:e,failover:t}),t&&this.logFailover(e),f=e}),n(e.info.throughputEstimation$,e=>this.updateContext({downloadSpeed:e})),n(e.info.httpConnectionType$,e=>{this.statContext.firstConnectionType||this.updateContext({firstConnectionType:de(e)}),this.updateContext({connectionType:de(e)})}),n(e.info.httpConnectionReused$,e=>{D.isNullable(this.statContext.firstConnectionReused)&&this.updateContext({firstConnectionReused:e}),this.updateContext({connectionReused:e})}),n(e.info.surface$,e=>{this.uiEvents||this.updateContext({mode:pe(e,this.statContext.isPreviewPlayerView)})}),n(D.combine({current:e.info.currentTextTrack$,available:e.info.availableTextTracks$}),({current:e,available:t})=>{let n=t.find(({id:t})=>t===e),r=n&&(n.isAuto?`${n.language}_auto`:n.language);this.updateContext({subtitles:r?.split(`.`)[0]})}),this.player=e,this.uiEvents&&this.attachToUi(this.uiEvents),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToUi(e){this.uiEvents=e;let{subscription:t,subscribe:n}=z();if(!this.player)return t;n(e.actionRewind$,()=>{this.player&&(this.resetViewSession(),this.logActionPlay({position:this.player.info.position$.getValue()}))}),n(e.actionSeek$,this.seekAction$);let{surface$:r}=ye(this.uiEvents,n,this.player);return n(r,e=>{this.updateContext({mode:pe(e,this.statContext.isPreviewPlayerView)})}),n(e.actionSetSubtitle$,e=>this.updateContext({subtitles:e?.split(`.`)[0]})),n(e.nextMovie$,e=>this.logNextMovie(e)),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToAds(e){let{subscription:t,subscribe:n}=z();return n(e.slotRequested$,()=>this.logAdSlotRequest()),n(e.started$,e=>this.logAdStarted(e)),n(e.paused$,()=>this.logAdPaused()),n(e.resumed$,()=>this.logAdResumed()),n(e.ended$,()=>this.logAdEnded()),n(e.skipped$,()=>this.logAdSkipped()),n(e.clicked$,()=>this.logAdClicked()),n(e.error$,e=>this.logError({errorType:e,fatal:!1})),this.resubscribeBeforeunload(),this.subscription.add(t),t}attachToInteractive(e){let{subscription:t,subscribe:n}=z();return n(e.click$,e=>this.logInterfaceClick(e)),n(e.nextMovie$,e=>this.logNextMovie(e)),this.resubscribeBeforeunload(),this.subscription.add(t),t}authorize(e){return Promise.all(this.apis.map(t=>t.authorize(e)))}pause(){this.logger.pause(),this.oneStatDebugLog({message:`paused`})}resume(){this.logger.resume(),this.oneStatDebugLog({message:`resumed`})}destroy(){this.logger.flush(),this.subscription.unsubscribe(),this.apis.forEach(e=>e.destroy()),this.logger.destroy()}resetViewSession(){this.oneStatDebugLog({message:`VSID reset`}),k.generateVSID(this.config.useVsid64)}getDeviceId(){return this.deviceId}setFieldBlacklist(){}logInited(){}logReady(e){this.log({operation:`player_ready`,param:String(e.time)})}logStarted(...e){this.logActionPlay(...e)}logPlay(){this.log({operation:`play_toggle`})}logPlaying(){this.log({operation:`playing`})}logPause(e){this.log({operation:`pause`,param:String(Math.round(e.position))})}logSeek(e){this.log({operation:`seek`,param:e.action,time:e.time})}logFirstBytes(e){this.log({operation:`first_bytes`,param:String(e.time)})}logFirstFrame(e){this.log({operation:`first_frame`,param:String(e.time)})}logError(e){this.log({operation:`content_error`,param:`${e.fatal?`fatal`:`recoverable`}_${e.errorType}`})}logWatchCoverageRecord(e){this.log({operation:`watch_coverage_record`,param:`${e.start}-${e.end}`},{in_history:this.statContext.inHistory?1:void 0})}logWatchCoverageLive(e){let t=this.timeSynchronisation?.getOffset()??0,n=e.start+t,r=e.end+t;this.log({operation:`watch_coverage_live`,param:`${n}-${r}`},{in_history:this.statContext.inHistory?1:void 0,latency:e.latency,buffer_latency:e.bufferLatency})}logEmptyBuffer(e){this.log({operation:`empty_buffer`,param:String(e.duration)},D.isNonNullable(e.liveAtStall)?{live:e.liveAtStall?1:void 0}:{})}logDownloadSpeed(){}logAdSlotRequest(){this.log({operation:`adv`,param:`slot_request`})}logAdStarted(e){this.log({operation:`adv`,param:e})}logAdPaused(){this.log({operation:`adv`,param:`pause`})}logAdResumed(){this.log({operation:`adv`,param:`resume`})}logAdEnded(){this.log({operation:`adv`,param:`ended`})}logAdSkipped(){this.log({operation:`adv`,param:`skip`})}logAdClicked(){this.log({operation:`adv`,param:`click`})}logInterfaceClick(e){this.log({operation:`player_interface_click`,param:e})}logNextMovie(e){this.log({operation:`next_movie`,param:String(e)})}subscribe(){this.resubscribeBeforeunload();let e=D.getWindow().navigator.connection;e&&`onchange`in e&&`effectiveType`in e&&this.subscription.add(D.merge(D.fromEvent(e,`change`),D.observableFrom([`init`])).subscribe(()=>this.updateContext({network:me(e.effectiveType)}))),this.config.debugLog&&this.debugLogger.log$.subscribe(e=>{console.debug(`%c stat `,`background:#fa6470;`,`component: ${e.component}.`,e.message)})}resubscribeBeforeunload(){this.beforeunloadSubscription?.unsubscribe(),this.beforeunloadSubscription=D.fromEvent(window,`beforeunload`).subscribe(()=>this.logger.flush({wantBeacon:!0,clearStorage:this.config.clearStorageAtUnload})),this.subscription.add(this.beforeunloadSubscription)}logPlayerReady(e){this.log({operation:`player_ready`,param:String(Math.round(e.duration))})}logActionPlay(e){this.log({operation:`action_play`,param:String(Math.round(e.position))})}logActionPlayInteractive(){this.statContext.projectId&&this.log({operation:`action_play_interactive`,param:String(this.statContext.movieId)},{vid:this.statContext.projectId})}logFailover(e){this.log({operation:`failover`,param:e})}logCloseAtEmptyBuffer(e){this.log({operation:`close_at_empty_buffer`,param:String(Math.round(e.duration))},D.isNonNullable(e.liveAtStall)?{live:e.liveAtStall?1:void 0}:{})}logActionStop(){this.log({operation:`action_stop`})}logTrackSwitch(e){this.log({operation:`track_switch`,param:e.id})}logQuality(e){this.log({operation:`quality`,param:e})}log(e,t={}){if(this.disabled){this.oneStatDebugLog({message:`operation ${e.operation} but statistics is disabled`});return}if(this.config.disabledOperations.includes(e.operation)){this.oneStatDebugLog({message:`operation ${e.operation} but it is disabled`});return}this.oneStatDebugLog({message:`operation ${e.operation} ${e.param}`});let n=this.createLogItem(e,t);this.logger.log(n)}createLogItem({operation:e,param:t,time:n},r={}){let i=this.timeSynchronisation?.now()??D.now(),a=this.vsid$.getValue();D.assertNonNullable(a);let o=this.isid$.getValue(),s=e===`empty_buffer`||e===`close_at_empty_buffer`?this.statContext.connectionType:this.statContext.firstConnectionType,c=this.statContext.firstConnectionReused,l,u;if(this.isEmbed||this.statContext.place===`embed`){l=this.embedParent;let e=[...new URLSearchParams(location.search).entries()].filter(([e,t])=>this.config.embedUrlParams.includes(e));u=new URLSearchParams(e).toString()}else if(this.statContext.place===`direct`){let e=this.statContext.refDomain||document.referrer;l=e&&d.urlCanParsePolyfilled(URL,e)?new URL(e).hostname:e,u=location.href.substring(0,1024)}let f={vsid:a,isid:o,vid:this.statContext.movieId,uid:this.zenUid$.getValue(),ct:this.statContext.contentType,place:this.isEmbed?`embed`:this.statContext.place,quality:this.statContext.quality,cdn_host:this.statContext.cdnHostname,stat_type:this.statContext.autoplay===!0?`auto`:this.statContext.autoplay===!1?``:void 0,param:t,vk_app_id:this.statContext.vkAppId,track_code:this.statContext.trackCode,connection_type:s,connection_reused:c===!0?1:c===!1?0:void 0,cached_data:this.statContext.cached===!0?1:this.statContext.cached===!1?0:void 0,live:this.statContext.liveEdge?1:void 0,muted:this.statContext.audible===!1?1:void 0,mode:this.statContext.mode,subtitles:this.statContext.subtitles,download_speed:this.statContext.downloadSpeed,manual_quality:this.statContext.autoQuality?void 0:1,ref_domain:l,direct_url:u,rate:this.statContext.rate===1||D.isNullable(this.statContext.rate)?void 0:this.statContext.rate.toFixed(2),view_360:this.statContext.is3d?1:void 0,codec_info:xe(this.statContext.videoCodec,this.statContext.audioCodec),aid:this.statContext.albumId,vk_playlist_id:this.statContext.vkPlaylistId,...r};for(let e of this.config.disabledCustomFields)delete f[e];return{operation:e,type:1,time:n,network:this.statContext.network,timestamp:i,custom:f}}}var Ce={};l(Ce,{VERSION:()=>A,ThinOneStat:()=>dt,ActionSeekType:()=>De});var B=require(`@vkontakte/videoplayer-shared`),V=require(`@vkontakte/videoplayer-core`),we=require(`@vkontakte/videoplayer-shared`),Te=[`play`,`manifest_request`,`first_byte_manifest`,`first_media_request`,`first_byte_media`,`first_video_frame_decoded`,`ready`,`playback_started`,`watched_n`],Ee;(e=>{e.AUTO=`auto`,e.AUTO_POOR=`auto_poor`,e.AUTO_RICH=`auto_rich`})(Ee||={});var De;(e=>{e.SLIDER=`slider`,e.RICH=`rich`,e.DOUBLE_TAP=`double_tap`,e.TIME_CODE=`time_code`,e.EPISODE=`episode`,e.REWIND=`rewind`,e.LIVE=`live`,e.UNKNOWN=`unknown`})(De||={});var Oe=`CIOPGQJGDIHBABABA`,H={prod:`https://api.ok.ru`,vk_alias:`https://api.mycdn.me`,videotest:`https://videotestapi.ok.ru/api`,test:`https://apitest.ok.ru`,okcdn:`https://api.okcdn.ru`,auto:``},ke={apiKey:Oe,apiEnv:`okcdn`,apiBaseUrl:null,apiUvRestBaseUrl:`https://uvapi.okcdn.ru`,apiForticom:!0,apiUvRest:!1,requestRetryCount:3,useVsid64:!1,flushFirstTime:5e3,flushDebounceTime:1e4,flushMaxWait:6e4,flushRetryFirstDelay:3e3,flushRetryMaxDelay:18e5,storageExpiration:2160*60*1e3,clearStorageAtUnload:!1,disabledEvents:[],disabledParams:[],synchronizeTime:!0,useBeacon:!1,alwaysSendDesktop:!1,watchCoverageHeartbeatInterval:3e4,watchCoverageExactTime:!1,watchedNDefaultTargetDuration:1e3,telemetryInterval:3e4,downloadBytesMaxTime:3e4},Ae=e=>we.fillWithDefault(e,ke),U=require(`@vkontakte/videoplayer-shared`);function je(e){let t=Error(`HTTP ${e.status}: ${e.statusText}`);return t.status=e.status,t.statusText=e.statusText,t.response=e,t}function W(){let e=Error(`Request timeout`);return e.isTimeout=!0,e}function Me(e){let t=Error(e.error_msg);return t.errorData=e,t}function G(e){if(!(!e||typeof e!=`object`)){if(`errorData`in e)return e.errorData?.error_code;if(`error_code`in e){let t=e.error_code;return typeof t==`number`?t:void 0}}}var Ne=({application:e,platform:t,product:n,...r})=>r;class Pe{id=`forticom`;authorized$;params;authToken;sessionKey;authorizePromise;refreshAuthTokenPromise;isDestroyed=!1;constructor(e){this.params=e,this.authorized$=new U.ValueSubject(!1)}async authorize(e){return this.authToken=e??await this.refreshAuthToken(),this.doAuthorize()}logBeacon(e){if(!e.length)return;let t=this.createLogParams(e),n=this.sessionKey;n&&(this.params.apiTransport.sendBeacon(`log.logUvStat`,t,n)||this.logRequest(e).catch(()=>{}))}async logRequest(e){if(!e.length)return;let t=`log.logUvStat`,n=this.createLogParams(e),r={};try{let e=await this.getValidSessionKey();if(!e){this.params.errors$.next({id:`ThinOneStat:Api:LogRequestNoSession`,category:U.ErrorCategory.NETWORK,message:`No session key available for log request`});return}await this.params.apiTransport.sendRequest(t,n,e,r)}catch(e){await this.handleLogRequestError(e,t,n)&&await this.retryLogRequestAfterSessionRecovery(t,n)}}async getValidSessionKey(){return this.sessionKey?this.sessionKey:this.doAuthorize()}async handleLogRequestError(e,t,n){let r=G(e);return r===void 0?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestUnknown`,category:U.ErrorCategory.NETWORK,message:`Unknown ${t} error`,thrown:e,data:{params:n}}),!1):this.isSessionExpiredError(r)?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestSessionError#${r}`,category:U.ErrorCategory.EXTERNAL_API,message:`Session error: ${r}`,data:{error:e,code:r}}),this.authorized$.next(!1),this.sessionKey=void 0,await this.doAuthorize()!==void 0):r===401?(this.params.errors$.next({id:`ThinOneStat:Api:LogRequestAuthLogin#${r}`,category:U.ErrorCategory.EXTERNAL_API,message:`Auth token expired`,data:{error:e,code:r}}),this.authorized$.next(!1),this.authToken=await this.refreshAuthToken(),this.sessionKey=void 0,await this.doAuthorize()!==void 0):(this.params.errors$.next({id:`ThinOneStat:Api:LogRequest#${r}`,category:U.ErrorCategory.EXTERNAL_API,message:`${t} error`,data:{error:e,code:r,params:n}}),!1)}async retryLogRequestAfterSessionRecovery(e,t){let n=this.sessionKey;if(!n){this.params.errors$.next({id:`ThinOneStat:Api:LogRequestRetryFailed`,category:U.ErrorCategory.NETWORK,message:`Cannot retry log request: no session key after recovery`});return}try{await this.params.apiTransport.sendRequest(e,t,n)}catch(e){let t=G(e);this.params.errors$.next({id:`ThinOneStat:Api:LogRequestRetryFailed#${t??`unknown`}`,category:U.ErrorCategory.EXTERNAL_API,message:`Log request failed after session recovery`,thrown:e,data:{originalError:e,retryCode:t}})}}isSessionExpiredError(e){return!!e&&[102,103,104].includes(e)}async doAuthorize(e=1){if(!this.isDestroyed)return this.authorizePromise?this.authorizePromise:(this.sessionKey=void 0,this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize(e).finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.sessionKey!==void 0)}),this.authorizePromise)}async executeAuthorize(e){let t=this.buildAuthParams();try{let e=await this.params.apiTransport.sendRequest(`auth.anonymLogin`,t);if(this.isDestroyed)return;if(!e?.session_key){this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeFailed`,category:U.ErrorCategory.EXTERNAL_API,message:`No session key in response`,data:e});return}return this.sessionKey=e.session_key,this.sessionKey}catch(t){if(this.isDestroyed)return;let n=G(t);if(n===401&&this.params.refreshAuthToken){if(this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeAuthLogin`,category:U.ErrorCategory.EXTERNAL_API,message:`Auth token expired during authorization`,data:{error:t,code:n,attempt:e}}),e>=2){this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeMaxAttemptsReached`,category:U.ErrorCategory.EXTERNAL_API,message:`Authorization failed after ${e} attempts`,data:{error:t,code:n}});return}return this.authToken=await this.refreshAuthToken(),this.isDestroyed?void 0:this.executeAuthorize(e+1)}n?this.params.errors$.next({id:`ThinOneStat:Api:Authorize#${n}`,category:U.ErrorCategory.EXTERNAL_API,message:`Authorize error`,data:{error:t,code:n}}):this.params.errors$.next({id:`ThinOneStat:Api:AuthorizeUnknown`,category:U.ErrorCategory.NETWORK,message:`Authorize error`,thrown:t});return}}buildAuthParams(){let e={session_data:{version:2,device_id:this.params.deviceId,client_version:this.getClientVersion(),client_type:`SDK_JS`}};return this.authToken!==void 0&&(e.session_data.auth_token=this.authToken,e.session_data.version=3),e}getClientVersion(){return A}async refreshAuthToken(){if(this.params.refreshAuthToken)return this.refreshAuthTokenPromise||=this.params.refreshAuthToken().catch(e=>{throw this.params.errors$.next({id:`ThinOneStat:Api:RefreshAuthTokenFailed`,category:U.ErrorCategory.NETWORK,message:`Failed to refresh auth token`,thrown:e}),e}).finally(()=>{this.refreshAuthTokenPromise=void 0}),this.refreshAuthTokenPromise}createLogParams(e){let t=e[0],n=t.application??`@vkontakte/videoplayer-statistics:${this.getClientVersion()}`;return this.params.vktvVersion&&(n=this.params.vktvVersion),{collector:`ok.mobile.apps.video`,uv_stat_data:JSON.stringify({application:n,platform:t.platform,product:t.product,events:e.map(Ne)})}}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.refreshAuthTokenPromise=void 0,this.sessionKey=void 0,this.authorized$.next(!1)}}class Fe{id=`uvrest`;authorized$;params;statToken;authorizePromise;isDestroyed=!1;constructor(e){this.params=e,this.authorized$=new U.ValueSubject(!1)}async authorize(e){if(!this.isDestroyed)return e?(this.statToken=e,this.authorized$.next(!0),e):this.authorizePromise?this.authorizePromise:(this.authorized$.next(!1),this.authorizePromise=this.executeAuthorize().finally(()=>{this.authorizePromise=void 0,this.isDestroyed||this.authorized$.next(this.statToken!==void 0)}),this.authorizePromise)}async logRequest(e){if(!e.length)return;let t=this.buildBody(e);try{let e=await this.getValidToken();if(!e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:LogRequestNoToken`,category:U.ErrorCategory.NETWORK,message:`No stat token available for log request`});return}await this.params.apiTransport.sendJsonRequest(`/uv/stat/logUvStat`,t,{Authorization:`Bearer ${e}`})}catch(e){await this.handleLogRequestError(e)&&await this.retryLogRequestAfterRecovery(t)}}logBeacon(e){if(!e.length)return;let t=this.statToken;if(!t)return;let n=this.buildBody(e);this.params.apiTransport.sendJsonBeacon(`/uv/stat/logUvStat`,n,{Authorization:`Bearer ${t}`})}destroy(){this.isDestroyed=!0,this.authorizePromise=void 0,this.statToken=void 0,this.authorized$.next(!1)}async getValidToken(){return this.statToken?this.statToken:this.authorize()}async executeAuthorize(){try{let e=await this.params.getAuthToken();if(this.isDestroyed)return;if(!e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:NoToken`,category:U.ErrorCategory.EXTERNAL_API,message:`getAuthToken returned no token`});return}return this.statToken=e,this.statToken}catch(e){if(this.isDestroyed)return;this.params.errors$.next({id:`ThinOneStat:UvRestApi:AuthorizeFailed`,category:U.ErrorCategory.NETWORK,message:`Failed to get stat token`,thrown:e});return}}async handleLogRequestError(e){let t=e?.status;return t===401?(this.params.errors$.next({id:`ThinOneStat:UvRestApi:TokenExpired`,category:U.ErrorCategory.EXTERNAL_API,message:`Stat token expired (401), refreshing`,thrown:e}),this.authorized$.next(!1),this.statToken=void 0,await this.authorize()!==void 0):(this.params.errors$.next({id:`ThinOneStat:UvRestApi:LogRequestFailed#${t??`unknown`}`,category:U.ErrorCategory.EXTERNAL_API,message:`Log request failed with status ${t}`,thrown:e}),!1)}async retryLogRequestAfterRecovery(e){let t=this.statToken;if(!t){this.params.errors$.next({id:`ThinOneStat:UvRestApi:RetryNoToken`,category:U.ErrorCategory.NETWORK,message:`Cannot retry log request: no token after recovery`});return}try{await this.params.apiTransport.sendJsonRequest(`/uv/stat/logUvStat`,e,{Authorization:`Bearer ${t}`})}catch(e){this.params.errors$.next({id:`ThinOneStat:UvRestApi:RetryFailed`,category:U.ErrorCategory.EXTERNAL_API,message:`Log request failed after token refresh`,thrown:e})}}buildBody(e){let t=e[0],n=t.application;return this.params.vktvVersion&&(n=this.params.vktvVersion),[{product:t.product,platform:t.platform,application:n,user_id:this.params.userId,events:e.map(Ne)}]}}var Ie=require(`@vkontakte/videoplayer-shared`),Le=require(`@vkontakte/videoplayer-shared`),K=require(`@vkontakte/videoplayer-shared`);class Re{params;apiKey;apiBaseUrl;apiUvRestBaseUrl;apiEnv;resolvedApiBaseUrl=null;resolveApiBaseUrlPromise=null;timeSynchronisation;fetchTimeout=3e4;maxTotalRequestTime=6e4;retryConfig;constructor(e){this.params=e,this.apiKey=e.config.apiKey,this.apiEnv=e.config.apiEnv,this.apiBaseUrl=e.config.apiBaseUrl??H[this.apiEnv],this.apiUvRestBaseUrl=e.config.apiUvRestBaseUrl,this.timeSynchronisation=e.timeSynchronisation,this.retryConfig={maxAttempts:e.config.requestRetryCount??3,baseDelay:3e3,maxDelay:3e4,retryableErrors:[`Failed to fetch`,`NetworkError when attempting to fetch resource.`,`Request timeout`,`Network request failed`],retryableStatuses:[408,429,500,502,503,504]}}sendBeacon(e,t,n){if(!K.getWindow().Blob||!K.getWindow().navigator.sendBeacon)return!1;let r=this.resolvedApiBaseUrl??this.apiBaseUrl,i=this.prepareQueryParams({method:e,queryParams:t,sessionKey:n}),a=K.getWindow().Blob,o=new a([i.toString()],{type:`application/x-www-form-urlencoded`});try{return K.getWindow().navigator.sendBeacon(`${r}/fb.do`,o)}catch(n){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendBeacon`,category:K.ErrorCategory.NETWORK,message:`Unhandled beacon error`,thrown:n,data:{method:e,params:t}})}return!1}async sendRequest(e,t,n,r){return this.sendRequestWithRetry(e,t,n,r)}async sendRequestWithRetry(e,t,n,r,i=1,a){let o=K.now(),s=a??o+this.maxTotalRequestTime;if(K.now()>=s){let r=Error(`Request deadline exceeded`);throw this.handleFinalError(e,t,n,o,r,i),r}try{return await this.executeRequest(e,t,n,r,o,i,s)}catch(a){if(this.shouldRetry(a,i)&&K.now()<s){let c=this.getRetryDelay(a,i);if(K.now()+c>=s)throw this.handleFinalError(e,t,n,o,a,i),a;return this.logRetryAttempt(e,i,a,c),await this.sleep(c),this.sendRequestWithRetry(e,t,n,r,i+1,s)}throw this.handleFinalError(e,t,n,o,a,i),a}}async executeRequest(e,t,n,r,i,a,o){let s=i??K.now(),c=await this.resolveApiBaseUrl(),l=o===void 0?this.fetchTimeout:o-K.now();if(l<=0)throw W();try{let i=await this.fetchWithTimeout(`${c}/fb.do`,{method:`post`,headers:{"Content-type":`application/x-www-form-urlencoded`,...r},body:this.prepareQueryParams({method:e,queryParams:t,sessionKey:n})},Math.min(this.fetchTimeout,l));return await this.processResponse(i,s)}catch(t){throw t instanceof Error&&(t.context={method:e,attempt:a,url:`${c}/fb.do`}),t}}async sendJsonRequest(e,t,n){return this.sendJsonRequestWithRetry(e,t,n)}sendJsonBeacon(e,t,n){let r=`${this.apiUvRestBaseUrl}${e}`;try{Ie.fetchPolyfilled(r,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t),keepalive:!0}).catch(t=>{this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonBeacon`,category:K.ErrorCategory.NETWORK,message:`Keepalive request failed`,thrown:t,data:{path:e}})})}catch(t){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonBeacon`,category:K.ErrorCategory.NETWORK,message:`Failed to initiate keepalive request`,thrown:t,data:{path:e}})}}async sendJsonRequestWithRetry(e,t,n,r=1,i){let a=K.now(),o=i??a+this.maxTotalRequestTime;if(K.now()>=o){let t=Error(`Request deadline exceeded`);throw this.handleJsonFinalError(e,a,t,r),t}try{return await this.executeJsonRequest(e,t,n,a,r,o)}catch(i){if(this.shouldRetry(i,r)&&K.now()<o){let s=this.getRetryDelay(i,r);if(K.now()+s>=o)throw this.handleJsonFinalError(e,a,i,r),i;return this.logRetryAttempt(e,r,i,s),await this.sleep(s),this.sendJsonRequestWithRetry(e,t,n,r+1,o)}throw this.handleJsonFinalError(e,a,i,r),i}}async executeJsonRequest(e,t,n,r,i,a){let o=r??K.now(),s=a===void 0?this.fetchTimeout:a-K.now();if(s<=0)throw W();let c=`${this.apiUvRestBaseUrl}${e}`;try{let e=await this.fetchWithTimeout(c,{method:`POST`,headers:{"Content-Type":`application/json`,...n},body:JSON.stringify(t)},Math.min(this.fetchTimeout,s));return await this.processResponse(e,o)}catch(t){throw t instanceof Error&&(t.context={method:e,attempt:i,url:c}),t}}async fetchWithTimeout(e,t,n=this.fetchTimeout){let r=new Le.AbortControllerPolyfilled,i=K.getWindow().setTimeout(()=>r.abort(),n);try{let n=await Ie.fetchPolyfilled(e,{...t,signal:r.signal});return K.getWindow().clearTimeout(i),n}catch(e){throw K.getWindow().clearTimeout(i),K.isAbortError(e)?W():e}}async processResponse(e,t){if(!e.ok){let t=je(e),n=this.parseRetryAfter(e);throw n!==void 0&&(t.retryAfter=n),t}let n=new Date(e.headers.get(`date`)??``).getTime(),r=K.now()-t;isFinite(n)&&this.timeSynchronisation?.addServerTime(n,r);let i=e.headers.get(`content-length`);if(i!==null&&Number(i)===0)return{};let a=await e.json();if(a&&typeof a==`object`&&`error_msg`in a)throw Me(a);return a}shouldRetry(e,t){if(t>=this.retryConfig.maxAttempts||!(e instanceof Error))return!1;let n=e;return!!(n.isTimeout||n.status&&this.retryConfig.retryableStatuses.includes(n.status)||this.retryConfig.retryableErrors.includes(e.message)||e.message.includes(`network`)||e.message.includes(`timeout`)||e.message.includes(`connection`))}logRetryAttempt(e,t,n,r){let i=n instanceof Error?n.message:String(n);this.params.errors$.next({id:`ThinOneStat:ApiTransport:retry`,category:K.ErrorCategory.NETWORK,message:`Retrying request (attempt ${t}/${this.retryConfig.maxAttempts}) after ${Math.round(r)}ms`,thrown:n,data:{method:e,attempt:t,delay:r,errorMessage:i}})}handleFinalError(e,t,n,r,i,a){let o=K.ErrorCategory.NETWORK;this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendRequest`,category:o,message:i instanceof Error?i.message:`Request failed after retries`,thrown:i,data:{method:e,params:t,sessionKey:n,time:K.now()-r,maxAttempts:this.retryConfig.maxAttempts,attempts:a}})}handleJsonFinalError(e,t,n,r){this.params.errors$.next({id:`ThinOneStat:ApiTransport:sendJsonRequest`,category:K.ErrorCategory.NETWORK,message:n instanceof Error?n.message:`JSON request failed after retries`,thrown:n,data:{path:e,time:K.now()-t,maxAttempts:this.retryConfig.maxAttempts,attempts:r}})}sleep(e){return new Promise(t=>K.getWindow().setTimeout(t,e))}resolveApiBaseUrl(){return this.apiEnv===`auto`?(this.resolveApiBaseUrlPromise||=this.fetchApiBaseUrl().then(e=>(e===H.vk_alias&&(this.resolveApiBaseUrlPromise=null),e)),this.resolveApiBaseUrlPromise):Promise.resolve(this.apiBaseUrl)}async fetchApiBaseUrl(){try{let e=atob(`aHR0cHM6Ly9kbnMuZ29vZ2xlL3Jlc29sdmU/bmFtZT12aWRlby5fZW5kcG9pbnQub2sucnUmdHlwZT1UWFQ=`),t=(await(await this.fetchWithTimeout(e,{method:`GET`,mode:`cors`,cache:`no-cache`},5e3)).json())?.Answer[0]?.data;if(!t)throw Error(`Wrong DNS response`);return this.resolvedApiBaseUrl=t,t}catch(e){return this.params.errors$.next({id:`ThinOneStat:ApiTransport:resolveApiBaseUrl`,category:K.ErrorCategory.NETWORK,message:`Unhandled resolve api base url error`,thrown:e}),H.vk_alias}}prepareQueryParams(e){let t=new URLSearchParams({format:`JSON`,method:e.method,application_key:this.apiKey});return e.sessionKey!==void 0&&t.append(`session_key`,e.sessionKey),Object.entries(e.queryParams).forEach(([e,n])=>t.append(e,typeof n==`string`?n:JSON.stringify(n))),t}parseRetryAfter(e){let t=e.headers.get(`Retry-After`);if(!t)return;let n=Number(t);if(!isNaN(n)&&n>=0)return n*1e3;let r=new Date(t).getTime();if(!isNaN(r))return Math.max(0,r-Date.now())}getRetryDelay(e,t){let n=e;return n?.retryAfter!==void 0&&n.retryAfter>0?n.retryAfter:K.getExponentialDelay(t,{start:this.retryConfig.baseDelay,max:this.retryConfig.maxDelay,factor:2,random:.2})}}var ze=require(`@vkontakte/videoplayer-shared`),q=require(`@vkontakte/videoplayer-shared`),Be=e=>t=>(t.client_time??q.now())+e>=q.now();class Ve{params;subscription;userSalt;unsaltedStorage=[];lastVsid;isPaused=!1;firstFlush;debouncedFlush;flushRetryCount=0;retryFlushTimerId;debugLog;flushPromise=null;flushSnapshotLength=0;onlineListener;urgentEvents=[`watch_coverage`,`watch_coverage_live`,`watched_n`,`playback_started`];STORAGE_MAX_ATTEMPTS=3;constructor(e){this.params=e,this.userSalt=e.userSalt,this.debugLog=e.debugLogger.createComponentLog(`batch-queue:${e.storageKey}`),this.firstFlush=q.debounceFn(()=>this.safeFlush(),this.params.config.flushFirstTime,{maxWait:this.params.config.flushFirstTime});let t=q.safeStorage.isPersistent()?this.params.config.flushMaxWait:0;this.debouncedFlush=q.debounceFn(()=>this.safeFlush(),this.params.config.flushDebounceTime,{maxWait:t}),this.subscription=this.params.canSend$.subscribe(e=>{e&&(this.debugLog({message:`Can send, triggering flush`}),this.debouncedFlush())}),this.onlineListener=()=>{this.debugLog({message:`Network online, triggering flush`}),this.cancelRetry(),this.safeFlush()},q.getWindow().addEventListener(`online`,this.onlineListener),this.housekeepStorage()}isUrgent=e=>this.urgentEvents.includes(e.event_name);scheduleRetry(){this.retryFlushTimerId!==void 0&&q.getWindow().clearTimeout(this.retryFlushTimerId);let e=this.params.config.flushRetryFirstDelay,t=this.params.config.flushRetryMaxDelay,n=Math.min(e*2**this.flushRetryCount,t);this.flushRetryCount++,this.debugLog({message:`Scheduling flush retry #${this.flushRetryCount} in ${n}ms`}),this.retryFlushTimerId=q.getWindow().setTimeout(()=>{this.retryFlushTimerId=void 0,this.safeFlush()},n)}cancelRetry(){this.retryFlushTimerId!==void 0&&(q.getWindow().clearTimeout(this.retryFlushTimerId),this.retryFlushTimerId=void 0),this.flushRetryCount=0}safeFlush(){if(this.flushPromise){this.debugLog({message:`Flush already in progress, skipping`});return}this.flushPromise=this.flush().finally(()=>{this.flushPromise=null})}isStorageAvailable(){let e=q.safeStorage.isPersistent?.()??!1;return e||this.debugLog({message:`Storage not available`}),e}readFromStorage(){if(!this.isStorageAvailable())return{};try{let e=q.safeStorage.get(this.params.storageKey),t=e?JSON.parse(e):{};return this.debugLog({message:`Read ${Object.keys(t).length} users from storage`}),t}catch(e){return this.params.errors$.next({id:`ThinOneStat:BatchQueue:ReadStorageError`,category:q.ErrorCategory.WTF,message:`Failed to read from storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to read from storage`}),{}}}writeToStorage(e){if(!this.isStorageAvailable())return!1;try{return q.safeStorage.set(this.params.storageKey,JSON.stringify(e)),this.debugLog({message:`Written ${Object.keys(e).length} users to storage`}),!0}catch(e){return this.params.errors$.next({id:`ThinOneStat:BatchQueue:WriteStorageError`,category:q.ErrorCategory.WTF,message:`Failed to write to storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to write to storage`}),!1}}addToStorage(e){if(!this.userSalt){this.unsaltedStorage.push(e),this.debugLog({message:`Added item to unsalted storage, total: ${this.unsaltedStorage.length}`});return}for(let t=0;t<this.STORAGE_MAX_ATTEMPTS;t++)try{let t=this.readFromStorage(),n=t[this.userSalt]??[];if(this.debugLog({message:`Adding item to storage for user ${this.userSalt}`}),this.writeToStorage({...t,[this.userSalt]:[...n,e]}))return}catch(n){t===this.STORAGE_MAX_ATTEMPTS-1&&(this.params.errors$.next({id:`ThinOneStat:BatchQueue:AddToStorageFailed`,category:q.ErrorCategory.WTF,message:`Failed to add item to storage after retries`,thrown:n,data:{item:e,attempt:t,storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to add item to storage after retries`}))}}getFromStorage(){if(!this.userSalt)return this.debugLog({message:`Retrieved ${this.unsaltedStorage.length} items from unsalted storage`}),this.flushSnapshotLength=this.unsaltedStorage.length,this.unsaltedStorage.slice();let e=this.readFromStorage(),t=e[this.userSalt]??[],n=t.filter(Be(this.params.config.storageExpiration));if(this.debugLog({message:`Retrieved from storage for user ${this.userSalt}`}),n.length!==t.length){let r=t.length-n.length;this.params.errors$.next({id:`ThinOneStat:BatchQueue:DropExpiredEvents`,category:q.ErrorCategory.WTF,message:`Dropped ${r} expired events from storage`,data:{droppedCount:r,expiration:this.params.config.storageExpiration,storageKey:this.params.storageKey}}),this.debugLog({message:`Cleaning ${r} expired items from storage`});let i={...e};n.length?i[this.userSalt]=n:delete i[this.userSalt],this.writeToStorage(i)}return this.flushSnapshotLength=n.length,n}markStorageSent(){if(!this.userSalt){let e=this.unsaltedStorage.splice(0,this.flushSnapshotLength);this.debugLog({message:`Cleared unsalted storage, removed ${e.length} items, ${this.unsaltedStorage.length} new items kept`}),this.flushSnapshotLength=0;return}let e=this.flushSnapshotLength;this.flushSnapshotLength=0;for(let t=0;t<this.STORAGE_MAX_ATTEMPTS;t++)try{let t=this.readFromStorage(),n=(t[this.userSalt]??[]).slice(e),r={...t};if(n.length?r[this.userSalt]=n:delete r[this.userSalt],this.writeToStorage(r)){this.debugLog({message:`Marked storage as sent for user ${this.userSalt}, removed ${e} items, ${n.length} kept`});return}}catch(e){t===this.STORAGE_MAX_ATTEMPTS-1&&(this.params.errors$.next({id:`ThinOneStat:BatchQueue:MarkStorageSentFailed`,category:q.ErrorCategory.WTF,message:`Failed to mark storage as sent after retries`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to mark storage as sent after retries`}))}}housekeepStorage(){if(this.isStorageAvailable())try{let e=this.readFromStorage(),t=!1,n=0;for(let[r,i]of Object.entries(e)){let a=i.filter(Be(this.params.config.storageExpiration)),o=i.length-a.length;n+=o,a.length?a.length!==i.length&&(e[r]=a,t=!0):(delete e[r],t=!0),this.debugLog({message:`${i.length} retrieved from storage, ${a.length} of them actual for ${r}`})}t?(this.debugLog({message:`Housekeeping completed, removed ${n} expired items total`}),this.writeToStorage(e),n>0&&this.params.errors$.next({id:`ThinOneStat:BatchQueue:DropExpiredEvents`,category:q.ErrorCategory.WTF,message:`Dropped ${n} expired events during housekeeping`,data:{droppedCount:n,expiration:this.params.config.storageExpiration,storageKey:this.params.storageKey}})):this.debugLog({message:`Housekeeping completed, no changes needed`})}catch(e){this.params.errors$.next({id:`ThinOneStat:BatchQueue:HousekeepStorageError`,category:q.ErrorCategory.WTF,message:`Failed to housekeep storage`,thrown:e,data:{storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to housekeep storage`})}}log(e){try{if(this.debugLog({message:`Logging item: ${e.event_name}`}),this.addToStorage(e),this.isPaused){this.debugLog({message:`Queue paused, skipping flush`});return}this.isUrgent(e)?(this.debugLog({message:`Urgent item, flushing immediately`}),this.safeFlush()):this.lastVsid===e.vsid?(this.debugLog({message:`Same VSID, scheduling debounced flush`}),this.debouncedFlush()):(this.debugLog({message:`VSID changed, triggering first flush`}),this.firstFlush()),this.lastVsid=e.vsid}catch(t){this.params.errors$.next({id:`ThinOneStat:BatchQueue:LogError`,category:q.ErrorCategory.WTF,message:`Failed to log item`,thrown:t,data:{item:e,storageKey:this.params.storageKey}}),this.debugLog({message:`Failed to log item`})}}flush(e={wantBeacon:!1,clearStorage:!0}){return this.flushPromise?(this.debugLog({message:`Flush already in progress, waiting`}),this.flushPromise):(this.flushPromise=this.flushInternal(e).finally(()=>{this.flushPromise=null}),this.flushPromise)}async flushInternal({wantBeacon:e,clearStorage:t}){this.debugLog({message:`Starting flush`});let n=this.getFromStorage();if(n.length===0){this.debugLog({message:`No items to flush`});return}let r=this.params.canSend$.getValue();if(this.debugLog({message:`Can send status: ${r}`}),!r){this.debugLog({message:`Cannot send, requesting ready state`}),this.params.requestReady();return}if(e&&this.params.config.useBeacon){this.debugLog({message:`Flushing ${n.length} events through beacon`}),this.params.sendBeacon(n),t&&this.markStorageSent(),this.firstFlush.cancel(),this.debouncedFlush.cancel();return}this.debugLog({message:`Flushing ${n.length} events`});try{await this.params.send(n),this.debugLog({message:`Flush completed successfully`}),this.cancelRetry(),t&&this.markStorageSent(),this.firstFlush.cancel(),this.debouncedFlush.cancel()}catch(e){throw this.params.errors$.next({id:`ThinOneStat:BatchQueue:FlushRequestError`,category:q.ErrorCategory.NETWORK,message:`Failed to send events, will retry later`,thrown:e,data:{itemsCount:n.length,storageKey:this.params.storageKey}}),this.debugLog({message:`Flush failed, scheduling retry`}),this.scheduleRetry(),e}}pause(){this.debugLog({message:`Queue paused`}),this.isPaused=!0,this.debouncedFlush.cancel(),this.firstFlush.cancel(),this.cancelRetry()}resume(){this.debugLog({message:`Queue resumed`}),this.isPaused=!1,this.debouncedFlush()}async destroy(){this.debugLog({message:`Destroying queue`}),this.flushPromise&&(this.debugLog({message:`Waiting for ongoing flush to complete`}),await this.flushPromise),this.cancelRetry(),this.onlineListener&&=(q.getWindow().removeEventListener(`online`,this.onlineListener),void 0),this.subscription.unsubscribe(),this.firstFlush.cancel(),this.debouncedFlush.cancel(),this.flushPromise=null,this.debugLog({message:`Queue destroyed`})}}class He{params;queues=[];constructor(e){this.params=e;for(let t of e.apis)this.queues.push(this.createQueueForApi(t))}createQueueForApi(e){let t=this.getStorageKeyForApi(e.id);return new Ve({config:this.params.config,storageKey:t,debugLogger:this.params.debugLogger,errors$:this.params.errors$,userSalt:this.params.userSalt,send:t=>e.logRequest(t),sendBeacon:t=>e.logBeacon(t),canSend$:e.authorized$,requestReady:()=>{e.authorize().catch(e=>{this.params.errors$.next({id:`ThinOneStat:Logger:RequestReadyFailed`,category:ze.ErrorCategory.NETWORK,message:`Failed to authorize on requestReady`,thrown:e,data:{storageKey:t}})})}})}getStorageKeyForApi(e){return`thinonestat_events_${e}`}log(e){this.queues.forEach(t=>t.log(e))}async flush(e){let t=await Promise.allSettled(this.queues.map(t=>t.flush(e)));if(t.length>0&&t.every(e=>e.status===`rejected`))throw t.find(e=>e.status===`rejected`).reason}pause(){this.queues.forEach(e=>e.pause())}resume(){this.queues.forEach(e=>e.resume())}async destroy(){await Promise.allSettled(this.queues.map(e=>e.destroy())),this.queues=[]}}function Ue(){let e=`0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz`.split(``),t=Array(36),n=0,r,i;for(i=0;i<36;i++)i===8||i===13||i===18||i===23?t[i]=`-`:i===14?t[i]=`4`:(n<=2&&(n=33554432+Math.random()*16777216|0),r=n&15,n>>=4,t[i]=e[i===19?r&3|8:r]);return t.join(``)}class We{offset=void 0;constructor(e){this.offset=e}getOffset(){return this.offset??0}setOffset(e){this.offset=e}now(){return Date.now()+(this.offset??0)}date(e=new Date){return e.setTime(e.getTime()+(this.offset??0)),e}addServerTime(e,t){let n=e-Date.now()-t/2;if(Math.abs(n)<1e3){this.offset=0;return}if(this.offset===void 0)this.offset=Math.round(n);else{let e=.2;this.offset=Math.round(e*n+(1-e)*this.offset)}}}var Ge=require(`@vkontakte/videoplayer-core`),J=require(`@vkontakte/videoplayer-shared`),Ke=(e,t,n)=>new J.Observable(r=>{let i=new J.Subscription,a,o=new J.ValueSubject(void 0),s;i.add(e.info.isLive$.pipe(J.filterChanged()).subscribe(t=>{s&&(s.unsubscribe(),o.next(void 0)),a=void 0,s=t?e.info.liveTime$.pipe(J.map(e=>e&&e/1e3)).subscribe(o):e.info.position$.subscribe(o),i.add(s)}));let{playing$:c,paused$:l,ended$:u,looped$:d}=e.events,f=e.events.willSeek$.pipe(J.filter(()=>e.info.playbackState$.getValue()===Ge.PlaybackState.PLAYING)),p=e.events.seeked$.pipe(J.filter(()=>e.info.playbackState$.getValue()===Ge.PlaybackState.PLAYING)),m=e.info.isStalled$.pipe(J.filter(e=>e)),h=!1,g=new J.Subject;i.add(f.subscribe(()=>{h||g.next(),h=!0})).add(p.subscribe(()=>h=!1));let _=J.merge(J.fromEvent(J.getWindow(),`beforeunload`),e.events.willDestruct$),v=e=>{a=e},y=e=>{let t=a;if(a=void 0,!(J.isNullable(t)||J.isNullable(e)||!isFinite(t)||!isFinite(e))){if(Math.round(e*1e3)<=Math.round(t*1e3)){a=e;return}r.next({from:t,to:e})}},b=new J.Subject,x=new J.Subject,S=()=>n.exactTime?e.info.isLive$.getValue()?e.getExactLiveTime():e.getExactTime():o.getValue()??0,C=J.merge(c,p,b).pipe(J.map(()=>S())),w=J.merge(d,J.merge(l,g,_,u,x,m).pipe(J.map(()=>S()))),T=t.started$&&t.ended$?J.merge(C,t.ended$):C,E=t.started$&&t.ended$?J.merge(w,t.started$):w;if(i.add(T.subscribe(v)).add(E.subscribe(y)),n.heartbeatInterval&&isFinite(n.heartbeatInterval)){let e=[],t=()=>{x.next(),b.next()};i.add(T.subscribe(()=>{let r=J.getWindow().setTimeout(t,n.heartbeatInterval);e.push(r)})).add(E.subscribe(()=>{e.forEach(e=>{J.getWindow().clearTimeout(e)}),e=[]}))}return i}),qe=require(`@vkontakte/videoplayer-core`),Je=require(`@vkontakte/videoplayer-shared`),Ye=(e,t)=>new Je.Observable(n=>{let r=new qe.Subscription;return r.add(Je.merge(e.info.position$,e.info.liveTime$).subscribe(()=>{let e=t.getTotalViewTime();e>=t.targetDuration&&(n.next({target_duration:t.targetDuration,current_tvt:e}),r.unsubscribe())})),r}),Xe=require(`@vkontakte/videoplayer-core`),Y=require(`@vkontakte/videoplayer-shared`),Ze=(e,{maxTimeWindow:t})=>new Y.Observable(n=>{let r=new Xe.Subscription,i={maxBytesThreshold:100*1024*1024,minBytesThreshold:100*1024,maxTimeWindow:t,minTimeWindow:2e3},a={bytes:0,startTime:0,lastEmitTime:0,timeoutId:0},o=()=>{a.timeoutId&&=(Y.getWindow().clearTimeout(a.timeoutId),null)},s=()=>{a.bytes=0,a.startTime=0,a.lastEmitTime=0,o()},c=e=>{if(a.bytes<i.minBytesThreshold||e-a.lastEmitTime<i.minTimeWindow&&a.lastEmitTime>0)return!1;let t=(a.startTime>0?e-a.startTime:0)>=i.maxTimeWindow,n=a.bytes>=i.maxBytesThreshold;return t||n},l=(e=!1)=>{let t=Date.now();if(!e&&!c(t))return;o();let r=a.startTime>0?t-a.startTime:Math.max(i.minTimeWindow,1),l=a.bytes/(r/1e3);n.next({download_bytes:Math.round(a.bytes),download_speed:Math.round(l)}),a.lastEmitTime=t,s()},u=()=>{if(o(),a.startTime>0&&a.bytes>=i.minBytesThreshold){let e=Date.now()-a.startTime,t=i.maxTimeWindow-e;t>100?a.timeoutId=Y.getWindow().setTimeout(()=>{a.timeoutId=null,l()},t):t>0&&l()}};return r.add(e.info.httpDownloadMetrics$.pipe(Y.filter(e=>e!==void 0)).subscribe(e=>{let t=Date.now();a.startTime===0&&(a.startTime=t),a.bytes+=e.bytes,c(t)?l():u()})),r.add(()=>{if(o(),a.bytes>0){let e=Date.now(),t=a.startTime>0?e-a.startTime:i.minTimeWindow,r=a.bytes/(t/1e3);n.next({download_bytes:Math.round(a.bytes),download_speed:Math.round(r)})}s()}),r}),X=require(`@vkontakte/videoplayer-shared`),Qe=e=>new X.Observable(t=>{let n=new X.Subscription,r=0,i=1,a=0,o=!1,s=!1,c=()=>{if(!o)return;let e=Date.now();a+=(e-r)*i,r=e},l=()=>{(!o||s)&&(r=Date.now(),o=!0,s=!1)},u=()=>{o&&=(c(),!1)},d=()=>{o&&(c(),o=!1,s=!0)},f=e=>{o&&c(),i=e,r=Date.now()},{started$:p,playing$:m,paused$:h,willSeek$:g,ended$:_,willDestruct$:v}=e.events,{currentPlaybackRate$:y}=e.info;return n.add(g.subscribe(d)).add(X.merge(p,m).subscribe(l)).add(X.merge(h,_,v).subscribe(u)).add(y.subscribe(f)),t.next(()=>{let e=a;return o&&(e+=(Date.now()-r)*i),Math.round(e)}),n}),$e=require(`@vkontakte/videoplayer-shared`),et=(e,t)=>new $e.Observable(n=>{let r=new $e.Subscription,i=0;return r.add($e.merge(e.info.position$,e.info.liveTime$).subscribe(()=>{let e=t.getTotalViewTime();e>i&&Math.floor(e/t.interval)>Math.floor(i/t.interval)&&(i=e,n.next())})),r}),tt=require(`@vkontakte/videoplayer-shared`),Z=require(`@vkontakte/videoplayer-core`),nt=e=>{if(!tt.isNullable(e)){if(typeof e==`string`)return e;try{return JSON.stringify(e)}catch{return String(e)}}},rt=e=>e&&{[Z.HttpConnectionType.HTTP1]:`http1`,[Z.HttpConnectionType.HTTP2]:`http2`,[Z.HttpConnectionType.QUIC]:`http3`}[e],Q=e=>{if(e!==void 0)switch(e){case Z.VideoFormat.MPEG:return`MP4`;case Z.VideoFormat.DASH:case Z.VideoFormat.DASH_LIVE:return`DASH`;case Z.VideoFormat.DASH_SEP:return`DASH_SEP`;case Z.VideoFormat.DASH_ONDEMAND:return`ONDEMAND_DASH`;case Z.VideoFormat.HLS_ONDEMAND:return`ONDEMAND_HLS`;case Z.VideoFormat.HLS:case Z.VideoFormat.HLS_LIVE:return`HLS`;case Z.VideoFormat.HLS_FMP4:return`HLS_FMP4`;case Z.VideoFormat.DASH_WEBM:case Z.VideoFormat.DASH_LIVE_WEBM:return`WEBM`;case Z.VideoFormat.DASH_LIVE_CMAF:return`ONDEMAND_DASH_LIVE`;case Z.VideoFormat.HLS_LIVE_CMAF:return`ONDEMAND_HLS_LIVE`;case Z.VideoFormat.WEB_RTC_LIVE:return`WEBRTC`;case Z.VideoFormat.DASH_WEBM_AV1:return`AV1`;case Z.VideoFormat.DASH_STREAMS:return`MULTI_DASH`;default:return tt.assertNever(e)}},$=e=>{if(e!==void 0)switch(e){case Z.VideoQuality.Q_144P:return`144p`;case Z.VideoQuality.Q_240P:return`240p`;case Z.VideoQuality.Q_360P:return`360p`;case Z.VideoQuality.Q_480P:return`480p`;case Z.VideoQuality.Q_720P:return`720p`;case Z.VideoQuality.Q_1080P:return`1080p`;case Z.VideoQuality.Q_1440P:return`1440p`;case Z.VideoQuality.Q_2160P:return`2160p`;case Z.VideoQuality.INVARIANT:case Z.VideoQuality.Q_576P:case Z.VideoQuality.Q_4320P:return`UNKNOWN`;default:return tt.assertNever(e)}},it=e=>{if(e!==void 0)switch(e){case`MP4`:case`ONDEMAND_DASH`:case`ONDEMAND_DASH_LIVE`:case`ONDEMAND_HLS`:case`ONDEMAND_HLS_LIVE`:case`MULTI_DASH`:case`WEBM`:case`AV1`:return`vod`;case`DASH`:case`HLS`:case`HLS_FMP4`:case`WEBRTC`:return`live`;default:return}},at=(e,t,n)=>{if(e&&t)switch(e){case Z.VideoFormat.MPEG:return n&&t[e]&&t[e][n];default:return t[e]?.url}},ot=(e,t=!1)=>{if(t)return`minimal`;switch(e){case void 0:case Z.Surface.NONE:return;case Z.Surface.INLINE:return`inline`;case Z.Surface.FULLSCREEN:return`fullscreen`;case Z.Surface.SECOND_SCREEN:return`chromecast`;case Z.Surface.PIP:return`pip_external`}},st=e=>{switch(e){case void 0:return;case Z.Surface.INVISIBLE:return`background`;default:return`foreground`}},ct=`_thin-one-stat_deviceId`,lt=`2.2.0`,ut=()=>{let e=new B.Subscription;return{subscription:e,subscribe:(t,n)=>{t&&e.add(t.subscribe(n))}}};class dt{apis=[];logger;config;subscription;lifecycleFlushSubscription;timeSynchronisation;statContext;deviceId;targetDuration;debugLogger=new B.Logger;thinOneStatDebugLog=this.debugLogger.createComponentLog(`ThinOneStat`);onceEvents=new Set(Te);sentOnceEvents=new Set;eventNumber=1;isStarted;isSeeking;inBufferStarvation;wasPaused;isLive;lastContentType;cdnHostname;isEmbed;embedHostname;connectionType;connectionReused;player;playerSize={width:0,height:0};currentSubtitle;userQuality;downloadQuality;downloadSpeed;networkType;uiEvents;playerSubscription;uiSubscription;adsSubscription;seekAction$=new B.ValueSubject(`unknown`);vsid$=new B.ValueSubject(void 0);isid$=new B.ValueSubject(void 0);errors$=new B.Subject;getTotalViewTime=()=>0;constructor(e,t){this.statContext=e,this.config=Ae(t.config??{}),this.subscription=new B.Subscription,this.config.synchronizeTime&&(this.timeSynchronisation=new We),this.subscription.add(this.errors$.subscribe(e=>{this.thinOneStatDebugLog({message:`[${e.category}] ${e.id}: ${e.message}`})}));let n=B.safeStorage.get(ct);this.statContext.deviceId?this.deviceId=this.statContext.deviceId:n?this.deviceId=n:(this.deviceId=Ue(),B.safeStorage.set(ct,this.deviceId)),this.statContext.targetDuration?this.targetDuration=this.statContext.targetDuration:this.targetDuration=this.config.watchedNDefaultTargetDuration,this.resetViewSession(),t.useIsid&&this.isid$.next(t.isid??ne());let r=new Re({errors$:this.errors$,config:this.config,timeSynchronisation:this.timeSynchronisation});this.config.apiForticom&&this.apis.push(new Pe({config:this.config,apiTransport:r,errors$:this.errors$,deviceId:this.deviceId,refreshAuthToken:t.refreshAuthToken})),this.config.apiUvRest&&(t.getUvRestAuthToken?this.apis.push(new Fe({config:this.config,apiTransport:r,errors$:this.errors$,userId:this.statContext.userId,getAuthToken:t.getUvRestAuthToken})):this.errors$.next({id:`ThinOneStat:Init:MissingGetUvRestAuthToken`,category:B.ErrorCategory.WTF,message:`thinStatToUvRest enabled but getUvRestAuthToken callback not provided`})),this.logger=new He({config:this.config,debugLogger:this.debugLogger,apis:this.apis,errors$:this.errors$,userSalt:t.userSalt});let{isEmbed:i,topOrigin:a}=B.detectEmbed();this.isEmbed=i,this.embedHostname=a?new URL(a).hostname:void 0}authorize(e){this.apis.forEach(t=>{t.authorize(e).catch(()=>{})})}reportError(e){let t={error_severity:e.severity,error_category:this.mapErrorCategory(e.category),player_error_code:e.code,player_error_trace:e.trace,error_message:e.message,content_type:this.getCurrentContentType()};this.logError(t),e.severity===`critical`&&this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}updateContext(e){this.statContext={...this.statContext,...e}}attachTo(e,t){this.playerSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to player, unsubscribing from previous`}),this.playerSubscription.unsubscribe());let n=new B.Subscription,r=(e,t)=>n.add(e.subscribe(t));r(Qe(e),e=>{this.getTotalViewTime=e}),r(e.info.isLive$,e=>this.isLive=e),r(e.info.currentFormat$,e=>{let t=Q(e);t&&(this.lastContentType=t)});let i;r(e.info.hostname$.pipe(B.filter(e=>e!==void 0),B.filterChanged()),e=>{if(i!==void 0&&i!==e){let t={cdn_host:e,connection_type:this.connectionType,connection_reused:this.connectionReused,content_type:this.getCurrentContentType()};this.logFailover(t),this.logTelemetry(this.collectTelemetryData())}this.cdnHostname=i=e}),r(e.info.httpConnectionType$,e=>{this.connectionType=rt(e)}),r(e.info.httpConnectionReused$,e=>{this.connectionReused=e}),r(Ke(e,t,{heartbeatInterval:this.config.watchCoverageHeartbeatInterval,exactTime:this.config.watchCoverageExactTime}),t=>{let n={watch_interval:`${Math.round(t.from*1e3)}-${Math.round(t.to*1e3)}`,in_history:!!this.statContext.inHistory,content_type:this.getCurrentContentType(),playback_quality:$(e.info.currentQuality$.getValue()),recom_info:this.statContext.recomInfo};if(!e.info.isLive$.getValue())this.logWatchCoverage(n);else{let t=e.info.atLiveEdge$.getValue();this.logWatchCoverageLive({...n,live:!!t})}}),r(et(e,{interval:this.statContext.telemetryInterval??this.config.telemetryInterval,getTotalViewTime:this.getTotalViewTime}),()=>{this.logTelemetry(this.collectTelemetryData())}),r(Ye(e,{targetDuration:this.targetDuration,getTotalViewTime:this.getTotalViewTime}),e=>{this.logWatchedN(e)}),r(e.events.willReady$,()=>{let t={isid:this.isid$.getValue(),stream_profile:it(this.getCurrentContentType()),dpi:Math.round(this.statContext.dpi??96*V.clientChecker.display.pixelRatio),web_layout:this.statContext.isMobile||V.clientChecker.device.isMobile?`mobile`:`desktop`,preloaded:this.statContext.preload,navigation:this.statContext.navigation,recom_info:this.statContext.recomInfo};if(this.isEmbed&&(t.iframe_host=this.embedHostname??`unknown`),this.logStartSession(t),this.statContext.preload){let t=e.info.currentFormat$.getValue(),n=e.info.availableSources$.getValue(),r=e.info.currentBuffer$.getValue(),i=e.info.currentQuality$.getValue(),a={target_buffer_time:this.calcBufferTime(r),playback_url:at(t,n,i),playback_quality:$(i)};this.logPreloadStarted(a)}});let a=()=>({playback_url:at(e.info.currentFormat$.getValue(),e.info.availableSources$.getValue(),e.info.currentQuality$.getValue()),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused});r(B.combine({manifestRequested$:e.events.manifestRequested$,connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$}).pipe(B.once()),()=>{this.logManifestRequest(a())}),r(B.combine({firstBytesManifest:e.events.firstBytesManifest$,connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$}).pipe(B.once()),()=>{this.logFirstByteManifest(a())}),r(e.events.manifestReceived$,()=>{this.logManifestReceived(a())}),r(e.events.firstBytesRequested$,()=>{this.logFirstMediaRequest(a())}),r(e.events.firstBytes$,()=>{this.logFirstByteMedia(a())}),r(B.merge(e.events.willStart$,e.events.looped$),e=>{e&&this.resetViewSession(),this.isStarted=!0}),r(e.events.started$,()=>{let t=e.info.currentBuffer$.getValue(),n=e.info.currentQuality$.getValue(),r=e.info.isAutoQualityEnabled$.getValue(),i=e.info.muted$.getValue(),a=e.info.volume$.getValue(),o=e.info.surface$.getValue(),s=e.info.availableTextTracks$.getValue(),c=e.info.currentTextTrack$.getValue(),l=e.info.currentAudioStream$.getValue(),u=e.info.currentPlaybackRate$.getValue(),d={playback_quality:$(n),user_quality:r?`auto`:$(n),player_width:Math.round(this.statContext.playerSize?.width??this.playerSize.width),player_height:Math.round(this.statContext.playerSize?.height??this.playerSize.height),muted:i,sound_volume:Math.round(a*100),buffer_time:this.calcBufferTime(t),mode:ot(o,this.statContext.isPreviewPlayerView),visibility:st(o),subtitles_enabled:c?!!c:this.currentSubtitle?.enabled,auto_subtitles:s.find(({id:e})=>e===c)?.isAuto??this.currentSubtitle?.auto,audio_track_lang:l?.language,playback_rate:u*100,recom_info:this.statContext.recomInfo};this.logPlaybackStarted(d),this.logTelemetry(this.collectTelemetryData())}),r(e.events.willStart$,()=>{this.logPlay()}),r(e.events.playing$,()=>{let t=e.info.currentBuffer$.getValue(),n={buffer_time:this.calcBufferTime(t)};this.logPlaying(n),this.inBufferStarvation=!1,this.wasPaused&&this.logResume()}),r(e.events.paused$,()=>{this.wasPaused=!0,this.logPause()}),r(e.events.ended$,()=>{this.logContentEndReached()}),r(e.events.willStop$,()=>{this.isStarted=!1,this.isSeeking=!1,this.inBufferStarvation=!1,this.logStop()}),r(B.merge(e.events.fetcherRecoverableError$,e.events.fatalError$),()=>{this.statContext.preload&&this.logPreloadError()}),r(e.events.fetcherRecoverableError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:this.mapErrorSeverity(t),error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:nt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.mapErrorSeverity(t)===`critical`&&this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}),r(e.events.managedError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:this.mapErrorSeverity(t),error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:nt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.logTelemetry(this.collectTelemetryData())}),r(e.events.fatalError$,({id:e,category:t,data:n,message:r,httpCode:i,UVBackendErrorCode:a})=>{let o={error_severity:`critical`,error_category:this.mapErrorCategory(t),player_error_code:e,player_error_trace:nt(n),http_error_code:i,uv_backend_error_code_subcode:a,error_message:r,content_type:this.getCurrentContentType()};this.logError(o),this.logCannotPlay(),this.logTelemetry(this.collectTelemetryData())}),r(B.combine({connectionType:e.info.httpConnectionType$,connectionReused:e.info.httpConnectionReused$,connectionMetrics:e.info.httpConnectionMetrics$}).pipe(B.filter(({connectionReused:e})=>e!==void 0),B.filter(({connectionMetrics:e})=>e!==void 0),B.filterChanged((e,t)=>e.connectionType===t.connectionType)),({connectionMetrics:t})=>{let n=e.info.currentFormat$.getValue(),r=e.info.availableSources$.getValue(),i=e.info.currentQuality$.getValue(),a=e.info.rttEstimation$.getValue();this.networkType=this.statContext.networkType??t?.networkType;let o=t?.dnsResolveTime,s=t?.tcpHandshakeTime,c=t?.tlsHandshakeTime,l={playback_url:at(n,r,i),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused,rtt:a?Math.round(a):void 0,network_type:this.networkType,dns_resolve_time:o?Math.round(o):void 0,tcp_handshake_time:s?Math.round(s):void 0,tls_handshake_time:c?Math.round(c):void 0};this.logConnectionEstablished(l)}),r(Ze(e,{maxTimeWindow:this.config.downloadBytesMaxTime}),e=>{this.downloadSpeed=e.download_speed,this.logDownloadBytes(e)}),r(e.events.firstFrame$,()=>{this.logFirstVideoFrameDecoded()}),r(e.events.willSeek$,()=>{this.isSeeking=!0;let e={seek_type:this.seekAction$.getValue()};this.isStarted&&this.logSeeking(e)}),r(e.events.seeked$,()=>{this.isSeeking=!1;let e={seek_type:this.seekAction$.getValue()};this.isStarted&&this.logSeeked(e),this.seekAction$.next(`unknown`)}),r(e.info.stallStartTime$,t=>{if(t<=0){this.inBufferStarvation=!1;return}this.inBufferStarvation||(this.inBufferStarvation=!0,this.logBufferStarvation({buffer_time:this.calcBufferTime(e.info.currentBuffer$.getValue())}),this.logTelemetry(this.collectTelemetryData()))}),r(e.info.currentBuffer$.pipe(B.filter(e=>!!this.calcBufferTime(e)),B.once()),e=>{let t=this.calcBufferTime(e);if(this.statContext.preload){let e={buffer_time:t};this.logPreloadEnded(e)}this.logReady({buffer_time:t})});let o;r(B.combine({videoStream:e.info.currentVideoStream$,audioStream:e.info.currentAudioStream$,quality:e.info.currentQuality$}).pipe(B.filter(({videoStream:e,audioStream:t})=>e!==void 0&&t!==void 0),B.filter(({quality:e})=>e!==void 0)),({videoStream:t,audioStream:n,quality:r})=>{if(o!==r){if(this.userQuality=e.info.isAutoQualityEnabled$.getValue()?`auto`:$(r),this.downloadQuality=$(r),this.isStarted){let e=t?.codec,r=n?.codec,i={user_quality:this.userQuality,playback_quality:$(o),download_quality:this.downloadQuality,codec_info:this.genCodecInfo(e,r)};this.logQualityChanged(i)}o=r}});let s,c;return r(e.info.surface$,e=>{let t=ot(e,this.statContext.isPreviewPlayerView);t&&!this.uiEvents&&this.isStarted&&this.logModeChanged({mode:t}),c=st(e),s!==c&&(this.isStarted&&this.logVisibilityChanged({visibility:c}),s=st(e))}),r(B.combine({muted:e.info.muted$,volume:e.info.volume$}).pipe(B.debounce(300)),({muted:e,volume:t})=>{if(this.isStarted){let n={muted:e,sound_volume:Math.round(t*100)};this.logSoundChanged(n)}}),r(e.info.currentFormat$.pipe(B.filter(e=>e!==void 0),B.skip(1)),e=>{let t={content_type:Q(e),cdn_host:this.cdnHostname,connection_type:this.connectionType,connection_reused:this.connectionReused};this.logContentTypeChange(t)}),r(e.info.currentAudioStream$.pipe(B.filter(e=>e!==void 0),B.skip(1)),e=>{let t={audio_track_lang:e.language,codec_info:e.codec};this.logAudioTrackSwitched(t)}),r(e.info.currentVideoStream$.pipe(B.filter(e=>e!==void 0),B.skip(1)),e=>{let t={hdr:e.hdr,codec_info:e.codec};this.logVideoTrackSwitched(t)}),this.player=e,this.playerSubscription=n,this.resubscribeLifecycleFlush(),this.subscription.add(n),n}attachToUi(e){this.uiSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to UI, unsubscribing from previous`}),this.uiSubscription.unsubscribe()),this.uiEvents=e;let{subscribe:t,subscription:n}=ut();t(e.actionSeek$,this.seekAction$),t(e.playerSize$,({width:e,height:t})=>{e&&t&&(this.playerSize={width:e,height:t},this.isStarted&&this.logViewPortChanged({player_width:Math.round(e),player_height:Math.round(t)}))}),t(e.isLoaderVisible$,e=>{if(!e){this.inBufferStarvation=!1;return}this.isStarted&&(this.logShowLoader(),!(this.isSeeking||this.inBufferStarvation||!this.player)&&(this.inBufferStarvation=!0,this.logBufferStarvation({buffer_time:this.calcBufferTime(this.player.info.currentBuffer$.getValue())}),this.logTelemetry(this.collectTelemetryData())))}),t(e.actionSubtitlesSwitched$,e=>{if(this.currentSubtitle={...e},this.isStarted){let t={subtitles_enabled:e.enabled,subtitles_track_lang:e.lang,auto_subtitles:e.auto};this.logSubtitlesSwitched(t)}});let{surface$:r}=ye(this.uiEvents,t,this.player);return t(r,e=>{this.isStarted&&this.logModeChanged({mode:ot(e)})}),t(e.actionQuality$,e=>{let t=e=>Object.values(Ee).includes(e),n=this.player?.info.currentVideoStream$.getValue()?.codec,r=this.player?.info.currentAudioStream$.getValue()?.codec,i={user_quality:t(e)?`auto`:$(e),playback_quality:$(this.player?.info.currentQuality$.getValue()),download_quality:t(e)?$(this.player?.info.currentQuality$?.getValue()):$(e),codec_info:this.genCodecInfo(n,r)};this.logQualityChangeRequested(i)}),this.uiSubscription=n,this.resubscribeLifecycleFlush(),this.subscription.add(n),n}attachToAds(e){this.adsSubscription&&(this.thinOneStatDebugLog({message:`Re-attaching to ads, unsubscribing from previous`}),this.adsSubscription.unsubscribe());let{subscription:t,subscribe:n}=ut();return n(e.init$,e=>{let t={slot:e};this.logAdvConfiguration(t)}),n(e.slotRequested$,e=>{let t={adv_section:e};this.logAdvRequest(t)}),n(e.started$,e=>{let t={adv_section:e};this.logAdvBreakStarted(t)}),n(e.ended$,({section:e,ordIds:t})=>{let n={adv_section:e,erids:t.join(`,`)};this.logAdvBreakEnded(n)}),n(e.error$,e=>{let t={error_message:e};this.logAdvError(t)}),this.adsSubscription=t,this.resubscribeLifecycleFlush(),this.subscription.add(t),t}getDeviceId(){return this.deviceId}resetViewSession(){this.thinOneStatDebugLog({message:`VSID reset`}),k.generateVSID(this.config.useVsid64),this.vsid$.next(k.getVSID()),this.eventNumber=0,this.sentOnceEvents.clear()}async destroy(){this.thinOneStatDebugLog({message:`Destroying ThinOneStat`}),this.subscription.unsubscribe(),this.lifecycleFlushSubscription?.unsubscribe(),await this.logger.destroy(),this.apis.forEach(e=>e.destroy()),this.sentOnceEvents.clear(),this.thinOneStatDebugLog({message:`ThinOneStat destroyed`})}calcBufferTime(e){return e?Math.round(((e.end??0)-(e.start??0))*1e3):void 0}mapErrorCategory(e){switch(e){case B.ErrorCategory.NETWORK:return`network`;case B.ErrorCategory.VIDEO_PIPELINE:return`video_pipeline`;case B.ErrorCategory.EXTERNAL_API:return`external_api`;case B.ErrorCategory.PARSER:return`parser`;case B.ErrorCategory.DOM:return`dom`;default:return`wtf`}}mapErrorSeverity(e){return B.ErrorCategory.FATAL===e?`critical`:`informative`}getCurrentContentType(){let e=Q(this.player?.info.currentFormat$.getValue());if(e)return e;if(this.lastContentType)return this.lastContentType;let t=this.player?.info.availableSources$.getValue();if(!t)return;let n=Object.keys(t)[0];return Q(n)}genCodecInfo(e,t){return e&&t?`${e},${t}`:void 0}getLatencyInfo(){let e=this.player?.info.atLiveEdge$.getValue(),t={client_time:Date.now(),live:e};return JSON.stringify(t)}collectTelemetryData(){if(!this.player)return{};let e=this.player.info.currentBuffer$.getValue(),t=this.player.experimental.element$.getValue(),n=0,r=0;t&&(n=t.getVideoPlaybackQuality().droppedVideoFrames,r=t.getVideoPlaybackQuality().totalVideoFrames);let i=this.getCurrentContentType(),a=this.player.info.currentVideoStream$.getValue()?.codec,o=this.player.info.currentAudioStream$.getValue()?.codec,s=this.player.info.videoBitrate$.getValue(),c=this.player.info.rttEstimation$.getValue();return{buffer_time:this.calcBufferTime(e),dropped_frames_count:n,rendered_frames_count:r,content_type:i,user_quality:this.userQuality,playback_quality:$(this.player.info.currentQuality$.getValue()),download_quality:this.downloadQuality,codec_info:this.genCodecInfo(a,o),bandwidth_from_manifest:s?Math.round(s):void 0,cdn_host:this.cdnHostname,download_speed:this.downloadSpeed?Math.round(this.downloadSpeed):void 0,network_type:this.networkType,rtt:c?Math.round(c):void 0,latency:this.getLatencyInfo()}}isValidPlatform(e=``){return[`web:desktop`,`web:mobile`,`smart_tv`].includes(e)}createRequiredParams(e){let t=this.vsid$.getValue();B.assertNonNullable(t);let n=this.eventNumber++,r=((this.isLive?this.player?.info.liveTime$.getValue():this.player?.info.position$.getValue())??0)*1e3,i={vsid:t,uv_movie_id:this.statContext.movieId,event_name:e,client_time:Date.now(),application:this.statContext.application??`@vkontakte/videoplayer-statistics:${A}`,platform:this.isValidPlatform(this.statContext.platform)?this.statContext.platform:`web:${this.statContext.isMobile||V.clientChecker.device.isMobile?`mobile`:`desktop`}`,product:this.statContext.product,event_number:n,playback_position:Math.round(r),current_tvt:this.getTotalViewTime(),cdn_host:this.cdnHostname};for(let e of this.config.disabledParams)delete i[e];return i}createRequiredFatParams(){let e={stats_version:lt,browser:this.statContext.browser??V.clientChecker.browser.current,browser_version:this.statContext.browserVersion??String(V.clientChecker.browser.currentVersion),os:this.statContext.os??V.clientChecker.device.os.name,os_version:this.statContext.osVersion??V.clientChecker.device.os.version,device_type:this.statContext.deviceType??(this.statContext.isMobile||V.clientChecker.device.isMobile?`mobile`:`desktop`),device_manufacturer:this.statContext.deviceManufacturer??V.clientChecker.device.details.vendor,device_model:this.statContext.deviceModel??V.clientChecker.device.details.model,navigation:this.statContext.navigation};for(let t of this.config.disabledParams)delete e[t];return e}log(e,t,n=!1){if(this.config.disabledEvents.includes(e))return;if(this.onceEvents.has(e)){if(this.sentOnceEvents.has(e))return;this.sentOnceEvents.add(e)}let r=this.createRequiredParams(e),i={};n&&(i=this.createRequiredFatParams()),this.logger.log({...r,...i,...t})}logWatchCoverage(e){this.log(`watch_coverage`,e,!0)}logWatchCoverageLive(e){this.log(`watch_coverage_live`,e,!0)}logWatchedN(e){this.log(`watched_n`,e,!0)}logStartSession(e){this.log(`start_session`,e,!0)}logPreloadStarted(e){this.log(`preload_started`,e)}logPreloadEnded(e){this.log(`preload_ended`,e)}logPreloadError(){this.log(`preload_error`)}logPlaybackStarted(e){this.log(`playback_started`,e,!0)}logPlay(){this.log(`play`)}logManifestRequest(e){this.log(`manifest_request`,e)}logFirstByteManifest(e){this.log(`first_byte_manifest`,e)}logFirstMediaRequest(e){this.log(`first_media_request`,e)}logFirstByteMedia(e){this.log(`first_byte_media`,e)}logManifestReceived(e){this.log(`manifest_received`,e)}logConnectionEstablished(e){this.log(`connection_established`,e)}logFirstVideoFrameDecoded(){this.log(`first_video_frame_decoded`)}logReady(e){this.log(`ready`,e)}logPlaying(e){this.log(`playing`,e)}logPause(){this.log(`pause`)}logResume(){this.log(`resume`)}logSeeking(e){this.log(`seeking`,e)}logSeeked(e){this.log(`seeked`,e)}logContentEndReached(){this.log(`content_end_reached`)}logStop(){this.log(`stop`)}logBufferStarvation(e){this.log(`buffer_starvation`,e)}logShowLoader(){this.log(`show_loader`)}logQualityChangeRequested(e){this.log(`quality_change_requested`,e)}logQualityChanged(e){this.log(`quality_changed`,e)}logCdnHostChanged(e){this.log(`cdn_host_changed`,e)}logViewPortChanged(e){this.log(`view_port_changed`,e)}logModeChanged(e){this.log(`mode_changed`,e)}logVisibilityChanged(e){this.log(`visibility_changed`,e)}logAudioTrackSwitched(e){this.log(`audio_track_switched`,e)}logVideoTrackSwitched(e){this.log(`video_track_switched`,e)}logSubtitlesSwitched(e){this.log(`subtitles_switched`,e)}logSoundChanged(e){this.log(`sound_changed`,e)}logFailover(e){this.log(`failover`,e)}logContentTypeChange(e){this.log(`content_type_change`,e)}logError(e){this.log(`error`,e,!0)}logCannotPlay(){this.log(`cannot_play`,{},!0)}logDownloadBytes(e){this.log(`download_bytes`,e)}logTelemetry(e){this.log(`telemetry`,e)}logAdvConfiguration(e){this.log(`adv_configuration`,e)}logAdvRequest(e){this.log(`adv_request`,e)}logAdvBreakStarted(e){this.log(`adv_break_started`,e)}logAdvBreakEnded(e){this.log(`adv_break_ended`,e)}logAdvError(e){this.log(`adv_error`,e)}resubscribeLifecycleFlush(){this.lifecycleFlushSubscription?.unsubscribe();let e=B.getWindow(),t=()=>{this.logger.flush({wantBeacon:!0,clearStorage:this.config.clearStorageAtUnload}).catch(()=>{})},n=new B.Subscription;n.add(B.fromEvent(e,`beforeunload`).subscribe(t)),n.add(B.fromEvent(e,`pagehide`).subscribe(t)),n.add(B.fromEvent(e.document,`visibilitychange`).subscribe(e=>{e.target?.visibilityState===`hidden`&&t()})),this.lifecycleFlushSubscription=n}}var ft=require(`@vkontakte/videoplayer-shared`),pt=require(`@vkontakte/videoplayer-shared`),mt=require(`@vkontakte/videoplayer-core`),ht=require(`@vkontakte/videoplayer-shared`);class gt{startTime=null;stopTime=null;accumulatedTime=0;get time(){return Math.floor(this.timeMs/1e3)}get timeMs(){return this.startTime===null?0:(this.stopTime??this.now())-this.startTime-this.accumulatedTime}get isRunning(){return this.startTime!==null&&this.stopTime===null}start(){this.isRunning||(this.stopTime===null?this.startTime=this.now():(this.accumulatedTime+=this.now()-this.stopTime,this.stopTime=null))}stop(){this.isRunning&&(this.stopTime=this.now())}reset(){this.startTime=null,this.stopTime=null,this.accumulatedTime=0}now(){return Date.now()}}var _t=()=>window.Image?new Image:document.createElement(`img`);class vt{params;position=0;started=!1;isActiveLive;heartbeatPixels=[];heartbeatLastTimeSent={};heartbeatInterval;heartbeatFirstTimeoutId;idleCallbackIds=[];log;stopwatch=new gt;subscription=new mt.Subscription;constructor(e){this.params=e,this.log=new ht.Logger().createComponentLog(`MediascopePixel`)}attachTo(e){this.subscription.add(e.info.playbackState$.subscribe(e=>this.onPlaybackState(e))).add(e.info.position$.subscribe(e=>this.onPosition(e))).add(e.info.atLiveEdge$.subscribe(e=>this.isActiveLive=e)),this.subscription.add(e.events.willSeek$.subscribe(()=>this.onWillSeek())).add(e.events.seeked$.subscribe(()=>this.onSeeked())).add(e.events.ended$.subscribe(()=>this.onEnded()))}destroy(){this.stopwatch.reset(),this.subscription.unsubscribe(),window.clearTimeout(this.heartbeatFirstTimeoutId),this.idleCallbackIds.forEach(e=>ft.cancelIdleCallbackPolyfilled(e)),this.send(`stop`)}onPlaybackState(e){e===mt.PlaybackState.PLAYING?this.started?this.play():this.start():e===mt.PlaybackState.PAUSED&&this.pause()}start(){this.started=!0,this.stopwatch.start(),this.send(`start`),this.heartbeatPixels=this.prepareHeartbeatPixels(),this.heartbeatInterval=this.heartbeatPixels[0]?.interval;let e=this.heartbeatPixels[0]?.url,t=this.heartbeatPixels[0]?.delay;if(t!==void 0&&e)try{this.heartbeatFirstTimeoutId=window.setTimeout(()=>{this.sendHeartbeat(e)},t*1e3)}catch(e){this.log({message:e.message})}}play(){this.stopwatch.start(),this.send(`resume`)}pause(){this.stopwatch.stop(),this.send(`pause`)}onPosition(e){if(this.position=e,this.heartbeatInterval&&this.stopwatch.time!==0&&this.stopwatch.time%this.heartbeatInterval===0)for(let[e,{url:t}]of this.heartbeatPixels.entries())this.heartbeatLastTimeSent[e]!==this.stopwatch.time&&(this.heartbeatLastTimeSent[e]=this.stopwatch.time,this.sendHeartbeat(t))}onWillSeek(){this.started&&this.send(`stop`)}onSeeked(){this.started&&this.send(`start`)}onEnded(){this.stopwatch.stop(),this.send(`stop`)}validatePixels(e,t){return e?.filter(e=>e.event===t.event&&t.keys.every(t=>!!e[t]))}getFrameTimestamp(){let e;if(this.isActiveLive){let t=Math.floor(Math.random()*25)+5;e=Date.now()/1e3-t}else e=this.position||0;return Math.floor(e)}getUTC(){return Math.floor(Date.now()/1e3)}prepareUrl(e){return e.replace(`{@fts_fake_sec}`,String(this.getFrameTimestamp())).replace(`{@utc_sec}`,String(this.getUTC())).split(`&`).filter(e=>!/={@[a-zA-Z_]+}/.test(e)).join(`&`)}preparePixels(e){return this.validatePixels(this.params.pixels,{event:e,keys:[`url`]})}prepareHeartbeatPixels(){return this.validatePixels(this.params.pixels,{event:`heartbeat`,keys:[`url`,`interval`]})}call(e){try{this.idleCallbackIds.push(pt.requestIdleCallbackPolyfilled(()=>_t().src=this.prepareUrl(e)))}catch(e){this.log({message:e.message})}}send(e){let t=this.preparePixels(e);for(let{url:e}of t)this.call(e)}sendHeartbeat(e){this.call(e)}}