Skip to content

Commit 8929d22

Browse files
author
Hagai Barel
committed
Add client unit tests and testdata
1 parent c46e9a0 commit 8929d22

File tree

7 files changed

+302
-3
lines changed

7 files changed

+302
-3
lines changed

internal/client/client.go

+12-3
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,9 @@ var (
2121
}
2222
//scraping endpoints for EMQ v3 api version
2323
targetsV3 = map[string]string{
24-
"node_metrics": "/api/v3/nodes/%s/metrics/",
25-
"node_stats": "/api/v3/nodes/%s/stats/",
26-
"nodes": "/api/v3/nodes/%s",
24+
"nodes_metrics": "/api/v3/nodes/%s/metrics/",
25+
"nodes_stats": "/api/v3/nodes/%s/stats/",
26+
"nodes": "/api/v3/nodes/%s",
2727
}
2828
)
2929

@@ -87,6 +87,11 @@ func (c *Client) Fetch() (map[string]interface{}, error) {
8787
return data, nil
8888
}
8989

90+
//set the host name for the client (mostly for testing purposes)
91+
func (c *Client) setHost(host string) {
92+
c.host = host
93+
}
94+
9095
//get preforms an http GET call to the provided path and returns the response
9196
func (c *Client) get(path string) (map[string]interface{}, error) {
9297

@@ -133,6 +138,10 @@ func (c *Client) newRequest(path string) (req *http.Request, err error) {
133138

134139
u := c.host + fmt.Sprintf(path, c.node)
135140

141+
if !strings.Contains(u, "://") {
142+
u = fmt.Sprintf("http://%s", u)
143+
}
144+
136145
log.Debugln("Fetching from", u)
137146

138147
req, err = http.NewRequest(http.MethodGet, u, nil)

internal/client/client_suite_test.go

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package client
2+
3+
import (
4+
"testing"
5+
6+
. "github.com/onsi/ginkgo"
7+
. "github.com/onsi/gomega"
8+
)
9+
10+
func TestClient(t *testing.T) {
11+
RegisterFailHandler(Fail)
12+
RunSpecs(t, "Client Suite")
13+
}

internal/client/client_test.go

+162
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,162 @@
1+
package client
2+
3+
import (
4+
"io/ioutil"
5+
"net/http"
6+
7+
. "github.com/onsi/ginkgo"
8+
. "github.com/onsi/gomega"
9+
"github.com/onsi/gomega/ghttp"
10+
)
11+
12+
//helper function to load json data from the testdata folder
13+
func loadData(path string) []byte {
14+
b, err := ioutil.ReadFile("testdata/" + path)
15+
if err != nil {
16+
panic(err)
17+
}
18+
return b
19+
}
20+
21+
var _ = Describe("Client", func() {
22+
23+
var (
24+
s *ghttp.Server
25+
c *Client
26+
)
27+
28+
BeforeEach(func() {
29+
s = ghttp.NewServer()
30+
//TODO(hagaibarel) add v2 tests
31+
c = NewClient(
32+
s.URL(),
33+
"emqx",
34+
"v3",
35+
"admin",
36+
"public",
37+
)
38+
})
39+
40+
AfterEach(func() {
41+
s.Close()
42+
})
43+
44+
Context("Successful fetching", func() {
45+
46+
BeforeEach(func() {
47+
//register route handlers
48+
s.RouteToHandler("GET", "/api/v3/nodes/emqx/metrics/", ghttp.CombineHandlers(
49+
ghttp.VerifyBasicAuth("admin", "public"),
50+
ghttp.VerifyHeader(http.Header{
51+
"Accept": []string{"application/json"},
52+
}),
53+
ghttp.RespondWith(200, loadData("metrics.json")),
54+
))
55+
56+
s.RouteToHandler("GET", "/api/v3/nodes/emqx/stats/", ghttp.CombineHandlers(
57+
ghttp.VerifyBasicAuth("admin", "public"),
58+
ghttp.VerifyHeader(http.Header{
59+
"Accept": []string{"application/json"},
60+
}),
61+
ghttp.RespondWith(200, loadData("stats.json")),
62+
))
63+
64+
s.RouteToHandler("GET", "/api/v3/nodes/emqx", ghttp.CombineHandlers(
65+
ghttp.VerifyBasicAuth("admin", "public"),
66+
ghttp.VerifyHeader(http.Header{
67+
"Accept": []string{"application/json"},
68+
}),
69+
ghttp.RespondWith(200, loadData("node.json")),
70+
))
71+
})
72+
73+
It("should succeed fetching the metrics", func() {
74+
res, err := c.Fetch()
75+
76+
Expect(err).ShouldNot(HaveOccurred())
77+
Expect(res).To(HaveKeyWithValue("nodes_version", "v3.0.1"))
78+
})
79+
})
80+
81+
Context("Failed requests", func() {
82+
83+
var (
84+
statusCode int
85+
body []byte
86+
path = "/api/v3/nodes/%s/stats"
87+
)
88+
89+
BeforeEach(func() {
90+
s.AppendHandlers(
91+
ghttp.CombineHandlers(
92+
ghttp.VerifyRequest("GET", "/api/v3/nodes/emqx/stats"),
93+
ghttp.VerifyBasicAuth("admin", "public"),
94+
ghttp.VerifyHeader(http.Header{
95+
"Accept": []string{"application/json"},
96+
}),
97+
ghttp.RespondWithPtr(&statusCode, &body),
98+
),
99+
)
100+
})
101+
102+
It("should fail when the response status code is not 200", func() {
103+
statusCode = http.StatusNotFound
104+
body = loadData("badresponse.json")
105+
106+
data, err := c.get(path)
107+
108+
Expect(err).To(HaveOccurred())
109+
Expect(err.Error()).To(ContainSubstring("Received status code not ok"))
110+
Expect(data).To(BeNil())
111+
})
112+
113+
It("should fail when the response body isn't valid json", func() {
114+
statusCode = http.StatusOK
115+
body = []byte("not valid json")
116+
117+
data, err := c.get(path)
118+
119+
Expect(err).To(HaveOccurred())
120+
Expect(err.Error()).To(ContainSubstring("Error in json decoder"))
121+
Expect(data).To(BeNil())
122+
})
123+
124+
It("should fail when the response body has Code != 0", func() {
125+
statusCode = http.StatusOK
126+
body = loadData("badresponse.json")
127+
128+
data, err := c.get(path)
129+
130+
Expect(err).To(HaveOccurred())
131+
Expect(err.Error()).To(ContainSubstring("Recvied code != 0"))
132+
Expect(data).To(BeNil())
133+
})
134+
135+
It("should fail to create a request for a bad path", func() {
136+
path := "/api/v3/nodes/emqx/stats"
137+
statusCode = http.StatusOK
138+
body = loadData("badresponse.json")
139+
140+
data, err := c.get(path)
141+
142+
Expect(err).To(HaveOccurred())
143+
Expect(err.Error()).To(ContainSubstring("Failed to create http request"))
144+
Expect(data).To(BeNil())
145+
146+
})
147+
148+
It("should fail for bad urls", func() {
149+
statusCode = http.StatusOK
150+
body = loadData("badresponse.json")
151+
152+
c.setHost("localhost:1859")
153+
154+
data, err := c.get(path)
155+
156+
Expect(err).To(HaveOccurred())
157+
Expect(err.Error()).To(ContainSubstring("Failed to get metrics"))
158+
Expect(data).To(BeNil())
159+
})
160+
161+
})
162+
})
+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
{
2+
"code": 1,
3+
"data": {
4+
"subscriptions/shared/max": 0,
5+
"subscriptions/max": 0,
6+
"subscribers/max": 0,
7+
"topics/count": 0,
8+
"subscriptions/count": 0,
9+
"suboptions/max": 0,
10+
"topics/max": 0,
11+
"sessions/persistent/max": 0,
12+
"connections/max": 0,
13+
"subscriptions/shared/count": 0,
14+
"sessions/persistent/count": 0,
15+
"retained/count": 3,
16+
"routes/count": 0,
17+
"sessions/count": 0,
18+
"retained/max": 3,
19+
"sessions/max": 0,
20+
"routes/max": 0,
21+
"subscribers/count": 0,
22+
"connections/count": 0
23+
}
24+
}

internal/client/testdata/metrics.json

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
{
2+
"code": 0,
3+
"data": {
4+
"messages/qos1/sent": 0,
5+
"packets/pubrel/missed": 0,
6+
"packets/puback/sent": 0,
7+
"messages/received": 0,
8+
"packets/unsuback": 0,
9+
"packets/pubrel/sent": 0,
10+
"packets/subscribe": 0,
11+
"packets/connack": 0,
12+
"packets/disconnect/sent": 0,
13+
"packets/pubcomp/sent": 0,
14+
"packets/unsubscribe": 0,
15+
"packets/auth": 0,
16+
"packets/suback": 0,
17+
"packets/pubrec/received": 0,
18+
"messages/expired": 0,
19+
"messages/qos2/received": 0,
20+
"packets/sent": 0,
21+
"packets/pubrel/received": 0,
22+
"messages/qos0/received": 0,
23+
"messages/forward": 0,
24+
"messages/dropped": 0,
25+
"messages/retained": 3,
26+
"messages/qos2/dropped": 0,
27+
"packets/pubrec/missed": 0,
28+
"packets/puback/missed": 0,
29+
"messages/qos2/sent": 0,
30+
"messages/qos2/expired": 0,
31+
"packets/pubrec/sent": 0,
32+
"messages/qos1/received": 0,
33+
"packets/puback/received": 0,
34+
"packets/connect": 0,
35+
"packets/pubcomp/received": 0,
36+
"messages/sent": 0,
37+
"messages/qos0/sent": 0,
38+
"packets/disconnect/received": 0,
39+
"packets/pingreq": 0,
40+
"packets/pubcomp/missed": 0,
41+
"packets/received": 0,
42+
"bytes/received": 0,
43+
"bytes/sent": 0,
44+
"packets/publish/sent": 0,
45+
"packets/publish/received": 0,
46+
"packets/pingresp": 0
47+
}
48+
}

internal/client/testdata/node.json

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
{
2+
"code": 0,
3+
"data": {
4+
"connections": 0,
5+
"load1": "2.04",
6+
"load15": "1.14",
7+
"load5": "1.25",
8+
"max_fds": 1048576,
9+
"memory_total": 154337280,
10+
"memory_used": 114375208,
11+
"name": "emqx",
12+
"node_status": "Running",
13+
"otp_release": "R21/10.2.1",
14+
"process_available": 2097152,
15+
"process_used": 388,
16+
"uptime": "1 hours, 39 minutes, 22 seconds",
17+
"version": "v3.0.1"
18+
}
19+
}

internal/client/testdata/stats.json

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
{
2+
"code": 0,
3+
"data": {
4+
"subscriptions/shared/max": 0,
5+
"subscriptions/max": 0,
6+
"subscribers/max": 0,
7+
"topics/count": 0,
8+
"subscriptions/count": 0,
9+
"suboptions/max": 0,
10+
"topics/max": 0,
11+
"sessions/persistent/max": 0,
12+
"connections/max": 0,
13+
"subscriptions/shared/count": 0,
14+
"sessions/persistent/count": 0,
15+
"retained/count": 3,
16+
"routes/count": 0,
17+
"sessions/count": 0,
18+
"retained/max": 3,
19+
"sessions/max": 0,
20+
"routes/max": 0,
21+
"subscribers/count": 0,
22+
"connections/count": 0
23+
}
24+
}

0 commit comments

Comments
 (0)