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 | ||
|  | } |