[core] all image columns should be statically or dynamically defined - nothing undefined

This commit is contained in:
Shish 2024-01-15 18:01:48 +00:00
parent fb034af699
commit 2a494504f3
9 changed files with 86 additions and 27 deletions

View file

@ -15,6 +15,26 @@ enum ImagePropType
case STRING; case STRING;
} }
function pop_val(array &$row, string $name): mixed
{
$value = $row[$name];
unset($row[$name]);
if(is_null($value)) {
return null;
}
return $value;
}
function pop_int(array &$row, string $name): ?int
{
$val = pop_val($row, $name);
return (is_null($val)) ? null : (int)$val;
}
function pop_bool(array &$row, string $name): ?bool
{
$val = pop_val($row, $name);
return (is_null($val)) ? null : bool_escape($val);
}
/** /**
* Class Image * Class Image
* *
@ -64,20 +84,7 @@ class Image implements \ArrayAccess
public ?string $tmp_file = null; public ?string $tmp_file = null;
/** @var array<string, ImagePropType> */ /** @var array<string, ImagePropType> */
public static array $prop_types = [ public static array $prop_types = [];
"id" => ImagePropType::INT,
"owner_id" => ImagePropType::INT,
"locked" => ImagePropType::BOOL,
"lossless" => ImagePropType::BOOL,
"video" => ImagePropType::BOOL,
"video_codec" => ImagePropType::STRING,
"image" => ImagePropType::BOOL,
"audio" => ImagePropType::BOOL,
"height" => ImagePropType::INT,
"width" => ImagePropType::INT,
"filesize" => ImagePropType::INT,
"length" => ImagePropType::INT,
];
/** @var array<string, mixed> */ /** @var array<string, mixed> */
private array $dynamic_props = []; private array $dynamic_props = [];
@ -88,30 +95,54 @@ class Image implements \ArrayAccess
public function __construct(?array $row = null) public function __construct(?array $row = null)
{ {
if (!is_null($row)) { if (!is_null($row)) {
$this->id = pop_int($row, 'id');
$this->hash = pop_val($row, 'hash');
$this->filename = pop_val($row, 'filename');
$this->filesize = pop_int($row, 'filesize');
$this->mime = pop_val($row, 'mime');
$this->ext = pop_val($row, 'ext');
$this->posted = pop_val($row, 'posted');
$this->source = pop_val($row, 'source');
$this->owner_id = pop_int($row, 'owner_id');
$this->owner_ip = pop_val($row, 'owner_ip');
$this->locked = pop_bool($row, 'locked');
$this->lossless = pop_bool($row, 'lossless');
$this->video = pop_bool($row, 'video');
$this->video_codec = pop_val($row, 'video_codec');
$this->image = pop_bool($row, 'image');
$this->audio = pop_bool($row, 'audio');
$this->height = pop_int($row, 'height');
$this->width = pop_int($row, 'width');
$this->length = pop_int($row, 'length');
// Any remaining fields are dynamic properties
foreach ($row as $name => $value) { foreach ($row as $name => $value) {
if (is_numeric($name)) { if (is_numeric($name)) {
continue; continue;
} }
if(property_exists($this, $name)) {
throw new \Exception("Property $name already exists on Image");
}
// some databases use table.name rather than name if(array_key_exists($name, static::$prop_types)) {
$name = str_replace("images.", "", $name); if (is_null($value)) {
$value = null;
if (is_null($value)) { } else {
$value = null;
} else {
if(array_key_exists($name, static::$prop_types)) {
$value = match(static::$prop_types[$name]) { $value = match(static::$prop_types[$name]) {
ImagePropType::BOOL => bool_escape((string)$value), ImagePropType::BOOL => bool_escape((string)$value),
ImagePropType::INT => int_escape((string)$value), ImagePropType::INT => int_escape((string)$value),
ImagePropType::STRING => (string)$value, ImagePropType::STRING => (string)$value,
}; };
} }
}
if(property_exists($this, $name)) {
$this->$name = $value;
} else {
$this->dynamic_props[$name] = $value; $this->dynamic_props[$name] = $value;
} else {
// Database table has a column we don't know about,
// it isn't static and it isn't a known prop_type -
// maybe from an old extension that has since been
// disabled? Just ignore it.
if(UNITTEST) {
throw new \Exception("Unknown column $name in images table");
}
} }
} }
} }
@ -119,10 +150,13 @@ class Image implements \ArrayAccess
public function offsetExists(mixed $offset): bool public function offsetExists(mixed $offset): bool
{ {
return isset($this->dynamic_props[$offset]); return array_key_exists($offset, $this->dynamic_props);
} }
public function offsetGet(mixed $offset): mixed public function offsetGet(mixed $offset): mixed
{ {
if(!$this->offsetExists($offset)) {
throw new \OutOfBoundsException("Undefined dynamic property: $offset");
}
return $this->dynamic_props[$offset]; return $this->dynamic_props[$offset];
} }
public function offsetSet(mixed $offset, mixed $value): void public function offsetSet(mixed $offset, mixed $value): void

View file

@ -24,6 +24,7 @@ class Approval extends Extension
$config->set_default_bool(ApprovalConfig::COMMENTS, false); $config->set_default_bool(ApprovalConfig::COMMENTS, false);
Image::$prop_types["approved"] = ImagePropType::BOOL; Image::$prop_types["approved"] = ImagePropType::BOOL;
Image::$prop_types["approved_by_id"] = ImagePropType::INT;
} }
public function onImageAddition(ImageAdditionEvent $event): void public function onImageAddition(ImageAdditionEvent $event): void

