|
|
| version 1.301, 2009/05/13 07:35:41 | version 1.321, 2010/05/25 03:58:25 |
|---|---|
| Line 49 const char* ORIGINS_CONTENT_TYPE="text/p | Line 49 const char* ORIGINS_CONTENT_TYPE="text/p |
| #define MAIN_METHOD_NAME "main" | #define MAIN_METHOD_NAME "main" |
| #define AUTO_METHOD_NAME "auto" | #define AUTO_METHOD_NAME "auto" |
| #define AUTOUSE_METHOD_NAME "autouse" | |
| #define BODY_NAME "body" | #define BODY_NAME "body" |
| #define EXCEPTION_TYPE_PART_NAME "type" | #define EXCEPTION_TYPE_PART_NAME "type" |
| #define EXCEPTION_SOURCE_PART_NAME "source" | #define EXCEPTION_SOURCE_PART_NAME "source" |
| Line 58 const char* ORIGINS_CONTENT_TYPE="text/p | Line 59 const char* ORIGINS_CONTENT_TYPE="text/p |
| const String main_method_name(MAIN_METHOD_NAME); | const String main_method_name(MAIN_METHOD_NAME); |
| const String auto_method_name(AUTO_METHOD_NAME); | const String auto_method_name(AUTO_METHOD_NAME); |
| const String content_transfer_encoding_name(CONTENT_TRANSFER_ENCODING_NAME); | const String autouse_method_name(AUTOUSE_METHOD_NAME); |
| const String content_disposition_name(CONTENT_DISPOSITION_NAME); | |
| const String content_disposition_inline(CONTENT_DISPOSITION_INLINE); | |
| const String content_disposition_attachment(CONTENT_DISPOSITION_ATTACHMENT); | |
| const String content_disposition_filename_name(CONTENT_DISPOSITION_FILENAME_NAME); | |
| const String body_name(BODY_NAME); | const String body_name(BODY_NAME); |
| const String exception_type_part_name(EXCEPTION_TYPE_PART_NAME); | const String exception_type_part_name(EXCEPTION_TYPE_PART_NAME); |
| const String exception_source_part_name(EXCEPTION_SOURCE_PART_NAME); | const String exception_source_part_name(EXCEPTION_SOURCE_PART_NAME); |
| Line 109 Request::Request(SAPI_Info& asapi_info, | Line 107 Request::Request(SAPI_Info& asapi_info, |
| wcontext(0), | wcontext(0), |
| flang(adefault_lang), | flang(adefault_lang), |
| fconnection(0), | fconnection(0), |
| finterrupted(false),fskip(SKIP_NOTHING), | finterrupted(false), |
| fskip(SKIP_NOTHING), | |
| fin_cycle(0), | |
| // public | // public |
| #ifdef RESOURCES_DEBUG | #ifdef RESOURCES_DEBUG |
| Line 198 Request::~Request() { | Line 198 Request::~Request() { |
| Value& Request::get_self() { return method_frame/*always have!*/->self(); } | Value& Request::get_self() { return method_frame/*always have!*/->self(); } |
| Value* Request::get_class(const String& name){ | |
| Value* result=classes().get(name); | |
| if(!result) | |
| if(Value* value=main_class.get_element(autouse_method_name)) | |
| if(Junction* junction=value->get_junction()) | |
| if(const Method *method=junction->method) { | |
| Value *vname=new VString(name); | |
| VMethodFrame frame(*junction, 0/*no parent*/); | |
| frame.set_self(main_class); | |
| frame.store_params(&vname, 1); | |
| // we don't need the result | |
| execute_method(frame, *method); | |
| result=classes().get(name); | |
| } | |
| return result; | |
| } | |
| static void load_charset(HashStringValue::key_type akey, | static void load_charset(HashStringValue::key_type akey, |
| HashStringValue::value_type avalue, | HashStringValue::value_type avalue, |
| Request_charsets* charsets) { | Request_charsets* charsets) { |
| Line 219 void Request::configure_admin(VStateless | Line 239 void Request::configure_admin(VStateless |
| ... | ... |
| ] | ] |
| */ | */ |
| if(Value* vcharsets=conf_class.get_element(charsets_name, conf_class, false)) { | if(Value* vcharsets=conf_class.get_element(charsets_name)) { |
| if(!vcharsets->is_string()) | if(!vcharsets->is_string()) |
| if(HashStringValue* charsets=vcharsets->get_hash()) | if(HashStringValue* charsets=vcharsets->get_hash()) |
| charsets->for_each<Request_charsets*>(load_charset, &this->charsets); | charsets->for_each<Request_charsets*>(load_charset, &this->charsets); |
| Line 283 void Request::configure() { | Line 303 void Request::configure() { |
| methoded_array().configure_user(*this); | methoded_array().configure_user(*this); |
| // $MAIN:MIME-TYPES | // $MAIN:MIME-TYPES |
| if(Value* element=main_class.get_element(mime_types_name, main_class, false)) | if(Value* element=main_class.get_element(mime_types_name)) |
| if(Table *table=element->get_table()) | if(Table *table=element->get_table()) |
| mime_types=table; | mime_types=table; |
| } | } |
| Line 312 gettimeofday(&mt[0],NULL); | Line 332 gettimeofday(&mt[0],NULL); |
| // loading config | // loading config |
| if(config_filespec) { | if(config_filespec) { |
| const String& filespec=*new String(config_filespec); | const String& filespec=*new String(config_filespec); |
| use_file(main_class, | use_file_directly(main_class, |
| filespec, 0 /*main_alias*/, | filespec, |
| true /*ignore class_path*/, | |
| config_fail_on_read_problem, | config_fail_on_read_problem, |
| true /*file must exist if 'fail on read problem' not set*/); | true /*file must exist if 'fail on read problem' not set*/); |
| } | } |
| Line 339 gettimeofday(&mt[0],NULL); | Line 358 gettimeofday(&mt[0],NULL); |
| String::L_CLEAN); | String::L_CLEAN); |
| sfile_spec << "/" AUTO_FILE_NAME; | sfile_spec << "/" AUTO_FILE_NAME; |
| use_file(main_class, | use_file_directly(main_class, |
| sfile_spec, | sfile_spec, |
| 0 /*main_alias*/, | |
| true /*ignore class_path*/, | |
| true /*fail on read problem*/, | true /*fail on read problem*/, |
| false /*but ignore absence, sole user*/); | false /*but ignore absence, sole user*/); |
| } | } |
| after=before+1; | for(after=before+1;*after=='/';after++); |
| } | } |
| } | } |
| Line 354 gettimeofday(&mt[0],NULL); | Line 371 gettimeofday(&mt[0],NULL); |
| // compile requested file | // compile requested file |
| String& spath_translated=*new String; | String& spath_translated=*new String; |
| spath_translated.append_help_length(request_info.path_translated, 0, String::L_TAINTED); | spath_translated.append_help_length(request_info.path_translated, 0, String::L_TAINTED); |
| use_file(main_class, | use_file_directly(main_class, spath_translated); |
| spath_translated, | |
| 0 /*main_alias*/, | |
| true /*ignore class_path*/); | |
| configure(); | configure(); |
| } catch(...) { | } catch(...) { |
| Line 388 gettimeofday(&mt[2],NULL); | Line 402 gettimeofday(&mt[2],NULL); |
| body_value=response.fields().get(body_name); // $response:body | body_value=response.fields().get(body_name); // $response:body |
| if(!body_value) | if(!body_value) |
| body_value=new VString(*body_string); // just result of ^main[] | body_value=new VString(*body_string); // just result of ^main[] |
| // ensure that body_value has no just L_TAINTED parts left | |
| if(body_value->is_string()) | |
| { | |
| String& untainted=*new String(); | |
| untainted.append(*body_value->get_string(), flang); | |
| body_value=new VString(untainted); | |
| } | |
| // @postprocess | // @postprocess |
| if(Value* value=main_class.get_element(post_process_method_name, main_class, false)) | if(Value* value=main_class.get_element(post_process_method_name)) |
| if(Junction* junction=value->get_junction()) | if(Junction* junction=value->get_junction()) |
| if(const Method *method=junction->method) { | if(const Method *method=junction->method) { |
| // preparing to pass parameters to | // preparing to pass parameters to |
| Line 409 gettimeofday(&mt[2],NULL); | Line 416 gettimeofday(&mt[2],NULL); |
| body_value=&execute_method(frame, *method).as_value(); | body_value=&execute_method(frame, *method).as_value(); |
| } | } |
| VFile* body_file=body_value->as_vfile( | VFile* body_file=body_value->as_vfile(flang, &charsets); |
| String::L_UNSPECIFIED, &charsets); | |
| #ifdef RESOURCES_DEBUG | #ifdef RESOURCES_DEBUG |
| //measure:after postprocess | //measure:after postprocess |
| Line 456 t[9]-t[3] | Line 462 t[9]-t[3] |
| // maybe we'd be lucky enough as to report an error | // maybe we'd be lucky enough as to report an error |
| // in a gracefull way... | // in a gracefull way... |
| if(Value* value=main_class.get_element( | if(Value* value=main_class.get_element(*new String(UNHANDLED_EXCEPTION_METHOD_NAME))) { |
| *new String(UNHANDLED_EXCEPTION_METHOD_NAME), | |
| main_class, | |
| false)) { | |
| if(Junction* junction=value->get_junction()) { | if(Junction* junction=value->get_junction()) { |
| if(const Method *method=junction->method) { | if(const Method *method=junction->method) { |
| // preparing to pass parameters to | // preparing to pass parameters to |
| Line 504 t[9]-t[3] | Line 507 t[9]-t[3] |
| // doing that ugly | // doing that ugly |
| // future $response:content-type | // future $response:content-type |
| response.fields().put(content_type_name, | response.fields().put(http_content_type, new VString(*new String(UNHANDLED_EXCEPTION_CONTENT_TYPE))); |
| new VString(*new String(UNHANDLED_EXCEPTION_CONTENT_TYPE))); | |
| // future $response:body | // future $response:body |
| body_string=new String(exception_cstr); | body_string=new String(exception_cstr); |
| } | } |
| VString body_vstring(*body_string); | VString body_vstring(*body_string); |
| VFile* body_file=body_vstring.as_vfile(String::L_UNSPECIFIED, &charsets); | VFile* body_file=body_vstring.as_vfile(flang, &charsets); |
| // conditionally log it | // conditionally log it |
| Value* vhandled=details.vhash.hash().get(exception_handled_part_name); | Value* vhandled=details.vhash.hash().get(exception_handled_part_name); |
| Line 541 uint Request::register_file(String::Body | Line 543 uint Request::register_file(String::Body |
| return file_list.count()-1; | return file_list.count()-1; |
| } | } |
| void Request::use_file(VStateless_class& aclass, | void Request::use_file_directly(VStateless_class& aclass, |
| const String& file_name, const String* main_alias, | const String& file_spec, |
| bool ignore_class_path, | bool fail_on_read_problem, |
| bool fail_on_read_problem, | bool fail_on_file_absence) { |
| bool fail_on_file_absence) { | |
| // cyclic dependence check | // cyclic dependence check |
| if(used_files.get(file_name)) | if(used_files.get(file_spec)) |
| return; | return; |
| used_files.put(file_name, true); | used_files.put(file_spec, true); |
| if(fail_on_read_problem && !fail_on_file_absence) // ignore file absence if asked for | |
| if(!entry_exists(file_spec)) | |
| return; | |
| if(const char* source=file_read_text(charsets, file_spec, fail_on_read_problem)) | |
| use_buf(aclass, source, 0, register_file(file_spec)); | |
| } | |
| const String* file_spec; | void Request::use_file(VStateless_class& aclass, |
| if(ignore_class_path) // ignore_class_path? | const String& file_name, |
| file_spec=&file_name; | const String* use_filespec/*absolute*/) { |
| else if(file_name.first_char()=='/') //absolute path? [no need to scan MAIN:CLASS_PATH] | |
| file_spec=&absolute(file_name); | const String* filespec=0; |
| else { | |
| file_spec=0; | if(file_name.first_char()=='/') //absolute path? [no need to scan MAIN:CLASS_PATH] |
| if(Value* element=main_class.get_element(class_path_name, main_class, false)) { | filespec=&absolute(file_name); |
| else if(use_filespec){ // search in current dir first | |
| int afterslash=lastposafter(*use_filespec, 0, "/", 1, true); | |
| if(afterslash>0) | |
| filespec=file_exist(use_filespec->mid(0, afterslash-1), file_name); // found in current dir? | |
| } | |
| if(!filespec){ | |
| // prevent multiple scan CLASS_PATH for searching one file | |
| if(searched_along_class_path.get(file_name)) | |
| return; | |
| searched_along_class_path.put(file_name, true); | |
| if(Value* element=main_class.get_element(class_path_name)) { | |
| if(element->is_string()) { | if(element->is_string()) { |
| file_spec=file_exist(absolute(element->as_string()), file_name); // found at class_path? | filespec=file_exist(absolute(element->as_string()), file_name); // found at class_path? |
| } else if(Table *table=element->get_table()) { | } else if(Table *table=element->get_table()) { |
| int size=table->count(); | for(size_t i=table->count(); i--; ) { |
| for(int i=size; i--; ) { | |
| const String& path=*(*table->get(i))[0]; | const String& path=*(*table->get(i))[0]; |
| if((file_spec=file_exist(absolute(path), file_name))) | if(filespec=file_exist(absolute(path), file_name)) |
| break; // found along class_path | break; // found along class_path |
| } | } |
| } else | } else |
| throw Exception(PARSER_RUNTIME, | throw Exception(PARSER_RUNTIME, |
| 0, | 0, |
| "$" CLASS_PATH_NAME " must be string or table"); | "$" CLASS_PATH_NAME " must be string or table"); |
| if(!file_spec) | if(!filespec) |
| throw Exception(PARSER_RUNTIME, | throw Exception(PARSER_RUNTIME, |
| &file_name, | &file_name, |
| "not found along " MAIN_CLASS_NAME ":" CLASS_PATH_NAME); | "not found along " MAIN_CLASS_NAME ":" CLASS_PATH_NAME); |
| } | } else |
| if(!file_spec) | |
| throw Exception(PARSER_RUNTIME, | throw Exception(PARSER_RUNTIME, |
| &file_name, | &file_name, |
| "usage failed - no $" MAIN_CLASS_NAME ":" CLASS_PATH_NAME " were specified"); | "usage failed - no $" MAIN_CLASS_NAME ":" CLASS_PATH_NAME " were specified"); |
| } | } |
| if(fail_on_read_problem && !fail_on_file_absence) // ignore file absence if asked for | use_file_directly(aclass, *filespec); |
| if(!entry_exists(*file_spec)) | |
| return; | |
| if(const char* source=file_read_text(charsets, *file_spec, fail_on_read_problem)) | |
| use_buf(aclass, source, main_alias, register_file(*file_spec)); | |
| } | } |
| void Request::use_buf(VStateless_class& aclass, | void Request::use_buf(VStateless_class& aclass, |
| const char* source, const String* main_alias, | const char* source, const String* main_alias, |
| uint file_no, | uint file_no, |
| int line_no_offset) { | int line_no_offset) { |
| // temporary zero @conf so to maybe-replace it in compiled code | // temporary zero @conf so to maybe-replace it in compiled code |
| Temp_method temp_method_conf(aclass, conf_method_name, 0); | Temp_method temp_method_conf(aclass, conf_method_name, 0); |
| // temporary zero @auto so to maybe-replace it in compiled code | // temporary zero @auto so to maybe-replace it in compiled code |
| Line 669 static void add_header_attribute( | Line 685 static void add_header_attribute( |
| || name==CHARSET_NAME) | || name==CHARSET_NAME) |
| return; | return; |
| const char* aname=String(name, String::L_URI).cstr(String::L_UNSPECIFIED, 0, &info->r.charsets); | const char* aname=String(name, String::L_URI).untaint_and_transcode_cstr(String::L_URI, &info->r.charsets); |
| SAPI::add_header_attribute(info->r.sapi_info, | SAPI::add_header_attribute(info->r.sapi_info, |
| aname, | aname, |
| attributed_meaning_to_string(*value, String::L_URI, false).cstr(String::L_UNSPECIFIED, 0, &info->r.charsets)); | attributed_meaning_to_string(*value, String::L_URI, false).untaint_and_transcode_cstr(String::L_URI, &info->r.charsets) |
| ); | |
| if(strcasecmp(aname, "last-modified")==0) | if(strcasecmp(aname, "last-modified")==0) |
| info->add_last_modified = false; | info->add_last_modified=false; |
| } | } |
| static void output_sole_piece(Request& r, | static void output_sole_piece(Request& r, |
| Line 690 static void output_sole_piece(Request& r | Line 707 static void output_sole_piece(Request& r |
| r.charsets.source(), | r.charsets.source(), |
| r.charsets.client()); | r.charsets.client()); |
| // prepare header: content-length | // prepare header: Content-Length |
| char content_length_cstr[MAX_NUMBER]; | SAPI::add_header_attribute(r.sapi_info, HTTP_CONTENT_LENGTH, format(output.length, "%u")); |
| snprintf(content_length_cstr, MAX_NUMBER, "%u", output.length); | |
| SAPI::add_header_attribute(r.sapi_info, "content-length", content_length_cstr); | |
| // send header | // send header |
| SAPI::send_header(r.sapi_info); | SAPI::send_header(r.sapi_info); |
| Line 710 struct Range | Line 725 struct Range |
| size_t end; | size_t end; |
| }; | }; |
| #endif | #endif |
| static void parse_range(const String* s, Array<Range> &ar) | static void parse_range(const String* s, Array<Range> &ar) { |
| { | |
| const char *p = s->cstr(); | const char *p = s->cstr(); |
| if(s->starts_with("bytes=")) | if(s->starts_with("bytes=")) |
| p += 6; | p += 6; |
| Line 740 static void output_pieces(Request& r, | Line 754 static void output_pieces(Request& r, |
| Value& date, | Value& date, |
| bool add_last_modified) | bool add_last_modified) |
| { | { |
| SAPI::add_header_attribute(r.sapi_info, "Accept-Ranges", "bytes"); | SAPI::add_header_attribute(r.sapi_info, "accept-ranges", "bytes"); |
| const size_t BUFSIZE = 10*0x400; | const size_t BUFSIZE = 10*0x400; |
| char buf[BUFSIZE]; | char buf[BUFSIZE]; |
| Line 754 static void output_pieces(Request& r, | Line 768 static void output_pieces(Request& r, |
| if(count == 1){ | if(count == 1){ |
| Range &rg = ar.get_ref(0); | Range &rg = ar.get_ref(0); |
| if(rg.start == (size_t)-1 && rg.end == (size_t)-1){ | if(rg.start == (size_t)-1 && rg.end == (size_t)-1){ |
| SAPI::add_header_attribute(r.sapi_info, "status", "416 Requested Range Not Satisfiable"); | SAPI::add_header_attribute(r.sapi_info, HTTP_STATUS, "416 Requested Range Not Satisfiable"); |
| return; | return; |
| } | } |
| if(rg.start == (size_t)-1 && rg.end != (size_t)-1){ | if(rg.start == (size_t)-1 && rg.end != (size_t)-1){ |
| Line 768 static void output_pieces(Request& r, | Line 782 static void output_pieces(Request& r, |
| part_length -= rg.start; | part_length -= rg.start; |
| } | } |
| if(part_length == 0){ | if(part_length == 0){ |
| SAPI::add_header_attribute(r.sapi_info, "status", "204 No Content"); | SAPI::add_header_attribute(r.sapi_info, HTTP_STATUS, "204 No Content"); |
| return; | return; |
| } | } |
| SAPI::add_header_attribute(r.sapi_info, "status", "206 Partial Content"); | SAPI::add_header_attribute(r.sapi_info, HTTP_STATUS, "206 Partial Content"); |
| snprintf(buf, BUFSIZE, "bytes %u-%u/%u", rg.start, rg.end, content_length); | snprintf(buf, BUFSIZE, "bytes %u-%u/%u", rg.start, rg.end, content_length); |
| SAPI::add_header_attribute(r.sapi_info, "Content-Range", buf); | SAPI::add_header_attribute(r.sapi_info, "content-range", buf); |
| }else if(count != 0){ | }else if(count != 0){ |
| SAPI::add_header_attribute(r.sapi_info, "status", "501 Not Implemented"); | SAPI::add_header_attribute(r.sapi_info, HTTP_STATUS, "501 Not Implemented"); |
| return; | return; |
| } | } |
| } | } |
| snprintf(buf, BUFSIZE, "%u", part_length); | SAPI::add_header_attribute(r.sapi_info, HTTP_CONTENT_LENGTH, format(part_length, "%u")); |
| SAPI::add_header_attribute(r.sapi_info, "Content-Length", buf); | |
| if(add_last_modified) | |
| SAPI::add_header_attribute(r.sapi_info, "last-modified", attributed_meaning_to_string(date, String::L_AS_IS, true).cstr()); | |
| if(add_last_modified){ | |
| const String &s = attributed_meaning_to_string(date, String::L_AS_IS, true); | |
| SAPI::add_header_attribute(r.sapi_info, "Last-Modified", s.cstr()); | |
| } | |
| SAPI::send_header(r.sapi_info); | SAPI::send_header(r.sapi_info); |
| const String& filespec=r.absolute(filename); | const String& filespec=r.absolute(filename); |
| Line 819 void Request::output_result(VFile* body_ | Line 831 void Request::output_result(VFile* body_ |
| // header: cookies | // header: cookies |
| cookie.output_result(sapi_info); | cookie.output_result(sapi_info); |
| // may be specified | // _file_ content-type might be specified |
| Value* body_file_content_type=body_file->fields().get(content_type_name); | Value* body_file_content_type=body_file->fields().get(content_type_name); |
| // content-disposition | // Content-Disposition |
| Value* vfile_name=body_file->fields().get(name_name); | Value* vfile_name=body_file->fields().get(name_name); |
| if(!vfile_name) { | if(!vfile_name) { |
| vfile_name=body_file->fields().get(response_body_file_name); | vfile_name=body_file->fields().get(response_body_file_name); |
| if(vfile_name) | if(vfile_name) { |
| { | char* name_cstr=vfile_name->as_string().cstrm(); |
| const String& sfile_name=vfile_name->as_string(); | |
| char* name_cstr=sfile_name.cstrm(); | |
| if(char *after_slash=rsplit(name_cstr, '\\')) | if(char *after_slash=rsplit(name_cstr, '\\')) |
| name_cstr=after_slash; | name_cstr=after_slash; |
| if(char *after_slash=rsplit(name_cstr, '/')) | if(char *after_slash=rsplit(name_cstr, '/')) |
| Line 844 void Request::output_result(VFile* body_ | Line 853 void Request::output_result(VFile* body_ |
| VHash& hash=*new VHash(); | VHash& hash=*new VHash(); |
| HashStringValue &h=hash.hash(); | HashStringValue &h=hash.hash(); |
| h.put(value_name, new VString( as_attachment ? content_disposition_attachment : content_disposition_inline )); | h.put(value_name, new VString( as_attachment ? content_disposition_attachment : content_disposition_inline )); |
| h.put(content_disposition_filename_name, vfile_name); | h.put(content_disposition_filename_name, vfile_name); |
| response.fields().put(content_disposition_name, &hash); | |
| response.fields().put(content_disposition, &hash); | |
| if(!body_file_content_type) | if(!body_file_content_type) |
| body_file_content_type=new VString(mime_type_of(sfile_name.cstr())); | body_file_content_type=new VString(mime_type_of(sfile_name.cstr())); |
| } | } |
| } | } |
| // set content-type | // set Content-Type |
| if(body_file_content_type) { | if(body_file_content_type) { |
| // body file content type | // body file content type |
| response.fields().put(content_type_name, body_file_content_type); | response.fields().put(content_type_name, body_file_content_type); |
| } else { | } else { |
| // default content type | // default content type |
| response.fields().put_dont_replace(content_type_name, | response.fields().put_dont_replace(content_type_name, new VString(*new String(DEFAULT_CONTENT_TYPE))); |
| new VString(*new String(DEFAULT_CONTENT_TYPE))); | |
| } | } |
| // prepare header: $response:fields without :body | // prepare header: $response:fields without :body, :download and :charset |
| Add_header_attribute_info info(*this); | Add_header_attribute_info info(*this); |
| response.fields().for_each<Add_header_attribute_info*>(add_header_attribute, &info); | response.fields().for_each<Add_header_attribute_info*>(add_header_attribute, &info); |
| if(body_file_content_type) | |
| if(HashStringValue *hash=body_file_content_type->get_hash()) | |
| body_file_content_type=hash->get(value_name); | |
| if(Value* vresponse_body_file=body_file->fields().get(response_body_file_name)) { | if(Value* vresponse_body_file=body_file->fields().get(response_body_file_name)) { |
| // $response:[download|body][$.file[filespec]] -- optput specified file | |
| const String& sresponse_body_file=vresponse_body_file->as_string(); | const String& sresponse_body_file=vresponse_body_file->as_string(); |
| size_t content_length=0; | size_t content_length=0; |
| time_t atime=0, mtime=0, ctime=0; | time_t atime=0, mtime=0, ctime=0; |
| Line 881 void Request::output_result(VFile* body_ | Line 886 void Request::output_result(VFile* body_ |
| VDate* vdate=0; | VDate* vdate=0; |
| if(Value* v=body_file->fields().get("mdate")) { | if(Value* v=body_file->fields().get("mdate")) { |
| if(Value* vdatep=v->as(VDATE_TYPE, false)) | if(Value* vdatep=v->as(VDATE_TYPE)) |
| vdate=static_cast<VDate*>(vdatep); | vdate=static_cast<VDate*>(vdatep); |
| else | else |
| throw Exception(PARSER_RUNTIME, 0, "mdate must be a date"); | throw Exception(PARSER_RUNTIME, 0, "mdate must be a date"); |
| Line 895 void Request::output_result(VFile* body_ | Line 900 void Request::output_result(VFile* body_ |
| *vdate, | *vdate, |
| info.add_last_modified); | info.add_last_modified); |
| } else { | } else { |
| if(body_file_content_type) | |
| if(HashStringValue *hash=body_file_content_type->get_hash()) | |
| body_file_content_type=hash->get(value_name); | |
| output_sole_piece(*this, header_only, | output_sole_piece(*this, header_only, |
| *body_file, body_file_content_type); | *body_file, body_file_content_type); |
| } | } |
| Line 906 const String& Request::mime_type_of(cons | Line 915 const String& Request::mime_type_of(cons |
| String sext(++cext); | String sext(++cext); |
| Table::Action_options options; | Table::Action_options options; |
| if(mime_types->locate(0, sext.change_case(charsets.source(), String::CC_LOWER), options)) | if(mime_types->locate(0, sext.change_case(charsets.source(), String::CC_LOWER), options)) |
| if(const String* result=mime_types->item(1)) | if(const String* result=mime_types->item(1)) |
| return *result; | return *result; |
| else | else |
| throw Exception(PARSER_RUNTIME, | throw Exception(PARSER_RUNTIME, |
| 0, | 0, |
| Line 917 const String& Request::mime_type_of(cons | Line 926 const String& Request::mime_type_of(cons |
| return *new String("application/octet-stream"); | return *new String("application/octet-stream"); |
| } | } |
| const String* Request::get_used_filename(uint file_no){ | |
| if(file_no < file_list.count()) | |
| return new String(file_list[file_no], String::L_TAINTED); | |
| return 0; | |
| } | |
| #ifdef XML | #ifdef XML |
| xmlChar* Request::transcode(const String& s) { | xmlChar* Request::transcode(const String& s) { |
| return charsets.source().transcode(s); | return charsets.source().transcode(s); |
| Line 963 Request::Exception_details Request::get_ | Line 978 Request::Exception_details Request::get_ |
| // $.source | // $.source |
| if(problem_source) { | if(problem_source) { |
| String& source=*new String; | String& source=*new String(*problem_source, String::L_TAINTED); |
| source.append(*problem_source, String::L_TAINTED, true/*forced*/); | |
| hash.put(exception_source_part_name, new VString(source)); | hash.put(exception_source_part_name, new VString(source)); |
| } | } |
| Line 981 Request::Exception_details Request::get_ | Line 994 Request::Exception_details Request::get_ |
| // $.comment | // $.comment |
| if(const char* comment=e.comment(true)) | if(const char* comment=e.comment(true)) |
| hash.put(exception_comment_part_name, | hash.put(exception_comment_part_name, |
| new VString(*new String(comment, true/*tainted*/))); | new VString(*new String(comment, String::L_TAINTED))); |
| // $.handled(0) | // $.handled(0) |
| hash.put(exception_handled_part_name, &VBool::get(false)); | hash.put(exception_handled_part_name, &VBool::get(false)); |