imposm3/writer/relations.go

282 lines
6.4 KiB
Go
Raw Permalink Normal View History

package writer
import (
"sync"
"time"
2014-08-04 17:19:35 +04:00
"github.com/omniscale/imposm3/cache"
"github.com/omniscale/imposm3/database"
"github.com/omniscale/imposm3/element"
"github.com/omniscale/imposm3/expire"
geomp "github.com/omniscale/imposm3/geom"
2016-01-04 13:19:28 +03:00
geosp "github.com/omniscale/imposm3/geom/geos"
2014-08-04 17:19:35 +04:00
"github.com/omniscale/imposm3/mapping"
"github.com/omniscale/imposm3/stats"
)
type RelationWriter struct {
2013-05-28 14:54:19 +04:00
OsmElemWriter
2016-01-04 13:19:28 +03:00
singleIdSpace bool
rel chan *element.Relation
polygonMatcher mapping.RelWayMatcher
relationMatcher mapping.RelationMatcher
relationMemberMatcher mapping.RelationMatcher
maxGap float64
}
func NewRelationWriter(
osmCache *cache.OSMCache,
diffCache *cache.DiffCache,
singleIdSpace bool,
rel chan *element.Relation,
inserter database.Inserter,
progress *stats.Statistics,
matcher mapping.RelWayMatcher,
2016-01-04 13:19:28 +03:00
relMatcher mapping.RelationMatcher,
relMemberMatcher mapping.RelationMatcher,
srid int,
) *OsmElemWriter {
maxGap := 1e-1 // 0.1m
if srid == 4326 {
maxGap = 1e-6 // ~0.1m
}
rw := RelationWriter{
2013-05-28 14:54:19 +04:00
OsmElemWriter: OsmElemWriter{
2013-10-28 11:26:51 +04:00
osmCache: osmCache,
diffCache: diffCache,
progress: progress,
wg: &sync.WaitGroup{},
inserter: inserter,
srid: srid,
2013-05-28 14:54:19 +04:00
},
2016-01-04 13:19:28 +03:00
singleIdSpace: singleIdSpace,
polygonMatcher: matcher,
relationMatcher: relMatcher,
relationMemberMatcher: relMemberMatcher,
rel: rel,
maxGap: maxGap,
}
2013-05-28 14:54:19 +04:00
rw.OsmElemWriter.writer = &rw
return &rw.OsmElemWriter
}
func (rw *RelationWriter) relId(id int64) int64 {
if !rw.singleIdSpace {
return -id
}
return element.RelIdOffset - id
}
func (rw *RelationWriter) loop() {
2016-01-04 13:19:28 +03:00
geos := geosp.NewGeos()
geos.SetHandleSrid(rw.srid)
2013-05-28 10:59:59 +04:00
defer geos.Finish()
2013-06-07 10:25:45 +04:00
NextRel:
for r := range rw.rel {
rw.progress.AddRelations(1)
err := rw.osmCache.Ways.FillMembers(r.Members)
2013-06-07 10:25:45 +04:00
if err != nil {
if err != cache.NotFound {
log.Warn(err)
2013-06-07 10:25:45 +04:00
}
2016-01-06 10:46:44 +03:00
continue
}
2016-01-04 13:19:28 +03:00
for i, m := range r.Members {
if m.Way == nil {
continue
}
err := rw.osmCache.Coords.FillWay(m.Way)
2013-06-07 10:25:45 +04:00
if err != nil {
if err != cache.NotFound {
log.Warn(err)
2013-06-07 10:25:45 +04:00
}
continue NextRel
}
2014-06-30 10:58:22 +04:00
rw.NodesToSrid(m.Way.Nodes)
2016-01-04 13:19:28 +03:00
r.Members[i].Elem = &m.Way.OSMElem
}
2016-01-06 10:46:44 +03:00
// handleRelation updates r.Members but we need all of them
// for the diffCache
allMembers := r.Members
2016-01-04 13:19:28 +03:00
2016-01-06 10:46:44 +03:00
inserted := false
2016-01-04 13:19:28 +03:00
2016-01-06 10:46:44 +03:00
if handleRelationMembers(rw, r, geos) {
inserted = true
}
if handleRelation(rw, r, geos) {
inserted = true
}
if handleMultiPolygon(rw, r, geos) {
inserted = true
}
2016-01-04 13:19:28 +03:00
2016-01-06 10:46:44 +03:00
if inserted && rw.diffCache != nil {
rw.diffCache.Ways.AddFromMembers(r.Id, allMembers)
rw.diffCache.CoordsRel.AddFromMembers(r.Id, allMembers)
2016-01-06 10:46:44 +03:00
for _, member := range allMembers {
if member.Way != nil {
rw.diffCache.Coords.AddFromWay(member.Way)
2016-01-04 13:19:28 +03:00
}
}
}
2016-01-06 10:46:44 +03:00
if inserted && rw.expireor != nil {
for _, m := range allMembers {
if m.Way != nil {
2016-11-23 16:43:04 +03:00
expire.ExpireProjectedNodes(rw.expireor, m.Way.Nodes, rw.srid, true)
2016-01-06 10:46:44 +03:00
}
}
}
2016-01-06 10:46:44 +03:00
}
rw.wg.Done()
}
2016-01-06 10:46:44 +03:00
func handleMultiPolygon(rw *RelationWriter, r *element.Relation, geos *geosp.Geos) bool {
matches := rw.polygonMatcher.MatchRelation(r)
if matches == nil {
return false
}
// prepare relation (build rings)
2016-01-06 10:46:44 +03:00
prepedRel, err := geomp.PrepareRelation(r, rw.srid, rw.maxGap)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
2016-01-06 10:46:44 +03:00
return false
}
// build the multipolygon
geom, err := prepedRel.Build()
if geom.Geom != nil {
defer geos.Destroy(geom.Geom)
}
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
2016-01-06 10:46:44 +03:00
return false
}
2016-01-06 10:46:44 +03:00
if rw.limiter != nil {
start := time.Now()
parts, err := rw.limiter.Clip(geom.Geom)
if err != nil {
2016-01-06 10:46:44 +03:00
log.Warn(err)
return false
}
2016-01-06 10:46:44 +03:00
if duration := time.Now().Sub(start); duration > time.Minute {
log.Warnf("clipping relation %d to -limitto took %s", r.Id, duration)
}
2017-02-01 13:59:41 +03:00
if len(parts) == 0 {
return false
}
2016-01-06 10:46:44 +03:00
for _, g := range parts {
rel := element.Relation(*r)
rel.Id = rw.relId(r.Id)
2016-01-06 10:46:44 +03:00
geom = geomp.Geometry{Geom: g, Wkb: geos.AsEwkbHex(g)}
err := rw.inserter.InsertPolygon(rel.OSMElem, geom, matches)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue
}
}
2016-01-06 10:46:44 +03:00
} else {
rel := element.Relation(*r)
rel.Id = rw.relId(r.Id)
err := rw.inserter.InsertPolygon(rel.OSMElem, geom, matches)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
2013-12-16 19:54:42 +04:00
log.Warn(err)
}
2016-01-06 10:46:44 +03:00
return false
}
2016-01-06 10:46:44 +03:00
}
return true
}
func handleRelation(rw *RelationWriter, r *element.Relation, geos *geosp.Geos) bool {
relMatches := rw.relationMatcher.MatchRelation(r)
if relMatches == nil {
return false
}
rel := element.Relation(*r)
rel.Id = rw.relId(r.Id)
rw.inserter.InsertPolygon(rel.OSMElem, geomp.Geometry{}, relMatches)
return true
}
func handleRelationMembers(rw *RelationWriter, r *element.Relation, geos *geosp.Geos) bool {
relMemberMatches := rw.relationMemberMatcher.MatchRelation(r)
if relMemberMatches == nil {
return false
}
for i, m := range r.Members {
if m.Type == element.RELATION {
mrel, err := rw.osmCache.Relations.GetRelation(m.Id)
if err != nil {
if err != cache.NotFound {
2016-01-06 10:46:44 +03:00
log.Warn(err)
}
return false
}
2016-01-06 10:46:44 +03:00
r.Members[i].Elem = &mrel.OSMElem
} else if m.Type == element.NODE {
nd, err := rw.osmCache.Nodes.GetNode(m.Id)
if err != nil {
if err == cache.NotFound {
nd, err = rw.osmCache.Coords.GetCoord(m.Id)
if err != nil {
if err != cache.NotFound {
log.Warn(err)
}
return false
}
} else {
log.Warn(err)
return false
}
}
2016-01-06 10:46:44 +03:00
rw.NodeToSrid(nd)
r.Members[i].Node = nd
r.Members[i].Elem = &nd.OSMElem
}
}
2016-01-06 10:46:44 +03:00
for _, m := range r.Members {
var g *geosp.Geom
var err error
if m.Node != nil {
g, err = geomp.Point(geos, *m.Node)
} else if m.Way != nil {
g, err = geomp.LineString(geos, m.Way.Nodes)
}
if err != nil {
log.Warn(err)
return false
}
var gelem geomp.Geometry
if g == nil {
g = geos.FromWkt("POLYGON EMPTY")
gelem = geomp.Geometry{Geom: g, Wkb: geos.AsEwkbHex(g)}
} else {
gelem, err = geomp.AsGeomElement(geos, g)
if err != nil {
log.Warn(err)
return false
}
}
rel := element.Relation(*r)
rel.Id = rw.relId(r.Id)
rw.inserter.InsertRelationMember(rel, m, gelem, relMemberMatches)
}
return true
}