Использование соккетов в PHP
 
(работа с Usenet-новостными серверами)

Armel Fauveau, Denis Roshchin.

PHP может открывать соккеты на локальной или удаленной машине. В данной статье будет рассмотрен пример использования соккетов для: соединения с Usenet-новостным сервером, ведения диалога с ним и скачивания некоторых статей.

Открываем соккет.

Для открываения соккета используется функция fsockopen(). Эта функция доступна, как в PHP3, так и PHP4. Вызов функции имеет следующий вид:

int fsockopen (string hostname, int port [, int errno [, string errstr [, double timeout]]])

Для UDP соединения, надо определить протокол: udp://hostname.

Больше информации о функции fsockopen() можно узнав здесь: http://www.php.net/manual/function.fsockopen.php

NNTP- протокол (Network News Transfer Protocol)

Для доступа к новостному usenet-серверу мы будем использовать NNTP-протокол.

Этот протокол детально описан в RFC977 (Request For Comment number 977). Полное описание присутствует в интернете: http://www.w3.org/Protocols/rfc977/rfc977.html

Этот документ детально описывает процедуру соединения и диалога с сервером.

Соединение (Connecting)

Для соединения с NNTP-сервером нам необходимо знать его имя (или IP-адрес) и порт.Так же необходимо указывать таймер, чтобы в случае невозможности подсоединения к серверу не “заморозили” application.


$cfgServer
= "your.news.host";
$cfgPort = 119;
$cfgTimeOut = 10;

// open a socket
if(!$cfgTimeOut)
// without timeout
$usenet_handle = fsockopen($cfgServer, $cfgPort);
else
// with timeout
$usenet_handle = fsockopen($cfgServer, $cfgPort, &$errno, &$errstr, $cfgTimeOut);

if(!
$usenet_handle) {
echo
"Connexion failed\n";
exit();
}
else {
echo
"Connected\n";
$tmp = fgets($usenet_handle, 1024);
}

?>

Ведение диалога с сервером.

Итак, теперь мы присоединились к серверу и можем вести диалог с ним, используя ранее открытый соккет. Для примера, попробуем достать десять последних сообщений с какой-либо группы. В RFC977 описано, что первый шаг – выбрать группу с помощью комманды:

GROUP ggg

Обязательный парамтор - ggg - имя группы, которую мы хотим выбрать (например, "net.news"). Список существующих групп может быть получен с помощью комманды LIST.Удачный выбор группы будет подтвержден ответом сервера, где будет сообщаться колличество новых, старых статей и общее колличество.

chrome:~$ telnet my.news.host 119 Trying aa.bb.cc.dd... Connected to my.news.host. Escape character is '^]'. 200 my.news.host InterNetNews NNRP server INN 2.2.2 13-Dec-1999 ready (posting ok). GROUP alt.test 211 232 222996 223235 alt.test quit 205 .

После получения комманды "GROUP alt.test", новостной сервер ответил "211 232 222996 223235 alt.test". 211 – определенный RFC спецификацией код (говоря обычным языком – 212 – означает, что команда была завершена с положительным результатом – смотрите документацию RFC для более полной характеристики). Следующая цифра – 232 – колличество имеющихся в текущий момент новых статей. 222996 – старых. 223235 – всего статей. 232+222996 не равно 223235. Почему? Возможно, недостающие семь статей были каким-либо образом удалены модератором или самим автором.

В зависимости от сервера (public или private вас могут попросить идентифицироваться. Так-же возможно, что идентификация понадобиться только при написании своих сообщений, а чтение может производиться без этого.


//$cfgUser = "xxxxxx";
//$cfgPasswd = "yyyyyy";
$cfgNewsGroup = "alt.php";

// identification required on private server
if($cfgUser) {
fputs($usenet_handle, "AUTHINFO USER ".$cfgUser."\n");
$tmp = fgets($usenet_handle, 1024);

fputs($usenet_handle, "AUTHINFO PASS ".$cfgPasswd."\n");
$tmp = fgets($usenet_handle, 1024);

// check error

if($tmp != "281 Ok\r\n") {
echo
"502 Authentication error\n";
exit();
}
}

// select newsgroup

fputs($usenet_handle, "GROUP ".$cfgNewsGroup."\n");
$tmp = fgets($usenet_handle, 1024);

if(
$tmp == "480 Authentication required for command\r\n") {
echo
"$tmp\n";
exit();
}

$info = split(" ", $tmp);
$first = $info[2];
$last = $info[3];

print
"First : $first\n";
print
"Last : $last\n";


?>

 

Скачивание некоторых статей

Теперь мы имеем номер последней статьи, так что сейчас нам не составит труда скачать десять последних статей. RFC977 спецификация допускает использование комманды ARTICLE, как с номером статьи, так и Message ID (Уникальный Номер Сообщения).

Будтье внимательны здесь – номер статьи отличен от Message ID. Если статья опубликованна на нескольких серверах, то она несомненно будет иметь разный номер оба раза, но одинаковый Message ID. Грубо говоря, номер статьи – присваивается каждый раз по-новому на сервере, и может меняться со временем; Message ID – у каждой статьи уникальный.


$cfgLimit
= 10;

// upload last articles

$boucle=$last-$cfgLimit;

while (
$boucle <= $last) {

set_time_limit(0);

fputs($usenet_handle, "ARTICLE $boucle\n");

$article="";
$tmp = fgets($usenet_handle, 4096);
if(
substr($tmp,0,3) != "220") {
echo
"+----------------------+\n";
echo
"Error on article $boucle\n";
echo
"+----------------------+\n";
}
else {
while(
$tmp!=".\r\n") {
$tmp = fgets($usenet_handle, 4096);
$article = $article.$tmp;
}

echo
"+----------------------+\n";
echo
"Article $boucle\n";
echo
"+----------------------+\n";
echo
"$article\n";
}

$boucle++;
}

?>

Так-же благодаря комманде HEAD возможно получить только хэадер (header) сообщения или-же только текст, используя команду BODY.

 

Отсоединяемся от сервера

Чтобы закрыть сессию с NNTP-сервером, просто закройте соккет используя fclose() (аналагично закрытию фаила).

Больше информации о функции fclose() можно узнать здес: http://www.php.net/manual/function.fclose.php


// close connexion

fclose($usenet_handle);

?>

Заключение

Мы только что видели как открыть, использовать и затем закрыть соккет – для соединения с NNTP-сервером и получения некоторых статей из новостных групп.

Для опубликования сообщения необходимо использовать POST комманду.

Примеры приложений работующих с новостными группами можно найти здесь: http://www.phpindex.com/ng/

Авторы статьи :
  • Armel Fauveau
  • Denis Roshchin
  •  
    Автор: Armel Fauveau, Denis Roshchin
     
    Оригинал статьи: http://www.woweb.ru/publ/59-1-0-9