| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // {{{ 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; | 
					
						
							|  |  |  | use App\Core\DB\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 | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |     public function version(): string | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         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
										 |  |  |      * | 
					
						
							|  |  |  |      * @param int $filesize | 
					
						
							|  |  |  |      * @param int $user_id | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws ClientException | 
					
						
							|  |  |  |      * @throws ServerException | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     { | 
					
						
							|  |  |  |         $query = <<<END | 
					
						
							|  |  |  | select sum(at.size) as total | 
					
						
							|  |  |  |     from attachment at | 
					
						
							| 
									
										
										
										
											2021-08-19 19:18:10 +01:00
										 |  |  |         join gsactor_to_attachment ua with at.id = ua.attachment_id | 
					
						
							|  |  |  |     where ua.gsactor_id = :actor_id and at.size is not null | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  | END; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 13:40:50 +01:00
										 |  |  |         $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. ', | 
					
						
							|  |  |  |                                          ['quota' => $max_file_size, 'size' => $filesize])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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[] = [ | 
					
						
							|  |  |  |             'name'        => 'FileQuota', | 
					
						
							|  |  |  |             'version'     => $this->version(), | 
					
						
							|  |  |  |             'author'      => 'Hugo Sales', | 
					
						
							|  |  |  |             'homepage'    => GNUSOCIAL_PROJECT_URL, | 
					
						
							|  |  |  |             'description' => // TRANS: Plugin description.
 | 
					
						
							|  |  |  |                 _m('Plugin to manage user quotas.'), | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  |         return Event::next; | 
					
						
							| 
									
										
										
										
											2021-04-30 13:14:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } |