Added support for JSON_UNESCAPED_UNICODE and fixed parsing string
parent
15dddf44fa
commit
ff0f833b3e
|
@ -78,8 +78,9 @@ class JsonFile
|
||||||
*
|
*
|
||||||
* @param array $hash writes hash into json file
|
* @param array $hash writes hash into json file
|
||||||
* @param Boolean $prettyPrint If true, output is pretty-printed
|
* @param Boolean $prettyPrint If true, output is pretty-printed
|
||||||
|
* @param Boolean $unescapeUnicode If true, unicode chars in output are unescaped
|
||||||
*/
|
*/
|
||||||
public function write(array $hash, $prettyPrint = true)
|
public function write(array $hash, $prettyPrint = true, $unescapeUnicode = true)
|
||||||
{
|
{
|
||||||
$dir = dirname($this->path);
|
$dir = dirname($this->path);
|
||||||
if (!is_dir($dir)) {
|
if (!is_dir($dir)) {
|
||||||
|
@ -94,7 +95,7 @@ class JsonFile
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
file_put_contents($this->path, static::encode($hash, $prettyPrint));
|
file_put_contents($this->path, static::encode($hash, $prettyPrint, $unescapeUnicode));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -105,17 +106,23 @@ class JsonFile
|
||||||
*
|
*
|
||||||
* @param array $hash Data to encode into a formatted JSON string
|
* @param array $hash Data to encode into a formatted JSON string
|
||||||
* @param Boolean $prettyPrint If true, output is pretty-printed
|
* @param Boolean $prettyPrint If true, output is pretty-printed
|
||||||
|
* @param Boolean $unescapeUnicode If true, unicode chars in output are unescaped
|
||||||
* @return string Indented version of the original JSON string
|
* @return string Indented version of the original JSON string
|
||||||
*/
|
*/
|
||||||
static public function encode(array $hash, $prettyPrint = true)
|
static public function encode(array $hash, $prettyPrint = true, $unescapeUnicode = true)
|
||||||
{
|
{
|
||||||
if ($prettyPrint && defined('JSON_PRETTY_PRINT')) {
|
if (version_compare(PHP_VERSION, '5.4', '>=')) {
|
||||||
return json_encode($hash, JSON_PRETTY_PRINT);
|
$options = $prettyPrint ? JSON_PRETTY_PRINT : 0;
|
||||||
|
if ($unescapeUnicode) {
|
||||||
|
$options |= JSON_UNESCAPED_UNICODE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json_encode($hash, $options);
|
||||||
}
|
}
|
||||||
|
|
||||||
$json = json_encode($hash);
|
$json = json_encode($hash);
|
||||||
|
|
||||||
if (!$prettyPrint) {
|
if (!$prettyPrint && !$unescapeUnicode) {
|
||||||
return $json;
|
return $json;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,21 +131,43 @@ class JsonFile
|
||||||
$strLen = strlen($json);
|
$strLen = strlen($json);
|
||||||
$indentStr = ' ';
|
$indentStr = ' ';
|
||||||
$newLine = "\n";
|
$newLine = "\n";
|
||||||
$prevChar = '';
|
|
||||||
$outOfQuotes = true;
|
$outOfQuotes = true;
|
||||||
|
$buffer = '';
|
||||||
|
$noescape = true;
|
||||||
|
|
||||||
for ($i = 0; $i <= $strLen; $i++) {
|
for ($i = 0; $i <= $strLen; $i++) {
|
||||||
// Grab the next character in the string
|
// Grab the next character in the string
|
||||||
$char = substr($json, $i, 1);
|
$char = substr($json, $i, 1);
|
||||||
|
|
||||||
// Are we inside a quoted string?
|
// Are we inside a quoted string?
|
||||||
if ('"' === $char && ('\\' !== $prevChar || '\\\\' === substr($json, $i-2, 2))) {
|
if ('"' === $char && $noescape) {
|
||||||
$outOfQuotes = !$outOfQuotes;
|
$outOfQuotes = !$outOfQuotes;
|
||||||
} elseif (':' === $char && $outOfQuotes) {
|
}
|
||||||
|
|
||||||
|
if (!$outOfQuotes) {
|
||||||
|
$buffer .= $char;
|
||||||
|
$noescape = '\\' === $char ? !$noescape : true;
|
||||||
|
continue;
|
||||||
|
} elseif ('' !== $buffer) {
|
||||||
|
if ($unescapeUnicode && function_exists('mb_convert_encoding')) {
|
||||||
|
// http://stackoverflow.com/questions/2934563/how-to-decode-unicode-escape-sequences-like-u00ed-to-proper-utf-8-encoded-cha
|
||||||
|
$result .= preg_replace_callback('/\\\\u([0-9a-f]{4})/i', function($match) {
|
||||||
|
return mb_convert_encoding(pack('H*', $match[1]), 'UTF-8', 'UCS-2BE');
|
||||||
|
}, $buffer.$char);
|
||||||
|
} else {
|
||||||
|
$result .= $buffer.$char;
|
||||||
|
}
|
||||||
|
|
||||||
|
$buffer = '';
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (':' === $char) {
|
||||||
// Add a space after the : character
|
// Add a space after the : character
|
||||||
$char .= ' ';
|
$char .= ' ';
|
||||||
} elseif (('}' === $char || ']' === $char) && $outOfQuotes) {
|
} elseif (('}' === $char || ']' === $char)) {
|
||||||
$pos--;
|
$pos--;
|
||||||
|
$prevChar = substr($json, $i - 1, 1);
|
||||||
|
|
||||||
if ('{' !== $prevChar && '[' !== $prevChar) {
|
if ('{' !== $prevChar && '[' !== $prevChar) {
|
||||||
// If this character is the end of an element,
|
// If this character is the end of an element,
|
||||||
|
@ -153,12 +182,11 @@ class JsonFile
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the character to the result string
|
|
||||||
$result .= $char;
|
$result .= $char;
|
||||||
|
|
||||||
// If the last character was the beginning of an element,
|
// If the last character was the beginning of an element,
|
||||||
// output a new line and indent the next line
|
// output a new line and indent the next line
|
||||||
if ((',' === $char || '{' === $char || '[' === $char) && $outOfQuotes) {
|
if (',' === $char || '{' === $char || '[' === $char) {
|
||||||
$result .= $newLine;
|
$result .= $newLine;
|
||||||
|
|
||||||
if ('{' === $char || '[' === $char) {
|
if ('{' === $char || '[' === $char) {
|
||||||
|
@ -169,8 +197,6 @@ class JsonFile
|
||||||
$result .= $indentStr;
|
$result .= $indentStr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$prevChar = $char;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $result;
|
return $result;
|
||||||
|
|
|
@ -121,6 +121,26 @@ class JsonFileTest extends \PHPUnit_Framework_TestCase
|
||||||
$this->assertJsonFormat($json, $data);
|
$this->assertJsonFormat($json, $data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testEscape()
|
||||||
|
{
|
||||||
|
$data = array("Metadata\\\"" => 'src/');
|
||||||
|
$json = '{
|
||||||
|
"Metadata\\\\\\"": "src\/"
|
||||||
|
}';
|
||||||
|
|
||||||
|
$this->assertJsonFormat($json, $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testUnicode()
|
||||||
|
{
|
||||||
|
$data = array("Žluťoučký \" kůň" => "úpěl ďábelské ódy za €");
|
||||||
|
$json = '{
|
||||||
|
"Žluťoučký \" kůň": "úpěl ďábelské ódy za €"
|
||||||
|
}';
|
||||||
|
|
||||||
|
$this->assertJsonFormat($json, $data);
|
||||||
|
}
|
||||||
|
|
||||||
private function expectParseException($text, $json)
|
private function expectParseException($text, $json)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
|
|
Loading…
Reference in New Issue