2 +--------------------------------------------------------------------+
4 +--------------------------------------------------------------------+
5 | Redistribution and use in source and binary forms, with or without |
6 | modification, are permitted provided that the conditions mentioned |
7 | in the accompanying LICENSE file are met. |
8 +--------------------------------------------------------------------+
9 | Copyright (c) 2004-2014, Michael Wallner <mike@php.net> |
10 +--------------------------------------------------------------------+
13 #include "php_http_api.h"
15 static void set_option(zval
*options
, const char *name_str
, size_t name_len
, int type
, void *value_ptr
, size_t value_len
)
17 if (Z_TYPE_P(options
) == IS_OBJECT
) {
21 zend_update_property_double(Z_OBJCE_P(options
), options
, name_str
, name_len
, *(double *)value_ptr
);
24 zend_update_property_long(Z_OBJCE_P(options
), options
, name_str
, name_len
, *(zend_long
*)value_ptr
);
27 zend_update_property_stringl(Z_OBJCE_P(options
), options
, name_str
, name_len
, value_ptr
, value_len
);
31 zend_update_property(Z_OBJCE_P(options
), options
, name_str
, name_len
, value_ptr
);
35 zend_update_property_null(Z_OBJCE_P(options
), options
, name_str
, name_len
);
38 convert_to_array(options
);
42 add_assoc_double_ex(options
, name_str
, name_len
, *(double *)value_ptr
);
45 add_assoc_long_ex(options
, name_str
, name_len
, *(zend_long
*)value_ptr
);
48 zend_string
*value
= zend_string_init(value_ptr
, value_len
, 0);
49 add_assoc_str_ex(options
, name_str
, name_len
, value
);
53 Z_ADDREF_P(value_ptr
);
54 add_assoc_zval_ex(options
, name_str
, name_len
, value_ptr
);
59 add_assoc_null_ex(options
, name_str
, name_len
);
63 static zval
*get_option(zval
*options
, const char *name_str
, size_t name_len
, zval
*tmp
)
67 if (Z_TYPE_P(options
) == IS_OBJECT
) {
68 val
= zend_read_property(Z_OBJCE_P(options
), options
, name_str
, name_len
, 0, tmp
);
69 } else if (Z_TYPE_P(options
) == IS_ARRAY
) {
70 val
= zend_symtable_str_find(Z_ARRVAL_P(options
), name_str
, name_len
);
79 static php_http_message_body_t
*get_body(zval
*options
)
81 zval zbody_tmp
, *zbody
;
82 php_http_message_body_t
*body
= NULL
;
84 if ((zbody
= get_option(options
, ZEND_STRL("body"), &zbody_tmp
))) {
85 if ((Z_TYPE_P(zbody
) == IS_OBJECT
) && instanceof_function(Z_OBJCE_P(zbody
), php_http_message_body_class_entry
)) {
86 php_http_message_body_object_t
*body_obj
= PHP_HTTP_OBJ(NULL
, zbody
);
88 body
= body_obj
->body
;
90 Z_TRY_DELREF_P(zbody
);
95 static php_http_message_t
*get_request(zval
*options
)
97 zval zrequest_tmp
, *zrequest
;
98 php_http_message_t
*request
= NULL
;
100 if ((zrequest
= get_option(options
, ZEND_STRL("request"), &zrequest_tmp
))) {
101 if (Z_TYPE_P(zrequest
) == IS_OBJECT
&& instanceof_function(Z_OBJCE_P(zrequest
), php_http_message_class_entry
)) {
102 php_http_message_object_t
*request_obj
= PHP_HTTP_OBJ(NULL
, zrequest
);
104 request
= request_obj
->message
;
106 Z_TRY_DELREF_P(zrequest
);
111 static void set_cookie(zval
*options
, zval
*zcookie_new
)
113 zval tmp
, zcookies_set_tmp
, *zcookies_set
;
114 php_http_arrkey_t key
;
115 php_http_cookie_object_t
*obj
= PHP_HTTP_OBJ(NULL
, zcookie_new
);
118 zcookies_set
= get_option(options
, ZEND_STRL("cookies"), &zcookies_set_tmp
);
119 if (zcookies_set
&& Z_TYPE_P(zcookies_set
) == IS_ARRAY
) {
120 array_copy(Z_ARRVAL_P(zcookies_set
), Z_ARRVAL(tmp
));
121 zval_ptr_dtor(zcookies_set
);
124 ZEND_HASH_FOREACH_KEY(&obj
->list
->cookies
, key
.h
, key
.key
)
126 Z_ADDREF_P(zcookie_new
);
128 add_assoc_zval_ex(&tmp
, key
.key
->val
, key
.key
->len
, zcookie_new
);
130 add_index_zval(&tmp
, key
.h
, zcookie_new
);
133 ZEND_HASH_FOREACH_END();
135 set_option(options
, ZEND_STRL("cookies"), IS_ARRAY
, &tmp
, 0);
139 php_http_cache_status_t
php_http_env_is_response_cached_by_etag(zval
*options
, const char *header_str
, size_t header_len
, php_http_message_t
*request
)
141 php_http_cache_status_t ret
= PHP_HTTP_CACHE_NO
;
142 char *header
= NULL
, *etag
= NULL
;
143 php_http_message_body_t
*body
;
144 zval zetag_tmp
, *zetag
;
147 if (!(body
= get_body(options
))) {
151 if ((zetag
= get_option(options
, ZEND_STRL("etag"), &zetag_tmp
)) && Z_TYPE_P(zetag
) != IS_NULL
) {
152 zend_string
*zs
= zval_get_string(zetag
);
153 etag
= estrndup(zs
->val
, zs
->len
);
154 zend_string_release(zs
);
155 zval_ptr_dtor(zetag
);
158 if (!etag
&& (etag
= php_http_message_body_etag(body
))) {
159 set_option(options
, ZEND_STRL("etag"), IS_STRING
, etag
, strlen(etag
));
162 if (etag
&& (header
= php_http_env_get_request_header(header_str
, header_len
, NULL
, request
))) {
163 ret
= php_http_match(header
, etag
, PHP_HTTP_MATCH_WORD
) ? PHP_HTTP_CACHE_HIT
: PHP_HTTP_CACHE_MISS
;
172 php_http_cache_status_t
php_http_env_is_response_cached_by_last_modified(zval
*options
, const char *header_str
, size_t header_len
, php_http_message_t
*request
)
174 php_http_cache_status_t ret
= PHP_HTTP_CACHE_NO
;
177 php_http_message_body_t
*body
;
180 if (!(body
= get_body(options
))) {
184 if ((zlm
= get_option(options
, ZEND_STRL("lastModified"), &zlm_tmp
))) {
185 lm
= zval_get_long(zlm
);
190 lm
= php_http_message_body_mtime(body
);
191 set_option(options
, ZEND_STRL("lastModified"), IS_LONG
, &lm
, 0);
194 if ((header
= php_http_env_get_request_header(header_str
, header_len
, NULL
, request
))) {
195 ums
= php_parse_date(header
, NULL
);
197 if (ums
> 0 && ums
>= lm
) {
198 ret
= PHP_HTTP_CACHE_HIT
;
200 ret
= PHP_HTTP_CACHE_MISS
;
208 static zend_bool
php_http_env_response_is_cacheable(php_http_env_response_t
*r
, php_http_message_t
*request
)
210 if (r
->ops
->get_status(r
) >= 400) {
214 if (php_http_env_got_request_header(ZEND_STRL("Authorization"), request
)) {
218 if (-1 == php_http_select_str(php_http_env_get_request_method(request
), 2, "HEAD", "GET")) {
225 static size_t output(void *context
, char *buf
, size_t len
)
227 php_http_env_response_t
*r
= context
;
229 if (SUCCESS
!= r
->ops
->write(r
, buf
, len
)) {
233 /* we really only need to flush when throttling is enabled,
234 because we push the data as fast as possible anyway if not */
235 if (r
->throttle
.delay
>= PHP_HTTP_DIFFSEC
) {
237 php_http_sleep(r
->throttle
.delay
);
242 static ZEND_RESULT_CODE
php_http_env_response_send_data(php_http_env_response_t
*r
, const char *buf
, size_t len
)
244 size_t chunks_sent
, chunk
= r
->throttle
.chunk
? r
->throttle
.chunk
: PHP_HTTP_SENDBUF_SIZE
;
246 if (r
->content
.encoder
) {
247 char *enc_str
= NULL
;
251 if (SUCCESS
!= php_http_encoding_stream_update(r
->content
.encoder
, buf
, len
, &enc_str
, &enc_len
)) {
255 if (SUCCESS
!= php_http_encoding_stream_finish(r
->content
.encoder
, &enc_str
, &enc_len
)) {
263 chunks_sent
= php_http_buffer_chunked_output(&r
->buffer
, enc_str
, enc_len
, buf
? chunk
: 0, output
, r
);
266 chunks_sent
= php_http_buffer_chunked_output(&r
->buffer
, buf
, len
, buf
? chunk
: 0, output
, r
);
269 return chunks_sent
!= (size_t) -1 ? SUCCESS
: FAILURE
;
272 static inline ZEND_RESULT_CODE
php_http_env_response_send_done(php_http_env_response_t
*r
)
274 return php_http_env_response_send_data(r
, NULL
, 0);
277 php_http_env_response_t
*php_http_env_response_init(php_http_env_response_t
*r
, zval
*options
, php_http_env_response_ops_t
*ops
, void *init_arg
)
282 r
= emalloc(sizeof(*r
));
284 memset(r
, 0, sizeof(*r
));
289 r
->ops
= php_http_env_response_get_sapi_ops();
292 r
->buffer
= php_http_buffer_init(NULL
);
294 ZVAL_COPY(&r
->options
, options
);
296 if (r
->ops
->init
&& (SUCCESS
!= r
->ops
->init(r
, init_arg
))) {
298 php_http_env_response_free(&r
);
300 php_http_env_response_dtor(r
);
308 void php_http_env_response_dtor(php_http_env_response_t
*r
)
313 php_http_buffer_free(&r
->buffer
);
314 zval_ptr_dtor(&r
->options
);
315 PTR_FREE(r
->content
.type
);
316 PTR_FREE(r
->content
.encoding
);
317 if (r
->content
.encoder
) {
318 php_http_encoding_stream_free(&r
->content
.encoder
);
322 void php_http_env_response_free(php_http_env_response_t
**r
)
325 php_http_env_response_dtor(*r
);
331 static ZEND_RESULT_CODE
php_http_env_response_send_head(php_http_env_response_t
*r
, php_http_message_t
*request
)
333 ZEND_RESULT_CODE ret
= SUCCESS
;
334 zval zoption_tmp
, *zoption
, *options
= &r
->options
;
340 if ((zoption
= get_option(options
, ZEND_STRL("headers"), &zoption_tmp
))) {
341 if (Z_TYPE_P(zoption
) == IS_ARRAY
) {
342 php_http_header_to_callback(Z_ARRVAL_P(zoption
), 0, (php_http_pass_format_callback_t
) r
->ops
->set_header
, r
);
344 zval_ptr_dtor(zoption
);
347 if (ret
!= SUCCESS
) {
351 if ((zoption
= get_option(options
, ZEND_STRL("responseCode"), &zoption_tmp
))) {
352 zend_long rc
= zval_get_long(zoption
);
354 zval_ptr_dtor(zoption
);
356 ret
= r
->ops
->set_status(r
, rc
);
360 if (ret
!= SUCCESS
) {
364 if ((zoption
= get_option(options
, ZEND_STRL("httpVersion"), &zoption_tmp
))) {
365 php_http_version_t v
;
366 zend_string
*zs
= zval_get_string(zoption
);
368 zval_ptr_dtor(zoption
);
369 if (zs
->len
&& php_http_version_parse(&v
, zs
->val
)) {
370 ret
= r
->ops
->set_protocol_version(r
, &v
);
371 php_http_version_dtor(&v
);
373 zend_string_release(zs
);
376 if (ret
!= SUCCESS
) {
380 if ((zoption
= get_option(options
, ZEND_STRL("cookies"), &zoption_tmp
))) {
381 if (Z_TYPE_P(zoption
) == IS_ARRAY
) {
384 ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(zoption
), zcookie
)
386 if (Z_TYPE_P(zcookie
) == IS_OBJECT
&& instanceof_function(Z_OBJCE_P(zcookie
), php_http_cookie_class_entry
)) {
387 php_http_cookie_object_t
*obj
= PHP_HTTP_OBJ(NULL
, zcookie
);
391 php_http_cookie_list_to_string(obj
->list
, &str
, &len
);
392 if (SUCCESS
!= (ret
= r
->ops
->add_header(r
, "Set-Cookie: %s", str
))) {
399 ZEND_HASH_FOREACH_END();
401 zval_ptr_dtor(zoption
);
404 if (ret
!= SUCCESS
) {
408 if ((zoption
= get_option(options
, ZEND_STRL("contentType"), &zoption_tmp
))) {
409 zend_string
*zs
= zval_get_string(zoption
);
411 zval_ptr_dtor(zoption
);
412 if (zs
->len
&& strchr(zs
->val
, '/')) {
413 if (SUCCESS
== (ret
= r
->ops
->set_header(r
, "Content-Type: %.*s", zs
->len
, zs
->val
))) {
414 r
->content
.type
= estrndup(zs
->val
, zs
->len
);
417 zend_string_release(zs
);
420 if (ret
!= SUCCESS
) {
424 if (r
->range
.status
== PHP_HTTP_RANGE_OK
) {
425 if (zend_hash_num_elements(&r
->range
.values
) == 1) {
426 zval
*range
, *begin
, *end
;
428 if ( 1 == php_http_array_list(&r
->range
.values
, 1, &range
)
429 && 2 == php_http_array_list(Z_ARRVAL_P(range
), 2, &begin
, &end
)
431 if (SUCCESS
== (ret
= r
->ops
->set_status(r
, 206))) {
432 ret
= r
->ops
->set_header(r
, "Content-Range: bytes %ld-%ld/%zu", Z_LVAL_P(begin
), Z_LVAL_P(end
), r
->content
.length
);
435 /* this should never happen */
436 zend_hash_destroy(&r
->range
.values
);
440 php_http_boundary(r
->range
.boundary
, sizeof(r
->range
.boundary
));
441 if (SUCCESS
== (ret
= r
->ops
->set_status(r
, 206))) {
442 ret
= r
->ops
->set_header(r
, "Content-Type: multipart/byteranges; boundary=%s", r
->range
.boundary
);
446 if ((zoption
= get_option(options
, ZEND_STRL("cacheControl"), &zoption_tmp
))) {
447 zend_string
*zs
= zval_get_string(zoption
);
449 zval_ptr_dtor(zoption
);
451 ret
= r
->ops
->set_header(r
, "Cache-Control: %.*s", zs
->len
, zs
->val
);
453 zend_string_release(zs
);
456 if (ret
!= SUCCESS
) {
460 if ((zoption
= get_option(options
, ZEND_STRL("contentDisposition"), &zoption_tmp
))) {
462 if (Z_TYPE_P(zoption
) == IS_ARRAY
) {
463 php_http_buffer_t buf
;
465 php_http_buffer_init(&buf
);
466 if (php_http_params_to_string(&buf
, Z_ARRVAL_P(zoption
), ZEND_STRL(","), ZEND_STRL(";"), ZEND_STRL("="), PHP_HTTP_PARAMS_DEFAULT
)) {
468 ret
= r
->ops
->set_header(r
, "Content-Disposition: %.*s", buf
.used
, buf
.data
);
472 php_http_buffer_dtor(&buf
);
474 zval_ptr_dtor(zoption
);
477 if (ret
!= SUCCESS
) {
481 if ((zoption
= get_option(options
, ZEND_STRL("contentEncoding"), &zoption_tmp
))) {
482 zend_long ce
= zval_get_long(zoption
);
484 HashTable
*result
= NULL
;
486 zval_ptr_dtor(zoption
);
488 case PHP_HTTP_CONTENT_ENCODING_GZIP
:
489 array_init(&zsupported
);
490 add_next_index_stringl(&zsupported
, ZEND_STRL("none"));
491 add_next_index_stringl(&zsupported
, ZEND_STRL("gzip"));
492 add_next_index_stringl(&zsupported
, ZEND_STRL("deflate"));
494 if ((result
= php_http_negotiate_encoding(Z_ARRVAL(zsupported
), request
))) {
495 zend_string
*key_str
= NULL
;
496 zend_ulong index
= 0;
498 zend_hash_internal_pointer_reset(result
);
499 if (HASH_KEY_IS_STRING
== zend_hash_get_current_key(result
, &key_str
, &index
)) {
500 if (zend_string_equals_literal(key_str
, "gzip")) {
501 if (!(r
->content
.encoder
= php_http_encoding_stream_init(NULL
, php_http_encoding_stream_get_deflate_ops(), PHP_HTTP_DEFLATE_TYPE_GZIP
))) {
503 } else if (SUCCESS
== (ret
= r
->ops
->set_header(r
, "Content-Encoding: gzip"))) {
504 r
->content
.encoding
= estrndup(key_str
->val
, key_str
->len
);
506 } else if (zend_string_equals_literal(key_str
, "deflate")) {
507 if (!(r
->content
.encoder
= php_http_encoding_stream_init(NULL
, php_http_encoding_stream_get_deflate_ops(), PHP_HTTP_DEFLATE_TYPE_ZLIB
))) {
509 } else if (SUCCESS
== (ret
= r
->ops
->set_header(r
, "Content-Encoding: deflate"))) {
510 r
->content
.encoding
= estrndup(key_str
->val
, key_str
->len
);
513 ret
= r
->ops
->del_header(r
, ZEND_STRL("Content-Encoding"));
516 if (SUCCESS
== ret
) {
517 ret
= r
->ops
->add_header(r
, "Vary: Accept-Encoding");
521 zend_hash_destroy(result
);
522 FREE_HASHTABLE(result
);
525 zval_dtor(&zsupported
);
528 case PHP_HTTP_CONTENT_ENCODING_NONE
:
530 ret
= r
->ops
->del_header(r
, ZEND_STRL("Content-Encoding"));
535 if (SUCCESS
!= ret
) {
539 if (php_http_env_response_is_cacheable(r
, request
)) {
540 switch (php_http_env_is_response_cached_by_etag(options
, ZEND_STRL("If-None-Match"), request
)) {
541 case PHP_HTTP_CACHE_MISS
:
544 case PHP_HTTP_CACHE_NO
:
545 if (PHP_HTTP_CACHE_HIT
!= php_http_env_is_response_cached_by_last_modified(options
, ZEND_STRL("If-Modified-Since"), request
)) {
550 case PHP_HTTP_CACHE_HIT
:
551 ret
= r
->ops
->set_status(r
, 304);
556 if ((zoption
= get_option(options
, ZEND_STRL("etag"), &zoption_tmp
))) {
557 zend_string
*zs
= zval_get_string(zoption
);
559 zval_ptr_dtor(zoption
);
560 if (*zs
->val
!= '"' && strncmp(zs
->val
, "W/\"", 3)) {
561 ret
= r
->ops
->set_header(r
, "ETag: \"%s\"", zs
->val
);
563 ret
= r
->ops
->set_header(r
, "ETag: %s", zs
->val
);
565 zend_string_release(zs
);
567 if ((zoption
= get_option(options
, ZEND_STRL("lastModified"), &zoption_tmp
))) {
568 zend_long lm
= zval_get_long(zoption
);
570 zval_ptr_dtor(zoption
);
572 zend_string
*date
= php_format_date(ZEND_STRL(PHP_HTTP_DATE_FORMAT
), lm
, 0);
574 ret
= r
->ops
->set_header(r
, "Last-Modified: %s", date
->val
);
575 zend_string_release(date
);
585 static ZEND_RESULT_CODE
php_http_env_response_send_body(php_http_env_response_t
*r
)
587 ZEND_RESULT_CODE ret
= SUCCESS
;
588 zval zoption_tmp
, *zoption
;
589 php_http_message_body_t
*body
;
595 if ((body
= get_body(&r
->options
))) {
596 if ((zoption
= get_option(&r
->options
, ZEND_STRL("throttleDelay"), &zoption_tmp
))) {
597 r
->throttle
.delay
= zval_get_double(zoption
);
598 zval_ptr_dtor(zoption
);
600 if ((zoption
= get_option(&r
->options
, ZEND_STRL("throttleChunk"), &zoption_tmp
))) {
601 r
->throttle
.chunk
= zval_get_long(zoption
);
602 zval_ptr_dtor(zoption
);
605 if (r
->range
.status
== PHP_HTTP_RANGE_OK
) {
606 if (zend_hash_num_elements(&r
->range
.values
) == 1) {
608 zval
*range
, *begin
, *end
;
610 if ( 1 == php_http_array_list(&r
->range
.values
, 1, &range
)
611 && 2 == php_http_array_list(Z_ARRVAL_P(range
), 2, &begin
, &end
)
614 ret
= php_http_message_body_to_callback(body
, (php_http_pass_callback_t
) php_http_env_response_send_data
, r
, Z_LVAL_P(begin
), Z_LVAL_P(end
) - Z_LVAL_P(begin
) + 1);
615 if (ret
== SUCCESS
) {
616 ret
= php_http_env_response_send_done(r
);
618 zend_hash_destroy(&r
->range
.values
);
620 /* this should never happen */
621 zend_hash_destroy(&r
->range
.values
);
622 r
->ops
->set_status(r
, 500);
627 /* send multipart/byte-ranges message */
630 ZEND_HASH_FOREACH_VAL(&r
->range
.values
, chunk
)
634 if (2 == php_http_array_list(Z_ARRVAL_P(chunk
), 2, &begin
, &end
)) {
635 php_http_buffer_appendf(r
->buffer
,
638 "Content-Type: %s" PHP_HTTP_CRLF
639 "Content-Range: bytes %ld-%ld/%zu" PHP_HTTP_CRLF PHP_HTTP_CRLF
,
642 r
->content
.type
? r
->content
.type
: "application/octet-stream",
647 ret
= php_http_message_body_to_callback(body
, (php_http_pass_callback_t
) php_http_env_response_send_data
, r
, Z_LVAL_P(begin
), Z_LVAL_P(end
) - Z_LVAL_P(begin
) + 1);
650 ZEND_HASH_FOREACH_END();
652 if (ret
== SUCCESS
) {
653 php_http_buffer_appendf(r
->buffer
, PHP_HTTP_CRLF
"--%s--", r
->range
.boundary
);
654 ret
= php_http_env_response_send_done(r
);
656 zend_hash_destroy(&r
->range
.values
);
660 ret
= php_http_message_body_to_callback(body
, (php_http_pass_callback_t
) php_http_env_response_send_data
, r
, 0, 0);
661 if (ret
== SUCCESS
) {
662 ret
= php_http_env_response_send_done(r
);
669 ZEND_RESULT_CODE
php_http_env_response_send(php_http_env_response_t
*r
)
671 php_http_message_t
*request
;
672 php_http_message_body_t
*body
;
674 request
= get_request(&r
->options
);
676 /* check for ranges */
677 if ((body
= get_body(&r
->options
))) {
678 r
->content
.length
= php_http_message_body_size(body
);
680 if (SUCCESS
!= r
->ops
->set_header(r
, "Accept-Ranges: bytes")) {
683 ZEND_INIT_SYMTABLE_EX(&r
->range
.values
, 0, 0);
684 r
->range
.status
= php_http_env_get_request_ranges(&r
->range
.values
, r
->content
.length
, request
);
686 switch (r
->range
.status
) {
687 case PHP_HTTP_RANGE_NO
:
688 zend_hash_destroy(&r
->range
.values
);
691 case PHP_HTTP_RANGE_ERR
:
692 if (php_http_env_got_request_header(ZEND_STRL("If-Range"), request
)) {
693 r
->range
.status
= PHP_HTTP_RANGE_NO
;
694 zend_hash_destroy(&r
->range
.values
);
697 zend_hash_destroy(&r
->range
.values
);
698 if (SUCCESS
!= r
->ops
->set_status(r
, 416)) {
701 if (SUCCESS
!= r
->ops
->set_header(r
, "Content-Range: bytes */%zu", r
->content
.length
)) {
707 case PHP_HTTP_RANGE_OK
:
708 if (PHP_HTTP_CACHE_MISS
== php_http_env_is_response_cached_by_etag(&r
->options
, ZEND_STRL("If-Range"), request
)
709 || PHP_HTTP_CACHE_MISS
== php_http_env_is_response_cached_by_last_modified(&r
->options
, ZEND_STRL("If-Range"), request
)
711 r
->range
.status
= PHP_HTTP_RANGE_NO
;
712 zend_hash_destroy(&r
->range
.values
);
715 if (PHP_HTTP_CACHE_MISS
== php_http_env_is_response_cached_by_etag(&r
->options
, ZEND_STRL("If-Match"), request
)
716 || PHP_HTTP_CACHE_MISS
== php_http_env_is_response_cached_by_last_modified(&r
->options
, ZEND_STRL("If-Unmodified-Since"), request
)
717 || PHP_HTTP_CACHE_MISS
== php_http_env_is_response_cached_by_last_modified(&r
->options
, ZEND_STRL("Unless-Modified-Since"), request
)
720 zend_hash_destroy(&r
->range
.values
);
721 if (SUCCESS
!= r
->ops
->set_status(r
, 412)) {
732 if (SUCCESS
!= php_http_env_response_send_head(r
, request
)) {
733 php_error_docref(NULL
, E_WARNING
, "Failed to send response headers");
737 if (SUCCESS
!= php_http_env_response_send_body(r
)) {
738 php_error_docref(NULL
, E_WARNING
, "Failed to send response body");
742 if (SUCCESS
!= r
->ops
->finish(r
)) {
743 php_error_docref(NULL
, E_WARNING
, "Failed to finish response");
750 static long php_http_env_response_sapi_get_status(php_http_env_response_t
*r
)
752 return php_http_env_get_response_code();
754 static ZEND_RESULT_CODE
php_http_env_response_sapi_set_status(php_http_env_response_t
*r
, long http_code
)
756 return php_http_env_set_response_code(http_code
);
758 static ZEND_RESULT_CODE
php_http_env_response_sapi_set_protocol_version(php_http_env_response_t
*r
, php_http_version_t
*v
)
761 return php_http_env_set_response_protocol_version(v
);
763 static ZEND_RESULT_CODE
php_http_env_response_sapi_set_header(php_http_env_response_t
*r
, const char *fmt
, ...)
765 ZEND_RESULT_CODE ret
;
769 ret
= php_http_env_set_response_header_va(0, 1, fmt
, args
);
774 static ZEND_RESULT_CODE
php_http_env_response_sapi_add_header(php_http_env_response_t
*r
, const char *fmt
, ...)
776 ZEND_RESULT_CODE ret
;
780 ret
= php_http_env_set_response_header_va(0, 0, fmt
, args
);
785 static ZEND_RESULT_CODE
php_http_env_response_sapi_del_header(php_http_env_response_t
*r
, const char *header_str
, size_t header_len
)
787 return php_http_env_set_response_header_value(0, header_str
, header_len
, NULL
, 1);
789 static ZEND_RESULT_CODE
php_http_env_response_sapi_write(php_http_env_response_t
*r
, const char *data_str
, size_t data_len
)
791 if (0 < PHPWRITE(data_str
, data_len
)) {
796 static ZEND_RESULT_CODE
php_http_env_response_sapi_flush(php_http_env_response_t
*r
)
798 if (php_output_get_level()) {
799 php_output_flush_all();
801 if (!(php_output_get_status() & PHP_OUTPUT_IMPLICITFLUSH
)) {
807 static ZEND_RESULT_CODE
php_http_env_response_sapi_finish(php_http_env_response_t
*r
)
812 static php_http_env_response_ops_t php_http_env_response_sapi_ops
= {
815 php_http_env_response_sapi_get_status
,
816 php_http_env_response_sapi_set_status
,
817 php_http_env_response_sapi_set_protocol_version
,
818 php_http_env_response_sapi_set_header
,
819 php_http_env_response_sapi_add_header
,
820 php_http_env_response_sapi_del_header
,
821 php_http_env_response_sapi_write
,
822 php_http_env_response_sapi_flush
,
823 php_http_env_response_sapi_finish
826 php_http_env_response_ops_t
*php_http_env_response_get_sapi_ops(void)
828 return &php_http_env_response_sapi_ops
;
831 typedef struct php_http_env_response_stream_ctx
{
833 php_http_version_t version
;
837 php_stream_filter
*chunked_filter
;
838 php_http_message_t
*request
;
843 } php_http_env_response_stream_ctx_t
;
845 static ZEND_RESULT_CODE
php_http_env_response_stream_init(php_http_env_response_t
*r
, void *init_arg
)
847 php_http_env_response_stream_ctx_t
*ctx
;
848 size_t buffer_size
= 0x1000;
850 ctx
= ecalloc(1, sizeof(*ctx
));
852 ctx
->stream
= init_arg
;
853 ++GC_REFCOUNT(ctx
->stream
->res
);
854 ZEND_INIT_SYMTABLE(&ctx
->header
);
855 php_http_version_init(&ctx
->version
, 1, 1);
856 php_stream_set_option(ctx
->stream
, PHP_STREAM_OPTION_WRITE_BUFFER
, PHP_STREAM_BUFFER_FULL
, &buffer_size
);
857 ctx
->status_code
= 200;
859 ctx
->request
= get_request(&r
->options
);
861 /* there are some limitations regarding TE:chunked, see https://tools.ietf.org/html/rfc7230#section-3.3.1 */
862 if (ctx
->request
&& ctx
->request
->http
.version
.major
== 1 && ctx
->request
->http
.version
.minor
== 0) {
863 ctx
->version
.minor
= 0;
870 static void php_http_env_response_stream_dtor(php_http_env_response_t
*r
)
872 php_http_env_response_stream_ctx_t
*ctx
= r
->ctx
;
874 if (ctx
->chunked_filter
) {
875 ctx
->chunked_filter
= php_stream_filter_remove(ctx
->chunked_filter
, 1);
877 zend_hash_destroy(&ctx
->header
);
878 zend_list_delete(ctx
->stream
->res
);
882 static void php_http_env_response_stream_header(php_http_env_response_stream_ctx_t
*ctx
, HashTable
*header
, php_http_buffer_t
*buf
)
886 ZEND_HASH_FOREACH_VAL(header
, val
)
888 if (Z_TYPE_P(val
) == IS_ARRAY
) {
889 php_http_env_response_stream_header(ctx
, Z_ARRVAL_P(val
), buf
);
891 zend_string
*zs
= zval_get_string(val
);
894 /* disable chunked transfer encoding if we've got an explicit content-length */
895 if (!strncasecmp(zs
->val
, "Content-Length:", lenof("Content-Length:"))) {
899 php_http_buffer_append(buf
, zs
->val
, zs
->len
);
900 php_http_buffer_appends(buf
, PHP_HTTP_CRLF
);
901 zend_string_release(zs
);
904 ZEND_HASH_FOREACH_END();
906 static ZEND_RESULT_CODE
php_http_env_response_stream_start(php_http_env_response_stream_ctx_t
*ctx
)
908 php_http_buffer_t header_buf
;
910 if (ctx
->started
|| ctx
->finished
) {
914 php_http_buffer_init(&header_buf
);
915 php_http_buffer_appendf(&header_buf
, "HTTP/%u.%u %ld %s" PHP_HTTP_CRLF
, ctx
->version
.major
, ctx
->version
.minor
, ctx
->status_code
, php_http_env_get_response_status_for_code(ctx
->status_code
));
917 /* there are some limitations regarding TE:chunked, see https://tools.ietf.org/html/rfc7230#section-3.3.1 */
918 if (ctx
->version
.major
== 1 && ctx
->version
.minor
== 0) {
920 } else if (ctx
->status_code
== 204 || ctx
->status_code
/100 == 1) {
922 } else if (ctx
->request
&& ctx
->status_code
/100 == 2 && !strcasecmp(ctx
->request
->http
.info
.request
.method
, "CONNECT")) {
926 php_http_env_response_stream_header(ctx
, &ctx
->header
, &header_buf
);
928 /* enable chunked transfer encoding */
930 php_http_buffer_appends(&header_buf
, "Transfer-Encoding: chunked" PHP_HTTP_CRLF
);
933 php_http_buffer_appends(&header_buf
, PHP_HTTP_CRLF
);
935 if (header_buf
.used
== php_stream_write(ctx
->stream
, header_buf
.data
, header_buf
.used
)) {
938 php_http_buffer_dtor(&header_buf
);
939 php_stream_flush(ctx
->stream
);
942 ctx
->chunked_filter
= php_stream_filter_create("http.chunked_encode", NULL
, 0);
943 php_stream_filter_append(&ctx
->stream
->writefilters
, ctx
->chunked_filter
);
946 return ctx
->started
? SUCCESS
: FAILURE
;
948 static long php_http_env_response_stream_get_status(php_http_env_response_t
*r
)
950 php_http_env_response_stream_ctx_t
*ctx
= r
->ctx
;
952 return ctx
->status_code
;
954 static ZEND_RESULT_CODE
php_http_env_response_stream_set_status(php_http_env_response_t
*r
, long http_code
)
956 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
958 if (stream_ctx
->started
|| stream_ctx
->finished
) {
962 stream_ctx
->status_code
= http_code
;
966 static ZEND_RESULT_CODE
php_http_env_response_stream_set_protocol_version(php_http_env_response_t
*r
, php_http_version_t
*v
)
968 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
970 if (stream_ctx
->started
|| stream_ctx
->finished
) {
974 memcpy(&stream_ctx
->version
, v
, sizeof(stream_ctx
->version
));
978 static ZEND_RESULT_CODE
php_http_env_response_stream_set_header_ex(php_http_env_response_t
*r
, zend_bool replace
, const char *fmt
, va_list argv
)
980 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
981 char *header_end
, *header_str
= NULL
;
982 size_t header_len
= 0;
983 zval zheader
, *zheader_ptr
;
984 zend_string
*header_key
;
987 if (stream_ctx
->started
|| stream_ctx
->finished
) {
991 header_len
= vspprintf(&header_str
, 0, fmt
, argv
);
993 if (!(header_end
= strchr(header_str
, ':'))) {
998 header_key
= zend_string_init(header_str
, header_end
- header_str
, 0);
1000 if (!replace
&& (zheader_ptr
= zend_hash_find(&stream_ctx
->header
, header_key
))) {
1001 convert_to_array(zheader_ptr
);
1002 rv
= add_next_index_str(zheader_ptr
, php_http_cs2zs(header_str
, header_len
));
1004 ZVAL_STR(&zheader
, php_http_cs2zs(header_str
, header_len
));
1006 rv
= zend_hash_update(&stream_ctx
->header
, header_key
, &zheader
)
1007 ? SUCCESS
: FAILURE
;
1010 zend_string_release(header_key
);
1014 static ZEND_RESULT_CODE
php_http_env_response_stream_set_header(php_http_env_response_t
*r
, const char *fmt
, ...)
1016 ZEND_RESULT_CODE ret
;
1019 va_start(argv
, fmt
);
1020 ret
= php_http_env_response_stream_set_header_ex(r
, 1, fmt
, argv
);
1025 static ZEND_RESULT_CODE
php_http_env_response_stream_add_header(php_http_env_response_t
*r
, const char *fmt
, ...)
1027 ZEND_RESULT_CODE ret
;
1030 va_start(argv
, fmt
);
1031 ret
= php_http_env_response_stream_set_header_ex(r
, 0, fmt
, argv
);
1036 static ZEND_RESULT_CODE
php_http_env_response_stream_del_header(php_http_env_response_t
*r
, const char *header_str
, size_t header_len
)
1038 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
1040 if (stream_ctx
->started
|| stream_ctx
->finished
) {
1044 zend_hash_str_del(&stream_ctx
->header
, header_str
, header_len
);
1047 static ZEND_RESULT_CODE
php_http_env_response_stream_write(php_http_env_response_t
*r
, const char *data_str
, size_t data_len
)
1049 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
1051 if (stream_ctx
->finished
) {
1054 if (!stream_ctx
->started
) {
1055 if (SUCCESS
!= php_http_env_response_stream_start(stream_ctx
)) {
1060 if (data_len
!= php_stream_write(stream_ctx
->stream
, data_str
, data_len
)) {
1066 static ZEND_RESULT_CODE
php_http_env_response_stream_flush(php_http_env_response_t
*r
)
1068 php_http_env_response_stream_ctx_t
*stream_ctx
= r
->ctx
;
1070 if (stream_ctx
->finished
) {
1073 if (!stream_ctx
->started
) {
1074 if (SUCCESS
!= php_http_env_response_stream_start(stream_ctx
)) {
1079 return php_stream_flush(stream_ctx
->stream
);
1081 static ZEND_RESULT_CODE
php_http_env_response_stream_finish(php_http_env_response_t
*r
)
1083 php_http_env_response_stream_ctx_t
*ctx
= r
->ctx
;
1085 if (ctx
->finished
) {
1088 if (!ctx
->started
) {
1089 if (SUCCESS
!= php_http_env_response_stream_start(ctx
)) {
1094 php_stream_flush(ctx
->stream
);
1095 if (ctx
->chunked
&& ctx
->chunked_filter
) {
1096 php_stream_filter_flush(ctx
->chunked_filter
, 1);
1097 ctx
->chunked_filter
= php_stream_filter_remove(ctx
->chunked_filter
, 1 TSRMLS_CC
);
1105 static php_http_env_response_ops_t php_http_env_response_stream_ops
= {
1106 php_http_env_response_stream_init
,
1107 php_http_env_response_stream_dtor
,
1108 php_http_env_response_stream_get_status
,
1109 php_http_env_response_stream_set_status
,
1110 php_http_env_response_stream_set_protocol_version
,
1111 php_http_env_response_stream_set_header
,
1112 php_http_env_response_stream_add_header
,
1113 php_http_env_response_stream_del_header
,
1114 php_http_env_response_stream_write
,
1115 php_http_env_response_stream_flush
,
1116 php_http_env_response_stream_finish
1119 php_http_env_response_ops_t
*php_http_env_response_get_stream_ops(void)
1121 return &php_http_env_response_stream_ops
;
1124 #define PHP_HTTP_ENV_RESPONSE_OBJECT_INIT(obj) \
1126 if (!obj->message) { \
1127 obj->message = php_http_message_init_env(NULL, PHP_HTTP_RESPONSE); \
1131 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse___construct
, 0, 0, 0)
1132 ZEND_END_ARG_INFO();
1133 static PHP_METHOD(HttpEnvResponse
, __construct
)
1135 php_http_message_object_t
*obj
;
1137 php_http_expect(SUCCESS
== zend_parse_parameters_none(), invalid_arg
, return);
1139 obj
= PHP_HTTP_OBJ(NULL
, getThis());
1141 php_http_expect(obj
->message
= php_http_message_init_env(obj
->message
, PHP_HTTP_RESPONSE
), unexpected_val
, return);
1144 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse___invoke
, 0, 0, 1)
1145 ZEND_ARG_INFO(0, ob_string
)
1146 ZEND_ARG_INFO(0, ob_flags
)
1147 ZEND_END_ARG_INFO();
1148 static PHP_METHOD(HttpEnvResponse
, __invoke
)
1152 zend_long ob_flags
= 0;
1154 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "s|l", &ob_str
, &ob_len
, &ob_flags
)) {
1155 php_http_message_object_t
*obj
= PHP_HTTP_OBJ(NULL
, getThis());
1157 PHP_HTTP_ENV_RESPONSE_OBJECT_INIT(obj
);
1159 php_http_message_object_init_body_object(obj
);
1160 php_http_message_body_append(obj
->message
->body
, ob_str
, ob_len
);
1165 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setEnvRequest
, 0, 0, 1)
1166 ZEND_ARG_OBJ_INFO(0, env_request
, http
\\Message
, 1)
1167 ZEND_END_ARG_INFO();
1168 static PHP_METHOD(HttpEnvResponse
, setEnvRequest
)
1170 zval
*env_req
= NULL
;
1172 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|O", &env_req
, php_http_message_class_entry
), invalid_arg
, return);
1174 set_option(getThis(), ZEND_STRL("request"), IS_OBJECT
, env_req
, 0);
1175 RETVAL_ZVAL_FAST(getThis());
1178 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setContentType
, 0, 0, 1)
1179 ZEND_ARG_INFO(0, content_type
)
1180 ZEND_END_ARG_INFO();
1181 static PHP_METHOD(HttpEnvResponse
, setContentType
)
1183 char *ct_str
= NULL
;
1186 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "s!", &ct_str
, &ct_len
), invalid_arg
, return);
1188 set_option(getThis(), ZEND_STRL("contentType"), IS_STRING
, ct_str
, ct_len
);
1189 RETVAL_ZVAL_FAST(getThis());
1192 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setContentDisposition
, 0, 0, 1)
1193 ZEND_ARG_ARRAY_INFO(0, disposition_params
, 1)
1194 ZEND_END_ARG_INFO();
1195 static PHP_METHOD(HttpEnvResponse
, setContentDisposition
)
1199 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "a", &zdisposition
), invalid_arg
, return);
1201 zend_update_property(Z_OBJCE_P(getThis()), getThis(), ZEND_STRL("contentDisposition"), zdisposition
);
1202 RETVAL_ZVAL_FAST(getThis());
1205 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setContentEncoding
, 0, 0, 1)
1206 ZEND_ARG_INFO(0, content_encoding
)
1207 ZEND_END_ARG_INFO();
1208 static PHP_METHOD(HttpEnvResponse
, setContentEncoding
)
1212 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "l", &ce
), invalid_arg
, return);
1214 set_option(getThis(), ZEND_STRL("contentEncoding"), IS_LONG
, &ce
, 0);
1215 RETVAL_ZVAL_FAST(getThis());
1218 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setCacheControl
, 0, 0, 1)
1219 ZEND_ARG_INFO(0, cache_control
)
1220 ZEND_END_ARG_INFO();
1221 static PHP_METHOD(HttpEnvResponse
, setCacheControl
)
1223 char *cc_str
= NULL
;
1226 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "s!", &cc_str
, &cc_len
), invalid_arg
, return);
1228 set_option(getThis(), ZEND_STRL("cacheControl"), IS_STRING
, cc_str
, cc_len
);
1229 RETVAL_ZVAL_FAST(getThis());
1232 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setLastModified
, 0, 0, 1)
1233 ZEND_ARG_INFO(0, last_modified
)
1234 ZEND_END_ARG_INFO();
1235 static PHP_METHOD(HttpEnvResponse
, setLastModified
)
1237 zend_long last_modified
;
1239 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "l", &last_modified
), invalid_arg
, return);
1241 set_option(getThis(), ZEND_STRL("lastModified"), IS_LONG
, &last_modified
, 0);
1242 RETVAL_ZVAL_FAST(getThis());
1245 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_isCachedByLastModified
, 0, 0, 0)
1246 ZEND_ARG_INFO(0, header_name
)
1247 ZEND_END_ARG_INFO();
1248 static PHP_METHOD(HttpEnvResponse
, isCachedByLastModified
)
1250 char *header_name_str
= NULL
;
1251 size_t header_name_len
= 0;
1253 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|s!", &header_name_str
, &header_name_len
)) {
1254 if (!header_name_str
|| !header_name_len
) {
1255 header_name_str
= "If-Modified-Since";
1256 header_name_len
= lenof("If-Modified-Since");
1259 RETURN_LONG(php_http_env_is_response_cached_by_last_modified(getThis(), header_name_str
, header_name_len
, get_request(getThis())));
1263 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setEtag
, 0, 0, 1)
1264 ZEND_ARG_INFO(0, etag
)
1265 ZEND_END_ARG_INFO();
1266 static PHP_METHOD(HttpEnvResponse
, setEtag
)
1268 char *etag_str
= NULL
;
1269 size_t etag_len
= 0;
1271 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "s!", &etag_str
, &etag_len
), invalid_arg
, return);
1273 set_option(getThis(), ZEND_STRL("etag"), IS_STRING
, etag_str
, etag_len
);
1274 RETVAL_ZVAL_FAST(getThis());
1277 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_isCachedByEtag
, 0, 0, 0)
1278 ZEND_ARG_INFO(0, header_name
)
1279 ZEND_END_ARG_INFO();
1280 static PHP_METHOD(HttpEnvResponse
, isCachedByEtag
)
1282 char *header_name_str
= NULL
;
1283 size_t header_name_len
= 0;
1285 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|s!", &header_name_str
, &header_name_len
)) {
1286 if (!header_name_str
|| !header_name_len
) {
1287 header_name_str
= "If-None-Match";
1288 header_name_len
= lenof("If-None-Match");
1290 RETURN_LONG(php_http_env_is_response_cached_by_etag(getThis(), header_name_str
, header_name_len
, get_request(getThis())));
1294 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setThrottleRate
, 0, 0, 1)
1295 ZEND_ARG_INFO(0, chunk_size
)
1296 ZEND_ARG_INFO(0, delay
)
1297 ZEND_END_ARG_INFO();
1298 static PHP_METHOD(HttpEnvResponse
, setThrottleRate
)
1300 zend_long chunk_size
;
1303 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "l|d", &chunk_size
, &delay
), invalid_arg
, return);
1305 set_option(getThis(), ZEND_STRL("throttleDelay"), IS_DOUBLE
, &delay
, 0);
1306 set_option(getThis(), ZEND_STRL("throttleChunk"), IS_LONG
, &chunk_size
, 0);
1307 RETVAL_ZVAL_FAST(getThis());
1310 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_setCookie
, 0, 0, 1)
1311 ZEND_ARG_INFO(0, cookie
)
1312 ZEND_END_ARG_INFO();
1313 static PHP_METHOD(HttpEnvResponse
, setCookie
)
1315 zval
*zcookie_new
, tmp
;
1317 zend_error_handling zeh
;
1318 php_http_cookie_list_t
*list
= NULL
;
1320 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "z", &zcookie_new
), invalid_arg
, return);
1322 zend_replace_error_handling(EH_THROW
, php_http_exception_unexpected_val_class_entry
, &zeh
);
1323 switch (Z_TYPE_P(zcookie_new
)) {
1325 if (instanceof_function(Z_OBJCE_P(zcookie_new
), php_http_cookie_class_entry
)) {
1326 Z_ADDREF_P(zcookie_new
);
1331 list
= php_http_cookie_list_from_struct(NULL
, zcookie_new
);
1333 ZVAL_OBJECT(zcookie_new
, &php_http_cookie_object_new_ex(php_http_cookie_class_entry
, list
)->zo
, 1);
1337 zs
= zval_get_string(zcookie_new
);
1338 list
= php_http_cookie_list_parse(NULL
, zs
->val
, zs
->len
, 0, NULL
);
1339 zend_string_release(zs
);
1341 ZVAL_OBJECT(zcookie_new
, &php_http_cookie_object_new_ex(php_http_cookie_class_entry
, list
)->zo
, 1);
1343 zend_restore_error_handling(&zeh
);
1345 set_cookie(getThis(), zcookie_new
);
1346 zval_ptr_dtor(zcookie_new
);
1348 RETVAL_ZVAL_FAST(getThis());
1351 ZEND_BEGIN_ARG_INFO_EX(ai_HttpEnvResponse_send
, 0, 0, 0)
1352 ZEND_ARG_INFO(0, stream
)
1353 ZEND_END_ARG_INFO();
1354 static PHP_METHOD(HttpEnvResponse
, send
)
1356 zval
*zstream
= NULL
;
1357 php_stream
*s
= NULL
;
1359 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|r", &zstream
)) {
1360 /* first flush the output layer to avoid conflicting headers and output;
1361 * also, ob_start($thisEnvResponse) might have been called */
1362 php_output_end_all();
1365 php_http_env_response_t
*r
;
1367 php_stream_from_zval(s
, zstream
);
1368 r
= php_http_env_response_init(NULL
, getThis(), php_http_env_response_get_stream_ops(), s
);
1373 RETVAL_BOOL(SUCCESS
== php_http_env_response_send(r
));
1374 php_http_env_response_free(&r
);
1376 php_http_env_response_t r
;
1378 if (!php_http_env_response_init(&r
, getThis(), NULL
, NULL
)) {
1382 RETVAL_BOOL(SUCCESS
== php_http_env_response_send(&r
));
1383 php_http_env_response_dtor(&r
);
1388 static zend_function_entry php_http_env_response_methods
[] = {
1389 PHP_ME(HttpEnvResponse
, __construct
, ai_HttpEnvResponse___construct
, ZEND_ACC_PUBLIC
|ZEND_ACC_CTOR
)
1390 PHP_ME(HttpEnvResponse
, __invoke
, ai_HttpEnvResponse___invoke
, ZEND_ACC_PUBLIC
)
1391 PHP_ME(HttpEnvResponse
, setEnvRequest
, ai_HttpEnvResponse_setEnvRequest
, ZEND_ACC_PUBLIC
)
1392 PHP_ME(HttpEnvResponse
, setCookie
, ai_HttpEnvResponse_setCookie
, ZEND_ACC_PUBLIC
)
1393 PHP_ME(HttpEnvResponse
, setContentType
, ai_HttpEnvResponse_setContentType
, ZEND_ACC_PUBLIC
)
1394 PHP_ME(HttpEnvResponse
, setContentDisposition
, ai_HttpEnvResponse_setContentDisposition
, ZEND_ACC_PUBLIC
)
1395 PHP_ME(HttpEnvResponse
, setContentEncoding
, ai_HttpEnvResponse_setContentEncoding
, ZEND_ACC_PUBLIC
)
1396 PHP_ME(HttpEnvResponse
, setCacheControl
, ai_HttpEnvResponse_setCacheControl
, ZEND_ACC_PUBLIC
)
1397 PHP_ME(HttpEnvResponse
, setLastModified
, ai_HttpEnvResponse_setLastModified
, ZEND_ACC_PUBLIC
)
1398 PHP_ME(HttpEnvResponse
, isCachedByLastModified
, ai_HttpEnvResponse_isCachedByLastModified
, ZEND_ACC_PUBLIC
)
1399 PHP_ME(HttpEnvResponse
, setEtag
, ai_HttpEnvResponse_setEtag
, ZEND_ACC_PUBLIC
)
1400 PHP_ME(HttpEnvResponse
, isCachedByEtag
, ai_HttpEnvResponse_isCachedByEtag
, ZEND_ACC_PUBLIC
)
1401 PHP_ME(HttpEnvResponse
, setThrottleRate
, ai_HttpEnvResponse_setThrottleRate
, ZEND_ACC_PUBLIC
)
1402 PHP_ME(HttpEnvResponse
, send
, ai_HttpEnvResponse_send
, ZEND_ACC_PUBLIC
)
1403 EMPTY_FUNCTION_ENTRY
1406 zend_class_entry
*php_http_env_response_class_entry
;
1408 PHP_MINIT_FUNCTION(http_env_response
)
1410 zend_class_entry ce
= {0};
1412 INIT_NS_CLASS_ENTRY(ce
, "http\\Env", "Response", php_http_env_response_methods
);
1413 php_http_env_response_class_entry
= zend_register_internal_class_ex(&ce
, php_http_message_class_entry
);
1415 zend_declare_class_constant_long(php_http_env_response_class_entry
, ZEND_STRL("CONTENT_ENCODING_NONE"), PHP_HTTP_CONTENT_ENCODING_NONE
);
1416 zend_declare_class_constant_long(php_http_env_response_class_entry
, ZEND_STRL("CONTENT_ENCODING_GZIP"), PHP_HTTP_CONTENT_ENCODING_GZIP
);
1418 zend_declare_class_constant_long(php_http_env_response_class_entry
, ZEND_STRL("CACHE_NO"), PHP_HTTP_CACHE_NO
);
1419 zend_declare_class_constant_long(php_http_env_response_class_entry
, ZEND_STRL("CACHE_HIT"), PHP_HTTP_CACHE_HIT
);
1420 zend_declare_class_constant_long(php_http_env_response_class_entry
, ZEND_STRL("CACHE_MISS"), PHP_HTTP_CACHE_MISS
);
1422 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("request"), ZEND_ACC_PROTECTED
);
1423 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("cookies"), ZEND_ACC_PROTECTED
);
1424 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("contentType"), ZEND_ACC_PROTECTED
);
1425 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("contentDisposition"), ZEND_ACC_PROTECTED
);
1426 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("contentEncoding"), ZEND_ACC_PROTECTED
);
1427 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("cacheControl"), ZEND_ACC_PROTECTED
);
1428 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("etag"), ZEND_ACC_PROTECTED
);
1429 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("lastModified"), ZEND_ACC_PROTECTED
);
1430 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("throttleDelay"), ZEND_ACC_PROTECTED
);
1431 zend_declare_property_null(php_http_env_response_class_entry
, ZEND_STRL("throttleChunk"), ZEND_ACC_PROTECTED
);
1442 * vim600: noet sw=4 ts=4 fdm=marker
1443 * vim<600: noet sw=4 ts=4