| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // {{{ License
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:30:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | // This file is part of GNU social - https://www.gnu.org/software/social
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social is free software: you can redistribute it and/or modify
 | 
					
						
							|  |  |  | // it under the terms of the GNU Affero General Public License as published by
 | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or
 | 
					
						
							|  |  |  | // (at your option) any later version.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social is distributed in the hope that it will be useful,
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					
						
							|  |  |  | // GNU Affero General Public License for more details.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU Affero General Public License
 | 
					
						
							|  |  |  | // along with GNU social.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:30:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | // }}}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Component\Posting; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:30:12 +00:00
										 |  |  | use App\Core\Cache; | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | use App\Core\DB\DB; | 
					
						
							|  |  |  | use App\Core\Event; | 
					
						
							|  |  |  | use App\Core\Form; | 
					
						
							| 
									
										
										
										
											2021-09-20 12:34:28 +01:00
										 |  |  | use App\Core\GSFile; | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | use function App\Core\I18n\_m; | 
					
						
							| 
									
										
										
										
											2021-04-18 02:17:57 +01:00
										 |  |  | use App\Core\Modules\Component; | 
					
						
							| 
									
										
										
										
											2021-09-20 17:02:35 +01:00
										 |  |  | use App\Core\Security; | 
					
						
							| 
									
										
										
										
											2021-09-18 03:22:27 +01:00
										 |  |  | use App\Entity\Actor; | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  | use App\Entity\ActorToAttachment; | 
					
						
							| 
									
										
										
										
											2021-04-27 20:53:59 +00:00
										 |  |  | use App\Entity\Attachment; | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  | use App\Entity\AttachmentToNote; | 
					
						
							| 
									
										
										
										
											2020-09-10 20:35:57 +00:00
										 |  |  | use App\Entity\Note; | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | use App\Util\Common; | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:33 +01:00
										 |  |  | use App\Util\Exception\ClientException; | 
					
						
							| 
									
										
										
										
											2021-04-18 02:17:57 +01:00
										 |  |  | use App\Util\Exception\InvalidFormException; | 
					
						
							| 
									
										
										
										
											2020-09-05 21:28:53 +00:00
										 |  |  | use App\Util\Exception\RedirectException; | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  | use App\Util\Exception\ServerException; | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  | use App\Util\Formatting; | 
					
						
							| 
									
										
										
										
											2020-08-22 01:24:55 +01:00
										 |  |  | use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | 
					
						
							| 
									
										
										
										
											2020-08-20 00:40:06 +00:00
										 |  |  | use Symfony\Component\Form\Extension\Core\Type\FileType; | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | use Symfony\Component\Form\Extension\Core\Type\SubmitType; | 
					
						
							|  |  |  | use Symfony\Component\Form\Extension\Core\Type\TextareaType; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-18 02:17:57 +01:00
										 |  |  | class Posting extends Component | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * HTML render event handler responsible for adding and handling | 
					
						
							|  |  |  |      * the result of adding the note submission form, only if a user is logged in | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @throws ClientException | 
					
						
							|  |  |  |      * @throws RedirectException | 
					
						
							|  |  |  |      * @throws ServerException | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-09-18 07:27:17 +01:00
										 |  |  |     public function onAppendRightPostingBlock(array $vars, array &$res): bool | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-09-17 21:51:55 +01:00
										 |  |  |         if (($user = Common::user()) === null) { | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |             return Event::next; | 
					
						
							| 
									
										
										
										
											2020-08-20 00:40:06 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 21:28:53 +00:00
										 |  |  |         $actor_id = $user->getId(); | 
					
						
							| 
									
										
										
										
											2020-08-28 20:16:26 +00:00
										 |  |  |         $to_tags  = []; | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         $tags     = Cache::get("actor-circle-{$actor_id}", | 
					
						
							| 
									
										
										
										
											2021-09-18 03:22:27 +01:00
										 |  |  |                                fn () => DB::dql('select c.tag from App\Entity\ActorCircle c where c.tagger = :tagger', ['tagger' => $actor_id])); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:30:12 +00:00
										 |  |  |         foreach ($tags as $t) { | 
					
						
							| 
									
										
										
										
											2020-08-28 07:15:56 +01:00
										 |  |  |             $t           = $t['tag']; | 
					
						
							|  |  |  |             $to_tags[$t] = $t; | 
					
						
							| 
									
										
										
										
											2020-08-22 01:24:55 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-06 21:01:20 +01:00
										 |  |  |         $placeholder_strings = ['How are you feeling?', 'Have something to share?', 'How was your day?']; | 
					
						
							|  |  |  |         Event::handle('PostingPlaceHolderString', [&$placeholder_strings]); | 
					
						
							|  |  |  |         $placeholder = $placeholder_strings[array_rand($placeholder_strings)]; | 
					
						
							| 
									
										
										
										
											2020-08-27 03:25:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |         $initial_content = ''; | 
					
						
							|  |  |  |         Event::handle('PostingInitialContent', [&$initial_content]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         $available_content_types = ['Plain Text' => 'text/plain']; | 
					
						
							|  |  |  |         Event::handle('PostingAvailableContentTypes', [&$available_content_types]); | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $request     = $vars['request']; | 
					
						
							|  |  |  |         $form_params = [ | 
					
						
							| 
									
										
										
										
											2021-09-13 14:17:58 +01:00
										 |  |  |             ['content',     TextareaType::class, ['label' => _m('Content:'), 'data' => $initial_content, 'attr' => ['placeholder' => _m($placeholder)]]], | 
					
						
							|  |  |  |             ['attachments', FileType::class,     ['label' => _m('Attachments:'), 'data' => null, 'multiple' => true, 'required' => false]], | 
					
						
							|  |  |  |             ['visibility',  ChoiceType::class,   ['label' => _m('Visibility:'), 'multiple' => false, 'expanded' => false, 'data' => 'public', 'choices' => [_m('Public') => 'public', _m('Instance') => 'instance', _m('Private') => 'private']]], | 
					
						
							|  |  |  |             ['to',          ChoiceType::class,   ['label' => _m('To:'), 'multiple' => false, 'expanded' => false, 'choices' => $to_tags]], | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |         ]; | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         if (count($available_content_types) > 1) { | 
					
						
							|  |  |  |             $form_params[] = ['content_type', ChoiceType::class, | 
					
						
							|  |  |  |                 ['label'      => _m('Text format:'), 'multiple' => false, 'expanded' => false, | 
					
						
							|  |  |  |                     'data'    => $available_content_types[array_key_first($available_content_types)], | 
					
						
							|  |  |  |                     'choices' => $available_content_types, ], ]; | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         $form_params[] = ['post_note',   SubmitType::class,   ['label' => _m('Post')]]; | 
					
						
							|  |  |  |         $form          = Form::create($form_params); | 
					
						
							| 
									
										
										
										
											2020-08-26 07:56:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  |         $form->handleRequest($request); | 
					
						
							|  |  |  |         if ($form->isSubmitted()) { | 
					
						
							|  |  |  |             $data = $form->getData(); | 
					
						
							|  |  |  |             if ($form->isValid()) { | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |                 $content_type = $data['content_type'] ?? $available_content_types[array_key_first($available_content_types)]; | 
					
						
							|  |  |  |                 self::storeLocalNote($user->getActor(), $data['content'], $content_type, $data['attachments']); | 
					
						
							| 
									
										
										
										
											2020-09-05 21:28:53 +00:00
										 |  |  |                 throw new RedirectException(); | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2020-09-08 00:12:33 +00:00
										 |  |  |                 throw new InvalidFormException(); | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 07:27:17 +01:00
										 |  |  |         $res = $form->createView(); | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return Event::next; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-10 20:35:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Store the given note with $content and $attachments, created by | 
					
						
							|  |  |  |      * $actor_id, possibly as a reply to note $reply_to and with flag | 
					
						
							|  |  |  |      * $is_local. Sanitizes $content and $attachments | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Actor     $actor | 
					
						
							|  |  |  |      * @param string    $content | 
					
						
							|  |  |  |      * @param string    $content_type | 
					
						
							|  |  |  |      * @param array     $attachments | 
					
						
							|  |  |  |      * @param null|Note $reply_to | 
					
						
							|  |  |  |      * @param null|Note $repeat_of | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws ClientException | 
					
						
							|  |  |  |      * @throws ServerException | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-09-18 03:22:27 +01:00
										 |  |  |     public static function storeLocalNote(Actor $actor, string $content, string $content_type, array $attachments, ?Note $reply_to = null, ?Note $repeat_of = null) | 
					
						
							| 
									
										
										
										
											2020-09-10 20:35:57 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         $rendered = null; | 
					
						
							|  |  |  |         Event::handle('RenderNoteContent', [$content, $content_type, &$rendered, $actor, $reply_to]); | 
					
						
							| 
									
										
										
										
											2021-07-22 13:02:09 +01:00
										 |  |  |         $note = Note::create([ | 
					
						
							| 
									
										
										
										
											2021-09-18 03:22:27 +01:00
										 |  |  |             'actor_id'     => $actor->getId(), | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |             'content'      => $content, | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |             'content_type' => $content_type, | 
					
						
							|  |  |  |             'rendered'     => $rendered, | 
					
						
							|  |  |  |             'is_local'     => true, | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |         ]); | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $processed_attachments = []; | 
					
						
							|  |  |  |         /** @var \Symfony\Component\HttpFoundation\File\UploadedFile[] $attachments */ | 
					
						
							|  |  |  |         foreach ($attachments as $f) { | 
					
						
							|  |  |  |             $filesize      = $f->getSize(); | 
					
						
							|  |  |  |             $max_file_size = Common::config('attachments', 'file_quota'); | 
					
						
							|  |  |  |             if ($max_file_size < $filesize) { | 
					
						
							|  |  |  |                 throw new ClientException(_m('No file may be larger than {quota} bytes and the file you sent was {size} bytes. ' . | 
					
						
							|  |  |  |                     'Try to upload a smaller version.', ['quota' => $max_file_size, 'size' => $filesize])); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Event::handle('EnforceUserFileQuota', [$filesize, $actor->getId()]); | 
					
						
							|  |  |  |             $processed_attachments[] = [GSFile::sanitizeAndStoreFileAsAttachment($f), $f->getClientOriginalName()]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         DB::persist($note); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Need file and note ids for the next step
 | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         Event::handle('ProcessNoteContent', [$note->getId(), $content, $content_type]); | 
					
						
							| 
									
										
										
										
											2020-09-10 20:35:57 +00:00
										 |  |  |         DB::flush(); | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($processed_attachments != []) { | 
					
						
							|  |  |  |             foreach ($processed_attachments as [$a, $fname]) { | 
					
						
							| 
									
										
										
										
											2021-09-20 12:34:28 +01:00
										 |  |  |                 if (DB::count('actor_to_attachment', $args = ['attachment_id' => $a->getId(), 'actor_id' => $actor->getId()]) === 0) { | 
					
						
							| 
									
										
										
										
											2021-09-18 03:44:02 +01:00
										 |  |  |                     DB::persist(ActorToAttachment::create($args)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 DB::persist(AttachmentToNote::create(['attachment_id' => $a->getId(), 'note_id' => $note->getId(), 'title' => $fname])); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-14 16:47:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 03:22:27 +01:00
										 |  |  |     public function onRenderNoteContent(string $content, string $content_type, ?string &$rendered, Actor $author, ?Note $reply_to = null) | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-09-20 17:02:35 +01:00
										 |  |  |         switch ($content_type) { | 
					
						
							|  |  |  |             case 'text/plain': | 
					
						
							|  |  |  |                 $rendered = Formatting::renderPlainText($content); | 
					
						
							|  |  |  |                 $rendered = Formatting::linkifyMentions($rendered, $author, $reply_to); | 
					
						
							|  |  |  |                 return Event::stop; | 
					
						
							|  |  |  |             case 'text/html': | 
					
						
							|  |  |  |                 // TODO: It has to linkify and stuff as well
 | 
					
						
							|  |  |  |                 $rendered = Security::sanitize($content); | 
					
						
							|  |  |  |                 return Event::stop; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 return Event::next; | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-10 20:35:57 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-27 20:53:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get a unique representation of a file on disk | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * This can be used in the future to deduplicate images by visual content | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @param string      $filename | 
					
						
							|  |  |  |      * @param null|string $out_hash | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							| 
									
										
										
										
											2021-04-27 20:53:59 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  |     public function onHashFile(string $filename, ?string &$out_hash): bool | 
					
						
							| 
									
										
										
										
											2021-04-27 20:53:59 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         $out_hash = hash_file(Attachment::FILEHASH_ALGO, $filename); | 
					
						
							|  |  |  |         return Event::stop; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-28 21:53:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  |      * Fill the list with allowed sizes for an attachment, to prevent potential DoS'ing by requesting thousands of different thumbnail sizes | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param null|array $sizes | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							| 
									
										
										
										
											2021-04-28 21:53:02 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-08-31 18:33:58 +01:00
										 |  |  |     public function onGetAllowedThumbnailSizes(?array &$sizes): bool | 
					
						
							| 
									
										
										
										
											2021-04-28 21:53:02 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         $sizes[] = ['width' => Common::config('thumbnail', 'width'), 'height' => Common::config('thumbnail', 'height')]; | 
					
						
							|  |  |  |         return Event::next; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-14 15:46:08 +00:00
										 |  |  | } |