' . NL; const DATEFMT = 'DateFmt'; const GRAPHNAME ='GraphName'; const BARMAXWIDTH = 250; # px const BARCELLWIDTH = '260px'; const NULLUSERAGENT = '_nulluseragentvalue_'; const USERAGENTEMPTY = 'User Agent empty'; // These constants are cell or column names in the TotalCounter array. DO NOT change. const PREVIOUSYEARS = 'LastYears'; const LASTYEAR = 'LastYear'; const LASTMONTH = 'LastYear'; const LASTWEEK = 'LastWeek'; const LASTDAY = 'LastDay'; const KEYPAGES = 'Pages'; const KEYUSERS = 'Users'; const KEYBROWSERS = 'Browsers'; const KEYOSES = 'OSes'; const KEYREFERERS = 'Referers'; # spelling as per mispelling in HTTP header spec const KEYLOCATIONS = 'Locations'; const KEYBOTS = 'Bots'; const KEYLANGUAGES = 'Languages'; const KEYPAGESTODAYCOUNTER = 'PagesTodayCounter'; const KEYPAGESTODAYDAY = 'PagesTodayDay'; const KEYLASTTIMESTAMP = 'LastTimestamp'; const KEYDATECREATED = 'DateCreated'; const KEYTOTAL = 'Total'; const KEYTOTALCOUNTERVERSION = 'TotalCounterVersion'; const UNKNOWN = 'Unknown'; # const ACTIONNAMECHECK = 'totalcountercheck'; // ?action=totalcountercheck \SDV($TotalCounterActionName, 'totalcounter'); // ?action=totalcounter \SDV($TotalCounterAuthLevel, 'read'); \SDV($TotalCounterMaxItems, 30); // default 30 \SDV($TotalCounterEnableLookup, 0); // default 0 \SDV($TotalCounterBarColor, '#5af'); // default '#5af' \SDV($TotalCounterCountBots, 0); // default 0 \SDV($TotalCounterShowNumbers, 1); // default 1 \SDV($TotalCounterEnableGeoIp, 0); // default 0 \SDV($TotalCounterGeoIPData, "$WorkDir/GeoIP.dat"); \SDV($TotalCounterEnableDownload, 0); //default 0 \SDV($TotalCounterDownloadManager, "$WorkDir/.download.manager"); \SDV($TotalCounterEnableChmods, 1); // default 1 \SDV($TotalCounterEnableUsers, 0); // default 0 \SDV($TotalCounterFile, "$WorkDir/totalcounter.stat"); \SDV($TotalCounterLockfile, "$WorkDir/totalcounter.lock"); \SDV($TotalCounterLogfile, "$WorkDir/totalcounter.log"); \SDV($TotalCounterEnableLog, 0); # default 0 (off), 1 (on, captures unknowns), 2 log server details (verbose) /*## \SDV($TotalCounterBrowsersUnset, ''); // used to remove individual totals from file, don't use this unless you know what you are doing \SDV($TotalCounterOSesUnset, ''); // used to remove individual totals from file, don't use this unless you know what you are doing \SDV($TotalCounterLocationsUnset, ''); // used to remove individual totals from file, don't use this unless you know what you are doing \SDV($TotalCounterBotsUnset, ''); // used to remove individual totals from file, don't use this unless you know what you are doing \SDV($TotalCounterReferersUnset, ''); // used to remove individual totals from file, don't use this unless you know what you are doing ##*/ \SDV($HTMLStylesFmt[TOTALCOUNTERNAME], '.TCbar {background-color:$TotalCounterBarColor; min-height:13px; width:13px; color:#fff;}' .NL . '.TCtxtr {text-align:right;}' .NL . '.TCtxtl {text-align:left;}' .NL . '.TCtxth {font-weight: bold;}' .NL . '.TCprogress {margin-left:auto; margin-right:auto;}' . NL . 'table.totalcounter td {font-size:x-small; text-align:left}' . NL); \SDVA($TotalCounterMonthsShort, array('Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec')); \SDV($TotalCounterBlacklist[KEYPAGES], array ()); \SDV($TotalCounterBlacklist[KEYUSERS], array ()); \SDV($TotalCounterBlacklist[KEYBROWSERS], array ()); \SDV($TotalCounterBlacklist[KEYOSES], array ()); \SDV($TotalCounterBlacklist[KEYREFERERS], array ()); \SDV($TotalCounterBlacklist[KEYLOCATIONS], array ()); \SDV($TotalCounterBlacklist[KEYBOTS], array ()); \SDV($TotalCounterBlacklist[KEYLANGUAGES], array ()); ## by MateuszCzaplinski ## last day, last week, ... - data & display descriptions \SDVA($TotalCounterTimeBins, array( LASTDAY => array( # LastDay = 24 hours; 1 hour = 60*60sec GRAPHNAME=>'$[Last] $[day] ($[hours])', 'max'=>24, 'atom'=>60*60, // DATEFMT => function($now, $atom, $maxnr, $nr) { return date("H:00", $now - $atom * ($maxnr - 1 - $nr)); } ), LASTWEEK => array( # LastWeek = 7 days GRAPHNAME=>'$[Last] $[week]', 'max'=>7, 'atom'=>24*60*60, // DATEFMT => function($now, $atom, $maxnr, $nr) { return date("D", $now - $atom * ($maxnr - 1 - $nr)); } ), LASTMONTH => array( GRAPHNAME=>'$[Last] $[month]', 'max'=>31, 'atom'=>24*60*60, // DATEFMT => function($now, $atom, $maxnr, $nr) { return date("j", $now - $atom * ($maxnr - 1 - $nr)); } ), LASTYEAR => array( # date('n') is the month of the year GRAPHNAME=>'$[Last] $[year]', 'max'=>12, 'atom'=>'n', // DATEFMT => function($now, $atom, $maxnr, $nr) use ($TotalCounterMonthsShort) { return $TotalCounterMonthsShort[(12 + intval(date($atom, $now)) - $maxnr + $nr) % 12]; } ), PREVIOUSYEARS => array( GRAPHNAME=>'$[Previous] $[years]', 'max'=>30, 'atom'=>'Y', // DATEFMT => function($now, $atom, $maxnr, $nr) { return strval(intval(date($atom, $now)) - ($maxnr - 1 - $nr)); } ) )); \SDVA($HandleActions, array ( $TotalCounterActionName => __NAMESPACE__ . '\HandleTotalCounter' )); \SDVA($HandleAuth, array ( $TotalCounterActionName => $TotalCounterAuthLevel )); global $TotalCounter; \SDV($TotalCounterDebug, false); # set default debug setting # set debug flag $totalCounter_debugOn = boolval ($TotalCounterDebug); # if on writes input and output to web page if ($totalCounter_debugOn) { tcmsg (__FILE__, $RecipeInfo[TOTALCOUNTERNAME]['Version'] . ' using "' . $WorkDir . '" with action=' . $action . ', log=' . $TotalCounterEnableLog . ', IP lookup: '. $TotalCounterEnableLookup); } $TotalCounterLog = boolval ($TotalCounterEnableLog > 0); $TotalCounterLogVerbose = boolval ($TotalCounterEnableLog == 2); if ($TotalCounterMaxItems <= 0) $TotalCounterMaxItems = 1; # set up some debug actions for fixing the TotalCounter file /* if ($totalCounter_debugOn) { $HandleActions [ACTIONNAMECHECK] = __NAMESPACE__ . '\HandleTotalCounterCheck'; $HandleAuth [ACTIONNAMECHECK] = 'admin'; } # end if debug */ $statfilename = $TotalCounterFile; $lockfilename = $TotalCounterLockfile; $logfilename = $TotalCounterLogfile; $psft = function_exists('\PSFT') ? '\PSFT' : 'strftime'; $logfiletime = $psft ("%Y-%m-%d %H:%M:%S "); # identify this run $geoIpFile = $TotalCounterGeoIPData; // clear cached information about file clearstatcache(); // script to carry on working after the user has cancelled request or browser session closed ignore_user_abort(true); //------------------------------------------------------------------------------------ if ($TotalCounterLog) { $logfilehandle = fopen($logfilename, 'a'); # create or open logfile for appending if ($logfilehandle === false) { tcmsg ('fopen failed', $logfilename, error_get_last()); if ($totalCounter_debugOn) \Abort ($MessagesFmt [TOTALCOUNTERNAME]); } } if ($TotalCounterLogVerbose) { # write for every call, this can be very verbose \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . 'UA: "' . $HttpUserAgent . '" Rf: "' . $_SERVER['HTTP_REFERER'] . '" URf: "' . $_SERVER['HTTP_USER_REFERER'] . '" XXF: "' . $_SERVER['HTTP_X_FORWARDED_FOR'] . '" Fw: "' . $_SERVER['HTTP_FORWARDED'] . '" XFH: "' . $_SERVER['HTTP_X_FORWARDED_HOST'] . '" RH: "' . $_SERVER['REMOTE_HOST'] . '" RA: "' . $_SERVER['REMOTE_ADDR'] . '"' . NL); \Lock(0); # release lock if (false === $fwritestatus) { tcmsg ('fwrite failed', $logfilename, error_get_last()); if ($totalCounter_debugOn) \Abort ($MessagesFmt [TOTALCOUNTERNAME]); } } if (function_exists('\ResolvePageName')) { $tc_pagename = \ResolvePageName($pagename); } else { $tc_pagename = str_replace('/', '.', $pagename); # consistemmtly use "." to separate Groupname and Pagename } // end if $tcPageExists = true; switch (true) { case (empty($tc_pagename)): $tcPageExists = false; $tc_pagename = UNKNOWN; #"$DefaultGroup.$DefaultName"; if ($TotalCounterLog) { \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . 'Called with empty pagename:"' . $pagename . '"' . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } break; case (\PageExists($tc_pagename)): # PmWiki function break; default: # pagename does not exist $tcPageExists = false; if ($TotalCounterLog) { \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . 'Called with non-existent page:"' . $tc_pagename . '"' . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } } # end switch if (!$tcPageExists) { # page does not exist return; # finished TotalCounter processing <=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<=<= } # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // update counts from page being browsed if ($action == 'browse') { //find users if (isset ($AuthId)) { $tc_user = $AuthId; } else { if (isset ($Author)) { $tc_user = $Author; } else { session_start(); if (isset ($_SESSION['authid'])) { $tc_user = $_SESSION['authid'][0]; } else { $tc_user = 'Guest (not authenticated)'; } // end isset $_SESSION } // end if else isset $Author } // end if else isset $AuthId ## detect user agent $tc_bot = NULLUSERAGENT; $tc_browser = NULLUSERAGENT; $tcBotFound = false; $tcBrowserFound = false; switch (true) { case empty($_SERVER['HTTP_USER_AGENT']): # it happens often if ($TotalCounterLogVerbose) { \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . 'UA empty RH:"' . $_SERVER['REMOTE_HOST'] . '" action: ' . $action . '"' . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } $tc_bot = USERAGENTEMPTY; $tc_browser = USERAGENTEMPTY; break; default: # user agent not empty $tc_bot = detectWebBot ($_SERVER['HTTP_USER_AGENT']); if ($tc_bot !== NULLUSERAGENT) { # found a bot $tcBotFound = true; break; } $tc_browser = detectBrowser ($_SERVER['HTTP_USER_AGENT']); if ($tc_browser !== NULLUSERAGENT) { $tcBrowserFound = true; break; } $tc_browser = UNKNOWN; $tcBrowserFound = false; # don't count unknown; if ($TotalCounterLog) { $logMsg = 'Browser unknown: '; # initialise if (!empty ($_SERVER['HTTP_X_FORWARDED_FOR'])) $logMsg .= 'XFF:"' . $_SERVER['HTTP_X_FORWARDED_FOR'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED'])) $logMsg .= 'Fw:"' . $_SERVER['HTTP_FORWARDED'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED-HOST'])) $logMsg .= 'FH:"' . $_SERVER['HTTP_FORWARDED-HOST'] . '" '; if (!empty ($_SERVER['REMOTE_HOST'])) $logMsg .= 'RH:"' . $_SERVER['REMOTE_HOST'] . '" '; if (!empty ($_SERVER['REMOTE_ADDR'])) $logMsg .= 'RA:"' . $_SERVER['REMOTE_ADDR'] . '" '; if (!empty ($_SERVER['HTTP_USER_AGENT'])) $logMsg .= 'UA:"' . $_SERVER['HTTP_USER_AGENT'] . '" '; \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . $logMsg . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } } # end switch // decide if we are counting this visit $tc_count_visit = ((!$tcBotFound) // don't count bots || ($TotalCounterCountBots == 1)); // count bots (all visits) $tcOSFound = false; if ($tc_count_visit) { # don't count bots by default // find operating system if ($tc_browser == USERAGENTEMPTY) { $tc_os = USERAGENTEMPTY; } else { $tc_os = detectOS ($_SERVER['HTTP_USER_AGENT']); if ($tc_os == NULLUSERAGENT) { $tc_os = UNKNOWN; if ($TotalCounterLog) { $logMsg = 'OpSystem unknown: '; # initialise if (!empty ($_SERVER['HTTP_X_FORWARDED_FOR'])) $logMsg .= 'XFF:"' . $_SERVER['HTTP_X_FORWARDED_FOR'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED'])) $logMsg .= 'Fw:"' . $_SERVER['HTTP_FORWARDED'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED-HOST'])) $logMsg .= 'FH:"' . $_SERVER['HTTP_FORWARDED-HOST'] . '" '; if (!empty ($_SERVER['REMOTE_HOST'])) $logMsg .= 'RH:"' . $_SERVER['REMOTE_HOST'] . '" '; if (!empty ($_SERVER['REMOTE_ADDR'])) $logMsg .= 'RA:"' . $_SERVER['REMOTE_ADDR'] . '" '; if (!empty ($_SERVER['HTTP_USER_AGENT'])) $logMsg .= 'UA:"' . $_SERVER['HTTP_USER_AGENT'] . '" '; \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . $logMsg . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } } else { $tcOSFound = true; } } } // end find OS if ($tc_count_visit) { # don't count bots by default // find referrer domain $matches = []; $referer = ''; $tc_referer = UNKNOWN; if (!empty ($_SERVER['HTTP_REFERER'])) { # remove the schema, see https://regex101.com/r/epmzHv/2 if (1 == preg_match("/^(?:https?:\/\/)?([^\/:\r\n]+)/", $_SERVER['HTTP_REFERER'], $matches)) { $referer = $matches[1]; } } if (!empty($referer)) { $tc_referer = $referer; } if ($tc_referer == UNKNOWN) { if ($tcBotFound) { # skip referer if it is a bot and referer not identified // 1.10.0 unset ($tc_referer); }; // end !empty $tc_bot if ($TotalCounterLog and !empty($_SERVER['HTTP_REFERER'])) { $logMsg = 'Referer unknown: '; # initialise if (!empty ($_SERVER['HTTP_X_FORWARDED_FOR'])) $logMsg .= 'XFF:"' . $_SERVER['HTTP_X_FORWARDED_FOR'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED'])) $logMsg .= 'Fw:"' . $_SERVER['HTTP_FORWARDED'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED-HOST'])) $logMsg .= 'FH:"' . $_SERVER['HTTP_FORWARDED-HOST'] . '" '; if (!empty ($_SERVER['REMOTE_HOST'])) $logMsg .= 'RH:"' . $_SERVER['REMOTE_HOST'] . '" '; if (!empty ($_SERVER['REMOTE_ADDR'])) $logMsg .= 'RA:"' . $_SERVER['REMOTE_ADDR'] . '" '; if (!empty ($_SERVER['HTTP_USER_AGENT'])) $logMsg .= 'UA:"' . $_SERVER['HTTP_USER_AGENT'] . '" '; \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . $logMsg . '" ^' . $referer . '^ [' . implode (', ', $matches) . '] ' . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } } // end find referrer } // end count visit referrer if ($tc_count_visit) { # don't count bots by default // find location $dbgloc = ''; # de-facto standard header for identifying the originating IP address of a client connecting to a web server through an HTTP proxy or a load balancer $dbgloc = ''; # initialise $thehost = ''; # initialise switch (true) { case (!empty($_SERVER['HTTP_X_FORWARDED_FOR'])): $dbgloc .= 'XFF'; if (false !== strpos($_SERVER['HTTP_X_FORWARDED_FOR'], ',')) { $ips = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']); $thehost = trim($ips[0]); # see https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Forwarded-For } else { $thehost = $_SERVER['HTTP_X_FORWARDED_FOR']; } break; case (!empty($_SERVER['HTTP_FORWARDED'])): $dbgloc = 'Fw'; $posn = strpos($_SERVER['HTTP_FORWARDED'], 'for='); if (false !== $posn) { $ips = explode(';', substr($_SERVER['HTTP_FORWARDED'], $posn + 4)); #for=192.0.2.60;proto=http $ips = explode(',', $ips[0]); # 192.0.2.43, for=198.51.100.17 $thehost = trim($ips[0]); } else { $thehost = $_SERVER['HTTP_FORWARDED']; } break; case (!empty($_SERVER['REMOTE_HOST'])): if (false !== strpos($_SERVER['REMOTE_HOST'], ',')) { $dbgloc = 'RH+'; # empty($_SERVER['HTTP_X_FORWARDED_FOR']) and empty($_SERVER['HTTP_FORWARDED']) $ips = explode(',', $_SERVER['REMOTE_HOST']); $thehost = trim($ips[0]); } else { $dbgloc = 'RH'; $thehost = $_SERVER['REMOTE_HOST']; } break; default: # no header information ! break; } # end switch /* this code replaced by the above switch statement if (!empty($_SERVER['HTTP_X_FORWARDED_FOR'])) { $dbgloc .= 'XFF'; if (false !== strpos($_SERVER['HTTP_X_FORWARDED_FOR'], ',')) { $ips = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']); $thehost = trim($ips[0]); # see https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Forwarded-For } else { $thehost = $_SERVER['HTTP_X_FORWARDED_FOR']; } } elseif (!empty($_SERVER['HTTP_FORWARDED'])) { $dbgloc = 'Fw'; $posn = strpos($_SERVER['HTTP_FORWARDED'], 'for='); if (false !== $posn) { $ips = explode(';', substr($_SERVER['HTTP_FORWARDED'], $posn + 4)); #for=192.0.2.60;proto=http $ips = explode(',', $ips[0]); # 192.0.2.43, for=198.51.100.17 $thehost = trim($ips[0]); } else { $thehost = $_SERVER['HTTP_FORWARDED']; } } elseif (false !== strpos($_SERVER['REMOTE_HOST'], ',')) { $dbgloc = 'RH+'; # empty($_SERVER['HTTP_X_FORWARDED_FOR']) and empty($_SERVER['HTTP_FORWARDED']) $ips = explode(',', $_SERVER['REMOTE_HOST']); $thehost = trim($ips[0]); } else { $dbgloc = 'RH'; $thehost = $_SERVER['REMOTE_HOST']; } */ // match any character not digits or period backwards from end of string if (1 == preg_match("/[^\.0-9]+$/", $thehost, $matches)) { $loc = $matches[0]; } $tc_location = UNKNOWN; while (true) { if (!empty($loc)) { $dbgloc .= '='; $tc_location = $loc; break; } if ($TotalCounterEnableLookup == 1) { $hostbyaddr = gethostbyaddr($_SERVER['REMOTE_ADDR']); $dbgloc .= 'L^' . $hostbyaddr . '^'; if (false === $hostbyaddr) { $tc_location = UNKNOWN; } else { $prmRetVal = preg_match("/[^\.0-9]+$/", $hostbyaddr, $matches); // match any character not digits or period (IP address?) switch (true) { case ($prmRetVal === false): # error occurred $tc_location = UNKNOWN; break; case ($prmRetVal == 1): # match found $gloc = $matches[0]; break; case ($prmRetVal == 0): # match not found $tc_location = UNKNOWN; break; } # end switch if (!empty($gloc)) { $tc_location = $gloc; break; } } } if ($TotalCounterEnableGeoIp == 1) { $dbgloc .= 'G'; include ('geoip/geoip.inc'); # return the two letter country code corresponding to a hostname or an IP address $gi = geoip_open($geoIpFile, GEOIP_STANDARD); # https://www.php.net/manual/en/function.geoip-country-code-by-name.php $gccba = geoip_country_code_by_addr($gi, $_SERVER['REMOTE_ADDR']); geoip_close($gi); if (!false === $gccba) { $tc_location = $gccba; break; } } # https://regex101.com/r/ONcmD7/1 if (1 == preg_match("/(?:10\.[0-9]{1,3}|172\.(?:1[6-9]|2[0-9]|3[0-1])|192\.168)(?:\.[0-2]?[0-9]{1,2}){2}/", $thehost, $matches)) { $dbgloc .= 'P'; # match for private IP address https://en.wikipedia.org/wiki/Private_network $tc_location = 'Private IP (' . stristr ($matches [0], '.', true) . ')'; break; } $dbgloc .= '.'; break; } # end while true if ($tc_location == UNKNOWN) { if ($tcBotFound) { # skip location if it is a bot and location not identified unset ($tc_location); } elseif ($TotalCounterLogVerbose) { # this happens so often we log only if verbose $logMsg = 'Location unknown: '; # initialise if (!empty ($_SERVER['HTTP_X_FORWARDED_FOR'])) $logMsg .= 'XFF:"' . $_SERVER['HTTP_X_FORWARDED_FOR'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED'])) $logMsg .= 'Fw:"' . $_SERVER['HTTP_FORWARDED'] . '" '; if (!empty ($_SERVER['HTTP_FORWARDED-HOST'])) $logMsg .= 'FH:"' . $_SERVER['HTTP_FORWARDED-HOST'] . '" '; if (!empty ($_SERVER['REMOTE_HOST'])) $logMsg .= 'RH:"' . $_SERVER['REMOTE_HOST'] . '" '; if (!empty ($_SERVER['REMOTE_ADDR'])) $logMsg .= 'RA:"' . $_SERVER['REMOTE_ADDR'] . '" '; # if (!empty ($_SERVER['HTTP_USER_AGENT'])) $logMsg .= 'UA:"' . $_SERVER['HTTP_USER_AGENT'] . '" '; # don't lok this variable \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . $logMsg . ' (' . $thehost . ') ' . $dbgloc . NL); \Lock(0); # release lock if ($fwritestatus === false) { tcmsg ('fwrite failed', $logfilename, error_get_last()); } } } else { //end = Unknown $tc_location = strtolower($tc_location); $tc_location = str_ireplace ([UNKNOWN, 'private ip'], [UNKNOWN, 'Private IP'], $tc_location); } } // end count visits location } // end if action = browse # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= $oldumask = umask(0); # set the default file permissions for newly created files and directories $TotalCounterDownloads = FALSE; if ($TotalCounterEnableDownload == 1) { $downloadfile = $TotalCounterDownloadManager; if (file_exists($downloadfile)) { $TotalCounterDownloadsFileContents = tc_file_get_contents($downloadfile); if (FALSE === $TotalCounterDownloadsFileContents) { tcmsg ('tc_file_get_contents failed', $downloadfile, error_get_last()); } else { $TotalCounterDownloads = unserialize($TotalCounterDownloadsFileContents); if (FALSE === $TotalCounterDownloads) tcmsg ('unserialize failed', $downloadfile, error_get_last()); } } } if (file_exists($statfilename)) { $TotalCounterFileContents = tc_file_get_contents($statfilename); if (FALSE === $TotalCounterFileContents) { tcmsg ('tc_file_get_contents failed', $statfilename, error_get_last()); echo $MessagesFmt [TOTALCOUNTERNAME]; return ($MessagesFmt [TOTALCOUNTERNAME]); # failed to read file, lets get out of here } $TotalCounter = unserialize($TotalCounterFileContents); if (FALSE === $TotalCounter) { tcmsg ('tc_file unserialize failed', $statfilename, error_get_last()); echo $MessagesFmt [TOTALCOUNTERNAME]; return ($MessagesFmt [TOTALCOUNTERNAME]); # failed to unserialise file, lets get out of here } } else { touch($statfilename); # create the stat file $TotalCounter = []; $TotalCounter[KEYDATECREATED] = date ('c'); # ISO 8601 format $TotalCounter[KEYTOTAL] = 0; $TotalCounter[KEYPAGES][$tc_pagename] = 0; } # $PageCount = 0; # initialise $TotalCount = 0; # initialise # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= if (($action == 'browse') && ($tcPageExists)) { if( dblock($statfilename) ) { $TotalCounterFileContents = tc_file_get_contents($statfilename); if (FALSE === $TotalCounterFileContents) { tcmsg ('tc_file_get_contents failed', $statfilename, error_get_last()); } $TotalCounter = unserialize($TotalCounterFileContents); if (FALSE === $TotalCounter) { tcmsg ('tc_file unserialize failed', $statfilename, error_get_last()); } /*## // code intended only for developers to clean up entries in the stats file // use at your own risk if ($totalCounter_debugOn) { if (!empty ($TotalCounterBrowsersUnset)) { if (isset($TotalCounter[KEYBROWSERS][$TotalCounterBrowsersUnset])) { unset ($TotalCounter[KEYBROWSERS][$TotalCounterBrowsersUnset]); tcmsg ('unset Browser', $TotalCounterBrowsersUnset); } } # end Browsers if (!empty ($TotalCounterOSesUnset)) { if (isset($TotalCounter[KEYOSES][$TotalCounterOSesUnset])) { unset ($TotalCounter[KEYOSES][$TotalCounterOSesUnset]); tcmsg ('unset OS', $TotalCounterOSesUnset); } } # end OSes if (!empty ($TotalCounterLocationsUnset)) { if (isset($TotalCounter[KEYLOCATIONS][$TotalCounterLocationsUnset])) { unset ($TotalCounter[KEYLOCATIONS][$TotalCounterLocationsUnset]); tcmsg ('unset Location', $TotalCounterLocationsUnset); } } # end Locations if (!empty ($TotalCounterBotsUnset)) { if (isset($TotalCounter[KEYBOTS][$TotalCounterBotsUnset])) { unset ($TotalCounter[KEYBOTS][$TotalCounterBotsUnset]); tcmsg ('unset Bot', $TotalCounterBotsUnset); } } # end Bots if (!empty ($TotalCounterReferersUnset)) { if (isset($TotalCounter[KEYREFERERS][$TotalCounterReferersUnset])) { unset ($TotalCounter[KEYREFERERS][$TotalCounterReferersUnset]); tcmsg ('unset Referer', $TotalCounterReferersUnset); } } # end Referers } # $totalCounter_debugOn // end clean up code ##*/ $TotalCount = ++ $TotalCounter[KEYTOTAL]; $blacklisted = false; if (in_array($tc_user, $TotalCounterBlacklist[KEYUSERS])) $blacklisted = true; if (!$blacklisted && !in_array($tc_user, $TotalCounterBlacklist[KEYUSERS])) { if (is_array($TotalCounterBlacklist[KEYUSERS])) { foreach ($TotalCounterBlacklist[KEYUSERS] as $value) if (substr($value, 0, 1) == '/' && preg_match($value, $tc_user) > 0) $blacklisted = true; } if (isset ($tc_user)) { incrementCount(KEYUSERS, $tc_user); } } if (!$blacklisted && !in_array($tc_pagename, $TotalCounterBlacklist[KEYPAGES])) { if (is_array($TotalCounterBlacklist[KEYPAGES])) foreach ($TotalCounterBlacklist[KEYPAGES] as $value) if (substr($value, 0, 1) == '/') if (preg_match($value, $tc_pagename) > 0) $blacklisted = true; if (!$blacklisted) { if (empty ($TotalCounter[KEYPAGES][$tc_pagename])) { # initialise $TotalCounter[KEYPAGES][$tc_pagename] = 0; $TotalCounter[KEYPAGESTODAYDAY][$tc_pagename] = date("%y%m%d"); $TotalCounter[KEYPAGESTODAYCOUNTER][$tc_pagename] = 0; } $PageCount = ++ $TotalCounter[KEYPAGES][$tc_pagename]; ## handles the daily counter if ($TotalCounter[KEYPAGESTODAYDAY][$tc_pagename] == date("%y%m%d")) $PageCountToday = ++ $TotalCounter[KEYPAGESTODAYCOUNTER][$tc_pagename]; else { $TotalCounter[KEYPAGESTODAYDAY][$tc_pagename] = date("%y%m%d"); $TotalCounter[KEYPAGESTODAYCOUNTER][$tc_pagename] = 1; } } else { $PageCount = 0; // blacklisted } } if (!$blacklisted && defined('MULTILANGUAGE')) { if (isset ($userlang2)) { incrementCount(KEYLANGUAGES, $userlang2); } } if (!$blacklisted && ($tcBrowserFound) && !in_array($tc_browser, $TotalCounterBlacklist[KEYBROWSERS])) { incrementCount(KEYBROWSERS, $tc_browser); } if (!$blacklisted && ($tcBotFound) && !in_array($tc_bot, $TotalCounterBlacklist[KEYBOTS])) { incrementCount(KEYBOTS, $tc_bot); } if (!$blacklisted && ($tcOSFound) && !in_array($tc_os, $TotalCounterBlacklist[KEYOSES])) { // 1.10.0 isset incrementCount(KEYOSES, $tc_os); } switch (true) { case $blacklisted: break; case !isset ($TotalCounterBlacklist[KEYREFERERS]): break; case !isset ($tc_referer): break; case !is_array($TotalCounterBlacklist[KEYREFERERS]): break; case in_array($tc_referer, $TotalCounterBlacklist[KEYREFERERS]): break; default: foreach ($TotalCounterBlacklist[KEYREFERERS] as $value) { if (substr($value, 0, 1) == '/') if (preg_match($value, $tc_referer) > 0) $blacklisted = true; } if (!$blacklisted) incrementCount(KEYREFERERS, $tc_referer); } # end switch # if (!$blacklisted && isset ($tc_location) && !in_array($tc_location, $TotalCounterBlacklist[KEYLOCATIONS])) { // 1.10.0 isset incrementCount(KEYLOCATIONS, $tc_location); } if (!$blacklisted && defined('MULTILANGUAGE')) { if (! in_array($tc_location, $TotalCounterBlacklist[KEYLANGUAGES])) incrementCount(KEYLANGUAGES, $userlang2); } ## by MateuszCzaplinski ## last day, last week, ... - collect data if (!$blacklisted && (!$tcBotFound)) { // don't count if bot $TCnow = time(); # fix current time for duration of processing foreach ($TotalCounterTimeBins as $n=>$a) TCbins($n, $a['max'], $a['atom'], $TCnow); $TotalCounter[KEYLASTTIMESTAMP] = $TCnow; $TotalCounter[KEYTOTALCOUNTERVERSION] = TOTALCOUNTERV; } dbexport_unlock($statfilename, serialize($TotalCounter), 'w'); } else { // could not acquire a lockfile // check if the lockfile isn't a stale one, try to delete it if so dblock_remove_stale($statfilename); } # end if browse } else { $TotalCount = $TotalCounter[KEYTOTAL]; $PageCount = empty ($TotalCounter[KEYPAGES][$tc_pagename]) ? 0 : $TotalCounter[KEYPAGES][$tc_pagename]; ## by Schlaefer (==?) - fixed incrementCount(KEYPAGESTODAYCOUNTER, $tc_pagename); if (empty ($TotalCounter[KEYPAGESTODAYDAY][$tc_pagename])) { $TotalCounter[KEYPAGESTODAYDAY][$tc_pagename] = date("%y%m%d"); } } # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= //add the {$PageCount} and {$TotalCount} markup $FmtPV['$PageCount'] = "'" . number_format ($PageCount) . "'"; // return count for current page $FmtPV['$TotalCount'] = "'" . number_format ((float) $TotalCount) . "'"; // return total count for wiki ## by Schlaefer ## adds vars for the input form $FmtPV['$TotalCounterMaxItems'] = "'" . (!empty($_REQUEST['TotalCounterMaxItems']) ? $_REQUEST['TotalCounterMaxItems'] : $TotalCounterMaxItems) . "'"; # //add the {$PageViews} page variable (this appears to duplicate $PageCount above) $FmtPV['$PageViews'] = 'number_format ($GLOBALS["TotalCounter"]["Pages"][$pagename])'; # allows use in PageLists ## by Schlaefer ## add the {$PagesTodayCounter} page variable $FmtPV['$PageCountToday'] = 'number_format ($GLOBALS["TotalCounter"]["PagesTodayCounter"][$pagename])'; # allows use in PageLists return; # finished TotalCounter processing //===================================================================================================================== function incrementCount (string $countType, $counter) { // create variable if it does not exist to avoid PHP 8 errors # counter may be of type string or int global $TotalCounter; if (empty($TotalCounter[$countType])) { // If not, initialise it as an empty array $TotalCounter[$countType] = []; } if (empty ($TotalCounter[$countType][$counter])) { $TotalCounter[$countType][$counter] = 1; } else { $TotalCounter[$countType][$counter]++; } } # end incrementCount //===================================================================================================================== function HandleTotalCounter(string $pagename, string $auth = 'read') { // handle PmWiki action=totalcounter global $action, $TotalCounter, $TotalCounterMaxItems, $TotalCounterBarColor, $TotalCounterShowNumbers; global $TotalCount, $TotalCounterEnableDownload, $TotalCounterDownloads, $TotalCounterTimeBins, $TotalCounterBinsFmt, $TotalCounterEnableUsers; global $PageStartFmt, $PageEndFmt, $MessagesFmt, $totalCounter_debugOn, $TotalCounterLog, $logfilehandle, $tcPageExists; //$page = RetrieveAuthPage($pagename, $auth, true, READPAGE_CURRENT); $page = \RetrieveAuthPage($pagename, $auth); # PmWiki function if (!$page) { \Abort('?you are not permitted to perform this action "' . $action . '"'); # PmWiki function } if (!$tcPageExists) { \Abort('?action "'. $action . '" is not available on a non-existent page "' . $pagename . '"'); # PmWiki function } # $all_locations = SetAllLocations (); # top level domains #$Action = 'TotalCounter statistics'; ## by Schlaefer ## sets the max items if provided by the form if (array_key_exists ('TotalCounterMaxItems', $_REQUEST)) $TotalCounterMaxItems = $_REQUEST['TotalCounterMaxItems']; $html = '

Total Counter $[statistics]

' . NL; //------------------------------------------------------------------------------------------------------------ // PAGES $html .= graphHead ('$[Page] $[views]', true); arsort($TotalCounter[KEYPAGES]); // sort high to low $tar = array_slice($TotalCounter[KEYPAGES], 0, $TotalCounterMaxItems); $tar2 = array_slice($TotalCounter[KEYPAGES], $TotalCounterMaxItems, $TotalCounterMaxItems); $tot = $TotalCount; $max = current($tar); $i = 0; if (is_array($tar) && $tot) // by Florian Xaver foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, true, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); if (is_array($tar2)) { $html .= '
'; $html .= '' . '$[More] $[pages]' . '' . NL; $html .= graphHead ('$[More] $[pages]', true); foreach ($tar2 as $pn => $cnt) { $html .= graphLine ($pn, true, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); $html .= '
'; } ## by Schlaefer //------------------------------------------------------------------------------------------------------------ ## PAGES daily $html .= graphHead ('$[Page] $[views] $[today]', true); $pageviews = array (); foreach ($TotalCounter[KEYPAGESTODAYCOUNTER] as $pn => $cnt) { if ($TotalCounter[KEYPAGESTODAYDAY][$pn] === date("%y%m%d")) $pageviews[$pn] = $cnt; } arsort($pageviews); $tot = array_sum($pageviews); $tar = array_slice($pageviews, 0, $TotalCounterMaxItems); $tar2 = array_slice($pageviews, $TotalCounterMaxItems, $TotalCounterMaxItems); $max = current($tar); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, true, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); if (is_array($tar2)) { $html .= '
'; $html .= '' . '$[More] $[Page] $[views]' . '' . NL; $html .= graphHead ('$[More] $[page] $[views]', true); foreach ($tar2 as $pn => $cnt) { $html .= graphLine ($pn, true, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); $html .= '
'; } //------------------------------------------------------------------------------------------------------------ // USERS # TotalCounterEnableUsers if ($TotalCounterEnableUsers == 1) { $html .= graphHead ('$[Users]', true); arsort($TotalCounter[KEYUSERS]); $tar = array_slice($TotalCounter[KEYUSERS], 0, $TotalCounterMaxItems); $max = current($tar); $tot = array_sum($TotalCounter[KEYUSERS]); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); } //------------------------------------------------------------------------------------------------------------ // LANGUAGES if (defined('MULTILANGUAGE')) { $html .= graphHead ('$[Languages]', true); arsort($TotalCounter[KEYLANGUAGES]); $tar = array_slice($TotalCounter[KEYLANGUAGES], 0, $TotalCounterMaxItems); $max = current($tar); $tot = array_sum($TotalCounter[KEYLANGUAGES]); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); } //------------------------------------------------------------------------------------------------------------ // BROWSERS $html .= graphHead ('$[Browsers]', true); if (is_array($TotalCounter[KEYBROWSERS])) { arsort ($TotalCounter[KEYBROWSERS]); $tot = array_sum($TotalCounter[KEYBROWSERS]); $tar = array_slice($TotalCounter[KEYBROWSERS], 0, $TotalCounterMaxItems); } else { # should never happen $tar = []; $tot = 0; } $max = current($tar); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); //------------------------------------------------------------------------------------------------------------ // OPERATING SYSTEMS $html .= graphHead ('$[Operating systems]', true); if (is_array($TotalCounter[KEYOSES])) { arsort($TotalCounter[KEYOSES]); $tar = array_slice($TotalCounter[KEYOSES], 0, $TotalCounterMaxItems); $tot = array_sum($TotalCounter[KEYOSES]); } else { # should never happen $tar = []; $tot = 0; } $max = current($tar); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); //------------------------------------------------------------------------------------------------------------ // REFERERS $html .= graphHead ('$[Referers]', true); if (is_array($TotalCounter[KEYREFERERS])) { arsort($TotalCounter[KEYREFERERS]); $tar = array_slice($TotalCounter[KEYREFERERS], 0, $TotalCounterMaxItems); $tot = array_sum($TotalCounter[KEYREFERERS]); } else { # should never happen $tar = []; $tot = 0; } $max = current($tar); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); //------------------------------------------------------------------------------------------------------------ // LOCATIONS $html .= graphHead ('$[Locations]', true); if (is_array($TotalCounter[KEYLOCATIONS])) { arsort($TotalCounter[KEYLOCATIONS]); $tar = array_slice($TotalCounter[KEYLOCATIONS], 0, $TotalCounterMaxItems); $tot = array_sum($TotalCounter[KEYLOCATIONS]); } else { # should never happen $tar = []; $tot = 0; } $max = current($tar); $i = 0; if (is_array($tar)) foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); //------------------------------------------------------------------------------------------------------------ // WEB BOTS $html .= graphHead ('$[Web bots]', true); arsort($TotalCounter[KEYBOTS]); $tar = array_slice($TotalCounter[KEYBOTS], 0, $TotalCounterMaxItems); $tar2 = array_slice($TotalCounter[KEYBOTS], $TotalCounterMaxItems, $TotalCounterMaxItems); $max = current($tar); $tot = array_sum($TotalCounter[KEYBOTS]); $i = 0; if (is_array($tar)) { foreach ($tar as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } } $html .= graphHead ('', false); if (is_array($tar2)) { $html .= '
'; $html .= '' . '$[More] $[Web bots]' . '' . NL; $html .= graphHead ('$[More] $[Web bots]', true); foreach ($tar2 as $pn => $cnt) { $html .= graphLine ($pn, false, $cnt, $tot, $max, ++ $i); } $html .= graphHead ('', false); $html .= '
'; } //------------------------------------------------------------------------------------------------------------ // Downloads if ($TotalCounterEnableDownload == 1) { $html .= graphHead ('$[File] $[downloads]', true); arsort($TotalCounterDownloads); $max = count($TotalCounterDownloads); $tot = array_sum($TotalCounterDownloads); $i = 0; if (is_array($TotalCounterDownloads)) { for ($row = 0; $row < $max; $row++) { $tablerow = each($TotalCounterDownloads); $value = $tablerow['value']; $html .= '' . ($TotalCounterShowNumbers ? TD1 . ++ $i . '.' : '') . '' . $tablerow['key'] . '' . '' . ' ' . $value . '' . '' . NL; } } $html .= graphHead ('', false); } //------------------------------------------------------------------------------------------------------------ // Time statistics ## by MateuszCzaplinski foreach( $TotalCounterTimeBins as $n=>$a ) { $name = $a[GRAPHNAME]; $dateFmt = $a[DATEFMT]; $html .= BR . '
' . NL . '

' . $name. '

' . NL . '' . NL . '' . '' . NL . '' . NL; $rows = []; # initialise $maxcount = max($TotalCounter[$n]); $direction = 'width'; $maxnr = $a['max']; $atom = $a['atom']; for ($nr = 0; $nr < $maxnr; $nr++) { if (!isset ($TotalCounter[$n][$nr])) continue; # does not exist $count = $TotalCounter[$n][$nr]; $row = ''; # initialise if (($n !== PREVIOUSYEARS) || ($n === PREVIOUSYEARS && $count > 0)) { # should improve this to cater for varying number of days in a month // Directly evaluate the format function to display date column $output = $dateFmt(time(), $atom, $maxnr, $nr); $row .= "' . NL; $row .= "" . NL; # display count column $row .= "' . NL; $rows[] = $row; } } # end for $nr $html .= ''.implode(''.NL.'',$rows).'
' . $name . ' $[Count]
 " . $output . '" . NL; $row .= "
" . NL; $row .= "
" . number_format($count) . '
' . NL; // 1.10.0 } //------------------------------------------------------------------------------------------------------------ if ($totalCounter_debugOn) { $html .= '

Messages

' . NL . $MessagesFmt [TOTALCOUNTERNAME] . NL; } $html .= '

TotalCounter ' . TOTALCOUNTERV . '

