@vonage/ml-transformers 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (33) hide show
  1. package/README.md +9 -0
  2. package/dist/docs/.nojekyll +1 -0
  3. package/dist/docs/assets/highlight.css +85 -0
  4. package/dist/docs/assets/icons.css +1043 -0
  5. package/dist/docs/assets/icons.png +0 -0
  6. package/dist/docs/assets/icons@2x.png +0 -0
  7. package/dist/docs/assets/main.js +52 -0
  8. package/dist/docs/assets/search.js +1 -0
  9. package/dist/docs/assets/style.css +1413 -0
  10. package/dist/docs/assets/widgets.png +0 -0
  11. package/dist/docs/assets/widgets@2x.png +0 -0
  12. package/dist/docs/enums/BlurRadius.html +7 -0
  13. package/dist/docs/index.html +15 -0
  14. package/dist/docs/modules.html +30 -0
  15. package/dist/ml-transformers.es.js +1254 -0
  16. package/dist/ml-transformers.umd.js +2 -0
  17. package/dist/models/selfie_segmentation_landscape.tflite +0 -0
  18. package/dist/tflite-simd.wasm +0 -0
  19. package/dist/types/main.d.ts +95 -0
  20. package/dist/types/src/transformers/BackgroundTransformer.d.ts +34 -0
  21. package/dist/types/src/transformers/BlurPostProcess.d.ts +10 -0
  22. package/dist/types/src/transformers/PostProcessInterface.d.ts +5 -0
  23. package/dist/types/src/transformers/Queue.d.ts +13 -0
  24. package/dist/types/src/transformers/SilueteBlurPostProcess.d.ts +10 -0
  25. package/dist/types/src/transformers/VideoPostProcess.d.ts +11 -0
  26. package/dist/types/src/transformers/VirtualPostProcess.d.ts +11 -0
  27. package/dist/types/src/transformers/VonageTFLite.d.ts +28 -0
  28. package/dist/types/src/transformers/VonageTransformerConfig.d.ts +1 -0
  29. package/dist/types/src/types.d.ts +65 -0
  30. package/dist/types/src/utils/utils.d.ts +6 -0
  31. package/dist/types/src/vonage/VonageMediaProcessorThread.d.ts +1 -0
  32. package/dist/types/src/vonage/VonageMediaProcessorWorker.d.ts +10 -0
  33. package/package.json +65 -0
