Skip to content

Commit 5b0125c

Browse files
committed
postgis: create one composite PRIMARY KEY instead of two indices
Create a composite PRIMARY KEY for OSM ID and the unique `id` colum (serial) instead of a separeate index for `id` and OSM ID. This can save a few GB for large tables. OSM ID is first, so lookups for diff imports are still fast. The `id` makes the index unique as OSM IDs are not unique (e.g. multiple mappings).
1 parent 2d7f426 commit 5b0125c

File tree

2 files changed

+25
-4
lines changed

2 files changed

+25
-4
lines changed

database/postgis/postgis.go

+12-1
Original file line numberDiff line numberDiff line change
@@ -196,6 +196,13 @@ func (pg *PostGIS) Finish() error {
196196
}
197197

198198
func createIndex(pg *PostGIS, tableName string, columns []ColumnSpec) error {
199+
foundIDCol := false
200+
for _, cs := range columns {
201+
if cs.Name == "id" {
202+
foundIDCol = true
203+
}
204+
}
205+
199206
for _, col := range columns {
200207
if col.Type.Name() == "GEOMETRY" {
201208
sql := fmt.Sprintf(`CREATE INDEX "%s_geom" ON "%s"."%s" USING GIST ("%s")`,
@@ -207,7 +214,11 @@ func createIndex(pg *PostGIS, tableName string, columns []ColumnSpec) error {
207214
return err
208215
}
209216
}
210-
if col.FieldType.Name == "id" {
217+
if col.FieldType.Name == "id" && foundIDCol {
218+
// Create index for OSM ID required for diff updates, but only if
219+
// the table does have an `id` column.
220+
// The explicit `id` column prevented the creation of our composite
221+
// PRIMARY KEY index of id (serial) and OSM ID.
211222
sql := fmt.Sprintf(`CREATE INDEX "%s_%s_idx" ON "%s"."%s" USING BTREE ("%s")`,
212223
tableName, col.Name, pg.Config.ImportSchema, tableName, col.Name)
213224
step := log.Step(fmt.Sprintf("Creating OSM id index on %s", tableName))

database/postgis/spec.go

+13-3
Original file line numberDiff line numberDiff line change
@@ -43,17 +43,21 @@ func (col *ColumnSpec) AsSQL() string {
4343

4444
func (spec *TableSpec) CreateTableSQL() string {
4545
foundIDCol := false
46+
pkCols := []string{}
4647
for _, cs := range spec.Columns {
4748
if cs.Name == "id" {
4849
foundIDCol = true
4950
}
51+
if cs.FieldType.Name == "id" {
52+
pkCols = append(pkCols, cs.Name)
53+
}
5054
}
5155

5256
cols := []string{}
5357
if !foundIDCol {
54-
// only add id column if there is no id configured
55-
// TODO allow to disable id column?
56-
cols = append(cols, "id SERIAL PRIMARY KEY")
58+
// Create explicit id column only if there is no id configured.
59+
cols = append(cols, "id SERIAL")
60+
pkCols = append(pkCols, "id")
5761
}
5862

5963
for _, col := range spec.Columns {
@@ -62,6 +66,12 @@ func (spec *TableSpec) CreateTableSQL() string {
6266
}
6367
cols = append(cols, col.AsSQL())
6468
}
69+
70+
// Make composite PRIMARY KEY of serial `id` and OSM ID. But only if the
71+
// user did not provide a custom `id` colum which might not be unique.
72+
if pkCols != nil && !foundIDCol {
73+
cols = append(cols, `PRIMARY KEY ("`+strings.Join(pkCols, `", "`)+`")`)
74+
}
6575
columnSQL := strings.Join(cols, ",\n")
6676
return fmt.Sprintf(`
6777
CREATE TABLE IF NOT EXISTS "%s"."%s" (

0 commit comments

Comments
 (0)