From d4b43305a767c7c3c07bf67635b95d99d7ef5470 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 00:16:36 +0800 Subject: [PATCH 01/10] change: rateCaptureFrequency adjusted to 50ms --- speedtest/data_manager.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/speedtest/data_manager.go b/speedtest/data_manager.go index a56bb66..9fe90c5 100644 --- a/speedtest/data_manager.go +++ b/speedtest/data_manager.go @@ -95,7 +95,7 @@ func NewDataManager() *DataManager { ret := &DataManager{ nThread: runtime.NumCPU(), captureTime: time.Second * 10, - rateCaptureFrequency: time.Millisecond * 100, + rateCaptureFrequency: time.Millisecond * 50, Snapshot: &Snapshot{}, } ret.dFn = &funcGroup{manager: ret} From 41c72eacad93540f0853e9cb7a3a6796a423fa80 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 00:17:44 +0800 Subject: [PATCH 02/10] change: add welford alg with moving window --- example/compare/main.go | 37 ++++++++++++++++ example/ip/main.go | 39 +++++++++++++++++ example/sd/welford.go | 87 ++++++++++++++++++++++++++++++++++++++ example/sd/welford_test.go | 82 +++++++++++++++++++++++++++++++++++ 4 files changed, 245 insertions(+) create mode 100644 example/compare/main.go create mode 100644 example/ip/main.go create mode 100644 example/sd/welford.go create mode 100644 example/sd/welford_test.go diff --git a/example/compare/main.go b/example/compare/main.go new file mode 100644 index 0000000..6599a25 --- /dev/null +++ b/example/compare/main.go @@ -0,0 +1,37 @@ +package main + +import ( + "context" + "fmt" + "github.com/showwin/speedtest-go/speedtest" + "time" +) + +func main() { + server, err := speedtest.CustomServer("http://speedtest.139play.com:8080/speedtest/upload.php") + if err != nil { + fmt.Println(err) + return + } + //timeoutContext, cancel := context.WithTimeout(context.Background(), time.Second*10) + //defer cancel() + //_, err = server.TCPPing(timeoutContext, 5, time.Second, func(latency time.Duration) { + // fmt.Println(latency) + //}) + //if err != nil { + // fmt.Println(err) + // return + //} + + fmt.Println() + //timeoutContext1, cancel1 := context.WithTimeout(context.Background(), time.Second*10) + //defer cancel1() + _, err = server.HTTPPing(context.TODO(), 5, time.Second, func(latency time.Duration) { + fmt.Println(latency * 2) + //fmt.Println(latency) + }) + if err != nil { + fmt.Println(err) + return + } +} diff --git a/example/ip/main.go b/example/ip/main.go new file mode 100644 index 0000000..737b2f7 --- /dev/null +++ b/example/ip/main.go @@ -0,0 +1,39 @@ +package main + +import ( + "fmt" + "github.com/showwin/speedtest-go/speedtest" + "syscall" +) + +func customControl(network string, address string, conn syscall.RawConn) error { + fmt.Printf("%s,%s\n", network, address) + return nil +} + +func main() { + var speedtestClient = speedtest.New() + serverList, _ := speedtestClient.FetchServers() + targets, _ := serverList.FindServer([]int{}) + + var d []float64 + + speedtestClient.CallbackDownloadRate(func(downRate float64) { + d = append(d, downRate) + }) + + for _, s := range targets { + // Please make sure your host can access this test server, + // otherwise you will get an error. + // It is recommended to replace a server at this time + s.DownloadTest() + fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) + s.Context.Reset() // reset counter + } + + fmt.Print("data := []float64{") + for _, val := range d { + fmt.Printf("%v, ", val) + } + fmt.Print("}") +} diff --git a/example/sd/welford.go b/example/sd/welford.go new file mode 100644 index 0000000..b19af89 --- /dev/null +++ b/example/sd/welford.go @@ -0,0 +1,87 @@ +package main + +import ( + "fmt" + "math" +) + +// Welford Fast standard deviation calculation with moving window +// ref Welford, B. P. (1962). Note on a Method for Calculating Corrected Sums of Squares and Products. Technometrics, 4(3), 419–420. https://doi.org/10.1080/00401706.1962.10490022 +type Welford struct { + n int // data size + mean float64 // mean + sum float64 // sum + vector []float64 // data set + eraseIndex int // the value will be erased next time + cap int + currentStdDev float64 + tolerance float64 + factor float64 + prevStdDev float64 + consecutiveStableIterations int + maxConsecutiveStableIterations int +} + +// NewWelford recommended windowSize = moving time window / sampling frequency +func NewWelford(windowSize int) *Welford { + return &Welford{ + eraseIndex: 0, + vector: make([]float64, windowSize), + cap: windowSize, + factor: 1e-3, + tolerance: 1e-6, + maxConsecutiveStableIterations: 3, + } +} + +func (w *Welford) Add(value float64) { + if w.n == w.cap { + delta := w.vector[w.eraseIndex] - w.mean + w.mean -= delta / float64(w.n-1) + w.sum -= delta * (w.vector[w.eraseIndex] - w.mean) + // the calc error is approximated to zero + if w.sum < 0 { + w.sum = 0 + } + w.vector[w.eraseIndex] = value + w.eraseIndex++ + if w.eraseIndex == w.cap { + w.eraseIndex = 0 + } + } else { + w.vector[w.n] = value + w.n++ + } + delta := value - w.mean + w.mean += delta / float64(w.n) + w.sum += delta * (value - w.mean) +} + +func (w *Welford) Mean() float64 { + return w.mean +} + +func (w *Welford) Variance() float64 { + if w.n < 2 { + return 0 + } + return w.sum / float64(w.n-1) +} + +func (w *Welford) StandardDeviation() float64 { + w.currentStdDev = math.Sqrt(w.Variance()) + return w.currentStdDev +} + +func (w *Welford) Convergence() bool { + if w.n > 1 { + w.tolerance = w.currentStdDev * w.factor // dynamic tolerance + conv := math.Abs(w.currentStdDev - w.prevStdDev) + fmt.Printf("conv: %f, tolerance: %f\n", conv, w.tolerance) + if conv < w.tolerance && w.cap/2 < w.n { + w.consecutiveStableIterations++ + } + } + w.prevStdDev = w.currentStdDev + return w.consecutiveStableIterations >= w.maxConsecutiveStableIterations +} diff --git a/example/sd/welford_test.go b/example/sd/welford_test.go new file mode 100644 index 0000000..e22cecc --- /dev/null +++ b/example/sd/welford_test.go @@ -0,0 +1,82 @@ +package main + +import ( + "fmt" + "github.com/showwin/speedtest-go/speedtest" + "math/rand" + "testing" + "time" +) + +func BenchmarkWOM(b *testing.B) { + w := NewWelford(10) + rd := rand.New(rand.NewSource(0)) + var arr []float64 + for i := 0; i < 100; i++ { + arr = append(arr, rd.Float64()) + } + for i := 0; i < b.N; i++ { + w.Add(arr[i%100]) + w.Mean() + w.StandardDeviation() + } +} + +func BenchmarkName(b *testing.B) { + data := []int64{55, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5} + for i := 0; i < b.N; i++ { + speedtest.StandardDeviation(data) + } +} + +func TestWOM(t *testing.T) { + data := []float64{55, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5} + w := NewWelford(20) + for i, x := range data { + w.Add(x) + fmt.Printf("[%d] Mean: %.2f Standard Deviation: %.2f\n", i, w.Mean(), w.StandardDeviation()) + } +} + +func TestWOMRealTime(t *testing.T) { + data := []float64{6.91552, 18.721692307692308, 23.04116556291391, 28.059485148514852, 31.118470119521916, 34.21727152317881, 35.15127065527066, 36.74378054862843, 38.05981374722838, 39.122272, 38.76271506352087, 39.60149084858569, 40.23165538461539, 40.72287589158345, 41.3182940397351, 41.36879800498754, 41.848093896713614, 42.29560044395117, 42.45441684210526, 42.27466666666667, 42.67902476190476, 42.68457142857143, 42.80731190269331, 42.75244, 42.6367104, 42.65657801691007, 42.68948038490007, 42.81048394004282, 42.64527084769124, 42.802427145708585, 42.92410838709678, 43.15598501872659, 43.32482909090909, 43.46838800705467, 43.5166464877213, 43.523846751804555, 43.73983171521036, 43.870409258285115, 43.84913230769231, 43.72752023988006, 43.85322926829268, 43.964127436994765, 44.027410506741056, 43.99742169768498, 44.093007552199026, 44.241321164710996, 44.287108464483204, 44.079746772178254, 44.180878367346935, 44.1128768, 44.17403607843137, 44.0556123076923, 44.20335873255375, 44.19607703703704, 44.2603155216285, 44.202986438258385, 44.2371857042747, 44.17902344827586, 44.31277559322034, 44.30876912840985, 44.378359108781126, 44.3999845410628, 44.37440913415794, 44.4760624609619, 44.471889264841586, 44.454861296184134, 44.37803098927294, 44.37345251396648, 44.47449043478261, 44.43488914285714, 44.50349070422535, 44.45223882254929, 44.433060821917806, 44.46777081081081, 44.4554752, 44.43440842105263, 44.55939356178609, 44.59433376057421, 44.633178734177214, 44.5508462884279, 44.58072493827161, 44.67904608632041, 44.662959537572256, 44.66403237324446, 44.59772449459332, 44.53588837209303, 44.586143382352944, 44.66949738933031, 44.66949618320611, 44.69251965356429, 44.628363956043955, 44.63443946968051, 44.607556129032254, 44.66488851063829, 44.66553062513155, 44.72077, 44.75405813234384, 44.821391020408164, 44.88321874368814, 44.9364192, 44.94112373787369, 44.958842352941176, 44.97548797517455, 44.94797846153846, 44.9455939047619, 44.910117647058826, 44.88617040358744, 44.86545696835092, 44.86558503026968, 44.90363345454546, 44.87501062871554, 44.93104802713802, 44.93590804597701, 44.90068409051044, 44.87467130434782, 44.891225650749874, 44.9156841025641, 44.92164610169491, 44.89092118971601, 44.9027482086319, 44.86017701453105, 44.87692428711898, 44.8651103526735, 44.8483070967742, 44.79754023356263, 44.787605776860815, 44.7891175822446, 44.778797499999996, 44.778773283743995, 44.79586406274028, 44.77517923664122, 44.62143350499849, 44.48119933904161, 44.38141850746269, 44.425831753554505, 44.390321423320096, 44.31049810218978, 44.124642318840586, 44.01107798561151, 44.02091885714285, 44.02881950942861, 44.019682253521125, 44.071147412587415, 44.08087753401833, 44.09411310344827, 44.12980821917809, 44.07751870493811, 44.09083445854713, 44.09876506104924, 44.08840634582056, 44.13032264900663, 44.09793658729115, 44.106114509803916, 44.06832567199065, 44.096165139981935, 44.116, 44.10581480066234, 44.09257851448817, 44.075851106639846, 44.074916, 44.091910073282826, 44.09682567901235, 44.07930895705522, 44.053896892138944, 44.06240019391589, 44.06116150788872, 44.1180531736527, 44.15990285714285, 44.134685126020585, 44.18049829431832, 44.158170252572496, 44.21358883720931, 44.16577849710982, 44.1291108045977, 44.16022671694664, 44.1493200045439, 44.1491597740113, 44.15741842696629, 44.177169162011175, 44.21226797022553, 44.166602585349686, 44.17985497692815, 44.168419672131144, 44.19028366481904, 44.173513455095645, 44.162090322580646, 44.123481608212145, 44.155531914893615, 44.18765714285714, 44.224294736842104, 44.21350366492147, 44.168821324448146, 44.168, 44.17472356215214, 44.187647999999996, 44.18084571428571, 44.18716962744899, 44.19440137359862, 44.1429404079992, 44.15704} + //data := []float64{1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} + size := 5 * time.Second / (time.Millisecond * 50) + w := NewWelford(int(size)) + for i, x := range data { + w.Add(x) + fmt.Printf("[%d] Mean: %.2f Standard Deviation: %.2f\n", i, w.Mean(), w.StandardDeviation()) + if w.Convergence() { + fmt.Println("Convergence") + break + } + time.Sleep(time.Millisecond * 50) + } +} + +func TestDownTestData(t *testing.T) { + var speedtestClient = speedtest.New() + serverList, _ := speedtestClient.FetchServers() + targets, _ := serverList.FindServer([]int{}) + + var d []float64 + + speedtestClient.CallbackDownloadRate(func(downRate float64) { + d = append(d, downRate) + }) + + for _, s := range targets { + // Please make sure your host can access this test server, + // otherwise you will get an error. + // It is recommended to replace a server at this time + s.DownloadTest() + fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) + s.Context.Reset() // reset counter + } + + fmt.Print("data := []float64{") + for _, val := range d { + fmt.Printf("%v, ", val) + } + fmt.Print("}") +} From 956b518c7816cdd4b820e95b67be0f63e625f576 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 02:25:00 +0800 Subject: [PATCH 03/10] change: add c.v scalar --- example/sd/welford_test.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/example/sd/welford_test.go b/example/sd/welford_test.go index e22cecc..f4d44f3 100644 --- a/example/sd/welford_test.go +++ b/example/sd/welford_test.go @@ -39,13 +39,13 @@ func TestWOM(t *testing.T) { } func TestWOMRealTime(t *testing.T) { - data := []float64{6.91552, 18.721692307692308, 23.04116556291391, 28.059485148514852, 31.118470119521916, 34.21727152317881, 35.15127065527066, 36.74378054862843, 38.05981374722838, 39.122272, 38.76271506352087, 39.60149084858569, 40.23165538461539, 40.72287589158345, 41.3182940397351, 41.36879800498754, 41.848093896713614, 42.29560044395117, 42.45441684210526, 42.27466666666667, 42.67902476190476, 42.68457142857143, 42.80731190269331, 42.75244, 42.6367104, 42.65657801691007, 42.68948038490007, 42.81048394004282, 42.64527084769124, 42.802427145708585, 42.92410838709678, 43.15598501872659, 43.32482909090909, 43.46838800705467, 43.5166464877213, 43.523846751804555, 43.73983171521036, 43.870409258285115, 43.84913230769231, 43.72752023988006, 43.85322926829268, 43.964127436994765, 44.027410506741056, 43.99742169768498, 44.093007552199026, 44.241321164710996, 44.287108464483204, 44.079746772178254, 44.180878367346935, 44.1128768, 44.17403607843137, 44.0556123076923, 44.20335873255375, 44.19607703703704, 44.2603155216285, 44.202986438258385, 44.2371857042747, 44.17902344827586, 44.31277559322034, 44.30876912840985, 44.378359108781126, 44.3999845410628, 44.37440913415794, 44.4760624609619, 44.471889264841586, 44.454861296184134, 44.37803098927294, 44.37345251396648, 44.47449043478261, 44.43488914285714, 44.50349070422535, 44.45223882254929, 44.433060821917806, 44.46777081081081, 44.4554752, 44.43440842105263, 44.55939356178609, 44.59433376057421, 44.633178734177214, 44.5508462884279, 44.58072493827161, 44.67904608632041, 44.662959537572256, 44.66403237324446, 44.59772449459332, 44.53588837209303, 44.586143382352944, 44.66949738933031, 44.66949618320611, 44.69251965356429, 44.628363956043955, 44.63443946968051, 44.607556129032254, 44.66488851063829, 44.66553062513155, 44.72077, 44.75405813234384, 44.821391020408164, 44.88321874368814, 44.9364192, 44.94112373787369, 44.958842352941176, 44.97548797517455, 44.94797846153846, 44.9455939047619, 44.910117647058826, 44.88617040358744, 44.86545696835092, 44.86558503026968, 44.90363345454546, 44.87501062871554, 44.93104802713802, 44.93590804597701, 44.90068409051044, 44.87467130434782, 44.891225650749874, 44.9156841025641, 44.92164610169491, 44.89092118971601, 44.9027482086319, 44.86017701453105, 44.87692428711898, 44.8651103526735, 44.8483070967742, 44.79754023356263, 44.787605776860815, 44.7891175822446, 44.778797499999996, 44.778773283743995, 44.79586406274028, 44.77517923664122, 44.62143350499849, 44.48119933904161, 44.38141850746269, 44.425831753554505, 44.390321423320096, 44.31049810218978, 44.124642318840586, 44.01107798561151, 44.02091885714285, 44.02881950942861, 44.019682253521125, 44.071147412587415, 44.08087753401833, 44.09411310344827, 44.12980821917809, 44.07751870493811, 44.09083445854713, 44.09876506104924, 44.08840634582056, 44.13032264900663, 44.09793658729115, 44.106114509803916, 44.06832567199065, 44.096165139981935, 44.116, 44.10581480066234, 44.09257851448817, 44.075851106639846, 44.074916, 44.091910073282826, 44.09682567901235, 44.07930895705522, 44.053896892138944, 44.06240019391589, 44.06116150788872, 44.1180531736527, 44.15990285714285, 44.134685126020585, 44.18049829431832, 44.158170252572496, 44.21358883720931, 44.16577849710982, 44.1291108045977, 44.16022671694664, 44.1493200045439, 44.1491597740113, 44.15741842696629, 44.177169162011175, 44.21226797022553, 44.166602585349686, 44.17985497692815, 44.168419672131144, 44.19028366481904, 44.173513455095645, 44.162090322580646, 44.123481608212145, 44.155531914893615, 44.18765714285714, 44.224294736842104, 44.21350366492147, 44.168821324448146, 44.168, 44.17472356215214, 44.187647999999996, 44.18084571428571, 44.18716962744899, 44.19440137359862, 44.1429404079992, 44.15704} - //data := []float64{1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} + //data := []float64{6.91552, 18.721692307692308, 23.04116556291391, 28.059485148514852, 31.118470119521916, 34.21727152317881, 35.15127065527066, 36.74378054862843, 38.05981374722838, 39.122272, 38.76271506352087, 39.60149084858569, 40.23165538461539, 40.72287589158345, 41.3182940397351, 41.36879800498754, 41.848093896713614, 42.29560044395117, 42.45441684210526, 42.27466666666667, 42.67902476190476, 42.68457142857143, 42.80731190269331, 42.75244, 42.6367104, 42.65657801691007, 42.68948038490007, 42.81048394004282, 42.64527084769124, 42.802427145708585, 42.92410838709678, 43.15598501872659, 43.32482909090909, 43.46838800705467, 43.5166464877213, 43.523846751804555, 43.73983171521036, 43.870409258285115, 43.84913230769231, 43.72752023988006, 43.85322926829268, 43.964127436994765, 44.027410506741056, 43.99742169768498, 44.093007552199026, 44.241321164710996, 44.287108464483204, 44.079746772178254, 44.180878367346935, 44.1128768, 44.17403607843137, 44.0556123076923, 44.20335873255375, 44.19607703703704, 44.2603155216285, 44.202986438258385, 44.2371857042747, 44.17902344827586, 44.31277559322034, 44.30876912840985, 44.378359108781126, 44.3999845410628, 44.37440913415794, 44.4760624609619, 44.471889264841586, 44.454861296184134, 44.37803098927294, 44.37345251396648, 44.47449043478261, 44.43488914285714, 44.50349070422535, 44.45223882254929, 44.433060821917806, 44.46777081081081, 44.4554752, 44.43440842105263, 44.55939356178609, 44.59433376057421, 44.633178734177214, 44.5508462884279, 44.58072493827161, 44.67904608632041, 44.662959537572256, 44.66403237324446, 44.59772449459332, 44.53588837209303, 44.586143382352944, 44.66949738933031, 44.66949618320611, 44.69251965356429, 44.628363956043955, 44.63443946968051, 44.607556129032254, 44.66488851063829, 44.66553062513155, 44.72077, 44.75405813234384, 44.821391020408164, 44.88321874368814, 44.9364192, 44.94112373787369, 44.958842352941176, 44.97548797517455, 44.94797846153846, 44.9455939047619, 44.910117647058826, 44.88617040358744, 44.86545696835092, 44.86558503026968, 44.90363345454546, 44.87501062871554, 44.93104802713802, 44.93590804597701, 44.90068409051044, 44.87467130434782, 44.891225650749874, 44.9156841025641, 44.92164610169491, 44.89092118971601, 44.9027482086319, 44.86017701453105, 44.87692428711898, 44.8651103526735, 44.8483070967742, 44.79754023356263, 44.787605776860815, 44.7891175822446, 44.778797499999996, 44.778773283743995, 44.79586406274028, 44.77517923664122, 44.62143350499849, 44.48119933904161, 44.38141850746269, 44.425831753554505, 44.390321423320096, 44.31049810218978, 44.124642318840586, 44.01107798561151, 44.02091885714285, 44.02881950942861, 44.019682253521125, 44.071147412587415, 44.08087753401833, 44.09411310344827, 44.12980821917809, 44.07751870493811, 44.09083445854713, 44.09876506104924, 44.08840634582056, 44.13032264900663, 44.09793658729115, 44.106114509803916, 44.06832567199065, 44.096165139981935, 44.116, 44.10581480066234, 44.09257851448817, 44.075851106639846, 44.074916, 44.091910073282826, 44.09682567901235, 44.07930895705522, 44.053896892138944, 44.06240019391589, 44.06116150788872, 44.1180531736527, 44.15990285714285, 44.134685126020585, 44.18049829431832, 44.158170252572496, 44.21358883720931, 44.16577849710982, 44.1291108045977, 44.16022671694664, 44.1493200045439, 44.1491597740113, 44.15741842696629, 44.177169162011175, 44.21226797022553, 44.166602585349686, 44.17985497692815, 44.168419672131144, 44.19028366481904, 44.173513455095645, 44.162090322580646, 44.123481608212145, 44.155531914893615, 44.18765714285714, 44.224294736842104, 44.21350366492147, 44.168821324448146, 44.168, 44.17472356215214, 44.187647999999996, 44.18084571428571, 44.18716962744899, 44.19440137359862, 44.1429404079992, 44.15704} + data := []float64{1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} size := 5 * time.Second / (time.Millisecond * 50) w := NewWelford(int(size)) for i, x := range data { w.Add(x) - fmt.Printf("[%d] Mean: %.2f Standard Deviation: %.2f\n", i, w.Mean(), w.StandardDeviation()) + fmt.Printf("[%d] Mean: %.2f, Standard Deviation: %.2f, C.V: %.2f\n", i, w.Mean(), w.StandardDeviation(), w.StandardDeviation()/w.Mean()) if w.Convergence() { fmt.Println("Convergence") break From efb35cd28f89640dd213bf6414baf1a52dee9967 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 02:50:24 +0800 Subject: [PATCH 04/10] change: add readme.md --- example/sd/README.md | 6 ++++++ 1 file changed, 6 insertions(+) create mode 100644 example/sd/README.md diff --git a/example/sd/README.md b/example/sd/README.md new file mode 100644 index 0000000..03663b9 --- /dev/null +++ b/example/sd/README.md @@ -0,0 +1,6 @@ +# + +1. Use welford alg to quickly calculate standard deviation and mean. +2. The welford method integrated moving window feature, This allows us to ignore early data with excessive volatility. +3. Use the coefficient of variation(c.v) to reflect the confidence of the test result datasets. +4. When the c.v is less than 0.05, we terminate this test. \ No newline at end of file From 9d100b2aeddf53f1222b60a9192260c3921c9442 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 04:08:08 +0800 Subject: [PATCH 05/10] change: update readme.md --- example/sd/README.md | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/example/sd/README.md b/example/sd/README.md index 03663b9..7bd48d4 100644 --- a/example/sd/README.md +++ b/example/sd/README.md @@ -1,6 +1,11 @@ -# +# Issue 192 + +SpeedTest-Go 1. Use welford alg to quickly calculate standard deviation and mean. -2. The welford method integrated moving window feature, This allows us to ignore early data with excessive volatility. -3. Use the coefficient of variation(c.v) to reflect the confidence of the test result datasets. -4. When the c.v is less than 0.05, we terminate this test. \ No newline at end of file +2. (a)The welford method integrated moving window feature, This allows us to ignore early data with excessive volatility. +3. (b)The welford method integrated moving Average feature, compare with 2a, this causes early data to be diluted gradually rather than immediately. +4. Use the coefficient of variation(c.v) to reflect the confidence of the test result datasets. +5. When the c.v is less than 0.05, we terminate this test. + +Is there a better way? \ No newline at end of file From 5028301a9fc264330fcec749b557f952cf17c4b9 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 14:31:57 +0800 Subject: [PATCH 06/10] change: update EWMA --- example/compare/main.go | 37 ------- example/ip/main.go | 39 ------- example/sd/welford.go | 87 ---------------- example/sd/welford_test.go | 82 --------------- {example/sd => speedtest/internal}/README.md | 0 speedtest/internal/welford.go | 102 +++++++++++++++++++ speedtest/internal/welford_test.go | 63 ++++++++++++ 7 files changed, 165 insertions(+), 245 deletions(-) delete mode 100644 example/compare/main.go delete mode 100644 example/ip/main.go delete mode 100644 example/sd/welford.go delete mode 100644 example/sd/welford_test.go rename {example/sd => speedtest/internal}/README.md (100%) create mode 100644 speedtest/internal/welford.go create mode 100644 speedtest/internal/welford_test.go diff --git a/example/compare/main.go b/example/compare/main.go deleted file mode 100644 index 6599a25..0000000 --- a/example/compare/main.go +++ /dev/null @@ -1,37 +0,0 @@ -package main - -import ( - "context" - "fmt" - "github.com/showwin/speedtest-go/speedtest" - "time" -) - -func main() { - server, err := speedtest.CustomServer("http://speedtest.139play.com:8080/speedtest/upload.php") - if err != nil { - fmt.Println(err) - return - } - //timeoutContext, cancel := context.WithTimeout(context.Background(), time.Second*10) - //defer cancel() - //_, err = server.TCPPing(timeoutContext, 5, time.Second, func(latency time.Duration) { - // fmt.Println(latency) - //}) - //if err != nil { - // fmt.Println(err) - // return - //} - - fmt.Println() - //timeoutContext1, cancel1 := context.WithTimeout(context.Background(), time.Second*10) - //defer cancel1() - _, err = server.HTTPPing(context.TODO(), 5, time.Second, func(latency time.Duration) { - fmt.Println(latency * 2) - //fmt.Println(latency) - }) - if err != nil { - fmt.Println(err) - return - } -} diff --git a/example/ip/main.go b/example/ip/main.go deleted file mode 100644 index 737b2f7..0000000 --- a/example/ip/main.go +++ /dev/null @@ -1,39 +0,0 @@ -package main - -import ( - "fmt" - "github.com/showwin/speedtest-go/speedtest" - "syscall" -) - -func customControl(network string, address string, conn syscall.RawConn) error { - fmt.Printf("%s,%s\n", network, address) - return nil -} - -func main() { - var speedtestClient = speedtest.New() - serverList, _ := speedtestClient.FetchServers() - targets, _ := serverList.FindServer([]int{}) - - var d []float64 - - speedtestClient.CallbackDownloadRate(func(downRate float64) { - d = append(d, downRate) - }) - - for _, s := range targets { - // Please make sure your host can access this test server, - // otherwise you will get an error. - // It is recommended to replace a server at this time - s.DownloadTest() - fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) - s.Context.Reset() // reset counter - } - - fmt.Print("data := []float64{") - for _, val := range d { - fmt.Printf("%v, ", val) - } - fmt.Print("}") -} diff --git a/example/sd/welford.go b/example/sd/welford.go deleted file mode 100644 index b19af89..0000000 --- a/example/sd/welford.go +++ /dev/null @@ -1,87 +0,0 @@ -package main - -import ( - "fmt" - "math" -) - -// Welford Fast standard deviation calculation with moving window -// ref Welford, B. P. (1962). Note on a Method for Calculating Corrected Sums of Squares and Products. Technometrics, 4(3), 419–420. https://doi.org/10.1080/00401706.1962.10490022 -type Welford struct { - n int // data size - mean float64 // mean - sum float64 // sum - vector []float64 // data set - eraseIndex int // the value will be erased next time - cap int - currentStdDev float64 - tolerance float64 - factor float64 - prevStdDev float64 - consecutiveStableIterations int - maxConsecutiveStableIterations int -} - -// NewWelford recommended windowSize = moving time window / sampling frequency -func NewWelford(windowSize int) *Welford { - return &Welford{ - eraseIndex: 0, - vector: make([]float64, windowSize), - cap: windowSize, - factor: 1e-3, - tolerance: 1e-6, - maxConsecutiveStableIterations: 3, - } -} - -func (w *Welford) Add(value float64) { - if w.n == w.cap { - delta := w.vector[w.eraseIndex] - w.mean - w.mean -= delta / float64(w.n-1) - w.sum -= delta * (w.vector[w.eraseIndex] - w.mean) - // the calc error is approximated to zero - if w.sum < 0 { - w.sum = 0 - } - w.vector[w.eraseIndex] = value - w.eraseIndex++ - if w.eraseIndex == w.cap { - w.eraseIndex = 0 - } - } else { - w.vector[w.n] = value - w.n++ - } - delta := value - w.mean - w.mean += delta / float64(w.n) - w.sum += delta * (value - w.mean) -} - -func (w *Welford) Mean() float64 { - return w.mean -} - -func (w *Welford) Variance() float64 { - if w.n < 2 { - return 0 - } - return w.sum / float64(w.n-1) -} - -func (w *Welford) StandardDeviation() float64 { - w.currentStdDev = math.Sqrt(w.Variance()) - return w.currentStdDev -} - -func (w *Welford) Convergence() bool { - if w.n > 1 { - w.tolerance = w.currentStdDev * w.factor // dynamic tolerance - conv := math.Abs(w.currentStdDev - w.prevStdDev) - fmt.Printf("conv: %f, tolerance: %f\n", conv, w.tolerance) - if conv < w.tolerance && w.cap/2 < w.n { - w.consecutiveStableIterations++ - } - } - w.prevStdDev = w.currentStdDev - return w.consecutiveStableIterations >= w.maxConsecutiveStableIterations -} diff --git a/example/sd/welford_test.go b/example/sd/welford_test.go deleted file mode 100644 index f4d44f3..0000000 --- a/example/sd/welford_test.go +++ /dev/null @@ -1,82 +0,0 @@ -package main - -import ( - "fmt" - "github.com/showwin/speedtest-go/speedtest" - "math/rand" - "testing" - "time" -) - -func BenchmarkWOM(b *testing.B) { - w := NewWelford(10) - rd := rand.New(rand.NewSource(0)) - var arr []float64 - for i := 0; i < 100; i++ { - arr = append(arr, rd.Float64()) - } - for i := 0; i < b.N; i++ { - w.Add(arr[i%100]) - w.Mean() - w.StandardDeviation() - } -} - -func BenchmarkName(b *testing.B) { - data := []int64{55, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5} - for i := 0; i < b.N; i++ { - speedtest.StandardDeviation(data) - } -} - -func TestWOM(t *testing.T) { - data := []float64{55, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5} - w := NewWelford(20) - for i, x := range data { - w.Add(x) - fmt.Printf("[%d] Mean: %.2f Standard Deviation: %.2f\n", i, w.Mean(), w.StandardDeviation()) - } -} - -func TestWOMRealTime(t *testing.T) { - //data := []float64{6.91552, 18.721692307692308, 23.04116556291391, 28.059485148514852, 31.118470119521916, 34.21727152317881, 35.15127065527066, 36.74378054862843, 38.05981374722838, 39.122272, 38.76271506352087, 39.60149084858569, 40.23165538461539, 40.72287589158345, 41.3182940397351, 41.36879800498754, 41.848093896713614, 42.29560044395117, 42.45441684210526, 42.27466666666667, 42.67902476190476, 42.68457142857143, 42.80731190269331, 42.75244, 42.6367104, 42.65657801691007, 42.68948038490007, 42.81048394004282, 42.64527084769124, 42.802427145708585, 42.92410838709678, 43.15598501872659, 43.32482909090909, 43.46838800705467, 43.5166464877213, 43.523846751804555, 43.73983171521036, 43.870409258285115, 43.84913230769231, 43.72752023988006, 43.85322926829268, 43.964127436994765, 44.027410506741056, 43.99742169768498, 44.093007552199026, 44.241321164710996, 44.287108464483204, 44.079746772178254, 44.180878367346935, 44.1128768, 44.17403607843137, 44.0556123076923, 44.20335873255375, 44.19607703703704, 44.2603155216285, 44.202986438258385, 44.2371857042747, 44.17902344827586, 44.31277559322034, 44.30876912840985, 44.378359108781126, 44.3999845410628, 44.37440913415794, 44.4760624609619, 44.471889264841586, 44.454861296184134, 44.37803098927294, 44.37345251396648, 44.47449043478261, 44.43488914285714, 44.50349070422535, 44.45223882254929, 44.433060821917806, 44.46777081081081, 44.4554752, 44.43440842105263, 44.55939356178609, 44.59433376057421, 44.633178734177214, 44.5508462884279, 44.58072493827161, 44.67904608632041, 44.662959537572256, 44.66403237324446, 44.59772449459332, 44.53588837209303, 44.586143382352944, 44.66949738933031, 44.66949618320611, 44.69251965356429, 44.628363956043955, 44.63443946968051, 44.607556129032254, 44.66488851063829, 44.66553062513155, 44.72077, 44.75405813234384, 44.821391020408164, 44.88321874368814, 44.9364192, 44.94112373787369, 44.958842352941176, 44.97548797517455, 44.94797846153846, 44.9455939047619, 44.910117647058826, 44.88617040358744, 44.86545696835092, 44.86558503026968, 44.90363345454546, 44.87501062871554, 44.93104802713802, 44.93590804597701, 44.90068409051044, 44.87467130434782, 44.891225650749874, 44.9156841025641, 44.92164610169491, 44.89092118971601, 44.9027482086319, 44.86017701453105, 44.87692428711898, 44.8651103526735, 44.8483070967742, 44.79754023356263, 44.787605776860815, 44.7891175822446, 44.778797499999996, 44.778773283743995, 44.79586406274028, 44.77517923664122, 44.62143350499849, 44.48119933904161, 44.38141850746269, 44.425831753554505, 44.390321423320096, 44.31049810218978, 44.124642318840586, 44.01107798561151, 44.02091885714285, 44.02881950942861, 44.019682253521125, 44.071147412587415, 44.08087753401833, 44.09411310344827, 44.12980821917809, 44.07751870493811, 44.09083445854713, 44.09876506104924, 44.08840634582056, 44.13032264900663, 44.09793658729115, 44.106114509803916, 44.06832567199065, 44.096165139981935, 44.116, 44.10581480066234, 44.09257851448817, 44.075851106639846, 44.074916, 44.091910073282826, 44.09682567901235, 44.07930895705522, 44.053896892138944, 44.06240019391589, 44.06116150788872, 44.1180531736527, 44.15990285714285, 44.134685126020585, 44.18049829431832, 44.158170252572496, 44.21358883720931, 44.16577849710982, 44.1291108045977, 44.16022671694664, 44.1493200045439, 44.1491597740113, 44.15741842696629, 44.177169162011175, 44.21226797022553, 44.166602585349686, 44.17985497692815, 44.168419672131144, 44.19028366481904, 44.173513455095645, 44.162090322580646, 44.123481608212145, 44.155531914893615, 44.18765714285714, 44.224294736842104, 44.21350366492147, 44.168821324448146, 44.168, 44.17472356215214, 44.187647999999996, 44.18084571428571, 44.18716962744899, 44.19440137359862, 44.1429404079992, 44.15704} - data := []float64{1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} - size := 5 * time.Second / (time.Millisecond * 50) - w := NewWelford(int(size)) - for i, x := range data { - w.Add(x) - fmt.Printf("[%d] Mean: %.2f, Standard Deviation: %.2f, C.V: %.2f\n", i, w.Mean(), w.StandardDeviation(), w.StandardDeviation()/w.Mean()) - if w.Convergence() { - fmt.Println("Convergence") - break - } - time.Sleep(time.Millisecond * 50) - } -} - -func TestDownTestData(t *testing.T) { - var speedtestClient = speedtest.New() - serverList, _ := speedtestClient.FetchServers() - targets, _ := serverList.FindServer([]int{}) - - var d []float64 - - speedtestClient.CallbackDownloadRate(func(downRate float64) { - d = append(d, downRate) - }) - - for _, s := range targets { - // Please make sure your host can access this test server, - // otherwise you will get an error. - // It is recommended to replace a server at this time - s.DownloadTest() - fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) - s.Context.Reset() // reset counter - } - - fmt.Print("data := []float64{") - for _, val := range d { - fmt.Printf("%v, ", val) - } - fmt.Print("}") -} diff --git a/example/sd/README.md b/speedtest/internal/README.md similarity index 100% rename from example/sd/README.md rename to speedtest/internal/README.md diff --git a/speedtest/internal/welford.go b/speedtest/internal/welford.go new file mode 100644 index 0000000..b8b9431 --- /dev/null +++ b/speedtest/internal/welford.go @@ -0,0 +1,102 @@ +package internal + +import ( + "fmt" + "math" +) + +// Welford Fast standard deviation calculation with moving window +// ref Welford, B. P. (1962). Note on a Method for Calculating Corrected Sums of Squares and Products. Technometrics, 4(3), 419–420. https://doi.org/10.1080/00401706.1962.10490022 +type Welford struct { + n int // data size + mean float64 // mean + sum float64 // sum + vector []float64 // data set + eraseIndex int // the value will be erased next time + cap int + currentStdDev float64 + consecutiveStableIterations int + consecutiveStableIterationsThreshold int + cv float64 + ewmaMean float64 +} + +// NewWelford recommended windowSize = moving time window / sampling frequency +func NewWelford(windowSize int) *Welford { + return &Welford{ + vector: make([]float64, windowSize), + cap: windowSize, + consecutiveStableIterationsThreshold: 201, + } +} + +// Update Enter the given value into the measuring system. +// return bool stability evaluation +func (w *Welford) Update(value float64) bool { + if w.n == w.cap { + delta := w.vector[w.eraseIndex] - w.mean + w.mean -= delta / float64(w.n-1) + w.sum -= delta * (w.vector[w.eraseIndex] - w.mean) + // the calc error is approximated to zero + if w.sum < 0 { + w.sum = 0 + } + w.vector[w.eraseIndex] = value + w.eraseIndex++ + if w.eraseIndex == w.cap { + w.eraseIndex = 0 + } + } else { + w.vector[w.n] = value + w.n++ + } + delta := value - w.mean + w.mean += delta / float64(w.n) + w.sum += delta * (value - w.mean) + w.currentStdDev = math.Sqrt(w.Variance()) + // update C.V + if w.mean == 0 { + w.cv = 1 + } else { + w.cv = w.currentStdDev / w.mean + if w.cv > 1 { + w.cv = 1 + } + } + // ewma beta ratio + // TODO: w.cv needs normalization + beta := w.cv*0.618 + 0.381 + w.ewmaMean = value*beta + w.ewmaMean*(1-beta) + // acc consecutiveStableIterations + if w.cap/2 < w.n && w.cv < 0.03 { + w.consecutiveStableIterations++ + } + return w.consecutiveStableIterations >= w.consecutiveStableIterationsThreshold +} + +func (w *Welford) Mean() float64 { + return w.mean +} + +func (w *Welford) CV() float64 { + return w.cv +} + +func (w *Welford) Variance() float64 { + if w.n < 2 { + return 0 + } + return w.sum / float64(w.n-1) +} + +func (w *Welford) StandardDeviation() float64 { + return w.currentStdDev +} + +func (w *Welford) EWMA() float64 { + return w.ewmaMean +} + +func (w *Welford) String() string { + return fmt.Sprintf("Mean: %.2f, Standard Deviation: %.2f, C.V: %.2f, EWMA: %.2f", w.Mean(), w.StandardDeviation(), w.CV(), w.EWMA()) +} diff --git a/speedtest/internal/welford_test.go b/speedtest/internal/welford_test.go new file mode 100644 index 0000000..16c6edd --- /dev/null +++ b/speedtest/internal/welford_test.go @@ -0,0 +1,63 @@ +package internal + +import ( + "fmt" + "github.com/showwin/speedtest-go/speedtest" + "math/rand" + "testing" + "time" +) + +func BenchmarkWOM(b *testing.B) { + w := NewWelford(10) + rd := rand.New(rand.NewSource(0)) + var arr []float64 + for i := 0; i < 100; i++ { + arr = append(arr, rd.Float64()) + } + for i := 0; i < b.N; i++ { + w.Update(arr[i%100]) + } +} + +func TestWOM(t *testing.T) { + //data := []float64{0, 6.91552, 18.721692307692308, 23.04116556291391, 28.059485148514852, 31.118470119521916, 34.21727152317881, 35.15127065527066, 36.74378054862843, 38.05981374722838, 39.122272, 38.76271506352087, 39.60149084858569, 40.23165538461539, 40.72287589158345, 41.3182940397351, 41.36879800498754, 41.848093896713614, 42.29560044395117, 42.45441684210526, 42.27466666666667, 42.67902476190476, 42.68457142857143, 42.80731190269331, 42.75244, 42.6367104, 42.65657801691007, 42.68948038490007, 42.81048394004282, 42.64527084769124, 42.802427145708585, 42.92410838709678, 43.15598501872659, 43.32482909090909, 43.46838800705467, 43.5166464877213, 43.523846751804555, 43.73983171521036, 43.870409258285115, 43.84913230769231, 43.72752023988006, 43.85322926829268, 43.964127436994765, 44.027410506741056, 43.99742169768498, 44.093007552199026, 44.241321164710996, 44.287108464483204, 44.079746772178254, 44.180878367346935, 44.1128768, 44.17403607843137, 44.0556123076923, 44.20335873255375, 44.19607703703704, 44.2603155216285, 44.202986438258385, 44.2371857042747, 44.17902344827586, 44.31277559322034, 44.30876912840985, 44.378359108781126, 44.3999845410628, 44.37440913415794, 44.4760624609619, 44.471889264841586, 44.454861296184134, 44.37803098927294, 44.37345251396648, 44.47449043478261, 44.43488914285714, 44.50349070422535, 44.45223882254929, 44.433060821917806, 44.46777081081081, 44.4554752, 44.43440842105263, 44.55939356178609, 44.59433376057421, 44.633178734177214, 44.5508462884279, 44.58072493827161, 44.67904608632041, 44.662959537572256, 44.66403237324446, 44.59772449459332, 44.53588837209303, 44.586143382352944, 44.66949738933031, 44.66949618320611, 44.69251965356429, 44.628363956043955, 44.63443946968051, 44.607556129032254, 44.66488851063829, 44.66553062513155, 44.72077, 44.75405813234384, 44.821391020408164, 44.88321874368814, 44.9364192, 44.94112373787369, 44.958842352941176, 44.97548797517455, 44.94797846153846, 44.9455939047619, 44.910117647058826, 44.88617040358744, 44.86545696835092, 44.86558503026968, 44.90363345454546, 44.87501062871554, 44.93104802713802, 44.93590804597701, 44.90068409051044, 44.87467130434782, 44.891225650749874, 44.9156841025641, 44.92164610169491, 44.89092118971601, 44.9027482086319, 44.86017701453105, 44.87692428711898, 44.8651103526735, 44.8483070967742, 44.79754023356263, 44.787605776860815, 44.7891175822446, 44.778797499999996, 44.778773283743995, 44.79586406274028, 44.77517923664122, 44.62143350499849, 44.48119933904161, 44.38141850746269, 44.425831753554505, 44.390321423320096, 44.31049810218978, 44.124642318840586, 44.01107798561151, 44.02091885714285, 44.02881950942861, 44.019682253521125, 44.071147412587415, 44.08087753401833, 44.09411310344827, 44.12980821917809, 44.07751870493811, 44.09083445854713, 44.09876506104924, 44.08840634582056, 44.13032264900663, 44.09793658729115, 44.106114509803916, 44.06832567199065, 44.096165139981935, 44.116, 44.10581480066234, 44.09257851448817, 44.075851106639846, 44.074916, 44.091910073282826, 44.09682567901235, 44.07930895705522, 44.053896892138944, 44.06240019391589, 44.06116150788872, 44.1180531736527, 44.15990285714285, 44.134685126020585, 44.18049829431832, 44.158170252572496, 44.21358883720931, 44.16577849710982, 44.1291108045977, 44.16022671694664, 44.1493200045439, 44.1491597740113, 44.15741842696629, 44.177169162011175, 44.21226797022553, 44.166602585349686, 44.17985497692815, 44.168419672131144, 44.19028366481904, 44.173513455095645, 44.162090322580646, 44.123481608212145, 44.155531914893615, 44.18765714285714, 44.224294736842104, 44.21350366492147, 44.168821324448146, 44.168, 44.17472356215214, 44.187647999999996, 44.18084571428571, 44.18716962744899, 44.19440137359862, 44.1429404079992, 44.15704} + data := []float64{0, 1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} + size := 5 * time.Second / (time.Millisecond * 50) + w := NewWelford(int(size)) + for i, x := range data { + if w.Update(x) { + fmt.Println("Convergence") + break + } + fmt.Printf("[%d] %s\n", i, w) + time.Sleep(time.Millisecond * 50) + } +} + +func TestGenTestData(t *testing.T) { + var speedtestClient = speedtest.New() + serverList, _ := speedtestClient.FetchServers() + targets, _ := serverList.FindServer([]int{}) + + var d []float64 + + speedtestClient.CallbackDownloadRate(func(downRate float64) { + d = append(d, downRate) + }) + + for _, s := range targets { + // Please make sure your host can access this test server, + // otherwise you will get an error. + // It is recommended to replace a server at this time + s.DownloadTest() + fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) + s.Context.Reset() // reset counter + } + + fmt.Print("data := []float64{") + for _, val := range d { + fmt.Printf("%v, ", val) + } + fmt.Print("}") +} From f6723df653787392a183c23590e32db1b4ed6445 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Thu, 2 May 2024 14:34:12 +0800 Subject: [PATCH 07/10] change: remove test --- speedtest/internal/welford_test.go | 28 ---------------------------- 1 file changed, 28 deletions(-) diff --git a/speedtest/internal/welford_test.go b/speedtest/internal/welford_test.go index 16c6edd..396c64b 100644 --- a/speedtest/internal/welford_test.go +++ b/speedtest/internal/welford_test.go @@ -2,7 +2,6 @@ package internal import ( "fmt" - "github.com/showwin/speedtest-go/speedtest" "math/rand" "testing" "time" @@ -34,30 +33,3 @@ func TestWOM(t *testing.T) { time.Sleep(time.Millisecond * 50) } } - -func TestGenTestData(t *testing.T) { - var speedtestClient = speedtest.New() - serverList, _ := speedtestClient.FetchServers() - targets, _ := serverList.FindServer([]int{}) - - var d []float64 - - speedtestClient.CallbackDownloadRate(func(downRate float64) { - d = append(d, downRate) - }) - - for _, s := range targets { - // Please make sure your host can access this test server, - // otherwise you will get an error. - // It is recommended to replace a server at this time - s.DownloadTest() - fmt.Printf("Latency: %s, Download: %f, Upload: %f\n", s.Latency, s.DLSpeed, s.ULSpeed) - s.Context.Reset() // reset counter - } - - fmt.Print("data := []float64{") - for _, val := range d { - fmt.Printf("%v, ", val) - } - fmt.Print("}") -} From 02ae89c08790013518a2cc454b0c5ab2c5eaeeba Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Fri, 3 May 2024 02:54:29 +0800 Subject: [PATCH 08/10] change: done --- speedtest.go | 18 ++- speedtest/data_manager.go | 214 ++++++++++++++++------------- speedtest/data_manager_test.go | 4 +- speedtest/internal/welford.go | 6 +- speedtest/internal/welford_test.go | 7 +- speedtest/request.go | 26 ++-- speedtest/request_test.go | 5 +- 7 files changed, 159 insertions(+), 121 deletions(-) diff --git a/speedtest.go b/speedtest.go index cf21fc4..ce7f52c 100644 --- a/speedtest.go +++ b/speedtest.go @@ -142,11 +142,11 @@ func main() { latencies = append(latencies, latency...) } }() - ticker := speedtestClient.CallbackDownloadRate(func(downRate float64) { + speedtestClient.SetCallbackDownload(func(downRate float64) { if lc == 0 { - task.Printf("Download: %.2fMbps (latency: --)", downRate) + task.Printf("Download: %.2fMbps (latency: --)", downRate*8/1000000) } else { - task.Printf("Download: %.2fMbps (latency: %dms)", downRate, lc/1000000) + task.Printf("Download: %.2fMbps (latency: %dms)", downRate*8/1000000, lc/1000000) } }) if *multi { @@ -154,9 +154,8 @@ func main() { } else { task.CheckError(server.DownloadTest()) } - ticker.Stop() mean, _, std, minL, maxL := speedtest.StandardDeviation(latencies) - task.Printf("Download: %.2fMbps (used: %.2fMB) (latency: %dms jitter: %dms min: %dms max: %dms)", server.DLSpeed, float64(server.Context.Manager.GetTotalDownload())/1000/1000, mean/1000000, std/1000000, minL/1000000, maxL/1000000) + task.Printf("Download: %.2fMbps (used: %.2fMB) (latency: %dms jitter: %dms min: %dms max: %dms)", server.DLSpeed*8/1000000, float64(server.Context.Manager.GetTotalDownload())/1000/1000, mean/1000000, std/1000000, minL/1000000, maxL/1000000) task.Complete() }) @@ -177,11 +176,11 @@ func main() { latencies = append(latencies, latency...) } }() - ticker := speedtestClient.CallbackUploadRate(func(upRate float64) { + speedtestClient.SetCallbackUpload(func(upRate float64) { if lc == 0 { - task.Printf("Upload: %.2fMbps (latency: --)", upRate) + task.Printf("Upload: %.2fMbps (latency: --)", upRate*8/1000000) } else { - task.Printf("Upload: %.2fMbps (latency: %dms)", upRate, lc/1000000) + task.Printf("Upload: %.2fMbps (latency: %dms)", upRate*8/1000000, lc/1000000) } }) if *multi { @@ -189,10 +188,9 @@ func main() { } else { task.CheckError(server.UploadTest()) } - ticker.Stop() quit = true mean, _, std, minL, maxL := speedtest.StandardDeviation(latencies) - task.Printf("Upload: %.2fMbps (used: %.2fMB) (latency: %dms jitter: %dms min: %dms max: %dms)", server.ULSpeed, float64(server.Context.Manager.GetTotalUpload())/1000/1000, mean/1000000, std/1000000, minL/1000000, maxL/1000000) + task.Printf("Upload: %.2fMbps (used: %.2fMB) (latency: %dms jitter: %dms min: %dms max: %dms)", server.ULSpeed*8/1000000, float64(server.Context.Manager.GetTotalUpload())/1000/1000, mean/1000000, std/1000000, minL/1000000, maxL/1000000) task.Complete() }) taskManager.Reset() diff --git a/speedtest/data_manager.go b/speedtest/data_manager.go index 9fe90c5..1969eb9 100644 --- a/speedtest/data_manager.go +++ b/speedtest/data_manager.go @@ -4,6 +4,7 @@ import ( "bytes" "context" "errors" + "github.com/showwin/speedtest-go/speedtest/internal" "io" "math" "runtime" @@ -26,11 +27,14 @@ type Manager interface { GetAvgDownloadRate() float64 GetAvgUploadRate() float64 - CallbackDownloadRate(callback func(downRate float64)) *time.Ticker - CallbackUploadRate(callback func(upRate float64)) *time.Ticker + GetEWMADownloadRate() float64 + GetEWMAUploadRate() float64 - RegisterDownloadHandler(fn func()) *funcGroup - RegisterUploadHandler(fn func()) *funcGroup + SetCallbackDownload(callback func(downRate float64)) + SetCallbackUpload(callback func(upRate float64)) + + RegisterDownloadHandler(fn func()) *TestDirection + RegisterUploadHandler(fn func()) *TestDirection // Wait for the upload or download task to end to avoid errors caused by core occupation Wait() @@ -55,13 +59,18 @@ const readChunkSize = 1024 * 32 // 32 KBytes type DataType int32 -const typeEmptyChunk = 0 -const typeDownload = 1 -const typeUpload = 2 +const ( + typeEmptyChunk = iota + typeDownload + typeUpload +) + +var ( + ErrorUninitializedManager = errors.New("uninitialized manager") +) type funcGroup struct { - fns []func() - manager *DataManager + fns []func() } func (f *funcGroup) Add(fn func()) { @@ -69,12 +78,6 @@ func (f *funcGroup) Add(fn func()) { } type DataManager struct { - totalDownload int64 - totalUpload int64 - - DownloadRateSequence []int64 - UploadRateSequence []int64 - SnapshotStore *Snapshots Snapshot *Snapshot sync.Mutex @@ -87,8 +90,27 @@ type DataManager struct { running bool - dFn *funcGroup - uFn *funcGroup + download *TestDirection + upload *TestDirection +} + +type TestDirection struct { + TestType int // test type + manager *DataManager // manager + totalDataVolume int64 // total send/receive data volume + RateSequence []int64 // rate history sequence + welford *internal.Welford // std/EWMA/mean + captureCallback func(realTimeRate float64) // user callback + closeFunc func() // close func + *funcGroup // actually exec function +} + +func (dm *DataManager) NewDataDirection(testType int) *TestDirection { + return &TestDirection{ + TestType: testType, + manager: dm, + funcGroup: &funcGroup{}, + } } func NewDataManager() *DataManager { @@ -98,32 +120,22 @@ func NewDataManager() *DataManager { rateCaptureFrequency: time.Millisecond * 50, Snapshot: &Snapshot{}, } - ret.dFn = &funcGroup{manager: ret} - ret.uFn = &funcGroup{manager: ret} - ret.SnapshotStore = newRecentSnapshots(maxSnapshotSize) + ret.download = ret.NewDataDirection(typeDownload) + ret.upload = ret.NewDataDirection(typeUpload) + ret.SnapshotStore = newHistorySnapshots(maxSnapshotSize) return ret } -func (dm *DataManager) CallbackDownloadRate(callback func(downRate float64)) *time.Ticker { - ticker := time.NewTicker(dm.rateCaptureFrequency) - go func() { - sTime := time.Now() - for range ticker.C { - callback((float64(dm.GetTotalDownload()) * 8 / 1000000) / float64(time.Since(sTime).Milliseconds()) * 1000) - } - }() - return ticker +func (dm *DataManager) SetCallbackDownload(callback func(downRate float64)) { + if dm.download != nil { + dm.download.captureCallback = callback + } } -func (dm *DataManager) CallbackUploadRate(callback func(upRate float64)) *time.Ticker { - ticker := time.NewTicker(dm.rateCaptureFrequency) - go func() { - sTime := time.Now() - for range ticker.C { - callback((float64(dm.GetTotalUpload()) * 8 / 1000000) / float64(time.Since(sTime).Milliseconds()) * 1000) - } - }() - return ticker +func (dm *DataManager) SetCallbackUpload(callback func(upRate float64)) { + if dm.upload != nil { + dm.upload.captureCallback = callback + } } func (dm *DataManager) Wait() { @@ -143,65 +155,69 @@ func (dm *DataManager) Wait() { } } -func (dm *DataManager) RegisterUploadHandler(fn func()) *funcGroup { - if len(dm.uFn.fns) < dm.nThread { - dm.uFn.Add(fn) +func (dm *DataManager) RegisterUploadHandler(fn func()) *TestDirection { + if len(dm.upload.fns) < dm.nThread { + dm.upload.Add(fn) } - return dm.uFn + return dm.upload } -func (dm *DataManager) RegisterDownloadHandler(fn func()) *funcGroup { - if len(dm.dFn.fns) < dm.nThread { - dm.dFn.Add(fn) +func (dm *DataManager) RegisterDownloadHandler(fn func()) *TestDirection { + if len(dm.download.fns) < dm.nThread { + dm.download.Add(fn) } - return dm.dFn + return dm.download } -func (f *funcGroup) Start(cancel context.CancelFunc, mainRequestHandlerIndex int) { - if len(f.fns) == 0 { +func (td *TestDirection) Start(cancel context.CancelFunc, mainRequestHandlerIndex int) { + if len(td.fns) == 0 { panic("empty task stack") } - if mainRequestHandlerIndex > len(f.fns)-1 { + if mainRequestHandlerIndex > len(td.fns)-1 { mainRequestHandlerIndex = 0 } mainLoadFactor := 0.1 // When the number of processor cores is equivalent to the processing program, // the processing efficiency reaches the highest level (VT is not considered). - mainN := int(mainLoadFactor * float64(len(f.fns))) + mainN := int(mainLoadFactor * float64(len(td.fns))) if mainN == 0 { mainN = 1 } - if len(f.fns) == 1 { - mainN = f.manager.nThread + if len(td.fns) == 1 { + mainN = td.manager.nThread } - auxN := f.manager.nThread - mainN - dbg.Printf("Available fns: %d\n", len(f.fns)) + auxN := td.manager.nThread - mainN + dbg.Printf("Available fns: %d\n", len(td.fns)) dbg.Printf("mainN: %d\n", mainN) dbg.Printf("auxN: %d\n", auxN) wg := sync.WaitGroup{} - f.manager.running = true - stopCapture := f.manager.rateCapture() - time.AfterFunc(f.manager.captureTime, func() { + td.manager.running = true + stopCapture := td.rateCapture() + + // refresh once function + td.closeFunc = sync.OnceFunc(func() { stopCapture <- true close(stopCapture) - f.manager.running = false + td.manager.running = false cancel() dbg.Println("FuncGroup: Stop") }) + + time.AfterFunc(td.manager.captureTime, td.closeFunc) for i := 0; i < mainN; i++ { wg.Add(1) go func() { defer wg.Done() for { - if !f.manager.running { + if !td.manager.running { return } - f.fns[mainRequestHandlerIndex]() + td.fns[mainRequestHandlerIndex]() } }() } for j := 0; j < auxN; { - for i := range f.fns { + for i := range td.fns { if j == auxN { break } @@ -213,10 +229,10 @@ func (f *funcGroup) Start(cancel context.CancelFunc, mainRequestHandlerIndex int go func() { defer wg.Done() for { - if !f.manager.running { + if !td.manager.running { return } - f.fns[t]() + td.fns[t]() } }() j++ @@ -225,27 +241,29 @@ func (f *funcGroup) Start(cancel context.CancelFunc, mainRequestHandlerIndex int wg.Wait() } -func (dm *DataManager) rateCapture() chan bool { - ticker := time.NewTicker(dm.rateCaptureFrequency) - oldTotalDownload := dm.totalDownload - oldTotalUpload := dm.totalUpload +func (td *TestDirection) rateCapture() chan bool { + ticker := time.NewTicker(td.manager.rateCaptureFrequency) + var prevTotalDataVolume int64 = 0 stopCapture := make(chan bool) + td.welford = internal.NewWelford(int(5 * time.Second / td.manager.rateCaptureFrequency)) + sTime := time.Now() go func(t *time.Ticker) { defer t.Stop() for { select { case <-t.C: - newTotalDownload := dm.totalDownload - newTotalUpload := dm.totalUpload - deltaDownload := newTotalDownload - oldTotalDownload - deltaUpload := newTotalUpload - oldTotalUpload - oldTotalDownload = newTotalDownload - oldTotalUpload = newTotalUpload - if deltaDownload != 0 { - dm.DownloadRateSequence = append(dm.DownloadRateSequence, deltaDownload) - } - if deltaUpload != 0 { - dm.UploadRateSequence = append(dm.UploadRateSequence, deltaUpload) + newTotalDataVolume := td.totalDataVolume + deltaDataVolume := newTotalDataVolume - prevTotalDataVolume + prevTotalDataVolume = newTotalDataVolume + if deltaDataVolume != 0 { + td.RateSequence = append(td.RateSequence, deltaDataVolume) + if td.captureCallback != nil { + globalAvg := (float64(td.totalDataVolume)) / float64(time.Since(sTime).Milliseconds()) * 1000 + if td.welford.Update(globalAvg) { + go td.closeFunc() + } + td.captureCallback(td.welford.EWMA()) + } } case stop := <-stopCapture: if stop { @@ -267,19 +285,19 @@ func (dm *DataManager) NewChunk() Chunk { } func (dm *DataManager) AddTotalDownload(value int64) { - atomic.AddInt64(&dm.totalDownload, value) + atomic.AddInt64(&dm.download.totalDataVolume, value) } func (dm *DataManager) AddTotalUpload(value int64) { - atomic.AddInt64(&dm.totalUpload, value) + atomic.AddInt64(&dm.upload.totalDataVolume, value) } func (dm *DataManager) GetTotalDownload() int64 { - return dm.totalDownload + return dm.download.totalDataVolume } func (dm *DataManager) GetTotalUpload() int64 { - return dm.totalUpload + return dm.upload.totalDataVolume } func (dm *DataManager) SetRateCaptureFrequency(duration time.Duration) Manager { @@ -306,24 +324,34 @@ func (dm *DataManager) Snapshots() *Snapshots { } func (dm *DataManager) Reset() { - dm.totalDownload = 0 - dm.totalUpload = 0 dm.SnapshotStore.push(dm.Snapshot) dm.Snapshot = &Snapshot{} - dm.DownloadRateSequence = []int64{} - dm.UploadRateSequence = []int64{} - dm.dFn.fns = []func(){} - dm.uFn.fns = []func(){} + dm.download = dm.NewDataDirection(typeDownload) + dm.upload = dm.NewDataDirection(typeUpload) } func (dm *DataManager) GetAvgDownloadRate() float64 { unit := float64(dm.captureTime / time.Millisecond) - return float64(dm.totalDownload*8/1000) / unit + return float64(dm.download.totalDataVolume*8/1000) / unit +} + +func (dm *DataManager) GetEWMADownloadRate() float64 { + if dm.download.welford != nil { + return dm.download.welford.EWMA() + } + return 0 } func (dm *DataManager) GetAvgUploadRate() float64 { unit := float64(dm.captureTime / time.Millisecond) - return float64(dm.totalUpload*8/1000) / unit + return float64(dm.upload.totalDataVolume*8/1000) / unit +} + +func (dm *DataManager) GetEWMAUploadRate() float64 { + if dm.upload.welford != nil { + return dm.upload.welford.EWMA() + } + return 0 } type DataChunk struct { @@ -379,7 +407,7 @@ func (dc *DataChunk) DownloadHandler(r io.Reader) error { rs := int64(readSize) dc.remainOrDiscardSize += rs - atomic.AddInt64(&dc.manager.totalDownload, rs) + atomic.AddInt64(&dc.manager.download.totalDataVolume, rs) if dc.err != nil { if dc.err == io.EOF { return nil @@ -430,7 +458,7 @@ func (dc *DataChunk) Read(b []byte) (n int, err error) { } n64 := int64(n) dc.remainOrDiscardSize -= n64 - atomic.AddInt64(&dc.manager.totalUpload, n64) + atomic.AddInt64(&dc.manager.upload.totalDataVolume, n64) return } @@ -513,7 +541,7 @@ type Snapshots struct { maxSize int } -func newRecentSnapshots(size int) *Snapshots { +func newHistorySnapshots(size int) *Snapshots { return &Snapshots{ sp: make([]*Snapshot, 0, size), maxSize: size, diff --git a/speedtest/data_manager_test.go b/speedtest/data_manager_test.go index 14c956e..6f59db6 100644 --- a/speedtest/data_manager_test.go +++ b/speedtest/data_manager_test.go @@ -35,14 +35,14 @@ func TestDataManager_AddTotalDownload(t *testing.T) { }() } wg.Wait() - if dmp.totalDownload != 43521000000 { + if dmp.download.totalDataVolume != 43521000000 { t.Fatal() } } func TestDataManager_GetAvgDownloadRate(t *testing.T) { dm := NewDataManager() - dm.totalDownload = 3000000 + dm.download.totalDataVolume = 3000000 dm.captureTime = time.Second * 10 result := dm.GetAvgDownloadRate() diff --git a/speedtest/internal/welford.go b/speedtest/internal/welford.go index b8b9431..ffeaf49 100644 --- a/speedtest/internal/welford.go +++ b/speedtest/internal/welford.go @@ -26,7 +26,7 @@ func NewWelford(windowSize int) *Welford { return &Welford{ vector: make([]float64, windowSize), cap: windowSize, - consecutiveStableIterationsThreshold: 201, + consecutiveStableIterationsThreshold: 10, } } @@ -65,8 +65,8 @@ func (w *Welford) Update(value float64) bool { } // ewma beta ratio // TODO: w.cv needs normalization - beta := w.cv*0.618 + 0.381 - w.ewmaMean = value*beta + w.ewmaMean*(1-beta) + beta := w.cv*0.381 + 0.618 + w.ewmaMean = w.mean*beta + w.ewmaMean*(1-beta) // acc consecutiveStableIterations if w.cap/2 < w.n && w.cv < 0.03 { w.consecutiveStableIterations++ diff --git a/speedtest/internal/welford_test.go b/speedtest/internal/welford_test.go index 396c64b..415fd30 100644 --- a/speedtest/internal/welford_test.go +++ b/speedtest/internal/welford_test.go @@ -24,12 +24,15 @@ func TestWOM(t *testing.T) { data := []float64{0, 1550.14016, 1102.92032, 759.1852799999999, 601.88672, 502.06617600000004, 435.71381333333335, 385.54944, 348.24736, 316.40547555555554, 289.261344, 267.38202181818184, 252.83786666666668, 245.42818461538462, 245.35842285714287, 235.63543466666667, 227.58704000000003, 220.20446117647057, 213.48643555555554, 207.48786526315791, 201.919424, 197.19868952380952, 192.2912581818182, 188.60752695652172, 185.09968, 181.7553664, 178.6282953846154, 175.99228444444446, 173.34433142857142, 170.91121655172415, 168.496128, 166.37638193548386, 163.99491999999998, 161.9206012121212, 160.24624, 158.76464457142856, 157.2420711111111, 155.45751351351353, 154.00374736842107, 152.63651282051282, 151.112704, 149.88837463414634, 149.02825142857142, 147.3117618604651, 144.63512, 145.58946844444446, 144.72223304347824, 143.76531744680852, 143.01374, 142.06400653061226, 141.0796992, 140.27383843137252, 139.51694153846157, 138.54825660377358, 138.22770370370372, 137.21800727272725, 136.98238857142854, 135.9439045614035, 135.45676137931034, 135.40977898305084, 134.70067622540847, 134.12748048540504, 133.63318709677418, 133.16014222222222, 132.703765, 132.20392861538463, 131.7192387878788, 131.27342328358208, 130.71865882352944, 130.23932753623188, 129.99216914285716, 129.28652169014086, 129.16500888888888, 128.67712438356165, 128.25776864864866, 127.85582506666665, 127.45831157894739, 127.03613922077922, 126.90191179487178, 126.59213772151901, 126.425936, 126.0965688888889, 125.77238634146343, 125.49601735357918, 125.1707276190476, 124.89910964705884, 124.60520186046512, 124.29531218390805, 124.02556363636364, 123.47192808988764, 123.56910222222223, 123.33042637362638, 123.09693913043478, 122.809328172043, 122.58635234042553, 122.41840168421052, 122.15547666666666, 121.92927999999999, 121.73630367346938, 121.48462222222223, 121.29761599999999, 121.06910415841584, 120.87151686274511, 120.68683805825242, 120.51469538461538, 120.2012220952381, 120.15477433962265, 119.2894474766355, 119.77253037037036, 119.57481394495413, 119.31698327272728, 119.10490234234233, 118.99255142857143, 118.76022088495574, 118.8147452631579, 118.6306587826087, 118.51604413793103, 118.38956854700855, 118.1484366101695, 117.98398924369748, 117.95854933333334, 117.80418115702479, 117.68285901639344, 117.32433170731707, 117.36720000000001, 117.21502720000001, 117.10605714285714, 117.01992566929134, 116.88315749999998, 116.77652093023256, 116.65271384615386, 116.51296488549619, 116.39656969696969, 116.29855518796992, 116.18375402985075, 116.00662992592594, 115.96002117647059, 115.85702540145985, 115.77244985507247, 115.67459145200748, 115.48589942857143, 115.44551716312057, 115.35635154929577, 115.21038993006992, 115.11760666666666, 115.01394537931034, 114.97817863013698, 114.80941278911564, 114.72184648648648, 114.64405261744967, 114.55408426666666, 114.41580291390729, 114.3308, 114.27059869281045, 114.18505260423433, 114.14871122580645, 113.99949128205128, 113.94442191082803, 113.83386734177216, 113.59835371069182, 113.515388, 113.6068849689441, 113.52565728395062, 113.47399852760736, 113.35184, 113.3104446060606, 113.19637204819277, 113.12842730538924, 113.09605333333333, 113.08127621301774, 112.81183811764706, 112.9294203508772, 112.81810418604651, 112.69997317919075, 112.63358712643677, 112.5452672, 112.50175090909092, 112.52005423728812, 112.43650516853933, 112.34864983240223, 112.20527644444445, 112.20484950276244, 112.19415912087912, 112.13252546448088, 112.04997913043476, 112.02885535135137, 111.93645075268819, 111.89874994652408, 111.80677787234042, 111.73926264550266, 111.75360336842105, 111.62827225130889, 111.64060833333332, 111.58019481865286, 111.51077113402062, 111.32106666666667, 111.43125714285713, 111.37313757741902, 111.2588913131313, 111.27499738693469, 111.22497689768976} size := 5 * time.Second / (time.Millisecond * 50) w := NewWelford(int(size)) + ok := false for i, x := range data { if w.Update(x) { - fmt.Println("Convergence") + ok = true break } fmt.Printf("[%d] %s\n", i, w) - time.Sleep(time.Millisecond * 50) + } + if !ok { + t.Fatal("TestWOM failed") } } diff --git a/speedtest/request.go b/speedtest/request.go index 2257d39..f72bc0d 100644 --- a/speedtest/request.go +++ b/speedtest/request.go @@ -39,20 +39,24 @@ func (s *Server) MultiDownloadTestContext(ctx context.Context, servers Servers) return errors.New("not found available servers") } mainIDIndex := 0 - var fp *funcGroup + var td *TestDirection _context, cancel := context.WithCancel(ctx) + defer cancel() for i, server := range *ss { if server.ID == s.ID { mainIDIndex = i } sp := server dbg.Printf("Register Download Handler: %s\n", sp.URL) - fp = server.Context.RegisterDownloadHandler(func() { + td = server.Context.RegisterDownloadHandler(func() { _ = downloadRequest(_context, sp, 3) }) } - fp.Start(cancel, mainIDIndex) // block here - s.DLSpeed = fp.manager.GetAvgDownloadRate() + if td == nil { + return ErrorUninitializedManager + } + td.Start(cancel, mainIDIndex) // block here + s.DLSpeed = td.manager.GetEWMADownloadRate() return nil } @@ -66,20 +70,24 @@ func (s *Server) MultiUploadTestContext(ctx context.Context, servers Servers) er return errors.New("not found available servers") } mainIDIndex := 0 - var fp *funcGroup + var td *TestDirection _context, cancel := context.WithCancel(ctx) + defer cancel() for i, server := range *ss { if server.ID == s.ID { mainIDIndex = i } sp := server dbg.Printf("Register Upload Handler: %s\n", sp.URL) - fp = server.Context.RegisterUploadHandler(func() { + td = server.Context.RegisterUploadHandler(func() { _ = uploadRequest(_context, sp, 3) }) } - fp.Start(cancel, mainIDIndex) // block here - s.ULSpeed = fp.manager.GetAvgUploadRate() + if td == nil { + return ErrorUninitializedManager + } + td.Start(cancel, mainIDIndex) // block here + s.ULSpeed = td.manager.GetEWMAUploadRate() return nil } @@ -104,7 +112,7 @@ func (s *Server) downloadTestContext(ctx context.Context, downloadRequest downlo _ = downloadRequest(_context, s, 3) }).Start(cancel, 0) duration := time.Since(start) - s.DLSpeed = s.Context.GetAvgDownloadRate() + s.DLSpeed = s.Context.GetEWMADownloadRate() s.TestDuration.Download = &duration s.testDurationTotalCount() return nil diff --git a/speedtest/request_test.go b/speedtest/request_test.go index 08740a1..da18d8f 100644 --- a/speedtest/request_test.go +++ b/speedtest/request_test.go @@ -29,8 +29,9 @@ func TestDownloadTestContext(t *testing.T) { if err != nil { t.Errorf(err.Error()) } - if server.DLSpeed < idealSpeed*(1-delta) || idealSpeed*(1+delta) < server.DLSpeed { - t.Errorf("got unexpected server.DLSpeed '%v', expected between %v and %v", server.DLSpeed, idealSpeed*(1-delta), idealSpeed*(1+delta)) + value := server.Context.Manager.GetAvgDownloadRate() + if value < idealSpeed*(1-delta) || idealSpeed*(1+delta) < value { + t.Errorf("got unexpected server.DLSpeed '%v', expected between %v and %v", value, idealSpeed*(1-delta), idealSpeed*(1+delta)) } if server.TestDuration.Download == nil || *server.TestDuration.Download != *server.TestDuration.Total { t.Errorf("can't count test duration, server.TestDuration.Download=%v, server.TestDuration.Total=%v", server.TestDuration.Download, server.TestDuration.Total) From 650af1ccc6c9ebb60b1289ade8a1a2b71c96e437 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Fri, 3 May 2024 02:58:05 +0800 Subject: [PATCH 09/10] change: consecutiveStableIterationsThreshold --- speedtest/internal/welford.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/speedtest/internal/welford.go b/speedtest/internal/welford.go index ffeaf49..d57e7e2 100644 --- a/speedtest/internal/welford.go +++ b/speedtest/internal/welford.go @@ -26,7 +26,7 @@ func NewWelford(windowSize int) *Welford { return &Welford{ vector: make([]float64, windowSize), cap: windowSize, - consecutiveStableIterationsThreshold: 10, + consecutiveStableIterationsThreshold: 42, } } From 6ebcf1ea729e26740ad89e5af1f882bd13fcec40 Mon Sep 17 00:00:00 2001 From: r3inbowari Date: Fri, 3 May 2024 03:07:58 +0800 Subject: [PATCH 10/10] change: version tag and back to sync.once in close fn --- speedtest/data_manager.go | 17 ++++++++++------- speedtest/speedtest.go | 2 +- 2 files changed, 11 insertions(+), 8 deletions(-) diff --git a/speedtest/data_manager.go b/speedtest/data_manager.go index 1969eb9..409c486 100644 --- a/speedtest/data_manager.go +++ b/speedtest/data_manager.go @@ -195,13 +195,16 @@ func (td *TestDirection) Start(cancel context.CancelFunc, mainRequestHandlerInde stopCapture := td.rateCapture() // refresh once function - td.closeFunc = sync.OnceFunc(func() { - stopCapture <- true - close(stopCapture) - td.manager.running = false - cancel() - dbg.Println("FuncGroup: Stop") - }) + once := sync.Once{} + td.closeFunc = func() { + once.Do(func() { + stopCapture <- true + close(stopCapture) + td.manager.running = false + cancel() + dbg.Println("FuncGroup: Stop") + }) + } time.AfterFunc(td.manager.captureTime, td.closeFunc) for i := 0; i < mainN; i++ { diff --git a/speedtest/speedtest.go b/speedtest/speedtest.go index 2eac251..371c9eb 100644 --- a/speedtest/speedtest.go +++ b/speedtest/speedtest.go @@ -13,7 +13,7 @@ import ( ) var ( - version = "1.6.12" + version = "1.7.0-dev/stability_evaluation" DefaultUserAgent = fmt.Sprintf("showwin/speedtest-go %s", version) )