. /** * Plugin to throttle subscriptions by a user * * @category Throttle * @package GNUsocial * @author Evan Prodromou * @copyright 2010 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ defined('GNUSOCIAL') || die(); /** * Subscription throttle * * @category Throttle * @package GNUsocial * @author Evan Prodromou * @copyright 2010 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class SubscriptionThrottlePlugin extends Plugin { const PLUGIN_VERSION = '2.0.0'; public $subLimits = [86400 => 100, 3600 => 50]; public $groupLimits = [86400 => 50, 3600 => 25]; /** * Filter subscriptions to see if they're coming too fast. * * @param Profile $profile The profile subscribing * @param Profile $other The profile being subscribed to * * @return boolean hook value */ public function onStartSubscribe(Profile $profile, $other) { foreach ($this->subLimits as $seconds => $limit) { $sub = $this->_getNthSub($profile, $limit); if (!empty($sub)) { $subtime = strtotime($sub->created); $now = time(); if ($now - $subtime < $seconds) { // TRANS: Exception thrown when subscribing too quickly. throw new Exception(_m('Too many subscriptions. Take a break and try again later.')); } } } return true; } /** * Filter group joins to see if they're coming too fast. * * @param Group $group The group being joined * @param Profile $profile The profile joining * * @return boolean hook value */ public function onStartJoinGroup($group, $profile) { foreach ($this->groupLimits as $seconds => $limit) { $mem = $this->_getNthMem($profile, $limit); if (!empty($mem)) { $jointime = strtotime($mem->created); $now = time(); if ($now - $jointime < $seconds) { // TRANS: Exception thrown when joing groups too quickly. throw new Exception(_m('Too many memberships. Take a break and try again later.')); } } } return true; } /** * Get the Nth most recent subscription for this profile * * @param Profile $profile profile to get subscriptions for * @param integer $n How far to count back * * @return Subscription a subscription or null */ private function _getNthSub(Profile $profile, $n) { $sub = new Subscription(); $sub->subscriber = $profile->id; $sub->orderBy('created DESC'); $sub->limit($n - 1, 1); if ($sub->find(true)) { return $sub; } else { return null; } } /** * Get the Nth most recent group membership for this profile * * @param Profile $profile The user to get memberships for * @param integer $n How far to count back * * @return Group_member a membership or null */ private function _getNthMem(Profile $profile, $n) { $mem = new Group_member(); $mem->profile_id = $profile->id; $mem->orderBy('created DESC, group_id DESC'); $mem->limit($n - 1, 1); if ($mem->find(true)) { return $mem; } else { return null; } } /** * Return plugin version data for display * * @param array &$versions Array of version arrays * * @return boolean hook value */ public function onPluginVersion(array &$versions): bool { $versions[] = array('name' => 'SubscriptionThrottle', 'version' => self::PLUGIN_VERSION, 'author' => 'Evan Prodromou', 'homepage' => GNUSOCIAL_ENGINE_REPO_URL . 'tree/master/plugins/SubscriptionThrottle', 'rawdescription' => // TRANS: Plugin description. _m('Configurable limits for subscriptions and group memberships.')); return true; } }