2017-02-09 19:50:04 +01:00
|
|
|
<?php
|
|
|
|
/* Icinga Web 2 | (c) 2017 Icinga Development Team | GPLv2+ */
|
|
|
|
|
|
|
|
namespace Icinga\Util;
|
|
|
|
|
|
|
|
use Icinga\Exception\Json\JsonDecodeException;
|
|
|
|
use Icinga\Exception\Json\JsonEncodeException;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wrap {@link json_encode()} and {@link json_decode()} with error handling
|
|
|
|
*/
|
|
|
|
class Json
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* {@link json_encode()} wrapper
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
|
|
|
* @param int $options
|
|
|
|
* @param int $depth
|
2018-06-20 18:03:21 +02:00
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* @throws JsonEncodeException
|
|
|
|
*/
|
|
|
|
public static function encode($value, $options = 0, $depth = 512)
|
|
|
|
{
|
|
|
|
return static::encodeAndSanitize($value, $options, $depth, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@link json_encode()} wrapper, automatically sanitizes bad UTF-8
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
|
|
|
* @param int $options
|
|
|
|
* @param int $depth
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* @throws JsonEncodeException
|
|
|
|
*/
|
|
|
|
public static function sanitize($value, $options = 0, $depth = 512)
|
|
|
|
{
|
|
|
|
return static::encodeAndSanitize($value, $options, $depth, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@link json_encode()} wrapper, sanitizes bad UTF-8
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
|
|
|
* @param int $options
|
|
|
|
* @param int $depth
|
2018-05-14 10:19:32 +02:00
|
|
|
* @param bool $autoSanitize Automatically sanitize invalid UTF-8 (if any)
|
2017-02-09 19:50:04 +01:00
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* @throws JsonEncodeException
|
|
|
|
*/
|
2018-06-20 18:03:21 +02:00
|
|
|
protected static function encodeAndSanitize($value, $options, $depth, $autoSanitize)
|
2017-02-09 19:50:04 +01:00
|
|
|
{
|
2018-05-08 09:16:47 +02:00
|
|
|
$encoded = json_encode($value, $options, $depth);
|
2018-04-27 17:15:54 +02:00
|
|
|
|
|
|
|
switch (json_last_error()) {
|
|
|
|
case JSON_ERROR_NONE:
|
|
|
|
return $encoded;
|
|
|
|
|
2018-05-14 10:19:32 +02:00
|
|
|
/** @noinspection PhpMissingBreakStatementInspection */
|
2018-04-27 17:15:54 +02:00
|
|
|
case JSON_ERROR_UTF8:
|
2018-05-14 10:19:32 +02:00
|
|
|
if ($autoSanitize) {
|
|
|
|
return static::encode(static::sanitizeUtf8Recursive($value), $options, $depth);
|
|
|
|
}
|
2018-06-22 09:57:10 +02:00
|
|
|
// Fallthrough
|
2018-04-27 17:15:54 +02:00
|
|
|
|
|
|
|
default:
|
2018-05-08 09:16:47 +02:00
|
|
|
throw new JsonEncodeException('%s: %s', json_last_error_msg(), var_export($value, true));
|
2017-02-09 19:50:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@link json_decode()} wrapper
|
|
|
|
*
|
|
|
|
* @param string $json
|
|
|
|
* @param bool $assoc
|
|
|
|
* @param int $depth
|
|
|
|
* @param int $options
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
* @throws JsonDecodeException
|
|
|
|
*/
|
|
|
|
public static function decode($json, $assoc = false, $depth = 512, $options = 0)
|
|
|
|
{
|
2017-12-15 11:17:07 +01:00
|
|
|
$decoded = json_decode($json, $assoc, $depth, $options);
|
2017-06-13 09:29:59 +02:00
|
|
|
|
2017-02-09 19:50:04 +01:00
|
|
|
if (json_last_error() !== JSON_ERROR_NONE) {
|
2018-05-08 09:16:47 +02:00
|
|
|
throw new JsonDecodeException('%s: %s', json_last_error_msg(), var_export($json, true));
|
2017-02-09 19:50:04 +01:00
|
|
|
}
|
|
|
|
return $decoded;
|
|
|
|
}
|
|
|
|
|
2018-04-27 17:15:54 +02:00
|
|
|
/**
|
|
|
|
* Replace bad byte sequences in UTF-8 strings inside the given JSON-encodable structure with question marks
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
protected static function sanitizeUtf8Recursive($value)
|
|
|
|
{
|
|
|
|
switch (gettype($value)) {
|
|
|
|
case 'string':
|
|
|
|
return static::sanitizeUtf8String($value);
|
|
|
|
|
|
|
|
case 'array':
|
|
|
|
$sanitized = array();
|
|
|
|
|
|
|
|
foreach ($value as $key => $val) {
|
|
|
|
if (is_string($key)) {
|
|
|
|
$key = static::sanitizeUtf8String($key);
|
|
|
|
}
|
|
|
|
|
|
|
|
$sanitized[$key] = static::sanitizeUtf8Recursive($val);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sanitized;
|
|
|
|
|
|
|
|
case 'object':
|
|
|
|
$sanitized = array();
|
|
|
|
|
|
|
|
foreach ($value as $key => $val) {
|
|
|
|
if (is_string($key)) {
|
|
|
|
$key = static::sanitizeUtf8String($key);
|
|
|
|
}
|
|
|
|
|
|
|
|
$sanitized[$key] = static::sanitizeUtf8Recursive($val);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (object) $sanitized;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Replace bad byte sequences in the given UTF-8 string with question marks
|
|
|
|
*
|
|
|
|
* @param string $string
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected static function sanitizeUtf8String($string)
|
|
|
|
{
|
|
|
|
return mb_convert_encoding($string, 'UTF-8', 'UTF-8');
|
|
|
|
}
|
2017-02-09 19:50:04 +01:00
|
|
|
}
|