leilukin-tumbleblog/modules/pingable/pingable.php

414 lines
13 KiB
PHP
Raw Normal View History

2024-06-20 14:10:42 +00:00
<?php
require_once "model".DIR."Pingback.php";
class Pingable extends Modules {
# Array: $caches
# Query caches for methods.
private $caches = array();
2025-01-13 09:56:01 +00:00
public static function __install(
): void {
2024-06-20 14:10:42 +00:00
Pingback::install();
Group::add_permission("edit_pingback", "Edit Webmentions");
Group::add_permission("delete_pingback", "Delete Webmentions");
}
2025-01-13 09:56:01 +00:00
public static function __uninstall(
$confirm
): void {
2024-06-20 14:10:42 +00:00
if ($confirm)
Pingback::uninstall();
Group::remove_permission("edit_pingback");
Group::remove_permission("delete_pingback");
}
2025-01-13 09:56:01 +00:00
public function list_permissions(
$names = array()
): array {
2024-06-20 14:10:42 +00:00
$names["edit_pingback"] = __("Edit Webmentions", "pingable");
$names["delete_pingback"] = __("Delete Webmentions", "pingable");
return $names;
}
2025-01-13 09:56:01 +00:00
public function webmention(
$post,
$from,
$to
): void {
2024-06-20 14:10:42 +00:00
$count = SQL::current()->count(
tables:"pingbacks",
conds:array(
"post_id" => $post->id,
"source" => $from
)
);
if (!empty($count))
error(
__("Error"),
__("A webmention from your URL is already registered.", "pingable"),
code:422
);
if (strlen($from) > 2048)
error(
__("Error"),
__("Your URL is too long to be stored in our database.", "pingable"),
code:413
);
Pingback::add(
post_id:$post->id,
source:$from,
title:preg_replace("~(https?://|^)([^/:]+).*~i", "$2", $from)
);
}
2025-01-13 09:56:01 +00:00
public function admin_edit_pingback(
$admin
): void {
2024-06-20 14:10:42 +00:00
if (empty($_GET['id']) or !is_numeric($_GET['id']))
error(
__("No ID Specified"),
__("An ID is required to edit a webmention.", "pingable"),
code:400
);
$pingback = new Pingback($_GET['id']);
if ($pingback->no_results)
2024-09-05 17:51:48 +00:00
show_404(
__("Not Found"),
__("Webmention not found.", "pingable")
2024-06-20 14:10:42 +00:00
);
if (!$pingback->editable())
show_403(
__("Access Denied"),
__("You do not have sufficient privileges to edit this webmention.", "pingable")
);
$admin->display(
"pages".DIR."edit_pingback",
array("pingback" => $pingback)
);
}
2025-01-13 09:56:01 +00:00
public function admin_update_pingback(
$admin
): never {
2024-06-20 14:10:42 +00:00
if (!isset($_POST['hash']) or !Session::check_token($_POST['hash']))
show_403(
__("Access Denied"),
__("Invalid authentication token.")
);
if (empty($_POST['id']) or !is_numeric($_POST['id']))
error(
__("No ID Specified"),
__("An ID is required to update a webmention.", "pingable"),
code:400
);
if (empty($_POST['title']))
error(
__("No Title Specified", "pingable"),
__("A title is required to update a webmention.", "pingable"),
code:400
);
$pingback = new Pingback($_POST['id']);
if ($pingback->no_results)
show_404(
__("Not Found"),
__("Webmention not found.", "pingable")
);
if (!$pingback->editable())
show_403(
__("Access Denied"),
__("You do not have sufficient privileges to edit this webmention.", "pingable")
);
$pingback = $pingback->update($_POST['title']);
Flash::notice(
__("Webmention updated.", "pingable"),
"manage_pingbacks"
);
}
2025-01-13 09:56:01 +00:00
public function admin_delete_pingback(
$admin
): void {
2024-06-20 14:10:42 +00:00
if (empty($_GET['id']) or !is_numeric($_GET['id']))
error(
__("No ID Specified"),
__("An ID is required to delete a webmention.", "pingable"),
code:400
);
$pingback = new Pingback($_GET['id']);
if ($pingback->no_results)
2024-09-05 17:51:48 +00:00
show_404(
__("Not Found"),
__("Webmention not found.", "pingable")
2024-06-20 14:10:42 +00:00
);
if (!$pingback->deletable())
show_403(
__("Access Denied"),
__("You do not have sufficient privileges to delete this webmention.", "pingable")
);
$admin->display(
"pages".DIR."delete_pingback",
array("pingback" => $pingback)
);
}
2025-01-13 09:56:01 +00:00
public function admin_destroy_pingback(
): never {
2024-06-20 14:10:42 +00:00
if (!isset($_POST['hash']) or !Session::check_token($_POST['hash']))
show_403(
__("Access Denied"),
__("Invalid authentication token.")
);
if (empty($_POST['id']) or !is_numeric($_POST['id']))
error(
__("No ID Specified"),
__("An ID is required to delete a webmention.", "pingable"),
code:400
);
if (!isset($_POST['destroy']) or $_POST['destroy'] != "indubitably")
redirect("manage_pingbacks");
$pingback = new Pingback($_POST['id']);
if ($pingback->no_results)
show_404(
__("Not Found"),
__("Webmention not found.", "pingable")
);
if (!$pingback->deletable())
show_403(
__("Access Denied"),
__("You do not have sufficient privileges to delete this webmention.", "pingable")
);
Pingback::delete($pingback->id);
Flash::notice(
__("Webmention deleted.", "pingable"),
"manage_pingbacks"
);
}
2025-01-13 09:56:01 +00:00
public function admin_manage_pingbacks(
$admin
): void {
2024-06-20 14:10:42 +00:00
if (!Visitor::current()->group->can("edit_pingback", "delete_pingback"))
show_403(
__("Access Denied"),
__("You do not have sufficient privileges to manage webmentions.", "pingable")
);
# Redirect searches to a clean URL or dirty GET depending on configuration.
if (isset($_POST['query']))
redirect(
"manage_pingbacks/query/".
2024-09-05 17:51:48 +00:00
str_ireplace(
array("%2F", "%5C"),
"%5F",
urlencode($_POST['query'])
).
2024-06-20 14:10:42 +00:00
"/"
);
fallback($_GET['query'], "");
list($where, $params, $order) = keywords(
$_GET['query'],
"title LIKE :query",
"pingbacks"
);
$admin->display(
"pages".DIR."manage_pingbacks",
array(
"pingbacks" => new Paginator(
Pingback::find(
array(
"placeholders" => true,
"where" => $where,
"params" => $params,
"order" => $order
)
),
$admin->post_limit
)
)
);
}
2025-01-13 09:56:01 +00:00
public function manage_nav(
$navs
): array {
2024-06-20 14:10:42 +00:00
if (Visitor::current()->group->can("edit_pingback", "delete_pingback"))
$navs["manage_pingbacks"] = array(
"title" => __("Webmentions", "pingable"),
"selected" => array(
"edit_pingback",
"delete_pingback"
)
);
return $navs;
}
2025-01-13 09:56:01 +00:00
public function admin_determine_action(
$action
): ?string {
2024-06-20 14:10:42 +00:00
$visitor = Visitor::current();
if (
$action == "manage" and
$visitor->group->can("edit_pingback", "delete_pingback")
)
return "manage_pingbacks";
return null;
}
2025-01-13 09:56:01 +00:00
public function manage_posts_column_header(
): string {
2024-06-20 14:10:42 +00:00
return '<th class="post_pingbacks value">'.
__("Webmentions", "pingable").
'</th>';
}
2025-01-13 09:56:01 +00:00
public function manage_posts_column(
$post
): string {
2024-06-20 14:10:42 +00:00
return '<td class="post_pingbacks value"><a href="'.
url("manage_pingbacks/query/".urlencode("post_id:".$post->id)).
'">'.
$post->pingback_count.
'</a></td>';
}
2025-01-13 09:56:01 +00:00
public function post(
$post
): void {
2024-06-20 14:10:42 +00:00
$post->has_many[] = "pingbacks";
}
2025-01-13 09:56:01 +00:00
public function delete_post(
$post
): void {
2024-06-20 14:10:42 +00:00
SQL::current()->delete(
table:"pingbacks",
conds:array("post_id" => $post->id)
);
}
2025-01-13 09:56:01 +00:00
private function get_post_pingback_count(
$post_id
): int {
2024-06-20 14:10:42 +00:00
if (!isset($this->caches["post_pingback_counts"])) {
$counts = SQL::current()->select(
tables:"pingbacks",
fields:array("COUNT(post_id) AS total", "post_id AS post_id"),
group:"post_id"
)->fetchAll();
$this->caches["post_pingback_counts"] = array();
foreach ($counts as $count) {
$id = $count["post_id"];
$total = (int) $count["total"];
$this->caches["post_pingback_counts"][$id] = $total;
}
}
return fallback($this->caches["post_pingback_counts"][$post_id], 0);
}
2025-01-13 09:56:01 +00:00
public function post_pingback_count_attr(
$attr,
$post
): int {
2024-06-20 14:10:42 +00:00
if ($post->no_results)
return 0;
return $this->get_post_pingback_count($post->id);
}
2025-01-13 09:56:01 +00:00
public function import_chyrp_post(
$entry,
$post
): void {
2024-06-20 14:10:42 +00:00
$chyrp = $entry->children(
"http://chyrp.net/export/1.0/"
);
if (!isset($chyrp->pingback))
return;
foreach ($chyrp->pingback as $pingback) {
$title = $pingback->children(
"http://www.w3.org/2005/Atom"
)->title;
2024-09-05 17:51:48 +00:00
2024-06-20 14:10:42 +00:00
$source = $pingback->children(
"http://www.w3.org/2005/Atom"
)->link["href"];
2024-09-05 17:51:48 +00:00
2024-06-20 14:10:42 +00:00
$created_at = $pingback->children(
"http://www.w3.org/2005/Atom"
)->published;
Pingback::add(
post_id:$post->id,
source:unfix((string) $source),
title:unfix((string) $title),
created_at:datetime((string) $created_at)
);
}
}
2025-01-13 09:56:01 +00:00
public function posts_export(
$atom,
$post
): string {
2024-09-05 17:51:48 +00:00
$pingbacks = Pingback::find(
array("where" => array("post_id" => $post->id))
);
2024-06-20 14:10:42 +00:00
foreach ($pingbacks as $pingback) {
$atom.= '<chyrp:pingback>'."\n".
'<title type="html">'.
2024-09-05 17:51:48 +00:00
fix($pingback->title, false, true).
2024-06-20 14:10:42 +00:00
'</title>'."\n".
'<link rel="via" href="'.
2024-09-05 17:51:48 +00:00
fix($pingback->source, true).
2024-06-20 14:10:42 +00:00
'" />'."\n".
'<published>'.
2024-09-05 17:51:48 +00:00
when(DATE_ATOM, $pingback->created_at).
2024-06-20 14:10:42 +00:00
'</published>'."\n".
2024-09-05 17:51:48 +00:00
'<chyrp:etag>'.
fix($pingback->etag(), false, true).
'</chyrp:etag>'."\n".
2024-06-20 14:10:42 +00:00
'</chyrp:pingback>'."\n";
}
return $atom;
}
}