Your IP : 18.216.89.144


Current Path : /var/www/www-root/data/www/info.monolith-realty.ru/bitrix/modules/scale/lib/
Upload File :
Current File : /var/www/www-root/data/www/info.monolith-realty.ru/bitrix/modules/scale/lib/monitoring.php

<?php
namespace Bitrix\Scale;

use \Bitrix\Main\Localization\Loc;
Loc::loadMessages(__FILE__);

/**
 * Class Monitoring
 * @package Bitrix\Scale
 */
class Monitoring
{
	protected static $rrdPath = "/var/lib/munin";
	protected static $monitoringCategories = array();

	/**
	 * Checks if database files are created
	 * @return bool
	 */
	public static function isDatabaseCreated($hostname)
	{
		$dir = new \Bitrix\Main\IO\Directory(static::$rrdPath."/".$hostname);
		return $dir->isExists();
	}

	/**
	 * Checks if monitoring is enabled
	 * @return bool
	 */
	public static function isEnabled()
	{
		$result = false;
		$command = "sudo -u root /opt/webdir/bin/bx-monitor -o json";

		try
		{
			$action =  new Action("is_monitoring_enabled", array(
				"START_COMMAND_TEMPLATE" => $command,
				"LOG_LEVEL" => Logger::LOG_LEVEL_DISABLE
				), "", array());

			if(!$action->start())
			{
				return false;
			}
		}
		catch(\Exception $e)
		{
			return false;
		}

		$actRes = $action->getResult();
		if(isset($actRes["is_monitoring_enabled"]["OUTPUT"]["DATA"]["params"]["monitor"]["monitoring_status"]))
		{
			$result = ($actRes["is_monitoring_enabled"]["OUTPUT"]["DATA"]["params"]["monitor"]["monitoring_status"] == "enable");
		}
		else
		{
			$result = false;
		}

		return $result;
	}

	/**
	 * Returns value for server role loadbar (thermometr)
	 * @param $hostname
	 * @param $roleId
	 * @return bool|float
	 * @throws \Bitrix\Main\ArgumentNullException
	 * @throws \Exception
	 * @throws \Bitrix\Main\IO\FileNotFoundException
	 */
	public static function getLoadBarValue($hostname, $roleId)
	{
		if (!extension_loaded('rrd'))
			throw new \Exception("Extension rrd not loaded!");

		if($hostname == '')
			throw new \Bitrix\Main\ArgumentNullException("hostname");

		if($roleId == '')
			throw new \Bitrix\Main\ArgumentNullException("roleId");

		$role = RolesData::getRole($roleId);

		if(empty($role))
			throw new \Exception("Role with id = ".$roleId." was not defined.");

		if(!isset($role["LOADBAR_INFO"]) || $role["LOADBAR_INFO"] == '')
			throw new \Exception("Role ".$roleId." has no correctly defined LOADBAR_INFO param .");

		$rrdFile = str_replace('##HOSTNAME##', $hostname, $role["LOADBAR_INFO"]);
		$rrdPath = "/var/lib/munin/".$hostname."/".$rrdFile;
		$file = new \Bitrix\Main\IO\File($rrdPath);

		if(!$file->isExists())
			throw new \Bitrix\Main\IO\FileNotFoundException($rrdPath);

		$data = \rrd_lastupdate($rrdPath);

		$result = static::extractRrdValue($data);

		return $result;
	}

	public static function getInfoTableCategory($hostname, $categoryId)
	{
		if($hostname == '')
			throw new \Bitrix\Main\ArgumentNullException("hostname");

		if($categoryId == '')
			throw new \Bitrix\Main\ArgumentNullException("paramId");

		$categories = self::getInfoTableCategoriesList($hostname);
		$result = array();

		if(isset($categories[$categoryId]))
			$result = $categories[$categoryId];

		return $result;
	}

