2019-02-14 02:28:41 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2019-02-19 21:49:16 +01:00
|
|
|
"log"
|
2019-02-14 02:28:41 +01:00
|
|
|
"os"
|
2019-02-19 21:49:16 +01:00
|
|
|
"runtime"
|
2019-02-14 02:28:41 +01:00
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/kataras/iris"
|
2019-02-22 20:24:10 +01:00
|
|
|
"github.com/kataras/iris/websocket"
|
2019-02-14 02:28:41 +01:00
|
|
|
)
|
|
|
|
|
2019-02-23 06:23:10 +01:00
|
|
|
const (
|
|
|
|
endpoint = "localhost:8080"
|
|
|
|
totalClients = 16000 // max depends on the OS.
|
|
|
|
verbose = false
|
|
|
|
maxC = 0
|
|
|
|
)
|
2019-02-14 02:28:41 +01:00
|
|
|
|
|
|
|
func main() {
|
|
|
|
ws := websocket.New(websocket.Config{})
|
|
|
|
ws.OnConnection(handleConnection)
|
2019-02-19 21:49:16 +01:00
|
|
|
|
|
|
|
// websocket.Config{PingPeriod: ((60 * time.Second) * 9) / 10}
|
2019-02-14 02:28:41 +01:00
|
|
|
|
|
|
|
go func() {
|
2019-02-23 06:23:10 +01:00
|
|
|
dur := 4 * time.Second
|
2019-02-19 21:49:16 +01:00
|
|
|
if totalClients >= 64000 {
|
2019-02-23 06:23:10 +01:00
|
|
|
// if more than 64000 then let's perform those checks every 24 seconds instead,
|
|
|
|
// either way works.
|
2019-02-19 21:49:16 +01:00
|
|
|
dur = 24 * time.Second
|
|
|
|
}
|
|
|
|
t := time.NewTicker(dur)
|
2019-02-23 17:35:29 +01:00
|
|
|
defer func() {
|
|
|
|
t.Stop()
|
|
|
|
printMemUsage()
|
|
|
|
os.Exit(0)
|
|
|
|
}()
|
2019-02-19 21:49:16 +01:00
|
|
|
|
|
|
|
var started bool
|
2019-02-14 02:28:41 +01:00
|
|
|
for {
|
|
|
|
<-t.C
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
n := ws.GetTotalConnections()
|
|
|
|
if n > 0 {
|
|
|
|
started = true
|
2019-02-23 06:23:10 +01:00
|
|
|
if maxC > 0 && n > maxC {
|
|
|
|
log.Printf("current connections[%d] > MaxConcurrentConnections[%d]", n, maxC)
|
|
|
|
return
|
|
|
|
}
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
2019-02-19 21:49:16 +01:00
|
|
|
if started {
|
2019-02-23 06:23:10 +01:00
|
|
|
disconnectedN := atomic.LoadUint64(&totalDisconnected)
|
|
|
|
connectedN := atomic.LoadUint64(&totalConnected)
|
|
|
|
if disconnectedN == totalClients && connectedN == totalClients {
|
2019-02-19 21:49:16 +01:00
|
|
|
if n != 0 {
|
|
|
|
log.Println("ALL CLIENTS DISCONNECTED BUT LEFTOVERS ON CONNECTIONS LIST.")
|
|
|
|
} else {
|
|
|
|
log.Println("ALL CLIENTS DISCONNECTED SUCCESSFULLY.")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
} else if n == 0 {
|
2019-02-23 06:23:10 +01:00
|
|
|
log.Printf("%d/%d CLIENTS WERE NOT CONNECTED AT ALL. CHECK YOUR OS NET SETTINGS. THE REST CLIENTS WERE DISCONNECTED SUCCESSFULLY.\n",
|
2019-02-19 21:49:16 +01:00
|
|
|
totalClients-totalConnected, totalClients)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2019-02-22 20:24:10 +01:00
|
|
|
app := iris.New()
|
2019-02-23 17:35:29 +01:00
|
|
|
app.Get("/", func(ctx iris.Context) {
|
|
|
|
c := ws.Upgrade(ctx)
|
|
|
|
handleConnection(c)
|
|
|
|
c.Wait()
|
|
|
|
})
|
2019-02-23 06:23:10 +01:00
|
|
|
app.Run(iris.Addr(endpoint), iris.WithoutServerError(iris.ErrServerClosed))
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
2019-02-23 06:23:10 +01:00
|
|
|
var totalConnected uint64
|
|
|
|
|
2019-02-14 02:28:41 +01:00
|
|
|
func handleConnection(c websocket.Connection) {
|
2019-02-23 06:23:10 +01:00
|
|
|
if c.Err() != nil {
|
|
|
|
log.Fatalf("[%d] upgrade failed: %v", atomic.LoadUint64(&totalConnected)+1, c.Err())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
atomic.AddUint64(&totalConnected, 1)
|
2019-02-14 02:28:41 +01:00
|
|
|
c.OnError(func(err error) { handleErr(c, err) })
|
|
|
|
c.OnDisconnect(func() { handleDisconnect(c) })
|
|
|
|
c.On("chat", func(message string) {
|
|
|
|
c.To(websocket.Broadcast).Emit("chat", c.ID()+": "+message)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-02-23 06:23:10 +01:00
|
|
|
var totalDisconnected uint64
|
2019-02-14 02:28:41 +01:00
|
|
|
|
|
|
|
func handleDisconnect(c websocket.Connection) {
|
2019-02-23 06:23:10 +01:00
|
|
|
newC := atomic.AddUint64(&totalDisconnected, 1)
|
2019-02-22 20:24:10 +01:00
|
|
|
if verbose {
|
2019-02-23 06:23:10 +01:00
|
|
|
log.Printf("[%d] client [%s] disconnected!\n", newC, c.ID())
|
2019-02-22 20:24:10 +01:00
|
|
|
}
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func handleErr(c websocket.Connection, err error) {
|
2019-02-23 17:35:29 +01:00
|
|
|
log.Printf("client [%s] errorred: %v\n", c.ID(), err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func toMB(b uint64) uint64 {
|
|
|
|
return b / 1024 / 1024
|
|
|
|
}
|
|
|
|
|
|
|
|
func printMemUsage() {
|
|
|
|
var m runtime.MemStats
|
|
|
|
runtime.ReadMemStats(&m)
|
|
|
|
log.Printf("Alloc = %v MiB", toMB(m.Alloc))
|
|
|
|
log.Printf("\tTotalAlloc = %v MiB", toMB(m.TotalAlloc))
|
|
|
|
log.Printf("\tSys = %v MiB", toMB(m.Sys))
|
|
|
|
log.Printf("\tNumGC = %v\n", m.NumGC)
|
|
|
|
log.Printf("\tNumGoRoutines = %d\n", runtime.NumGoroutine())
|
2019-02-14 02:28:41 +01:00
|
|
|
}
|