| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2021-10-10 09:26:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | declare(strict_types = 1); | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | // {{{ License
 | 
					
						
							|  |  |  | // 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/>.
 | 
					
						
							|  |  |  | // }}}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @author    Hugo Sales <hugo@hsal.es> | 
					
						
							|  |  |  |  * @copyright 2021 Free Software Foundation, Inc http://www.fsf.org | 
					
						
							|  |  |  |  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Plugin\Reply\Controller; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-07 23:31:20 +00:00
										 |  |  | use App\Core\Controller\FeedController; | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | use App\Core\DB\DB; | 
					
						
							|  |  |  | use App\Core\Form; | 
					
						
							|  |  |  | use function App\Core\I18n\_m; | 
					
						
							|  |  |  | use App\Core\Log; | 
					
						
							|  |  |  | use App\Core\Router\Router; | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  | use App\Entity\Actor; | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | use App\Entity\Note; | 
					
						
							|  |  |  | use App\Util\Common; | 
					
						
							|  |  |  | use App\Util\Exception\ClientException; | 
					
						
							|  |  |  | use App\Util\Exception\InvalidFormException; | 
					
						
							| 
									
										
										
										
											2021-09-06 19:49:03 +01:00
										 |  |  | use App\Util\Exception\NoSuchNoteException; | 
					
						
							| 
									
										
										
										
											2021-12-01 01:31:02 +00:00
										 |  |  | use App\Util\Exception\NotImplementedException; | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | use App\Util\Exception\RedirectException; | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  | use App\Util\Form\FormFields; | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  | use Component\Posting\Posting; | 
					
						
							|  |  |  | use Plugin\Reply\Entity\NoteReply; | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | use Symfony\Component\Form\Extension\Core\Type\FileType; | 
					
						
							|  |  |  | use Symfony\Component\Form\Extension\Core\Type\SubmitType; | 
					
						
							|  |  |  | use Symfony\Component\Form\Extension\Core\Type\TextareaType; | 
					
						
							|  |  |  | use Symfony\Component\HttpFoundation\Request; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-07 23:31:20 +00:00
										 |  |  | class Reply extends FeedController | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Controller for the note reply non-JS page | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @throws \App\Util\Exception\NoLoggedInUser | 
					
						
							|  |  |  |      * @throws \App\Util\Exception\ServerException | 
					
						
							|  |  |  |      * @throws ClientException | 
					
						
							|  |  |  |      * @throws InvalidFormException | 
					
						
							|  |  |  |      * @throws NoSuchNoteException | 
					
						
							|  |  |  |      * @throws RedirectException | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |     public function replyAddNote(Request $request, int $id) | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         $user     = Common::ensureLoggedIn(); | 
					
						
							|  |  |  |         $actor_id = $user->getId(); | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $note = Note::getWithPK($id); | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |         if (\is_null($note) || !$note->isVisibleTo($user)) { | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |             throw new NoSuchNoteException(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  |         // TODO shouldn't this be the posting form?
 | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |         $form = Form::create([ | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  |             ['content', TextareaType::class, ['label' => _m('Reply'), 'label_attr' => ['class' => 'section-form-label'], 'help' => _m('Please input your reply.')]], | 
					
						
							| 
									
										
										
										
											2021-11-28 12:25:23 +00:00
										 |  |  |             FormFields::language($user->getActor(), context_actor: $note->getActor(), label: _m('Note language')), | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |             ['attachments', FileType::class, ['label' => ' ', 'multiple' => true, 'required' => false]], | 
					
						
							|  |  |  |             ['replyform', SubmitType::class, ['label' => _m('Submit')]], | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |         ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $form->handleRequest($request); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($form->isSubmitted()) { | 
					
						
							|  |  |  |             $data = $form->getData(); | 
					
						
							|  |  |  |             if ($form->isValid()) { | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |                 // Create a new note with the same content as the original
 | 
					
						
							|  |  |  |                 $reply = Posting::storeLocalNote( | 
					
						
							|  |  |  |                     actor: Actor::getWithPK($actor_id), | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |                     content: $data['content'], | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |                     content_type: 'text/plain', // TODO
 | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  |                     language: $data['language'], | 
					
						
							| 
									
										
										
										
											2021-09-09 03:46:30 +01:00
										 |  |  |                     attachments: $data['attachments'], | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |                 ); | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 // Update DB
 | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                 DB::persist($reply); | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |                 DB::flush(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Find the id of the note we just created
 | 
					
						
							|  |  |  |                 $reply_id = $reply->getId(); | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                 $og_id    = $note->getId(); | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 // Add it to note_repeat table
 | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                 if (!\is_null($reply_id)) { | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |                     DB::persist(NoteReply::create([ | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                         'note_id'  => $reply_id, | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |                         'actor_id' => $actor_id, | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                         'reply_to' => $og_id, | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |                     ])); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Update DB one last time
 | 
					
						
							|  |  |  |                 DB::flush(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                 // Redirect user to where they came from
 | 
					
						
							|  |  |  |                 // Prevent open redirect
 | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  |                 if (!\is_null($from = $this->string('from'))) { | 
					
						
							|  |  |  |                     if (Router::isAbsolute($from)) { | 
					
						
							|  |  |  |                         Log::warning("Actor {$actor_id} attempted to reply to a note and then get redirected to another host, or the URL was invalid ({$from})"); | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |                         throw new ClientException(_m('Can not redirect to outside the website from here'), 400); // 400 Bad request (deceptive)
 | 
					
						
							|  |  |  |                     } else { | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                         // TODO anchor on element id
 | 
					
						
							| 
									
										
										
										
											2021-11-25 23:08:30 +00:00
										 |  |  |                         throw new RedirectException($from); | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2021-11-16 19:36:17 +00:00
										 |  |  |                     // If we don't have a URL to return to, go to the instance root
 | 
					
						
							|  |  |  |                     throw new RedirectException('root'); | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 throw new InvalidFormException(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return [ | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |             '_template' => 'reply/add_reply.html.twig', | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |             'note'      => $note, | 
					
						
							| 
									
										
										
										
											2021-11-07 01:32:06 +00:00
										 |  |  |             'add_reply' => $form->createView(), | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  |         ]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-01 01:31:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public function replies(Request $request) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // TODO replies
 | 
					
						
							|  |  |  |         throw new NotImplementedException; | 
					
						
							|  |  |  |         $actor_id = Common::ensureLoggedIn()->getId(); | 
					
						
							|  |  |  |         $notes    = DB::dql('select n from App\Entity\Note n ' | 
					
						
							|  |  |  |                             . 'where n.reply_to is not null and n.actor_id = :id ' | 
					
						
							|  |  |  |                             . 'order by n.created DESC', ['id' => $actor_id], ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $notes_out = null; | 
					
						
							|  |  |  |         Event::handle('FormatNoteList', [$notes, &$notes_out]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-07 23:31:20 +00:00
										 |  |  |         return $this->process_feed([ | 
					
						
							| 
									
										
										
										
											2021-12-01 01:31:02 +00:00
										 |  |  |             '_template'  => 'feeds/feed.html.twig', | 
					
						
							|  |  |  |             'notes'      => $notes_out, | 
					
						
							|  |  |  |             'page_title' => 'Replies feed', | 
					
						
							| 
									
										
										
										
											2021-12-07 23:31:20 +00:00
										 |  |  |         ]); | 
					
						
							| 
									
										
										
										
											2021-12-01 01:31:02 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-05 18:25:53 +01:00
										 |  |  | } |