Handling bad incoming request (keep a check assertion to help during debugging period)

This commit is contained in:
Jocelyn Fiat
2011-12-12 10:44:50 +01:00
parent 59505ccdc4
commit 8b4f774bab
3 changed files with 54 additions and 20 deletions

View File

@@ -26,6 +26,7 @@ feature {NONE} -- Initialization
reset
do
has_error := False
create method.make_empty
create uri.make_empty
create request_header.make_empty
@@ -48,7 +49,14 @@ feature -- Execution
end
analyze_request_message (client_socket)
if has_error then
check catch_bad_incoming_connection: False end
if is_verbose then
log ("ERROR: invalid HTTP incoming request")
end
else
process_request (Current, client_socket)
end
reset
end
@@ -57,6 +65,7 @@ feature -- Request processing
process_request (a_handler: HTTP_CONNECTION_HANDLER; a_socket: TCP_STREAM_SOCKET)
-- Process request ...
require
no_error: not has_error
a_handler_attached: a_handler /= Void
a_uri_attached: a_handler.uri /= Void
a_method_attached: a_handler.method /= Void
@@ -74,6 +83,9 @@ feature -- Access
request_header_map : HASH_TABLE [STRING,STRING]
-- Contains key:value of the header
has_error: BOOLEAN
-- Error occurred during `analyze_request_message'
method: STRING
-- http verb
@@ -85,10 +97,12 @@ feature -- Access
--| unused for now
remote_info: detachable TUPLE [addr: STRING; hostname: STRING; port: INTEGER]
-- Information related to remote client
feature -- Parsing
analyze_request_message (a_socket: TCP_STREAM_SOCKET)
-- Analyze message extracted from `a_socket' as HTTP request
require
input_readable: a_socket /= Void and then a_socket.is_open_read
local
@@ -97,23 +111,31 @@ feature -- Parsing
line : detachable STRING
k, val: STRING
txt: STRING
l_is_verbose: BOOLEAN
do
create txt.make (64)
line := next_line (a_socket)
if line /= Void then
analyze_request_line (line)
txt.append (line)
txt.append_character ('%N')
request_header := txt
if attached next_line (a_socket) as l_request_line and then not l_request_line.is_empty then
txt.append (l_request_line)
txt.append_character ('%N')
analyze_request_line (l_request_line)
else
has_error := True
end
l_is_verbose := is_verbose
if not has_error or l_is_verbose then
-- if `is_verbose' we can try to print the request, even if it is a bad HTTP request
from
line := next_line (a_socket)
until
line = Void or end_of_stream
loop
n := line.count
if is_verbose then
print ("%N" + line)
if l_is_verbose then
log (line)
end
pos := line.index_of (':',1)
if pos > 0 then
@@ -139,26 +161,26 @@ feature -- Parsing
end
analyze_request_line (line: STRING)
-- Analyze `line' as a HTTP request line
require
line /= Void
valid_line: line /= Void and then not line.is_empty
local
pos, next_pos: INTEGER
do
if is_verbose then
print ("%N## Parse HTTP request line ##")
print ("%N")
print (line)
log ("%N## Parse HTTP request line ##")
log (line)
end
pos := line.index_of (' ', 1)
method := line.substring (1, pos - 1)
next_pos := line.index_of (' ', pos + 1)
uri := line.substring (pos + 1, next_pos - 1)
version := line.substring (next_pos + 1, line.count)
ensure
not_void_method: method /= Void
has_error := method.is_empty
end
next_line (a_socket: TCP_STREAM_SOCKET): detachable STRING
-- Next line fetched from `a_socket' is available.
require
is_readable: a_socket.is_open_read
do

View File

@@ -32,6 +32,7 @@ feature -- Output
-- Log `a_message'
do
io.put_string (a_message)
io.put_new_line
end
feature -- Inherited Features
@@ -50,14 +51,14 @@ feature -- Inherited Features
create l_listening_socket.make_server_by_port (l_http_port)
if not l_listening_socket.is_bound then
if is_verbose then
log ("Socket could not be bound on port " + l_http_port.out )
log ("Socket could not be bound on port " + l_http_port.out)
end
else
l_http_port := l_listening_socket.port
from
l_listening_socket.listen (max_tcp_clients)
if is_verbose then
log ("%NHTTP Connection Server ready on port " + l_http_port.out +" : http://localhost:" + l_http_port.out + "/%N")
log ("%NHTTP Connection Server ready on port " + l_http_port.out +" : http://localhost:" + l_http_port.out + "/")
end
on_launched (l_http_port)
until
@@ -102,12 +103,15 @@ feature -- Inherited Features
-- Process incoming connection
do
if is_verbose then
log ("Incoming connection...%N")
log ("Incoming connection...(socket:" + a_socket.descriptor.out + ")")
end
--| FIXME jfiat [2011/11/03] : should use a Pool of Threads/Handler to process this connection
--| also handle permanent connection...?
receive_message_and_send_reply (a_socket)
a_socket.cleanup
if is_verbose then
log ("connection completed...")
end
ensure
socket_closed: a_socket.is_closed
end

View File

@@ -22,8 +22,8 @@ feature -- Initialization
a_http_handler_valid: a_http_handler /= Void
do
if configuration.is_verbose then
print("%N%N%N")
print ("Starting Web Application Server (port="+ configuration.http_server_port.out +"):%N")
log ("%N%N%N")
log ("Starting Web Application Server (port="+ configuration.http_server_port.out +"):%N")
end
stop_requested := False
a_http_handler.execute
@@ -42,6 +42,14 @@ feature -- Access
stop_requested: BOOLEAN
-- Stops the server
feature -- Output
log (a_message: READABLE_STRING_8)
-- Log `a_message'
do
io.put_string (a_message)
end
;note
copyright: "2011-2011, Javier Velilla and others"
license: "Eiffel Forum License v2 (see http://www.eiffel.com/licensing/forum.txt)"