Files
Grav/vendor/ornicar/php-user-agent/lib/phpUserAgentStringParser.php

322 lines
7.8 KiB
PHP
Raw Normal View History

<?php
/**
* Simple PHP User Agent string parser
*/
class phpUserAgentStringParser
{
/**
* Parse a user agent string.
*
* @param string $userAgentString defaults to $_SERVER['HTTP_USER_AGENT'] if empty
* @return array ( the user agent informations
* 'browser_name' => 'firefox',
* 'browser_version' => '3.6',
* 'operating_system' => 'linux'
* )
*/
public function parse($userAgentString = null)
{
// use current user agent string as default
if(!$userAgentString)
{
$userAgentString = isset($_SERVER['HTTP_USER_AGENT']) ? $_SERVER['HTTP_USER_AGENT'] : null;
}
// parse quickly (with medium accuracy)
$informations = $this->doParse($userAgentString);
// run some filters to increase accuracy
foreach($this->getFilters() as $filter)
{
$this->$filter($informations);
}
return $informations;
}
/**
* Detect quickly informations from the user agent string
*
* @param string $userAgentString user agent string
* @return array user agent informations array
*/
protected function doParse($userAgentString)
{
$userAgent = array(
'string' => $this->cleanUserAgentString($userAgentString),
'browser_name' => null,
'browser_version' => null,
'operating_system' => null,
'engine' => null
);
if(empty($userAgent['string']))
{
return $userAgent;
}
// build regex that matches phrases for known browsers
// (e.g. "Firefox/2.0" or "MSIE 6.0" (This only matches the major and minor
// version numbers. E.g. "2.0.0.6" is parsed as simply "2.0"
$pattern = '#('.join('|', $this->getKnownBrowsers()).')[/ ]+([0-9]+(?:\.[0-9]+)?)#';
// Find all phrases (or return empty array if none found)
if (preg_match_all($pattern, $userAgent['string'], $matches))
{
// Since some UAs have more than one phrase (e.g Firefox has a Gecko phrase,
// Opera 7,8 have a MSIE phrase), use the last one found (the right-most one
// in the UA). That's usually the most correct.
$i = count($matches[1])-1;
if (isset($matches[1][$i]))
{
$userAgent['browser_name'] = $matches[1][$i];
}
if (isset($matches[2][$i]))
{
$userAgent['browser_version'] = $matches[2][$i];
}
}
// Find operating system
$pattern = '#'.join('|', $this->getKnownOperatingSystems()).'#';
if (preg_match($pattern, $userAgent['string'], $match))
{
if (isset($match[0]))
{
$userAgent['operating_system'] = $match[0];
}
}
// Find engine
$pattern = '#'.join('|', $this->getKnownEngines()).'#';
if (preg_match($pattern, $userAgent['string'], $match))
{
if (isset($match[0]))
{
$userAgent['engine'] = $match[0];
}
}
return $userAgent;
}
/**
* Make user agent string lowercase, and replace browser aliases
*
* @param string $userAgentString the dirty user agent string
* @return string the clean user agent string
*/
public function cleanUserAgentString($userAgentString)
{
// clean up the string
$userAgentString = trim(strtolower($userAgentString));
// replace browser names with their aliases
$userAgentString = strtr($userAgentString, $this->getKnownBrowserAliases());
// replace operating system names with their aliases
$userAgentString = strtr($userAgentString, $this->getKnownOperatingSystemAliases());
// replace engine names with their aliases
$userAgentString = strtr($userAgentString, $this->getKnownEngineAliases());
return $userAgentString;
}
/**
* Get the list of filters that get called when parsing a user agent
*
* @return array list of valid callables
*/
public function getFilters()
{
return array(
'filterAndroid',
'filterGoogleChrome',
'filterSafariVersion',
'filterOperaVersion',
'filterYahoo',
'filterMsie',
);
}
/**
* Add a filter to be called when parsing a user agent
*
* @param string $filter name of the filter method
*/
public function addFilter($filter)
{
$this->filters += $filter;
}
/**
* Get known browsers
*
* @return array the browsers
*/
protected function getKnownBrowsers()
{
return array(
'msie',
'firefox',
'safari',
'webkit',
'opera',
'netscape',
'konqueror',
'gecko',
'chrome',
'googlebot',
'iphone',
'msnbot',
'applewebkit'
);
}
/**
* Get known browser aliases
*
* @return array the browser aliases
*/
protected function getKnownBrowserAliases()
{
return array(
'shiretoko' => 'firefox',
'namoroka' => 'firefox',
'shredder' => 'firefox',
'minefield' => 'firefox',
'granparadiso' => 'firefox'
);
}
/**
* Get known operating system
*
* @return array the operating systems
*/
protected function getKnownOperatingSystems()
{
return array(
'windows',
'macintosh',
'linux',
'freebsd',
'unix',
'iphone'
);
}
/**
* Get known operating system aliases
*
* @return array the operating system aliases
*/
protected function getKnownOperatingSystemAliases()
{
return array();
}
/**
* Get known engines
*
* @return array the engines
*/
protected function getKnownEngines()
{
return array(
'gecko',
'webkit',
'trident',
'presto'
);
}
/**
* Get known engines aliases
*
* @return array the engines aliases
*/
protected function getKnownEngineAliases()
{
return array();
}
/**
* Filters
*/
/**
* Google chrome has a safari like signature
*/
protected function filterGoogleChrome(array &$userAgent)
{
if ('safari' === $userAgent['browser_name'] && strpos($userAgent['string'], 'chrome/'))
{
$userAgent['browser_name'] = 'chrome';
$userAgent['browser_version'] = preg_replace('|.+chrome/([0-9]+(?:\.[0-9]+)?).+|', '$1', $userAgent['string']);
}
}
/**
* Safari version is not encoded "normally"
*/
protected function filterSafariVersion(array &$userAgent)
{
if ('safari' === $userAgent['browser_name'] && strpos($userAgent['string'], ' version/'))
{
$userAgent['browser_version'] = preg_replace('|.+\sversion/([0-9]+(?:\.[0-9]+)?).+|', '$1', $userAgent['string']);
}
}
/**
* Opera 10.00 (and higher) version number is located at the end
*/
protected function filterOperaVersion(array &$userAgent)
{
if('opera' === $userAgent['browser_name'] && strpos($userAgent['string'], ' version/'))
{
$userAgent['browser_version'] = preg_replace('|.+\sversion/([0-9]+\.[0-9]+)\s*.*|', '$1', $userAgent['string']);
}
}
/**
* Yahoo bot has a special user agent string
*/
protected function filterYahoo(array &$userAgent)
{
if (null === $userAgent['browser_name'] && strpos($userAgent['string'], 'yahoo! slurp'))
{
$userAgent['browser_name'] = 'yahoobot';
}
}
/**
* MSIE does not always declare its engine
*/
protected function filterMsie(array &$userAgent)
{
if ('msie' === $userAgent['browser_name'] && empty($userAgent['engine']))
{
$userAgent['engine'] = 'trident';
}
}
/**
* Android has a safari like signature
*/
protected function filterAndroid(array &$userAgent) {
if ('safari' === $userAgent['browser_name'] && strpos($userAgent['string'], 'android ')) {
$userAgent['browser_name'] = 'android';
$userAgent['operating_system'] = 'android';
$userAgent['browser_version'] = preg_replace('|.+android ([0-9]+(?:\.[0-9]+)+).+|', '$1', $userAgent['string']);
}
}
}