Сайт вебмастера

Сайт на нескольких CMS одновременно

17-01-2016Время чтения ~ 3 мин.Albireo Framework / CMS 9233

Необычная задача. В результате можно добиться того, чтобы какие-то страницы сайта создавались разными CMS. У меня стояла задача обеспечить одновременную работу LPF и MaxSite CMS. То есть был установлен сайт на MaxSite CMS, который замечательно работает. LPF нужно было использовать только для главной страницы и нескольких других (произвольных).

Эта статья описывает уже несуществующий Landing Page Framework (сейчас это Albireo). Ценность статьи в том, что она описывает общие алгоритмы, которые подходят для любых других php-систем.

Таким образом задача состоит в том, чтобы загружать только реально созданные LPF-страницы, во всех остальных случаях должна работать MaxSite CMS.

Как правило, такие вещи делаются «в лоб»: вторая система устанавливается в отдельный каталог сайта. Но тогда приходится мириться с дополнительным сегментом в адресе страниц. Ну и в этом способе сложно обеспечить работу главной.

Как правило CMS подключается через корневой index.php. Если сайт использует ЧПУ, то первоначальная настройка URL-запроса происходит в .htaccess. Входящий запрос обрабатывается и в «скрытом» виде подключается index.php, который уже дальше сам рулит как ему нужно.

Очевидно, что для одновременной работы нескольких CMS потребуется переименовать index.php. Например для MaxSite CMS это будет index-mso.php, а для LPF — index-lpf.php. Таким образом на сервере будут два корневых файла разных систем.

Что касается других одноименных файлов/каталогов. Очевидно, что они не должны пересекаться. Если этого добиться не получается, значит одновременная работа систем будет невозможна.

Теперь, очевидно, что нам нужен некий диспетчер, который будет выполнять подключение (через require) нужного index-файла. Диспетчер нужно разместить в новом index.php. Таким образом у нас получается:

  • index.php — диспетчер
  • index-mso.php — MaxSite CMS
  • index-lpf.php — LPF

Но у нас есть еще .htaccess в котором формируется нужный URL-запрос.

Если открыть эти файлы, то различие по сути будет только в правилах RewriteRule.

MaxSite CMS
RewriteRule ^(.*)$ /index.php/$1 [L,QSA]
 
LPF 
RewriteRule ^(.*)$ /index.php?page=$1 [QSA]

Если открыть другие CMS, то их правила будут другими. Здесь главное определить какое правило оставить (или даже переписать), чтобы обе системы нормально работали. В нашем случае должен остаться .htaccess от LPF, поскольку он формирует более сложный GET-запрос (добавляет page=...).

Дальше следует определить какие именно GET-запросы после прохождения .htaccess сформировались. Это довольно просто увидеть в php-переменной $_SERVER['QUERY_STRING'], которую можно вывести в диспетчере.

Вот именно на определенном $_SERVER['QUERY_STRING'] и должен подключаться соответствующий index-файл.

Самый простой способ прописать адреса вручную.

if (
	$_SERVER['QUERY_STRING'] == '' or
	$_SERVER['QUERY_STRING'] == 'page=hello'
	)
	require 'index-lpf.php';
else
	require 'index-mso.php';

В данном примере главная страница и /hello отдаст LPF, а все остальные страницы — MaxSite CMS. Обратите внимание, что GET-запрос от LPF формируется с page=, поскольку так работает .htaccess.

Мне хотелось, чтобы LPF отдавал все свои страницы. Поскольку страницы в LPF — это каталоги в lpf-content/pages/, то достаточно автоматом их получить и после проверять на вхождение в get-запрос.

Привожу готовый рабочий код index.php (диспетчер).

<?php
$lpf_pages = _directory_map(dirname(realpath(__FILE__)) . '/lpf-content/pages/', 1);
$lpf_pages = array_map('_add_route', $lpf_pages);
 
if (
	$_SERVER['QUERY_STRING'] == '' or // главная идет от LPF 
	in_array($_SERVER['QUERY_STRING'], $lpf_pages)
	)
	require 'index-lpf.php';
else
	require 'index-mso.php';
 
 
function _add_route($s)
{
	return 'page=' . $s;
}
 
function _directory_map($source_dir, $directory_depth = 0, $hidden = FALSE)
{
	if ($fp = @opendir($source_dir))
	{
		$filedata	= array();
		$new_depth	= $directory_depth - 1;
		$source_dir	= rtrim($source_dir, DIRECTORY_SEPARATOR).DIRECTORY_SEPARATOR;
 
		while (FALSE !== ($file = readdir($fp)))
		{
			if (!trim($file, '.') OR ($hidden == FALSE && $file[0] == '.')) continue;
 
			if (($directory_depth < 1 OR $new_depth > 0) && @is_dir($source_dir.$file))
				$filedata[$file] = _directory_map($source_dir . $file . DIRECTORY_SEPARATOR, $new_depth, $hidden);
			else
				$filedata[] = $file;
		}
 
		closedir($fp);
		return $filedata;
	}
 
	return FALSE;
}
 
# end of flie

Функция _directory_map — это хелпер из CodeIgniter. Функция _add_route добавляет к имени каталога page=, чтобы уже без ухищрений использовать в условии in_array.

Данный подход универсальный к любым CMS, главное подобрать get-запрос в .htaccess, который будет работать на этих системах.

Похожие записи