Skip to content
Snippets Groups Projects
StoreController.php 20.3 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 Store Controller
     * @author dev@maarch.org
     * @ingroup core
     */
    
    
    namespace Resource\controllers;
    
    use Slim\Http\Request;
    use Slim\Http\Response;
    
    use Attachment\models\AttachmentModel;
    
    use Contact\models\ContactModel;
    
    Giovannoni Laurent's avatar
    Giovannoni Laurent committed
    use Docserver\controllers\DocserverController;
    
    use Resource\models\ChronoModel;
    
    Damien's avatar
    Damien committed
    use SrcCore\models\DatabaseModel;
    
    use SrcCore\models\ValidatorModel;
    
    use Respect\Validation\Validator;
    
    Damien's avatar
    Damien committed
    use Entity\models\EntityModel;
    
    Damien's avatar
    Damien committed
    use Resource\models\ResModel;
    
    Damien's avatar
    Damien committed
    use SrcCore\models\CoreConfigModel;
    
    Damien's avatar
    Damien committed
    use User\models\UserModel;
    
    
    class StoreController
    {
    
        public function checkFileUpload(Request $request, Response $response, array $aArgs)
        {
            $body = $request->getParsedBody();
    
            if (!Validator::notEmpty()->validate($body['size'])) {
                return $response->withStatus(400)->withJson(['errors' => 'filesize is empty']);
            } else if (!Validator::notEmpty()->validate($body['type'])) {
                return $response->withStatus(400)->withJson(['errors' => 'no mime type detected']);
            } else if (!Validator::notEmpty()->validate($body['extension'])) {
                return $response->withStatus(400)->withJson(['errors' => 'this filename has no extension']);
            }
    
            if (!StoreController::isFileAllowed($body)) {
                return $response->withStatus(400)->withJson(['errors' => _FILE_NOT_ALLOWED_INFO_1.' "'.$body['extension'].'" '._FILE_NOT_ALLOWED_INFO_2.' "'. $body['type']. '" '._FILE_NOT_ALLOWED_INFO_3]);
            }
    
            $maxFilesizeMo = ini_get('upload_max_filesize');
            $maxFilesizeKo = ini_get('upload_max_filesize')*1024;
    
            if ($body['size']/1024 > $maxFilesizeKo) {
                return $response->withStatus(400)->withJson(['errors' => _MAX_SIZE_UPLOAD_REACHED.' ('.round($maxFilesizeMo).'Mo Max.)']);
            }
            return $response->withJson(['success']);
        }
    
        private static function isFileAllowed(array $args)
        {
            ValidatorModel::notEmpty($args, ['extension', 'type']);
            ValidatorModel::stringType($args, ['extension', 'type']);
            $loadedXml = CoreConfigModel::getXmlLoaded(['path' => 'apps/maarch_entreprise/xml/extensions.xml']);
            if ($loadedXml) {
                foreach ($loadedXml->FORMAT as $value) {
                    if (strtolower((string)$value->name) == strtolower($args['extension']) && strtolower((string)$value->mime) == strtolower($args['type'])) {
                        return true;
                    }
                }
            }
    
            return false;
        }
    
        private static function getAllowedMime()
        {
            $loadedXml = CoreConfigModel::getXmlLoaded(['path' => 'apps/maarch_entreprise/xml/extensions.xml']);
            $mimeList = [];
            
            if ($loadedXml) {
                foreach ($loadedXml->FORMAT as $value) {
                    $mimeList[] = (string)$value->mime;
                }
            }
    
            return array_unique($mimeList);
        }
    
    
        public static function storeResource(array $aArgs)
    
    Damien's avatar
    Damien committed
        {
            ValidatorModel::notEmpty($aArgs, ['encodedFile', 'format', 'status', 'type_id', 'category_id']);
            ValidatorModel::stringType($aArgs, ['format', 'status']);
    
            $mlbColumns = [
                'category_id', 'exp_contact_id', 'exp_user_id', 'dest_contact_id', 'dest_user_id',
    
                'nature_id', 'alt_identifier', 'admission_date', 'process_limit_date', 'closing_date', 'address_id'
    
    Damien's avatar
    Damien committed
            ];
            try {
    
                foreach ($aArgs as $column => $value) {
                    if (empty($value)) {
                        unset($aArgs[$column]);
                    }
                }
    
    Damien's avatar
    Damien committed
                $fileContent    = base64_decode(str_replace(['-', '_'], ['+', '/'], $aArgs['encodedFile']));
    
                $storeResult = DocserverController::storeResourceOnDocServer([
                    'collId'            => 'letterbox_coll',
                    'docserverTypeId'   => 'DOC',
    
    Damien's avatar
    Damien committed
                    'encodedResource'   => base64_encode($fileContent),
                    'format'            => $aArgs['format']
    
    Damien's avatar
    Damien committed
                ]);
                if (!empty($storeResult['errors'])) {
                    return ['errors' => '[storeResource] ' . $storeResult['errors']];
                }
                unset($aArgs['encodedFile']);
    
    
    Damien's avatar
    Damien committed
                $resId = DatabaseModel::getNextSequenceValue(['sequenceId' => 'res_id_mlb_seq']);
    
    
    Damien's avatar
    Damien committed
                $data = [
                    'docserver_id'  => $storeResult['docserver_id'],
                    'filename'      => $storeResult['file_destination_name'],
                    'filesize'      => $storeResult['fileSize'],
                    'path'          => $storeResult['destination_dir'],
    
    Damien's avatar
    Damien committed
                    'fingerprint'   => $storeResult['fingerPrint'],
                    'res_id'        => $resId
    
    Damien's avatar
    Damien committed
                ];
                $data = array_merge($aArgs, $data);
                $data = StoreController::prepareStorage($data);
    
                $dataMlb = [];
                foreach ($data as $key => $value) {
                    if (in_array($key, $mlbColumns)) {
                        $dataMlb[$key] = $value;
                        unset($data[$key]);
                    }
                }
    
    Damien's avatar
    Damien committed
                ResModel::create($data);
    
    Damien's avatar
    Damien committed
    
                $dataMlb['res_id'] = $resId;
                ResModel::createExt($dataMlb);
    
                return $resId;
            } catch (\Exception $e) {
                return ['errors' => '[storeResource] ' . $e->getMessage()];
            }
        }
    
        public static function storeResourceRes(array $aArgs)
    
        {
            ValidatorModel::notEmpty($aArgs, ['encodedFile', 'data', 'collId', 'table', 'fileFormat', 'status']);
            ValidatorModel::stringType($aArgs, ['collId', 'table', 'fileFormat', 'status']);
            ValidatorModel::arrayType($aArgs, ['data']);
    
    
            if (!in_array($aArgs['table'], ['res_letterbox', 'res_attachments', 'res_version_attachments'])) {
    
                return ['errors' => '[storeResource] Table not valid'];
            }
    
            try {
                $fileContent    = base64_decode(str_replace(['-', '_'], ['+', '/'], $aArgs['encodedFile']));
    
    
                $storeResult = DocserverController::storeResourceOnDocServer([
    
                    'collId'            => $aArgs['collId'],
                    'docserverTypeId'   => 'DOC',
    
    Damien's avatar
    Damien committed
                    'encodedResource'   => base64_encode($fileContent),
                    'format'            => $aArgs['fileFormat']
    
                ]);
                if (!empty($storeResult['errors'])) {
                    return ['errors' => '[storeResource] ' . $storeResult['errors']];
                }
    
    
    Damien's avatar
    Damien committed
                $data = StoreController::prepareStorageRes([
    
                    'data'          => $aArgs['data'],
                    'docserverId'   => $storeResult['docserver_id'],
                    'status'        => $aArgs['status'],
                    'fileName'      => $storeResult['file_destination_name'],
                    'fileFormat'    => $aArgs['fileFormat'],
                    'fileSize'      => $storeResult['fileSize'],
                    'path'          => $storeResult['destination_dir'],
                    'fingerPrint'   => $storeResult['fingerPrint']
                ]);
    
                $resId = false;
                if ($aArgs['table'] == 'res_letterbox') {
                    $resId = ResModel::create($data);
                } elseif ($aArgs['table'] == 'res_attachments') {
    
    Damien's avatar
    Damien committed
                    $resId = AttachmentModel::create($data);
    
                } elseif ($aArgs['table'] == 'res_version_attachments') {
                    $resId = AttachmentModel::createVersion($data);
    
                }
    
                return $resId;
            } catch (\Exception $e) {
                return ['errors' => '[storeResource] ' . $e->getMessage()];
            }
        }
    
        public static function controlFingerPrint(array $aArgs)
        {
            ValidatorModel::notEmpty($aArgs, ['pathInit', 'pathTarget']);
            ValidatorModel::stringType($aArgs, ['pathInit', 'pathTarget', 'fingerprintMode']);
    
            if (!file_exists($aArgs['pathInit'])) {
                return ['errors' => '[controlFingerprint] PathInit does not exist'];
            }
            if (!file_exists($aArgs['pathTarget'])) {
                return ['errors' => '[controlFingerprint] PathTarget does not exist'];
            }
    
            $fingerprint1 = StoreController::getFingerPrint(['filePath' => $aArgs['pathInit'], 'mode' => $aArgs['fingerprintMode']]);
            $fingerprint2 = StoreController::getFingerPrint(['filePath' => $aArgs['pathTarget'], 'mode' => $aArgs['fingerprintMode']]);
    
            if ($fingerprint1 != $fingerprint2) {
                return ['errors' => '[controlFingerprint] Fingerprints do not match: ' . $aArgs['pathInit'] . ' and ' . $aArgs['pathTarget']];
            }
    
            return true;
        }
    
        public static function getFingerPrint(array $aArgs)
        {
            ValidatorModel::notEmpty($aArgs, ['filePath']);
            ValidatorModel::stringType($aArgs, ['filePath', 'mode']);
    
            if (empty($aArgs['mode']) || $aArgs['mode'] == 'NONE') {
    
    Damien's avatar
    Damien committed
                $aArgs['mode'] = 'sha512';
    
            }
    
            return hash_file(strtolower($aArgs['mode']), $aArgs['filePath']);
        }
    
        public static function prepareStorage(array $aArgs)
    
    Damien's avatar
    Damien committed
        {
    
    Damien's avatar
    Damien committed
            ValidatorModel::notEmpty($aArgs, ['docserver_id', 'filename', 'format', 'filesize', 'path', 'fingerprint', 'status', 'res_id']);
    
    Damien's avatar
    Damien committed
            ValidatorModel::stringType($aArgs, ['docserver_id', 'filename', 'format', 'path', 'fingerprint', 'status']);
    
    Damien's avatar
    Damien committed
            ValidatorModel::intVal($aArgs, ['filesize', 'res_id']);
    
    Damien's avatar
    Damien committed
            if (empty($aArgs['typist'])) {
                $aArgs['typist'] = 'auto';
    
    Damien's avatar
    Damien committed
            unset($aArgs['alt_identifier']);
            if (!empty($aArgs['chrono'])) {
    
    Damien's avatar
    Damien committed
                $aArgs['alt_identifier'] = ChronoModel::getChrono(['id' => $aArgs['category_id'], 'entityId' => $aArgs['destination'], 'typeId' => $aArgs['type_id'], 'resId' => $aArgs['res_id']]);
    
    Damien's avatar
    Damien committed
            }
    
    Damien's avatar
    Damien committed
            unset($aArgs['chrono']);
    
            if (empty($aArgs['process_limit_date'])) {
    
    Damien's avatar
    Damien committed
                $processLimitDate = ResModel::getStoredProcessLimitDate(['typeId' => $aArgs['type_id'], 'admissionDate' => $aArgs['admission_date']]);
                $aArgs['process_limit_date'] = $processLimitDate;
            }
    
    Damien's avatar
    Damien committed
    
            if (!empty($aArgs['exp_contact_id']) && !is_numeric($aArgs['exp_contact_id'])) {
                $mail = explode('<', str_replace('>', '', $aArgs['exp_contact_id']));
                $contact = ContactModel::getByEmail(['email' => $mail[count($mail) - 1], 'select' => ['contacts_v2.contact_id']]);
                if (!empty($contact['contact_id'])) {
                    $aArgs['exp_contact_id'] = $contact['contact_id'];
                } else {
                    $aArgs['exp_contact_id'] = 0;
                }
            }
    
            if (!empty($aArgs['address_id']) && !is_numeric($aArgs['address_id'])) {
                $mail = explode('<', str_replace('>', '', $aArgs['address_id']));
                $contact = ContactModel::getByEmail(['email' => $mail[count($mail) - 1], 'select' => ['contact_addresses.id']]);
                if (!empty($contact['id'])) {
                    $aArgs['address_id'] = $contact['id'];
                } else {
                    $aArgs['address_id'] = 0;
                }
            }
    
    
    Damien's avatar
    Damien committed
            unset($aArgs['external_id']);
            if (!empty($aArgs['externalId'])) {
                if (is_array($aArgs['externalId'])) {
                    $aArgs['external_id'] = json_encode($aArgs['externalId']);
                }
                unset($aArgs['externalId']);
            }
    
    
    Damien's avatar
    Damien committed
            $aArgs['creation_date'] = 'CURRENT_TIMESTAMP';
    
            return $aArgs;
        }
    
        public static function prepareStorageRes(array $aArgs)
    
            ValidatorModel::notEmpty($aArgs, ['data', 'docserverId', 'fileName', 'fileFormat', 'fileSize', 'path', 'fingerPrint']);
    
            ValidatorModel::stringType($aArgs, ['docserverId', 'status', 'fileName', 'fileFormat', 'path', 'fingerPrint']);
            ValidatorModel::arrayType($aArgs, ['data']);
            ValidatorModel::intVal($aArgs, ['fileSize']);
    
            $statusFound        = false;
            $typistFound        = false;
            $toAddressFound     = false;
            $userPrimaryEntity  = false;
    
            foreach ($aArgs['data'] as $key => $value) {
                $aArgs['data'][$key]['column'] = strtolower($value['column']);
            }
    
            foreach ($aArgs['data'] as $key => $value) {
                if (strtolower($value['type']) == 'integer' || strtolower($value['type']) == 'float') {
                    if (empty($value['value'])) {
                        $aArgs['data'][$key]['value'] = '0';
                    }
    
                } elseif (strtolower($value['type']) == 'string') {
    
                    $aArgs['data'][$key]['value'] = str_replace(';', '', $value['value']);
                    $aArgs['data'][$key]['value'] = str_replace('--', '', $value['value']);
                }
    
                if ($value['column'] == 'status') {
                    $statusFound = true;
    
                } elseif ($value['column'] == 'typist') {
    
                    $typistFound = true;
    
                } elseif ($value['column'] == 'custom_t10') {
    
                    $theString = str_replace('>', '', $value['value']);
                    $mail = explode("<", $theString);
                    $user =  UserModel::getByEmail(['mail' => $mail[count($mail) -1], 'select' => ['user_id']]);
                    if (!empty($user[0]['user_id'])) {
                        $toAddressFound = true;
                        $destUser = $user[0]['user_id'];
    
    Damien's avatar
    Damien committed
                        $entity = EntityModel::getByLogin(['login' => $destUser, 'select' => ['entity_id']]);
    
                        if (!empty($entity[0]['entity_id'])) {
                            $userEntity = $entity[0]['entity_id'];
                            $userPrimaryEntity = true;
                        }
                    } else {
                        $entity = EntityModel::getByEmail(['email' => $mail[count($mail) -1], 'select' => ['entity_id']]);
                        if (!empty($entity[0]['entity_id'])) {
                            $userPrimaryEntity = true;
                        }
                    }
                }
            }
    
            $destUser   = empty($destUser) ? '' : $destUser;
            $userEntity = empty($userEntity) ? '' : $userEntity;
    
            if (!$typistFound && !$toAddressFound) {
                $aArgs['data'][] = [
                    'column'    => 'typist',
                    'value'     => 'auto',
                    'type'      => 'string'
                ];
            }
            if (!$statusFound) {
                $aArgs['data'][] = [
                    'column'    => 'status',
                    'value'     => $aArgs['status'],
                    'type'      => 'string'
                ];
            }
            if ($toAddressFound) {
                $aArgs['data'][] = [
                    'column'    => 'dest_user',
                    'value'     => $destUser,
                    'type'      => 'string'
                ];
    
    Damien's avatar
    Damien committed
                if (!$typistFound) {
                    $aArgs['data'][] = [
                        'column'    => 'typist',
                        'value'     => $destUser,
                        'type'      => 'string'
                    ];
                }
    
            }
            if ($userPrimaryEntity) {
                $destinationFound = false;
                $initiatorFound = false;
                foreach ($aArgs['data'] as $key => $value) {
                    if ($value['column'] == 'destination') {
                        if (empty($value['value'])) {
                            $aArgs['data'][$key]['value'] = $userEntity;
                        }
                        $destinationFound = true;
    
                    } elseif ($value['column'] == 'initiator') {
    
                        if (empty($value['value'])) {
                            $aArgs['data'][$key]['value'] = $userEntity;
                        }
                        $initiatorFound = true;
                    }
                }
                if (!$destinationFound) {
                    $aArgs['data'][] = [
                        'column'    => 'destination',
                        'value'     => $userEntity,
                        'type'      => 'string'
                    ];
                }
                if (!$initiatorFound) {
                    $aArgs['data'][] = [
                        'column'    => 'initiator',
                        'value'     => $userEntity,
                        'type'      => 'string'
                    ];
                }
            }
    
            $aArgs['data'][] = [
                'column'    => 'docserver_id',
                'value'     => $aArgs['docserverId'],
                'type'      => 'string'
            ];
            $aArgs['data'][] = [
                'column'    => 'creation_date',
                'value'     => 'CURRENT_TIMESTAMP',
                'type'      => 'function'
            ];
            $aArgs['data'][] = [
                'column'    => 'path',
                'value'     => $aArgs['path'],
                'type'      => 'string'
            ];
            $aArgs['data'][] = [
                'column'    => 'fingerprint',
                'value'     => $aArgs['fingerPrint'],
                'type'      => 'string'
            ];
            $aArgs['data'][] = [
                'column'    => 'filename',
                'value'     => $aArgs['fileName'],
                'type'      => 'string'
            ];
            $aArgs['data'][] = [
                'column'    => 'format',
                'value'     => $aArgs['fileFormat'],
                'type'      => 'string'
            ];
            $aArgs['data'][] = [
                'column'    => 'filesize',
                'value'     => $aArgs['fileSize'],
                'type'      => 'int'
            ];
    
            $formatedData = [];
            foreach ($aArgs['data'] as $value) {
                $formatedData[$value['column']] = $value['value'];
            }
    
            return $formatedData;
        }
    
    Damien's avatar
    Damien committed
    
        public static function prepareExtStorage(array $aArgs)
        {
            ValidatorModel::notEmpty($aArgs, ['data', 'resId']);
            ValidatorModel::arrayType($aArgs, ['data']);
            ValidatorModel::intVal($aArgs, ['resId']);
    
            $processLimitDateFound  = false;
    
            $admissionDate          = null;
    
    Damien's avatar
    Damien committed
    
            foreach ($aArgs['data'] as $key => $value) {
                $aArgs['data'][$key]['column'] = strtolower($value['column']);
            }
    
            foreach ($aArgs['data'] as $value) {
                if ($value['column'] == 'process_limit_date') {
                    $processLimitDateFound = true;
                }
                if ($value['column'] == 'category_id') {
                    $categoryId = $value['value'];
                }
    
                if ($value['column'] == 'admission_date') {
                    $admissionDate = $value['value'];
                }
    
    Damien's avatar
    Damien committed
            }
    
            if (!$processLimitDateFound) {
    
                $processLimitDate = ResModel::getStoredProcessLimitDate(['resId' => $aArgs['resId'], 'admissionDate' => $admissionDate]);
    
    Damien's avatar
    Damien committed
    
                $aArgs['data'][] = [
                    'column'    => 'process_limit_date',
                    'value'     => $processLimitDate,
                    'type'      => 'date'
                ];
            }
    
            foreach ($aArgs['data'] as $key => $value) {
                if (strtolower($value['type']) == 'integer' || strtolower($value['type']) == 'float') {
                    if ($value['value'] == '') {
                        $aArgs['data'][$key]['value'] = '0';
                    }
                    $aArgs['data'][$key]['value'] = str_replace(',', '.', $value['value']);
                }
                if ($value['column'] == 'alt_identifier' && empty($value['value']) && !empty($categoryId)) {
                    $document = ResModel::getById(['resId' => $aArgs['resId'], 'select' => ['destination, type_id']]);
                    $aArgs['data'][$key]['value'] = ChronoModel::getChrono(['id' => $categoryId, 'entityId' => $document['destination'], 'typeId' => $document['type_id']]);
                } elseif ($value['column'] == 'exp_contact_id' && !empty($value['value']) && !is_numeric($value['value'])) {
                    $mail = explode('<', str_replace('>', '', $value['value']));
    
                    $contact = ContactModel::getByEmail(['email' => $mail[count($mail) - 1], 'select' => ['contacts_v2.contact_id']]);
                    if (!empty($contact['contact_id'])) {
                        $aArgs['data'][$key]['value'] = $contact['contact_id'];
    
    Damien's avatar
    Damien committed
                    } else {
                        $aArgs['data'][$key]['value'] = 0;
                    }
                } elseif ($value['column'] == 'address_id' && !empty($value['value']) && !is_numeric($value['value'])) {
                    $mail = explode('<', str_replace('>', '', $value['value']));
    
                    $contact = ContactModel::getByEmail(['email' => $mail[count($mail) - 1], 'select' => ['contact_addresses.id']]);
                    if (!empty($contact['id'])) {
                        $aArgs['data'][$key]['value'] = $contact['ca_id'];
    
    Damien's avatar
    Damien committed
                    } else {
                        $aArgs['data'][$key]['value'] = 0;
                    }
                }
            }
    
            $formatedData = [];
            foreach ($aArgs['data'] as $value) {
                $formatedData[$value['column']] = $value['value'];
            }
            $formatedData['res_id'] = $aArgs['resId'];
    
            return $formatedData;
        }