pi3web_sapi.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Pi3Web version 2.0 |
  16. +----------------------------------------------------------------------+
  17. | This file is committed by the Pi3 development group. |
  18. | (pi3web.sourceforge.net) |
  19. | |
  20. | Author: Holger Zimmermann (zimpel@users.sourceforge.net) |
  21. +----------------------------------------------------------------------+
  22. */
  23. /* $Id$ */
  24. #define ZEND_INCLUDE_FULL_WINDOWS_HEADERS
  25. #include "php.h"
  26. #include "php_main.h"
  27. #include "php_variables.h"
  28. #include "SAPI.h"
  29. #include "php_globals.h"
  30. #include "ext/standard/info.h"
  31. #include "zend_highlight.h"
  32. #include "zend_indent.h"
  33. #include "zend_alloc.h"
  34. #include "ext/standard/basic_functions.h"
  35. #include "TSRM/TSRM.h"
  36. #include "PiAPI.h"
  37. #include "Pi3API.h"
  38. #include "pi3web_sapi.h"
  39. #define PI3WEB_SERVER_VAR_BUF_SIZE 1024
  40. int IWasLoaded=0;
  41. static void php_info_pi3web(ZEND_MODULE_INFO_FUNC_ARGS)
  42. {
  43. char variable_buf[PI3WEB_SERVER_VAR_BUF_SIZE];
  44. DWORD variable_len;
  45. LPCONTROL_BLOCK lpCB = (LPCONTROL_BLOCK) SG(server_context);
  46. PIDB *pDB = (PIDB *)lpCB->GetVariableNames(lpCB->ConnID);
  47. PIDBIterator *pIter = PIDB_getIterator( pDB, PIDBTYPE_STRING, 0, 0 );
  48. PUTS("<table border=0 cellpadding=3 cellspacing=1 width=600 align=center>\n");
  49. PUTS("<tr><th colspan=2 bgcolor=\"" PHP_HEADER_COLOR "\">Pi3Web Server Information</th></tr>\n");
  50. php_info_print_table_header(2, "Information Field", "Value");
  51. php_info_print_table_row(2, "Pi3Web SAPI module version", "$Id$");
  52. php_info_print_table_row(2, "Server Name Stamp", HTTPCore_getServerStamp());
  53. snprintf(variable_buf, 511, "%d", HTTPCore_debugEnabled());
  54. php_info_print_table_row(2, "Debug Enabled", variable_buf);
  55. PIPlatform_getCurrentDirectory( variable_buf, PI3WEB_SERVER_VAR_BUF_SIZE);
  56. php_info_print_table_row(2, "Current Path", variable_buf);
  57. if (lpCB->GetServerVariable(lpCB->ConnID, "SERVER_NAME", variable_buf, &variable_len)
  58. && variable_buf[0]) {
  59. php_info_print_table_row(2, "Main Virtual Hostname", variable_buf);
  60. };
  61. snprintf(variable_buf, 511, "%d", PIPlatform_getProcessId());
  62. php_info_print_table_row(2, "Server PID", variable_buf);
  63. php_info_print_table_row(2, "Server Platform", PIPlatform_getDescription());
  64. PUTS("</table><br />");
  65. PUTS("<table border=0 cellpadding=3 cellspacing=1 width=600 align=center>\n");
  66. PUTS("<tr><th colspan=2 bgcolor=\"" PHP_HEADER_COLOR "\">HTTP Request Information</th></tr>\n");
  67. php_info_print_table_row(2, "HTTP Request Line", lpCB->lpszReq);
  68. PUTS("<tr><th colspan=2 bgcolor=\"" PHP_HEADER_COLOR "\">HTTP Headers</th></tr>\n");
  69. php_info_print_table_header(2, "Server Variable", "Value");
  70. /* --- loop over all registered server variables --- */
  71. for(; pIter && PIDBIterator_atValidElement( pIter ); PIDBIterator_next( pIter ) )
  72. {
  73. PCHAR pKey;
  74. PIDBIterator_current( pIter, &pKey );
  75. if ( !pKey ) { /* sanity */ continue; };
  76. variable_len = PI3WEB_SERVER_VAR_BUF_SIZE;
  77. if (lpCB->GetServerVariable(lpCB->ConnID, pKey, variable_buf, &variable_len)
  78. && variable_buf[0]) {
  79. php_info_print_table_row(2, pKey, variable_buf);
  80. } else if (PIPlatform_getLastError() == PIAPI_EINVAL) {
  81. char *tmp_variable_buf;
  82. tmp_variable_buf = (char *) emalloc(variable_len);
  83. if (lpCB->GetServerVariable(lpCB->ConnID, pKey, tmp_variable_buf, &variable_len)
  84. && variable_buf[0]) {
  85. php_info_print_table_row(2, pKey, tmp_variable_buf);
  86. }
  87. efree(tmp_variable_buf);
  88. }
  89. }
  90. PUTS("</table>");
  91. }
  92. static zend_module_entry php_pi3web_module = {
  93. STANDARD_MODULE_HEADER,
  94. "PI3WEB",
  95. NULL,
  96. NULL,
  97. NULL,
  98. NULL,
  99. NULL,
  100. php_info_pi3web,
  101. NULL,
  102. STANDARD_MODULE_PROPERTIES
  103. };
  104. static int zend_pi3web_ub_write(const char *str, uint str_length TSRMLS_DC)
  105. {
  106. DWORD num_bytes = str_length;
  107. LPCONTROL_BLOCK cb;
  108. cb = (LPCONTROL_BLOCK) SG(server_context);
  109. if ( !IWasLoaded ) return 0;
  110. cb->WriteClient(cb->ConnID, (char *) str, &num_bytes, 0 );
  111. if (num_bytes != str_length)
  112. php_handle_aborted_connection();
  113. return num_bytes;
  114. }
  115. static int sapi_pi3web_header_handler(sapi_header_struct *sapi_header, sapi_headers_struct *sapi_headers TSRMLS_DC)
  116. {
  117. return SAPI_HEADER_ADD;
  118. }
  119. static void accumulate_header_length(sapi_header_struct *sapi_header, uint *total_length TSRMLS_DC)
  120. {
  121. *total_length += sapi_header->header_len+2;
  122. }
  123. static void concat_header(sapi_header_struct *sapi_header, char **combined_headers_ptr TSRMLS_DC)
  124. {
  125. memcpy(*combined_headers_ptr, sapi_header->header, sapi_header->header_len);
  126. *combined_headers_ptr += sapi_header->header_len;
  127. **combined_headers_ptr = '\r';
  128. (*combined_headers_ptr)++;
  129. **combined_headers_ptr = '\n';
  130. (*combined_headers_ptr)++;
  131. }
  132. static int sapi_pi3web_send_headers(sapi_headers_struct *sapi_headers TSRMLS_DC)
  133. {
  134. uint total_length = 2; /* account for the trailing \r\n */
  135. char *combined_headers, *combined_headers_ptr;
  136. LPCONTROL_BLOCK lpCB = (LPCONTROL_BLOCK) SG(server_context);
  137. sapi_header_struct default_content_type;
  138. if ( !IWasLoaded ) return SAPI_HEADER_SENT_SUCCESSFULLY;
  139. if (SG(sapi_headers).send_default_content_type) {
  140. sapi_get_default_content_type_header(&default_content_type TSRMLS_CC);
  141. accumulate_header_length(&default_content_type, (void *) &total_length TSRMLS_CC);
  142. }
  143. zend_llist_apply_with_argument(&SG(sapi_headers).headers, (llist_apply_with_arg_func_t) accumulate_header_length, (void *) &total_length TSRMLS_CC);
  144. /* Generate headers */
  145. combined_headers = (char *) emalloc(total_length+1);
  146. combined_headers_ptr = combined_headers;
  147. if (SG(sapi_headers).send_default_content_type) {
  148. concat_header(&default_content_type, (void *) &combined_headers_ptr TSRMLS_CC);
  149. sapi_free_header(&default_content_type); /* we no longer need it */
  150. }
  151. zend_llist_apply_with_argument(&SG(sapi_headers).headers, (llist_apply_with_arg_func_t) concat_header, (void *) &combined_headers_ptr TSRMLS_CC);
  152. *combined_headers_ptr++ = '\r';
  153. *combined_headers_ptr++ = '\n';
  154. *combined_headers_ptr = 0;
  155. lpCB->dwHttpStatusCode = SG(sapi_headers).http_response_code;
  156. lpCB->SendHeaderFunction(lpCB->ConnID, &total_length, (LPDWORD) combined_headers);
  157. efree(combined_headers);
  158. if (SG(sapi_headers).http_status_line) {
  159. efree(SG(sapi_headers).http_status_line);
  160. SG(sapi_headers).http_status_line = 0;
  161. }
  162. return SAPI_HEADER_SENT_SUCCESSFULLY;
  163. }
  164. static int php_pi3web_startup(sapi_module_struct *sapi_module)
  165. {
  166. if (php_module_startup(sapi_module, &php_pi3web_module, 1)==FAILURE) {
  167. return FAILURE;
  168. } else {
  169. return SUCCESS;
  170. }
  171. }
  172. static int sapi_pi3web_read_post(char *buffer, uint count_bytes TSRMLS_DC)
  173. {
  174. LPCONTROL_BLOCK lpCB = (LPCONTROL_BLOCK) SG(server_context);
  175. DWORD read_from_buf=0;
  176. DWORD read_from_input=0;
  177. DWORD total_read=0;
  178. if ((DWORD)SG(read_post_bytes) < lpCB->cbAvailable) {
  179. read_from_buf = MIN(lpCB->cbAvailable-SG(read_post_bytes), count_bytes);
  180. memcpy(buffer, lpCB->lpbData+SG(read_post_bytes), read_from_buf);
  181. total_read += read_from_buf;
  182. }
  183. if (read_from_buf<count_bytes
  184. && (SG(read_post_bytes)+read_from_buf) < lpCB->cbTotalBytes) {
  185. DWORD cbRead=0, cbSize;
  186. read_from_input = MIN(count_bytes-read_from_buf, lpCB->cbTotalBytes-SG(read_post_bytes)-read_from_buf);
  187. while (cbRead < read_from_input) {
  188. cbSize = read_from_input - cbRead;
  189. if (!lpCB->ReadClient(lpCB->ConnID, buffer+read_from_buf+cbRead, &cbSize) || cbSize==0) {
  190. break;
  191. }
  192. cbRead += cbSize;
  193. }
  194. total_read += cbRead;
  195. }
  196. /* removed after re-testing POST with Pi3Web 2.0.2 */
  197. /* SG(read_post_bytes) += total_read; */
  198. return total_read;
  199. }
  200. static char *sapi_pi3web_read_cookies(TSRMLS_D)
  201. {
  202. LPCONTROL_BLOCK lpCB = (LPCONTROL_BLOCK) SG(server_context);
  203. char variable_buf[PI3WEB_SERVER_VAR_BUF_SIZE];
  204. DWORD variable_len = PI3WEB_SERVER_VAR_BUF_SIZE;
  205. if (lpCB->GetServerVariable(lpCB->ConnID, "HTTP_COOKIE", variable_buf, &variable_len)) {
  206. return estrndup(variable_buf, variable_len);
  207. } else if (PIPlatform_getLastError()==PIAPI_EINVAL) {
  208. char *tmp_variable_buf = (char *) emalloc(variable_len+1);
  209. if (lpCB->GetServerVariable(lpCB->ConnID, "HTTP_COOKIE", tmp_variable_buf, &variable_len)) {
  210. tmp_variable_buf[variable_len] = 0;
  211. return tmp_variable_buf;
  212. } else {
  213. efree(tmp_variable_buf);
  214. }
  215. }
  216. return NULL;
  217. }
  218. static void init_request_info(LPCONTROL_BLOCK lpCB TSRMLS_DC)
  219. {
  220. SG(server_context) = lpCB;
  221. SG(request_info).request_method = lpCB->lpszMethod;
  222. SG(request_info).query_string = lpCB->lpszQueryString;
  223. SG(request_info).path_translated = lpCB->lpszPathTranslated;
  224. SG(request_info).request_uri = lpCB->lpszUri;
  225. SG(request_info).content_type = lpCB->lpszContentType;
  226. SG(request_info).content_length = lpCB->cbTotalBytes;
  227. SG(request_info).auth_user = (lpCB->lpszUser) ? (char *)estrdup((const char *)(lpCB->lpszUser)) : 0;
  228. SG(request_info).auth_password = (lpCB->lpszPassword) ? (char *)estrdup((const char *)(lpCB->lpszPassword)) : 0;
  229. SG(sapi_headers).http_response_code = 200;
  230. }
  231. static void sapi_pi3web_register_variables(zval *track_vars_array TSRMLS_DC)
  232. {
  233. char static_variable_buf[PI3WEB_SERVER_VAR_BUF_SIZE];
  234. char *variable_buf;
  235. DWORD variable_len = PI3WEB_SERVER_VAR_BUF_SIZE;
  236. LPCONTROL_BLOCK lpCB = (LPCONTROL_BLOCK) SG(server_context);
  237. PIDB *pDB = (PIDB *)lpCB->GetVariableNames(lpCB->ConnID);
  238. PIDBIterator *pIter = PIDB_getIterator( pDB, PIDBTYPE_STRING, 0, 0 );
  239. /* --- loop over all registered server variables --- */
  240. for(; pIter && PIDBIterator_atValidElement( pIter ); PIDBIterator_next( pIter ) )
  241. {
  242. PCHAR pKey;
  243. PIDBIterator_current( pIter, &pKey );
  244. if ( !pKey ) { /* sanity */ continue; };
  245. variable_len = PI3WEB_SERVER_VAR_BUF_SIZE;
  246. if (lpCB->GetServerVariable(lpCB->ConnID, pKey, static_variable_buf, &variable_len)
  247. && (variable_len > 1)) {
  248. php_register_variable(pKey, static_variable_buf, track_vars_array TSRMLS_CC);
  249. } else if (PIPlatform_getLastError()==PIAPI_EINVAL) {
  250. variable_buf = (char *) emalloc(variable_len);
  251. if (lpCB->GetServerVariable(lpCB->ConnID, pKey, variable_buf, &variable_len)) {
  252. php_register_variable(pKey, variable_buf, track_vars_array TSRMLS_CC);
  253. }
  254. efree(variable_buf);
  255. }
  256. }
  257. /* PHP_SELF support */
  258. variable_len = PI3WEB_SERVER_VAR_BUF_SIZE;
  259. if (lpCB->GetServerVariable(lpCB->ConnID, "SCRIPT_NAME", static_variable_buf, &variable_len)
  260. && (variable_len > 1)) {
  261. php_register_variable("PHP_SELF", static_variable_buf, track_vars_array TSRMLS_CC);
  262. }
  263. }
  264. static sapi_module_struct pi3web_sapi_module = {
  265. "pi3web", /* name */
  266. "PI3WEB", /* pretty name */
  267. php_pi3web_startup, /* startup */
  268. php_module_shutdown_wrapper, /* shutdown */
  269. NULL, /* activate */
  270. NULL, /* deactivate */
  271. zend_pi3web_ub_write, /* unbuffered write */
  272. NULL, /* flush */
  273. NULL, /* get uid */
  274. NULL, /* getenv */
  275. php_error, /* error handler */
  276. sapi_pi3web_header_handler, /* header handler */
  277. sapi_pi3web_send_headers, /* send headers handler */
  278. NULL, /* send header handler */
  279. sapi_pi3web_read_post, /* read POST data */
  280. sapi_pi3web_read_cookies, /* read Cookies */
  281. sapi_pi3web_register_variables, /* register server variables */
  282. NULL, /* Log message */
  283. NULL, /* Get request time */
  284. NULL, /* Child terminate */
  285. STANDARD_SAPI_MODULE_PROPERTIES
  286. };
  287. MODULE_API DWORD PHP5_wrapper(LPCONTROL_BLOCK lpCB)
  288. {
  289. zend_file_handle file_handle = {0};
  290. int iRet = PIAPI_COMPLETED;
  291. TSRMLS_FETCH();
  292. zend_first_try {
  293. file_handle.filename = lpCB->lpszFileName;
  294. file_handle.free_filename = 0;
  295. file_handle.type = ZEND_HANDLE_FILENAME;
  296. file_handle.opened_path = NULL;
  297. init_request_info(lpCB TSRMLS_CC);
  298. php_request_startup(TSRMLS_C);
  299. switch ( lpCB->dwBehavior ) {
  300. case PHP_MODE_STANDARD:
  301. iRet = ( php_execute_script( &file_handle TSRMLS_CC ) ) ?
  302. PIAPI_COMPLETED : PIAPI_ERROR;
  303. break;
  304. case PHP_MODE_HIGHLIGHT: {
  305. zend_syntax_highlighter_ini syntax_highlighter_ini;
  306. if ( open_file_for_scanning( &file_handle TSRMLS_CC ) == SUCCESS )
  307. {
  308. php_get_highlight_struct( &syntax_highlighter_ini );
  309. zend_highlight( &syntax_highlighter_ini TSRMLS_CC );
  310. }
  311. else
  312. {
  313. iRet = PIAPI_ERROR;
  314. };
  315. };
  316. break;
  317. case PHP_MODE_INDENT: {
  318. sapi_header_line ctr = {0};
  319. ctr.line = "Content-Type: text/plain";
  320. ctr.line_len = strlen(ctr.line);
  321. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  322. }
  323. if ( open_file_for_scanning( &file_handle TSRMLS_CC ) == SUCCESS )
  324. {
  325. zend_indent();
  326. }
  327. else
  328. {
  329. iRet = PIAPI_ERROR;
  330. };
  331. break;
  332. case PHP_MODE_LINT:
  333. iRet = (php_lint_script(&file_handle TSRMLS_CC) == SUCCESS) ?
  334. PIAPI_COMPLETED : PIAPI_ERROR;
  335. break;
  336. default:
  337. iRet = PIAPI_ERROR;;
  338. }
  339. if (SG(request_info).cookie_data) {
  340. efree(SG(request_info).cookie_data);
  341. };
  342. php_request_shutdown(NULL);
  343. } zend_catch {
  344. iRet = PIAPI_ERROR;
  345. } zend_end_try();
  346. return iRet;
  347. }
  348. MODULE_API BOOL PHP5_startup() {
  349. tsrm_startup(1, 1, 0, NULL);
  350. sapi_startup(&pi3web_sapi_module);
  351. if (pi3web_sapi_module.startup) {
  352. pi3web_sapi_module.startup(&pi3web_sapi_module);
  353. };
  354. IWasLoaded = 1;
  355. return IWasLoaded;
  356. };
  357. MODULE_API BOOL PHP5_shutdown() {
  358. if (pi3web_sapi_module.shutdown) {
  359. pi3web_sapi_module.shutdown(&pi3web_sapi_module);
  360. };
  361. sapi_shutdown();
  362. tsrm_shutdown();
  363. IWasLoaded = 0;
  364. return !IWasLoaded;
  365. };
  366. /*
  367. * Local variables:
  368. * tab-width: 4
  369. * c-basic-offset: 4
  370. * End:
  371. */