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 #include <ext/standard/php_lcg.h>
17 #define BOUNDARY_OPEN(body) \
19 size_t size = php_http_message_body_size(body); \
21 php_stream_truncate_set_size(php_http_message_body_stream(body), size - lenof("--" PHP_HTTP_CRLF)); \
22 php_http_message_body_append(body, ZEND_STRL(PHP_HTTP_CRLF)); \
24 php_http_message_body_appendf(body, "--%s" PHP_HTTP_CRLF, php_http_message_body_boundary(body)); \
28 #define BOUNDARY_CLOSE(body) \
29 php_http_message_body_appendf(body, PHP_HTTP_CRLF "--%s--" PHP_HTTP_CRLF, php_http_message_body_boundary(body))
31 static STATUS
add_recursive_fields(php_http_message_body_t
*body
, const char *name
, zval
*value
);
32 static STATUS
add_recursive_files(php_http_message_body_t
*body
, const char *name
, zval
*value
);
34 php_http_message_body_t
*php_http_message_body_init(php_http_message_body_t
**body_ptr
, php_stream
*stream TSRMLS_DC
)
36 php_http_message_body_t
*body
;
38 if (body_ptr
&& *body_ptr
) {
44 body
= ecalloc(1, sizeof(php_http_message_body_t
));
48 php_stream_auto_cleanup(stream
);
49 body
->stream_id
= php_stream_get_resource_id(stream
);
50 zend_list_addref(body
->stream_id
);
52 stream
= php_stream_temp_create(TEMP_STREAM_DEFAULT
, 0xffff);
53 php_stream_auto_cleanup(stream
);
54 body
->stream_id
= php_stream_get_resource_id(stream
);
56 TSRMLS_SET_CTX(body
->ts
);
65 unsigned php_http_message_body_addref(php_http_message_body_t
*body
)
67 return ++body
->refcount
;
70 php_http_message_body_t
*php_http_message_body_copy(php_http_message_body_t
*from
, php_http_message_body_t
*to
)
73 TSRMLS_FETCH_FROM_CTX(from
->ts
);
76 php_stream_truncate_set_size(php_http_message_body_stream(to
), 0);
78 to
= php_http_message_body_init(NULL
, NULL TSRMLS_CC
);
80 php_http_message_body_to_stream(from
, php_http_message_body_stream(to
), 0, 0);
86 to
->boundary
= estrdup(from
->boundary
);
94 void php_http_message_body_free(php_http_message_body_t
**body_ptr
)
97 php_http_message_body_t
*body
= *body_ptr
;
99 if (!--body
->refcount
) {
100 TSRMLS_FETCH_FROM_CTX(body
->ts
);
101 /* NOFIXME: shows leakinfo in DEBUG mode */
102 zend_list_delete(body
->stream_id
);
103 PTR_FREE(body
->boundary
);
110 const php_stream_statbuf
*php_http_message_body_stat(php_http_message_body_t
*body
)
112 TSRMLS_FETCH_FROM_CTX(body
->ts
);
113 php_stream_stat(php_http_message_body_stream(body
), &body
->ssb
);
117 const char *php_http_message_body_boundary(php_http_message_body_t
*body
)
119 if (!body
->boundary
) {
120 union { double dbl
; int num
[2]; } data
;
121 TSRMLS_FETCH_FROM_CTX(body
->ts
);
123 data
.dbl
= php_combined_lcg(TSRMLS_C
);
124 spprintf(&body
->boundary
, 0, "%x.%x", data
.num
[0], data
.num
[1]);
126 return body
->boundary
;
129 char *php_http_message_body_etag(php_http_message_body_t
*body
)
131 php_http_etag_t
*etag
;
132 php_stream
*s
= php_http_message_body_stream(body
);
133 TSRMLS_FETCH_FROM_CTX(body
->ts
);
135 /* real file or temp buffer ? */
136 if (s
->ops
!= &php_stream_temp_ops
&& s
->ops
!= &php_stream_memory_ops
) {
137 php_stream_stat(php_http_message_body_stream(body
), &body
->ssb
);
139 if (body
->ssb
.sb
.st_mtime
) {
142 spprintf(&etag
, 0, "%lx-%lx-%lx", body
->ssb
.sb
.st_ino
, body
->ssb
.sb
.st_mtime
, body
->ssb
.sb
.st_size
);
148 if ((etag
= php_http_etag_init(PHP_HTTP_G
->env
.etag_mode TSRMLS_CC
))) {
149 php_http_message_body_to_callback(body
, (php_http_pass_callback_t
) php_http_etag_update
, etag
, 0, 0);
150 return php_http_etag_finish(etag
);
156 void php_http_message_body_to_string(php_http_message_body_t
*body
, char **buf
, size_t *len
, off_t offset
, size_t forlen
)
158 php_stream
*s
= php_http_message_body_stream(body
);
159 TSRMLS_FETCH_FROM_CTX(body
->ts
);
161 php_stream_seek(s
, offset
, SEEK_SET
);
165 *len
= php_stream_copy_to_mem(s
, buf
, forlen
, 0);
168 STATUS
php_http_message_body_to_stream(php_http_message_body_t
*body
, php_stream
*dst
, off_t offset
, size_t forlen
)
170 php_stream
*s
= php_http_message_body_stream(body
);
171 TSRMLS_FETCH_FROM_CTX(body
->ts
);
173 php_stream_seek(s
, offset
, SEEK_SET
);
178 return php_stream_copy_to_stream_ex(s
, dst
, forlen
, NULL
);
181 STATUS
php_http_message_body_to_callback(php_http_message_body_t
*body
, php_http_pass_callback_t cb
, void *cb_arg
, off_t offset
, size_t forlen
)
183 php_stream
*s
= php_http_message_body_stream(body
);
184 char *buf
= emalloc(0x1000);
185 TSRMLS_FETCH_FROM_CTX(body
->ts
);
187 php_stream_seek(s
, offset
, SEEK_SET
);
192 while (!php_stream_eof(s
)) {
193 size_t read
= php_stream_read(s
, buf
, MIN(forlen
, 0x1000));
196 if (-1 == cb(cb_arg
, buf
, read
)) {
201 if (read
< MIN(forlen
, sizeof(buf
))) {
205 if (forlen
&& !(forlen
-= read
)) {
214 size_t php_http_message_body_append(php_http_message_body_t
*body
, const char *buf
, size_t len
)
218 TSRMLS_FETCH_FROM_CTX(body
->ts
);
220 if (!(s
= php_http_message_body_stream(body
))) {
225 php_stream_seek(s
, 0, SEEK_END
);
228 written
= php_stream_write(s
, buf
, len
);
230 if (written
!= len
) {
231 php_error_docref(NULL TSRMLS_CC
, E_WARNING
, "Failed to append %zu bytes to body; wrote %zu", len
, written
);
237 size_t php_http_message_body_appendf(php_http_message_body_t
*body
, const char *fmt
, ...)
244 print_len
= vspprintf(&print_str
, 0, fmt
, argv
);
247 print_len
= php_http_message_body_append(body
, print_str
, print_len
);
253 STATUS
php_http_message_body_add_form(php_http_message_body_t
*body
, HashTable
*fields
, HashTable
*files
)
258 INIT_PZVAL_ARRAY(&tmp
, fields
);
259 if (SUCCESS
!= add_recursive_fields(body
, NULL
, &tmp
)) {
264 INIT_PZVAL_ARRAY(&tmp
, files
);
265 if (SUCCESS
!= add_recursive_files(body
, NULL
, &tmp
)) {
273 void php_http_message_body_add_part(php_http_message_body_t
*body
, php_http_message_t
*part
)
275 TSRMLS_FETCH_FROM_CTX(body
->ts
);
278 php_http_message_to_callback(part
, (php_http_pass_callback_t
) php_http_message_body_append
, body
);
279 BOUNDARY_CLOSE(body
);
283 STATUS
php_http_message_body_add_form_field(php_http_message_body_t
*body
, const char *name
, const char *value_str
, size_t value_len
)
286 TSRMLS_FETCH_FROM_CTX(body
->ts
);
288 safe_name
= php_addslashes(estrdup(name
), strlen(name
), NULL
, 1 TSRMLS_CC
);
291 php_http_message_body_appendf(
293 "Content-Disposition: form-data; name=\"%s\"" PHP_HTTP_CRLF
297 php_http_message_body_append(body
, value_str
, value_len
);
298 BOUNDARY_CLOSE(body
);
304 STATUS
php_http_message_body_add_form_file(php_http_message_body_t
*body
, const char *name
, const char *ctype
, const char *path
, php_stream
*in
)
306 char *safe_name
, *path_dup
= estrdup(path
), *bname
;
308 TSRMLS_FETCH_FROM_CTX(body
->ts
);
310 safe_name
= php_addslashes(estrdup(name
), strlen(name
), NULL
, 1 TSRMLS_CC
);
312 php_basename(path_dup
, strlen(path_dup
), NULL
, 0, &bname
, &bname_len TSRMLS_CC
);
315 php_http_message_body_appendf(
317 "Content-Disposition: form-data; name=\"%s\"; filename=\"%s\"" PHP_HTTP_CRLF
318 "Content-Transfer-Encoding: binary" PHP_HTTP_CRLF
319 "Content-Type: %s" PHP_HTTP_CRLF
321 safe_name
, bname
, ctype
323 php_stream_copy_to_stream_ex(in
, php_http_message_body_stream(body
), PHP_STREAM_COPY_ALL
, NULL
);
324 BOUNDARY_CLOSE(body
);
333 static inline char *format_key(uint type
, char *str
, ulong num
, const char *prefix
) {
334 char *new_key
= NULL
;
336 if (prefix
&& *prefix
) {
337 if (type
== HASH_KEY_IS_STRING
) {
338 spprintf(&new_key
, 0, "%s[%s]", prefix
, str
);
340 spprintf(&new_key
, 0, "%s[%lu]", prefix
, num
);
342 } else if (type
== HASH_KEY_IS_STRING
) {
343 new_key
= estrdup(str
);
345 new_key
= estrdup("");
351 static STATUS
add_recursive_fields(php_http_message_body_t
*body
, const char *name
, zval
*value
)
353 if (Z_TYPE_P(value
) == IS_ARRAY
|| Z_TYPE_P(value
) == IS_OBJECT
) {
357 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0);
358 TSRMLS_FETCH_FROM_CTX(body
->ts
);
361 if (!ht
->nApplyCount
) {
363 FOREACH_KEYVAL(pos
, value
, key
, val
) {
364 char *str
= format_key(key
.type
, key
.str
, key
.num
, name
);
365 if (SUCCESS
!= add_recursive_fields(body
, str
, *val
)) {
375 zval
*cpy
= php_http_ztyp(IS_STRING
, value
);
376 php_http_message_body_add_form_field(body
, name
, Z_STRVAL_P(cpy
), Z_STRLEN_P(cpy
));
383 static STATUS
add_recursive_files(php_http_message_body_t
*body
, const char *name
, zval
*value
)
385 zval
**zdata
= NULL
, **zfile
, **zname
, **ztype
;
387 TSRMLS_FETCH_FROM_CTX(body
->ts
);
389 if (Z_TYPE_P(value
) != IS_ARRAY
&& Z_TYPE_P(value
) != IS_OBJECT
) {
390 php_error_docref(NULL TSRMLS_CC
, E_WARNING
, "Expected array or object (name, type, file) for message body file to add");
396 if ((SUCCESS
!= zend_hash_find(ht
, ZEND_STRS("name"), (void *) &zname
))
397 || (SUCCESS
!= zend_hash_find(ht
, ZEND_STRS("type"), (void *) &ztype
))
398 || (SUCCESS
!= zend_hash_find(ht
, ZEND_STRS("file"), (void *) &zfile
))
402 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0);
404 if (!ht
->nApplyCount
) {
406 FOREACH_HASH_KEYVAL(pos
, ht
, key
, val
) {
407 if (Z_TYPE_PP(val
) == IS_ARRAY
|| Z_TYPE_PP(val
) == IS_OBJECT
) {
408 char *str
= format_key(key
.type
, key
.str
, key
.num
, name
);
410 if (SUCCESS
!= add_recursive_files(body
, str
, *val
)) {
423 zval
*zfc
= php_http_ztyp(IS_STRING
, *zfile
);
425 if (SUCCESS
== zend_hash_find(ht
, ZEND_STRS("data"), (void *) &zdata
)) {
426 if (Z_TYPE_PP(zdata
) == IS_RESOURCE
) {
427 php_stream_from_zval_no_verify(stream
, zdata
);
429 zval
*tmp
= php_http_ztyp(IS_STRING
, *zdata
);
431 stream
= php_stream_memory_open(TEMP_STREAM_READONLY
, Z_STRVAL_P(tmp
), Z_STRLEN_P(tmp
));
435 stream
= php_stream_open_wrapper(Z_STRVAL_P(zfc
), "r", REPORT_ERRORS
|USE_PATH
, NULL
);
442 zval
*znc
= php_http_ztyp(IS_STRING
, *zname
), *ztc
= php_http_ztyp(IS_STRING
, *ztype
);
443 char *key
= format_key(HASH_KEY_IS_STRING
, Z_STRVAL_P(znc
), 0, name
);
444 STATUS ret
= php_http_message_body_add_form_file(body
, key
, Z_STRVAL_P(ztc
), Z_STRVAL_P(zfc
), stream
);
450 if (!zdata
|| Z_TYPE_PP(zdata
) != IS_RESOURCE
) {
451 php_stream_close(stream
);
459 struct splitbody_arg
{
460 php_http_buffer_t buf
;
461 php_http_message_parser_t
*parser
;
467 static size_t splitbody(void *opaque
, char *buf
, size_t len TSRMLS_DC
)
469 struct splitbody_arg
*arg
= opaque
;
470 const char *boundary
= NULL
;
475 first_boundary
= !(consumed
|| arg
->consumed
);
477 if ((boundary
= php_http_locate_str(buf
, len
, arg
->boundary_str
+ first_boundary
, arg
->boundary_len
- first_boundary
))) {
478 size_t real_boundary_len
= arg
->boundary_len
- 1, cut
;
479 const char *real_boundary
= boundary
+ !first_boundary
;
482 if (buf
+ len
<= real_boundary
+ real_boundary_len
) {
483 /* if we just have enough data for the boundary, it's just a byte too less */
484 arg
->consumed
+= consumed
;
488 if (!first_boundary
) {
490 /* this is not the first boundary, read rest of this message */
491 php_http_buffer_append(&arg
->buf
, buf
, real_boundary
- buf
);
492 st
=php_http_message_parser_parse(arg
->parser
, &arg
->buf
, 0, &arg
->parser
->message
);
493 //fprintf(stderr, "1 st=%d\n",st);
496 /* move after the boundary */
497 cut
= real_boundary
- buf
+ real_boundary_len
;
502 if (buf
== php_http_locate_bin_eol(buf
, len
, &eol_len
)) {
508 if (!first_boundary
) {
509 /* advance messages */
510 php_http_message_t
*msg
;
512 msg
= php_http_message_init(NULL
, 0, NULL TSRMLS_CC
);
513 msg
->parent
= arg
->parser
->message
;
514 arg
->parser
->message
= msg
;
517 /* is this the last boundary? */
519 /* ignore the rest */
523 /* let this be garbage */
524 php_error_docref(NULL TSRMLS_CC
, E_WARNING
, "Malformed multipart boundary at pos %zu", consumed
);
529 } while (boundary
&& len
);
531 /* let there be room for the next boundary */
532 if (len
> arg
->boundary_len
) {
534 consumed
+= len
- arg
->boundary_len
;
535 php_http_buffer_append(&arg
->buf
, buf
, len
- arg
->boundary_len
);
536 st
=php_http_message_parser_parse(arg
->parser
, &arg
->buf
, 0, &arg
->parser
->message
);
537 //fprintf(stderr, "2 st=%d\n", st);
540 arg
->consumed
+= consumed
;
544 php_http_message_t
*php_http_message_body_split(php_http_message_body_t
*body
, const char *boundary
)
546 php_stream
*s
= php_http_message_body_stream(body
);
547 php_http_buffer_t
*tmp
= NULL
;
548 php_http_message_t
*msg
= NULL
;
549 struct splitbody_arg arg
;
550 TSRMLS_FETCH_FROM_CTX(body
->ts
);
552 php_http_buffer_init(&arg
.buf
);
553 arg
.parser
= php_http_message_parser_init(NULL TSRMLS_CC
);
554 arg
.boundary_len
= spprintf(&arg
.boundary_str
, 0, "\n--%s", boundary
);
557 php_stream_rewind(s
);
558 while (!php_stream_eof(s
)) {
559 php_http_buffer_passthru(&tmp
, 0x1000, (php_http_buffer_pass_func_t
) _php_stream_read
, s
, splitbody
, &arg TSRMLS_CC
);
562 msg
= arg
.parser
->message
;
563 arg
.parser
->message
= NULL
;
565 php_http_buffer_free(&tmp
);
566 php_http_message_parser_free(&arg
.parser
);
567 php_http_buffer_dtor(&arg
.buf
);
568 PTR_FREE(arg
.boundary_str
);
573 static zend_object_handlers php_http_message_body_object_handlers
;
575 zend_object_value
php_http_message_body_object_new(zend_class_entry
*ce TSRMLS_DC
)
577 return php_http_message_body_object_new_ex(ce
, NULL
, NULL TSRMLS_CC
);
580 zend_object_value
php_http_message_body_object_new_ex(zend_class_entry
*ce
, php_http_message_body_t
*body
, php_http_message_body_object_t
**ptr TSRMLS_DC
)
582 php_http_message_body_object_t
*o
;
584 o
= ecalloc(1, sizeof(php_http_message_body_object_t
));
585 zend_object_std_init((zend_object
*) o
, php_http_message_body_class_entry TSRMLS_CC
);
586 object_properties_init((zend_object
*) o
, ce
);
596 o
->zv
.handle
= zend_objects_store_put((zend_object
*) o
, NULL
, php_http_message_body_object_free
, NULL TSRMLS_CC
);
597 o
->zv
.handlers
= &php_http_message_body_object_handlers
;
602 zend_object_value
php_http_message_body_object_clone(zval
*object TSRMLS_DC
)
604 zend_object_value new_ov
;
605 php_http_message_body_object_t
*new_obj
= NULL
;
606 php_http_message_body_object_t
*old_obj
= zend_object_store_get_object(object TSRMLS_CC
);
607 php_http_message_body_t
*body
= php_http_message_body_copy(old_obj
->body
, NULL
);
609 new_ov
= php_http_message_body_object_new_ex(old_obj
->zo
.ce
, body
, &new_obj TSRMLS_CC
);
610 zend_objects_clone_members(&new_obj
->zo
, new_ov
, &old_obj
->zo
, Z_OBJ_HANDLE_P(object
) TSRMLS_CC
);
615 void php_http_message_body_object_free(void *object TSRMLS_DC
)
617 php_http_message_body_object_t
*obj
= object
;
619 php_http_message_body_free(&obj
->body
);
620 zend_object_std_dtor((zend_object
*) obj TSRMLS_CC
);
624 #define PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj) \
627 obj->body = php_http_message_body_init(NULL, NULL TSRMLS_CC); \
631 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody___construct
, 0, 0, 0)
632 ZEND_ARG_INFO(0, stream
)
634 PHP_METHOD(HttpMessageBody
, __construct
)
636 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
637 zval
*zstream
= NULL
;
640 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|r!", &zstream
), invalid_arg
, return);
643 php_http_expect(php_stream_from_zval_no_verify(stream
, &zstream
), unexpected_val
, return);
646 php_http_message_body_free(&obj
->body
);
648 obj
->body
= php_http_message_body_init(NULL
, stream TSRMLS_CC
);
652 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody___toString
, 0, 0, 0)
654 PHP_METHOD(HttpMessageBody
, __toString
)
656 if (SUCCESS
== zend_parse_parameters_none()) {
657 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
661 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
663 php_http_message_body_to_string(obj
->body
, &str
, &len
, 0, 0);
665 RETURN_STRINGL(str
, len
, 0);
668 RETURN_EMPTY_STRING();
671 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_unserialize
, 0, 0, 1)
672 ZEND_ARG_INFO(0, serialized
)
674 PHP_METHOD(HttpMessageBody
, unserialize
)
679 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "s", &us_str
, &us_len
)) {
680 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
681 php_stream
*s
= php_stream_memory_open(0, us_str
, us_len
);
683 obj
->body
= php_http_message_body_init(NULL
, s TSRMLS_CC
);
687 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_toStream
, 0, 0, 1)
688 ZEND_ARG_INFO(0, stream
)
689 ZEND_ARG_INFO(0, offset
)
690 ZEND_ARG_INFO(0, maxlen
)
692 PHP_METHOD(HttpMessageBody
, toStream
)
695 long offset
= 0, forlen
= 0;
697 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "r|ll", &zstream
, &offset
, &forlen
)) {
699 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
701 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
703 php_stream_from_zval(stream
, &zstream
);
704 php_http_message_body_to_stream(obj
->body
, stream
, offset
, forlen
);
705 RETURN_ZVAL(getThis(), 1, 0);
709 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_toCallback
, 0, 0, 1)
710 ZEND_ARG_INFO(0, callback
)
711 ZEND_ARG_INFO(0, offset
)
712 ZEND_ARG_INFO(0, maxlen
)
714 PHP_METHOD(HttpMessageBody
, toCallback
)
716 php_http_pass_fcall_arg_t fcd
;
717 long offset
= 0, forlen
= 0;
719 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "f|ll", &fcd
.fci
, &fcd
.fcc
, &offset
, &forlen
)) {
720 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
722 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
726 TSRMLS_SET_CTX(fcd
.ts
);
728 php_http_message_body_to_callback(obj
->body
, php_http_pass_fcall_callback
, &fcd
, offset
, forlen
);
729 zend_fcall_info_args_clear(&fcd
.fci
, 1);
731 zval_ptr_dtor(&fcd
.fcz
);
732 RETURN_ZVAL(getThis(), 1, 0);
736 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_getResource
, 0, 0, 0)
738 PHP_METHOD(HttpMessageBody
, getResource
)
740 if (SUCCESS
== zend_parse_parameters_none()) {
741 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
743 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
745 zend_list_addref(obj
->body
->stream_id
);
746 RETVAL_RESOURCE(obj
->body
->stream_id
);
750 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_getBoundary
, 0, 0, 0)
752 PHP_METHOD(HttpMessageBody
, getBoundary
)
754 if (SUCCESS
== zend_parse_parameters_none()) {
755 php_http_message_body_object_t
* obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
757 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
759 if (obj
->body
->boundary
) {
760 RETURN_STRING(obj
->body
->boundary
, 1);
765 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_append
, 0, 0, 1)
766 ZEND_ARG_INFO(0, string
)
768 PHP_METHOD(HttpMessageBody
, append
)
772 php_http_message_body_object_t
*obj
;
774 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "s", &str
, &len
), invalid_arg
, return);
776 obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
778 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
780 php_http_expect(len
== php_http_message_body_append(obj
->body
, str
, len
), runtime
, return);
782 RETURN_ZVAL(getThis(), 1, 0);
785 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_addForm
, 0, 0, 0)
786 ZEND_ARG_ARRAY_INFO(0, fields
, 1)
787 ZEND_ARG_ARRAY_INFO(0, files
, 1)
789 PHP_METHOD(HttpMessageBody
, addForm
)
791 HashTable
*fields
= NULL
, *files
= NULL
;
792 php_http_message_body_object_t
*obj
;
794 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|h!h!", &fields
, &files
), invalid_arg
, return);
796 obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
798 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
800 php_http_expect(SUCCESS
== php_http_message_body_add_form(obj
->body
, fields
, files
), runtime
, return);
802 RETURN_ZVAL(getThis(), 1, 0);
805 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_addPart
, 0, 0, 1)
806 ZEND_ARG_OBJ_INFO(0, message
, http
\\Message
, 0)
808 PHP_METHOD(HttpMessageBody
, addPart
)
811 php_http_message_body_object_t
*obj
;
812 php_http_message_object_t
*mobj
;
813 zend_error_handling zeh
;
815 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "O", &zobj
, php_http_message_class_entry
), invalid_arg
, return);
817 obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
818 mobj
= zend_object_store_get_object(zobj TSRMLS_CC
);
820 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
822 zend_replace_error_handling(EH_THROW
, php_http_exception_runtime_class_entry
, &zeh TSRMLS_CC
);
823 php_http_message_body_add_part(obj
->body
, mobj
->message
);
824 zend_restore_error_handling(&zeh TSRMLS_CC
);
826 if (!EG(exception
)) {
827 RETURN_ZVAL(getThis(), 1, 0);
831 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_etag
, 0, 0, 0)
833 PHP_METHOD(HttpMessageBody
, etag
)
835 if (SUCCESS
== zend_parse_parameters_none()) {
836 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
839 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
841 if ((etag
= php_http_message_body_etag(obj
->body
))) {
842 RETURN_STRING(etag
, 0);
849 ZEND_BEGIN_ARG_INFO_EX(ai_HttpMessageBody_stat
, 0, 0, 0)
850 ZEND_ARG_INFO(0, field
)
852 PHP_METHOD(HttpMessageBody
, stat
)
854 char *field_str
= NULL
;
857 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|s", &field_str
, &field_len
)) {
858 php_http_message_body_object_t
*obj
= zend_object_store_get_object(getThis() TSRMLS_CC
);
859 const php_stream_statbuf
*sb
;
861 PHP_HTTP_MESSAGE_BODY_OBJECT_INIT(obj
);
863 if ((sb
= php_http_message_body_stat(obj
->body
))) {
864 if (field_str
&& field_len
) {
865 switch (*field_str
) {
868 RETURN_LONG(sb
->sb
.st_size
);
872 RETURN_LONG(sb
->sb
.st_atime
);
876 RETURN_LONG(sb
->sb
.st_mtime
);
880 RETURN_LONG(sb
->sb
.st_ctime
);
883 php_error_docref(NULL TSRMLS_CC
, E_WARNING
, "Unknown stat field: '%s' (should be one of [s]ize, [a]time, [m]time or [c]time)", field_str
);
887 object_init(return_value
);
888 add_property_long_ex(return_value
, ZEND_STRS("size"), sb
->sb
.st_size TSRMLS_CC
);
889 add_property_long_ex(return_value
, ZEND_STRS("atime"), sb
->sb
.st_atime TSRMLS_CC
);
890 add_property_long_ex(return_value
, ZEND_STRS("mtime"), sb
->sb
.st_mtime TSRMLS_CC
);
891 add_property_long_ex(return_value
, ZEND_STRS("ctime"), sb
->sb
.st_ctime TSRMLS_CC
);
897 static zend_function_entry php_http_message_body_methods
[] = {
898 PHP_ME(HttpMessageBody
, __construct
, ai_HttpMessageBody___construct
, ZEND_ACC_PUBLIC
|ZEND_ACC_CTOR
)
899 PHP_ME(HttpMessageBody
, __toString
, ai_HttpMessageBody___toString
, ZEND_ACC_PUBLIC
)
900 PHP_MALIAS(HttpMessageBody
, toString
, __toString
, ai_HttpMessageBody___toString
, ZEND_ACC_PUBLIC
)
901 PHP_MALIAS(HttpMessageBody
, serialize
, __toString
, ai_HttpMessageBody___toString
, ZEND_ACC_PUBLIC
)
902 PHP_ME(HttpMessageBody
, unserialize
, ai_HttpMessageBody_unserialize
, ZEND_ACC_PUBLIC
)
903 PHP_ME(HttpMessageBody
, toStream
, ai_HttpMessageBody_toStream
, ZEND_ACC_PUBLIC
)
904 PHP_ME(HttpMessageBody
, toCallback
, ai_HttpMessageBody_toCallback
, ZEND_ACC_PUBLIC
)
905 PHP_ME(HttpMessageBody
, getResource
, ai_HttpMessageBody_getResource
, ZEND_ACC_PUBLIC
)
906 PHP_ME(HttpMessageBody
, getBoundary
, ai_HttpMessageBody_getBoundary
, ZEND_ACC_PUBLIC
)
907 PHP_ME(HttpMessageBody
, append
, ai_HttpMessageBody_append
, ZEND_ACC_PUBLIC
)
908 PHP_ME(HttpMessageBody
, addForm
, ai_HttpMessageBody_addForm
, ZEND_ACC_PUBLIC
)
909 PHP_ME(HttpMessageBody
, addPart
, ai_HttpMessageBody_addPart
, ZEND_ACC_PUBLIC
)
910 PHP_ME(HttpMessageBody
, etag
, ai_HttpMessageBody_etag
, ZEND_ACC_PUBLIC
)
911 PHP_ME(HttpMessageBody
, stat
, ai_HttpMessageBody_stat
, ZEND_ACC_PUBLIC
)
915 zend_class_entry
*php_http_message_body_class_entry
;
917 PHP_MINIT_FUNCTION(http_message_body
)
919 zend_class_entry ce
= {0};
921 INIT_NS_CLASS_ENTRY(ce
, "http\\Message", "Body", php_http_message_body_methods
);
922 php_http_message_body_class_entry
= zend_register_internal_class(&ce TSRMLS_CC
);
923 php_http_message_body_class_entry
->create_object
= php_http_message_body_object_new
;
924 memcpy(&php_http_message_body_object_handlers
, zend_get_std_object_handlers(), sizeof(zend_object_handlers
));
925 php_http_message_body_object_handlers
.clone_obj
= php_http_message_body_object_clone
;
926 zend_class_implements(php_http_message_body_class_entry TSRMLS_CC
, 1, zend_ce_serializable
);
936 * vim600: noet sw=4 ts=4 fdm=marker
937 * vim<600: noet sw=4 ts=4