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
*/
*/
protected array $context = [];
+ /**
+ * Count integer
+ */
+ protected int $count;
+
/**
* Locale string
*/
protected string $locale;
+ /**
+ * Page integer
+ */
+ protected int $page;
+
/**
* Request instance
*/
/**
* 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();
//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 ???
];
}
+ /**
+ * 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
*
- * Display file tree
+ * Display index
+ *
+ * @param Request $request The request instance
+ * @return Response The rendered view
+ */
+ public function index(Request $request): Response {
+ //Get user
+ $user = $this->security->getUser();
+
+ //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'));
+ }
+
+ //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');
+ }
+
+ //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);
+ }
+
+ /**
+ * The directory page
+ *
+ * Display directory
+ *
+ * @param Request $request The request instance
+ * @param string $path The directory path
+ * @return Response The rendered view
+ */
+ public function directory(Request $request, string $path): Response {
+ header('Content-Type: text/plain');
+ var_dump($path);
+ exit;
+
+ //Render template
+ $response = $this->render('@RapsysTree/directory.html.twig', $this->context);
+
+ $response->setEtag(md5($response->getContent()));
+ $response->setPublic();
+ $response->isNotModified($request);
+
+ //Return response
+ return $response;
+ }
+
+ /**
+ * The document page
+ *
+ * Display document
*
* @param Request $request The request instance
+ * @param string $path The directory path
* @return Response The rendered view
*/
- public function index(Request $request, string $path): Response {
+ public function document(Request $request, string $path): Response {
//Render template
- $response = $this->render('@RapsysTree/index.html.twig', $this->context);
+ $response = $this->render('@RapsysTree/document.html.twig', $this->context);
$response->setEtag(md5($response->getContent()));
$response->setPublic();
}
//With empty facebook title and title
- if (empty($parameters['facebook']['og:title']) && !empty($parameters['title'])) {
+ if (empty($parameters['facebook']['og:title']) && !empty($parameters['title']['page'])) {
//Set facebook title
- $parameters['facebook']['og:title'] = $parameters['title'];
+ $parameters['facebook']['og:title'] = $parameters['title']['page'];
}
//With empty facebook description and description