| modification, are permitted provided that the conditions mentioned |
| in the accompanying LICENSE file are met. |
+--------------------------------------------------------------------+
- | Copyright (c) 2004-2005, Michael Wallner <mike@php.net> |
+ | Copyright (c) 2004-2006, Michael Wallner <mike@php.net> |
+--------------------------------------------------------------------+
*/
/* $Id$ */
-#ifdef HAVE_CONFIG_H
-# include "config.h"
-#endif
-
#define HTTP_WANT_CURL
#include "php_http.h"
# include "php_http_request_object.h"
#endif
-ZEND_EXTERN_MODULE_GLOBALS(http);
-
/* {{{ char *http_request_methods[] */
static const char *const http_request_methods[] = {
"UNKNOWN",
};
/* }}} */
+/* {{{ */
PHP_MINIT_FUNCTION(http_request_method)
{
/* HTTP/1.1 */
return SUCCESS;
}
+PHP_RINIT_FUNCTION(http_request_method)
+{
+ HTTP_G->request.methods.custom.entries = ecalloc(1, sizeof(http_request_method_entry *));
+
+ if (HTTP_G->request.methods.custom.ini && *HTTP_G->request.methods.custom.ini) {
+ HashPosition pos;
+ HashTable methods;
+ zval **data;
+
+ zend_hash_init(&methods, 0, NULL, ZVAL_PTR_DTOR, 0);
+ http_parse_params(HTTP_G->request.methods.custom.ini, HTTP_PARAMS_DEFAULT, &methods);
+ FOREACH_HASH_VAL(pos, &methods, data) {
+ if (Z_TYPE_PP(data) == IS_STRING) {
+ http_request_method_register(Z_STRVAL_PP(data), Z_STRLEN_PP(data));
+ }
+ }
+ zend_hash_destroy(&methods);
+ }
+ return SUCCESS;
+}
+
PHP_RSHUTDOWN_FUNCTION(http_request_method)
{
- int i, c = zend_hash_num_elements(&HTTP_G(request).methods.custom);
+ int i;
+ http_request_method_entry **ptr = HTTP_G->request.methods.custom.entries;
- for (i = 0; i < c; ++i) {
- http_request_method_unregister(HTTP_MAX_REQUEST_METHOD + i);
+ for (i = 0; i < HTTP_G->request.methods.custom.count; ++i) {
+ if (ptr[i]) {
+ http_request_method_unregister(HTTP_CUSTOM_REQUEST_METHOD_START + i);
+ }
}
+ efree(HTTP_G->request.methods.custom.entries);
return SUCCESS;
}
+/* }}} */
/* {{{ char *http_request_method_name(http_request_method) */
PHP_HTTP_API const char *_http_request_method_name(http_request_method m TSRMLS_DC)
{
- zval **meth;
+ http_request_method_entry **ptr = HTTP_G->request.methods.custom.entries;
if (HTTP_STD_REQUEST_METHOD(m)) {
return http_request_methods[m];
}
- if (SUCCESS == zend_hash_index_find(&HTTP_G(request).methods.custom, HTTP_CUSTOM_REQUEST_METHOD(m), (void **) &meth)) {
- return Z_STRVAL_PP(meth);
+ if ( (HTTP_CUSTOM_REQUEST_METHOD(m) >= 0) &&
+ (HTTP_CUSTOM_REQUEST_METHOD(m) < HTTP_G->request.methods.custom.count) &&
+ (ptr[HTTP_CUSTOM_REQUEST_METHOD(m)])) {
+ return ptr[HTTP_CUSTOM_REQUEST_METHOD(m)]->name;
}
return http_request_methods[0];
}
/* }}} */
-/* {{{ ulong http_request_method_exists(zend_bool, ulong, char *) */
-PHP_HTTP_API ulong _http_request_method_exists(zend_bool by_name, ulong id, const char *name TSRMLS_DC)
+/* {{{ int http_request_method_exists(zend_bool, ulong, char *) */
+PHP_HTTP_API int _http_request_method_exists(zend_bool by_name, http_request_method id, const char *name TSRMLS_DC)
{
+ int i;
+ http_request_method_entry **ptr = HTTP_G->request.methods.custom.entries;
+
if (by_name) {
- unsigned i;
-
- for (i = HTTP_NO_REQUEST_METHOD + 1; i < HTTP_MAX_REQUEST_METHOD; ++i) {
- if (!strcmp(name, http_request_methods[i])) {
+ for (i = HTTP_MIN_REQUEST_METHOD; i < HTTP_MAX_REQUEST_METHOD; ++i) {
+ if (!strcasecmp(name, http_request_methods[i])) {
return i;
}
}
- {
- zval **data;
- char *key;
- ulong idx;
- HashPosition pos;
-
- FOREACH_HASH_KEYVAL(pos, &HTTP_G(request).methods.custom, key, idx, data) {
- if (!strcmp(name, Z_STRVAL_PP(data))) {
- return idx + HTTP_MAX_REQUEST_METHOD;
- }
+ for (i = 0; i < HTTP_G->request.methods.custom.count; ++i) {
+ if (ptr[i] && !strcasecmp(name, ptr[i]->name)) {
+ return HTTP_CUSTOM_REQUEST_METHOD_START + i;
}
}
- return 0;
- } else {
- return HTTP_STD_REQUEST_METHOD(id) || zend_hash_index_exists(&HTTP_G(request).methods.custom, HTTP_CUSTOM_REQUEST_METHOD(id)) ? id : 0;
+ } else if (HTTP_STD_REQUEST_METHOD(id)) {
+ return id;
+ } else if ( (HTTP_CUSTOM_REQUEST_METHOD(id) >= 0) &&
+ (HTTP_CUSTOM_REQUEST_METHOD(id) < HTTP_G->request.methods.custom.count) &&
+ (ptr[HTTP_CUSTOM_REQUEST_METHOD(id)])) {
+ return id;
}
+
+ return 0;
}
/* }}} */
-/* {{{ ulong http_request_method_register(char *) */
-PHP_HTTP_API ulong _http_request_method_register(const char *method_name, size_t method_name_len TSRMLS_DC)
+/* {{{ int http_request_method_register(char *) */
+PHP_HTTP_API int _http_request_method_register(const char *method_name, int method_name_len TSRMLS_DC)
{
- zval array;
- char *http_method, *method;
- ulong i, meth_num = HTTP_G(request).methods.custom.nNextFreeElement + HTTP_MAX_REQUEST_METHOD;
-
+ int i, meth_num;
+ char *http_method, *method, *mconst;
+ http_request_method_entry **ptr = HTTP_G->request.methods.custom.entries;
+
+ if (!isalpha(*method_name)) {
+ http_error_ex(HE_WARNING, HTTP_E_REQUEST_METHOD, "Request method does not start with a character (%s)", method_name);
+ return 0;
+ }
+
+ if (http_request_method_exists(1, 0, method_name)) {
+ http_error_ex(HE_WARNING, HTTP_E_REQUEST_METHOD, "Request method does already exist (%s)", method_name);
+ return 0;
+ }
+
method = emalloc(method_name_len + 1);
+ mconst = emalloc(method_name_len + 1);
for (i = 0; i < method_name_len; ++i) {
- method[i] = toupper(method_name[i]);
+ switch (method_name[i]) {
+ case '-':
+ method[i] = '-';
+ mconst[i] = '_';
+ break;
+
+ default:
+ if (!isalnum(method_name[i])) {
+ efree(method);
+ efree(mconst);
+ http_error_ex(HE_WARNING, HTTP_E_REQUEST_METHOD, "Request method contains illegal characters (%s)", method_name);
+ return 0;
+ }
+ mconst[i] = method[i] = toupper(method_name[i]);
+ break;
+ }
}
method[method_name_len] = '\0';
+ mconst[method_name_len] = '\0';
- INIT_ZARR(array, &HTTP_G(request).methods.custom);
- add_next_index_stringl(&array, method, method_name_len, 0);
+ ptr = erealloc(ptr, sizeof(http_request_method_entry *) * (HTTP_G->request.methods.custom.count + 1));
+ HTTP_G->request.methods.custom.entries = ptr;
+ ptr[HTTP_G->request.methods.custom.count] = emalloc(sizeof(http_request_method_entry));
+ ptr[HTTP_G->request.methods.custom.count]->name = method;
+ ptr[HTTP_G->request.methods.custom.count]->cnst = mconst;
+ meth_num = HTTP_CUSTOM_REQUEST_METHOD_START + HTTP_G->request.methods.custom.count++;
- method_name_len = spprintf(&http_method, 0, "HTTP_METH_%s", method);
+ method_name_len = spprintf(&http_method, 0, "HTTP_METH_%s", mconst);
zend_register_long_constant(http_method, method_name_len + 1, meth_num, CONST_CS, http_module_number TSRMLS_CC);
efree(http_method);
#if defined(ZEND_ENGINE_2) && defined(HTTP_HAVE_CURL) && !defined(WONKY)
- method_name_len = spprintf(&http_method, 0, "METH_%s", method);
+ method_name_len = spprintf(&http_method, 0, "METH_%s", mconst);
zend_declare_class_constant_long(http_request_object_ce, http_method, method_name_len, meth_num TSRMLS_CC);
efree(http_method);
#endif
-
+
return meth_num;
}
/* }}} */
-/* {{{ STATUS http_request_method_unregister(usngigned long) */
-PHP_HTTP_API STATUS _http_request_method_unregister(ulong method TSRMLS_DC)
+/* {{{ STATUS http_request_method_unregister(int) */
+PHP_HTTP_API STATUS _http_request_method_unregister(int method TSRMLS_DC)
{
- zval **zmethod;
char *http_method;
int method_len;
-
- if (SUCCESS != zend_hash_index_find(&HTTP_G(request).methods.custom, HTTP_CUSTOM_REQUEST_METHOD(method), (void **) &zmethod)) {
- http_error_ex(HE_NOTICE, HTTP_E_REQUEST_METHOD, "Request method with id %lu does not exist", method);
+ http_request_method_entry **ptr = HTTP_G->request.methods.custom.entries;
+
+ if (HTTP_STD_REQUEST_METHOD(method)) {
+ http_error_ex(HE_WARNING, HTTP_E_REQUEST_METHOD, "Standard request methods cannot be unregistered");
return FAILURE;
}
+ if ( (HTTP_CUSTOM_REQUEST_METHOD(method) < 0) ||
+ (HTTP_CUSTOM_REQUEST_METHOD(method) > HTTP_G->request.methods.custom.count) ||
+ (!ptr[HTTP_CUSTOM_REQUEST_METHOD(method)])) {
+ http_error_ex(HE_NOTICE, HTTP_E_REQUEST_METHOD, "Custom request method with id %d does not exist", method);
+ return FAILURE;
+ }
+
#if defined(ZEND_ENGINE_2) && defined(HTTP_HAVE_CURL) && !defined(WONKY)
- method_len = spprintf(&http_method, 0, "METH_%s", Z_STRVAL_PP(zmethod));
- if ((SUCCESS != zend_hash_del(&http_request_object_ce->constants_table, http_method, method_len + 1))) {
+ method_len = spprintf(&http_method, 0, "METH_%s", ptr[HTTP_CUSTOM_REQUEST_METHOD(method)]->cnst);
+ if (SUCCESS != zend_hash_del(&http_request_object_ce->constants_table, http_method, method_len + 1)) {
http_error_ex(HE_NOTICE, HTTP_E_REQUEST_METHOD, "Could not unregister request method: HttpRequest::%s", http_method);
efree(http_method);
return FAILURE;
efree(http_method);
#endif
- method_len = spprintf(&http_method, 0, "HTTP_METH_%s", Z_STRVAL_PP(zmethod));
- if ( (SUCCESS != zend_hash_index_del(&HTTP_G(request).methods.custom, HTTP_CUSTOM_REQUEST_METHOD(method)))
- || (SUCCESS != zend_hash_del(EG(zend_constants), http_method, method_len + 1))) {
+ method_len = spprintf(&http_method, 0, "HTTP_METH_%s", ptr[HTTP_CUSTOM_REQUEST_METHOD(method)]->cnst);
+ if (SUCCESS != zend_hash_del(EG(zend_constants), http_method, method_len + 1)) {
http_error_ex(HE_NOTICE, HTTP_E_REQUEST_METHOD, "Could not unregister request method: %s", http_method);
efree(http_method);
return FAILURE;
}
efree(http_method);
+ efree(ptr[HTTP_CUSTOM_REQUEST_METHOD(method)]->name);
+ efree(ptr[HTTP_CUSTOM_REQUEST_METHOD(method)]->cnst);
+ efree(ptr[HTTP_CUSTOM_REQUEST_METHOD(method)]);
+ ptr[HTTP_CUSTOM_REQUEST_METHOD(method)] = NULL;
+
return SUCCESS;
}
/* }}} */