connect = $connect; $this->logService = $logService; $this->pollRepository = RepositoryFactory::pollRepository(); $this->slotRepository = RepositoryFactory::slotRepository(); $this->voteRepository = RepositoryFactory::voteRepository(); $this->commentRepository = RepositoryFactory::commentRepository(); } /** * Find a poll from its ID. * * @param $poll_id int The ID of the poll * @return \stdClass|null The found poll, or null */ function findById($poll_id) { if (preg_match(POLL_REGEX, $poll_id)) { return $this->pollRepository->findById($poll_id); } return null; } public function findByAdminId($admin_poll_id) { if (preg_match(ADMIN_POLL_REGEX, $admin_poll_id)) { return $this->pollRepository->findByAdminId($admin_poll_id); } return null; } function allCommentsByPollId($poll_id) { return $this->commentRepository->findAllByPollId($poll_id); } function allVotesByPollId($poll_id) { return $this->voteRepository->allUserVotesByPollId($poll_id); } function allSlotsByPoll($poll) { $slots = $this->slotRepository->listByPollId($poll->id); if ($poll->format === 'D') { $this->sortSlorts($slots); } return $slots; } /** * @param $poll_id * @param $vote_id * @param $name * @param $choices * @param $slots_hash * @throws ConcurrentEditionException * @throws ConcurrentVoteException * @return bool */ public function updateVote($poll_id, $vote_id, $name, $choices, $slots_hash) { $poll = $this->findById($poll_id); // Check that no-one voted in the meantime and it conflicts the maximum votes constraint $this->checkMaxVotes($choices, $poll, $poll_id); // Check if slots are still the same $this->checkThatSlotsDidntChanged($poll, $slots_hash); // Update vote $choices = implode($choices); return $this->voteRepository->update($poll_id, $vote_id, $name, $choices); } /** * @param $poll_id * @param $name * @param $choices * @param $slots_hash * @throws AlreadyExistsException * @throws ConcurrentEditionException * @throws ConcurrentVoteException * @return \stdClass */ function addVote($poll_id, $name, $choices, $slots_hash) { $poll = $this->findById($poll_id); // Check that no-one voted in the meantime and it conflicts the maximum votes constraint $this->checkMaxVotes($choices, $poll, $poll_id); // Check if slots are still the same $this->checkThatSlotsDidntChanged($poll, $slots_hash); // Check if vote already exists if ($this->voteRepository->existsByPollIdAndName($poll_id, $name)) { throw new AlreadyExistsException(); } // Insert new vote $choices = implode($choices); $token = $this->random(16); return $this->voteRepository->insert($poll_id, $name, $choices, $token); } function addComment($poll_id, $name, $comment) { if ($this->commentRepository->exists($poll_id, $name, $comment)) { return true; } return $this->commentRepository->insert($poll_id, $name, $comment); } /** * @param Form $form * @return array */ function createPoll(Form $form) { // Generate poll IDs, loop while poll ID already exists if (empty($form->id)) { // User want us to generate an id for him do { $poll_id = $this->random(16); } while ($this->pollRepository->existsById($poll_id)); $admin_poll_id = $poll_id . $this->random(8); } else { // User have choosen the poll id $poll_id = $form->id; do { $admin_poll_id = $this->random(24); } while ($this->pollRepository->existsByAdminId($admin_poll_id)); } // Insert poll + slots $this->pollRepository->beginTransaction(); $this->pollRepository->insertPoll($poll_id, $admin_poll_id, $form); $this->slotRepository->insertSlots($poll_id, $form->getChoices()); $this->pollRepository->commit(); $this->logService->log('CREATE_POLL', 'id:' . $poll_id . ', title: ' . $form->title . ', format:' . $form->format . ', admin:' . $form->admin_name . ', mail:' . $form->admin_mail); return [$poll_id, $admin_poll_id]; } public function findAllByAdminMail($mail) { return $this->pollRepository->findAllByAdminMail($mail); } /** * @param \stdClass $poll * @return array */ private function computeEmptyBestChoices($poll) { $result = ['y' => [], 'inb' => []]; // if there is no votes, calculates the number of slot $slots = $this->allSlotsByPoll($poll); if ($poll->format === 'A') { // poll format classic for ($i = 0; $i < count($slots); $i++) { $result['y'][] = 0; $result['inb'][] = 0; } } else { // poll format date $slots = $this->splitSlots($slots); foreach ($slots as $slot) { for ($i = 0; $i < count($slot->moments); $i++) { $result['y'][] = 0; $result['inb'][] = 0; } } } return $result; } /** * @param array $votes * @param \stdClass $poll * @return array */ public function computeBestChoices($votes, $poll) { if (0 === count($votes)) { return $this->computeEmptyBestChoices($poll); } $result = ['y' => [], 'inb' => []]; // if there are votes foreach ($votes as $vote) { $choices = str_split($vote->choices); foreach ($choices as $i => $choice) { if (!isset($result['y'][$i])) { $result['inb'][$i] = 0; $result['y'][$i] = 0; } if ($choice === "1") { $result['inb'][$i]++; } if ($choice === "2") { $result['y'][$i]++; } } } return $result; } function splitSlots($slots) { $splitted = []; foreach ($slots as $slot) { $obj = new \stdClass(); $obj->day = $slot->title; $obj->moments = explode(',', $slot->moments); $splitted[] = $obj; } return $splitted; } /** * @param $slots array The slots to hash * @return string The hash */ public function hashSlots($slots) { return md5(array_reduce($slots, function($carry, $item) { return $carry . $item->id . '@' . $item->moments . ';'; })); } function splitVotes($votes) { $splitted = []; foreach ($votes as $vote) { $obj = new \stdClass(); $obj->id = $vote->id; $obj->name = $vote->name; $obj->uniqId = $vote->uniqId; $obj->choices = str_split($vote->choices); $splitted[] = $obj; } return $splitted; } /** * @return int The max timestamp allowed for expiry date */ public function maxExpiryDate() { global $config; return time() + (86400 * $config['default_poll_duration']); } /** * @return int The min timestamp allowed for expiry date */ public function minExpiryDate() { return time() + 86400; } /** * @return mixed */ public function sortSlorts(&$slots) { uasort($slots, function ($a, $b) { return $a->title > $b->title; }); return $slots; } private function random($length) { return Token::getToken($length); } /** * This method checks if the hash send by the user is the same as the computed hash. * * @param $poll /stdClass The poll * @param $slots_hash string The hash sent by the user * @throws ConcurrentEditionException Thrown when hashes are differents */ private function checkThatSlotsDidntChanged($poll, $slots_hash) { $slots = $this->allSlotsByPoll($poll); if ($slots_hash !== $this->hashSlots($slots)) { throw new ConcurrentEditionException(); } } /** * This method checks if the votes doesn't conflicts the maximum votes constraint * * @param $user_choice * @param \stdClass $poll * @param string $poll_id * @throws ConcurrentVoteException */ private function checkMaxVotes($user_choice, $poll, $poll_id) { $votes = $this->allVotesByPollId($poll_id); if (count($votes) <= 0) { return; } $best_choices = $this->computeBestChoices($votes, $poll); foreach ($best_choices['y'] as $i => $nb_choice) { // if for this option we have reached maximum value and user wants to add itself too if ($poll->ValueMax !== null && $nb_choice >= $poll->ValueMax && $user_choice[$i] === "2") { throw new ConcurrentVoteException(); } } } }