2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2008-09-07 05:57:28 +00:00
|
|
|
|
2023-01-10 22:44:09 +00:00
|
|
|
namespace Shimmie2;
|
|
|
|
|
2023-02-04 13:27:27 +00:00
|
|
|
use GQLA\Type;
|
|
|
|
use GQLA\Field;
|
|
|
|
use GQLA\Query;
|
2023-02-04 14:47:26 +00:00
|
|
|
use GQLA\Mutation;
|
2023-02-01 12:50:00 +00:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
class SendPMEvent extends Event
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public PM $pm;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
public function __construct(PM $pm)
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-05-28 17:59:38 +01:00
|
|
|
$this->pm = $pm;
|
|
|
|
}
|
2008-09-07 13:55:28 +00:00
|
|
|
}
|
|
|
|
|
2023-02-04 14:47:26 +00:00
|
|
|
#[Type(name: "PrivateMessage")]
|
2019-05-28 17:59:38 +01:00
|
|
|
class PM
|
|
|
|
{
|
2023-02-04 14:47:26 +00:00
|
|
|
public int $id = -1;
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $from_id;
|
|
|
|
public string $from_ip;
|
|
|
|
public int $to_id;
|
2023-02-04 13:27:27 +00:00
|
|
|
public mixed $sent_date;
|
|
|
|
#[Field]
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $subject;
|
2023-02-04 13:27:27 +00:00
|
|
|
#[Field]
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $message;
|
2023-02-04 13:27:27 +00:00
|
|
|
#[Field]
|
2021-03-14 23:43:50 +00:00
|
|
|
public bool $is_read;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
2023-02-04 14:47:26 +00:00
|
|
|
#[Field]
|
|
|
|
public function from(): User
|
|
|
|
{
|
|
|
|
return User::by_id($this->from_id);
|
|
|
|
}
|
|
|
|
|
2023-02-13 22:28:50 +00:00
|
|
|
#[Field(name: "pm_id")]
|
|
|
|
public function graphql_oid(): int
|
|
|
|
{
|
|
|
|
return $this->id;
|
|
|
|
}
|
|
|
|
#[Field(name: "id")]
|
|
|
|
public function graphql_guid(): string
|
|
|
|
{
|
|
|
|
return "pm:{$this->id}";
|
|
|
|
}
|
|
|
|
|
2023-02-04 14:47:26 +00:00
|
|
|
public function __construct(
|
|
|
|
int $from_id,
|
|
|
|
string $from_ip,
|
|
|
|
int $to_id,
|
|
|
|
string $subject,
|
|
|
|
string $message,
|
|
|
|
bool $is_read = false
|
|
|
|
) {
|
|
|
|
$this->from_id = $from_id;
|
|
|
|
$this->from_ip = $from_ip;
|
|
|
|
$this->to_id = $to_id;
|
|
|
|
$this->subject = $subject;
|
|
|
|
$this->message = $message;
|
|
|
|
$this->is_read = $is_read;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function from_row(array $row): PM
|
|
|
|
{
|
|
|
|
$pm = new PM(
|
|
|
|
(int)$row["from_id"],
|
|
|
|
$row["from_ip"],
|
|
|
|
(int)$row["to_id"],
|
|
|
|
$row["subject"],
|
|
|
|
$row["message"],
|
|
|
|
bool_escape($row["is_read"]),
|
|
|
|
);
|
|
|
|
$pm->id = (int)$row["id"];
|
|
|
|
$pm->sent_date = $row["sent_date"];
|
|
|
|
return $pm;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Field(extends: "User", name: "private_messages", type: "[PrivateMessage!]")]
|
|
|
|
public static function get_pms(User $duser): ?array
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2023-02-04 14:47:26 +00:00
|
|
|
global $database, $user;
|
|
|
|
|
|
|
|
if (!$user->can(Permissions::READ_PM)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (($duser->id != $user->id) && !$user->can(Permissions::VIEW_OTHER_PMS)) {
|
|
|
|
return null;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2023-02-04 14:47:26 +00:00
|
|
|
|
|
|
|
$pms = [];
|
|
|
|
$arr = $database->get_all(
|
|
|
|
"SELECT * FROM private_message WHERE to_id = :to_id ORDER BY sent_date DESC",
|
|
|
|
["to_id" => $duser->id]
|
|
|
|
);
|
|
|
|
foreach ($arr as $pm) {
|
|
|
|
$pms[] = PM::from_row($pm);
|
|
|
|
}
|
|
|
|
return $pms;
|
|
|
|
}
|
|
|
|
|
2023-02-07 13:27:20 +00:00
|
|
|
#[Field(extends: "User", name: "private_message_unread_count")]
|
|
|
|
public static function count_unread_pms(User $duser): ?int
|
|
|
|
{
|
|
|
|
global $database, $user;
|
|
|
|
|
|
|
|
if (!$user->can(Permissions::READ_PM)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (($duser->id != $user->id) && !$user->can(Permissions::VIEW_OTHER_PMS)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (int)$database->get_one(
|
|
|
|
"SELECT COUNT(*) FROM private_message WHERE to_id = :to_id AND is_read = :is_read",
|
|
|
|
["is_read" => false, "to_id" => $duser->id]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-02-04 18:24:34 +00:00
|
|
|
#[Mutation(name: "create_private_message")]
|
2023-02-13 22:28:50 +00:00
|
|
|
public static function send_pm(int $to_user_id, string $subject, string $message): bool
|
2023-02-04 14:47:26 +00:00
|
|
|
{
|
|
|
|
global $user;
|
|
|
|
if (!$user->can(Permissions::SEND_PM)) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-02-13 22:28:50 +00:00
|
|
|
send_event(new SendPMEvent(new PM($user->id, get_real_ip(), $to_user_id, $subject, $message)));
|
2023-02-04 14:47:26 +00:00
|
|
|
return true;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2009-10-08 17:44:25 +01:00
|
|
|
}
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
class PrivMsg extends Extension
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
/** @var PrivMsgTheme */
|
2023-06-27 15:56:49 +01:00
|
|
|
protected Themelet $theme;
|
2020-01-26 13:19:35 +00:00
|
|
|
|
2019-11-03 17:19:37 +00:00
|
|
|
public function onDatabaseUpgrade(DatabaseUpgradeEvent $event)
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2020-10-29 00:57:58 +00:00
|
|
|
global $database;
|
2008-09-07 05:57:28 +00:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
// shortcut to latest
|
2020-10-29 00:57:58 +00:00
|
|
|
if ($this->get_version("pm_version") < 1) {
|
2019-05-28 17:59:38 +01:00
|
|
|
$database->create_table("private_message", "
|
2009-05-11 14:09:24 -07:00
|
|
|
id SCORE_AIPK,
|
|
|
|
from_id INTEGER NOT NULL,
|
|
|
|
from_ip SCORE_INET NOT NULL,
|
|
|
|
to_id INTEGER NOT NULL,
|
2019-11-03 19:25:51 +00:00
|
|
|
sent_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
2009-05-11 14:09:24 -07:00
|
|
|
subject VARCHAR(64) NOT NULL,
|
|
|
|
message TEXT NOT NULL,
|
2020-10-26 22:37:25 +00:00
|
|
|
is_read BOOLEAN NOT NULL DEFAULT FALSE,
|
2012-03-11 01:52:25 +00:00
|
|
|
FOREIGN KEY (from_id) REFERENCES users(id) ON DELETE CASCADE,
|
|
|
|
FOREIGN KEY (to_id) REFERENCES users(id) ON DELETE CASCADE
|
2009-05-11 14:09:24 -07:00
|
|
|
");
|
2019-05-28 17:59:38 +01:00
|
|
|
$database->execute("CREATE INDEX private_message__to_id ON private_message(to_id)");
|
2020-10-29 00:57:58 +00:00
|
|
|
$this->set_version("pm_version", 3);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
|
2020-10-29 00:57:58 +00:00
|
|
|
if ($this->get_version("pm_version") < 2) {
|
2019-05-28 17:59:38 +01:00
|
|
|
log_info("pm", "Adding foreign keys to private messages");
|
2020-10-25 21:34:52 +00:00
|
|
|
$database->execute("delete from private_message where to_id not in (select id from users);");
|
|
|
|
$database->execute("delete from private_message where from_id not in (select id from users);");
|
|
|
|
$database->execute("ALTER TABLE private_message
|
2012-03-11 01:52:25 +00:00
|
|
|
ADD FOREIGN KEY (from_id) REFERENCES users(id) ON DELETE CASCADE,
|
|
|
|
ADD FOREIGN KEY (to_id) REFERENCES users(id) ON DELETE CASCADE;");
|
2020-10-29 00:57:58 +00:00
|
|
|
$this->set_version("pm_version", 2);
|
2020-10-26 17:13:41 +00:00
|
|
|
}
|
|
|
|
|
2020-10-29 00:57:58 +00:00
|
|
|
if ($this->get_version("pm_version") < 3) {
|
2020-10-26 17:28:21 +00:00
|
|
|
$database->standardise_boolean("private_message", "is_read", true);
|
2020-10-29 00:57:58 +00:00
|
|
|
$this->set_version("pm_version", 3);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-02 14:54:48 -05:00
|
|
|
public function onPageSubNavBuilding(PageSubNavBuildingEvent $event)
|
|
|
|
{
|
|
|
|
global $user;
|
2019-09-29 14:30:55 +01:00
|
|
|
if ($event->parent==="user") {
|
2019-12-15 20:40:05 +00:00
|
|
|
if ($user->can(Permissions::READ_PM)) {
|
2019-08-02 14:54:48 -05:00
|
|
|
$count = $this->count_pms($user);
|
|
|
|
$h_count = $count > 0 ? " <span class='unread'>($count)</span>" : "";
|
|
|
|
$event->add_nav_link("pm", new Link('user#private-messages'), "Private Messages$h_count");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
public function onUserBlockBuilding(UserBlockBuildingEvent $event)
|
|
|
|
{
|
|
|
|
global $user;
|
2019-12-15 20:40:05 +00:00
|
|
|
if ($user->can(Permissions::READ_PM)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
$count = $this->count_pms($user);
|
|
|
|
$h_count = $count > 0 ? " <span class='unread'>($count)</span>" : "";
|
2020-03-28 14:11:14 +00:00
|
|
|
$event->add_link("Private Messages$h_count", make_link("user", null, "private-messages"));
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function onUserPageBuilding(UserPageBuildingEvent $event)
|
|
|
|
{
|
|
|
|
global $page, $user;
|
|
|
|
$duser = $event->display_user;
|
|
|
|
if (!$user->is_anonymous() && !$duser->is_anonymous()) {
|
2023-02-04 14:47:26 +00:00
|
|
|
$pms = PM::get_pms($duser);
|
|
|
|
if (!is_null($pms)) {
|
|
|
|
$this->theme->display_pms($page, $pms);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
if ($user->id != $duser->id) {
|
|
|
|
$this->theme->display_composer($page, $user, $duser);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function onPageRequest(PageRequestEvent $event)
|
|
|
|
{
|
2019-10-02 10:49:32 +01:00
|
|
|
global $cache, $database, $page, $user;
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($event->page_matches("pm")) {
|
2019-12-15 20:40:05 +00:00
|
|
|
switch ($event->get_arg(0)) {
|
|
|
|
case "read":
|
|
|
|
if ($user->can(Permissions::READ_PM)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
$pm_id = int_escape($event->get_arg(1));
|
|
|
|
$pm = $database->get_row("SELECT * FROM private_message WHERE id = :id", ["id" => $pm_id]);
|
|
|
|
if (is_null($pm)) {
|
|
|
|
$this->theme->display_error(404, "No such PM", "There is no PM #$pm_id");
|
2019-07-09 09:10:21 -05:00
|
|
|
} elseif (($pm["to_id"] == $user->id) || $user->can(Permissions::VIEW_OTHER_PMS)) {
|
2020-01-26 19:44:36 +00:00
|
|
|
$from_user = User::by_id((int)$pm["from_id"]);
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($pm["to_id"] == $user->id) {
|
2020-10-26 17:13:41 +00:00
|
|
|
$database->execute("UPDATE private_message SET is_read=true WHERE id = :id", ["id" => $pm_id]);
|
2019-10-02 10:49:32 +01:00
|
|
|
$cache->delete("pm-count-{$user->id}");
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2023-02-04 14:47:26 +00:00
|
|
|
$this->theme->display_message($page, $from_user, $user, PM::from_row($pm));
|
2019-05-28 17:59:38 +01:00
|
|
|
} else {
|
2019-11-04 01:04:08 +00:00
|
|
|
$this->theme->display_permission_denied();
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2019-12-15 20:40:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "delete":
|
|
|
|
if ($user->can(Permissions::READ_PM)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($user->check_auth_token()) {
|
|
|
|
$pm_id = int_escape($_POST["pm_id"]);
|
|
|
|
$pm = $database->get_row("SELECT * FROM private_message WHERE id = :id", ["id" => $pm_id]);
|
|
|
|
if (is_null($pm)) {
|
|
|
|
$this->theme->display_error(404, "No such PM", "There is no PM #$pm_id");
|
2019-07-09 09:10:21 -05:00
|
|
|
} elseif (($pm["to_id"] == $user->id) || $user->can(Permissions::VIEW_OTHER_PMS)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
$database->execute("DELETE FROM private_message WHERE id = :id", ["id" => $pm_id]);
|
2019-10-02 10:49:32 +01:00
|
|
|
$cache->delete("pm-count-{$user->id}");
|
2019-05-28 17:59:38 +01:00
|
|
|
log_info("pm", "Deleted PM #$pm_id", "PM deleted");
|
2019-06-18 20:58:28 -05:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2020-03-27 19:41:34 +00:00
|
|
|
$page->set_redirect(referer_or(make_link()));
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
2019-12-15 20:40:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "send":
|
|
|
|
if ($user->can(Permissions::SEND_PM)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($user->check_auth_token()) {
|
|
|
|
$to_id = int_escape($_POST["to_id"]);
|
|
|
|
$from_id = $user->id;
|
|
|
|
$subject = $_POST["subject"];
|
|
|
|
$message = $_POST["message"];
|
2022-01-17 17:06:20 +00:00
|
|
|
send_event(new SendPMEvent(new PM($from_id, get_real_ip(), $to_id, $subject, $message)));
|
2019-12-15 19:47:18 +00:00
|
|
|
$page->flash("PM sent");
|
2019-06-18 20:58:28 -05:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2020-03-27 19:41:34 +00:00
|
|
|
$page->set_redirect(referer_or(make_link()));
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2019-12-15 20:40:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$this->theme->display_error(400, "Invalid action", "That's not something you can do with a PM");
|
|
|
|
break;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function onSendPM(SendPMEvent $event)
|
|
|
|
{
|
2019-10-02 10:49:32 +01:00
|
|
|
global $cache, $database;
|
2019-05-28 17:59:38 +01:00
|
|
|
$database->execute(
|
|
|
|
"
|
2009-05-11 14:09:24 -07:00
|
|
|
INSERT INTO private_message(
|
|
|
|
from_id, from_ip, to_id,
|
|
|
|
sent_date, subject, message)
|
2011-02-22 17:29:31 -05:00
|
|
|
VALUES(:fromid, :fromip, :toid, now(), :subject, :message)",
|
2019-05-28 17:59:38 +01:00
|
|
|
["fromid" => $event->pm->from_id, "fromip" => $event->pm->from_ip,
|
|
|
|
"toid" => $event->pm->to_id, "subject" => $event->pm->subject, "message" => $event->pm->message]
|
|
|
|
);
|
2019-10-02 10:49:32 +01:00
|
|
|
$cache->delete("pm-count-{$event->pm->to_id}");
|
2019-05-28 17:59:38 +01:00
|
|
|
log_info("pm", "Sent PM to User #{$event->pm->to_id}");
|
|
|
|
}
|
2008-09-07 05:57:28 +00:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
private function count_pms(User $user)
|
|
|
|
{
|
2019-10-02 10:49:32 +01:00
|
|
|
global $cache, $database;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
2019-10-02 10:49:32 +01:00
|
|
|
$count = $cache->get("pm-count:{$user->id}");
|
2023-02-02 16:04:35 +00:00
|
|
|
if (is_null($count)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
$count = $database->get_one("
|
2012-03-02 17:41:25 +00:00
|
|
|
SELECT count(*)
|
|
|
|
FROM private_message
|
2012-03-02 18:03:34 +00:00
|
|
|
WHERE to_id = :to_id
|
2012-03-02 17:41:25 +00:00
|
|
|
AND is_read = :is_read
|
2020-10-26 17:13:41 +00:00
|
|
|
", ["to_id" => $user->id, "is_read" => false]);
|
2019-10-02 10:49:32 +01:00
|
|
|
$cache->set("pm-count:{$user->id}", $count, 600);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
return $count;
|
|
|
|
}
|
2008-09-07 05:57:28 +00:00
|
|
|
}
|