Перезапись URL-адреса G-WAN для домашней страницы не работает

Я пытаюсь переписать URL-адрес для домашней страницы сайта. Вот упрощенная версия моего обработчика.

int init(int argc, char *argv[])
{
   u32 *states = (u32*)get_env(argv, US_HANDLER_STATES);
   *states =  (1 << HDL_AFTER_READ);
   return 0;
}

int main(int argc, char *argv[])
{
   xbuf_t *read_xbuf = (xbuf_t*)get_env(argv, READ_XBUF);
   xbuf_replfrto(read_xbuf, read_xbuf->ptr, read_xbuf->ptr + 16, " / ", " /?home ");
   return 255;
}

void clean(int argc, char *argv[]) 
{}

По сути, это просто замена «/» на «/?home». Поэтому, когда пользователь загружает «www.domain.com», он дает ему содержимое «home.c». Вот результат перезаписи. Все выглядит правильно, я не уверен, что вызывает проблему.

Оригинальный запрос:

GET / HTTP/1.1
Host: localhost:8000
User-Agent: Mozilla/5.0 (X11; Ubuntu; Linux i686; rv:12.0) Gecko/20100101 Firefox/12.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us,en;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive

Переписанный запрос:

GET /?home HTTP/1.1
Host: localhost:8000
User-Agent: Mozilla/5.0 (X11; Ubuntu; Linux i686; rv:12.0) Gecko/20100101 Firefox/12.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us,en;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive

После перепрошивки вот такой результат.

GET http://localhost:8000/


 -- response --
0 

person Richard Heath    schedule 20.11.2012    source источник


Ответы (2)


Вы забыли return 255; в функции main().

Помните, что коды возврата обработчика соединения имеют значение:

return 255; // execute next connection step
return   0; // close connection

Кроме того, даже если он останется пустым, вы должны объявить функцию clean() в обработчике соединения:

void clean(int argc, char *argv[]) 
{}

Наконец, вы ДОЛЖНЫ проверить состояние обработчика в main().

Это дает нам протестированный код ниже:

// ============================================================================
// Handler C script for the G-WAN Web Application Server (http://gwan.ch/)
// ----------------------------------------------------------------------------
// main.c: basic rewrite example
// ============================================================================
#include "gwan.h"    // G-WAN exported functions

#include <stdio.h> // puts(), printf()
// ----------------------------------------------------------------------------
// init() will initialize your data structures, load your files, etc.
// ----------------------------------------------------------------------------
// init() should return -1 if failure (to allocate memory for example)
int init(int argc, char *argv[])
{
   // define which handler states we want to be notified in main():
   // enum HANDLER_ACT { 
   //  HDL_INIT = 0, 
   //  HDL_AFTER_ACCEPT, // just after accept (only client IP address setup)
   //  HDL_AFTER_READ,   // each time a read was done until HTTP request OK
   //  HDL_BEFORE_PARSE, // HTTP verb/URI validated but HTTP headers are not 
   //  HDL_AFTER_PARSE,  // HTTP headers validated, ready to build reply
   //  HDL_BEFORE_WRITE, // after a reply was built, but before it is sent
   //  HDL_HTTP_ERRORS,  // when G-WAN is going to reply with an HTTP error
   //  HDL_CLEANUP };
   u32 *states = (u32*)get_env(argv, US_HANDLER_STATES);
   *states =  (1 << HDL_AFTER_READ);
   return 0;
}
// ----------------------------------------------------------------------------
// clean() will free any allocated memory and possibly log summarized stats
// ----------------------------------------------------------------------------
void clean(int argc, char *argv[])
{}
// ----------------------------------------------------------------------------
// main() does the job for all the connection states below:
// (see 'HTTP_Env' in gwan.h for all the values you can fetch with get_env())
// ----------------------------------------------------------------------------
int main(int argc, char *argv[])
{
   // HDL_HTTP_ERRORS return values:
   //   0: Close the client connection
   //   2: Send a server reply based on a custom reply buffer
   // 255: Continue (send a reply based on the request HTTP code)
   const int state = (long)argv[0];
   if(state != HDL_AFTER_READ)
      return 255;

   xbuf_t *read_xbuf = (xbuf_t*)get_env(argv, READ_XBUF);
   printf("req_1: %.20s\n", read_xbuf->ptr);
   xbuf_replfrto(read_xbuf, read_xbuf->ptr, read_xbuf->ptr + 16, " / ", " /?home ");
   printf("req_2: %.20s\n-------------------\n\n", read_xbuf->ptr);

   return 255; // continue G-WAN's default execution path
}
// ============================================================================
// End of Source Code
// ============================================================================
person Gil    schedule 21.11.2012
comment
Я изменил свой пример кода. Я получил чистый и вернул 255 на свой исходный код. Но все та же проблема. - person Richard Heath; 21.11.2012
comment
Я переписал ответ выше, чтобы добавить полный (проверенный) исходный код. - person Gil; 21.11.2012
comment
Все такой же. Я использовал ваш код, но я использую виртуальную машину. Это может вызвать проблему. Вечером попробую из дома. - person Richard Heath; 21.11.2012
comment
Я получил req_2: GET /?home HTTP/1. после перезаписи, но все еще ничего. - person Richard Heath; 25.11.2012
comment
Доказательство того, что обработчик перезаписи работает. Теперь проверьте файлы журнала, чтобы узнать, почему файл не обслуживается (404: не найден?). - person Gil; 26.11.2012
comment
[Пятница, 09 ноября, 01:19:43 2012 GMT] [клиент 127.0.0.1] 0: READ:Tmo 565 байт: - Это сообщение об ошибке. home.c существует и отлично работает. - person Richard Heath; 26.11.2012
comment
Это означает, что HTTP-запрос был неполным (или поврежден перезаписью). Вы должны использовать printf(req_2: %*v\n-------------------\n\n, read_xbuf-›len, read_xbuf-›ptr); чтобы увидеть, где находится шестнадцатеричный дамп, созданный этой командой. - person Gil; 27.11.2012
comment
Вы используете 3.3.28? Я использовал ваш проверенный код, но он все еще не работает на моей машине. Я предполагаю, что это связано с длиной запроса. Мое переписывание работает, когда я не добавляю char в результат, например /blog, в /?log. Но из/в/?домой это не так. - person Richard Heath; 28.11.2012
comment
Я использовал версию 3.11.9, чтобы исправить вашу перезапись версии 3.3, используйте шестнадцатеричный дамп %*v выше, чтобы увидеть, где перезапись нарушает запрос (или дождитесь следующего общедоступного выпуска, который работает для меня). - person Gil; 28.11.2012
comment
Я сделал сравнение по переписанному запросу, и это правильно. Я просто подожду следующего релиза. Возможно, это просто проблема 3.3. Я попробую это снова в следующем выпуске. Спасибо за помощь. - person Richard Heath; 29.11.2012

Эта проблема решена в G-WAN версии 4+.

person Richard Heath    schedule 13.02.2013