101 lines
2.1 KiB
Go
101 lines
2.1 KiB
Go
![]() |
// For more tutorials: https://blog.learngoprogramming.com
|
||
|
//
|
||
|
// Copyright © 2018 Inanc Gumus
|
||
|
// Learn Go Programming Course
|
||
|
// License: https://creativecommons.org/licenses/by-nc-sa/4.0/
|
||
|
//
|
||
|
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"sort"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
// result stores details about a log line
|
||
|
type result struct {
|
||
|
domain string
|
||
|
visits int
|
||
|
// add more metrics when needed
|
||
|
}
|
||
|
|
||
|
// parser keep tracks of the parsing
|
||
|
type parser struct {
|
||
|
sum map[string]result // metrics per domain
|
||
|
domains []string // unique domain names
|
||
|
total int // total visits for all domains
|
||
|
lines int // number of parsed lines (for the error messages)
|
||
|
lerr error // the last error occurred
|
||
|
}
|
||
|
|
||
|
// newParser creates and returns a new parser
|
||
|
func newParser() *parser {
|
||
|
return &parser{sum: make(map[string]result)}
|
||
|
}
|
||
|
|
||
|
// parse result from a log line
|
||
|
func (p *parser) parse(line string) (r result) {
|
||
|
if p.lerr != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
p.lines++
|
||
|
|
||
|
fields := strings.Fields(line)
|
||
|
if len(fields) != 2 {
|
||
|
p.lerr = fmt.Errorf("wrong input: %v (line #%d)", fields, p.lines)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
var err error
|
||
|
|
||
|
r.domain = fields[0]
|
||
|
r.visits, err = strconv.Atoi(fields[1])
|
||
|
|
||
|
if r.visits < 0 || err != nil {
|
||
|
p.lerr = fmt.Errorf("wrong input: %q (line #%d)", fields[1], p.lines)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// update the parsing results
|
||
|
func (p *parser) update(r result) {
|
||
|
if p.lerr != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// collect unique domains for easy access to sum map
|
||
|
if _, ok := p.sum[r.domain]; !ok {
|
||
|
p.domains = append(p.domains, r.domain)
|
||
|
}
|
||
|
|
||
|
// keep track of total visits
|
||
|
p.total += r.visits
|
||
|
|
||
|
// group the log lines by domain
|
||
|
p.sum[r.domain] = result{
|
||
|
domain: r.domain,
|
||
|
visits: r.visits + p.sum[r.domain].visits,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// summarize the parsing results
|
||
|
func (p *parser) summarize() {
|
||
|
sort.Strings(p.domains)
|
||
|
|
||
|
fmt.Printf("%-30s %10s\n", "DOMAIN", "VISITS")
|
||
|
fmt.Println(strings.Repeat("-", 45))
|
||
|
|
||
|
for _, domain := range p.domains {
|
||
|
fmt.Printf("%-30s %10d\n", domain, p.sum[domain].visits)
|
||
|
}
|
||
|
fmt.Printf("\n%-30s %10d\n", "TOTAL", p.total)
|
||
|
}
|
||
|
|
||
|
// err returns the last error encountered
|
||
|
func (p *parser) err() error {
|
||
|
return p.lerr
|
||
|
}
|