123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409 |
- // Copyright 2014 Google Inc. All Rights Reserved.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- //[START sample]
- // Package gcsdemo is an example App Engine app using the Google Cloud Storage API.
- package gcsdemo
- //[START imports]
- import (
- "bytes"
- "fmt"
- "io"
- "io/ioutil"
- "net/http"
- "strings"
- "golang.org/x/net/context"
- "google.golang.org/appengine"
- "google.golang.org/appengine/file"
- "google.golang.org/appengine/log"
- "google.golang.org/cloud/storage"
- )
- //[END imports]
- // bucket is a local cache of the app's default bucket name.
- var bucket string // or: var bucket = "<your-app-id>.appspot.com"
- func init() {
- http.HandleFunc("/", handler)
- }
- // demo struct holds information needed to run the various demo functions.
- type demo struct {
- bucket *storage.BucketHandle
- client *storage.Client
- w http.ResponseWriter
- ctx context.Context
- // cleanUp is a list of filenames that need cleaning up at the end of the demo.
- cleanUp []string
- // failed indicates that one or more of the demo steps failed.
- failed bool
- }
- func (d *demo) errorf(format string, args ...interface{}) {
- d.failed = true
- log.Errorf(d.ctx, format, args...)
- }
- // handler is the main demo entry point that calls the GCS operations.
- func handler(w http.ResponseWriter, r *http.Request) {
- if r.URL.Path != "/" {
- http.NotFound(w, r)
- return
- }
- ctx := appengine.NewContext(r)
- if bucket == "" {
- var err error
- if bucket, err = file.DefaultBucketName(ctx); err != nil {
- log.Errorf(ctx, "failed to get default GCS bucket name: %v", err)
- return
- }
- }
- client, err := storage.NewClient(ctx)
- if err != nil {
- log.Errorf(ctx, "failed to get default GCS bucket name: %v", err)
- return
- }
- defer client.Close()
- w.Header().Set("Content-Type", "text/plain; charset=utf-8")
- fmt.Fprintf(w, "Demo GCS Application running from Version: %v\n", appengine.VersionID(ctx))
- fmt.Fprintf(w, "Using bucket name: %v\n\n", bucket)
- d := &demo{
- w: w,
- ctx: ctx,
- client: client,
- bucket: client.Bucket(bucket),
- }
- n := "demo-testfile-go"
- d.createFile(n)
- d.readFile(n)
- d.copyFile(n)
- d.statFile(n)
- d.createListFiles()
- d.listBucket()
- d.listBucketDirMode()
- d.defaultACL()
- d.putDefaultACLRule()
- d.deleteDefaultACLRule()
- d.bucketACL()
- d.putBucketACLRule()
- d.deleteBucketACLRule()
- d.acl(n)
- d.putACLRule(n)
- d.deleteACLRule(n)
- d.deleteFiles()
- if d.failed {
- io.WriteString(w, "\nDemo failed.\n")
- } else {
- io.WriteString(w, "\nDemo succeeded.\n")
- }
- }
- //[START write]
- // createFile creates a file in Google Cloud Storage.
- func (d *demo) createFile(fileName string) {
- fmt.Fprintf(d.w, "Creating file /%v/%v\n", bucket, fileName)
- wc := d.bucket.Object(fileName).NewWriter(d.ctx)
- wc.ContentType = "text/plain"
- wc.Metadata = map[string]string{
- "x-goog-meta-foo": "foo",
- "x-goog-meta-bar": "bar",
- }
- d.cleanUp = append(d.cleanUp, fileName)
- if _, err := wc.Write([]byte("abcde\n")); err != nil {
- d.errorf("createFile: unable to write data to bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- if _, err := wc.Write([]byte(strings.Repeat("f", 1024*4) + "\n")); err != nil {
- d.errorf("createFile: unable to write data to bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- if err := wc.Close(); err != nil {
- d.errorf("createFile: unable to close bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- }
- //[END write]
- //[START read]
- // readFile reads the named file in Google Cloud Storage.
- func (d *demo) readFile(fileName string) {
- io.WriteString(d.w, "\nAbbreviated file content (first line and last 1K):\n")
- rc, err := d.bucket.Object(fileName).NewReader(d.ctx)
- if err != nil {
- d.errorf("readFile: unable to open file from bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- defer rc.Close()
- slurp, err := ioutil.ReadAll(rc)
- if err != nil {
- d.errorf("readFile: unable to read data from bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- fmt.Fprintf(d.w, "%s\n", bytes.SplitN(slurp, []byte("\n"), 2)[0])
- if len(slurp) > 1024 {
- fmt.Fprintf(d.w, "...%s\n", slurp[len(slurp)-1024:])
- } else {
- fmt.Fprintf(d.w, "%s\n", slurp)
- }
- }
- //[END read]
- // copyFile copies a file in Google Cloud Storage.
- func (d *demo) copyFile(fileName string) {
- copyName := fileName + "-copy"
- fmt.Fprintf(d.w, "Copying file /%v/%v to /%v/%v:\n", bucket, fileName, bucket, copyName)
- obj, err := d.client.CopyObject(d.ctx, bucket, fileName, bucket, copyName, nil)
- if err != nil {
- d.errorf("copyFile: unable to copy /%v/%v to bucket %q, file %q: %v", bucket, fileName, bucket, copyName, err)
- return
- }
- d.cleanUp = append(d.cleanUp, copyName)
- d.dumpStats(obj)
- }
- func (d *demo) dumpStats(obj *storage.ObjectAttrs) {
- fmt.Fprintf(d.w, "(filename: /%v/%v, ", obj.Bucket, obj.Name)
- fmt.Fprintf(d.w, "ContentType: %q, ", obj.ContentType)
- fmt.Fprintf(d.w, "ACL: %#v, ", obj.ACL)
- fmt.Fprintf(d.w, "Owner: %v, ", obj.Owner)
- fmt.Fprintf(d.w, "ContentEncoding: %q, ", obj.ContentEncoding)
- fmt.Fprintf(d.w, "Size: %v, ", obj.Size)
- fmt.Fprintf(d.w, "MD5: %q, ", obj.MD5)
- fmt.Fprintf(d.w, "CRC32C: %q, ", obj.CRC32C)
- fmt.Fprintf(d.w, "Metadata: %#v, ", obj.Metadata)
- fmt.Fprintf(d.w, "MediaLink: %q, ", obj.MediaLink)
- fmt.Fprintf(d.w, "StorageClass: %q, ", obj.StorageClass)
- if !obj.Deleted.IsZero() {
- fmt.Fprintf(d.w, "Deleted: %v, ", obj.Deleted)
- }
- fmt.Fprintf(d.w, "Updated: %v)\n", obj.Updated)
- }
- // statFile reads the stats of the named file in Google Cloud Storage.
- func (d *demo) statFile(fileName string) {
- io.WriteString(d.w, "\nFile stat:\n")
- obj, err := d.bucket.Object(fileName).Attrs(d.ctx)
- if err != nil {
- d.errorf("statFile: unable to stat file from bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- d.dumpStats(obj)
- }
- // createListFiles creates files that will be used by listBucket.
- func (d *demo) createListFiles() {
- io.WriteString(d.w, "\nCreating more files for listbucket...\n")
- for _, n := range []string{"foo1", "foo2", "bar", "bar/1", "bar/2", "boo/"} {
- d.createFile(n)
- }
- }
- // listBucket lists the contents of a bucket in Google Cloud Storage.
- func (d *demo) listBucket() {
- io.WriteString(d.w, "\nListbucket result:\n")
- query := &storage.Query{Prefix: "foo"}
- for query != nil {
- objs, err := d.bucket.List(d.ctx, query)
- if err != nil {
- d.errorf("listBucket: unable to list bucket %q: %v", bucket, err)
- return
- }
- query = objs.Next
- for _, obj := range objs.Results {
- d.dumpStats(obj)
- }
- }
- }
- func (d *demo) listDir(name, indent string) {
- query := &storage.Query{Prefix: name, Delimiter: "/"}
- for query != nil {
- objs, err := d.bucket.List(d.ctx, query)
- if err != nil {
- d.errorf("listBucketDirMode: unable to list bucket %q: %v", bucket, err)
- return
- }
- query = objs.Next
- for _, obj := range objs.Results {
- fmt.Fprint(d.w, indent)
- d.dumpStats(obj)
- }
- for _, dir := range objs.Prefixes {
- fmt.Fprintf(d.w, "%v(directory: /%v/%v)\n", indent, bucket, dir)
- d.listDir(dir, indent+" ")
- }
- }
- }
- // listBucketDirMode lists the contents of a bucket in dir mode in Google Cloud Storage.
- func (d *demo) listBucketDirMode() {
- io.WriteString(d.w, "\nListbucket directory mode result:\n")
- d.listDir("b", "")
- }
- // dumpDefaultACL prints out the default object ACL for this bucket.
- func (d *demo) dumpDefaultACL() {
- acl, err := d.bucket.ACL().List(d.ctx)
- if err != nil {
- d.errorf("defaultACL: unable to list default object ACL for bucket %q: %v", bucket, err)
- return
- }
- for _, v := range acl {
- fmt.Fprintf(d.w, "Scope: %q, Permission: %q\n", v.Entity, v.Role)
- }
- }
- // defaultACL displays the default object ACL for this bucket.
- func (d *demo) defaultACL() {
- io.WriteString(d.w, "\nDefault object ACL:\n")
- d.dumpDefaultACL()
- }
- // putDefaultACLRule adds the "allUsers" default object ACL rule for this bucket.
- func (d *demo) putDefaultACLRule() {
- io.WriteString(d.w, "\nPut Default object ACL Rule:\n")
- err := d.bucket.DefaultObjectACL().Set(d.ctx, storage.AllUsers, storage.RoleReader)
- if err != nil {
- d.errorf("putDefaultACLRule: unable to save default object ACL rule for bucket %q: %v", bucket, err)
- return
- }
- d.dumpDefaultACL()
- }
- // deleteDefaultACLRule deleted the "allUsers" default object ACL rule for this bucket.
- func (d *demo) deleteDefaultACLRule() {
- io.WriteString(d.w, "\nDelete Default object ACL Rule:\n")
- err := d.bucket.DefaultObjectACL().Delete(d.ctx, storage.AllUsers)
- if err != nil {
- d.errorf("deleteDefaultACLRule: unable to delete default object ACL rule for bucket %q: %v", bucket, err)
- return
- }
- d.dumpDefaultACL()
- }
- // dumpBucketACL prints out the bucket ACL.
- func (d *demo) dumpBucketACL() {
- acl, err := d.bucket.ACL().List(d.ctx)
- if err != nil {
- d.errorf("dumpBucketACL: unable to list bucket ACL for bucket %q: %v", bucket, err)
- return
- }
- for _, v := range acl {
- fmt.Fprintf(d.w, "Scope: %q, Permission: %q\n", v.Entity, v.Role)
- }
- }
- // bucketACL displays the bucket ACL for this bucket.
- func (d *demo) bucketACL() {
- io.WriteString(d.w, "\nBucket ACL:\n")
- d.dumpBucketACL()
- }
- // putBucketACLRule adds the "allUsers" bucket ACL rule for this bucket.
- func (d *demo) putBucketACLRule() {
- io.WriteString(d.w, "\nPut Bucket ACL Rule:\n")
- err := d.bucket.ACL().Set(d.ctx, storage.AllUsers, storage.RoleReader)
- if err != nil {
- d.errorf("putBucketACLRule: unable to save bucket ACL rule for bucket %q: %v", bucket, err)
- return
- }
- d.dumpBucketACL()
- }
- // deleteBucketACLRule deleted the "allUsers" bucket ACL rule for this bucket.
- func (d *demo) deleteBucketACLRule() {
- io.WriteString(d.w, "\nDelete Bucket ACL Rule:\n")
- err := d.bucket.ACL().Delete(d.ctx, storage.AllUsers)
- if err != nil {
- d.errorf("deleteBucketACLRule: unable to delete bucket ACL rule for bucket %q: %v", bucket, err)
- return
- }
- d.dumpBucketACL()
- }
- // dumpACL prints out the ACL of the named file.
- func (d *demo) dumpACL(fileName string) {
- acl, err := d.bucket.Object(fileName).ACL().List(d.ctx)
- if err != nil {
- d.errorf("dumpACL: unable to list file ACL for bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- for _, v := range acl {
- fmt.Fprintf(d.w, "Scope: %q, Permission: %q\n", v.Entity, v.Role)
- }
- }
- // acl displays the ACL for the named file.
- func (d *demo) acl(fileName string) {
- fmt.Fprintf(d.w, "\nACL for file %v:\n", fileName)
- d.dumpACL(fileName)
- }
- // putACLRule adds the "allUsers" ACL rule for the named file.
- func (d *demo) putACLRule(fileName string) {
- fmt.Fprintf(d.w, "\nPut ACL rule for file %v:\n", fileName)
- err := d.bucket.Object(fileName).ACL().Set(d.ctx, storage.AllUsers, storage.RoleReader)
- if err != nil {
- d.errorf("putACLRule: unable to save ACL rule for bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- d.dumpACL(fileName)
- }
- // deleteACLRule deleted the "allUsers" ACL rule for the named file.
- func (d *demo) deleteACLRule(fileName string) {
- fmt.Fprintf(d.w, "\nDelete ACL rule for file %v:\n", fileName)
- err := d.bucket.Object(fileName).ACL().Delete(d.ctx, storage.AllUsers)
- if err != nil {
- d.errorf("deleteACLRule: unable to delete ACL rule for bucket %q, file %q: %v", bucket, fileName, err)
- return
- }
- d.dumpACL(fileName)
- }
- // deleteFiles deletes all the temporary files from a bucket created by this demo.
- func (d *demo) deleteFiles() {
- io.WriteString(d.w, "\nDeleting files...\n")
- for _, v := range d.cleanUp {
- fmt.Fprintf(d.w, "Deleting file %v\n", v)
- if err := d.bucket.Object(v).Delete(d.ctx); err != nil {
- d.errorf("deleteFiles: unable to delete bucket %q, file %q: %v", bucket, v, err)
- return
- }
- }
- }
- //[END sample]
|