| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-10 09:26:18 +01:00
										 |  |  | declare(strict_types = 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00: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/>.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // }}}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Plugin\FileQuota; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use App\Core\Cache; | 
					
						
							| 
									
										
										
										
											2022-03-27 15:19:09 +01:00
										 |  |  | use App\Core\DB; | 
					
						
							| 
									
										
										
										
											2021-08-14 16:17:15 +01:00
										 |  |  | use App\Core\Event; | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  | use function App\Core\I18n\_m; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | use App\Core\Modules\Plugin; | 
					
						
							|  |  |  | use App\Util\Common; | 
					
						
							|  |  |  | use App\Util\Exception\ClientException; | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  | use App\Util\Exception\ServerException; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Check attachment file size quotas | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package   GNUsocial | 
					
						
							|  |  |  |  * @ccategory Attachment | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |  * @author    Hugo Sales <hugo@hsal.es> | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |  * @copyright 2021 Free Software Foundation, Inc http://www.fsf.org | 
					
						
							|  |  |  |  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class FileQuota extends Plugin | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-03-28 21:18:44 +01:00
										 |  |  |     public static function version(): string | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         return '1.0.0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |      * Check file size to ensure it respects configured file size | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |      * quotas. Handles per file, per user and per user-month quotas. | 
					
						
							|  |  |  |      * Throws on quota violations | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @throws ClientException | 
					
						
							|  |  |  |      * @throws ServerException | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |     public function onEnforceUserFileQuota(int $filesize, int $user_id): bool | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-10-10 09:26:18 +01:00
										 |  |  |         $query = <<<'END' | 
					
						
							|  |  |  |             select sum(at.size) as total | 
					
						
							|  |  |  |                 from attachment at | 
					
						
							|  |  |  |                     join actor_to_attachment ua with at.id = ua.attachment_id | 
					
						
							|  |  |  |                 where ua.actor_id = :actor_id and at.size is not null | 
					
						
							|  |  |  |             END; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |         $max_user_quota = Common::config('attachments', 'user_quota'); | 
					
						
							|  |  |  |         if ($max_user_quota !== false) { // If not disabled
 | 
					
						
							|  |  |  |             $cache_key_user_total = "FileQuota-total-user-{$user_id}"; | 
					
						
							| 
									
										
										
										
											2021-07-22 20:57:03 +01:00
										 |  |  |             $user_total           = Cache::get($cache_key_user_total, fn () => DB::dql($query, ['actor_id' => $user_id])[0]['total']); | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |             Cache::set($cache_key_user_total, $user_total + $filesize); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |             if ($user_total + $filesize > $max_user_quota) { | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |                 // TRANS: Message given if an upload would exceed user quota.
 | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |                 throw new ClientException(_m('A file this large would exceed your user quota of {quota} bytes.', ['quota' => $max_user_quota])); | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $query .= ' AND MONTH(at.modified) = MONTH(CURRENT_DATE())' | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |             . ' AND YEAR(at.modified)  = YEAR(CURRENT_DATE())'; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $monthly_quota = Common::config('attachments', 'monthly_quota'); | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |         if ($monthly_quota !== false) { // If not disabled
 | 
					
						
							|  |  |  |             $cache_key_user_monthly = "FileQuota-monthly-user-{$user_id}"; | 
					
						
							| 
									
										
										
										
											2021-07-22 20:57:03 +01:00
										 |  |  |             $monthly_total          = Cache::get($cache_key_user_monthly, fn () => DB::dql($query, ['actor_id' => $user_id])[0]['total']); | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |             Cache::set($cache_key_user_monthly, $monthly_total + $filesize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($monthly_total + $filesize > $monthly_quota) { | 
					
						
							|  |  |  |                 // TRANS: Message given if an upload would exceed user quota.
 | 
					
						
							|  |  |  |                 throw new ClientException(_m('A file this large would exceed your monthly quota of {quota} bytes.', ['quota' => $monthly_quota])); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Event::next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Event raised when GNU social polls the plugin for information about it. | 
					
						
							|  |  |  |      * Adds this plugin's version information to $versions array | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2021-09-06 23:47:28 +01:00
										 |  |  |      * @param array $versions inherited from parent | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return bool true hook value | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function onPluginVersion(array &$versions): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $versions[] = [ | 
					
						
							| 
									
										
										
										
											2021-10-10 09:26:18 +01:00
										 |  |  |             'name'     => 'FileQuota', | 
					
						
							|  |  |  |             'version'  => $this->version(), | 
					
						
							|  |  |  |             'author'   => 'Hugo Sales', | 
					
						
							|  |  |  |             'homepage' => GNUSOCIAL_PROJECT_URL, | 
					
						
							|  |  |  |             'description', // TRANS: Plugin description. => _m('Plugin to manage user quotas.'),
 | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |         ]; | 
					
						
							|  |  |  |         return Event::next; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } |