[core] use ArrayAccess rather than dynamic props for extra columns on the images table

This commit is contained in:
Shish 2024-01-15 17:12:36 +00:00
parent 589ff69eea
commit bbea373c19
22 changed files with 125 additions and 99 deletions

View file

@ -89,7 +89,7 @@ class BaseThemelet
"data-post-id" => $id,
];
if(Extension::is_enabled(RatingsInfo::KEY)) {
$attrs["data-rating"] = $image->rating;
$attrs["data-rating"] = $image['rating'];
}
return A(

View file

@ -17,9 +17,8 @@ use GQLA\Query;
* image per se, but could be a video, sound file, or any
* other supported upload type.
*/
#[\AllowDynamicProperties]
#[Type(name: "Post")]
class Image
class Image implements \ArrayAccess
{
public const IMAGE_DIR = "images";
public const THUMBNAIL_DIR = "thumbs";
@ -57,6 +56,8 @@ class Image
public ?int $length = null;
public ?string $tmp_file = null;
/** @var array<string, mixed> */
private array $dynamic_props = [];
public static array $bool_props = ["locked", "lossless", "video", "audio", "image"];
public static array $int_props = ["id", "owner_id", "height", "width", "filesize", "length"];
@ -75,20 +76,40 @@ class Image
// some databases use table.name rather than name
$name = str_replace("images.", "", $name);
// hax, this is likely the cause of much scrutinizer-ci complaints.
if (is_null($value)) {
$this->$name = null;
$value = null;
} elseif (in_array($name, self::$bool_props)) {
$this->$name = bool_escape((string)$value);
$value = bool_escape((string)$value);
} elseif (in_array($name, self::$int_props)) {
$this->$name = int_escape((string)$value);
} else {
$value = int_escape((string)$value);
}
if(property_exists($this, $name)) {
$this->$name = $value;
} else {
$this->dynamic_props[$name] = $value;
}
}
}
}
public function offsetExists(mixed $offset): bool
{
return isset($this->dynamic_props[$offset]);
}
public function offsetGet(mixed $offset): mixed
{
return $this->dynamic_props[$offset];
}
public function offsetSet(mixed $offset, mixed $value): void
{
$this->dynamic_props[$offset] = $value;
}
public function offsetUnset(mixed $offset): void
{
unset($this->dynamic_props[$offset]);
}
#[Field(name: "post_id")]
public function graphql_oid(): int
{
@ -262,6 +283,15 @@ class Image
);
}
// For the future: automatically save dynamic props instead of
// requiring each extension to do it manually.
/*
$props_sql = "UPDATE images SET ";
$props_sql .= implode(", ", array_map(fn ($prop) => "$prop = :$prop", array_keys($this->dynamic_props)));
$props_sql .= " WHERE id = :id";
$database->execute($props_sql, array_merge($this->dynamic_props, ["id" => $this->id]));
*/
$database->execute(
"UPDATE images SET ".
"lossless = :lossless, ".

View file

@ -208,7 +208,7 @@ class Approval extends Extension
{
global $user, $config;
if ($config->get_bool(ApprovalConfig::IMAGES) && $image->approved === false && !$user->can(Permissions::APPROVE_IMAGE) && $user->id !== $image->owner_id) {
if ($config->get_bool(ApprovalConfig::IMAGES) && $image['approved'] === false && !$user->can(Permissions::APPROVE_IMAGE) && $user->id !== $image->owner_id) {
return false;
}
return true;

View file

@ -14,7 +14,7 @@ class ApprovalTheme extends Themelet
{
public function get_image_admin_html(Image $image): HTMLElement
{
if ($image->approved === true) {
if ($image['approved'] === true) {
$form = SHM_SIMPLE_FORM(
'disapprove_image/'.$image->id,
INPUT(["type" => 'hidden', "name" => 'image_id', "value" => $image->id]),

View file

@ -312,14 +312,14 @@ class CommentList extends Extension
$image = Image::by_id((int)$row["image_id"]);
if (
Extension::is_enabled(RatingsInfo::KEY) && !is_null($image) &&
!in_array($image->rating, $user_ratings)
!in_array($image['rating'], $user_ratings)
) {
$image = null; // this is "clever", I may live to regret it
}
if (
Extension::is_enabled(ApprovalInfo::KEY) && !is_null($image) &&
$config->get_bool(ApprovalConfig::IMAGES) &&
$image->approved !== true
$image['approved'] !== true
) {
$image = null;
}

View file

@ -107,7 +107,7 @@ class Favorites extends Extension
public function onParseLinkTemplate(ParseLinkTemplateEvent $event): void
{
$event->replace('$favorites', (string)$event->image->favorites);
$event->replace('$favorites', (string)$event->image['favorites']);
}
public function onUserBlockBuilding(UserBlockBuildingEvent $event): void

View file

@ -65,7 +65,7 @@ class Featured extends Extension
);
if (!is_null($image)) {
if (Extension::is_enabled(RatingsInfo::KEY)) {
if (!in_array($image->rating, Ratings::get_user_class_privs($user))) {
if (!in_array($image['rating'], Ratings::get_user_class_privs($user))) {
return;
}
}

View file

@ -282,7 +282,7 @@ class NumericScore extends Extension
public function onParseLinkTemplate(ParseLinkTemplateEvent $event): void
{
$event->replace('$score', (string)$event->image->numeric_score);
$event->replace('$score', (string)$event->image['numeric_score']);
}
public function onHelpPageBuilding(HelpPageBuildingEvent $event): void

View file

@ -10,10 +10,7 @@ class NumericScoreTheme extends Themelet
{
global $user, $page;
$i_image_id = $image->id;
if (is_string($image->numeric_score)) {
$image->numeric_score = (int)$image->numeric_score;
}
$i_score = $image->numeric_score;
$i_score = (int)$image['numeric_score'];
$html = "
Current Score: $i_score

View file

@ -74,12 +74,12 @@ class _SafeOuroborosImage
if (Extension::is_enabled(RatingsInfo::KEY) !== false) {
// 'u' is not a "valid" rating
if ($img->rating == 's' || $img->rating == 'q' || $img->rating == 'e') {
$this->rating = $img->rating;
if ($img['rating'] == 's' || $img['rating'] == 'q' || $img['rating'] == 'e') {
$this->rating = $img['rating'];
}
}
if (Extension::is_enabled(NumericScoreInfo::KEY) !== false) {
$this->score = $img->numeric_score;
$this->score = $img['numeric_score'];
}
$this->source = $img->source;

View file

@ -77,8 +77,9 @@ class PostTitles extends Extension
public function onBulkExport(BulkExportEvent $event): void
{
$event->fields["title"] = $event->image->title;
$event->fields["title"] = $event->image['title'];
}
public function onBulkImport(BulkImportEvent $event): void
{
if (array_key_exists("title", $event->fields) && $event->fields['title'] != null) {
@ -97,7 +98,7 @@ class PostTitles extends Extension
{
global $config;
$title = $image->title ?? "";
$title = $image['title'] ?? "";
if (empty($title) && $config->get_bool(PostTitlesConfig::DEFAULT_TO_FILENAME)) {
$info = pathinfo($image->filename);
if (array_key_exists("extension", $info)) {

View file

@ -117,7 +117,7 @@ class PrivateImage extends Extension
{
global $user, $page;
if ($event->image->private === true && $event->image->owner_id != $user->id && !$user->can(Permissions::SET_OTHERS_PRIVATE_IMAGES)) {
if ($event->image['private'] === true && $event->image->owner_id != $user->id && !$user->can(Permissions::SET_OTHERS_PRIVATE_IMAGES)) {
$page->set_mode(PageMode::REDIRECT);
$page->set_redirect(make_link());
}

View file

@ -10,7 +10,7 @@ class PrivateImageTheme extends Themelet
{
public function get_image_admin_html(Image $image): string
{
if ($image->private === false) {
if ($image['private'] === false) {
$html = SHM_SIMPLE_FORM(
'privatize_image/'.$image->id,
INPUT(["type" => 'hidden', "name" => 'image_id', "value" => $image->id]),

View file

@ -104,7 +104,7 @@ class Ratings extends Extension
global $user;
$user_view_level = Ratings::get_user_class_privs($user);
if (!in_array($image->rating, $user_view_level)) {
if (!in_array($image['rating'], $user_view_level)) {
return false;
}
return true;
@ -185,7 +185,7 @@ class Ratings extends Extension
public function onBulkExport(BulkExportEvent $event): void
{
$event->fields["rating"] = $event->image->rating;
$event->fields["rating"] = $event->image['rating'];
}
public function onBulkImport(BulkImportEvent $event): void
{
@ -198,10 +198,10 @@ class Ratings extends Extension
public function onRatingSet(RatingSetEvent $event): void
{
if (empty($event->image->rating)) {
if (empty($event->image['rating'])) {
$old_rating = "";
} else {
$old_rating = $event->image->rating;
$old_rating = $event->image['rating'];
}
$this->set_rating($event->image->id, $event->rating, $old_rating);
}
@ -212,7 +212,7 @@ class Ratings extends Extension
$event->add_part(
$this->theme->get_rater_html(
$event->image->id,
$event->image->rating,
$event->image['rating'],
$user->can(Permissions::EDIT_IMAGE_RATING)
),
80
@ -232,8 +232,8 @@ class Ratings extends Extension
public function onParseLinkTemplate(ParseLinkTemplateEvent $event): void
{
if(!is_null($event->image->rating)) {
$event->replace('$rating', $this->rating_to_human($event->image->rating));
if(!is_null($event->image['rating'])) {
$event->replace('$rating', $this->rating_to_human($event->image['rating']));
}
}

View file

@ -136,12 +136,12 @@ class Relationships extends Extension
{
global $database;
if (bool_escape($event->image->has_children)) {
if (bool_escape($event->image['has_children'])) {
$database->execute("UPDATE images SET parent_id = NULL WHERE parent_id = :iid", ["iid" => $event->image->id]);
}
if ($event->image->parent_id !== null) {
$this->set_has_children($event->image->parent_id);
if ($event->image['parent_id'] !== null) {
$this->set_has_children($event->image['parent_id']);
}
}

View file

@ -24,12 +24,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_2 = Image::by_id($image_id_2);
$image_3 = Image::by_id($image_id_3);
$this->assertNull($image_1->parent_id);
$this->assertNull($image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertFalse($image_1->has_children);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertNull($image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertFalse($image_1['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -46,12 +46,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_2 = Image::by_id($image_2->id);
$image_3 = Image::by_id($image_3->id);
$this->assertNull($image_1->parent_id);
$this->assertEquals($image_1->id, $image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertTrue($image_1->has_children);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertEquals($image_1->id, $image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertTrue($image_1['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -67,12 +67,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_2 = Image::by_id($image_2->id);
$image_3 = Image::by_id($image_3->id);
$this->assertNull($image_1->parent_id);
$this->assertEquals($image_3->id, $image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_2->has_children);
$this->assertTrue($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertEquals($image_3->id, $image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertTrue($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -107,12 +107,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_2 = Image::by_id($image_2->id);
$image_3 = Image::by_id($image_3->id);
$this->assertNull($image_1->parent_id);
$this->assertNull($image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertNull($image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_3['has_children']);
}
//=================================================================
@ -130,12 +130,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_2 = Image::by_id($image_id_2);
$image_3 = Image::by_id($image_id_3);
$this->assertNull($image_1->parent_id);
$this->assertNull($image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertFalse($image_1->has_children);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertNull($image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertFalse($image_1['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -153,12 +153,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_3 = Image::by_id($image_3->id);
$this->assertEquals(["pbx"], $image_2->get_tag_array());
$this->assertNull($image_1->parent_id);
$this->assertEquals($image_1->id, $image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertTrue($image_1->has_children);
$this->assertFalse($image_2->has_children);
$this->assertFalse($image_3->has_children);
$this->assertNull($image_1['parent_id']);
$this->assertEquals($image_1->id, $image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertTrue($image_1['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertFalse($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -176,12 +176,12 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
$image_3 = Image::by_id($image_3->id);
$this->assertEquals(["pbx"], $image_3->get_tag_array());
$this->assertEquals($image_3->id, $image_1->parent_id);
$this->assertEquals($image_1->id, $image_2->parent_id);
$this->assertNull($image_3->parent_id);
$this->assertTrue($image_1->has_children);
$this->assertFalse($image_2->has_children);
$this->assertTrue($image_3->has_children);
$this->assertEquals($image_3->id, $image_1['parent_id']);
$this->assertEquals($image_1->id, $image_2['parent_id']);
$this->assertNull($image_3['parent_id']);
$this->assertTrue($image_1['has_children']);
$this->assertFalse($image_2['has_children']);
$this->assertTrue($image_3['has_children']);
return [$image_1, $image_2, $image_3];
}
@ -193,7 +193,7 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
assert(!is_null($image_3));
// check parent is set
$this->assertEquals($image_2->parent_id, $image_1->id);
$this->assertEquals($image_2['parent_id'], $image_1->id);
// un-set it
send_event(new TagSetEvent($image_2, ["pbx", "parent:none"]));
@ -203,6 +203,6 @@ class RelationshipsTest extends ShimmiePHPUnitTestCase
// check it was unset
$this->assertEquals(["pbx"], $image_2->get_tag_array());
$this->assertNull($image_2->parent_id);
$this->assertNull($image_2['parent_id']);
}
}

View file

@ -14,12 +14,12 @@ class RelationshipsTheme extends Themelet
{
global $page, $database;
if ($image->parent_id !== null) {
$a = "<a href='".make_link("post/view/".$image->parent_id)."'>parent post</a>";
if ($image['parent_id'] !== null) {
$a = "<a href='".make_link("post/view/".$image['parent_id'])."'>parent post</a>";
$page->add_block(new Block(null, "This post belongs to a $a.", "main", 5, "ImageHasParent"));
}
if (bool_escape($image->has_children)) {
if (bool_escape($image['has_children'])) {
$ids = $database->get_col("SELECT id FROM images WHERE parent_id = :iid", ["iid" => $image->id]);
$html = "This post has <a href='".search_link(['parent='.$image->id])."'>".(count($ids) > 1 ? "child posts" : "a child post")."</a>";
@ -39,8 +39,8 @@ class RelationshipsTheme extends Themelet
return SHM_POST_INFO(
"Parent",
strval($image->parent_id) ?: "None",
!$user->is_anonymous() ? INPUT(["type" => "number", "name" => "tag_edit__parent", "value" => $image->parent_id]) : null
strval($image['parent_id']) ?: "None",
!$user->is_anonymous() ? INPUT(["type" => "number", "name" => "tag_edit__parent", "value" => $image['parent_id']]) : null
);
}

View file

@ -49,7 +49,7 @@ class Trash extends Extension
{
global $user;
if ($image->trash === true && !$user->can(Permissions::VIEW_TRASH)) {
if ($image['trash'] === true && !$user->can(Permissions::VIEW_TRASH)) {
return false;
}
return true;
@ -77,7 +77,7 @@ class Trash extends Extension
public function onImageDeletion(ImageDeletionEvent $event): void
{
if ($event->force !== true && $event->image->trash !== true) {
if ($event->force !== true && $event->image['trash'] !== true) {
self::set_trash($event->image->id, true);
$event->stop_processing = true;
}
@ -157,7 +157,7 @@ class Trash extends Extension
public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event): void
{
global $user;
if ($event->image->trash === true && $user->can(Permissions::VIEW_TRASH)) {
if ($event->image['trash'] === true && $user->can(Permissions::VIEW_TRASH)) {
$event->add_part($this->theme->get_image_admin_html($event->image->id));
}
}

View file

@ -6,8 +6,6 @@ parameters:
- ../ext
- ../tests
- ../themes
ignoreErrors:
- '#Access to an undefined property Shimmie2\\Image::\$#'
dynamicConstantNames:
- DEBUG
- SPEED_HAX

View file

@ -51,10 +51,10 @@ class CustomViewPostTheme extends ViewPostTheme
}
if (Extension::is_enabled(RatingsInfo::KEY)) {
if ($image->rating === null || $image->rating == "?") {
$image->rating = "?";
if ($image['rating'] === null || $image['rating'] == "?") {
$image['rating'] = "?";
}
$h_rating = Ratings::rating_to_human($image->rating);
$h_rating = Ratings::rating_to_human($image['rating']);
$html .= "<br>Rating: $h_rating";
}

View file

@ -52,12 +52,12 @@ class CustomViewPostTheme extends ViewPostTheme
}
if (Extension::is_enabled(RatingsInfo::KEY)) {
if ($image->rating === null || $image->rating == "?") {
$image->rating = "?";
if ($image['rating'] === null || $image['rating'] == "?") {
$image['rating'] = "?";
}
// @phpstan-ignore-next-line - ???
if (Extension::is_enabled(RatingsInfo::KEY)) {
$h_rating = Ratings::rating_to_human($image->rating);
$h_rating = Ratings::rating_to_human($image['rating']);
$html .= "<br>Rating: $h_rating";
}
}

View file

@ -56,10 +56,10 @@ class CustomViewPostTheme extends ViewPostTheme
}
if (Extension::is_enabled(RatingsInfo::KEY)) {
if ($image->rating === null || $image->rating == "?") {
$image->rating = "?";
if ($image['rating'] === null || $image['rating'] == "?") {
$image['rating'] = "?";
}
$h_rating = Ratings::rating_to_human($image->rating);
$h_rating = Ratings::rating_to_human($image['rating']);
$html .= "<br>Rating: $h_rating";
}