Компьютерные науки - Учебники на русском языке - Скачать бесплатно
Виктор Хименко
HotMail своими руками, или что может PHP
МИР ПК #02/99
В этой статье я хотел бы продемонстрировать, что создание работоспособных и
полезных динамических Web-узлов вовсе не такая сложная задача, как может
показаться при чтении текстов, описывающих интерфейс CGI. На самом деле подобная
работа (по крайней мере, при использовании Linux) больше всего напоминает игру
The Incredible Machines, где нужно кремнем высечь искру, чтобы свет упал на
солнечную батарею, от которой срабатывает сверхсовременный компьютер. Здесь тоже
все, как правило, собирается из компонентов, которые оказались под рукой.
В качестве примера мы рассмотрим создание почтовой службы на Web-сервере -
WebMail - на основе языка PHP3 с максимальным использованием готовых
компонентов. За кадром останутся создание новых пользователей, их регистрация в
базе данных и т. п. Наша почта, кроме того, не будет обрабатывать письма с
вложениями и в кодировках, отличных от KOI8-R, поддерживать записные книжки и т.
п. - все это лишь увеличило бы размеры сценариев, не дав ничего принципиально
нового.
Компоненты
Вначале о "готовых компонентах". Я уже упомянул о том, что все описанное ниже
рассчитано на использование Linux. Однако существующие дистрибутивы этой ОС
настолько различаются по комплектации, что вряд ли удастся построить пример,
который годился бы для любой версии. Поэтому я ограничусь версией, с которой
лучше всего знаком, - KSI-Linux 2.0 (http://www.ksi-linux.com). Если у вас
другой вариант Linux, например RedHat 5.2 или Debian 2.0, то вам, скорее всего,
потребуется получить некоторые компоненты из Internet или перекомпилировать уже
имеющиеся.
Нам понадобятся Web-сервер Russian Apache 1.3.x rus/PL 27.4, PHP 3.0.6 с
поддержкой протокола IMAP4r1 и почтовый сервер IMAP4r1. Благодаря Russian Apache
мы будем, по крайней мере отчасти, избавлены от проблем с кодировками:
независимо от кодировки, используемой клиентом, данные на сервер всегда будут
посылаться в KOI8-R (это настройка Russian Apache по умолчанию). Сервер IMAP4r1
позволит нам манипулировать почтовым ящиком, не описывая в явной форме права
доступа (связываясь с этим сервером, вы указываете свое имя и пароль, и он сам
решает все проблемы с правами). И наконец, PHP 3.0.6 с поддержкой протокола
IMAP4r1 даст возможность работать с сервером IMAP4r1, не реализуя самостоятельно
соответствующий протокол, который весьма и весьма нетривиален.
Здесь многие зададут вопрос: а почему не perl? Дело в том, что, хотя язык perl -
стандартная "рабочая лошадка" для создания Web-узлов, у него есть определенные
недостатки. Интерпретатор perl существенно сложнее интерпретатора PHP и требует
для работы больше ресурсов компьютера. Кроме того, это не интерпретатор в точном
смысле слова: он компилирует программу в специализированный псевдокод, который
затем немедленно исполняется. Если в программе есть циклы, такой подход может
дать существенный выигрыш, но если нет (как это часто бывает в простых сценариях
для Web-сервера - например, во всей нашей реализации WebMail будет всего один
цикл), накладные расходы не окупаются. Применение mod_perl - дополнительного
модуля Apache - позволяет избежать многократной перекомпиляции сценариев, но при
этом к памяти сервера предъявляются еще более высокие требования.
Коротко о PHP
Теперь несколько слов о том, что же такое PHP. Это интерпретируемый язык для
создания активных Web-страниц. Программа на PHP, подобно тексту на JavaScript,
VBScript или ASP, вставляется в HTML-файл. Начало и конец программы отмечаются
специальными скобками
. Текст вне этих скобок PHP не интерпретирует:
он передается Web-браузеру "как есть". В листинге 1 приведена реализация на PHP
"вечного" примера - счетчика. Как видите, это совершенно обычный HTML-файл,
однако в том месте, где должно стоять количество посещений, стоит сценарий на
PHP3, который в качестве результата своей работы выводит число посещений
страницы1.
Синтаксис PHP основан на синтаксисе языков Си, Java и perl и довольно подробно
описан в руководстве, которое входит в комплект поставки (его также можно взять
на узле http://www.php.net/). Способы заставить сервер правильно реагировать на
HTML-файлы со вставками на PHP, вообще говоря, различны для разных серверов, но
чаще всего бывает достаточно дать имени файла расширение .php3.
Итак, мы хотели бы иметь возможность читать и отправлять почту с помощью
Web-браузера. Видимо, будет разумно сделать интерфейс похожим, скажем, на
Netscape Messenger: окно разделено по горизонтали на две части, в верхней
находится список писем в нашем почтовом ящике на сервере, в нижней - текущее
письмо. Но перед тем как показать пользователю HTML-файл с описанием фреймов, мы
потребуем от него ввести свое имя и пароль (при неправильно введенном пароле он
получит файл, содержащий сообщение об ошибке). Эту функцию будет осуществлять
файл index.php3, показанный в листинге 2. Давайте посмотрим на него поближе.
Прежде всего стоит обратить внимание на то, что скобка
PHP-сценарий, стоит в самом начале файла. Это не случайно. Дело в том, что не
нужно сообщать браузеру о том, что документ состоит из нескольких фреймов, пока
пользователь не введет правильные имя и пароль. Чтобы запросить пароль, мы
меняем "ответ сервера" с обычного "200 OK" на "401 Auth Required"2. Это очень
просто: специально для подобных случаев в PHP предусмотрена функция Header(). Но
эта функция работает только при условии, что перед ее вызовом вывод документа
еще не был начат, и следовательно, скобка
файла...
Дальнейшее очевидно: мы смотрим на введенное пользователем имя и пароль,
пытаемся связаться с сервером IMAP4r1 и, если все прошло успешно, просто выдаем
информацию о наборе фреймов. Однако теперь во все сценарии в том подкаталоге,
где размещается наш стартовый файл index.php3, будет передаваться информация об
имени и пароле!3
Заметим также, что перед именем функции imap_open стоит символ "@". Он означает,
что сообщения о возможных ошибках при работе функции должны не выводиться в
текст на HTML (поведение PHP по умолчанию), а сохраняться в специальной
переменной. Это необходимо для того, чтобы воспользоваться функцией Header():
она не будет работать, если в тело документа уже выведен какой бы то ни было
текст, в том числе и сообщение об ошибке.
Другие файлы
Теперь нам нужно создать файлы top.php3 и main.php3, на которые имеются ссылки в
index.php3. Начнем с top.php3 (листинг 3). В нем мы строим таблицу, в которой
каждая строка соответствует одному письму и содержит его порядковый номер, тему,
имя (адрес) отправителя и дату отправки. Вся информация извлекается из
соответствующих полей заголовка письма.
При оформлении ссылок (HREF) мы должны не забыть передать номер письма (как при
работе с обычным CGI-сценарием). Файл main.php3 (листинг 4) проверяет,
установлена ли переменная $mail. Если нет, то выводится только надпись New
message с соответствующей ссылкой, а если да, добавляются еще две ссылки - Reply
и Delete. Кроме того, ссылки, встречающиеся в тексте письма, заменяются на
ссылки HTML. Файлы mail.php3, del.php3 и send.php3 (листинги 5-7) устроены
предельно просто, однако, так как файлы del.php3 и send.php3 содержат вызовы
функции Header(), они начинаются со скобки
для того, чтобы сообщение об успешной посылке или удалении письма заменилось на
основное окно автоматически, без каких бы то ни было действий со стороны
пользователя.
Чего мы не сделали
Вот и все. Служба WebMail готова (см. рисунок). Как несложно заметить, мы всего
лишь соединили готовые части - все наши шесть файлов не насчитывают и двух сотен
строк - и получили почти настоящую почтовую программу. Конечно, с ней связаны и
некоторые проблемы. Например, такие.
Что будет, если с помощью нашей WebMail попробовать читать почту одновременно
с двух компьютеров? При удалении письма на одном компьютере нумерация
"съедет", и на втором начнут происходить весьма неприятные вещи.
Как уже было замечено, на сервере с не полностью контролируемым содержанием
использование нашей программы может привести к попаданию паролей в нечистые
руки.
Нет поддержки кодировок, отличных от KOI8-R, для почты (при том, что для
Web-браузера благодаря Russian Apache поддерживаются все распространенные
кодировки).
Нет поддержки вложений.
Нет записной книжки.
Нет поддержки папок.
Впрочем, мы и не ставили себе целью создать программу, способную сравниться с
Eudora или Pegasus (но, кстати, программа Imap webMail Program - см.
http://web.horde.org/imp/ - очень близка к этому).
Сделаем еще несколько замечаний по поводу возможных решений первой и второй
проблемы. Эти проблемы связаны с обеспечением безопасности, а значит, требуют
особого внимания. Для решения первой проблемы можно передавать в дополнение к
номеру письма еще и идентификатор сообщения (MessageID); это, однако, повлечет
существенное усложнение программы, так как нужно будет организовать подробное
"разбирательство" в случае несовпадения идентификаторов, а также корректное
обновление списка писем.
Вторая проблема решается путем добавления еще одной формы на входе и передачи
имени пользователя и пароля от сценария к сценарию через URL - так, как сейчас
передается номер письма. Но тогда пароль будет появляться в адресной строке
браузера. С этим можно бороться двумя способами - либо зашифровывая пароль перед
посылкой и расшифровывая при получении, либо создав еще два фрейма: первый не
используется никак (или в нем размещается реклама, что, в общем, то же самое),
во втором происходит вся работа. Можно и скомбинировать названные способы.
Дерзайте!
ЛИСТИНГ 1 Файл counter.php3 (счетчик числа посещений Web-страницы)
vlink="#800080" alink="#FF0000">
Число посещениий:
$filename = "counter.dat";
$fp = @fopen($filename,"r");
if ($fp) {
$counter=fgets($fp,10);
fclose($fp);
} else {
$counter=0;
}
$counter++;
print $counter;
$fp = fopen($filename,"w");
if ($fp) {
$counter=fputs($fp,$counter);
fclose($fp);
}
?>
ЛИСТИНГ 2 Файл index.php3 (идентификация пользователя, установление контакта с
почтовым сервером и создание набора фреймов для вывода списка писем и текста
текущего письма)
$REALM = "Web mail";
$POPSERVER = '127.0.0.1';
$LOGERRORS = 1;
if(!isset($PHP_AUTH_USER)):
Header( "WWW-Authenticate: Basic realm=\"$REALM\"");
Header( "HTTP/1.0 401 Unauthorized");
echo "
Authorization Required\n";
exit;
else:
if(!($imap_stream=@imap_open("{127.0.0.1:143}Inbox","$PHP_
AUTH_USER","$PHP_AUTH_PW",OP_READONLY))):
Header( "WWW-Authenticate: Basic realm=\"$REALM\"");
Header( "HTTP/1.0 401 Auth Required");
echo "
Authorization Required\n";
exit;
else:
imap_close($imap_stream); ?>
|