Skip to content

Instantly share code, notes, and snippets.

@LaPingvino
Created October 31, 2020 18:47
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save LaPingvino/d992074bf89ac4ba8ec061839646125e to your computer and use it in GitHub Desktop.
Save LaPingvino/d992074bf89ac4ba8ec061839646125e to your computer and use it in GitHub Desktop.
package main
import (
"os"
"fmt"
"bufio"
"encoding/json"
"strings"
"strconv"
)
type Name struct {
Name string
Names map[string]string
AdminLevel int
Place string
Lat float64
Long float64
}
func main() {
if len(os.Args) < 2 {
panic("need to invoke with planet.osm file")
}
file, err := os.Open(os.Args[1])
if err != nil {
panic("problem opening file: " + err.Error())
}
fb := bufio.NewReader(file)
var node, place bool
var line string
var name Name
var counter, old int64
for err == nil {
line, err = fb.ReadString('\n')
counter += int64(len(line))
if counter > old + 1000000000 {
old = counter
print(".")
}
if len(line) < 7 {
continue
}
switch {
case strings.Contains(line,"<node"):
node = true
lati := strings.Index(line, "lat=\"")
if lati == -1 {
node = false
continue
}
lats := line[lati+5:strings.Index(line[lati+5:], "\"")+lati+5]
loni := strings.Index(line, "lon=\"")
if loni == -1 {
node = false
continue
}
lons := line[loni+5:strings.Index(line[loni+5:], "\"")+loni+5]
lat, _ := strconv.ParseFloat(lats, 64)
lon, _ := strconv.ParseFloat(lons, 64)
name = Name{Lat: lat, Long: lon, Names: map[string]string{}}
case strings.Contains(line, "k=\"name\""):
if !node {
continue
}
namei := strings.Index(line, "v=\"")
if namei == -1 {
node = false
continue
}
name.Name = line[namei+3:strings.Index(line[namei+3:],"\"")+namei+3]
case strings.Contains(line, "k=\"name"):
if !node {
continue
}
langi := strings.Index(line, "k=\"name:")
if langi == -1 {
node = false
continue
}
namei := strings.Index(line, "v=\"")
if namei == -1 {
node = false
continue
}
lang := line[langi+8:strings.Index(line[langi+8:],"\"")+langi+8]
name.Names[lang] = line[namei+3:strings.Index(line[namei+3:],"\"")+namei+3]
case strings.Contains(line, "k=\"alt_name"):
if !node {
continue
}
langi := strings.Index(line, "k=\"")
if langi == -1 {
node = false
continue
}
namei := strings.Index(line, "v=\"")
if namei == -1 {
node = false
continue
}
lang := line[langi+3:strings.Index(line[langi+3:],"\"")+langi+3]
name.Names[lang] = line[namei+3:strings.Index(line[namei+3:],"\"")+namei+3]
case strings.Contains(line, "k=\"place\""):
if !node {
continue
}
placei := strings.Index(line, "v=\"")
if placei == -1 {
node = false
continue
}
name.Place = line[placei+3:strings.Index(line[placei+3:],"\"")+placei+3]
place = true
case strings.Contains(line, "k=\"admin_level\""):
if !node {
continue
}
admi := strings.Index(line, "v=\"")
if admi == -1 {
node = false
continue
}
admlvl := line[admi+3:strings.Index(line[admi+3:],"\"")+admi+3]
name.AdminLevel, _ = strconv.Atoi(admlvl)
case strings.Contains(line, "</node>"):
if !node {
continue
}
if !place {
node = false
place = false
continue
}
if name.Name == "" {
node = false
continue
}
output, _ := json.Marshal(name)
fmt.Println(string(output))
node = false
place = false
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment