Completed
Branch PHPRenderer (6c13a8)
by Josh
10:40
created

XPath::export()   A

Complexity

Conditions 4
Paths 4

Size

Total Lines 18
Code Lines 8

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 8
CRAP Score 4

Importance

Changes 0
Metric Value
dl 0
loc 18
rs 9.2
c 0
b 0
f 0
ccs 8
cts 8
cp 1
cc 4
eloc 8
nc 4
nop 1
crap 4
1
<?php
2
3
/**
4
* @package   s9e\TextFormatter
5
* @copyright Copyright (c) 2010-2017 The s9e Authors
6
* @license   http://www.opensource.org/licenses/mit-license.php The MIT License
7
*/
8
namespace s9e\TextFormatter\Utils;
9
10
use InvalidArgumentException;
11
12
abstract class XPath
13
{
14
	/**
15
	* Export a literal as an XPath expression
16
	*
17
	* @param  mixed  $value Literal, e.g. "foo"
18
	* @return string        XPath expression, e.g. "'foo'"
19
	*/
20 8
	public static function export($value)
21
	{
22 8
		if (!is_scalar($value))
23
		{
24 1
			throw new InvalidArgumentException(__METHOD__ . '() cannot export non-scalar values');
25
		}
26 7
		if (is_int($value))
27
		{
28 1
			return (string) $value;
29
		}
30 6
		if (is_float($value))
31
		{
32
			// Avoid locale issues by using sprintf()
33 2
			return preg_replace('(\\.?0+$)', '', sprintf('%F', $value));
34
		}
35
36 4
		return self::exportString($value);
0 ignored issues
show
Bug introduced by
It seems like $value defined by parameter $value on line 20 can also be of type boolean; however, s9e\TextFormatter\Utils\XPath::exportString() does only seem to accept string, maybe add an additional type check?

This check looks at variables that have been passed in as parameters and are passed out again to other methods.

If the outgoing method call has stricter type requirements than the method itself, an issue is raised.

An additional type check may prevent trouble.

Loading history...
37
	}
38
39
	/**
40
	* Export a string as an XPath expression
41
	*
42
	* @param  string $str Literal, e.g. "foo"
43
	* @return string      XPath expression, e.g. "'foo'"
44
	*/
45 4
	protected static function exportString($str)
46
	{
47
		// foo becomes 'foo'
48 4
		if (strpos($str, "'") === false)
49
		{
50 1
			return "'" . $str . "'";
51
		}
52
53
		// d'oh becomes "d'oh"
54 3
		if (strpos($str, '"') === false)
55
		{
56 1
			return '"' . $str . '"';
57
		}
58
59
		// This string contains both ' and ". XPath 1.0 doesn't have a mechanism to escape quotes,
60
		// so we have to get creative and use concat() to join chunks in single quotes and chunks
61
		// in double quotes
62 2
		$toks = [];
63 2
		$c    = '"';
64 2
		$pos  = 0;
65 2
		while ($pos < strlen($str))
66
		{
67 2
			$spn = strcspn($str, $c, $pos);
68 2
			if ($spn)
69
			{
70 2
				$toks[] = $c . substr($str, $pos, $spn) . $c;
71 2
				$pos   += $spn;
72
			}
73 2
			$c = ($c === '"') ? "'" : '"';
74
		}
75
76 2
		return 'concat(' . implode(',', $toks) . ')';
77
	}
78
}