imposm3/writer/ways.go

155 lines
3.5 KiB
Go
Raw Normal View History

package writer
import (
"goposm/cache"
"goposm/element"
"goposm/geom"
2013-05-28 10:59:59 +04:00
"goposm/geom/clipper"
"goposm/geom/geos"
"goposm/mapping"
"goposm/proj"
"goposm/stats"
"log"
"runtime"
"sync"
)
type WayWriter struct {
osmCache *cache.OSMCache
ways chan *element.Way
lineStringTagMatcher *mapping.TagMatcher
polygonTagMatcher *mapping.TagMatcher
progress *stats.Statistics
insertBuffer *InsertBuffer
wg *sync.WaitGroup
2013-05-28 10:59:59 +04:00
clipper *clipper.Clipper
}
func NewWayWriter(osmCache *cache.OSMCache, ways chan *element.Way,
insertBuffer *InsertBuffer, lineStringTagMatcher *mapping.TagMatcher,
polygonTagMatcher *mapping.TagMatcher, progress *stats.Statistics) *WayWriter {
ww := WayWriter{
osmCache: osmCache,
ways: ways,
insertBuffer: insertBuffer,
lineStringTagMatcher: lineStringTagMatcher,
polygonTagMatcher: polygonTagMatcher,
progress: progress,
wg: &sync.WaitGroup{},
}
2013-05-28 10:59:59 +04:00
return &ww
}
func (ww *WayWriter) SetClipper(clipper *clipper.Clipper) {
ww.clipper = clipper
}
func (ww *WayWriter) Start() {
for i := 0; i < runtime.NumCPU(); i++ {
ww.wg.Add(1)
go ww.loop()
}
}
func (ww *WayWriter) Close() {
ww.wg.Wait()
}
func (ww *WayWriter) loop() {
geos := geos.NewGeos()
defer geos.Finish()
for w := range ww.ways {
ww.progress.AddWays(1)
inserted, err := ww.osmCache.InsertedWays.IsInserted(w.Id)
if err != nil {
log.Println(err)
continue
}
if inserted {
continue
}
err = ww.osmCache.Coords.FillWay(w)
if err != nil {
continue
}
proj.NodesToMerc(w.Nodes)
2013-05-23 15:09:47 +04:00
if matches := ww.lineStringTagMatcher.Match(&w.Tags); len(matches) > 0 {
// make copy to avoid interference with polygon matches
way := element.Way(*w)
way.Geom, err = geom.LineStringWkb(geos, way.Nodes)
if err != nil {
if err, ok := err.(ErrorLevel); ok {
if err.Level() <= 0 {
continue
}
}
log.Println(err)
continue
}
2013-05-28 10:59:59 +04:00
if ww.clipper != nil {
parts, err := ww.clipper.Clip(way.Geom.Geom)
if err != nil {
log.Println(err)
continue
}
for _, g := range parts {
way := element.Way(*w)
way.Geom = &element.Geometry{g, geos.AsWkb(g)}
for _, match := range matches {
row := match.Row(&way.OSMElem)
ww.insertBuffer.Insert(match.Table.Name, row)
}
}
} else {
for _, match := range matches {
row := match.Row(&way.OSMElem)
ww.insertBuffer.Insert(match.Table.Name, row)
}
}
}
if w.IsClosed() {
2013-05-23 15:09:47 +04:00
if matches := ww.polygonTagMatcher.Match(&w.Tags); len(matches) > 0 {
way := element.Way(*w)
way.Geom, err = geom.PolygonWkb(geos, way.Nodes)
if err != nil {
if err, ok := err.(ErrorLevel); ok {
if err.Level() <= 0 {
continue
}
}
log.Println(err)
continue
}
2013-05-28 10:59:59 +04:00
if ww.clipper != nil {
parts, err := ww.clipper.Clip(way.Geom.Geom)
if err != nil {
log.Println(err)
continue
}
for _, g := range parts {
way := element.Way(*w)
way.Geom = &element.Geometry{g, geos.AsWkb(g)}
for _, match := range matches {
row := match.Row(&way.OSMElem)
ww.insertBuffer.Insert(match.Table.Name, row)
}
}
} else {
for _, match := range matches {
row := match.Row(&way.OSMElem)
ww.insertBuffer.Insert(match.Table.Name, row)
}
}
}
}
// if *diff {
// ww.diffCache.Coords.AddFromWay(w)
// }
}
ww.wg.Done()
}