]> git.openstreetmap.org Git - nominatim.git/blob - src/nominatim_api/reverse.py
switch to subtags for tourism=information and natural=water
[nominatim.git] / src / nominatim_api / reverse.py
1 # SPDX-License-Identifier: GPL-3.0-or-later
2 #
3 # This file is part of Nominatim. (https://nominatim.org)
4 #
5 # Copyright (C) 2024 by the Nominatim developer community.
6 # For a full list of authors see the git log.
7 """
8 Implementation of reverse geocoding.
9 """
10 from typing import Optional, List, Callable, Type, Tuple, Dict, Any, cast, Union
11 import functools
12
13 import sqlalchemy as sa
14
15 from .typing import SaColumn, SaSelect, SaFromClause, SaLabel, SaRow, \
16                     SaBind, SaLambdaSelect
17 from .sql.sqlalchemy_types import Geometry
18 from .connection import SearchConnection
19 from . import results as nres
20 from .logging import log
21 from .types import AnyPoint, DataLayer, ReverseDetails, GeometryFormat, Bbox
22
23
24 RowFunc = Callable[[Optional[SaRow], Type[nres.ReverseResult]], Optional[nres.ReverseResult]]
25
26 WKT_PARAM: SaBind = sa.bindparam('wkt', type_=Geometry)
27 MAX_RANK_PARAM: SaBind = sa.bindparam('max_rank')
28
29
30 def no_index(expr: SaColumn) -> SaColumn:
31     """ Wrap the given expression, so that the query planner will
32         refrain from using the expression for index lookup.
33     """
34     return sa.func.coalesce(sa.null(), expr)
35
36
37 def _select_from_placex(t: SaFromClause, use_wkt: bool = True) -> SaSelect:
38     """ Create a select statement with the columns relevant for reverse
39         results.
40     """
41     if not use_wkt:
42         distance = t.c.distance
43         centroid = t.c.centroid
44     else:
45         distance = t.c.geometry.ST_Distance(WKT_PARAM)
46         centroid = sa.case((t.c.geometry.is_line_like(), t.c.geometry.ST_ClosestPoint(WKT_PARAM)),
47                            else_=t.c.centroid).label('centroid')
48
49     return sa.select(t.c.place_id, t.c.osm_type, t.c.osm_id, t.c.name,
50                      t.c.class_, t.c.type,
51                      t.c.address, t.c.extratags,
52                      t.c.housenumber, t.c.postcode, t.c.country_code,
53                      t.c.importance, t.c.wikipedia,
54                      t.c.parent_place_id, t.c.rank_address, t.c.rank_search,
55                      centroid,
56                      t.c.linked_place_id, t.c.admin_level,
57                      distance.label('distance'),
58                      t.c.geometry.ST_Expand(0).label('bbox'))
59
60
61 def _interpolated_housenumber(table: SaFromClause) -> SaLabel:
62     return sa.cast(table.c.startnumber
63                    + sa.func.round(((table.c.endnumber - table.c.startnumber) * table.c.position)
64                                    / table.c.step) * table.c.step,
65                    sa.Integer).label('housenumber')
66
67
68 def _interpolated_position(table: SaFromClause) -> SaLabel:
69     fac = sa.cast(table.c.step, sa.Float) / (table.c.endnumber - table.c.startnumber)
70     rounded_pos = sa.func.round(table.c.position / fac) * fac
71     return sa.case(
72         (table.c.endnumber == table.c.startnumber, table.c.linegeo.ST_Centroid()),
73         else_=table.c.linegeo.ST_LineInterpolatePoint(rounded_pos)).label('centroid')
74
75
76 def _locate_interpolation(table: SaFromClause) -> SaLabel:
77     """ Given a position, locate the closest point on the line.
78     """
79     return sa.case((table.c.linegeo.is_line_like(),
80                     table.c.linegeo.ST_LineLocatePoint(WKT_PARAM)),
81                    else_=0).label('position')
82
83
84 def _get_closest(*rows: Optional[SaRow]) -> Optional[SaRow]:
85     return min(rows, key=lambda row: 1000 if row is None else row.distance)
86
87
88 class ReverseGeocoder:
89     """ Class implementing the logic for looking up a place from a
90         coordinate.
91     """
92
93     def __init__(self, conn: SearchConnection, params: ReverseDetails,
94                  restrict_to_country_areas: bool = False) -> None:
95         self.conn = conn
96         self.params = params
97         self.restrict_to_country_areas = restrict_to_country_areas
98
99         self.bind_params: Dict[str, Any] = {'max_rank': params.max_rank}
100
101     @property
102     def max_rank(self) -> int:
103         """ Return the maximum configured rank.
104         """
105         return self.params.max_rank
106
107     def has_geometries(self) -> bool:
108         """ Check if any geometries are requested.
109         """
110         return bool(self.params.geometry_output)
111
112     def layer_enabled(self, *layer: DataLayer) -> bool:
113         """ Return true when any of the given layer types are requested.
114         """
115         return any(self.params.layers & ly for ly in layer)
116
117     def layer_disabled(self, *layer: DataLayer) -> bool:
118         """ Return true when none of the given layer types is requested.
119         """
120         return not any(self.params.layers & ly for ly in layer)
121
122     def has_feature_layers(self) -> bool:
123         """ Return true if any layer other than ADDRESS or POI is requested.
124         """
125         return self.layer_enabled(DataLayer.RAILWAY, DataLayer.MANMADE, DataLayer.NATURAL)
126
127     def _add_geometry_columns(self, sql: SaLambdaSelect, col: SaColumn) -> SaSelect:
128         out = []
129
130         if self.params.geometry_simplification > 0.0:
131             col = sa.func.ST_SimplifyPreserveTopology(col, self.params.geometry_simplification)
132
133         if self.params.geometry_output & GeometryFormat.GEOJSON:
134             out.append(sa.func.ST_AsGeoJSON(col, 7).label('geometry_geojson'))
135         if self.params.geometry_output & GeometryFormat.TEXT:
136             out.append(sa.func.ST_AsText(col).label('geometry_text'))
137         if self.params.geometry_output & GeometryFormat.KML:
138             out.append(sa.func.ST_AsKML(col, 7).label('geometry_kml'))
139         if self.params.geometry_output & GeometryFormat.SVG:
140             out.append(sa.func.ST_AsSVG(col, 0, 7).label('geometry_svg'))
141
142         return sql.add_columns(*out)
143
144     def _filter_by_layer(self, table: SaFromClause) -> SaColumn:
145         if self.layer_enabled(DataLayer.MANMADE):
146             exclude = []
147             if self.layer_disabled(DataLayer.RAILWAY):
148                 exclude.append('railway')
149             if self.layer_disabled(DataLayer.NATURAL):
150                 exclude.extend(('natural', 'water', 'waterway'))
151             return table.c.class_.not_in(tuple(exclude))
152
153         include = []
154         if self.layer_enabled(DataLayer.RAILWAY):
155             include.append('railway')
156         if self.layer_enabled(DataLayer.NATURAL):
157             include.extend(('natural', 'water', 'waterway'))
158         return table.c.class_.in_(tuple(include))
159
160     async def _find_closest_street_or_poi(self, distance: float) -> Optional[SaRow]:
161         """ Look up the closest rank 26+ place in the database, which
162             is closer than the given distance.
163         """
164         t = self.conn.t.placex
165
166         # PostgreSQL must not get the distance as a parameter because
167         # there is a danger it won't be able to properly estimate index use
168         # when used with prepared statements
169         diststr = sa.text(f"{distance}")
170
171         sql: SaLambdaSelect = sa.lambda_stmt(
172             lambda: _select_from_placex(t)
173             .where(t.c.geometry.within_distance(WKT_PARAM, diststr))
174             .where(t.c.indexed_status == 0)
175             .where(t.c.linked_place_id == None)
176             .where(sa.or_(sa.not_(t.c.geometry.is_area()),
177                           t.c.centroid.ST_Distance(WKT_PARAM) < diststr))
178             .order_by('distance')
179             .limit(2))
180
181         if self.has_geometries():
182             sql = self._add_geometry_columns(sql, t.c.geometry)
183
184         restrict: List[Union[SaColumn, Callable[[], SaColumn]]] = []
185
186         if self.layer_enabled(DataLayer.ADDRESS):
187             max_rank = min(29, self.max_rank)
188             restrict.append(lambda: no_index(t.c.rank_address).between(26, max_rank))
189             if self.max_rank == 30:
190                 restrict.append(lambda: sa.func.IsAddressPoint(t))
191         if self.layer_enabled(DataLayer.POI) and self.max_rank == 30:
192             restrict.append(lambda: sa.and_(no_index(t.c.rank_search) == 30,
193                                             t.c.class_.not_in(('place', 'building')),
194                                             sa.not_(t.c.geometry.is_line_like())))
195         if self.has_feature_layers():
196             restrict.append(sa.and_(no_index(t.c.rank_search).between(26, MAX_RANK_PARAM),
197                                     no_index(t.c.rank_address) == 0,
198                                     self._filter_by_layer(t)))
199
200         if not restrict:
201             return None
202
203         sql = sql.where(sa.or_(*restrict))
204
205         # If the closest object is inside an area, then check if there is a
206         # POI node nearby and return that.
207         prev_row = None
208         for row in await self.conn.execute(sql, self.bind_params):
209             if prev_row is None:
210                 if row.rank_search <= 27 or row.osm_type == 'N' or row.distance > 0:
211                     return row
212                 prev_row = row
213             else:
214                 if row.rank_search > 27 and row.osm_type == 'N'\
215                    and row.distance < 0.0001:
216                     return row
217
218         return prev_row
219
220     async def _find_housenumber_for_street(self, parent_place_id: int) -> Optional[SaRow]:
221         t = self.conn.t.placex
222
223         def _base_query() -> SaSelect:
224             return _select_from_placex(t)\
225                 .where(t.c.geometry.within_distance(WKT_PARAM, 0.001))\
226                 .where(t.c.parent_place_id == parent_place_id)\
227                 .where(sa.func.IsAddressPoint(t))\
228                 .where(t.c.indexed_status == 0)\
229                 .where(t.c.linked_place_id == None)\
230                 .order_by('distance')\
231                 .limit(1)
232
233         sql: SaLambdaSelect
234         if self.has_geometries():
235             sql = self._add_geometry_columns(_base_query(), t.c.geometry)
236         else:
237             sql = sa.lambda_stmt(_base_query)
238
239         return (await self.conn.execute(sql, self.bind_params)).one_or_none()
240
241     async def _find_interpolation_for_street(self, parent_place_id: Optional[int],
242                                              distance: float) -> Optional[SaRow]:
243         t = self.conn.t.osmline
244
245         sql = sa.select(t,
246                         t.c.linegeo.ST_Distance(WKT_PARAM).label('distance'),
247                         _locate_interpolation(t))\
248                 .where(t.c.linegeo.within_distance(WKT_PARAM, distance))\
249                 .where(t.c.startnumber != None)\
250                 .order_by('distance')\
251                 .limit(1)
252
253         if parent_place_id is not None:
254             sql = sql.where(t.c.parent_place_id == parent_place_id)
255
256         inner = sql.subquery('ipol')
257
258         sql = sa.select(inner.c.place_id, inner.c.osm_id,
259                         inner.c.parent_place_id, inner.c.address,
260                         _interpolated_housenumber(inner),
261                         _interpolated_position(inner),
262                         inner.c.postcode, inner.c.country_code,
263                         inner.c.distance)
264
265         if self.has_geometries():
266             sub = sql.subquery('geom')
267             sql = self._add_geometry_columns(sa.select(sub), sub.c.centroid)
268
269         return (await self.conn.execute(sql, self.bind_params)).one_or_none()
270
271     async def _find_tiger_number_for_street(self, parent_place_id: int) -> Optional[SaRow]:
272         t = self.conn.t.tiger
273
274         def _base_query() -> SaSelect:
275             inner = sa.select(t,
276                               t.c.linegeo.ST_Distance(WKT_PARAM).label('distance'),
277                               _locate_interpolation(t))\
278                       .where(t.c.linegeo.within_distance(WKT_PARAM, 0.001))\
279                       .where(t.c.parent_place_id == parent_place_id)\
280                       .order_by('distance')\
281                       .limit(1)\
282                       .subquery('tiger')
283
284             return sa.select(inner.c.place_id,
285                              inner.c.parent_place_id,
286                              _interpolated_housenumber(inner),
287                              _interpolated_position(inner),
288                              inner.c.postcode,
289                              inner.c.distance)
290
291         sql: SaLambdaSelect
292         if self.has_geometries():
293             sub = _base_query().subquery('geom')
294             sql = self._add_geometry_columns(sa.select(sub), sub.c.centroid)
295         else:
296             sql = sa.lambda_stmt(_base_query)
297
298         return (await self.conn.execute(sql, self.bind_params)).one_or_none()
299
300     async def lookup_street_poi(self) -> Tuple[Optional[SaRow], RowFunc]:
301         """ Find a street or POI/address for the given WKT point.
302         """
303         log().section('Reverse lookup on street/address level')
304         distance = 0.006
305         parent_place_id = None
306
307         row = await self._find_closest_street_or_poi(distance)
308         row_func: RowFunc = nres.create_from_placex_row
309         log().var_dump('Result (street/building)', row)
310
311         # If the closest result was a street, but an address was requested,
312         # check for a housenumber nearby which is part of the street.
313         if row is not None:
314             if self.max_rank > 27 \
315                and self.layer_enabled(DataLayer.ADDRESS) \
316                and row.rank_address <= 27:
317                 distance = 0.001
318                 parent_place_id = row.place_id
319                 log().comment('Find housenumber for street')
320                 addr_row = await self._find_housenumber_for_street(parent_place_id)
321                 log().var_dump('Result (street housenumber)', addr_row)
322
323                 if addr_row is not None:
324                     row = addr_row
325                     row_func = nres.create_from_placex_row
326                     distance = addr_row.distance
327                 elif row.country_code == 'us' and parent_place_id is not None:
328                     log().comment('Find TIGER housenumber for street')
329                     addr_row = await self._find_tiger_number_for_street(parent_place_id)
330                     log().var_dump('Result (street Tiger housenumber)', addr_row)
331
332                     if addr_row is not None:
333                         row_func = cast(RowFunc,
334                                         functools.partial(nres.create_from_tiger_row,
335                                                           osm_type=row.osm_type,
336                                                           osm_id=row.osm_id))
337                         row = addr_row
338             else:
339                 distance = row.distance
340
341         # Check for an interpolation that is either closer than our result
342         # or belongs to a close street found.
343         if self.max_rank > 27 and self.layer_enabled(DataLayer.ADDRESS):
344             log().comment('Find interpolation for street')
345             addr_row = await self._find_interpolation_for_street(parent_place_id,
346                                                                  distance)
347             log().var_dump('Result (street interpolation)', addr_row)
348             if addr_row is not None:
349                 row = addr_row
350                 row_func = nres.create_from_osmline_row
351
352         return row, row_func
353
354     async def _lookup_area_address(self) -> Optional[SaRow]:
355         """ Lookup large addressable areas for the given WKT point.
356         """
357         log().comment('Reverse lookup by larger address area features')
358         t = self.conn.t.placex
359
360         def _base_query() -> SaSelect:
361             # The inner SQL brings results in the right order, so that
362             # later only a minimum of results needs to be checked with ST_Contains.
363             inner = sa.select(t, sa.literal(0.0).label('distance'))\
364                       .where(t.c.rank_search.between(5, MAX_RANK_PARAM))\
365                       .where(t.c.rank_address != 5)\
366                       .where(t.c.rank_address != 11)\
367                       .where(t.c.geometry.intersects(WKT_PARAM))\
368                       .where(sa.func.PlacexGeometryReverseLookuppolygon())\
369                       .order_by(sa.desc(t.c.rank_search))\
370                       .limit(50)\
371                       .subquery('area')
372
373             return _select_from_placex(inner, False)\
374                 .where(inner.c.geometry.ST_Contains(WKT_PARAM))\
375                 .order_by(sa.desc(inner.c.rank_search))\
376                 .limit(1)
377
378         sql: SaLambdaSelect = sa.lambda_stmt(_base_query)
379         if self.has_geometries():
380             sql = self._add_geometry_columns(sql, sa.literal_column('area.geometry'))
381
382         address_row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
383         log().var_dump('Result (area)', address_row)
384
385         if address_row is not None and address_row.rank_search < self.max_rank:
386             log().comment('Search for better matching place nodes inside the area')
387
388             address_rank = address_row.rank_search
389             address_id = address_row.place_id
390
391             def _place_inside_area_query() -> SaSelect:
392                 inner = \
393                     sa.select(t, t.c.geometry.ST_Distance(WKT_PARAM).label('distance'))\
394                     .where(t.c.rank_search > address_rank)\
395                     .where(t.c.rank_search <= MAX_RANK_PARAM)\
396                     .where(t.c.indexed_status == 0)\
397                     .where(sa.func.IntersectsReverseDistance(t, WKT_PARAM))\
398                     .order_by(sa.desc(t.c.rank_search))\
399                     .limit(50)\
400                     .subquery('places')
401
402                 touter = t.alias('outer')
403                 return _select_from_placex(inner, False)\
404                     .join(touter, touter.c.geometry.ST_Contains(inner.c.geometry))\
405                     .where(touter.c.place_id == address_id)\
406                     .where(sa.func.IsBelowReverseDistance(inner.c.distance, inner.c.rank_search))\
407                     .order_by(sa.desc(inner.c.rank_search), inner.c.distance)\
408                     .limit(1)
409
410             if self.has_geometries():
411                 sql = self._add_geometry_columns(_place_inside_area_query(),
412                                                  sa.literal_column('places.geometry'))
413             else:
414                 sql = sa.lambda_stmt(_place_inside_area_query)
415
416             place_address_row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
417             log().var_dump('Result (place node)', place_address_row)
418
419             if place_address_row is not None:
420                 return place_address_row
421
422         return address_row
423
424     async def _lookup_area_others(self) -> Optional[SaRow]:
425         t = self.conn.t.placex
426
427         inner = sa.select(t, t.c.geometry.ST_Distance(WKT_PARAM).label('distance'))\
428                   .where(t.c.rank_address == 0)\
429                   .where(t.c.rank_search.between(5, MAX_RANK_PARAM))\
430                   .where(t.c.name != None)\
431                   .where(t.c.indexed_status == 0)\
432                   .where(t.c.linked_place_id == None)\
433                   .where(self._filter_by_layer(t))\
434                   .where(t.c.geometry.intersects(sa.func.ST_Expand(WKT_PARAM, 0.007)))\
435                   .order_by(sa.desc(t.c.rank_search))\
436                   .order_by('distance')\
437                   .limit(50)\
438                   .subquery()
439
440         sql = _select_from_placex(inner, False)\
441             .where(sa.or_(sa.not_(inner.c.geometry.is_area()),
442                           inner.c.geometry.ST_Contains(WKT_PARAM)))\
443             .order_by(sa.desc(inner.c.rank_search), inner.c.distance)\
444             .limit(1)
445
446         if self.has_geometries():
447             sql = self._add_geometry_columns(sql, inner.c.geometry)
448
449         row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
450         log().var_dump('Result (non-address feature)', row)
451
452         return row
453
454     async def lookup_area(self) -> Optional[SaRow]:
455         """ Lookup large areas for the current search.
456         """
457         log().section('Reverse lookup by larger area features')
458
459         if self.layer_enabled(DataLayer.ADDRESS):
460             address_row = await self._lookup_area_address()
461         else:
462             address_row = None
463
464         if self.has_feature_layers():
465             other_row = await self._lookup_area_others()
466         else:
467             other_row = None
468
469         return _get_closest(address_row, other_row)
470
471     async def lookup_country_codes(self) -> List[str]:
472         """ Lookup the country for the current search.
473         """
474         log().section('Reverse lookup by country code')
475         t = self.conn.t.country_grid
476         sql = sa.select(t.c.country_code).distinct()\
477                 .where(t.c.geometry.ST_Contains(WKT_PARAM))
478
479         ccodes = [cast(str, r[0]) for r in await self.conn.execute(sql, self.bind_params)]
480         log().var_dump('Country codes', ccodes)
481         return ccodes
482
483     async def lookup_country(self, ccodes: List[str]) -> Tuple[Optional[SaRow], RowFunc]:
484         """ Lookup the country for the current search.
485         """
486         row_func = nres.create_from_placex_row
487         if not ccodes:
488             ccodes = await self.lookup_country_codes()
489
490         if not ccodes:
491             return None, row_func
492
493         t = self.conn.t.placex
494         if self.max_rank > 4:
495             log().comment('Search for place nodes in country')
496
497             def _base_query() -> SaSelect:
498                 inner = sa.select(t, t.c.geometry.ST_Distance(WKT_PARAM).label('distance'))\
499                           .where(t.c.rank_search > 4)\
500                           .where(t.c.rank_search <= MAX_RANK_PARAM)\
501                           .where(t.c.indexed_status == 0)\
502                           .where(t.c.country_code.in_(ccodes))\
503                           .where(sa.func.IntersectsReverseDistance(t, WKT_PARAM))\
504                           .order_by(sa.desc(t.c.rank_search))\
505                           .limit(50)\
506                           .subquery('area')
507
508                 return _select_from_placex(inner, False)\
509                     .where(sa.func.IsBelowReverseDistance(inner.c.distance, inner.c.rank_search))\
510                     .order_by(sa.desc(inner.c.rank_search), inner.c.distance)\
511                     .limit(1)
512
513             sql: SaLambdaSelect
514             if self.has_geometries():
515                 sql = self._add_geometry_columns(_base_query(),
516                                                  sa.literal_column('area.geometry'))
517             else:
518                 sql = sa.lambda_stmt(_base_query)
519
520             address_row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
521             log().var_dump('Result (addressable place node)', address_row)
522         else:
523             address_row = None
524
525         if address_row is None:
526             # Still nothing, then return a country with the appropriate country code.
527             def _country_base_query() -> SaSelect:
528                 return _select_from_placex(t)\
529                          .where(t.c.country_code.in_(ccodes))\
530                          .where(t.c.rank_address == 4)\
531                          .where(t.c.rank_search == 4)\
532                          .where(t.c.linked_place_id == None)\
533                          .order_by('distance')\
534                          .limit(1)
535
536             if self.has_geometries():
537                 sql = self._add_geometry_columns(_country_base_query(), t.c.geometry)
538             else:
539                 sql = sa.lambda_stmt(_country_base_query)
540
541             address_row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
542
543         if address_row is None:
544             # finally fall back to country table
545             t = self.conn.t.country_name
546             tgrid = self.conn.t.country_grid
547
548             sql = sa.select(tgrid.c.country_code,
549                             tgrid.c.geometry.ST_Centroid().ST_Collect().ST_Centroid()
550                                  .label('centroid'),
551                             tgrid.c.geometry.ST_Collect().ST_Expand(0).label('bbox'))\
552                     .where(tgrid.c.country_code.in_(ccodes))\
553                     .group_by(tgrid.c.country_code)
554
555             sub = sql.subquery('grid')
556             sql = sa.select(t.c.country_code,
557                             t.c.name.merge(t.c.derived_name).label('name'),
558                             sub.c.centroid, sub.c.bbox)\
559                     .join(sub, t.c.country_code == sub.c.country_code)\
560                     .order_by(t.c.country_code)\
561                     .limit(1)
562
563             sql = self._add_geometry_columns(sql, sub.c.centroid)
564
565             address_row = (await self.conn.execute(sql, self.bind_params)).one_or_none()
566             row_func = nres.create_from_country_row
567
568         return address_row, row_func
569
570     async def lookup(self, coord: AnyPoint) -> Optional[nres.ReverseResult]:
571         """ Look up a single coordinate. Returns the place information,
572             if a place was found near the coordinates or None otherwise.
573         """
574         log().function('reverse_lookup', coord=coord, params=self.params)
575
576         self.bind_params['wkt'] = f'POINT({coord[0]} {coord[1]})'
577
578         row: Optional[SaRow] = None
579         row_func: RowFunc = nres.create_from_placex_row
580
581         if self.max_rank >= 26:
582             row, tmp_row_func = await self.lookup_street_poi()
583             if row is not None:
584                 row_func = tmp_row_func
585
586         if row is None:
587             if self.restrict_to_country_areas:
588                 ccodes = await self.lookup_country_codes()
589                 if not ccodes:
590                     return None
591             else:
592                 ccodes = []
593
594             if self.max_rank > 4:
595                 row = await self.lookup_area()
596             if row is None and self.layer_enabled(DataLayer.ADDRESS):
597                 row, row_func = await self.lookup_country(ccodes)
598
599         result = row_func(row, nres.ReverseResult)
600         if result is not None:
601             assert row is not None
602             result.distance = getattr(row,  'distance', 0)
603             if hasattr(row, 'bbox'):
604                 result.bbox = Bbox.from_wkb(row.bbox)
605             await nres.add_result_details(self.conn, [result], self.params)
606
607         return result