Skip to content
Snippets Groups Projects
FolderController.php 13.2 KiB
Newer Older
  • Learn to ignore specific revisions
  • <?php
    
    /**
     * Copyright Maarch since 2008 under licence GPLv3.
     * See LICENCE.txt file at the root folder for more details.
     * This file is part of Maarch software.
     */
    
    /**
     * @brief Folder Controller
     *
     * @author dev@maarch.org
     */
    
    namespace Folder\controllers;
    
    
    use Entity\models\EntityModel;
    
    use Folder\models\EntityFolderModel;
    
    use Folder\models\FolderModel;
    use History\controllers\HistoryController;
    use Respect\Validation\Validator;
    use Slim\Http\Request;
    use Slim\Http\Response;
    
    use SrcCore\models\DatabaseModel;
    
    use User\models\UserModel;
    
    
    class FolderController
    {
        public function get(Request $request, Response $response)
        {
    
            $folders = FolderController::getScopeFolders(['login' => $GLOBALS['userId']]);
    
    
            $tree = [];
            foreach ($folders as $folder) {
                $insert = [
                    'name'       => $folder['label'],
                    'id'         => $folder['id'],
                    'label'      => $folder['label'],
                    'public'     => $folder['public'],
                    'user_id'    => $folder['user_id'],
                    'parent_id'  => $folder['parent_id'],
                    'level'      => $folder['level'],
                ];
                if ($folder['level'] == 0) {
                    $tree[] = $insert;
    
                    $found = false;
    
                    foreach ($tree as $key => $branch) {
                        if ($branch['id'] == $folder['parent_id']) {
                            array_splice($tree, $key + 1, 0, [$insert]);
    
                            $found = true;
    
                    if (!$found) {
                        $tree[] = $insert;
                    }
    
            return $response->withJson(['folders' => $tree]);
    
        }
    
        public function getById(Request $request, Response $response, array $aArgs)
        {
            if (!Validator::numeric()->notEmpty()->validate($aArgs['id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Query id is empty or not an integer']);
            }
    
    
            $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $aArgs['id']]);
            if (empty($folder[0])) {
                return $response->withStatus(400)->withJson(['errors' => 'Folder not found or out of your perimeter']);
    
            $folder[0]['sharing']['entities'] = [];
            if ($folder[0]['public']) {
    
                $entitiesFolder = EntityFolderModel::getByFolderId(['folder_id' => $aArgs['id']]);
                foreach ($entitiesFolder as $value) {
    
                    $folder[0]['sharing']['entities'][] = ['entity_id' => $value['entity_id'], 'edition' => $value['edition']];
    
            //TODO Get resources
    
    
            return $response->withJson(['folder' => $folder[0]]);
    
        }
    
        public function create(Request $request, Response $response)
        {
            $data = $request->getParams();
    
            if (!Validator::stringType()->notEmpty()->validate($data['label'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body label is empty or not a string']);
            }
            if (!empty($data['parent_id']) && !Validator::intval()->validate($data['parent_id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body parent_id is not a numeric']);
            }
    
    
            if (empty($data['parent_id'])) {
                $data['parent_id'] = 0;
                $owner  = $GLOBALS['id'];
                $public = false;
    
                $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $data['parent_id'], 'edition' => true]);
                if (empty($folder[0])) {
                    return $response->withStatus(400)->withJson(['errors' => 'Parent Folder not found or out of your perimeter']);
                }
                $owner  = $folder[0]['user_id'];
                $public = $folder[0]['public'];
                $level  = $folder[0]['level'] + 1;
    
            $id = FolderModel::create([
    
                'label'     => $data['label'],
                'public'    => $public,
                'user_id'   => $owner,
                'parent_id' => $data['parent_id'],
                'level'     => $level
    
            if ($public) {
                $entitiesSharing = EntityFolderModel::getByFolderId(['folder_id' => $data['parent_id']]);
    
                foreach ($entitiesSharing as $entity) {
    
                    EntityFolderModel::create([
                        'folder_id' => $id,
                        'entity_id' => $entity['entity_id'],
                        'edition'   => $entity['edition'],
                    ]);
                }
            }
    
    
            HistoryController::add([
                'tableName' => 'folders',
                'recordId'  => $id,
                'eventType' => 'ADD',
    
                'info'      => _FOLDER_CREATION . " : {$data['label']}",
    
                'moduleId'  => 'folder',
                'eventId'   => 'folderCreation',
            ]);
    
            return $response->withJson(['folder' => $id]);
        }
    
        public function update(Request $request, Response $response, array $aArgs)
        {
            $data = $request->getParams();
    
            if (!Validator::numeric()->notEmpty()->validate($aArgs['id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Query id is empty or not an integer']);
            }
            if (!Validator::stringType()->notEmpty()->validate($data['label'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body label is empty or not a string']);
            }
            if (!empty($data['parent_id']) &&!Validator::intval()->validate($data['parent_id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body parent_id is not a numeric']);
            }
    
    
            $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $aArgs['id'], 'edition' => true]);
            if (empty($folder[0])) {
                return $response->withStatus(400)->withJson(['errors' => 'Folder not found or out of your perimeter']);
            }
    
            if (empty($data['parent_id'])) {
                $data['parent_id'] = 0;
    
                $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $data['parent_id']]);
                if (empty($folder[0])) {
                    return $response->withStatus(400)->withJson(['errors' => 'Parent Folder not found or out of your perimeter']);
                }
                $level = $folder[0]['level'] + 1;
    
            FolderModel::update([
                'set' => [
                    'label'      => $data['label'],
    
                    'parent_id'  => $data['parent_id'],
                    'level'      => $level
    
                ],
                'where' => ['id = ?'],
                'data' => [$aArgs['id']]
            ]);
    
            HistoryController::add([
                'tableName' => 'folders',
                'recordId'  => $aArgs['id'],
                'eventType' => 'UP',
                'info'      => _FOLDER_MODIFICATION . " : {$data['label']}",
                'moduleId'  => 'folder',
                'eventId'   => 'folderModification',
            ]);
    
            return $response->withStatus(200);
        }
    
        public function sharing(Request $request, Response $response, array $aArgs)
        {
            $data = $request->getParams();
    
            if (!Validator::numeric()->notEmpty()->validate($aArgs['id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Query id is empty or not an integer']);
            }
            if (!Validator::boolVal()->validate($data['public'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body public is empty or not a boolean']);
            }
    
            if ($data['public'] && !isset($data['sharing']['entities'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Body sharing/entities does not exists']);
            }
    
            DatabaseModel::beginTransaction();
            $sharing = FolderController::folderSharing(['folderId' => $aArgs['id'], 'public' => $data['public'], 'sharing' => $data['sharing']]);
            if (!$sharing) {
                DatabaseModel::rollbackTransaction();
                return $response->withStatus(400)->withJson(['errors' => 'Can not share/unshare folder because almost one folder is out of your perimeter']);
            }
            DatabaseModel::commitTransaction();
    
            HistoryController::add([
                'tableName' => 'folders',
                'recordId'  => $aArgs['id'],
                'eventType' => 'UP',
                'info'      => _FOLDER_SHARING_MODIFICATION . " : {$data['label']}",
                'moduleId'  => 'folder',
                'eventId'   => 'folderModification',
            ]);
    
            return $response->withStatus(200);
        }
    
        public function folderSharing($aArgs = [])
        {
            $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $aArgs['folderId'], 'edition' => true]);
    
            if (empty($folder[0])) {
    
    
            FolderModel::update([
                'set' => [
    
                    'public' => empty($aArgs['public']) ? 'false' : 'true',
    
                ],
                'where' => ['id = ?'],
    
                'data' => [$aArgs['folderId']]
    
            EntityFolderModel::deleteByFolderId(['folder_id' => $aArgs['folderId']]);
    
            if ($aArgs['public'] && !empty($aArgs['sharing']['entities'])) {
                foreach ($aArgs['sharing']['entities'] as $entity) {
    
                    EntityFolderModel::create([
    
                        'folder_id' => $aArgs['folderId'],
    
                        'entity_id' => $entity['entity_id'],
                        'edition'   => $entity['edition'],
                    ]);
    
            $folderChild = FolderModel::getChild(['id' => $aArgs['folderId'], 'select' => ['id']]);
            if (!empty($folderChild)) {
                foreach ($folderChild as $child) {
                    $sharing = FolderController::folderSharing(['folderId' => $child['id'], 'public' => $aArgs['public'], 'sharing' => $aArgs['sharing']]);
                    if (!$sharing) {
                        return false;
                    }
                }
            }
            return true;
    
        }
    
        public function delete(Request $request, Response $response, array $aArgs)
        {
            if (!Validator::numeric()->notEmpty()->validate($aArgs['id'])) {
                return $response->withStatus(400)->withJson(['errors' => 'Query id is empty or not an integer']);
            }
    
    
            $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $aArgs['id'], 'edition' => true]);
    
            DatabaseModel::beginTransaction();
            $deletion = FolderController::folderDeletion(['folderId' => $aArgs['id']]);
            if (!$deletion) {
                DatabaseModel::rollbackTransaction();
                return $response->withStatus(400)->withJson(['errors' => 'Can not delete because almost one folder is out of your perimeter']);
            }
            DatabaseModel::commitTransaction();
    
            HistoryController::add([
                'tableName' => 'folder',
                'recordId'  => $aArgs['id'],
                'eventType' => 'DEL',
    
                'info'      => _FOLDER_SUPPRESSION . " : {$folder[0]['label']}",
    
                'moduleId'  => 'folder',
                'eventId'   => 'folderSuppression',
            ]);
    
            return $response->withStatus(200);
        }
    
        public function folderDeletion($aArgs = [])
        {
            $folder = FolderController::getScopeFolders(['login' => $GLOBALS['userId'], 'folderId' => $aArgs['folderId'], 'edition' => true]);
            if (empty($folder[0])) {
                return false;
            }
    
            FolderModel::delete(['where' => ['id = ?'], 'data' => [$aArgs['folderId']]]);
            EntityFolderModel::deleteByFolderId(['folder_id' => $aArgs['folderId']]);
    
            $folderChild = FolderModel::getChild(['id' => $aArgs['folderId'], 'select' => ['id']]);
            if (!empty($folderChild)) {
                foreach ($folderChild as $child) {
                    $deletion = FolderController::folderDeletion(['folderId' => $child['id']]);
                    if (!$deletion) {
                        return false;
                    }
                }
            }
            return true;
        }
    
    
        // login (string) : Login of user connected
        // folderId (integer) : Check specific folder
        // edition (boolean) : whether user can edit or not
        private static function getScopeFolders($aArgs = [])
        {
            $login = $aArgs['login'];
            $userEntities = EntityModel::getEntitiesByUserId([
                'select'  => ['entities.id'],
                'user_id' => $login
            ]);
    
            $userEntities = array_column($userEntities, 'id');
            if (empty($userEntities)) {
                $userEntities = 0;
            }
    
            $user = UserModel::getByLogin(['login' => $login, 'select' => ['id']]);
    
    
            if ($aArgs['edition']) {
                $edition = [true];
            } else {
                $edition = ['false', 'true', null];
            }
    
            $where = ['(user_id = ? OR (entity_id in (?) AND entities_folders.edition in (?)))'];
            $data = [$user['id'], $userEntities, $edition];
    
    
            if (!empty($aArgs['folderId'])) {
                $where[] = 'folders.id = ?';
    
                $data[]  = $aArgs['folderId'];
    
            }
    
            $folders = FolderModel::get([
    
                'select'   => ['distinct (folders.id)', 'folders.*'],
    
                'where'    => $where,
                'data'     => $data,
                'order_by' => ['level']
            ]);
    
            return $folders;
        }