Я нахожусь на пути разработки простого бота для irc в php.
Половина выше — реализация IRC-бота с нуля (подключение сокета… и т. Д.)
Функция, которую я хочу добавить, это «Расписание уведомлений»
Когда наступает конкретное время, некоторые сообщения отправляются.
Например,
когда время Tue Apr 19 16:32
приходит, какое-то уведомление отправлено.
Так что если вы установите что-то вроде (date("D") == "Tue" && date("H") == 15)
,
Предполагается, что отправка сообщения будет продолжаться до 16:00.
Но как только бот попадает в канал, он прекращает отправку сообщений.
Я предполагаю, что это вызвано соединением сокета, однако я действительно не знаю ключ.
<?php
// Time zone setting
date_default_timezone_set('Asia/Tokyo');
// Our bot's configuration parameters.
$server = '192.168.59.103';
$port = 6667;
$nickname = 'Bot';
$ident = 'Bot';
$gecos = 'Bot v1.0';
$channel = '#bot-test';// Connect to the network
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
$error = socket_connect($socket, $server, $port);
// Add some error handling in case connection was not successful.
if ($socket === false){
$errorCode = socket_last_error();
$errorString = socket_strerror($errorCode);
die("Error $errorCode: $errorString \n");
}
// Send the registration info.
socket_write($socket, "NICK $nickname\r\n");
socket_write($socket, "USER $ident * 8 :$gecos\r\n");
// Finally, loop until the socket closes.
while (is_resource($socket)) {
// Fetch the data from the socket
$data = trim(socket_read($socket, 1024, PHP_NORMAL_READ));
echo $data . "\n";
// Splitting the data into chunks
$d = explode(' ', $data);
// Padding the array avoids ugly undefined offset erros.
$d = array_pad ($d, 10, '');
// Our ping handler.
// Ping: $servername.
if ($d[0] === 'PING') {
socket_write($socket, 'PONG ' . $d[1] . "\r\n");
}
if ($d[1] === '376' || $d[1] === '422') {
socket_write($socket, 'JOIN ' . $channel . "\r\n");
}
// Bot collections
// "$d" parameter format
// [0] [1] [2] [3]
// :Nickname!ident@hostname PRIVMSG #bot-test :@arukas.
// Scheduler bot
if (date("D") == "Tue" && date("H") == 15) {
$saying = "REALLY SLEEPY!!";
socket_write($socket, 'PRIVMSG ' . "CIRC1989" . " :$saying\r\n");
}
}
Ваш код поврежден в части логики чтения / записи — Ваш текущий код всегда предполагает, что что-то прочитано (будет спать до тех пор, пока что-то не произойдет), а затем что-то напишет. Вам нужно добавить буферы и использовать опрос / выбор. Я полагаю, php, по крайней мере, один из них.
Псевдокод, который должен работать:
readbuffer[]
writebuffer[]
while (no_error)
{
if (writebuffer not empty)
{
select(socket, want_to_write, want_to_read, timeout_until_next_event);
} else {
select(socket, 0, want_to_read, timeout_until_next_event);
}
if (select return can write)
{
retval = write(socket, writebuffer);
if (no_error)
writebuffer.removefromstart(retval);
}
if (select return can read)
{
retval = read(socket, readbuffer + offset already filled);
if (no_error)
{
parse as much as possible in readbuffer, removing data as parsed;
}
}
check_timers();
}
Других решений пока нет …