parallel hashing
This commit is contained in:
parent
f82decc6dc
commit
5e10c8f583
126
check.go
126
check.go
@ -6,13 +6,13 @@ import (
|
||||
"crypto/sha1"
|
||||
"crypto/sha256"
|
||||
"crypto/sha512"
|
||||
"errors"
|
||||
"fmt"
|
||||
"hash"
|
||||
"io"
|
||||
"log"
|
||||
"os"
|
||||
"strings"
|
||||
"sync"
|
||||
)
|
||||
|
||||
type checksum struct {
|
||||
@ -43,61 +43,113 @@ func parseCS(line string) (checksum, error) {
|
||||
return checksum{filename: f, hash: hsh, checksum: cs}, nil
|
||||
}
|
||||
|
||||
func check(files []string) error {
|
||||
streams := []io.ReadCloser{}
|
||||
defer func() {
|
||||
for _, stream := range streams {
|
||||
stream.Close()
|
||||
}
|
||||
}()
|
||||
type input struct {
|
||||
f io.ReadCloser
|
||||
err error
|
||||
}
|
||||
|
||||
type work struct {
|
||||
cs checksum
|
||||
err error
|
||||
}
|
||||
|
||||
func streams(files []string) chan input {
|
||||
r := make(chan input)
|
||||
|
||||
go func() {
|
||||
for _, name := range files {
|
||||
f, err := os.Open(name)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
streams = append(streams, f)
|
||||
r <- input{f, err}
|
||||
}
|
||||
if len(files) == 0 {
|
||||
streams = append(streams, os.Stdin)
|
||||
r <- input{f: os.Stdin}
|
||||
}
|
||||
close(r)
|
||||
}()
|
||||
|
||||
jobs := []checksum{}
|
||||
for _, stream := range streams {
|
||||
s := bufio.NewScanner(stream)
|
||||
return r
|
||||
}
|
||||
|
||||
func check(files []string) chan error {
|
||||
jobs := make(chan work)
|
||||
|
||||
go func() {
|
||||
for stream := range streams(files) {
|
||||
if stream.err != nil {
|
||||
jobs <- work{err: stream.err}
|
||||
break
|
||||
}
|
||||
s := bufio.NewScanner(stream.f)
|
||||
for s.Scan() {
|
||||
cs, err := parseCS(s.Text())
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
jobs = append(jobs, cs)
|
||||
jobs <- work{cs, err}
|
||||
}
|
||||
stream.f.Close()
|
||||
if s.Err() != nil {
|
||||
return s.Err()
|
||||
jobs <- work{err: s.Err()}
|
||||
}
|
||||
}
|
||||
close(jobs)
|
||||
}()
|
||||
|
||||
results := []<-chan error{}
|
||||
|
||||
workers := 8
|
||||
for w := 0; w < workers; w++ {
|
||||
results = append(results, compute(jobs))
|
||||
}
|
||||
|
||||
errs := 0
|
||||
for _, job := range jobs {
|
||||
f, err := os.Open(job.filename)
|
||||
if err != nil {
|
||||
return fmt.Errorf("open: %v", err)
|
||||
return merge(results)
|
||||
}
|
||||
|
||||
func merge(cs []<-chan error) chan error {
|
||||
out := make(chan error)
|
||||
|
||||
var wg sync.WaitGroup
|
||||
|
||||
output := func(c <-chan error) {
|
||||
for n := range c {
|
||||
out <- n
|
||||
}
|
||||
if _, err := io.Copy(job.hash, f); err != nil {
|
||||
wg.Done()
|
||||
}
|
||||
|
||||
wg.Add(len(cs))
|
||||
for _, c := range cs {
|
||||
go output(c)
|
||||
}
|
||||
|
||||
go func() {
|
||||
wg.Wait()
|
||||
close(out)
|
||||
}()
|
||||
return out
|
||||
}
|
||||
|
||||
func compute(jobs chan work) chan error {
|
||||
r := make(chan error)
|
||||
go func() {
|
||||
for job := range jobs {
|
||||
if job.err != nil {
|
||||
log.Printf("%+v", job.err)
|
||||
continue
|
||||
}
|
||||
f, err := os.Open(job.cs.filename)
|
||||
if err != nil {
|
||||
r <- fmt.Errorf("open: %v", err)
|
||||
continue
|
||||
}
|
||||
if _, err := io.Copy(job.cs.hash, f); err != nil {
|
||||
log.Printf("%+v", err)
|
||||
}
|
||||
f.Close()
|
||||
if fmt.Sprintf("%x", job.hash.Sum(nil)) == job.checksum {
|
||||
fmt.Printf("%s: OK\n", job.filename)
|
||||
if fmt.Sprintf("%x", job.cs.hash.Sum(nil)) == job.cs.checksum {
|
||||
fmt.Printf("%s: OK\n", job.cs.filename)
|
||||
} else {
|
||||
errs++
|
||||
fmt.Fprintf(os.Stderr, "%s: bad\n", job.filename)
|
||||
r <- fmt.Errorf("%s: bad", job.cs.filename)
|
||||
}
|
||||
}
|
||||
|
||||
var err error
|
||||
if errs != 0 {
|
||||
err = errors.New("bad files found")
|
||||
}
|
||||
return err
|
||||
close(r)
|
||||
}()
|
||||
return r
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user