PHP:不允许对闭包进行序列化


PHP : Serialization of a closure is not allowed

整个场景是我在我的opencart商店中添加了facebook登录扩展,当一个新用户用facebook点击登录时,一切都很好。添加电子邮件和密码并点击登录,然后会出现一个确认登录对话框,其中有两个按钮"取消"answers"登录"。如果用户点击登录,它会将用户带到我的商店,但当用户点击"取消"时,它应该会将用户带回我商店的登录页面,但它会显示错误"Fatal error : uncaught exception with a message serialization of closure is not allowed
这条线是哪条"$_SESSION["HA::STORE"][$key] = serialize($value);我正在使用包含此行的以下功能:

public function set( $key, $value)
    {
        $key = strtolower( $key );
        $_SESSION["HA::STORE"][$key] = serialize($value);
    }

我试过这个var_dump(serialize($value((;它返回一个字符串如何序列化?我已经搜索过了,但没有找到任何有用的解决方案

更新:

function login()
    {
        Hybrid_Logger::info( "Enter Hybrid_Provider_Adapter::login( {$this->id} ) " );
        if( ! $this->adapter ){
            throw new Exception( "Hybrid_Provider_Adapter::login() should not directly used." );
        }
        // clear all unneeded params
        foreach( Hybrid_Auth::$config["providers"] as $idpid => $params ){
            Hybrid_Auth::storage()->delete( "hauth_session.{$idpid}.hauth_return_to"    );
            Hybrid_Auth::storage()->delete( "hauth_session.{$idpid}.hauth_endpoint"     );
            Hybrid_Auth::storage()->delete( "hauth_session.{$idpid}.id_provider_params" );
        }
        // make a fresh start
        $this->logout();
        # get hybridauth base url
        $HYBRID_AUTH_URL_BASE = Hybrid_Auth::$config["base_url"];
        # we make use of session_id() as storage hash to identify the current user
        # using session_regenerate_id() will be a problem, but ..
        $this->params["hauth_token"] = session_id();
        # set request timestamp
        $this->params["hauth_time"]  = time();
        # for default HybridAuth endpoint url hauth_login_start_url
        #   auth.start  required  the IDp ID
        #   auth.time   optional  login request timestamp
        $this->params["login_start"] = $HYBRID_AUTH_URL_BASE . ( strpos( $HYBRID_AUTH_URL_BASE, '?' ) ? '&' : '?' ) . "hauth.start={$this->id}&hauth.time={$this->params["hauth_time"]}";
        # for default HybridAuth endpoint url hauth_login_done_url
        #   auth.done   required  the IDp ID
        $this->params["login_done"]  = $HYBRID_AUTH_URL_BASE . ( strpos( $HYBRID_AUTH_URL_BASE, '?' ) ? '&' : '?' ) . "hauth.done={$this->id}";
        Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.hauth_return_to"    , $this->params["hauth_return_to"] );
        Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.hauth_endpoint"     , $this->params["login_done"] ); 
        Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.id_provider_params" , $this->params );
        // store config to be used by the end point 
        Hybrid_Auth::storage()->config( "CONFIG", Hybrid_Auth::$config );
        // move on
        Hybrid_Logger::debug( "Hybrid_Provider_Adapter::login( {$this->id} ), redirect the user to login_start URL." );
        Hybrid_Auth::redirect( $this->params["login_start"] );
    }

在函数loggin中,set((由以下函数调用:

Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.hauth_return_to"    , $this->params["hauth_return_to"] );
        Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.hauth_endpoint"     , $this->params["login_done"] ); 
        Hybrid_Auth::storage()->set( "hauth_session.{$this->id}.id_provider_params" , $this->params );

我猜您正在传递或正在获得一个闭包作为set()$value参数。

您需要使用反射检查和忽略闭包

public function set( $key, $value)
{
    if (is_object($value)) {
        try {
            $reflection = new ReflectionFunction($value);
            if ($reflection->isClosure()) {
                //Trigger a E_USER_NOTICE if you want to avoid silently failing
                trigger_error("You cannot pass a closure as the second parameter of set()");
                return; // Do nothing else
            }
        } catch ('ReflectionException $e) {
            // Catch the exception thrown if $value is not a closure
        }
    }
    $key = strtolower( $key );
    $_SESSION["HA::STORE"][$key] = serialize($value);
}

或者您可以捕获并忽略异常:

function set( $key, $value)
{
    $key = strtolower( $key );
    try{
        $_SESSION["HA::STORE"][$key] = serialize($value);
    } catch ('Exception $e) {
        // Catch the exception thrown and handle it however you want
    }
}

或者看看这个github项目来序列化闭包