X-Git-Url: https://git.openstreetmap.org./nominatim.git/blobdiff_plain/6e89310a9285f1ad15d8002bf68f578eada367a0..f6e792f31b105447d86c78ffa10ac343a6b5e332:/src/nominatim_db/clicmd/api.py diff --git a/src/nominatim_db/clicmd/api.py b/src/nominatim_db/clicmd/api.py index 20553b40..fac88bdd 100644 --- a/src/nominatim_db/clicmd/api.py +++ b/src/nominatim_db/clicmd/api.py @@ -7,17 +7,19 @@ """ Subcommand definitions for API calls from the command line. """ -from typing import Dict, Any +from typing import Dict, Any, Optional import argparse import logging import json import sys +from functools import reduce import nominatim_api as napi import nominatim_api.v1 as api_output from nominatim_api.v1.helpers import zoom_to_rank, deduplicate_results from nominatim_api.v1.format import dispatch as formatting import nominatim_api.logging as loglib +from ..errors import UsageError from .args import NominatimArgs # Do not repeat documentation of subcommand classes. @@ -61,6 +63,48 @@ def _add_api_output_arguments(parser: argparse.ArgumentParser) -> None: "Parameter is difference tolerance in degrees.")) +def _get_geometry_output(args: NominatimArgs) -> napi.GeometryFormat: + """ Get the requested geometry output format in a API-compatible + format. + """ + if not args.polygon_output: + return napi.GeometryFormat.NONE + if args.polygon_output == 'geojson': + return napi.GeometryFormat.GEOJSON + if args.polygon_output == 'kml': + return napi.GeometryFormat.KML + if args.polygon_output == 'svg': + return napi.GeometryFormat.SVG + if args.polygon_output == 'text': + return napi.GeometryFormat.TEXT + + try: + return napi.GeometryFormat[args.polygon_output.upper()] + except KeyError as exp: + raise UsageError(f"Unknown polygon output format '{args.polygon_output}'.") from exp + + +def _get_locales(args: NominatimArgs, default: Optional[str]) -> napi.Locales: + """ Get the locales from the language parameter. + """ + if args.lang: + return napi.Locales.from_accept_languages(args.lang) + if default: + return napi.Locales.from_accept_languages(default) + + return napi.Locales() + + +def _get_layers(args: NominatimArgs, default: napi.DataLayer) -> Optional[napi.DataLayer]: + """ Get the list of selected layers as a DataLayer enum. + """ + if not args.layers: + return default + + return reduce(napi.DataLayer.__or__, + (napi.DataLayer[s.upper()] for s in args.layers)) + + class APISearch: """\ Execute a search query. @@ -105,13 +149,13 @@ class APISearch: params: Dict[str, Any] = {'max_results': args.limit + min(args.limit, 10), 'address_details': True, # needed for display name - 'geometry_output': args.get_geometry_output(), + 'geometry_output': _get_geometry_output(args), 'geometry_simplification': args.polygon_threshold, 'countries': args.countrycodes, 'excluded': args.exclude_place_ids, 'viewbox': args.viewbox, 'bounded_viewbox': args.bounded, - 'locales': args.get_locales(api.config.DEFAULT_LANGUAGE) + 'locales': _get_locales(args, api.config.DEFAULT_LANGUAGE) } if args.query: @@ -183,11 +227,11 @@ class APIReverse: result = api.reverse(napi.Point(args.lon, args.lat), max_rank=zoom_to_rank(args.zoom or 18), - layers=args.get_layers(napi.DataLayer.ADDRESS | napi.DataLayer.POI), + layers=_get_layers(args, napi.DataLayer.ADDRESS | napi.DataLayer.POI), address_details=True, # needed for display name - geometry_output=args.get_geometry_output(), + geometry_output=_get_geometry_output(args), geometry_simplification=args.polygon_threshold, - locales=args.get_locales(api.config.DEFAULT_LANGUAGE)) + locales=_get_locales(args, api.config.DEFAULT_LANGUAGE)) if args.format == 'debug': print(loglib.get_and_disable()) @@ -247,9 +291,9 @@ class APILookup: results = api.lookup(places, address_details=True, # needed for display name - geometry_output=args.get_geometry_output(), + geometry_output=_get_geometry_output(args), geometry_simplification=args.polygon_threshold or 0.0, - locales=args.get_locales(api.config.DEFAULT_LANGUAGE)) + locales=_get_locales(args, api.config.DEFAULT_LANGUAGE)) output = api_output.format_result( results, @@ -323,7 +367,7 @@ class APIDetails: api = napi.NominatimAPI(args.project_dir) - locales = args.get_locales(api.config.DEFAULT_LANGUAGE) + locales = _get_locales(args, api.config.DEFAULT_LANGUAGE) result = api.details(place, address_details=args.addressdetails, linked_places=args.linkedplaces,