head.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  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. | Author: Rasmus Lerdorf <rasmus@lerdorf.on.ca> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #include <stdio.h>
  20. #include "php.h"
  21. #include "ext/standard/php_standard.h"
  22. #include "ext/date/php_date.h"
  23. #include "SAPI.h"
  24. #include "php_main.h"
  25. #include "head.h"
  26. #ifdef TM_IN_SYS_TIME
  27. #include <sys/time.h>
  28. #else
  29. #include <time.h>
  30. #endif
  31. #include "php_globals.h"
  32. /* Implementation of the language Header() function */
  33. /* {{{ proto void header(string header [, bool replace, [int http_response_code]])
  34. Sends a raw HTTP header */
  35. PHP_FUNCTION(header)
  36. {
  37. zend_bool rep = 1;
  38. sapi_header_line ctr = {0};
  39. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|bl", &ctr.line,
  40. &ctr.line_len, &rep, &ctr.response_code) == FAILURE)
  41. return;
  42. sapi_header_op(rep ? SAPI_HEADER_REPLACE:SAPI_HEADER_ADD, &ctr TSRMLS_CC);
  43. }
  44. /* }}} */
  45. /* {{{ proto void header_remove([string name])
  46. Removes an HTTP header previously set using header() */
  47. PHP_FUNCTION(header_remove)
  48. {
  49. sapi_header_line ctr = {0};
  50. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|s", &ctr.line,
  51. &ctr.line_len) == FAILURE)
  52. return;
  53. sapi_header_op(ZEND_NUM_ARGS() == 0 ? SAPI_HEADER_DELETE_ALL : SAPI_HEADER_DELETE, &ctr TSRMLS_CC);
  54. }
  55. /* }}} */
  56. PHPAPI int php_header(TSRMLS_D)
  57. {
  58. if (sapi_send_headers(TSRMLS_C)==FAILURE || SG(request_info).headers_only) {
  59. return 0; /* don't allow output */
  60. } else {
  61. return 1; /* allow output */
  62. }
  63. }
  64. PHPAPI int php_setcookie(char *name, int name_len, char *value, int value_len, time_t expires, char *path, int path_len, char *domain, int domain_len, int secure, int url_encode, int httponly TSRMLS_DC)
  65. {
  66. char *cookie, *encoded_value = NULL;
  67. int len=sizeof("Set-Cookie: ");
  68. char *dt;
  69. sapi_header_line ctr = {0};
  70. int result;
  71. if (name && strpbrk(name, "=,; \t\r\n\013\014") != NULL) { /* man isspace for \013 and \014 */
  72. zend_error( E_WARNING, "Cookie names cannot contain any of the following '=,; \\t\\r\\n\\013\\014'" );
  73. return FAILURE;
  74. }
  75. if (!url_encode && value && strpbrk(value, ",; \t\r\n\013\014") != NULL) { /* man isspace for \013 and \014 */
  76. zend_error( E_WARNING, "Cookie values cannot contain any of the following ',; \\t\\r\\n\\013\\014'" );
  77. return FAILURE;
  78. }
  79. len += name_len;
  80. if (value && url_encode) {
  81. int encoded_value_len;
  82. encoded_value = php_url_encode(value, value_len, &encoded_value_len);
  83. len += encoded_value_len;
  84. } else if ( value ) {
  85. encoded_value = estrdup(value);
  86. len += value_len;
  87. }
  88. if (path) {
  89. len += path_len;
  90. }
  91. if (domain) {
  92. len += domain_len;
  93. }
  94. cookie = emalloc(len + 100);
  95. if (value == NULL || value_len == 0) {
  96. /*
  97. * MSIE doesn't delete a cookie when you set it to a null value
  98. * so in order to force cookies to be deleted, even on MSIE, we
  99. * pick an expiry date in the past
  100. */
  101. dt = php_format_date("D, d-M-Y H:i:s T", sizeof("D, d-M-Y H:i:s T")-1, 1, 0 TSRMLS_CC);
  102. snprintf(cookie, len + 100, "Set-Cookie: %s=deleted; expires=%s; Max-Age=0", name, dt);
  103. efree(dt);
  104. } else {
  105. snprintf(cookie, len + 100, "Set-Cookie: %s=%s", name, value ? encoded_value : "");
  106. if (expires > 0) {
  107. const char *p;
  108. char tsdelta[13];
  109. strlcat(cookie, "; expires=", len + 100);
  110. dt = php_format_date("D, d-M-Y H:i:s T", sizeof("D, d-M-Y H:i:s T")-1, expires, 0 TSRMLS_CC);
  111. /* check to make sure that the year does not exceed 4 digits in length */
  112. p = zend_memrchr(dt, '-', strlen(dt));
  113. if (!p || *(p + 5) != ' ') {
  114. efree(dt);
  115. efree(cookie);
  116. efree(encoded_value);
  117. zend_error(E_WARNING, "Expiry date cannot have a year greater than 9999");
  118. return FAILURE;
  119. }
  120. strlcat(cookie, dt, len + 100);
  121. efree(dt);
  122. snprintf(tsdelta, sizeof(tsdelta), "%li", (long) difftime(expires, time(NULL)));
  123. strlcat(cookie, "; Max-Age=", len + 100);
  124. strlcat(cookie, tsdelta, len + 100);
  125. }
  126. }
  127. if (encoded_value) {
  128. efree(encoded_value);
  129. }
  130. if (path && path_len > 0) {
  131. strlcat(cookie, "; path=", len + 100);
  132. strlcat(cookie, path, len + 100);
  133. }
  134. if (domain && domain_len > 0) {
  135. strlcat(cookie, "; domain=", len + 100);
  136. strlcat(cookie, domain, len + 100);
  137. }
  138. if (secure) {
  139. strlcat(cookie, "; secure", len + 100);
  140. }
  141. if (httponly) {
  142. strlcat(cookie, "; httponly", len + 100);
  143. }
  144. ctr.line = cookie;
  145. ctr.line_len = strlen(cookie);
  146. result = sapi_header_op(SAPI_HEADER_ADD, &ctr TSRMLS_CC);
  147. efree(cookie);
  148. return result;
  149. }
  150. /* php_set_cookie(name, value, expires, path, domain, secure) */
  151. /* {{{ proto bool setcookie(string name [, string value [, int expires [, string path [, string domain [, bool secure[, bool httponly]]]]]])
  152. Send a cookie */
  153. PHP_FUNCTION(setcookie)
  154. {
  155. char *name, *value = NULL, *path = NULL, *domain = NULL;
  156. long expires = 0;
  157. zend_bool secure = 0, httponly = 0;
  158. int name_len, value_len = 0, path_len = 0, domain_len = 0;
  159. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|slssbb", &name,
  160. &name_len, &value, &value_len, &expires, &path,
  161. &path_len, &domain, &domain_len, &secure, &httponly) == FAILURE) {
  162. return;
  163. }
  164. if (php_setcookie(name, name_len, value, value_len, expires, path, path_len, domain, domain_len, secure, 1, httponly TSRMLS_CC) == SUCCESS) {
  165. RETVAL_TRUE;
  166. } else {
  167. RETVAL_FALSE;
  168. }
  169. }
  170. /* }}} */
  171. /* {{{ proto bool setrawcookie(string name [, string value [, int expires [, string path [, string domain [, bool secure[, bool httponly]]]]]])
  172. Send a cookie with no url encoding of the value */
  173. PHP_FUNCTION(setrawcookie)
  174. {
  175. char *name, *value = NULL, *path = NULL, *domain = NULL;
  176. long expires = 0;
  177. zend_bool secure = 0, httponly = 0;
  178. int name_len, value_len = 0, path_len = 0, domain_len = 0;
  179. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|slssbb", &name,
  180. &name_len, &value, &value_len, &expires, &path,
  181. &path_len, &domain, &domain_len, &secure, &httponly) == FAILURE) {
  182. return;
  183. }
  184. if (php_setcookie(name, name_len, value, value_len, expires, path, path_len, domain, domain_len, secure, 0, httponly TSRMLS_CC) == SUCCESS) {
  185. RETVAL_TRUE;
  186. } else {
  187. RETVAL_FALSE;
  188. }
  189. }
  190. /* }}} */
  191. /* {{{ proto bool headers_sent([string &$file [, int &$line]])
  192. Returns true if headers have already been sent, false otherwise */
  193. PHP_FUNCTION(headers_sent)
  194. {
  195. zval *arg1 = NULL, *arg2 = NULL;
  196. const char *file="";
  197. int line=0;
  198. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|zz", &arg1, &arg2) == FAILURE)
  199. return;
  200. if (SG(headers_sent)) {
  201. line = php_output_get_start_lineno(TSRMLS_C);
  202. file = php_output_get_start_filename(TSRMLS_C);
  203. }
  204. switch(ZEND_NUM_ARGS()) {
  205. case 2:
  206. zval_dtor(arg2);
  207. ZVAL_LONG(arg2, line);
  208. case 1:
  209. zval_dtor(arg1);
  210. if (file) {
  211. ZVAL_STRING(arg1, file, 1);
  212. } else {
  213. ZVAL_STRING(arg1, "", 1);
  214. }
  215. break;
  216. }
  217. if (SG(headers_sent)) {
  218. RETURN_TRUE;
  219. } else {
  220. RETURN_FALSE;
  221. }
  222. }
  223. /* }}} */
  224. /* {{{ php_head_apply_header_list_to_hash
  225. Turn an llist of sapi_header_struct headers into a numerically indexed zval hash */
  226. static void php_head_apply_header_list_to_hash(void *data, void *arg TSRMLS_DC)
  227. {
  228. sapi_header_struct *sapi_header = (sapi_header_struct *)data;
  229. if (arg && sapi_header) {
  230. add_next_index_string((zval *)arg, (char *)(sapi_header->header), 1);
  231. }
  232. }
  233. /* {{{ proto array headers_list(void)
  234. Return list of headers to be sent / already sent */
  235. PHP_FUNCTION(headers_list)
  236. {
  237. if (zend_parse_parameters_none() == FAILURE) {
  238. return;
  239. }
  240. array_init(return_value);
  241. zend_llist_apply_with_argument(&SG(sapi_headers).headers, php_head_apply_header_list_to_hash, return_value TSRMLS_CC);
  242. }
  243. /* }}} */
  244. /* {{{ proto long http_response_code([int response_code])
  245. Sets a response code, or returns the current HTTP response code */
  246. PHP_FUNCTION(http_response_code)
  247. {
  248. long response_code = 0;
  249. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|l", &response_code) == FAILURE) {
  250. return;
  251. }
  252. if (response_code)
  253. {
  254. long old_response_code;
  255. old_response_code = SG(sapi_headers).http_response_code;
  256. SG(sapi_headers).http_response_code = response_code;
  257. if (old_response_code) {
  258. RETURN_LONG(old_response_code);
  259. }
  260. RETURN_TRUE;
  261. }
  262. if (!SG(sapi_headers).http_response_code) {
  263. RETURN_FALSE;
  264. }
  265. RETURN_LONG(SG(sapi_headers).http_response_code);
  266. }
  267. /* }}} */
  268. /*
  269. * Local variables:
  270. * tab-width: 4
  271. * c-basic-offset: 4
  272. * vim600: sw=4 ts=4 fdm=marker
  273. * vim<600: sw=4 ts=4 * End:
  274. */