| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889 |
- <?php
- namespace MailPoet\Subscription;
- if (!defined('ABSPATH')) exit;
- use MailPoet\Entities\SubscriberIPEntity;
- use MailPoet\Subscribers\SubscriberIPsRepository;
- use MailPoet\Util\Helpers;
- use MailPoet\WP\Functions as WPFunctions;
- class Throttling {
- /** @var SubscriberIPsRepository */
- private $subscriberIPsRepository;
- /** @var WPFunctions */
- private $wp;
- public function __construct(
- SubscriberIPsRepository $subscriberIPsRepository,
- WPFunctions $wp
- ) {
- $this->wp = $wp;
- $this->subscriberIPsRepository = $subscriberIPsRepository;
- }
- public function throttle() {
- $subscriptionLimitEnabled = $this->wp->applyFilters('mailpoet_subscription_limit_enabled', true);
- $subscriptionLimitWindow = (int)$this->wp->applyFilters('mailpoet_subscription_limit_window', DAY_IN_SECONDS);
- $subscriptionLimitBase = (int)$this->wp->applyFilters('mailpoet_subscription_limit_base', MINUTE_IN_SECONDS);
- $subscriberIp = Helpers::getIP();
- if ($subscriptionLimitEnabled && !$this->isUserExemptFromThrottling()) {
- if (!empty($subscriberIp)) {
- $subscriptionCount = $this->subscriberIPsRepository->getCountByIPAndCreatedAtAfterTimeInSeconds($subscriberIp, $subscriptionLimitWindow);
- if ($subscriptionCount > 0) {
- $timeout = $subscriptionLimitBase * pow(2, $subscriptionCount - 1);
- $existingUser = $this->subscriberIPsRepository->findOneByIPAndCreatedAtAfterTimeInSeconds($subscriberIp, $timeout);
- if (!empty($existingUser)) {
- return $timeout;
- }
- }
- }
- }
- if ($subscriberIp !== null) {
- $ip = new SubscriberIPEntity($subscriberIp);
- $existingIp = $this->subscriberIPsRepository->findOneBy(['ip' => $ip->getIP(), 'createdAt' => $ip->getCreatedAt()]);
- if (!$existingIp) {
- $this->subscriberIPsRepository->persist($ip);
- $this->subscriberIPsRepository->flush();
- }
- }
- $this->purge();
- return false;
- }
- public function purge(): void {
- $interval = $this->wp->applyFilters('mailpoet_subscription_purge_window', MONTH_IN_SECONDS);
- $this->subscriberIPsRepository->deleteCreatedAtBeforeTimeInSeconds($interval);
- }
- public function secondsToTimeString($seconds): string {
- $hrs = floor($seconds / 3600);
- $min = floor($seconds % 3600 / 60);
- $sec = $seconds % 3600 % 60;
- $result = [
- 'hours' => $hrs ? sprintf(__('%d hours', 'mailpoet'), $hrs) : '',
- 'minutes' => $min ? sprintf(__('%d minutes', 'mailpoet'), $min) : '',
- 'seconds' => $sec ? sprintf(__('%d seconds', 'mailpoet'), $sec) : '',
- ];
- return join(' ', array_filter($result));
- }
- private function isUserExemptFromThrottling(): bool {
- if (!$this->wp->isUserLoggedIn()) {
- return false;
- }
- $user = $this->wp->wpGetCurrentUser();
- $roles = $this->wp->applyFilters('mailpoet_subscription_throttling_exclude_roles', ['administrator', 'editor']);
- return !empty(array_intersect($roles, (array)$user->roles));
- }
- }
|