	public static function getInfoTableCategoriesList($hostname)
	{
		$result = array();

		$result["HDD"] = array(
			"NAME" => Loc::getMessage("SCALE_MONITORING_HDD"),
			"PARAMS" => static::getHddsParams($hostname)
		);

		$result["NET"] = array(
			"NAME" => Loc::getMessage("SCALE_MONITORING_NET"),
			"PARAMS" => static::getNetParams($hostname)
		);

		$result["HDDACT"] = array(
			"NAME" => Helper::nbsp(Loc::getMessage("SCALE_MONITORING_HDDACT")),
			"PARAMS" => static::getHddsUtilization($hostname)
		);

		$result["MEMORY"] = array(
			"NAME" => Loc::getMessage("SCALE_MONITORING_MEMORY"),
			"PARAMS" => array(
				array(
					"NAME" => Loc::getMessage("SCALE_MONITORING_MEMORY_PARAMS"),
					"TYPE" => "ARRAY",
					"ITEMS" => array(
						array(
							"VALUE_FUNC" => '\Bitrix\Scale\Monitoring::getMemoryUsage',
							"FUNC_PARAMS" => array($hostname)
						),
						array(
							"VALUE_FUNC" => '\Bitrix\Scale\Monitoring::getMemoryUsageValue',
							"FUNC_PARAMS" => array($hostname),
							"TYPE"=>"LOADBAR"
						)
					)
				)
			)
		);

		$result["AVG_LOAD"] = array(
				"NAME" => Helper::nbsp(Loc::getMessage("SCALE_ITS_AVG_LOAD_NAME")),
				"PARAMS" => array(
					"CURR" => array(
						"NAME" => Loc::getMessage("SCALE_ITS_AVG_LOAD_CURR"),
						"RRD" => "##HOSTNAME##-load-load-g.rrd",
						"CF" => "LAST",
						"FORMAT" => "%2.2lf"
					),
					"MIN" => array(
						"NAME" => Loc::getMessage("SCALE_ITS_AVG_LOAD_MIN"),
						"RRD" => "##HOSTNAME##-load-load-g.rrd",
						"CF" => "MIN",
						"FORMAT" => "%2.2lf"
					),
					"MAX" => array(
						"NAME" => Loc::getMessage("SCALE_ITS_AVG_LOAD_MAX"),
						"RRD" => "##HOSTNAME##-load-load-g.rrd",
						"CF" => "MAX",
						"FORMAT" => "%2.2lf"
					)
				)
			);

		return $result;
	}

	public static function getValue($hostname, $categoryId, $param)
	{
		if($hostname == '')
			throw new \Bitrix\Main\ArgumentNullException("hostname");

		if($categoryId == '')
			throw new \Bitrix\Main\ArgumentNullException("categoryId");

		if($param == '')
			throw new \Bitrix\Main\ArgumentNullException("param");

		$arCat = static::getInfoTableCategory($hostname, $categoryId);

		if(!$arCat)
			throw new \Exception("Monitoring category ".$categoryId." not exist.");

		if(!$arCat["PARAMS"][$param])
			throw new \Exception("Monitoring param ".$param." in category ".$categoryId." not exist.");

		$monParam = $arCat["PARAMS"][$param];

		if(isset($monParam["TYPE"]) && $monParam["TYPE"] == "ARRAY")
		{
			if(!isset($monParam["ITEMS"]) || !is_array($monParam["ITEMS"]))
				throw new \Exception("Monitoring param ".$param." in category ".$categoryId." hasn't field ITEMS.");

			$result = array();
			foreach($monParam["ITEMS"] as $item)
			{
				$result[] = static::getItemValue($hostname, $categoryId, $item, $param);
			}
		}
		else
		{
			$result = static::getItemValue($hostname, $categoryId, $monParam, $param);
		}

		return $result;
	}

