Исключение: Сериализация "Закрытие" не допускается
Так что я точно не знаю, что мне нужно было бы показать вам, ребята, как бы то ни было, если вам нужно больше кода, пожалуйста, не стесняйтесь спрашивать:
Таким образом, этот метод настроит initMailer для Zend в нашем приложении:
protected function _initMailer()
{
if ('testing' !== APPLICATION_ENV) {
$this->bootstrap('Config');
$options = $this->getOptions();
$mail = new Zend_Application_Resource_Mail($options['mail']);
}elseif ('testing' === APPLICATION_ENV) {
//change the mail transport only if dev or test
if (APPLICATION_ENV <> 'production') {
$callback = function()
{
return 'ZendMail_' . microtime(true) .'.tmp';
};
$mail = new Zend_Mail_Transport_File(
array('path' => '/tmp/mail/',
'callback'=>$callback
)
);
Zend_Mail::setDefaultTransport($mail);
}
}
return $mail;
}
Вы можете увидеть закрытие, которое находится внутри. Когда я запускаю какие-либо тесты, которые используют этот код, я получаю:
Exception: Serialization of 'Closure' is not allowed
и, следовательно, все тесты в связи с этим "закрытием" терпят неудачу. Поэтому я здесь прошу вас, ребята, что я должен делать.
Для разъяснения вышеизложенного все делали это, говоря, что любое электронное письмо, которое мы отправляем, мы хотим хранить информацию об этом письме в папке в каталоге/tmp/mail/в файле.
Ответы
Ответ 1
По-видимому, анонимные функции не могут быть сериализованы.
Пример
$function = function () {
return "ABC";
};
serialize($function); // would throw error
Из вашего кода вы используете Closure
$callback = function () // <---------------------- Issue
{
return 'ZendMail_' . microtime(true) . '.tmp';
};
Решение 1: Заменить нормальной функцией Пример
function emailCallback() {
return 'ZendMail_' . microtime(true) . '.tmp';
}
$callback = "emailCallback" ;
Решение 2: Непрямой вызов метода по переменной массива
Если вы посмотрите http://docs.mnkras.com/libraries_23rdparty_2_zend_2_mail_2_transport_2file_8php_source.html
public function __construct($options = null)
63 {
64 if ($options instanceof Zend_Config) {
65 $options = $options->toArray();
66 } elseif (!is_array($options)) {
67 $options = array();
68 }
69
70 // Making sure we have some defaults to work with
71 if (!isset($options['path'])) {
72 $options['path'] = sys_get_temp_dir();
73 }
74 if (!isset($options['callback'])) {
75 $options['callback'] = array($this, 'defaultCallback'); <- here
76 }
77
78 $this->setOptions($options);
79 }
Вы можете использовать тот же подход для отправки обратного вызова
$callback = array($this,"aMethodInYourClass");
Ответ 2
Сериализация Direct Closure не допускается PHP. Но вы можете использовать класс powefull, например PHP Super Closure: https://github.com/jeremeamia/super_closure
Этот класс очень прост в использовании и входит в структуру laravel для менеджера очередей.
Из документации github:
$helloWorld = new SerializableClosure(function ($name = 'World') use ($greeting) {
echo "{$greeting}, {$name}!\n";
});
$serialized = serialize($helloWorld);
Ответ 3
Как уже говорилось: закрытие, из коробки, не может быть сериализовано.
Однако, используя магические методы и рефлексию __sleep()
, __wakeup()
и CAN вручную, можно сделать сериализуемые замыкания. Подробнее см. extending-php-5-3-closures-with-serialization-and-reflection
Это использует отражение и функцию php eval. Обратите внимание, что это открывает возможность инъекции CODE, поэтому, пожалуйста, обратите внимание на то, что вы сериализуете.
Ответ 4
Вам нужно отключить глобальные блокировки
/**
* @backupGlobals disabled
*/