Your IP : 18.119.134.252


Current Path : /var/www/www-root/data/webdav/www/www.monolith-realty.ru/bitrix/modules/rest/lib/
Upload File :
Current File : /var/www/www-root/data/webdav/www/www.monolith-realty.ru/bitrix/modules/rest/lib/app.php

<?php
namespace Bitrix\Rest;

use Bitrix\Main;
use Bitrix\Main\ArgumentException;
use Bitrix\Main\Data\Cache;
use Bitrix\Main\Engine\CurrentUser;
use Bitrix\Main\Event;
use Bitrix\Main\EventManager;
use Bitrix\Main\EventResult;
use Bitrix\Main\Localization\Loc;
use Bitrix\Main\ObjectPropertyException;
use Bitrix\Main\SystemException;
use Bitrix\Main\Web\HttpClient;
use Bitrix\Main\Web\Uri;
use Bitrix\Rest\Engine\Access;
use Bitrix\Rest\Event\Sender;
use Bitrix\Rest\FormConfig\EventType;
use Bitrix\Rest\Marketplace\Client;
use Bitrix\Main\ORM\Fields\BooleanField;
use Bitrix\Main\ORM\Fields\Relations\OneToMany;
use Bitrix\Rest\Preset\EventController;

Loc::loadMessages(__FILE__);


/**
 * Class AppTable
 *
 * Fields:
 * <ul>
 * <li> ID int mandatory
 * <li> CLIENT_ID string(128) mandatory
 * <li> CODE string(128) mandatory
 * <li> ACTIVE bool optional default 'Y'
 * <li> INSTALLED bool optional default 'N'
 * <li> URL string(1000) mandatory
 * <li> URL_DEMO string(1000) optional
 * <li> URL_INSTALL string(1000) optional
 * <li> VERSION string(4) mandatory
 * <li> SCOPE string(2000) mandatory
 * <li> STATUS string(1) mandatory default 'F'
 * <li> DATE_FINISH date optional
 * <li> IS_TRIALED bool optional default 'N'
 * <li> SHARED_KEY string(32) optional
 * <li> CLIENT_SECRET string(100) optional
 * <li> APP_NAME string(1000) optional
 * <li> ACCESS string(2000) optional
 * </ul>
 *
 * @package Bitrix\Rest
 *
 * DO NOT WRITE ANYTHING BELOW THIS
 *
 * <<< ORMENTITYANNOTATION
 * @method static EO_App_Query query()
 * @method static EO_App_Result getByPrimary($primary, array $parameters = array())
 * @method static EO_App_Result getById($id)
 * @method static EO_App_Result getList(array $parameters = array())
 * @method static EO_App_Entity getEntity()
 * @method static \Bitrix\Rest\EO_App createObject($setDefaultValues = true)
 * @method static \Bitrix\Rest\EO_App_Collection createCollection()
 * @method static \Bitrix\Rest\EO_App wakeUpObject($row)
 * @method static \Bitrix\Rest\EO_App_Collection wakeUpCollection($rows)
 */
class AppTable extends Main\Entity\DataManager
{
	const ACTIVE = 'Y';
	const INACTIVE = 'N';
	const INSTALLED = 'Y';
	const NOT_INSTALLED = 'N';
	const TRIALED = 'Y';
	const NOT_TRIALED = 'N';

	const TYPE_STANDARD = 'N';
	const TYPE_ONLY_API = 'A';
	const TYPE_CONFIGURATION = 'C';
	const TYPE_SMART_ROBOTS = 'R';
	const TYPE_BIC_DASHBOARD = 'B';

	const MODE_SITE = 'S';

	const STATUS_LOCAL = 'L';
	const STATUS_FREE = 'F';
	const STATUS_PAID = 'P';
	const STATUS_DEMO = 'D';
	const STATUS_TRIAL = 'T';
	const STATUS_SUBSCRIPTION = 'S';

	const PAID_NOTIFY_DAYS = 5;
	const PAID_GRACE_PERIOD = -14;

	const CACHE_TTL = 86400;
	const CACHE_PATH = '/rest/app/';

	private static $skipRemoteUpdate = false;

	protected static $licenseLang = null;

	protected static $applicationCache = array();

	protected static $localAppDeniedScope = array(
		'landing_cloud', 'rating',
	);

	/**
	 * Returns DB table name for entity.
	 *
	 * @return string
	 */
	public static function getTableName()
	{
		return 'b_rest_app';
	}

