package main import ( "bufio" "database/sql" "fmt" "io" "regexp" "strconv" "strings" ) func speckStream(r io.ReadCloser, c chan exoSystem) { defer close(c) defer r.Close() keep := regexp.MustCompile(`^\s*[\d-]`) br := bufio.NewReader(r) for { line, err := br.ReadBytes('\n') switch err { case io.EOF: return case nil: break default: log_error("unable to stream speck file: %v", err) return } if !keep.Match(line) { continue } planet := parseSpeckLine(line) c <- *planet } } type exoSystem struct { x, y, z float64 planets int name string } func (e exoSystem) Store(db *sql.DB) { _, err := db.Exec(` insert into planets (name, x, y, z, planets) values (?, ?, ?, ?, ?) ;`, e.name, e.x, e.y, e.z, e.planets) if err != nil { log_error("%v", err) } } func countPlanets() (int, error) { row := db.QueryRow(`select count(*) from planets`) var n int err := row.Scan(&n) return n, err } func (e exoSystem) String() string { return fmt.Sprintf("", e.name, e.x, e.y, e.z, e.planets) } func parseSpeckLine(line []byte) *exoSystem { parts := strings.Split(string(line), " ") var err error var g errorGroup s := new(exoSystem) s.x, err = strconv.ParseFloat(parts[0], 64) g.AddError(err) s.y, err = strconv.ParseFloat(parts[1], 64) g.AddError(err) s.z, err = strconv.ParseFloat(parts[2], 64) g.AddError(err) s.planets, err = strconv.Atoi(parts[3]) g.AddError(err) s.name = strings.TrimSpace(strings.Join(parts[7:], " ")) if g != nil { log_error("%v", g) } return s }