106

Hotaru CMS と Image Upload プラグインを使用しています。投稿に画像を添付しようとすると、次のエラーが表示されます。それ以外の場合は、エラーは発生しません。

unserialize() [function.unserialize]: オフセットでのエラー

問題のあるコード (エラーは ** のある行を指しています):

/**
     * Retrieve submission step data
     *
     * @param $key - empty when setting
     * @return bool
     */
    public function loadSubmitData($h, $key = '')
    {
        // delete everything in this table older than 30 minutes:
        $this->deleteTempData($h->db);

        if (!$key) { return false; }

        $cleanKey = preg_replace('/[^a-z0-9]+/','',$key);
        if (strcmp($key,$cleanKey) != 0) {
            return false;
        } else {
            $sql = "SELECT tempdata_value FROM " . TABLE_TEMPDATA . " WHERE tempdata_key = %s ORDER BY tempdata_updatedts DESC LIMIT 1";
            $submitted_data = $h->db->get_var($h->db->prepare($sql, $key));
            **if ($submitted_data) { return unserialize($submitted_data); } else { return false; }** 
        }
    }

テーブルからのデータ、最後のビットに画像情報があることに注意してください。私はPHPの専門家ではないので、皆さん/ギャルがどう思うか疑問に思っていましたか?

tempdata_value:

a:10:{s:16:"submit_editorial";b:0;s:15:"submit_orig_url";s:13:"www.bbc.co.uk";s:12:"submit_title";s:14:"No title found";s:14:"submit_content";s:12:"dnfsdkfjdfdf";s:15:"submit_category";i:2;s:11:"submit_tags";s:3:"bbc";s:9:"submit_id";b:0;s:16:"submit_subscribe";i:0;s:15:"submit_comments";s:4:"open";s:5:"image";s:19:"C:fakepath100.jpg";}

編集:シリアル化ビットを見つけたと思います...

/**
     * Save submission step data
     *
     * @return bool
     */
    public function saveSubmitData($h)
    {
        // delete everything in this table older than 30 minutes:
        $this->deleteTempData($h->db);

        $sid = preg_replace('/[^a-z0-9]+/i', '', session_id());
        $key = md5(microtime() . $sid . rand());
        $sql = "INSERT INTO " . TABLE_TEMPDATA . " (tempdata_key, tempdata_value, tempdata_updateby) VALUES (%s,%s, %d)";
        $h->db->query($h->db->prepare($sql, $key, serialize($h->vars['submitted_data']), $h->currentUser->id));
        return $key;
    }
4

17 に答える 17

228

unserialize() [function.unserialize]: Error at offsetinvalid serialization data無効な長さによるものでした

クイックフィックス

あなたができることはrecalculating the length、シリアル化された配列の要素です

現在のシリアル化されたデータ

$data = 'a:10:{s:16:"submit_editorial";b:0;s:15:"submit_orig_url";s:13:"www.bbc.co.uk";s:12:"submit_title";s:14:"No title found";s:14:"submit_content";s:12:"dnfsdkfjdfdf";s:15:"submit_category";i:2;s:11:"submit_tags";s:3:"bbc";s:9:"submit_id";b:0;s:16:"submit_subscribe";i:0;s:15:"submit_comments";s:4:"open";s:5:"image";s:19:"C:fakepath100.jpg";}';

再計算なしの例

var_dump(unserialize($data));

出力

Notice: unserialize() [function.unserialize]: Error at offset 337 of 338 bytes

再計算

$data = preg_replace('!s:(\d+):"(.*?)";!e', "'s:'.strlen('$2').':\"$2\";'", $data);
var_dump(unserialize($data));

出力

array
  'submit_editorial' => boolean false
  'submit_orig_url' => string 'www.bbc.co.uk' (length=13)
  'submit_title' => string 'No title found' (length=14)
  'submit_content' => string 'dnfsdkfjdfdf' (length=12)
  'submit_category' => int 2
  'submit_tags' => string 'bbc' (length=3)
  'submit_id' => boolean false
  'submit_subscribe' => int 0
  'submit_comments' => string 'open' (length=4)
  'image' => string 'C:fakepath100.jpg' (length=17)

推奨事項..私

この種のクイックフィックスを使用する代わりに...私はあなたが質問を更新することをアドバイスします

  • データをどのようにシリアル化するか

  • どのようにそれを保存していますか..

================================編集1================ ===============

エラー

"一重引用符の代わりに二重引用符を使用したためにエラーが生成されたため、'C:\fakepath\100.png変換されましたC:fakepath100.jpg

エラーを修正するには

