package writer import ( "fmt" "goposm/cache" "goposm/database" "goposm/element" "goposm/geom" "goposm/geom/geos" "goposm/mapping" "goposm/proj" "goposm/stats" "log" "sync" ) type RelationWriter struct { OsmElemWriter rel chan *element.Relation tagMatcher *mapping.TagMatcher } func NewRelationWriter(osmCache *cache.OSMCache, diffCache *cache.DiffCache, rel chan *element.Relation, insertBuffer database.RowInserter, tagMatcher *mapping.TagMatcher, progress *stats.Statistics) *OsmElemWriter { rw := RelationWriter{ OsmElemWriter: OsmElemWriter{ osmCache: osmCache, diffCache: diffCache, progress: progress, wg: &sync.WaitGroup{}, insertBuffer: insertBuffer, }, rel: rel, tagMatcher: tagMatcher, } rw.OsmElemWriter.writer = &rw return &rw.OsmElemWriter } func (rw *RelationWriter) loop() { geos := geos.NewGeos() defer geos.Finish() NextRel: for r := range rw.rel { rw.progress.AddRelations(1) err := rw.osmCache.Ways.FillMembers(r.Members) if err != nil { if err != cache.NotFound { fmt.Println(err) } continue NextRel } for _, m := range r.Members { if m.Way == nil { continue } err := rw.osmCache.Coords.FillWay(m.Way) if err != nil { if err != cache.NotFound { fmt.Println(err) } continue NextRel } proj.NodesToMerc(m.Way.Nodes) } // BuildRelation updates r.Members but we need all of them // for the diffCache allMembers := r.Members err = geom.BuildRelation(r) if err != nil { if r.Geom != nil && r.Geom.Geom != nil { geos.Destroy(r.Geom.Geom) } if err, ok := err.(ErrorLevel); ok { if err.Level() <= 0 { continue NextRel } } log.Println(err) continue NextRel } if matches := rw.tagMatcher.Match(&r.Tags); len(matches) > 0 { if rw.clipper != nil { parts, err := rw.clipper.Clip(r.Geom.Geom) if err != nil { log.Println(err) continue NextRel } for _, g := range parts { rel := element.Relation(*r) rel.Geom = &element.Geometry{g, geos.AsWkb(g)} rw.insertMatches(&rel.OSMElem, matches) } } else { rw.insertMatches(&r.OSMElem, matches) } err := rw.osmCache.InsertedWays.PutMembers(r.Members) if err != nil { fmt.Println(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) } } } } geos.Destroy(r.Geom.Geom) } rw.wg.Done() }