View file

@ -24,6 +24,11 @@ class Artists extends Extension
/** @var ArtistsTheme */ /** @var ArtistsTheme */
protected Themelet $theme; protected Themelet $theme;
public function onInitExt(InitExtEvent $event): void
{
Image::$prop_types["author"] = ImagePropType::STRING;
}
public function onImageInfoSet(ImageInfoSetEvent $event): void public function onImageInfoSet(ImageInfoSetEvent $event): void
{ {
global $user; global $user;

View file

@ -27,6 +27,11 @@ class Favorites extends Extension
/** @var FavoritesTheme */ /** @var FavoritesTheme */
protected Themelet $theme; protected Themelet $theme;
public function onInitExt(InitExtEvent $event): void
{
Image::$prop_types["favorites"] = ImagePropType::INT;
}
public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event): void public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event): void
{ {
global $database, $user; global $database, $user;

View file

@ -9,6 +9,11 @@ class Notes extends Extension
/** @var NotesTheme */ /** @var NotesTheme */
protected Themelet $theme; protected Themelet $theme;
public function onInitExt(InitExtEvent $event): void
{
Image::$prop_types["notes"] = ImagePropType::INT;
}
public function onDatabaseUpgrade(DatabaseUpgradeEvent $event): void public function onDatabaseUpgrade(DatabaseUpgradeEvent $event): void
{ {
global $config, $database; global $config, $database;

View file

@ -104,6 +104,11 @@ class NumericScore extends Extension
/** @var NumericScoreTheme */ /** @var NumericScoreTheme */
protected Themelet $theme; protected Themelet $theme;
public function onInitExt(InitExtEvent $event): void
{
Image::$prop_types["numeric_score"] = ImagePropType::INT;
}
public function onDisplayingImage(DisplayingImageEvent $event): void public function onDisplayingImage(DisplayingImageEvent $event): void
{ {
global $user; global $user;

View file

@ -23,6 +23,7 @@ class PostTitles extends Extension
$config->set_default_bool(PostTitlesConfig::DEFAULT_TO_FILENAME, false); $config->set_default_bool(PostTitlesConfig::DEFAULT_TO_FILENAME, false);
$config->set_default_bool(PostTitlesConfig::SHOW_IN_WINDOW_TITLE, false); $config->set_default_bool(PostTitlesConfig::SHOW_IN_WINDOW_TITLE, false);
Image::$prop_types["title"] = ImagePropType::STRING;
} }
public function onDatabaseUpgrade(DatabaseUpgradeEvent $event): void public function onDatabaseUpgrade(DatabaseUpgradeEvent $event): void

View file

@ -97,6 +97,8 @@ class Ratings extends Extension
} }
$config->set_default_array("ext_rating_" . $key . "_privs", array_keys($_shm_ratings)); $config->set_default_array("ext_rating_" . $key . "_privs", array_keys($_shm_ratings));
} }
Image::$prop_types["rating"] = ImagePropType::STRING;
} }
private function check_permissions(Image $image): bool private function check_permissions(Image $image): bool

View file

@ -14,3 +14,4 @@ parameters:
- BASE_HREF - BASE_HREF
- STATSD_HOST - STATSD_HOST
- TRACE_FILE - TRACE_FILE
- UNITTEST