X-Git-Url: https://git.m6w6.name/?p=m6w6%2Fext-http;a=blobdiff_plain;f=http_exception_object.c;h=c883863036ed74bfc28d86944b97e15992fbf2a2;hp=5af0dc568156b3dad3f7b01f165e285916204902;hb=refs%2Fheads%2Fv1.7.x;hpb=fcebff9b60cdeede7970f5b4bb7b01318b4415c3 diff --git a/http_exception_object.c b/http_exception_object.c index 5af0dc5..c883863 100644 --- a/http_exception_object.c +++ b/http_exception_object.c @@ -1,43 +1,109 @@ /* - +----------------------------------------------------------------------+ - | PECL :: http | - +----------------------------------------------------------------------+ - | This source file is subject to version 3.0 of the PHP license, that | - | is bundled with this package in the file LICENSE, and is available | - | through the world-wide-web at http://www.php.net/license/3_0.txt. | - | If you did not receive a copy of the PHP license and are unable to | - | obtain it through the world-wide-web, please send a note to | - | license@php.net so we can mail you a copy immediately. | - +----------------------------------------------------------------------+ - | Copyright (c) 2004-2005 Michael Wallner | - +----------------------------------------------------------------------+ + +--------------------------------------------------------------------+ + | PECL :: http | + +--------------------------------------------------------------------+ + | Redistribution and use in source and binary forms, with or without | + | modification, are permitted provided that the conditions mentioned | + | in the accompanying LICENSE file are met. | + +--------------------------------------------------------------------+ + | Copyright (c) 2004-2010, Michael Wallner | + +--------------------------------------------------------------------+ */ /* $Id$ */ - -#ifdef HAVE_CONFIG_H -# include "config.h" -#endif - -#include "php.h" - #include "php_http.h" -#include "php_http_std_defs.h" #ifdef ZEND_ENGINE_2 -#include "php_http_exception_object.h" +#ifndef HTTP_DBG_EXCEPTIONS +# define HTTP_DBG_EXCEPTIONS 0 +#endif + +#include "zend_interfaces.h" #include "zend_exceptions.h" +#include "php_http_exception_object.h" zend_class_entry *http_exception_object_ce; +zend_class_entry *HTTP_EX_CE(runtime); +zend_class_entry *HTTP_EX_CE(header); +zend_class_entry *HTTP_EX_CE(malformed_headers); +zend_class_entry *HTTP_EX_CE(request_method); +zend_class_entry *HTTP_EX_CE(message_type); +zend_class_entry *HTTP_EX_CE(invalid_param); +zend_class_entry *HTTP_EX_CE(encoding); +zend_class_entry *HTTP_EX_CE(request); +zend_class_entry *HTTP_EX_CE(request_pool); +zend_class_entry *HTTP_EX_CE(socket); +zend_class_entry *HTTP_EX_CE(response); +zend_class_entry *HTTP_EX_CE(url); +zend_class_entry *HTTP_EX_CE(querystring); + +#define HTTP_EMPTY_ARGS(method) HTTP_EMPTY_ARGS_EX(HttpException, method, 0) +#define HTTP_EXCEPTION_ME(method, visibility) PHP_ME(HttpException, method, HTTP_ARGS(HttpException, method), visibility) + +HTTP_EMPTY_ARGS(__toString); + zend_function_entry http_exception_object_fe[] = { - {NULL, NULL, NULL} + HTTP_EXCEPTION_ME(__toString, ZEND_ACC_PUBLIC) + + EMPTY_FUNCTION_ENTRY }; -void _http_exception_object_init(INIT_FUNC_ARGS) +#if HTTP_DBG_EXCEPTIONS +static void http_exception_hook(zval *ex TSRMLS_DC) { - HTTP_REGISTER_CLASS(HttpException, http_exception_object, zend_exception_get_default(), 0); + if (ex) { + zval *m = zend_read_property(Z_OBJCE_P(ex), ex, "message", lenof("message"), 0 TSRMLS_CC); + fprintf(stderr, "*** Threw exception '%s'\n", Z_STRVAL_P(m)); + } else { + fprintf(stderr, "*** Threw NULL exception\n"); + } +} +#endif + +PHP_MINIT_FUNCTION(http_exception_object) +{ + HTTP_REGISTER_CLASS(HttpException, http_exception_object, ZEND_EXCEPTION_GET_DEFAULT(), 0); + + zend_declare_property_null(HTTP_EX_DEF_CE, "innerException", lenof("innerException"), ZEND_ACC_PUBLIC TSRMLS_CC); + + HTTP_REGISTER_EXCEPTION(HttpRuntimeException, HTTP_EX_CE(runtime), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpInvalidParamException, HTTP_EX_CE(invalid_param), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpHeaderException, HTTP_EX_CE(header), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpMalformedHeadersException, HTTP_EX_CE(malformed_headers), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpRequestMethodException, HTTP_EX_CE(request_method), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpMessageTypeException, HTTP_EX_CE(message_type), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpEncodingException, HTTP_EX_CE(encoding), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpRequestException, HTTP_EX_CE(request), HTTP_EX_DEF_CE); + + zend_declare_property_long(HTTP_EX_CE(request), "curlCode", lenof("curlCode"), 0, ZEND_ACC_PUBLIC TSRMLS_CC); + + HTTP_REGISTER_EXCEPTION(HttpRequestPoolException, HTTP_EX_CE(request_pool), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpSocketException, HTTP_EX_CE(socket), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpResponseException, HTTP_EX_CE(response), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpUrlException, HTTP_EX_CE(url), HTTP_EX_DEF_CE); + HTTP_REGISTER_EXCEPTION(HttpQueryStringException, HTTP_EX_CE(querystring), HTTP_EX_DEF_CE); + + HTTP_LONG_CONSTANT("HTTP_E_RUNTIME", HTTP_E_RUNTIME); + HTTP_LONG_CONSTANT("HTTP_E_INVALID_PARAM", HTTP_E_INVALID_PARAM); + HTTP_LONG_CONSTANT("HTTP_E_HEADER", HTTP_E_HEADER); + HTTP_LONG_CONSTANT("HTTP_E_MALFORMED_HEADERS", HTTP_E_MALFORMED_HEADERS); + HTTP_LONG_CONSTANT("HTTP_E_REQUEST_METHOD", HTTP_E_REQUEST_METHOD); + HTTP_LONG_CONSTANT("HTTP_E_MESSAGE_TYPE", HTTP_E_MESSAGE_TYPE); + HTTP_LONG_CONSTANT("HTTP_E_ENCODING", HTTP_E_ENCODING); + HTTP_LONG_CONSTANT("HTTP_E_REQUEST", HTTP_E_REQUEST); + HTTP_LONG_CONSTANT("HTTP_E_REQUEST_POOL", HTTP_E_REQUEST_POOL); + HTTP_LONG_CONSTANT("HTTP_E_SOCKET", HTTP_E_SOCKET); + HTTP_LONG_CONSTANT("HTTP_E_RESPONSE", HTTP_E_RESPONSE); + HTTP_LONG_CONSTANT("HTTP_E_URL", HTTP_E_URL); + HTTP_LONG_CONSTANT("HTTP_E_QUERYSTRING", HTTP_E_QUERYSTRING); + +#if HTTP_DBG_EXCEPTIONS + zend_throw_exception_hook=http_exception_hook; +#endif + + return SUCCESS; } zend_class_entry *_http_exception_get_default() @@ -45,17 +111,66 @@ zend_class_entry *_http_exception_get_default() return http_exception_object_ce; } -void _http_exception_throw_ce_ex(zend_class_entry *ce, int code TSRMLS_DC) +zend_class_entry *_http_exception_get_for_code(long code) { - static char * const errors[] = { - "Unkown Error" - }; + zend_class_entry *ex = http_exception_object_ce; + + switch (code) { + case HTTP_E_RUNTIME: ex = HTTP_EX_CE(runtime); break; + case HTTP_E_INVALID_PARAM: ex = HTTP_EX_CE(invalid_param); break; + case HTTP_E_HEADER: ex = HTTP_EX_CE(header); break; + case HTTP_E_MALFORMED_HEADERS: ex = HTTP_EX_CE(malformed_headers); break; + case HTTP_E_REQUEST_METHOD: ex = HTTP_EX_CE(request_method); break; + case HTTP_E_MESSAGE_TYPE: ex = HTTP_EX_CE(message_type); break; + case HTTP_E_ENCODING: ex = HTTP_EX_CE(encoding); break; + case HTTP_E_REQUEST: ex = HTTP_EX_CE(request); break; + case HTTP_E_REQUEST_POOL: ex = HTTP_EX_CE(request_pool); break; + case HTTP_E_SOCKET: ex = HTTP_EX_CE(socket); break; + case HTTP_E_RESPONSE: ex = HTTP_EX_CE(response); break; + case HTTP_E_URL: ex = HTTP_EX_CE(url); break; + case HTTP_E_QUERYSTRING: ex = HTTP_EX_CE(querystring); break; + } + + return ex; +} + +PHP_METHOD(HttpException, __toString) +{ + phpstr full_str; + zend_class_entry *ce; + zval *zobj = getThis(), *retval = NULL, *m, *f, *l; + + phpstr_init(&full_str); + + do { + ce = Z_OBJCE_P(zobj); + + m = zend_read_property(ce, zobj, "message", lenof("message"), 0 TSRMLS_CC); + f = zend_read_property(ce, zobj, "file", lenof("file"), 0 TSRMLS_CC); + l = zend_read_property(ce, zobj, "line", lenof("line"), 0 TSRMLS_CC); + + if (m && f && l && Z_TYPE_P(m) == IS_STRING && Z_TYPE_P(f) == IS_STRING && Z_TYPE_P(l) == IS_LONG) { + if (zobj != getThis()) { + phpstr_appends(&full_str, " inner "); + } + + phpstr_appendf(&full_str, "exception '%.*s' with message '%.*s' in %.*s:%ld" PHP_EOL, + ce->name_length, ce->name, Z_STRLEN_P(m), Z_STRVAL_P(m), Z_STRLEN_P(f), Z_STRVAL_P(f), Z_LVAL_P(l) + ); + } else { + break; + } + + zobj = zend_read_property(ce, zobj, "innerException", lenof("innerException"), 0 TSRMLS_CC); + } while (Z_TYPE_P(zobj) == IS_OBJECT); - if (!ce) { - ce = http_exception_get_default(); + if (zend_call_method_with_0_params(&getThis(), Z_OBJCE_P(getThis()), NULL, "gettraceasstring", &retval) && Z_TYPE_P(retval) == IS_STRING) { + phpstr_appends(&full_str, "Stack trace:" PHP_EOL); + phpstr_append(&full_str, Z_STRVAL_P(retval), Z_STRLEN_P(retval)); + zval_ptr_dtor(&retval); } - zend_throw_exception(ce, errors[code], code TSRMLS_CC); + RETURN_PHPSTR_VAL(&full_str); } #endif