	/**
	 * Returns entity map definition.
	 *
	 * @return array
	 */
	public static function getMap()
	{
		return array(
			'ID' => array(
				'data_type' => 'integer',
				'primary' => true,
				'autocomplete' => true,
			),
			'CLIENT_ID' => array(
				'data_type' => 'string',
				'required' => true,
				'validation' => array(__CLASS__, 'validateClientId'),
			),
			'CODE' => array(
				'data_type' => 'string',
				'required' => true,
				'validation' => array(__CLASS__, 'validateCode'),
			),
			'ACTIVE' => array(
				'data_type' => 'boolean',
				'values' => array(static::INACTIVE, static::ACTIVE),
			),
			'INSTALLED' => array(
				'data_type' => 'boolean',
				'values' => array(static::NOT_INSTALLED, static::INSTALLED),
			),
			'URL' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateUrl'),
			),
			'URL_DEMO' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateUrlDemo'),
			),
			'URL_INSTALL' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateUrlInstall'),
			),
			'URL_SETTINGS' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateUrl'),
			),
			'VERSION' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateVersion'),
			),
			'SCOPE' => array(
				'data_type' => 'string',
				'required' => true,
				'validation' => array(__CLASS__, 'validateScope'),
			),
			'STATUS' => array(
				'data_type' => 'enum',
				'required' => true,
				'values' => array(
					static::STATUS_LOCAL,
					static::STATUS_FREE,
					static::STATUS_PAID,
					static::STATUS_DEMO,
					static::STATUS_TRIAL,
					static::STATUS_SUBSCRIPTION,
				),
			),
			'DATE_FINISH' => array(
				'data_type' => 'date',
			),
			'IS_TRIALED' => array(
				'data_type' => 'boolean',
				'values' => array(static::NOT_TRIALED, static::TRIALED),
			),
			'SHARED_KEY' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateSharedKey'),
			),
			'CLIENT_SECRET' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateClientSecret'),
			),
			'APP_NAME' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateAppName'),
			),
			'ACCESS' => array(
				'data_type' => 'string',
				'validation' => array(__CLASS__, 'validateAccess'),
			),
			'APPLICATION_TOKEN' => array(
				'data_type' => 'string',
			),
			'MOBILE' => array(
				'data_type' => 'boolean',
				'values' => array(static::INACTIVE, static::ACTIVE),
			),
			'USER_INSTALL' => array(
				'data_type' => 'boolean',
				'values' => array(static::INACTIVE, static::ACTIVE),
			),
			'LANG' => array(
				'data_type' => 'Bitrix\Rest\AppLangTable',
				'reference' => array(
					'=this.ID' => 'ref.APP_ID',
					'=ref.LANGUAGE_ID' => new Main\DB\SqlExpression('?s', LANGUAGE_ID),
				),
			),
			'LANG_DEFAULT' => array(
				'data_type' => 'Bitrix\Rest\AppLangTable',
				'reference' => array(
					'=this.ID' => 'ref.APP_ID',
					'=ref.LANGUAGE_ID' => new Main\DB\SqlExpression('?s', Loc::getDefaultLang(LANGUAGE_ID)),
				),
			),
			'LANG_LICENSE' => array(
				'data_type' => 'Bitrix\Rest\AppLangTable',
				'reference' => array(
					'=this.ID' => 'ref.APP_ID',
					'=ref.LANGUAGE_ID' => new Main\DB\SqlExpression('?s', static::getLicenseLanguage()),
				),
			),
			(new OneToMany('LANG_ALL', AppLangTable::class, 'APP'))
				->configureJoinType('left')
		);
	}

	/**
	 * Holds sending changed data to oauth.
	 *
	 * @param $v bool
	 */
	public static function setSkipRemoteUpdate($v)
	{
		static::$skipRemoteUpdate = $v;
	}

	/**
	 * Event on before add application.
	 *
	 * @param Main\Entity\Event $event
	 * @return Main\Entity\EventResult
	 */
	public static function onBeforeAdd(Main\Entity\Event $event)
	{
		$result = new Main\Entity\EventResult();
		$data = $event->getParameters();

		if($data['fields']['STATUS'] == static::STATUS_LOCAL && !$data['fields']['CLIENT_ID'])
		{
			$rnd = Main\Security\Random::getString(8);
			$dummyClientId = 'no_client_id_'.$rnd;
			$dummyClientSecret = 'no_client_secret_'.$rnd;

			$result->modifyFields(array(
				"CLIENT_ID" => $dummyClientId,
				"CLIENT_SECRET" => $dummyClientSecret,
				"CODE" => $dummyClientId,
			));
		}

		return $result;
	}

	/**
	 * Event on after add application.
	 *
	 * @param Main\Entity\Event $event
	 * @return bool
	 * @throws OAuthException
	 */
	public static function onAfterAdd(Main\Entity\Event $event)
	{
		EventController::onAddApp($event);
		$data = $event->getParameters();
		if(!static::$skipRemoteUpdate)
		{
			if(
				$data['fields']['STATUS'] === static::STATUS_LOCAL
				&& OAuthService::getEngine()->isRegistered()
			)
			{
				try
				{
					$appFields = array(
						'TITLE' => $data['fields']['APP_NAME'],
						'REDIRECT_URI' => $data['fields']['URL'],
						'SCOPE' => $data['fields']['SCOPE'],
					);

					$clientData = OAuthService::getEngine()
						->getClient()
						->addApplication($appFields);
				}
				catch(Main\SystemException $e)
				{
					$clientData = array(
						"error" => $e->getCode(),
						"error_description" => $e->getMessage(),
					);
				}

				if(is_array($clientData))
				{
					if($clientData['result'])
					{
						static::$skipRemoteUpdate = true;

						static::clearClientCache($clientData['result']['client_id']);

						$updateResult = static::update($data['id'], array(
							'CLIENT_ID' => $clientData['result']['client_id'],
							'CLIENT_SECRET' => $clientData['result']['client_secret'],
							'STATUS' => static::STATUS_LOCAL,
							'SHARED_KEY' => md5(\CRestUtil::getMemberId().$clientData['result']['client_secret']),
							'CODE' => $clientData['result']['client_id'],
						));
						static::$skipRemoteUpdate = false;

						if($updateResult->isSuccess())
						{
							return true;
						}
						else
						{
							$clientData = array('error' => $updateResult->getErrorMessages());
						}
					}
				}
				else
				{
					$clientData = array('error' => 'Unknown error');
				}

				static::$skipRemoteUpdate = true;
				static::delete($data['id']);
				static::$skipRemoteUpdate = false;

				throw new OAuthException($clientData);
			}
		}

		if($data['fields']['STATUS'] !== static::STATUS_LOCAL)
		{
			\Bitrix\Rest\Engine\Access::getActiveEntity(true);
		}

		return true;
	}

	/**
	 * Event on after update application.
	 *
	 * @param Main\Entity\Event $event
	 * @return bool
	 * @throws OAuthException
	 */
	public static function onAfterUpdate(Main\Entity\Event $event)
	{
		$data = $event->getParameters();
		static::clearClientCache($data['primary']['ID']);

		if(!static::$skipRemoteUpdate)
		{
			if(
				isset($data['fields']['STATUS'])
				&& $data['fields']['STATUS'] === static::STATUS_LOCAL
				&& OAuthService::getEngine()->isRegistered()
			)
			{
				$app = static::getByClientId($data['primary']['ID']);

				try
				{
					$updateResult = OAuthService::getEngine()
						->getClient()
						->updateApplication(array(
							"CLIENT_ID" => $app['CLIENT_ID'],
							'TITLE' => $data['fields']['APP_NAME'],
							'REDIRECT_URI' => $data['fields']['URL'],
							'SCOPE' => $data['fields']['SCOPE'],
						));

					if($updateResult['result'])
					{
						return true;
					}
				}
				catch(Main\SystemException $e)
				{
					$updateResult = array(
						"error" => $e->getCode(),
						"error_description" => $e->getMessage(),
					);
				}

				throw new OAuthException($updateResult);
			}
		}

		if(isset($data['fields']['STATUS']) && $data['fields']['STATUS'] !== static::STATUS_LOCAL)
		{
			\Bitrix\Rest\Engine\Access::getActiveEntity(true);
		}

		return true;
	}

	/**
	 * Event on before delete application.
	 *
	 * @param Main\Entity\Event $event
	 */
	public static function onDelete(Main\Entity\Event $event)
	{
		if(!static::$skipRemoteUpdate)
		{
			$data = $event->getParameters();
			$app = static::getByClientId($data['primary']['ID']);

			if($app['STATUS'] == AppTable::STATUS_LOCAL)
			{
				if(OAuthService::getEngine()->isRegistered())
				{
					\CRestUtil::cleanApp($app["ID"], true);

					try
					{
						OAuthService::getEngine()
							->getClient()
							->deleteApplication(array(
								'CLIENT_ID' => $app['CLIENT_ID'],
							));
					}
					catch(\Bitrix\Main\SystemException $e)
					{
					}
				}
			}
		}
	}

	/**
	 * Event on after delete application.
	 *
	 * @param Main\Entity\Event $event
	 */
	public static function onAfterDelete(Main\Entity\Event $event)
	{
		$data = $event->getParameters();

		static::clearClientCache($data['primary']['ID']);
		AppLangTable::deleteByApp($data['primary']['ID']);
	}

	public static function isInstalled(int $appId): bool
	{
		$row = self::getRowById($appId);
		if (!$row)
		{
			return false;
		}

		return $row['INSTALLED'] === self::INSTALLED;
	}

	public static function install($appId)
	{
		$appInfo = static::getByClientId($appId);
		if($appInfo)
		{
			$eventFields = array(
				'APP_ID' => $appId,
				'VERSION' => $appInfo['VERSION'],
				'ACTIVE' => $appInfo['ACTIVE'],
				'INSTALLED' => $appInfo['INSTALLED'],
			);

			if ($appInfo['ACTIVE'] === self::ACTIVE && $appInfo['INSTALLED'] === self::INSTALLED)
			{
				$res = PlacementTable::getList(
					[
						'filter' => [
							'=APP_ID' => $appInfo['ID'],
						],
						'select' => [
							'ID',
							'PLACEMENT',
							'USER_ID',
						],
					]
				);
				while ($item = $res->fetch())
				{
					$event = new Event(
						'rest',
						PlacementTable::PREFIX_EVENT_ON_AFTER_ADD . $item['PLACEMENT'],
						[
							'ID' => $item['ID'],
							'PLACEMENT' => $item['PLACEMENT'],
							'USER_ID' => $item['USER_ID'],
						]
					);
					EventManager::getInstance()->send($event);
				}
			}

			foreach(GetModuleEvents("rest", "OnRestAppInstall", true) as $eventHandler)
			{
				ExecuteModuleEventEx($eventHandler, array($eventFields));
			}
		}
	}

	/**
	 * Uninstalls application.
	 *
	 * @param string|int $appId
	 * @param int $clean
	 */
	public static function uninstall($appId, $clean = 0)
	{
		$appInfo = static::getByClientId($appId);
		if($appInfo)
		{
			\CRestUtil::cleanApp($appId, $clean);

			if($appInfo['STATUS'] !== static::STATUS_LOCAL)
			{
				OAuthService::getEngine()->getClient()->unInstallApplication(array(
					'CLIENT_ID' => $appInfo['CLIENT_ID']
				));
			}
		}
	}

	/**
	 * Checks opportunity of deleting application.
	 *
	 * @param int $appId
	 * @param int $clean
	 * @return Main\ErrorCollection
	 */
	public static function checkUninstallAvailability($appId, $clean = 0)
	{
		$event = new Main\Event('rest', 'onBeforeApplicationUninstall', [
			'ID' => $appId,
			'CLEAN' => $clean
		]);
		$event->send();

		$result = new Main\ErrorCollection();
		if ($event->getResults())
		{
			/** @var \Bitrix\Main\EventResult $eventResult */
			foreach ($event->getResults() as $eventResult)
			{
				if($eventResult->getType() === EventResult::ERROR)
				{
					$eventResultData = $eventResult->getParameters();
					if ($eventResultData instanceof Main\Error)
					{
						$result->add([$eventResultData]);
					}
				}
			}
		}

		return $result;
	}

	/**
	 * Updates applications status from OAuth.
	 *
	 * @throws Main\ArgumentException
	 * @throws Main\ObjectPropertyException
	 * @throws Main\SystemException
	 */
	public static function updateAppStatusInfo()
	{
		$appList = OAuthService::getEngine()->getClient()->getApplicationList();

		if(is_array($appList) && is_array($appList['result']))
		{
			$dbApps = static::getList(array(
				'filter' => array(
					'!=STATUS' => static::STATUS_LOCAL,
				),
				'select' => array(
					'ID', 'CLIENT_ID', 'STATUS', 'DATE_FINISH',
				)
			));

			$localApps = array();
			while($app = $dbApps->fetch())
			{
				$localApps[$app['CLIENT_ID']] = $app;
			}

			foreach($appList['result'] as $app)
			{
				if(array_key_exists($app['client_id'], $localApps))
				{
					$dateFinishLocal = $localApps[$app['client_id']]['DATE_FINISH']
						? $localApps[$app['client_id']]['DATE_FINISH']->getTimestamp()
						: '';
					$dateFinishRemote = ($app['date_finish'] ?? null) ? Main\Type\Date::createFromTimestamp($app['date_finish'])->getTimestamp() : '';

					if(
						$localApps[$app['client_id']]['STATUS'] !== $app['status']
						|| $dateFinishRemote != $dateFinishLocal
					)
					{
						$appFields = array(
							'STATUS' => $app['status'],
							'DATE_FINISH' => $app['date_finish']
								? Main\Type\Date::createFromTimestamp($app['date_finish'])
								: '',
						);

						static::setSkipRemoteUpdate(true);
						$result = static::update($localApps[$app['client_id']]['ID'], $appFields);
						static::setSkipRemoteUpdate(false);

						if(
							$result->isSuccess()
							&& $appFields['STATUS'] === static::STATUS_PAID
						)
						{
							static::callAppPaymentEvent($localApps[$app['client_id']]['ID']);
						}
					}
				}
				else
				{
					$appFields = array(
						'CLIENT_ID' => $app['client_id'],
						'CODE' => $app['code'],
						'ACTIVE' => $app['active'] ? static::ACTIVE : static::INACTIVE,
						'INSTALLED' => static::NOT_INSTALLED,
						'VERSION' => $app['version'],
						'STATUS' => $app['status'],
						'SCOPE' => $app['scope'],
					);

					if(!empty($app['date_finish']))
					{
						$appFields['DATE_FINISH'] = Main\Type\Date::createFromTimestamp($app['date_finish']);
					}

					$result = static::add($appFields);

					if($result->isSuccess() && $appFields['STATUS'] === static::STATUS_PAID)
					{
						static::callAppPaymentEvent($result->getId());
					}
				}
			}
		}
	}

	/**
	 * Sends event applications payment information.
	 *
	 * @param $appId
	 */
	public static function callAppPaymentEvent($appId)
	{
		// for compatibility purpose module_id is bitrix24 here
		foreach(GetModuleEvents('bitrix24', 'OnAfterAppPaid', true) as $event)
		{
			ExecuteModuleEventEx($event, array($appId));
		}
	}

	/**
	 * Returns applications information.
	 *
	 * @param mixed $app
	 * @param string $detailUrl
	 * @return array
	 */
	public static function getAppStatusInfo($app, $detailUrl)
	{
		$res = array();

		if (
			!empty($app)
			&& (
				is_string($app)
				|| is_integer($app)
			)
		)
		{
			$appInfo = $app = static::getByClientId($app);
		}
		elseif (isset($app['CODE']))
		{
			$appInfo = static::getByClientId($app['CODE']);
		}

		if(is_array($app))
		{
			$res['STATUS'] = $app['STATUS'];
			$res['PAYMENT_NOTIFY'] = 'N';
			$res['PAYMENT_EXPIRED'] = 'N';
			$res['PAYMENT_ALLOW'] = 'Y';

			if ($app['STATUS'] === self::STATUS_SUBSCRIPTION)
			{
				if (!\Bitrix\Rest\Marketplace\Client::isSubscriptionAvailable())
				{
					$res['MESSAGE_REPLACE'] = array(
						'#DETAIL_URL#' => $detailUrl,
						'#DAYS#' => 0,
						'#CODE#' => urlencode($app['CODE'])
					);
					$res['PAYMENT_NOTIFY'] = 'Y';
					$res['PAYMENT_EXPIRED'] = 'Y';
					$res['PAYMENT_ALLOW'] = 'N';
				}
				else
				{
					$dateFinish = \Bitrix\Rest\Marketplace\Client::getSubscriptionFinalDate();
					if (!is_null($dateFinish))
					{
						$res['DAYS_LEFT'] = floor(($dateFinish->getTimestamp() - \CTimeZone::getOffset() - time()) / 86400);
						if($res['DAYS_LEFT'] < 0)
						{
							$res['MESSAGE_REPLACE'] = array(
								'#DETAIL_URL#' => $detailUrl,
								'#DAYS#' => $res['DAYS_LEFT'],
								'#CODE#' => urlencode($app['CODE'])
							);
							$res['PAYMENT_NOTIFY'] = 'Y';
							$res['PAYMENT_EXPIRED'] = 'Y';
							$res['PAYMENT_ALLOW'] = 'N';
						}
						elseif ($res['DAYS_LEFT'] < static::PAID_NOTIFY_DAYS)
						{
							$res['MESSAGE_REPLACE'] = array(
								'#DETAIL_URL#' => $detailUrl,
								'#DAYS#' => $res['DAYS_LEFT'],
								'#CODE#' => urlencode($app['CODE'])
							);
							$res['PAYMENT_NOTIFY'] = 'Y';
						}
					}
				}
			}
			elseif($app['DATE_FINISH'] <> '' && $app['STATUS'] != self::STATUS_FREE)
			{
				$res['DAYS_LEFT'] = floor(
					(MakeTimeStamp($app['DATE_FINISH']) - \CTimeZone::getOffset() - time()) / 86400
				);

				if(
					$res['DAYS_LEFT'] < static::PAID_NOTIFY_DAYS
					|| $app['STATUS'] == static::STATUS_TRIAL)
				{
					$res['PAYMENT_NOTIFY'] = 'Y';

					if($res['DAYS_LEFT'] < 0)
					{
						$res['PAYMENT_EXPIRED'] = 'Y';

						if($app['STATUS'] == static::STATUS_TRIAL)
						{
							$res['PAYMENT_ALLOW'] = 'N';
						}
						elseif(
							$app['STATUS'] == static::STATUS_PAID
							&& $res['DAYS_LEFT'] < static::PAID_GRACE_PERIOD
						)
						{
							if($app['IS_TRIALED'] == 'N' && $app['URL_DEMO'] <> '')
							{
								$res['STATUS'] = static::STATUS_DEMO;
							}
							else
							{
								$res['PAYMENT_ALLOW'] = 'N';
							}
						}
					}
				}

				$res['MESSAGE_REPLACE'] = array(
					"#DETAIL_URL#" => $detailUrl,
					"#DAYS#" => $res['DAYS_LEFT'],
					"#CODE#" => urlencode($app['CODE']),
				);
			}
			elseif($app['STATUS'] == self::STATUS_DEMO)
			{
				$res['PAYMENT_NOTIFY'] = 'Y';
				$res['MESSAGE_REPLACE'] = array(
					"#DETAIL_URL#" => $detailUrl,
					"#CODE#" => urlencode($app['CODE'])
				);
			}
			else
			{
				$res['MESSAGE_REPLACE'] = array(
					"#DETAIL_URL#" => $detailUrl,
					"#CODE#" => urlencode($app['CODE'])
				);
			}

			$res['MESSAGE_SUFFIX'] = '_'.$res['STATUS'].'_'.$res['PAYMENT_EXPIRED'].'_'.$res['PAYMENT_ALLOW'];

		}

		if (!empty($appInfo['CLIENT_ID']))
		{
			$isHold = \Bitrix\Rest\Engine\Access\HoldEntity::is(
				\Bitrix\Rest\Engine\Access\HoldEntity::TYPE_APP,
				$appInfo['CLIENT_ID']
			);
			if ($isHold)
			{
				$res['MESSAGE_SUFFIX'] = '_HOLD_OVERLOAD';
				$res['PAYMENT_NOTIFY'] = 'Y';
			}
		}

		return $res;
	}

	/**
	 * Returns message with applications status.
	 *
	 * @param string $suffix
	 * @param array|null $replace
	 * @param bool $checkAdmin
	 * @param string|null $language
	 *
	 * @return string
	 */
	public static function getStatusMessage($suffix, $replace = null, $checkAdmin = true, $language = null)
	{
		if ($checkAdmin && \CRestUtil::isAdmin())
		{
			$suffix .= '_A';
		}

		if (
			is_array($replace)
			&& array_key_exists('#DAYS#', $replace)
			&& (
				is_int($replace['#DAYS#'])
				|| preg_match('/^(-|)\d+$/', $replace['#DAYS#'])
			)
		)
		{
			$replace['#DAYS#'] = FormatDate('ddiff', time(), time() + 24 * 60 * 60 * $replace['#DAYS#']);
		}

		return Loc::getMessage('PAYMENT_MESSAGE' . $suffix, $replace, $language);
	}

	/**
	 * @param string|int $appId
	 * @return array|false
	 */
	public static function getAccess($appId)
	{
		$appInfo = static::getByClientId($appId);
		if($appInfo)
		{
			if($appInfo['ACCESS'] <> '')
			{
				$rightsList = explode(",", $appInfo["ACCESS"]);

				$access = new \CAccess();
				$accessNames = $access->getNames($rightsList);

				$result = array();
				foreach($rightsList as $right)
				{
					$result[$right] = array(
						"id" => $right,
						"provider" => $accessNames[$right]["provider_id"],
						"name" => $accessNames[$right]["name"]
					);
				}
				return $result;
			}
		}

		return false;
	}

	/**
	 * @param string|int $appId
	 * @param array $newRights
	 * @throws \Exception
	 */
	public static function setAccess($appId, $newRights = array())
	{
		$appInfo = static::getByClientId($appId);
		if($appInfo)
		{
			$rights = '';
			if(is_array($newRights) && !empty($newRights))
			{
				$rightIdList = array();
				foreach($newRights as $key => $rightsList)
				{
					foreach($rightsList as $rightId => $ar)
					{
						$rightIdList[] = $rightId;
					}
				}
				$rights = implode(",", $rightIdList);
			}

			static::update($appId, array(
				'ACCESS' => $rights,
			));
		}

		if(defined("BX_COMP_MANAGED_CACHE"))
		{
			global $CACHE_MANAGER;
			$CACHE_MANAGER->ClearByTag('bitrix24_left_menu');
		}
	}

	/**
	 * @param string|int $clientId
	 * @return mixed
	 * @throws Main\ArgumentException
	 * @throws Main\ObjectPropertyException
	 * @throws Main\SystemException
	 */
	public static function getByClientId($clientId)
	{
		if(!array_key_exists($clientId, static::$applicationCache))
		{
			if(strval(intval($clientId)) == $clientId)
			{
				$filter = array('=ID' => $clientId);
			}
			else
			{
				$filter = array(
					array(
						'LOGIC' => 'OR',
						'=CODE' => $clientId,
						'=CLIENT_ID' => $clientId,
					),
				);
			}

			$dbRes = static::getList(
				[
					'filter' => $filter,
					'select' => [
						'*',
						'MENU_NAME' => 'LANG.MENU_NAME',
						'MENU_NAME_DEFAULT' => 'LANG_DEFAULT.MENU_NAME',
						'MENU_NAME_LICENSE' => 'LANG_LICENSE.MENU_NAME',
					],
					'limit' => 1,
				]
			);

			foreach ($dbRes->fetchCollection() as $app)
			{
				$appInfo = [
					'ID' => $app->getId(),
					'MENU_NAME' => !is_null($app->getLang()) ? $app->getLang()->getMenuName() : '',
					'MENU_NAME_DEFAULT' => !is_null($app->getLangDefault()) ? $app->getLangDefault()->getMenuName() : '',
					'MENU_NAME_LICENSE' => !is_null($app->getLangLicense()) ? $app->getLangLicense()->getMenuName() : '',
				];
				foreach ($app->sysGetEntity()->getScalarFields() as $field)
				{
					$fieldName = $field->getName();
					if ($field instanceof BooleanField)
					{
						$appInfo[$fieldName] = $app->get($fieldName) ? 'Y' : 'N';
					}
					else
					{
						$appInfo[$fieldName] = $app->get($fieldName);
					}
				}
				$app->fillLangAll();
				if (!is_null($app->getLangAll()))
				{
					foreach ($app->getLangAll() as $lang)
					{
						$appInfo['LANG_ALL'][$lang->getLanguageId()] = [
							'MENU_NAME' => $lang->getMenuName(),
						];
					}
				}
				if ($appInfo['MENU_NAME'] === '')
				{
					$appInfo = Lang::mergeFromLangAll($appInfo);
				}
			}

			if (isset($appInfo) && is_array($appInfo))
			{
				static::$applicationCache[$appInfo['ID']] = $appInfo;
				static::$applicationCache[$appInfo['CLIENT_ID']] = $appInfo;
				static::$applicationCache[$appInfo['CODE']] = $appInfo;
			}
		}

		return static::$applicationCache[$clientId];
	}

	protected static function clearClientCache($clientId)
	{
		if(array_key_exists($clientId, static::$applicationCache))
		{
			$app = static::$applicationCache[$clientId];
			if(is_array($app))
			{
				unset(static::$applicationCache[$app['ID']]);
				unset(static::$applicationCache[$app['CLIENT_ID']]);
			}
			else
			{
				unset(static::$applicationCache[$clientId]);
			}
		}
	}

	protected static function getLicenseLanguage()
	{
		if(static::$licenseLang === null)
		{
			if(Main\Loader::includeModule('bitrix24'))
			{
				static::$licenseLang = \CBitrix24::getLicensePrefix();
			}
			else
			{
				$dbSites = \CSite::getList('sort', 'asc', array('DEFAULT' => 'Y', 'ACTIVE' => 'Y'));
				$site = $dbSites->fetch();

				static::$licenseLang = is_array($site) && isset($site['LANGUAGE_ID']) ? $site['LANGUAGE_ID'] : LANGUAGE_ID;
			}

			if(static::$licenseLang === null)
			{
				static::$licenseLang = 'en';
			}
		}

		return static::$licenseLang;
	}


	/**
	 * Returns validators for CLIENT_ID field.
	 *
	 * @return array
	 */
	public static function validateClientId()
	{
		return array(
			new Main\Entity\Validator\Length(null, 128),
			new Main\Entity\Validator\Unique(),
		);
	}

	/**
	 * Returns validators for CODE field.
	 *
	 * @return array
	 */
	public static function validateCode()
	{
		return array(
			new Main\Entity\Validator\Length(null, 128),
			new Main\Entity\Validator\Unique(),
		);
	}

	/**
	 * Returns validators for URL field.
	 *
	 * @return array
	 */
	public static function validateUrl()
	{
		return array(
			new Main\Entity\Validator\Length(null, 1000),
		);
	}

	/**
	 * Returns validators for URL_DEMO field.
	 *
	 * @return array
	 */
	public static function validateUrlDemo()
	{
		return array(
			new Main\Entity\Validator\Length(null, 1000),
		);
	}

	/**
	 * Returns validators for URL_INSTALL field.
	 *
	 * @return array
	 */
	public static function validateUrlInstall()
	{
		return array(
			new Main\Entity\Validator\Length(null, 1000),
			function ($value, $primary, array $row, Main\Entity\Field $field)
			{
				$checkResult = true;

				if($value)
				{
					try
					{
						if(!HandlerHelper::checkCallback($value, $row, false))
						{
							$checkResult = false;
						}
					}
					catch(RestException $e)
					{
						$checkResult = false;
					}

					if(!$checkResult)
					{
						return Loc::getMessage("MP_ERROR_INCORRECT_URL_INSTALL");
					}
				}

				return true;
			}
		);
	}

	/**
	 * Returns validators for VERSION field.
	 *
	 * @return array
	 */
	public static function validateVersion()
	{
		return array(
			new Main\Entity\Validator\Length(null, 4),
		);
	}

	/**
	 * Returns validators for SCOPE field.
	 *
	 * @return array
	 */
	public static function validateScope()
	{
		return array(
			new Main\Entity\Validator\Length(null, 2000),
		);
	}

	/**
	 * Returns validators for SHARED_KEY field.
	 *
	 * @return array
	 */
	public static function validateSharedKey()
	{
		return array(
			new Main\Entity\Validator\Length(null, 32),
		);
	}

	/**
	 * Returns validators for APP_SECRET_ID field.
	 *
	 * @return array
	 */
	public static function validateClientSecret()
	{
		return array(
			new Main\Entity\Validator\Length(null, 100),
		);
	}

	/**
	 * Returns validators for APP_NAME field.
	 *
	 * @return array
	 */
	public static function validateAppName()
	{
		return array(
			new Main\Entity\Validator\Length(null, 1000),
		);
	}

	/**
	 * Returns validators for ACCESS field.
	 *
	 * @return array
	 */
	public static function validateAccess()
	{
		return array(
			new Main\Entity\Validator\Length(null, 2000),
		);
	}

	/**
	 * @param array $permissionList
	 * @return array
	 */
	public static function cleanLocalPermissionList(array $permissionList)
	{
		foreach($permissionList as $key => $perm)
		{
			if(in_array($perm, static::$localAppDeniedScope))
			{
				unset($permissionList[$key]);
			}
		}

		return array_values($permissionList);
	}

	/**
	 * @param string $code
	 * @param false $version
	 * @return bool
	 */
	public static function canUninstallByType($code, $version = false)
	{
		$result = true;

		$type = static::getAppType($code, $version);
		if($type == static::TYPE_CONFIGURATION)
		{
			$appList = \Bitrix\Rest\Configuration\Helper::getInstance()->getBasicAppList();
			if(in_array($code, $appList))
			{
				$result = false;
			}
		}

		return $result;
	}

	/**
	 * @param $code
	 * @param false $version
	 * @return false|mixed
	 */
	public static function getAppType($code, $version = false)
	{
		$result = false;
		$cache = Cache::createInstance();
		if ($cache->initCache(static::CACHE_TTL, 'appType'.md5($code.$version), static::CACHE_PATH))
		{
			$result = $cache->getVars();
		}
		elseif ($cache->startDataCache())
		{
			$appDetailInfo = Client::getInstall($code, $version);
			$result = ($appDetailInfo['ITEMS']['TYPE'])?:false;
			$cache->endDataCache($result);
		}

		return $result;
	}
}

class App
{
	public function __construct(
		private string $clientId
	)
	{
	}

	/**
	 * @throws ObjectPropertyException
	 * @throws SystemException
	 * @throws ArgumentException
	 */
	public function fetchAppFormConfig(array $formData, EventType $type): string
	{
		$appData = AppTable::getByClientId($this->clientId);
		if (empty($appData['URL_SETTINGS']))
		{
			throw new ArgumentException('Property "URL_SETTINGS" is empty', 'URL_SETTINGS');
		}
		$uri = new Uri($appData['URL_SETTINGS']);
		$httpClient = new HttpClient();
		$params = Sender::getDefaultEventParams();
		$params['sendRefreshToken'] = true;
		$event = [
			'event' => $type->value,
			'data' => $formData,
			'auth' => Sender::getAuth(
				$this->clientId,
				CurrentUser::get()->getId() ?? 0,
				[],
				$params
			)
		];

		return $httpClient->post($uri, $event);
	}
}