Newer
Older
* Piwik - free/libre analytics platform
*
* @link http://piwik.org
* @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
*/
namespace Piwik\Tests\Unit;
use Exception;
use PHPUnit_Framework_TestCase;
use Piwik\Common;
use Piwik\Filesystem;
/**
* @backupGlobals enabled
class Core_CommonTest extends PHPUnit_Framework_TestCase
{
/**
* Dataprovider for testSanitizeInputValues
*/
public function getInputValues()
{
return array( // input, output
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// sanitize an array OK
array(
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52),
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52)
),
array(
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52,
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52),
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52),
array(array(array(array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52)))
)),
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52,
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52),
array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52),
array(array(array(array('test1' => 't1', 't45', "teatae", 4568, array('test'), 1.52)))
))
),
// sanitize an array with bad value level1
array(
array('test1' => 't1', 't45', 'tea1"ta"e', 568, 1 => array('t<e"st'), 1.52),
array('test1' => 't1', 't45', 'tea1"ta"e', 568, 1 => array('t<e"st'), 1.52)
),
// sanitize an array with bad value level2
array(
array('tea1"ta"e' => array('t<e"st' => array('tgeag454554"t')), 1.52),
array('tea1"ta"e' => array('t<e"st' => array('tgeag454554"t')), 1.52)
),
// sanitize a string unicode => no change
array(
" Поиск в Интернете Поgqegиск страниц на рgeqg8978усском",
" Поиск в Интернете Поgqegиск страниц на рgeqg8978усском"
),
// sanitize a bad string
array(
'& " < > 123abc\'',
'& " < > 123abc''
),
// test filter - expect new line and null byte to be filtered out
array(
"Null\0Byte",
'NullByte'
),
// double encoded - no change (document as user error)
array(
'%48%45%4C%00%4C%4F+%57%4F%52%4C%44',
'%48%45%4C%00%4C%4F+%57%4F%52%4C%44'
),
// sanitize an integer
array('121564564', '121564564'),
array('121564564.0121', '121564564.0121'),
array(121564564.0121, 121564564.0121),
array(12121, 12121),
// sanitize HTML
array(
"<test toto='mama' piwik=\"cool\">Piwik!!!!!</test>",
"<test toto='mama' piwik="cool">Piwik!!!!!</test>"
),
// sanitize a SQL query
array(
"SELECT piwik FROM piwik_tests where test= 'super\"value' AND cool=toto #comment here",
"SELECT piwik FROM piwik_tests where test= 'super"value' AND cool=toto #comment here"
),
// sanitize php variables
array(true, true),
array(false, false),
array(null, null),
array("", ""),
);
/**
* @dataProvider getInputValues
*/
public function testSanitizeInputValues($input, $output)
{
if (version_compare(PHP_VERSION, '5.4') < 0) {
$this->assertTrue(@set_magic_quotes_runtime(1));
$this->assertEquals(1, @get_magic_quotes_runtime());
$this->assertEquals($output, Common::sanitizeInputValues($input));
$this->assertTrue(@set_magic_quotes_runtime(0));
$this->assertEquals(0, @get_magic_quotes_runtime());
$this->assertEquals($output, Common::sanitizeInputValues($input));
/**
* emptyvarname => exception
*/
public function testGetRequestVarEmptyVarName()
try {
$_GET[''] = 1;
Common::getRequestVar('');
} catch (Exception $e) {
return;
}
$this->fail('Expected exception not raised');
/**
* nodefault Notype Novalue => exception
*/
public function testGetRequestVarNoDefaultNoTypeNoValue()
try {
Common::getRequestVar('test');
} catch (Exception $e) {
return;
}
$this->fail('Expected exception not raised');
/**
* nodefault Notype WithValue => value
*/
public function testGetRequestVarNoDefaultNoTypeWithValue()
{
$_GET['test'] = 1413.431413;
$this->assertEquals($_GET['test'], Common::getRequestVar('test'));
public function testGetRequestVar_GetStringFloatGiven()
{
$_GET['test'] = 1413.431413;
$value = Common::getRequestVar('test', null, 'string');
$this->assertEquals('1413.431413', $value);
}
public function testGetRequestVar_GetStringIntegerGiven()
{
$_GET['test'] = 1413;
$value = Common::getRequestVar('test', null, 'string');
$this->assertEquals('1413', $value);
}
/**
* nodefault Withtype WithValue => exception cos type not matching
* @expectedException \Exception
* @expectedExceptionMessage The parameter 'test' isn't set in the Request
public function testGetRequestVarNoDefaultWithTypeWithValue()
$_GET['test'] = false;
Common::getRequestVar('test', null, 'string');
}
/**
* nodefault Withtype WithValue => exception cos type not matching
*/
public function testGetRequestVarNoDefaultWithTypeWithValue2()
try {
Common::getRequestVar('test', null, 'string');
} catch (Exception $e) {
return;
}
$this->fail('Expected exception not raised');
/**
* Dataprovider for testGetRequestVar
*/
public function getRequestVarValues()
{
return array( // value of request var, default value, var type, expected
array(1413.431413, 2, 'int', 2), // withdefault Withtype WithValue => value casted as type
array(null, 'default', null, 'default'), // withdefault Notype NoValue => default value
array(null, 'default', 'string', 'default'), // withdefault Withtype NoValue =>default value casted as type
// integer as a default value / types
array('', 45, 'int', 45),
array(1413.431413, 45, 'int', 45),
array('', 45, 'integer', 45),
array('', 45.0, 'float', 45.0),
array('', 45.25, 'float', 45.25),
// string as a default value / types
array('1413.431413', 45, 'int', 45),
array('1413.431413', 45, 'string', '1413.431413'),
array('', 45, 'string', '45'),
array('', 'geaga', 'string', 'geaga'),
array('', ''}{}}{}{}'', 'string', ''}{}}{}{}''),
array('', 'http://url?arg1=val1&arg2=val2', 'string', 'http://url?arg1=val1&arg2=val2'),
array('http://url?arg1=val1&arg2=val2', 'http://url?arg1=val1&arg2=val4', 'string', 'http://url?arg1=val1&arg2=val2'),
array(array("test", 1345524, array("gaga")), array(), 'array', array("test", 1345524, array("gaga"))), // array as a default value / types
array(array("test", 1345524, array("gaga")), 45, 'string', "45"),
array(array("test", 1345524, array("gaga")), array(1), 'array', array("test", 1345524, array("gaga"))),
array(array("test", 1345524, "Start of hello\nworld\n\t", array("gaga")), array(1), 'array', array("test", 1345524, "Start of hello\nworld\n\t", array("gaga"))),
array(array("test", 1345524, array("gaga")), 4, 'int', 4),
array('', array(1), 'array', array(1)),
array('', array(), 'array', array()),
// we give a number in a string and request for a number => it should give the string casted as a number
array('45645646', 1, 'int', 45645646),
array('45645646', 45, 'integer', 45645646),
array('45645646', '45454', 'string', '45645646'),
array('45645646', array(), 'array', array()),
);
/**
* @dataProvider getRequestVarValues
* @group Core
*/
public function testGetRequestVar($varValue, $default, $type, $expected)
{
$_GET['test'] = $varValue;
$return = Common::getRequestVar('test', $default, $type);
$this->assertEquals($expected, $return);
// validate correct type
switch ($type) {
case 'int':
case 'integer':
$this->assertTrue(is_int($return));
break;
case 'float':
$this->assertTrue(is_float($return));
break;
case 'string':
$this->assertTrue(is_string($return));
break;
case 'array':
$this->assertTrue(is_array($return));
break;
}
}
public function testIsValidFilenameValidValues()
{
$valid = array(
);
foreach ($valid as $toTest) {
$this->assertTrue(Filesystem::isValidFilename($toTest), $toTest . " not valid!");
}
}
public function testIsValidFilenameNotValidValues()
{
$notvalid = array(
"../test",
"/etc/htpasswd",
'$var',
';test',
'[bizarre]',
'',
".htaccess",
"very long long eogaioge ageja geau ghaeihieg heiagie aiughaeui hfilename",
"WHITE SPACE",
);
foreach ($notvalid as $toTest) {
$this->assertFalse(Filesystem::isValidFilename($toTest), $toTest . " valid but shouldn't!");
}
}
/**
* Dataprovider for testGetBrowserLanguage
*/
public function getBrowserLanguageData()
{
return array( // user agent, browser language
array("en-gb", "en-gb"),
// filter quality attribute
array("en-us,en;q=0.5", "en-us,en"),
// bad user agents
array("en-us,chrome://global/locale/intl.properties", "en-us"),
// unregistered language tag
array("en,en-securid", "en"),
array("en-securid,en", "en"),
array("en-us,en-securid,en", "en-us,en"),
// accept private sub tags
array("en-us,x-en-securid", "en-us,x-en-securid"),
array("en-us,en-x-securid", "en-us,en-x-securid"),
// filter arbitrary white space
array("en-us, en", "en-us,en"),
array("en-ca, en-us ,en", "en-ca,en-us,en"),
// handle comments
array(" ( comment ) en-us (another comment) ", "en-us"),
// handle quoted pairs (embedded in comments)
array(" ( \( start ) en-us ( \) end ) ", "en-us"),
array(" ( \) en-ca, \( ) en-us ( \) ,en ) ", "en-us"),
);
/**
* @dataProvider getBrowserLanguageData
* @group Core
*/
public function testGetBrowserLanguage($useragent, $browserLanguage)
{
$res = Common::getBrowserLanguage($useragent);
$this->assertEquals($browserLanguage, $res);
/**
* Dataprovider for testExtractCountryCodeFromBrowserLanguage
*/
public function getCountryCodeTestData()
{
return array( // browser language, valid countries, expected result
array("", array(), "xx"),
array("", array("us" => 'amn'), "xx"),
array("en", array("us" => 'amn'), "xx"),
array("en-us", array("us" => 'amn'), "us"),
array("en-ca", array("us" => 'amn'), "xx"),
array("en-ca", array("us" => 'amn', "ca" => 'amn'), "ca"),
array("fr-fr,fr-ca", array("us" => 'amn', "ca" => 'amn'), "ca"),
array("fr-fr;q=1.0,fr-ca;q=0.9", array("us" => 'amn', "ca" => 'amn'), "ca"),
array("fr-ca,fr;q=0.1", array("us" => 'amn', "ca" => 'amn'), "ca"),
array("en-us,en;q=0.5", Common::getCountriesList(), "us"),
array("fr-ca,fr;q=0.1", array("fr" => 'eur', "us" => 'amn', "ca" => 'amn'), "ca"),
array("fr-fr,fr-ca", array("fr" => 'eur', "us" => 'amn', "ca" => 'amn'), "fr")
);
}
/**
* @dataProvider getCountryCodeTestData
* @group Core
*/
public function testExtractCountryCodeFromBrowserLanguage($browserLanguage, $validCountries, $expected)
{
include 'DataFiles/LanguageToCountry.php';
$this->assertEquals($expected, Common::extractCountryCodeFromBrowserLanguage($browserLanguage, $validCountries, true));
$this->assertEquals($expected, Common::extractCountryCodeFromBrowserLanguage($browserLanguage, $validCountries, false));
}
/**
* Dataprovider for testExtractCountryCodeFromBrowserLanguageInfer
*/
public function getCountryCodeTestDataInfer()
{
return array( // browser language, valid countries, expected result (non-guess vs guess)
array("fr,en-us", array("us" => 'amn', "ca" => 'amn'), "us", "fr"),
array("fr,en-us", array("fr" => 'eur', "us" => 'amn', "ca" => 'amn'), "us", "fr"),
array("fr,fr-fr,en-us", array("fr" => 'eur', "us" => 'amn', "ca" => 'amn'), "fr", "fr"),
array("fr-fr,fr,en-us", array("fr" => 'eur', "us" => 'amn', "ca" => 'amn'), "fr", "fr")
);
}
/**
* @dataProvider getCountryCodeTestDataInfer
* @group Core
*/
public function testExtractCountryCodeFromBrowserLanguageInfer($browserLanguage, $validCountries, $expected, $expectedInfer)
{
include "DataFiles/LanguageToCountry.php";
// do not infer country from language
$this->assertEquals($expected, Common::extractCountryCodeFromBrowserLanguage($browserLanguage, $validCountries, $enableLanguageToCountryGuess = false));
// infer country from language
$this->assertEquals($expectedInfer, Common::extractCountryCodeFromBrowserLanguage($browserLanguage, $validCountries, $enableLanguageToCountryGuess = true));
}
/**
* Dataprovider for testExtractLanguageAndRegionCodeFromBrowserLanguage
public function getLanguageDataToExtractLanguageRegionCode()
// browser language, valid languages (with optional region), expected result
array("fr-ca", array("fr"), "fr-ca"),
array("fr-ca", array("ca"), "xx"),
array("", array(), "xx"),
array("", array("en"), "xx"),
array("fr", array("en"), "xx"),
array("en", array("en"), "en"),
array("en", array("en-ca"), "xx"),
array("en-ca", array("en-ca"), "en-ca"),
array("en-ca", array("en"), "en-ca"),
array("fr,en-us", array("fr", "en"), "fr"),
array("fr,en-us", array("en", "fr"), "fr"),
array("fr-fr,fr-ca", array("fr"), "fr-fr"),
array("fr-fr,fr-ca", array("fr-ca"), "fr-ca"),
array("-ca", array("fr","ca"), "xx"),
array("fr-fr;q=1.0,fr-ca;q=0.9", array("fr-ca"), "fr-ca"),
array("es,en,fr;q=0.7,de;q=0.3", array("fr", "es", "de", "en"), "es"),
array("zh-sg,de;q=0.3", array("zh", "es", "de"), "zh-sg"),
array("fr-ca,fr;q=0.1", array("fr-ca"), "fr-ca"),
array("r5,fr;q=1,de", array("fr", "de"), "fr"),
array("Zen§gq1", array("en"), "xx"),
);
}
/**
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
* @dataProvider getLanguageDataToExtractLanguageRegionCode
*/
public function testExtractLanguageAndRegionCodeFromBrowserLanguage($browserLanguage, $validLanguages, $expected)
{
$this->assertEquals($expected, Common::extractLanguageAndRegionCodeFromBrowserLanguage($browserLanguage, $validLanguages), "test with {$browserLanguage} failed, expected {$expected}");
}
/**
* Dataprovider for testExtractLanguageCodeFromBrowserLanguage
*/
public function getLanguageDataToExtractLanguageCode()
{
return array(
// browser language, valid languages, expected result
array("fr-ca", array("fr"), "fr"),
array("fr-ca", array("ca"), "xx"),
array("", array("en"), "xx"),
array("fr", array("en"), "xx"),
array("en", array("en"), "en"),
array("en", array("en-ca"), "xx"),
array("en-ca", array("en"), "en"),
array("fr,en-us", array("fr", "en"), "fr"),
array("fr,en-us", array("en", "fr"), "fr"),
array("fr-fr,fr-ca", array("fr"), "fr"),
array("-ca", array("fr","ca"), "xx"),
array("es,en,fr;q=0.7,de;q=0.3", array("fr", "es", "de", "en"), "es"),
array("zh-sg,de;q=0.3", array("zh", "es", "de"), "zh"),
array("r5,fr;q=1,de", array("fr", "de"), "fr"),
array("Zen§gq1", array("en"), "xx"),
);
}
/**
* @dataProvider getLanguageDataToExtractLanguageCode
*/
public function testExtractLanguageCodeFromBrowserLanguage($browserLanguage, $validLanguages, $expected)
{
$this->assertEquals($expected, Common::extractLanguageCodeFromBrowserLanguage($browserLanguage, $validLanguages), "test with {$browserLanguage} failed, expected {$expected}");
public function testSearchEnginesDefinedCorrectly()
{
include "DataFiles/SearchEngines.php";
$searchEngines = array();
foreach ($GLOBALS['Piwik_SearchEngines'] as $host => $info) {
if (isset($info[2]) && $info[2] !== false) {
$this->assertTrue(strrpos($info[2], "{k}") !== false, $host . " search URL is not defined correctly, must contain the macro {k}");
}
if (!array_key_exists($info[0], $searchEngines)) {
$searchEngines[$info[0]] = true;
$this->assertTrue(strpos($host, '{}') === false, $host . " search URL is the master record and should not contain {}");
}
if (isset($info[3]) && $info[3] !== false) {
$this->assertTrue(is_array($info[3]) || is_string($info[3]), $host . ' encoding must be either a string or an array');
if (is_string($info[3])) {
$this->assertTrue(trim($info[3]) !== '', $host . ' encoding cannot be an empty string');
$this->assertTrue(strpos($info[3], ' ') === false, $host . ' encoding cannot contain spaces');
}
if (is_array($info[3])) {
$this->assertTrue(count($info[3]) > 0, $host . ' encodings cannot be an empty array');
$this->assertTrue(strpos(serialize($info[3]), '""') === false, $host . ' encodings in array cannot be empty stringss');
$this->assertTrue(strpos(serialize($info[3]), ' ') === false, $host . ' encodings in array cannot contain spaces');
}
}
}
}