| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- #!/usr/bin/env python
- #
- # Copyright 2018 Espressif Systems (Shanghai) PTE LTD
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- from __future__ import print_function, unicode_literals
- import argparse
- import http.client
- from builtins import str
- from tiny_test_fw import Utility
- def verbose_print(verbosity, *args):
- if (verbosity):
- Utility.console_log(''.join(str(elems) for elems in args))
- def test_val(text, expected, received):
- if expected != received:
- Utility.console_log(' Fail!')
- Utility.console_log(' [reason] ' + text + ':')
- Utility.console_log(' expected: ' + str(expected))
- Utility.console_log(' received: ' + str(received))
- return False
- return True
- def test_get_handler(ip, port, verbosity=False):
- verbose_print(verbosity, '======== GET HANDLER TEST =============')
- # Establish HTTP connection
- verbose_print(verbosity, 'Connecting to => ' + ip + ':' + port)
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- uri = '/hello?query1=value1&query2=value2&query3=value3'
- # GET hello response
- test_headers = {'Test-Header-1':'Test-Value-1', 'Test-Header-2':'Test-Value-2'}
- verbose_print(verbosity, 'Sending GET to URI : ', uri)
- verbose_print(verbosity, 'Sending additional headers : ')
- for k, v in test_headers.items():
- verbose_print(verbosity, '\t', k, ': ', v)
- sess.request('GET', url=uri, headers=test_headers)
- resp = sess.getresponse()
- resp_hdrs = resp.getheaders()
- resp_data = resp.read().decode()
- # Close HTTP connection
- sess.close()
- if not (
- test_val('Status code mismatch', 200, resp.status) and
- test_val('Response mismatch', 'Custom-Value-1', resp.getheader('Custom-Header-1')) and
- test_val('Response mismatch', 'Custom-Value-2', resp.getheader('Custom-Header-2')) and
- test_val('Response mismatch', 'Hello World!', resp_data)
- ):
- return False
- verbose_print(verbosity, 'vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv')
- verbose_print(verbosity, 'Server response to GET /hello')
- verbose_print(verbosity, 'Response Headers : ')
- for k, v in resp_hdrs:
- verbose_print(verbosity, '\t', k, ': ', v)
- verbose_print(verbosity, 'Response Data : ' + resp_data)
- verbose_print(verbosity, '========================================\n')
- return True
- def test_post_handler(ip, port, msg, verbosity=False):
- verbose_print(verbosity, '======== POST HANDLER TEST ============')
- # Establish HTTP connection
- verbose_print(verbosity, 'Connecting to => ' + ip + ':' + port)
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- # POST message to /echo and get back response
- sess.request('POST', url='/echo', body=msg)
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- verbose_print(verbosity, 'Server response to POST /echo (' + msg + ')')
- verbose_print(verbosity, 'vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv')
- verbose_print(verbosity, resp_data)
- verbose_print(verbosity, '========================================\n')
- # Close HTTP connection
- sess.close()
- return test_val('Response mismatch', msg, resp_data)
- def test_put_handler(ip, port, verbosity=False):
- verbose_print(verbosity, '======== PUT HANDLER TEST =============')
- # Establish HTTP connection
- verbose_print(verbosity, 'Connecting to => ' + ip + ':' + port)
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- # PUT message to /ctrl to disable /hello and /echo URI handlers
- # and set 404 error handler to custom http_404_error_handler()
- verbose_print(verbosity, 'Disabling /hello and /echo handlers')
- sess.request('PUT', url='/ctrl', body='0')
- resp = sess.getresponse()
- resp.read()
- try:
- # Send HTTP request to /hello URI
- sess.request('GET', url='/hello')
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- # 404 Error must be returned from server as URI /hello is no longer available.
- # But the custom error handler http_404_error_handler() will not close the
- # session if the requested URI is /hello
- if not test_val('Status code mismatch', 404, resp.status):
- raise AssertionError
- # Compare error response string with expectation
- verbose_print(verbosity, 'Response on GET /hello : ' + resp_data)
- if not test_val('Response mismatch', '/hello URI is not available', resp_data):
- raise AssertionError
- # Using same session for sending an HTTP request to /echo, as it is expected
- # that the custom error handler http_404_error_handler() would not have closed
- # the session
- sess.request('POST', url='/echo', body='Some content')
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- # 404 Error must be returned from server as URI /hello is no longer available.
- # The custom error handler http_404_error_handler() will close the session
- # this time as the requested URI is /echo
- if not test_val('Status code mismatch', 404, resp.status):
- raise AssertionError
- # Compare error response string with expectation
- verbose_print(verbosity, 'Response on POST /echo : ' + resp_data)
- if not test_val('Response mismatch', '/echo URI is not available', resp_data):
- raise AssertionError
- try:
- # Using same session should fail as by now the session would have closed
- sess.request('POST', url='/hello', body='Some content')
- resp = sess.getresponse()
- resp.read().decode()
- # If control reaches this point then the socket was not closed.
- # This is not expected
- verbose_print(verbosity, 'Socket not closed by server')
- raise AssertionError
- except http.client.HTTPException:
- # Catch socket error as we tried to communicate with an already closed socket
- pass
- except http.client.HTTPException:
- verbose_print(verbosity, 'Socket closed by server')
- return False
- except AssertionError:
- return False
- finally:
- # Close HTTP connection
- sess.close()
- verbose_print(verbosity, 'Enabling /hello handler')
- # Create new connection
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- # PUT message to /ctrl to enable /hello URI handler
- # and restore 404 error handler to default
- sess.request('PUT', url='/ctrl', body='1')
- resp = sess.getresponse()
- resp.read()
- # Close HTTP connection
- sess.close()
- # Create new connection
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- try:
- # Sending HTTP request to /hello should work now
- sess.request('GET', url='/hello')
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- if not test_val('Status code mismatch', 200, resp.status):
- raise AssertionError
- verbose_print(verbosity, 'Response on GET /hello : ' + resp_data)
- if not test_val('Response mismatch', 'Hello World!', resp_data):
- raise AssertionError
- # 404 Error handler should have been restored to default
- sess.request('GET', url='/invalid')
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- if not test_val('Status code mismatch', 404, resp.status):
- raise AssertionError
- verbose_print(verbosity, 'Response on GET /invalid : ' + resp_data)
- if not test_val('Response mismatch', 'This URI does not exist', resp_data):
- raise AssertionError
- except http.client.HTTPException:
- verbose_print(verbosity, 'Socket closed by server')
- return False
- except AssertionError:
- return False
- finally:
- # Close HTTP connection
- sess.close()
- return True
- def test_custom_uri_query(ip, port, query, verbosity=False):
- verbose_print(verbosity, '======== GET HANDLER TEST =============')
- # Establish HTTP connection
- verbose_print(verbosity, 'Connecting to => ' + ip + ':' + port)
- sess = http.client.HTTPConnection(ip + ':' + port, timeout=15)
- uri = '/hello?' + query
- # GET hello response
- verbose_print(verbosity, 'Sending GET to URI : ', uri)
- sess.request('GET', url=uri, headers={})
- resp = sess.getresponse()
- resp_data = resp.read().decode()
- verbose_print(verbosity, 'vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv')
- verbose_print(verbosity, 'Server response to GET /hello')
- verbose_print(verbosity, 'Response Data : ' + resp_data)
- verbose_print(verbosity, '========================================\n')
- # Close HTTP connection
- sess.close()
- return 'Hello World!' == resp_data
- if __name__ == '__main__':
- # Configure argument parser
- parser = argparse.ArgumentParser(description='Run HTTPd Test')
- parser.add_argument('IP', metavar='IP', type=str, help='Server IP')
- parser.add_argument('port', metavar='port', type=str, help='Server port')
- parser.add_argument('msg', metavar='message', type=str, help='Message to be sent to server')
- args = vars(parser.parse_args())
- # Get arguments
- ip = args['IP']
- port = args['port']
- msg = args['msg']
- if not (
- test_get_handler(ip, port, True) and
- test_put_handler(ip, port, True) and
- test_post_handler(ip, port, msg, True)
- ):
- Utility.console_log('Failed!')
|