This repository has been archived by the owner on May 24, 2020. It is now read-only.
forked from ant0ine/go-json-rest
-
Notifications
You must be signed in to change notification settings - Fork 0
/
status.go
111 lines (94 loc) · 2.61 KB
/
status.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
package rest
import (
"fmt"
"net/http"
"os"
"sync"
"time"
)
func (self *ResourceHandler) statusWrapper(h http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
// call the handler
h(w, r)
if self.statusService != nil {
self.statusService.update(
self.env.getVar(r, "statusCode").(int),
self.env.getVar(r, "elapsedTime").(*time.Duration),
)
}
}
}
type statusService struct {
lock sync.Mutex
start time.Time
pid int
responseCounts map[string]int
totalResponseTime time.Time
}
func newStatusService() *statusService {
return &statusService{
start: time.Now(),
pid: os.Getpid(),
responseCounts: map[string]int{},
totalResponseTime: time.Time{},
}
}
func (self *statusService) getRoute() Route {
return Route{
HttpMethod: "GET",
PathExp: "/.status",
Func: func(writer *ResponseWriter, request *Request) {
self.getStatus(writer, request)
},
}
}
func (self *statusService) update(statusCode int, responseTime *time.Duration) {
self.lock.Lock()
self.responseCounts[fmt.Sprintf("%d", statusCode)]++
self.totalResponseTime = self.totalResponseTime.Add(*responseTime)
self.lock.Unlock()
}
type status struct {
Pid int
UpTime string
UpTimeSec float64
Time string
TimeUnix int64
StatusCodeCount map[string]int
TotalCount int
TotalResponseTime string
TotalResponseTimeSec float64
AverageResponseTime string
AverageResponseTimeSec float64
}
func (self *statusService) getStatus(w *ResponseWriter, r *Request) {
now := time.Now()
uptime := now.Sub(self.start)
totalCount := 0
for _, count := range self.responseCounts {
totalCount += count
}
totalResponseTime := self.totalResponseTime.Sub(time.Time{})
averageResponseTime := time.Duration(0)
if totalCount > 0 {
avgNs := int64(totalResponseTime) / int64(totalCount)
averageResponseTime = time.Duration(avgNs)
}
st := &status{
Pid: self.pid,
UpTime: uptime.String(),
UpTimeSec: uptime.Seconds(),
Time: now.String(),
TimeUnix: now.Unix(),
StatusCodeCount: self.responseCounts,
TotalCount: totalCount,
TotalResponseTime: totalResponseTime.String(),
TotalResponseTimeSec: totalResponseTime.Seconds(),
AverageResponseTime: averageResponseTime.String(),
AverageResponseTimeSec: averageResponseTime.Seconds(),
}
err := w.WriteJson(st)
if err != nil {
http.Error(w, err.Error(), 500)
}
}