integration_test.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. // Copyright 2014 Google Inc. All Rights Reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package storage
  15. import (
  16. "bytes"
  17. "crypto/md5"
  18. "fmt"
  19. "io"
  20. "io/ioutil"
  21. "log"
  22. "math/rand"
  23. "net/http"
  24. "strings"
  25. "testing"
  26. "time"
  27. "golang.org/x/net/context"
  28. "google.golang.org/cloud"
  29. "google.golang.org/cloud/internal/testutil"
  30. )
  31. var (
  32. contents = make(map[string][]byte)
  33. objects = []string{"obj1", "obj2", "obj/with/slashes"}
  34. aclObjects = []string{"acl1", "acl2"}
  35. copyObj = "copy-object"
  36. )
  37. const envBucket = "GCLOUD_TESTS_GOLANG_PROJECT_ID"
  38. // testConfig returns the Client used to access GCS and the default bucket
  39. // name to use.
  40. func testConfig(ctx context.Context, t *testing.T) (*Client, string) {
  41. ts := cloud.WithTokenSource(testutil.TokenSource(ctx, ScopeFullControl))
  42. if ts == nil {
  43. t.Skip("Integration tests skipped. See CONTRIBUTING.md for details")
  44. }
  45. p := testutil.ProjID()
  46. if p == "" {
  47. log.Fatal("The project ID must be set. See CONTRIBUTING.md for details")
  48. }
  49. client, err := NewClient(ctx, ts)
  50. if err != nil {
  51. t.Fatalf("NewClient: %v", err)
  52. }
  53. return client, p
  54. }
  55. func TestAdminClient(t *testing.T) {
  56. ctx := context.Background()
  57. projectID := testutil.ProjID()
  58. newBucket := projectID + "copy"
  59. client, err := NewAdminClient(ctx, projectID, cloud.WithTokenSource(testutil.TokenSource(ctx, ScopeFullControl)))
  60. if err != nil {
  61. t.Fatalf("Could not create client: %v", err)
  62. }
  63. defer client.Close()
  64. if err := client.CreateBucket(ctx, newBucket, nil); err != nil {
  65. t.Errorf("CreateBucket(%v, %v) failed %v", newBucket, nil, err)
  66. }
  67. if err := client.DeleteBucket(ctx, newBucket); err != nil {
  68. t.Errorf("DeleteBucket(%v) failed %v", newBucket, err)
  69. t.Logf("TODO: Warning this test left a new bucket in the cloud project, it must be deleted manually")
  70. }
  71. attrs := BucketAttrs{
  72. DefaultObjectACL: []ACLRule{{Entity: "domain-google.com", Role: RoleReader}},
  73. }
  74. if err := client.CreateBucket(ctx, newBucket, &attrs); err != nil {
  75. t.Errorf("CreateBucket(%v, %v) failed %v", newBucket, attrs, err)
  76. }
  77. if err := client.DeleteBucket(ctx, newBucket); err != nil {
  78. t.Errorf("DeleteBucket(%v) failed %v", newBucket, err)
  79. t.Logf("TODO: Warning this test left a new bucket in the cloud project, it must be deleted manually")
  80. }
  81. }
  82. func TestObjects(t *testing.T) {
  83. if testing.Short() {
  84. t.Skip("Integration tests skipped in short mode")
  85. }
  86. ctx := context.Background()
  87. client, bucket := testConfig(ctx, t)
  88. defer client.Close()
  89. bkt := client.Bucket(bucket)
  90. // Cleanup.
  91. cleanup(t, "obj")
  92. const defaultType = "text/plain"
  93. // Test Writer.
  94. for _, obj := range objects {
  95. t.Logf("Writing %v", obj)
  96. wc := bkt.Object(obj).NewWriter(ctx)
  97. wc.ContentType = defaultType
  98. c := randomContents()
  99. if _, err := wc.Write(c); err != nil {
  100. t.Errorf("Write for %v failed with %v", obj, err)
  101. }
  102. if err := wc.Close(); err != nil {
  103. t.Errorf("Close for %v failed with %v", obj, err)
  104. }
  105. contents[obj] = c
  106. }
  107. // Test Reader.
  108. for _, obj := range objects {
  109. t.Logf("Creating a reader to read %v", obj)
  110. rc, err := bkt.Object(obj).NewReader(ctx)
  111. if err != nil {
  112. t.Errorf("Can't create a reader for %v, errored with %v", obj, err)
  113. }
  114. slurp, err := ioutil.ReadAll(rc)
  115. if err != nil {
  116. t.Errorf("Can't ReadAll object %v, errored with %v", obj, err)
  117. }
  118. if got, want := slurp, contents[obj]; !bytes.Equal(got, want) {
  119. t.Errorf("Contents (%q) = %q; want %q", obj, got, want)
  120. }
  121. if got, want := rc.Size(), len(contents[obj]); got != int64(want) {
  122. t.Errorf("Size (%q) = %d; want %d", obj, got, want)
  123. }
  124. if got, want := rc.ContentType(), "text/plain"; got != want {
  125. t.Errorf("ContentType (%q) = %q; want %q", obj, got, want)
  126. }
  127. rc.Close()
  128. // Test SignedURL
  129. opts := &SignedURLOptions{
  130. GoogleAccessID: "xxx@clientid",
  131. PrivateKey: dummyKey("rsa"),
  132. Method: "GET",
  133. MD5: []byte("202cb962ac59075b964b07152d234b70"),
  134. Expires: time.Date(2020, time.October, 2, 10, 0, 0, 0, time.UTC),
  135. ContentType: "application/json",
  136. Headers: []string{"x-header1", "x-header2"},
  137. }
  138. u, err := SignedURL(bucket, obj, opts)
  139. if err != nil {
  140. t.Fatalf("SignedURL(%q, %q) errored with %v", bucket, obj, err)
  141. }
  142. res, err := client.hc.Get(u)
  143. if err != nil {
  144. t.Fatalf("Can't get URL %q: %v", u, err)
  145. }
  146. slurp, err = ioutil.ReadAll(res.Body)
  147. if err != nil {
  148. t.Fatalf("Can't ReadAll signed object %v, errored with %v", obj, err)
  149. }
  150. if got, want := slurp, contents[obj]; !bytes.Equal(got, want) {
  151. t.Errorf("Contents (%v) = %q; want %q", obj, got, want)
  152. }
  153. res.Body.Close()
  154. }
  155. // Test NotFound.
  156. _, err := bkt.Object("obj-not-exists").NewReader(ctx)
  157. if err != ErrObjectNotExist {
  158. t.Errorf("Object should not exist, err found to be %v", err)
  159. }
  160. name := objects[0]
  161. // Test StatObject.
  162. o, err := bkt.Object(name).Attrs(ctx)
  163. if err != nil {
  164. t.Error(err)
  165. }
  166. if got, want := o.Name, name; got != want {
  167. t.Errorf("Name (%v) = %q; want %q", name, got, want)
  168. }
  169. if got, want := o.ContentType, defaultType; got != want {
  170. t.Errorf("ContentType (%v) = %q; want %q", name, got, want)
  171. }
  172. // Test object copy.
  173. copy, err := client.CopyObject(ctx, bucket, name, bucket, copyObj, nil)
  174. if err != nil {
  175. t.Errorf("CopyObject failed with %v", err)
  176. }
  177. if copy.Name != copyObj {
  178. t.Errorf("Copy object's name = %q; want %q", copy.Name, copyObj)
  179. }
  180. if copy.Bucket != bucket {
  181. t.Errorf("Copy object's bucket = %q; want %q", copy.Bucket, bucket)
  182. }
  183. // Test UpdateAttrs.
  184. updated, err := bkt.Object(name).Update(ctx, ObjectAttrs{
  185. ContentType: "text/html",
  186. ACL: []ACLRule{{Entity: "domain-google.com", Role: RoleReader}},
  187. })
  188. if err != nil {
  189. t.Errorf("UpdateAttrs failed with %v", err)
  190. }
  191. if want := "text/html"; updated.ContentType != want {
  192. t.Errorf("updated.ContentType == %q; want %q", updated.ContentType, want)
  193. }
  194. // Test checksums.
  195. checksumCases := []struct {
  196. name string
  197. contents [][]byte
  198. size int64
  199. md5 string
  200. crc32c uint32
  201. }{
  202. {
  203. name: "checksum-object",
  204. contents: [][]byte{[]byte("hello"), []byte("world")},
  205. size: 10,
  206. md5: "fc5e038d38a57032085441e7fe7010b0",
  207. crc32c: 1456190592,
  208. },
  209. {
  210. name: "zero-object",
  211. contents: [][]byte{},
  212. size: 0,
  213. md5: "d41d8cd98f00b204e9800998ecf8427e",
  214. crc32c: 0,
  215. },
  216. }
  217. for _, c := range checksumCases {
  218. wc := bkt.Object(c.name).NewWriter(ctx)
  219. for _, data := range c.contents {
  220. if _, err := wc.Write(data); err != nil {
  221. t.Errorf("Write(%q) failed with %q", data, err)
  222. }
  223. }
  224. if err = wc.Close(); err != nil {
  225. t.Errorf("%q: close failed with %q", c.name, err)
  226. }
  227. obj := wc.Attrs()
  228. if got, want := obj.Size, c.size; got != want {
  229. t.Errorf("Object (%q) Size = %v; want %v", c.name, got, want)
  230. }
  231. if got, want := fmt.Sprintf("%x", obj.MD5), c.md5; got != want {
  232. t.Errorf("Object (%q) MD5 = %q; want %q", c.name, got, want)
  233. }
  234. if got, want := obj.CRC32C, c.crc32c; got != want {
  235. t.Errorf("Object (%q) CRC32C = %v; want %v", c.name, got, want)
  236. }
  237. }
  238. // Test public ACL.
  239. publicObj := objects[0]
  240. if err = bkt.Object(publicObj).ACL().Set(ctx, AllUsers, RoleReader); err != nil {
  241. t.Errorf("PutACLEntry failed with %v", err)
  242. }
  243. publicClient, err := NewClient(ctx, cloud.WithBaseHTTP(http.DefaultClient))
  244. if err != nil {
  245. t.Fatal(err)
  246. }
  247. rc, err := publicClient.Bucket(bucket).Object(publicObj).NewReader(ctx)
  248. if err != nil {
  249. t.Error(err)
  250. }
  251. slurp, err := ioutil.ReadAll(rc)
  252. if err != nil {
  253. t.Errorf("ReadAll failed with %v", err)
  254. }
  255. if !bytes.Equal(slurp, contents[publicObj]) {
  256. t.Errorf("Public object's content: got %q, want %q", slurp, contents[publicObj])
  257. }
  258. rc.Close()
  259. // Test writer error handling.
  260. wc := publicClient.Bucket(bucket).Object(publicObj).NewWriter(ctx)
  261. if _, err := wc.Write([]byte("hello")); err != nil {
  262. t.Errorf("Write unexpectedly failed with %v", err)
  263. }
  264. if err = wc.Close(); err == nil {
  265. t.Error("Close expected an error, found none")
  266. }
  267. // DeleteObject object.
  268. // The rest of the other object will be deleted during
  269. // the initial cleanup. This tests exists, so we still can cover
  270. // deletion if there are no objects on the bucket to clean.
  271. if err := bkt.Object(copyObj).Delete(ctx); err != nil {
  272. t.Errorf("Deletion of %v failed with %v", copyObj, err)
  273. }
  274. _, err = bkt.Object(copyObj).Attrs(ctx)
  275. if err != ErrObjectNotExist {
  276. t.Errorf("Copy is expected to be deleted, stat errored with %v", err)
  277. }
  278. }
  279. func TestACL(t *testing.T) {
  280. if testing.Short() {
  281. t.Skip("Integration tests skipped in short mode")
  282. }
  283. ctx := context.Background()
  284. client, bucket := testConfig(ctx, t)
  285. defer client.Close()
  286. bkt := client.Bucket(bucket)
  287. cleanup(t, "acl")
  288. entity := ACLEntity("domain-google.com")
  289. if err := client.Bucket(bucket).DefaultObjectACL().Set(ctx, entity, RoleReader); err != nil {
  290. t.Errorf("Can't put default ACL rule for the bucket, errored with %v", err)
  291. }
  292. for _, obj := range aclObjects {
  293. t.Logf("Writing %v", obj)
  294. wc := bkt.Object(obj).NewWriter(ctx)
  295. c := randomContents()
  296. if _, err := wc.Write(c); err != nil {
  297. t.Errorf("Write for %v failed with %v", obj, err)
  298. }
  299. if err := wc.Close(); err != nil {
  300. t.Errorf("Close for %v failed with %v", obj, err)
  301. }
  302. }
  303. name := aclObjects[0]
  304. o := bkt.Object(name)
  305. acl, err := o.ACL().List(ctx)
  306. if err != nil {
  307. t.Errorf("Can't retrieve ACL of %v", name)
  308. }
  309. aclFound := false
  310. for _, rule := range acl {
  311. if rule.Entity == entity && rule.Role == RoleReader {
  312. aclFound = true
  313. }
  314. }
  315. if !aclFound {
  316. t.Error("Expected to find an ACL rule for google.com domain users, but not found")
  317. }
  318. if err := o.ACL().Delete(ctx, entity); err != nil {
  319. t.Errorf("Can't delete the ACL rule for the entity: %v", entity)
  320. }
  321. if err := bkt.ACL().Set(ctx, "user-jbd@google.com", RoleReader); err != nil {
  322. t.Errorf("Error while putting bucket ACL rule: %v", err)
  323. }
  324. bACL, err := bkt.ACL().List(ctx)
  325. if err != nil {
  326. t.Errorf("Error while getting the ACL of the bucket: %v", err)
  327. }
  328. bACLFound := false
  329. for _, rule := range bACL {
  330. if rule.Entity == "user-jbd@google.com" && rule.Role == RoleReader {
  331. bACLFound = true
  332. }
  333. }
  334. if !bACLFound {
  335. t.Error("Expected to find an ACL rule for jbd@google.com user, but not found")
  336. }
  337. if err := bkt.ACL().Delete(ctx, "user-jbd@google.com"); err != nil {
  338. t.Errorf("Error while deleting bucket ACL rule: %v", err)
  339. }
  340. }
  341. func TestValidObjectNames(t *testing.T) {
  342. if testing.Short() {
  343. t.Skip("Integration tests skipped in short mode")
  344. }
  345. ctx := context.Background()
  346. client, bucket := testConfig(ctx, t)
  347. defer client.Close()
  348. bkt := client.Bucket(bucket)
  349. validNames := []string{
  350. "gopher",
  351. "Гоферови",
  352. "a",
  353. strings.Repeat("a", 1024),
  354. }
  355. for _, name := range validNames {
  356. w := bkt.Object(name).NewWriter(ctx)
  357. if _, err := w.Write([]byte("data")); err != nil {
  358. t.Errorf("Object %q write failed: %v. Want success", name, err)
  359. continue
  360. }
  361. if err := w.Close(); err != nil {
  362. t.Errorf("Object %q close failed: %v. Want success", name, err)
  363. continue
  364. }
  365. defer bkt.Object(name).Delete(ctx)
  366. }
  367. invalidNames := []string{
  368. "", // Too short.
  369. strings.Repeat("a", 1025), // Too long.
  370. "new\nlines",
  371. "bad\xffunicode",
  372. }
  373. for _, name := range invalidNames {
  374. w := bkt.Object(name).NewWriter(ctx)
  375. // Invalid object names will either cause failure during Write or Close.
  376. if _, err := w.Write([]byte("data")); err != nil {
  377. continue
  378. }
  379. if err := w.Close(); err != nil {
  380. continue
  381. }
  382. defer bkt.Object(name).Delete(ctx)
  383. t.Errorf("%q should have failed. Didn't", name)
  384. }
  385. }
  386. func cleanup(t *testing.T, prefix string) {
  387. if testing.Short() {
  388. t.Skip("Integration tests cleanup skipped in short mode")
  389. }
  390. ctx := context.Background()
  391. client, bucket := testConfig(ctx, t)
  392. defer client.Close()
  393. var q *Query = &Query{
  394. Prefix: prefix,
  395. }
  396. for {
  397. o, err := client.Bucket(bucket).List(ctx, q)
  398. if err != nil {
  399. t.Fatalf("Cleanup List for bucket %v failed with error: %v", bucket, err)
  400. }
  401. for _, obj := range o.Results {
  402. t.Logf("Cleanup deletion of %v", obj.Name)
  403. if err = client.Bucket(bucket).Object(obj.Name).Delete(ctx); err != nil {
  404. t.Fatalf("Cleanup Delete for object %v failed with %v", obj.Name, err)
  405. }
  406. }
  407. if o.Next == nil {
  408. break
  409. }
  410. q = o.Next
  411. }
  412. }
  413. func randomContents() []byte {
  414. h := md5.New()
  415. io.WriteString(h, fmt.Sprintf("hello world%d", rand.Intn(100000)))
  416. return h.Sum(nil)
  417. }