Fromを変更する必要があります$h->vars['submitted_data'](単一の点に注意してください'

交換

 $h->vars['submitted_data']['image'] = "C:\fakepath\100.png" ;

 $h->vars['submitted_data']['image'] = 'C:\fakepath\100.png' ;

追加のフィルター

シリアル化を呼び出す前に、この単純なフィルターを追加することもできます

function satitize(&$value, $key)
{
    $value = addslashes($value);
}

array_walk($h->vars['submitted_data'], "satitize");

UTF文字がある場合は、実行することもできます

 $h->vars['submitted_data'] = array_map("utf8_encode",$h->vars['submitted_data']);

将来のシリアル化されたデータで問題を検出する方法

  findSerializeError ( $data1 ) ;

出力

Diffrence 9 != 7
    -> ORD number 57 != 55
    -> Line Number = 315
    -> Section Data1  = pen";s:5:"image";s:19:"C:fakepath100.jpg
    -> Section Data2  = pen";s:5:"image";s:17:"C:fakepath100.jpg
                                            ^------- The Error (Element Length)

findSerializeError働き

function findSerializeError($data1) {
    echo "<pre>";
    $data2 = preg_replace ( '!s:(\d+):"(.*?)";!e', "'s:'.strlen('$2').':\"$2\";'",$data1 );
    $max = (strlen ( $data1 ) > strlen ( $data2 )) ? strlen ( $data1 ) : strlen ( $data2 );

    echo $data1 . PHP_EOL;
    echo $data2 . PHP_EOL;

    for($i = 0; $i < $max; $i ++) {

        if (@$data1 {$i} !== @$data2 {$i}) {

            echo "Diffrence ", @$data1 {$i}, " != ", @$data2 {$i}, PHP_EOL;
            echo "\t-> ORD number ", ord ( @$data1 {$i} ), " != ", ord ( @$data2 {$i} ), PHP_EOL;
            echo "\t-> Line Number = $i" . PHP_EOL;

            $start = ($i - 20);
            $start = ($start < 0) ? 0 : $start;
            $length = 40;

            $point = $max - $i;
            if ($point < 20) {
                $rlength = 1;
                $rpoint = - $point;
            } else {
                $rpoint = $length - 20;
                $rlength = 1;
            }

            echo "\t-> Section Data1  = ", substr_replace ( substr ( $data1, $start, $length ), "<b style=\"color:green\">{$data1 {$i}}</b>", $rpoint, $rlength ), PHP_EOL;
            echo "\t-> Section Data2  = ", substr_replace ( substr ( $data2, $start, $length ), "<b style=\"color:red\">{$data2 {$i}}</b>", $rpoint, $rlength ), PHP_EOL;
        }

    }

}

データベースに保存するためのより良い方法

$toDatabse = base64_encode(serialize($data));  // Save to database
$fromDatabase = unserialize(base64_decode($data)); //Getting Save Format 
于 2012-04-14T10:48:12.980 に答える
16

unserialize()シリアル化されたデータをデータベースに不適切に配置したために失敗した別の理由があります こちらの公式説明を参照してください。バイナリ データを返し、php 変数はエンコード方法を気にしないためserialize()、TEXT に入れると、VARCHAR() でこのエラーが発生します。

解決策: シリアル化されたデータをテーブルの BLOB に格納します。

于 2013-12-27T08:54:54.100 に答える
6

このエラーは、文字セットが間違っているために発生します。

開始タグの後に文字セットを設定:

header('Content-Type: text/html; charset=utf-8');

データベースに charset utf8 を設定します。

mysql_query("SET NAMES 'utf8'");
于 2013-04-24T14:56:19.667 に答える
1

照合タイプを変更する必要がutf8_unicode_ciあり、問題は修正されます。

于 2014-02-16T12:32:09.227 に答える
1

このページでいくつかのことを試しても成功しなかった後、ページソースを見て、シリアル化された文字列のすべての引用符が html-entities に置き換えられていることに気付きました。これらのエンティティをデコードすると、多くの頭痛の種を回避できます。

$myVar = html_entity_decode($myVar);
于 2017-09-14T13:08:56.107 に答える
0

これはすべての場合に使用できます。

$newdata = preg_replace_callback(
    '/(?<=^|\{|;)s:(\d+):\"(.*?)\";(?=[asbdiO]\:\d|N;|\}|$)/s',
    function($m){
        return 's:' . strlen($m[2]) . ':"' . $m[2] . '";';
    },
    $badData
);
于 2021-11-16T10:42:39.740 に答える
0

特定のフィールドの列サイズを変更する (LONGTEXT)

于 2021-03-27T03:47:32.997 に答える