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-2006, Michael Wallner <mike@php.net> |
10 +--------------------------------------------------------------------+
19 #include "zend_interfaces.h"
21 #include "php_http_api.h"
22 #include "php_http_url_api.h"
23 #include "php_http_querystring_object.h"
24 #include "php_http_exception_object.h"
26 #define HTTP_BEGIN_ARGS(method, ret_ref, req_args) HTTP_BEGIN_ARGS_EX(HttpQueryString, method, ret_ref, req_args)
27 #define HTTP_EMPTY_ARGS(method, ret_ref) HTTP_EMPTY_ARGS_EX(HttpQueryString, method, ret_ref)
28 #define HTTP_QUERYSTRING_ME(method, visibility) PHP_ME(HttpQueryString, method, HTTP_ARGS(HttpQueryString, method), visibility)
30 HTTP_BEGIN_ARGS(__construct
, 0, 0)
31 HTTP_ARG_VAL(global
, 0)
34 HTTP_BEGIN_ARGS(getInstance
, 0, 0)
35 HTTP_ARG_VAL(global
, 0)
38 HTTP_EMPTY_ARGS(__toString
, 0);
40 HTTP_BEGIN_ARGS(get
, 0, 0)
43 HTTP_ARG_VAL(defval
, 0)
46 HTTP_BEGIN_ARGS(set
, 0, 2)
48 HTTP_ARG_VAL(value
, 0)
51 HTTP_BEGIN_ARGS(del
, 0, 1)
52 HTTP_ARG_VAL(params
, 0)
55 HTTP_BEGIN_ARGS(mod
, 0, 1)
56 HTTP_ARG_VAL(params
, 0)
59 #define http_querystring_object_declare_default_properties() _http_querystring_object_declare_default_properties(TSRMLS_C)
60 static inline void _http_querystring_object_declare_default_properties(TSRMLS_D
);
62 #define GET_STATIC_PROP(n) *GET_STATIC_PROP_EX(http_querystring_object_ce, n)
63 #define SET_STATIC_PROP(n, v) SET_STATIC_PROP_EX(http_querystring_object_ce, n, v)
64 #define OBJ_PROP_CE http_querystring_object_ce
65 zend_class_entry
*http_querystring_object_ce
;
66 zend_function_entry http_querystring_object_fe
[] = {
67 HTTP_QUERYSTRING_ME(__construct
, ZEND_ACC_PUBLIC
|ZEND_ACC_CTOR
|ZEND_ACC_FINAL
)
68 HTTP_QUERYSTRING_ME(__toString
, ZEND_ACC_PUBLIC
)
69 HTTP_QUERYSTRING_ME(get
, ZEND_ACC_PUBLIC
)
70 HTTP_QUERYSTRING_ME(set
, ZEND_ACC_PUBLIC
)
71 HTTP_QUERYSTRING_ME(del
, ZEND_ACC_PUBLIC
)
72 HTTP_QUERYSTRING_ME(mod
, ZEND_ACC_PUBLIC
)
73 HTTP_QUERYSTRING_ME(getInstance
, ZEND_ACC_PUBLIC
|ZEND_ACC_STATIC
)
77 static zend_object_handlers http_querystring_object_handlers
;
79 PHP_MINIT_FUNCTION(http_querystring_object
)
81 HTTP_REGISTER_CLASS_EX(HttpQueryString
, http_querystring_object
, NULL
, 0);
83 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_BOOL", HTTP_QUERYSTRING_TYPE_BOOL
);
84 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_INT", HTTP_QUERYSTRING_TYPE_INT
);
85 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_FLOAT", HTTP_QUERYSTRING_TYPE_FLOAT
);
86 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_STRING", HTTP_QUERYSTRING_TYPE_STRING
);
87 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_ARRAY", HTTP_QUERYSTRING_TYPE_ARRAY
);
88 HTTP_LONG_CONSTANT("HTTP_QUERYSTRING_TYPE_OBJECT", HTTP_QUERYSTRING_TYPE_OBJECT
);
93 zend_object_value
_http_querystring_object_new(zend_class_entry
*ce TSRMLS_DC
)
95 return http_querystring_object_new_ex(ce
, NULL
);
98 zend_object_value
_http_querystring_object_new_ex(zend_class_entry
*ce
, http_querystring_object
**ptr TSRMLS_DC
)
100 zend_object_value ov
;
101 http_querystring_object
*o
;
103 o
= ecalloc(1, sizeof(http_querystring_object
));
110 ALLOC_HASHTABLE(OBJ_PROP(o
));
111 zend_hash_init(OBJ_PROP(o
), 0, NULL
, ZVAL_PTR_DTOR
, 0);
112 zend_hash_copy(OBJ_PROP(o
), &ce
->default_properties
, (copy_ctor_func_t
) zval_add_ref
, NULL
, sizeof(zval
*));
114 ov
.handle
= putObject(http_querystring_object
, o
);
115 ov
.handlers
= &http_querystring_object_handlers
;
120 static inline void _http_querystring_object_declare_default_properties(TSRMLS_D
)
122 zend_class_entry
*ce
= http_querystring_object_ce
;
124 DCL_STATIC_PROP_N(PRIVATE
, instance
);
126 DCL_PROP_N(PRIVATE
, queryArray
);
127 DCL_PROP(PRIVATE
, string
, queryString
, "");
130 DCL_CONST(long, "TYPE_BOOL", HTTP_QUERYSTRING_TYPE_BOOL
);
131 DCL_CONST(long, "TYPE_INT", HTTP_QUERYSTRING_TYPE_INT
);
132 DCL_CONST(long, "TYPE_FLOAT", HTTP_QUERYSTRING_TYPE_FLOAT
);
133 DCL_CONST(long, "TYPE_STRING", HTTP_QUERYSTRING_TYPE_STRING
);
134 DCL_CONST(long, "TYPE_ARRAY", HTTP_QUERYSTRING_TYPE_ARRAY
);
135 DCL_CONST(long, "TYPE_OBJECT", HTTP_QUERYSTRING_TYPE_OBJECT
);
139 void _http_querystring_object_free(zend_object
*object TSRMLS_DC
)
141 http_querystring_object
*o
= (http_querystring_object
*) object
;
144 zend_hash_destroy(OBJ_PROP(o
));
145 FREE_HASHTABLE(OBJ_PROP(o
));
150 #define http_querystring_update(qa, qs) _http_querystring_update((qa), (qs) TSRMLS_CC)
151 static inline void _http_querystring_update(zval
*qarray
, zval
*qstring TSRMLS_DC
)
156 if (Z_TYPE_P(qarray
) != IS_ARRAY
) {
157 convert_to_array(qarray
);
159 if (SUCCESS
== http_urlencode_hash_ex(Z_ARRVAL_P(qarray
), 0, NULL
, 0, &s
, &l
)) {
161 ZVAL_STRINGL(qstring
, s
, l
, 0);
163 http_error(HE_WARNING
, HTTP_E_QUERYSTRING
, "Failed to update query string");
167 #define http_querystring_modify_array(a, k, l, v) _http_querystring_modify_array((a), (k), (l), (v) TSRMLS_CC)
168 static inline int _http_querystring_modify_array(zval
*qarray
, char *key
, uint keylen
, zval
*data TSRMLS_DC
)
170 if (Z_TYPE_P(data
) == IS_NULL
) {
171 if (SUCCESS
!= zend_hash_del(Z_ARRVAL_P(qarray
), key
, keylen
+ 1)) {
176 add_assoc_zval(qarray
, key
, data
);
181 #define http_querystring_instantiate(g) _http_querystring_instantiate((g) TSRMLS_CC)
182 static inline zval
*_http_querystring_instantiate(zend_bool global TSRMLS_DC
)
184 zval
*zobj
, *zglobal
;
186 MAKE_STD_ZVAL(zglobal
);
187 ZVAL_BOOL(zglobal
, global
);
190 Z_TYPE_P(zobj
) = IS_OBJECT
;
191 Z_OBJVAL_P(zobj
) = http_querystring_object_new(http_querystring_object_ce
);
192 zend_call_method_with_1_params(&zobj
, Z_OBJCE_P(zobj
), NULL
, "__construct", NULL
, zglobal
);
194 zval_ptr_dtor(&zglobal
);
199 /* {{{ proto void HttpQueryString::__construct([bool global = true])
201 * Creates a new HttpQueryString object instance.
202 * Operates on and modifies $_GET and $_SERVER['QUERY_STRING'] if global is TRUE.
204 PHP_METHOD(HttpQueryString
, __construct
)
206 zend_bool global
= 1;
207 zval
*qarray
= NULL
, *qstring
= NULL
, **_GET
, **_SERVER
, **QUERY_STRING
;
210 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|b", &global
)) {
212 if ( (SUCCESS
== zend_hash_find(&EG(symbol_table
), "_SERVER", sizeof("_SERVER"), (void **) &_SERVER
)) &&
213 (Z_TYPE_PP(_SERVER
) == IS_ARRAY
) &&
214 (SUCCESS
== zend_hash_find(Z_ARRVAL_PP(_SERVER
), "QUERY_STRING", sizeof("QUERY_STRING"), (void **) &QUERY_STRING
))) {
216 qstring
= *QUERY_STRING
;
218 if ((SUCCESS
== zend_hash_find(&EG(symbol_table
), "_GET", sizeof("_GET"), (void **) &_GET
)) && (Z_TYPE_PP(_GET
) == IS_ARRAY
)) {
221 http_error(HE_WARNING
, HTTP_E_QUERYSTRING
, "Could not acquire reference to superglobal GET array");
224 http_error(HE_WARNING
, HTTP_E_QUERYSTRING
, "Could not acquire reference to QUERY_STRING");
227 if (qarray
&& qstring
) {
228 if (Z_TYPE_P(qstring
) != IS_STRING
) {
229 convert_to_string(qstring
);
232 SET_PROP(queryArray
, qarray
);
233 SET_PROP(queryString
, qstring
);
234 GET_PROP(queryArray
)->is_ref
= 1;
235 GET_PROP(queryString
)->is_ref
= 1;
238 qarray
= ecalloc(1, sizeof(zval
));
240 SET_PROP(queryArray
, qarray
);
241 UPD_STRL(queryString
, "", 0);
248 /* {{{ proto string HttpQueryString::__toString()
250 * Returns the string representation.
252 PHP_METHOD(HttpQueryString
, __toString
)
255 RETURN_PROP(queryString
);
259 /* {{{ proto mixed HttpQueryString::get([string key[, mixed type = 0[, mixed defval = NULL[, bool delete = false]]]])
261 * Get (part of) the query string.
263 * The type parameter is either one of the HttpQueryString::TYPE_* constants or a type abbreviation like
264 * "b" for bool, "i" for int, "f" for float, "s" for string, "a" for array and "o" for a stdClass object.
266 PHP_METHOD(HttpQueryString
, get
)
272 zval
*ztype
= NULL
, *defval
= NULL
;
274 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|szzb", &name
, &name_len
, &ztype
, &defval
, &del
)) {
275 if (name
&& name_len
) {
276 zval
**arrval
, *qarray
= GET_PROP(queryArray
);
278 if (SUCCESS
== zend_hash_find(Z_ARRVAL_P(qarray
), name
, name_len
+ 1, (void **) &arrval
)) {
279 RETVAL_ZVAL(*arrval
, 1, 0);
282 if (Z_TYPE_P(ztype
) == IS_LONG
) {
283 type
= Z_LVAL_P(ztype
);
284 } else if(Z_TYPE_P(ztype
) == IS_STRING
) {
285 switch (tolower(Z_STRVAL_P(ztype
)[0]))
288 type
= HTTP_QUERYSTRING_TYPE_BOOL
;
291 type
= HTTP_QUERYSTRING_TYPE_INT
;
294 type
= HTTP_QUERYSTRING_TYPE_FLOAT
;
297 type
= HTTP_QUERYSTRING_TYPE_STRING
;
300 type
= HTTP_QUERYSTRING_TYPE_ARRAY
;
303 type
= HTTP_QUERYSTRING_TYPE_OBJECT
;
308 convert_to_type(type
, return_value
);
312 if (del
&& (SUCCESS
== zend_hash_del(Z_ARRVAL_P(qarray
), name
, name_len
+ 1))) {
313 http_querystring_update(qarray
, GET_PROP(queryString
));
316 RETURN_ZVAL(defval
, 1, 0);
319 RETURN_PROP(queryString
);
325 /* {{{ proto string HttpQueryString::set(string name, mixed value)
327 * Set a query string entry.
329 PHP_METHOD(HttpQueryString
, set
)
335 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "sz", &name
, &name_len
, &value
)) {
336 zval
*qarray
= GET_PROP(queryArray
);
338 if (http_querystring_modify_array(qarray
, name
, name_len
, value
)) {
339 http_querystring_update(qarray
, GET_PROP(queryString
));
344 RETURN_PROP(queryString
);
349 /* {{{ proto string HttpQueryString::del(mixed param)
351 * Deletes entry/entries from the query string.
353 PHP_METHOD(HttpQueryString
, del
)
357 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "z", ¶ms
)) {
358 zval
*qarray
= GET_PROP(queryArray
);
360 if (Z_TYPE_P(params
) == IS_ARRAY
) {
364 FOREACH_VAL(pos
, params
, name
) {
366 convert_to_string_ex(name
);
367 zend_hash_del(Z_ARRVAL_P(qarray
), Z_STRVAL_PP(name
), Z_STRLEN_PP(name
) + 1);
371 http_querystring_update(qarray
, GET_PROP(queryString
));
374 convert_to_string_ex(¶ms
);
375 if (SUCCESS
== zend_hash_del(Z_ARRVAL_P(qarray
), Z_STRVAL_P(params
), Z_STRLEN_P(params
) + 1)) {
376 http_querystring_update(qarray
, GET_PROP(queryString
));
378 zval_ptr_dtor(¶ms
);
382 RETURN_PROP(queryString
);
387 /* {{{ proto string HttpQueryString::mod(array params)
389 * Modifies the query string according to params. NULL values will unset the variable.
391 PHP_METHOD(HttpQueryString
, mod
)
395 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "a|b", ¶ms
)) {
396 zval
**value
, *qarray
= GET_PROP(queryArray
);
402 FOREACH_KEYLENVAL(pos
, params
, key
, keylen
, idx
, value
) {
404 http_querystring_modify_array(qarray
, key
, keylen
, *value
);
406 keylen
= spprintf(&key
, 0, "%lu", idx
);
407 http_querystring_modify_array(qarray
, key
, keylen
, *value
);
413 http_querystring_update(qarray
, GET_PROP(queryString
));
417 RETURN_PROP(queryString
);
422 /* {{{ proto HttpQueryString HttpQueryString::getInstance([bool global = true])
424 * Get a single instance (differentiates between the global setting).
426 PHP_METHOD(HttpQueryString
, getInstance
)
428 zend_bool global
= 1;
429 zval
*instance
= GET_STATIC_PROP(instance
);
432 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|b", &global
)) {
433 zval
**zobj_ptr
= NULL
, *zobj
= NULL
;
435 if (Z_TYPE_P(instance
) == IS_ARRAY
) {
436 if (SUCCESS
== zend_hash_index_find(Z_ARRVAL_P(instance
), global
, (void **) &zobj_ptr
)) {
437 RETVAL_ZVAL(*zobj_ptr
, 1, 0);
439 zobj
= http_querystring_instantiate(global
);
440 add_index_zval(instance
, global
, zobj
);
441 RETVAL_OBJECT(zobj
, 1);
444 MAKE_STD_ZVAL(instance
);
445 array_init(instance
);
447 zobj
= http_querystring_instantiate(global
);
448 add_index_zval(instance
, global
, zobj
);
449 RETVAL_OBJECT(zobj
, 1);
451 SET_STATIC_PROP(instance
, instance
);
452 zval_ptr_dtor(&instance
);
459 #endif /* ZEND_ENGINE_2 */
466 * vim600: noet sw=4 ts=4 fdm=marker
467 * vim<600: noet sw=4 ts=4