' . NL; \PrintFmt($pagename, array ( # PmWiki function & $PageStartFmt, $html, & $PageEndFmt )); if ($TotalCounterLog) { $fclosestatus = fclose($logfilehandle); if ($fclosestatus === false) { tcmsg ('fclose failed', '"' . $logfilename, error_get_last()); if ($totalCounter_debugOn) \Abort ($MessagesFmt [TOTALCOUNTERNAME]); } } return; } # end HandleTotalCounter // ## by MateuszCzaplinski ## Manages an array of counters, each for a specified time interval. ## In $TotalCounter[$name] array there are $max counters. Each counter ## is for time interval of $atom length. ## Note: if $atom is a number, it is a length of interval measured ## in seconds. If $atom is a string, it means date($atom) is executed ## and the result is the index of an interval. ## NOTE: See TODO below //===================================================================================================================== function tc_file_get_contents ($filename) { \Lock(1); # acquire shared lock if (function_exists("file_get_contents")) { $contents = file_get_contents ($filename); } else { $contents = file($filename); if (!FALSE === $contents) { $contents = implode('', $contents); } } \Lock(0); # release lock return $contents; } # end function tc_file_get_contents //===================================================================================================================== function TCbins(string $name, int $max, $atom, $TCnow) { global $TotalCounter; $lastTS = $TotalCounter[KEYLASTTIMESTAMP]; if( $TCnow < $lastTS ) return; // some error? if( !$lastTS ) $TotalCounter[$name] = array_fill(0,$max,0); if( is_string($atom) ) { $diff = intval (date($atom,$TCnow)) - intval (date($atom,$lastTS)); if( $diff < 0 ) $diff += $max; # TODO: handle time delta > $max # Until fixed, if the site has no visitor for about a # year, statistics will get falsified (empty years will compress) } else $diff = intval ($TCnow/$atom) - intval ($lastTS/$atom); if( $diff < 0 ) return; if( $diff > 0 ) { $a = array_slice($TotalCounter[$name], $diff, max(0,$max-$diff)); if(!$a) $a = array(); $a = array_pad($a, $max, 0); $TotalCounter[$name] = $a; } incrementCount($name, $max-1); // put our visit in last bin } # end function TCbins //===================================================================================================================== // generate graph head and tail function graphHead (string $headline='', bool $heading=true):string { global $TotalCounterShowNumbers; $retval = ''; if ($heading) { # create table heading $retval .= BR . '

' . $headline . '