@@ -0,0 +1,2 @@
1
+ (function(s,K){typeof exports=="object"&&typeof module!="undefined"?K(exports):typeof define=="function"&&define.amd?define(["exports"],K):(s=typeof globalThis!="undefined"?globalThis:s||self,K(s["ml-transformers"]={}))})(this,function(s){"use strict";function K(){return new Promise((Z,l)=>{typeof MediaStreamTrackProcessor=="undefined"||typeof MediaStreamTrackGenerator=="undefined"?l("Your browser does not support the MediaStreamTrack API for Insertable Streams of Media."):Z()})}var D={exports:{}},A=function(l,d){return function(){for(var c=new Array(arguments.length),m=0;m<c.length;m++)c[m]=arguments[m];return l.apply(d,c)}},Yl=A,y=Object.prototype.toString;function k(Z){return Array.isArray(Z)}function U(Z){return typeof Z=="undefined"}function Jl(Z){return Z!==null&&!U(Z)&&Z.constructor!==null&&!U(Z.constructor)&&typeof Z.constructor.isBuffer=="function"&&Z.constructor.isBuffer(Z)}function _(Z){return y.call(Z)==="[object ArrayBuffer]"}function Il(Z){return y.call(Z)==="[object FormData]"}function Kl(Z){var l;return typeof ArrayBuffer!="undefined"&&ArrayBuffer.isView?l=ArrayBuffer.isView(Z):l=Z&&Z.buffer&&_(Z.buffer),l}function Nl(Z){return typeof Z=="string"}function Hl(Z){return typeof Z=="number"}function q(Z){return Z!==null&&typeof Z=="object"}function H(Z){if(y.call(Z)!=="[object Object]")return!1;var l=Object.getPrototypeOf(Z);return l===null||l===Object.prototype}function Sl(Z){return y.call(Z)==="[object Date]"}function vl(Z){return y.call(Z)==="[object File]"}function Ll(Z){return y.call(Z)==="[object Blob]"}function $(Z){return y.call(Z)==="[object Function]"}function Fl(Z){return q(Z)&&$(Z.pipe)}function zl(Z){return y.call(Z)==="[object URLSearchParams]"}function fl(Z){return Z.trim?Z.trim():Z.replace(/^\s+|\s+$/g,"")}function xl(){return typeof navigator!="undefined"&&(navigator.product==="ReactNative"||navigator.product==="NativeScript"||navigator.product==="NS")?!1:typeof window!="undefined"&&typeof document!="undefined"}function B(Z,l){if(!(Z===null||typeof Z=="undefined"))if(typeof Z!="object"&&(Z=[Z]),k(Z))for(var d=0,b=Z.length;d<b;d++)l.call(null,Z[d],d,Z);else for(var c in Z)Object.prototype.hasOwnProperty.call(Z,c)&&l.call(null,Z[c],c,Z)}function T(){var Z={};function l(c,m){H(Z[m])&&H(c)?Z[m]=T(Z[m],c):H(c)?Z[m]=T({},c):k(c)?Z[m]=c.slice():Z[m]=c}for(var d=0,b=arguments.length;d<b;d++)B(arguments[d],l);return Z}function Cl(Z,l,d){return B(l,function(c,m){d&&typeof c=="function"?Z[m]=Yl(c,d):Z[m]=c}),Z}function kl(Z){return Z.charCodeAt(0)===65279&&(Z=Z.slice(1)),Z}var W={isArray:k,isArrayBuffer:_,isBuffer:Jl,isFormData:Il,isArrayBufferView:Kl,isString:Nl,isNumber:Hl,isObject:q,isPlainObject:H,isUndefined:U,isDate:Sl,isFile:vl,isBlob:Ll,isFunction:$,isStream:Fl,isURLSearchParams:zl,isStandardBrowserEnv:xl,forEach:B,merge:T,extend:Cl,trim:fl,stripBOM:kl},R=W;function ll(Z){return encodeURIComponent(Z).replace(/%3A/gi,":").replace(/%24/g,"$").replace(/%2C/gi,",").replace(/%20/g,"+").replace(/%5B/gi,"[").replace(/%5D/gi,"]")}var Zl=function(l,d,b){if(!d)return l;var c;if(b)c=b(d);else if(R.isURLSearchParams(d))c=d.toString();else{var m=[];R.forEach(d,function(e,X){e===null||typeof e=="undefined"||(R.isArray(e)?X=X+"[]":e=[e],R.forEach(e,function(u){R.isDate(u)?u=u.toISOString():R.isObject(u)&&(u=JSON.stringify(u)),m.push(ll(X)+"="+ll(u))}))}),c=m.join("&")}if(c){var t=l.indexOf("#");t!==-1&&(l=l.slice(0,t)),l+=(l.indexOf("?")===-1?"?":"&")+c}return l},Ul=W;function S(){this.handlers=[]}S.prototype.use=function(l,d,b){return this.handlers.push({fulfilled:l,rejected:d,synchronous:b?b.synchronous:!1,runWhen:b?b.runWhen:null}),this.handlers.length-1},S.prototype.eject=function(l){this.handlers[l]&&(this.handlers[l]=null)},S.prototype.forEach=function(l){Ul.forEach(this.handlers,function(b){b!==null&&l(b)})};var Bl=S,Tl=W,gl=function(l,d){Tl.forEach(l,function(c,m){m!==d&&m.toUpperCase()===d.toUpperCase()&&(l[d]=c,delete l[m])})},dl=function(l,d,b,c,m){return l.config=d,b&&(l.code=b),l.request=c,l.response=m,l.isAxiosError=!0,l.toJSON=function(){return{message:this.message,name:this.name,description:this.description,number:this.number,fileName:this.fileName,lineNumber:this.lineNumber,columnNumber:this.columnNumber,stack:this.stack,config:this.config,code:this.code,status:this.response&&this.response.status?this.response.status:null}},l},wl=dl,cl=function(l,d,b,c,m){var t=new Error(l);return wl(t,d,b,c,m)},Ql=cl,Pl=function(l,d,b){var c=b.config.validateStatus;!b.status||!c||c(b.status)?l(b):d(Ql("Request failed with status code "+b.status,b.config,null,b.request,b))},v=W,jl=v.isStandardBrowserEnv()?function(){return{write:function(d,b,c,m,t,i){var e=[];e.push(d+"="+encodeURIComponent(b)),v.isNumber(c)&&e.push("expires="+new Date(c).toGMTString()),v.isString(m)&&e.push("path="+m),v.isString(t)&&e.push("domain="+t),i===!0&&e.push("secure"),document.cookie=e.join("; ")},read:function(d){var b=document.cookie.match(new RegExp("(^|;\\s*)("+d+")=([^;]*)"));return b?decodeURIComponent(b[3]):null},remove:function(d){this.write(d,"",Date.now()-864e5)}}}():function(){return{write:function(){},read:function(){return null},remove:function(){}}}(),Ml=function(l){return/^([a-z][a-z\d+\-.]*:)?\/\//i.test(l)},El=function(l,d){return d?l.replace(/\/+$/,"")+"/"+d.replace(/^\/+/,""):l},Ol=Ml,Dl=El,Al=function(l,d){return l&&!Ol(d)?Dl(l,d):d},g=W,_l=["age","authorization","content-length","content-type","etag","expires","from","host","if-modified-since","if-unmodified-since","last-modified","location","max-forwards","proxy-authorization","referer","retry-after","user-agent"],ql=function(l){var d={},b,c,m;return l&&g.forEach(l.split(`
2
+ `),function(i){if(m=i.indexOf(":"),b=g.trim(i.substr(0,m)).toLowerCase(),c=g.trim(i.substr(m+1)),b){if(d[b]&&_l.indexOf(b)>=0)return;b==="set-cookie"?d[b]=(d[b]?d[b]:[]).concat([c]):d[b]=d[b]?d[b]+", "+c:c}}),d},bl=W,$l=bl.isStandardBrowserEnv()?function(){var l=/(msie|trident)/i.test(navigator.userAgent),d=document.createElement("a"),b;function c(m){var t=m;return l&&(d.setAttribute("href",t),t=d.href),d.setAttribute("href",t),{href:d.href,protocol:d.protocol?d.protocol.replace(/:$/,""):"",host:d.host,search:d.search?d.search.replace(/^\?/,""):"",hash:d.hash?d.hash.replace(/^#/,""):"",hostname:d.hostname,port:d.port,pathname:d.pathname.charAt(0)==="/"?d.pathname:"/"+d.pathname}}return b=c(window.location.href),function(t){var i=bl.isString(t)?c(t):t;return i.protocol===b.protocol&&i.host===b.host}}():function(){return function(){return!0}}();function w(Z){this.message=Z}w.prototype.toString=function(){return"Cancel"+(this.message?": "+this.message:"")},w.prototype.__CANCEL__=!0;var L=w,F=W,lZ=Pl,ZZ=jl,dZ=Zl,cZ=Al,bZ=ql,mZ=$l,Q=cl,nZ=f,tZ=L,ml=function(l){return new Promise(function(b,c){var m=l.data,t=l.headers,i=l.responseType,e;function X(){l.cancelToken&&l.cancelToken.unsubscribe(e),l.signal&&l.signal.removeEventListener("abort",e)}F.isFormData(m)&&delete t["Content-Type"];var n=new XMLHttpRequest;if(l.auth){var u=l.auth.username||"",p=l.auth.password?unescape(encodeURIComponent(l.auth.password)):"";t.Authorization="Basic "+btoa(u+":"+p)}var o=cZ(l.baseURL,l.url);n.open(l.method.toUpperCase(),dZ(o,l.params,l.paramsSerializer),!0),n.timeout=l.timeout;function rl(){if(!!n){var h="getAllResponseHeaders"in n?bZ(n.getAllResponseHeaders()):null,I=!i||i==="text"||i==="json"?n.responseText:n.response,r={data:I,status:n.status,statusText:n.statusText,headers:h,config:l,request:n};lZ(function(O){b(O),X()},function(O){c(O),X()},r),n=null}}if("onloadend"in n?n.onloadend=rl:n.onreadystatechange=function(){!n||n.readyState!==4||n.status===0&&!(n.responseURL&&n.responseURL.indexOf("file:")===0)||setTimeout(rl)},n.onabort=function(){!n||(c(Q("Request aborted",l,"ECONNABORTED",n)),n=null)},n.onerror=function(){c(Q("Network Error",l,null,n)),n=null},n.ontimeout=function(){var I=l.timeout?"timeout of "+l.timeout+"ms exceeded":"timeout exceeded",r=l.transitional||nZ.transitional;l.timeoutErrorMessage&&(I=l.timeoutErrorMessage),c(Q(I,l,r.clarifyTimeoutError?"ETIMEDOUT":"ECONNABORTED",n)),n=null},F.isStandardBrowserEnv()){var Rl=(l.withCredentials||mZ(o))&&l.xsrfCookieName?ZZ.read(l.xsrfCookieName):void 0;Rl&&(t[l.xsrfHeaderName]=Rl)}"setRequestHeader"in n&&F.forEach(t,function(I,r){typeof m=="undefined"&&r.toLowerCase()==="content-type"?delete t[r]:n.setRequestHeader(r,I)}),F.isUndefined(l.withCredentials)||(n.withCredentials=!!l.withCredentials),i&&i!=="json"&&(n.responseType=l.responseType),typeof l.onDownloadProgress=="function"&&n.addEventListener("progress",l.onDownloadProgress),typeof l.onUploadProgress=="function"&&n.upload&&n.upload.addEventListener("progress",l.onUploadProgress),(l.cancelToken||l.signal)&&(e=function(h){!n||(c(!h||h&&h.type?new tZ("canceled"):h),n.abort(),n=null)},l.cancelToken&&l.cancelToken.subscribe(e),l.signal&&(l.signal.aborted?e():l.signal.addEventListener("abort",e))),m||(m=null),n.send(m)})},a=W,nl=gl,iZ=dl,eZ={"Content-Type":"application/x-www-form-urlencoded"};function tl(Z,l){!a.isUndefined(Z)&&a.isUndefined(Z["Content-Type"])&&(Z["Content-Type"]=l)}function uZ(){var Z;return(typeof XMLHttpRequest!="undefined"||typeof process!="undefined"&&Object.prototype.toString.call(process)==="[object process]")&&(Z=ml),Z}function XZ(Z,l,d){if(a.isString(Z))try{return(l||JSON.parse)(Z),a.trim(Z)}catch(b){if(b.name!=="SyntaxError")throw b}return(d||JSON.stringify)(Z)}var z={transitional:{silentJSONParsing:!0,forcedJSONParsing:!0,clarifyTimeoutError:!1},adapter:uZ(),transformRequest:[function(l,d){return nl(d,"Accept"),nl(d,"Content-Type"),a.isFormData(l)||a.isArrayBuffer(l)||a.isBuffer(l)||a.isStream(l)||a.isFile(l)||a.isBlob(l)?l:a.isArrayBufferView(l)?l.buffer:a.isURLSearchParams(l)?(tl(d,"application/x-www-form-urlencoded;charset=utf-8"),l.toString()):a.isObject(l)||d&&d["Content-Type"]==="application/json"?(tl(d,"application/json"),XZ(l)):l}],transformResponse:[function(l){var d=this.transitional||z.transitional,b=d&&d.silentJSONParsing,c=d&&d.forcedJSONParsing,m=!b&&this.responseType==="json";if(m||c&&a.isString(l)&&l.length)try{return JSON.parse(l)}catch(t){if(m)throw t.name==="SyntaxError"?iZ(t,this,"E_JSON_PARSE"):t}return l}],timeout:0,xsrfCookieName:"XSRF-TOKEN",xsrfHeaderName:"X-XSRF-TOKEN",maxContentLength:-1,maxBodyLength:-1,validateStatus:function(l){return l>=200&&l<300},headers:{common:{Accept:"application/json, text/plain, */*"}}};a.forEach(["delete","get","head"],function(l){z.headers[l]={}}),a.forEach(["post","put","patch"],function(l){z.headers[l]=a.merge(eZ)});var f=z,aZ=W,WZ=f,oZ=function(l,d,b){var c=this||WZ;return aZ.forEach(b,function(t){l=t.call(c,l,d)}),l},il=function(l){return!!(l&&l.__CANCEL__)},el=W,P=oZ,GZ=il,sZ=f,hZ=L;function j(Z){if(Z.cancelToken&&Z.cancelToken.throwIfRequested(),Z.signal&&Z.signal.aborted)throw new hZ("canceled")}var VZ=function(l){j(l),l.headers=l.headers||{},l.data=P.call(l,l.data,l.headers,l.transformRequest),l.headers=el.merge(l.headers.common||{},l.headers[l.method]||{},l.headers),el.forEach(["delete","get","head","post","put","patch","common"],function(c){delete l.headers[c]});var d=l.adapter||sZ.adapter;return d(l).then(function(c){return j(l),c.data=P.call(l,c.data,c.headers,l.transformResponse),c},function(c){return GZ(c)||(j(l),c&&c.response&&(c.response.data=P.call(l,c.response.data,c.response.headers,l.transformResponse))),Promise.reject(c)})},G=W,ul=function(l,d){d=d||{};var b={};function c(n,u){return G.isPlainObject(n)&&G.isPlainObject(u)?G.merge(n,u):G.isPlainObject(u)?G.merge({},u):G.isArray(u)?u.slice():u}function m(n){if(G.isUndefined(d[n])){if(!G.isUndefined(l[n]))return c(void 0,l[n])}else return c(l[n],d[n])}function t(n){if(!G.isUndefined(d[n]))return c(void 0,d[n])}function i(n){if(G.isUndefined(d[n])){if(!G.isUndefined(l[n]))return c(void 0,l[n])}else return c(void 0,d[n])}function e(n){if(n in d)return c(l[n],d[n]);if(n in l)return c(void 0,l[n])}var X={url:t,method:t,data:t,baseURL:i,transformRequest:i,transformResponse:i,paramsSerializer:i,timeout:i,timeoutMessage:i,withCredentials:i,adapter:i,responseType:i,xsrfCookieName:i,xsrfHeaderName:i,onUploadProgress:i,onDownloadProgress:i,decompress:i,maxContentLength:i,maxBodyLength:i,transport:i,httpAgent:i,httpsAgent:i,cancelToken:i,socketPath:i,responseEncoding:i,validateStatus:e};return G.forEach(Object.keys(l).concat(Object.keys(d)),function(u){var p=X[u]||m,o=p(u);G.isUndefined(o)&&p!==e||(b[u]=o)}),b},Xl={version:"0.25.0"},pZ=Xl.version,M={};["object","boolean","number","function","string","symbol"].forEach(function(Z,l){M[Z]=function(b){return typeof b===Z||"a"+(l<1?"n ":" ")+Z}});var al={};M.transitional=function(l,d,b){function c(m,t){return"[Axios v"+pZ+"] Transitional option '"+m+"'"+t+(b?". "+b:"")}return function(m,t,i){if(l===!1)throw new Error(c(t," has been removed"+(d?" in "+d:"")));return d&&!al[t]&&(al[t]=!0,console.warn(c(t," has been deprecated since v"+d+" and will be removed in the near future"))),l?l(m,t,i):!0}};function yZ(Z,l,d){if(typeof Z!="object")throw new TypeError("options must be an object");for(var b=Object.keys(Z),c=b.length;c-- >0;){var m=b[c],t=l[m];if(t){var i=Z[m],e=i===void 0||t(i,m,Z);if(e!==!0)throw new TypeError("option "+m+" must be "+e);continue}if(d!==!0)throw Error("Unknown option "+m)}}var rZ={assertOptions:yZ,validators:M},Wl=W,RZ=Zl,ol=Bl,Gl=VZ,x=ul,sl=rZ,Y=sl.validators;function N(Z){this.defaults=Z,this.interceptors={request:new ol,response:new ol}}N.prototype.request=function(l,d){if(typeof l=="string"?(d=d||{},d.url=l):d=l||{},!d.url)throw new Error("Provided config url is not valid");d=x(this.defaults,d),d.method?d.method=d.method.toLowerCase():this.defaults.method?d.method=this.defaults.method.toLowerCase():d.method="get";var b=d.transitional;b!==void 0&&sl.assertOptions(b,{silentJSONParsing:Y.transitional(Y.boolean),forcedJSONParsing:Y.transitional(Y.boolean),clarifyTimeoutError:Y.transitional(Y.boolean)},!1);var c=[],m=!0;this.interceptors.request.forEach(function(o){typeof o.runWhen=="function"&&o.runWhen(d)===!1||(m=m&&o.synchronous,c.unshift(o.fulfilled,o.rejected))});var t=[];this.interceptors.response.forEach(function(o){t.push(o.fulfilled,o.rejected)});var i;if(!m){var e=[Gl,void 0];for(Array.prototype.unshift.apply(e,c),e=e.concat(t),i=Promise.resolve(d);e.length;)i=i.then(e.shift(),e.shift());return i}for(var X=d;c.length;){var n=c.shift(),u=c.shift();try{X=n(X)}catch(p){u(p);break}}try{i=Gl(X)}catch(p){return Promise.reject(p)}for(;t.length;)i=i.then(t.shift(),t.shift());return i},N.prototype.getUri=function(l){if(!l.url)throw new Error("Provided config url is not valid");return l=x(this.defaults,l),RZ(l.url,l.params,l.paramsSerializer).replace(/^\?/,"")},Wl.forEach(["delete","get","head","options"],function(l){N.prototype[l]=function(d,b){return this.request(x(b||{},{method:l,url:d,data:(b||{}).data}))}}),Wl.forEach(["post","put","patch"],function(l){N.prototype[l]=function(d,b,c){return this.request(x(c||{},{method:l,url:d,data:b}))}});var YZ=N,JZ=L;function J(Z){if(typeof Z!="function")throw new TypeError("executor must be a function.");var l;this.promise=new Promise(function(c){l=c});var d=this;this.promise.then(function(b){if(!!d._listeners){var c,m=d._listeners.length;for(c=0;c<m;c++)d._listeners[c](b);d._listeners=null}}),this.promise.then=function(b){var c,m=new Promise(function(t){d.subscribe(t),c=t}).then(b);return m.cancel=function(){d.unsubscribe(c)},m},Z(function(c){d.reason||(d.reason=new JZ(c),l(d.reason))})}J.prototype.throwIfRequested=function(){if(this.reason)throw this.reason},J.prototype.subscribe=function(l){if(this.reason){l(this.reason);return}this._listeners?this._listeners.push(l):this._listeners=[l]},J.prototype.unsubscribe=function(l){if(!!this._listeners){var d=this._listeners.indexOf(l);d!==-1&&this._listeners.splice(d,1)}},J.source=function(){var l,d=new J(function(c){l=c});return{token:d,cancel:l}};var IZ=J,KZ=function(l){return function(b){return l.apply(null,b)}},NZ=W,HZ=function(l){return NZ.isObject(l)&&l.isAxiosError===!0},hl=W,SZ=A,C=YZ,vZ=ul,LZ=f;function Vl(Z){var l=new C(Z),d=SZ(C.prototype.request,l);return hl.extend(d,C.prototype,l),hl.extend(d,l),d.create=function(c){return Vl(vZ(Z,c))},d}var V=Vl(LZ);V.Axios=C,V.Cancel=L,V.CancelToken=IZ,V.isCancel=il,V.VERSION=Xl.version,V.all=function(l){return Promise.all(l)},V.spread=KZ,V.isAxiosError=HZ,D.exports=V,D.exports.default=V;for(var FZ=[],E=0;E<256;++E)FZ.push((E+256).toString(16).substr(1));class zZ{constructor(){this.processor_=null,this.generator_=null}init(l){return new Promise((d,b)=>{try{this.processor_=new MediaStreamTrackProcessor(l)}catch(c){console.log(`[InsertableStreamHelper] MediaStreamTrackProcessor failed: ${c}`),b(c)}try{l.kind==="audio"?this.generator_=new MediaStreamTrackGenerator({kind:"audio"}):l.kind==="video"?this.generator_=new MediaStreamTrackGenerator({kind:"video"}):b("kind not supported")}catch(c){console.log(`[InsertableStreamHelper] MediaStreamTrackGenerator failed: ${c}`),b(c)}d()})}getReadable(){return this.processor_.readable}getWriteable(){return this.generator_.writable}getProccesorTrack(){return this.generator_}}class fZ{constructor(l){this.insertableStreamHelper_=new zZ,this.mediaProcessor_=l}setTrack(l){return new Promise((d,b)=>{this.insertableStreamHelper_.init(l).then(()=>{this.mediaProcessor_.transform(this.insertableStreamHelper_.getReadable(),this.insertableStreamHelper_.getWriteable()).then(()=>{d(this.insertableStreamHelper_.getProccesorTrack())}).catch(c=>{b(c)})}).catch(c=>{b(c)})})}destroy(){return new Promise((l,d)=>{this.mediaProcessor_?this.mediaProcessor_.destroy().then(()=>{l()}).catch(b=>{d(b)}):d("no processor")})}}const pl="KGZ1bmN0aW9uKCl7InVzZSBzdHJpY3QiO2NsYXNzIER7fUQudXBkYXRlcz17dHJhbnNmb3JtZXJfbmV3OiJOZXcgdHJhbnNmb3JtZXIiLHRyYW5zZm9ybWVyX251bGw6Ik51bGwgdHJhbnNmb3JtZXIifSxELmVycm9ycz17dHJhbnNmb3JtZXJfbm9uZToiTm8gdHJhbnNmb3JtZXJzIHByb3ZpZGVkIix0cmFuc2Zvcm1lcl9zdGFydDoiQ2Fubm90IHN0YXJ0IHRyYW5zZm9ybWVyIix0cmFuc2Zvcm1lcl90cmFuc2Zvcm06IkNhbm5vdCB0cmFuc2Zvcm0gZnJhbWUiLHRyYW5zZm9ybWVyX2ZsdXNoOiJDYW5ub3QgZmx1c2ggdHJhbnNmb3JtZXIifTt2YXIgX2U9e2V4cG9ydHM6e319LHhlPWZ1bmN0aW9uKGUsdCl7cmV0dXJuIGZ1bmN0aW9uKCl7Zm9yKHZhciBzPW5ldyBBcnJheShhcmd1bWVudHMubGVuZ3RoKSxhPTA7YTxzLmxlbmd0aDthKyspc1thXT1hcmd1bWVudHNbYV07cmV0dXJuIGUuYXBwbHkodCxzKX19LG10PXhlLGs9T2JqZWN0LnByb3RvdHlwZS50b1N0cmluZztmdW5jdGlvbiBnZShyKXtyZXR1cm4gQXJyYXkuaXNBcnJheShyKX1mdW5jdGlvbiB2ZShyKXtyZXR1cm4gdHlwZW9mIHI9PSJ1bmRlZmluZWQifWZ1bmN0aW9uIF90KHIpe3JldHVybiByIT09bnVsbCYmIXZlKHIpJiZyLmNvbnN0cnVjdG9yIT09bnVsbCYmIXZlKHIuY29uc3RydWN0b3IpJiZ0eXBlb2Ygci5jb25zdHJ1Y3Rvci5pc0J1ZmZlcj09ImZ1bmN0aW9uIiYmci5jb25zdHJ1Y3Rvci5pc0J1ZmZlcihyKX1mdW5jdGlvbiBOZShyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgQXJyYXlCdWZmZXJdIn1mdW5jdGlvbiBndChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgRm9ybURhdGFdIn1mdW5jdGlvbiB2dChyKXt2YXIgZTtyZXR1cm4gdHlwZW9mIEFycmF5QnVmZmVyIT0idW5kZWZpbmVkIiYmQXJyYXlCdWZmZXIuaXNWaWV3P2U9QXJyYXlCdWZmZXIuaXNWaWV3KHIpOmU9ciYmci5idWZmZXImJk5lKHIuYnVmZmVyKSxlfWZ1bmN0aW9uIHl0KHIpe3JldHVybiB0eXBlb2Ygcj09InN0cmluZyJ9ZnVuY3Rpb24gd3Qocil7cmV0dXJuIHR5cGVvZiByPT0ibnVtYmVyIn1mdW5jdGlvbiBMZShyKXtyZXR1cm4gciE9PW51bGwmJnR5cGVvZiByPT0ib2JqZWN0In1mdW5jdGlvbiBLKHIpe2lmKGsuY2FsbChyKSE9PSJbb2JqZWN0IE9iamVjdF0iKXJldHVybiExO3ZhciBlPU9iamVjdC5nZXRQcm90b3R5cGVPZihyKTtyZXR1cm4gZT09PW51bGx8fGU9PT1PYmplY3QucHJvdG90eXBlfWZ1bmN0aW9uIEN0KHIpe3JldHVybiBrLmNhbGwocik9PT0iW29iamVjdCBEYXRlXSJ9ZnVuY3Rpb24gYnQocil7cmV0dXJuIGsuY2FsbChyKT09PSJbb2JqZWN0IEZpbGVdIn1mdW5jdGlvbiBUdChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgQmxvYl0ifWZ1bmN0aW9uIFVlKHIpe3JldHVybiBrLmNhbGwocik9PT0iW29iamVjdCBGdW5jdGlvbl0ifWZ1bmN0aW9uIEV0KHIpe3JldHVybiBMZShyKSYmVWUoci5waXBlKX1mdW5jdGlvbiBSdChyKXtyZXR1cm4gay5jYWxsKHIpPT09IltvYmplY3QgVVJMU2VhcmNoUGFyYW1zXSJ9ZnVuY3Rpb24gUHQocil7cmV0dXJuIHIudHJpbT9yLnRyaW0oKTpyLnJlcGxhY2UoL15ccyt8XHMrJC9nLCIiKX1mdW5jdGlvbiBPdCgpe3JldHVybiB0eXBlb2YgbmF2aWdhdG9yIT0idW5kZWZpbmVkIiYmKG5hdmlnYXRvci5wcm9kdWN0PT09IlJlYWN0TmF0aXZlInx8bmF2aWdhdG9yLnByb2R1Y3Q9PT0iTmF0aXZlU2NyaXB0Inx8bmF2aWdhdG9yLnByb2R1Y3Q9PT0iTlMiKT8hMTp0eXBlb2Ygd2luZG93IT0idW5kZWZpbmVkIiYmdHlwZW9mIGRvY3VtZW50IT0idW5kZWZpbmVkIn1mdW5jdGlvbiB5ZShyLGUpe2lmKCEocj09PW51bGx8fHR5cGVvZiByPT0idW5kZWZpbmVkIikpaWYodHlwZW9mIHIhPSJvYmplY3QiJiYocj1bcl0pLGdlKHIpKWZvcih2YXIgdD0wLG49ci5sZW5ndGg7dDxuO3QrKyllLmNhbGwobnVsbCxyW3RdLHQscik7ZWxzZSBmb3IodmFyIHMgaW4gcilPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwocixzKSYmZS5jYWxsKG51bGwscltzXSxzLHIpfWZ1bmN0aW9uIHdlKCl7dmFyIHI9e307ZnVuY3Rpb24gZShzLGEpe0soclthXSkmJksocyk/clthXT13ZShyW2FdLHMpOksocyk/clthXT13ZSh7fSxzKTpnZShzKT9yW2FdPXMuc2xpY2UoKTpyW2FdPXN9Zm9yKHZhciB0PTAsbj1hcmd1bWVudHMubGVuZ3RoO3Q8bjt0KyspeWUoYXJndW1lbnRzW3RdLGUpO3JldHVybiByfWZ1bmN0aW9uIFN0KHIsZSx0KXtyZXR1cm4geWUoZSxmdW5jdGlvbihzLGEpe3QmJnR5cGVvZiBzPT0iZnVuY3Rpb24iP3JbYV09bXQocyx0KTpyW2FdPXN9KSxyfWZ1bmN0aW9uIEl0KHIpe3JldHVybiByLmNoYXJDb2RlQXQoMCk9PT02NTI3OSYmKHI9ci5zbGljZSgxKSkscn12YXIgUD17aXNBcnJheTpnZSxpc0FycmF5QnVmZmVyOk5lLGlzQnVmZmVyOl90LGlzRm9ybURhdGE6Z3QsaXNBcnJheUJ1ZmZlclZpZXc6dnQsaXNTdHJpbmc6eXQsaXNOdW1iZXI6d3QsaXNPYmplY3Q6TGUsaXNQbGFpbk9iamVjdDpLLGlzVW5kZWZpbmVkOnZlLGlzRGF0ZTpDdCxpc0ZpbGU6YnQsaXNCbG9iOlR0LGlzRnVuY3Rpb246VWUsaXNTdHJlYW06RXQsaXNVUkxTZWFyY2hQYXJhbXM6UnQsaXNTdGFuZGFyZEJyb3dzZXJFbnY6T3QsZm9yRWFjaDp5ZSxtZXJnZTp3ZSxleHRlbmQ6U3QsdHJpbTpQdCxzdHJpcEJPTTpJdH0sVz1QO2Z1bmN0aW9uIEJlKHIpe3JldHVybiBlbmNvZGVVUklDb21wb25lbnQocikucmVwbGFjZSgvJTNBL2dpLCI6IikucmVwbGFjZSgvJTI0L2csIiQiKS5yZXBsYWNlKC8lMkMvZ2ksIiwiKS5yZXBsYWNlKC8lMjAvZywiKyIpLnJlcGxhY2UoLyU1Qi9naSwiWyIpLnJlcGxhY2UoLyU1RC9naSwiXSIpfXZhciBrZT1mdW5jdGlvbihlLHQsbil7aWYoIXQpcmV0dXJuIGU7dmFyIHM7aWYobilzPW4odCk7ZWxzZSBpZihXLmlzVVJMU2VhcmNoUGFyYW1zKHQpKXM9dC50b1N0cmluZygpO2Vsc2V7dmFyIGE9W107Vy5mb3JFYWNoKHQsZnVuY3Rpb24oaCxnKXtoPT09bnVsbHx8dHlwZW9mIGg9PSJ1bmRlZmluZWQifHwoVy5pc0FycmF5KGgpP2c9ZysiW10iOmg9W2hdLFcuZm9yRWFjaChoLGZ1bmN0aW9uKHApe1cuaXNEYXRlKHApP3A9cC50b0lTT1N0cmluZygpOlcuaXNPYmplY3QocCkmJihwPUpTT04uc3RyaW5naWZ5KHApKSxhLnB1c2goQmUoZykrIj0iK0JlKHApKX0pKX0pLHM9YS5qb2luKCImIil9aWYocyl7dmFyIGY9ZS5pbmRleE9mKCIjIik7ZiE9PS0xJiYoZT1lLnNsaWNlKDAsZikpLGUrPShlLmluZGV4T2YoIj8iKT09PS0xPyI/IjoiJiIpK3N9cmV0dXJuIGV9LE10PVA7ZnVuY3Rpb24gWSgpe3RoaXMuaGFuZGxlcnM9W119WS5wcm90b3R5cGUudXNlPWZ1bmN0aW9uKGUsdCxuKXtyZXR1cm4gdGhpcy5oYW5kbGVycy5wdXNoKHtmdWxmaWxsZWQ6ZSxyZWplY3RlZDp0LHN5bmNocm9ub3VzOm4/bi5zeW5jaHJvbm91czohMSxydW5XaGVuOm4/bi5ydW5XaGVuOm51bGx9KSx0aGlzLmhhbmRsZXJzLmxlbmd0aC0xfSxZLnByb3RvdHlwZS5lamVjdD1mdW5jdGlvbihlKXt0aGlzLmhhbmRsZXJzW2VdJiYodGhpcy5oYW5kbGVyc1tlXT1udWxsKX0sWS5wcm90b3R5cGUuZm9yRWFjaD1mdW5jdGlvbihlKXtNdC5mb3JFYWNoKHRoaXMuaGFuZGxlcnMsZnVuY3Rpb24obil7biE9PW51bGwmJmUobil9KX07dmFyIEF0PVksRnQ9UCx4dD1mdW5jdGlvbihlLHQpe0Z0LmZvckVhY2goZSxmdW5jdGlvbihzLGEpe2EhPT10JiZhLnRvVXBwZXJDYXNlKCk9PT10LnRvVXBwZXJDYXNlKCkmJihlW3RdPXMsZGVsZXRlIGVbYV0pfSl9LEhlPWZ1bmN0aW9uKGUsdCxuLHMsYSl7cmV0dXJuIGUuY29uZmlnPXQsbiYmKGUuY29kZT1uKSxlLnJlcXVlc3Q9cyxlLnJlc3BvbnNlPWEsZS5pc0F4aW9zRXJyb3I9ITAsZS50b0pTT049ZnVuY3Rpb24oKXtyZXR1cm57bWVzc2FnZTp0aGlzLm1lc3NhZ2UsbmFtZTp0aGlzLm5hbWUsZGVzY3JpcHRpb246dGhpcy5kZXNjcmlwdGlvbixudW1iZXI6dGhpcy5udW1iZXIsZmlsZU5hbWU6dGhpcy5maWxlTmFtZSxsaW5lTnVtYmVyOnRoaXMubGluZU51bWJlcixjb2x1bW5OdW1iZXI6dGhpcy5jb2x1bW5OdW1iZXIsc3RhY2s6dGhpcy5zdGFjayxjb25maWc6dGhpcy5jb25maWcsY29kZTp0aGlzLmNvZGUsc3RhdHVzOnRoaXMucmVzcG9uc2UmJnRoaXMucmVzcG9uc2Uuc3RhdHVzP3RoaXMucmVzcG9uc2Uuc3RhdHVzOm51bGx9fSxlfSxOdD1IZSxEZT1mdW5jdGlvbihlLHQsbixzLGEpe3ZhciBmPW5ldyBFcnJvcihlKTtyZXR1cm4gTnQoZix0LG4scyxhKX0sTHQ9RGUsVXQ9ZnVuY3Rpb24oZSx0LG4pe3ZhciBzPW4uY29uZmlnLnZhbGlkYXRlU3RhdHVzOyFuLnN0YXR1c3x8IXN8fHMobi5zdGF0dXMpP2Uobik6dChMdCgiUmVxdWVzdCBmYWlsZWQgd2l0aCBzdGF0dXMgY29kZSAiK24uc3RhdHVzLG4uY29uZmlnLG51bGwsbi5yZXF1ZXN0LG4pKX0sWj1QLEJ0PVouaXNTdGFuZGFyZEJyb3dzZXJFbnYoKT9mdW5jdGlvbigpe3JldHVybnt3cml0ZTpmdW5jdGlvbih0LG4scyxhLGYsbCl7dmFyIGg9W107aC5wdXNoKHQrIj0iK2VuY29kZVVSSUNvbXBvbmVudChuKSksWi5pc051bWJlcihzKSYmaC5wdXNoKCJleHBpcmVzPSIrbmV3IERhdGUocykudG9HTVRTdHJpbmcoKSksWi5pc1N0cmluZyhhKSYmaC5wdXNoKCJwYXRoPSIrYSksWi5pc1N0cmluZyhmKSYmaC5wdXNoKCJkb21haW49IitmKSxsPT09ITAmJmgucHVzaCgic2VjdXJlIiksZG9jdW1lbnQuY29va2llPWguam9pbigiOyAiKX0scmVhZDpmdW5jdGlvbih0KXt2YXIgbj1kb2N1bWVudC5jb29raWUubWF0Y2gobmV3IFJlZ0V4cCgiKF58O1xccyopKCIrdCsiKT0oW147XSopIikpO3JldHVybiBuP2RlY29kZVVSSUNvbXBvbmVudChuWzNdKTpudWxsfSxyZW1vdmU6ZnVuY3Rpb24odCl7dGhpcy53cml0ZSh0LCIiLERhdGUubm93KCktODY0ZTUpfX19KCk6ZnVuY3Rpb24oKXtyZXR1cm57d3JpdGU6ZnVuY3Rpb24oKXt9LHJlYWQ6ZnVuY3Rpb24oKXtyZXR1cm4gbnVsbH0scmVtb3ZlOmZ1bmN0aW9uKCl7fX19KCksa3Q9ZnVuY3Rpb24oZSl7cmV0dXJuL14oW2Etel1bYS16XGQrXC0uXSo6KT9cL1wvL2kudGVzdChlKX0sSHQ9ZnVuY3Rpb24oZSx0KXtyZXR1cm4gdD9lLnJlcGxhY2UoL1wvKyQvLCIiKSsiLyIrdC5yZXBsYWNlKC9eXC8rLywiIik6ZX0sRHQ9a3QsJHQ9SHQsV3Q9ZnVuY3Rpb24oZSx0KXtyZXR1cm4gZSYmIUR0KHQpPyR0KGUsdCk6dH0sQ2U9UCxxdD1bImFnZSIsImF1dGhvcml6YXRpb24iLCJjb250ZW50LWxlbmd0aCIsImNvbnRlbnQtdHlwZSIsImV0YWciLCJleHBpcmVzIiwiZnJvbSIsImhvc3QiLCJpZi1tb2RpZmllZC1zaW5jZSIsImlmLXVubW9kaWZpZWQtc2luY2UiLCJsYXN0LW1vZGlmaWVkIiwibG9jYXRpb24iLCJtYXgtZm9yd2FyZHMiLCJwcm94eS1hdXRob3JpemF0aW9uIiwicmVmZXJlciIsInJldHJ5LWFmdGVyIiwidXNlci1hZ2VudCJdLGp0PWZ1bmN0aW9uKGUpe3ZhciB0PXt9LG4scyxhO3JldHVybiBlJiZDZS5mb3JFYWNoKGUuc3BsaXQoYApgKSxmdW5jdGlvbihsKXtpZihhPWwuaW5kZXhPZigiOiIpLG49Q2UudHJpbShsLnN1YnN0cigwLGEpKS50b0xvd2VyQ2FzZSgpLHM9Q2UudHJpbShsLnN1YnN0cihhKzEpKSxuKXtpZih0W25dJiZxdC5pbmRleE9mKG4pPj0wKXJldHVybjtuPT09InNldC1jb29raWUiP3Rbbl09KHRbbl0/dFtuXTpbXSkuY29uY2F0KFtzXSk6dFtuXT10W25dP3Rbbl0rIiwgIitzOnN9fSksdH0sJGU9UCxWdD0kZS5pc1N0YW5kYXJkQnJvd3NlckVudigpP2Z1bmN0aW9uKCl7dmFyIGU9Lyhtc2llfHRyaWRlbnQpL2kudGVzdChuYXZpZ2F0b3IudXNlckFnZW50KSx0PWRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoImEiKSxuO2Z1bmN0aW9uIHMoYSl7dmFyIGY9YTtyZXR1cm4gZSYmKHQuc2V0QXR0cmlidXRlKCJocmVmIixmKSxmPXQuaHJlZiksdC5zZXRBdHRyaWJ1dGUoImhyZWYiLGYpLHtocmVmOnQuaHJlZixwcm90b2NvbDp0LnByb3RvY29sP3QucHJvdG9jb2wucmVwbGFjZSgvOiQvLCIiKToiIixob3N0OnQuaG9zdCxzZWFyY2g6dC5zZWFyY2g/dC5zZWFyY2gucmVwbGFjZSgvXlw/LywiIik6IiIsaGFzaDp0Lmhhc2g/dC5oYXNoLnJlcGxhY2UoL14jLywiIik6IiIsaG9zdG5hbWU6dC5ob3N0bmFtZSxwb3J0OnQucG9ydCxwYXRobmFtZTp0LnBhdGhuYW1lLmNoYXJBdCgwKT09PSIvIj90LnBhdGhuYW1lOiIvIit0LnBhdGhuYW1lfX1yZXR1cm4gbj1zKHdpbmRvdy5sb2NhdGlvbi5ocmVmKSxmdW5jdGlvbihmKXt2YXIgbD0kZS5pc1N0cmluZyhmKT9zKGYpOmY7cmV0dXJuIGwucHJvdG9jb2w9PT1uLnByb3RvY29sJiZsLmhvc3Q9PT1uLmhvc3R9fSgpOmZ1bmN0aW9uKCl7cmV0dXJuIGZ1bmN0aW9uKCl7cmV0dXJuITB9fSgpO2Z1bmN0aW9uIGJlKHIpe3RoaXMubWVzc2FnZT1yfWJlLnByb3RvdHlwZS50b1N0cmluZz1mdW5jdGlvbigpe3JldHVybiJDYW5jZWwiKyh0aGlzLm1lc3NhZ2U/IjogIit0aGlzLm1lc3NhZ2U6IiIpfSxiZS5wcm90b3R5cGUuX19DQU5DRUxfXz0hMDt2YXIgZWU9YmUsdGU9UCx6dD1VdCxHdD1CdCxKdD1rZSxRdD1XdCxYdD1qdCxLdD1WdCxUZT1EZSxZdD1uZSxadD1lZSxXZT1mdW5jdGlvbihlKXtyZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24obixzKXt2YXIgYT1lLmRhdGEsZj1lLmhlYWRlcnMsbD1lLnJlc3BvbnNlVHlwZSxoO2Z1bmN0aW9uIGcoKXtlLmNhbmNlbFRva2VuJiZlLmNhbmNlbFRva2VuLnVuc3Vic2NyaWJlKGgpLGUuc2lnbmFsJiZlLnNpZ25hbC5yZW1vdmVFdmVudExpc3RlbmVyKCJhYm9ydCIsaCl9dGUuaXNGb3JtRGF0YShhKSYmZGVsZXRlIGZbIkNvbnRlbnQtVHlwZSJdO3ZhciBvPW5ldyBYTUxIdHRwUmVxdWVzdDtpZihlLmF1dGgpe3ZhciBwPWUuYXV0aC51c2VybmFtZXx8IiIsUj1lLmF1dGgucGFzc3dvcmQ/dW5lc2NhcGUoZW5jb2RlVVJJQ29tcG9uZW50KGUuYXV0aC5wYXNzd29yZCkpOiIiO2YuQXV0aG9yaXphdGlvbj0iQmFzaWMgIitidG9hKHArIjoiK1IpfXZhciB5PVF0KGUuYmFzZVVSTCxlLnVybCk7by5vcGVuKGUubWV0aG9kLnRvVXBwZXJDYXNlKCksSnQoeSxlLnBhcmFtcyxlLnBhcmFtc1NlcmlhbGl6ZXIpLCEwKSxvLnRpbWVvdXQ9ZS50aW1lb3V0O2Z1bmN0aW9uIFMoKXtpZighIW8pe3ZhciBUPSJnZXRBbGxSZXNwb25zZUhlYWRlcnMiaW4gbz9YdChvLmdldEFsbFJlc3BvbnNlSGVhZGVycygpKTpudWxsLE09IWx8fGw9PT0idGV4dCJ8fGw9PT0ianNvbiI/by5yZXNwb25zZVRleHQ6by5yZXNwb25zZSxBPXtkYXRhOk0sc3RhdHVzOm8uc3RhdHVzLHN0YXR1c1RleHQ6by5zdGF0dXNUZXh0LGhlYWRlcnM6VCxjb25maWc6ZSxyZXF1ZXN0Om99O3p0KGZ1bmN0aW9uKHope24oeiksZygpfSxmdW5jdGlvbih6KXtzKHopLGcoKX0sQSksbz1udWxsfX1pZigib25sb2FkZW5kImluIG8/by5vbmxvYWRlbmQ9UzpvLm9ucmVhZHlzdGF0ZWNoYW5nZT1mdW5jdGlvbigpeyFvfHxvLnJlYWR5U3RhdGUhPT00fHxvLnN0YXR1cz09PTAmJiEoby5yZXNwb25zZVVSTCYmby5yZXNwb25zZVVSTC5pbmRleE9mKCJmaWxlOiIpPT09MCl8fHNldFRpbWVvdXQoUyl9LG8ub25hYm9ydD1mdW5jdGlvbigpeyFvfHwocyhUZSgiUmVxdWVzdCBhYm9ydGVkIixlLCJFQ09OTkFCT1JURUQiLG8pKSxvPW51bGwpfSxvLm9uZXJyb3I9ZnVuY3Rpb24oKXtzKFRlKCJOZXR3b3JrIEVycm9yIixlLG51bGwsbykpLG89bnVsbH0sby5vbnRpbWVvdXQ9ZnVuY3Rpb24oKXt2YXIgTT1lLnRpbWVvdXQ/InRpbWVvdXQgb2YgIitlLnRpbWVvdXQrIm1zIGV4Y2VlZGVkIjoidGltZW91dCBleGNlZWRlZCIsQT1lLnRyYW5zaXRpb25hbHx8WXQudHJhbnNpdGlvbmFsO2UudGltZW91dEVycm9yTWVzc2FnZSYmKE09ZS50aW1lb3V0RXJyb3JNZXNzYWdlKSxzKFRlKE0sZSxBLmNsYXJpZnlUaW1lb3V0RXJyb3I/IkVUSU1FRE9VVCI6IkVDT05OQUJPUlRFRCIsbykpLG89bnVsbH0sdGUuaXNTdGFuZGFyZEJyb3dzZXJFbnYoKSl7dmFyIEk9KGUud2l0aENyZWRlbnRpYWxzfHxLdCh5KSkmJmUueHNyZkNvb2tpZU5hbWU/R3QucmVhZChlLnhzcmZDb29raWVOYW1lKTp2b2lkIDA7SSYmKGZbZS54c3JmSGVhZGVyTmFtZV09SSl9InNldFJlcXVlc3RIZWFkZXIiaW4gbyYmdGUuZm9yRWFjaChmLGZ1bmN0aW9uKE0sQSl7dHlwZW9mIGE9PSJ1bmRlZmluZWQiJiZBLnRvTG93ZXJDYXNlKCk9PT0iY29udGVudC10eXBlIj9kZWxldGUgZltBXTpvLnNldFJlcXVlc3RIZWFkZXIoQSxNKX0pLHRlLmlzVW5kZWZpbmVkKGUud2l0aENyZWRlbnRpYWxzKXx8KG8ud2l0aENyZWRlbnRpYWxzPSEhZS53aXRoQ3JlZGVudGlhbHMpLGwmJmwhPT0ianNvbiImJihvLnJlc3BvbnNlVHlwZT1lLnJlc3BvbnNlVHlwZSksdHlwZW9mIGUub25Eb3dubG9hZFByb2dyZXNzPT0iZnVuY3Rpb24iJiZvLmFkZEV2ZW50TGlzdGVuZXIoInByb2dyZXNzIixlLm9uRG93bmxvYWRQcm9ncmVzcyksdHlwZW9mIGUub25VcGxvYWRQcm9ncmVzcz09ImZ1bmN0aW9uIiYmby51cGxvYWQmJm8udXBsb2FkLmFkZEV2ZW50TGlzdGVuZXIoInByb2dyZXNzIixlLm9uVXBsb2FkUHJvZ3Jlc3MpLChlLmNhbmNlbFRva2VufHxlLnNpZ25hbCkmJihoPWZ1bmN0aW9uKFQpeyFvfHwocyghVHx8VCYmVC50eXBlP25ldyBadCgiY2FuY2VsZWQiKTpUKSxvLmFib3J0KCksbz1udWxsKX0sZS5jYW5jZWxUb2tlbiYmZS5jYW5jZWxUb2tlbi5zdWJzY3JpYmUoaCksZS5zaWduYWwmJihlLnNpZ25hbC5hYm9ydGVkP2goKTplLnNpZ25hbC5hZGRFdmVudExpc3RlbmVyKCJhYm9ydCIsaCkpKSxhfHwoYT1udWxsKSxvLnNlbmQoYSl9KX0sdz1QLHFlPXh0LGVyPUhlLHRyPXsiQ29udGVudC1UeXBlIjoiYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkIn07ZnVuY3Rpb24gamUocixlKXshdy5pc1VuZGVmaW5lZChyKSYmdy5pc1VuZGVmaW5lZChyWyJDb250ZW50LVR5cGUiXSkmJihyWyJDb250ZW50LVR5cGUiXT1lKX1mdW5jdGlvbiBycigpe3ZhciByO3JldHVybih0eXBlb2YgWE1MSHR0cFJlcXVlc3QhPSJ1bmRlZmluZWQifHx0eXBlb2YgcHJvY2VzcyE9InVuZGVmaW5lZCImJk9iamVjdC5wcm90b3R5cGUudG9TdHJpbmcuY2FsbChwcm9jZXNzKT09PSJbb2JqZWN0IHByb2Nlc3NdIikmJihyPVdlKSxyfWZ1bmN0aW9uIG5yKHIsZSx0KXtpZih3LmlzU3RyaW5nKHIpKXRyeXtyZXR1cm4oZXx8SlNPTi5wYXJzZSkociksdy50cmltKHIpfWNhdGNoKG4pe2lmKG4ubmFtZSE9PSJTeW50YXhFcnJvciIpdGhyb3cgbn1yZXR1cm4odHx8SlNPTi5zdHJpbmdpZnkpKHIpfXZhciByZT17dHJhbnNpdGlvbmFsOntzaWxlbnRKU09OUGFyc2luZzohMCxmb3JjZWRKU09OUGFyc2luZzohMCxjbGFyaWZ5VGltZW91dEVycm9yOiExfSxhZGFwdGVyOnJyKCksdHJhbnNmb3JtUmVxdWVzdDpbZnVuY3Rpb24oZSx0KXtyZXR1cm4gcWUodCwiQWNjZXB0IikscWUodCwiQ29udGVudC1UeXBlIiksdy5pc0Zvcm1EYXRhKGUpfHx3LmlzQXJyYXlCdWZmZXIoZSl8fHcuaXNCdWZmZXIoZSl8fHcuaXNTdHJlYW0oZSl8fHcuaXNGaWxlKGUpfHx3LmlzQmxvYihlKT9lOncuaXNBcnJheUJ1ZmZlclZpZXcoZSk/ZS5idWZmZXI6dy5pc1VSTFNlYXJjaFBhcmFtcyhlKT8oamUodCwiYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkO2NoYXJzZXQ9dXRmLTgiKSxlLnRvU3RyaW5nKCkpOncuaXNPYmplY3QoZSl8fHQmJnRbIkNvbnRlbnQtVHlwZSJdPT09ImFwcGxpY2F0aW9uL2pzb24iPyhqZSh0LCJhcHBsaWNhdGlvbi9qc29uIiksbnIoZSkpOmV9XSx0cmFuc2Zvcm1SZXNwb25zZTpbZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcy50cmFuc2l0aW9uYWx8fHJlLnRyYW5zaXRpb25hbCxuPXQmJnQuc2lsZW50SlNPTlBhcnNpbmcscz10JiZ0LmZvcmNlZEpTT05QYXJzaW5nLGE9IW4mJnRoaXMucmVzcG9uc2VUeXBlPT09Impzb24iO2lmKGF8fHMmJncuaXNTdHJpbmcoZSkmJmUubGVuZ3RoKXRyeXtyZXR1cm4gSlNPTi5wYXJzZShlKX1jYXRjaChmKXtpZihhKXRocm93IGYubmFtZT09PSJTeW50YXhFcnJvciI/ZXIoZix0aGlzLCJFX0pTT05fUEFSU0UiKTpmfXJldHVybiBlfV0sdGltZW91dDowLHhzcmZDb29raWVOYW1lOiJYU1JGLVRPS0VOIix4c3JmSGVhZGVyTmFtZToiWC1YU1JGLVRPS0VOIixtYXhDb250ZW50TGVuZ3RoOi0xLG1heEJvZHlMZW5ndGg6LTEsdmFsaWRhdGVTdGF0dXM6ZnVuY3Rpb24oZSl7cmV0dXJuIGU+PTIwMCYmZTwzMDB9LGhlYWRlcnM6e2NvbW1vbjp7QWNjZXB0OiJhcHBsaWNhdGlvbi9qc29uLCB0ZXh0L3BsYWluLCAqLyoifX19O3cuZm9yRWFjaChbImRlbGV0ZSIsImdldCIsImhlYWQiXSxmdW5jdGlvbihlKXtyZS5oZWFkZXJzW2VdPXt9fSksdy5mb3JFYWNoKFsicG9zdCIsInB1dCIsInBhdGNoIl0sZnVuY3Rpb24oZSl7cmUuaGVhZGVyc1tlXT13Lm1lcmdlKHRyKX0pO3ZhciBuZT1yZSxzcj1QLGlyPW5lLG9yPWZ1bmN0aW9uKGUsdCxuKXt2YXIgcz10aGlzfHxpcjtyZXR1cm4gc3IuZm9yRWFjaChuLGZ1bmN0aW9uKGYpe2U9Zi5jYWxsKHMsZSx0KX0pLGV9LFZlPWZ1bmN0aW9uKGUpe3JldHVybiEhKGUmJmUuX19DQU5DRUxfXyl9LHplPVAsRWU9b3IsYXI9VmUsdXI9bmUsZnI9ZWU7ZnVuY3Rpb24gUmUocil7aWYoci5jYW5jZWxUb2tlbiYmci5jYW5jZWxUb2tlbi50aHJvd0lmUmVxdWVzdGVkKCksci5zaWduYWwmJnIuc2lnbmFsLmFib3J0ZWQpdGhyb3cgbmV3IGZyKCJjYW5jZWxlZCIpfXZhciBscj1mdW5jdGlvbihlKXtSZShlKSxlLmhlYWRlcnM9ZS5oZWFkZXJzfHx7fSxlLmRhdGE9RWUuY2FsbChlLGUuZGF0YSxlLmhlYWRlcnMsZS50cmFuc2Zvcm1SZXF1ZXN0KSxlLmhlYWRlcnM9emUubWVyZ2UoZS5oZWFkZXJzLmNvbW1vbnx8e30sZS5oZWFkZXJzW2UubWV0aG9kXXx8e30sZS5oZWFkZXJzKSx6ZS5mb3JFYWNoKFsiZGVsZXRlIiwiZ2V0IiwiaGVhZCIsInBvc3QiLCJwdXQiLCJwYXRjaCIsImNvbW1vbiJdLGZ1bmN0aW9uKHMpe2RlbGV0ZSBlLmhlYWRlcnNbc119KTt2YXIgdD1lLmFkYXB0ZXJ8fHVyLmFkYXB0ZXI7cmV0dXJuIHQoZSkudGhlbihmdW5jdGlvbihzKXtyZXR1cm4gUmUoZSkscy5kYXRhPUVlLmNhbGwoZSxzLmRhdGEscy5oZWFkZXJzLGUudHJhbnNmb3JtUmVzcG9uc2UpLHN9LGZ1bmN0aW9uKHMpe3JldHVybiBhcihzKXx8KFJlKGUpLHMmJnMucmVzcG9uc2UmJihzLnJlc3BvbnNlLmRhdGE9RWUuY2FsbChlLHMucmVzcG9uc2UuZGF0YSxzLnJlc3BvbnNlLmhlYWRlcnMsZS50cmFuc2Zvcm1SZXNwb25zZSkpKSxQcm9taXNlLnJlamVjdChzKX0pfSxPPVAsR2U9ZnVuY3Rpb24oZSx0KXt0PXR8fHt9O3ZhciBuPXt9O2Z1bmN0aW9uIHMobyxwKXtyZXR1cm4gTy5pc1BsYWluT2JqZWN0KG8pJiZPLmlzUGxhaW5PYmplY3QocCk/Ty5tZXJnZShvLHApOk8uaXNQbGFpbk9iamVjdChwKT9PLm1lcmdlKHt9LHApOk8uaXNBcnJheShwKT9wLnNsaWNlKCk6cH1mdW5jdGlvbiBhKG8pe2lmKE8uaXNVbmRlZmluZWQodFtvXSkpe2lmKCFPLmlzVW5kZWZpbmVkKGVbb10pKXJldHVybiBzKHZvaWQgMCxlW29dKX1lbHNlIHJldHVybiBzKGVbb10sdFtvXSl9ZnVuY3Rpb24gZihvKXtpZighTy5pc1VuZGVmaW5lZCh0W29dKSlyZXR1cm4gcyh2b2lkIDAsdFtvXSl9ZnVuY3Rpb24gbChvKXtpZihPLmlzVW5kZWZpbmVkKHRbb10pKXtpZighTy5pc1VuZGVmaW5lZChlW29dKSlyZXR1cm4gcyh2b2lkIDAsZVtvXSl9ZWxzZSByZXR1cm4gcyh2b2lkIDAsdFtvXSl9ZnVuY3Rpb24gaChvKXtpZihvIGluIHQpcmV0dXJuIHMoZVtvXSx0W29dKTtpZihvIGluIGUpcmV0dXJuIHModm9pZCAwLGVbb10pfXZhciBnPXt1cmw6ZixtZXRob2Q6ZixkYXRhOmYsYmFzZVVSTDpsLHRyYW5zZm9ybVJlcXVlc3Q6bCx0cmFuc2Zvcm1SZXNwb25zZTpsLHBhcmFtc1NlcmlhbGl6ZXI6bCx0aW1lb3V0OmwsdGltZW91dE1lc3NhZ2U6bCx3aXRoQ3JlZGVudGlhbHM6bCxhZGFwdGVyOmwscmVzcG9uc2VUeXBlOmwseHNyZkNvb2tpZU5hbWU6bCx4c3JmSGVhZGVyTmFtZTpsLG9uVXBsb2FkUHJvZ3Jlc3M6bCxvbkRvd25sb2FkUHJvZ3Jlc3M6bCxkZWNvbXByZXNzOmwsbWF4Q29udGVudExlbmd0aDpsLG1heEJvZHlMZW5ndGg6bCx0cmFuc3BvcnQ6bCxodHRwQWdlbnQ6bCxodHRwc0FnZW50OmwsY2FuY2VsVG9rZW46bCxzb2NrZXRQYXRoOmwscmVzcG9uc2VFbmNvZGluZzpsLHZhbGlkYXRlU3RhdHVzOmh9O3JldHVybiBPLmZvckVhY2goT2JqZWN0LmtleXMoZSkuY29uY2F0KE9iamVjdC5rZXlzKHQpKSxmdW5jdGlvbihwKXt2YXIgUj1nW3BdfHxhLHk9UihwKTtPLmlzVW5kZWZpbmVkKHkpJiZSIT09aHx8KG5bcF09eSl9KSxufSxKZT17dmVyc2lvbjoiMC4yNS4wIn0sY3I9SmUudmVyc2lvbixQZT17fTtbIm9iamVjdCIsImJvb2xlYW4iLCJudW1iZXIiLCJmdW5jdGlvbiIsInN0cmluZyIsInN5bWJvbCJdLmZvckVhY2goZnVuY3Rpb24ocixlKXtQZVtyXT1mdW5jdGlvbihuKXtyZXR1cm4gdHlwZW9mIG49PT1yfHwiYSIrKGU8MT8ibiAiOiIgIikrcn19KTt2YXIgUWU9e307UGUudHJhbnNpdGlvbmFsPWZ1bmN0aW9uKGUsdCxuKXtmdW5jdGlvbiBzKGEsZil7cmV0dXJuIltBeGlvcyB2IitjcisiXSBUcmFuc2l0aW9uYWwgb3B0aW9uICciK2ErIiciK2YrKG4/Ii4gIituOiIiKX1yZXR1cm4gZnVuY3Rpb24oYSxmLGwpe2lmKGU9PT0hMSl0aHJvdyBuZXcgRXJyb3IocyhmLCIgaGFzIGJlZW4gcmVtb3ZlZCIrKHQ/IiBpbiAiK3Q6IiIpKSk7cmV0dXJuIHQmJiFRZVtmXSYmKFFlW2ZdPSEwLGNvbnNvbGUud2FybihzKGYsIiBoYXMgYmVlbiBkZXByZWNhdGVkIHNpbmNlIHYiK3QrIiBhbmQgd2lsbCBiZSByZW1vdmVkIGluIHRoZSBuZWFyIGZ1dHVyZSIpKSksZT9lKGEsZixsKTohMH19O2Z1bmN0aW9uIGhyKHIsZSx0KXtpZih0eXBlb2YgciE9Im9iamVjdCIpdGhyb3cgbmV3IFR5cGVFcnJvcigib3B0aW9ucyBtdXN0IGJlIGFuIG9iamVjdCIpO2Zvcih2YXIgbj1PYmplY3Qua2V5cyhyKSxzPW4ubGVuZ3RoO3MtLSA+MDspe3ZhciBhPW5bc10sZj1lW2FdO2lmKGYpe3ZhciBsPXJbYV0saD1sPT09dm9pZCAwfHxmKGwsYSxyKTtpZihoIT09ITApdGhyb3cgbmV3IFR5cGVFcnJvcigib3B0aW9uICIrYSsiIG11c3QgYmUgIitoKTtjb250aW51ZX1pZih0IT09ITApdGhyb3cgRXJyb3IoIlVua25vd24gb3B0aW9uICIrYSl9fXZhciBkcj17YXNzZXJ0T3B0aW9uczpocix2YWxpZGF0b3JzOlBlfSxYZT1QLHByPWtlLEtlPUF0LFllPWxyLHNlPUdlLFplPWRyLHE9WmUudmFsaWRhdG9ycztmdW5jdGlvbiBHKHIpe3RoaXMuZGVmYXVsdHM9cix0aGlzLmludGVyY2VwdG9ycz17cmVxdWVzdDpuZXcgS2UscmVzcG9uc2U6bmV3IEtlfX1HLnByb3RvdHlwZS5yZXF1ZXN0PWZ1bmN0aW9uKGUsdCl7aWYodHlwZW9mIGU9PSJzdHJpbmciPyh0PXR8fHt9LHQudXJsPWUpOnQ9ZXx8e30sIXQudXJsKXRocm93IG5ldyBFcnJvcigiUHJvdmlkZWQgY29uZmlnIHVybCBpcyBub3QgdmFsaWQiKTt0PXNlKHRoaXMuZGVmYXVsdHMsdCksdC5tZXRob2Q/dC5tZXRob2Q9dC5tZXRob2QudG9Mb3dlckNhc2UoKTp0aGlzLmRlZmF1bHRzLm1ldGhvZD90Lm1ldGhvZD10aGlzLmRlZmF1bHRzLm1ldGhvZC50b0xvd2VyQ2FzZSgpOnQubWV0aG9kPSJnZXQiO3ZhciBuPXQudHJhbnNpdGlvbmFsO24hPT12b2lkIDAmJlplLmFzc2VydE9wdGlvbnMobix7c2lsZW50SlNPTlBhcnNpbmc6cS50cmFuc2l0aW9uYWwocS5ib29sZWFuKSxmb3JjZWRKU09OUGFyc2luZzpxLnRyYW5zaXRpb25hbChxLmJvb2xlYW4pLGNsYXJpZnlUaW1lb3V0RXJyb3I6cS50cmFuc2l0aW9uYWwocS5ib29sZWFuKX0sITEpO3ZhciBzPVtdLGE9ITA7dGhpcy5pbnRlcmNlcHRvcnMucmVxdWVzdC5mb3JFYWNoKGZ1bmN0aW9uKHkpe3R5cGVvZiB5LnJ1bldoZW49PSJmdW5jdGlvbiImJnkucnVuV2hlbih0KT09PSExfHwoYT1hJiZ5LnN5bmNocm9ub3VzLHMudW5zaGlmdCh5LmZ1bGZpbGxlZCx5LnJlamVjdGVkKSl9KTt2YXIgZj1bXTt0aGlzLmludGVyY2VwdG9ycy5yZXNwb25zZS5mb3JFYWNoKGZ1bmN0aW9uKHkpe2YucHVzaCh5LmZ1bGZpbGxlZCx5LnJlamVjdGVkKX0pO3ZhciBsO2lmKCFhKXt2YXIgaD1bWWUsdm9pZCAwXTtmb3IoQXJyYXkucHJvdG90eXBlLnVuc2hpZnQuYXBwbHkoaCxzKSxoPWguY29uY2F0KGYpLGw9UHJvbWlzZS5yZXNvbHZlKHQpO2gubGVuZ3RoOylsPWwudGhlbihoLnNoaWZ0KCksaC5zaGlmdCgpKTtyZXR1cm4gbH1mb3IodmFyIGc9dDtzLmxlbmd0aDspe3ZhciBvPXMuc2hpZnQoKSxwPXMuc2hpZnQoKTt0cnl7Zz1vKGcpfWNhdGNoKFIpe3AoUik7YnJlYWt9fXRyeXtsPVllKGcpfWNhdGNoKFIpe3JldHVybiBQcm9taXNlLnJlamVjdChSKX1mb3IoO2YubGVuZ3RoOylsPWwudGhlbihmLnNoaWZ0KCksZi5zaGlmdCgpKTtyZXR1cm4gbH0sRy5wcm90b3R5cGUuZ2V0VXJpPWZ1bmN0aW9uKGUpe2lmKCFlLnVybCl0aHJvdyBuZXcgRXJyb3IoIlByb3ZpZGVkIGNvbmZpZyB1cmwgaXMgbm90IHZhbGlkIik7cmV0dXJuIGU9c2UodGhpcy5kZWZhdWx0cyxlKSxwcihlLnVybCxlLnBhcmFtcyxlLnBhcmFtc1NlcmlhbGl6ZXIpLnJlcGxhY2UoL15cPy8sIiIpfSxYZS5mb3JFYWNoKFsiZGVsZXRlIiwiZ2V0IiwiaGVhZCIsIm9wdGlvbnMiXSxmdW5jdGlvbihlKXtHLnByb3RvdHlwZVtlXT1mdW5jdGlvbih0LG4pe3JldHVybiB0aGlzLnJlcXVlc3Qoc2Uobnx8e30se21ldGhvZDplLHVybDp0LGRhdGE6KG58fHt9KS5kYXRhfSkpfX0pLFhlLmZvckVhY2goWyJwb3N0IiwicHV0IiwicGF0Y2giXSxmdW5jdGlvbihlKXtHLnByb3RvdHlwZVtlXT1mdW5jdGlvbih0LG4scyl7cmV0dXJuIHRoaXMucmVxdWVzdChzZShzfHx7fSx7bWV0aG9kOmUsdXJsOnQsZGF0YTpufSkpfX0pO3ZhciBtcj1HLF9yPWVlO2Z1bmN0aW9uIGoocil7aWYodHlwZW9mIHIhPSJmdW5jdGlvbiIpdGhyb3cgbmV3IFR5cGVFcnJvcigiZXhlY3V0b3IgbXVzdCBiZSBhIGZ1bmN0aW9uLiIpO3ZhciBlO3RoaXMucHJvbWlzZT1uZXcgUHJvbWlzZShmdW5jdGlvbihzKXtlPXN9KTt2YXIgdD10aGlzO3RoaXMucHJvbWlzZS50aGVuKGZ1bmN0aW9uKG4pe2lmKCEhdC5fbGlzdGVuZXJzKXt2YXIgcyxhPXQuX2xpc3RlbmVycy5sZW5ndGg7Zm9yKHM9MDtzPGE7cysrKXQuX2xpc3RlbmVyc1tzXShuKTt0Ll9saXN0ZW5lcnM9bnVsbH19KSx0aGlzLnByb21pc2UudGhlbj1mdW5jdGlvbihuKXt2YXIgcyxhPW5ldyBQcm9taXNlKGZ1bmN0aW9uKGYpe3Quc3Vic2NyaWJlKGYpLHM9Zn0pLnRoZW4obik7cmV0dXJuIGEuY2FuY2VsPWZ1bmN0aW9uKCl7dC51bnN1YnNjcmliZShzKX0sYX0scihmdW5jdGlvbihzKXt0LnJlYXNvbnx8KHQucmVhc29uPW5ldyBfcihzKSxlKHQucmVhc29uKSl9KX1qLnByb3RvdHlwZS50aHJvd0lmUmVxdWVzdGVkPWZ1bmN0aW9uKCl7aWYodGhpcy5yZWFzb24pdGhyb3cgdGhpcy5yZWFzb259LGoucHJvdG90eXBlLnN1YnNjcmliZT1mdW5jdGlvbihlKXtpZih0aGlzLnJlYXNvbil7ZSh0aGlzLnJlYXNvbik7cmV0dXJufXRoaXMuX2xpc3RlbmVycz90aGlzLl9saXN0ZW5lcnMucHVzaChlKTp0aGlzLl9saXN0ZW5lcnM9W2VdfSxqLnByb3RvdHlwZS51bnN1YnNjcmliZT1mdW5jdGlvbihlKXtpZighIXRoaXMuX2xpc3RlbmVycyl7dmFyIHQ9dGhpcy5fbGlzdGVuZXJzLmluZGV4T2YoZSk7dCE9PS0xJiZ0aGlzLl9saXN0ZW5lcnMuc3BsaWNlKHQsMSl9fSxqLnNvdXJjZT1mdW5jdGlvbigpe3ZhciBlLHQ9bmV3IGooZnVuY3Rpb24ocyl7ZT1zfSk7cmV0dXJue3Rva2VuOnQsY2FuY2VsOmV9fTt2YXIgZ3I9aix2cj1mdW5jdGlvbihlKXtyZXR1cm4gZnVuY3Rpb24obil7cmV0dXJuIGUuYXBwbHkobnVsbCxuKX19LHlyPVAsd3I9ZnVuY3Rpb24oZSl7cmV0dXJuIHlyLmlzT2JqZWN0KGUpJiZlLmlzQXhpb3NFcnJvcj09PSEwfSxldD1QLENyPXhlLGllPW1yLGJyPUdlLFRyPW5lO2Z1bmN0aW9uIHR0KHIpe3ZhciBlPW5ldyBpZShyKSx0PUNyKGllLnByb3RvdHlwZS5yZXF1ZXN0LGUpO3JldHVybiBldC5leHRlbmQodCxpZS5wcm90b3R5cGUsZSksZXQuZXh0ZW5kKHQsZSksdC5jcmVhdGU9ZnVuY3Rpb24ocyl7cmV0dXJuIHR0KGJyKHIscykpfSx0fXZhciBOPXR0KFRyKTtOLkF4aW9zPWllLE4uQ2FuY2VsPWVlLE4uQ2FuY2VsVG9rZW49Z3IsTi5pc0NhbmNlbD1WZSxOLlZFUlNJT049SmUudmVyc2lvbixOLmFsbD1mdW5jdGlvbihlKXtyZXR1cm4gUHJvbWlzZS5hbGwoZSl9LE4uc3ByZWFkPXZyLE4uaXNBeGlvc0Vycm9yPXdyLF9lLmV4cG9ydHM9TixfZS5leHBvcnRzLmRlZmF1bHQ9Tjt2YXIgRXI9X2UuZXhwb3J0cztjbGFzcyBke2lzRW1wdHkoKXtyZXR1cm4hdGhpcy5pc1ByZXNlbnQoKX1zdGF0aWMgb2YoZSl7aWYoZSE9bnVsbClyZXR1cm4gbmV3IHJ0KGUpO3Rocm93IG5ldyBUeXBlRXJyb3IoIlRoZSBwYXNzZWQgdmFsdWUgd2FzIG51bGwgb3IgdW5kZWZpbmVkLiIpfXN0YXRpYyBvZk5vbk51bGwoZSl7cmV0dXJuIGQub2YoZSl9c3RhdGljIG9mTnVsbGFibGUoZSl7cmV0dXJuIGUhPW51bGw/bmV3IHJ0KGUpOm5ldyBudH1zdGF0aWMgZW1wdHkoKXtyZXR1cm4gbmV3IG50fXN0YXRpYyBmcm9tKGUpe3N3aXRjaChlLmtpbmQpe2Nhc2UicHJlc2VudCI6cmV0dXJuIGQub2YoZS52YWx1ZSk7Y2FzZSJlbXB0eSI6cmV0dXJuIGQuZW1wdHkoKTtkZWZhdWx0OnRocm93IG5ldyBUeXBlRXJyb3IoIlRoZSBwYXNzZWQgdmFsdWUgd2FzIG5vdCBhbiBPcHRpb24gdHlwZS4iKX19fWNsYXNzIHJ0IGV4dGVuZHMgZHtjb25zdHJ1Y3RvcihlKXtzdXBlcigpO3RoaXMucGF5bG9hZD1lfWlzUHJlc2VudCgpe3JldHVybiEwfWdldCgpe3JldHVybiB0aGlzLnBheWxvYWR9aWZQcmVzZW50KGUpe2UodGhpcy5wYXlsb2FkKX1pZlByZXNlbnRPckVsc2UoZSx0KXtlKHRoaXMucGF5bG9hZCl9ZmlsdGVyKGUpe3JldHVybiBlKHRoaXMucGF5bG9hZCk/dGhpczpkLmVtcHR5KCl9bWFwKGUpe2NvbnN0IHQ9ZSh0aGlzLnBheWxvYWQpO3JldHVybiBkLm9mTnVsbGFibGUodCl9ZmxhdE1hcChlKXtyZXR1cm4gZSh0aGlzLnBheWxvYWQpfW9yKGUpe3JldHVybiB0aGlzfW9yRWxzZShlKXtyZXR1cm4gdGhpcy5wYXlsb2FkfW9yRWxzZUdldChlKXtyZXR1cm4gdGhpcy5wYXlsb2FkfW9yRWxzZVRocm93KGUpe3JldHVybiB0aGlzLnBheWxvYWR9b3JOdWxsKCl7cmV0dXJuIHRoaXMucGF5bG9hZH1vclVuZGVmaW5lZCgpe3JldHVybiB0aGlzLnBheWxvYWR9dG9PcHRpb24oKXtyZXR1cm57a2luZDoicHJlc2VudCIsdmFsdWU6dGhpcy5wYXlsb2FkfX1tYXRjaGVzKGUpe3JldHVybiBlLnByZXNlbnQodGhpcy5wYXlsb2FkKX10b0pTT04oZSl7cmV0dXJuIHRoaXMucGF5bG9hZH19Y2xhc3MgbnQgZXh0ZW5kcyBke2lzUHJlc2VudCgpe3JldHVybiExfWNvbnN0cnVjdG9yKCl7c3VwZXIoKX1nZXQoKXt0aHJvdyBuZXcgVHlwZUVycm9yKCJUaGUgb3B0aW9uYWwgaXMgbm90IHByZXNlbnQuIil9aWZQcmVzZW50KGUpe31pZlByZXNlbnRPckVsc2UoZSx0KXt0KCl9ZmlsdGVyKGUpe3JldHVybiB0aGlzfW1hcChlKXtyZXR1cm4gZC5lbXB0eSgpfWZsYXRNYXAoZSl7cmV0dXJuIGQuZW1wdHkoKX1vcihlKXtyZXR1cm4gZSgpfW9yRWxzZShlKXtyZXR1cm4gZX1vckVsc2VHZXQoZSl7cmV0dXJuIHRoaXMub3JFbHNlKGUoKSl9b3JFbHNlVGhyb3coZSl7dGhyb3cgZSgpfW9yTnVsbCgpe3JldHVybiBudWxsfW9yVW5kZWZpbmVkKCl7fXRvT3B0aW9uKCl7cmV0dXJue2tpbmQ6ImVtcHR5In19bWF0Y2hlcyhlKXtyZXR1cm4gZS5lbXB0eSgpfXRvSlNPTihlKXtyZXR1cm4gbnVsbH19Y29uc3QgUnI9IjEuMC4xIjtjbGFzcyBMe2NvbnN0cnVjdG9yKCl7dGhpcy5fcmVwb3J0PXthY3Rpb246ZC5lbXB0eSgpLGFwcGxpY2F0aW9uSWQ6ZC5vZk51bGxhYmxlKG51bGwpLHRpbWVzdGFtcDpEYXRlLm5vdygpLGZwczpkLmVtcHR5KCksZnJhbWVzVHJhbnNmb3JtZWQ6ZC5lbXB0eSgpLGd1aWQ6ZC5lbXB0eSgpLGhpZ2hlc3RGcmFtZVRyYW5zZm9ybUNwdTpkLmVtcHR5KCksbWVzc2FnZTpkLmVtcHR5KCksc291cmNlOmQub2ZOdWxsYWJsZShudWxsKSx0cmFuc2Zvcm1lZEZwczpkLmVtcHR5KCksdHJhbnNmb3JtZXJUeXBlOmQuZW1wdHkoKSx2YXJpYXRpb246ZC5lbXB0eSgpLHZpZGVvSGVpZ2h0OmQuZW1wdHkoKSx2aWRlb1dpZHRoOmQuZW1wdHkoKSx2ZXJzaW9uOlJyfX1hY3Rpb24oZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5hY3Rpb249ZC5vZk51bGxhYmxlKGUpLHRoaXN9ZnJhbWVzVHJhbnNmb3JtZWQoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5mcmFtZXNUcmFuc2Zvcm1lZD1kLm9mTnVsbGFibGUoZSksdGhpc31mcHMoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC5mcHM9ZC5vZk51bGxhYmxlKGUpLHRoaXN9Z3VpZChlKXtyZXR1cm4gdGhpcy5fcmVwb3J0Lmd1aWQ9ZC5vZk51bGxhYmxlKGUpLHRoaXN9bWVzc2FnZShlKXtyZXR1cm4gdGhpcy5fcmVwb3J0Lm1lc3NhZ2U9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dHJhbnNmb3JtZWRGcHMoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC50cmFuc2Zvcm1lZEZwcz1kLm9mTnVsbGFibGUoZSksdGhpc310cmFuc2Zvcm1lclR5cGUoZSl7cmV0dXJuIHRoaXMuX3JlcG9ydC50cmFuc2Zvcm1lclR5cGU9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dmFyaWF0aW9uKGUpe3JldHVybiB0aGlzLl9yZXBvcnQudmFyaWF0aW9uPWQub2ZOdWxsYWJsZShlKSx0aGlzfXZpZGVvSGVpZ2h0KGUpe3JldHVybiB0aGlzLl9yZXBvcnQudmlkZW9IZWlnaHQ9ZC5vZk51bGxhYmxlKGUpLHRoaXN9dmlkZW9XaWR0aChlKXtyZXR1cm4gdGhpcy5fcmVwb3J0LnZpZGVvV2lkdGg9ZC5vZk51bGxhYmxlKGUpLHRoaXN9YnVpbGQoKXtyZXR1cm4gdGhpcy5fcmVwb3J0fX1jb25zdCBQcj1yPT5KU09OLnN0cmluZ2lmeShyLChlLHQpPT57aWYodCE9PW51bGwpcmV0dXJuIHR9KTtjbGFzcyBVe3N0YXRpYyByZXBvcnQoZSl7cmV0dXJuIG5ldyBQcm9taXNlKCh0LG4pPT57bGV0IHM9RXIuY3JlYXRlKCksYT17dGltZW91dDoxZTQsdGltZW91dEVycm9yTWVzc2FnZToiUmVxdWVzdCB0aW1lb3V0IixoZWFkZXJzOnsiQ29udGVudC1UeXBlIjoiYXBwbGljYXRpb24vanNvbiJ9fTtjb25zdCBmPSJodHRwczovL2hsZy5kZXYudG9rYm94LmNvbS9kZXYvbG9nZ2luZy92Y3Bfd2VicnRjIjtzLnBvc3QoZixQcihlKSxhKS50aGVuKGw9Pntjb25zb2xlLmxvZyhsKSx0KCJzdWNjZXNzIil9KS5jYXRjaChsPT57Y29uc29sZS5sb2cobCksbihsKX0pfSl9fXZhciBvZSxPcj1uZXcgVWludDhBcnJheSgxNik7ZnVuY3Rpb24gU3IoKXtpZighb2UmJihvZT10eXBlb2YgY3J5cHRvIT0idW5kZWZpbmVkIiYmY3J5cHRvLmdldFJhbmRvbVZhbHVlcyYmY3J5cHRvLmdldFJhbmRvbVZhbHVlcy5iaW5kKGNyeXB0byl8fHR5cGVvZiBtc0NyeXB0byE9InVuZGVmaW5lZCImJnR5cGVvZiBtc0NyeXB0by5nZXRSYW5kb21WYWx1ZXM9PSJmdW5jdGlvbiImJm1zQ3J5cHRvLmdldFJhbmRvbVZhbHVlcy5iaW5kKG1zQ3J5cHRvKSwhb2UpKXRocm93IG5ldyBFcnJvcigiY3J5cHRvLmdldFJhbmRvbVZhbHVlcygpIG5vdCBzdXBwb3J0ZWQuIFNlZSBodHRwczovL2dpdGh1Yi5jb20vdXVpZGpzL3V1aWQjZ2V0cmFuZG9tdmFsdWVzLW5vdC1zdXBwb3J0ZWQiKTtyZXR1cm4gb2UoT3IpfXZhciBJcj0vXig/OlswLTlhLWZdezh9LVswLTlhLWZdezR9LVsxLTVdWzAtOWEtZl17M30tWzg5YWJdWzAtOWEtZl17M30tWzAtOWEtZl17MTJ9fDAwMDAwMDAwLTAwMDAtMDAwMC0wMDAwLTAwMDAwMDAwMDAwMCkkL2k7ZnVuY3Rpb24gTXIocil7cmV0dXJuIHR5cGVvZiByPT0ic3RyaW5nIiYmSXIudGVzdChyKX1mb3IodmFyIEM9W10sT2U9MDtPZTwyNTY7KytPZSlDLnB1c2goKE9lKzI1NikudG9TdHJpbmcoMTYpLnN1YnN0cigxKSk7ZnVuY3Rpb24gQXIocil7dmFyIGU9YXJndW1lbnRzLmxlbmd0aD4xJiZhcmd1bWVudHNbMV0hPT12b2lkIDA/YXJndW1lbnRzWzFdOjAsdD0oQ1tyW2UrMF1dK0NbcltlKzFdXStDW3JbZSsyXV0rQ1tyW2UrM11dKyItIitDW3JbZSs0XV0rQ1tyW2UrNV1dKyItIitDW3JbZSs2XV0rQ1tyW2UrN11dKyItIitDW3JbZSs4XV0rQ1tyW2UrOV1dKyItIitDW3JbZSsxMF1dK0NbcltlKzExXV0rQ1tyW2UrMTJdXStDW3JbZSsxM11dK0NbcltlKzE0XV0rQ1tyW2UrMTVdXSkudG9Mb3dlckNhc2UoKTtpZighTXIodCkpdGhyb3cgVHlwZUVycm9yKCJTdHJpbmdpZmllZCBVVUlEIGlzIGludmFsaWQiKTtyZXR1cm4gdH1mdW5jdGlvbiBzdChyLGUsdCl7cj1yfHx7fTt2YXIgbj1yLnJhbmRvbXx8KHIucm5nfHxTcikoKTtpZihuWzZdPW5bNl0mMTV8NjQsbls4XT1uWzhdJjYzfDEyOCxlKXt0PXR8fDA7Zm9yKHZhciBzPTA7czwxNjsrK3MpZVt0K3NdPW5bc107cmV0dXJuIGV9cmV0dXJuIEFyKG4pfWNvbnN0IEZyPTUwMDtjbGFzcyB4cntjb25zdHJ1Y3RvcihlKXt0aGlzLnV1aWRfPXN0KCksdGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF89MCx0aGlzLnRyYW5zZm9ybWVyXz1lLHRoaXMuc2hvdWxkU3RvcF89ITEsdGhpcy5pc0ZsYXNoZWRfPSExLHRoaXMuZnJhbWVzRnJvbVNvdXJjZV89MCx0aGlzLmZwc189MCx0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnRTdGFydFRpbWVzdGFtcF89MCx0aGlzLnZpZGVvSGVpZ2h0Xz0wLHRoaXMudmlkZW9XaWR0aF89MCx0aGlzLnRyYW5zZm9ybWVyVHlwZV89IkN1c3RvbSIsImdldFRyYW5zZm9ybWVyVHlwZSJpbiBlJiYodGhpcy50cmFuc2Zvcm1lclR5cGVfPWUuZ2V0VHJhbnNmb3JtZXJUeXBlKCkpO2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhVHJhbnNmb3JtZXIiKS5ndWlkKHRoaXMudXVpZF8pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZhcmlhdGlvbigiQ3JlYXRlIikuYnVpbGQoKTtVLnJlcG9ydCh0KX1hc3luYyBzdGFydChlKXtpZih0aGlzLnRyYW5zZm9ybWVyXyYmdHlwZW9mIHRoaXMudHJhbnNmb3JtZXJfLnN0YXJ0PT0iZnVuY3Rpb24iKXRyeXthd2FpdCB0aGlzLnRyYW5zZm9ybWVyXy5zdGFydChlKX1jYXRjaHtjb25zdCBuPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX3N0YXJ0KS50cmFuc2Zvcm1lclR5cGUodGhpcy50cmFuc2Zvcm1lclR5cGVfKS52YXJpYXRpb24oIkVycm9yIikuYnVpbGQoKTtVLnJlcG9ydChuKX19YXN5bmMgdHJhbnNmb3JtKGUsdCl7dmFyIG4scyxhLGY7aWYodGhpcy5tZWRpYVRyYW5zZm9ybWVyUW9zUmVwb3J0U3RhcnRUaW1lc3RhbXBfPT09MCYmKHRoaXMubWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydFN0YXJ0VGltZXN0YW1wXz1EYXRlLm5vdygpKSx0aGlzLnZpZGVvSGVpZ2h0Xz0obj1lPT1udWxsP3ZvaWQgMDplLmRpc3BsYXlIZWlnaHQpIT1udWxsP246MCx0aGlzLnZpZGVvV2lkdGhfPShzPWU9PW51bGw/dm9pZCAwOmUuZGlzcGxheVdpZHRoKSE9bnVsbD9zOjAsKyt0aGlzLmZyYW1lc0Zyb21Tb3VyY2VfLHRoaXMudHJhbnNmb3JtZXJfKWlmKHRoaXMuc2hvdWxkU3RvcF8pZS5jbG9zZSgpLHRoaXMuZmx1c2godCksdC50ZXJtaW5hdGUoKTtlbHNlIHRyeXthd2FpdCgoZj0oYT10aGlzLnRyYW5zZm9ybWVyXykudHJhbnNmb3JtKT09bnVsbD92b2lkIDA6Zi5jYWxsKGEsZSx0KSksKyt0aGlzLmZyYW1lc1RyYW5zZm9ybWVkXyx0aGlzLmZyYW1lc1RyYW5zZm9ybWVkXz09PUZyJiZ0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnQoKX1jYXRjaHtjb25zdCBoPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX3RyYW5zZm9ybSkudHJhbnNmb3JtZXJUeXBlKHRoaXMudHJhbnNmb3JtZXJUeXBlXykudmFyaWF0aW9uKCJFcnJvciIpLmJ1aWxkKCk7VS5yZXBvcnQoaCl9fWFzeW5jIGZsdXNoKGUpe2lmKHRoaXMudHJhbnNmb3JtZXJfJiZ0eXBlb2YgdGhpcy50cmFuc2Zvcm1lcl8uZmx1c2g9PSJmdW5jdGlvbiImJiF0aGlzLmlzRmxhc2hlZF8pe3RoaXMuaXNGbGFzaGVkXz0hMDt0cnl7YXdhaXQgdGhpcy50cmFuc2Zvcm1lcl8uZmx1c2goZSl9Y2F0Y2h7Y29uc3Qgcz1uZXcgTCgpLmFjdGlvbigiTWVkaWFUcmFuc2Zvcm1lciIpLmd1aWQodGhpcy51dWlkXykubWVzc2FnZShELmVycm9ycy50cmFuc2Zvcm1lcl90cmFuc2Zvcm0pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZhcmlhdGlvbigiRXJyb3IiKS5idWlsZCgpO1UucmVwb3J0KHMpfX10aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnQoKTtjb25zdCB0PW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZ3VpZCh0aGlzLnV1aWRfKS50cmFuc2Zvcm1lclR5cGUodGhpcy50cmFuc2Zvcm1lclR5cGVfKS52YXJpYXRpb24oIkRlbGV0ZSIpLmJ1aWxkKCk7VS5yZXBvcnQodCl9c3RvcCgpe2NvbnNvbGUubG9nKCJbUGlwZWxpbmVdIFN0b3Agc3RyZWFtLiIpLHRoaXMuc2hvdWxkU3RvcF89ITB9bWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydCgpe2xldCBlPShEYXRlLm5vdygpLXRoaXMubWVkaWFUcmFuc2Zvcm1lclFvc1JlcG9ydFN0YXJ0VGltZXN0YW1wXykvMWUzLHQ9dGhpcy5mcmFtZXNGcm9tU291cmNlXy9lLG49dGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF8vZTtjb25zdCBzPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVRyYW5zZm9ybWVyIikuZnBzKHQpLnRyYW5zZm9ybWVkRnBzKG4pLmZyYW1lc1RyYW5zZm9ybWVkKHRoaXMuZnJhbWVzVHJhbnNmb3JtZWRfKS5ndWlkKHRoaXMudXVpZF8pLnRyYW5zZm9ybWVyVHlwZSh0aGlzLnRyYW5zZm9ybWVyVHlwZV8pLnZpZGVvSGVpZ2h0KHRoaXMudmlkZW9IZWlnaHRfKS52aWRlb1dpZHRoKHRoaXMudmlkZW9XaWR0aF8pLnZhcmlhdGlvbigiUW9TIikuYnVpbGQoKTtVLnJlcG9ydChzKSx0aGlzLm1lZGlhVHJhbnNmb3JtZXJRb3NSZXBvcnRTdGFydFRpbWVzdGFtcF89MCx0aGlzLmZyYW1lc0Zyb21Tb3VyY2VfPTAsdGhpcy5mcmFtZXNUcmFuc2Zvcm1lZF89MH19Y2xhc3MgTnJ7Y29uc3RydWN0b3IoZSl7dGhpcy50cmFuc2Zvcm1lcnNfPVtdO2ZvcihsZXQgdCBvZiBlKXRoaXMudHJhbnNmb3JtZXJzXy5wdXNoKG5ldyB4cih0KSl9YXN5bmMgc3RhcnQoZSx0KXtpZighdGhpcy50cmFuc2Zvcm1lcnNffHx0aGlzLnRyYW5zZm9ybWVyc18ubGVuZ3RoPT09MCl7Y29uc29sZS5sb2coIltQaXBlbGluZV0gTm8gdHJhbnNmb3JtZXJzLiIpO3JldHVybn10cnl7bGV0IG49ZTtmb3IobGV0IHMgb2YgdGhpcy50cmFuc2Zvcm1lcnNfKWU9ZS5waXBlVGhyb3VnaChuZXcgVHJhbnNmb3JtU3RyZWFtKHMpKTtlLnBpcGVUbyh0KS50aGVuKGFzeW5jKCk9Pntjb25zb2xlLmxvZygiW1BpcGVsaW5lXSBTZXR1cC4iKSxhd2FpdCB0LmFib3J0KCksYXdhaXQgbi5jYW5jZWwoKX0pLmNhdGNoKGFzeW5jIHM9PntlLmNhbmNlbCgpLnRoZW4oKCk9Pntjb25zb2xlLmxvZygiW1BpcGVsaW5lXSBTaHV0dGluZyBkb3duIHN0cmVhbXMgYWZ0ZXIgYWJvcnQuIil9KS5jYXRjaChhPT57Y29uc29sZS5lcnJvcigiW1BpcGVsaW5lXSBFcnJvciBmcm9tIHN0cmVhbSB0cmFuc2Zvcm06IixhKX0pLGF3YWl0IHQuYWJvcnQocyksYXdhaXQgbi5jYW5jZWwocyl9KX1jYXRjaHt0aGlzLmRlc3Ryb3koKTtyZXR1cm59Y29uc29sZS5sb2coIltQaXBlbGluZV0gUGlwZWxpbmUgc3RhcnRlZC4iKX1hc3luYyBkZXN0cm95KCl7Y29uc29sZS5sb2coIltQaXBlbGluZV0gRGVzdHJveWluZyBQaXBlbGluZS4iKTtmb3IobGV0IGUgb2YgdGhpcy50cmFuc2Zvcm1lcnNfKWUuc3RvcCgpfX1jbGFzcyBMcntjb25zdHJ1Y3Rvcigpe3RoaXMudXVpZF89c3QoKTtjb25zdCBlPW5ldyBMKCkuYWN0aW9uKCJNZWRpYVByb2Nlc3NvciIpLmd1aWQodGhpcy51dWlkXykudmFyaWF0aW9uKCJDcmVhdGUiKS5idWlsZCgpO1UucmVwb3J0KGUpfXRyYW5zZm9ybShlLHQpe3JldHVybiB0aGlzLnJlYWRhYmxlXz1lLHRoaXMud3JpdGFibGVfPXQsdGhpcy50cmFuc2Zvcm1JbnRlcm5hbCgpfXRyYW5zZm9ybUludGVybmFsKCl7cmV0dXJuIG5ldyBQcm9taXNlKChlLHQpPT57aWYoIXRoaXMudHJhbnNmb3JtZXJzX3x8dGhpcy50cmFuc2Zvcm1lcnNfLmxlbmd0aD09PTApe2NvbnN0IG49bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQuZXJyb3JzLnRyYW5zZm9ybWVyX25vbmUpLnZhcmlhdGlvbigiRXJyb3IiKS5idWlsZCgpO1UucmVwb3J0KG4pLHQoIltNZWRpYVByb2Nlc3Nvcl0gTmVlZCB0byBzZXQgdHJhbnNmb3JtZXJzLiIpfXRoaXMucGlwZWxpbmVfJiZ0aGlzLnBpcGVsaW5lXy5kZXN0cm95KCksdGhpcy5waXBlbGluZV89bmV3IE5yKHRoaXMudHJhbnNmb3JtZXJzXyksdGhpcy5waXBlbGluZV8uc3RhcnQodGhpcy5yZWFkYWJsZV8sdGhpcy53cml0YWJsZV8pLnRoZW4oKCk9PntlKCl9KS5jYXRjaChuPT57dChuKX0pfSl9c2V0VHJhbnNmb3JtZXJzKGUpe2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS5tZXNzYWdlKEQudXBkYXRlcy50cmFuc2Zvcm1lcl9uZXcpLnZhcmlhdGlvbigiVXBkYXRlIikuYnVpbGQoKTtyZXR1cm4gVS5yZXBvcnQodCksdGhpcy50cmFuc2Zvcm1lcnNfPWUsdGhpcy5yZWFkYWJsZV8mJnRoaXMud3JpdGFibGVfP3RoaXMudHJhbnNmb3JtSW50ZXJuYWwoKTpQcm9taXNlLnJlc29sdmUoKX1kZXN0cm95KCl7cmV0dXJuIG5ldyBQcm9taXNlKGU9Pnt0aGlzLnBpcGVsaW5lXyYmdGhpcy5waXBlbGluZV8uZGVzdHJveSgpO2NvbnN0IHQ9bmV3IEwoKS5hY3Rpb24oIk1lZGlhUHJvY2Vzc29yIikuZ3VpZCh0aGlzLnV1aWRfKS52YXJpYXRpb24oIkRlbGV0ZSIpLmJ1aWxkKCk7VS5yZXBvcnQodCksZSgpfSl9fXZhciBhZTsoZnVuY3Rpb24ocil7ci5Mb3c9IkxvdyIsci5IaWdoPSJIaWdoIn0pKGFlfHwoYWU9e30pKTtjbGFzcyBpdHtjb25zdHJ1Y3RvcihlKXt0aGlzLmJsdXJGaWx0ZXJfPWU9PT1hZS5IaWdoPyJibHVyKDE1cHgpIjoiYmx1cig1cHgpIn1ydW5Qb3N0UHJvY2Vzc2luZyhlLHQpe3RyeXtlLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iZGVzdGluYXRpb24tb3ZlciIsZS5maWx0ZXI9dGhpcy5ibHVyRmlsdGVyXyxlLmRyYXdJbWFnZSh0LDAsMCx0LndpZHRoLHQuaGVpZ2h0KX1jYXRjaChuKXtjb25zb2xlLmxvZygiW0JsdXJQb3N0UHJvY2Vzc10gRmFpbGVkIHRvIGRyYXcgY2FudmFzIixuKX19fWNsYXNzIFVye2NvbnN0cnVjdG9yKGUpe2lmKHRoaXMuaW1hZ2VDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLmltYWdlQ3R4Xz10aGlzLmltYWdlQ2FudmFzXy5nZXRDb250ZXh0KCIyZCIse2FscGhhOiExLHdpbGxSZWFkRnJlcXVlbnRseTohMCxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLmltYWdlQ3R4Xyl0aHJvdyBuZXcgRXJyb3IoIlVuYWJsZSB0byBjcmVhdGUgT2Zmc2NyZWVuQ2FudmFzUmVuZGVyaW5nQ29udGV4dDJEIik7dGhpcy5pbWFnZUJsb2JfPXZvaWQgMCxmZXRjaChlKS50aGVuKHQ9Pnt0LmJsb2IoKS50aGVuKG49Pnt0aGlzLmltYWdlQmxvYl89bn0pLmNhdGNoKG49Pntjb25zb2xlLmVycm9yKCJibG9iIGVycm9yIixuKX0pfSkuY2F0Y2godD0+e2NvbnNvbGUuZXJyb3IoImZldGNoIGVycm9yIix0KX0pfXJ1blBvc3RQcm9jZXNzaW5nKGUsdCl7dHJ5e2lmKHR5cGVvZiB0aGlzLmltYWdlQmxvYl8hPSJ1bmRlZmluZWQiJiYodGhpcy5pbWFnZUNhbnZhc18ud2lkdGghPXQud2lkdGh8fHRoaXMuaW1hZ2VDYW52YXNfLmhlaWdodCE9dC5oZWlnaHQpKXt0aGlzLmltYWdlQ2FudmFzXy53aWR0aD10LndpZHRoLHRoaXMuaW1hZ2VDYW52YXNfLmhlaWdodD10LmhlaWdodDtjb25zdCBuPSJoaWdoIixzPXtyZXNpemVXaWR0aDp0LndpZHRoLHJlc2l6ZUhlaWdodDp0LmhlaWdodCxyZXNpemVRdWFsaXR5Om59LGE9Y3JlYXRlSW1hZ2VCaXRtYXAodGhpcy5pbWFnZUJsb2JfLHMpLnRoZW4oZj0+e3RoaXMuaW1hZ2VDdHhfLmRyYXdJbWFnZShmLDAsMCl9KS5jYXRjaChmPT57Y29uc29sZS5lcnJvcigiZXJyb3IgY3JlYXRlSW1hZ2VCaXRtYXAiLGYpfSl9ZS5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImRlc3RpbmF0aW9uLW92ZXIiLGUuZHJhd0ltYWdlKHRoaXMuaW1hZ2VDYW52YXNfLDAsMCx0aGlzLmltYWdlQ2FudmFzXy53aWR0aCx0aGlzLmltYWdlQ2FudmFzXy5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbVmlydHVhbFBvc3RQcm9jZXNzXSBGYWlsZWQgdG8gZHJhdyBjYW52YXMiLG4pfX19Y2xhc3MgQnJ7Y29uc3RydWN0b3IoZT0xLzApe3RoaXMuY2FwYWNpdHk9ZSx0aGlzLnN0b3JhZ2U9W119ZW5xdWV1ZShlKXtpZih0aGlzLnNpemUoKT09PXRoaXMuY2FwYWNpdHkpdGhyb3cgRXJyb3IoIlF1ZXVlIGhhcyByZWFjaGVkIG1heCBjYXBhY2l0eSwgeW91IGNhbm5vdCBhZGQgbW9yZSBpdGVtcyIpO3RoaXMuc3RvcmFnZS5wdXNoKGUpfWRlcXVldWUoKXtyZXR1cm4gdGhpcy5zdG9yYWdlLnNoaWZ0KCl9c2l6ZSgpe3JldHVybiB0aGlzLnN0b3JhZ2UubGVuZ3RofX1jbGFzcyBrcntjb25zdHJ1Y3RvcihlKXtjb25zb2xlLmxvZygiW1ZpZGVvUG9zdFByb2Nlc3NdIEN0b3IiKSx0aGlzLnF1ZXVlXz1uZXcgQnIsZSYmKHRoaXMuZnJhbWVSZWFkZXJfPWUuZ2V0UmVhZGVyKCksdGhpcy5mcmFtZVJlYWRlcl8ucmVhZCgpLnRoZW4odGhpcy5wcm9jZXNzRnJhbWUuYmluZCh0aGlzKSkuY2F0Y2godD0+Y29uc29sZS5lcnJvcih0KSkpfXByb2Nlc3NGcmFtZShlKXtlLmRvbmV8fCh0aGlzLnF1ZXVlXy5lbnF1ZXVlKGUudmFsdWUpLHRoaXMuZnJhbWVSZWFkZXJfLnJlYWQoKS50aGVuKHRoaXMucHJvY2Vzc0ZyYW1lLmJpbmQodGhpcykpLmNhdGNoKHQ9PmNvbnNvbGUuZXJyb3IodCkpKX1ydW5Qb3N0UHJvY2Vzc2luZyhlLHQpe3RyeXtpZihlLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iZGVzdGluYXRpb24tb3ZlciIsdGhpcy5xdWV1ZV8uc2l6ZSgpPjApe2NvbnN0IG49dGhpcy5xdWV1ZV8uZGVxdWV1ZSgpO2UuZHJhd0ltYWdlKG4sMCwwLHQud2lkdGgsdC5oZWlnaHQpLG4uY2xvc2UoKX1lbHNlIGUuZHJhd0ltYWdlKHQsMCwwLHQud2lkdGgsdC5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbVmlkZW9Qb3N0UHJvY2Vzc10gRmFpbGVkIHRvIGRyYXcgY2FudmFzIixuKX19fWZ1bmN0aW9uIEhyKCl7cmV0dXJuInNlbGZpZV9zZWdtZW50YXRpb25fbGFuZHNjYXBlIn12YXIgdWU9ZnVuY3Rpb24oKXt2YXIgcj10eXBlb2YgZG9jdW1lbnQhPSJ1bmRlZmluZWQiJiZkb2N1bWVudC5jdXJyZW50U2NyaXB0P2RvY3VtZW50LmN1cnJlbnRTY3JpcHQuc3JjOnZvaWQgMDtyZXR1cm4gZnVuY3Rpb24oZSl7ZT1lfHx7fTt2YXIgdD10eXBlb2YgZSE9InVuZGVmaW5lZCI/ZTp7fSxuLHM7dC5yZWFkeT1uZXcgUHJvbWlzZShmdW5jdGlvbihpLHUpe249aSxzPXV9KTt2YXIgYT17fSxmO2ZvcihmIGluIHQpdC5oYXNPd25Qcm9wZXJ0eShmKSYmKGFbZl09dFtmXSk7dmFyIGw9Ii4vdGhpcy5wcm9ncmFtIixoPWZ1bmN0aW9uKGksdSl7dGhyb3cgdX0sZz0hMCxvPSIiO2Z1bmN0aW9uIHAoaSl7cmV0dXJuIHQubG9jYXRlRmlsZT90LmxvY2F0ZUZpbGUoaSxvKTpvK2l9dmFyIFI7dHlwZW9mIGRvY3VtZW50IT0idW5kZWZpbmVkIiYmZG9jdW1lbnQuY3VycmVudFNjcmlwdCYmKG89ZG9jdW1lbnQuY3VycmVudFNjcmlwdC5zcmMpLHImJihvPXIpLG8uaW5kZXhPZigiYmxvYjoiKSE9PTA/bz1vLnN1YnN0cigwLG8ucmVwbGFjZSgvWz8jXS4qLywiIikubGFzdEluZGV4T2YoIi8iKSsxKTpvPSIiO3ZhciB5PXQucHJpbnR8fGNvbnNvbGUubG9nLmJpbmQoY29uc29sZSksUz10LnByaW50RXJyfHxjb25zb2xlLndhcm4uYmluZChjb25zb2xlKTtmb3IoZiBpbiBhKWEuaGFzT3duUHJvcGVydHkoZikmJih0W2ZdPWFbZl0pO2E9bnVsbCx0LmFyZ3VtZW50cyx0LnRoaXNQcm9ncmFtJiYobD10LnRoaXNQcm9ncmFtKSx0LnF1aXQmJihoPXQucXVpdCk7dmFyIEk7dC53YXNtQmluYXJ5JiYoST10Lndhc21CaW5hcnkpO3ZhciBUPXQubm9FeGl0UnVudGltZXx8ITA7dHlwZW9mIFdlYkFzc2VtYmx5IT0ib2JqZWN0IiYmSCgibm8gbmF0aXZlIHdhc20gc3VwcG9ydCBkZXRlY3RlZCIpO3ZhciBNLEE9ITEsZmU9dHlwZW9mIFRleHREZWNvZGVyIT0idW5kZWZpbmVkIj9uZXcgVGV4dERlY29kZXIoInV0ZjgiKTp2b2lkIDA7ZnVuY3Rpb24geihpLHUsYyl7Zm9yKHZhciBtPXUrYyx2PXU7aVt2XSYmISh2Pj1tKTspKyt2O2lmKHYtdT4xNiYmaS5zdWJhcnJheSYmZmUpcmV0dXJuIGZlLmRlY29kZShpLnN1YmFycmF5KHUsdikpO2Zvcih2YXIgYj0iIjt1PHY7KXt2YXIgXz1pW3UrK107aWYoIShfJjEyOCkpe2IrPVN0cmluZy5mcm9tQ2hhckNvZGUoXyk7Y29udGludWV9dmFyIEU9aVt1KytdJjYzO2lmKChfJjIyNCk9PTE5Mil7Yis9U3RyaW5nLmZyb21DaGFyQ29kZSgoXyYzMSk8PDZ8RSk7Y29udGludWV9dmFyIEI9aVt1KytdJjYzO2lmKChfJjI0MCk9PTIyND9fPShfJjE1KTw8MTJ8RTw8NnxCOl89KF8mNyk8PDE4fEU8PDEyfEI8PDZ8aVt1KytdJjYzLF88NjU1MzYpYis9U3RyaW5nLmZyb21DaGFyQ29kZShfKTtlbHNle3ZhciBwdD1fLTY1NTM2O2IrPVN0cmluZy5mcm9tQ2hhckNvZGUoNTUyOTZ8cHQ+PjEwLDU2MzIwfHB0JjEwMjMpfX1yZXR1cm4gYn1mdW5jdGlvbiBsZShpLHUpe3JldHVybiBpP3ooSixpLHUpOiIifWZ1bmN0aW9uIHFyKGksdSxjKXtmb3IodmFyIG09MDttPGkubGVuZ3RoOysrbSljZVt1Kys+PjBdPWkuY2hhckNvZGVBdChtKTtjfHwoY2VbdT4+MF09MCl9ZnVuY3Rpb24ganIoaSx1KXtyZXR1cm4gaSV1PjAmJihpKz11LWkldSksaX12YXIgb3QsY2UsSixGO2Z1bmN0aW9uIGF0KGkpe290PWksdC5IRUFQOD1jZT1uZXcgSW50OEFycmF5KGkpLHQuSEVBUDE2PW5ldyBJbnQxNkFycmF5KGkpLHQuSEVBUDMyPUY9bmV3IEludDMyQXJyYXkoaSksdC5IRUFQVTg9Sj1uZXcgVWludDhBcnJheShpKSx0LkhFQVBVMTY9bmV3IFVpbnQxNkFycmF5KGkpLHQuSEVBUFUzMj1uZXcgVWludDMyQXJyYXkoaSksdC5IRUFQRjMyPW5ldyBGbG9hdDMyQXJyYXkoaSksdC5IRUFQRjY0PW5ldyBGbG9hdDY0QXJyYXkoaSl9dC5JTklUSUFMX01FTU9SWTt2YXIgSWUsdXQ9W10sZnQ9W10sbHQ9W10sVnI9MDtmdW5jdGlvbiB6cigpe3JldHVybiBUfHxWcj4wfWZ1bmN0aW9uIEdyKCl7aWYodC5wcmVSdW4pZm9yKHR5cGVvZiB0LnByZVJ1bj09ImZ1bmN0aW9uIiYmKHQucHJlUnVuPVt0LnByZVJ1bl0pO3QucHJlUnVuLmxlbmd0aDspWHIodC5wcmVSdW4uc2hpZnQoKSk7TWUodXQpfWZ1bmN0aW9uIEpyKCl7TWUoZnQpfWZ1bmN0aW9uIFFyKCl7aWYodC5wb3N0UnVuKWZvcih0eXBlb2YgdC5wb3N0UnVuPT0iZnVuY3Rpb24iJiYodC5wb3N0UnVuPVt0LnBvc3RSdW5dKTt0LnBvc3RSdW4ubGVuZ3RoOylZcih0LnBvc3RSdW4uc2hpZnQoKSk7TWUobHQpfWZ1bmN0aW9uIFhyKGkpe3V0LnVuc2hpZnQoaSl9ZnVuY3Rpb24gS3IoaSl7ZnQudW5zaGlmdChpKX1mdW5jdGlvbiBZcihpKXtsdC51bnNoaWZ0KGkpfXZhciAkPTAsUT1udWxsO2Z1bmN0aW9uIFpyKGkpeyQrKyx0Lm1vbml0b3JSdW5EZXBlbmRlbmNpZXMmJnQubW9uaXRvclJ1bkRlcGVuZGVuY2llcygkKX1mdW5jdGlvbiBlbihpKXtpZigkLS0sdC5tb25pdG9yUnVuRGVwZW5kZW5jaWVzJiZ0Lm1vbml0b3JSdW5EZXBlbmRlbmNpZXMoJCksJD09MCYmUSl7dmFyIHU9UTtRPW51bGwsdSgpfX10LnByZWxvYWRlZEltYWdlcz17fSx0LnByZWxvYWRlZEF1ZGlvcz17fTtmdW5jdGlvbiBIKGkpe3Qub25BYm9ydCYmdC5vbkFib3J0KGkpLGk9IkFib3J0ZWQoIitpKyIpIixTKGkpLEE9ITAsaSs9Ii4gQnVpbGQgd2l0aCAtcyBBU1NFUlRJT05TPTEgZm9yIG1vcmUgaW5mby4iO3ZhciB1PW5ldyBXZWJBc3NlbWJseS5SdW50aW1lRXJyb3IoaSk7dGhyb3cgcyh1KSx1fXZhciB0bj0iZGF0YTphcHBsaWNhdGlvbi9vY3RldC1zdHJlYW07YmFzZTY0LCI7ZnVuY3Rpb24gY3QoaSl7cmV0dXJuIGkuc3RhcnRzV2l0aCh0bil9dmFyIHg7eD0idGZsaXRlLXNpbWQud2FzbSIsY3QoeCl8fCh4PXAoeCkpO2Z1bmN0aW9uIGh0KGkpe3RyeXtpZihpPT14JiZJKXJldHVybiBuZXcgVWludDhBcnJheShJKTtpZighUil0aHJvdyJib3RoIGFzeW5jIGFuZCBzeW5jIGZldGNoaW5nIG9mIHRoZSB3YXNtIGZhaWxlZCJ9Y2F0Y2godSl7SCh1KX19ZnVuY3Rpb24gcm4oKXtyZXR1cm4hSSYmZyYmdHlwZW9mIGZldGNoPT0iZnVuY3Rpb24iP2ZldGNoKHgse2NyZWRlbnRpYWxzOiJzYW1lLW9yaWdpbiJ9KS50aGVuKGZ1bmN0aW9uKGkpe2lmKCFpLm9rKXRocm93ImZhaWxlZCB0byBsb2FkIHdhc20gYmluYXJ5IGZpbGUgYXQgJyIreCsiJyI7cmV0dXJuIGkuYXJyYXlCdWZmZXIoKX0pLmNhdGNoKGZ1bmN0aW9uKCl7cmV0dXJuIGh0KHgpfSk6UHJvbWlzZS5yZXNvbHZlKCkudGhlbihmdW5jdGlvbigpe3JldHVybiBodCh4KX0pfWZ1bmN0aW9uIG5uKCl7dmFyIGk9e2E6Um59O2Z1bmN0aW9uIHUoXyxFKXt2YXIgQj1fLmV4cG9ydHM7dC5hc209QixNPXQuYXNtLnEsYXQoTS5idWZmZXIpLEllPXQuYXNtLkQsS3IodC5hc20uciksZW4oKX1acigpO2Z1bmN0aW9uIGMoXyl7dShfLmluc3RhbmNlKX1mdW5jdGlvbiBtKF8pe3JldHVybiBybigpLnRoZW4oZnVuY3Rpb24oRSl7cmV0dXJuIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlKEUsaSl9KS50aGVuKGZ1bmN0aW9uKEUpe3JldHVybiBFfSkudGhlbihfLGZ1bmN0aW9uKEUpe1MoImZhaWxlZCB0byBhc3luY2hyb25vdXNseSBwcmVwYXJlIHdhc206ICIrRSksSChFKX0pfWZ1bmN0aW9uIHYoKXtyZXR1cm4hSSYmdHlwZW9mIFdlYkFzc2VtYmx5Lmluc3RhbnRpYXRlU3RyZWFtaW5nPT0iZnVuY3Rpb24iJiYhY3QoeCkmJnR5cGVvZiBmZXRjaD09ImZ1bmN0aW9uIj9mZXRjaCh4LHtjcmVkZW50aWFsczoic2FtZS1vcmlnaW4ifSkudGhlbihmdW5jdGlvbihfKXt2YXIgRT1XZWJBc3NlbWJseS5pbnN0YW50aWF0ZVN0cmVhbWluZyhfLGkpO3JldHVybiBFLnRoZW4oYyxmdW5jdGlvbihCKXtyZXR1cm4gUygid2FzbSBzdHJlYW1pbmcgY29tcGlsZSBmYWlsZWQ6ICIrQiksUygiZmFsbGluZyBiYWNrIHRvIEFycmF5QnVmZmVyIGluc3RhbnRpYXRpb24iKSxtKGMpfSl9KTptKGMpfWlmKHQuaW5zdGFudGlhdGVXYXNtKXRyeXt2YXIgYj10Lmluc3RhbnRpYXRlV2FzbShpLHUpO3JldHVybiBifWNhdGNoKF8pe3JldHVybiBTKCJNb2R1bGUuaW5zdGFudGlhdGVXYXNtIGNhbGxiYWNrIGZhaWxlZCB3aXRoIGVycm9yOiAiK18pLCExfXJldHVybiB2KCkuY2F0Y2gocykse319ZnVuY3Rpb24gTWUoaSl7Zm9yKDtpLmxlbmd0aD4wOyl7dmFyIHU9aS5zaGlmdCgpO2lmKHR5cGVvZiB1PT0iZnVuY3Rpb24iKXt1KHQpO2NvbnRpbnVlfXZhciBjPXUuZnVuYzt0eXBlb2YgYz09Im51bWJlciI/dS5hcmc9PT12b2lkIDA/SWUuZ2V0KGMpKCk6SWUuZ2V0KGMpKHUuYXJnKTpjKHUuYXJnPT09dm9pZCAwP251bGw6dS5hcmcpfX1mdW5jdGlvbiBzbihpLHUsYyxtKXtIKCJBc3NlcnRpb24gZmFpbGVkOiAiK2xlKGkpKyIsIGF0OiAiK1t1P2xlKHUpOiJ1bmtub3duIGZpbGVuYW1lIixjLG0/bGUobSk6InVua25vd24gZnVuY3Rpb24iXSl9ZnVuY3Rpb24gb24oaSx1KXtIKCJUbyB1c2UgZGxvcGVuLCB5b3UgbmVlZCB0byB1c2UgRW1zY3JpcHRlbidzIGxpbmtpbmcgc3VwcG9ydCwgc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9lbXNjcmlwdGVuLWNvcmUvZW1zY3JpcHRlbi93aWtpL0xpbmtpbmciKX1mdW5jdGlvbiBhbihpLHUpe0goIlRvIHVzZSBkbG9wZW4sIHlvdSBuZWVkIHRvIHVzZSBFbXNjcmlwdGVuJ3MgbGlua2luZyBzdXBwb3J0LCBzZWUgaHR0cHM6Ly9naXRodWIuY29tL2Vtc2NyaXB0ZW4tY29yZS9lbXNjcmlwdGVuL3dpa2kvTGlua2luZyIpfWZ1bmN0aW9uIHVuKCl7SCgiIil9dmFyIGhlO2hlPWZ1bmN0aW9uKCl7cmV0dXJuIHBlcmZvcm1hbmNlLm5vdygpfTt2YXIgZm49ITA7ZnVuY3Rpb24gbG4oaSl7cmV0dXJuIEZbZHQoKT4+Ml09aSxpfWZ1bmN0aW9uIGNuKGksdSl7dmFyIGM7aWYoaT09PTApYz1EYXRlLm5vdygpO2Vsc2UgaWYoKGk9PT0xfHxpPT09NCkmJmZuKWM9aGUoKTtlbHNlIHJldHVybiBsbigyOCksLTE7cmV0dXJuIEZbdT4+Ml09Yy8xZTN8MCxGW3UrND4+Ml09YyUxZTMqMWUzKjFlM3wwLDB9ZnVuY3Rpb24gaG4oKXtyZXR1cm4gMjE0NzQ4MzY0OH1mdW5jdGlvbiBkbihpLHUsYyl7Si5jb3B5V2l0aGluKGksdSx1K2MpfWZ1bmN0aW9uIHBuKGkpe3RyeXtyZXR1cm4gTS5ncm93KGktb3QuYnl0ZUxlbmd0aCs2NTUzNT4+PjE2KSxhdChNLmJ1ZmZlciksMX1jYXRjaHt9fWZ1bmN0aW9uIG1uKGkpe3ZhciB1PUoubGVuZ3RoO2k9aT4+PjA7dmFyIGM9MjE0NzQ4MzY0ODtpZihpPmMpcmV0dXJuITE7Zm9yKHZhciBtPTE7bTw9NDttKj0yKXt2YXIgdj11KigxKy4yL20pO3Y9TWF0aC5taW4odixpKzEwMDY2MzI5Nik7dmFyIGI9TWF0aC5taW4oYyxqcihNYXRoLm1heChpLHYpLDY1NTM2KSksXz1wbihiKTtpZihfKXJldHVybiEwfXJldHVybiExfWZ1bmN0aW9uIF9uKGkpe2Zvcih2YXIgdT1oZSgpO2hlKCktdTxpOyk7fXZhciBBZT17fTtmdW5jdGlvbiBnbigpe3JldHVybiBsfHwiLi90aGlzLnByb2dyYW0ifWZ1bmN0aW9uIFgoKXtpZighWC5zdHJpbmdzKXt2YXIgaT0odHlwZW9mIG5hdmlnYXRvcj09Im9iamVjdCImJm5hdmlnYXRvci5sYW5ndWFnZXMmJm5hdmlnYXRvci5sYW5ndWFnZXNbMF18fCJDIikucmVwbGFjZSgiLSIsIl8iKSsiLlVURi04Iix1PXtVU0VSOiJ3ZWJfdXNlciIsTE9HTkFNRToid2ViX3VzZXIiLFBBVEg6Ii8iLFBXRDoiLyIsSE9NRToiL2hvbWUvd2ViX3VzZXIiLExBTkc6aSxfOmduKCl9O2Zvcih2YXIgYyBpbiBBZSlBZVtjXT09PXZvaWQgMD9kZWxldGUgdVtjXTp1W2NdPUFlW2NdO3ZhciBtPVtdO2Zvcih2YXIgYyBpbiB1KW0ucHVzaChjKyI9Iit1W2NdKTtYLnN0cmluZ3M9bX1yZXR1cm4gWC5zdHJpbmdzfXZhciBkZT17bWFwcGluZ3M6e30sYnVmZmVyczpbbnVsbCxbXSxbXV0scHJpbnRDaGFyOmZ1bmN0aW9uKGksdSl7dmFyIGM9ZGUuYnVmZmVyc1tpXTt1PT09MHx8dT09PTEwPygoaT09PTE/eTpTKSh6KGMsMCkpLGMubGVuZ3RoPTApOmMucHVzaCh1KX0sdmFyYXJnczp2b2lkIDAsZ2V0OmZ1bmN0aW9uKCl7ZGUudmFyYXJncys9NDt2YXIgaT1GW2RlLnZhcmFyZ3MtND4+Ml07cmV0dXJuIGl9LGdldFN0cjpmdW5jdGlvbihpKXt2YXIgdT1sZShpKTtyZXR1cm4gdX0sZ2V0NjQ6ZnVuY3Rpb24oaSx1KXtyZXR1cm4gaX19O2Z1bmN0aW9uIHZuKGksdSl7dmFyIGM9MDtyZXR1cm4gWCgpLmZvckVhY2goZnVuY3Rpb24obSx2KXt2YXIgYj11K2M7RltpK3YqND4+Ml09YixxcihtLGIpLGMrPW0ubGVuZ3RoKzF9KSwwfWZ1bmN0aW9uIHluKGksdSl7dmFyIGM9WCgpO0ZbaT4+Ml09Yy5sZW5ndGg7dmFyIG09MDtyZXR1cm4gYy5mb3JFYWNoKGZ1bmN0aW9uKHYpe20rPXYubGVuZ3RoKzF9KSxGW3U+PjJdPW0sMH1mdW5jdGlvbiB3bihpKXtPbihpKX1mdW5jdGlvbiBDbihpKXtyZXR1cm4gMH1mdW5jdGlvbiBibihpLHUsYyxtLHYpe31mdW5jdGlvbiBUbihpLHUsYyxtKXtmb3IodmFyIHY9MCxiPTA7YjxjO2IrKyl7Zm9yKHZhciBfPUZbdStiKjg+PjJdLEU9Rlt1KyhiKjgrNCk+PjJdLEI9MDtCPEU7QisrKWRlLnByaW50Q2hhcihpLEpbXytCXSk7dis9RX1yZXR1cm4gRlttPj4yXT12LDB9ZnVuY3Rpb24gRW4oKXtpZih0eXBlb2YgY3J5cHRvPT0ib2JqZWN0IiYmdHlwZW9mIGNyeXB0by5nZXRSYW5kb21WYWx1ZXM9PSJmdW5jdGlvbiIpe3ZhciBpPW5ldyBVaW50OEFycmF5KDEpO3JldHVybiBmdW5jdGlvbigpe3JldHVybiBjcnlwdG8uZ2V0UmFuZG9tVmFsdWVzKGkpLGlbMF19fWVsc2UgcmV0dXJuIGZ1bmN0aW9uKCl7SCgicmFuZG9tRGV2aWNlIil9fWZ1bmN0aW9uIHBlKGksdSl7cGUucmFuZG9tRGV2aWNlfHwocGUucmFuZG9tRGV2aWNlPUVuKCkpO2Zvcih2YXIgYz0wO2M8dTtjKyspY2VbaStjPj4wXT1wZS5yYW5kb21EZXZpY2UoKTtyZXR1cm4gMH12YXIgUm49e2E6c24sZTpvbixkOmFuLGI6dW4sbjpjbixoOmhuLGw6ZG4sbTptbixwOl9uLGY6dm4sZzp5bixqOnduLGk6Q24sazpibixjOlRuLG86cGV9O25uKCksdC5fX193YXNtX2NhbGxfY3RvcnM9ZnVuY3Rpb24oKXtyZXR1cm4odC5fX193YXNtX2NhbGxfY3RvcnM9dC5hc20ucikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRNb2RlbEJ1ZmZlck1lbW9yeU9mZnNldD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRNb2RlbEJ1ZmZlck1lbW9yeU9mZnNldD10LmFzbS5zKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX2dldElucHV0TWVtb3J5T2Zmc2V0PWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2dldElucHV0TWVtb3J5T2Zmc2V0PXQuYXNtLnQpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fZ2V0SW5wdXRIZWlnaHQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0SW5wdXRIZWlnaHQ9dC5hc20udSkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRJbnB1dFdpZHRoPWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2dldElucHV0V2lkdGg9dC5hc20udikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRJbnB1dENoYW5uZWxDb3VudD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRJbnB1dENoYW5uZWxDb3VudD10LmFzbS53KS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX2dldE91dHB1dE1lbW9yeU9mZnNldD1mdW5jdGlvbigpe3JldHVybih0Ll9nZXRPdXRwdXRNZW1vcnlPZmZzZXQ9dC5hc20ueCkuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRPdXRwdXRIZWlnaHQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0SGVpZ2h0PXQuYXNtLnkpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fZ2V0T3V0cHV0V2lkdGg9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0V2lkdGg9dC5hc20ueikuYXBwbHkobnVsbCxhcmd1bWVudHMpfSx0Ll9nZXRPdXRwdXRDaGFubmVsQ291bnQ9ZnVuY3Rpb24oKXtyZXR1cm4odC5fZ2V0T3V0cHV0Q2hhbm5lbENvdW50PXQuYXNtLkEpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sdC5fbG9hZE1vZGVsPWZ1bmN0aW9uKCl7cmV0dXJuKHQuX2xvYWRNb2RlbD10LmFzbS5CKS5hcHBseShudWxsLGFyZ3VtZW50cyl9LHQuX3J1bkluZmVyZW5jZT1mdW5jdGlvbigpe3JldHVybih0Ll9ydW5JbmZlcmVuY2U9dC5hc20uQykuYXBwbHkobnVsbCxhcmd1bWVudHMpfTt2YXIgZHQ9dC5fX19lcnJub19sb2NhdGlvbj1mdW5jdGlvbigpe3JldHVybihkdD10Ll9fX2Vycm5vX2xvY2F0aW9uPXQuYXNtLkUpLmFwcGx5KG51bGwsYXJndW1lbnRzKX0sbWU7ZnVuY3Rpb24gUG4oaSl7dGhpcy5uYW1lPSJFeGl0U3RhdHVzIix0aGlzLm1lc3NhZ2U9IlByb2dyYW0gdGVybWluYXRlZCB3aXRoIGV4aXQoIitpKyIpIix0aGlzLnN0YXR1cz1pfVE9ZnVuY3Rpb24gaSgpe21lfHxGZSgpLG1lfHwoUT1pKX07ZnVuY3Rpb24gRmUoaSl7aWYoJD4wfHwoR3IoKSwkPjApKXJldHVybjtmdW5jdGlvbiB1KCl7bWV8fChtZT0hMCx0LmNhbGxlZFJ1bj0hMCwhQSYmKEpyKCksbih0KSx0Lm9uUnVudGltZUluaXRpYWxpemVkJiZ0Lm9uUnVudGltZUluaXRpYWxpemVkKCksUXIoKSkpfXQuc2V0U3RhdHVzPyh0LnNldFN0YXR1cygiUnVubmluZy4uLiIpLHNldFRpbWVvdXQoZnVuY3Rpb24oKXtzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7dC5zZXRTdGF0dXMoIiIpfSwxKSx1KCl9LDEpKTp1KCl9dC5ydW49RmU7ZnVuY3Rpb24gT24oaSx1KXtTbihpKX1mdW5jdGlvbiBTbihpKXt6cigpfHwodC5vbkV4aXQmJnQub25FeGl0KGkpLEE9ITApLGgoaSxuZXcgUG4oaSkpfWlmKHQucHJlSW5pdClmb3IodHlwZW9mIHQucHJlSW5pdD09ImZ1bmN0aW9uIiYmKHQucHJlSW5pdD1bdC5wcmVJbml0XSk7dC5wcmVJbml0Lmxlbmd0aD4wOyl0LnByZUluaXQucG9wKCkoKTtyZXR1cm4gRmUoKSxlLnJlYWR5fX0oKTt0eXBlb2YgZXhwb3J0cz09Im9iamVjdCImJnR5cGVvZiBtb2R1bGU9PSJvYmplY3QiP21vZHVsZS5leHBvcnRzPXVlOnR5cGVvZiBkZWZpbmU9PSJmdW5jdGlvbiImJmRlZmluZS5hbWQ/ZGVmaW5lKFtdLGZ1bmN0aW9uKCl7cmV0dXJuIHVlfSk6dHlwZW9mIGV4cG9ydHM9PSJvYmplY3QiJiYoZXhwb3J0cy5jcmVhdGVWb25hZ2VURkxpdGVTaW1kTW9kdWxlPXVlKTtjbGFzcyBEcntjb25zdHJ1Y3Rvcigpe3RoaXMuaXNTSU1EU3VwcG9ydGVkXz0hMSx0aGlzLmlzVGhyZWFkc1N1cHBvcnRlZF89ITEsdGhpcy5pbnB1dFdpZHRoXz0wLHRoaXMuaW5wdXRIZWlnaHRfPTAsdGhpcy5pbnB1dENoYW5uZWxDb3VudF89MCx0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xz0wLHRoaXMub3V0cHV0V2lkdGhfPTAsdGhpcy5vdXRwdXRIZWlnaHRfPTAsdGhpcy5vdXRwdXRDaGFubmVsQ291bnRfPTAsdGhpcy5vdXRwdXRNZW1vcnlPZmZzZXRfPTB9fWFzeW5jIGZ1bmN0aW9uICRyKHIpe2xldCBlPW5ldyBEcjt2YXIgdD1uZXcgUHJvbWlzZShmdW5jdGlvbihuLHMpe2FzeW5jIGZ1bmN0aW9uIGEoKXtyZXR1cm4gdWUoe2xvY2F0ZUZpbGU6ZnVuY3Rpb24oaCl7cmV0dXJuIHIraH19KX1hc3luYyBmdW5jdGlvbiBmKGwpe3JldHVybiBuZXcgUHJvbWlzZShhc3luYyhoLGcpPT57Y29uc3Qgbz1sO2lmKHR5cGVvZiBvPT0idW5kZWZpbmVkIilyZXR1cm4gZygiVEZMaXRlIGJhY2tlbmQgdW5hdmFpbGFibGU6IHdhc21TaW1kIik7Y29uc3QgcD1IcigpO2NvbnNvbGUubG9nKCJMb2FkaW5nIHRmbGl0ZSBtb2RlbDoiLHApO2xldCBSPXIrYG1vZGVscy8ke3B9LnRmbGl0ZWA7Y29uc29sZS5sb2coIlRGTGl0ZSBtb2RlbFVybDogIitSKTt0cnl7Y29uc3QgUz1hd2FpdChhd2FpdCBmZXRjaChSKSkuYXJyYXlCdWZmZXIoKTtjb25zb2xlLmxvZygiTW9kZWwgYnVmZmVyIHNpemU6IixTLmJ5dGVMZW5ndGgpO2NvbnN0IEk9by5fZ2V0TW9kZWxCdWZmZXJNZW1vcnlPZmZzZXQoKTtjb25zb2xlLmxvZygiTW9kZWwgYnVmZmVyIG1lbW9yeSBvZmZzZXQ6IixJKSxjb25zb2xlLmxvZygiTG9hZGluZyBtb2RlbCBidWZmZXIuLi4iKSxvLkhFQVBVOC5zZXQobmV3IFVpbnQ4QXJyYXkoUyksSSk7Y29uc3QgVD1vLl9sb2FkTW9kZWwoUy5ieXRlTGVuZ3RoKTtpZihjb25zb2xlLmxvZygiTG9hZCBtb2RlbCByZXN1bHQ6IixUKSxUIT09MCl7ZygiQ2Fubm90IGxvYWQgbW9kZWwiKTtyZXR1cm59ZS5pbnB1dFdpZHRoXz1vLl9nZXRJbnB1dFdpZHRoKCksZS5pbnB1dEhlaWdodF89by5fZ2V0SW5wdXRIZWlnaHQoKSxlLmlucHV0Q2hhbm5lbENvdW50Xz1vLl9nZXRJbnB1dENoYW5uZWxDb3VudCgpLGUuaW5wdXRNZW1vcnlPZmZzZXRfPW8uX2dldElucHV0TWVtb3J5T2Zmc2V0KCksY29uc29sZS5sb2coIklucHV0IG1lbW9yeSBvZmZzZXQ6IixlLmlucHV0TWVtb3J5T2Zmc2V0XyksY29uc29sZS5sb2coIklucHV0IGhlaWdodDoiLGUuaW5wdXRIZWlnaHRfKSxjb25zb2xlLmxvZygiSW5wdXQgd2lkdGg6IixlLmlucHV0V2lkdGhfKSxjb25zb2xlLmxvZygiSW5wdXQgY2hhbm5lbHM6IixlLmlucHV0Q2hhbm5lbENvdW50XyksZS5vdXRwdXRNZW1vcnlPZmZzZXRfPW8uX2dldE91dHB1dE1lbW9yeU9mZnNldCgpLGUub3V0cHV0V2lkdGhfPW8uX2dldE91dHB1dFdpZHRoKCksZS5vdXRwdXRIZWlnaHRfPW8uX2dldE91dHB1dEhlaWdodCgpLGUub3V0cHV0Q2hhbm5lbENvdW50Xz1vLl9nZXRPdXRwdXRDaGFubmVsQ291bnQoKSxjb25zb2xlLmxvZygiT3V0cHV0IG1lbW9yeSBvZmZzZXQ6IixlLm91dHB1dE1lbW9yeU9mZnNldF8pLGNvbnNvbGUubG9nKCJPdXRwdXQgaGVpZ2h0OiIsZS5vdXRwdXRIZWlnaHRfKSxjb25zb2xlLmxvZygiT3V0cHV0IHdpZHRoOiIsZS5vdXRwdXRXaWR0aF8pLGNvbnNvbGUubG9nKCJPdXRwdXQgY2hhbm5lbHM6IixlLm91dHB1dENoYW5uZWxDb3VudF8pLGUuc2VsZWN0ZWRURkxpdGVfPW99Y2F0Y2goeSl7Zyh5KTtyZXR1cm59aCgpfSl9YSgpLnRoZW4obD0+e2lmKGw9PT1udWxsKXRocm93IGNvbnNvbGUuZXJyb3IoImxvYWRURkxpdGUgbnVsbCIpLCJsb2FkVEZMaXRlIG51bGwiO2YobCkudGhlbihoPT57bihlKX0pLmNhdGNoKGg9Pntjb25zb2xlLmVycm9yKCJsb2FkVEZMaXRlTW9kZWwgZXJyb3IiKSxzKCJsb2FkVEZMaXRlTW9kZWwgZXJyb3IiKX0pfSkuY2F0Y2gobD0+e2NvbnNvbGUuZXJyb3IoImxvYWRURkxpdGUgZXJyb3I6IixsKSxzKCJsb2FkVEZMaXRlIGVycm9yOiIrbCl9KX0pO3JldHVybiB0fWNsYXNzIFdye2NvbnN0cnVjdG9yKGUpe3RoaXMuYmx1ckZpbHRlcl89ZT09PWFlLkhpZ2g/ImJsdXIoMTVweCkiOiJibHVyKDVweCkifXJ1blBvc3RQcm9jZXNzaW5nKGUsdCl7dHJ5e2UuZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uPSJzb3VyY2UtaW4iLGUuZmlsdGVyPXRoaXMuYmx1ckZpbHRlcl8sZS5kcmF3SW1hZ2UodCwwLDAsdC53aWR0aCx0LmhlaWdodCksZS5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImRlc3RpbmF0aW9uLW92ZXIiLGUuZmlsdGVyPSJibHVyKDBweCkiLGUuZHJhd0ltYWdlKHQsMCwwLHQud2lkdGgsdC5oZWlnaHQpfWNhdGNoKG4pe2NvbnNvbGUubG9nKCJbQmx1clBvc3RQcm9jZXNzXSBGYWlsZWQgdG8gZHJhdyBjYW52YXMiLG4pfX19Y29uc3QgVj1jbGFzc3tjb25zdHJ1Y3RvcihyKXtpZih0aGlzLmNvbmZpZ189cix0aGlzLmZyYW1lQ291bnRlcl89MCx0aGlzLmxhc3RQcm9jZXNzZWRUaW1lc3RhbXBfPTAsdGhpcy5yZXN1bHRDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLnJlc3VsdEN0eF89dGhpcy5yZXN1bHRDYW52YXNfLmdldENvbnRleHQoIjJkIix7YWxwaGE6ITEsZGVzeW5jaHJvbml6ZWQ6ITB9KSwhdGhpcy5yZXN1bHRDdHhfKXRocm93IG5ldyBFcnJvcigiVW5hYmxlIHRvIGNyZWF0ZSBPZmZzY3JlZW5DYW52YXNSZW5kZXJpbmdDb250ZXh0MkQiKTtpZih0aGlzLnNlZ21lbnRhdGlvbk1hc2tDYW52YXNfPW5ldyBPZmZzY3JlZW5DYW52YXMoMSwxKSx0aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfPXRoaXMuc2VnbWVudGF0aW9uTWFza0NhbnZhc18uZ2V0Q29udGV4dCgiMmQiLHthbHBoYTohMSxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfKXRocm93IG5ldyBFcnJvcigiVW5hYmxlIHRvIGNyZWF0ZSBPZmZzY3JlZW5DYW52YXNSZW5kZXJpbmdDb250ZXh0MkQiKTtpZih0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc189bmV3IE9mZnNjcmVlbkNhbnZhcygxLDEpLHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ3R4Xz10aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18uZ2V0Q29udGV4dCgiMmQiLHthbHBoYTohMSxkZXN5bmNocm9uaXplZDohMH0pLCF0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUN0eF8pdGhyb3cgbmV3IEVycm9yKCJVbmFibGUgdG8gY3JlYXRlIE9mZnNjcmVlbkNhbnZhc1JlbmRlcmluZ0NvbnRleHQyRCIpO3RoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF89MCx0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xz0wLHRoaXMub3V0cHV0TWVtb3J5T2Zmc2V0Xz0wLHIudHJhbnNmb3JtZXJUeXBlPT09IlZpcnR1YWxCYWNrZ3JvdW5kIj8odGhpcy50aW1lckdhcF89Vi5WSVJUVUFMX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgVXIoci5iYWNrZ3JvdW5kQXNzZXRVcmkpKTpyLnRyYW5zZm9ybWVyVHlwZT09PSJCYWNrZ3JvdW5kQmx1ciI/KHRoaXMudGltZXJHYXBfPVYuQkxVUl9QUk9DRVNTX0dBUF9NSUNST19TRUNPTkQsdGhpcy5wb3N0UHJvY2Vzc0ludGVyZmFjZV89bmV3IGl0KHIucmFkaXVzKSk6ci50cmFuc2Zvcm1lclR5cGU9PT0iVmlkZW9CYWNrZ3JvdW5kIj90aGlzLnRpbWVyR2FwXz1WLkJMVVJfUFJPQ0VTU19HQVBfTUlDUk9fU0VDT05EOnIudHJhbnNmb3JtZXJUeXBlPT09IlNpbHVldGVCbHVyIj8odGhpcy50aW1lckdhcF89Vi5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgV3Ioci5yYWRpdXMpKToodGhpcy50aW1lckdhcF89Vi5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORCx0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXz1uZXcgaXQpfWFzeW5jIHN0YXJ0KCl7YXdhaXQgJHIodGhpcy5jb25maWdfLndhc21Bc3NldFVyaVBhdGgpLnRoZW4ocj0+e3ImJih0aGlzLnZvYW5nZVRGTGl0ZUluZm9fPXIsdGhpcy5zZWdtZW50YXRpb25NYXNrQ2FudmFzXy53aWR0aD10aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMuc2VnbWVudGF0aW9uTWFza0NhbnZhc18uaGVpZ2h0PXRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfLHRoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dFdpZHRoXyp0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0SGVpZ2h0Xyx0aGlzLnNlZ21lbnRhdGlvbk1hc2tfPW5ldyBJbWFnZURhdGEodGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dFdpZHRoXyx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0SGVpZ2h0XyksdGhpcy5pbnB1dE1lbW9yeU9mZnNldF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dE1lbW9yeU9mZnNldF8vNCx0aGlzLm91dHB1dE1lbW9yeU9mZnNldF89dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5vdXRwdXRNZW1vcnlPZmZzZXRfLzQpfSkuY2F0Y2gocj0+e2NvbnNvbGUubG9nKCJlcnJvciB1c2VURkxpdGU6IixyKX0pfWFzeW5jIHRyYW5zZm9ybShyLGUpeyh0aGlzLnJlc3VsdENhbnZhc18ud2lkdGghPXIuZGlzcGxheVdpZHRofHx0aGlzLnJlc3VsdENhbnZhc18uaGVpZ2h0IT1yLmRpc3BsYXlIZWlnaHQpJiYodGhpcy5yZXN1bHRDYW52YXNfLndpZHRoPXIuZGlzcGxheVdpZHRoLHRoaXMucmVzdWx0Q2FudmFzXy5oZWlnaHQ9ci5kaXNwbGF5SGVpZ2h0KSwodGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLndpZHRoIT1yLmRpc3BsYXlXaWR0aHx8dGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLmhlaWdodCE9ci5kaXNwbGF5SGVpZ2h0KSYmKHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXy53aWR0aD1yLmRpc3BsYXlXaWR0aCx0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18uaGVpZ2h0PXIuZGlzcGxheUhlaWdodCk7Y29uc3QgdD1yLnRpbWVzdGFtcDtjcmVhdGVJbWFnZUJpdG1hcChyKS50aGVuKG49PntyLmNsb3NlKCksdGhpcy5wcm9jZXNzRnJhbWUoZSxuLHQpfSkuY2F0Y2gobj0+e2NvbnNvbGUuZXJyb3IoImNyZWF0ZUltYWdlQml0bWFwIixuKSxlLmVucXVldWUocil9KX1wcm9jZXNzRnJhbWUocixlLHQpe3QtdGhpcy5sYXN0UHJvY2Vzc2VkVGltZXN0YW1wXz49dGhpcy50aW1lckdhcF8mJih0aGlzLmxhc3RQcm9jZXNzZWRUaW1lc3RhbXBfPXQsdGhpcy5wcm9jZXNzU291cmNlKGUpKSx0aGlzLnJlc3VsdEN0eF8uZ2xvYmFsQ29tcG9zaXRlT3BlcmF0aW9uPSJjb3B5Iix0aGlzLnJlc3VsdEN0eF8uZHJhd0ltYWdlKHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXywwLDAsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLndpZHRoLHRoaXMuc2VnbWVudGF0aW9uTWFza0ZyYW1lQ2FudmFzXy5oZWlnaHQpLHRoaXMucmVzdWx0Q3R4Xy5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249InNvdXJjZS1hdG9wIix0aGlzLnJlc3VsdEN0eF8uZmlsdGVyPSJub25lIix0aGlzLnJlc3VsdEN0eF8uZHJhd0ltYWdlKGUsMCwwLGUud2lkdGgsZS5oZWlnaHQpLHRoaXMucG9zdFByb2Nlc3NJbnRlcmZhY2VfJiZ0aGlzLnBvc3RQcm9jZXNzSW50ZXJmYWNlXy5ydW5Qb3N0UHJvY2Vzc2luZyh0aGlzLnJlc3VsdEN0eF8sZSksci5lbnF1ZXVlKG5ldyBWaWRlb0ZyYW1lKHRoaXMucmVzdWx0Q2FudmFzXyx7dGltZXN0YW1wOnQsYWxwaGE6ImRpc2NhcmQifSkpLHRoaXMuZnJhbWVDb3VudGVyXysrfXByb2Nlc3NTb3VyY2Uocil7dGhpcy5zZWdtZW50YXRpb25NYXNrQ3R4Xy5nbG9iYWxDb21wb3NpdGVPcGVyYXRpb249ImNvcHkiLHRoaXMuc2VnbWVudGF0aW9uTWFza0N0eF8uZHJhd0ltYWdlKHIsMCwwLHIud2lkdGgsci5oZWlnaHQsMCwwLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRXaWR0aF8sdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5pbnB1dEhlaWdodF8pO2xldCBlPXRoaXMuc2VnbWVudGF0aW9uTWFza0N0eF8uZ2V0SW1hZ2VEYXRhKDAsMCx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfKTtmb3IobGV0IHQ9MDt0PHRoaXMuc2VnbWVudGF0aW9uUGl4ZWxDb3VudF87dCsrKXRoaXMudm9hbmdlVEZMaXRlSW5mb18uc2VsZWN0ZWRURkxpdGVfLkhFQVBGMzJbdGhpcy5pbnB1dE1lbW9yeU9mZnNldF8rdCozXT1lLmRhdGFbdCo0XS8yNTUsdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xyt0KjMrMV09ZS5kYXRhW3QqNCsxXS8yNTUsdGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLmlucHV0TWVtb3J5T2Zmc2V0Xyt0KjMrMl09ZS5kYXRhW3QqNCsyXS8yNTU7dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uX3J1bkluZmVyZW5jZSgpO2ZvcihsZXQgdD0wO3Q8dGhpcy5zZWdtZW50YXRpb25QaXhlbENvdW50Xzt0Kyspe2NvbnN0IG49dGhpcy52b2FuZ2VURkxpdGVJbmZvXy5zZWxlY3RlZFRGTGl0ZV8uSEVBUEYzMlt0aGlzLm91dHB1dE1lbW9yeU9mZnNldF8rdF07dGhpcy5zZWdtZW50YXRpb25NYXNrXy5kYXRhW3QqNCszXT0yNTUqbn10aGlzLnNlZ21lbnRhdGlvbk1hc2tDdHhfLnB1dEltYWdlRGF0YSh0aGlzLnNlZ21lbnRhdGlvbk1hc2tfLDAsMCksdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDdHhfLmdsb2JhbENvbXBvc2l0ZU9wZXJhdGlvbj0iY29weSIsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDdHhfLmRyYXdJbWFnZSh0aGlzLnNlZ21lbnRhdGlvbk1hc2tDYW52YXNfLDAsMCx0aGlzLnZvYW5nZVRGTGl0ZUluZm9fLmlucHV0V2lkdGhfLHRoaXMudm9hbmdlVEZMaXRlSW5mb18uaW5wdXRIZWlnaHRfLDAsMCx0aGlzLnNlZ21lbnRhdGlvbk1hc2tGcmFtZUNhbnZhc18ud2lkdGgsdGhpcy5zZWdtZW50YXRpb25NYXNrRnJhbWVDYW52YXNfLmhlaWdodCl9Zmx1c2goKXt9Z2V0VHJhbnNmb3JtZXJUeXBlKCl7cmV0dXJuIkJhY2tncm91bmRUcmFuc2Zvcm1lciJ9c2V0VmlkZW9CR1JlYWRhYmxlKHIpe3RoaXMuY29uZmlnXy50cmFuc2Zvcm1lclR5cGU9PT0iVmlkZW9CYWNrZ3JvdW5kIiYmKHRoaXMucG9zdFByb2Nlc3NJbnRlcmZhY2VfPW5ldyBrcihyKSl9fTtsZXQgU2U9VjtpZihTZS5CTFVSX1BST0NFU1NfR0FQX01JQ1JPX1NFQ09ORD0yZTUsU2UuVklSVFVBTF9QUk9DRVNTX0dBUF9NSUNST19TRUNPTkQ9MTVlNCx0eXBlb2YgaW1wb3J0U2NyaXB0cz09ImZ1bmN0aW9uIil7bGV0IHI9bmV3IExyO29ubWVzc2FnZT1hc3luYyBlPT57Y29uc3R7b3BlcmF0aW9uOnR9PWUuZGF0YTtpZihjb25zb2xlLmxvZygib3BlcmF0aW9uPSIsdCksdD09PSJ0cmFuc2Zvcm0iKXtjb25zdHtyZWFkYWJsZTpuLHdyaXRhYmxlOnMsY29uZmlnOmEsdmlkZW9CR1JlYWRhYmxlOmZ9PWUuZGF0YTtsZXQgbD1bXTtjb25zdCBoPW5ldyBTZShKU09OLnBhcnNlKGEpKTtoLnNldFZpZGVvQkdSZWFkYWJsZShmKSxsLnB1c2goaCksci5zZXRUcmFuc2Zvcm1lcnMobCkudGhlbigoKT0+e3IudHJhbnNmb3JtKG4scykudGhlbigoKT0+e3Bvc3RNZXNzYWdlKCJzdWNjZXNzIil9KS5jYXRjaChnPT57cG9zdE1lc3NhZ2UoImVycm9yOiIrZyl9KX0pLmNhdGNoKGc9Pntwb3N0TWVzc2FnZSgiZXJyb3I6IitnKX0pfWVsc2UgdD09PSJkZXN0cm95IiYmci5kZXN0cm95KCkudGhlbigoKT0+e3Bvc3RNZXNzYWdlKCJzdWNjZXNzIil9KS5jYXRjaChuPT57cG9zdE1lc3NhZ2UoImVycm9yOiIrbil9KX19fSkoKTsK",yl=typeof window!="undefined"&&window.Blob&&new Blob([atob(pl)],{type:"text/javascript;charset=utf-8"});function xZ(){const Z=yl&&(window.URL||window.webkitURL).createObjectURL(yl);try{return Z?new Worker(Z):new Worker("data:application/javascript;base64,"+pl,{type:"module"})}finally{Z&&(window.URL||window.webkitURL).revokeObjectURL(Z)}}class CZ{constructor(l){this.config_=l,this.worker_=new xZ}transform(l,d){return new Promise(async(b,c)=>{console.log("Created a worker thread.");const m=new Promise((t,i)=>{this.worker_.addEventListener("message",function(X){X.data==="success"?t():i(X.data)})});if(this.config_.transformerType==="VideoBackground"){const t=document.createElement("video");t.src=this.config_.backgroundAssetUri,t.muted=!0;try{await t.play()}catch(n){c(n);return}const i=t.captureStream().getVideoTracks()[0],X=new MediaStreamTrackProcessor(i).readable;this.worker_.postMessage({operation:"transform",readable:l,writable:d,config:JSON.stringify(this.config_),videoBGReadable:X},[l,d,X])}else this.worker_.postMessage({operation:"transform",readable:l,writable:d,config:JSON.stringify(this.config_)},[l,d]);m.then(()=>{console.log("Worker inited"),b()}).catch(t=>{console.log("Worker inited error"),c(t)})})}destroy(){return new Promise(async(l,d)=>{console.log("destroy a worker thread.");const b=new Promise((c,m)=>{this.worker_.addEventListener("message",function(i){i.data==="success"?c():m(i.data)})});this.worker_.postMessage({operation:"destroy"}),b.then(()=>{console.log("Worker destroyed"),l()}).catch(c=>{console.log("Worker destroy error"),d(c)})})}}s.BlurRadius=void 0,function(Z){Z.Low="Low",Z.High="High"}(s.BlurRadius||(s.BlurRadius={}));const kZ=async()=>WebAssembly.validate(new Uint8Array([0,97,115,109,1,0,0,0,1,5,1,96,0,1,123,3,2,1,0,10,10,1,8,0,65,0,253,15,253,98,11]));function UZ(){return new Promise(async(Z,l)=>{try{await K()}catch(d){l(d);return}kZ().then(d=>{if(d===!1){l("Your browser does not support WebAssembly features.");return}Z()}).catch(d=>{l(d)})})}function BZ(Z){return new Promise(async(l,d)=>{const b=new fZ(new CZ(Z));l(b)})}s.createVonageMediaProcessorConnector=BZ,s.isSupported=UZ,Object.defineProperty(s,"__esModule",{value:!0}),s[Symbol.toStringTag]="Module"});
Binary file
@@ -0,0 +1,95 @@
1
+ import { MediaProcessorConnectorInterface } from '@vonage/media-processor';
2
+ import { BlurRadius, BackgroundBlurConfig, VirtualBackgroundConfig, AssetBackgroundConfig, VideoBackgroundConfig, MediaProcessorBaseConfig, SilueteBlurConfig } from './src/types';
3
+ import { isSupported } from './src/utils/utils';
4
+ /**
5
+ * MediaProcessorConfig specifies the transformer logic to be performed.
6
+ *
7
+ * @example
8
+ *
9
+ * ```ts
10
+ * let config: MediaProcessorConfig;
11
+ * if(typeType === 'blur'){
12
+ * config = {
13
+ * wasmAssetUriPath: location.href
14
+ * transformerType: 'BackgroundBlur',
15
+ * radius: blurLevelType === 'low' ? BlurRadius.Low : BlurRadius.High,
16
+ * };
17
+ * } else if(typeType === 'virtual'){
18
+ * config = {
19
+ * wasmAssetUriPath: location.href,
20
+ * transformerType: 'VirtualBackground',
21
+ * backgroundAssetUri: location.href + "images/vonage.jpeg"
22
+ * }
23
+ * } else if(typeType === 'video') {
24
+ * config = {
25
+ * wasmAssetUriPath: location.href,
26
+ * transformerType: 'VideoBackground',
27
+ * backgroundAssetUri: location.href + 'videos/bbb.mp4'
28
+ * };
29
+ * } else if(typeType === 'siluete') {
30
+ * config = {
31
+ * wasmAssetUriPath: location.href,
32
+ * transformerType: 'SilueteBlur',
33
+ * radius: blurLevelType === 'low' ? BlurRadius.Low : BlurRadius.High,
34
+ * };
35
+ * }
36
+ * ```
37
+ */
38
+ declare type MediaProcessorConfig = MediaProcessorBaseConfig & (BackgroundBlurConfig | ((VirtualBackgroundConfig | VideoBackgroundConfig) & AssetBackgroundConfig) | SilueteBlurConfig);
39
+ /**
40
+ * Factory function to create new MediaProcessorConnectorInterface instances.
41
+ *
42
+ * @param config Specifies the transformer logic that the MediaProcessorConnectorInterface instance will be enabled with.
43
+ * @returns A promise that is resolved with a given MediaProcessorConnectorInterface instance.
44
+ *
45
+ * @example
46
+ *
47
+ * ```ts
48
+ * createVonageMediaProcessorConnector(config).then(mediaProcessorConnector => {
49
+ * setMediaProcessorConnector(mediaProcessorConnector);
50
+ * })
51
+ * .catch(e => {
52
+ * console.error(e)
53
+ * });
54
+ *
55
+ * // ..
56
+ *
57
+ * setMediaProcessorConnector(mediaProcessorConnector: MediaProcessorConnectorInterface): Promise<void> {
58
+ * return new Promise<void>(async (resolve, reject) => {
59
+ * this.mediaProcessorConnector_ = mediaProcessorConnector;
60
+ * if (!this.stream_)
61
+ * {
62
+ * console.log('[CameraSource] Requesting camera.');
63
+ * reject("no stream")
64
+ * }
65
+ * this.mediaProcessorConnector_.setTrack(this.videoTrack_).then(newTrack => {
66
+ * let processedStream = new MediaStream();
67
+ * processedStream.addTrack(newTrack);
68
+ * this.sink_.setMediaStream(processedStream);
69
+ * resolve();
70
+ * })
71
+ * .catch(e => {
72
+ * reject(e)
73
+ * })
74
+ * });
75
+ * }
76
+ *
77
+ * // ..
78
+ *
79
+ * async stopMediaProcessorConnector() {
80
+ * if(this.mediaProcessorConnector_){
81
+ * this.mediaProcessorConnector_.destroy().then(() => {
82
+ * let processedStream = new MediaStream();
83
+ * processedStream.addTrack(this.videoTrack_);
84
+ * this.sink_.setMediaStream(processedStream);
85
+ * })
86
+ * .catch(e => {
87
+ * console.error(e);
88
+ * });
89
+ * }
90
+ * }
91
+ * ```
92
+ */
93
+ declare function createVonageMediaProcessorConnector(config: MediaProcessorConfig): Promise<MediaProcessorConnectorInterface>;
94
+ export type { MediaProcessorBaseConfig, BackgroundBlurConfig, VirtualBackgroundConfig, VideoBackgroundConfig, AssetBackgroundConfig, SilueteBlurConfig, MediaProcessorConfig };
95
+ export { BlurRadius, createVonageMediaProcessorConnector, isSupported, };
@@ -0,0 +1,34 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ import { MediaProcessorConfig } from '../../main';
4
+ import { PostProcessInterface } from './PostProcessInterface';
5
+ import { VonageTFLiteInfo } from './VonageTFLite';
6
+ declare class BackgroundTransformer {
7
+ static BLUR_PROCESS_GAP_MICRO_SECOND: number;
8
+ static VIRTUAL_PROCESS_GAP_MICRO_SECOND: number;
9
+ voangeTFLiteInfo_: VonageTFLiteInfo;
10
+ frameCounter_: number;
11
+ lastProcessedTimestamp_: number;
12
+ timerGap_: number;
13
+ segmentationMaskCanvas_: OffscreenCanvas;
14
+ segmentationMaskCtx_: OffscreenCanvasRenderingContext2D | null;
15
+ segmentationPixelCount_: number;
16
+ segmentationMask_: ImageData;
17
+ segmentationMaskFrameCanvas_: OffscreenCanvas;
18
+ segmentationMaskFrameCtx_: OffscreenCanvasRenderingContext2D | null;
19
+ resultCanvas_: OffscreenCanvas;
20
+ resultCtx_: OffscreenCanvasRenderingContext2D | null;
21
+ inputMemoryOffset_: number;
22
+ outputMemoryOffset_: number;
23
+ postProcessInterface_?: PostProcessInterface;
24
+ config_: MediaProcessorConfig;
25
+ constructor(config: MediaProcessorConfig);
26
+ start(): Promise<void>;
27
+ transform(frame: any, controller: TransformStreamDefaultController): Promise<void>;
28
+ processFrame(controller: TransformStreamDefaultController, image: ImageBitmap, timestamp: number): void;
29
+ processSource(image: ImageBitmap): void;
30
+ flush(): void;
31
+ getTransformerType(): string;
32
+ setVideoBGReadable(videoBGReadable: ReadableStream): void;
33
+ }
34
+ export default BackgroundTransformer;
@@ -0,0 +1,10 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ import { PostProcessInterface } from "./PostProcessInterface";
4
+ import { BlurRadius } from "../types";
5
+ declare class BlurPostProcess implements PostProcessInterface {
6
+ blurFilter_: string;
7
+ constructor(radius?: BlurRadius);
8
+ runPostProcessing(resultCtx: OffscreenCanvasRenderingContext2D | null, image: ImageBitmap): void;
9
+ }
10
+ export default BlurPostProcess;
@@ -0,0 +1,5 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ export interface PostProcessInterface {
4
+ runPostProcessing(resultCtx: OffscreenCanvasRenderingContext2D | null, image: ImageBitmap): void;
5
+ }
@@ -0,0 +1,13 @@
1
+ export interface IQueue<T> {
2
+ enqueue(item: T): void;
3
+ dequeue(): T | undefined;
4
+ size(): number;
5
+ }
6
+ export declare class Queue<T> implements IQueue<T> {
7
+ private capacity;
8
+ private storage;
9
+ constructor(capacity?: number);
10
+ enqueue(item: T): void;
11
+ dequeue(): T | undefined;
12
+ size(): number;
13
+ }
@@ -0,0 +1,10 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ import { PostProcessInterface } from './PostProcessInterface';
4
+ import { BlurRadius } from '../types';
5
+ declare class SilueteBlurPostProcess implements PostProcessInterface {
6
+ blurFilter_: string;
7
+ constructor(radius?: BlurRadius);
8
+ runPostProcessing(resultCtx: OffscreenCanvasRenderingContext2D | null, image: ImageBitmap): void;
9
+ }
10
+ export default SilueteBlurPostProcess;
@@ -0,0 +1,11 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ import { PostProcessInterface } from './PostProcessInterface';
4
+ import { Queue } from './Queue';
5
+ export declare class VideoPostProcess implements PostProcessInterface {
6
+ queue_: Queue<any>;
7
+ frameReader_: any;
8
+ constructor(videoBGReadable: ReadableStream);
9
+ processFrame(result: any): void;
10
+ runPostProcessing(resultCtx: OffscreenCanvasRenderingContext2D | null, image: ImageBitmap): void;
11
+ }
@@ -0,0 +1,11 @@
1
+ /// <reference types="dom-webcodecs" />
2
+ /// <reference types="offscreencanvas" />
3
+ import { PostProcessInterface } from "./PostProcessInterface";
4
+ declare class VirtualPostProcess implements PostProcessInterface {
5
+ imageCanvas_: OffscreenCanvas;
6
+ imageCtx_: OffscreenCanvasRenderingContext2D | null;
7
+ imageBlob_: any;
8
+ constructor(imageUri: string);
9
+ runPostProcessing(resultCtx: OffscreenCanvasRenderingContext2D | null, image: ImageBitmap): void;
10
+ }
11
+ export default VirtualPostProcess;
@@ -0,0 +1,28 @@
1
+ /// <reference types="emscripten" />
2
+ export interface TFLite extends EmscriptenModule {
3
+ _getModelBufferMemoryOffset(): number;
4
+ _getInputMemoryOffset(): number;
5
+ _getInputHeight(): number;
6
+ _getInputWidth(): number;
7
+ _getInputChannelCount(): number;
8
+ _getOutputMemoryOffset(): number;
9
+ _getOutputHeight(): number;
10
+ _getOutputWidth(): number;
11
+ _getOutputChannelCount(): number;
12
+ _loadModel(bufferSize: number): number;
13
+ _runInference(): number;
14
+ }
15
+ export declare class VonageTFLiteInfo {
16
+ selectedTFLite_: TFLite;
17
+ isSIMDSupported_: boolean;
18
+ isThreadsSupported_: boolean;
19
+ inputWidth_: number;
20
+ inputHeight_: number;
21
+ inputChannelCount_: number;
22
+ inputMemoryOffset_: number;
23
+ outputWidth_: number;
24
+ outputHeight_: number;
25
+ outputChannelCount_: number;
26
+ outputMemoryOffset_: number;
27
+ }
28
+ export declare function VonageTFLite(uri: string): Promise<VonageTFLiteInfo>;
@@ -0,0 +1 @@
1
+ export declare function getVonageTFLiteModelFileName(): string;
@@ -0,0 +1,65 @@
1
+ /**
2
+ * VonageTransformerType which type of transformer to create.
3
+ */
4
+ export declare type VonageTransformerType = 'BackgroundBlur' | 'VirtualBackground' | 'VideoBackground' | 'SilueteBlur';
5
+ /**
6
+ * BlurRadius specifies how much bluring filter to apply by a given transformer.
7
+ */
8
+ export declare enum BlurRadius {
9
+ /**
10
+ * Low bluring filtering.
11
+ */
12
+ Low = "Low",
13
+ /**
14
+ * High bluring filtering.
15
+ */
16
+ High = "High"
17
+ }
18
+ /**
19
+ * MediaProcessorBaseConfig specifies the absolute URL for the path where the Wasm modules used by the transformers from this library are stored. ML models files must be located under a folder called `models` under this same path.
20
+ *
21
+ * @example
22
+ *
23
+ * ```ts
24
+ * {
25
+ * wasmAssetUriPath: 'http://localhost:8080/assets'
26
+ * }
27
+ * ```
28
+ *
29
+ * Wasm modules must be found under *assets/* folder as well as the ML models files under *assets/models* folder.
30
+ */
31
+ export declare type MediaProcessorBaseConfig = {
32
+ wasmAssetUriPath: string;
33
+ };
34
+ /**
35
+ * BackgroundBlurConfig specifies the configuration for a transformer logic that performs background blurring.
36
+ */
37
+ export declare type BackgroundBlurConfig = {
38
+ transformerType: 'BackgroundBlur';
39
+ radius?: BlurRadius;
40
+ };
41
+ /**
42
+ * AssetBackgroundConfig specifies the absolute URL to the image or video file used for background replacement.
43
+ */
44
+ export declare type AssetBackgroundConfig = {
45
+ backgroundAssetUri: string;
46
+ };
47
+ /**
48
+ * VirtualBackgroundConfig specifies the configuration for a transformer logic that performs background replacement with a given image.
49
+ */
50
+ export declare type VirtualBackgroundConfig = {
51
+ transformerType: 'VirtualBackground';
52
+ };
53
+ /**
54
+ * VirtualBackgroundConfig specifies the configuration for a transformer logic that performs background replacement with a given video.
55
+ */
56
+ export declare type VideoBackgroundConfig = {
57
+ transformerType: 'VideoBackground';
58
+ };
59
+ /**
60
+ * BackgroundBlurConfig specifies the configuration for a transformer logic that performs siluete blurring.
61
+ */
62
+ export declare type SilueteBlurConfig = {
63
+ transformerType: 'SilueteBlur';
64
+ radius?: BlurRadius;
65
+ };
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Check if the current browser is officially supported by the library.
3
+ *
4
+ * @returns The promise will resolve or reject depending whether the browser is supported or not.
5
+ */
6
+ export declare function isSupported(): Promise<void>;
@@ -0,0 +1,10 @@
1
+ import { MediaProcessorInterface } from '@vonage/media-processor';
2
+ import { MediaProcessorConfig } from '../../main';
3
+ declare class VonageMediaProcessorWorker implements MediaProcessorInterface {
4
+ worker_: any;
5
+ config_: MediaProcessorConfig;
6
+ constructor(config: MediaProcessorConfig);
7
+ transform(readable: ReadableStream, writable: WritableStream): Promise<void>;
8
+ destroy(): Promise<void>;
9
+ }
10
+ export default VonageMediaProcessorWorker;
package/package.json ADDED
@@ -0,0 +1,65 @@
1
+ {
2
+ "name": "@vonage/ml-transformers",
3
+ "version": "1.0.1",
4
+ "author": "Guy Mininberg <guy.mininberg@vonage.com>",
5
+ "contributors": [
6
+ "Guy Mininberg <guy.mininberg@vonage.com>",
7
+ "Jose Olivera <joseantonio.oliveraortega@vonage.com>"
8
+ ],
9
+ "keywords": [
10
+ "vonage",
11
+ "vonage api",
12
+ "client",
13
+ "media processing",
14
+ "insertable streams",
15
+ "transformers",
16
+ "machine learning",
17
+ "background blur",
18
+ "virtual background",
19
+ "video background",
20
+ "face masking",
21
+ "opentok",
22
+ "nexmo"
23
+ ],
24
+ "scripts": {
25
+ "build": "npm-run-all build:js doc",
26
+ "build:js": "tsc && vite build && tsc --declaration --emitDeclarationOnly --outDir dist/types",
27
+ "doc": "typedoc --excludePrivate --excludeProtected --disableSources --out dist/docs lib/main.ts",
28
+ "dev": "vite",
29
+ "preview": "vite preview"
30
+ },
31
+ "dependencies": {
32
+ "@types/emscripten": "^1.39.6",
33
+ "wasm-feature-detect": "^1.2.11",
34
+ "@types/offscreencanvas": "^2019.6.4",
35
+ "@types/dom-mediacapture-transform": "^0.1.2",
36
+ "@types/node": "^17.0.10",
37
+ "axios": "^0.25.0",
38
+ "uuid": "^8.3.2",
39
+ "typescript-optional": "3.0.0-alpha.3",
40
+ "@vonage/media-processor": "1.0.1"
41
+ },
42
+ "files": [
43
+ "dist"
44
+ ],
45
+ "types": "./dist/types/main.d.ts",
46
+ "main": "./dist/ml-transformers.umd.js",
47
+ "module": "./dist/ml-transformers.es.js",
48
+ "exports": {
49
+ ".": {
50
+ "import": "./dist/ml-transformers.es.js",
51
+ "require": "./dist/ml-transformers.umd.js"
52
+ }
53
+ },
54
+ "devDependencies": {
55
+ "@tsconfig/node12": "^1.0.1",
56
+ "typescript": "^4.5.4",
57
+ "vite": "^2.7.2",
58
+ "npm-run-all": "^4.1.5",
59
+ "typedoc": "^0.22.12"
60
+ },
61
+ "repository": "https://github.com/vonage/ml-transformers",
62
+ "publishConfig": {
63
+ "registry": "https://registry.npmjs.org"
64
+ }
65
+ }