	protected  static function getItemValue($hostname, $categoryId, $item, $param)
	{
		if(isset($item["VALUE"]))
			return $item["VALUE"];

		if(isset($item["VALUE_FUNC"]))
		{
			return call_user_func_array($item["VALUE_FUNC"], (isset($item["FUNC_PARAMS"]) ? $item["FUNC_PARAMS"] : array()));
		}

		if((!$item["RRD"] || !$item["CF"]) && !$item["OPTIONS"])
			throw new \Exception("Monitoring param item in category ".$categoryId." has no RRD or CF fields.");

		if(isset($item["RRD"]))
		{
			$rrdFile = str_replace('##HOSTNAME##', $hostname, $item["RRD"]);
			$rrdPath = static::$rrdPath."/".$hostname."/".$rrdFile;
			$file = new \Bitrix\Main\IO\File($rrdPath);

			if(!$file->isExists())
				throw new \Bitrix\Main\IO\FileNotFoundException($rrdPath);

			$first = \rrd_first($rrdPath);
			$last = \rrd_last($rrdPath);
		}

		if(isset($item["OPTIONS"]))
		{
			$arOpts = $item["OPTIONS"];
			$arOpts = str_replace('##HOSTNAME##', $hostname, $arOpts);
		}
		else
		{
			if($item["CF"] == "MIN")
			{
				$agr = "MINIMUM";
			}
			elseif($item["CF"] == "MAX")
			{
				$agr = "MAXIMUM";
			}
			else
			{
				$agr = $item["CF"];
			}

			if($item["CF"] == "LAST")
				$item["CF"] = "AVERAGE";

			$format = isset($item["FORMAT"]) ? $item["FORMAT"] : "%6.2lf";

			$arOpts = array(
				"DEF:val=".$rrdPath.":42:".$item["CF"],
				"VDEF:vval=val,".$agr,
				"PRINT:vval:".$format);
		}

		if(isset($item["RRD"]))
		{
			$arOpts[] = "--start";
			$arOpts[] = $first;
			$arOpts[] =  "--end";
			$arOpts[] = $last;
		}

		$data = \rrd_graph( "/dev/null", $arOpts);

		if(isset($item["DATA_FUNC"]) && is_callable($item["DATA_FUNC"]))
		{
			$result = $item["DATA_FUNC"]($data);
		}
		else
		{
			if(isset($data["calcpr"]))
			{
				$data["data"] = $data["calcpr"];
			}

			$result = static::extractRrdValue($data);
		}

		return $result;
	}

	protected static function extractRrdValue($data)
	{
		$result = false;

		if(isset($data["data"]) && is_array($data["data"]))
		{
			reset($data["data"]);
			$result = current($data["data"]);
		}

		return trim($result);
	}

	protected static function getAnsibleSetup($hostname)
	{
		static $info = array();

		if(!isset($info[$hostname]))
		{
			$shellAdapter = new ShellAdapter();
			$execRes = $shellAdapter->syncExec("sudo -u root /usr/bin/ansible ".$hostname." -m setup");
			$serversData = $shellAdapter->getLastOutput();
			$serversData = mb_substr($serversData, mb_strpos($serversData, "{"));

			if($execRes)
			{
				$info[$hostname] = json_decode($serversData, true);
			}
			else
			{
				$info[$hostname] = array();
			}
		}

		return $info[$hostname];
	}

	protected static function getHddsParams($hostname)
	{
		$result = array();
		$arData = static::getAnsibleSetup($hostname);

		if(isset($arData["ansible_facts"]["ansible_mounts"]))
		{
			foreach($arData["ansible_facts"]["ansible_mounts"] as $mountId => $mount)
			{
				$result[$mountId] = array(
					"NAME" => $mount["device"]." ".Loc::getMessage("SCALE_MONITORING_HDD_PARAMS"),
					"TYPE" => "ARRAY",
					"ITEMS" => array(
						array(
							"VALUE_FUNC" => '\Bitrix\Scale\Monitoring::getHddsValues',
							"FUNC_PARAMS" => array(
								$hostname,
								$mountId
							)
						),
						array(
							"TYPE" => "LOADBAR",
							"VALUE_FUNC" => '\Bitrix\Scale\Monitoring::getHddsUsed',
							"FUNC_PARAMS" => array(
								$hostname,
								$mountId
							)

						)
					)
				);
			}
		}

		return $result;
	}

	protected static function getHddsUsed($hostname, $param)
	{
		$arData = static::getAnsibleSetup($hostname);

		if(isset($arData["ansible_facts"]["ansible_mounts"][$param]["size_total"]) && isset($arData["ansible_facts"]["ansible_mounts"][$param]["size_available"]))
		{
			$mount = $arData["ansible_facts"]["ansible_mounts"][$param];
			$result = ($mount["size_total"]-$mount["size_available"])/$mount["size_total"]*100;
		}
		else
		{
			$result = "0";
		}

		return $result;
	}

	protected static function getHddsValues($hostname, $param)
	{
		$arData = static::getAnsibleSetup($hostname);

		if(isset($arData["ansible_facts"]["ansible_mounts"][$param]["size_total"]) && isset($arData["ansible_facts"]["ansible_mounts"][$param]["size_available"]))
		{
			$mount = $arData["ansible_facts"]["ansible_mounts"][$param];

			$result =  static::formatSize($mount["size_total"], 2)."&nbsp;/&nbsp;".
				static::formatSize(($mount["size_total"]-$mount["size_available"]), 2)."&nbsp;/&nbsp;".
				static::formatSize($mount["size_available"], 2);
		}
		else
		{
			$result = "0";
		}

		return $result;
	}

	protected static function getNetParams($hostname)
	{
		$dir = new \Bitrix\Main\IO\Directory(static::$rrdPath."/".$hostname);

		if(!$dir->isExists())
			return array();

		$arChildren = $dir->getChildren();
		$result = array();
		
		foreach ($arChildren as $child)
		{
			if(!$child->isFile())
				continue;

			$name = $child->getName();
			$pos1 = mb_strpos($name, "-if_");
			$pos2 = mb_strpos($name, "-up-");

			if($pos1 !== false && $pos2 !== false)
			{
				$pos1 += 4;
				$dev = mb_substr($name, $pos1, $pos2 - $pos1);

				$result[$dev] = array(
					"NAME" => $dev." ".Loc::getMessage("SCALE_MONITORING_NET_PARAMS"),
					"RRD" => $hostname."-if_".$dev."-up-d.rrd",
					"OPTIONS" => array(
						"DEF:in=".static::$rrdPath."/".$hostname."/".$hostname."-if_".$dev."-up-d.rrd:42:AVERAGE:start=now-600;end=now",
						"DEF:out=".static::$rrdPath."/".$hostname."/".$hostname."-if_".$dev."-down-d.rrd:42:AVERAGE:start=now-600;end=now",
						"VDEF:vin=in,TOTAL",
						"VDEF:vout=out,TOTAL",
						"PRINT:vin:%1.2lf",
						"PRINT:vout:%1.2lf"
					),
					"DATA_FUNC" => '\Bitrix\Scale\Monitoring::formatNetParamsValue'
				);
			}
		}

		return $result;
	}

	protected static function formatNetParamsValue(array $data)
	{
		$result = false;

		if(isset($data['calcpr'][0], $data['calcpr'][1]))
		{
			$result = static::formatSize((int)$data['calcpr'][0]/600) .
				'&nbsp;/&nbsp;' .
				static::formatSize((int)$data['calcpr'][1]/600).'&nbsp;' .
				Helper::nbsp(Loc::getMessage('SCALE_MONITORING_NET_SEC'));
		}

		return $result;
	}

