Subversion Repository Public Repository

Nextrek

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
<?php

namespace Jeremeamia\SuperClosure;

/**
 * This class allows you to do the impossible - serialize closures! With the combined power of the nikic/php-parser
 * library, the Reflection API, and infamous eval, you can serialize a closure, unserialize it in a different PHP
 * process, and execute it. It's almost as cool as time travel!
 *
 * @copyright Jeremy Lindblom 2010-2013
 */
class SerializableClosure implements \Serializable
{
    /**
     * @var \Closure The closure being made serializable
     */
    protected $closure;

    /**
     * @var \ReflectionFunction The reflected closure
     */
    protected $reflection;

    /**
     * @var array The calculated state to serialize
     */
    protected $state;

    /**
     * @param \Closure $closure
     */
    public function __construct(\Closure $closure)
    {
        $this->closure = $closure;
    }

    /**
     * @return \ReflectionFunction
     */
    public function getReflection()
    {
        if (!$this->reflection) {
            $this->reflection = new \ReflectionFunction($this->closure);
        }

        return $this->reflection;
    }

    /**
     * @return \Closure
     */
    public function getClosure()
    {
        return $this->closure;
    }

    /**
     * Invokes the original closure
     *
     * @return mixed
     */
    public function __invoke()
    {
        return $this->getReflection()->invokeArgs(func_get_args());
    }

    /**
     * Serialize the code and of context of the closure
     *
     * @return string
     */
    public function serialize()
    {
        if (!$this->state) {
            $this->createState();
        }

        return serialize($this->state);
    }

    /**
     * Unserializes the closure data and recreates the closure. Attempts to recreate the closure's context as well by
     * extracting the used variables into the scope. Variables names in this method are surrounded with underlines in
     * order to prevent collisions with the variables in the context. NOTE: There be dragons here! Both `eval` and
     * `extract` are used in this method
     *
     * @param string $__serialized__
     */
    public function unserialize($__serialized__)
    {
        // Unserialize the data we need to reconstruct the SuperClosure
        $this->state = unserialize($__serialized__);
        list($__code__, $__context__) = $this->state;

        // Simulate the original context the Closure was created in
        extract($__context__);

        // Evaluate the code to recreate the Closure
        eval("\$this->closure = {$__code__};");
    }

    /**
     * Uses the closure parser to fetch the closure's code and context
     */
    protected function createState()
    {
        $parser = new ClosureParser($this->getReflection());
        $this->state = array($parser->getCode());
        // Add the used variables (context) to the state, but wrap all closures with SerializableClosure
        $this->state[] = array_map(function ($var) {
            return ($var instanceof \Closure) ? new self($var) : $var;
        }, $parser->getUsedVariables());
    }
}

Commits for Nextrek/Aiba_backup/vendor/jeremeamia/SuperClosure/src/Jeremeamia/SuperClosure/SerializableClosure.php

Diff revisions: vs.
Revision Author Commited Message
1464 MOliva picture MOliva Tue 13 Oct, 2020 11:16:56 +0000