14287: Test timestamp precision is maintained by response munging.
[arvados.git] / lib / controller / router / router_test.go
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 package router
6
7 import (
8         "bytes"
9         "encoding/json"
10         "io"
11         "net/http"
12         "net/http/httptest"
13         "os"
14         "testing"
15         "time"
16
17         "git.curoverse.com/arvados.git/sdk/go/arvados"
18         "git.curoverse.com/arvados.git/sdk/go/arvadostest"
19         check "gopkg.in/check.v1"
20 )
21
22 // Gocheck boilerplate
23 func Test(t *testing.T) {
24         check.TestingT(t)
25 }
26
27 var _ = check.Suite(&RouterSuite{})
28
29 type RouterSuite struct {
30         rtr *router
31 }
32
33 func (s *RouterSuite) SetUpTest(c *check.C) {
34         cluster := &arvados.Cluster{
35                 TLS: arvados.TLS{Insecure: true},
36         }
37         arvadostest.SetServiceURL(&cluster.Services.RailsAPI, "https://"+os.Getenv("ARVADOS_TEST_API_HOST"))
38         s.rtr = New(cluster)
39 }
40
41 func (s *RouterSuite) TearDownTest(c *check.C) {
42         err := arvados.NewClientFromEnv().RequestAndDecode(nil, "POST", "database/reset", nil, nil)
43         c.Check(err, check.IsNil)
44 }
45
46 func (s *RouterSuite) doRequest(c *check.C, token, method, path string, hdrs http.Header, body io.Reader) (*http.Request, *httptest.ResponseRecorder, map[string]interface{}) {
47         req := httptest.NewRequest(method, path, body)
48         for k, v := range hdrs {
49                 req.Header[k] = v
50         }
51         req.Header.Set("Authorization", "Bearer "+token)
52         rw := httptest.NewRecorder()
53         s.rtr.ServeHTTP(rw, req)
54         c.Logf("response body: %s", rw.Body.String())
55         var jresp map[string]interface{}
56         err := json.Unmarshal(rw.Body.Bytes(), &jresp)
57         c.Check(err, check.IsNil)
58         return req, rw, jresp
59 }
60
61 func (s *RouterSuite) TestCollectionParams(c *check.C) {
62         token := arvadostest.ActiveTokenV2
63
64         _, rw, jresp := s.doRequest(c, token, "GET", `/arvados/v1/collections?include_trash=true`, nil, nil)
65         c.Check(rw.Code, check.Equals, http.StatusOK)
66         c.Check(jresp["items_available"], check.FitsTypeOf, float64(0))
67
68         _, rw, jresp = s.doRequest(c, token, "GET", `/arvados/v1/collections`, nil, bytes.NewBufferString(`{"include_trash":true}`))
69         c.Check(rw.Code, check.Equals, http.StatusOK)
70         c.Check(jresp["items"], check.FitsTypeOf, []interface{}{})
71
72         _, rw, jresp = s.doRequest(c, token, "POST", `/arvados/v1/collections`, nil, bytes.NewBufferString(`ensure_unique_name=true`))
73         c.Check(rw.Code, check.Equals, http.StatusOK)
74         c.Check(jresp["uuid"], check.FitsTypeOf, "")
75
76         _, rw, jresp = s.doRequest(c, token, "POST", `/arvados/v1/collections?ensure_unique_name=true`, nil, nil)
77         c.Check(rw.Code, check.Equals, http.StatusOK)
78         c.Check(jresp["uuid"], check.FitsTypeOf, "")
79 }
80
81 func (s *RouterSuite) TestContainerList(c *check.C) {
82         token := arvadostest.ActiveTokenV2
83
84         _, rw, jresp := s.doRequest(c, token, "GET", `/arvados/v1/containers?limit=0`, nil, nil)
85         c.Check(rw.Code, check.Equals, http.StatusOK)
86         c.Check(jresp["items_available"], check.FitsTypeOf, float64(0))
87         c.Check(jresp["items_available"].(float64) > 2, check.Equals, true)
88         c.Check(jresp["items"], check.HasLen, 0)
89
90         _, rw, jresp = s.doRequest(c, token, "GET", `/arvados/v1/containers?limit=2&select=["uuid","command"]`, nil, nil)
91         c.Check(rw.Code, check.Equals, http.StatusOK)
92         c.Check(jresp["items_available"], check.FitsTypeOf, float64(0))
93         c.Check(jresp["items_available"].(float64) > 2, check.Equals, true)
94         c.Check(jresp["items"], check.HasLen, 2)
95         item0 := jresp["items"].([]interface{})[0].(map[string]interface{})
96         c.Check(item0["uuid"], check.HasLen, 27)
97         c.Check(item0["command"], check.FitsTypeOf, []interface{}{})
98         c.Check(item0["command"].([]interface{})[0], check.FitsTypeOf, "")
99         c.Check(item0["mounts"], check.IsNil)
100
101         _, rw, jresp = s.doRequest(c, token, "GET", `/arvados/v1/containers`, nil, nil)
102         c.Check(rw.Code, check.Equals, http.StatusOK)
103         c.Check(jresp["items_available"], check.FitsTypeOf, float64(0))
104         c.Check(jresp["items_available"].(float64) > 2, check.Equals, true)
105         avail := int(jresp["items_available"].(float64))
106         c.Check(jresp["items"], check.HasLen, avail)
107         item0 = jresp["items"].([]interface{})[0].(map[string]interface{})
108         c.Check(item0["uuid"], check.HasLen, 27)
109         c.Check(item0["command"], check.FitsTypeOf, []interface{}{})
110         c.Check(item0["command"].([]interface{})[0], check.FitsTypeOf, "")
111         c.Check(item0["mounts"], check.NotNil)
112 }
113
114 func (s *RouterSuite) TestContainerLock(c *check.C) {
115         uuid := arvadostest.QueuedContainerUUID
116         token := arvadostest.AdminToken
117         _, rw, jresp := s.doRequest(c, token, "POST", "/arvados/v1/containers/"+uuid+"/lock", nil, nil)
118         c.Check(rw.Code, check.Equals, http.StatusOK)
119         c.Check(jresp["uuid"], check.HasLen, 27)
120         c.Check(jresp["state"], check.Equals, "Locked")
121         _, rw, jresp = s.doRequest(c, token, "POST", "/arvados/v1/containers/"+uuid+"/lock", nil, nil)
122         c.Check(rw.Code, check.Equals, http.StatusUnprocessableEntity)
123         c.Check(rw.Body.String(), check.Not(check.Matches), `.*"uuid":.*`)
124         _, rw, jresp = s.doRequest(c, token, "POST", "/arvados/v1/containers/"+uuid+"/unlock", nil, nil)
125         c.Check(rw.Code, check.Equals, http.StatusOK)
126         c.Check(jresp["uuid"], check.HasLen, 27)
127         c.Check(jresp["state"], check.Equals, "Queued")
128         c.Check(jresp["environment"], check.IsNil)
129         _, rw, jresp = s.doRequest(c, token, "POST", "/arvados/v1/containers/"+uuid+"/unlock", nil, nil)
130         c.Check(rw.Code, check.Equals, http.StatusUnprocessableEntity)
131         c.Check(jresp["uuid"], check.IsNil)
132 }
133
134 func (s *RouterSuite) TestFullTimestampsInResponse(c *check.C) {
135         uuid := arvadostest.CollectionReplicationDesired2Confirmed2UUID
136         token := arvadostest.ActiveTokenV2
137
138         _, rw, jresp := s.doRequest(c, token, "GET", `/arvados/v1/collections/`+uuid, nil, nil)
139         c.Check(rw.Code, check.Equals, http.StatusOK)
140         c.Check(jresp["uuid"], check.Equals, uuid)
141         expectNS := map[string]int{
142                 "created_at":  596506000, // fixture says 596506247, but truncated by postgresql
143                 "modified_at": 596338000, // fixture says 596338465, but truncated by postgresql
144         }
145         for key, ns := range expectNS {
146                 mt, ok := jresp[key].(string)
147                 c.Logf("jresp[%q] == %q", key, mt)
148                 c.Assert(ok, check.Equals, true)
149                 t, err := time.Parse(time.RFC3339Nano, mt)
150                 c.Check(err, check.IsNil)
151                 c.Check(t.Nanosecond(), check.Equals, ns)
152         }
153 }
154
155 func (s *RouterSuite) TestSelectParam(c *check.C) {
156         uuid := arvadostest.QueuedContainerUUID
157         token := arvadostest.ActiveTokenV2
158         for _, sel := range [][]string{
159                 {"uuid", "command"},
160                 {"uuid", "command", "uuid"},
161                 {"", "command", "uuid"},
162         } {
163                 j, err := json.Marshal(sel)
164                 c.Assert(err, check.IsNil)
165                 _, rw, resp := s.doRequest(c, token, "GET", "/arvados/v1/containers/"+uuid+"?select="+string(j), nil, nil)
166                 c.Check(rw.Code, check.Equals, http.StatusOK)
167
168                 c.Check(resp["uuid"], check.HasLen, 27)
169                 c.Check(resp["command"], check.HasLen, 2)
170                 c.Check(resp["mounts"], check.IsNil)
171                 _, hasMounts := resp["mounts"]
172                 c.Check(hasMounts, check.Equals, false)
173         }
174 }