1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773 |
- /*
- +----------------------------------------------------------------------+
- | Copyright (c) The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | https://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Author: Moriyoshi Koizumi <moriyoshi@php.net> |
- | Xinchen Hui <laruence@php.net> |
- +----------------------------------------------------------------------+
- */
- #include <stdio.h>
- #include <stdlib.h>
- #include <fcntl.h>
- #include <assert.h>
- #ifdef PHP_WIN32
- # include <process.h>
- # include <io.h>
- # include "win32/time.h"
- # include "win32/signal.h"
- # include "win32/php_registry.h"
- # include <sys/timeb.h>
- #else
- # include "php_config.h"
- #endif
- #ifdef __riscos__
- #include <unixlib/local.h>
- #endif
- #if HAVE_SYS_TIME_H
- #include <sys/time.h>
- #endif
- #if HAVE_UNISTD_H
- #include <unistd.h>
- #endif
- #include <signal.h>
- #include <locale.h>
- #if HAVE_DLFCN_H
- #include <dlfcn.h>
- #endif
- #include "SAPI.h"
- #include "php.h"
- #include "php_ini.h"
- #include "php_main.h"
- #include "php_globals.h"
- #include "php_variables.h"
- #include "zend_hash.h"
- #include "zend_modules.h"
- #include "fopen_wrappers.h"
- #include "http_status_codes.h"
- #include "zend_compile.h"
- #include "zend_execute.h"
- #include "zend_highlight.h"
- #include "zend_exceptions.h"
- #include "php_getopt.h"
- #ifndef PHP_WIN32
- # define php_select(m, r, w, e, t) select(m, r, w, e, t)
- # define SOCK_EINVAL EINVAL
- # define SOCK_EAGAIN EAGAIN
- # define SOCK_EINTR EINTR
- # define SOCK_EADDRINUSE EADDRINUSE
- #else
- # include "win32/select.h"
- # define SOCK_EINVAL WSAEINVAL
- # define SOCK_EAGAIN WSAEWOULDBLOCK
- # define SOCK_EINTR WSAEINTR
- # define SOCK_EADDRINUSE WSAEADDRINUSE
- #endif
- #include "ext/standard/file.h" /* for php_set_sock_blocking() :-( */
- #include "zend_smart_str.h"
- #include "ext/standard/html.h"
- #include "ext/standard/url.h" /* for php_raw_url_decode() */
- #include "ext/standard/php_string.h" /* for php_dirname() */
- #include "ext/date/php_date.h" /* for php_format_date() */
- #include "php_network.h"
- #include "php_http_parser.h"
- #include "php_cli_server.h"
- #include "php_cli_server_arginfo.h"
- #include "mime_type_map.h"
- #include "php_cli_process_title.h"
- #include "php_cli_process_title_arginfo.h"
- #define OUTPUT_NOT_CHECKED -1
- #define OUTPUT_IS_TTY 1
- #define OUTPUT_NOT_TTY 0
- #if HAVE_FORK
- # include <sys/wait.h>
- static pid_t php_cli_server_master;
- static pid_t *php_cli_server_workers;
- static zend_long php_cli_server_workers_max;
- #endif
- typedef struct php_cli_server_poller {
- fd_set rfds, wfds;
- struct {
- fd_set rfds, wfds;
- } active;
- php_socket_t max_fd;
- } php_cli_server_poller;
- typedef struct php_cli_server_request {
- enum php_http_method request_method;
- int protocol_version;
- char *request_uri;
- size_t request_uri_len;
- char *vpath;
- size_t vpath_len;
- char *path_translated;
- size_t path_translated_len;
- char *path_info;
- size_t path_info_len;
- char *query_string;
- size_t query_string_len;
- HashTable headers;
- HashTable headers_original_case;
- char *content;
- size_t content_len;
- const char *ext;
- size_t ext_len;
- zend_stat_t sb;
- } php_cli_server_request;
- typedef struct php_cli_server_chunk {
- struct php_cli_server_chunk *next;
- enum php_cli_server_chunk_type {
- PHP_CLI_SERVER_CHUNK_HEAP,
- PHP_CLI_SERVER_CHUNK_IMMORTAL
- } type;
- union {
- struct { void *block; char *p; size_t len; } heap;
- struct { const char *p; size_t len; } immortal;
- } data;
- } php_cli_server_chunk;
- typedef struct php_cli_server_buffer {
- php_cli_server_chunk *first;
- php_cli_server_chunk *last;
- } php_cli_server_buffer;
- typedef struct php_cli_server_content_sender {
- php_cli_server_buffer buffer;
- } php_cli_server_content_sender;
- typedef struct php_cli_server_client {
- struct php_cli_server *server;
- php_socket_t sock;
- struct sockaddr *addr;
- socklen_t addr_len;
- char *addr_str;
- size_t addr_str_len;
- php_http_parser parser;
- unsigned int request_read:1;
- char *current_header_name;
- size_t current_header_name_len;
- unsigned int current_header_name_allocated:1;
- char *current_header_value;
- size_t current_header_value_len;
- enum { HEADER_NONE=0, HEADER_FIELD, HEADER_VALUE } last_header_element;
- size_t post_read_offset;
- php_cli_server_request request;
- unsigned int content_sender_initialized:1;
- php_cli_server_content_sender content_sender;
- int file_fd;
- } php_cli_server_client;
- typedef struct php_cli_server {
- php_socket_t server_sock;
- php_cli_server_poller poller;
- int is_running;
- char *host;
- int port;
- int address_family;
- char *document_root;
- size_t document_root_len;
- char *router;
- size_t router_len;
- socklen_t socklen;
- HashTable clients;
- HashTable extension_mime_types;
- } php_cli_server;
- typedef struct php_cli_server_http_response_status_code_pair {
- int code;
- const char *str;
- } php_cli_server_http_response_status_code_pair;
- static php_cli_server_http_response_status_code_pair template_map[] = {
- { 400, "<h1>%s</h1><p>Your browser sent a request that this server could not understand.</p>" },
- { 404, "<h1>%s</h1><p>The requested resource <code class=\"url\">%s</code> was not found on this server.</p>" },
- { 500, "<h1>%s</h1><p>The server is temporarily unavailable.</p>" },
- { 501, "<h1>%s</h1><p>Request method not supported.</p>" }
- };
- #define PHP_CLI_SERVER_LOG_PROCESS 1
- #define PHP_CLI_SERVER_LOG_ERROR 2
- #define PHP_CLI_SERVER_LOG_MESSAGE 3
- static int php_cli_server_log_level = 3;
- #if HAVE_UNISTD_H || defined(PHP_WIN32)
- static int php_cli_output_is_tty = OUTPUT_NOT_CHECKED;
- #endif
- static const char php_cli_server_request_error_unexpected_eof[] = "Unexpected EOF";
- static size_t php_cli_server_client_send_through(php_cli_server_client *client, const char *str, size_t str_len);
- static php_cli_server_chunk *php_cli_server_chunk_heap_new_self_contained(size_t len);
- static void php_cli_server_buffer_append(php_cli_server_buffer *buffer, php_cli_server_chunk *chunk);
- static void php_cli_server_logf(int type, const char *format, ...);
- static void php_cli_server_log_response(php_cli_server_client *client, int status, const char *message);
- ZEND_DECLARE_MODULE_GLOBALS(cli_server)
- /* {{{ static char php_cli_server_css[]
- * copied from ext/standard/info.c
- */
- static const char php_cli_server_css[] = "<style>\n" \
- "body { background-color: #fcfcfc; color: #333333; margin: 0; padding:0; }\n" \
- "h1 { font-size: 1.5em; font-weight: normal; background-color: #9999cc; min-height:2em; line-height:2em; border-bottom: 1px inset black; margin: 0; }\n" \
- "h1, p { padding-left: 10px; }\n" \
- "code.url { background-color: #eeeeee; font-family:monospace; padding:0 2px;}\n" \
- "</style>\n";
- /* }}} */
- #ifdef PHP_WIN32
- int php_cli_server_get_system_time(char *buf) {
- struct _timeb system_time;
- errno_t err;
- if (buf == NULL) {
- return -1;
- }
- _ftime(&system_time);
- err = ctime_s(buf, 52, &(system_time.time) );
- if (err) {
- return -1;
- }
- return 0;
- }
- #else
- int php_cli_server_get_system_time(char *buf) {
- struct timeval tv;
- struct tm tm;
- gettimeofday(&tv, NULL);
- /* TODO: should be checked for NULL tm/return value */
- php_localtime_r(&tv.tv_sec, &tm);
- php_asctime_r(&tm, buf);
- return 0;
- }
- #endif
- static void char_ptr_dtor_p(zval *zv) /* {{{ */
- {
- pefree(Z_PTR_P(zv), 1);
- } /* }}} */
- static char *get_last_error(void) /* {{{ */
- {
- return pestrdup(strerror(errno), 1);
- } /* }}} */
- static int status_comp(const void *a, const void *b) /* {{{ */
- {
- const http_response_status_code_pair *pa = (const http_response_status_code_pair *) a;
- const http_response_status_code_pair *pb = (const http_response_status_code_pair *) b;
- if (pa->code < pb->code) {
- return -1;
- } else if (pa->code > pb->code) {
- return 1;
- }
- return 0;
- } /* }}} */
- static const char *get_status_string(int code) /* {{{ */
- {
- http_response_status_code_pair needle = {code, NULL},
- *result = NULL;
- result = bsearch(&needle, http_status_map, http_status_map_len, sizeof(needle), status_comp);
- if (result) {
- return result->str;
- }
- /* Returning NULL would require complicating append_http_status_line() to
- * not segfault in that case, so let's just return a placeholder, since RFC
- * 2616 requires a reason phrase. This is basically what a lot of other Web
- * servers do in this case anyway. */
- return "Unknown Status Code";
- } /* }}} */
- static const char *get_template_string(int code) /* {{{ */
- {
- size_t e = (sizeof(template_map) / sizeof(php_cli_server_http_response_status_code_pair));
- size_t s = 0;
- while (e != s) {
- size_t c = MIN((e + s + 1) / 2, e - 1);
- int d = template_map[c].code;
- if (d > code) {
- e = c;
- } else if (d < code) {
- s = c;
- } else {
- return template_map[c].str;
- }
- }
- return NULL;
- } /* }}} */
- static void append_http_status_line(smart_str *buffer, int protocol_version, int response_code, int persistent) /* {{{ */
- {
- if (!response_code) {
- response_code = 200;
- }
- smart_str_appendl_ex(buffer, "HTTP", 4, persistent);
- smart_str_appendc_ex(buffer, '/', persistent);
- smart_str_append_long_ex(buffer, protocol_version / 100, persistent);
- smart_str_appendc_ex(buffer, '.', persistent);
- smart_str_append_long_ex(buffer, protocol_version % 100, persistent);
- smart_str_appendc_ex(buffer, ' ', persistent);
- smart_str_append_long_ex(buffer, response_code, persistent);
- smart_str_appendc_ex(buffer, ' ', persistent);
- smart_str_appends_ex(buffer, get_status_string(response_code), persistent);
- smart_str_appendl_ex(buffer, "\r\n", 2, persistent);
- } /* }}} */
- static void append_essential_headers(smart_str* buffer, php_cli_server_client *client, int persistent) /* {{{ */
- {
- char *val;
- struct timeval tv = {0};
- if (NULL != (val = zend_hash_str_find_ptr(&client->request.headers, "host", sizeof("host")-1))) {
- smart_str_appends_ex(buffer, "Host: ", persistent);
- smart_str_appends_ex(buffer, val, persistent);
- smart_str_appends_ex(buffer, "\r\n", persistent);
- }
- if (!gettimeofday(&tv, NULL)) {
- zend_string *dt = php_format_date("D, d M Y H:i:s", sizeof("D, d M Y H:i:s") - 1, tv.tv_sec, 0);
- smart_str_appends_ex(buffer, "Date: ", persistent);
- smart_str_appends_ex(buffer, dt->val, persistent);
- smart_str_appends_ex(buffer, " GMT\r\n", persistent);
- zend_string_release_ex(dt, 0);
- }
- smart_str_appendl_ex(buffer, "Connection: close\r\n", sizeof("Connection: close\r\n") - 1, persistent);
- } /* }}} */
- static const char *get_mime_type(const php_cli_server *server, const char *ext, size_t ext_len) /* {{{ */
- {
- char *ret;
- ALLOCA_FLAG(use_heap)
- char *ext_lower = do_alloca(ext_len + 1, use_heap);
- zend_str_tolower_copy(ext_lower, ext, ext_len);
- ret = zend_hash_str_find_ptr(&server->extension_mime_types, ext_lower, ext_len);
- free_alloca(ext_lower, use_heap);
- return (const char*)ret;
- } /* }}} */
- PHP_FUNCTION(apache_request_headers) /* {{{ */
- {
- php_cli_server_client *client;
- HashTable *headers;
- zend_string *key;
- char *value;
- zval tmp;
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- client = SG(server_context);
- headers = &client->request.headers_original_case;
- array_init_size(return_value, zend_hash_num_elements(headers));
- ZEND_HASH_FOREACH_STR_KEY_PTR(headers, key, value) {
- ZVAL_STRING(&tmp, value);
- zend_symtable_update(Z_ARRVAL_P(return_value), key, &tmp);
- } ZEND_HASH_FOREACH_END();
- }
- /* }}} */
- static void add_response_header(sapi_header_struct *h, zval *return_value) /* {{{ */
- {
- char *s, *p;
- ptrdiff_t len;
- ALLOCA_FLAG(use_heap)
- if (h->header_len > 0) {
- p = strchr(h->header, ':');
- len = p - h->header;
- if (p && (len > 0)) {
- while (len > 0 && (h->header[len-1] == ' ' || h->header[len-1] == '\t')) {
- len--;
- }
- if (len) {
- s = do_alloca(len + 1, use_heap);
- memcpy(s, h->header, len);
- s[len] = 0;
- do {
- p++;
- } while (*p == ' ' || *p == '\t');
- add_assoc_stringl_ex(return_value, s, (uint32_t)len, p, h->header_len - (p - h->header));
- free_alloca(s, use_heap);
- }
- }
- }
- }
- /* }}} */
- PHP_FUNCTION(apache_response_headers) /* {{{ */
- {
- if (zend_parse_parameters_none() == FAILURE) {
- RETURN_THROWS();
- }
- array_init(return_value);
- zend_llist_apply_with_argument(&SG(sapi_headers).headers, (llist_apply_with_arg_func_t)add_response_header, return_value);
- }
- /* }}} */
- /* {{{ cli_server module */
- static void cli_server_init_globals(zend_cli_server_globals *cg)
- {
- cg->color = 0;
- }
- PHP_INI_BEGIN()
- STD_PHP_INI_BOOLEAN("cli_server.color", "0", PHP_INI_ALL, OnUpdateBool, color, zend_cli_server_globals, cli_server_globals)
- PHP_INI_END()
- static PHP_MINIT_FUNCTION(cli_server)
- {
- ZEND_INIT_MODULE_GLOBALS(cli_server, cli_server_init_globals, NULL);
- REGISTER_INI_ENTRIES();
- return SUCCESS;
- }
- static PHP_MSHUTDOWN_FUNCTION(cli_server)
- {
- UNREGISTER_INI_ENTRIES();
- return SUCCESS;
- }
- static PHP_MINFO_FUNCTION(cli_server)
- {
- DISPLAY_INI_ENTRIES();
- }
- zend_module_entry cli_server_module_entry = {
- STANDARD_MODULE_HEADER,
- "cli_server",
- NULL,
- PHP_MINIT(cli_server),
- PHP_MSHUTDOWN(cli_server),
- NULL,
- NULL,
- PHP_MINFO(cli_server),
- PHP_VERSION,
- STANDARD_MODULE_PROPERTIES
- };
- /* }}} */
- const zend_function_entry server_additional_functions[] = {
- PHP_FE(cli_set_process_title, arginfo_cli_set_process_title)
- PHP_FE(cli_get_process_title, arginfo_cli_get_process_title)
- PHP_FE(apache_request_headers, arginfo_apache_request_headers)
- PHP_FE(apache_response_headers, arginfo_apache_response_headers)
- PHP_FALIAS(getallheaders, apache_request_headers, arginfo_getallheaders)
- PHP_FE_END
- };
- static int sapi_cli_server_startup(sapi_module_struct *sapi_module) /* {{{ */
- {
- return php_module_startup(sapi_module, &cli_server_module_entry, 1);
- } /* }}} */
- static size_t sapi_cli_server_ub_write(const char *str, size_t str_length) /* {{{ */
- {
- php_cli_server_client *client = SG(server_context);
- if (!client) {
- return 0;
- }
- return php_cli_server_client_send_through(client, str, str_length);
- } /* }}} */
- static void sapi_cli_server_flush(void *server_context) /* {{{ */
- {
- php_cli_server_client *client = server_context;
- if (!client) {
- return;
- }
- if (!ZEND_VALID_SOCKET(client->sock)) {
- php_handle_aborted_connection();
- return;
- }
- if (!SG(headers_sent)) {
- sapi_send_headers();
- SG(headers_sent) = 1;
- }
- } /* }}} */
- static int sapi_cli_server_discard_headers(sapi_headers_struct *sapi_headers) /* {{{ */{
- return SAPI_HEADER_SENT_SUCCESSFULLY;
- }
- /* }}} */
- static int sapi_cli_server_send_headers(sapi_headers_struct *sapi_headers) /* {{{ */
- {
- php_cli_server_client *client = SG(server_context);
- smart_str buffer = { 0 };
- sapi_header_struct *h;
- zend_llist_position pos;
- if (client == NULL || SG(request_info).no_headers) {
- return SAPI_HEADER_SENT_SUCCESSFULLY;
- }
- if (SG(sapi_headers).http_status_line) {
- smart_str_appends(&buffer, SG(sapi_headers).http_status_line);
- smart_str_appendl(&buffer, "\r\n", 2);
- } else {
- append_http_status_line(&buffer, client->request.protocol_version, SG(sapi_headers).http_response_code, 0);
- }
- append_essential_headers(&buffer, client, 0);
- h = (sapi_header_struct*)zend_llist_get_first_ex(&sapi_headers->headers, &pos);
- while (h) {
- if (h->header_len) {
- smart_str_appendl(&buffer, h->header, h->header_len);
- smart_str_appendl(&buffer, "\r\n", 2);
- }
- h = (sapi_header_struct*)zend_llist_get_next_ex(&sapi_headers->headers, &pos);
- }
- smart_str_appendl(&buffer, "\r\n", 2);
- php_cli_server_client_send_through(client, ZSTR_VAL(buffer.s), ZSTR_LEN(buffer.s));
- smart_str_free(&buffer);
- return SAPI_HEADER_SENT_SUCCESSFULLY;
- }
- /* }}} */
- static char *sapi_cli_server_read_cookies(void) /* {{{ */
- {
- php_cli_server_client *client = SG(server_context);
- char *val;
- if (NULL == (val = zend_hash_str_find_ptr(&client->request.headers, "cookie", sizeof("cookie")-1))) {
- return NULL;
- }
- return val;
- } /* }}} */
- static size_t sapi_cli_server_read_post(char *buf, size_t count_bytes) /* {{{ */
- {
- php_cli_server_client *client = SG(server_context);
- if (client->request.content) {
- size_t content_len = client->request.content_len;
- size_t nbytes_copied = MIN(client->post_read_offset + count_bytes, content_len) - client->post_read_offset;
- memmove(buf, client->request.content + client->post_read_offset, nbytes_copied);
- client->post_read_offset += nbytes_copied;
- return nbytes_copied;
- }
- return 0;
- } /* }}} */
- static void sapi_cli_server_register_variable(zval *track_vars_array, const char *key, const char *val) /* {{{ */
- {
- char *new_val = (char *)val;
- size_t new_val_len;
- if (NULL == val) {
- return;
- }
- if (sapi_module.input_filter(PARSE_SERVER, (char*)key, &new_val, strlen(val), &new_val_len)) {
- php_register_variable_safe((char *)key, new_val, new_val_len, track_vars_array);
- }
- } /* }}} */
- static int sapi_cli_server_register_entry_cb(char **entry, int num_args, va_list args, zend_hash_key *hash_key) /* {{{ */ {
- zval *track_vars_array = va_arg(args, zval *);
- if (hash_key->key) {
- char *real_key, *key;
- uint32_t i;
- key = estrndup(ZSTR_VAL(hash_key->key), ZSTR_LEN(hash_key->key));
- for(i=0; i<ZSTR_LEN(hash_key->key); i++) {
- if (key[i] == '-') {
- key[i] = '_';
- } else {
- key[i] = toupper(key[i]);
- }
- }
- spprintf(&real_key, 0, "%s_%s", "HTTP", key);
- if (strcmp(key, "CONTENT_TYPE") == 0 || strcmp(key, "CONTENT_LENGTH") == 0) {
- sapi_cli_server_register_variable(track_vars_array, key, *entry);
- }
- sapi_cli_server_register_variable(track_vars_array, real_key, *entry);
- efree(key);
- efree(real_key);
- }
- return ZEND_HASH_APPLY_KEEP;
- }
- /* }}} */
- static void sapi_cli_server_register_variables(zval *track_vars_array) /* {{{ */
- {
- php_cli_server_client *client = SG(server_context);
- sapi_cli_server_register_variable(track_vars_array, "DOCUMENT_ROOT", client->server->document_root);
- {
- char *tmp;
- if ((tmp = strrchr(client->addr_str, ':'))) {
- char addr[64], port[8];
- const char *addr_start = client->addr_str, *addr_end = tmp;
- if (addr_start[0] == '[') addr_start++;
- if (addr_end[-1] == ']') addr_end--;
- strncpy(port, tmp + 1, 8);
- port[7] = '\0';
- strncpy(addr, addr_start, addr_end - addr_start);
- addr[addr_end - addr_start] = '\0';
- sapi_cli_server_register_variable(track_vars_array, "REMOTE_ADDR", addr);
- sapi_cli_server_register_variable(track_vars_array, "REMOTE_PORT", port);
- } else {
- sapi_cli_server_register_variable(track_vars_array, "REMOTE_ADDR", client->addr_str);
- }
- }
- {
- char *tmp;
- spprintf(&tmp, 0, "PHP %s Development Server", PHP_VERSION);
- sapi_cli_server_register_variable(track_vars_array, "SERVER_SOFTWARE", tmp);
- efree(tmp);
- }
- {
- char *tmp;
- spprintf(&tmp, 0, "HTTP/%d.%d", client->request.protocol_version / 100, client->request.protocol_version % 100);
- sapi_cli_server_register_variable(track_vars_array, "SERVER_PROTOCOL", tmp);
- efree(tmp);
- }
- sapi_cli_server_register_variable(track_vars_array, "SERVER_NAME", client->server->host);
- {
- char *tmp;
- spprintf(&tmp, 0, "%i", client->server->port);
- sapi_cli_server_register_variable(track_vars_array, "SERVER_PORT", tmp);
- efree(tmp);
- }
- sapi_cli_server_register_variable(track_vars_array, "REQUEST_URI", client->request.request_uri);
- sapi_cli_server_register_variable(track_vars_array, "REQUEST_METHOD", SG(request_info).request_method);
- sapi_cli_server_register_variable(track_vars_array, "SCRIPT_NAME", client->request.vpath);
- if (SG(request_info).path_translated) {
- sapi_cli_server_register_variable(track_vars_array, "SCRIPT_FILENAME", SG(request_info).path_translated);
- } else if (client->server->router) {
- sapi_cli_server_register_variable(track_vars_array, "SCRIPT_FILENAME", client->server->router);
- }
- if (client->request.path_info) {
- sapi_cli_server_register_variable(track_vars_array, "PATH_INFO", client->request.path_info);
- }
- if (client->request.path_info_len) {
- char *tmp;
- spprintf(&tmp, 0, "%s%s", client->request.vpath, client->request.path_info);
- sapi_cli_server_register_variable(track_vars_array, "PHP_SELF", tmp);
- efree(tmp);
- } else {
- sapi_cli_server_register_variable(track_vars_array, "PHP_SELF", client->request.vpath);
- }
- if (client->request.query_string) {
- sapi_cli_server_register_variable(track_vars_array, "QUERY_STRING", client->request.query_string);
- }
- zend_hash_apply_with_arguments(&client->request.headers, (apply_func_args_t)sapi_cli_server_register_entry_cb, 1, track_vars_array);
- } /* }}} */
- static void sapi_cli_server_log_write(int type, const char *msg) /* {{{ */
- {
- char buf[52];
- if (php_cli_server_log_level < type) {
- return;
- }
- if (php_cli_server_get_system_time(buf) != 0) {
- memmove(buf, "unknown time, can't be fetched", sizeof("unknown time, can't be fetched"));
- } else {
- size_t l = strlen(buf);
- if (l > 0) {
- buf[l - 1] = '\0';
- } else {
- memmove(buf, "unknown", sizeof("unknown"));
- }
- }
- #ifdef HAVE_FORK
- if (php_cli_server_workers_max > 1) {
- fprintf(stderr, "[%ld] [%s] %s\n", (long) getpid(), buf, msg);
- } else {
- fprintf(stderr, "[%s] %s\n", buf, msg);
- }
- #else
- fprintf(stderr, "[%s] %s\n", buf, msg);
- #endif
- } /* }}} */
- static void sapi_cli_server_log_message(const char *msg, int syslog_type_int) /* {{{ */
- {
- sapi_cli_server_log_write(PHP_CLI_SERVER_LOG_MESSAGE, msg);
- } /* }}} */
- /* {{{ sapi_module_struct cli_server_sapi_module */
- sapi_module_struct cli_server_sapi_module = {
- "cli-server", /* name */
- "Built-in HTTP server", /* pretty name */
- sapi_cli_server_startup, /* startup */
- php_module_shutdown_wrapper, /* shutdown */
- NULL, /* activate */
- NULL, /* deactivate */
- sapi_cli_server_ub_write, /* unbuffered write */
- sapi_cli_server_flush, /* flush */
- NULL, /* get uid */
- NULL, /* getenv */
- php_error, /* error handler */
- NULL, /* header handler */
- sapi_cli_server_send_headers, /* send headers handler */
- NULL, /* send header handler */
- sapi_cli_server_read_post, /* read POST data */
- sapi_cli_server_read_cookies, /* read Cookies */
- sapi_cli_server_register_variables, /* register server variables */
- sapi_cli_server_log_message, /* Log message */
- NULL, /* Get request time */
- NULL, /* Child terminate */
- STANDARD_SAPI_MODULE_PROPERTIES
- }; /* }}} */
- static int php_cli_server_poller_ctor(php_cli_server_poller *poller) /* {{{ */
- {
- FD_ZERO(&poller->rfds);
- FD_ZERO(&poller->wfds);
- poller->max_fd = -1;
- return SUCCESS;
- } /* }}} */
- static void php_cli_server_poller_add(php_cli_server_poller *poller, int mode, php_socket_t fd) /* {{{ */
- {
- if (mode & POLLIN) {
- PHP_SAFE_FD_SET(fd, &poller->rfds);
- }
- if (mode & POLLOUT) {
- PHP_SAFE_FD_SET(fd, &poller->wfds);
- }
- if (fd > poller->max_fd) {
- poller->max_fd = fd;
- }
- } /* }}} */
- static void php_cli_server_poller_remove(php_cli_server_poller *poller, int mode, php_socket_t fd) /* {{{ */
- {
- if (mode & POLLIN) {
- PHP_SAFE_FD_CLR(fd, &poller->rfds);
- }
- if (mode & POLLOUT) {
- PHP_SAFE_FD_CLR(fd, &poller->wfds);
- }
- #ifndef PHP_WIN32
- if (fd == poller->max_fd) {
- while (fd > 0) {
- fd--;
- if (PHP_SAFE_FD_ISSET(fd, &poller->rfds) || PHP_SAFE_FD_ISSET(fd, &poller->wfds)) {
- break;
- }
- }
- poller->max_fd = fd;
- }
- #endif
- } /* }}} */
- static int php_cli_server_poller_poll(php_cli_server_poller *poller, struct timeval *tv) /* {{{ */
- {
- memmove(&poller->active.rfds, &poller->rfds, sizeof(poller->rfds));
- memmove(&poller->active.wfds, &poller->wfds, sizeof(poller->wfds));
- return php_select(poller->max_fd + 1, &poller->active.rfds, &poller->active.wfds, NULL, tv);
- } /* }}} */
- static int php_cli_server_poller_iter_on_active(php_cli_server_poller *poller, void *opaque, int(*callback)(void *, php_socket_t fd, int events)) /* {{{ */
- {
- int retval = SUCCESS;
- #ifdef PHP_WIN32
- struct socket_entry {
- SOCKET fd;
- int events;
- } entries[FD_SETSIZE * 2];
- size_t i;
- struct socket_entry *n = entries, *m;
- for (i = 0; i < poller->active.rfds.fd_count; i++) {
- n->events = POLLIN;
- n->fd = poller->active.rfds.fd_array[i];
- n++;
- }
- m = n;
- for (i = 0; i < poller->active.wfds.fd_count; i++) {
- struct socket_entry *e;
- SOCKET fd = poller->active.wfds.fd_array[i];
- for (e = entries; e < m; e++) {
- if (e->fd == fd) {
- e->events |= POLLOUT;
- }
- }
- if (e == m) {
- assert(n < entries + FD_SETSIZE * 2);
- n->events = POLLOUT;
- n->fd = fd;
- n++;
- }
- }
- {
- struct socket_entry *e = entries;
- for (; e < n; e++) {
- if (SUCCESS != callback(opaque, e->fd, e->events)) {
- retval = FAILURE;
- }
- }
- }
- #else
- php_socket_t fd;
- const php_socket_t max_fd = poller->max_fd;
- for (fd=0 ; fd<=max_fd ; fd++) {
- if (PHP_SAFE_FD_ISSET(fd, &poller->active.rfds)) {
- if (SUCCESS != callback(opaque, fd, POLLIN)) {
- retval = FAILURE;
- }
- }
- if (PHP_SAFE_FD_ISSET(fd, &poller->active.wfds)) {
- if (SUCCESS != callback(opaque, fd, POLLOUT)) {
- retval = FAILURE;
- }
- }
- }
- #endif
- return retval;
- } /* }}} */
- static size_t php_cli_server_chunk_size(const php_cli_server_chunk *chunk) /* {{{ */
- {
- switch (chunk->type) {
- case PHP_CLI_SERVER_CHUNK_HEAP:
- return chunk->data.heap.len;
- case PHP_CLI_SERVER_CHUNK_IMMORTAL:
- return chunk->data.immortal.len;
- }
- return 0;
- } /* }}} */
- static void php_cli_server_chunk_dtor(php_cli_server_chunk *chunk) /* {{{ */
- {
- switch (chunk->type) {
- case PHP_CLI_SERVER_CHUNK_HEAP:
- if (chunk->data.heap.block != chunk) {
- pefree(chunk->data.heap.block, 1);
- }
- break;
- case PHP_CLI_SERVER_CHUNK_IMMORTAL:
- break;
- }
- } /* }}} */
- static void php_cli_server_buffer_dtor(php_cli_server_buffer *buffer) /* {{{ */
- {
- php_cli_server_chunk *chunk, *next;
- for (chunk = buffer->first; chunk; chunk = next) {
- next = chunk->next;
- php_cli_server_chunk_dtor(chunk);
- pefree(chunk, 1);
- }
- } /* }}} */
- static void php_cli_server_buffer_ctor(php_cli_server_buffer *buffer) /* {{{ */
- {
- buffer->first = NULL;
- buffer->last = NULL;
- } /* }}} */
- static void php_cli_server_buffer_append(php_cli_server_buffer *buffer, php_cli_server_chunk *chunk) /* {{{ */
- {
- php_cli_server_chunk *last;
- for (last = chunk; last->next; last = last->next);
- if (!buffer->last) {
- buffer->first = chunk;
- } else {
- buffer->last->next = chunk;
- }
- buffer->last = last;
- } /* }}} */
- static void php_cli_server_buffer_prepend(php_cli_server_buffer *buffer, php_cli_server_chunk *chunk) /* {{{ */
- {
- php_cli_server_chunk *last;
- for (last = chunk; last->next; last = last->next);
- last->next = buffer->first;
- if (!buffer->last) {
- buffer->last = last;
- }
- buffer->first = chunk;
- } /* }}} */
- static size_t php_cli_server_buffer_size(const php_cli_server_buffer *buffer) /* {{{ */
- {
- php_cli_server_chunk *chunk;
- size_t retval = 0;
- for (chunk = buffer->first; chunk; chunk = chunk->next) {
- retval += php_cli_server_chunk_size(chunk);
- }
- return retval;
- } /* }}} */
- static php_cli_server_chunk *php_cli_server_chunk_immortal_new(const char *buf, size_t len) /* {{{ */
- {
- php_cli_server_chunk *chunk = pemalloc(sizeof(php_cli_server_chunk), 1);
- chunk->type = PHP_CLI_SERVER_CHUNK_IMMORTAL;
- chunk->next = NULL;
- chunk->data.immortal.p = buf;
- chunk->data.immortal.len = len;
- return chunk;
- } /* }}} */
- static php_cli_server_chunk *php_cli_server_chunk_heap_new(void *block, char *buf, size_t len) /* {{{ */
- {
- php_cli_server_chunk *chunk = pemalloc(sizeof(php_cli_server_chunk), 1);
- chunk->type = PHP_CLI_SERVER_CHUNK_HEAP;
- chunk->next = NULL;
- chunk->data.heap.block = block;
- chunk->data.heap.p = buf;
- chunk->data.heap.len = len;
- return chunk;
- } /* }}} */
- static php_cli_server_chunk *php_cli_server_chunk_heap_new_self_contained(size_t len) /* {{{ */
- {
- php_cli_server_chunk *chunk = pemalloc(sizeof(php_cli_server_chunk) + len, 1);
- chunk->type = PHP_CLI_SERVER_CHUNK_HEAP;
- chunk->next = NULL;
- chunk->data.heap.block = chunk;
- chunk->data.heap.p = (char *)(chunk + 1);
- chunk->data.heap.len = len;
- return chunk;
- } /* }}} */
- static void php_cli_server_content_sender_dtor(php_cli_server_content_sender *sender) /* {{{ */
- {
- php_cli_server_buffer_dtor(&sender->buffer);
- } /* }}} */
- static void php_cli_server_content_sender_ctor(php_cli_server_content_sender *sender) /* {{{ */
- {
- php_cli_server_buffer_ctor(&sender->buffer);
- } /* }}} */
- static int php_cli_server_content_sender_send(php_cli_server_content_sender *sender, php_socket_t fd, size_t *nbytes_sent_total) /* {{{ */
- {
- php_cli_server_chunk *chunk, *next;
- size_t _nbytes_sent_total = 0;
- for (chunk = sender->buffer.first; chunk; chunk = next) {
- #ifdef PHP_WIN32
- int nbytes_sent;
- #else
- ssize_t nbytes_sent;
- #endif
- next = chunk->next;
- switch (chunk->type) {
- case PHP_CLI_SERVER_CHUNK_HEAP:
- #ifdef PHP_WIN32
- nbytes_sent = send(fd, chunk->data.heap.p, (int)chunk->data.heap.len, 0);
- #else
- nbytes_sent = send(fd, chunk->data.heap.p, chunk->data.heap.len, 0);
- #endif
- if (nbytes_sent < 0) {
- *nbytes_sent_total = _nbytes_sent_total;
- return php_socket_errno();
- #ifdef PHP_WIN32
- } else if (nbytes_sent == chunk->data.heap.len) {
- #else
- } else if (nbytes_sent == (ssize_t)chunk->data.heap.len) {
- #endif
- php_cli_server_chunk_dtor(chunk);
- pefree(chunk, 1);
- sender->buffer.first = next;
- if (!next) {
- sender->buffer.last = NULL;
- }
- } else {
- chunk->data.heap.p += nbytes_sent;
- chunk->data.heap.len -= nbytes_sent;
- }
- _nbytes_sent_total += nbytes_sent;
- break;
- case PHP_CLI_SERVER_CHUNK_IMMORTAL:
- #ifdef PHP_WIN32
- nbytes_sent = send(fd, chunk->data.immortal.p, (int)chunk->data.immortal.len, 0);
- #else
- nbytes_sent = send(fd, chunk->data.immortal.p, chunk->data.immortal.len, 0);
- #endif
- if (nbytes_sent < 0) {
- *nbytes_sent_total = _nbytes_sent_total;
- return php_socket_errno();
- #ifdef PHP_WIN32
- } else if (nbytes_sent == chunk->data.immortal.len) {
- #else
- } else if (nbytes_sent == (ssize_t)chunk->data.immortal.len) {
- #endif
- php_cli_server_chunk_dtor(chunk);
- pefree(chunk, 1);
- sender->buffer.first = next;
- if (!next) {
- sender->buffer.last = NULL;
- }
- } else {
- chunk->data.immortal.p += nbytes_sent;
- chunk->data.immortal.len -= nbytes_sent;
- }
- _nbytes_sent_total += nbytes_sent;
- break;
- }
- }
- *nbytes_sent_total = _nbytes_sent_total;
- return 0;
- } /* }}} */
- static int php_cli_server_content_sender_pull(php_cli_server_content_sender *sender, int fd, size_t *nbytes_read) /* {{{ */
- {
- #ifdef PHP_WIN32
- int _nbytes_read;
- #else
- ssize_t _nbytes_read;
- #endif
- php_cli_server_chunk *chunk = php_cli_server_chunk_heap_new_self_contained(131072);
- #ifdef PHP_WIN32
- _nbytes_read = read(fd, chunk->data.heap.p, (unsigned int)chunk->data.heap.len);
- #else
- _nbytes_read = read(fd, chunk->data.heap.p, chunk->data.heap.len);
- #endif
- if (_nbytes_read < 0) {
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- char *errstr = get_last_error();
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR, "%s", errstr);
- pefree(errstr, 1);
- }
- php_cli_server_chunk_dtor(chunk);
- pefree(chunk, 1);
- return 1;
- }
- chunk->data.heap.len = _nbytes_read;
- php_cli_server_buffer_append(&sender->buffer, chunk);
- *nbytes_read = _nbytes_read;
- return 0;
- } /* }}} */
- #if HAVE_UNISTD_H
- static int php_cli_is_output_tty(void) /* {{{ */
- {
- if (php_cli_output_is_tty == OUTPUT_NOT_CHECKED) {
- php_cli_output_is_tty = isatty(STDOUT_FILENO);
- }
- return php_cli_output_is_tty;
- } /* }}} */
- #elif defined(PHP_WIN32)
- static int php_cli_is_output_tty() /* {{{ */
- {
- if (php_cli_output_is_tty == OUTPUT_NOT_CHECKED) {
- php_cli_output_is_tty = php_win32_console_fileno_is_console(STDOUT_FILENO) && php_win32_console_fileno_has_vt100(STDOUT_FILENO);
- }
- return php_cli_output_is_tty;
- } /* }}} */
- #endif
- static void php_cli_server_log_response(php_cli_server_client *client, int status, const char *message) /* {{{ */
- {
- int color = 0, effective_status = status;
- char *basic_buf, *message_buf = "", *error_buf = "";
- bool append_error_message = 0;
- if (PG(last_error_message)) {
- if (PG(last_error_type) & E_FATAL_ERRORS) {
- if (status == 200) {
- /* the status code isn't changed by a fatal error, so fake it */
- effective_status = 500;
- }
- append_error_message = 1;
- }
- }
- #if HAVE_UNISTD_H || defined(PHP_WIN32)
- if (CLI_SERVER_G(color) && php_cli_is_output_tty() == OUTPUT_IS_TTY) {
- if (effective_status >= 500) {
- /* server error: red */
- color = 1;
- } else if (effective_status >= 400) {
- /* client error: yellow */
- color = 3;
- } else if (effective_status >= 200) {
- /* success: green */
- color = 2;
- }
- }
- #endif
- /* basic */
- spprintf(&basic_buf, 0, "%s [%d]: %s %s", client->addr_str, status, php_http_method_str(client->request.request_method), client->request.request_uri);
- if (!basic_buf) {
- return;
- }
- /* message */
- if (message) {
- spprintf(&message_buf, 0, " - %s", message);
- if (!message_buf) {
- efree(basic_buf);
- return;
- }
- }
- /* error */
- if (append_error_message) {
- spprintf(&error_buf, 0, " - %s in %s on line %d",
- ZSTR_VAL(PG(last_error_message)), ZSTR_VAL(PG(last_error_file)), PG(last_error_lineno));
- if (!error_buf) {
- efree(basic_buf);
- if (message) {
- efree(message_buf);
- }
- return;
- }
- }
- if (color) {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_MESSAGE, "\x1b[3%dm%s%s%s\x1b[0m", color, basic_buf, message_buf, error_buf);
- } else {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_MESSAGE, "%s%s%s", basic_buf, message_buf, error_buf);
- }
- efree(basic_buf);
- if (message) {
- efree(message_buf);
- }
- if (append_error_message) {
- efree(error_buf);
- }
- } /* }}} */
- static void php_cli_server_logf(int type, const char *format, ...) /* {{{ */
- {
- char *buf = NULL;
- va_list ap;
- if (php_cli_server_log_level < type) {
- return;
- }
- va_start(ap, format);
- vspprintf(&buf, 0, format, ap);
- va_end(ap);
- if (!buf) {
- return;
- }
- sapi_cli_server_log_write(type, buf);
- efree(buf);
- } /* }}} */
- static php_socket_t php_network_listen_socket(const char *host, int *port, int socktype, int *af, socklen_t *socklen, zend_string **errstr) /* {{{ */
- {
- php_socket_t retval = SOCK_ERR;
- int err = 0;
- struct sockaddr *sa = NULL, **p, **sal;
- int num_addrs = php_network_getaddresses(host, socktype, &sal, errstr);
- if (num_addrs == 0) {
- return -1;
- }
- for (p = sal; *p; p++) {
- if (sa) {
- pefree(sa, 1);
- sa = NULL;
- }
- retval = socket((*p)->sa_family, socktype, 0);
- if (retval == SOCK_ERR) {
- continue;
- }
- switch ((*p)->sa_family) {
- #if HAVE_GETADDRINFO && HAVE_IPV6
- case AF_INET6:
- sa = pemalloc(sizeof(struct sockaddr_in6), 1);
- *(struct sockaddr_in6 *)sa = *(struct sockaddr_in6 *)*p;
- ((struct sockaddr_in6 *)sa)->sin6_port = htons(*port);
- *socklen = sizeof(struct sockaddr_in6);
- break;
- #endif
- case AF_INET:
- sa = pemalloc(sizeof(struct sockaddr_in), 1);
- *(struct sockaddr_in *)sa = *(struct sockaddr_in *)*p;
- ((struct sockaddr_in *)sa)->sin_port = htons(*port);
- *socklen = sizeof(struct sockaddr_in);
- break;
- default:
- /* Unknown family */
- *socklen = 0;
- closesocket(retval);
- continue;
- }
- #ifdef SO_REUSEADDR
- {
- int val = 1;
- setsockopt(retval, SOL_SOCKET, SO_REUSEADDR, (char*)&val, sizeof(val));
- }
- #endif
- if (bind(retval, sa, *socklen) == SOCK_CONN_ERR) {
- err = php_socket_errno();
- if (err == SOCK_EINVAL || err == SOCK_EADDRINUSE) {
- goto out;
- }
- closesocket(retval);
- retval = SOCK_ERR;
- continue;
- }
- err = 0;
- *af = sa->sa_family;
- if (*port == 0) {
- if (getsockname(retval, sa, socklen)) {
- err = php_socket_errno();
- goto out;
- }
- switch (sa->sa_family) {
- #if HAVE_GETADDRINFO && HAVE_IPV6
- case AF_INET6:
- *port = ntohs(((struct sockaddr_in6 *)sa)->sin6_port);
- break;
- #endif
- case AF_INET:
- *port = ntohs(((struct sockaddr_in *)sa)->sin_port);
- break;
- }
- }
- break;
- }
- if (retval == SOCK_ERR) {
- goto out;
- }
- if (listen(retval, SOMAXCONN)) {
- err = php_socket_errno();
- goto out;
- }
- out:
- if (sa) {
- pefree(sa, 1);
- }
- if (sal) {
- php_network_freeaddresses(sal);
- }
- if (err) {
- if (ZEND_VALID_SOCKET(retval)) {
- closesocket(retval);
- }
- if (errstr) {
- *errstr = php_socket_error_str(err);
- }
- return SOCK_ERR;
- }
- return retval;
- } /* }}} */
- static int php_cli_server_request_ctor(php_cli_server_request *req) /* {{{ */
- {
- req->protocol_version = 0;
- req->request_uri = NULL;
- req->request_uri_len = 0;
- req->vpath = NULL;
- req->vpath_len = 0;
- req->path_translated = NULL;
- req->path_translated_len = 0;
- req->path_info = NULL;
- req->path_info_len = 0;
- req->query_string = NULL;
- req->query_string_len = 0;
- zend_hash_init(&req->headers, 0, NULL, char_ptr_dtor_p, 1);
- zend_hash_init(&req->headers_original_case, 0, NULL, NULL, 1);
- req->content = NULL;
- req->content_len = 0;
- req->ext = NULL;
- req->ext_len = 0;
- return SUCCESS;
- } /* }}} */
- static void php_cli_server_request_dtor(php_cli_server_request *req) /* {{{ */
- {
- if (req->request_uri) {
- pefree(req->request_uri, 1);
- }
- if (req->vpath) {
- pefree(req->vpath, 1);
- }
- if (req->path_translated) {
- pefree(req->path_translated, 1);
- }
- if (req->path_info) {
- pefree(req->path_info, 1);
- }
- if (req->query_string) {
- pefree(req->query_string, 1);
- }
- zend_hash_destroy(&req->headers);
- zend_hash_destroy(&req->headers_original_case);
- if (req->content) {
- pefree(req->content, 1);
- }
- } /* }}} */
- static void php_cli_server_request_translate_vpath(php_cli_server_request *request, const char *document_root, size_t document_root_len) /* {{{ */
- {
- zend_stat_t sb;
- static const char *index_files[] = { "index.php", "index.html", NULL };
- char *buf = safe_pemalloc(1, request->vpath_len, 1 + document_root_len + 1 + sizeof("index.html"), 1);
- char *p = buf, *prev_path = NULL, *q, *vpath;
- size_t prev_path_len = 0;
- int is_static_file = 0;
- memmove(p, document_root, document_root_len);
- p += document_root_len;
- vpath = p;
- if (request->vpath_len > 0 && request->vpath[0] != '/') {
- *p++ = DEFAULT_SLASH;
- }
- q = request->vpath + request->vpath_len;
- while (q > request->vpath) {
- if (*q-- == '.') {
- is_static_file = 1;
- break;
- }
- }
- memmove(p, request->vpath, request->vpath_len);
- #ifdef PHP_WIN32
- q = p + request->vpath_len;
- do {
- if (*q == '/') {
- *q = '\\';
- }
- } while (q-- > p);
- #endif
- p += request->vpath_len;
- *p = '\0';
- q = p;
- while (q > buf) {
- if (!php_sys_stat(buf, &sb)) {
- if (sb.st_mode & S_IFDIR) {
- const char **file = index_files;
- if (q[-1] != DEFAULT_SLASH) {
- *q++ = DEFAULT_SLASH;
- }
- while (*file) {
- size_t l = strlen(*file);
- memmove(q, *file, l + 1);
- if (!php_sys_stat(buf, &sb) && (sb.st_mode & S_IFREG)) {
- q += l;
- break;
- }
- file++;
- }
- if (!*file || is_static_file) {
- if (prev_path) {
- pefree(prev_path, 1);
- }
- pefree(buf, 1);
- return;
- }
- }
- break; /* regular file */
- }
- if (prev_path) {
- pefree(prev_path, 1);
- *q = DEFAULT_SLASH;
- }
- while (q > buf && *(--q) != DEFAULT_SLASH);
- prev_path_len = p - q;
- prev_path = pestrndup(q, prev_path_len, 1);
- *q = '\0';
- }
- if (prev_path) {
- request->path_info_len = prev_path_len;
- #ifdef PHP_WIN32
- while (prev_path_len--) {
- if (prev_path[prev_path_len] == '\\') {
- prev_path[prev_path_len] = '/';
- }
- }
- #endif
- request->path_info = prev_path;
- pefree(request->vpath, 1);
- request->vpath = pestrndup(vpath, q - vpath, 1);
- request->vpath_len = q - vpath;
- request->path_translated = buf;
- request->path_translated_len = q - buf;
- } else {
- pefree(request->vpath, 1);
- request->vpath = pestrndup(vpath, q - vpath, 1);
- request->vpath_len = q - vpath;
- request->path_translated = buf;
- request->path_translated_len = q - buf;
- }
- #ifdef PHP_WIN32
- {
- uint32_t i = 0;
- for (;i<request->vpath_len;i++) {
- if (request->vpath[i] == '\\') {
- request->vpath[i] = '/';
- }
- }
- }
- #endif
- request->sb = sb;
- } /* }}} */
- static void normalize_vpath(char **retval, size_t *retval_len, const char *vpath, size_t vpath_len, int persistent) /* {{{ */
- {
- char *decoded_vpath = NULL;
- char *decoded_vpath_end;
- char *p;
- *retval = NULL;
- *retval_len = 0;
- decoded_vpath = pestrndup(vpath, vpath_len, persistent);
- if (!decoded_vpath) {
- return;
- }
- decoded_vpath_end = decoded_vpath + php_raw_url_decode(decoded_vpath, (int)vpath_len);
- #ifdef PHP_WIN32
- {
- char *p = decoded_vpath;
- do {
- if (*p == '\\') {
- *p = '/';
- }
- } while (*p++);
- }
- #endif
- p = decoded_vpath;
- if (p < decoded_vpath_end && *p == '/') {
- char *n = p;
- while (n < decoded_vpath_end && *n == '/') n++;
- memmove(++p, n, decoded_vpath_end - n);
- decoded_vpath_end -= n - p;
- }
- while (p < decoded_vpath_end) {
- char *n = p;
- while (n < decoded_vpath_end && *n != '/') n++;
- if (n - p == 2 && p[0] == '.' && p[1] == '.') {
- if (p > decoded_vpath) {
- --p;
- for (;;) {
- if (p == decoded_vpath) {
- if (*p == '/') {
- p++;
- }
- break;
- }
- if (*(--p) == '/') {
- p++;
- break;
- }
- }
- }
- while (n < decoded_vpath_end && *n == '/') n++;
- memmove(p, n, decoded_vpath_end - n);
- decoded_vpath_end -= n - p;
- } else if (n - p == 1 && p[0] == '.') {
- while (n < decoded_vpath_end && *n == '/') n++;
- memmove(p, n, decoded_vpath_end - n);
- decoded_vpath_end -= n - p;
- } else {
- if (n < decoded_vpath_end) {
- char *nn = n;
- while (nn < decoded_vpath_end && *nn == '/') nn++;
- p = n + 1;
- memmove(p, nn, decoded_vpath_end - nn);
- decoded_vpath_end -= nn - p;
- } else {
- p = n;
- }
- }
- }
- *decoded_vpath_end = '\0';
- *retval = decoded_vpath;
- *retval_len = decoded_vpath_end - decoded_vpath;
- } /* }}} */
- /* {{{ php_cli_server_client_read_request */
- static int php_cli_server_client_read_request_on_message_begin(php_http_parser *parser)
- {
- return 0;
- }
- static int php_cli_server_client_read_request_on_path(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- {
- char *vpath;
- size_t vpath_len;
- if (UNEXPECTED(client->request.vpath != NULL)) {
- return 1;
- }
- normalize_vpath(&vpath, &vpath_len, at, length, 1);
- client->request.vpath = vpath;
- client->request.vpath_len = vpath_len;
- }
- return 0;
- }
- static int php_cli_server_client_read_request_on_query_string(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- if (EXPECTED(client->request.query_string == NULL)) {
- client->request.query_string = pestrndup(at, length, 1);
- client->request.query_string_len = length;
- } else {
- ZEND_ASSERT(length <= PHP_HTTP_MAX_HEADER_SIZE && PHP_HTTP_MAX_HEADER_SIZE - length >= client->request.query_string_len);
- client->request.query_string = perealloc(client->request.query_string, client->request.query_string_len + length + 1, 1);
- memcpy(client->request.query_string + client->request.query_string_len, at, length);
- client->request.query_string_len += length;
- client->request.query_string[client->request.query_string_len] = '\0';
- }
- return 0;
- }
- static int php_cli_server_client_read_request_on_url(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- if (EXPECTED(client->request.request_uri == NULL)) {
- client->request.request_method = parser->method;
- client->request.request_uri = pestrndup(at, length, 1);
- client->request.request_uri_len = length;
- } else {
- ZEND_ASSERT(client->request.request_method == parser->method);
- ZEND_ASSERT(length <= PHP_HTTP_MAX_HEADER_SIZE && PHP_HTTP_MAX_HEADER_SIZE - length >= client->request.query_string_len);
- client->request.request_uri = perealloc(client->request.request_uri, client->request.request_uri_len + length + 1, 1);
- memcpy(client->request.request_uri + client->request.request_uri_len, at, length);
- client->request.request_uri_len += length;
- client->request.request_uri[client->request.request_uri_len] = '\0';
- }
- return 0;
- }
- static int php_cli_server_client_read_request_on_fragment(php_http_parser *parser, const char *at, size_t length)
- {
- return 0;
- }
- static void php_cli_server_client_save_header(php_cli_server_client *client)
- {
- /* strip off the colon */
- zend_string *orig_header_name = zend_string_init(client->current_header_name, client->current_header_name_len, 1);
- zend_string *lc_header_name = zend_string_alloc(client->current_header_name_len, 1);
- zend_str_tolower_copy(ZSTR_VAL(lc_header_name), client->current_header_name, client->current_header_name_len);
- GC_MAKE_PERSISTENT_LOCAL(orig_header_name);
- GC_MAKE_PERSISTENT_LOCAL(lc_header_name);
- zend_hash_add_ptr(&client->request.headers, lc_header_name, client->current_header_value);
- zend_hash_add_ptr(&client->request.headers_original_case, orig_header_name, client->current_header_value);
- zend_string_release_ex(lc_header_name, 1);
- zend_string_release_ex(orig_header_name, 1);
- if (client->current_header_name_allocated) {
- pefree(client->current_header_name, 1);
- client->current_header_name_allocated = 0;
- }
- client->current_header_name = NULL;
- client->current_header_name_len = 0;
- client->current_header_value = NULL;
- client->current_header_value_len = 0;
- }
- static int php_cli_server_client_read_request_on_header_field(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- switch (client->last_header_element) {
- case HEADER_VALUE:
- php_cli_server_client_save_header(client);
- ZEND_FALLTHROUGH;
- case HEADER_NONE:
- client->current_header_name = (char *)at;
- client->current_header_name_len = length;
- break;
- case HEADER_FIELD:
- if (client->current_header_name_allocated) {
- size_t new_length = client->current_header_name_len + length;
- client->current_header_name = perealloc(client->current_header_name, new_length + 1, 1);
- memcpy(client->current_header_name + client->current_header_name_len, at, length);
- client->current_header_name[new_length] = '\0';
- client->current_header_name_len = new_length;
- } else {
- size_t new_length = client->current_header_name_len + length;
- char* field = pemalloc(new_length + 1, 1);
- memcpy(field, client->current_header_name, client->current_header_name_len);
- memcpy(field + client->current_header_name_len, at, length);
- field[new_length] = '\0';
- client->current_header_name = field;
- client->current_header_name_len = new_length;
- client->current_header_name_allocated = 1;
- }
- break;
- }
- client->last_header_element = HEADER_FIELD;
- return 0;
- }
- static int php_cli_server_client_read_request_on_header_value(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- switch (client->last_header_element) {
- case HEADER_FIELD:
- client->current_header_value = pestrndup(at, length, 1);
- client->current_header_value_len = length;
- break;
- case HEADER_VALUE:
- {
- size_t new_length = client->current_header_value_len + length;
- client->current_header_value = perealloc(client->current_header_value, new_length + 1, 1);
- memcpy(client->current_header_value + client->current_header_value_len, at, length);
- client->current_header_value[new_length] = '\0';
- client->current_header_value_len = new_length;
- }
- break;
- case HEADER_NONE:
- // can't happen
- assert(0);
- break;
- }
- client->last_header_element = HEADER_VALUE;
- return 0;
- }
- static int php_cli_server_client_read_request_on_headers_complete(php_http_parser *parser)
- {
- php_cli_server_client *client = parser->data;
- switch (client->last_header_element) {
- case HEADER_NONE:
- break;
- case HEADER_FIELD:
- client->current_header_value = pemalloc(1, 1);
- *client->current_header_value = '\0';
- client->current_header_value_len = 0;
- ZEND_FALLTHROUGH;
- case HEADER_VALUE:
- php_cli_server_client_save_header(client);
- break;
- }
- client->last_header_element = HEADER_NONE;
- return 0;
- }
- static int php_cli_server_client_read_request_on_body(php_http_parser *parser, const char *at, size_t length)
- {
- php_cli_server_client *client = parser->data;
- if (!client->request.content) {
- client->request.content = pemalloc(parser->content_length, 1);
- client->request.content_len = 0;
- }
- client->request.content = perealloc(client->request.content, client->request.content_len + length, 1);
- memmove(client->request.content + client->request.content_len, at, length);
- client->request.content_len += length;
- return 0;
- }
- static int php_cli_server_client_read_request_on_message_complete(php_http_parser *parser)
- {
- php_cli_server_client *client = parser->data;
- client->request.protocol_version = parser->http_major * 100 + parser->http_minor;
- php_cli_server_request_translate_vpath(&client->request, client->server->document_root, client->server->document_root_len);
- {
- const char *vpath = client->request.vpath, *end = vpath + client->request.vpath_len, *p = end;
- client->request.ext = end;
- client->request.ext_len = 0;
- while (p > vpath) {
- --p;
- if (*p == '.') {
- ++p;
- client->request.ext = p;
- client->request.ext_len = end - p;
- break;
- }
- }
- }
- client->request_read = 1;
- return 0;
- }
- static int php_cli_server_client_read_request(php_cli_server_client *client, char **errstr)
- {
- char buf[16384];
- static const php_http_parser_settings settings = {
- php_cli_server_client_read_request_on_message_begin,
- php_cli_server_client_read_request_on_path,
- php_cli_server_client_read_request_on_query_string,
- php_cli_server_client_read_request_on_url,
- php_cli_server_client_read_request_on_fragment,
- php_cli_server_client_read_request_on_header_field,
- php_cli_server_client_read_request_on_header_value,
- php_cli_server_client_read_request_on_headers_complete,
- php_cli_server_client_read_request_on_body,
- php_cli_server_client_read_request_on_message_complete
- };
- size_t nbytes_consumed;
- int nbytes_read;
- if (client->request_read) {
- return 1;
- }
- nbytes_read = recv(client->sock, buf, sizeof(buf) - 1, 0);
- if (nbytes_read < 0) {
- int err = php_socket_errno();
- if (err == SOCK_EAGAIN) {
- return 0;
- }
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- *errstr = php_socket_strerror(err, NULL, 0);
- }
- return -1;
- } else if (nbytes_read == 0) {
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- *errstr = estrdup(php_cli_server_request_error_unexpected_eof);
- }
- return -1;
- }
- client->parser.data = client;
- nbytes_consumed = php_http_parser_execute(&client->parser, &settings, buf, nbytes_read);
- if (nbytes_consumed != (size_t)nbytes_read) {
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- if ((buf[0] & 0x80) /* SSLv2 */ || buf[0] == 0x16 /* SSLv3/TLSv1 */) {
- *errstr = estrdup("Unsupported SSL request");
- } else {
- *errstr = estrdup("Malformed HTTP request");
- }
- }
- return -1;
- }
- if (client->current_header_name) {
- char *header_name = safe_pemalloc(client->current_header_name_len, 1, 1, 1);
- memmove(header_name, client->current_header_name, client->current_header_name_len);
- client->current_header_name = header_name;
- client->current_header_name_allocated = 1;
- }
- return client->request_read ? 1: 0;
- }
- /* }}} */
- static size_t php_cli_server_client_send_through(php_cli_server_client *client, const char *str, size_t str_len) /* {{{ */
- {
- struct timeval tv = { 10, 0 };
- #ifdef PHP_WIN32
- int nbytes_left = (int)str_len;
- #else
- ssize_t nbytes_left = (ssize_t)str_len;
- #endif
- do {
- #ifdef PHP_WIN32
- int nbytes_sent;
- #else
- ssize_t nbytes_sent;
- #endif
- nbytes_sent = send(client->sock, str + str_len - nbytes_left, nbytes_left, 0);
- if (nbytes_sent < 0) {
- int err = php_socket_errno();
- if (err == SOCK_EAGAIN) {
- int nfds = php_pollfd_for(client->sock, POLLOUT, &tv);
- if (nfds > 0) {
- continue;
- } else {
- /* error or timeout */
- php_handle_aborted_connection();
- return nbytes_left;
- }
- } else {
- php_handle_aborted_connection();
- return nbytes_left;
- }
- }
- nbytes_left -= nbytes_sent;
- } while (nbytes_left > 0);
- return str_len;
- } /* }}} */
- static void php_cli_server_client_populate_request_info(const php_cli_server_client *client, sapi_request_info *request_info) /* {{{ */
- {
- char *val;
- request_info->request_method = php_http_method_str(client->request.request_method);
- request_info->proto_num = client->request.protocol_version;
- request_info->request_uri = client->request.request_uri;
- request_info->path_translated = client->request.path_translated;
- request_info->query_string = client->request.query_string;
- request_info->content_length = client->request.content_len;
- request_info->auth_user = request_info->auth_password = request_info->auth_digest = NULL;
- if (NULL != (val = zend_hash_str_find_ptr(&client->request.headers, "content-type", sizeof("content-type")-1))) {
- request_info->content_type = val;
- }
- } /* }}} */
- static void destroy_request_info(sapi_request_info *request_info) /* {{{ */
- {
- } /* }}} */
- static int php_cli_server_client_ctor(php_cli_server_client *client, php_cli_server *server, php_socket_t client_sock, struct sockaddr *addr, socklen_t addr_len) /* {{{ */
- {
- client->server = server;
- client->sock = client_sock;
- client->addr = addr;
- client->addr_len = addr_len;
- {
- zend_string *addr_str = 0;
- php_network_populate_name_from_sockaddr(addr, addr_len, &addr_str, NULL, 0);
- client->addr_str = pestrndup(ZSTR_VAL(addr_str), ZSTR_LEN(addr_str), 1);
- client->addr_str_len = ZSTR_LEN(addr_str);
- zend_string_release_ex(addr_str, 0);
- }
- php_http_parser_init(&client->parser, PHP_HTTP_REQUEST);
- client->request_read = 0;
- client->last_header_element = HEADER_NONE;
- client->current_header_name = NULL;
- client->current_header_name_len = 0;
- client->current_header_name_allocated = 0;
- client->current_header_value = NULL;
- client->current_header_value_len = 0;
- client->post_read_offset = 0;
- if (FAILURE == php_cli_server_request_ctor(&client->request)) {
- return FAILURE;
- }
- client->content_sender_initialized = 0;
- client->file_fd = -1;
- return SUCCESS;
- } /* }}} */
- static void php_cli_server_client_dtor(php_cli_server_client *client) /* {{{ */
- {
- php_cli_server_request_dtor(&client->request);
- if (client->file_fd >= 0) {
- close(client->file_fd);
- client->file_fd = -1;
- }
- pefree(client->addr, 1);
- pefree(client->addr_str, 1);
- if (client->content_sender_initialized) {
- php_cli_server_content_sender_dtor(&client->content_sender);
- }
- } /* }}} */
- static void php_cli_server_close_connection(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_MESSAGE, "%s Closing", client->addr_str);
- zend_hash_index_del(&server->clients, client->sock);
- } /* }}} */
- static int php_cli_server_send_error_page(php_cli_server *server, php_cli_server_client *client, int status) /* {{{ */
- {
- zend_string *escaped_request_uri = NULL;
- const char *status_string = get_status_string(status);
- const char *content_template = get_template_string(status);
- char *errstr = get_last_error();
- assert(status_string && content_template);
- php_cli_server_content_sender_ctor(&client->content_sender);
- client->content_sender_initialized = 1;
- escaped_request_uri = php_escape_html_entities_ex((const unsigned char *) client->request.request_uri, client->request.request_uri_len, 0, ENT_QUOTES, NULL, /* double_encode */ 0, /* quiet */ 0);
- {
- static const char prologue_template[] = "<!doctype html><html><head><title>%d %s</title>";
- php_cli_server_chunk *chunk = php_cli_server_chunk_heap_new_self_contained(strlen(prologue_template) + 3 + strlen(status_string) + 1);
- if (!chunk) {
- goto fail;
- }
- snprintf(chunk->data.heap.p, chunk->data.heap.len, prologue_template, status, status_string);
- chunk->data.heap.len = strlen(chunk->data.heap.p);
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- {
- php_cli_server_chunk *chunk = php_cli_server_chunk_immortal_new(php_cli_server_css, sizeof(php_cli_server_css) - 1);
- if (!chunk) {
- goto fail;
- }
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- {
- static const char template[] = "</head><body>";
- php_cli_server_chunk *chunk = php_cli_server_chunk_immortal_new(template, sizeof(template) - 1);
- if (!chunk) {
- goto fail;
- }
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- {
- php_cli_server_chunk *chunk = php_cli_server_chunk_heap_new_self_contained(strlen(content_template) + ZSTR_LEN(escaped_request_uri) + 3 + strlen(status_string) + 1);
- if (!chunk) {
- goto fail;
- }
- snprintf(chunk->data.heap.p, chunk->data.heap.len, content_template, status_string, ZSTR_VAL(escaped_request_uri));
- chunk->data.heap.len = strlen(chunk->data.heap.p);
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- {
- static const char epilogue_template[] = "</body></html>";
- php_cli_server_chunk *chunk = php_cli_server_chunk_immortal_new(epilogue_template, sizeof(epilogue_template) - 1);
- if (!chunk) {
- goto fail;
- }
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- {
- php_cli_server_chunk *chunk;
- smart_str buffer = { 0 };
- append_http_status_line(&buffer, client->request.protocol_version, status, 1);
- if (!buffer.s) {
- /* out of memory */
- goto fail;
- }
- append_essential_headers(&buffer, client, 1);
- smart_str_appends_ex(&buffer, "Content-Type: text/html; charset=UTF-8\r\n", 1);
- smart_str_appends_ex(&buffer, "Content-Length: ", 1);
- smart_str_append_unsigned_ex(&buffer, php_cli_server_buffer_size(&client->content_sender.buffer), 1);
- smart_str_appendl_ex(&buffer, "\r\n", 2, 1);
- smart_str_appendl_ex(&buffer, "\r\n", 2, 1);
- chunk = php_cli_server_chunk_heap_new(buffer.s, ZSTR_VAL(buffer.s), ZSTR_LEN(buffer.s));
- if (!chunk) {
- smart_str_free_ex(&buffer, 1);
- goto fail;
- }
- php_cli_server_buffer_prepend(&client->content_sender.buffer, chunk);
- }
- php_cli_server_log_response(client, status, errstr ? errstr : "?");
- php_cli_server_poller_add(&server->poller, POLLOUT, client->sock);
- if (errstr) {
- pefree(errstr, 1);
- }
- zend_string_free(escaped_request_uri);
- return SUCCESS;
- fail:
- if (errstr) {
- pefree(errstr, 1);
- }
- zend_string_free(escaped_request_uri);
- return FAILURE;
- } /* }}} */
- static int php_cli_server_dispatch_script(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- if (strlen(client->request.path_translated) != client->request.path_translated_len) {
- /* can't handle paths that contain nul bytes */
- return php_cli_server_send_error_page(server, client, 400);
- }
- {
- zend_file_handle zfd;
- zend_stream_init_filename(&zfd, SG(request_info).path_translated);
- zfd.primary_script = 1;
- zend_try {
- php_execute_script(&zfd);
- } zend_end_try();
- zend_destroy_file_handle(&zfd);
- }
- php_cli_server_log_response(client, SG(sapi_headers).http_response_code, NULL);
- return SUCCESS;
- } /* }}} */
- static int php_cli_server_begin_send_static(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- int fd;
- int status = 200;
- if (client->request.path_translated && strlen(client->request.path_translated) != client->request.path_translated_len) {
- /* can't handle paths that contain nul bytes */
- return php_cli_server_send_error_page(server, client, 400);
- }
- #ifdef PHP_WIN32
- /* The win32 namespace will cut off trailing dots and spaces. Since the
- VCWD functionality isn't used here, a sophisticated functionality
- would have to be reimplemented to know ahead there are no files
- with invalid names there. The simplest is just to forbid invalid
- filenames, which is done here. */
- if (client->request.path_translated &&
- ('.' == client->request.path_translated[client->request.path_translated_len-1] ||
- ' ' == client->request.path_translated[client->request.path_translated_len-1])) {
- return php_cli_server_send_error_page(server, client, 500);
- }
- fd = client->request.path_translated ? php_win32_ioutil_open(client->request.path_translated, O_RDONLY): -1;
- #else
- fd = client->request.path_translated ? open(client->request.path_translated, O_RDONLY): -1;
- #endif
- if (fd < 0) {
- return php_cli_server_send_error_page(server, client, 404);
- }
- php_cli_server_content_sender_ctor(&client->content_sender);
- client->content_sender_initialized = 1;
- client->file_fd = fd;
- {
- php_cli_server_chunk *chunk;
- smart_str buffer = { 0 };
- const char *mime_type = get_mime_type(server, client->request.ext, client->request.ext_len);
- append_http_status_line(&buffer, client->request.protocol_version, status, 1);
- if (!buffer.s) {
- /* out of memory */
- php_cli_server_log_response(client, 500, NULL);
- return FAILURE;
- }
- append_essential_headers(&buffer, client, 1);
- if (mime_type) {
- smart_str_appendl_ex(&buffer, "Content-Type: ", sizeof("Content-Type: ") - 1, 1);
- smart_str_appends_ex(&buffer, mime_type, 1);
- if (strncmp(mime_type, "text/", 5) == 0) {
- smart_str_appends_ex(&buffer, "; charset=UTF-8", 1);
- }
- smart_str_appendl_ex(&buffer, "\r\n", 2, 1);
- }
- smart_str_appends_ex(&buffer, "Content-Length: ", 1);
- smart_str_append_unsigned_ex(&buffer, client->request.sb.st_size, 1);
- smart_str_appendl_ex(&buffer, "\r\n", 2, 1);
- smart_str_appendl_ex(&buffer, "\r\n", 2, 1);
- chunk = php_cli_server_chunk_heap_new(buffer.s, ZSTR_VAL(buffer.s), ZSTR_LEN(buffer.s));
- if (!chunk) {
- smart_str_free_ex(&buffer, 1);
- php_cli_server_log_response(client, 500, NULL);
- return FAILURE;
- }
- php_cli_server_buffer_append(&client->content_sender.buffer, chunk);
- }
- php_cli_server_log_response(client, 200, NULL);
- php_cli_server_poller_add(&server->poller, POLLOUT, client->sock);
- return SUCCESS;
- }
- /* }}} */
- static int php_cli_server_request_startup(php_cli_server *server, php_cli_server_client *client) { /* {{{ */
- char *auth;
- php_cli_server_client_populate_request_info(client, &SG(request_info));
- if (NULL != (auth = zend_hash_str_find_ptr(&client->request.headers, "authorization", sizeof("authorization")-1))) {
- php_handle_auth_data(auth);
- }
- SG(sapi_headers).http_response_code = 200;
- if (FAILURE == php_request_startup()) {
- return FAILURE;
- }
- PG(during_request_startup) = 0;
- return SUCCESS;
- }
- /* }}} */
- static int php_cli_server_request_shutdown(php_cli_server *server, php_cli_server_client *client) { /* {{{ */
- php_request_shutdown(0);
- php_cli_server_close_connection(server, client);
- destroy_request_info(&SG(request_info));
- SG(server_context) = NULL;
- SG(rfc1867_uploaded_files) = NULL;
- return SUCCESS;
- }
- /* }}} */
- static int php_cli_server_dispatch_router(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- int decline = 0;
- zend_file_handle zfd;
- char *old_cwd;
- ALLOCA_FLAG(use_heap)
- old_cwd = do_alloca(MAXPATHLEN, use_heap);
- old_cwd[0] = '\0';
- php_ignore_value(VCWD_GETCWD(old_cwd, MAXPATHLEN - 1));
- zend_stream_init_filename(&zfd, server->router);
- zfd.primary_script = 1;
- zend_try {
- zval retval;
- ZVAL_UNDEF(&retval);
- if (SUCCESS == zend_execute_scripts(ZEND_REQUIRE, &retval, 1, &zfd)) {
- if (Z_TYPE(retval) != IS_UNDEF) {
- decline = Z_TYPE(retval) == IS_FALSE;
- zval_ptr_dtor(&retval);
- }
- }
- } zend_end_try();
- zend_destroy_file_handle(&zfd);
- if (old_cwd[0] != '\0') {
- php_ignore_value(VCWD_CHDIR(old_cwd));
- }
- free_alloca(old_cwd, use_heap);
- return decline;
- }
- /* }}} */
- static int php_cli_server_dispatch(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- int is_static_file = 0;
- const char *ext = client->request.ext;
- SG(server_context) = client;
- if (client->request.ext_len != 3
- || (ext[0] != 'p' && ext[0] != 'P') || (ext[1] != 'h' && ext[1] != 'H') || (ext[2] != 'p' && ext[2] != 'P')
- || !client->request.path_translated) {
- is_static_file = 1;
- }
- if (server->router || !is_static_file) {
- if (FAILURE == php_cli_server_request_startup(server, client)) {
- php_cli_server_request_shutdown(server, client);
- return SUCCESS;
- }
- }
- if (server->router) {
- if (!php_cli_server_dispatch_router(server, client)) {
- php_cli_server_request_shutdown(server, client);
- return SUCCESS;
- }
- }
- if (!is_static_file) {
- if (SUCCESS == php_cli_server_dispatch_script(server, client)
- || SUCCESS != php_cli_server_send_error_page(server, client, 500)) {
- if (SG(sapi_headers).http_response_code == 304) {
- SG(sapi_headers).send_default_content_type = 0;
- }
- php_cli_server_request_shutdown(server, client);
- return SUCCESS;
- }
- } else {
- if (server->router) {
- static int (*send_header_func)(sapi_headers_struct *);
- send_header_func = sapi_module.send_headers;
- /* do not generate default content type header */
- SG(sapi_headers).send_default_content_type = 0;
- /* we don't want headers to be sent */
- sapi_module.send_headers = sapi_cli_server_discard_headers;
- php_request_shutdown(0);
- sapi_module.send_headers = send_header_func;
- SG(sapi_headers).send_default_content_type = 1;
- SG(rfc1867_uploaded_files) = NULL;
- }
- if (SUCCESS != php_cli_server_begin_send_static(server, client)) {
- php_cli_server_close_connection(server, client);
- }
- SG(server_context) = NULL;
- return SUCCESS;
- }
- SG(server_context) = NULL;
- destroy_request_info(&SG(request_info));
- return SUCCESS;
- }
- /* }}} */
- static int php_cli_server_mime_type_ctor(php_cli_server *server, const php_cli_server_ext_mime_type_pair *mime_type_map) /* {{{ */
- {
- const php_cli_server_ext_mime_type_pair *pair;
- zend_hash_init(&server->extension_mime_types, 0, NULL, NULL, 1);
- for (pair = mime_type_map; pair->ext; pair++) {
- size_t ext_len = strlen(pair->ext);
- zend_hash_str_add_ptr(&server->extension_mime_types, pair->ext, ext_len, (void*)pair->mime_type);
- }
- return SUCCESS;
- } /* }}} */
- static void php_cli_server_dtor(php_cli_server *server) /* {{{ */
- {
- zend_hash_destroy(&server->clients);
- zend_hash_destroy(&server->extension_mime_types);
- if (ZEND_VALID_SOCKET(server->server_sock)) {
- closesocket(server->server_sock);
- }
- if (server->host) {
- pefree(server->host, 1);
- }
- if (server->document_root) {
- pefree(server->document_root, 1);
- }
- if (server->router) {
- pefree(server->router, 1);
- }
- #if HAVE_FORK
- if (php_cli_server_workers_max > 1 &&
- php_cli_server_workers &&
- getpid() == php_cli_server_master) {
- zend_long php_cli_server_worker;
- for (php_cli_server_worker = 0;
- php_cli_server_worker < php_cli_server_workers_max;
- php_cli_server_worker++) {
- int php_cli_server_worker_status;
- do {
- if (waitpid(php_cli_server_workers[php_cli_server_worker],
- &php_cli_server_worker_status,
- 0) == FAILURE) {
- /* an extremely bad thing happened */
- break;
- }
- } while (!WIFEXITED(php_cli_server_worker_status) &&
- !WIFSIGNALED(php_cli_server_worker_status));
- }
- pefree(php_cli_server_workers, 1);
- }
- #endif
- } /* }}} */
- static void php_cli_server_client_dtor_wrapper(zval *zv) /* {{{ */
- {
- php_cli_server_client *p = Z_PTR_P(zv);
- shutdown(p->sock, SHUT_RDWR);
- closesocket(p->sock);
- php_cli_server_poller_remove(&p->server->poller, POLLIN | POLLOUT, p->sock);
- php_cli_server_client_dtor(p);
- pefree(p, 1);
- } /* }}} */
- /**
- * Parse the host and port portions of an address specifier in
- * one of the following forms:
- * - hostOrIP:port
- * - [hostOrIP]:port
- */
- static char *php_cli_server_parse_addr(const char *addr, int *pport) {
- const char *p, *end;
- long port;
- if (addr[0] == '[') {
- /* Encapsulated [hostOrIP]:port */
- const char *start = addr + 1;
- end = strchr(start, ']');
- if (!end) {
- /* No ending ] delimiter to match [ */
- return NULL;
- }
- p = end + 1;
- if (*p != ':') {
- /* Invalid char following address/missing port */
- return NULL;
- }
- port = strtol(p + 1, (char**)&p, 10);
- if (p && *p) {
- /* Non-numeric in port */
- return NULL;
- }
- if (port < 0 || port > 65535) {
- /* Invalid port */
- return NULL;
- }
- /* Full [hostOrIP]:port provided */
- *pport = (int)port;
- return pestrndup(start, end - start, 1);
- }
- end = strchr(addr, ':');
- if (!end) {
- /* Missing port */
- return NULL;
- }
- port = strtol(end + 1, (char**)&p, 10);
- if (p && *p) {
- /* Non-numeric port */
- return NULL;
- }
- if (port < 0 || port > 65535) {
- /* Invalid port */
- return NULL;
- }
- *pport = (int)port;
- return pestrndup(addr, end - addr, 1);
- }
- static void php_cli_server_startup_workers(void) {
- char *workers = getenv("PHP_CLI_SERVER_WORKERS");
- if (!workers) {
- return;
- }
- #if HAVE_FORK
- php_cli_server_workers_max = ZEND_ATOL(workers);
- if (php_cli_server_workers_max > 1) {
- zend_long php_cli_server_worker;
- php_cli_server_workers = pecalloc(
- php_cli_server_workers_max, sizeof(pid_t), 1);
- php_cli_server_master = getpid();
- for (php_cli_server_worker = 0;
- php_cli_server_worker < php_cli_server_workers_max;
- php_cli_server_worker++) {
- pid_t pid = fork();
- if (pid == FAILURE) {
- /* no more forks allowed, work with what we have ... */
- php_cli_server_workers_max =
- php_cli_server_worker + 1;
- return;
- } else if (pid == SUCCESS) {
- return;
- } else {
- php_cli_server_workers[php_cli_server_worker] = pid;
- }
- }
- } else {
- fprintf(stderr, "number of workers must be larger than 1\n");
- }
- #else
- fprintf(stderr, "forking is not supported on this platform\n");
- #endif
- }
- static int php_cli_server_ctor(php_cli_server *server, const char *addr, const char *document_root, const char *router) /* {{{ */
- {
- int retval = SUCCESS;
- char *host = NULL;
- zend_string *errstr = NULL;
- char *_document_root = NULL;
- char *_router = NULL;
- int err = 0;
- int port = 3000;
- php_socket_t server_sock = SOCK_ERR;
- host = php_cli_server_parse_addr(addr, &port);
- if (!host) {
- fprintf(stderr, "Invalid address: %s\n", addr);
- retval = FAILURE;
- goto out;
- }
- server_sock = php_network_listen_socket(host, &port, SOCK_STREAM, &server->address_family, &server->socklen, &errstr);
- if (server_sock == SOCK_ERR) {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR, "Failed to listen on %s:%d (reason: %s)", host, port, errstr ? ZSTR_VAL(errstr) : "?");
- if (errstr) {
- zend_string_release_ex(errstr, 0);
- }
- retval = FAILURE;
- goto out;
- }
- server->server_sock = server_sock;
- php_cli_server_startup_workers();
- err = php_cli_server_poller_ctor(&server->poller);
- if (SUCCESS != err) {
- goto out;
- }
- php_cli_server_poller_add(&server->poller, POLLIN, server_sock);
- server->host = host;
- server->port = port;
- zend_hash_init(&server->clients, 0, NULL, php_cli_server_client_dtor_wrapper, 1);
- {
- size_t document_root_len = strlen(document_root);
- _document_root = pestrndup(document_root, document_root_len, 1);
- if (!_document_root) {
- retval = FAILURE;
- goto out;
- }
- server->document_root = _document_root;
- server->document_root_len = document_root_len;
- }
- if (router) {
- size_t router_len = strlen(router);
- _router = pestrndup(router, router_len, 1);
- if (!_router) {
- retval = FAILURE;
- goto out;
- }
- server->router = _router;
- server->router_len = router_len;
- } else {
- server->router = NULL;
- server->router_len = 0;
- }
- if (php_cli_server_mime_type_ctor(server, mime_type_map) == FAILURE) {
- retval = FAILURE;
- goto out;
- }
- server->is_running = 1;
- out:
- if (retval != SUCCESS) {
- if (host) {
- pefree(host, 1);
- }
- if (_document_root) {
- pefree(_document_root, 1);
- }
- if (_router) {
- pefree(_router, 1);
- }
- if (server_sock > -1) {
- closesocket(server_sock);
- }
- }
- return retval;
- } /* }}} */
- static int php_cli_server_recv_event_read_request(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- char *errstr = NULL;
- int status = php_cli_server_client_read_request(client, &errstr);
- if (status < 0) {
- if (errstr) {
- if (strcmp(errstr, php_cli_server_request_error_unexpected_eof) == 0 && client->parser.state == s_start_req) {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_MESSAGE,
- "%s Closed without sending a request; it was probably just an unused speculative preconnection", client->addr_str);
- } else {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR, "%s Invalid request (%s)", client->addr_str, errstr);
- }
- efree(errstr);
- }
- php_cli_server_close_connection(server, client);
- return FAILURE;
- } else if (status == 1 && client->request.request_method == PHP_HTTP_NOT_IMPLEMENTED) {
- return php_cli_server_send_error_page(server, client, 501);
- } else if (status == 1) {
- php_cli_server_poller_remove(&server->poller, POLLIN, client->sock);
- php_cli_server_dispatch(server, client);
- } else {
- php_cli_server_poller_add(&server->poller, POLLIN, client->sock);
- }
- return SUCCESS;
- } /* }}} */
- static int php_cli_server_send_event(php_cli_server *server, php_cli_server_client *client) /* {{{ */
- {
- if (client->content_sender_initialized) {
- if (client->file_fd >= 0 && !client->content_sender.buffer.first) {
- size_t nbytes_read;
- if (php_cli_server_content_sender_pull(&client->content_sender, client->file_fd, &nbytes_read)) {
- php_cli_server_close_connection(server, client);
- return FAILURE;
- }
- if (nbytes_read == 0) {
- close(client->file_fd);
- client->file_fd = -1;
- }
- }
- {
- size_t nbytes_sent;
- int err = php_cli_server_content_sender_send(&client->content_sender, client->sock, &nbytes_sent);
- if (err && err != SOCK_EAGAIN) {
- php_cli_server_close_connection(server, client);
- return FAILURE;
- }
- }
- if (!client->content_sender.buffer.first && client->file_fd < 0) {
- php_cli_server_close_connection(server, client);
- }
- }
- return SUCCESS;
- }
- /* }}} */
- typedef struct php_cli_server_do_event_for_each_fd_callback_params {
- php_cli_server *server;
- int(*rhandler)(php_cli_server*, php_cli_server_client*);
- int(*whandler)(php_cli_server*, php_cli_server_client*);
- } php_cli_server_do_event_for_each_fd_callback_params;
- static int php_cli_server_do_event_for_each_fd_callback(void *_params, php_socket_t fd, int event) /* {{{ */
- {
- php_cli_server_do_event_for_each_fd_callback_params *params = _params;
- php_cli_server *server = params->server;
- if (server->server_sock == fd) {
- php_cli_server_client *client = NULL;
- php_socket_t client_sock;
- socklen_t socklen = server->socklen;
- struct sockaddr *sa = pemalloc(server->socklen, 1);
- client_sock = accept(server->server_sock, sa, &socklen);
- if (!ZEND_VALID_SOCKET(client_sock)) {
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- char *errstr = php_socket_strerror(php_socket_errno(), NULL, 0);
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR,
- "Failed to accept a client (reason: %s)", errstr);
- efree(errstr);
- }
- pefree(sa, 1);
- return SUCCESS;
- }
- if (SUCCESS != php_set_sock_blocking(client_sock, 0)) {
- pefree(sa, 1);
- closesocket(client_sock);
- return SUCCESS;
- }
- client = pemalloc(sizeof(php_cli_server_client), 1);
- if (FAILURE == php_cli_server_client_ctor(client, server, client_sock, sa, socklen)) {
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR, "Failed to create a new request object");
- pefree(sa, 1);
- closesocket(client_sock);
- return SUCCESS;
- }
- php_cli_server_logf(PHP_CLI_SERVER_LOG_MESSAGE, "%s Accepted", client->addr_str);
- zend_hash_index_update_ptr(&server->clients, client_sock, client);
- php_cli_server_poller_add(&server->poller, POLLIN, client->sock);
- } else {
- php_cli_server_client *client;
- if (NULL != (client = zend_hash_index_find_ptr(&server->clients, fd))) {
- if (event & POLLIN) {
- params->rhandler(server, client);
- }
- if (event & POLLOUT) {
- params->whandler(server, client);
- }
- }
- }
- return SUCCESS;
- } /* }}} */
- static void php_cli_server_do_event_for_each_fd(php_cli_server *server, int(*rhandler)(php_cli_server*, php_cli_server_client*), int(*whandler)(php_cli_server*, php_cli_server_client*)) /* {{{ */
- {
- php_cli_server_do_event_for_each_fd_callback_params params = {
- server,
- rhandler,
- whandler
- };
- php_cli_server_poller_iter_on_active(&server->poller, ¶ms, php_cli_server_do_event_for_each_fd_callback);
- } /* }}} */
- static int php_cli_server_do_event_loop(php_cli_server *server) /* {{{ */
- {
- int retval = SUCCESS;
- while (server->is_running) {
- struct timeval tv = { 1, 0 };
- int n = php_cli_server_poller_poll(&server->poller, &tv);
- if (n > 0) {
- php_cli_server_do_event_for_each_fd(server,
- php_cli_server_recv_event_read_request,
- php_cli_server_send_event);
- } else if (n == 0) {
- /* do nothing */
- } else {
- int err = php_socket_errno();
- if (err != SOCK_EINTR) {
- if (php_cli_server_log_level >= PHP_CLI_SERVER_LOG_ERROR) {
- char *errstr = php_socket_strerror(err, NULL, 0);
- php_cli_server_logf(PHP_CLI_SERVER_LOG_ERROR, "%s", errstr);
- efree(errstr);
- }
- retval = FAILURE;
- goto out;
- }
- }
- }
- out:
- return retval;
- } /* }}} */
- static php_cli_server server;
- static void php_cli_server_sigint_handler(int sig) /* {{{ */
- {
- server.is_running = 0;
- }
- /* }}} */
- int do_cli_server(int argc, char **argv) /* {{{ */
- {
- char *php_optarg = NULL;
- int php_optind = 1;
- int c;
- const char *server_bind_address = NULL;
- extern const opt_struct OPTIONS[];
- const char *document_root = NULL;
- #ifdef PHP_WIN32
- char document_root_tmp[MAXPATHLEN];
- size_t k;
- #endif
- const char *router = NULL;
- char document_root_buf[MAXPATHLEN];
- while ((c = php_getopt(argc, argv, OPTIONS, &php_optarg, &php_optind, 0, 2))!=-1) {
- switch (c) {
- case 'S':
- server_bind_address = php_optarg;
- break;
- case 't':
- #ifndef PHP_WIN32
- document_root = php_optarg;
- #else
- k = strlen(php_optarg);
- if (k + 1 > MAXPATHLEN) {
- fprintf(stderr, "Document root path is too long.\n");
- return 1;
- }
- memmove(document_root_tmp, php_optarg, k + 1);
- /* Clean out any trailing garbage that might have been passed
- from a batch script. */
- do {
- document_root_tmp[k] = '\0';
- k--;
- } while ('"' == document_root_tmp[k] || ' ' == document_root_tmp[k]);
- document_root = document_root_tmp;
- #endif
- break;
- case 'q':
- if (php_cli_server_log_level > 1) {
- php_cli_server_log_level--;
- }
- break;
- }
- }
- if (document_root) {
- zend_stat_t sb;
- if (php_sys_stat(document_root, &sb)) {
- fprintf(stderr, "Directory %s does not exist.\n", document_root);
- return 1;
- }
- if (!S_ISDIR(sb.st_mode)) {
- fprintf(stderr, "%s is not a directory.\n", document_root);
- return 1;
- }
- if (VCWD_REALPATH(document_root, document_root_buf)) {
- document_root = document_root_buf;
- }
- } else {
- char *ret = NULL;
- #if HAVE_GETCWD
- ret = VCWD_GETCWD(document_root_buf, MAXPATHLEN);
- #elif HAVE_GETWD
- ret = VCWD_GETWD(document_root_buf);
- #endif
- document_root = ret ? document_root_buf: ".";
- }
- if (argc > php_optind) {
- router = argv[php_optind];
- }
- if (FAILURE == php_cli_server_ctor(&server, server_bind_address, document_root, router)) {
- return 1;
- }
- sapi_module.phpinfo_as_text = 0;
- {
- bool ipv6 = strchr(server.host, ':');
- php_cli_server_logf(
- PHP_CLI_SERVER_LOG_PROCESS,
- "PHP %s Development Server (http://%s%s%s:%d) started",
- PHP_VERSION, ipv6 ? "[" : "", server.host,
- ipv6 ? "]" : "", server.port);
- }
- #if defined(SIGINT)
- signal(SIGINT, php_cli_server_sigint_handler);
- #endif
- #if defined(SIGPIPE)
- signal(SIGPIPE, SIG_IGN);
- #endif
- zend_signal_init();
- php_cli_server_do_event_loop(&server);
- php_cli_server_dtor(&server);
- return 0;
- } /* }}} */
|