' . NL; $retval .= '' . NL; $retval .= '' . '' . '' . NL; $retval .= '' . NL; } else { # create table footer $retval .= '
' . $headline . ' $[Percent]$[Count]
' . NL; } return $retval; } # end function graphHead //===================================================================================================================== // generate graph line function graphLine (string $pname, bool $bpage, int $pcnt, int $ptotal, int $pmax, int $linenr) :string { global $TotalCounterShowNumbers; $retval = '' . NL; if ($TotalCounterShowNumbers) { $retval .= '' . $linenr . '.'; } $retval .= ''; if ($bpage) { $retval .= "$pname "; } else { $retval .= $pname; } $retval .= '' . NL; if (0 == $perCent = Round(100 * $pcnt / $ptotal)) $perCent = Round(100 * $pcnt / $ptotal, 1); $retval .= '' . $perCent . '%' . NL; $retval .= '
'; $retval .= ' ' . number_format ($pcnt) . ''; return $retval . '' . NL; } # end function graphLine //===================================================================================================================== // https://www.exakat.io/en/prevent-multiple-php-scripts-at-the-same-time/ // https://stackoverflow.com/questions/6967553/php-flock-alternative // Modified (breaks and returns 0 on failure, // or returns 1 on success) by Mateusz Czaplinski, 22.01.2008 function acquireLock(string $wp) { //Check if lock doesn't exist or our target is unwritable if (!is_writable($wp)) return FALSE; $lfName = "$wp.l"; \Lock(2); # lock exclusive; stop race condition if(file_exists($lfName)) { $retVal = FALSE; } else { //create the lock - hide warnings and pass empty if already created from racing $retVal = fopen($lfName, 'x'); } \Lock(0); # release lock return $retVal; } //===================================================================================================================== function dblock(string $wp):bool { global $TotalCounterEnableChmods; //Check for lockfile handle - if empty , another process raced the lock so report a failure $ftw = acquireLock($wp); if( $ftw === FALSE) return FALSE; if($TotalCounterEnableChmods) chmod($wp, 0444); //set the target file to read-only $fwStatus = fwrite($ftw, 'lock'); //write the lockfile with 4bytes if($TotalCounterEnableChmods) chmod("$wp.l", 0444); //set the lockfile to read only (OPTIONAL) fclose($ftw); //close our lockfile clearstatcache(); //Clear the stat cache return TRUE; } //===================================================================================================================== // Note: don't call it if 'dblock()' returned 0 ! function dbexport_unlock(string $wp, $data, string $meth) { global $TotalCounterEnableChmods; if($TotalCounterEnableChmods) chmod($wp, 0666); //Set the target file to read+write //Write the passed string to the target file then close \Lock(2); # acquire exclusive lock fwrite($ftw = fopen($wp, $meth), $data); fclose($ftw); \Lock(0); # release lock //Validate the written data using a string comparison $check = tc_file_get_contents($wp); if ($check != $data) echo "Data Mismatch - Locking FAILED!". BR; chmod("$wp.l", 0666); //Set the lockfile to read+write (OPTIONAL) unlink("$wp.l"); //Release the lockfile by removing it } //===================================================================================================================== function dblock_remove_stale(string $wp) { if (false === $modTime = filemtime("$wp.l")) return; // 75 minutes - to make absolutely sure we're not tricked by Daylight // Savings on Windows - see https://www.php.net/manual/en/function.stat.php#58404 if ($modTime+(75*60) < time()) unlink("$wp.l"); } //===================================================================================================================== function detectWebBot (string $HttpUserAgent): string { // find web bot https://developers.whatismybrowser.com/useragents/explore/, https://bigdata-madesimple.com/top-50-open-source-web-crawlers-for-data-mining/ $tc_bot = NULLUSERAGENT; $tc_bot_info = ''; # initialise if (preg_match('/ia_archiver/i', $HttpUserAgent)) {$tc_bot = 'Alexa'; $tc_bot_info = 'https://support.alexa.com/hc/en-us/articles/200450194-Alexa-s-Web-and-Site-Audit-Crawlers';} elseif (preg_match('/360Spider/i', $HttpUserAgent)) {$tc_bot = '360Spider';} # elseif (preg_match('/A6-Indexer/i', $HttpUserAgent)) {$tc_bot = 'A6'; $tc_bot_info = 'http://www.a6corp.com/a6-web-scraping-policy/';} elseif (preg_match('/Abonti/i', $HttpUserAgent)) {$tc_bot = 'Abonti'; $tc_bot_info = 'http://www.abonti.com';} elseif (preg_match('/acebookexternalhit/i', $HttpUserAgent)) {$tc_bot = 'Facebook External hit'; $tc_bot_info = 'http://www.facebook.com/externalhit_uatext.php';} elseif (preg_match('/Adsbot/i', $HttpUserAgent)) {$tc_bot = 'Adsbot'; $tc_bot_info = 'https://seostar.co/robot/';} elseif (preg_match('/AhrefsBot/i', $HttpUserAgent)) {$tc_bot = 'Ahrefs'; $tc_bot_info = 'https://ahrefs.com/robot/';} elseif (preg_match('/aiohttp/i', $HttpUserAgent)) {$tc_bot = 'aiohttp';} # elseif (preg_match('/ALittle/i', $HttpUserAgent)) {$tc_bot = 'ALittle Client';} # elseif (preg_match('/AnyEvent/i', $HttpUserAgent)) {$tc_bot = 'AnyEvent'; $tc_bot_info = 'http://software.schmorp.de/pkg/AnyEvent';} elseif (preg_match('/AppEngine-Google/i', $HttpUserAgent)) {$tc_bot = 'AppEngine-Google'; $tc_bot_info = 'http://code.google.com/appengine';} elseif (preg_match('/applebot/i', $HttpUserAgent)) {$tc_bot = 'Applebot'; $tc_bot_info = 'https://support.apple.com/en-us/HT204683';} elseif (preg_match('/archive.org_bot/i', $HttpUserAgent)) {$tc_bot = 'Web archive'; $tc_bot_info = 'https://webarchive.jira.com/wiki/display/ARIH/Robots+Exclusion+Protocol';} elseif (preg_match('/AntBot/i', $HttpUserAgent)) {$tc_bot = 'Ant'; $tc_bot_info = 'http://www.ant.com';} elseif (preg_match('/ask jeeves/i', $HttpUserAgent)) {$tc_bot = 'Ask Jeeves';} # elseif (preg_match('/AwarioSmartBot/i', $HttpUserAgent)) {$tc_bot = 'AwarioSmartBot'; $tc_bot_info = 'https://awario.com/bots.html';} elseif (preg_match('/baiduspider/i', $HttpUserAgent)) {$tc_bot = 'Baidu'; $tc_bot_info = 'http://www.baidu.com/search/spider.html';} elseif (preg_match('/becomebot/i', $HttpUserAgent)) {$tc_bot = 'Become';} # elseif (preg_match('/bibalex.org_bot/i', $HttpUserAgent)) {$tc_bot = 'Bibalex'; $tc_bot_info = 'http://archive.bibalex.org/bot/';} elseif (preg_match('/bingbot/i', $HttpUserAgent)) {$tc_bot = 'Bing'; $tc_bot_info = 'http://www.bing.com/bingbot.htm';} elseif (preg_match('/BLEXBot/i', $HttpUserAgent)) {$tc_bot = 'WebMeUp'; $tc_bot_info = 'http://webmeup-crawler.com/';} elseif (preg_match('/Bytespider/i', $HttpUserAgent)) {$tc_bot = 'Bytespider'; $tc_bot_info = 'spider-feedback@bytedance.com';} elseif (preg_match('/CATExplorador/i', $HttpUserAgent)) {$tc_bot = 'CATExplorador'; $tc_bot_info = 'https://domini.cat/catexplorador/';} elseif (preg_match('/CCBot/i', $HttpUserAgent)) {$tc_bot = 'Common Crawl'; $tc_bot_info = 'http://commoncrawl.org/faqs/';} elseif (preg_match('/CensysInspect/i', $HttpUserAgent)) {$tc_bot = 'Censys Inspect'; $tc_bot_info = 'https://about.censys.io/';} elseif (preg_match('/centuryb/i', $HttpUserAgent)) {$tc_bot = 'centuryb'; $tc_bot_info = 'centuryb.o.t9@gmail.com';} elseif (preg_match('/ChatGPT-User/i', $HttpUserAgent)) {$tc_bot = 'ChatGPT'; $tc_bot_info = 'https://openai.com/bot';} elseif (preg_match('/CheckMarkNetwork/i', $HttpUserAgent)) {$tc_bot = 'CheckMark Network'; $tc_bot_info = 'http://www.checkmarknetwork.com/spider.html';} elseif (preg_match('/Cincraw/i', $HttpUserAgent)) {$tc_bot = 'Cincraw'; $tc_bot_info = 'http://cincrawdata.net/bot/';} elseif (preg_match('/ClaudeBot/i', $HttpUserAgent)) {$tc_bot = 'ClaudeBot'; $tc_bot_info = 'claudebot@anthropic.com';} elseif (preg_match('/coccocbot/i', $HttpUserAgent)) {$tc_bot = 'Coccocbot'; $tc_bot_info = 'http://help.coccoc.com/searchengine';} elseif (preg_match('/colly/i', $HttpUserAgent)) {$tc_bot = 'Colly'; $tc_bot_info = 'https://github.com/gocolly/colly/v2';} elseif (preg_match('/Crawlbot\/Nutch/i', $HttpUserAgent)) {$tc_bot = 'Crawlbot/Nutch'; $tc_bot_info = 'https://nutch.apache.org/';} elseif (preg_match('/Crawlson/i', $HttpUserAgent)) {$tc_bot = 'Crawlson'; $tc_bot_info = 'https://www.crawlson.com/about';} elseif (preg_match('/Dalvik/i', $HttpUserAgent)) {$tc_bot = 'Dalvik'; } # elseif (preg_match('/DataForSeoBot/i', $HttpUserAgent)) {$tc_bot = 'DataForSeoBot'; $tc_bot_info = 'https://dataforseo.com/dataforseo-bot';} elseif (preg_match('/Daum/i', $HttpUserAgent)) {$tc_bot = 'Daum'; $tc_bot_info = 'http://cs.daum.net/faq/15/4118.html?faqId=28966';} elseif (preg_match('/deepnoc/i', $HttpUserAgent)) {$tc_bot = 'Deepnoc'; $tc_bot_info = 'https://deepnoc.com/bot';} elseif (preg_match('/Discordbot/i', $HttpUserAgent)) {$tc_bot = 'Discordbot'; $tc_bot_info = 'https://discordapp.com';} elseif (preg_match('/DIVD/i', $HttpUserAgent)) {$tc_bot = 'DIVD'; $tc_bot_info = 'https://csirt.divd.nl/';} elseif (preg_match('/Domains Project/i', $HttpUserAgent)) {$tc_bot = 'Domains Project'; $tc_bot_info = 'https://domainsproject.org/';} elseif (preg_match('/DomainStatsBot/i', $HttpUserAgent)) {$tc_bot = 'DomainStatsBot'; $tc_bot_info = 'https://domainstats.com/pages/our-bot';} elseif (preg_match('/dotbot/i', $HttpUserAgent)) {$tc_bot = 'DotBot'; $tc_bot_info = 'http://www.opensiteexplorer.org/dotbot';} elseif (preg_match('/DuckDuckGo-Favicons-Bot/i', $HttpUserAgent)) {$tc_bot = 'DuckDuckGo-Favicons-Bot'; $tc_bot_info = 'http://duckduckgo.com';} elseif (preg_match('/DuckDuckBot/i', $HttpUserAgent)) {$tc_bot = 'DuckDuckBot'; $tc_bot_info = 'https://duckduckgo.com/duckduckbot';} elseif (preg_match('/Dy robot/i', $HttpUserAgent)) {$tc_bot = 'Dy robot';} # elseif (preg_match('/EntferBot/i', $HttpUserAgent)) {$tc_bot = 'EntferBot'; $tc_bot_info = 'https://entfer.com';} elseif (preg_match('/exabot/i', $HttpUserAgent)) {$tc_bot = 'Exalead';} # elseif (preg_match('/Expanse/i', $HttpUserAgent)) {$tc_bot = 'Expanse'; $tc_bot_info = 'scaninfo@expanseinc.com';} elseif (preg_match('/facebookexternalhit/i', $HttpUserAgent)) {$tc_bot = 'Facebook'; $tc_bot_info = 'http://www.facebook.com/externalhit_uatext.php';} elseif (preg_match('/fast/i', $HttpUserAgent)) {$tc_bot = 'Fast/Alltheweb';} # elseif (preg_match('/gigabot/i', $HttpUserAgent) // http://www.gigablast.com/spider.html || preg_match('/GigablastOpenSource/i', $HttpUserAgent)) {$tc_bot = 'Gigablast'; $tc_bot_info = 'https://github.com/gigablast/open-source-search-engine';} elseif (preg_match('/Go-http-client/i', $HttpUserAgent)) {$tc_bot = 'Go-http-client'; $tc_bot_info = 'Go-http-client';} elseif (preg_match('/googlebot/i', $HttpUserAgent)) {$tc_bot = 'Google'; $tc_bot_info = 'http://www.google.com/bot.html';} elseif (preg_match('/GPTBot/i', $HttpUserAgent)) {$tc_bot = 'GPTBot'; $tc_bot_info = 'https://openai.com/gptbot';} elseif (preg_match('/Grammarly/i', $HttpUserAgent)) {$tc_bot = 'Grammarly'; $tc_bot_info = 'http://www.grammarly.com';} elseif (preg_match('/hgfAlphaXCrawl/i', $HttpUserAgent)) {$tc_bot = 'hgfAlphaXCrawl'; $tc_bot_info = 'https://www.fim.uni-passau.de/data-science/forschung/open-search';} elseif (preg_match('/InfoTigerBot/i', $HttpUserAgent)) {$tc_bot = 'InfoTigerBot'; $tc_bot_info = 'https://infotiger.com/bot';} elseif (preg_match('/InternetMeasurement/i', $HttpUserAgent)) {$tc_bot = 'InternetMeasurement'; $tc_bot_info = 'https://internet-measurement.com/';} elseif (preg_match('/James BOT/i', $HttpUserAgent)) {$tc_bot = 'CognitiveSEO'; $tc_bot_info = 'http://cognitiveseo.com/bot.html';} elseif (preg_match('/libwww-perl/i', $HttpUserAgent)) {$tc_bot = 'Libwww-perl'; $tc_bot_info = 'libwww-perl';} elseif (preg_match('/linkdexbot/i', $HttpUserAgent)) {$tc_bot = 'Linkdex'; $tc_bot_info = 'http://www.linkdex.com/bots/';} elseif (preg_match('/linkfluence/i', $HttpUserAgent)) {$tc_bot = 'Linkfluence'; $tc_bot_info = 'https://linkfluence.com/';} elseif (preg_match('/Mediapartners-Google/i', $HttpUserAgent)) {$tc_bot = 'Google'; $tc_bot_info = 'https://support.google.com/webmasters/answer/1061943?hl=en';} elseif (preg_match('/AdsBot-Google/i', $HttpUserAgent)) {$tc_bot = 'Google';} # elseif (preg_match('/grub-client/i', $HttpUserAgent)) {$tc_bot = 'Grub';} # elseif (preg_match('/java/i', $HttpUserAgent)) {$tc_bot = 'Java';} # elseif (preg_match('/libcurl/i', $HttpUserAgent)) {$tc_bot = 'cURL';} # elseif (preg_match('/curl/i', $HttpUserAgent)) {$tc_bot = 'cURL';} # elseif (preg_match('/slurp@inktomi.com/i', $HttpUserAgent)) {$tc_bot = 'Inktomi';} # elseif (preg_match('/Keybot Translation-Search-Machine/i', $HttpUserAgent)) {$tc_bot = 'Keybot Translation-Search-Machine';} elseif (preg_match('/Knowledge AI/i', $HttpUserAgent)) {$tc_bot = 'Knowledge AI';} # elseif (preg_match('/Linespider/i', $HttpUserAgent)) {$tc_bot = 'Linespider'; $tc_bot_info = 'https://lin.ee/4dwXkTH';} elseif (preg_match('/ltx71/i', $HttpUserAgent)) {$tc_bot = 'ltx71'; $tc_bot_info = 'http://ltx71.com/';} elseif (preg_match('/Mail\.RU_Bot/i', $HttpUserAgent)) {$tc_bot = 'mail.ru'; $tc_bot_info = 'http://go.mail.ru/help/robots';} elseif (preg_match('/marginalia/i', $HttpUserAgent)) {$tc_bot = 'marginalia'; $tc_bot_info = 'https://search.marginalia.nu';} elseif (preg_match('/meanpathbot/i', $HttpUserAgent)) {$tc_bot = 'meanpath';} # elseif (preg_match('/MJ12bot/i', $HttpUserAgent)) {$tc_bot = 'Majestic'; $tc_bot_info = 'http://www.majestic12.co.uk/bot.php';} elseif (preg_match('/MojeekBot/i', $HttpUserAgent)) {$tc_bot = 'MojeekBot'; $tc_bot_info = 'https://www.mojeek.com/bot.html';} elseif (preg_match('/msnbot/i', $HttpUserAgent)) {$tc_bot = 'MSN';} # elseif (preg_match('/NerdyBot/i', $HttpUserAgent)) {$tc_bot = 'Nerdy data';} # elseif (preg_match('/NetcraftSurveyAgent/i', $HttpUserAgent)) {$tc_bot = 'Netcraft'; $tc_bot_info = 'info@netcraft.com';} elseif (preg_match('/Netcraft Web Server Survey/i', $HttpUserAgent)) {$tc_bot = 'Netcraft'; } # elseif (preg_match('/NetpeakCheckerBot/i', $HttpUserAgent)) {$tc_bot = 'NetpeakCheckerBot'; $tc_bot_info = 'https://netpeaksoftware.com/checker';} elseif (preg_match('/NetSystemsResearch/i', $HttpUserAgent)) {$tc_bot = 'NetSystemsResearch'; $tc_bot_info = 'https://netsystemsresearch.com';} elseif (preg_match('/Neevabot/i', $HttpUserAgent)) {$tc_bot = 'Neevabot'; $tc_bot_info = 'https://neeva.com/neevabot';} elseif (preg_match('/Nimo Software/i', $HttpUserAgent)) {$tc_bot = 'Nimo Software HTTP Retriever';} # elseif (preg_match('/NLNZ_IAHarvester/i', $HttpUserAgent)) {$tc_bot = 'NLNZ_IAHarvester'; $tc_bot_info = 'https://natlib.govt.nz/publishers-and-authors/web-harvesting/domain-harvest';} elseif (preg_match('/node-fetch/i', $HttpUserAgent)) {$tc_bot = 'Node-fetch'; $tc_bot_info = 'https://github.com/bitinn/node-fetch';} elseif (preg_match('/OAI-SearchBot/i', $HttpUserAgent)) {$tc_bot = 'OpenAI'; $tc_bot_info = 'https://openai.com/searchbot';} elseif (preg_match('/Pandalytics/i', $HttpUserAgent)) {$tc_bot = 'Pandalytics'; $tc_bot_info = 'https://domainsbot.com/pandalytics/';} elseif (preg_match('/panscient/i', $HttpUserAgent)) {$tc_bot = 'Panscient'; $tc_bot_info = 'panscient.com';} elseif (preg_match('/PerplexityBot/i', $HttpUserAgent)) {$tc_bot = 'PerplexityBot'; $tc_bot_info = 'https://docs.perplexity.ai/docs/perplexity-bot';} elseif (preg_match('/petalbot/i', $HttpUserAgent)) {$tc_bot = 'Petalbot'; $tc_bot_info = 'https://petalsearch.com/';} elseif (preg_match('/PHP-Curl-Class/i', $HttpUserAgent)) {$tc_bot = 'PHP-Curl-Class'; $tc_bot_info = 'https://github.com/php-curl-class/php-curl-class';} elseif (preg_match('/Pinterest/i', $HttpUserAgent)) {$tc_bot = 'Pinterest'; $tc_bot_info = 'http://www.pinterest.com';} elseif (preg_match('/PocketParser/i', $HttpUserAgent)) {$tc_bot = 'PocketParser'; $tc_bot_info = 'https://getpocket.com/pocketparser_ua';} elseif (preg_match('/python-requests/i', $HttpUserAgent)) {$tc_bot = 'Python-requests';} // # python-requests elseif (preg_match('/python-urllib/i', $HttpUserAgent)) {$tc_bot = 'Python-urllib';} // # python-urllib elseif (preg_match('/Qwantify/i', $HttpUserAgent)) {$tc_bot = 'Qwantify'; $tc_bot_info = 'https://www.qwant.com/';} elseif (preg_match('/RepoLookoutBot/i', $HttpUserAgent)) {$tc_bot = 'RepoLookoutBot'; $tc_bot_info = 'abuse reports to abuse@repo-lookout.org';} elseif (preg_match('/SafeDNSBot/i', $HttpUserAgent)) {$tc_bot = 'SafeDNSBot'; $tc_bot_info = 'https://www.safedns.com/searchbot';} elseif (preg_match('/Scrapy/i', $HttpUserAgent)) {$tc_bot = 'Scrapy'; $tc_bot_info = 'https://scrapy.org';} elseif (preg_match('/scooter/i', $HttpUserAgent)) {$tc_bot = 'Altavista'; $tc_bot_info = 'deprecated http://www.siteware.ch/webresources/useragents/spiders/altavista.html';} elseif (preg_match('/Screaming/i', $HttpUserAgent)) {$tc_bot = 'Screaming Frog SEO Spider'; $tc_bot_info = 'https://www.screamingfrog.co.uk/seo-spider/';} elseif (preg_match('/SISTRIX/i', $HttpUserAgent)) {$tc_bot = 'SISTRIX'; $tc_bot_info = 'http://crawler.007ac9.net/';} elseif (preg_match('/Crawler/i', $HttpUserAgent)) {$tc_bot = 'SISTRIX'; $tc_bot_info = 'http://crawler.007ac9.net/';} elseif (preg_match('/SeekportBot/i', $HttpUserAgent)) {$tc_bot = 'SeekportBot'; $tc_bot_info = 'https://bot.seekport.com';} elseif (preg_match('/SemrushBot/i', $HttpUserAgent)) {$tc_bot = 'SemrushBot'; $tc_bot_info = 'http://www.semrush.com/bot.html';} elseif (preg_match('/SenutoBot/i', $HttpUserAgent)) {$tc_bot = 'SenutoBot'; $tc_bot_info = 'https://www.senuto.com/';} elseif (preg_match('/SEOkicks/i', $HttpUserAgent)) {$tc_bot = 'SEOkicks'; $tc_bot_info = 'https://www.seokicks.de/robot.html';} elseif (preg_match('/serpstatbot/i', $HttpUserAgent)) {$tc_bot = 'Serpstatbot'; $tc_bot_info = 'https://serpstatbot.com/; abuse@serpstatbot.com';} elseif (preg_match('/SeznamBot/i', $HttpUserAgent)) {$tc_bot = 'SeznamBot'; $tc_bot_info = 'http://napoveda.seznam.cz/en/seznambot-intro/';} elseif (preg_match('/Sidetrade/i', $HttpUserAgent)) {$tc_bot = 'Sidetrade indexer bot'; } # elseif (preg_match('/SiteExplorer/i', $HttpUserAgent)) {$tc_bot = 'Site Explorer'; $tc_bot_info = 'http://siteexplorer.info/';} elseif (preg_match('/snapchat/i', $HttpUserAgent)) {$tc_bot = 'Snapchat URL Preview Service'; $tc_bot_info = 'https://developers.snap.com/robots';} elseif (preg_match('/Sogou/i', $HttpUserAgent)) {$tc_bot = 'Sogou web spider'; $tc_bot_info = 'http://www.sogou.com/docs/help/webmasters.htm#07';} elseif (preg_match('/SpiceworksAgentShell/i', $HttpUserAgent)) {$tc_bot = 'Spiceworks Agent Shell'; $tc_bot_info = 'https://community.spiceworks.com/support/inventory-online/docs/deploy-agent';} elseif (preg_match('/SurdotlyBot/i', $HttpUserAgent)) {$tc_bot = 'SurdotlyBot'; $tc_bot_info = ' http://sur.ly/bot.html';} elseif (preg_match('/synapse/i', $HttpUserAgent)) {$tc_bot = 'Synapse';} # elseif (preg_match('/t3versionsBot/i', $HttpUserAgent)) {$tc_bot = 'T3versionsBot'; $tc_bot_info = 'https://www.t3versions.com/bot';} elseif (preg_match('/TenMillionDomainsBot/i', $HttpUserAgent)) {$tc_bot = 'TenMillionDomainsBot'; $tc_bot = 'https://github.com/tonywangcn/ten-million-domains';} elseif (preg_match('/ThinkChaos/i', $HttpUserAgent)) {$tc_bot = 'ThinkChaos'; } # elseif (preg_match('/Timpibot/i', $HttpUserAgent)) {$tc_bot = 'Timpibot'; $tc_bot_info = 'http://www.timpi.io';} elseif (preg_match('/Turnitin/i', $HttpUserAgent)) {$tc_bot = 'Turnitin'; $tc_bot_info = 'https://bit.ly/2UvnfoQ';} elseif (preg_match('/Twisted PageGetter/i', $HttpUserAgent)) {$tc_bot = 'Twisted PageGetter'; $tc_bot_info = 'https://twistedmatrix.com/trac/';} elseif (preg_match('/Twitterbot/i', $HttpUserAgent)) {$tc_bot = 'Twitterbot'; } # elseif (preg_match('/unirest-java/i', $HttpUserAgent)) {$tc_bot = 'unirest-java'; } # elseif (preg_match('/W3C-checklink/i', $HttpUserAgent)) {$tc_bot = 'W3C-checklink'; } # elseif (preg_match('/webpage-inspector/i', $HttpUserAgent)) {$tc_bot = 'Webpage Inspector'; $tc_bot_info = 'webpage-inspector.com';} elseif (preg_match('/webprosbot/i', $HttpUserAgent)) {$tc_bot = 'Webprosbot'; $tc_bot_info = 'mailto:abuse-6337@webpros.com';} elseif (preg_match('/WebTarantula/i', $HttpUserAgent)) {$tc_bot = 'WebTarantula'; $tc_bot_info = 'http://webtarantula.com/';} elseif (preg_match('/wget/i', $HttpUserAgent)) {$tc_bot = 'wget'; $tc_bot_info = 'https://www.gnu.org/software/wget/';} elseif (preg_match('/woorankreview/i', $HttpUserAgent)) {$tc_bot = 'WooRankReview'; $tc_bot_info = 'https://www.woorank.com/';} elseif (preg_match('/wp_is_mobile/i', $HttpUserAgent)) {$tc_bot = 'Wp_is_mobile'; } # elseif (preg_match('/Xenu/i', $HttpUserAgent)) {$tc_bot = 'Xenu Link Sleuth'; } # elseif (preg_match('/XoviBot/i', $HttpUserAgent)) {$tc_bot = 'Xovi'; $tc_bot_info = 'http://www.xovibot.net/';} elseif (preg_match('/yahoo! slurp/i', $HttpUserAgent)) {$tc_bot = 'Yahoo!'; $tc_bot_info = 'http://help.yahoo.com/help/us/ysearch/slurp';} elseif (preg_match('/YandexBot/i', $HttpUserAgent)) {$tc_bot = 'Yandex'; $tc_bot_info = 'http://yandex.com/bots';} elseif (preg_match('/Yeti/i', $HttpUserAgent)) {$tc_bot = 'Yeti'; $tc_bot_info = 'http://naver.me/spd';} elseif (preg_match('/YisouSpider/i', $HttpUserAgent)) {$tc_bot = 'YisouSpider'; } # elseif (preg_match('/YottaaMonitor/i', $HttpUserAgent)) {$tc_bot = 'Yottaa'; $tc_bot_info = 'http://www.yottaa.com/blog/bid/223629/Google-Analytics-How-to-Segment-and-Filter-Robot-Traffic';} elseif (preg_match('/zyborg/i', $HttpUserAgent) || preg_match('/zealbot/i', $HttpUserAgent)) {$tc_bot = 'WiseNut!';} # elseif (preg_match('/2ip bot/i', $HttpUserAgent)) {$tc_bot = '2ip bot'; $tc_bot_info = 'http://2ip.io';} # return $tc_bot; } # end detectWebBot //===================================================================================================================== function detectBrowser (string $HttpUserAgent): string { // not a bot, so find the browser, https://developer.mozilla.org/en-US/docs/Web/HTTP/Browser_detection_using_the_user_agent $tc_browser = NULLUSERAGENT; if (preg_match('/arachne/i', $HttpUserAgent)) $tc_browser = 'Arachne GPL'; elseif (preg_match('/vivaldi/i', $HttpUserAgent)) $tc_browser = 'Vivaldi'; elseif (preg_match('/blazer/i', $HttpUserAgent)) $tc_browser = 'Blazer'; elseif (preg_match('/brave/i', $HttpUserAgent)) $tc_browser = 'Brave'; elseif (preg_match('/opera/i', $HttpUserAgent) || preg_match('/OPR/i', $HttpUserAgent)) $tc_browser = 'Opera'; # must be before Chrome below elseif (preg_match('/webtv/i', $HttpUserAgent)) $tc_browser = 'WebTV'; elseif (preg_match('/camino/i', $HttpUserAgent)) $tc_browser = 'Camino'; elseif (preg_match('/MAXTHON/i', $HttpUserAgent)) $tc_browser = 'MAXTHON'; # must be before msie below # http://www.maxthon.com/ elseif (preg_match('/netpositive/i', $HttpUserAgent)) $tc_browser = 'NetPositive'; elseif (preg_match('/internet explorer/i', $HttpUserAgent) || preg_match('/msie/i', $HttpUserAgent) || preg_match('/IEMobile/i', $HttpUserAgent) || preg_match('/mspie/i', $HttpUserAgent) || preg_match('/trident/i', $HttpUserAgent) ) $tc_browser = 'MS Internet Explorer'; # add trident elseif (preg_match('/avant browser/i', $HttpUserAgent) || preg_match('/advanced browser/i', $HttpUserAgent)) $tc_browser = 'Avant Browser'; elseif (preg_match('/galeon/i', $HttpUserAgent)) $tc_browser = 'Galeon'; elseif (preg_match('/konqueror/i', $HttpUserAgent)) $tc_browser = 'Konqueror'; elseif (preg_match('/icab/i', $HttpUserAgent)) $tc_browser = 'iCab'; elseif (preg_match('/FBAN\/FBIOS/i', $HttpUserAgent)) $tc_browser = 'Facebook In-App Browser'; elseif (preg_match('/Nmap Scripting Engine/i', $HttpUserAgent)) $tc_browser = 'Nmap'; # http://nmap.org/book/nse.html elseif (preg_match('/omniweb/i', $HttpUserAgent)) $tc_browser = 'OmniWeb'; elseif (preg_match('/phoenix/i', $HttpUserAgent)) $tc_browser = 'Phoenix'; elseif (preg_match('/firebird/i', $HttpUserAgent)) $tc_browser = 'Firebird'; elseif (preg_match('/seamonkey/i', $HttpUserAgent)) $tc_browser = 'Seamonkey'; # must be before Firefox elseif (preg_match('/firefox/i', $HttpUserAgent)) $tc_browser = 'Firefox'; elseif (preg_match('/netscape/i', $HttpUserAgent)) $tc_browser = 'Netscape'; # must be before Mozilla below elseif (preg_match('/minimo/i', $HttpUserAgent)) $tc_browser = 'Minimo'; elseif (preg_match('/mozilla/i', $HttpUserAgent) && preg_match('/rv:[0-9].[0-9][a-b]/i', $HttpUserAgent)) $tc_browser = 'Mozilla'; # elseif (preg_match('/mozilla/i', $HttpUserAgent) && preg_match('/rv:[0-9].[0-9]/i', $HttpUserAgent)) $tc_browser = 'Mozilla'; # elseif (preg_match('/YaBrowser/i', $HttpUserAgent)) $tc_browser = 'Yandex browser'; # http://help.yandex.ru/yabrowser/?lang=en elseif (preg_match('/libwww/i', $HttpUserAgent)) { if (preg_match('/amaya/i', $HttpUserAgent)) { $tc_browser = 'Amaya'; } else { $tc_browser = 'Text browser'; } } elseif (preg_match('/edge/i', $HttpUserAgent)) $tc_browser = 'Edge'; // must be before Safari elseif (preg_match('/chromium/i', $HttpUserAgent)) $tc_browser = 'Chromium'; // must be before Chrome elseif (preg_match('/chrome/i', $HttpUserAgent)) $tc_browser = 'Chrome'; // must be before Safari elseif (preg_match('/safari/i', $HttpUserAgent)) $tc_browser = 'Safari'; // must be after Chrome above elseif (preg_match('/elinks/i', $HttpUserAgent)) $tc_browser = 'ELinks'; elseif (preg_match('/offbyone/i', $HttpUserAgent)) $tc_browser = 'Off By One'; elseif (preg_match('/playstation portable/i', $HttpUserAgent)) $tc_browser = 'PlayStation Portable'; elseif (preg_match('/links/i', $HttpUserAgent)) $tc_browser = 'Links'; elseif (preg_match('/ibrowse/i', $HttpUserAgent)) $tc_browser = 'iBrowse'; elseif (preg_match('/w3m/i', $HttpUserAgent)) $tc_browser = 'w3m'; elseif (preg_match('/aweb/i', $HttpUserAgent)) $tc_browser = 'AWeb'; elseif (preg_match('/voyager/i', $HttpUserAgent)) $tc_browser = 'Voyager'; elseif (preg_match('/oregano/i', $HttpUserAgent)) $tc_browser = 'Oregano'; return $tc_browser; } # end detectBrowser //===================================================================================================================== function detectOS (string $HttpUserAgent):string { // find operating system $tc_os = NULLUSERAGENT; if (preg_match('/android/i', $HttpUserAgent)) $tc_os = 'Android'; // # must be before linux below elseif (preg_match('/linux/i', $HttpUserAgent)) $tc_os = 'Linux'; elseif (preg_match('/irix/i', $HttpUserAgent)) $tc_os = 'IRIX'; elseif (preg_match('/hp-ux/i', $HttpUserAgent)) $tc_os = 'HP-Unix'; elseif (preg_match('/os2/i', $HttpUserAgent)) $tc_os = 'OS/2'; elseif (preg_match('/beos/i', $HttpUserAgent)) $tc_os = 'BeOS'; elseif (preg_match('/sunos/i', $HttpUserAgent)) $tc_os = 'SunOS'; elseif (preg_match('/palm/i', $HttpUserAgent)) $tc_os = 'PalmOS'; elseif (preg_match('/cygwin/i', $HttpUserAgent)) $tc_os = 'Cygwin'; elseif (preg_match('/amiga/i', $HttpUserAgent)) $tc_os = 'Amiga'; elseif (preg_match('/unix/i', $HttpUserAgent)) $tc_os = 'Unix'; elseif (preg_match('/qnx/i', $HttpUserAgent)) $tc_os = 'QNX'; elseif (preg_match('/Windows Phone/i', $HttpUserAgent)) $tc_os = 'Windows Phone'; # must be before Windows below elseif (preg_match('/windows/i', $HttpUserAgent)) $tc_os = 'Windows'; # elseif (preg_match('/openbsd/i', $HttpUserAgent)) $tc_os = 'OpenBSD'; # elseif (preg_match('/iphone os/i', $HttpUserAgent)) $tc_os = 'iPhone'; # must be before Mac OS below elseif (preg_match('/mac os/i', $HttpUserAgent)) $tc_os = 'Mac'; elseif (preg_match('/cros/i', $HttpUserAgent)) $tc_os = 'Chrome OS'; elseif (preg_match('/symbian/i', $HttpUserAgent)) $tc_os = 'Symbian'; elseif (preg_match('/risc/i', $HttpUserAgent)) $tc_os = 'RISC'; elseif (preg_match('/dreamcast/i', $HttpUserAgent)) $tc_os = 'Dreamcast'; elseif (preg_match('/freebsd/i', $HttpUserAgent)) $tc_os = 'FreeBSD'; elseif (preg_match('/dos/i', $HttpUserAgent)) $tc_os = 'dos'; return $tc_os; } # end detectOS //===================================================================================================================== function SetAllLocations ():array { // Define top level domains return array ( 'localhost' => 'localhost', UNKNOWN => 'Unknown', # original top level domains 'com' => 'Commercial', 'net' => 'Networks', 'org' => 'Organizations', 'int' => 'International organizations', 'edu' => 'US higher Education', 'gov' => 'US Government', 'mil' => 'US Dept of Defense', # selected ICANN TLDs 'academy' => 'Academy', 'aero' => 'Aviation', 'biz' => 'Business organizations', 'church' => 'Churches', 'city' => 'City', 'club' => 'Clubs', 'community' => 'Community', 'coop' => 'Co-operative organizations', 'education' => 'Education insitiutes', 'info' => 'Information', 'international' => 'International entities', 'mobi' => 'mobile devices', 'museum' => 'Museums', 'name' => 'Personal', 'place' => 'Place', 'travel' => 'Travelling', 'universite' => 'University', 'wiki' => 'Wikis', # selected geographic TLDs 'africa' => 'Africa', 'asia' => 'Asia', 'berlin' => 'Berlin', 'brussels' => 'Brussels', 'kiwi' => 'Kiwi', 'london' => 'London', 'paris' => 'Paris', 'quebec' => 'Quebec', 'scot' => 'Scotland', # country code top level https://icannwiki.org/Country_code_top-level_domain # updates from https://isotc.iso.org/livelink/livelink?func=ll&objId=16944257&objAction=browse&viewType=1 'ac' => 'Ascension Island', 'ad' => 'Andorra', 'ae' => 'United Arab Emirates', 'af' => 'Afghanistan', 'ag' => 'Antigua & Barbuda', 'ai' => 'Anguilla', 'al' => 'Albania', 'am' => 'Armenia', 'an' => 'Netherlands Antilles', 'ao' => 'Angola', 'aq' => 'Antarctica', 'ar' => 'Argentina', 'as' => 'American Samoa', 'at' => 'Austria', 'au' => 'Australia', 'aw' => 'Aruba', 'ax' => 'Åland', 'az' => 'Azerbaijan', 'ba' => 'Bosnia & Herzegovina', 'bb' => 'Barbados', 'bd' => 'Bangladesh', 'be' => 'Belgium', 'bf' => 'Burkina Faso', 'bg' => 'Bulgaria', 'bh' => 'Bahrain', 'bi' => 'Burundi', 'bj' => 'Benin', 'bm' => 'Bermuda', 'bn' => 'Brunei Darussalam', 'bo' => 'Bolivia', 'br' => 'Brazil', 'bs' => 'Bahamas', 'bt' => 'Bhutan', 'bv' => 'Bouvet Island', 'bw' => 'Botswana', 'by' => 'Belarus', 'bz' => 'Belize', 'ca' => 'Canada', 'cc' => 'Cocos (Keeling) Islands', 'cd' => 'Democratic republic of Congo', 'cf' => 'Central African Republic', 'cg' => 'Congo', 'ch' => 'Switzerland', 'ci' => 'Ivory Coast', 'ck' => 'Cook Islands', 'cl' => 'Chile', 'cm' => 'Cameroon', 'cn' => 'China', 'co' => 'Colombia', 'cr' => 'Costa Rica', 'cs' => 'Czechoslovakia/Sebia & Montenegro', // deleted 'cu' => 'Cuba', 'cv' => 'Cape Verde', 'cw' => 'Curaçao', 'cx' => 'Christmas Island', 'cy' => 'Cyprus', 'cz' => 'Czech Republic', 'de' => 'Germany', 'dj' => 'Djibouti', 'dk' => 'Denmark', 'dm' => 'Dominica', 'do' => 'Dominican Republic', 'dz' => 'Algeria', 'ec' => 'Ecuador', 'ee' => 'Estonia', 'eg' => 'Egypt', 'eh' => 'Western Sahara', 'er' => 'Eritrea', 'es' => 'Spain', 'et' => 'Ethiopia', 'eu' => 'European Union', 'fi' => 'Finland', 'fj' => 'Fiji', 'fk' => 'Falkland Islands', 'fm' => 'Micronesia', 'fo' => 'Faroe Islands', 'fr' => 'France', 'ga' => 'Gabon', 'gb' => 'United Kingdom', 'gd' => 'Grenada', 'ge' => 'Georgia', 'gf' => 'French Guiana', 'gg' => 'Guernsey', 'gh' => 'Ghana', 'gi' => 'Gibraltar', 'gl' => 'Greenland', 'gm' => 'Gambia', 'gn' => 'Guinea', 'gp' => 'Guadeloupe', 'gq' => 'Equatorial Guinea', 'gr' => 'Greece', 'gs' => 'South Georgia & South Sandwich Islands', 'gt' => 'Guatemala', 'gu' => 'Guam', 'gw' => 'Guinea-Bissau', 'gy' => 'Guyana', 'hk' => 'Hong Kong', 'hm' => 'Heard & McDonald Islands', 'hn' => 'Honduras', 'hr' => 'Croatia', 'ht' => 'Haiti', 'hu' => 'Hungary', 'id' => 'Indonesia', 'ie' => 'Ireland', 'il' => 'Israel', 'im' => 'Isle of Man', 'in' => 'India', 'io' => 'British Indian Ocean Territory', 'iq' => 'Iraq', 'ir' => 'Iran', 'is' => 'Iceland', 'it' => 'Italy', 'je' => 'Jersey', 'jm' => 'Jamaica', 'jo' => 'Jordan', 'jp' => 'Japan', 'ke' => 'Kenya', 'kg' => 'Kyrgyzstan', 'kh' => 'Cambodia', 'ki' => 'Kiribati', 'km' => 'Comoros', 'kn' => 'Saint Kitts & Nevis', 'kp' => 'North Korea', 'kr' => 'South Korea', 'kw' => 'Kuwait', 'ky' => 'Cayman Islands', 'kz' => 'Kazakhstan', 'la' => 'Laos', 'lb' => 'Lebanon', 'lc' => 'Saint Lucia', 'li' => 'Liechtenstein', 'lk' => 'Sri Lanka', 'lr' => 'Liberia', 'ls' => 'Lesotho', 'lt' => 'Lithuania', 'lu' => 'Luxembourg', 'lv' => 'Latvia', 'ly' => 'Libyan Arab Jamahiriya', 'ma' => 'Morocco', 'mc' => 'Monaco', 'md' => 'Moldova', 'me' => 'Montenegro', 'mg' => 'Madagascar', 'mh' => 'Marshall Islands', 'mk' => 'North Macedonia', 'ml' => 'Mali', 'mm' => 'Myanmar', 'mn' => 'Mongolia', 'mo' => 'Macau', 'mp' => 'Northern Mariana Islands', 'mq' => 'Martinique', 'mr' => 'Mauritania', 'ms' => 'Montserrat', 'mt' => 'Malta', 'mu' => 'Mauritius', 'mv' => 'Maldives', 'mw' => 'Malawi', 'mx' => 'Mexico', 'my' => 'Malaysia', 'mz' => 'Mozambique', 'na' => 'Namibia', 'nc' => 'New Caledonia', 'ne' => 'Niger', 'nf' => 'Norfolk Island', 'ng' => 'Nigeria', 'ni' => 'Nicaragua', 'nl' => 'The Netherlands', 'no' => 'Norway', 'np' => 'Nepal', 'nr' => 'Nauru', 'nu' => 'Niue', 'nz' => 'New Zealand', 'om' => 'Oman', 'pa' => 'Panama', 'pe' => 'Peru', 'pf' => 'French Polynesia', 'pg' => 'Papua New Guinea', 'ph' => 'Philippines', 'pk' => 'Pakistan', 'pl' => 'Poland', 'pm' => 'St. Pierre & Miquelon', 'pn' => 'Pitcairn', 'pr' => 'Puerto Rico', 'ps' => 'Palestine', 'pt' => 'Portugal', 'pw' => 'Palau', 'py' => 'Paraguay', 'qa' => 'Qatar', 're' => 'Réunion', 'ro' => 'Romania', 'rs' => 'Serbia', 'ru' => 'Russia', 'rw' => 'Rwanda', 'sa' => 'Saudi Arabia', 'sb' => 'Solomon Islands', 'sc' => 'Seychelles', 'sd' => 'Sudan', 'se' => 'Sweden', 'sg' => 'Singapore', 'sh' => 'St. Helena', 'si' => 'Slovenia', 'sj' => 'Svalbard & Jan Mayen Islands', 'sk' => 'Slovakia', 'sl' => 'Sierra Leone', 'sm' => 'San Marino', 'sn' => 'Senegal', 'so' => 'Somalia', 'sr' => 'Surinam', 'st' => 'Sao Tome & Principe', 'su' => 'USSR', 'sv' => 'El Salvador', 'sy' => 'Syrian Arab Republic', 'sz' => 'Swaziland', 'tc' => 'The Turks & Caicos Islands', 'td' => 'Chad', 'tf' => 'French Southern Territories', 'tg' => 'Togo', 'th' => 'Thailand', 'tj' => 'Tajikistan', 'tk' => 'Tokelau', 'tl' => 'Timor-Leste', 'tm' => 'Turkmenistan', 'tn' => 'Tunisia', 'to' => 'Tonga', 'tp' => 'East Timor', 'tr' => 'Turkey', 'tt' => 'Trinidad & Tobago', 'tv' => 'Tuvalu', 'tw' => 'Taiwan', 'tz' => 'Tanzania', 'ua' => 'Ukraine', 'ug' => 'Uganda', 'uk' => 'United Kingdom', 'um' => 'United States Minor Outlying Islands', 'us' => 'United States', 'uy' => 'Uruguay', 'uz' => 'Uzbekistan', 'va' => 'Vatican City', 'vc' => 'Saint Vincent & the Grenadines', 've' => 'Venezuela', 'vg' => 'British Virgin Islands', 'vi' => 'US Virgin Islands', 'vn' => 'Vietnam', 'vu' => 'Vanuatu', 'wf' => 'Wallis & Futuna Islands', 'ws' => 'Samoa', 'ye' => 'Yemen', 'yt' => 'Mayotte', 'yu' => 'Yugoslavia', 'za' => 'South Africa', 'zm' => 'Zambia', 'zr' => 'Zaire', // deprecated 'zw' => 'Zimbabwe', ); } //===================================================================================================================== // Record message to PmWiki for display by (:message:) directive function tcmsg(string $smsgprefix, string $smsgdata, array $LastError = []) { global $MessagesFmt, $TotalCounterLog, $logfilehandle, $logfiletime; if (!isset ($MessagesFmt [__NAMESPACE__ . '\\' . TOTALCOUNTERNAME])) $MessagesFmt [__NAMESPACE__ . '\\' . TOTALCOUNTERNAME] = []; $TcMsgs = ''; $TcMsgs .= '' . $smsgprefix . ': ' . \PHSC($smsgdata); if (!empty ($LastError)) $TcMsgs .= ' ' . TOTALCOUNTERNAME . ' {' . implode (', ', $LastError) . '}'; $MessagesFmt [__NAMESPACE__ . '\\' . TOTALCOUNTERNAME] [] = $TcMsgs . BR; if ($TotalCounterLog) { # also write message to the totalcounter log $TcLogMsg = ''; $TcLogMsg .= $smsgprefix . '> ' . \PHSC($smsgdata); if (!empty ($LastError)) $TcLogMsg .= ' ' . TOTALCOUNTERNAME . ' {' . implode (', ', $LastError) . '}'; \Lock(2); # acquire exclusive lock $fwritestatus = fwrite($logfilehandle, $logfiletime . $TcLogMsg . NL); \Lock(0); # release lock } } # end TotalCOunter