2013-08-13 13:47:01 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"flag"
|
2013-08-17 16:14:08 +02:00
|
|
|
vegeta "github.com/tsenart/vegeta/lib"
|
2013-08-14 17:48:51 +02:00
|
|
|
"io"
|
2013-08-13 13:47:01 +02:00
|
|
|
"log"
|
2013-08-13 18:51:00 +02:00
|
|
|
"os"
|
2013-08-14 18:24:20 +02:00
|
|
|
"runtime"
|
2013-08-13 13:47:01 +02:00
|
|
|
"time"
|
|
|
|
|
)
|
|
|
|
|
|
2013-08-14 18:24:20 +02:00
|
|
|
func init() {
|
|
|
|
|
runtime.GOMAXPROCS(runtime.NumCPU())
|
|
|
|
|
}
|
|
|
|
|
|
2013-08-13 13:47:01 +02:00
|
|
|
func main() {
|
|
|
|
|
var (
|
2013-08-16 20:08:10 +02:00
|
|
|
rate = flag.Uint64("rate", 50, "Requests per second")
|
2013-08-13 18:51:00 +02:00
|
|
|
targetsf = flag.String("targets", "targets.txt", "Targets file")
|
|
|
|
|
ordering = flag.String("ordering", "random", "Attack ordering [sequential, random]")
|
|
|
|
|
duration = flag.Duration("duration", 10*time.Second, "Duration of the test")
|
2013-08-17 19:52:29 +02:00
|
|
|
reporter = flag.String("reporter", "text", "Reporter to use [text, plot:timings]")
|
2013-08-14 17:48:51 +02:00
|
|
|
output = flag.String("output", "stdout", "Reporter output file")
|
2013-08-13 13:47:01 +02:00
|
|
|
)
|
|
|
|
|
flag.Parse()
|
|
|
|
|
|
2013-08-13 18:51:00 +02:00
|
|
|
if flag.NFlag() == 0 {
|
|
|
|
|
flag.Usage()
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if *rate == 0 {
|
|
|
|
|
log.Fatal("rate can't be zero")
|
2013-08-13 13:47:01 +02:00
|
|
|
}
|
2013-08-16 19:09:06 +02:00
|
|
|
|
2013-08-17 16:14:08 +02:00
|
|
|
targets, err := vegeta.NewTargetsFromFile(*targetsf)
|
2013-08-13 15:01:41 +02:00
|
|
|
if err != nil {
|
2013-08-13 13:47:01 +02:00
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
2013-08-16 19:11:42 +02:00
|
|
|
switch *ordering {
|
|
|
|
|
case "random":
|
2013-08-16 19:38:57 +02:00
|
|
|
targets.Shuffle(time.Now().UnixNano())
|
2013-08-16 19:11:42 +02:00
|
|
|
case "sequential":
|
|
|
|
|
break
|
|
|
|
|
default:
|
2013-08-13 18:51:00 +02:00
|
|
|
log.Fatalf("Unknown ordering %s", *ordering)
|
2013-08-13 13:47:01 +02:00
|
|
|
}
|
|
|
|
|
|
2013-08-13 18:51:00 +02:00
|
|
|
if *duration == 0 {
|
|
|
|
|
log.Fatal("Duration provided is invalid")
|
2013-08-13 13:47:01 +02:00
|
|
|
}
|
|
|
|
|
|
2013-08-17 16:14:08 +02:00
|
|
|
var rep vegeta.Reporter
|
2013-08-13 18:51:00 +02:00
|
|
|
switch *reporter {
|
|
|
|
|
case "text":
|
2013-08-17 16:14:08 +02:00
|
|
|
rep = vegeta.NewTextReporter()
|
2013-08-17 19:52:29 +02:00
|
|
|
case "plot:timings":
|
|
|
|
|
rep = vegeta.NewTimingsPlotReporter()
|
2013-08-13 18:51:00 +02:00
|
|
|
default:
|
2013-08-17 19:52:29 +02:00
|
|
|
log.Println("Reporter provided is not supported. using text")
|
2013-08-17 16:14:08 +02:00
|
|
|
rep = vegeta.NewTextReporter()
|
2013-08-13 18:51:00 +02:00
|
|
|
}
|
|
|
|
|
|
2013-08-16 19:13:49 +02:00
|
|
|
var out io.Writer
|
2013-08-14 17:48:51 +02:00
|
|
|
switch *output {
|
|
|
|
|
case "stdout":
|
|
|
|
|
out = os.Stdout
|
|
|
|
|
default:
|
2013-08-16 19:13:49 +02:00
|
|
|
file, err := os.Create(*output)
|
2013-08-14 17:48:51 +02:00
|
|
|
if err != nil {
|
2013-08-14 18:04:20 +02:00
|
|
|
log.Fatalf("Couldn't open `%s` for writing report: %s", *output, err)
|
2013-08-14 17:48:51 +02:00
|
|
|
}
|
2013-08-16 19:13:49 +02:00
|
|
|
defer file.Close()
|
|
|
|
|
out = file
|
2013-08-14 17:48:51 +02:00
|
|
|
}
|
2013-08-14 18:04:20 +02:00
|
|
|
|
|
|
|
|
log.Printf("Vegeta is attacking %d targets in %s order for %s...\n", len(targets), *ordering, *duration)
|
2013-08-17 16:14:08 +02:00
|
|
|
vegeta.Attack(targets, *rate, *duration, rep)
|
2013-08-14 18:04:20 +02:00
|
|
|
log.Println("Done!")
|
|
|
|
|
|
2013-08-14 17:48:51 +02:00
|
|
|
log.Printf("Writing report to '%s'...", *output)
|
2013-08-23 16:17:52 +02:00
|
|
|
if err = rep.Report(out); err != nil {
|
|
|
|
|
log.Printf("Failed to report: %s", err)
|
2013-08-13 13:47:01 +02:00
|
|
|
}
|
|
|
|
|
}
|