imposm3/writer/ways.go

156 lines
3.5 KiB
Go
Raw Normal View History

package writer
import (
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"
"github.com/omniscale/imposm3/geom"
"github.com/omniscale/imposm3/geom/geos"
"github.com/omniscale/imposm3/mapping"
"github.com/omniscale/imposm3/stats"
"sync"
)
type WayWriter struct {
2013-05-28 14:54:19 +04:00
OsmElemWriter
ways chan *element.Way
lineMatcher mapping.WayMatcher
polygonMatcher mapping.WayMatcher
}
func NewWayWriter(
osmCache *cache.OSMCache,
diffCache *cache.DiffCache,
ways chan *element.Way,
2013-10-28 14:37:58 +04:00
inserter database.Inserter,
progress *stats.Statistics,
polygonMatcher mapping.WayMatcher,
lineMatcher mapping.WayMatcher,
srid int,
) *OsmElemWriter {
ww := WayWriter{
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
},
lineMatcher: lineMatcher,
polygonMatcher: polygonMatcher,
ways: ways,
}
2013-05-28 14:54:19 +04:00
ww.OsmElemWriter.writer = &ww
return &ww.OsmElemWriter
}
func (ww *WayWriter) loop() {
geos := geos.NewGeos()
geos.SetHandleSrid(ww.srid)
defer geos.Finish()
for w := range ww.ways {
ww.progress.AddWays(1)
2013-07-15 13:41:03 +04:00
if len(w.Tags) == 0 {
continue
}
insertedAsRelation, err := ww.osmCache.InsertedWays.IsInserted(w.Id)
if err != nil {
log.Warn(err)
continue
}
err = ww.osmCache.Coords.FillWay(w)
if err != nil {
continue
}
2014-06-30 10:58:22 +04:00
ww.NodesToSrid(w.Nodes)
2013-07-12 16:57:06 +04:00
inserted := false
if matches := ww.lineMatcher.MatchWay(w); len(matches) > 0 {
err := ww.buildAndInsert(geos, w, matches, false)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue
}
inserted = true
}
if w.IsClosed() && !insertedAsRelation {
// only add polygons that were not inserted as a MultiPolygon relation
if matches := ww.polygonMatcher.MatchWay(w); len(matches) > 0 {
err := ww.buildAndInsert(geos, w, matches, true)
if err != nil {
if errl, ok := err.(ErrorLevel); !ok || errl.Level() > 0 {
log.Warn(err)
}
continue
}
inserted = true
}
}
2013-11-08 19:03:58 +04:00
if inserted && ww.expireor != nil {
expire.ExpireNodes(ww.expireor, w.Nodes)
}
2013-05-30 14:00:11 +04:00
if ww.diffCache != nil {
ww.diffCache.Coords.AddFromWay(w)
}
}
ww.wg.Done()
}
2013-05-28 14:54:19 +04:00
func (ww *WayWriter) buildAndInsert(g *geos.Geos, w *element.Way, matches []mapping.Match, isPolygon bool) error {
2013-05-28 14:54:19 +04:00
var err error
var geosgeom *geos.Geom
2013-05-28 14:54:19 +04:00
// make copy to avoid interference with polygon/linestring matches
way := element.Way(*w)
if isPolygon {
geosgeom, err = geom.Polygon(g, way.Nodes)
} else {
geosgeom, err = geom.LineString(g, way.Nodes)
}
2013-05-28 14:54:19 +04:00
if err != nil {
return err
2013-05-28 14:54:19 +04:00
}
way.Geom, err = geom.AsGeomElement(g, geosgeom)
if err != nil {
return err
}
2013-07-30 10:17:47 +04:00
if ww.limiter != nil {
parts, err := ww.limiter.Clip(way.Geom.Geom)
2013-05-28 14:54:19 +04:00
if err != nil {
return err
2013-05-28 14:54:19 +04:00
}
for _, p := range parts {
2013-05-28 14:54:19 +04:00
way := element.Way(*w)
2013-11-13 18:41:34 +04:00
way.Geom = &element.Geometry{Geom: p, Wkb: g.AsEwkbHex(p)}
if isPolygon {
if err := ww.inserter.InsertPolygon(way.OSMElem, matches); err != nil {
return err
}
} else {
if err := ww.inserter.InsertLineString(way.OSMElem, matches); err != nil {
return err
}
}
2013-05-28 14:54:19 +04:00
}
} else {
if isPolygon {
if err := ww.inserter.InsertPolygon(way.OSMElem, matches); err != nil {
return err
}
} else {
if err := ww.inserter.InsertLineString(way.OSMElem, matches); err != nil {
return err
}
}
2013-05-28 14:54:19 +04:00
}
return nil
2013-05-28 14:54:19 +04:00
}