mirror of
https://github.com/osm-search/Nominatim.git
synced 2026-03-09 11:34:07 +00:00
add simple reverse API tests
This commit is contained in:
130
test/bdd/api/reverse/simple.feature
Normal file
130
test/bdd/api/reverse/simple.feature
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
@APIDB
|
||||||
|
Feature: Simple Reverse Tests
|
||||||
|
Simple tests for internal server errors and response format.
|
||||||
|
|
||||||
|
Scenario Outline: Simple reverse-geocoding
|
||||||
|
When sending reverse coordinates <lat>,<lon>
|
||||||
|
Then the result is valid xml
|
||||||
|
When sending xml reverse coordinates <lat>,<lon>
|
||||||
|
Then the result is valid xml
|
||||||
|
When sending json reverse coordinates <lat>,<lon>
|
||||||
|
Then the result is valid json
|
||||||
|
When sending jsonv2 reverse coordinates <lat>,<lon>
|
||||||
|
Then the result is valid json
|
||||||
|
When sending html reverse coordinates <lat>,<lon>
|
||||||
|
Then the result is valid html
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| lat | lon |
|
||||||
|
| 0.0 | 0.0 |
|
||||||
|
| -34.830 | -56.105 |
|
||||||
|
| 45.174 | -103.072 |
|
||||||
|
| 21.156 | -12.2744 |
|
||||||
|
|
||||||
|
Scenario Outline: Testing different parameters
|
||||||
|
When sending reverse coordinates 53.603,10.041
|
||||||
|
| param | value |
|
||||||
|
| <parameter> | <value> |
|
||||||
|
Then the result is valid xml
|
||||||
|
When sending html reverse coordinates 53.603,10.041
|
||||||
|
| param | value |
|
||||||
|
| <parameter> | <value> |
|
||||||
|
Then the result is valid html
|
||||||
|
When sending xml reverse coordinates 53.603,10.041
|
||||||
|
| param | value |
|
||||||
|
| <parameter> | <value> |
|
||||||
|
Then the result is valid xml
|
||||||
|
When sending json reverse coordinates 53.603,10.041
|
||||||
|
| param | value |
|
||||||
|
| <parameter> | <value> |
|
||||||
|
Then the result is valid json
|
||||||
|
When sending jsonv2 reverse coordinates 53.603,10.041
|
||||||
|
| param | value |
|
||||||
|
| <parameter> | <value> |
|
||||||
|
Then the result is valid json
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| parameter | value |
|
||||||
|
| polygon | 1 |
|
||||||
|
| polygon | 0 |
|
||||||
|
| polygon_text | 1 |
|
||||||
|
| polygon_text | 0 |
|
||||||
|
| polygon_kml | 1 |
|
||||||
|
| polygon_kml | 0 |
|
||||||
|
| polygon_geojson | 1 |
|
||||||
|
| polygon_geojson | 0 |
|
||||||
|
| polygon_svg | 1 |
|
||||||
|
| polygon_svg | 0 |
|
||||||
|
|
||||||
|
Scenario Outline: Wrapping of legal jsonp requests
|
||||||
|
When sending <format> reverse coordinates 67.3245,0.456
|
||||||
|
| json_callback |
|
||||||
|
| foo |
|
||||||
|
Then the result is valid json
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| format |
|
||||||
|
| json |
|
||||||
|
| jsonv2 |
|
||||||
|
|
||||||
|
Scenario Outline: Reverse-geocoding without address
|
||||||
|
When sending <format> reverse coordinates 53.603,10.041
|
||||||
|
| addressdetails |
|
||||||
|
| 0 |
|
||||||
|
Then exactly 1 result is returned
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| format |
|
||||||
|
| json |
|
||||||
|
| jsonv2 |
|
||||||
|
| html |
|
||||||
|
| xml |
|
||||||
|
|
||||||
|
Scenario Outline: Reverse-geocoding with zoom
|
||||||
|
When sending <format> reverse coordinates 53.603,10.041
|
||||||
|
| zoom |
|
||||||
|
| 10 |
|
||||||
|
Then exactly 1 result is returned
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| format |
|
||||||
|
| json |
|
||||||
|
| jsonv2 |
|
||||||
|
| html |
|
||||||
|
| xml |
|
||||||
|
|
||||||
|
Scenario: Missing lon parameter
|
||||||
|
When sending reverse coordinates 52.52,
|
||||||
|
Then a HTTP 400 is returned
|
||||||
|
|
||||||
|
Scenario: Missing lat parameter
|
||||||
|
When sending reverse coordinates ,52.52
|
||||||
|
Then a HTTP 400 is returned
|
||||||
|
|
||||||
|
Scenario: Missing osm_id parameter
|
||||||
|
When sending reverse coordinates ,
|
||||||
|
| osm_type |
|
||||||
|
| N |
|
||||||
|
Then a HTTP 400 is returned
|
||||||
|
|
||||||
|
Scenario: Missing osm_type parameter
|
||||||
|
When sending reverse coordinates ,
|
||||||
|
| osm_id |
|
||||||
|
| 3498564 |
|
||||||
|
Then a HTTP 400 is returned
|
||||||
|
|
||||||
|
Scenario Outline: Bad format for lat or lon
|
||||||
|
When sending reverse coordinates ,
|
||||||
|
| lat | lon |
|
||||||
|
| <lat> | <lon> |
|
||||||
|
Then a HTTP 400 is returned
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
| lat | lon |
|
||||||
|
| 48.9660 | 8,4482 |
|
||||||
|
| 48,9660 | 8.4482 |
|
||||||
|
| 48,9660 | 8,4482 |
|
||||||
|
| 48.966.0 | 8.4482 |
|
||||||
|
| 48.966 | 8.448.2 |
|
||||||
|
| Nan | 8.448 |
|
||||||
|
| 48.966 | Nan |
|
||||||
@@ -148,6 +148,71 @@ class SearchResponse(object):
|
|||||||
return [ x[prop] for x in self.result ]
|
return [ x[prop] for x in self.result ]
|
||||||
|
|
||||||
|
|
||||||
|
class ReverseResponse(object):
|
||||||
|
|
||||||
|
def __init__(self, page, fmt='json', errorcode=200):
|
||||||
|
self.page = page
|
||||||
|
self.format = fmt
|
||||||
|
self.errorcode = errorcode
|
||||||
|
self.result = []
|
||||||
|
self.header = dict()
|
||||||
|
|
||||||
|
if errorcode == 200:
|
||||||
|
getattr(self, 'parse_' + fmt)()
|
||||||
|
|
||||||
|
def parse_html(self):
|
||||||
|
content, errors = tidy_document(self.page,
|
||||||
|
options={'char-encoding' : 'utf8'})
|
||||||
|
#eq_(len(errors), 0 , "Errors found in HTML document:\n%s" % errors)
|
||||||
|
|
||||||
|
b = content.find('nominatim_results =')
|
||||||
|
e = content.find('</script>')
|
||||||
|
content = content[b:e]
|
||||||
|
b = content.find('[')
|
||||||
|
e = content.rfind(']')
|
||||||
|
|
||||||
|
self.result = json.JSONDecoder(object_pairs_hook=OrderedDict).decode(content[b:e+1])
|
||||||
|
|
||||||
|
def parse_json(self):
|
||||||
|
m = re.fullmatch(r'([\w$][^(]*)\((.*)\)', self.page)
|
||||||
|
if m is None:
|
||||||
|
code = self.page
|
||||||
|
else:
|
||||||
|
code = m.group(2)
|
||||||
|
self.header['json_func'] = m.group(1)
|
||||||
|
self.result = [json.JSONDecoder(object_pairs_hook=OrderedDict).decode(code)]
|
||||||
|
|
||||||
|
def parse_xml(self):
|
||||||
|
et = ET.fromstring(self.page)
|
||||||
|
|
||||||
|
self.header = dict(et.attrib)
|
||||||
|
self.result = []
|
||||||
|
|
||||||
|
for child in et:
|
||||||
|
if child.tag == 'result':
|
||||||
|
eq_(0, len(self.result), "More than one result in reverse result")
|
||||||
|
self.result.append(dict(child.attrib))
|
||||||
|
elif child.tag == 'addressparts':
|
||||||
|
address = {}
|
||||||
|
for sub in child:
|
||||||
|
address[sub.tag] = sub.text
|
||||||
|
self.result[0]['address'] = address
|
||||||
|
elif child.tag == 'extratags':
|
||||||
|
self.result[0]['extratags'] = {}
|
||||||
|
for tag in child:
|
||||||
|
self.result[0]['extratags'][tag.attrib['key']] = tag.attrib['value']
|
||||||
|
elif child.tag == 'namedetails':
|
||||||
|
self.result[0]['namedetails'] = {}
|
||||||
|
for tag in child:
|
||||||
|
self.result[0]['namedetails'][tag.attrib['desc']] = tag.text
|
||||||
|
elif child.tag in ('geokml'):
|
||||||
|
self.result[0][child.tag] = True
|
||||||
|
else:
|
||||||
|
assert child.tag == 'error', \
|
||||||
|
"Unknown XML tag %s on page: %s" % (child.tag, self.page)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@when(u'searching for "(?P<query>.*)"(?P<dups> with dups)?')
|
@when(u'searching for "(?P<query>.*)"(?P<dups> with dups)?')
|
||||||
def query_cmd(context, query, dups):
|
def query_cmd(context, query, dups):
|
||||||
""" Query directly via PHP script.
|
""" Query directly via PHP script.
|
||||||
@@ -172,18 +237,9 @@ def query_cmd(context, query, dups):
|
|||||||
|
|
||||||
context.response = SearchResponse(outp.decode('utf-8'), 'json')
|
context.response = SearchResponse(outp.decode('utf-8'), 'json')
|
||||||
|
|
||||||
|
def send_api_query(endpoint, params, fmt, context):
|
||||||
@when(u'sending (?P<fmt>\S+ )?search query "(?P<query>.*)"(?P<addr> with address)?')
|
|
||||||
def website_search_request(context, fmt, query, addr):
|
|
||||||
env = BASE_SERVER_ENV
|
|
||||||
|
|
||||||
params = {}
|
|
||||||
if query:
|
|
||||||
params['q'] = query
|
|
||||||
if fmt is not None:
|
if fmt is not None:
|
||||||
params['format'] = fmt.strip()
|
params['format'] = fmt.strip()
|
||||||
if addr is not None:
|
|
||||||
params['addressdetails'] = '1'
|
|
||||||
if context.table:
|
if context.table:
|
||||||
if context.table.headings[0] == 'param':
|
if context.table.headings[0] == 'param':
|
||||||
for line in context.table:
|
for line in context.table:
|
||||||
@@ -191,15 +247,18 @@ def website_search_request(context, fmt, query, addr):
|
|||||||
else:
|
else:
|
||||||
for h in context.table.headings:
|
for h in context.table.headings:
|
||||||
params[h] = context.table[0][h]
|
params[h] = context.table[0][h]
|
||||||
|
|
||||||
|
env = BASE_SERVER_ENV
|
||||||
env['QUERY_STRING'] = urlencode(params)
|
env['QUERY_STRING'] = urlencode(params)
|
||||||
|
|
||||||
env['REQUEST_URI'] = '/search.php?' + env['QUERY_STRING']
|
env['SCRIPT_NAME'] = '/%s.php' % endpoint
|
||||||
env['SCRIPT_NAME'] = '/search.php'
|
env['REQUEST_URI'] = '%s?%s' % (env['SCRIPT_NAME'], env['QUERY_STRING'])
|
||||||
env['CONTEXT_DOCUMENT_ROOT'] = os.path.join(context.nominatim.build_dir, 'website')
|
env['CONTEXT_DOCUMENT_ROOT'] = os.path.join(context.nominatim.build_dir, 'website')
|
||||||
env['SCRIPT_FILENAME'] = os.path.join(context.nominatim.build_dir, 'website', 'search.php')
|
env['SCRIPT_FILENAME'] = os.path.join(env['CONTEXT_DOCUMENT_ROOT'],
|
||||||
|
'%s.php' % endpoint)
|
||||||
env['NOMINATIM_SETTINGS'] = context.nominatim.local_settings_file
|
env['NOMINATIM_SETTINGS'] = context.nominatim.local_settings_file
|
||||||
|
|
||||||
cmd = [ '/usr/bin/php-cgi', env['SCRIPT_FILENAME']]
|
cmd = ['/usr/bin/php-cgi', env['SCRIPT_FILENAME']]
|
||||||
for k,v in params.items():
|
for k,v in params.items():
|
||||||
cmd.append("%s=%s" % (k, v))
|
cmd.append("%s=%s" % (k, v))
|
||||||
|
|
||||||
@@ -221,7 +280,20 @@ def website_search_request(context, fmt, query, addr):
|
|||||||
status = 200
|
status = 200
|
||||||
|
|
||||||
content_start = outp.find('\r\n\r\n')
|
content_start = outp.find('\r\n\r\n')
|
||||||
assert_less(11, content_start)
|
|
||||||
|
return outp[content_start + 4:], status
|
||||||
|
|
||||||
|
|
||||||
|
@when(u'sending (?P<fmt>\S+ )?search query "(?P<query>.*)"(?P<addr> with address)?')
|
||||||
|
def website_search_request(context, fmt, query, addr):
|
||||||
|
|
||||||
|
params = {}
|
||||||
|
if query:
|
||||||
|
params['q'] = query
|
||||||
|
if addr is not None:
|
||||||
|
params['addressdetails'] = '1'
|
||||||
|
|
||||||
|
outp, status = send_api_query('search', params, fmt, context)
|
||||||
|
|
||||||
if fmt is None:
|
if fmt is None:
|
||||||
outfmt = 'html'
|
outfmt = 'html'
|
||||||
@@ -230,7 +302,27 @@ def website_search_request(context, fmt, query, addr):
|
|||||||
else:
|
else:
|
||||||
outfmt = fmt.strip()
|
outfmt = fmt.strip()
|
||||||
|
|
||||||
context.response = SearchResponse(outp[content_start + 4:], outfmt, status)
|
context.response = SearchResponse(outp, outfmt, status)
|
||||||
|
|
||||||
|
@when(u'sending (?P<fmt>\S+ )?reverse coordinates (?P<lat>[0-9.-]+)?,(?P<lon>[0-9.-]+)?')
|
||||||
|
def website_reverse_request(context, fmt, lat, lon):
|
||||||
|
params = {}
|
||||||
|
if lat is not None:
|
||||||
|
params['lat'] = lat
|
||||||
|
if lon is not None:
|
||||||
|
params['lon'] = lon
|
||||||
|
|
||||||
|
outp, status = send_api_query('reverse', params, fmt, context)
|
||||||
|
|
||||||
|
if fmt is None:
|
||||||
|
outfmt = 'xml'
|
||||||
|
elif fmt == 'jsonv2 ':
|
||||||
|
outfmt = 'json'
|
||||||
|
else:
|
||||||
|
outfmt = fmt.strip()
|
||||||
|
|
||||||
|
context.response = ReverseResponse(outp, outfmt, status)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@step(u'(?P<operator>less than|more than|exactly|at least|at most) (?P<number>\d+) results? (?:is|are) returned')
|
@step(u'(?P<operator>less than|more than|exactly|at least|at most) (?P<number>\d+) results? (?:is|are) returned')
|
||||||
@@ -246,6 +338,7 @@ def check_http_return_status(context, status):
|
|||||||
|
|
||||||
@then(u'the result is valid (?P<fmt>\w+)')
|
@then(u'the result is valid (?P<fmt>\w+)')
|
||||||
def step_impl(context, fmt):
|
def step_impl(context, fmt):
|
||||||
|
context.execute_steps("Then a HTTP 200 is returned")
|
||||||
eq_(context.response.format, fmt)
|
eq_(context.response.format, fmt)
|
||||||
|
|
||||||
@then(u'result header contains')
|
@then(u'result header contains')
|
||||||
|
|||||||
Reference in New Issue
Block a user