2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2023-01-10 22:44:09 +00:00
|
|
|
|
|
|
|
namespace Shimmie2;
|
|
|
|
|
2009-07-19 08:38:13 +01:00
|
|
|
/**
|
|
|
|
* Generic parent class for all events.
|
|
|
|
*
|
|
|
|
* An event is anything that can be passed around via send_event($blah)
|
2007-04-16 11:58:25 +00:00
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
abstract class Event
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public bool $stop_processing = false;
|
2019-06-25 18:47:06 -05:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
}
|
2019-12-07 22:49:02 +00:00
|
|
|
|
2021-03-14 23:43:50 +00:00
|
|
|
public function __toString(): string
|
2019-12-07 22:49:02 +00:00
|
|
|
{
|
|
|
|
return var_export($this, true);
|
|
|
|
}
|
2007-04-16 11:58:25 +00:00
|
|
|
}
|
2007-12-06 10:21:23 +00:00
|
|
|
|
|
|
|
|
2009-07-19 08:38:13 +01:00
|
|
|
/**
|
2009-07-21 04:18:40 +01:00
|
|
|
* A wake-up call for extensions. Upon recieving an InitExtEvent an extension
|
|
|
|
* should check that it's database tables are there and install them if not,
|
|
|
|
* and set any defaults with Config::set_default_int() and such.
|
2015-08-03 15:40:20 +01:00
|
|
|
*
|
|
|
|
* This event is sent before $user is set to anything
|
2007-12-06 10:21:23 +00:00
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class InitExtEvent extends Event
|
|
|
|
{
|
|
|
|
}
|
2007-12-06 10:21:23 +00:00
|
|
|
|
|
|
|
|
2009-07-19 08:38:13 +01:00
|
|
|
/**
|
|
|
|
* A signal that a page has been requested.
|
|
|
|
*
|
2009-01-03 13:00:09 -08:00
|
|
|
* User requests /view/42 -> an event is generated with $args = array("view",
|
|
|
|
* "42"); when an event handler asks $event->page_matches("view"), it returns
|
|
|
|
* true and ignores the matched part, such that $event->count_args() = 1 and
|
|
|
|
* $event->get_arg(0) = "42"
|
2007-12-06 10:21:23 +00:00
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class PageRequestEvent extends Event
|
|
|
|
{
|
2024-01-01 02:32:13 +00:00
|
|
|
public string $method;
|
2024-02-09 12:43:53 +00:00
|
|
|
public string $path;
|
2024-02-09 16:36:57 +00:00
|
|
|
/** @var array<string, string|string[]> */
|
|
|
|
public array $GET;
|
|
|
|
/** @var array<string, string|string[]> */
|
|
|
|
public array $POST;
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
/**
|
2021-03-14 23:43:50 +00:00
|
|
|
* @var string[]
|
2019-05-28 17:59:38 +01:00
|
|
|
*/
|
2024-02-09 15:10:36 +00:00
|
|
|
public array $args;
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $arg_count;
|
|
|
|
public int $part_count;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
2024-02-09 16:36:57 +00:00
|
|
|
/**
|
|
|
|
* @param string $method The HTTP method used to make the request
|
|
|
|
* @param string $path The path of the request
|
|
|
|
* @param array<string, string|string[]> $get The GET parameters
|
|
|
|
* @param array<string, string|string[]> $post The POST parameters
|
|
|
|
*/
|
|
|
|
public function __construct(string $method, string $path, array $get, array $post)
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-05-28 17:59:38 +01:00
|
|
|
global $config;
|
|
|
|
|
2024-01-01 02:32:13 +00:00
|
|
|
$this->method = $method;
|
|
|
|
|
2024-02-09 12:43:53 +00:00
|
|
|
// if we're looking at the root of the install,
|
|
|
|
// use the default front page
|
|
|
|
if ($path == "") {
|
2019-08-02 14:40:03 -05:00
|
|
|
$path = $config->get_string(SetupConfig::FRONT_PAGE);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2024-02-09 12:43:53 +00:00
|
|
|
$this->path = $path;
|
2024-02-09 16:36:57 +00:00
|
|
|
$this->GET = $get;
|
|
|
|
$this->POST = $post;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
// break the path into parts
|
|
|
|
$args = explode('/', $path);
|
|
|
|
|
|
|
|
$this->args = $args;
|
|
|
|
$this->arg_count = count($args);
|
|
|
|
}
|
|
|
|
|
2024-02-09 16:36:57 +00:00
|
|
|
public function get_GET(string $key): ?string
|
|
|
|
{
|
|
|
|
if(array_key_exists($key, $this->GET)) {
|
|
|
|
if(is_array($this->GET[$key])) {
|
|
|
|
throw new SCoreException("GET parameter {$key} is an array, expected single value");
|
|
|
|
}
|
|
|
|
return $this->GET[$key];
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-10 00:08:55 +00:00
|
|
|
public function req_GET(string $key): string
|
|
|
|
{
|
|
|
|
$value = $this->get_GET($key);
|
|
|
|
if($value === null) {
|
|
|
|
throw new UserErrorException("Missing GET parameter {$key}");
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
2024-02-09 16:36:57 +00:00
|
|
|
public function get_POST(string $key): ?string
|
|
|
|
{
|
|
|
|
if(array_key_exists($key, $this->POST)) {
|
|
|
|
if(is_array($this->POST[$key])) {
|
|
|
|
throw new SCoreException("POST parameter {$key} is an array, expected single value");
|
|
|
|
}
|
|
|
|
return $this->POST[$key];
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-10 00:08:55 +00:00
|
|
|
public function req_POST(string $key): string
|
|
|
|
{
|
|
|
|
$value = $this->get_POST($key);
|
|
|
|
if($value === null) {
|
|
|
|
throw new UserErrorException("Missing POST parameter {$key}");
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
/**
|
2024-02-10 00:46:35 +00:00
|
|
|
* @return string[]|null
|
|
|
|
*/
|
|
|
|
public function get_POST_array(string $key): ?array
|
|
|
|
{
|
|
|
|
if(array_key_exists($key, $this->POST)) {
|
|
|
|
if(!is_array($this->POST[$key])) {
|
|
|
|
throw new SCoreException("POST parameter {$key} is a single value, expected array");
|
|
|
|
}
|
|
|
|
return $this->POST[$key];
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
|
|
|
public function req_POST_array(string $key): array
|
|
|
|
{
|
|
|
|
$value = $this->get_POST_array($key);
|
|
|
|
if($value === null) {
|
|
|
|
throw new UserErrorException("Missing POST parameter {$key}");
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-05-28 17:59:38 +01:00
|
|
|
* Test if the requested path matches a given pattern.
|
|
|
|
*
|
|
|
|
* If it matches, store the remaining path elements in $args
|
|
|
|
*/
|
|
|
|
public function page_matches(string $name): bool
|
|
|
|
{
|
|
|
|
$parts = explode("/", $name);
|
|
|
|
$this->part_count = count($parts);
|
|
|
|
|
|
|
|
if ($this->part_count > $this->arg_count) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-11-11 21:49:12 +00:00
|
|
|
for ($i = 0; $i < $this->part_count; $i++) {
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($parts[$i] != $this->args[$i]) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the n th argument of the page request (if it exists.)
|
|
|
|
*/
|
2019-11-04 00:40:10 +00:00
|
|
|
public function get_arg(int $n): string
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
$offset = $this->part_count + $n;
|
|
|
|
if ($offset >= 0 && $offset < $this->arg_count) {
|
2023-12-24 21:12:59 +00:00
|
|
|
return rawurldecode($this->args[$offset]);
|
2019-05-28 17:59:38 +01:00
|
|
|
} else {
|
2020-03-27 14:41:24 +00:00
|
|
|
$nm1 = $this->arg_count - 1;
|
2021-11-16 14:55:37 +00:00
|
|
|
throw new UserErrorException("Requested an invalid page argument {$offset} / {$nm1}");
|
2019-11-04 00:40:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-27 14:41:24 +00:00
|
|
|
/**
|
|
|
|
* If page arg $n is set, then treat that as a 1-indexed page number
|
|
|
|
* and return a 0-indexed page number less than $max; else return 0
|
|
|
|
*/
|
2023-11-11 21:49:12 +00:00
|
|
|
public function try_page_num(int $n, ?int $max = null): int
|
2019-11-04 00:42:06 +00:00
|
|
|
{
|
|
|
|
if ($this->count_args() > $n) {
|
2019-11-04 00:40:10 +00:00
|
|
|
$i = $this->get_arg($n);
|
2024-01-15 20:34:53 +00:00
|
|
|
if (is_numberish($i) && int_escape($i) > 0) {
|
2020-03-27 14:41:24 +00:00
|
|
|
return page_number($i, $max);
|
2019-11-04 00:42:06 +00:00
|
|
|
} else {
|
2020-03-27 14:41:24 +00:00
|
|
|
return 0;
|
2019-11-04 00:40:10 +00:00
|
|
|
}
|
2019-11-04 00:42:06 +00:00
|
|
|
} else {
|
2020-03-27 14:41:24 +00:00
|
|
|
return 0;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the number of arguments the page request has.
|
|
|
|
*/
|
|
|
|
public function count_args(): int
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
return $this->arg_count - $this->part_count;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Many things use these functions
|
|
|
|
*/
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
public function get_search_terms(): array
|
|
|
|
{
|
|
|
|
$search_terms = [];
|
|
|
|
if ($this->count_args() === 2) {
|
2023-12-24 21:12:59 +00:00
|
|
|
$str = $this->get_arg(0);
|
|
|
|
|
|
|
|
// decode legacy caret-encoding just in case
|
|
|
|
// somebody has bookmarked such a URL
|
|
|
|
$from_caret = [
|
|
|
|
"^" => "^",
|
|
|
|
"s" => "/",
|
|
|
|
"b" => "\\",
|
|
|
|
"q" => "?",
|
|
|
|
"a" => "&",
|
|
|
|
"d" => ".",
|
|
|
|
];
|
|
|
|
$out = "";
|
|
|
|
$length = strlen($str);
|
|
|
|
for ($i = 0; $i < $length; $i++) {
|
|
|
|
if ($str[$i] == "^") {
|
|
|
|
$i++;
|
|
|
|
$out .= $from_caret[$str[$i]] ?? '';
|
|
|
|
} else {
|
|
|
|
$out .= $str[$i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$str = $out;
|
|
|
|
// end legacy
|
|
|
|
|
|
|
|
$search_terms = Tag::explode($str);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
return $search_terms;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function get_page_number(): int
|
|
|
|
{
|
|
|
|
$page_number = 1;
|
|
|
|
if ($this->count_args() === 1) {
|
|
|
|
$page_number = int_escape($this->get_arg(0));
|
|
|
|
} elseif ($this->count_args() === 2) {
|
|
|
|
$page_number = int_escape($this->get_arg(1));
|
|
|
|
}
|
|
|
|
if ($page_number === 0) {
|
|
|
|
$page_number = 1;
|
|
|
|
} // invalid -> 0
|
|
|
|
return $page_number;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function get_page_size(): int
|
|
|
|
{
|
|
|
|
global $config;
|
2019-08-16 09:18:14 -05:00
|
|
|
return $config->get_int(IndexConfig::IMAGES);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2007-12-06 10:21:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-01-11 00:55:05 +00:00
|
|
|
class CliGenEvent extends Event
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2024-01-11 00:55:05 +00:00
|
|
|
public function __construct(
|
|
|
|
public \Symfony\Component\Console\Application $app
|
|
|
|
) {
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2012-06-17 20:05:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-19 08:38:13 +01:00
|
|
|
/**
|
|
|
|
* A signal that some text needs formatting, the event carries
|
|
|
|
* both the text and the result
|
2007-12-06 10:21:23 +00:00
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class TextFormattingEvent extends Event
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* For reference
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $original;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* with formatting applied
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $formatted;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* with formatting removed
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $stripped;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
public function __construct(string $text)
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2020-02-01 20:01:25 +00:00
|
|
|
// We need to escape before formatting, instead of at display time,
|
|
|
|
// because formatters will add their own HTML tags into the mix and
|
|
|
|
// we don't want to escape those.
|
2019-05-28 17:59:38 +01:00
|
|
|
$h_text = html_escape(trim($text));
|
|
|
|
$this->original = $h_text;
|
|
|
|
$this->formatted = $h_text;
|
|
|
|
$this->stripped = $h_text;
|
|
|
|
}
|
2007-12-06 10:21:23 +00:00
|
|
|
}
|
2009-05-08 03:52:29 -07:00
|
|
|
|
|
|
|
|
2009-07-19 08:38:13 +01:00
|
|
|
/**
|
|
|
|
* A signal that something needs logging
|
2009-05-08 03:52:29 -07:00
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class LogEvent extends Event
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* a category, normally the extension name
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $section;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* See python...
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $priority = 0;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Free text to be logged
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $message;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The time that the event was created
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $time;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Extra data to be held separate
|
|
|
|
*
|
2021-03-14 23:43:50 +00:00
|
|
|
* @var string[]
|
2019-05-28 17:59:38 +01:00
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
public array $args;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
2020-02-01 21:37:07 +00:00
|
|
|
public function __construct(string $section, int $priority, string $message)
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-05-28 17:59:38 +01:00
|
|
|
$this->section = $section;
|
|
|
|
$this->priority = $priority;
|
|
|
|
$this->message = $message;
|
|
|
|
$this->time = time();
|
|
|
|
}
|
2009-05-08 03:52:29 -07:00
|
|
|
}
|
2019-11-03 17:19:37 +00:00
|
|
|
|
2019-11-03 18:28:38 +00:00
|
|
|
class DatabaseUpgradeEvent extends Event
|
|
|
|
{
|
2019-11-03 17:19:37 +00:00
|
|
|
}
|