36681dcdea
- Remove the X-Mailer header in e-mails, as this causes some email servers to see emails sent by Framadate as spam -----BEGIN PGP SIGNATURE----- iQIzBAABCAAdFiEExMITpfxOHHCvHn8FoGG53eDKB3MFAmHElvwACgkQoGG53eDK B3PO3w/+LyyC4Y7fDtl4hm9ClIswp62ozhxlIzRDJbza2S2S37A0ssXJMYCf2VOR ak+vWFUs9xj3o6G3oGlhLF1KMpWJ/nFVhyhCGgV1mQmUHSX2gnn6S9PBFenOGc9w zD2onJhVhJ/tbCJbF0Yl6zPitkfiucJI3FFIauByhkZ8wI3MPIYo/2m+H7KeVwLp YgJU0VqUQNMYl1clC+9Vu6gpSk/f6RklEVzJfEgV4MWMfoM6hLFqo0F6IDyWmgKp eNi1KHmFBGKFIexQeI+AvDyJyYsZH7uJrB0F7bV4wJGS7MTPOJlOJG5MXlLDcJgK EKNFUqrb9pQT5TWkO5FbJjfHwbP5bhO7TAEEvdIzhFqziR6gqdr2uXi+HiwG7yHO 21YzDMnRZPqiYTBJuP0XIhsJrgpIBQDuQ3u9XJTs4pfyxQDSR8m9sHdiPSEZySRN p7XJSsDyLhcjjAobdodLuZLSWqOvZfaHzXEEKZO3HdQmccbwRHvniXDEy0FHEt2A L7LVvd4Qpa4wi6T2b3UIK4ubuC4xYdtYBfpzOklcti5TwDT2jbSTJRbv5oGAPX1b HfMzMoi/os3ZDSnPPRUaBxxrbjFQ9bX4bdxfQZeC2XIBWMBOcF3SeSbC1vVlcu+3 bgwo3mehLvBCY0uMCGrjdymcAQq2IwX5Df8dGS7jFWUS5rAUi7s= =OVGk -----END PGP SIGNATURE----- Merge tag '1.1.19' into chapril-1.1.19
324 lines
10 KiB
PHP
324 lines
10 KiB
PHP
<?php
|
|
namespace Framadate\Services;
|
|
|
|
use Framadate\Exception\MomentAlreadyExistsException;
|
|
use Framadate\FramaDB;
|
|
use Framadate\Repositories\RepositoryFactory;
|
|
use Framadate\Utils;
|
|
|
|
/**
|
|
* Class AdminPollService
|
|
*
|
|
* @package Framadate\Services
|
|
*/
|
|
class AdminPollService {
|
|
private $connect;
|
|
private $pollService;
|
|
private $logService;
|
|
|
|
private $pollRepository;
|
|
private $slotRepository;
|
|
private $voteRepository;
|
|
private $commentRepository;
|
|
|
|
public function __construct(FramaDB $connect, PollService $pollService, LogService $logService) {
|
|
$this->connect = $connect;
|
|
$this->pollService = $pollService;
|
|
$this->logService = $logService;
|
|
$this->pollRepository = RepositoryFactory::pollRepository();
|
|
$this->slotRepository = RepositoryFactory::slotRepository();
|
|
$this->voteRepository = RepositoryFactory::voteRepository();
|
|
$this->commentRepository = RepositoryFactory::commentRepository();
|
|
}
|
|
|
|
public function updatePoll($poll): bool
|
|
{
|
|
global $config;
|
|
|
|
$end_date = strtotime($poll->end_date);
|
|
|
|
if ($end_date < strtotime($poll->creation_date)) {
|
|
$poll->end_date = $poll->creation_date;
|
|
} elseif ($end_date > $this->pollService->maxExpiryDate()) {
|
|
$poll->end_date = utf8_encode(strftime('%Y-%m-%d', $this->pollService->maxExpiryDate()));
|
|
}
|
|
|
|
return $this->pollRepository->update($poll);
|
|
}
|
|
|
|
/**
|
|
* Delete a comment from a poll.
|
|
*
|
|
* @param string $poll_id The ID of the poll
|
|
* @param $comment_id int The ID of the comment
|
|
* @return mixed true is action succeeded
|
|
*/
|
|
public function deleteComment(string $poll_id, int $comment_id) {
|
|
return $this->commentRepository->deleteById($poll_id, $comment_id);
|
|
}
|
|
|
|
/**
|
|
* Remove all comments of a poll.
|
|
*
|
|
* @param string $poll_id The ID a the poll
|
|
* @return bool|null true is action succeeded
|
|
*/
|
|
public function cleanComments(string $poll_id): ?bool
|
|
{
|
|
$this->logService->log("CLEAN_COMMENTS", "id:$poll_id");
|
|
return $this->commentRepository->deleteByPollId($poll_id);
|
|
}
|
|
|
|
/**
|
|
* Delete a vote from a poll.
|
|
*
|
|
* @param string $poll_id The ID of the poll
|
|
* @param $vote_id int The ID of the vote
|
|
* @return bool true is action succeeded
|
|
*/
|
|
public function deleteVote(string $poll_id, int $vote_id): bool
|
|
{
|
|
return $this->voteRepository->deleteById($poll_id, $vote_id);
|
|
}
|
|
|
|
/**
|
|
* Remove all votes of a poll.
|
|
*
|
|
* @param string $poll_id The ID of the poll
|
|
* @return bool|null true is action succeeded
|
|
*/
|
|
public function cleanVotes(string $poll_id): ?bool
|
|
{
|
|
$this->logService->log('CLEAN_VOTES', 'id:' . $poll_id);
|
|
return $this->voteRepository->deleteByPollId($poll_id);
|
|
}
|
|
|
|
/**
|
|
* Delete the entire given poll.
|
|
*
|
|
* @param $poll_id string The ID of the poll
|
|
* @return bool true is action succeeded
|
|
*/
|
|
public function deleteEntirePoll(string $poll_id): bool
|
|
{
|
|
$poll = $this->pollRepository->findById($poll_id);
|
|
$this->logService->log('DELETE_POLL', "id:$poll->id, format:$poll->format, admin:$poll->admin_name, mail:$poll->admin_mail");
|
|
|
|
// Delete the entire poll
|
|
$this->voteRepository->deleteByPollId($poll_id);
|
|
$this->commentRepository->deleteByPollId($poll_id);
|
|
$this->slotRepository->deleteByPollId($poll_id);
|
|
$this->pollRepository->deleteById($poll_id);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Delete a slot from a poll.
|
|
*
|
|
* @param object $poll The ID of the poll
|
|
* @param object $slot The slot informations (datetime + moment)
|
|
* @return bool true if action succeeded
|
|
*/
|
|
public function deleteDateSlot(object $poll, object $slot): bool
|
|
{
|
|
$this->logService->log('DELETE_SLOT', 'id:' . $poll->id . ', slot:' . json_encode($slot));
|
|
|
|
$datetime = $slot->title;
|
|
$moment = $slot->moment;
|
|
|
|
$slots = $this->pollService->allSlotsByPoll($poll);
|
|
|
|
// We can't delete the last slot
|
|
if ($poll->format === 'D' && count($slots) === 1 && strpos($slots[0]->moments, ',') === false) {
|
|
return false;
|
|
}
|
|
|
|
if ($poll->format === 'A' && count($slots) === 1) {
|
|
return false;
|
|
}
|
|
|
|
$index = 0;
|
|
$indexToDelete = -1;
|
|
$newMoments = [];
|
|
|
|
// Search the index of the slot to delete
|
|
foreach ($slots as $aSlot) {
|
|
$moments = explode(',', $aSlot->moments);
|
|
|
|
foreach ($moments as $rowMoment) {
|
|
if ($datetime === $aSlot->title) {
|
|
if ($moment === $rowMoment) {
|
|
$indexToDelete = $index;
|
|
} else {
|
|
$newMoments[] = $rowMoment;
|
|
}
|
|
}
|
|
$index++;
|
|
}
|
|
}
|
|
|
|
// Remove votes
|
|
$this->connect->beginTransaction();
|
|
$this->voteRepository->deleteByIndex($poll->id, $indexToDelete);
|
|
if (count($newMoments) > 0) {
|
|
$this->slotRepository->update($poll->id, $datetime, implode(',', $newMoments));
|
|
} else {
|
|
$this->slotRepository->deleteByDateTime($poll->id, $datetime);
|
|
}
|
|
$this->connect->commit();
|
|
|
|
return true;
|
|
}
|
|
|
|
public function deleteClassicSlot($poll, string $slot_title): bool
|
|
{
|
|
$this->logService->log('DELETE_SLOT', 'id:' . $poll->id . ', slot:' . $slot_title);
|
|
|
|
$slots = $this->pollService->allSlotsByPoll($poll);
|
|
|
|
if (count($slots) === 1) {
|
|
return false;
|
|
}
|
|
|
|
$index = 0;
|
|
$indexToDelete = -1;
|
|
|
|
// Search the index of the slot to delete
|
|
foreach ($slots as $aSlot) {
|
|
if ($slot_title === $aSlot->title) {
|
|
$indexToDelete = $index;
|
|
}
|
|
$index++;
|
|
}
|
|
|
|
// Remove votes
|
|
$this->connect->beginTransaction();
|
|
$this->voteRepository->deleteByIndex($poll->id, $indexToDelete);
|
|
$this->slotRepository->deleteByDateTime($poll->id, $slot_title);
|
|
$this->connect->commit();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Add a new slot to a date poll. And insert default values for user's votes.
|
|
* <ul>
|
|
* <li>Create a new slot if no one exists for the given date</li>
|
|
* <li>Create a new moment if a slot already exists for the given date</li>
|
|
* </ul>
|
|
*
|
|
* @param string $poll_id The ID of the poll
|
|
* @param $datetime int The datetime
|
|
* @param $new_moment string The moment's name
|
|
* @throws MomentAlreadyExistsException When the moment to add already exists in database
|
|
*/
|
|
public function addDateSlot(string $poll_id, int $datetime, string $new_moment): void
|
|
{
|
|
$this->logService->log('ADD_COLUMN', 'id:' . $poll_id . ', datetime:' . $datetime . ', moment:' . $new_moment);
|
|
|
|
$slots = $this->slotRepository->listByPollId($poll_id);
|
|
$result = $this->findInsertPosition($slots, $datetime);
|
|
|
|
// Begin transaction
|
|
$this->connect->beginTransaction();
|
|
|
|
if ($result->slot !== null) {
|
|
$slot = $result->slot;
|
|
$moments = explode(',', $slot->moments);
|
|
|
|
// Check if moment already exists (maybe not necessary)
|
|
if (in_array($new_moment, $moments, true)) {
|
|
throw new MomentAlreadyExistsException();
|
|
}
|
|
|
|
// Update found slot
|
|
$moments[] = $new_moment;
|
|
$this->slotRepository->update($poll_id, $datetime, implode(',', $moments));
|
|
} else {
|
|
$this->slotRepository->insert($poll_id, $datetime, $new_moment);
|
|
}
|
|
|
|
$this->voteRepository->insertDefault($poll_id, $result->insert);
|
|
|
|
// Commit transaction
|
|
$this->connect->commit();
|
|
}
|
|
|
|
/**
|
|
* Add a new slot to a classic poll. And insert default values for user's votes.
|
|
* <ul>
|
|
* <li>Create a new slot if no one exists for the given title</li>
|
|
* </ul>
|
|
*
|
|
* @param string $poll_id The ID of the poll
|
|
* @param string $title The title
|
|
* @throws MomentAlreadyExistsException When the moment to add already exists in database
|
|
*/
|
|
public function addClassicSlot(string $poll_id, string $title): void
|
|
{
|
|
$this->logService->log('ADD_COLUMN', 'id:' . $poll_id . ', title:' . $title);
|
|
|
|
$slots = $this->slotRepository->listByPollId($poll_id);
|
|
|
|
// Check if slot already exists
|
|
$titles = array_map(static function ($slot) {
|
|
return $slot->title;
|
|
}, $slots);
|
|
if (in_array($title, $titles, true)) {
|
|
// The moment already exists
|
|
throw new MomentAlreadyExistsException();
|
|
}
|
|
|
|
// Begin transaction
|
|
$this->connect->beginTransaction();
|
|
|
|
// New slot
|
|
$this->slotRepository->insert($poll_id, $title, null);
|
|
// Set default votes
|
|
$this->voteRepository->insertDefault($poll_id, count($slots));
|
|
|
|
// Commit transaction
|
|
$this->connect->commit();
|
|
}
|
|
|
|
/**
|
|
* This method find where to insert a datatime+moment into a list of slots.<br/>
|
|
* Return the {insert:X}, where X is the index of the moment into the whole poll (ex: X=0 => Insert to the first column).
|
|
* Return {slot:Y}, where Y is not null if there is a slot existing for the given datetime.
|
|
*
|
|
* @param $slots array All the slots of the poll
|
|
* @param $datetime int The datetime of the new slot
|
|
* @return \stdClass An object like this one: {insert:X, slot:Y} where Y can be null.
|
|
*/
|
|
private function findInsertPosition(array $slots, int $datetime) {
|
|
$result = new \stdClass();
|
|
$result->slot = null;
|
|
$result->insert = 0;
|
|
|
|
// Sort slots before searching where to insert
|
|
$this->pollService->sortSlorts($slots);
|
|
|
|
// Search where to insert new column
|
|
foreach ($slots as $k=>$slot) {
|
|
$rowDatetime = (int) $slot->title;
|
|
$moments = explode(',', $slot->moments);
|
|
|
|
if ($datetime === $rowDatetime) {
|
|
// Here we have to insert at the end of a slot
|
|
$result->insert += count($moments);
|
|
$result->slot = $slot;
|
|
break;
|
|
}
|
|
|
|
if ($datetime < $rowDatetime) {
|
|
// We have to insert before this slot
|
|
break;
|
|
}
|
|
$result->insert += count($moments);
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
}
|