X-Git-Url: https://git.rapsys.eu/airbundle/blobdiff_plain/f0a1de5ce9ec93c81eaf4f2593fefda43a1dfef8..d0d8de5a7844aacda1fa790267fd648fe59b5b2d:/Controller/ApplicationController.php?ds=sidebyside diff --git a/Controller/ApplicationController.php b/Controller/ApplicationController.php index cf7e7ce..8a9a8b4 100644 --- a/Controller/ApplicationController.php +++ b/Controller/ApplicationController.php @@ -3,6 +3,10 @@ namespace Rapsys\AirBundle\Controller; use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\Routing\RequestContext; +use Symfony\Component\Form\FormError; +use Symfony\Component\Routing\Exception\MethodNotAllowedException; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; use Rapsys\AirBundle\Entity\Slot; use Rapsys\AirBundle\Entity\User; use Rapsys\AirBundle\Entity\Session; @@ -24,6 +28,11 @@ class ApplicationController extends DefaultController { //Prevent non-guest to access here $this->denyAccessUnlessGranted('ROLE_GUEST', null, $this->translator->trans('Unable to access this page without role %role%!', ['%role%' => $this->translator->trans('Guest')])); + //Reject non post requests + if (!$request->isMethod('POST')) { + throw new \RuntimeException('Request method MUST be POST'); + } + //Create ApplicationType form $form = $this->createForm('Rapsys\AirBundle\Form\ApplicationType', null, [ //Set the action @@ -39,24 +48,19 @@ class ApplicationController extends DefaultController { 'slot' => $this->getDoctrine()->getRepository(Slot::class)->findOneById(3) ]); - //Reject non post requests - if (!$request->isMethod('POST')) { - throw new \RuntimeException('Request method MUST be POST'); - } - //Refill the fields in case of invalid form $form->handleRequest($request); //Handle invalid form if (!$form->isValid()) { //Set section - $section = $this->translator->trans('Application Add'); + $section = $this->translator->trans('Application add'); //Set title $title = $section.' - '.$this->translator->trans($this->config['site']['title']); //Render the view - return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'section' => $section, 'form' => $form]+$this->context); + return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'section' => $section, 'form' => $form->createView()]+$this->context); } //Get doctrine @@ -68,6 +72,15 @@ class ApplicationController extends DefaultController { //Get data $data = $form->getData(); + //Count session at location in last month for guest + if (!$this->isGranted('ROLE_REGULAR') && !empty($session = $doctrine->getRepository(Session::class)->findOneWithinLastMonthByLocationUser($data['location']->getId(), $this->getUser()->getId()))) { + //Add warning in flash message + $this->addFlash('warning', $this->translator->trans('Monthly application %location% already exists', ['%location%' => $this->translator->trans('at '.$data['location'])])); + + //Redirect to cleanup the form + return $this->redirectToRoute('rapsys_air_session_view', ['id' => $session['id']]); + } + //Protect session fetching try { //Fetch session @@ -82,6 +95,231 @@ class ApplicationController extends DefaultController { $session->setCreated(new \DateTime('now')); $session->setUpdated(new \DateTime('now')); + //Get short location + $short = $data['location']->getShort(); + + //Get slot + $slot = $data['slot']->getTitle(); + + //Set premium + $session->setPremium($premium = false); + + //Check if slot is afternoon + //XXX: premium is stored only for Afternoon and Evening + if ($slot == 'Afternoon') { + //Compute premium + //XXX: a session is considered premium a day off + $session->setPremium($premium = $this->isPremium($data['date'])); + //Check if slot is evening + //XXX: premium is stored only for Afternoon and Evening + } elseif ($slot == 'Evening') { + //Compute premium + //XXX: a session is considered premium the eve of a day off + $session->setPremium($premium = $this->isPremium((clone $data['date'])->add(new \DateInterval('P1D')))); + //Check if slot is after + } elseif ($slot == 'After') { + //Compute premium + //XXX: a session is considered premium the eve of a day off + $premium = $this->isPremium((clone $data['date'])->add(new \DateInterval('P1D'))); + } + + //Set default length at 6h + //XXX: date part will be truncated on save + $session->setLength(new \DateTime('06:00:00')); + + //Check if admin + if ($this->isGranted('ROLE_ADMIN')) { + //Check if morning + if ($slot == 'Morning') { + //Set begin at 9h + $session->setBegin(new \DateTime('09:00:00')); + + //Set length at 5h + $session->setLength(new \DateTime('05:00:00')); + //Check if afternoon + } elseif ($slot == 'Afternoon') { + //Set begin at 14h + $session->setBegin(new \DateTime('14:00:00')); + + //Set length at 5h + $session->setLength(new \DateTime('05:00:00')); + //Check if evening + } elseif ($slot == 'Evening') { + //Set begin at 19h + $session->setBegin(new \DateTime('19:00:00')); + + //Check if next day is premium + if ($premium) { + //Set length at 7h + $session->setLength(new \DateTime('07:00:00')); + } + //Check if after + } else { + //Set begin at 1h + $session->setBegin(new \DateTime('01:00:00')); + + //Set length at 4h + $session->setLength(new \DateTime('04:00:00')); + + //Check if next day is premium + if ($premium) { + //Set begin at 2h + $session->setBegin(new \DateTime('02:00:00')); + + //Set length at 3h + $session->setLength(new \DateTime('03:00:00')); + } + } + //Docks => 14h -> 19h | 19h -> 01/02h + //XXX: remove Garnier from here to switch back to 21h + } elseif (in_array($short, ['Docks', 'Garnier']) && in_array($slot, ['Afternoon', 'Evening', 'After'])) { + //Check if afternoon + if ($slot == 'Afternoon') { + //Set begin at 14h + $session->setBegin(new \DateTime('14:00:00')); + + //Set length at 5h + $session->setLength(new \DateTime('05:00:00')); + //Check if evening + } elseif ($slot == 'Evening') { + //Set begin at 19h + $session->setBegin(new \DateTime('19:00:00')); + + //Check if next day is premium + if ($premium) { + //Set length at 7h + $session->setLength(new \DateTime('07:00:00')); + } + //Check if after + } else { + //Set begin at 1h + $session->setBegin(new \DateTime('01:00:00')); + + //Set length at 4h + $session->setLength(new \DateTime('04:00:00')); + + //Check if next day is premium + if ($premium) { + //Set begin at 2h + $session->setBegin(new \DateTime('02:00:00')); + + //Set length at 3h + $session->setLength(new \DateTime('03:00:00')); + } + } + //Garnier => 21h -> 01/02h + } elseif ($short == 'Garnier' && in_array($slot, ['Evening', 'After'])) { + //Check if evening + if ($slot == 'Evening') { + //Set begin at 21h + $session->setBegin(new \DateTime('21:00:00')); + + //Set length at 5h + $session->setLength(new \DateTime('05:00:00')); + + //Check if next day is premium + if ($premium) { + //Set length at 6h + $session->setLength(new \DateTime('06:00:00')); + } + //Check if after + } else { + //Set begin at 1h + $session->setBegin(new \DateTime('01:00:00')); + + //Set length at 4h + $session->setLength(new \DateTime('04:00:00')); + + //Check if next day is premium + if ($premium) { + //Set begin at 2h + $session->setBegin(new \DateTime('02:00:00')); + + //Set length at 3h + $session->setLength(new \DateTime('03:00:00')); + } + } + //Trocadero|Tokyo|Swan|Honore|Orsay => 19h -> 01/02h + } elseif (in_array($short, ['Trocadero', 'Tokyo', 'Swan', 'Honore', 'Orsay']) && in_array($slot, ['Evening', 'After'])) { + //Check if evening + if ($slot == 'Evening') { + //Set begin at 19h + $session->setBegin(new \DateTime('19:00:00')); + + //Check if next day is premium + if ($premium) { + //Set length at 7h + $session->setLength(new \DateTime('07:00:00')); + } + //Check if after + } else { + //Set begin at 1h + $session->setBegin(new \DateTime('01:00:00')); + + //Set length at 4h + $session->setLength(new \DateTime('04:00:00')); + + //Check if next day is premium + if ($premium) { + //Set begin at 2h + $session->setBegin(new \DateTime('02:00:00')); + + //Set length at 3h + $session->setLength(new \DateTime('03:00:00')); + } + } + //La Villette => 14h -> 19h + } elseif ($short == 'Villette' && $slot == 'Afternoon') { + //Set begin at 14h + $session->setBegin(new \DateTime('14:00:00')); + + //Set length at 5h + $session->setLength(new \DateTime('05:00:00')); + //Place Colette => 14h -> 21h + //TODO: add check here that it's a millegaux account ? + } elseif ($short == 'Colette' && $slot == 'Afternoon') { + //Set begin at 14h + $session->setBegin(new \DateTime('14:00:00')); + + //Set length at 7h + $session->setLength(new \DateTime('07:00:00')); + //Galerie d'Orléans => 14h -> 18h + } elseif ($short == 'Orleans' && $slot == 'Afternoon') { + //Set begin at 14h + $session->setBegin(new \DateTime('14:00:00')); + + //Set length at 4h + $session->setLength(new \DateTime('04:00:00')); + //Combination not supported + } else { + //Add error in flash message + $this->addFlash('error', $this->translator->trans('Session on %date% %location% %slot% not yet supported', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')])); + + //Set section + $section = $this->translator->trans('Application add'); + + //Set title + $title = $section.' - '.$this->translator->trans($this->config['site']['title']); + + //Render the view + return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'section' => $section, 'form' => $form->createView()]+$this->context); + } + + //Check if admin + if (!$this->isGranted('ROLE_ADMIN') && $session->getStart() < new \DateTime('00:00:00')) { + //Add error in flash message + $this->addFlash('error', $this->translator->trans('Session in the past on %date% %location% %slot% not yet supported', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')])); + + //Set section + $section = $this->translator->trans('Application add'); + + //Set title + $title = $section.' - '.$this->translator->trans($this->config['site']['title']); + + //Render the view + return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'section' => $section, 'form' => $form->createView()]+$this->context); + } + //Queue session save $manager->persist($session); @@ -104,14 +342,8 @@ class ApplicationController extends DefaultController { //Retrieve application $application = $doctrine->getRepository(Application::class)->findOneBySessionUser($session, $user); - //Add notice in flash message - //TODO: set warning about application already exists bla bla bla... - #$this->addFlash('notice', $this->translator->trans('Application request the %date% for %location% on the slot %slot% saved', ['%location%' => $data['location']->getTitle(), '%slot%' => $data['slot']->getTitle(), '%date%' => $data['date']->format('Y-m-d')])); - - //Add error message to mail field - #$form->get('slot')->addError(new FormError($this->translator->trans('Application already exists'))); - - //TODO: redirect anyway on uri with application highlighted + //Add warning in flash message + $this->addFlash('warning', $this->translator->trans('Application on %date% %location% %slot% already exists', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')])); //Catch no application and session without identifier (not persisted&flushed) cases } catch (\Doctrine\ORM\NoResultException|\Doctrine\ORM\ORMInvalidArgumentException $e) { //Create the application @@ -131,23 +363,11 @@ class ApplicationController extends DefaultController { $manager->persist($application); //Flush to get the ids - #$manager->flush(); + $manager->flush(); //Add notice in flash message $this->addFlash('notice', $this->translator->trans('Application on %date% %location% %slot% created', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')])); } - - //Try unshort return field - if ( - !empty($data['return']) && - ($unshort = $this->slugger->unshort($data['return'])) && - ($route = json_decode($unshort, true)) !== null - ) { - $return = $this->generateUrl($route['_route'], ['session' => $session->getId()?:1]+$route['_route_params']); - } - - //XXX: Debug - header('Content-Type: text/plain'); //Extract and process referer if ($referer = $request->headers->get('referer')) { @@ -165,174 +385,142 @@ class ApplicationController extends DefaultController { //Try with referer path try { - var_dump($this->router); - exit; - var_dump($path = '/location'); - var_dump($this->router->match($path)); - var_dump($path = '/fr/emplacement'); - exit; - var_dump($this->router->match()); - exit; - var_dump($path); - var_dump($query); - exit; + //Save old context + $oldContext = $this->router->getContext(); + + //Force clean context + //XXX: prevent MethodNotAllowedException because current context method is POST in onevendor/symfony/routing/Matcher/Dumper/CompiledUrlMatcherTrait.php+42 + $this->router->setContext(new RequestContext()); + //Retrieve route matching path $route = $this->router->match($path); - var_dump($route); - exit; - //Verify that it differ from current one - if (($name = $route['_route']) == $logout) { - throw new ResourceNotFoundException('Identical referer and logout route'); - } + //Reset context + $this->router->setContext($oldContext); + + //Clear old context + unset($oldContext); + + //Extract name + $name = $route['_route']; //Remove route and controller from route defaults unset($route['_route'], $route['_controller']); //Generate url - $url = $this->router->generate($name, $route); + return $this->redirectToRoute($name, ['session' => $session->getId()]+$route); //No route matched - } catch(ResourceNotFoundException $e) { + } catch(MethodNotAllowedException|ResourceNotFoundException $e) { //Unset referer to fallback to default route unset($referer); } } - var_dump($request->headers->get('referer')); - #var_dump($request->get('_route')); - - var_dump($return); - exit; - //Fetch slugger helper - $slugger = $this->get('rapsys.slugger'); - - var_dump($short = $slugger->short(json_encode(['_route' => $request->get('_route'), '_route_params' => $request->get('_route_params')]))); - $short[12] = 'T'; - var_dump($ret = json_decode($slugger->unshort($short), true)); - var_dump($ret); - var_dump($this->generateUrl($ret['_route'], $ret['_route_params'])); - #var_dump(json_decode($slugger->unshort($data['return']))); - #var_dump($application->getId()); - exit; + //Redirect to cleanup the form + return $this->redirectToRoute('rapsys_air', ['session' => $session->getId()]); + } - //Init application - $application = false; + /** + * Compute eastern for selected year + * + * @param int $year The eastern year + * + * @return DateTime The eastern date + */ + function getEastern($year) { + //Set static + static $data = null; + //Check if already computed + if (isset($data[$year])) { + //Return computed eastern + return $data[$year]; + //Check if data is null + } elseif (is_null($data)) { + //Init data array + $data = []; + } + $d = (19 * ($year % 19) + 24) % 30; + $e = (2 * ($year % 4) + 4 * ($year % 7) + 6 * $d + 5) % 7; + + $day = 22 + $d + $e; + $month = 3; + + if ($day > 31) { + $day = $d + $e - 9; + $month = 4; + } elseif ($d == 29 && $e == 6) { + $day = 10; + $month = 4; + } elseif ($d == 28 && $e == 6) { + $day = 18; + $month = 4; + } - //Protect application fetching - try { - //TODO: handle admin case where we provide a user in extra - $application = $doctrine->getRepository(Application::class)->findOneBySessionUser($session, $this->getUser()); + //Store eastern in data + return ($data[$year] = new \DateTime(sprintf('%04d-%02d-%02d', $year, $month, $day))); + } - //Add error message to mail field - $form->get('slot')->addError(new FormError($this->translator->trans('Application already exists'))); - //Catch no application cases - //XXX: combine these catch when php 7.1 is available - } catch (\Doctrine\ORM\NoResultException $e) { - //Catch invalid argument because session is not already persisted - } catch(\Doctrine\ORM\ORMInvalidArgumentException $e) { + /** + * Check if date is a premium day + * + * @desc Consider as premium a day off + * + * @param DateTime $date The date to check + * @return bool Whether the date is off or not + */ + function isPremium($date) { + //Get day number + $w = $date->format('w'); + + //Check if weekend day + if ($w == 0 || $w == 6) { + //Date is weekend day + return true; } - //Create new application if none found - if (!$application) { - //Create the application - $application = new Application(); - $application->setSession($session); - //TODO: handle admin case where we provide a user in extra - $application->setUser($this->getUser()); - $application->setCreated(new \DateTime('now')); - $application->setUpdated(new \DateTime('now')); - $manager->persist($application); + //Get date day + $d = $date->format('d'); - //Flush to get the ids - $manager->flush(); - - //Add notice in flash message - $this->addFlash('notice', $this->translator->trans('Application request the %date% for %location% on the slot %slot% saved', ['%location%' => $data['location']->getTitle(), '%slot%' => $data['slot']->getTitle(), '%date%' => $data['date']->format('Y-m-d')])); + //Get date month + $m = $date->format('m'); - //Redirect to cleanup the form - return $this->redirectToRoute('rapsys_air_admin'); + //Check if fixed holiday + if ( + //Check if 1st january + ($d == 1 && $m == 1) || + //Check if 1st may + ($d == 1 && $m == 5) || + //Check if 8st may + ($d == 8 && $m == 5) || + //Check if 14st july + ($d == 14 && $m == 7) || + //Check if 15st august + ($d == 15 && $m == 8) || + //Check if 1st november + ($d == 1 && $m == 11) || + //Check if 11st november + ($d == 11 && $m == 11) || + //Check if 25st december + ($d == 25 && $m == 12) + ) { + //Date is a fixed holiday + return true; } - } - - function test(Request $request) { - - //Compute period - $period = new \DatePeriod( - //Start from first monday of week - new \DateTime('Monday this week'), - //Iterate on each day - new \DateInterval('P1D'), - //End with next sunday and 4 weeks - new \DateTime('Monday this week + 5 week') - ); - - //Fetch sessions - $sessions = $doctrine->getRepository(Session::class)->findAllByDatePeriod($period); - - //Init calendar - $calendar = []; - - //Init month - $month = null; - - //Iterate on each day - foreach($period as $date) { - //Init day in calendar - $calendar[$Ymd = $date->format('Ymd')] = [ - 'title' => $date->format('d'), - 'class' => [], - 'sessions' => [] - ]; - //Append month for first day of month - if ($month != $date->format('m')) { - $month = $date->format('m'); - $calendar[$Ymd]['title'] .= '/'.$month; - } - //Deal with today - if ($date->format('U') == ($today = strtotime('today'))) { - $calendar[$Ymd]['title'] .= '/'.$month; - $calendar[$Ymd]['current'] = true; - $calendar[$Ymd]['class'][] = 'current'; - } - //Disable passed days - if ($date->format('U') < $today) { - $calendar[$Ymd]['disabled'] = true; - $calendar[$Ymd]['class'][] = 'disabled'; - } - //Set next month days - if ($date->format('m') > date('m')) { - $calendar[$Ymd]['next'] = true; - $calendar[$Ymd]['class'][] = 'next'; - } - //Iterate on each session to find the one of the day - foreach($sessions as $session) { - if (($sessionYmd = $session->getDate()->format('Ymd')) == $Ymd) { - //Count number of application - $count = count($session->getApplications()); - - //Compute classes - $class = []; - if ($session->getApplication()) { - $class[] = 'granted'; - } elseif ($count == 0) { - $class[] = 'orphaned'; - } elseif ($count > 1) { - $class[] = 'disputed'; - } else { - $class[] = 'pending'; - } - //Add the session - $calendar[$Ymd]['sessions'][$session->getSlot()->getId().$session->getLocation()->getId()] = [ - 'id' => $session->getId(), - 'title' => ($count > 1?'['.$count.'] ':'').$session->getSlot()->getTitle().' '.$session->getLocation()->getTitle(), - 'class' => $class - ]; - } - } + //Get eastern + $eastern = $this->getEastern($date->format('Y')); - //Sort sessions - ksort($calendar[$Ymd]['sessions']); + //Check dynamic holidays + if ( + (clone $eastern)->add(new \DateInterval('P1D')) == $date || + (clone $eastern)->add(new \DateInterval('P39D')) == $date || + (clone $eastern)->add(new \DateInterval('P50D')) == $date + ) { + //Date is a dynamic holiday + return true; } + + //Date is not a holiday and week day + return false; } }