2019-02-14 02:28:41 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2019-02-19 21:49:16 +01:00
|
|
|
"log"
|
2019-02-14 02:28:41 +01:00
|
|
|
"math/rand"
|
2019-02-19 21:49:16 +01:00
|
|
|
"net"
|
2019-02-14 02:28:41 +01:00
|
|
|
"os"
|
|
|
|
"sync"
|
2019-02-18 03:42:57 +01:00
|
|
|
"sync/atomic"
|
2019-02-14 02:28:41 +01:00
|
|
|
"time"
|
|
|
|
|
2019-02-17 03:39:41 +01:00
|
|
|
"github.com/kataras/iris/websocket2"
|
2019-02-14 02:28:41 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2019-02-19 21:49:16 +01:00
|
|
|
url = "ws://localhost:8080"
|
2019-02-14 02:28:41 +01:00
|
|
|
f *os.File
|
|
|
|
)
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
const totalClients = 16000 // max depends on the OS.
|
2019-02-18 03:42:57 +01:00
|
|
|
|
|
|
|
var connectionFailures uint64
|
|
|
|
|
|
|
|
var (
|
|
|
|
disconnectErrors []error
|
|
|
|
connectErrors []error
|
|
|
|
errMu sync.Mutex
|
|
|
|
)
|
|
|
|
|
|
|
|
func collectError(op string, err error) {
|
|
|
|
errMu.Lock()
|
|
|
|
defer errMu.Unlock()
|
|
|
|
|
|
|
|
switch op {
|
|
|
|
case "disconnect":
|
|
|
|
disconnectErrors = append(disconnectErrors, err)
|
|
|
|
case "connect":
|
|
|
|
connectErrors = append(connectErrors, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2019-02-14 02:28:41 +01:00
|
|
|
|
|
|
|
func main() {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Println("--------======Running tests...==========--------------")
|
2019-02-14 02:28:41 +01:00
|
|
|
var err error
|
|
|
|
f, err = os.Open("./test.data")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
2019-02-18 03:42:57 +01:00
|
|
|
start := time.Now()
|
|
|
|
|
2019-02-14 02:28:41 +01:00
|
|
|
wg := new(sync.WaitGroup)
|
2019-02-18 03:42:57 +01:00
|
|
|
for i := 0; i < totalClients/4; i++ {
|
2019-02-14 02:28:41 +01:00
|
|
|
wg.Add(1)
|
|
|
|
go connect(wg, 5*time.Second)
|
|
|
|
}
|
|
|
|
|
2019-02-18 03:42:57 +01:00
|
|
|
for i := 0; i < totalClients/4; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
waitTime := time.Duration(rand.Intn(5)) * time.Millisecond
|
|
|
|
time.Sleep(waitTime)
|
|
|
|
go connect(wg, 7*time.Second+waitTime)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < totalClients/4; i++ {
|
2019-02-14 02:28:41 +01:00
|
|
|
wg.Add(1)
|
|
|
|
waitTime := time.Duration(rand.Intn(10)) * time.Millisecond
|
|
|
|
time.Sleep(waitTime)
|
2019-02-19 21:49:16 +01:00
|
|
|
go connect(wg, 9*time.Second+waitTime)
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
2019-02-18 03:42:57 +01:00
|
|
|
for i := 0; i < totalClients/4; i++ {
|
|
|
|
wg.Add(1)
|
2019-02-19 21:49:16 +01:00
|
|
|
waitTime := time.Duration(rand.Intn(5)) * time.Millisecond
|
2019-02-18 03:42:57 +01:00
|
|
|
time.Sleep(waitTime)
|
2019-02-19 21:49:16 +01:00
|
|
|
go connect(wg, 14*time.Second+waitTime)
|
2019-02-18 03:42:57 +01:00
|
|
|
}
|
|
|
|
|
2019-02-14 02:28:41 +01:00
|
|
|
wg.Wait()
|
2019-02-19 21:49:16 +01:00
|
|
|
|
|
|
|
log.Printf("execution time [%s]", time.Since(start))
|
|
|
|
log.Println()
|
2019-02-18 03:42:57 +01:00
|
|
|
|
|
|
|
if connectionFailures > 0 {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("Finished with %d/%d connection failures. Please close the server-side manually.\n", connectionFailures, totalClients)
|
2019-02-18 03:42:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if n := len(connectErrors); n > 0 {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("Finished with %d connect errors:\n", n)
|
2019-02-18 03:42:57 +01:00
|
|
|
var lastErr error
|
|
|
|
var sameC int
|
|
|
|
|
|
|
|
for i, err := range connectErrors {
|
|
|
|
if lastErr != nil {
|
|
|
|
if lastErr.Error() == err.Error() {
|
|
|
|
sameC++
|
|
|
|
continue
|
2019-02-19 21:49:16 +01:00
|
|
|
} else {
|
|
|
|
_, ok := lastErr.(*net.OpError)
|
|
|
|
if ok {
|
|
|
|
if _, ok = err.(*net.OpError); ok {
|
|
|
|
sameC++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2019-02-18 03:42:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if sameC > 0 {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("and %d more like this...\n", sameC)
|
2019-02-18 03:42:57 +01:00
|
|
|
sameC = 0
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("[%d] - %v\n", i+1, err)
|
2019-02-18 03:42:57 +01:00
|
|
|
lastErr = err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if n := len(disconnectErrors); n > 0 {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("Finished with %d disconnect errors\n", n)
|
2019-02-18 03:42:57 +01:00
|
|
|
for i, err := range disconnectErrors {
|
|
|
|
if err == websocket.ErrAlreadyDisconnected {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("[%d] - %v\n", i+1, err)
|
2019-02-18 03:42:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if connectionFailures == 0 && len(connectErrors) == 0 && len(disconnectErrors) == 0 {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Println("ALL OK.")
|
2019-02-18 03:42:57 +01:00
|
|
|
}
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Println("--------================--------------")
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func connect(wg *sync.WaitGroup, alive time.Duration) {
|
2019-02-17 03:39:41 +01:00
|
|
|
c, err := websocket.Dial(nil, url, websocket.ConnectionConfig{})
|
2019-02-14 02:28:41 +01:00
|
|
|
if err != nil {
|
2019-02-18 03:42:57 +01:00
|
|
|
atomic.AddUint64(&connectionFailures, 1)
|
|
|
|
collectError("connect", err)
|
|
|
|
wg.Done()
|
|
|
|
return
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
c.OnError(func(err error) {
|
2019-02-19 21:49:16 +01:00
|
|
|
log.Printf("error: %v", err)
|
2019-02-14 02:28:41 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
disconnected := false
|
|
|
|
c.OnDisconnect(func() {
|
2019-02-19 21:49:16 +01:00
|
|
|
// log.Printf("I am disconnected after [%s].\n", alive)
|
2019-02-14 02:28:41 +01:00
|
|
|
disconnected = true
|
|
|
|
})
|
|
|
|
|
|
|
|
c.On("chat", func(message string) {
|
2019-02-19 21:49:16 +01:00
|
|
|
// log.Printf("\n%s\n", message)
|
2019-02-14 02:28:41 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
time.Sleep(alive)
|
|
|
|
if err := c.Disconnect(); err != nil {
|
2019-02-18 03:42:57 +01:00
|
|
|
collectError("disconnect", err)
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
scanner := bufio.NewScanner(f)
|
|
|
|
for !disconnected {
|
|
|
|
if !scanner.Scan() || scanner.Err() != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2019-02-18 03:42:57 +01:00
|
|
|
if text := scanner.Text(); len(text) > 1 {
|
|
|
|
c.Emit("chat", text)
|
|
|
|
}
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
}
|