Documentation ¶
Overview ¶
Example (Stats) ¶
ExampleStats demonstrates how to read a full file and gather some stats. This is similar to `osmconvert --out-statistics`
package main import ( "context" "fmt" "math" "os" "time" "github.com/nextmv-io/osm" "github.com/nextmv-io/osm/osmpbf" ) // ExampleStats demonstrates how to read a full file and gather some stats. // This is similar to `osmconvert --out-statistics` func main() { f, err := os.Open("../testdata/delaware-latest.osm.pbf") if err != nil { fmt.Printf("could not open file: %v", err) os.Exit(1) } defer f.Close() nodes, ways, relations := 0, 0, 0 stats := newElementStats() minLat, maxLat := math.MaxFloat64, -math.MaxFloat64 minLon, maxLon := math.MaxFloat64, -math.MaxFloat64 minTS, maxTS := time.Date(2100, 1, 1, 0, 0, 0, 0, time.UTC), time.Time{} var ( maxNodeRefs int maxNodeRefsID osm.WayID ) var ( maxRelRefs int maxRelRefsID osm.RelationID ) scanner := osmpbf.New(context.Background(), f, 3) defer scanner.Close() for scanner.Scan() { var ts time.Time switch e := scanner.Object().(type) { case *osm.Node: nodes++ ts = e.Timestamp stats.Add(e.ElementID(), e.Tags) if e.Lat > maxLat { maxLat = e.Lat } if e.Lat < minLat { minLat = e.Lat } if e.Lon > maxLon { maxLon = e.Lon } if e.Lon < minLon { minLon = e.Lon } case *osm.Way: ways++ ts = e.Timestamp stats.Add(e.ElementID(), e.Tags) if l := len(e.Nodes); l > maxNodeRefs { maxNodeRefs = l maxNodeRefsID = e.ID } case *osm.Relation: relations++ ts = e.Timestamp stats.Add(e.ElementID(), e.Tags) if l := len(e.Members); l > maxRelRefs { maxRelRefs = l maxRelRefsID = e.ID } } if ts.After(maxTS) { maxTS = ts } if ts.Before(minTS) { minTS = ts } } if err := scanner.Err(); err != nil { fmt.Printf("scanner returned error: %v", err) os.Exit(1) } fmt.Println("timestamp min:", minTS.Format(time.RFC3339)) fmt.Println("timestamp max:", maxTS.Format(time.RFC3339)) fmt.Printf("lon min: %0.7f\n", minLon) fmt.Printf("lon max: %0.7f\n", maxLon) fmt.Printf("lat min: %0.7f\n", minLat) fmt.Printf("lat max: %0.7f\n", maxLat) fmt.Println("nodes:", nodes) fmt.Println("ways:", ways) fmt.Println("relations:", relations) fmt.Println("version max:", stats.MaxVersion) fmt.Println("node id min:", stats.Ranges[osm.TypeNode].Min) fmt.Println("node id max:", stats.Ranges[osm.TypeNode].Max) fmt.Println("way id min:", stats.Ranges[osm.TypeWay].Min) fmt.Println("way id max:", stats.Ranges[osm.TypeWay].Max) fmt.Println("relation id min:", stats.Ranges[osm.TypeRelation].Min) fmt.Println("relation id max:", stats.Ranges[osm.TypeRelation].Max) fmt.Println("keyval pairs max:", stats.MaxTags) fmt.Println("keyval pairs max object:", stats.MaxTagsID.Type(), stats.MaxTagsID.Ref()) fmt.Println("noderefs max:", maxNodeRefs) fmt.Println("noderefs max object: way", maxNodeRefsID) fmt.Println("relrefs max:", maxRelRefs) fmt.Println("relrefs max object: relation", maxRelRefsID) } // Stats is a shared bit of code to accumulate stats from the element ids. type elementStats struct { Ranges map[osm.Type]*idRange MaxVersion int MaxTags int MaxTagsID osm.ElementID } type idRange struct { Min, Max int64 } func newElementStats() *elementStats { return &elementStats{ Ranges: map[osm.Type]*idRange{ osm.TypeNode: {Min: math.MaxInt64}, osm.TypeWay: {Min: math.MaxInt64}, osm.TypeRelation: {Min: math.MaxInt64}, }, } } func (s *elementStats) Add(id osm.ElementID, tags osm.Tags) { s.Ranges[id.Type()].Add(id.Ref()) if v := id.Version(); v > s.MaxVersion { s.MaxVersion = v } if l := len(tags); l > s.MaxTags { s.MaxTags = l s.MaxTagsID = id } } func (r *idRange) Add(ref int64) { if ref > r.Max { r.Max = ref } if ref < r.Min { r.Min = ref } }
Output: timestamp min: 2007-10-16T15:59:24Z timestamp max: 2016-08-10T17:32:02Z lon min: -76.1748935 lon max: -74.4929376 lat min: 38.0273717 lat max: 39.9688859 nodes: 723870 ways: 73144 relations: 1644 version max: 421 node id min: 75385503 node id max: 4343778904 way id min: 9650669 way id max: 436488690 relation id min: 82010 relation id max: 6462005 keyval pairs max: 276 keyval pairs max object: relation 148838 noderefs max: 1811 noderefs max object: way 318739264 relrefs max: 7177 relrefs max object: relation 4799100
Index ¶
- func EncodeDenseNode(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, ...)
- func EncodeLatLon(value float64, offset int64, granularity int32) int64
- func EncodeNode(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, ...) *osmpbf.Node
- func EncodeRelation(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, ...) *osmpbf.Relation
- func EncodeString(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, value string) int32
- func EncodeTimestamp(timestamp time.Time, dateGranularity int32) int64
- func EncodeWay(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, way *osm.Way) *osmpbf.Way
- type Encoder
- type Header
- type Scanner
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func EncodeDenseNode ¶
func EncodeDenseNode(block *osmpbf.PrimitiveBlock, reverseStringTable map[string]int, groupDense *osmpbf.DenseNodes, current, previous *osm.Node)
func EncodeNode ¶
func EncodeRelation ¶
func EncodeString ¶
Types ¶
type Encoder ¶
Encoder is an interface for encoding osm data. The format written is the osm pbf format.
type Header ¶
type Header struct { Bounds *osm.Bounds RequiredFeatures []string OptionalFeatures []string WritingProgram string Source string ReplicationTimestamp time.Time ReplicationSeqNum uint64 ReplicationBaseURL string }
Header contains the contents of the header in the pbf file.
type Scanner ¶
type Scanner struct { // Skip element types that are not needed. The data is skipped // at the encoded protobuf level, but each block still needs to be decompressed. SkipNodes bool SkipWays bool SkipRelations bool // If the Filter function is false, the element well be skipped // at the decoding level. The functions should be fast, they block the // decoder, there are `procs` number of concurrent decoders. // Elements can be stored if the function returns true. Memory is // reused if the filter returns false. FilterNode func(*osm.Node) bool FilterWay func(*osm.Way) bool FilterRelation func(*osm.Relation) bool // contains filtered or unexported fields }
Scanner provides a convenient interface reading a stream of osm data from a file or url. Successive calls to the Scan method will step through the data.
Scanning stops unrecoverably at EOF, the first I/O error, the first xml error or the context being cancelled. When a scan stops, the reader may have advanced arbitrarily far past the last token.
The Scanner API is based on bufio.Scanner https://golang.org/pkg/bufio/#Scanner
func New ¶
New returns a new Scanner to read from r. procs indicates amount of paralellism, when reading blocks which will off load the unzipping/decoding to multiple cpus.
func (*Scanner) Close ¶
Close cleans up all the reading goroutines, it does not close the underlying reader.
func (*Scanner) FullyScannedBytes ¶
FullyScannedBytes returns the number of bytes that have been read and fully scanned. OSM protobuf files contain data blocks with 8000 nodes each. The returned value contains the bytes for the blocks that have been fully scanned.
A user can use this number of seek forward in a file and begin reading mid-data. Note that while elements are usually sorted by Type, ID, Version in OSM protobuf files, versions of given element may span blocks.
func (*Scanner) Header ¶
Header returns the pbf file header with interesting information about how it was created.
func (*Scanner) Object ¶
Object returns the most recent token generated by a call to Scan as a new osm.Object. Currently osm.pbf files only contain nodes, ways and relations. This method returns an object so match the osm.Scanner interface and allows this Scanner to share an interface with osmxml.Scanner.
func (*Scanner) PreviousFullyScannedBytes ¶
PreviousFullyScannedBytes returns the previous value of FullyScannedBytes. This is interesting because it's not totally clear if a feature spans a block. For example, if one quits after finding the first relation, upon restarting there is no way of knowing if the first relation is complete, so skip it. But if this relation is the first relation in the file we'll skip a full relation.
func (*Scanner) Scan ¶
Scan advances the Scanner to the next element, which will then be available through the Element method. It returns false when the scan stops, either by reaching the end of the input, an io error, an xml error or the context being cancelled. After Scan returns false, the Err method will return any error that occurred during scanning, except that if it was io.EOF, Err will return nil.