2020-01-26 13:19:35 +00:00
|
|
|
<?php declare(strict_types=1);
|
2018-11-05 22:30:18 +00:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\
|
|
|
|
* Event API *
|
|
|
|
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
/** @private */
|
|
|
|
global $_shm_event_listeners;
|
2019-05-28 16:59:38 +00:00
|
|
|
$_shm_event_listeners = [];
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2019-05-29 17:23:29 +00:00
|
|
|
function _load_event_listeners(): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2019-07-05 18:14:09 +00:00
|
|
|
global $_shm_event_listeners;
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
$cache_path = data_path("cache/shm_event_listeners.php");
|
2020-01-27 19:48:20 +00:00
|
|
|
if (SPEED_HAX && file_exists($cache_path)) {
|
2019-05-28 16:59:38 +00:00
|
|
|
require_once($cache_path);
|
|
|
|
} else {
|
|
|
|
_set_event_listeners();
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2020-01-27 19:48:20 +00:00
|
|
|
if (SPEED_HAX) {
|
2019-05-28 16:59:38 +00:00
|
|
|
_dump_event_listeners($_shm_event_listeners, $cache_path);
|
|
|
|
}
|
|
|
|
}
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-07-19 09:25:07 +00:00
|
|
|
function _clear_cached_event_listeners(): void
|
|
|
|
{
|
|
|
|
if (file_exists(data_path("cache/shm_event_listeners.php"))) {
|
2019-09-29 13:30:55 +00:00
|
|
|
unlink(data_path("cache/shm_event_listeners.php"));
|
2019-07-19 09:25:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 17:23:29 +00:00
|
|
|
function _set_event_listeners(): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $_shm_event_listeners;
|
|
|
|
$_shm_event_listeners = [];
|
|
|
|
|
|
|
|
foreach (get_declared_classes() as $class) {
|
|
|
|
$rclass = new ReflectionClass($class);
|
2020-01-26 13:19:35 +00:00
|
|
|
if (!$rclass->isAbstract() && is_subclass_of($class, "Extension")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
/** @var Extension $extension */
|
|
|
|
$extension = new $class();
|
|
|
|
|
|
|
|
// skip extensions which don't support our current database
|
2019-08-07 19:53:59 +00:00
|
|
|
if (!$extension->info->is_supported()) {
|
2019-05-28 16:59:38 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (get_class_methods($extension) as $method) {
|
|
|
|
if (substr($method, 0, 2) == "on") {
|
|
|
|
$event = substr($method, 2) . "Event";
|
|
|
|
$pos = $extension->get_priority() * 100;
|
|
|
|
while (isset($_shm_event_listeners[$event][$pos])) {
|
|
|
|
$pos += 1;
|
|
|
|
}
|
|
|
|
$_shm_event_listeners[$event][$pos] = $extension;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
function _dump_event_listeners(array $event_listeners, string $path): void
|
|
|
|
{
|
|
|
|
$p = "<"."?php\n";
|
|
|
|
|
|
|
|
foreach (get_declared_classes() as $class) {
|
|
|
|
$rclass = new ReflectionClass($class);
|
2020-01-26 13:19:35 +00:00
|
|
|
if (!$rclass->isAbstract() && is_subclass_of($class, "Extension")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
$p .= "\$$class = new $class(); ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$p .= "\$_shm_event_listeners = array(\n";
|
|
|
|
foreach ($event_listeners as $event => $listeners) {
|
|
|
|
$p .= "\t'$event' => array(\n";
|
|
|
|
foreach ($listeners as $id => $listener) {
|
|
|
|
$p .= "\t\t$id => \$".get_class($listener).",\n";
|
|
|
|
}
|
|
|
|
$p .= "\t),\n";
|
|
|
|
}
|
|
|
|
$p .= ");\n";
|
|
|
|
|
|
|
|
$p .= "?".">";
|
|
|
|
file_put_contents($path, $p);
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @private */
|
|
|
|
global $_shm_event_count;
|
|
|
|
$_shm_event_count = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send an event to all registered Extensions.
|
|
|
|
*/
|
2020-01-28 21:19:59 +00:00
|
|
|
function send_event(Event $event): Event
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2019-07-31 13:52:17 +00:00
|
|
|
global $tracer_enabled;
|
2019-08-07 19:53:59 +00:00
|
|
|
|
2019-07-05 19:49:47 +00:00
|
|
|
global $_shm_event_listeners, $_shm_event_count, $_tracer;
|
2019-05-28 16:59:38 +00:00
|
|
|
if (!isset($_shm_event_listeners[get_class($event)])) {
|
2020-01-28 21:19:59 +00:00
|
|
|
return $event;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
$method_name = "on".str_replace("Event", "", get_class($event));
|
|
|
|
|
|
|
|
// send_event() is performance sensitive, and with the number
|
2019-07-07 12:51:19 +00:00
|
|
|
// of times tracer gets called the time starts to add up
|
2019-09-29 13:30:55 +00:00
|
|
|
if ($tracer_enabled) {
|
|
|
|
$_tracer->begin(get_class($event));
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
// SHIT: http://bugs.php.net/bug.php?id=35106
|
|
|
|
$my_event_listeners = $_shm_event_listeners[get_class($event)];
|
|
|
|
ksort($my_event_listeners);
|
2019-06-25 23:47:06 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
foreach ($my_event_listeners as $listener) {
|
2019-09-29 13:30:55 +00:00
|
|
|
if ($tracer_enabled) {
|
|
|
|
$_tracer->begin(get_class($listener));
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
if (method_exists($listener, $method_name)) {
|
|
|
|
$listener->$method_name($event);
|
|
|
|
}
|
2019-09-29 13:30:55 +00:00
|
|
|
if ($tracer_enabled) {
|
|
|
|
$_tracer->end();
|
|
|
|
}
|
|
|
|
if ($event->stop_processing===true) {
|
2019-06-25 23:47:06 +00:00
|
|
|
break;
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
$_shm_event_count++;
|
2019-09-29 13:30:55 +00:00
|
|
|
if ($tracer_enabled) {
|
|
|
|
$_tracer->end();
|
|
|
|
}
|
2020-01-28 21:19:59 +00:00
|
|
|
|
|
|
|
return $event;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|