]> git.openstreetmap.org Git - nominatim.git/blob - src/nominatim_db/tools/exec_utils.py
make nominatim callable with themepark style
[nominatim.git] / src / nominatim_db / tools / exec_utils.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 Helper functions for executing external programs.
9 """
10 from typing import Any, Mapping, List, Optional
11 import logging
12 import os
13 import re
14 import subprocess
15 import shutil
16
17 from ..db.connection import get_pg_env
18 from ..errors import UsageError
19 from ..version import OSM2PGSQL_REQUIRED_VERSION
20
21 LOG = logging.getLogger()
22
23
24 def run_osm2pgsql(options: Mapping[str, Any]) -> None:
25     """ Run osm2pgsql with the given options.
26     """
27     _check_osm2pgsql_version(options['osm2pgsql'])
28
29     env = get_pg_env(options['dsn'])
30
31     cmd = [_find_osm2pgsql_cmd(options['osm2pgsql']),
32            '--append' if options['append'] else '--create',
33            '--slim',
34            '--log-progress', 'true',
35            '--number-processes', '1' if options['append'] else str(options['threads']),
36            '--cache', str(options['osm2pgsql_cache']),
37            '--style', str(options['osm2pgsql_style'])
38            ]
39
40     if str(options['osm2pgsql_style']).endswith('.lua'):
41         env['LUA_PATH'] = ';'.join((str(options['osm2pgsql_style_path'] / '?.lua'),
42                                     os.environ.get('LUA_PATH', ';')))
43         env['THEMEPARK_PATH'] = str(options['osm2pgsql_style_path'] / 'themes')
44         if 'THEMEPARK_PATH' in os.environ:
45             env['THEMEPARK_PATH'] += ':' + os.environ['THEMEPARK_PATH']
46         cmd.extend(('--output', 'flex'))
47
48         for flavour in ('data', 'index'):
49             if options['tablespaces'][f"main_{flavour}"]:
50                 env[f"NOMINATIM_TABLESPACE_PLACE_{flavour.upper()}"] = \
51                     options['tablespaces'][f"main_{flavour}"]
52     else:
53         cmd.extend(('--output', 'gazetteer', '--hstore', '--latlon'))
54         cmd.extend(_mk_tablespace_options('main', options))
55
56     if options['flatnode_file']:
57         cmd.extend(('--flat-nodes', options['flatnode_file']))
58
59     cmd.extend(_mk_tablespace_options('slim', options))
60
61     if options.get('disable_jit', False):
62         env['PGOPTIONS'] = '-c jit=off -c max_parallel_workers_per_gather=0'
63
64     if 'import_data' in options:
65         cmd.extend(('-r', 'xml', '-'))
66     elif isinstance(options['import_file'], list):
67         for fname in options['import_file']:
68             cmd.append(str(fname))
69     else:
70         cmd.append(str(options['import_file']))
71
72     subprocess.run(cmd, cwd=options.get('cwd', '.'),
73                    input=options.get('import_data'),
74                    env=env, check=True)
75
76
77 def _mk_tablespace_options(ttype: str, options: Mapping[str, Any]) -> List[str]:
78     cmds: List[str] = []
79     for flavour in ('data', 'index'):
80         if options['tablespaces'][f"{ttype}_{flavour}"]:
81             cmds.extend((f"--tablespace-{ttype}-{flavour}",
82                          options['tablespaces'][f"{ttype}_{flavour}"]))
83
84     return cmds
85
86
87 def _find_osm2pgsql_cmd(cmdline: Optional[str]) -> str:
88     if cmdline is not None:
89         return cmdline
90
91     in_path = shutil.which('osm2pgsql')
92     if in_path is None:
93         raise UsageError('osm2pgsql executable not found. Please install osm2pgsql first.')
94
95     return str(in_path)
96
97
98 def _check_osm2pgsql_version(cmdline: Optional[str]) -> None:
99     cmd = [_find_osm2pgsql_cmd(cmdline), '--version']
100
101     result = subprocess.run(cmd, capture_output=True, check=True)
102
103     if not result.stderr:
104         raise UsageError("osm2pgsql does not print version information.")
105
106     verinfo = result.stderr.decode('UTF-8')
107
108     match = re.search(r'osm2pgsql version (\d+)\.(\d+)', verinfo)
109     if match is None:
110         raise UsageError(f"No version information found in output: {verinfo}")
111
112     if (int(match[1]), int(match[2])) < OSM2PGSQL_REQUIRED_VERSION:
113         raise UsageError(f"osm2pgsql is too old. Found version {match[1]}.{match[2]}. "
114                          f"Need at least version {'.'.join(map(str, OSM2PGSQL_REQUIRED_VERSION))}.")