	protected static function getHddsUtilization($hostname)
	{
		$dir = new \Bitrix\Main\IO\Directory(static::$rrdPath."/".$hostname);

		if(!$dir->isExists())
			return array();

		$arChildren = $dir->getChildren();
		$result = array();

		foreach ($arChildren as $child)
		{
			if(!$child->isFile())
				continue;

			$name = $child->getName();
			$pos1 = mb_strpos($name, "-diskstats_utilization-");
			$pos2 = mb_strpos($name, "-util-");
			if($pos1 !== false && $pos2 !== false)
			{
				$pos1 += 23; //strlen("-diskstats_utilization-")
				$dev = mb_substr($name, $pos1, $pos2 - $pos1);

				$result[$dev] = array(
					"NAME" => $dev." ".Loc::getMessage("SCALE_MONITORING_HDDACT_PARAMS"),
					"TYPE" => "ARRAY",
					"ITEMS" => array(
						array(
							"OPTIONS" => array(
								"DEF:r=".static::$rrdPath."/".$hostname."/".$hostname."-diskstats_throughput-".$dev."-rdbytes-g.rrd:42:AVERAGE:start=now-600;end=now",
								"DEF:w=".static::$rrdPath."/".$hostname."/".$hostname."-diskstats_throughput-".$dev."-wrbytes-g.rrd:42:AVERAGE:start=now-600;end=now",
								"VDEF:vr=r,TOTAL",
								"VDEF:vw=w,TOTAL",
								"PRINT:vr:%1.2lf",
								"PRINT:vw:%1.2lf"
							),
							"DATA_FUNC" => '\Bitrix\Scale\Monitoring::formatHddsUtilizationValue'
						),
						array(
							"RRD" => $hostname."-diskstats_utilization-".$dev."-util-g.rrd",
							"CF" => "LAST",
							"FORMAT" => "%2.2lf",
							"TYPE" => "LOADBAR"
						),
					)
				);
			}
		}

		return $result;
	}

	protected static function formatHddsUtilizationValue(array $data)
	{
		$result = false;
		
		if(isset($data['calcpr'][0], $data['calcpr'][1]))
		{
			$result = static::formatSize((int)$data['calcpr'][0]/600) .
				'&nbsp;/&nbsp;' .
				static::formatSize((int)$data['calcpr'][1]/600) .
				'&nbsp;'.Loc::getMessage('SCALE_MONITORING_NET_SEC');
		}

		return $result;
	}

	/**
	 * @param float $size
	 * @param int $precision
	 * @return string
	 */
	public static function formatSize($size, $precision = 2)
	{
		static $a = array("b", "Kb", "Mb", "Gb", "Tb");
		$pos = 0;
		while($size >= 1024 && $pos < 4)
		{
			$size /= 1024;
			$pos++;
		}
		return round($size, $precision)."&nbsp;".$a[$pos];
	}

	protected static function getMemoryUsage($hostname)
	{
		$result = "0";
		$arData = static::getAnsibleSetup($hostname);

		if(isset($arData["ansible_facts"]["ansible_memtotal_mb"]) && isset($arData["ansible_facts"]["ansible_memfree_mb"]))
			$result = $arData["ansible_facts"]["ansible_memtotal_mb"]." / ".(intval($arData["ansible_facts"]["ansible_memtotal_mb"])-intval($arData["ansible_facts"]["ansible_memfree_mb"]))." / ".$arData["ansible_facts"]["ansible_memfree_mb"]." Mb";

		return $result;
	}

	protected static function getMemoryUsageValue($hostname)
	{
		$result = "0";
		$arData = static::getAnsibleSetup($hostname);

		if(isset($arData["ansible_facts"]["ansible_memtotal_mb"]) && isset($arData["ansible_facts"]["ansible_memfree_mb"]) && intval($arData["ansible_facts"]["ansible_memtotal_mb"]) != 0)
			$result = (intval($arData["ansible_facts"]["ansible_memtotal_mb"]) - intval($arData["ansible_facts"]["ansible_memfree_mb"]))/intval($arData["ansible_facts"]["ansible_memtotal_mb"])*100;

		return $result;
	}
}