X-Git-Url: https://git.rapsys.eu/treebundle/blobdiff_plain/6081ccbb83091e8ce6045f73b911bebfd2963ebd..5ec7117fb4c06ccddc35f5de89dbbec77e3565ea:/Controller/TreeController.php diff --git a/Controller/TreeController.php b/Controller/TreeController.php index a6e2ac1..7c2a38f 100644 --- a/Controller/TreeController.php +++ b/Controller/TreeController.php @@ -11,24 +11,38 @@ namespace Rapsys\TreeBundle\Controller; +use Doctrine\Persistence\ManagerRegistry; + use Psr\Container\ContainerInterface; +use Rapsys\PackBundle\Util\FacebookUtil; +use Rapsys\TreeBundle\Entity\Album; +use Rapsys\TreeBundle\Entity\Element; +use Rapsys\TreeBundle\RapsysTreeBundle; +use Rapsys\UserBundle\RapsysUserBundle; + use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; +use Symfony\Bundle\SecurityBundle\Security; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\RequestStack; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\Routing\Generator\UrlGeneratorInterface; use Symfony\Component\Routing\RouterInterface; +use Symfony\Component\Security\Core\Authorization\AuthorizationCheckerInterface; + use Symfony\Contracts\Translation\TranslatorInterface; use Twig\Environment; -use Rapsys\TreeBundle\RapsysTreeBundle; - /** * {@inheritdoc} */ class TreeController extends AbstractController { + /** + * Alias string + */ + protected string $alias; + /** * Config array */ @@ -39,11 +53,21 @@ class TreeController extends AbstractController { */ protected array $context = []; + /** + * Count integer + */ + protected int $count; + /** * Locale string */ protected string $locale; + /** + * Page integer + */ + protected int $page; + /** * Request instance */ @@ -62,15 +86,20 @@ class TreeController extends AbstractController { /** * Creates a new tree controller * + * @param AuthorizationCheckerInterface $checker The container instance * @param ContainerInterface $container The ContainerInterface instance + * @param ManagerRegistry $doctrine The doctrine instance + * @param FacebookUtil $facebook The facebook instance * @param RouterInterface $router The router instance + * @param Security $security The security instance * @param RequestStack $stack The stack instance * @param TranslatorInterface $translator The translator instance * @param Environment $twig The twig environment instance + * @param integer $limit The page limit */ - function __construct(protected ContainerInterface $container, protected RouterInterface $router, protected RequestStack $stack, protected TranslatorInterface $translator, protected Environment $twig) { + function __construct(protected AuthorizationCheckerInterface $checker, protected ContainerInterface $container, protected ManagerRegistry $doctrine, protected FacebookUtil $facebook, protected RouterInterface $router, protected Security $security, protected RequestStack $stack, protected TranslatorInterface $translator, protected Environment $twig, protected int $limit = 5) { //Retrieve config - $this->config = $container->getParameter(RapsysTreeBundle::getAlias()); + $this->config = $container->getParameter($this->alias = RapsysTreeBundle::getAlias()); //Get main request $this->request = $this->stack->getMainRequest(); @@ -84,6 +113,14 @@ class TreeController extends AbstractController { //Set alternates $alternates = []; + //Get current page + $this->page = (int) $this->request->query->get('page'); + + //With negative page + if ($this->page < 0) { + $this->page = 0; + } + //Set route //TODO: default to not found route ??? //TODO: pour une url not found, cet attribut n'est pas défini, comment on fait ??? @@ -153,6 +190,273 @@ class TreeController extends AbstractController { ]; } + /** + * The album page + * + * Display album + * + * @param Request $request The request instance + * @param int $id The album id + * @param string $path The album path + * @param string $slug The album slug + * @return Response The rendered view + */ + public function album(Request $request, int $id, string $path, string $slug): Response { + //Get user + $user = $this->security->getUser(); + + //Check admin role + if (!$this->checker->isGranted('ROLE_'.strtoupper($this->container->getParameter(RapsysUserBundle::getAlias().'.default.admin')))) { + //Throw access denied + //XXX: prevent slugger reverse engineering by not displaying decoded mail + throw $this->createAccessDeniedException($this->translator->trans('Unable to access album', [], $this->alias)); + } + + //Without album + if (!($this->context['album'] = $this->doctrine->getRepository(Album::class)->findOneByIdPathAsArray($id, $path))) { + //Throw 404 + throw $this->createNotFoundException($this->translator->trans('Unable to find album')); + } + + //With slug not matching + if ($this->context['album']['slug'] !== $slug) { + //Redirect on clean slug + return $this->redirectToRoute($this->route, [ 'slug' => $this->context['album']['slug'] ]+$this->routeParams); + } + + //Set modified + $this->modified = $this->context['album']['modified']; + + //Create response + $response = new Response(); + + //With logged user + if ($this->checker->isGranted('IS_AUTHENTICATED_REMEMBERED')) { + //Set last modified + $response->setLastModified(new \DateTime('-1 year')); + + //Set as private + $response->setPrivate(); + //Without logged user + } else { + //Set etag + //XXX: only for public to force revalidation by last modified + $response->setEtag(md5(serialize($this->context['album']))); + + //Set last modified + $response->setLastModified($this->modified); + + //Set as public + $response->setPublic(); + + //Without role and modification + if ($response->isNotModified($request)) { + //Return 304 response + return $response; + } + } + + //Set keywords + /*$this->context['head']['keywords'] = implode( + ', ', + //Use closure to extract each unique article keywords sorted + (function ($t) { + //Return array + $r = []; + + //Iterate on articles + foreach($t as $a) { + //Non empty keywords + if (!empty($a['keywords'])) { + //Iterate on keywords + foreach($a['keywords'] as $k) { + //Set keyword + $r[$k['title']] = $k['title']; + } + } + } + + //Sort array + sort($r); + + //Return array + return $r; + })($this->context['articles']) + ); + //Get albums + $this->context['albums'] = $this->config['albums'];*/ + + //Return rendered response + return $this->render('@RapsysTree/album.html.twig', $this->context, $response); + } + + + /** + * The element page + * + * Display element + * + * @param Request $request The request instance + * @param int $id The element id + * @param ?string $path The element path + * @return Response The rendered view + */ + public function element(Request $request, int $id, string $path): Response { + //Get user + $user = $this->security->getUser(); + + //Without element + if (!($this->context['element'] = $this->doctrine->getRepository(Element::class)->findOneByUidIdPathAsArray($user?->getId(), $id, $path))) { + //Throw 404 + throw $this->createNotFoundException($this->translator->trans('Unable to find element')); + } + + //With realpath not matching path + //XXX: extra slashes removed by rewrite rule + if (($this->context['path'] = substr($this->context['element']['realpath'], strlen($this->context['element']['album']['path'])+1)) !== $path) { + //Redirect on clean path + return $this->redirectToRoute($this->route, [ 'path' => $this->context['path'] ]+$this->routeParams); + } + + //Set modified + $this->modified = $this->context['element']['modified']; + + //Create response + $response = new Response(); + + //With logged user + if ($this->checker->isGranted('IS_AUTHENTICATED_REMEMBERED')) { + //Set last modified + $response->setLastModified(new \DateTime('-1 year')); + + //Set as private + $response->setPrivate(); + //Without logged user + } else { + //Set etag + //XXX: only for public to force revalidation by last modified + $response->setEtag(md5(serialize($this->context['element']))); + + //Set last modified + $response->setLastModified($this->modified); + + //Set as public + $response->setPublic(); + + //Without role and modification + if ($response->isNotModified($request)) { + //Return 304 response + return $response; + } + } + + //TODO: move that in a shared function to use in album member function too + + /* + //With directory + if (is_dir($realpath)) { + //Set element directories + $this->context['element'] += [ + //Directories + 'directories' => [], + //Files + 'files' => [] + ]; + + //Iterate on directory + foreach(array_diff(scandir($realpath), ['.', '..']) as $item) { + //Check item + if ( + //Without item realpath + !($itempath = realpath($realpath.'/'.$item)) || + //With item realpath not matching element path + ( + $this->context['element']['album']['path'].$this->context['element']['path'] !== + substr($itempath, 0, strlen($this->context['element']['album']['path'].$this->context['element']['path'])) + ) + ) { + //Skip + continue; + } + + //With directory + if (is_dir($itempath)) { + //Append directory + $this->context['element']['directories'][$item] = $this->router->generate($this->route, [ 'path' => $this->context['path'].'/'.$item ]+$this->routeParams); + //With file + } elseif (is_file($itempath)) { + //Append file + $this->context['element']['files'][$item] = [ + //Set mime + 'mime' => mime_content_type($itempath), + //Set size + 'size' => filesize($itempath), + //Set link + 'link' => $this->router->generate($this->route, [ 'path' => $this->context['path'].'/'.$item ]+$this->routeParams) + ]; + //With unknown type + } else { + //Throw 404 + throw $this->createNotFoundException($this->translator->trans('Unable to process element')); + } + } + //With file + } elseif (is_file($realpath)) { + //Append file + $this->context['element']['file'] = [ + //Set mime + 'mime' => mime_content_type($realpath), + //Set size + 'size' => filesize($realpath), + //TODO: extra fields ? (preview, miniature, etc ?) + ]; + //TODO: mimetype decided extra fields ? (.pdf for doc, webm preview, img preview, etc ?) + //TODO: XXX: finish this !!! + //With unknown type + } else { + //Throw 404 + throw $this->createNotFoundException($this->translator->trans('Unable to process element')); + } + */ + + //Set keywords + /*$this->context['head']['keywords'] = implode( + ', ', + //Use closure to extract each unique article keywords sorted + (function ($t) { + //Return array + $r = []; + + //Iterate on articles + foreach($t as $a) { + //Non empty keywords + if (!empty($a['keywords'])) { + //Iterate on keywords + foreach($a['keywords'] as $k) { + //Set keyword + $r[$k['title']] = $k['title']; + } + } + } + + //Sort array + sort($r); + + //Return array + return $r; + })($this->context['articles']) + ); + //Get albums + $this->context['albums'] = $this->config['albums'];*/ + + #header('Content-Type: text/plain'); + #var_dump($this->context['element']); + #exit; + + //Return rendered response + return $this->render('@RapsysTree/element.html.twig', $this->context, $response); + } + /** * The index page * @@ -162,18 +466,86 @@ class TreeController extends AbstractController { * @return Response The rendered view */ public function index(Request $request): Response { - //Get roots - $this->context['roots'] = $this->config['roots']; + //Get user + $user = $this->security->getUser(); - //Render template - $response = $this->render('@RapsysTree/index.html.twig', $this->context); + //With not enough albums + if (($this->count = $this->doctrine->getRepository(Album::class)->countByUidAsInt($user?->getId())) < $this->page * $this->limit) { + //Throw 404 + throw $this->createNotFoundException($this->translator->trans('Unable to find albums')); + } - $response->setEtag(md5($response->getContent())); - $response->setPublic(); - $response->isNotModified($request); + //Get albums + if ($this->context['albums'] = $this->doctrine->getRepository(Album::class)->findByUidAsArray($user?->getId(), $this->page, $this->limit)) { + //Set modified + $this->modified = max(array_map(function ($v) { return $v['modified']; }, $this->context['albums'])); + //Without albums + } else { + //Set empty modified + $this->modified = new \DateTime('-1 year'); + } - //Return response - return $response; + //Create response + $response = new Response(); + + //With logged user + if ($this->checker->isGranted('IS_AUTHENTICATED_REMEMBERED')) { + //Set last modified + $response->setLastModified(new \DateTime('-1 year')); + + //Set as private + $response->setPrivate(); + //Without logged user + } else { + //Set etag + //XXX: only for public to force revalidation by last modified + $response->setEtag(md5(serialize($this->context['albums']))); + + //Set last modified + $response->setLastModified($this->modified); + + //Set as public + $response->setPublic(); + + //Without role and modification + if ($response->isNotModified($request)) { + //Return 304 response + return $response; + } + } + + //Set keywords + /*$this->context['head']['keywords'] = implode( + ', ', + //Use closure to extract each unique article keywords sorted + (function ($t) { + //Return array + $r = []; + + //Iterate on articles + foreach($t as $a) { + //Non empty keywords + if (!empty($a['keywords'])) { + //Iterate on keywords + foreach($a['keywords'] as $k) { + //Set keyword + $r[$k['title']] = $k['title']; + } + } + } + + //Sort array + sort($r); + + //Return array + return $r; + })($this->context['articles']) + ); + //Get albums + $this->context['albums'] = $this->config['albums'];*/ + + //Return rendered response + return $this->render('@RapsysTree/index.html.twig', $this->context, $response); } /**