mirror of
https://github.com/osm-search/Nominatim.git
synced 2024-11-22 12:06:27 +03:00
change partition from char to int, partition search_name on country
This commit is contained in:
parent
31b205ecf9
commit
bfe56e6a45
File diff suppressed because one or more lines are too long
@ -336,7 +336,7 @@ void nominatim_exportPlace(uint64_t place_id, PGconn * conn, xmlTextWriterPtr wr
|
||||
xmlTextWriterEndElement(writer);
|
||||
}
|
||||
|
||||
if (PQntuples(resAddress)> 0)
|
||||
if (PQntuples(resAddress) > 0)
|
||||
{
|
||||
xmlTextWriterStartElement(writer, BAD_CAST "address");
|
||||
for(i = 0; i < PQntuples(resAddress); i++)
|
||||
|
@ -133,12 +133,11 @@ void nominatim_index(int rank_min, int rank_max, int num_threads, const char *co
|
||||
writer = nominatim_exportXMLStart(structuredoutputfile);
|
||||
}
|
||||
|
||||
fprintf(stderr, "Starting indexing rank (%i > %i ) using %i treads\n", rank_min, rank_max, num_threads);
|
||||
fprintf(stderr, "Starting indexing rank (%i to %i) using %i treads\n", rank_min, rank_max, num_threads);
|
||||
|
||||
for (rank = rank_min; rank <= rank_max; rank++)
|
||||
{
|
||||
printf("Starting rank %d\n", rank);
|
||||
rankStartTime = time(0);
|
||||
rankCountTuples = 0;
|
||||
rankPerSecond = 0;
|
||||
|
||||
@ -175,6 +174,7 @@ void nominatim_index(int rank_min, int rank_max, int num_threads, const char *co
|
||||
rankTotalTuples += PGint64(*((uint64_t *)PQgetvalue(resSectors, iSector, 1)));
|
||||
}
|
||||
|
||||
rankStartTime = time(0);
|
||||
for (iSector = 0; iSector < PQntuples(resSectors); iSector++)
|
||||
{
|
||||
sector = PGint32(*((uint32_t *)PQgetvalue(resSectors, iSector, 0)));
|
||||
@ -271,13 +271,14 @@ void *nominatim_indexThread(void * thread_data_in)
|
||||
{
|
||||
struct index_thread_data * thread_data = (struct index_thread_data * )thread_data_in;
|
||||
|
||||
PGresult * res;
|
||||
PGresult *res;
|
||||
|
||||
const char *paramValues[1];
|
||||
int paramLengths[1];
|
||||
int paramFormats[1];
|
||||
uint32_t paramPlaceID;
|
||||
uint32_t place_id;
|
||||
uint32_t place_id;
|
||||
time_t updateStartTime;
|
||||
|
||||
while(1)
|
||||
{
|
||||
@ -293,7 +294,9 @@ void *nominatim_indexThread(void * thread_data_in)
|
||||
|
||||
pthread_mutex_unlock( thread_data->count_mutex );
|
||||
|
||||
// printf(" Processing place_id %d\n", place_id);
|
||||
if (verbose) printf(" Processing place_id %d\n", place_id);
|
||||
|
||||
updateStartTime = time(0);
|
||||
paramPlaceID = PGint32(place_id);
|
||||
paramValues[0] = (char *)¶mPlaceID;
|
||||
paramLengths[0] = sizeof(paramPlaceID);
|
||||
@ -306,6 +309,7 @@ void *nominatim_indexThread(void * thread_data_in)
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
PQclear(res);
|
||||
if (difftime(time(0), updateStartTime) > 1) printf(" Slow place_id %d\n", place_id);
|
||||
|
||||
if (thread_data->writer)
|
||||
{
|
||||
|
@ -112,14 +112,6 @@ int main(int argc, char *argv[])
|
||||
|
||||
fprintf(stderr, "nominatim SVN version %s\n\n", VERSION);
|
||||
|
||||
if (sizeof(int*) == 4) {
|
||||
fprintf(stderr, "\n!! You are running this on 32bit system, so at most\n");
|
||||
fprintf(stderr, "!! 3GB of RAM can be used. If you encounter unexpected\n");
|
||||
fprintf(stderr, "!! exceptions during import, you should try running in slim\n");
|
||||
fprintf(stderr, "!! mode using parameter -s.\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
while (1) {
|
||||
int c, option_index = 0;
|
||||
static struct option long_options[] = {
|
||||
|
@ -46,7 +46,7 @@ END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
|
||||
CREATE OR REPLACE FUNCTION geometry_sector(place geometry) RETURNS INTEGER
|
||||
CREATE OR REPLACE FUNCTION geometry_sector(partition INTEGER, place geometry) RETURNS INTEGER
|
||||
AS $$
|
||||
DECLARE
|
||||
NEWgeometry geometry;
|
||||
@ -59,7 +59,7 @@ BEGIN
|
||||
RETURN 0;
|
||||
END IF;
|
||||
END IF;
|
||||
RETURN (500-ST_X(ST_Centroid(NEWgeometry))::integer)*1000 + (500-ST_Y(ST_Centroid(NEWgeometry))::integer);
|
||||
RETURN (partition*1000000) + (500-ST_X(ST_Centroid(NEWgeometry))::integer)*1000 + (500-ST_Y(ST_Centroid(NEWgeometry))::integer);
|
||||
END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
@ -85,26 +85,6 @@ END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
|
||||
CREATE OR REPLACE FUNCTION geometry_index(place geometry, indexed BOOLEAN, name HSTORE) RETURNS INTEGER
|
||||
AS $$
|
||||
BEGIN
|
||||
IF indexed THEN RETURN NULL; END IF;
|
||||
IF name is null THEN RETURN NULL; END IF;
|
||||
RETURN geometry_sector(place);
|
||||
END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
|
||||
CREATE OR REPLACE FUNCTION geometry_index(sector integer, indexed BOOLEAN, name HSTORE) RETURNS INTEGER
|
||||
AS $$
|
||||
BEGIN
|
||||
IF indexed THEN RETURN NULL; END IF;
|
||||
IF name is null THEN RETURN NULL; END IF;
|
||||
RETURN sector;
|
||||
END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
|
||||
CREATE OR REPLACE FUNCTION transliteration(text) RETURNS text
|
||||
AS '{modulepath}/nominatim.so', 'transliteration'
|
||||
LANGUAGE c IMMUTABLE STRICT;
|
||||
@ -465,7 +445,7 @@ BEGIN
|
||||
--RAISE WARNING 'start: %', ST_AsText(place_centre);
|
||||
|
||||
-- Try for a OSM polygon first
|
||||
FOR nearcountry IN select country_code from location_area_country where country_code is not null and st_contains(geometry, place_centre) limit 1
|
||||
FOR nearcountry IN select country_code from location_area_country where country_code is not null and not isguess and st_contains(geometry, place_centre) limit 1
|
||||
LOOP
|
||||
RETURN nearcountry.country_code;
|
||||
END LOOP;
|
||||
@ -481,10 +461,16 @@ BEGIN
|
||||
--RAISE WARNING 'natural earth: %', ST_AsText(place_centre);
|
||||
|
||||
-- Natural earth data (first fallback)
|
||||
-- FOR nearcountry IN select country_code from country_naturalearthdata where st_contains(geometry, place_centre) limit 1
|
||||
-- LOOP
|
||||
-- RETURN nearcountry.country_code;
|
||||
-- END LOOP;
|
||||
FOR nearcountry IN select country_code from country_naturalearthdata where st_contains(geometry, place_centre) limit 1
|
||||
LOOP
|
||||
RETURN nearcountry.country_code;
|
||||
END LOOP;
|
||||
|
||||
-- Natural earth data (first fallback)
|
||||
FOR nearcountry IN select country_code from country_naturalearthdata where st_distance(geometry, place_centre) < 0.5 limit 1
|
||||
LOOP
|
||||
RETURN nearcountry.country_code;
|
||||
END LOOP;
|
||||
|
||||
--RAISE WARNING 'in country: %', ST_AsText(place_centre);
|
||||
|
||||
@ -527,7 +513,7 @@ CREATE OR REPLACE FUNCTION get_country_language_code(search_country_code VARCHAR
|
||||
DECLARE
|
||||
nearcountry RECORD;
|
||||
BEGIN
|
||||
FOR nearcountry IN select distinct country_default_language_code from country where country_code = search_country_code limit 1
|
||||
FOR nearcountry IN select distinct country_default_language_code from country_name where country_code = search_country_code limit 1
|
||||
LOOP
|
||||
RETURN lower(nearcountry.country_default_language_code);
|
||||
END LOOP;
|
||||
@ -536,17 +522,17 @@ END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
|
||||
CREATE OR REPLACE FUNCTION get_partition(place geometry, in_country_code VARCHAR(10)) RETURNS TEXT
|
||||
CREATE OR REPLACE FUNCTION get_partition(place geometry, in_country_code VARCHAR(10)) RETURNS INTEGER
|
||||
AS $$
|
||||
DECLARE
|
||||
place_centre GEOMETRY;
|
||||
nearcountry RECORD;
|
||||
BEGIN
|
||||
FOR nearcountry IN select country_code from country_name where country_code = in_country_code
|
||||
FOR nearcountry IN select partition from country_name where country_code = in_country_code
|
||||
LOOP
|
||||
RETURN nearcountry.country_code;
|
||||
RETURN nearcountry.partition;
|
||||
END LOOP;
|
||||
RETURN 'none';
|
||||
RETURN 0;
|
||||
END;
|
||||
$$
|
||||
LANGUAGE plpgsql IMMUTABLE;
|
||||
@ -565,7 +551,7 @@ LANGUAGE plpgsql;
|
||||
CREATE OR REPLACE FUNCTION add_location(
|
||||
place_id INTEGER,
|
||||
country_code varchar(2),
|
||||
partition varchar(10),
|
||||
partition INTEGER,
|
||||
keywords INTEGER[],
|
||||
rank_search INTEGER,
|
||||
rank_address INTEGER,
|
||||
@ -584,6 +570,7 @@ DECLARE
|
||||
lat INTEGER;
|
||||
centroid GEOMETRY;
|
||||
secgeo GEOMETRY;
|
||||
secbox GEOMETRY;
|
||||
diameter FLOAT;
|
||||
x BOOLEAN;
|
||||
BEGIN
|
||||
@ -610,9 +597,12 @@ BEGIN
|
||||
ELSE
|
||||
FOR lon IN xmin..(xmax-1) LOOP
|
||||
FOR lat IN ymin..(ymax-1) LOOP
|
||||
secgeo := st_intersection(geometry, ST_SetSRID(ST_MakeBox2D(ST_Point(lon,lat),ST_Point(lon+1,lat+1)),4326));
|
||||
IF NOT ST_IsEmpty(secgeo) AND ST_GeometryType(secgeo) in ('ST_Polygon','ST_MultiPolygon') THEN
|
||||
x := insertLocationAreaLarge(partition, place_id, country_code, keywords, rank_search, rank_address, false, centroid, secgeo);
|
||||
secbox := ST_SetSRID(ST_MakeBox2D(ST_Point(lon,lat),ST_Point(lon+1,lat+1)),4326);
|
||||
IF st_intersects(geometry, secbox) THEN
|
||||
secgeo := st_intersection(geometry, secbox);
|
||||
IF NOT ST_IsEmpty(secgeo) AND ST_GeometryType(secgeo) in ('ST_Polygon','ST_MultiPolygon') THEN
|
||||
x := insertLocationAreaLarge(partition, place_id, country_code, keywords, rank_search, rank_address, false, centroid, secgeo);
|
||||
END IF;
|
||||
END IF;
|
||||
END LOOP;
|
||||
END LOOP;
|
||||
@ -629,6 +619,8 @@ BEGIN
|
||||
diameter := 0.15;
|
||||
ELSEIF rank_search = 17 THEN
|
||||
diameter := 0.05;
|
||||
ELSEIF rank_search = 21 THEN
|
||||
diameter := 0.01;
|
||||
ELSEIF rank_search = 25 THEN
|
||||
diameter := 0.005;
|
||||
END IF;
|
||||
@ -654,6 +646,7 @@ $$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
CREATE OR REPLACE FUNCTION update_location(
|
||||
partition INTEGER,
|
||||
place_id INTEGER,
|
||||
place_country_code varchar(2),
|
||||
name hstore,
|
||||
@ -666,7 +659,7 @@ CREATE OR REPLACE FUNCTION update_location(
|
||||
DECLARE
|
||||
b BOOLEAN;
|
||||
BEGIN
|
||||
b := delete_location(place_id);
|
||||
b := deleteLocationArea(partition, place_id);
|
||||
RETURN add_location(place_id, place_country_code, name, rank_search, rank_address, geometry);
|
||||
END;
|
||||
$$
|
||||
@ -865,6 +858,7 @@ DECLARE
|
||||
postcode TEXT;
|
||||
result BOOLEAN;
|
||||
country_code VARCHAR(2);
|
||||
default_language VARCHAR(10);
|
||||
diameter FLOAT;
|
||||
BEGIN
|
||||
-- RAISE WARNING '%',NEW.osm_id;
|
||||
@ -897,8 +891,20 @@ BEGIN
|
||||
NEW.indexed_status := 1; --STATUS_NEW
|
||||
|
||||
NEW.country_code := get_country_code(NEW.geometry, NEW.country_code);
|
||||
NEW.geometry_sector := geometry_sector(NEW.geometry);
|
||||
NEW.partition := get_partition(NEW.geometry, NEW.country_code);
|
||||
NEW.geometry_sector := geometry_sector(NEW.partition, NEW.geometry);
|
||||
|
||||
-- copy 'name' to or from the default language (if there is a default language)
|
||||
IF NEW.name is not null THEN
|
||||
default_language := get_country_language_code(NEW.country_code);
|
||||
IF default_language IS NOT NULL THEN
|
||||
IF NEW.name ? 'name' AND NOT NEW.name ? ('name:'||default_language) THEN
|
||||
NEW.name := NEW.name || (('name:'||default_language) => (NEW.name -> 'name'));
|
||||
ELSEIF NEW.name ? ('name:'||default_language) AND NOT NEW.name ? 'name' THEN
|
||||
NEW.name := NEW.name || ('name' => (NEW.name -> 'name:'||default_language));
|
||||
END IF;
|
||||
END IF;
|
||||
END IF;
|
||||
|
||||
IF NEW.admin_level > 15 THEN
|
||||
NEW.admin_level := 15;
|
||||
@ -1180,7 +1186,7 @@ BEGIN
|
||||
RETURN NEW;
|
||||
END IF;
|
||||
|
||||
DELETE FROM search_name WHERE place_id = NEW.place_id;
|
||||
result := deleteSearchName(NEW.partition, NEW.place_id);
|
||||
DELETE FROM place_addressline WHERE place_id = NEW.place_id;
|
||||
DELETE FROM place_boundingbox where place_id = NEW.place_id;
|
||||
|
||||
@ -1204,6 +1210,10 @@ BEGIN
|
||||
name_vector := name_vector + tagpairid;
|
||||
END IF;
|
||||
|
||||
FOR i IN 1..28 LOOP
|
||||
address_havelevel[i] := false;
|
||||
END LOOP;
|
||||
|
||||
--RAISE WARNING '% %', NEW.place_id, NEW.rank_search;
|
||||
|
||||
-- For low level elements we inherit from our parent road
|
||||
@ -1310,42 +1320,38 @@ BEGIN
|
||||
address_street_word_id := get_name_id(make_standard_name(NEW.street));
|
||||
--RAISE WARNING 'street: % %', NEW.street, address_street_word_id;
|
||||
IF address_street_word_id IS NOT NULL THEN
|
||||
FOR location IN SELECT place_id,ST_distance(NEW.geometry, search_name.centroid) as distance
|
||||
FROM search_name WHERE search_name.name_vector @> ARRAY[address_street_word_id]
|
||||
AND ST_DWithin(NEW.geometry, search_name.centroid, 0.01) and search_rank between 22 and 27
|
||||
ORDER BY ST_distance(NEW.geometry, search_name.centroid) ASC limit 1
|
||||
LOOP
|
||||
FOR location IN SELECT * from getNearestNamedRoadFeature(NEW.partition, place_centroid, address_street_word_id) LOOP
|
||||
--RAISE WARNING 'streetname found nearby %',location;
|
||||
NEW.parent_place_id := location.place_id;
|
||||
END LOOP;
|
||||
END IF;
|
||||
-- Failed, fall back to nearest - don't just stop
|
||||
IF NEW.parent_place_id IS NULL THEN
|
||||
--RAISE WARNING 'unable to find streetname nearby % %',NEW.street,address_street_word_id;
|
||||
-- RETURN null;
|
||||
END IF;
|
||||
END IF;
|
||||
|
||||
--RAISE WARNING 'x4';
|
||||
|
||||
IF NEW.parent_place_id IS NULL THEN
|
||||
FOR location IN SELECT place_id FROM getNearRoads(NEW.partition, place_centroid) LOOP
|
||||
-- Still nothing, just use the nearest road
|
||||
-- IF NEW.parent_place_id IS NULL THEN
|
||||
-- FOR location IN SELECT place_id FROM getNearRoads(NEW.partition, place_centroid) LOOP
|
||||
-- NEW.parent_place_id := location.place_id;
|
||||
-- END LOOP;
|
||||
-- END IF;
|
||||
|
||||
search_diameter := 0.00005;
|
||||
WHILE NEW.parent_place_id IS NULL AND search_diameter < 0.1 LOOP
|
||||
FOR location IN SELECT place_id FROM placex
|
||||
WHERE ST_DWithin(place_centroid, placex.geometry, search_diameter) and rank_search between 22 and 27
|
||||
ORDER BY ST_distance(NEW.geometry, placex.geometry) ASC limit 1
|
||||
LOOP
|
||||
NEW.parent_place_id := location.place_id;
|
||||
END LOOP;
|
||||
END IF;
|
||||
search_diameter := search_diameter * 2;
|
||||
END LOOP;
|
||||
|
||||
--RAISE WARNING 'x6 %',NEW.parent_place_id;
|
||||
|
||||
-- If we didn't find any road fallback to standard method
|
||||
IF NEW.parent_place_id IS NOT NULL THEN
|
||||
|
||||
-- Some unnamed roads won't have been indexed, index now if needed
|
||||
-- ALL are now indexed!
|
||||
-- select count(*) from place_addressline where place_id = NEW.parent_place_id INTO parent_place_id_count;
|
||||
-- IF parent_place_id_count = 0 THEN
|
||||
-- UPDATE placex set indexed = true where indexed = false and place_id = NEW.parent_place_id;
|
||||
-- END IF;
|
||||
|
||||
-- Add the street to the address as zero distance to force to front of list
|
||||
INSERT INTO place_addressline VALUES (NEW.place_id, NEW.parent_place_id, true, true, 0, 26);
|
||||
address_havelevel[26] := true;
|
||||
@ -1370,8 +1376,10 @@ BEGIN
|
||||
|
||||
-- Performance, it would be more acurate to do all the rest of the import process but it takes too long
|
||||
-- Just be happy with inheriting from parent road only
|
||||
INSERT INTO search_name values (NEW.place_id, NEW.rank_search, NEW.rank_address, 0, NEW.country_code,
|
||||
name_vector, nameaddress_vector, place_centroid);
|
||||
result := insertSearchName(NEW.partition, NEW.place_id, NEW.country_code, name_vector, nameaddress_vector, NEW.rank_search, NEW.rank_address, place_centroid);
|
||||
|
||||
-- INSERT INTO search_name values (NEW.place_id, NEW.rank_search, NEW.rank_address, 0, NEW.country_code,
|
||||
-- name_vector, nameaddress_vector, place_centroid);
|
||||
|
||||
return NEW;
|
||||
END IF;
|
||||
@ -1423,16 +1431,10 @@ BEGIN
|
||||
IF array_upper(isin_tokens, 1) IS NOT NULL THEN
|
||||
FOR i IN 1..array_upper(isin_tokens, 1) LOOP
|
||||
|
||||
FOR location IN SELECT place_id,search_name.name_vector,address_rank,
|
||||
ST_Distance(place_centroid, search_name.centroid) as distance
|
||||
FROM search_name
|
||||
WHERE search_name.name_vector @> ARRAY[isin_tokens[i]]
|
||||
AND search_rank < NEW.rank_search
|
||||
AND (country_code = NEW.country_code OR address_rank < 4)
|
||||
ORDER BY ST_distance(NEW.geometry, centroid) ASC limit 1
|
||||
LOOP
|
||||
nameaddress_vector := array_merge(nameaddress_vector, location.name_vector);
|
||||
INSERT INTO place_addressline VALUES (NEW.place_id, location.place_id, false, NOT address_havelevel[location.address_rank], location.distance, location.address_rank);
|
||||
FOR location IN SELECT * from getNearestNamedFeature(NEW.partition, place_centroid, search_maxrank, isin_tokens[i]) LOOP
|
||||
nameaddress_vector := array_merge(nameaddress_vector, location.keywords::integer[]);
|
||||
INSERT INTO place_addressline VALUES (NEW.place_id, location.place_id, false, NOT address_havelevel[location.rank_address], location.distance, location.rank_address);
|
||||
address_havelevel[location.rank_address] := true;
|
||||
END LOOP;
|
||||
|
||||
END LOOP;
|
||||
@ -1445,8 +1447,9 @@ BEGIN
|
||||
result := add_location(NEW.place_id, NEW.country_code, NEW.partition, name_vector, NEW.rank_search, NEW.rank_address, NEW.geometry);
|
||||
END IF;
|
||||
|
||||
INSERT INTO search_name values (NEW.place_id, NEW.rank_search, NEW.rank_search, 0, NEW.country_code,
|
||||
name_vector, nameaddress_vector, place_centroid);
|
||||
result := insertSearchName(NEW.partition, NEW.place_id, NEW.country_code, name_vector, nameaddress_vector, NEW.rank_search, NEW.rank_address, place_centroid);
|
||||
|
||||
-- INSERT INTO search_name values (NEW.place_id, NEW.rank_search, NEW.rank_search, 0, NEW.country_code, name_vector, nameaddress_vector, place_centroid);
|
||||
END IF;
|
||||
|
||||
END IF;
|
||||
@ -1459,19 +1462,16 @@ LANGUAGE plpgsql;
|
||||
CREATE OR REPLACE FUNCTION placex_delete() RETURNS TRIGGER
|
||||
AS $$
|
||||
DECLARE
|
||||
b BOOLEAN;
|
||||
BEGIN
|
||||
|
||||
--IF OLD.rank_search < 26 THEN
|
||||
--RAISE WARNING 'delete % % % % %',OLD.place_id,OLD.osm_type,OLD.osm_id,OLD.class,OLD.type;
|
||||
--END IF;
|
||||
|
||||
-- mark everything linked to this place for re-indexing
|
||||
UPDATE placex set indexed_status = 2 from place_addressline where address_place_id = OLD.place_id
|
||||
and placex.place_id = place_addressline.place_id and indexed_status = 0;
|
||||
|
||||
-- do the actual delete
|
||||
DELETE FROM location_area where place_id = OLD.place_id;
|
||||
DELETE FROM search_name where place_id = OLD.place_id;
|
||||
b := deleteLocationArea(OLD.partition, OLD.place_id);
|
||||
b := deleteSearchName(OLD.partition, OLD.place_id);
|
||||
DELETE FROM place_addressline where place_id = OLD.place_id;
|
||||
DELETE FROM place_addressline where address_place_id = OLD.place_id;
|
||||
|
||||
|
@ -14,28 +14,41 @@ CREATE TABLE location_area_country () INHERITS (location_area_large);
|
||||
CREATE INDEX idx_location_area_country_place_id ON location_area_country USING BTREE (place_id);
|
||||
CREATE INDEX idx_location_area_country_geometry ON location_area_country USING GIST (geometry);
|
||||
|
||||
CREATE TABLE search_name_country () INHERITS (search_name_blank);
|
||||
CREATE INDEX idx_search_name_country_place_id ON search_name_country USING BTREE (place_id);
|
||||
CREATE INDEX idx_search_name_country_centroid ON search_name_country USING GIST (centroid);
|
||||
CREATE INDEX idx_search_name_country_name_vector ON search_name_country USING GIN (name_vector gin__int_ops);
|
||||
CREATE INDEX idx_search_name_country_nameaddress_vector ON search_name_country USING GIN (nameaddress_vector gin__int_ops);
|
||||
|
||||
-- start
|
||||
CREATE TABLE location_area_large_-partition- () INHERITS (location_area_large);
|
||||
CREATE INDEX idx_location_area_large_-partition-_place_id ON location_area_large_-partition- USING BTREE (place_id);
|
||||
CREATE INDEX idx_location_area_large_-partition-_geometry ON location_area_large_-partition- USING GIST (geometry);
|
||||
|
||||
CREATE TABLE location_area_roadnear_-partition- () INHERITS (location_area_roadnear);
|
||||
CREATE INDEX idx_location_area_roadnear_-partition-_place_id ON location_area_roadfar_-partition- USING BTREE (place_id);
|
||||
CREATE INDEX idx_location_area_roadnear_-partition-_place_id ON location_area_roadnear_-partition- USING BTREE (place_id);
|
||||
CREATE INDEX idx_location_area_roadnear_-partition-_geometry ON location_area_roadnear_-partition- USING GIST (geometry);
|
||||
|
||||
CREATE TABLE location_area_roadfar_-partition- () INHERITS (location_area_roadfar);
|
||||
CREATE INDEX idx_location_area_roadfar_-partition-_place_id ON location_area_roadfar_-partition- USING BTREE (place_id);
|
||||
CREATE INDEX idx_location_area_roadfar_-partition-_geometry ON location_area_roadfar_-partition- USING GIST (geometry);
|
||||
|
||||
CREATE TABLE search_name_-partition- () INHERITS (search_name_blank);
|
||||
CREATE INDEX idx_search_name_-partition-_place_id ON search_name_-partition- USING BTREE (place_id);
|
||||
CREATE INDEX idx_search_name_-partition-_centroid ON search_name_-partition- USING GIST (centroid);
|
||||
CREATE INDEX idx_search_name_-partition-_name_vector ON search_name_-partition- USING GIN (name_vector gin__int_ops);
|
||||
CREATE INDEX idx_search_name_-partition-_nameaddress_vector ON search_name_-partition- USING GIN (nameaddress_vector gin__int_ops);
|
||||
|
||||
-- end
|
||||
|
||||
create or replace function getNearRoads(in_partition TEXT, point GEOMETRY) RETURNS setof nearplace AS $$
|
||||
create or replace function getNearRoads(in_partition INTEGER, point GEOMETRY) RETURNS setof nearplace AS $$
|
||||
DECLARE
|
||||
r nearplace%rowtype;
|
||||
a BOOLEAN;
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
a := FALSE;
|
||||
FOR r IN SELECT place_id FROM location_area_roadnear_-partition- WHERE ST_Contains(geometry, point) ORDER BY ST_Distance(point, centroid) ASC LIMIT 1 LOOP
|
||||
a := TRUE;
|
||||
@ -55,13 +68,13 @@ END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function getNearFeatures(in_partition TEXT, point GEOMETRY, maxrank INTEGER, isin_tokens INT[]) RETURNS setof nearfeature AS $$
|
||||
create or replace function getNearFeatures(in_partition INTEGER, point GEOMETRY, maxrank INTEGER, isin_tokens INT[]) RETURNS setof nearfeature AS $$
|
||||
DECLARE
|
||||
r nearfeature%rowtype;
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
FOR r IN
|
||||
SELECT place_id, keywords, rank_address, rank_search, ST_Distance(point, centroid) as distance FROM (
|
||||
SELECT * FROM location_area_large_-partition- WHERE ST_Contains(geometry, point) and rank_search < maxrank
|
||||
@ -81,12 +94,12 @@ END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function deleteLocationArea(in_partition TEXT, in_place_id integer) RETURNS BOOLEAN AS $$
|
||||
create or replace function deleteLocationArea(in_partition INTEGER, in_place_id integer) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
-- DELETE from location_area_large_-partition- WHERE place_id = in_place_id;
|
||||
-- DELETE from location_area_roadnear_-partition- WHERE place_id = in_place_id;
|
||||
-- DELETE from location_area_roadfar_-partition- WHERE place_id = in_place_id;
|
||||
@ -102,7 +115,7 @@ $$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function insertLocationAreaLarge(
|
||||
in_partition TEXT, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_partition INTEGER, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_rank_search INTEGER, in_rank_address INTEGER, in_estimate BOOLEAN,
|
||||
in_centroid GEOMETRY, in_geometry GEOMETRY) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
@ -114,7 +127,7 @@ BEGIN
|
||||
END IF;
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
INSERT INTO location_area_large_-partition- values (in_partition, in_place_id, in_country_code, in_keywords, in_rank_search, in_rank_address, in_estimate, in_centroid, in_geometry);
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
@ -127,14 +140,14 @@ $$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function insertLocationAreaRoadNear(
|
||||
in_partition TEXT, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_partition INTEGER, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_rank_search INTEGER, in_rank_address INTEGER, in_estimate BOOLEAN,
|
||||
in_centroid GEOMETRY, in_geometry GEOMETRY) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
INSERT INTO location_area_roadnear_-partition- values (in_partition, in_place_id, in_country_code, in_keywords, in_rank_search, in_rank_address, in_estimate, in_centroid, in_geometry);
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
@ -147,14 +160,14 @@ $$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function insertLocationAreaRoadFar(
|
||||
in_partition TEXT, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_partition INTEGER, in_place_id integer, in_country_code VARCHAR(2), in_keywords INTEGER[],
|
||||
in_rank_search INTEGER, in_rank_address INTEGER, in_estimate BOOLEAN,
|
||||
in_centroid GEOMETRY, in_geometry GEOMETRY) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = '-partition-' THEN
|
||||
IF in_partition = -partition- THEN
|
||||
INSERT INTO location_area_roadfar_-partition- values (in_partition, in_place_id, in_country_code, in_keywords, in_rank_search, in_rank_address, in_estimate, in_centroid, in_geometry);
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
@ -165,3 +178,114 @@ BEGIN
|
||||
END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function getNearestNamedFeature(in_partition INTEGER, point GEOMETRY, maxrank INTEGER, isin_token INTEGER) RETURNS setof nearfeature AS $$
|
||||
DECLARE
|
||||
r nearfeature%rowtype;
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = -partition- THEN
|
||||
FOR r IN
|
||||
SELECT place_id, name_vector, address_rank, search_rank,
|
||||
ST_Distance(centroid, point) as distance
|
||||
FROM search_name_-partition-
|
||||
WHERE name_vector @> ARRAY[isin_token]
|
||||
AND search_rank < maxrank
|
||||
UNION ALL
|
||||
SELECT place_id, name_vector, address_rank, search_rank,
|
||||
ST_Distance(centroid, point) as distance
|
||||
FROM search_name_country
|
||||
WHERE name_vector @> ARRAY[isin_token]
|
||||
AND search_rank < maxrank
|
||||
ORDER BY distance ASC limit 1
|
||||
LOOP
|
||||
RETURN NEXT r;
|
||||
END LOOP;
|
||||
RETURN;
|
||||
END IF;
|
||||
-- end
|
||||
|
||||
RAISE EXCEPTION 'Unknown partition %', in_partition;
|
||||
END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function getNearestNamedRoadFeature(in_partition INTEGER, point GEOMETRY, isin_token INTEGER) RETURNS setof nearfeature AS $$
|
||||
DECLARE
|
||||
r nearfeature%rowtype;
|
||||
BEGIN
|
||||
|
||||
-- start
|
||||
IF in_partition = -partition- THEN
|
||||
FOR r IN
|
||||
SELECT place_id, name_vector, address_rank, search_rank,
|
||||
ST_Distance(centroid, point) as distance
|
||||
FROM search_name_-partition-
|
||||
WHERE name_vector @> ARRAY[isin_token]
|
||||
AND ST_DWithin(centroid, point, 0.01)
|
||||
AND search_rank between 22 and 27
|
||||
ORDER BY distance ASC limit 1
|
||||
LOOP
|
||||
RETURN NEXT r;
|
||||
END LOOP;
|
||||
RETURN;
|
||||
END IF;
|
||||
-- end
|
||||
|
||||
RAISE EXCEPTION 'Unknown partition %', in_partition;
|
||||
END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function insertSearchName(
|
||||
in_partition INTEGER, in_place_id integer, in_country_code VARCHAR(2),
|
||||
in_name_vector INTEGER[], in_nameaddress_vector INTEGER[],
|
||||
in_rank_search INTEGER, in_rank_address INTEGER,
|
||||
in_centroid GEOMETRY) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
BEGIN
|
||||
|
||||
INSERT INTO search_name values (in_place_id, in_rank_search, in_rank_address, 0, in_country_code,
|
||||
in_name_vector, in_nameaddress_vector, in_centroid);
|
||||
|
||||
IF in_rank_search <= 4 THEN
|
||||
INSERT INTO search_name_country values (in_place_id, in_rank_search, in_rank_address, 0, in_country_code,
|
||||
in_name_vector, in_nameaddress_vector, in_centroid);
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
|
||||
-- start
|
||||
IF in_partition = -partition- THEN
|
||||
INSERT INTO search_name_-partition- values (in_place_id, in_rank_search, in_rank_address, 0, in_country_code,
|
||||
in_name_vector, in_nameaddress_vector, in_centroid);
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
-- end
|
||||
|
||||
RAISE EXCEPTION 'Unknown partition %', in_partition;
|
||||
RETURN FALSE;
|
||||
END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
||||
create or replace function deleteSearchName(in_partition INTEGER, in_place_id integer) RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
BEGIN
|
||||
|
||||
DELETE from search_name WHERE place_id = in_place_id;
|
||||
DELETE from search_name_country WHERE place_id = in_place_id;
|
||||
|
||||
-- start
|
||||
IF in_partition = -partition- THEN
|
||||
DELETE from search_name_-partition- WHERE place_id = in_place_id;
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
-- end
|
||||
|
||||
RAISE EXCEPTION 'Unknown partition %', in_partition;
|
||||
|
||||
RETURN FALSE;
|
||||
END
|
||||
$$
|
||||
LANGUAGE plpgsql;
|
||||
|
@ -76,7 +76,7 @@ CREATE SEQUENCE seq_word start 1;
|
||||
|
||||
drop table IF EXISTS location_area CASCADE;
|
||||
CREATE TABLE location_area (
|
||||
partition varchar(10),
|
||||
partition integer,
|
||||
place_id INTEGER,
|
||||
country_code VARCHAR(2),
|
||||
keywords INTEGER[],
|
||||
@ -91,8 +91,8 @@ CREATE TABLE location_area_large () INHERITS (location_area);
|
||||
CREATE TABLE location_area_roadnear () INHERITS (location_area);
|
||||
CREATE TABLE location_area_roadfar () INHERITS (location_area);
|
||||
|
||||
drop table IF EXISTS search_name;
|
||||
CREATE TABLE search_name (
|
||||
drop table IF EXISTS search_name_blank CASCADE;
|
||||
CREATE TABLE search_name_blank (
|
||||
place_id INTEGER,
|
||||
search_rank integer,
|
||||
address_rank integer,
|
||||
@ -101,9 +101,12 @@ CREATE TABLE search_name (
|
||||
name_vector integer[],
|
||||
nameaddress_vector integer[]
|
||||
);
|
||||
SELECT AddGeometryColumn('search_name_blank', 'centroid', 4326, 'GEOMETRY', 2);
|
||||
|
||||
drop table IF EXISTS search_name;
|
||||
CREATE TABLE search_name () INHERITS (search_name_blank);
|
||||
CREATE INDEX search_name_name_vector_idx ON search_name USING GIN (name_vector gin__int_ops);
|
||||
CREATE INDEX searchnameplacesearch_search_nameaddress_vector_idx ON search_name USING GIN (nameaddress_vector gin__int_ops);
|
||||
SELECT AddGeometryColumn('search_name', 'centroid', 4326, 'GEOMETRY', 2);
|
||||
CREATE INDEX idx_search_name_centroid ON search_name USING GIST (centroid);
|
||||
CREATE INDEX idx_search_name_place_id ON search_name USING BTREE (place_id);
|
||||
|
||||
@ -160,7 +163,7 @@ CREATE INDEX idx_country_geometry ON country USING GIST (geometry);
|
||||
drop table placex;
|
||||
CREATE TABLE placex (
|
||||
place_id INTEGER NOT NULL,
|
||||
partition varchar(10),
|
||||
partition integer,
|
||||
osm_type char(1),
|
||||
osm_id INTEGER,
|
||||
class TEXT NOT NULL,
|
||||
|
@ -47,6 +47,7 @@
|
||||
pgsqlRunScriptFile(CONST_Path_Postgresql_Postgis.'/postgis.sql');
|
||||
pgsqlRunScriptFile(CONST_Path_Postgresql_Postgis.'/spatial_ref_sys.sql');
|
||||
pgsqlRunScriptFile(CONST_BasePath.'/data/country_name.sql');
|
||||
pgsqlRunScriptFile(CONST_BasePath.'/data/country_naturaleathdata.sql');
|
||||
pgsqlRunScriptFile(CONST_BasePath.'/data/country_osm_grid.sql');
|
||||
pgsqlRunScriptFile(CONST_BasePath.'/data/gb_postcode.sql');
|
||||
pgsqlRunScriptFile(CONST_BasePath.'/data/us_statecounty.sql');
|
||||
@ -84,13 +85,13 @@
|
||||
{
|
||||
$bDidSomething = true;
|
||||
$oDB =& getDB();
|
||||
$sSQL = 'select distinct country_code from country_name order by country_code';
|
||||
$sSQL = 'select partition from country_name order by country_code';
|
||||
$aPartitions = $oDB->getCol($sSQL);
|
||||
if (PEAR::isError($aPartitions))
|
||||
{
|
||||
fail($aPartitions->getMessage());
|
||||
}
|
||||
$aPartitions[] = 'none';
|
||||
$aPartitions[] = 0;
|
||||
|
||||
$sTemplate = file_get_contents(CONST_BasePath.'/sql/partitions.src.sql');
|
||||
preg_match_all('#^-- start(.*?)^-- end#ms', $sTemplate, $aMatches, PREG_SET_ORDER);
|
||||
@ -111,12 +112,20 @@
|
||||
$bDidSomething = true;
|
||||
|
||||
$oDB =& getDB();
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE word')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE placex')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE place_addressline')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE place_boundingbox')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE location_area')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE search_name')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'TRUNCATE search_name_blank')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'DROP SEQUENCE seq_place')) fail(pg_last_error($oDB->connection));
|
||||
echo '.';
|
||||
if (!pg_query($oDB->connection, 'CREATE SEQUENCE seq_place start 100000')) fail(pg_last_error($oDB->connection));
|
||||
@ -130,6 +139,7 @@
|
||||
$sSQL = 'insert into placex (osm_type, osm_id, class, type, name, admin_level, ';
|
||||
$sSQL .= 'housenumber, street, isin, postcode, country_code, extratags, ';
|
||||
$sSQL .= 'geometry) select * from place where osm_id % '.$iInstances.' = '.$i;
|
||||
var_dump($sSQL);
|
||||
if (!pg_send_query($aDBInstances[$i]->connection, $sSQL)) fail(pg_last_error($oDB->connection));
|
||||
}
|
||||
$bAnyBusy = true;
|
||||
|
Loading…
Reference in New Issue
Block a user