/home/ivoiecob/email.hirewise-va.com/modules/Files/Module.php
<?php
/**
 * This code is licensed under AGPLv3 license or Afterlogic Software License
 * if commercial version of the product was purchased.
 * For full statements of the licenses see LICENSE-AFTERLOGIC and LICENSE-AGPL3 files.
 */

namespace Aurora\Modules\Files;

use Afterlogic\DAV\Constants;
use Afterlogic\DAV\Server;
use Aurora\Api;
use Aurora\Modules\Core\Models\User;
use Aurora\Modules\Core\Models\Tenant;
use Aurora\Modules\Core\Module as CoreModule;
use Aurora\Modules\Files\Enums\ErrorCodes;
use Aurora\Modules\Files\Models\FavoriteFile;
use Aurora\System\Classes\InheritedAttributes;
use Aurora\System\EventEmitter;
use Aurora\System\Exceptions\ApiException;
use Aurora\System\Utils;

/**
 * Main Files module. It provides PHP and Web APIs for managing files.
 *
 * @license https://www.gnu.org/licenses/agpl-3.0.html AGPL-3.0
 * @license https://afterlogic.com/products/common-licensing Afterlogic Software License
 * @copyright Copyright (c) 2023, Afterlogic Corp.
 *
 * @property Settings $oModuleSettings
 *
 * @package Modules
 */
class Module extends \Aurora\System\Module\AbstractModule
{
    protected static $sStorageType = '';

    /*
     * @var $oApiFileCache \Aurora\System\Managers\Filecache
     */
    public $oApiFileCache = null;

    /**
     *
     * @var \Aurora\Modules\Min\Module
     */
    protected $oMinModuleDecorator = null;

    public function getFilecacheManager()
    {
        if ($this->oApiFileCache === null) {
            $this->oApiFileCache = new \Aurora\System\Managers\Filecache();
        }

        return $this->oApiFileCache;
    }

    /**
     * @return Module
     */
    public static function getInstance()
    {
        return parent::getInstance();
    }

    /**
     * @return Module
     */
    public static function Decorator()
    {
        return parent::Decorator();
    }

    /**
     * @return Settings
     */
    public function getModuleSettings()
    {
        return $this->oModuleSettings;
    }


    /***** private functions *****/
    /**
     * Initializes Files Module.
     *
     * @ignore
     */
    public function init()
    {
        $this->subscribeEvent('Files::GetItems', array($this, 'onGetItems'));
        $this->subscribeEvent('Files::GetItems::after', array($this, 'onAfterGetItems'), 10000);
        $this->subscribeEvent('Files::GetStorages::after', array($this, 'onAfterGetStorages'), 1000);
        $this->subscribeEvent('Core::CreateUser::after', array($this, 'onAfterCreateUser'), 1000);
        $this->subscribeEvent('Files::Rename::after', array($this, 'onAfterRename'), 1000);
        $this->subscribeEvent('Files::Move::after', array($this, 'onAfterMove'), 1000);

        $this->AddEntries(
            array(
                'upload' => 'UploadFileData',
                'download-file' => 'EntryDownloadFile'
            )
        );
        $this->denyMethodsCallByWebApi(['getRawFile', 'getRawFileData', 'GetItems']);

        $this->aErrors = [
            Enums\ErrorCodes::NotFound	=> $this->i18N('INFO_NOTFOUND'),
            Enums\ErrorCodes::NotPermitted	=> $this->i18N('INFO_NOTPERMITTED'),
            Enums\ErrorCodes::AlreadeExists	=> $this->i18N('ERROR_ITEM_ALREADY_EXISTS'),
            Enums\ErrorCodes::CantDeleteSharedItem	=> $this->i18N('ERROR_CANNOT_DELETE_SHARED_ITEM'),
            Enums\ErrorCodes::CannotCopyOrMoveItemToItself	=> $this->i18N('ERROR_CANNOT_COPY_OR_MOVE_ITEM_TO_ITSELF'),
            Enums\ErrorCodes::NotPossibleToMoveSharedFileToCorporateStorage => $this->i18N('ERROR_NOT_POSSIBLE_TO_MOVE_SHARED_FILE_OR_DIR_TO_CORPORATE_STORAGE'),
        ];

        InheritedAttributes::addAttributes(User::class, ['Files::UserSpaceLimitMb']);
        InheritedAttributes::addAttributes(Tenant::class, ['Files::UserSpaceLimitMb', 'Files::TenantSpaceLimitMb']);
    }

    /**
    * Returns Min module decorator.
    *
    * @return \Aurora\Modules\Min\Module
    */
    private function getMinModuleDecorator()
    {
        return \Aurora\System\Api::GetModuleDecorator('Min');
    }

    /**
     * Checks if storage type is personal or corporate.
     *
     * @param string $Type Storage type.
     * @return bool
     */
    protected function checkStorageType($Type)
    {
        return $Type === static::$sStorageType;
    }

