До недавних пор я создавал сайты и плагины на WordPress, арендуя виртуальные хостинги у провайдеров. Для себя еще давно выделил панель ISP за удобность и практичность. Так случилось, что все время работал на Windows, следовательно, Linux для меня — темный лес с диким животными. Сайты со временем «росли» и становились более требовательны, как минимум к дисковому пространству и иногда к оперативной памяти.
Пару месяцев назад по некоторым соображениям решил арендовать виртуальный сервер на Linux и самостоятельно установить туда ISP и BILL для создания и управления услугами.
Поколдовав несколько часов с документацией и SSH консолью, я запустил свой первый сервер на CentOS. В течение недели выяснил: почему gmail.ru и mail.ru не хотят принимать письма с моего
И дело тут не в самом BILL, а именно в ISP Lite — как мне ответила техподдержка от ISP System:
Настройки дополнения «Оперативная память» в биллинге влияет на оперативную память, выделяемую пользователю в ISPmanager. Но у вас используется ISPmanager Lite, где нет ограничений на оперативную память. Ограничение на оперативную память есть только в ISPmanager Business.
Печалька. Поторчав часик-другой в гугле, ничего путного и бесплатного не нашел. В свою очередь, есть только возможность «ручками» менять параметр:
1. Изменяя файл php.ini в папке пользователя:
2. Открыть возможность клиенту в ISPmanager самостоятельно менять параметры PHP, что по сути меняет тот же файл php.ini и к тому же на усмотрение клиента:
Эти способы «убивали» всю автоматику покупки и обслуживания
Альфа-решение
В первую очередь был простой план: найти в БД билла установленные данные для каждого пользователя и переписать их в файлах php.ini простеньким скриптом PHP. А сам PHP запускать через CRON. Но вспоминая ситуации, когда для сайта вложенных 128Мб не хватало, понимал следующее: данные случаи происходили случайно и требовали чуть ли не мгновенного решения. Заставить CRON каждые 5 минут запускать этот скрипт — слишком безграмотное решение задачи с холостой работой 99% времени.
Бета-решение
Стал искать способ повесить скрипт на событие смены доп-услуг виртуального
Скрипт работал на событие и справлялся с поставленной задачей, но с чем я не разобрался, так это с папками php-bin, и вложенным в них файлом php.ini. А именно: сначала данные папки находились и прекрасно работали по пути /var/www/{user}/data/php-bin/, но время спустя после моих «издевательств» над ISP и сервером в целом, они сменили местоположение на /var/www/php-bin/{user}/… По гуглу и вышеупомянутой не-до-документации от ISP для этого есть опция DisableSecurePhpBin, но она не влияла никак. В следствие чего я добавил в свой плагинчик поиск файла php.ini в обоих местах, а так же поиск и файла с именем /var/www/{user}/data/php-bin/.php.ini(с точкой вначале).
Что нужно сделать
Долго расписывать не буду, дам просто готовую инструкцию по использованию моего мини-плагинчика для BILLmanager 5:
1. Нужно создать 2 файла с правами root и следующим содержанием:
1.1. /usr/local/mgr5/etc/xml/billmgr_mod_hiweb_vhostram.xml
<?xml version="1.0" encoding="UTF-8"?>
<mgrdata>
<handler name="hiweb_vhostram.php" type="cgi">
<event name="vhost.edit" after="yes"/>
</handler>
</mgrdata>
1.2. /usr/local/mgr5/addon/hiweb_vhostram.php
#!/usr/bin/php
<?php
$logFile = '/var/www/log/hiweb_vhostram.log';
$logInfo = array();
////////
//Получение данных для подключения к ДБ Билла
$fileSettings = dirname( dirname( __FILE__ ) ) . '/etc/billmgr.conf.d/db.conf';
if( file_exists( $fileSettings ) ){
$fileSettingsContent = file( $fileSettings, FILE_IGNORE_NEW_LINES );
foreach( $fileSettingsContent as $line ){
$arr = explode( ' ', trim( $line ) );
if( count( $arr ) != 2 )
continue;
list( $lineKey, $lineValue ) = $arr;
if( $lineKey == 'DBHost' )
$DBHost = $lineValue;
if( $lineKey == 'DBUser' )
$DBUser = $lineValue;
if( $lineKey == 'DBPassword' )
$DBPassword = $lineValue;
if( $lineKey == 'DBName' )
$DBName = $lineValue;
}
$mysql = new mysqli( $DBHost, $DBUser, $DBPassword, $DBName );
//$R = array($_GET, $_POST, $_SERVER);
if( $mysql ){
//Поиск индификатора купленной опции RAM
$priceListResult = $mysql->query( 'SELECT id FROM pricelist WHERE itemtype=(SELECT id FROM itemtype WHERE name="RAM" AND statparam != "")' );
$priceList = reset( $priceListResult->fetch_assoc() );
///Поиск мегабайт памяти
if(!isset($_SERVER['PARAM_addon_'.$priceList])){
$logInfo[] = 'error: no parameter [PARAM_addon_'.$priceList.'] in $_SERVER';
}else{
$additionMemory = $_SERVER['PARAM_addon_'.$priceList];
///Поиск пользователя
$userLogin = $_SERVER['PARAM_username'];
if( trim( $userLogin ) == '' ){
//do nothing
}
else{
$logInfo[] = 'select user ['.$userLogin.']';
///
//$userResult = $mysql->query( 'SELECT * FROM itemparam WHERE intname="username" AND item="' . $userId . '"' );
//$userData = $userResult->fetch_assoc();
//$userLogin = $userData['value'];
///
///Файлы установок
$phpIniFiles = array(
'/var/www/php-bin/' . $userLogin . '/php.ini', '/var/www/' . $userLogin . '/data/php-bin/.php.ini', '/var/www/' . $userLogin . '/data/php-bin/php.ini'
);
//Перебор файлов
foreach( $phpIniFiles as $phpIni ){
$B = '';
if( !file_exists( $phpIni ) ){
$logInfo[] = 'error: file not exists!';
}elseif( !is_readable( $phpIni ) ){
$logInfo[] = 'error: file not readable!';
}elseif( !is_file( $phpIni ) ){
$logInfo[] = 'error: this is dir, not file!!!';
}else{
$content = file_get_contents( $phpIni );
preg_match('/^memory_limit = (d){1,4}[a-zA-Z]?$/im',$content, $match);
if(count($match) > 0){
$line = trim(reset($match));
//Получение строки с новыми значениями
$lineExplode = explode( ' = ', $line );
$lineExplode[1] = $additionMemory . 'M';
$newLine = implode( ' = ', $lineExplode );
//Сравнение на изменение и запись в файл новых значений
if( trim( $line ) != trim( $newLine ) ){
$logInfo[] = file_put_contents($phpIni, str_replace($line,$newLine,$content)) ? $line.' → '.$newLine : 'error: file_put_content';
}else{
$logInfo[] = 'not change: '.$line.' → '.$newLine;
}
}else{
$add = 'memory_limit = ' . $additionMemory.'M';
$content .= chr( 13 ) . chr( 10 ) . $add . chr( 13 ) . chr( 10 );
$logInfo[] = file_put_contents($phpIni, $content) ? '+ '.$add : 'error add: file_put_content';
}
}
}
}
}
} else {
$logInfo[] = 'error: mysql not connected!';
}
}
file_put_contents($logFile, print_r( $logInfo, 1 ) );
echo '<?xml version="1.0" encoding="UTF-8"?><doc>';
echo '</doc>';
?>
2. Добавить к необходимому тарифу дополнительную опцию «Оперативная память» — у меня она была доступна сразу после установки ISP, затем BILL менеджеров. Осталось ее только включить в тариф, указав пару данных:
3. Также необходимо перевести WWW-домен → PHP в режим работы «CGI»:
4. Попробовать «докупить» оперативной памяти для виртуального
5. На всякий случай, файлик PHP для проверки работы смены ограничения оперативной памяти на виртуальном
<?php
function return_bytes( $val ){
$val = trim( $val );
$last = strtolower( $val[ strlen( $val ) - 1 ] );
switch( $last ){
// The 'G' modifier is available since PHP 5.1.0
case 'g':
$val *= 1024;
case 'm':
$val *= 1024;
case 'k':
$val *= 1024;
}
return $val;
}
/**
* Возвращает форматированный вид размера файла из байтов
* @param $size - INT килобайты
* @return string
*/
function size_format( $size ){
$size = intval( $size );
if( $size < 1024 ){
return $size . " bytes";
}else if( $size < ( 1024 * 1024 ) ){
$size = round( $size / 1024, 1 );
return $size . " KB";
}else if( $size < ( 1024 * 1024 * 1024 ) ){
$size = round( $size / ( 1024 * 1024 ), 1 );
return $size . " MB";
}else{
$size = round( $size / ( 1024 * 1024 * 1024 ), 1 );
return $size . " GB";
}
}
$memory = size_format( return_bytes( ini_get( 'memory_limit' ) ) );
echo '<phpmem>';
echo '<val>' . $memory . '</val>';
echo '</phpmem>';
Вывод
Актуальная документация позволяет сэкономить уйму времени! )) Данный плагинчик — первый мой опыт написания подобных дополнений и не факт, что будет работать корректно или вообще работать, так как я только разбираюсь в сложносплетениях Linux и Серверного ПО. Думаю, со временем доработаю его, если будет смысл.
Автор: DenMedia