4 require_once(dirname(dirname(__FILE__)).'/settings/settings.php');
5 require_once(CONST_BasePath.'/lib/init-cmd.php');
6 ini_set('memory_limit', '800M');
9 "Create and setup nominatim search system",
10 array('help', 'h', 0, 1, 0, 0, false, 'Show Help'),
11 array('quiet', 'q', 0, 1, 0, 0, 'bool', 'Quiet output'),
12 array('verbose', 'v', 0, 1, 0, 0, 'bool', 'Verbose output'),
14 array('osm-file', '', 0, 1, 1, 1, 'realpath', 'File to import'),
15 array('threads', '', 0, 1, 1, 1, 'int', 'Number of threads (where possible)'),
17 array('all', '', 0, 1, 0, 0, 'bool', 'Do the complete process'),
19 array('create-db', '', 0, 1, 0, 0, 'bool', 'Create nominatim db'),
20 array('setup-db', '', 0, 1, 0, 0, 'bool', 'Build a blank nominatim db'),
21 array('import-data', '', 0, 1, 0, 0, 'bool', 'Import a osm file'),
22 array('osm2pgsql-cache', '', 0, 1, 1, 1, 'int', 'Cache size used by osm2pgsql'),
23 array('create-functions', '', 0, 1, 0, 0, 'bool', 'Create functions'),
24 array('enable-diff-updates', '', 0, 1, 0, 0, 'bool', 'Turn on the code required to make diff updates work'),
25 array('enable-debug-statements', '', 0, 1, 0, 0, 'bool', 'Include debug warning statements in pgsql commands'),
26 array('ignore-errors', '', 0, 1, 0, 0, 'bool', 'Continue import even when errors in SQL are present (EXPERT)'),
27 array('create-tables', '', 0, 1, 0, 0, 'bool', 'Create main tables'),
28 array('create-partition-tables', '', 0, 1, 0, 0, 'bool', 'Create required partition tables'),
29 array('create-partition-functions', '', 0, 1, 0, 0, 'bool', 'Create required partition triggers'),
30 array('no-partitions', '', 0, 1, 0, 0, 'bool', "Do not partition search indices (speeds up import of single country extracts)"),
31 array('import-wikipedia-articles', '', 0, 1, 0, 0, 'bool', 'Import wikipedia article dump'),
32 array('load-data', '', 0, 1, 0, 0, 'bool', 'Copy data to live tables from import table'),
33 array('disable-token-precalc', '', 0, 1, 0, 0, 'bool', 'Disable name precalculation (EXPERT)'),
34 array('import-tiger-data', '', 0, 1, 0, 0, 'bool', 'Import tiger data (not included in \'all\')'),
35 array('calculate-postcodes', '', 0, 1, 0, 0, 'bool', 'Calculate postcode centroids'),
36 array('osmosis-init', '', 0, 1, 0, 0, 'bool', 'Generate default osmosis configuration'),
37 array('index', '', 0, 1, 0, 0, 'bool', 'Index the data'),
38 array('index-noanalyse', '', 0, 1, 0, 0, 'bool', 'Do not perform analyse operations during index (EXPERT)'),
39 array('create-search-indices', '', 0, 1, 0, 0, 'bool', 'Create additional indices required for search and update'),
40 array('create-website', '', 0, 1, 1, 1, 'realpath', 'Create symlinks to setup web directory'),
41 array('drop', '', 0, 1, 0, 0, 'bool', 'Drop tables needed for updates, making the database readonly (EXPERIMENTAL)'),
43 getCmdOpt($_SERVER['argv'], $aCMDOptions, $aCMDResult, true, true);
45 $bDidSomething = false;
47 // Check if osm-file is set and points to a valid file if --all or --import-data is given
48 if ($aCMDResult['import-data'] || $aCMDResult['all'])
50 if (!isset($aCMDResult['osm-file']))
52 fail('missing --osm-file for data import');
55 if (!file_exists($aCMDResult['osm-file']))
57 fail('the path supplied to --osm-file does not exist');
60 if (!is_readable($aCMDResult['osm-file']))
62 fail('osm-file "'.$aCMDResult['osm-file'].'" not readable');
67 // This is a pretty hard core default - the number of processors in the box - 1
68 $iInstances = isset($aCMDResult['threads'])?$aCMDResult['threads']:(getProcessorCount()-1);
72 echo "WARNING: resetting threads to $iInstances\n";
74 if ($iInstances > getProcessorCount())
76 $iInstances = getProcessorCount();
77 echo "WARNING: resetting threads to $iInstances\n";
80 // Assume we can steal all the cache memory in the box (unless told otherwise)
81 if (isset($aCMDResult['osm2pgsql-cache']))
83 $iCacheMemory = $aCMDResult['osm2pgsql-cache'];
87 $iCacheMemory = getCacheMemoryMB();
90 $aDSNInfo = DB::parseDSN(CONST_Database_DSN);
91 if (!isset($aDSNInfo['port']) || !$aDSNInfo['port']) $aDSNInfo['port'] = 5432;
93 if ($aCMDResult['create-db'] || $aCMDResult['all'])
96 $bDidSomething = true;
97 $oDB =& DB::connect(CONST_Database_DSN, false);
98 if (!PEAR::isError($oDB))
100 fail('database already exists ('.CONST_Database_DSN.')');
102 passthruCheckReturn('createdb -E UTF-8 -p '.$aDSNInfo['port'].' '.$aDSNInfo['database']);
105 if ($aCMDResult['setup-db'] || $aCMDResult['all'])
108 $bDidSomething = true;
109 // TODO: path detection, detection memory, etc.
113 $fPostgresVersion = getPostgresVersion($oDB);
114 echo 'Postgres version found: '.$fPostgresVersion."\n";
116 if ($fPostgresVersion < 9.1)
118 fail("Minimum supported version of Postgresql is 9.1.");
121 pgsqlRunScript('CREATE EXTENSION IF NOT EXISTS hstore');
122 pgsqlRunScript('CREATE EXTENSION IF NOT EXISTS postgis');
124 // For extratags and namedetails the hstore_to_json converter is
125 // needed which is only available from Postgresql 9.3+. For older
126 // versions add a dummy function that returns nothing.
127 $iNumFunc = $oDB->getOne("select count(*) from pg_proc where proname = 'hstore_to_json'");
128 if (PEAR::isError($iNumFunc))
130 fail("Cannot query stored procedures.", $iNumFunc);
134 pgsqlRunScript("create function hstore_to_json(dummy hstore) returns text AS 'select null::text' language sql immutable");
135 echo "WARNING: Postgresql is too old. extratags and namedetails API not available.";
138 $fPostgisVersion = getPostgisVersion($oDB);
139 echo 'Postgis version found: '.$fPostgisVersion."\n";
141 if ($fPostgisVersion < 2.1)
143 // Function was renamed in 2.1 and throws an annoying deprecation warning
144 pgsqlRunScript('ALTER FUNCTION st_line_interpolate_point(geometry, double precision) RENAME TO ST_LineInterpolatePoint');
147 pgsqlRunScriptFile(CONST_BasePath.'/data/country_name.sql');
148 pgsqlRunScriptFile(CONST_BasePath.'/data/country_naturalearthdata.sql');
149 pgsqlRunScriptFile(CONST_BasePath.'/data/country_osm_grid.sql');
150 pgsqlRunScriptFile(CONST_BasePath.'/data/gb_postcode_table.sql');
151 if (file_exists(CONST_BasePath.'/data/gb_postcode_data.sql.gz'))
153 pgsqlRunScriptFile(CONST_BasePath.'/data/gb_postcode_data.sql.gz');
157 echo "WARNING: external UK postcode table not found.\n";
159 if (CONST_Use_Extra_US_Postcodes)
161 pgsqlRunScriptFile(CONST_BasePath.'/data/us_postcode.sql');
164 if ($aCMDResult['no-partitions'])
166 pgsqlRunScript('update country_name set partition = 0');
169 // the following will be needed by create_functions later but
170 // is only defined in the subsequently called create_tables.
171 // Create dummies here that will be overwritten by the proper
172 // versions in create-tables.
173 pgsqlRunScript('CREATE TABLE place_boundingbox ()');
174 pgsqlRunScript('create type wikipedia_article_match as ()');
177 if ($aCMDResult['import-data'] || $aCMDResult['all'])
180 $bDidSomething = true;
182 $osm2pgsql = CONST_Osm2pgsql_Binary;
183 if (!file_exists($osm2pgsql))
185 echo "Please download and build osm2pgsql.\nIf it is already installed, check the path in your local settings (settings/local.php) file.\n";
186 fail("osm2pgsql not found in '$osm2pgsql'");
189 if (!is_null(CONST_Osm2pgsql_Flatnode_File))
191 $osm2pgsql .= ' --flat-nodes '.CONST_Osm2pgsql_Flatnode_File;
193 if (CONST_Tablespace_Osm2pgsql_Data)
194 $osm2pgsql .= ' --tablespace-slim-data '.CONST_Tablespace_Osm2pgsql_Data;
195 if (CONST_Tablespace_Osm2pgsql_Index)
196 $osm2pgsql .= ' --tablespace-slim-index '.CONST_Tablespace_Osm2pgsql_Index;
197 if (CONST_Tablespace_Place_Data)
198 $osm2pgsql .= ' --tablespace-main-data '.CONST_Tablespace_Place_Data;
199 if (CONST_Tablespace_Place_Index)
200 $osm2pgsql .= ' --tablespace-main-index '.CONST_Tablespace_Place_Index;
201 $osm2pgsql .= ' -lsc -O gazetteer --hstore --number-processes 1';
202 $osm2pgsql .= ' -C '.$iCacheMemory;
203 $osm2pgsql .= ' -P '.$aDSNInfo['port'];
204 $osm2pgsql .= ' -d '.$aDSNInfo['database'].' '.$aCMDResult['osm-file'];
205 passthruCheckReturn($osm2pgsql);
208 $x = $oDB->getRow('select * from place limit 1');
209 if (PEAR::isError($x)) {
210 fail($x->getMessage());
212 if (!$x) fail('No Data');
215 if ($aCMDResult['create-functions'] || $aCMDResult['all'])
218 $bDidSomething = true;
219 if (!file_exists(CONST_InstallPath.'/module/nominatim.so')) fail("nominatim module not built");
220 $sTemplate = file_get_contents(CONST_BasePath.'/sql/functions.sql');
221 $sTemplate = str_replace('{modulepath}', CONST_InstallPath.'/module', $sTemplate);
222 if ($aCMDResult['enable-diff-updates'])
224 $sTemplate = str_replace('RETURN NEW; -- %DIFFUPDATES%', '--', $sTemplate);
226 if ($aCMDResult['enable-debug-statements'])
228 $sTemplate = str_replace('--DEBUG:', '', $sTemplate);
230 if (CONST_Limit_Reindexing)
232 $sTemplate = str_replace('--LIMIT INDEXING:', '', $sTemplate);
234 if (!CONST_Use_US_Tiger_Data)
236 $sTemplate = str_replace('-- %NOTIGERDATA% ', '', $sTemplate);
238 if (!CONST_Use_Aux_Location_data)
240 $sTemplate = str_replace('-- %NOAUXDATA% ', '', $sTemplate);
242 pgsqlRunScript($sTemplate);
245 if ($aCMDResult['create-tables'] || $aCMDResult['all'])
247 $bDidSomething = true;
250 $sTemplate = file_get_contents(CONST_BasePath.'/sql/tables.sql');
251 $sTemplate = str_replace('{www-user}', CONST_Database_Web_User, $sTemplate);
252 $sTemplate = replace_tablespace('{ts:address-data}',
253 CONST_Tablespace_Address_Data, $sTemplate);
254 $sTemplate = replace_tablespace('{ts:address-index}',
255 CONST_Tablespace_Address_Index, $sTemplate);
256 $sTemplate = replace_tablespace('{ts:search-data}',
257 CONST_Tablespace_Search_Data, $sTemplate);
258 $sTemplate = replace_tablespace('{ts:search-index}',
259 CONST_Tablespace_Search_Index, $sTemplate);
260 $sTemplate = replace_tablespace('{ts:aux-data}',
261 CONST_Tablespace_Aux_Data, $sTemplate);
262 $sTemplate = replace_tablespace('{ts:aux-index}',
263 CONST_Tablespace_Aux_Index, $sTemplate);
264 pgsqlRunScript($sTemplate, false);
266 // re-run the functions
268 $sTemplate = file_get_contents(CONST_BasePath.'/sql/functions.sql');
269 $sTemplate = str_replace('{modulepath}',
270 CONST_InstallPath.'/module', $sTemplate);
271 pgsqlRunScript($sTemplate);
274 if ($aCMDResult['create-partition-tables'] || $aCMDResult['all'])
276 echo "Partition Tables\n";
277 $bDidSomething = true;
279 $sSQL = 'select distinct partition from country_name';
280 $aPartitions = $oDB->getCol($sSQL);
281 if (PEAR::isError($aPartitions))
283 fail($aPartitions->getMessage());
285 if (!$aCMDResult['no-partitions']) $aPartitions[] = 0;
287 $sTemplate = file_get_contents(CONST_BasePath.'/sql/partition-tables.src.sql');
288 $sTemplate = replace_tablespace('{ts:address-data}',
289 CONST_Tablespace_Address_Data, $sTemplate);
290 $sTemplate = replace_tablespace('{ts:address-index}',
291 CONST_Tablespace_Address_Index, $sTemplate);
292 $sTemplate = replace_tablespace('{ts:search-data}',
293 CONST_Tablespace_Search_Data, $sTemplate);
294 $sTemplate = replace_tablespace('{ts:search-index}',
295 CONST_Tablespace_Search_Index, $sTemplate);
296 $sTemplate = replace_tablespace('{ts:aux-data}',
297 CONST_Tablespace_Aux_Data, $sTemplate);
298 $sTemplate = replace_tablespace('{ts:aux-index}',
299 CONST_Tablespace_Aux_Index, $sTemplate);
300 preg_match_all('#^-- start(.*?)^-- end#ms', $sTemplate, $aMatches, PREG_SET_ORDER);
301 foreach($aMatches as $aMatch)
304 foreach($aPartitions as $sPartitionName)
306 $sResult .= str_replace('-partition-', $sPartitionName, $aMatch[1]);
308 $sTemplate = str_replace($aMatch[0], $sResult, $sTemplate);
311 pgsqlRunScript($sTemplate);
315 if ($aCMDResult['create-partition-functions'] || $aCMDResult['all'])
317 echo "Partition Functions\n";
318 $bDidSomething = true;
320 $sSQL = 'select distinct partition from country_name';
321 $aPartitions = $oDB->getCol($sSQL);
322 if (PEAR::isError($aPartitions))
324 fail($aPartitions->getMessage());
326 if (!$aCMDResult['no-partitions']) $aPartitions[] = 0;
328 $sTemplate = file_get_contents(CONST_BasePath.'/sql/partition-functions.src.sql');
329 preg_match_all('#^-- start(.*?)^-- end#ms', $sTemplate, $aMatches, PREG_SET_ORDER);
330 foreach($aMatches as $aMatch)
333 foreach($aPartitions as $sPartitionName)
335 $sResult .= str_replace('-partition-', $sPartitionName, $aMatch[1]);
337 $sTemplate = str_replace($aMatch[0], $sResult, $sTemplate);
340 pgsqlRunScript($sTemplate);
343 if ($aCMDResult['import-wikipedia-articles'] || $aCMDResult['all'])
345 $bDidSomething = true;
346 $sWikiArticlesFile = CONST_BasePath.'/data/wikipedia_article.sql.bin';
347 $sWikiRedirectsFile = CONST_BasePath.'/data/wikipedia_redirect.sql.bin';
348 if (file_exists($sWikiArticlesFile))
350 echo "Importing wikipedia articles...";
351 pgsqlRunDropAndRestore($sWikiArticlesFile);
356 echo "WARNING: wikipedia article dump file not found - places will have default importance\n";
358 if (file_exists($sWikiRedirectsFile))
360 echo "Importing wikipedia redirects...";
361 pgsqlRunDropAndRestore($sWikiRedirectsFile);
366 echo "WARNING: wikipedia redirect dump file not found - some place importance values may be missing\n";
371 if ($aCMDResult['load-data'] || $aCMDResult['all'])
373 echo "Drop old Data\n";
374 $bDidSomething = true;
377 if (!pg_query($oDB->connection, 'TRUNCATE word')) fail(pg_last_error($oDB->connection));
379 if (!pg_query($oDB->connection, 'TRUNCATE placex')) fail(pg_last_error($oDB->connection));
381 if (!pg_query($oDB->connection, 'TRUNCATE place_addressline')) fail(pg_last_error($oDB->connection));
383 if (!pg_query($oDB->connection, 'TRUNCATE place_boundingbox')) fail(pg_last_error($oDB->connection));
385 if (!pg_query($oDB->connection, 'TRUNCATE location_area')) fail(pg_last_error($oDB->connection));
387 if (!pg_query($oDB->connection, 'TRUNCATE search_name')) fail(pg_last_error($oDB->connection));
389 if (!pg_query($oDB->connection, 'TRUNCATE search_name_blank')) fail(pg_last_error($oDB->connection));
391 if (!pg_query($oDB->connection, 'DROP SEQUENCE seq_place')) fail(pg_last_error($oDB->connection));
393 if (!pg_query($oDB->connection, 'CREATE SEQUENCE seq_place start 100000')) fail(pg_last_error($oDB->connection));
396 $sSQL = 'select distinct partition from country_name';
397 $aPartitions = $oDB->getCol($sSQL);
398 if (PEAR::isError($aPartitions))
400 fail($aPartitions->getMessage());
402 if (!$aCMDResult['no-partitions']) $aPartitions[] = 0;
403 foreach($aPartitions as $sPartition)
405 if (!pg_query($oDB->connection, 'TRUNCATE location_road_'.$sPartition)) fail(pg_last_error($oDB->connection));
409 // used by getorcreate_word_id to ignore frequent partial words
410 if (!pg_query($oDB->connection, 'CREATE OR REPLACE FUNCTION get_maxwordfreq() RETURNS integer AS $$ SELECT '.CONST_Max_Word_Frequency.' as maxwordfreq; $$ LANGUAGE SQL IMMUTABLE')) fail(pg_last_error($oDB->connection));
413 // pre-create the word list
414 if (!$aCMDResult['disable-token-precalc'])
416 echo "Loading word list\n";
417 pgsqlRunScriptFile(CONST_BasePath.'/data/words.sql');
421 $aDBInstances = array();
422 for($i = 0; $i < $iInstances; $i++)
424 $aDBInstances[$i] =& getDB(true);
425 $sSQL = 'insert into placex (osm_type, osm_id, class, type, name, admin_level, ';
426 $sSQL .= 'housenumber, street, addr_place, isin, postcode, country_code, extratags, ';
427 $sSQL .= 'geometry) select * from place where osm_id % '.$iInstances.' = '.$i;
428 if ($aCMDResult['verbose']) echo "$sSQL\n";
429 if (!pg_send_query($aDBInstances[$i]->connection, $sSQL)) fail(pg_last_error($oDB->connection));
435 for($i = 0; $i < $iInstances; $i++)
437 if (pg_connection_busy($aDBInstances[$i]->connection)) $bAnyBusy = true;
443 echo "Reanalysing database...\n";
444 pgsqlRunScript('ANALYSE');
447 if ($aCMDResult['import-tiger-data'])
449 $bDidSomething = true;
451 $sTemplate = file_get_contents(CONST_BasePath.'/sql/tiger_import_start.sql');
452 $sTemplate = str_replace('{www-user}', CONST_Database_Web_User, $sTemplate);
453 $sTemplate = replace_tablespace('{ts:aux-data}',
454 CONST_Tablespace_Aux_Data, $sTemplate);
455 $sTemplate = replace_tablespace('{ts:aux-index}',
456 CONST_Tablespace_Aux_Index, $sTemplate);
457 pgsqlRunScript($sTemplate, false);
459 $aDBInstances = array();
460 for($i = 0; $i < $iInstances; $i++)
462 $aDBInstances[$i] =& getDB(true);
465 foreach(glob(CONST_Tiger_Data_Path.'/*.sql') as $sFile)
468 $hFile = fopen($sFile, "r");
469 $sSQL = fgets($hFile, 100000);
474 for($i = 0; $i < $iInstances; $i++)
476 if (!pg_connection_busy($aDBInstances[$i]->connection))
478 while(pg_get_result($aDBInstances[$i]->connection));
479 $sSQL = fgets($hFile, 100000);
481 if (!pg_send_query($aDBInstances[$i]->connection, $sSQL)) fail(pg_last_error($oDB->connection));
499 for($i = 0; $i < $iInstances; $i++)
501 if (pg_connection_busy($aDBInstances[$i]->connection)) $bAnyBusy = true;
508 echo "Creating indexes\n";
509 $sTemplate = file_get_contents(CONST_BasePath.'/sql/tiger_import_finish.sql');
510 $sTemplate = str_replace('{www-user}', CONST_Database_Web_User, $sTemplate);
511 $sTemplate = replace_tablespace('{ts:aux-data}',
512 CONST_Tablespace_Aux_Data, $sTemplate);
513 $sTemplate = replace_tablespace('{ts:aux-index}',
514 CONST_Tablespace_Aux_Index, $sTemplate);
515 pgsqlRunScript($sTemplate, false);
518 if ($aCMDResult['calculate-postcodes'] || $aCMDResult['all'])
520 $bDidSomething = true;
522 if (!pg_query($oDB->connection, 'DELETE from placex where osm_type=\'P\'')) fail(pg_last_error($oDB->connection));
523 $sSQL = "insert into placex (osm_type,osm_id,class,type,postcode,calculated_country_code,geometry) ";
524 $sSQL .= "select 'P',nextval('seq_postcodes'),'place','postcode',postcode,calculated_country_code,";
525 $sSQL .= "ST_SetSRID(ST_Point(x,y),4326) as geometry from (select calculated_country_code,postcode,";
526 $sSQL .= "avg(st_x(st_centroid(geometry))) as x,avg(st_y(st_centroid(geometry))) as y ";
527 $sSQL .= "from placex where postcode is not null group by calculated_country_code,postcode) as x";
528 if (!pg_query($oDB->connection, $sSQL)) fail(pg_last_error($oDB->connection));
530 if (CONST_Use_Extra_US_Postcodes)
532 $sSQL = "insert into placex (osm_type,osm_id,class,type,postcode,calculated_country_code,geometry) ";
533 $sSQL .= "select 'P',nextval('seq_postcodes'),'place','postcode',postcode,'us',";
534 $sSQL .= "ST_SetSRID(ST_Point(x,y),4326) as geometry from us_postcode";
535 if (!pg_query($oDB->connection, $sSQL)) fail(pg_last_error($oDB->connection));
539 if ($aCMDResult['osmosis-init'] || ($aCMDResult['all'] && !$aCMDResult['drop'])) // no use doing osmosis-init when dropping update tables
541 $bDidSomething = true;
544 if (!file_exists(CONST_Osmosis_Binary))
546 echo "Please download osmosis.\nIf it is already installed, check the path in your local settings (settings/local.php) file.\n";
547 if (!$aCMDResult['all'])
549 fail("osmosis not found in '".CONST_Osmosis_Binary."'");
554 if (file_exists(CONST_InstallPath.'/settings/configuration.txt'))
556 echo "settings/configuration.txt already exists\n";
560 passthru(CONST_Osmosis_Binary.' --read-replication-interval-init '.CONST_InstallPath.'/settings');
561 // update osmosis configuration.txt with our settings
562 passthru("sed -i 's!baseUrl=.*!baseUrl=".CONST_Replication_Url."!' ".CONST_InstallPath.'/settings/configuration.txt');
563 passthru("sed -i 's:maxInterval = .*:maxInterval = ".CONST_Replication_MaxInterval.":' ".CONST_InstallPath.'/settings/configuration.txt');
566 // Find the last node in the DB
567 $iLastOSMID = $oDB->getOne("select max(osm_id) from place where osm_type = 'N'");
569 // Lookup the timestamp that node was created (less 3 hours for margin for changsets to be closed)
570 $sLastNodeURL = 'http://www.openstreetmap.org/api/0.6/node/'.$iLastOSMID."/1";
571 $sLastNodeXML = file_get_contents($sLastNodeURL);
572 preg_match('#timestamp="(([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})Z)"#', $sLastNodeXML, $aLastNodeDate);
573 $iLastNodeTimestamp = strtotime($aLastNodeDate[1]) - (3*60*60);
575 // Search for the correct state file - uses file timestamps so need to sort by date descending
576 $sRepURL = CONST_Replication_Url."/";
577 $sRep = file_get_contents($sRepURL."?C=M;O=D;F=1");
578 // download.geofabrik.de: <a href="000/">000/</a></td><td align="right">26-Feb-2013 11:53 </td>
579 // planet.openstreetmap.org: <a href="273/">273/</a> 2013-03-11 07:41 -
580 preg_match_all('#<a href="[0-9]{3}/">([0-9]{3}/)</a>\s*([-0-9a-zA-Z]+ [0-9]{2}:[0-9]{2})#', $sRep, $aRepMatches, PREG_SET_ORDER);
583 $aPrevRepMatch = false;
584 foreach($aRepMatches as $aRepMatch)
586 if (strtotime($aRepMatch[2]) < $iLastNodeTimestamp) break;
587 $aPrevRepMatch = $aRepMatch;
589 if ($aPrevRepMatch) $aRepMatch = $aPrevRepMatch;
591 $sRepURL .= $aRepMatch[1];
592 $sRep = file_get_contents($sRepURL."?C=M;O=D;F=1");
593 preg_match_all('#<a href="[0-9]{3}/">([0-9]{3}/)</a>\s*([-0-9a-zA-Z]+ [0-9]{2}:[0-9]{2})#', $sRep, $aRepMatches, PREG_SET_ORDER);
594 $aPrevRepMatch = false;
595 foreach($aRepMatches as $aRepMatch)
597 if (strtotime($aRepMatch[2]) < $iLastNodeTimestamp) break;
598 $aPrevRepMatch = $aRepMatch;
600 if ($aPrevRepMatch) $aRepMatch = $aPrevRepMatch;
602 $sRepURL .= $aRepMatch[1];
603 $sRep = file_get_contents($sRepURL."?C=M;O=D;F=1");
604 preg_match_all('#<a href="[0-9]{3}.state.txt">([0-9]{3}).state.txt</a>\s*([-0-9a-zA-Z]+ [0-9]{2}:[0-9]{2})#', $sRep, $aRepMatches, PREG_SET_ORDER);
605 $aPrevRepMatch = false;
606 foreach($aRepMatches as $aRepMatch)
608 if (strtotime($aRepMatch[2]) < $iLastNodeTimestamp) break;
609 $aPrevRepMatch = $aRepMatch;
611 if ($aPrevRepMatch) $aRepMatch = $aPrevRepMatch;
613 $sRepURL .= $aRepMatch[1].'.state.txt';
614 echo "Getting state file: $sRepURL\n";
615 $sStateFile = file_get_contents($sRepURL);
616 if (!$sStateFile || strlen($sStateFile) > 1000) fail("unable to obtain state file");
617 file_put_contents(CONST_InstallPath.'/settings/state.txt', $sStateFile);
618 echo "Updating DB status\n";
619 pg_query($oDB->connection, 'TRUNCATE import_status');
620 $sSQL = "INSERT INTO import_status VALUES('".$aRepMatch[2]."')";
621 pg_query($oDB->connection, $sSQL);
625 if (!$aCMDResult['all'])
627 fail("Cannot read state file directory.");
633 if ($aCMDResult['index'] || $aCMDResult['all'])
635 $bDidSomething = true;
637 $sBaseCmd = CONST_InstallPath.'/nominatim/nominatim -i -d '.$aDSNInfo['database'].' -P '.$aDSNInfo['port'].' -t '.$iInstances.$sOutputFile;
638 passthruCheckReturn($sBaseCmd.' -R 4');
639 if (!$aCMDResult['index-noanalyse']) pgsqlRunScript('ANALYSE');
640 passthruCheckReturn($sBaseCmd.' -r 5 -R 25');
641 if (!$aCMDResult['index-noanalyse']) pgsqlRunScript('ANALYSE');
642 passthruCheckReturn($sBaseCmd.' -r 26');
645 if ($aCMDResult['create-search-indices'] || $aCMDResult['all'])
647 echo "Search indices\n";
648 $bDidSomething = true;
650 $sSQL = 'select distinct partition from country_name';
651 $aPartitions = $oDB->getCol($sSQL);
652 if (PEAR::isError($aPartitions))
654 fail($aPartitions->getMessage());
656 if (!$aCMDResult['no-partitions']) $aPartitions[] = 0;
658 $sTemplate = file_get_contents(CONST_BasePath.'/sql/indices.src.sql');
659 $sTemplate = replace_tablespace('{ts:address-index}',
660 CONST_Tablespace_Address_Index, $sTemplate);
661 $sTemplate = replace_tablespace('{ts:search-index}',
662 CONST_Tablespace_Search_Index, $sTemplate);
663 $sTemplate = replace_tablespace('{ts:aux-index}',
664 CONST_Tablespace_Aux_Index, $sTemplate);
665 preg_match_all('#^-- start(.*?)^-- end#ms', $sTemplate, $aMatches, PREG_SET_ORDER);
666 foreach($aMatches as $aMatch)
669 foreach($aPartitions as $sPartitionName)
671 $sResult .= str_replace('-partition-', $sPartitionName, $aMatch[1]);
673 $sTemplate = str_replace($aMatch[0], $sResult, $sTemplate);
676 pgsqlRunScript($sTemplate);
679 if (isset($aCMDResult['create-website']))
681 $bDidSomething = true;
682 $sTargetDir = $aCMDResult['create-website'];
683 if (!is_dir($sTargetDir))
685 echo "You must create the website directory before calling this function.\n";
686 fail("Target directory does not exist.");
689 @symlink(CONST_InstallPath.'/website/details.php', $sTargetDir.'/details.php');
690 @symlink(CONST_InstallPath.'/website/reverse.php', $sTargetDir.'/reverse.php');
691 @symlink(CONST_InstallPath.'/website/search.php', $sTargetDir.'/search.php');
692 @symlink(CONST_InstallPath.'/website/search.php', $sTargetDir.'/index.php');
693 @symlink(CONST_InstallPath.'/website/lookup.php', $sTargetDir.'/lookup.php');
694 @symlink(CONST_InstallPath.'/website/deletable.php', $sTargetDir.'/deletable.php');
695 @symlink(CONST_InstallPath.'/website/polygons.php', $sTargetDir.'/polygons.php');
696 @symlink(CONST_InstallPath.'/website/status.php', $sTargetDir.'/status.php');
697 @symlink(CONST_BasePath.'/website/images', $sTargetDir.'/images');
698 @symlink(CONST_BasePath.'/website/js', $sTargetDir.'/js');
699 @symlink(CONST_BasePath.'/website/css', $sTargetDir.'/css');
700 echo "Symlinks created\n";
702 $sTestFile = @file_get_contents(CONST_Website_BaseURL.'js/tiles.js');
705 echo "\nWARNING: Unable to access the website at ".CONST_Website_BaseURL."\n";
706 echo "You may want to update settings/local.php with @define('CONST_Website_BaseURL', 'http://[HOST]/[PATH]/');\n";
710 if ($aCMDResult['drop'])
712 // The implementation is potentially a bit dangerous because it uses
713 // a positive selection of tables to keep, and deletes everything else.
714 // Including any tables that the unsuspecting user might have manually
715 // created. USE AT YOUR OWN PERIL.
716 $bDidSomething = true;
718 // tables we want to keep. everything else goes.
719 $aKeepTables = array(
724 "location_property*",
738 $aDropTables = array();
739 $aHaveTables = $oDB->getCol("SELECT tablename FROM pg_tables WHERE schemaname='public'");
740 if (PEAR::isError($aHaveTables))
742 fail($aPartitions->getMessage());
744 foreach($aHaveTables as $sTable)
747 foreach ($aKeepTables as $sKeep)
749 if (fnmatch($sKeep, $sTable))
755 if (!$bFound) array_push($aDropTables, $sTable);
758 foreach ($aDropTables as $sDrop)
760 if ($aCMDResult['verbose']) echo "dropping table $sDrop\n";
761 @pg_query($oDB->connection, "DROP TABLE $sDrop CASCADE");
762 // ignore warnings/errors as they might be caused by a table having
763 // been deleted already by CASCADE
766 if (!is_null(CONST_Osm2pgsql_Flatnode_File))
768 if ($aCMDResult['verbose']) echo "deleting ".CONST_Osm2pgsql_Flatnode_File."\n";
769 unlink(CONST_Osm2pgsql_Flatnode_File);
775 showUsage($aCMDOptions, true);
779 echo "Setup finished.\n";
782 function pgsqlRunScriptFile($sFilename)
784 if (!file_exists($sFilename)) fail('unable to find '.$sFilename);
786 // Convert database DSN to psql parameters
787 $aDSNInfo = DB::parseDSN(CONST_Database_DSN);
788 if (!isset($aDSNInfo['port']) || !$aDSNInfo['port']) $aDSNInfo['port'] = 5432;
789 $sCMD = 'psql -p '.$aDSNInfo['port'].' -d '.$aDSNInfo['database'];
792 if (preg_match('/\\.gz$/', $sFilename))
794 $aDescriptors = array(
795 0 => array('pipe', 'r'),
796 1 => array('pipe', 'w'),
797 2 => array('file', '/dev/null', 'a')
799 $hGzipProcess = proc_open('zcat '.$sFilename, $aDescriptors, $ahGzipPipes);
800 if (!is_resource($hGzipProcess)) fail('unable to start zcat');
801 $aReadPipe = $ahGzipPipes[1];
802 fclose($ahGzipPipes[0]);
806 $sCMD .= ' -f '.$sFilename;
807 $aReadPipe = array('pipe', 'r');
810 $aDescriptors = array(
812 1 => array('pipe', 'w'),
813 2 => array('file', '/dev/null', 'a')
816 $hProcess = proc_open($sCMD, $aDescriptors, $ahPipes);
817 if (!is_resource($hProcess)) fail('unable to start pgsql');
820 // TODO: error checking
821 while(!feof($ahPipes[1]))
823 echo fread($ahPipes[1], 4096);
827 $iReturn = proc_close($hProcess);
830 fail("pgsql returned with error code ($iReturn)");
834 fclose($ahGzipPipes[1]);
835 proc_close($hGzipProcess);
840 function pgsqlRunScript($sScript, $bfatal = true)
843 // Convert database DSN to psql parameters
844 $aDSNInfo = DB::parseDSN(CONST_Database_DSN);
845 if (!isset($aDSNInfo['port']) || !$aDSNInfo['port']) $aDSNInfo['port'] = 5432;
846 $sCMD = 'psql -p '.$aDSNInfo['port'].' -d '.$aDSNInfo['database'];
847 if ($bfatal && !$aCMDResult['ignore-errors'])
848 $sCMD .= ' -v ON_ERROR_STOP=1';
849 $aDescriptors = array(
850 0 => array('pipe', 'r'),
855 $hProcess = @proc_open($sCMD, $aDescriptors, $ahPipes);
856 if (!is_resource($hProcess)) fail('unable to start pgsql');
858 while(strlen($sScript))
860 $written = fwrite($ahPipes[0], $sScript);
861 if ($written <= 0) break;
862 $sScript = substr($sScript, $written);
865 $iReturn = proc_close($hProcess);
866 if ($bfatal && $iReturn > 0)
868 fail("pgsql returned with error code ($iReturn)");
872 function pgsqlRunRestoreData($sDumpFile)
874 // Convert database DSN to psql parameters
875 $aDSNInfo = DB::parseDSN(CONST_Database_DSN);
876 if (!isset($aDSNInfo['port']) || !$aDSNInfo['port']) $aDSNInfo['port'] = 5432;
877 $sCMD = 'pg_restore -p '.$aDSNInfo['port'].' -d '.$aDSNInfo['database'].' -Fc -a '.$sDumpFile;
879 $aDescriptors = array(
880 0 => array('pipe', 'r'),
881 1 => array('pipe', 'w'),
882 2 => array('file', '/dev/null', 'a')
885 $hProcess = proc_open($sCMD, $aDescriptors, $ahPipes);
886 if (!is_resource($hProcess)) fail('unable to start pg_restore');
890 // TODO: error checking
891 while(!feof($ahPipes[1]))
893 echo fread($ahPipes[1], 4096);
897 $iReturn = proc_close($hProcess);
900 function pgsqlRunDropAndRestore($sDumpFile)
902 // Convert database DSN to psql parameters
903 $aDSNInfo = DB::parseDSN(CONST_Database_DSN);
904 if (!isset($aDSNInfo['port']) || !$aDSNInfo['port']) $aDSNInfo['port'] = 5432;
905 $sCMD = 'pg_restore -p '.$aDSNInfo['port'].' -d '.$aDSNInfo['database'].' -Fc --clean '.$sDumpFile;
907 $aDescriptors = array(
908 0 => array('pipe', 'r'),
909 1 => array('pipe', 'w'),
910 2 => array('file', '/dev/null', 'a')
913 $hProcess = proc_open($sCMD, $aDescriptors, $ahPipes);
914 if (!is_resource($hProcess)) fail('unable to start pg_restore');
918 // TODO: error checking
919 while(!feof($ahPipes[1]))
921 echo fread($ahPipes[1], 4096);
925 $iReturn = proc_close($hProcess);
928 function passthruCheckReturn($cmd)
931 passthru($cmd, $result);
932 if ($result != 0) fail('Error executing external command: '.$cmd);
935 function replace_tablespace($sTemplate, $sTablespace, $sSql)
938 $sSql = str_replace($sTemplate, 'TABLESPACE "'.$sTablespace.'"',
941 $sSql = str_replace($sTemplate, '', $sSql);