1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
14 "git.curoverse.com/arvados.git/sdk/go/arvadostest"
15 "git.curoverse.com/arvados.git/sdk/go/ctxlog"
16 "git.curoverse.com/arvados.git/sdk/go/httpserver"
17 "github.com/prometheus/client_golang/prometheus"
18 check "gopkg.in/check.v1"
21 func (s *HandlerSuite) TestMounts(c *check.C) {
22 c.Assert(s.handler.setup(context.Background(), s.cluster, "", prometheus.NewRegistry(), testServiceURL), check.IsNil)
24 vols := s.handler.volmgr.AllWritable()
25 vols[0].Put(context.Background(), TestHash, TestBlock)
26 vols[1].Put(context.Background(), TestHash2, TestBlock2)
28 resp := s.call("GET", "/mounts", "", nil)
29 c.Check(resp.Code, check.Equals, http.StatusOK)
30 var mntList []struct {
31 UUID string `json:"uuid"`
32 DeviceID string `json:"device_id"`
33 ReadOnly bool `json:"read_only"`
34 Replication int `json:"replication"`
35 StorageClasses map[string]bool `json:"storage_classes"`
37 c.Log(resp.Body.String())
38 err := json.Unmarshal(resp.Body.Bytes(), &mntList)
39 c.Assert(err, check.IsNil)
40 c.Assert(len(mntList), check.Equals, 2)
41 for _, m := range mntList {
42 c.Check(len(m.UUID), check.Equals, 27)
43 c.Check(m.UUID[:12], check.Equals, "zzzzz-nyw5e-")
44 c.Check(m.DeviceID, check.Equals, "mock-device-id")
45 c.Check(m.ReadOnly, check.Equals, false)
46 c.Check(m.Replication, check.Equals, 1)
47 c.Check(m.StorageClasses, check.DeepEquals, map[string]bool{"default": true})
49 c.Check(mntList[0].UUID, check.Not(check.Equals), mntList[1].UUID)
52 for _, tok := range []string{"", "xyzzy"} {
53 resp = s.call("GET", "/mounts/"+mntList[1].UUID+"/blocks", tok, nil)
54 c.Check(resp.Code, check.Equals, http.StatusUnauthorized)
55 c.Check(resp.Body.String(), check.Equals, "Unauthorized\n")
58 tok := arvadostest.DataManagerToken
60 // Nonexistent mount UUID
61 resp = s.call("GET", "/mounts/X/blocks", tok, nil)
62 c.Check(resp.Code, check.Equals, http.StatusNotFound)
63 c.Check(resp.Body.String(), check.Equals, "mount not found\n")
65 // Complete index of first mount
66 resp = s.call("GET", "/mounts/"+mntList[0].UUID+"/blocks", tok, nil)
67 c.Check(resp.Code, check.Equals, http.StatusOK)
68 c.Check(resp.Body.String(), check.Matches, TestHash+`\+[0-9]+ [0-9]+\n\n`)
70 // Partial index of first mount (one block matches prefix)
71 resp = s.call("GET", "/mounts/"+mntList[0].UUID+"/blocks?prefix="+TestHash[:2], tok, nil)
72 c.Check(resp.Code, check.Equals, http.StatusOK)
73 c.Check(resp.Body.String(), check.Matches, TestHash+`\+[0-9]+ [0-9]+\n\n`)
75 // Complete index of second mount (note trailing slash)
76 resp = s.call("GET", "/mounts/"+mntList[1].UUID+"/blocks/", tok, nil)
77 c.Check(resp.Code, check.Equals, http.StatusOK)
78 c.Check(resp.Body.String(), check.Matches, TestHash2+`\+[0-9]+ [0-9]+\n\n`)
80 // Partial index of second mount (no blocks match prefix)
81 resp = s.call("GET", "/mounts/"+mntList[1].UUID+"/blocks/?prefix="+TestHash[:2], tok, nil)
82 c.Check(resp.Code, check.Equals, http.StatusOK)
83 c.Check(resp.Body.String(), check.Equals, "\n")
86 func (s *HandlerSuite) TestMetrics(c *check.C) {
87 reg := prometheus.NewRegistry()
88 c.Assert(s.handler.setup(context.Background(), s.cluster, "", reg, testServiceURL), check.IsNil)
89 instrumented := httpserver.Instrument(reg, ctxlog.TestLogger(c), s.handler.Handler)
90 s.handler.Handler = instrumented.ServeAPI(s.cluster.ManagementToken, instrumented)
92 s.call("PUT", "/"+TestHash, "", TestBlock)
93 s.call("PUT", "/"+TestHash2, "", TestBlock2)
94 resp := s.call("GET", "/metrics.json", "", nil)
95 c.Check(resp.Code, check.Equals, http.StatusUnauthorized)
96 resp = s.call("GET", "/metrics.json", "foobar", nil)
97 c.Check(resp.Code, check.Equals, http.StatusForbidden)
98 resp = s.call("GET", "/metrics.json", arvadostest.ManagementToken, nil)
99 c.Check(resp.Code, check.Equals, http.StatusOK)
115 json.NewDecoder(resp.Body).Decode(&j)
116 found := make(map[string]bool)
117 names := map[string]bool{}
118 for _, g := range j {
120 for _, m := range g.Metric {
121 if len(m.Label) == 2 && m.Label[0].Name == "code" && m.Label[0].Value == "200" && m.Label[1].Name == "method" && m.Label[1].Value == "put" {
122 c.Check(m.Summary.SampleCount, check.Equals, "2")
128 metricsNames := []string{
129 "arvados_keepstore_bufferpool_inuse_buffers",
130 "arvados_keepstore_bufferpool_max_buffers",
131 "arvados_keepstore_bufferpool_allocated_bytes",
132 "arvados_keepstore_pull_queue_inprogress_entries",
133 "arvados_keepstore_pull_queue_pending_entries",
134 "arvados_keepstore_trash_queue_inprogress_entries",
135 "arvados_keepstore_trash_queue_pending_entries",
136 "request_duration_seconds",
138 for _, m := range metricsNames {
140 c.Check(ok, check.Equals, true, check.Commentf("checking metric %q", m))
144 func (s *HandlerSuite) call(method, path, tok string, body []byte) *httptest.ResponseRecorder {
145 resp := httptest.NewRecorder()
146 req, _ := http.NewRequest(method, path, bytes.NewReader(body))
148 req.Header.Set("Authorization", "Bearer "+tok)
150 s.handler.ServeHTTP(resp, req)