    public function getRawFileData($iUserId, $sType, $sPath, $sFileName, $SharedHash = null, $sAction = '', $iOffset = 0, $iChunkSize = 0)
    {
        $bDownload = true;
        $bThumbnail = false;
        $mResult = false;

        switch ($sAction) {
            case 'view':
                $bDownload = false;
                $bThumbnail = false;
                break;
            case 'thumb':
                $bDownload = false;
                $bThumbnail = true;
                break;
            case 'download':
                $bDownload = true;
                $bThumbnail = false;

                break;
            default:
                $bDownload = true;
                $bThumbnail = false;
                break;
        }
        if (!$bDownload || $iChunkSize == 0) {
            $iLength = -1;
            $iOffset = -1;
        } else {
            $iLength = $iChunkSize;
            $iOffset = $iChunkSize * $iOffset;
        }

        $oModuleDecorator = $this->getMinModuleDecorator();
        $mMin = ($oModuleDecorator && $SharedHash !== null) ? $oModuleDecorator->GetMinByHash($SharedHash) : array();

        $iUserId = (!empty($mMin['__hash__'])) ? $mMin['UserId'] : $iUserId;

        try {
            if ($iUserId && $SharedHash !== null) {
                \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::Anonymous);
                \Afterlogic\DAV\Server::setUser($iUserId);
            } else {
                \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
                if ($iUserId !== \Aurora\System\Api::getAuthenticatedUserId()) {
                    throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::AccessDenied);
                }
            }
        } catch (\Aurora\System\Exceptions\ApiException $oEx) {
            echo 'Access denied';
            exit();
        }

        if (isset($sType, $sPath, $sFileName)) {
            $sContentType = (empty($sFileName)) ? 'text/plain' : \MailSo\Base\Utils::MimeContentType($sFileName);

            $mResult = false;
            if ($bThumbnail) {
                $sRawKey = (string) \Aurora\System\Router::getItemByIndex(1, '');
                if (!empty($sRawKey)) {
                    \Aurora\System\Managers\Response::verifyCacheByKey($sRawKey);
                }
                $mResult = \Aurora\System\Managers\Thumb::GetResourceCache($iUserId, $sFileName);
            }
            if (!$mResult) {
                $aArgs = array(
                    'UserId' => $iUserId,
                    'Type' => $sType,
                    'Path' => $sPath,
                    'Name' => &$sFileName,
                    'Id' => $sFileName,
                    'IsThumb' => $bThumbnail,
                    'Offset' => $iOffset,
                    'ChunkSize' => $iChunkSize
                );
                $this->broadcastEvent(
                    'GetFile',
                    $aArgs,
                    $mResult
                );
            }
        }

        return $mResult;
    }

    /**
     * Downloads file, views file or makes thumbnail for file.
     *
     * @param int $iUserId User identifier.
     * @param string $sType Storage type - personal, corporate.
     * @param string $sPath Path to folder contained file.
     * @param string $sFileName File name.
     * @param string $SharedHash Indicates if file should be downloaded or viewed.
     * @param string $sAction Indicates if thumbnail should be created for file.
     *
     * @return void
     */
    public function getRawFile($iUserId, $sType, $sPath, $sFileName, $SharedHash = null, $sAction = '', $iOffset = 0, $iChunkSize = 0, $bShared = false)
    {
        // SVG files should not be viewed because they may contain JS
        if ($sAction !== 'download' && strtolower(pathinfo($sFileName, PATHINFO_EXTENSION)) === 'svg') {
            $this->oHttp->StatusHeader(403);
            exit();
        }

        $bDownload = true;
        $bThumbnail = false;

        switch ($sAction) {
            case 'view':
                $bDownload = false;
                $bThumbnail = false;
                break;
            case 'thumb':
                $bDownload = false;
                $bThumbnail = true;
                break;
            case 'download':
                $bDownload = true;
                $bThumbnail = false;

                break;
            default:
                $bDownload = true;
                $bThumbnail = false;
                break;
        }
        if (!$bDownload || $iChunkSize == 0) {
            $iLength = -1;
            $iOffset = -1;
        } else {
            $iLength = $iChunkSize;
            $iOffset = $iChunkSize * $iOffset;
        }

        $oModuleDecorator = $this->getMinModuleDecorator();
        $mMin = ($oModuleDecorator && $SharedHash !== null) ? $oModuleDecorator->GetMinByHash($SharedHash) : array();

        $iUserId = (!empty($mMin['__hash__'])) ? $mMin['UserId'] : $iUserId;

        try {
            if ($iUserId && $SharedHash !== null) {
                \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::Anonymous);
                \Afterlogic\DAV\Server::setUser($iUserId);
            } else {
                \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
                if ($iUserId !== \Aurora\System\Api::getAuthenticatedUserId()) {
                    throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::AccessDenied);
                }
            }
        } catch (\Aurora\System\Exceptions\ApiException $oEx) {
            //			echo(\Aurora\System\Managers\Response::GetJsonFromObject('Json', \Aurora\System\Managers\Response::FalseResponse(__METHOD__, 403, 'Access denied')));
            $this->oHttp->StatusHeader(403);
            exit;
        }

        if ($sType) {
            $sContentType = ($sFileName === '') ? 'text/plain' : \MailSo\Base\Utils::MimeContentType($sFileName);

            $mResult = false;
            if ($bThumbnail) {
                $sRawKey = (string) \Aurora\System\Router::getItemByIndex(1, '');
                if (!empty($sRawKey)) {
                    \Aurora\System\Managers\Response::verifyCacheByKey($sRawKey);
                }
                $mResult = \Aurora\System\Managers\Thumb::GetResourceCache($iUserId, $sFileName);
                if ($mResult) {
                    $sContentType = \MailSo\Base\Utils::MimeContentType($sFileName);
                    \Aurora\System\Managers\Response::OutputHeaders($bDownload, $sContentType, $sFileName);
                    echo $mResult;
                    exit();
                }
            }

            if (!$mResult) {
                $aArgs = array(
                    'UserId' => $iUserId,
                    'Type' => $sType,
                    'Path' => $sPath,
                    'Name' => &$sFileName,
                    'Id' => $sFileName,
                    'IsThumb' => $bThumbnail,
                    'Offset' => $iOffset,
                    'ChunkSize' => $iChunkSize,
                    'Shared' => $bShared
                );
                $this->broadcastEvent(
                    'GetFile',
                    $aArgs,
                    $mResult
                );
            }

            if (false !== $mResult) {
                if (is_resource($mResult)) {
                    $sFileName = \trim($sFileName, DIRECTORY_SEPARATOR);
                    $sContentType = \MailSo\Base\Utils::MimeContentType($sFileName);
                    \Aurora\System\Managers\Response::OutputHeaders($bDownload, $sContentType, $sFileName);

                    \header('Cache-Control: no-cache', true);

                    if ($bThumbnail) {
                        return \Aurora\System\Managers\Thumb::GetResource(
                            $iUserId,
                            $mResult,
                            $sFileName
                        );
                    } elseif ($sContentType === 'text/html' && !$bDownload) {
                        echo(\MailSo\Base\HtmlUtils::ClearHtmlSimple(stream_get_contents($mResult, $iLength, $iOffset)));
                    } elseif ($sContentType === 'text/plain') {
                        echo(stream_get_contents($mResult, $iLength, $iOffset));
                    } else {
                        $aMetaData = stream_get_meta_data($mResult);
                        $isSeekable = isset($aMetaData['seekable']) ? $aMetaData['seekable'] : false;
                        if ($iLength > -1 && $iOffset > -1 && $isSeekable) {
                            \MailSo\Base\Utils::GetFilePart($mResult, $iLength, $iOffset);
                        } else {
                            \MailSo\Base\Utils::FpassthruWithTimeLimitReset($mResult);
                        }
                    }

                    @fclose($mResult);
                } else {
                    //					echo(\Aurora\System\Managers\Response::GetJsonFromObject('Json', \Aurora\System\Managers\Response::FalseResponse(__METHOD__, 403, 'Not Found')));
                    $this->oHttp->StatusHeader(404);
                    exit;
                }
            }
        }
    }
    /***** private functions *****/

    /***** public functions *****/

    /***** public functions might be called with web API *****/
    /**
     * Uploads file from client side.
     *
     * echo string "true" or "false"
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function UploadFileData()
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        $mResult = false;
        $aPaths = \Aurora\System\Application::GetPaths();
        if (isset($aPaths[1]) && strtolower($aPaths[1]) === strtolower(self::GetName())) {
            $sType = isset($aPaths[2]) ? strtolower($aPaths[2]) : 'personal';
            $rData = fopen("php://input", "r");
            $aFilePath = array_slice($aPaths, 3);
            $sFilePath = urldecode(implode('/', $aFilePath));

            $bOverwrite = true;
            if (strpos($sFilePath, '!') === 0) {
                $sFilePath = substr($sFilePath, 1);
                $bOverwrite = false;
            }

            $iUserId = \Aurora\System\Api::getAuthenticatedUserId(
                \Aurora\System\Api::getAuthTokenFromHeaders()
            );
            $oUser = \Aurora\System\Api::getAuthenticatedUser($iUserId);
            if ($oUser) {
                if ($rData) {
                    $aArgs = array(
                        'UserId' => $oUser->UUID,
                        'Type' => $sType,
                        'Path' => dirname($sFilePath),
                        'Name' => basename($sFilePath),
                        'Data' => $rData,
                        'Overwrite' => $bOverwrite,
                        'RangeType' => 0,
                        'Offset' => 0,
                        'ExtendedProps' => array()
                    );
                    $this->broadcastEvent(
                        'CreateFile',
                        $aArgs,
                        $mResult
                    );
                } else {
                    $mResult = false;
                }
            } else {
                $mResult = false;
            }
        }
        if ($mResult) {
            echo 'true';
        } else {
            echo 'false';
        }
    }

    /**
     * @apiDefine Files Files Module
     * Main Files module. It provides PHP and Web APIs for managing files.
     */

    /**
     * @api {post} ?/Api/ GetSettings
     * @apiName GetSettings
     * @apiGroup Files
     * @apiDescription Obtains list of module settings for authenticated user.
     *
     * @apiHeader {string} [Authorization] "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetSettings} Method Method name
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'GetSettings'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name.
     * @apiSuccess {string} Result.Method Method name.
     * @apiSuccess {mixed} Result.Result List of module settings in case of success, otherwise **false**.
     * @apiSuccess {bool} Result.Result.EnableUploadSizeLimit=false Indicates if upload size limit is enabled.
     * @apiSuccess {int} Result.Result.UploadSizeLimitMb=0 Value of upload size limit in Mb.
     * @apiSuccess {string} Result.Result.CustomTabTitle=&quot;&quot; Custom tab title.
     * @apiSuccess {string} [Result.Result.PublicHash=&quot;&quot;] Public hash.
     * @apiSuccess {string} [Result.Result.PublicFolderName=&quot;&quot;] Public folder name.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetSettings',
     *	Result: { EnableUploadSizeLimit: true, UploadSizeLimitMb: 5,
     *		CustomTabTitle: "", PublicHash: "", PublicFolderName: "" }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetSettings',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */
    /**
     * Obtains list of module settings for authenticated user.
     *
     * @return array
     */
    public function GetSettings()
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::Anonymous);

        $iPostMaxSizeMb = Utils::getSizeFromIni('post_max_size') / 1024 / 1024;
        $iUploadMaxFilesizeMb = Utils::getSizeFromIni('upload_max_filesize') / 1024 / 1024;

        $aAppData = array(
            'EnableUploadSizeLimit' => $this->oModuleSettings->EnableUploadSizeLimit,
            'UploadSizeLimitMb' => min([$iPostMaxSizeMb, $iUploadMaxFilesizeMb, $this->oModuleSettings->UploadSizeLimitMb]),
            'UserSpaceLimitMb' => $this->oModuleSettings->UserSpaceLimitMb,
            'ShowUserSpaceLimit' => $this->oModuleSettings->ShowUserSpaceLimit,
            'TenantSpaceLimitMb' => $this->oModuleSettings->TenantSpaceLimitMb,
            'AllowTrash' => $this->oModuleSettings->AllowTrash,
            'AllowFavorites' => $this->oModuleSettings->AllowFavorites,
        );

        $oAuthenticatedUser = \Aurora\System\Api::getAuthenticatedUser();
        if ($oAuthenticatedUser instanceof User
                && ($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::NormalUser
                || $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::TenantAdmin
                || $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin)) {
            $aAppData['Storages'] = \Aurora\Modules\Files\Module::Decorator()->GetStorages();
        }

        $sPublicHash = \Aurora\System\Router::getItemByIndex(1);
        if (isset($sPublicHash)) {
            $aAppData['PublicHash'] = $sPublicHash;
            $oModuleDecorator = $this->getMinModuleDecorator();
            $mMin = ($oModuleDecorator && $sPublicHash !== null) ? $oModuleDecorator->GetMinByHash($sPublicHash) : array();
            if (isset($mMin['__hash__']) && $mMin['IsFolder']) {
                $aAppData['PublicFolderName'] = $mMin['Name'];
            }
        }
        return $aAppData;
    }

    public function GetSettingsForEntity($EntityType, $EntityId)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::Anonymous);

        $aResult = [];
        if ($EntityType === 'Tenant') {
            \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::TenantAdmin);
            $oTenant = \Aurora\Api::getTenantById($EntityId);
            if ($oTenant instanceof Tenant) {
                $aResult = [
                    'TenantSpaceLimitMb' => $oTenant->getExtendedProp(self::GetName() . '::TenantSpaceLimitMb'),
                    'UserSpaceLimitMb' => $oTenant->getExtendedProp(self::GetName() . '::UserSpaceLimitMb'),
                    'AllocatedSpace' => $this->GetAllocatedSpaceForUsersInTenant($oTenant->Id)
                ];
            }
        }
        if ($EntityType === 'User') {
            \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
            $oUser = \Aurora\Api::getUserById($EntityId);
            if ($oUser instanceof User) {
                $aResult = [
                    'UserSpaceLimitMb' => $oUser->getExtendedProp(self::GetName() . '::UserSpaceLimitMb'),
                ];
            }
        }

        return $aResult;
    }

    /**
     * @api {post} ?/Api/ UpdateSettings
     * @apiName UpdateSettings
     * @apiGroup Files
     * @apiDescription Updates module's settings - saves them to config.json file.
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=UpdateSettings} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **EnableUploadSizeLimit** *bool* Enable file upload size limit setting.<br>
     * &emsp; **UploadSizeLimitMb** *int* Upload file size limit setting in Mb.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'UpdateSettings',
     *	Parameters: '{ EnableUploadSizeLimit: true, UploadSizeLimitMb: 5 }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if settings were updated successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'UpdateSettings',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'UpdateSettings',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */
    /**
     * Updates module's settings - saves them to config.json file.
     *
     * @param bool $EnableUploadSizeLimit Enable file upload size limit setting.
     * @param int $UploadSizeLimitMb Upload file size limit setting in Mb.
     * @return bool
     */
    public function UpdateSettings($EnableUploadSizeLimit, $UploadSizeLimitMb)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::TenantAdmin);

        $this->setConfig('EnableUploadSizeLimit', $EnableUploadSizeLimit);
        $this->setConfig('UploadSizeLimitMb', $UploadSizeLimitMb);
        return (bool) $this->saveModuleConfig();
    }

    /**
     * @api {post} ?/Upload/ UploadFile
     * @apiDescription Uploads file from client side.
     * @apiName UploadFile
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=UploadFile} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Path** *string* Path to folder than should contain uploaded file.<br>
     * &emsp; **UploadData** *string* Uploaded file information. Contains fields size, name, tmp_name.<br>
     * &emsp; **SubPatha** *string* Relative path to subfolder where file should be uploaded.<br>
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result File object in case of success, otherwise **false**.
     * @apiSuccess {string} Result.Result.Name Original file name.
     * @apiSuccess {string} Result.Result.TempName Temporary file name.
     * @apiSuccess {string} Result.Result.MimeType Mime type of file.
     * @apiSuccess {int} Result.Result.Size File size.
     * @apiSuccess {string} Result.Result.Hash Hash used for file download, file view or getting file thumbnail.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'UploadFile',
     *	Result: { File: { Name: 'image.png', TempName: 'upload-post-6149f2cda5c58c6951658cce9f2b1378',
     *		MimeType: 'image/png', Size: 1813 } }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'UploadFile',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */
    /**
     * Uploads file from client side.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param string $Path Path to folder than should contain uploaded file.
     * @param array $UploadData Uploaded file information. Contains fields size, name, tmp_name.
     * @param string $SubPath Relative path to subfolder where file should be uploaded.
     * @param bool $Overwrite Overwrite a file if it already exists.
     * @param int $RangeType The type of update we're doing.
     * *	0 - overwrite
     * *	1 - append
     * *	2 - update based on a start byte
     * *	3 - update based on an end byte
     *;
     * @param int $Offset The start or end byte.
     * @param array $ExtendedProps Additional parameters.
     * @return array {
     *		*string* **Name** Original file name.
     *		*string* **TempName** Temporary file name.
     *		*string* **MimeType** Mime type of file.
     *		*int* **Size** File size.
     *		*string* **Hash** Hash used for file download, file view or getting file thumbnail.
     * }
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function UploadFile($UserId, $Type, $Path, $UploadData, $SubPath = '', $Overwrite = true, $RangeType = 0, $Offset = 0, $ExtendedProps = [])
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        $sUserPublicId = \Aurora\System\Api::getUserPublicIdById($UserId);

        $sError = '';
        $mResponse = array();

        if ($sUserPublicId) {
            if (is_array($UploadData)) {
                if (isset($ExtendedProps['FirstChunk']) && $RangeType == 1 && self::Decorator()->IsFileExists($UserId, $Type, $Path, $UploadData['name'])) {// It is forbidden to write first сhunk to the end of a existing file
                    $sError = \Aurora\System\Notifications::FileAlreadyExists;
                } elseif (!isset($ExtendedProps['FirstChunk']) && $RangeType == 1 && !self::Decorator()->IsFileExists($UserId, $Type, $Path, $UploadData['name'])) { // It is forbidden to write to the end of a nonexistent file
                    $sError = \Aurora\System\Notifications::FileNotFound;
                } else {
                    $iSize = (int) $UploadData['size'];
                    $iUploadSizeLimitMb = $this->oModuleSettings->UploadSizeLimitMb;
                    if ($iUploadSizeLimitMb > 0 && $iSize / (1024 * 1024) > $iUploadSizeLimitMb) {
                        throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::CanNotUploadFileLimit);
                    }

                    if (!self::Decorator()->CheckQuota($UserId, $Type, $iSize)) {
                        throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::CanNotUploadFileQuota);
                    }

                    if ($SubPath !== '' && !self::Decorator()->IsFileExists($UserId, $Type, $Path, $SubPath)) {
                        try {
                            self::Decorator()->CreateFolder($UserId, $Type, $Path, $SubPath);
                        } catch (\Exception $oException) {
                            // Cannot catch here \Sabre\DAV\Exception\Conflict
                            // Parallel uplod of files with creation of a subfolder may cause the conflict
                            if ($oException->getMessage() !== 'Can\'t create a directory') {
                                throw $oException;
                            }
                        }
                    }

                    $sUploadName = $UploadData['name'];
                    $sMimeType = \MailSo\Base\Utils::MimeContentType($sUploadName);

                    $sSavedName = 'upload-post-' . md5($UploadData['name'] . $UploadData['tmp_name']);
                    $rData = false;
                    if (\is_resource($UploadData['tmp_name'])) {
                        $rData = $UploadData['tmp_name'];
                    } elseif ($this->getFilecacheManager()->moveUploadedFile($sUserPublicId, $sSavedName, $UploadData['tmp_name'], '', self::GetName())) {
                        $rData = $this->getFilecacheManager()->getFile($sUserPublicId, $sSavedName, '', self::GetName());
                    }
                    if ($rData) {
                        $aArgs = array(
                            'UserId' => $UserId,
                            'Type' => $Type,
                            'Path' => $SubPath === '' ? $Path : $Path . '/' . $SubPath,
                            'Name' => $sUploadName,
                            'Data' => $rData,
                            'Overwrite' => $Overwrite,
                            'RangeType' => $RangeType,
                            'Offset' => $Offset,
                            'ExtendedProps' => $ExtendedProps
                        );
                        $mResult = false;
                        $this->broadcastEvent(
                            'CreateFile',
                            $aArgs,
                            $mResult
                        );

                        if ($mResult) {
                            $mResponse['File'] = array(
                                'Name' => $sUploadName,
                                'TempName' => $sSavedName,
                                'MimeType' => $sMimeType,
                                'Size' =>  (int) $iSize
                            );
                        } else {
                            $mResponse = false;
                        }
                    } else {
                        throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::CanNotUploadFileErrorData);
                    }
                }
            } else {
                throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::InvalidInputParameter);
            }
        } else {
            $sError = 'auth';
        }

        if (0 < strlen($sError)) {
            $mResponse['Error'] = $sError;
        }

        return $mResponse;
    }

    public function EntryDownloadFile()
    {
        // checkUserRoleIsAtLeast is called in getRawFile

        $sHash = (string) \Aurora\System\Router::getItemByIndex(1, '');
        $sAction = (string) \Aurora\System\Router::getItemByIndex(2, 'download');
        $iOffset = (int) \Aurora\System\Router::getItemByIndex(3, '');
        $iChunkSize = (int) \Aurora\System\Router::getItemByIndex(4, '');

        $aValues = \Aurora\System\Api::DecodeKeyValues($sHash);

        $iUserId = isset($aValues['UserId']) ? (int) $aValues['UserId'] : 0;
        $sType = isset($aValues['Type']) ? $aValues['Type'] : '';
        $sPath = isset($aValues['Path']) ? $aValues['Path'] : '';
        $sFileName = isset($aValues['Name']) ? $aValues['Name'] : '';
        $sPublicHash = isset($aValues['PublicHash']) ? $aValues['PublicHash'] : null;
        $bShared = isset($aValues['Shared']) ? $aValues['Shared'] : null;

        $this->getRawFile($iUserId, $sType, $sPath, $sFileName, $sPublicHash, $sAction, $iOffset, $iChunkSize, $bShared);
    }

    /**
     * @api {post} ?/Api/ ViewFile
     * @apiDescription Views file.
     * @apiName ViewFile
     * @apiGroup Files
     *
     * @apiHeader {string} [Authorization] "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=ViewFile} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Storage type - personal, corporate.<br>
     * &emsp; **Path** *string* Path to folder contained file.<br>
     * &emsp; **Name** *string* File name.<br>
     * &emsp; **SharedHash** *string* Shared hash.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'ViewFile',
     *	Parameters: '{ Type: "personal", Path: "", Name: "image.png" }'
     * }
     *
     * @apiSuccess {string} Result Content of the file with headers for view.
     */

    /**
     * Views file.
     *
     * @param int $UserId User identifier.
     * @param string $Type Storage type - personal, corporate.
     * @param string $Path Path to folder contained file.
     * @param string $Name File name.
     * @param string $SharedHash Shared hash.
     * @return void
     */
    public function ViewFile($UserId, $Type, $Path, $Name, $SharedHash)
    {
        // checkUserRoleIsAtLeast is called in getRawFile
        $this->getRawFile(
            \Aurora\System\Api::getUserPublicIdById($UserId),
            $Type,
            $Path,
            $Name,
            $SharedHash,
            false
        );
    }

    /**
     * @api {post} ?/Api/ GetFileThumbnail
     * @apiDescription Makes thumbnail for file.
     * @apiName GetFileThumbnail
     * @apiGroup Files
     *
     * @apiHeader {string} [Authorization] "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetFileThumbnail} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Storage type - personal, corporate.<br>
     * &emsp; **Path** *string* Path to folder contained file.<br>
     * &emsp; **Name** *string* File name.<br>
     * &emsp; **SharedHash** *string* Shared hash.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'GetFileThumbnail',
     *	Parameters: '{ Type: "personal", Path: "", Name: "image.png" }'
     * }
     *
     * @apiSuccess {string} Result Content of the file thumbnail with headers for view.
     */

    /**
     * Makes thumbnail for file.
     *
     * @param int $UserId User identifier.
     * @param string $Type Storage type - personal, corporate.
     * @param string $Path Path to folder contained file.
     * @param string $Name File name.
     * @param string $SharedHash Shared hash.
     * @return bool
     */
    public function GetFileThumbnail($UserId, $Type, $Path, $Name, $SharedHash)
    {
        return false;
        // checkUserRoleIsAtLeast is called in getRawFile
        // return \base64_encode(
        // 	$this->getRawFile(
        // 		\Aurora\System\Api::getUserPublicIdById($UserId),
        // 		$Type,
        // 		$Path,
        // 		$Name,
        // 		$SharedHash,
        // 		false,
        // 		true
        // 	)
        // );
    }

    /**
     * @api {post} ?/Api/ GetStorages
     * @apiDescription Returns storages available for logged in user.
     * @apiName GetStorages
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetStorages} Method Method name
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'GetStorages'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result List of storages in case of success, otherwise **false**.
     * @apiSuccess {string} Result.Result.Type Storage type - personal, corporate.
     * @apiSuccess {string} Result.Result.DisplayName Storage display name.
     * @apiSuccess {bool} Result.Result.IsExternal Indicates if storage external or not.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetStorages',
     *	Result: [{ Type: "personal", DisplayName: "Personal", IsExternal: false },
     *		{ Type: "corporate", DisplayName: "Corporate", IsExternal: false },
     *		{ Type: "google", IsExternal: true, DisplayName: "GoogleDrive" }]
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetStorages',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Returns storages available for logged in user.
     *
     * @return array {
     *		*string* **Type** Storage type - personal, corporate.
     *		*string* **DisplayName** Storage display name.
     *		*bool* **IsExternal** Indicates if storage external or not.
     * }
     */
    public function GetStorages()
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return [];
    }

    /**
     * Returns submodules.
     */
    public function GetSubModules()
    {
        return [];
    }

    /**
     * @api {post} ?/Api/ GetQuota
     * @apiDescription Returns used space and space limit for specified user.
     * @apiName GetQuota
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetQuota} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **UserId** *int* User identifier.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'UpdateAccount',
     *	Parameters: '{ UserId: 123 }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result Object in case of success, otherwise **false**.
     * @apiSuccess {int} Result.Result.Used Amount of space used by user.
     * @apiSuccess {int} Result.Result.Limit Limit of space for user.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetQuota',
     *	Result: { Used: 21921, Limit: 62914560 }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetQuota',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */
    /**
     * Returns used space and space limit for specified user.
     *
     * @param int $UserId User identifier.
     * @return array {
     *		*int* **Used** Amount of space used by user.
     *		*int* **Limit** Limit of space for user.
     * }
     */
    public function GetQuota($UserId, $Type)
    {
        return [
            'Limit' => 0,
            'Used' => 0
        ];
    }

    public function CheckQuota($UserId, $Type, $Size)
    {
        return false;
    }

    public function GetItems($UserId, $Type, $Path, $Pattern, $PublicHash = null, $Shared = false)
    {
        $aArgs = [
            'UserId' => $UserId,
            'Type' => $Type,
            'Path' => $Path,
            'Pattern' => $Pattern,
            'PublicHash' => $PublicHash,
            'Shared' => $Shared
        ];
        $mResult = [];

        $this->broadcastEvent('GetItems', $aArgs, $mResult);

        $aItems = [];
        if (is_array($mResult)) {
            foreach ($mResult as $oItem) {
                if ($oItem instanceof Classes\FileItem) {
                    $aItems[] = self::Decorator()->PopulateFileItem($aArgs['UserId'], $oItem);
                }
            }

            $mResult = $aItems;
        }

        return $aItems;
    }

    /**
     * @api {post} ?/Api/ GetFiles
     * @apiDescription Returns file list and user quota information.
     * @apiName GetFiles

     * 	 * @apiGroup Files
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetFiles} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage.<br>
     * &emsp; **Path** *string* Path to folder files are obtained from.<br>
     * &emsp; **Pattern** *string* String for search files and folders with such string in name.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'GetFiles',
     *	Parameters: '{ Type: "personal", Path: "", Pattern: "" }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result Object in case of success, otherwise **false**.
     * @apiSuccess {array} Result.Result.Items Array of files objects.
     * @apiSuccess {array} Result.Result.Quota Array of items with fields Used, Limit.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetFiles',
     *	Result: { Items: [{ Id: "image.png", Type: "personal", Path: "", FullPath: "/image.png",
     * Name: "image.png", Size: 1813, IsFolder: false, IsLink: false, LinkType: "", LinkUrl: "",
     * LastModified: 1475498855, ContentType: "image/png", Thumb: true, ThumbnailLink: "", OembedHtml: "",
     * Shared: false, Owner: "", Content: "", IsExternal: false }], Quota: { Used: 21921, Limit: 62914560 } }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetFiles',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Returns file list and user quota information.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage.
     * @param string $Path Path to folder files are obtained from.
     * @param string $Pattern String for search files and folders with such string in name.
     * @return array {
     *		*array* **Items** Array of files objects.
     *		*array* **Quota** Array of items with fields Used, Limit.
     * }
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function GetFiles($UserId, $Type, $Path, $Pattern, $Shared = false)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return [
            'Items' => self::Decorator()->GetItems($UserId, $Type, $Path, $Pattern, null, $Shared),
            'Quota' => self::Decorator()->GetQuota($UserId, $Type)
        ];
    }

    /**
     *
     * @param array $aArgs
     * @param mixed $mResult
     */
    public function onAfterGetStorages($aArgs, &$mResult)
    {
        if (is_array($mResult)) {
            \usort($mResult, function ($aItem1, $aItem2) {
                $aItem1['Order'] = isset($aItem1['Order']) ? $aItem1['Order'] : 1000;
                $aItem2['Order'] = isset($aItem2['Order']) ? $aItem2['Order'] : 1000;

                return ($aItem1['Order'] == $aItem2['Order']) ? 0 : ($aItem1['Order'] > $aItem2['Order'] ? +1 : -1);
            });
        }
    }

    /**
     *
     * @param int $iUserId
     * @param int $sType
     * @param string $sPath
     * @param string $sName
     * @param int $sSize
     *
     * @return array
     */
    private function generateMinArray($iUserId, $sType, $sPath, $sName, $sSize, $bIsFolder = false)
    {
        $aData = null;
        if ($iUserId) {
            $aData = array(
                'UserId' => $iUserId,
                'Type' => $sType,
                'Path' => $sPath,
                'Name' => $sName,
                'Size' => $sSize,
                'IsFolder' => $bIsFolder
            );
        }

        return $aData;
    }

    protected function updateMinHash($iUserId, $sType, $sPath, $sName, $sNewType, $sNewPath, $sNewName, $bIsFolder)
    {
        $sUserPublicId = \Aurora\Api::getUserPublicIdById($iUserId);
        $sID = \Aurora\Modules\Min\Module::generateHashId([$sUserPublicId, $sType, $sPath, $sName]);
        $sNewID = \Aurora\Modules\Min\Module::generateHashId([$sUserPublicId, $sNewType, $sNewPath, $sNewName]);

        $mData = $this->getMinModuleDecorator()->GetMinByID($sID);

        if ($mData) {
            $aData = $this->generateMinArray($sUserPublicId, $sNewType, $sNewPath, $sNewName, $mData['Size'], $bIsFolder);
            if ($aData) {
                $this->getMinModuleDecorator()->UpdateMinByID($sID, $aData, $sNewID);
            }
        }
    }

    /**
     *
     * @param array $aArgs
     * @param mixed $mResult
     */
    public function onAfterRename($aArgs, &$mResult)
    {
        if ($mResult && isset($aArgs['UserId'])) {
            $this->updateMinHash($aArgs['UserId'], $aArgs['Type'], $aArgs['Path'], $aArgs['Name'], $aArgs['Type'], $aArgs['Path'], $aArgs['NewName'], $aArgs['IsFolder']);
        }
    }

    public function onAfterMove($aArgs, &$mResult)
    {
        if ($mResult && isset($aArgs['Files']) && is_array($aArgs['Files']) && count($aArgs['Files']) > 0) {
            foreach ($aArgs['Files'] as $aFile) {
                $this->updateMinHash(
                    $aArgs['UserId'],
                    $aFile['FromType'],
                    $aFile['FromPath'],
                    $aFile['Name'],
                    $aArgs['ToType'],
                    $aArgs['ToPath'],
                    $aFile['NewName'],
                    $aFile['IsFolder']
                );
            }
        }
    }

    /**
     *
     * @param array $aArgs
     * @param mixed $mResult
     */
    public function onAfterCreateUser($aArgs, &$mResult)
    {
        if ($mResult) {
            $oUser = \Aurora\Api::getUserById($mResult);
            if ($oUser) {
                $oTenant = \Aurora\Api::getTenantById($oUser->IdTenant);
                $oUser->setExtendedProp($this->GetName() . '::UserSpaceLimitMb', $oTenant->getExtendedProp($this->GetName() . '::UserSpaceLimitMb'));
                $oUser->save();
            }
        }
    }

    /**
     * @ignore
     * @param array $aArgs Arguments of event.
     * @param mixed $mResult Is passed by reference.
     */
    public function onGetItems($aArgs, &$mResult)
    {
        if ($aArgs['Type'] === 'favorites') {
            $UserId = $aArgs['UserId'];
            Api::CheckAccess($UserId);

            $sUserPiblicId = Api::getUserPublicIdById($UserId);

            $files = [];
            $favorites = $this->GetFavorites($UserId);
            foreach ($favorites as $favorite) {
                list($sPath, $sName) = \Sabre\Uri\split($favorite['FullPath']);
                $file = \Aurora\Modules\PersonalFiles\Module::getInstance()->getManager()->getFileInfo($sUserPiblicId, $favorite['Type'], $sPath, $sName);
                if ($file) {
                    $file->Name = $favorite['DisplayName'];
                    $files[] = $file;
                }
            }

            $mResult = array_merge(
                $mResult,
                $files
            );
        }
    }

    /**
     * @ignore
     * @param array $aArgs Arguments of event.
     * @param mixed $mResult Is passed by reference.
     */
    public function onAfterGetItems($aArgs, &$mResult)
    {
        if (is_array($mResult) && count($mResult) > 0) {
            $query = FavoriteFile::where('IdUser', $aArgs['UserId'])
                ->where(function ($subQuery) use ($mResult) {
                    foreach ($mResult as $item) {
                        $subQuery->orWhere(function ($q) use ($item) {
                            $q->where('Type', $item->TypeStr)
                            ->where('FullPath', $item->FullPath);
                        });
                    }
                });

            $favorites = $query->get()->map(function ($item) {
                return $item->Type . $item->FullPath;
            });

            foreach ($mResult as $item) {
                $item->IsFavorite = in_array($item->TypeStr . $item->FullPath, $favorites->toArray());
            }
        }
    }

    /**
     *
     * @param array $UserId
     * @param mixed $Item
     *
     * @return mixed
     */
    public function PopulateFileItem($UserId, $Item)
    {
        return $Item;
    }

    /**
     *
     * @param int $UserId
     * @param mixed $Items
     *
     * @return mixed
     */
    public function PopulateFileItems($UserId, $Items)
    {
        return $Items;
    }

    /**
     * Return content of a file.
     *
     * @param int $UserId
     * @param string $Type
     * @param string $Path
     * @param string $Name
     */
    public function GetFileContent($UserId, $Type, $Path, $Name)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        // File content is obtained in subscribers methods
    }

    /**
     * Return information about file. Subscribers of "Files::GetFileInfo::after" event are used for collecting information.
     *
     * @param int $UserId
     * @param string $Type
     * @param string $Path
     * @param string $Id
     * @return \Aurora\Modules\Files\Classes\FileItem
     */
    public function GetFileInfo($UserId, $Type, $Path, $Id)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        return null;
    }

    /**
     * @api {post} ?/Api/ GetPublicFiles
     * @apiDescription Returns list of public files.
     * @apiName GetPublicFiles
     * @apiGroup Files
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=GetPublicFiles} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Hash** *string* Hash to identify the list of files to return. Containes information about user identifier, type of storage, path to public folder, name of public folder.<br>
     * &emsp; **Path** *string* Path to folder contained files to return.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'GetPublicFiles',
     *	Parameters: '{ Hash: "hash_value", Path: "" }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result Object in case of success, otherwise **false**.
     * @apiSuccess {array} Result.Result.Items Array of files objects.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetPublicFiles',
     *	Result: { Items: [{ Id: "image.png", Type: "personal", Path: "/shared_folder",
     * FullPath: "/shared_folder/image.png", Name: "image.png", Size: 43549, IsFolder: false,
     * IsLink: false, LinkType: "", LinkUrl: "", LastModified: 1475500277, ContentType: "image/png",
     * Thumb: true, ThumbnailLink: "", OembedHtml: "", Shared: false, Owner: "62a6d548-892e-11e6-be21-0cc47a041d39",
     * Content: "", IsExternal: false }] }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'GetPublicFiles',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Returns list of public files.
     *
     * @param string $Hash Hash to identify the list of files to return. Containes information about user identifier, type of storage, path to public folder, name of public folder.
     * @param string $Path Path to folder contained files to return.
     * @return array {
     *		*array* **Items** Array of files objects.
     *		*array* **Quota** Array of items with fields Used, Limit.
     * }
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function GetPublicFiles($Hash, $Path)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::Anonymous);

        $mResult = [];

        $oMinDecorator =  $this->getMinModuleDecorator();
        if ($oMinDecorator) {
            $mMin = $oMinDecorator->GetMinByHash($Hash);
            if (!empty($mMin['__hash__'])) {
                $sUserPublicId = $mMin['UserId'];
                if ($sUserPublicId) {
                    $oUser = CoreModule::Decorator()->GetUserByPublicId($sUserPublicId);
                    if ($oUser) {
                        $bPrevState = \Aurora\System\Api::skipCheckUserRole(true);
                        $sMinPath = implode('/', array($mMin['Path'], $mMin['Name']));
                        $mPos = strpos($Path, $sMinPath);
                        if ($mPos === 0 || $Path === '') {
                            if ($mPos !== 0) {
                                $Path =  $sMinPath . $Path;
                            }
                            $Path = str_replace('.', '', $Path);
                            $mResult = [
                                'Items' => self::Decorator()->GetItems($oUser->Id, $mMin['Type'], $Path, '', $Hash)
                            ];
                        }
                        \Aurora\System\Api::skipCheckUserRole($bPrevState);
                    }
                }
            }
        }

        return $mResult;
    }

    /**
     * @api {post} ?/Api/ CreateFolder
     * @apiDescription Creates folder.
     * @apiName CreateFolder
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=CreateFolder} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Path** *string* Path to new folder.<br>
     * &emsp; **FolderName** *string* New folder name.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateFolder',
     *	Parameters: '{ Type: "personal", Path: "", FolderName: "new_folder" }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if folder was created successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateFolder',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateFolder',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Creates folder.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param string $Path Path to new folder.
     * @param string $FolderName New folder name.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function CreateFolder($UserId, $Type, $Path, $FolderName)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        return false;
    }

    /**
     * @api {post} ?/Api/ CreateLink
     * @apiDescription Creates link.
     * @apiName CreateLink
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=CreateLink} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Path** *string* Path to new link.<br>
     * &emsp; **Link** *string* Link value.<br>
     * &emsp; **Name** *string* Link name.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateLink',
     *	Parameters: '{ Type: "personal", Path: "", Link: "link_value", Name: "name_value" }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result Link object in case of success, otherwise **false**.
     * @apiSuccess {string} Result.Result.Type Type of storage.
     * @apiSuccess {string} Result.Result.Path Path to link.
     * @apiSuccess {string} Result.Result.Link Link URL.
     * @apiSuccess {string} Result.Result.Name Link name.
     * @apiSuccess {int} [Result.ErrorCode] Error code.
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateLink',
     *	Result: { Type: "personal", Path: "", Link: "https://www.youtube.com/watch?v=1WPn4NdQnlg&t=1124s",
     *		Name: "Endless Numbers counting 90 to 100 - Learn 123 Numbers for Kids" }
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreateLink',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Creates link.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param string $Path Path to new link.
     * @param string $Link Link value.
     * @param string $Name Link name.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function CreateLink($UserId, $Type, $Path, $Link, $Name)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return false;
    }

    /**
     * @api {post} ?/Api/ Delete
     * @apiDescription Deletes files and folder specified with list.
     * @apiName Delete
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=Delete} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Items** *array* Array of items to delete.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'Delete',
     *	Parameters: '{ Type: "personal", Items: [{ "Path": "", "Name": "2.png" },
     *		{ "Path": "", "Name": "logo.png" }] }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if files and (or) folders were deleted successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'Delete',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'Delete',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Deletes files and folder specified with list.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param array $Items Array of items to delete.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function Delete($UserId, $Type, $Items)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        return false;
    }

    /**
     * Restore files and folder specified with list from Trash.
     *
     * @param int $UserId User identifier.
     * @param array $Items Array of items to delete.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function Restore($UserId, $Items)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        return false;
    }

    /**
     * @api {post} ?/Api/ LeaveShare
     * @apiDescription leave shared files and folder specified with list.
     * @apiName LeaveShare
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=LeaveShare} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Items** *array* Array of items to leave share.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'LeaveShare',
     *	Parameters: '{ Type: "personal", Items: [{ "Path": "", "Name": "2.png" },
     *		{ "Path": "", "Name": "logo.png" }] }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if files and (or) folders were leave share successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'LeaveShare',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'LeaveShare',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Leave shared files and folder specified with list.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param array $Items Array of items to leave.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function LeaveShare($UserId, $Type, $Items)
    {
        $aArgs = [
            'UserId' => $UserId,
            'Type' => $Type,
            'Items' => $Items
        ];
        EventEmitter::getInstance()->emit('Files', 'Delete::before', $aArgs);

        $UserId = $aArgs['UserId'];
        $Type = $aArgs['Type'];
        $Items = $aArgs['Items'];

        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        $aNodes = [];
        $aItems = [];
        foreach ($Items as $aItem) {
            try {
                $oNode = Server::getNodeForPath(Constants::FILESTORAGE_PATH_ROOT . '/' . $Type . '/' . $aItem['Path'] . '/' . $aItem['Name']);
                $aItems[] = $oNode;
            } catch (\Exception $oEx) {
                Api::LogException($oEx);
                throw new ApiException(ErrorCodes::NotFound);
            }

            if (!($oNode instanceof \Afterlogic\DAV\FS\Shared\File || $oNode instanceof \Afterlogic\DAV\FS\Shared\Directory)) {
                throw new ApiException(ErrorCodes::CantDeleteSharedItem);
            }

            $oItem = new Classes\FileItem();
            $oItem->Id = $aItem['Name'];
            $oItem->Name = $aItem['Name'];
            $oItem->TypeStr = $Type;
            $oItem->Path = $aItem['Path'];

            self::Decorator()->DeletePublicLink($UserId, $Type, $aItem['Path'], $aItem['Name']);
            \Aurora\System\Managers\Thumb::RemoveFromCache($UserId, $oItem->getHash(), $aItem['Name']);
        }

        $aArgs = [
            'UserId' => $UserId,
            'Type' => $Type,
            'Items' => $aItems
        ];
        $mResult = false;

        EventEmitter::getInstance()->emit('Files', 'LeaveShare', $aArgs, $mResult);

        return $mResult;
    }

    /**
     * @api {post} ?/Api/ Rename
     * @apiDescription Renames folder, file or link.
     * @apiName Rename
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=Rename} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage - personal, corporate.<br>
     * &emsp; **Path** *string* Path to item to rename.<br>
     * &emsp; **Name** *string* Current name of the item.<br>
     * &emsp; **NewName** *string* New name of the item.<br>
     * &emsp; **IsLink** *bool* Indicates if the item is link or not.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'Rename',
     *	Parameters: '{ Type: "personal", Path: "", Name: "old_name.png", NewName: "new_name.png",
     *		IsLink: false }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if file or folder was renamed successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'Rename',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'Rename',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Renames folder, file or link.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage - personal, corporate.
     * @param string $Path Path to item to rename.
     * @param string $Name Current name of the item.
     * @param string $NewName New name of the item.
     * @param bool $IsLink Indicates if the item is link or not.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function Rename($UserId, $Type, $Path, $Name, $NewName, $IsLink)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        if ($Name === '') {
            throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::InvalidInputParameter);
        }

        $oItem = new Classes\FileItem();
        $oItem->Id = $Name;
        $oItem->Name = $Name;
        $oItem->TypeStr = $Type;
        $oItem->Path = $Path;

        \Aurora\System\Managers\Thumb::RemoveFromCache($UserId, $oItem->getHash(), $Name);
        // Actual renaming is proceeded in subscribed methods. Look for it by "Files::Rename::after"

        return false;
    }

    /**
     * @api {post} ?/Api/ Copy
     * @apiDescription Copies files and/or folders from one folder to another.
     * @apiName Copy
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=Copy} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **FromType** *string* Storage type of folder items will be copied from.<br>
     * &emsp; **ToType** *string* Storage type of folder items will be copied to.<br>
     * &emsp; **FromPath** *string* Folder items will be copied from.<br>
     * &emsp; **ToPath** *string* Folder items will be copied to.<br>
     * &emsp; **Files** *array* List of items to copy<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'Copy',
     *	Parameters: '{ FromType: "personal", ToType: "corporate", FromPath: "", ToPath: "",
     * Files: [{ Name: "logo.png", IsFolder: false }, { Name: "details.png", IsFolder: false }] }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if files and (or) folders were copied successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'Copy',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'Copy',
     *		Result: false,
     *		ErrorCode: 102
     *	}]
     * }
     */

    /**
     * Copies files and/or folders from one folder to another.
     *
     * @param int $UserId User identifier.
     * @param string $FromType storage type of folder items will be copied from.
     * @param string $ToType storage type of folder items will be copied to.
     * @param string $FromPath folder items will be copied from.
     * @param string $ToPath folder items will be copied to.
     * @param array $Files list of items to copy {
     *		*string* **Name** Name of item to copy.
     *		*bool* **IsFolder** Indicates if the item to copy is folder or not.
     * }
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function Copy($UserId, $FromType, $ToType, $FromPath, $ToPath, $Files)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return null;
    }

    /**
     * @api {post} ?/Api/ Move
     * @apiDescription Moves files and/or folders from one folder to another.
     * @apiName Move
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=Move} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **FromType** *string* Storage type of folder items will be moved from.<br>
     * &emsp; **ToType** *string* Storage type of folder items will be moved to.<br>
     * &emsp; **FromPath** *string* Folder items will be moved from.<br>
     * &emsp; **ToPath** *string* Folder items will be moved to.<br>
     * &emsp; **Files** *array* List of items to move<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'Move',
     *	Parameters: '{ FromType: "personal", ToType: "corporate", FromPath: "", ToPath: "",
     *		Files: [{ "Name": "logo.png", "IsFolder": false },
     *		{ "Name": "details.png", "IsFolder": false }] }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicates if files and (or) folders were moved successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'Move',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'Move',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Moves files and/or folders from one folder to another.
     *
     * @param int $UserId User identifier.
     * @param string $FromType storage type of folder items will be moved from.
     * @param string $ToType storage type of folder items will be moved to.
     * @param string $FromPath folder items will be moved from.
     * @param string $ToPath folder items will be moved to.
     * @param array $Files list of items to move {
     *		*string* **Name** Name of item to copy.
     *		*bool* **IsFolder** Indicates if the item to copy is folder or not.
     * }
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function Move($UserId, $FromType, $ToType, $FromPath, $ToPath, $Files)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        foreach ($Files as $aFile) {
            if (!$aFile['IsFolder']) {
                $oItem = new Classes\FileItem();
                $oItem->Id = $aFile['Name'];
                $oItem->Name = $aFile['Name'];
                $oItem->TypeStr = $FromType;
                $oItem->Path = $FromPath;

                \Aurora\System\Managers\Thumb::RemoveFromCache($UserId, $oItem->getHash(), $aFile['Name']);
            }
        }

        return false;
    }

    /**
     * @api {post} ?/Api/ CreatePublicLink
     * @apiDescription Creates public link for file or folder.
     * @apiName CreatePublicLink
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=CreatePublicLink} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage contains the item.<br>
     * &emsp; **Path** *string* Path to the item.<br>
     * &emsp; **Name** *string* Name of the item.<br>
     * &emsp; **Size** *int* Size of the file.<br>
     * &emsp; **IsFolder** *bool* Indicates if the item is folder or not.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'CreatePublicLink',
     *	Parameters: '{ Type: "personal", Path: "", Name: "image.png", Size: 100, "IsFolder": false }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {mixed} Result.Result Public link to the item in case of success, otherwise **false**.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreatePublicLink',
     *	Result: 'AppUrl/?/files-pub/shared_item_hash/list'
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'CreatePublicLink',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Creates public link for file or folder.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage contains the item.
     * @param string $Path Path to the item.
     * @param string $Name Name of the item.
     * @param int $Size Size of the file.
     * @param bool $IsFolder Indicates if the item is folder or not.
     * @return string|false Public link to the item.
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function CreatePublicLink($UserId, $Type, $Path, $Name, $Size, $IsFolder)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return false;
    }

    /**
     * @api {post} ?/Api/ DeletePublicLink
     * @apiDescription Deletes public link from file or folder.
     * @apiName DeletePublicLink
     * @apiGroup Files
     *
     * @apiHeader {string} Authorization "Bearer " + Authentication token which was received as the result of Core.Login method.
     * @apiHeaderExample {json} Header-Example:
     *	{
     *		"Authorization": "Bearer 32b2ecd4a4016fedc4abee880425b6b8"
     *	}
     *
     * @apiParam {string=Files} Module Module name
     * @apiParam {string=DeletePublicLink} Method Method name
     * @apiParam {string} Parameters JSON.stringified object <br>
     * {<br>
     * &emsp; **Type** *string* Type of storage contains the item.<br>
     * &emsp; **Path** *string* Path to the item.<br>
     * &emsp; **Name** *string* Name of the item.<br>
     * }
     *
     * @apiParamExample {json} Request-Example:
     * {
     *	Module: 'Files',
     *	Method: 'DeletePublicLink',
     *	Parameters: '{ Type: "personal", Path: "", Name: "image.png" }'
     * }
     *
     * @apiSuccess {object[]} Result Array of response objects.
     * @apiSuccess {string} Result.Module Module name
     * @apiSuccess {string} Result.Method Method name
     * @apiSuccess {bool} Result.Result Indicated if public link was deleted successfully.
     * @apiSuccess {int} [Result.ErrorCode] Error code
     *
     * @apiSuccessExample {json} Success response example:
     * {
     *	Module: 'Files',
     *	Method: 'DeletePublicLink',
     *	Result: true
     * }
     *
     * @apiSuccessExample {json} Error response example:
     * {
     *	Module: 'Files',
     *	Method: 'DeletePublicLink',
     *	Result: false,
     *	ErrorCode: 102
     * }
     */

    /**
     * Deletes public link from file or folder.
     *
     * @param int $UserId User identifier.
     * @param string $Type Type of storage contains the item.
     * @param string $Path Path to the item.
     * @param string $Name Name of the item.
     * @return bool
     * @throws \Aurora\System\Exceptions\ApiException
     */
    public function DeletePublicLink($UserId, $Type, $Path, $Name)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return false;
    }

    /**
     * Checks URL and returns information about it.
     *
     * @param string $Url URL to check.
     * @return array|bool {
     *		Name
     *		Thumb
     *		Size
     *		LinkType
     * }
     */
    public function CheckUrl($Url)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        $mResult = false;

        if (substr($Url, 0, 11) === 'javascript:') {
            $Url = substr($Url, 11);
        }

        $aArgs = array(
            'Url' => $Url
        );
        $this->broadcastEvent(
            'CheckUrl',
            $aArgs,
            $mResult
        );

        return $mResult;
    }

    /**
     * @return array
     */
    public function GetFilesForUpload($UserId, $Hashes = array())
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        $sUUID = \Aurora\System\Api::getUserUUIDById($UserId);

        $mResult = false;
        if (is_array($Hashes) && 0 < count($Hashes)) {
            $mResult = array();
            foreach ($Hashes as $sHash) {
                $aData = \Aurora\System\Api::DecodeKeyValues($sHash);
                if (\is_array($aData) && 0 < \count($aData)) {
                    $oFileInfo = self::Decorator()->GetFileInfo($UserId, $aData['Type'], $aData['Path'], $aData['Id']);

                    $aArgs = array(
                        'UserId' => $UserId,
                        'Type' => $aData['Type'],
                        'Path' => $aData['Path'],
                        'Name' => $aData['Name'],
                        'Id' => $aData['Id']
                    );
                    $rFile = false;
                    $this->broadcastEvent(
                        'GetFile',
                        $aArgs,
                        $rFile
                    );

                    $sTempName = md5('Files/Tmp/' . $aData['Type'] . $aData['Path'] . $aData['Name'] . microtime(true) . rand(1000, 9999));

                    if (is_resource($rFile) && $this->getFilecacheManager()->putFile($sUUID, $sTempName, $rFile)) {
                        $aItem = array(
                            'Name' => $oFileInfo->Name,
                            'TempName' => $sTempName,
                            'Size' => $oFileInfo->Size,
                            'Hash' => $sHash,
                            'MimeType' => ''
                        );

                        $aItem['MimeType'] = \MailSo\Base\Utils::MimeContentType($aItem['Name']);
                        $aItem['NewHash'] = \Aurora\System\Api::EncodeKeyValues(array(
                            'TempFile' => true,
                            'UserId' => $UserId,
                            'Name' => $aItem['Name'],
                            'TempName' => $sTempName
                        ));

                        $aActions = array(
                            'view' => array(
                                'url' => '?file-cache/' . $aItem['NewHash'] . '/view'
                            ),
                            'download' => array(
                                'url' => '?file-cache/' . $aItem['NewHash']
                            )
                        );
                        $aItem['Actions'] = $aActions;

                        $mResult[] = $aItem;

                        if (is_resource($rFile)) {
                            @fclose($rFile);
                        }
                    }
                }
            }
        } else {
            throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::InvalidInputParameter);
        }

        return $mResult;
    }

    /**
     * Checks if file exists.
     *
     * @param int $UserId
     * @param string $Type
     * @param string $Path
     * @param string $Name
     */
    public function IsFileExists($UserId, $Type, $Path, $Name)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return true;
    }

    /**
     *
     * @param int $UserId
     * @param string $Type
     * @param string $Path
     * @param string $Name
     */
    public function GetNonExistentFileName($UserId, $Type, $Path, $Name, $WithoutGroup = false)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);
        return $Name;
    }

    /**
     *
     * @param int $UserId
     * @param array $Files
     */
    public function SaveFilesAsTempFiles($UserId, $Files)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        $mResult = false;

        if (is_array($Files) && count($Files) > 0) {
            $mResult = array();
            foreach ($Files as $aFile) {
                $Storage = $aFile['Storage'];
                $Path = $aFile['Path'];
                $Name = $aFile['Name'];
                $Id = $aFile['Name'];

                $aArgs = array(
                    'UserId' => $UserId,
                    'Type' => $Storage,
                    'Path' => $Path,
                    'Name' => &$Name,
                    'Id' => $Id,
                    'IsThumb' => false,
                    'Offset' => 0,
                    'ChunkSize' => 0
                );
                $mFileResource = false;
                $this->broadcastEvent(
                    'GetFile',
                    $aArgs,
                    $mFileResource
                );

                if (is_resource($mFileResource)) {
                    $sUUID = \Aurora\System\Api::getUserUUIDById($UserId);
                    try {
                        $sTempName = md5($sUUID . $Storage . $Path . $Name);

                        if (!$this->getFilecacheManager()->isFileExists($sUUID, $sTempName)) {
                            $this->getFilecacheManager()->putFile($sUUID, $sTempName, $mFileResource);
                        }

                        if ($this->getFilecacheManager()->isFileExists($sUUID, $sTempName)) {
                            $mResult[] = \Aurora\System\Utils::GetClientFileResponse(
                                null,
                                $UserId,
                                $Name,
                                $sTempName,
                                $this->getFilecacheManager()->fileSize($sUUID, $sTempName)
                            );
                        }
                    } catch (\Exception $oException) {
                        throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::FilesNotAllowed, $oException);
                    }
                }
            }
        }

        return $mResult;
    }

    public function UpdateSettingsForEntity($EntityType, $EntityId, $UserSpaceLimitMb, $TenantSpaceLimitMb)
    {
        $bResult = false;
        $oAuthenticatedUser = \Aurora\System\Api::getAuthenticatedUser();

        if ($EntityType === '') {
            \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::SuperAdmin);
            $this->setConfig('TenantSpaceLimitMb', $TenantSpaceLimitMb);
            $this->setConfig('UserSpaceLimitMb', $UserSpaceLimitMb);
            return $this->saveModuleConfig();
        }
        if ($EntityType === 'Tenant') {
            \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::TenantAdmin);
            $oTenant = \Aurora\Api::getTenantById($EntityId);

            if ($oTenant instanceof Tenant
                    && $oAuthenticatedUser instanceof User
                    && (($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::TenantAdmin && $oTenant->Id === $oAuthenticatedUser->IdTenant)
                    || $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin)) {
                if ($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin) {
                    $oTenant->setExtendedProp(self::GetName() . '::TenantSpaceLimitMb', $TenantSpaceLimitMb);
                }
                if (is_int($UserSpaceLimitMb)) {
                    if ($UserSpaceLimitMb <= $TenantSpaceLimitMb || $TenantSpaceLimitMb === 0) {
                        $oTenant->setExtendedProp(self::GetName() . '::UserSpaceLimitMb', $UserSpaceLimitMb);
                    } else {
                        throw new \Aurora\System\Exceptions\ApiException(1, null, 'User space limit must be less then tenant space limit');
                    }
                }

                $bResult = CoreModule::getInstance()->getTenantsManager()->updateTenant($oTenant);
            }
        }
        if ($EntityType === 'User') {
            \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::TenantAdmin);
            $oUser = \Aurora\Api::getUserById($EntityId);

            if ($oUser instanceof \Aurora\Modules\Core\Models\User
                    && $oAuthenticatedUser instanceof \Aurora\Modules\Core\Models\User
                    && (($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::TenantAdmin && $oUser->IdTenant === $oAuthenticatedUser->IdTenant)
                    || $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin)) {
                $oTenant = \Aurora\Api::getTenantById($oUser->IdTenant);

                $iTenantSpaceLimitMb = $oTenant->getExtendedProp(self::GetName() . '::TenantSpaceLimitMb');
                if ($iTenantSpaceLimitMb > 0) {
                    $iAllocatedSpaceForUsersInTenant = $this->GetAllocatedSpaceForUsersInTenant($oUser->IdTenant);
                    $iNewAllocatedSpaceForUsersInTenant = $iAllocatedSpaceForUsersInTenant - $oUser->getExtendedProp(self::GetName() . '::UserSpaceLimitMb') + $UserSpaceLimitMb;
                    if ($iNewAllocatedSpaceForUsersInTenant > $iTenantSpaceLimitMb) {
                        throw new \Aurora\System\Exceptions\ApiException(1, null, 'Over quota');
                    }
                }

                $oUser->setExtendedProp(self::GetName() . '::UserSpaceLimitMb', $UserSpaceLimitMb);

                $bResult = \Aurora\Modules\Core\Module::Decorator()->UpdateUserObject($oUser);
            }
        }

        return $bResult;
    }

    public function UpdateUserSpaceLimit($UserId, $Limit)
    {
        $mResult = false;
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::TenantAdmin);

        $oAuthenticatedUser = \Aurora\System\Api::getAuthenticatedUser();
        $oUser = \Aurora\Api::getUserById($UserId);

        if ($oUser instanceof \Aurora\Modules\Core\Models\User && $oAuthenticatedUser instanceof \Aurora\Modules\Core\Models\User && (
            ($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::TenantAdmin && $oUser->IdTenant === $oAuthenticatedUser->IdTenant) ||
                $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin
        )
        ) {
            $oUser->setExtendedProp(self::GetName() . '::UserSpaceLimitMb', $Limit);
            $mResult = \Aurora\Modules\Core\Module::Decorator()->UpdateUserObject($oUser);
        }

        return $mResult;
    }

    public function UpdateTenantSpaceLimit($TenantId, $Limit)
    {
        $mResult = false;
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::SuperAdmin);

        $oAuthenticatedUser = \Aurora\System\Api::getAuthenticatedUser();
        $oTenant = \Aurora\Api::getTenantById($TenantId);

        if ($oTenant instanceof Tenant && $oAuthenticatedUser instanceof User && (
            ($oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::TenantAdmin && $oTenant->Id === $oAuthenticatedUser->IdTenant) ||
                $oAuthenticatedUser->Role === \Aurora\System\Enums\UserRole::SuperAdmin
        )
        ) {
            $oTenant->setExtendedProp(self::GetName() . '::UserSpaceLimitMb', $Limit);
            $mResult = \Aurora\Modules\Core\Module::Decorator()->UpdateUserObject($oTenant);
        }

        return $mResult;
    }

    public function GetAllocatedSpaceForUsersInTenant($TenantId)
    {
        return User::where('IdTenant', $TenantId)->sum('Properties->' . 'PersonalFiles::UsedSpace') / 1024 / 1024;
    }

    public function CheckAllocatedSpaceLimitForUsersInTenant($oTenant, $UserSpaceLimitMb)
    {
        $iTenantSpaceLimitMb = $oTenant->getExtendedProp(self::GetName() . '::TenantSpaceLimitMb');
        $iAllocatedSpaceForUsersInTenant = $this->GetAllocatedSpaceForUsersInTenant($oTenant->Id);

        if ($iTenantSpaceLimitMb > 0 && $iAllocatedSpaceForUsersInTenant + $UserSpaceLimitMb > $iTenantSpaceLimitMb) {
            throw new \Aurora\System\Exceptions\ApiException(1, null, 'Over quota');
        }
    }

    /**
     * Update ExtendedProps
     *
     * @param int $UserId
     * @param string $Type Type of storage contains the item.
     * @param string $Path Path to the item.
     * @param string $Name Name of the item.
     * @param array $ExtendedProps
     * @return bool
     */

    public function UpdateExtendedProps($UserId, $Type, $Path, $Name, $ExtendedProps)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        return false;
        // Actual updating is preceded in subscribed methods. Look for it by "Files::UpdateExtendedProps::after"
    }

    public function GetExtendedProps($UserId = null, $Type = null, $Path = null, $Name = null)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        if ($UserId === null || $Type === null || $Path === null || $Name === null) {
            throw new \Aurora\System\Exceptions\ApiException(\Aurora\System\Notifications::InvalidInputParameter);
        }

        return false;
        // Actual updating is preceded in subscribed methods. Look for it by "Files::GetExtendedProps::after"
    }

    public function GetAccessInfoForPath($UserId, $Type, $Path)
    {
        \Aurora\System\Api::checkUserRoleIsAtLeast(\Aurora\System\Enums\UserRole::NormalUser);

        // Actual updating is preceded in subscribed methods. Look for it by "Files::GetInfoForPath::after"
        return false;
    }

    /**
     * Summary of AddToFavorites
     * @param int $UserId
     * @param array $Items
     * @return bool
     */
    public function AddToFavorites($UserId, $Items)
    {
        $mResult = false;
        $sPublicUserId = Api::getUserPublicIdById($UserId);
        $insert = [];
        foreach ($Items as $aItem) {
            $oItem = Server::getNodeForPath('files/' . $aItem['Type'] . $aItem['Path'] . '/' . $aItem['Name'], $sPublicUserId);
            if ($oItem) {
                $insert[] = [
                    'IdUser' => $UserId,
                    'Type' => $aItem['Type'],
                    'FullPath' => $aItem['Path'] . '/' . $aItem['Name'],
                    'DisplayName' => $this->getNonExistentFavoriteName($UserId, basename($aItem['Name']))
                ];
            }
        }

        if (count($insert) > 0) {
            $mResult = Models\FavoriteFile::insert($insert);
        }

        return $mResult;
    }

    /**
     * Summary of RemoveFromFavorites
     * @param int $UserId
     * @param array $Items
     * @return bool
     */
    public function RemoveFromFavorites($UserId, $Items)
    {
        $mResult = false;
        $sPublicUserId = Api::getUserPublicIdById($UserId);
        $query = Models\FavoriteFile::query();
        $itemsCount = 0;
        foreach ($Items as $aItem) {
            $oItem = Server::getNodeForPath('files/' . $aItem['Type'] . $aItem['Path'] . '/' . $aItem['Name'], $sPublicUserId);
            if ($oItem) {
                $itemsCount++;
                $query->orWhere(function ($subQuery) use ($UserId, $aItem) {
                    $subQuery
                        ->where('IdUser', $UserId)
                        ->where('Type', $aItem['Type'])
                        ->where('FullPath', $aItem['Path'] . '/' . $aItem['Name']);
                });
            }
        }

        if ($itemsCount > 0) {
            $mResult = !!$query->delete();
        }

        return $mResult;
    }
    /**
     * Summary of GetFavorites
     * @param int $UserId
     * @return mixed|\Aurora\System\Classes\Model[]|\Illuminate\Database\Eloquent\Collection
     */
    public function GetFavorites($UserId)
    {
        return Models\FavoriteFile::where('IdUser', $UserId)->get()->toArray();
    }
    /***** public functions might be called with web API *****/

    /**
    * @param int $iUserId
    * @param string $sDisplayName
    *
    * @return string
    */
    protected function getNonExistentFavoriteName($iUserId, $sDisplayName)
    {
        $iIndex = 1;
        $sFileNamePathInfo = pathinfo($sDisplayName);
        $sNameExt = '';
        $sNameWOExt = $sDisplayName;
        if (isset($sFileNamePathInfo['extension'])) {
            $sNameExt = '.' . $sFileNamePathInfo['extension'];
        }

        if (isset($sFileNamePathInfo['filename'])) {
            $sNameWOExt = $sFileNamePathInfo['filename'];
        }

        while (Models\FavoriteFile::where('IdUser', $iUserId)->where('DisplayName', $sDisplayName)->first()) {
            $sDisplayName = $sNameWOExt . ' (' . $iIndex . ')' . $sNameExt;
            $iIndex++;
        }

        return $sDisplayName;
    }
}