How to keep json_encode() from dropping strings with invalid characters

后端 未结 6 678
醉话见心
醉话见心 2020-11-29 06:24

Is there a way to keep json_encode() from returning null for a string that contains an invalid (non-UTF-8) character?

It can be a pain in t

相关标签:
6条回答
  • 2020-11-29 06:50

    to get a informational error notification on json failures we use this helper:

    • installs temporarily a custom error handler to catch json errors for encoding/decoding
    • throws RuntimeException on error
    <?php
    
    /**
     * usage:
     * $json = HelperJson::encode(['bla'=>'foo']);
     * $array = HelperJson::decode('{"bla":"foo"}');
     * 
     * throws exception on failure
     * 
     */
    class HelperJson {
    
        /**
         * @var array
         */
        static private $jsonErrors = [
                JSON_ERROR_NONE => '',
                JSON_ERROR_UTF8 => 'Malformed UTF-8 characters, possibly incorrectly encoded',
                JSON_ERROR_DEPTH => 'Maximum stack depth exceeded',
                JSON_ERROR_STATE_MISMATCH => 'Underflow or the modes mismatch',
                JSON_ERROR_CTRL_CHAR => 'Unexpected control character found',
                JSON_ERROR_SYNTAX => 'Syntax error, malformed JSON',
        ];
    
        /**
         * ! assoc ! (reverse logic to php function)
         * @param string $jsonString
         * @param bool $assoc
         * @throws RuntimeException
         * @return array|null
         */
        static public function decode($jsonString, $assoc=true){
    
            HelperJson_ErrorHandler::reset(); // siehe unten
            set_error_handler('HelperJson_ErrorHandler::handleError');
    
            $result = json_decode($jsonString, $assoc);
    
            $errStr = HelperJson_ErrorHandler::getErrstr();
            restore_error_handler();
    
            $jsonError = json_last_error();
            if( $jsonError!=JSON_ERROR_NONE ) {
                $errorMsg = isset(self::$jsonErrors[$jsonError]) ? self::$jsonErrors[$jsonError] : 'unknown error code: '.$jsonError;
                throw new \RuntimeException('json decoding error: '.$errorMsg.' JSON: '.substr($jsonString,0, 250));
            }
            if( $errStr!='' ){
                throw new \RuntimeException('json decoding problem: '.$errStr.' JSON: '.substr($jsonString,0, 250));
            }
            return $result;
        }
    
        /**
         * encode with error "throwing"
         * @param mixed $data
         * @param int $options   $options=JSON_PRESERVE_ZERO_FRACTION+JSON_UNESCAPED_SLASHES : 1024 + 64 = 1088
         * @return string
         * @throws \RuntimeException
         */
        static public function encode($data, $options=1088){
    
            HelperJson_ErrorHandler::reset();// scheint notwendg da sonst bei utf-8 problemen nur eine warnung geflogen ist, die hier aber nicht durchschlug, verdacht der error handler macht selbst was mit json und reset damit json_last_error
            set_error_handler('HelperJson_ErrorHandler::handleError');
    
            $result = json_encode($data, $options);
    
            $errStr = HelperJson_ErrorHandler::getErrstr();
            restore_error_handler();
    
            $jsonError = json_last_error();
            if( $jsonError!=JSON_ERROR_NONE ){
                $errorMsg = isset(self::$jsonErrors[$jsonError]) ? self::$jsonErrors[$jsonError] : 'unknown error code: '.$jsonError;
                throw new \RuntimeException('json encoding error: '.$errorMsg);
            }
            if( $errStr!='' ){
                throw new \RuntimeException('json encoding problem: '.$errStr);
            }
            return $result;
        }
    
    }
    
    /**
    
    HelperJson_ErrorHandler::install();
    preg_match('~a','');
    $errStr = HelperJson_ErrorHandler::getErrstr();
    HelperJson_ErrorHandler::remove();
    
     *
     */
    class HelperJson_ErrorHandler {
    
        static protected  $errno = 0;
        static protected  $errstr = '';
        static protected  $errfile = '';
        static protected  $errline = '';
        static protected  $errcontext = array();
    
        /**
         * @param int $errno
         * @param string $errstr
         * @param string $errfile
         * @param int $errline
         * @param array $errcontext
         * @return bool
         */
        static public function handleError($errno, $errstr, $errfile, $errline, $errcontext){
            self::$errno = $errno;
            self::$errstr = $errstr;
            self::$errfile = $errfile;
            self::$errline = $errline;
            self::$errcontext = $errcontext;
            return true;
        }
    
        /**
         * @return int
         */
        static public function getErrno(){
            return self::$errno;
        }
        /**
         * @return int
         */
        static public function getErrstr(){
            return self::$errstr;
        }
        /**
         * @return int
         */
        static public function getErrfile(){
            return self::$errfile;
        }
        /**
         * @return int
         */
        static public function getErrline(){
            return self::$errline;
        }
        /**
         * @return array
         */
        static public function getErrcontext(){
            return self::$errcontext;
        }
        /**
         * reset last error
         */
        static public function reset(){
            self::$errno = 0;
            self::$errstr = '';
            self::$errfile = '';
            self::$errline = 0;
            self::$errcontext = array();
        }
    
        /**
         * set black-hole error handler
         */
        static public function install(){
            self::reset();
            set_error_handler('HelperJson_ErrorHandler::handleError');
        }
    
        /**
         * restore previous error handler
         */
        static function remove(){
            restore_error_handler();
        }
    }
    
    
    0 讨论(0)
  • 2020-11-29 06:52

    Instead of using the iconv function, you can direclty use the json_encode with the JSON_UNESCAPED_UNICODE option ( >= PHP5.4.0 )

    Make sure you put "charset=utf-8" in the header of your php file:

    header('Content-Type: application/json; charset=utf-8');

    0 讨论(0)
  • 2020-11-29 06:53

    php does try to spew an error, but only if you turn display_errors off. This is odd because the display_errors setting is only meant to control whether or not errors are printed to standard output, not whether or not an error is triggered. I want to emphasize that when you have display_errors on, even though you may see all kinds of other php errors, php doesn't just hide this error, it will not even trigger it. That means it will not show up in any error logs, nor will any custom error_handlers get called. The error just never occurs.

    Here's some code that demonstrates this:

    error_reporting(-1);//report all errors
    $invalid_utf8_char = chr(193);
    
    ini_set('display_errors', 1);//display errors to standard output
    var_dump(json_encode($invalid_utf8_char));
    var_dump(error_get_last());//nothing
    
    ini_set('display_errors', 0);//do not display errors to standard output
    var_dump(json_encode($invalid_utf8_char));
    var_dump(error_get_last());// json_encode(): Invalid UTF-8 sequence in argument
    

    That bizarre and unfortunate behavior is related to this bug https://bugs.php.net/bug.php?id=47494 and a few others, and doesn't look like it will ever be fixed.

    workaround:

    Cleaning the string before passing it to json_encode may be a workable solution.

    $stripped_of_invalid_utf8_chars_string = iconv('UTF-8', 'UTF-8//IGNORE', $orig_string);
    if ($stripped_of_invalid_utf8_chars_string !== $orig_string) {
        // one or more chars were invalid, and so they were stripped out.
        // if you need to know where in the string the first stripped character was, 
        // then see http://stackoverflow.com/questions/7475437/find-first-character-that-is-different-between-two-strings
    }
    $json = json_encode($stripped_of_invalid_utf8_chars_string);
    

    http://php.net/manual/en/function.iconv.php

    The manual says

    //IGNORE silently discards characters that are illegal in the target charset.

    So by first removing the problematic characters, in theory json_encode() shouldnt get anything it will choke on and fail with. I haven't verified that the output of iconv with the //IGNORE flag is perfectly compatible with json_encodes notion of what valid utf8 characters are, so buyer beware...as there may be edge cases where it still fails. ugh, I hate character set issues.

    Edit
    in php 7.2+, there seems to be some new flags for json_encode: JSON_INVALID_UTF8_IGNORE and JSON_INVALID_UTF8_SUBSTITUTE
    There's not much documentation yet, but for now, this test should help you understand expected behavior: https://github.com/php/php-src/blob/master/ext/json/tests/json_encode_invalid_utf8.phpt

    And, in php 7.3+ there's the new flag JSON_THROW_ON_ERROR. See http://php.net/manual/en/class.jsonexception.php

    0 讨论(0)
  • 2020-11-29 06:58
    $s = iconv('UTF-8', 'UTF-8//IGNORE', $s);
    

    This solved the problem. I am not sure why the guys from php haven't made the life easier by fixing json_encode().

    Anyway using the above allows json_encode() to create object even if the data contains special characters (swedish letters for example).

    You can then use the result in javascript without the need of decoding the data back to its original encoding (with escape(), unescape(), encodeURIComponent(), decodeURIComponent());

    I am using it like this in php (smarty):

    $template = iconv('UTF-8', 'UTF-8//IGNORE', $screen->fetch("my_template.tpl"));
    

    Then I am sending the result to javascript and just innerHTML the ready template (html peace) in my document.

    Simply said above line should be implemented in json_encode() somehow in order to allow it to work with any encoding.

    0 讨论(0)
  • 2020-11-29 06:58

    You need to know the encoding of all strings you're dealing with, or you're entering a world of pain.

    UTF-8 is an easy encoding to use. Also, JSON is defined to use UTF-8 (http://www.json.org/JSONRequest.html). So why not use it?

    Short answer: the way to avoid json_encode() dropping your strings is to make sure they are valid UTF-8.

    0 讨论(0)
  • 2020-11-29 06:59

    This function will remove all invalid UTF8 chars from a string:

    function removeInvalidChars( $text) {
        $regex = '/( [\x00-\x7F] | [\xC0-\xDF][\x80-\xBF] | [\xE0-\xEF][\x80-\xBF]{2} | [\xF0-\xF7][\x80-\xBF]{3} ) | ./x';
        return preg_replace($regex, '$1', $text);
    }
    

    I use it after converting an Excel document to json, as Excel docs aren't guaranteed to be in UTF8.

    I don't think there's a particularly sensible way of converting invalid chars to a visible but valid character. You could replace invalid chars with U+FFFD which is the unicode replacement character by turning the regex above around, but that really doesn't provide a better user experience than just dropping invalid chars.

    0 讨论(0)
提交回复
热议问题