imposm3/writer/relations.go

159 lines
3.5 KiB
Go
Raw Normal View History

package writer
import (
2013-08-29 17:44:15 +04:00
"imposm3/cache"
"imposm3/database"
"imposm3/element"
2013-11-08 19:03:58 +04:00
"imposm3/expire"
2013-08-29 17:44:15 +04:00
"imposm3/geom"
"imposm3/geom/geos"
"imposm3/proj"
"imposm3/stats"
"sync"
"time"
)
type RelationWriter struct {
2013-05-28 14:54:19 +04:00
OsmElemWriter
2013-10-28 14:37:58 +04:00
rel chan *element.Relation
}
func NewRelationWriter(osmCache *cache.OSMCache, diffCache *cache.DiffCache, rel chan *element.Relation,
2013-10-28 14:37:58 +04:00
inserter database.Inserter, progress *stats.Statistics,
srid int) *OsmElemWriter {
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
},
2013-10-28 14:37:58 +04:00
rel: rel,
}
2013-05-28 14:54:19 +04:00
rw.OsmElemWriter.writer = &rw
return &rw.OsmElemWriter
}
func (rw *RelationWriter) loop() {
2013-05-28 10:59:59 +04:00
geos := geos.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
}
continue NextRel
}
for _, 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
}
proj.NodesToMerc(m.Way.Nodes)
}
// BuildRelation updates r.Members but we need all of them
// for the diffCache
allMembers := r.Members
// prepare relation first (build rings and compute actual
// relation tags)
prepedRel, err := geom.PrepareRelation(r, rw.srid)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue NextRel
}
// check for matches befor building the geometry
ok, matches := rw.inserter.ProbePolygon(r.OSMElem)
if !ok {
continue NextRel
}
// build the multipolygon
r, err = prepedRel.Build()
if err != nil {
2013-06-07 10:25:45 +04:00
if r.Geom != nil && r.Geom.Geom != nil {
geos.Destroy(r.Geom.Geom)
}
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
2013-06-07 10:25:45 +04:00
continue NextRel
}
2013-06-07 10:25:45 +04:00
2013-07-30 10:17:47 +04:00
if rw.limiter != nil {
start := time.Now()
2013-07-30 10:17:47 +04:00
parts, err := rw.limiter.Clip(r.Geom.Geom)
if err != nil {
log.Warn(err)
continue NextRel
}
if duration := time.Now().Sub(start); duration > time.Minute {
log.Warnf("clipping relation %d to -limitto took %s", r.Id, duration)
}
for _, g := range parts {
rel := element.Relation(*r)
rel.Id = -r.Id
2013-11-13 18:41:34 +04:00
rel.Geom = &element.Geometry{Geom: g, Wkb: geos.AsEwkbHex(g)}
err := rw.inserter.InsertPolygon(rel.OSMElem, matches)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue
}
}
} else {
rel := element.Relation(*r)
rel.Id = -r.Id
err := rw.inserter.InsertPolygon(rel.OSMElem, matches)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue
}
}
for _, m := range rw.inserter.SelectRelationPolygons(r.Tags, r.Members) {
2013-12-16 19:54:42 +04:00
err = rw.osmCache.InsertedWays.PutWay(m.Way)
if err != nil {
log.Warn(err)
}
}
if rw.diffCache != nil {
rw.diffCache.Ways.AddFromMembers(r.Id, allMembers)
for _, member := range allMembers {
if member.Way != nil {
rw.diffCache.Coords.AddFromWay(member.Way)
}
}
}
2013-11-08 19:03:58 +04:00
if rw.expireor != nil {
for _, m := range allMembers {
if m.Way != nil {
2013-11-08 19:03:58 +04:00
expire.ExpireNodes(rw.expireor, m.Way.Nodes)
}
}
}
2013-06-07 10:25:45 +04:00
geos.Destroy(r.Geom.Geom)
}
rw.wg.Done()
}