3 * @copyright Copyright (C) 2010-2023, the Friendica project
5 * @license GNU AGPL version 3 or any later version
7 * This program is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU Affero General Public License as
9 * published by the Free Software Foundation, either version 3 of the
10 * License, or (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU Affero General Public License for more details.
17 * You should have received a copy of the GNU Affero General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
22 namespace Friendica\Model;
24 use Friendica\Core\ACL;
25 use Friendica\Core\Logger;
26 use Friendica\Core\System;
27 use Friendica\Core\Worker;
28 use Friendica\Database\DBA;
30 use Friendica\Protocol\Activity;
31 use Friendica\Protocol\Delivery;
32 use Friendica\Util\DateTimeFormat;
35 * Class to handle private messages
40 * Insert private message
43 * @param bool $notification
44 * @return int|boolean Message ID or false on error
45 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
46 * @throws \ImagickException
48 public static function insert(array $msg, bool $notification = true)
50 if (!isset($msg['reply'])) {
51 $msg['reply'] = DBA::exists('mail', ['parent-uri' => $msg['parent-uri']]);
54 if (empty($msg['convid'])) {
55 $mail = DBA::selectFirst('mail', ['convid'], ["`convid` != 0 AND `parent-uri` = ?", $msg['parent-uri']]);
56 if (DBA::isResult($mail)) {
57 $msg['convid'] = $mail['convid'];
61 if (empty($msg['guid'])) {
62 $msg['guid'] = Item::guidFromUri($msg['uri'], parse_url($msg['from-url'], PHP_URL_HOST));
65 $msg['created'] = (!empty($msg['created']) ? DateTimeFormat::utc($msg['created']) : DateTimeFormat::utcNow());
67 $msg['author-id'] = Contact::getIdForURL($msg['from-url'], 0, false);
68 $msg['uri-id'] = ItemURI::insert(['uri' => $msg['uri'], 'guid' => $msg['guid']]);
69 $msg['parent-uri-id'] = ItemURI::getIdByURI($msg['parent-uri']);
73 if (DBA::exists('mail', ['uri' => $msg['uri'], 'uid' => $msg['uid']])) {
75 Logger::info('duplicate message already delivered.');
79 if ($msg['reply'] && DBA::isResult($reply = DBA::selectFirst('mail', ['uri', 'uri-id'], ['parent-uri' => $msg['parent-uri'], 'reply' => false]))) {
80 $msg['thr-parent'] = $reply['uri'];
81 $msg['thr-parent-id'] = $reply['uri-id'];
83 $msg['thr-parent'] = $msg['uri'];
84 $msg['thr-parent-id'] = $msg['uri-id'];
87 DBA::insert('mail', $msg);
89 $msg['id'] = DBA::lastInsertId();
93 if (!empty($msg['convid'])) {
94 DBA::update('conv', ['updated' => DateTimeFormat::utcNow()], ['id' => $msg['convid']]);
98 $user = User::getById($msg['uid']);
99 // send notifications.
101 'type' => Notification\Type::MAIL,
102 'otype' => Notification\ObjectType::MAIL,
103 'verb' => Activity::POST,
104 'uid' => $user['uid'],
105 'cid' => $msg['contact-id'],
106 'link' => DI::baseUrl() . '/message/' . $msg['id'],
109 DI::notify()->createFromArray($notif_params);
111 Logger::info('Mail is processed, notification was sent.', ['id' => $msg['id'], 'uri' => $msg['uri']]);
118 * Send private message
120 * @param integer $sender_uid the user id of the sender
121 * @param integer $recipient recipient id, default 0
122 * @param string $body message body, default empty
123 * @param string $subject message subject, default empty
124 * @param string $replyto reply to
126 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
128 public static function send(int $sender_uid, int $recipient = 0, string $body = '', string $subject = '', string $replyto = ''): int
136 if (!strlen($subject)) {
137 $subject = DI::l10n()->t('[no subject]');
140 $me = DBA::selectFirst('contact', [], ['uid' => $sender_uid, 'self' => true]);
141 if (!DBA::isResult($me)) {
145 $contacts = ACL::getValidMessageRecipientsForUser($sender_uid);
147 $contactIndex = array_search($recipient, array_column($contacts, 'id'));
148 if ($contactIndex === false) {
152 $contact = $contacts[$contactIndex];
154 Photo::setPermissionFromBody($body, $sender_uid, $me['id'], '<' . $contact['id'] . '>', '', '', '');
156 $guid = System::createUUID();
157 $uri = Item::newURI($guid);
162 // look for any existing conversation structure
164 if (strlen($replyto)) {
166 $condition = ["`uid` = ? AND (`uri` = ? OR `parent-uri` = ?)",
167 $sender_uid, $replyto, $replyto];
168 $mail = DBA::selectFirst('mail', ['convid'], $condition);
169 if (DBA::isResult($mail)) {
170 $convid = $mail['convid'];
176 // create a new conversation
177 $conv_guid = System::createUUID();
178 $convuri = $contact['addr'] . ':' . $conv_guid;
180 $fields = ['uid' => $sender_uid, 'guid' => $conv_guid, 'creator' => $me['addr'],
181 'created' => DateTimeFormat::utcNow(), 'updated' => DateTimeFormat::utcNow(),
182 'subject' => $subject, 'recips' => $contact['addr'] . ';' . $me['addr']];
183 if (DBA::insert('conv', $fields)) {
184 $convid = DBA::lastInsertId();
189 Logger::warning('conversation not found.');
193 if (!strlen($replyto)) {
197 $post_id = self::insert(
199 'uid' => $sender_uid,
202 'from-name' => $me['name'],
203 'from-photo' => $me['thumb'],
204 'from-url' => $me['url'],
205 'contact-id' => $recipient,
212 'parent-uri' => $replyto,
213 'created' => DateTimeFormat::utcNow()
220 * When a photo was uploaded into the message using the (profile wall) ajax
221 * uploader, The permissions are initially set to disallow anybody but the
222 * owner from seeing it. This is because the permissions may not yet have been
223 * set for the post. If it's private, the photo permissions should be set
224 * appropriately. But we didn't know the final permissions on the post until
225 * now. So now we'll look for links of uploaded messages that are in the
226 * post and set them to the same permissions as the post itself.
230 if (preg_match_all("/\[img\](.*?)\[\/img\]/", $body, $match)) {
232 if (count($images)) {
233 foreach ($images as $image) {
234 $image_rid = Photo::ridFromURI($image);
235 if (!empty($image_rid)) {
236 Photo::update(['allow_cid' => '<' . $recipient . '>'], ['resource-id' => $image_rid, 'album' => 'Wall Photos', 'uid' => $sender_uid]);
243 Worker::add(Worker::PRIORITY_HIGH, "Notifier", Delivery::MAIL, $post_id);
244 return intval($post_id);
251 * @param array $recipient recipient, default empty
252 * @param string $body message body, default empty
253 * @param string $subject message subject, default empty
254 * @param string $replyto reply to, default empty
256 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
257 * @throws \ImagickException
259 public static function sendWall(array $recipient = [], string $body = '', string $subject = '', string $replyto = ''): int
265 if (!strlen($subject)) {
266 $subject = DI::l10n()->t('[no subject]');
269 $guid = System::createUUID();
270 $uri = Item::newURI($guid);
272 $me = Contact::getByURL($replyto);
277 $conv_guid = System::createUUID();
279 $recip_handle = $recipient['nickname'] . '@' . substr(DI::baseUrl(), strpos(DI::baseUrl(), '://') + 3);
281 $sender_handle = $me['addr'];
283 $handles = $recip_handle . ';' . $sender_handle;
286 $fields = ['uid' => $recipient['uid'], 'guid' => $conv_guid, 'creator' => $sender_handle,
287 'created' => DateTimeFormat::utcNow(), 'updated' => DateTimeFormat::utcNow(),
288 'subject' => $subject, 'recips' => $handles];
289 if (DBA::insert('conv', $fields)) {
290 $convid = DBA::lastInsertId();
294 Logger::warning('conversation not found.');
300 'uid' => $recipient['uid'],
303 'from-name' => $me['name'],
304 'from-photo' => $me['photo'],
305 'from-url' => $me['url'],
313 'parent-uri' => $me['url'],
314 'created' => DateTimeFormat::utcNow(),