// This tests the swift packagae // // It can be used with a real swift server which should be set up in // the environment variables SWIFT_API_USER, SWIFT_API_KEY and // SWIFT_AUTH_URL // In case those variables are not defined, a fake Swift server // is used instead - see Testing in README.md for more info // // The functions are designed to run in order and create things the // next function tests. This means that if it goes wrong it is likely // errors will propagate. You may need to tidy up the CONTAINER to // get it to run cleanly. package swift_test import ( "archive/tar" "bytes" "crypto/md5" "crypto/rand" "crypto/tls" "encoding/json" "encoding/xml" "fmt" "io" "io/ioutil" "net/http" "net/http/httptest" "os" "reflect" "strconv" "strings" "sync" "testing" "time" "github.com/ncw/swift" "github.com/ncw/swift/swifttest" ) var ( srv *swifttest.SwiftServer m1 = swift.Metadata{"Hello": "1", "potato-Salad": "2"} m2 = swift.Metadata{"hello": "", "potato-salad": ""} skipVersionTests = false ) const ( CONTAINER = "GoSwiftUnitTest" SEGMENTS_CONTAINER = "GoSwiftUnitTest_segments" VERSIONS_CONTAINER = "GoSwiftUnitTestVersions" CURRENT_CONTAINER = "GoSwiftUnitTestCurrent" OBJECT = "test_object" OBJECT2 = "test_object2" EMPTYOBJECT = "empty_test_object" CONTENTS = "12345" CONTENTS2 = "54321" CONTENT_SIZE = int64(len(CONTENTS)) CONTENT_MD5 = "827ccb0eea8a706c4c34a16891f84e7b" EMPTY_MD5 = "d41d8cd98f00b204e9800998ecf8427e" SECRET_KEY = "b3968d0207b54ece87cccc06515a89d4" ) type someTransport struct{ http.Transport } func makeConnection(t *testing.T) (*swift.Connection, func()) { var err error UserName := os.Getenv("SWIFT_API_USER") ApiKey := os.Getenv("SWIFT_API_KEY") AuthUrl := os.Getenv("SWIFT_AUTH_URL") Region := os.Getenv("SWIFT_REGION_NAME") EndpointType := os.Getenv("SWIFT_ENDPOINT_TYPE") Insecure := os.Getenv("SWIFT_AUTH_INSECURE") ConnectionChannelTimeout := os.Getenv("SWIFT_CONNECTION_CHANNEL_TIMEOUT") DataChannelTimeout := os.Getenv("SWIFT_DATA_CHANNEL_TIMEOUT") internalServer := false if UserName == "" || ApiKey == "" || AuthUrl == "" { srv, err = swifttest.NewSwiftServer("localhost") if err != nil && t != nil { t.Fatal("Failed to create server", err) } UserName = "swifttest" ApiKey = "swifttest" AuthUrl = srv.AuthURL internalServer = true } transport := &http.Transport{ Proxy: http.ProxyFromEnvironment, MaxIdleConnsPerHost: 2048, } if Insecure == "1" { transport.TLSClientConfig = &tls.Config{InsecureSkipVerify: true} } c := swift.Connection{ UserName: UserName, ApiKey: ApiKey, AuthUrl: AuthUrl, Region: Region, Transport: transport, ConnectTimeout: 60 * time.Second, Timeout: 60 * time.Second, EndpointType: swift.EndpointType(EndpointType), } if !internalServer { if isV3Api() { c.Tenant = os.Getenv("SWIFT_TENANT") c.Domain = os.Getenv("SWIFT_API_DOMAIN") } else { c.Tenant = os.Getenv("SWIFT_TENANT") c.TenantId = os.Getenv("SWIFT_TENANT_ID") } } var timeout int64 if ConnectionChannelTimeout != "" { timeout, err = strconv.ParseInt(ConnectionChannelTimeout, 10, 32) if err == nil { c.ConnectTimeout = time.Duration(timeout) * time.Second } } if DataChannelTimeout != "" { timeout, err = strconv.ParseInt(DataChannelTimeout, 10, 32) if err == nil { c.Timeout = time.Duration(timeout) * time.Second } } return &c, func() { if srv != nil { srv.Close() } } } func makeConnectionAuth(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnection(t) err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } return c, rollback } func makeConnectionWithContainer(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionAuth(t) err := c.ContainerCreate(CONTAINER, m1.ContainerHeaders()) if err != nil { t.Fatal(err) } return c, func() { c.ContainerDelete(CONTAINER) rollback() } } func makeConnectionWithObject(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithContainer(t) err := c.ObjectPutString(CONTAINER, OBJECT, CONTENTS, "") if err != nil { t.Fatal(err) } return c, func() { c.ObjectDelete(CONTAINER, OBJECT) rollback() } } func makeConnectionWithObjectHeaders(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithObject(t) err := c.ObjectUpdate(CONTAINER, OBJECT, m1.ObjectHeaders()) if err != nil { t.Fatal(err) } return c, rollback } func makeConnectionWithVersionsContainer(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionAuth(t) err := c.VersionContainerCreate(CURRENT_CONTAINER, VERSIONS_CONTAINER) newRollback := func() { c.ContainerDelete(CURRENT_CONTAINER) c.ContainerDelete(VERSIONS_CONTAINER) rollback() } if err != nil { if err == swift.Forbidden { skipVersionTests = true return c, newRollback } t.Fatal(err) } return c, newRollback } func makeConnectionWithVersionsObject(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithVersionsContainer(t) if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS, ""); err != nil { t.Fatal(err) } // Version 2 if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS2, ""); err != nil { t.Fatal(err) } // Version 3 if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS2, ""); err != nil { t.Fatal(err) } return c, func() { for i := 0; i < 3; i++ { c.ObjectDelete(CURRENT_CONTAINER, OBJECT) } rollback() } } func makeConnectionWithSegmentsContainer(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithContainer(t) err := c.ContainerCreate(SEGMENTS_CONTAINER, swift.Headers{}) if err != nil { t.Fatal(err) } return c, func() { err = c.ContainerDelete(SEGMENTS_CONTAINER) if err != nil { t.Fatal(err) } rollback() } } func makeConnectionWithDLO(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithSegmentsContainer(t) opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } for i := 0; i < 2; i++ { _, err = fmt.Fprintf(out, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } return c, func() { c.DynamicLargeObjectDelete(CONTAINER, OBJECT) rollback() } } func makeConnectionWithSLO(t *testing.T) (*swift.Connection, func()) { c, rollback := makeConnectionWithSegmentsContainer(t) opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.StaticLargeObjectCreate(&opts) if err != nil { if err == swift.SLONotSupported { t.Skip("SLO not supported") return c, rollback } t.Fatal(err) } for i := 0; i < 2; i++ { _, err = fmt.Fprintf(out, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } return c, func() { c.StaticLargeObjectDelete(CONTAINER, OBJECT) rollback() } } func isV3Api() bool { AuthUrl := os.Getenv("SWIFT_AUTH_URL") return strings.Contains(AuthUrl, "v3") } func TestTransport(t *testing.T) { c, rollback := makeConnection(t) defer rollback() tr := &someTransport{ Transport: http.Transport{ MaxIdleConnsPerHost: 2048, }, } Insecure := os.Getenv("SWIFT_AUTH_INSECURE") if Insecure == "1" { tr.TLSClientConfig = &tls.Config{InsecureSkipVerify: true} } c.Transport = tr err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } // The following Test functions are run in order - this one must come before the others! func TestV1V2Authenticate(t *testing.T) { if isV3Api() { return } c, rollback := makeConnection(t) defer rollback() err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } func TestV3AuthenticateWithDomainNameAndTenantId(t *testing.T) { if !isV3Api() { return } c, rollback := makeConnection(t) defer rollback() c.Tenant = "" c.Domain = os.Getenv("SWIFT_API_DOMAIN") c.TenantId = os.Getenv("SWIFT_TENANT_ID") c.DomainId = "" err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } func TestV3TrustWithTrustId(t *testing.T) { if !isV3Api() { return } c, rollback := makeConnection(t) defer rollback() c.TrustId = os.Getenv("SWIFT_TRUST_ID") err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } func TestV3AuthenticateWithDomainIdAndTenantId(t *testing.T) { if !isV3Api() { return } c, rollback := makeConnection(t) defer rollback() c.Tenant = "" c.Domain = "" c.TenantId = os.Getenv("SWIFT_TENANT_ID") c.DomainId = os.Getenv("SWIFT_API_DOMAIN_ID") err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } func TestV3AuthenticateWithDomainNameAndTenantName(t *testing.T) { if !isV3Api() { return } c, rollback := makeConnection(t) defer rollback() c.Tenant = os.Getenv("SWIFT_TENANT") c.Domain = os.Getenv("SWIFT_API_DOMAIN") c.TenantId = "" c.DomainId = "" err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } func TestV3AuthenticateWithDomainIdAndTenantName(t *testing.T) { if !isV3Api() { return } c, rollback := makeConnection(t) defer rollback() c.Tenant = os.Getenv("SWIFT_TENANT") c.Domain = "" c.TenantId = "" c.DomainId = os.Getenv("SWIFT_API_DOMAIN_ID") err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } } // Attempt to trigger a race in authenticate // // Run with -race to test func TestAuthenticateRace(t *testing.T) { c, rollback := makeConnection(t) defer rollback() var wg sync.WaitGroup for i := 0; i < 10; i++ { wg.Add(1) go func() { defer wg.Done() err := c.Authenticate() if err != nil { t.Fatal("Auth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } }() } wg.Wait() } // Test a connection can be serialized and unserialized with JSON func TestSerializeConnectionJson(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() serializedConnection, err := json.Marshal(c) if err != nil { t.Fatalf("Failed to serialize connection: %v", err) } c2 := new(swift.Connection) err = json.Unmarshal(serializedConnection, &c2) if err != nil { t.Fatalf("Failed to unserialize connection: %v", err) } if !c2.Authenticated() { t.Fatal("Should be authenticated") } _, _, err = c2.Account() if err != nil { t.Fatalf("Failed to use unserialized connection: %v", err) } } // Test a connection can be serialized and unserialized with XML func TestSerializeConnectionXml(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() serializedConnection, err := xml.Marshal(c) if err != nil { t.Fatalf("Failed to serialize connection: %v", err) } c2 := new(swift.Connection) err = xml.Unmarshal(serializedConnection, &c2) if err != nil { t.Fatalf("Failed to unserialize connection: %v", err) } if !c2.Authenticated() { t.Fatal("Should be authenticated") } _, _, err = c2.Account() if err != nil { t.Fatalf("Failed to use unserialized connection: %v", err) } } // Test the reauthentication logic func TestOnReAuth(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() c.UnAuthenticate() _, _, err := c.Account() if err != nil { t.Fatalf("Failed to reauthenticate: %v", err) } } func TestAccount(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() info, headers, err := c.Account() if err != nil { t.Fatal(err) } if headers["X-Account-Container-Count"] != fmt.Sprintf("%d", info.Containers) { t.Error("Bad container count") } if headers["X-Account-Bytes-Used"] != fmt.Sprintf("%d", info.BytesUsed) { t.Error("Bad bytes count") } if headers["X-Account-Object-Count"] != fmt.Sprintf("%d", info.Objects) { t.Error("Bad objects count") } } func compareMaps(t *testing.T, a, b map[string]string) { if len(a) != len(b) { t.Error("Maps different sizes", a, b) } for ka, va := range a { if vb, ok := b[ka]; !ok || va != vb { t.Error("Difference in key", ka, va, b[ka]) } } for kb, vb := range b { if va, ok := a[kb]; !ok || vb != va { t.Error("Difference in key", kb, vb, a[kb]) } } } func TestAccountUpdate(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() err := c.AccountUpdate(m1.AccountHeaders()) if err != nil { t.Fatal(err) } _, headers, err := c.Account() if err != nil { t.Fatal(err) } m := headers.AccountMetadata() delete(m, "temp-url-key") // remove X-Account-Meta-Temp-URL-Key if set compareMaps(t, m, map[string]string{"hello": "1", "potato-salad": "2"}) err = c.AccountUpdate(m2.AccountHeaders()) if err != nil { t.Fatal(err) } _, headers, err = c.Account() if err != nil { t.Fatal(err) } m = headers.AccountMetadata() delete(m, "temp-url-key") // remove X-Account-Meta-Temp-URL-Key if set compareMaps(t, m, map[string]string{}) } func TestContainerCreate(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() err := c.ContainerCreate(CONTAINER, m1.ContainerHeaders()) if err != nil { t.Fatal(err) } err = c.ContainerDelete(CONTAINER) if err != nil { t.Fatal(err) } } func TestContainer(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() info, headers, err := c.Container(CONTAINER) if err != nil { t.Fatal(err) } compareMaps(t, headers.ContainerMetadata(), map[string]string{"hello": "1", "potato-salad": "2"}) if CONTAINER != info.Name { t.Error("Bad container count") } if headers["X-Container-Bytes-Used"] != fmt.Sprintf("%d", info.Bytes) { t.Error("Bad bytes count") } if headers["X-Container-Object-Count"] != fmt.Sprintf("%d", info.Count) { t.Error("Bad objects count") } } func TestContainersAll(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() containers1, err := c.ContainersAll(nil) if err != nil { t.Fatal(err) } containers2, err := c.Containers(nil) if err != nil { t.Fatal(err) } if len(containers1) != len(containers2) { t.Fatal("Wrong length") } for i := range containers1 { if containers1[i] != containers2[i] { t.Fatal("Not the same") } } } func TestContainersAllWithLimit(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() containers1, err := c.ContainersAll(&swift.ContainersOpts{Limit: 1}) if err != nil { t.Fatal(err) } containers2, err := c.Containers(nil) if err != nil { t.Fatal(err) } if len(containers1) != len(containers2) { t.Fatal("Wrong length") } for i := range containers1 { if containers1[i] != containers2[i] { t.Fatal("Not the same") } } } func TestContainerUpdate(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ContainerUpdate(CONTAINER, m2.ContainerHeaders()) if err != nil { t.Fatal(err) } _, headers, err := c.Container(CONTAINER) if err != nil { t.Fatal(err) } compareMaps(t, headers.ContainerMetadata(), map[string]string{}) } func TestContainerNames(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() containers, err := c.ContainerNames(nil) if err != nil { t.Fatal(err) } ok := false for _, container := range containers { if container == CONTAINER { ok = true break } } if !ok { t.Errorf("Didn't find container %q in listing %q", CONTAINER, containers) } } func TestContainerNamesAll(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() containers1, err := c.ContainerNamesAll(nil) if err != nil { t.Fatal(err) } containers2, err := c.ContainerNames(nil) if err != nil { t.Fatal(err) } if len(containers1) != len(containers2) { t.Fatal("Wrong length") } for i := range containers1 { if containers1[i] != containers2[i] { t.Fatal("Not the same") } } } func TestContainerNamesAllWithLimit(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() containers1, err := c.ContainerNamesAll(&swift.ContainersOpts{Limit: 1}) if err != nil { t.Fatal(err) } containers2, err := c.ContainerNames(nil) if err != nil { t.Fatal(err) } if len(containers1) != len(containers2) { t.Fatal("Wrong length") } for i := range containers1 { if containers1[i] != containers2[i] { t.Fatal("Not the same") } } } func TestObjectPutString(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ObjectPutString(CONTAINER, OBJECT, CONTENTS, "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() info, _, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Error(err) } if info.ContentType != "application/octet-stream" { t.Error("Bad content type", info.ContentType) } if info.Bytes != CONTENT_SIZE { t.Error("Bad length") } if info.Hash != CONTENT_MD5 { t.Error("Bad length") } } func TestObjectPut(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() headers := swift.Headers{} // Set content size incorrectly - should produce an error headers["Content-Length"] = strconv.FormatInt(CONTENT_SIZE-1, 10) contents := bytes.NewBufferString(CONTENTS) h, err := c.ObjectPut(CONTAINER, OBJECT, contents, true, CONTENT_MD5, "text/plain", headers) if err == nil { t.Fatal("Expecting error but didn't get one") } // Now set content size correctly contents = bytes.NewBufferString(CONTENTS) headers["Content-Length"] = strconv.FormatInt(CONTENT_SIZE, 10) h, err = c.ObjectPut(CONTAINER, OBJECT, contents, true, CONTENT_MD5, "text/plain", headers) if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() if h["Etag"] != CONTENT_MD5 { t.Errorf("Bad Etag want %q got %q", CONTENT_MD5, h["Etag"]) } // Fetch object info and compare info, _, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Error(err) } if info.ContentType != "text/plain" { t.Error("Bad content type", info.ContentType) } if info.Bytes != CONTENT_SIZE { t.Error("Bad length") } if info.Hash != CONTENT_MD5 { t.Error("Bad length") } } func TestObjectEmpty(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ObjectPutString(CONTAINER, EMPTYOBJECT, "", "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, EMPTYOBJECT) if err != nil { t.Error(err) } }() info, _, err := c.Object(CONTAINER, EMPTYOBJECT) if err != nil { t.Error(err) } if info.ContentType != "application/octet-stream" { t.Error("Bad content type", info.ContentType) } if info.Bytes != 0 { t.Errorf("Bad length want 0 got %v", info.Bytes) } if info.Hash != EMPTY_MD5 { t.Errorf("Bad MD5 want %v got %v", EMPTY_MD5, info.Hash) } } func TestObjectPutBytes(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ObjectPutBytes(CONTAINER, OBJECT, []byte(CONTENTS), "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Error(err) } }() info, _, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Error(err) } if info.ContentType != "application/octet-stream" { t.Error("Bad content type", info.ContentType) } if info.Bytes != CONTENT_SIZE { t.Error("Bad length") } if info.Hash != CONTENT_MD5 { t.Error("Bad length") } } func TestObjectPutMimeType(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ObjectPutString(CONTAINER, "test.jpg", CONTENTS, "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, "test.jpg") if err != nil { t.Error(err) } }() info, _, err := c.Object(CONTAINER, "test.jpg") if err != nil { t.Error(err) } if info.ContentType != "image/jpeg" { t.Error("Bad content type", info.ContentType) } } func TestObjectCreate(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() out, err := c.ObjectCreate(CONTAINER, OBJECT2, true, "", "", nil) if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Error(err) } }() buf := &bytes.Buffer{} hash := md5.New() out2 := io.MultiWriter(out, buf, hash) for i := 0; i < 100; i++ { fmt.Fprintf(out2, "%d %s\n", i, CONTENTS) } // Ensure Headers fails if called prematurely _, err = out.Headers() if err == nil { t.Error("Headers should fail if called before Close()") } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT2) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } // Ensure Headers succeeds when called after a good upload headers, err := out.Headers() if err != nil { t.Error(err) } if len(headers) < 1 { t.Error("The Headers returned by Headers() should not be empty") } // Test writing on closed file n, err := out.Write([]byte{0}) if err == nil || n != 0 { t.Error("Expecting error and n == 0 writing on closed file", err, n) } // Now with hash instead out, err = c.ObjectCreate(CONTAINER, OBJECT2, false, fmt.Sprintf("%x", hash.Sum(nil)), "", nil) if err != nil { t.Fatal(err) } _, err = out.Write(buf.Bytes()) if err != nil { t.Error(err) } err = out.Close() if err != nil { t.Error(err) } contents, err = c.ObjectGetString(CONTAINER, OBJECT2) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } // Now with bad hash out, err = c.ObjectCreate(CONTAINER, OBJECT2, false, CONTENT_MD5, "", nil) if err != nil { t.Fatal(err) } // FIXME: work around bug which produces 503 not 422 for empty corrupted files fmt.Fprintf(out, "Sausage") err = out.Close() if err != swift.ObjectCorrupted { t.Error("Expecting object corrupted not", err) } } func TestObjectGetString(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if contents != CONTENTS { t.Error("Contents wrong") } } func TestObjectGetBytes(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() contents, err := c.ObjectGetBytes(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if string(contents) != CONTENTS { t.Error("Contents wrong") } } func TestObjectOpen(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() file, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, nil) if err != nil { t.Fatal(err) } var buf bytes.Buffer n, err := io.Copy(&buf, file) if err != nil { t.Fatal(err) } if n != CONTENT_SIZE { t.Fatal("Wrong length", n, CONTENT_SIZE) } if buf.String() != CONTENTS { t.Error("Contents wrong") } err = file.Close() if err != nil { t.Fatal(err) } } func TestObjectOpenPartial(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() file, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, nil) if err != nil { t.Fatal(err) } var buf bytes.Buffer n, err := io.CopyN(&buf, file, 1) if err != nil { t.Fatal(err) } if n != 1 { t.Fatal("Wrong length", n, CONTENT_SIZE) } if buf.String() != CONTENTS[:1] { t.Error("Contents wrong") } err = file.Close() if err != nil { t.Fatal(err) } } func TestObjectOpenLength(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() file, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, nil) if err != nil { t.Fatal(err) } // FIXME ideally this would check both branches of the Length() code n, err := file.Length() if err != nil { t.Fatal(err) } if n != CONTENT_SIZE { t.Fatal("Wrong length", n, CONTENT_SIZE) } err = file.Close() if err != nil { t.Fatal(err) } } func TestObjectOpenNotModified(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() _, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, swift.Headers{ "If-None-Match": CONTENT_MD5, }) if err != swift.NotModified { t.Fatal(err) } } func TestObjectOpenSeek(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() plan := []struct { whence int offset int64 result int64 }{ {-1, 0, 0}, {-1, 0, 1}, {-1, 0, 2}, {0, 0, 0}, {0, 0, 0}, {0, 1, 1}, {0, 2, 2}, {1, 0, 3}, {1, -2, 2}, {1, 1, 4}, {2, -1, 4}, {2, -3, 2}, {2, -2, 3}, {2, -5, 0}, {2, -4, 1}, } file, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, nil) if err != nil { t.Fatal(err) } for _, p := range plan { if p.whence >= 0 { var result int64 result, err = file.Seek(p.offset, p.whence) if err != nil { t.Fatal(err, p) } if result != p.result { t.Fatal("Seek result was", result, "expecting", p.result, p) } } var buf bytes.Buffer var n int64 n, err = io.CopyN(&buf, file, 1) if err != nil { t.Fatal(err, p) } if n != 1 { t.Fatal("Wrong length", n, p) } actual := buf.String() expected := CONTENTS[p.result : p.result+1] if actual != expected { t.Error("Contents wrong, expecting", expected, "got", actual, p) } } err = file.Close() if err != nil { t.Fatal(err) } } // Test seeking to the end to find the file size func TestObjectOpenSeekEnd(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() file, _, err := c.ObjectOpen(CONTAINER, OBJECT, true, nil) if err != nil { t.Fatal(err) } n, err := file.Seek(0, 2) // seek to end if err != nil { t.Fatal(err) } if n != CONTENT_SIZE { t.Fatal("Wrong offset", n) } // Now check reading returns EOF buf := make([]byte, 16) nn, err := io.ReadFull(file, buf) if err != io.EOF { t.Fatal(err) } if nn != 0 { t.Fatal("wrong length", n) } // Now seek back to start and check we can read the file n, err = file.Seek(0, 0) // seek to start if err != nil { t.Fatal(err) } if n != 0 { t.Fatal("Wrong offset", n) } // read file and check contents buf, err = ioutil.ReadAll(file) if err != nil { t.Fatal(err) } if string(buf) != CONTENTS { t.Fatal("wrong contents", string(buf)) } } func TestObjectUpdate(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() err := c.ObjectUpdate(CONTAINER, OBJECT, m1.ObjectHeaders()) if err != nil { t.Fatal(err) } } func checkTime(t *testing.T, when time.Time, low, high int) { dt := time.Now().Sub(when) if dt < time.Duration(low)*time.Second || dt > time.Duration(high)*time.Second { t.Errorf("Time is wrong: dt=%q, when=%q", dt, when) } } func TestObject(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() object, headers, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } compareMaps(t, headers.ObjectMetadata(), map[string]string{"hello": "1", "potato-salad": "2"}) if object.Name != OBJECT || object.Bytes != CONTENT_SIZE || object.ContentType != "application/octet-stream" || object.Hash != CONTENT_MD5 || object.PseudoDirectory != false || object.SubDir != "" { t.Error("Bad object info", object) } checkTime(t, object.LastModified, -10, 10) } func TestObjectUpdate2(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() err := c.ObjectUpdate(CONTAINER, OBJECT, m2.ObjectHeaders()) if err != nil { t.Fatal(err) } _, headers, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } compareMaps(t, headers.ObjectMetadata(), map[string]string{"hello": "", "potato-salad": ""}) } func TestContainers(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() containers, err := c.Containers(nil) if err != nil { t.Fatal(err) } ok := false for _, container := range containers { if container.Name == CONTAINER { ok = true // Container may or may not have the file contents in it // Swift updates may be behind if container.Count == 0 && container.Bytes == 0 { break } if container.Count == 1 && container.Bytes == CONTENT_SIZE { break } t.Errorf("Bad size of Container %q: %q", CONTAINER, container) break } } if !ok { t.Errorf("Didn't find container %q in listing %q", CONTAINER, containers) } } func TestObjectNames(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectNames(CONTAINER, nil) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0] != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjectNamesAll(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectNamesAll(CONTAINER, nil) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0] != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjectNamesAllWithLimit(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectNamesAll(CONTAINER, &swift.ObjectsOpts{Limit: 1}) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0] != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjectsWalk(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects := make([]string, 0) err := c.ObjectsWalk(container, nil, func(opts *swift.ObjectsOpts) (interface{}, error) { newObjects, err := c.ObjectNames(CONTAINER, opts) if err == nil { objects = append(objects, newObjects...) } return newObjects, err }) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0] != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjects(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.Objects(CONTAINER, &swift.ObjectsOpts{Delimiter: '/'}) if err != nil { t.Fatal(err) } if len(objects) != 1 { t.Fatal("Should only be 1 object") } object := objects[0] if object.Name != OBJECT || object.Bytes != CONTENT_SIZE || object.ContentType != "application/octet-stream" || object.Hash != CONTENT_MD5 || object.PseudoDirectory != false || object.SubDir != "" { t.Error("Bad object info", object) } checkTime(t, object.LastModified, -10, 10) } func TestObjectsDirectory(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() err := c.ObjectPutString(CONTAINER, "directory", "", "application/directory") if err != nil { t.Fatal(err) } defer c.ObjectDelete(CONTAINER, "directory") // Look for the directory object and check we aren't confusing // it with a pseudo directory object objects, err := c.Objects(CONTAINER, &swift.ObjectsOpts{Delimiter: '/'}) if err != nil { t.Fatal(err) } if len(objects) != 2 { t.Fatal("Should only be 2 objects") } found := false for i := range objects { object := objects[i] if object.Name == "directory" { found = true if object.Bytes != 0 || object.ContentType != "application/directory" || object.Hash != "d41d8cd98f00b204e9800998ecf8427e" || object.PseudoDirectory != false || object.SubDir != "" { t.Error("Bad object info", object) } checkTime(t, object.LastModified, -10, 10) } } if !found { t.Error("Didn't find directory object") } } func TestObjectsPseudoDirectory(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() err := c.ObjectPutString(CONTAINER, "directory/puppy.jpg", "cute puppy", "") if err != nil { t.Fatal(err) } defer c.ObjectDelete(CONTAINER, "directory/puppy.jpg") // Look for the pseudo directory objects, err := c.Objects(CONTAINER, &swift.ObjectsOpts{Delimiter: '/'}) if err != nil { t.Fatal(err) } if len(objects) != 2 { t.Fatal("Should only be 2 objects", objects) } found := false for i := range objects { object := objects[i] if object.Name == "directory/" { found = true if object.Bytes != 0 || object.ContentType != "application/directory" || object.Hash != "" || object.PseudoDirectory != true || object.SubDir != "directory/" && object.LastModified.IsZero() { t.Error("Bad object info", object) } } } if !found { t.Error("Didn't find directory object", objects) } // Look in the pseudo directory now objects, err = c.Objects(CONTAINER, &swift.ObjectsOpts{Delimiter: '/', Path: "directory/"}) if err != nil { t.Fatal(err) } if len(objects) != 1 { t.Fatal("Should only be 1 object", objects) } object := objects[0] if object.Name != "directory/puppy.jpg" || object.Bytes != 10 || object.ContentType != "image/jpeg" || object.Hash != "87a12ea22fca7f54f0cefef1da535489" || object.PseudoDirectory != false || object.SubDir != "" { t.Error("Bad object info", object) } checkTime(t, object.LastModified, -10, 10) } func TestObjectsAll(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectsAll(CONTAINER, nil) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0].Name != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjectsAllWithLimit(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectsAll(CONTAINER, &swift.ObjectsOpts{Limit: 1}) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0].Name != OBJECT { t.Error("Incorrect listing", objects) } } func TestObjectNamesWithPath(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() objects, err := c.ObjectNames(CONTAINER, &swift.ObjectsOpts{Delimiter: '/', Path: ""}) if err != nil { t.Fatal(err) } if len(objects) != 1 || objects[0] != OBJECT { t.Error("Bad listing with path", objects) } // fmt.Println(objects) objects, err = c.ObjectNames(CONTAINER, &swift.ObjectsOpts{Delimiter: '/', Path: "Downloads/"}) if err != nil { t.Fatal(err) } if len(objects) != 0 { t.Error("Bad listing with path", objects) } } func TestObjectCopy(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() _, err := c.ObjectCopy(CONTAINER, OBJECT, CONTAINER, OBJECT2, nil) if err != nil { t.Fatal(err) } err = c.ObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } } func TestObjectCopyDifficultName(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() const dest = OBJECT + "?param %30%31%32 £100" _, err := c.ObjectCopy(CONTAINER, OBJECT, CONTAINER, dest, nil) if err != nil { t.Fatal(err) } err = c.ObjectDelete(CONTAINER, dest) if err != nil { t.Fatal(err) } } func TestObjectCopyWithMetadata(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() m := swift.Metadata{} m["copy-special-metadata"] = "hello" m["hello"] = "9" h := m.ObjectHeaders() h["Content-Type"] = "image/jpeg" _, err := c.ObjectCopy(CONTAINER, OBJECT, CONTAINER, OBJECT2, h) if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } }() // Re-read the metadata to see if it is correct _, headers, err := c.Object(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } if headers["Content-Type"] != "image/jpeg" { t.Error("Didn't change content type") } compareMaps(t, headers.ObjectMetadata(), map[string]string{"hello": "9", "potato-salad": "2", "copy-special-metadata": "hello"}) } func TestObjectMove(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() err := c.ObjectMove(CONTAINER, OBJECT, CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } testExistenceAfterDelete(t, c, CONTAINER, OBJECT) _, _, err = c.Object(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } err = c.ObjectMove(CONTAINER, OBJECT2, CONTAINER, OBJECT) if err != nil { t.Fatal(err) } testExistenceAfterDelete(t, c, CONTAINER, OBJECT2) _, headers, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } compareMaps(t, headers.ObjectMetadata(), map[string]string{"hello": "1", "potato-salad": "2"}) } func TestObjectUpdateContentType(t *testing.T) { c, rollback := makeConnectionWithObjectHeaders(t) defer rollback() err := c.ObjectUpdateContentType(CONTAINER, OBJECT, "text/potato") if err != nil { t.Fatal(err) } // Re-read the metadata to see if it is correct _, headers, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if headers["Content-Type"] != "text/potato" { t.Error("Didn't change content type") } compareMaps(t, headers.ObjectMetadata(), map[string]string{"hello": "1", "potato-salad": "2"}) } func TestVersionContainerCreate(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() err := c.VersionContainerCreate(CURRENT_CONTAINER, VERSIONS_CONTAINER) defer func() { c.ContainerDelete(CURRENT_CONTAINER) c.ContainerDelete(VERSIONS_CONTAINER) }() if err != nil { if err == swift.Forbidden { t.Log("Server doesn't support Versions - skipping test") return } t.Fatal(err) } } func TestVersionObjectAdd(t *testing.T) { c, rollback := makeConnectionWithVersionsContainer(t) defer rollback() if skipVersionTests { t.Log("Server doesn't support Versions - skipping test") return } // Version 1 if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS, ""); err != nil { t.Fatal(err) } defer func() { err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() if contents, err := c.ObjectGetString(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } else if contents != CONTENTS { t.Error("Contents wrong") } // Version 2 if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS2, ""); err != nil { t.Fatal(err) } defer func() { err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() if contents, err := c.ObjectGetString(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } else if contents != CONTENTS2 { t.Error("Contents wrong") } // Version 3 if err := c.ObjectPutString(CURRENT_CONTAINER, OBJECT, CONTENTS2, ""); err != nil { t.Fatal(err) } defer func() { err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() } func TestVersionObjectList(t *testing.T) { c, rollback := makeConnectionWithVersionsObject(t) defer rollback() if skipVersionTests { t.Log("Server doesn't support Versions - skipping test") return } list, err := c.VersionObjectList(VERSIONS_CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if len(list) != 2 { t.Error("Version list should return 2 objects") } } func TestVersionObjectDelete(t *testing.T) { c, rollback := makeConnectionWithVersionsObject(t) defer rollback() if skipVersionTests { t.Log("Server doesn't support Versions - skipping test") return } // Delete Version 3 if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } // Delete Version 2 if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } // Contents should be reverted to Version 1 if contents, err := c.ObjectGetString(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } else if contents != CONTENTS { t.Error("Contents wrong") } } func TestVersionDeleteContent(t *testing.T) { c, rollback := makeConnectionWithVersionsObject(t) defer rollback() if skipVersionTests { t.Log("Server doesn't support Versions - skipping test") return } // Delete Version 3 if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } // Delete Version 2 if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } // Delete Version 1 if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != nil { t.Fatal(err) } if err := c.ObjectDelete(CURRENT_CONTAINER, OBJECT); err != swift.ObjectNotFound { t.Fatalf("Expecting Object not found error, got: %v", err) } } // Check for non existence after delete // May have to do it a few times to wait for swift to be consistent. func testExistenceAfterDelete(t *testing.T, c *swift.Connection, container, object string) { for i := 10; i <= 0; i-- { _, _, err := c.Object(container, object) if err == swift.ObjectNotFound { break } if i == 0 { t.Fatalf("Expecting object %q/%q not found not: err=%v", container, object, err) } time.Sleep(1 * time.Second) } } func TestObjectDelete(t *testing.T) { c, rollback := makeConnectionWithObject(t) defer rollback() err := c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } testExistenceAfterDelete(t, c, CONTAINER, OBJECT) err = c.ObjectDelete(CONTAINER, OBJECT) if err != swift.ObjectNotFound { t.Fatal("Expecting Object not found", err) } } func TestBulkDelete(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() result, err := c.BulkDelete(CONTAINER, []string{OBJECT}) if err == swift.Forbidden { t.Log("Server doesn't support BulkDelete - skipping test") return } if err != nil { t.Fatal(err) } if result.NumberNotFound != 1 { t.Error("Expected 1, actual:", result.NumberNotFound) } if result.NumberDeleted != 0 { t.Error("Expected 0, actual:", result.NumberDeleted) } err = c.ObjectPutString(CONTAINER, OBJECT, CONTENTS, "") if err != nil { t.Fatal(err) } result, err = c.BulkDelete(CONTAINER, []string{OBJECT2, OBJECT}) if err != nil { t.Fatal(err) } if result.NumberNotFound != 1 { t.Error("Expected 1, actual:", result.NumberNotFound) } if result.NumberDeleted != 1 { t.Error("Expected 1, actual:", result.NumberDeleted) } t.Log("Errors:", result.Errors) } func TestBulkUpload(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() buffer := new(bytes.Buffer) ds := tar.NewWriter(buffer) var files = []struct{ Name, Body string }{ {OBJECT, CONTENTS}, {OBJECT2, CONTENTS2}, } for _, file := range files { hdr := &tar.Header{ Name: file.Name, Size: int64(len(file.Body)), } if err := ds.WriteHeader(hdr); err != nil { t.Fatal(err) } if _, err := ds.Write([]byte(file.Body)); err != nil { t.Fatal(err) } } if err := ds.Close(); err != nil { t.Fatal(err) } result, err := c.BulkUpload(CONTAINER, buffer, swift.UploadTar, nil) if err == swift.Forbidden { t.Log("Server doesn't support BulkUpload - skipping test") return } if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } err = c.ObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } }() if result.NumberCreated != 2 { t.Error("Expected 2, actual:", result.NumberCreated) } t.Log("Errors:", result.Errors) _, _, err = c.Object(CONTAINER, OBJECT) if err != nil { t.Error("Expecting object to be found") } _, _, err = c.Object(CONTAINER, OBJECT2) if err != nil { t.Error("Expecting object to be found") } } func TestObjectDifficultName(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() const name = `hello? sausage/êé/Hello, 世界/ " ' @ < > & ?/` err := c.ObjectPutString(CONTAINER, name, CONTENTS, "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, name) if err != nil { t.Fatal(err) } }() objects, err := c.ObjectNamesAll(CONTAINER, nil) if err != nil { t.Error(err) } found := false for _, object := range objects { if object == name { found = true break } } if !found { t.Errorf("Couldn't find %q in listing %q", name, objects) } } func TestTempUrl(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ObjectPutBytes(CONTAINER, OBJECT, []byte(CONTENTS), "") if err != nil { t.Fatal(err) } defer func() { err = c.ObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() m := swift.Metadata{} m["temp-url-key"] = SECRET_KEY err = c.AccountUpdate(m.AccountHeaders()) if err != nil { t.Fatal(err) } expiresTime := time.Now().Add(20 * time.Minute) tempUrl := c.ObjectTempUrl(CONTAINER, OBJECT, SECRET_KEY, "GET", expiresTime) resp, err := http.Get(tempUrl) if err != nil { t.Fatal("Failed to retrieve file from temporary url") } defer resp.Body.Close() if resp.StatusCode == 401 { t.Log("Server doesn't support tempurl") } else if resp.StatusCode != 200 { t.Fatal("HTTP Error retrieving file from temporary url", resp.StatusCode) } else { var content []byte if content, err = ioutil.ReadAll(resp.Body); err != nil || string(content) != CONTENTS { t.Error("Bad content", err) } resp, err = http.Post(tempUrl, "image/jpeg", bytes.NewReader([]byte(CONTENTS))) if err != nil { t.Fatal("Failed to retrieve file from temporary url") } defer resp.Body.Close() if resp.StatusCode != 401 { t.Fatal("Expecting server to forbid access to object") } } } func TestQueryInfo(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() infos, err := c.QueryInfo() if err != nil { t.Log("Server doesn't support querying info") return } if _, ok := infos["swift"]; !ok { t.Fatal("No 'swift' section found in configuration") } } func TestDLOCreate(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } defer func() { err = c.DynamicLargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } info, _, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if info.ObjectType != swift.DynamicLargeObjectType { t.Errorf("Wrong ObjectType, expected %d, got: %d", swift.DynamicLargeObjectType, info.ObjectType) } if info.Bytes != int64(len(expected)) { t.Errorf("Wrong Bytes size, expected %d, got: %d", len(expected), info.Bytes) } } func TestDLOInsert(t *testing.T) { c, rollback := makeConnectionWithDLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, CheckHash: true, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreateFile(&opts) if err != nil { t.Fatal(err) } buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) _, err = fmt.Fprintf(multi, "%d%s\n", 0, CONTENTS) if err != nil { t.Fatal(err) } fmt.Fprintf(buf, "\n%d %s\n", 1, CONTENTS) err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOAppend(t *testing.T) { c, rollback := makeConnectionWithDLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, Flags: os.O_APPEND, CheckHash: true, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreateFile(&opts) if err != nil { t.Fatal(err) } contents, err := c.ObjectGetString(CONTAINER, OBJECT) buf := bytes.NewBuffer([]byte(contents)) multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i+10, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err = c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOTruncate(t *testing.T) { c, rollback := makeConnectionWithDLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, Flags: os.O_TRUNC, CheckHash: true, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreateFile(&opts) if err != nil { t.Fatal(err) } buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) _, err = fmt.Fprintf(multi, "%s", CONTENTS) if err != nil { t.Fatal(err) } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOMove(t *testing.T) { c, rollback := makeConnectionWithDLO(t) defer rollback() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } err = c.DynamicLargeObjectMove(CONTAINER, OBJECT, CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } defer func() { err = c.DynamicLargeObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } }() contents2, err := c.ObjectGetString(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } if contents2 != contents { t.Error("Contents wrong") } } func TestDLONoSegmentContainer(t *testing.T) { c, rollback := makeConnectionWithDLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", SegmentContainer: CONTAINER, } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOCreateMissingSegmentsInList(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() if srv == nil { t.Skipf("This test only runs with the fake swift server as it's needed to simulate eventual consistency problems.") return } listURL := "/v1/AUTH_" + swifttest.TEST_ACCOUNT + "/" + SEGMENTS_CONTAINER srv.SetOverride(listURL, func(w http.ResponseWriter, r *http.Request, recorder *httptest.ResponseRecorder) { for k, v := range recorder.HeaderMap { w.Header().Set(k, v[0]) } w.WriteHeader(recorder.Code) w.Write([]byte("null\n")) }) defer srv.UnsetOverride(listURL) headers := swift.Headers{} err := c.ContainerCreate(SEGMENTS_CONTAINER, headers) if err != nil { t.Fatal(err) } defer func() { err = c.ContainerDelete(SEGMENTS_CONTAINER) if err != nil { t.Fatal(err) } }() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } defer func() { err = c.DynamicLargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOCreateIncorrectSize(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() if srv == nil { t.Skipf("This test only runs with the fake swift server as it's needed to simulate eventual consistency problems.") return } listURL := "/v1/AUTH_" + swifttest.TEST_ACCOUNT + "/" + CONTAINER + "/" + OBJECT headCount := 0 expectedHeadCount := 5 srv.SetOverride(listURL, func(w http.ResponseWriter, r *http.Request, recorder *httptest.ResponseRecorder) { for k, v := range recorder.HeaderMap { w.Header().Set(k, v[0]) } if r.Method == "HEAD" { headCount++ if headCount < expectedHeadCount { w.Header().Set("Content-Length", "7") } } w.WriteHeader(recorder.Code) w.Write(recorder.Body.Bytes()) }) defer srv.UnsetOverride(listURL) headers := swift.Headers{} err := c.ContainerCreate(SEGMENTS_CONTAINER, headers) if err != nil { t.Fatal(err) } defer func() { err = c.ContainerDelete(SEGMENTS_CONTAINER) if err != nil { t.Fatal(err) } }() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } defer func() { err = c.DynamicLargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } if headCount != expectedHeadCount { t.Errorf("Unexpected HEAD requests count, expected %d, got: %d", expectedHeadCount, headCount) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestDLOConcurrentWrite(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() nConcurrency := 5 nChunks := 100 var chunkSize int64 = 1024 writeFn := func(i int) { objName := fmt.Sprintf("%s_concurrent_dlo_%d", OBJECT, i) opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: objName, ContentType: "image/jpeg", } out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } defer func() { err = c.DynamicLargeObjectDelete(CONTAINER, objName) if err != nil { t.Fatal(err) } }() buf := &bytes.Buffer{} for j := 0; j < nChunks; j++ { var data []byte var n int data, err = ioutil.ReadAll(io.LimitReader(rand.Reader, chunkSize)) if err != nil { t.Fatal(err) } multi := io.MultiWriter(buf, out) n, err = multi.Write(data) if err != nil { t.Fatal(err) } if int64(n) != chunkSize { t.Fatalf("expected to write %d, got: %d", chunkSize, n) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, objName) if err != nil { t.Error(err) } if contents != expected { t.Error("Contents wrong") } } wg := sync.WaitGroup{} for i := 0; i < nConcurrency; i++ { wg.Add(1) go func(i int) { defer wg.Done() writeFn(i) }(i) } wg.Wait() } func TestDLOSegmentation(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", ChunkSize: 6, NoBuffer: true, } testSegmentation(t, c, func() swift.LargeObjectFile { out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } return out }, []segmentTest{ { writes: []string{"0", "1", "2", "3", "4", "5", "6", "7", "8"}, expectedSegs: []string{"0", "1", "2", "3", "4", "5", "6", "7", "8"}, expectedValue: "012345678", }, { writes: []string{"012345", "012345"}, expectedSegs: []string{"012345", "012345"}, expectedValue: "012345012345", }, { writes: []string{"0123456", "0123456"}, expectedSegs: []string{"012345", "6", "012345", "6"}, expectedValue: "01234560123456", }, { writes: []string{"0123456", "0123456"}, seeks: []int{-4, 0}, expectedSegs: []string{"012012", "3456"}, expectedValue: "0120123456", }, { writes: []string{"0123456", "0123456", "abcde"}, seeks: []int{0, -11, 0}, expectedSegs: []string{"012abc", "d", "e12345", "6"}, expectedValue: "012abcde123456", }, { writes: []string{"0123456", "ab"}, seeks: []int{-4, 0}, expectedSegs: []string{"012ab5", "6"}, expectedValue: "012ab56", }, }) } func TestDLOSegmentationBuffered(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", ChunkSize: 6, } testSegmentation(t, c, func() swift.LargeObjectFile { out, err := c.DynamicLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } return out }, []segmentTest{ { writes: []string{"0", "1", "2", "3", "4", "5", "6", "7", "8"}, expectedSegs: []string{"012345", "678"}, expectedValue: "012345678", }, { writes: []string{"012345", "012345"}, expectedSegs: []string{"012345", "012345"}, expectedValue: "012345012345", }, { writes: []string{"0123456", "0123456"}, expectedSegs: []string{"012345", "6", "012345", "6"}, expectedValue: "01234560123456", }, { writes: []string{"0123456", "0123456"}, seeks: []int{-4, 0}, expectedSegs: []string{"012012", "3456"}, expectedValue: "0120123456", }, { writes: []string{"0123456", "0123456", "abcde"}, seeks: []int{0, -11, 0}, expectedSegs: []string{"012abc", "d", "e12345", "6"}, expectedValue: "012abcde123456", }, { writes: []string{"0123456", "ab"}, seeks: []int{-4, 0}, expectedSegs: []string{"012ab5", "6"}, expectedValue: "012ab56", }, }) } func TestSLOCreate(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.StaticLargeObjectCreate(&opts) if err != nil { if err == swift.SLONotSupported { t.Skip("SLO not supported") return } t.Fatal(err) } defer func() { err = c.StaticLargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } info, _, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if info.ObjectType != swift.StaticLargeObjectType { t.Errorf("Wrong ObjectType, expected %d, got: %d", swift.StaticLargeObjectType, info.ObjectType) } if info.Bytes != int64(len(expected)) { t.Errorf("Wrong Bytes size, expected %d, got: %d", len(expected), info.Bytes) } } func TestSLOInsert(t *testing.T) { c, rollback := makeConnectionWithSLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", } out, err := c.StaticLargeObjectCreateFile(&opts) if err != nil { t.Fatal(err) } buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) _, err = fmt.Fprintf(multi, "%d%s\n", 0, CONTENTS) if err != nil { t.Fatal(err) } fmt.Fprintf(buf, "\n%d %s\n", 1, CONTENTS) err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestSLOAppend(t *testing.T) { c, rollback := makeConnectionWithSLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, Flags: os.O_APPEND, CheckHash: true, ContentType: "image/jpeg", } out, err := c.StaticLargeObjectCreateFile(&opts) if err != nil { t.Fatal(err) } contents, err := c.ObjectGetString(CONTAINER, OBJECT) buf := bytes.NewBuffer([]byte(contents)) multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i+10, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err = c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } } func TestSLOMove(t *testing.T) { c, rollback := makeConnectionWithSLO(t) defer rollback() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } err = c.StaticLargeObjectMove(CONTAINER, OBJECT, CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } defer func() { err = c.StaticLargeObjectDelete(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } }() contents2, err := c.ObjectGetString(CONTAINER, OBJECT2) if err != nil { t.Fatal(err) } if contents2 != contents { t.Error("Contents wrong") } } func TestSLONoSegmentContainer(t *testing.T) { c, rollback := makeConnectionWithSLO(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", SegmentContainer: CONTAINER, } out, err := c.StaticLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } buf := &bytes.Buffer{} multi := io.MultiWriter(buf, out) for i := 0; i < 2; i++ { _, err = fmt.Fprintf(multi, "%d %s\n", i, CONTENTS) if err != nil { t.Fatal(err) } } err = out.Close() if err != nil { t.Error(err) } expected := buf.String() contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != expected { t.Errorf("Contents wrong, expected %q, got: %q", expected, contents) } err = c.StaticLargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } } func TestSLOMinChunkSize(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() if srv == nil { t.Skipf("This test only runs with the fake swift server as it's needed to simulate min segment size.") return } srv.SetOverride("/info", func(w http.ResponseWriter, r *http.Request, recorder *httptest.ResponseRecorder) { w.Write([]byte(`{"slo": {"min_segment_size": 4}}`)) }) defer srv.UnsetOverride("/info") c.QueryInfo() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", ChunkSize: 6, MinChunkSize: 0, NoBuffer: true, } testSLOSegmentation(t, c, func() swift.LargeObjectFile { out, err := c.StaticLargeObjectCreate(&opts) if err != nil { t.Fatal(err) } return out }) } func TestSLOSegmentation(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", ChunkSize: 6, MinChunkSize: 4, NoBuffer: true, } testSLOSegmentation(t, c, func() swift.LargeObjectFile { out, err := c.StaticLargeObjectCreate(&opts) if err != nil { if err == swift.SLONotSupported { t.Skip("SLO not supported") } t.Fatal(err) } return out }) } func TestSLOSegmentationBuffered(t *testing.T) { c, rollback := makeConnectionWithSegmentsContainer(t) defer rollback() opts := swift.LargeObjectOpts{ Container: CONTAINER, ObjectName: OBJECT, ContentType: "image/jpeg", ChunkSize: 6, MinChunkSize: 4, } testSegmentation(t, c, func() swift.LargeObjectFile { out, err := c.StaticLargeObjectCreate(&opts) if err != nil { if err == swift.SLONotSupported { t.Skip("SLO not supported") } t.Fatal(err) } return out }, []segmentTest{ { writes: []string{"0", "1", "2", "3", "4", "5", "6", "7", "8"}, expectedSegs: []string{"012345", "678"}, expectedValue: "012345678", }, { writes: []string{"012345", "012345"}, expectedSegs: []string{"012345", "012345"}, expectedValue: "012345012345", }, { writes: []string{"0123456", "0123456"}, expectedSegs: []string{"012345", "601234", "56"}, expectedValue: "01234560123456", }, { writes: []string{"0123456", "0123456"}, seeks: []int{-4, 0}, expectedSegs: []string{"012012", "3456"}, expectedValue: "0120123456", }, { writes: []string{"0123456", "0123456", "abcde"}, seeks: []int{0, -11, 0}, expectedSegs: []string{"012abc", "de1234", "56"}, expectedValue: "012abcde123456", }, { writes: []string{"0123456", "ab"}, seeks: []int{-4, 0}, expectedSegs: []string{"012ab5", "6"}, expectedValue: "012ab56", }, }) } func testSLOSegmentation(t *testing.T, c *swift.Connection, createObj func() swift.LargeObjectFile) { testCases := []segmentTest{ { writes: []string{"0", "1", "2", "3", "4", "5", "6", "7", "8"}, expectedSegs: []string{"0123", "4567", "8"}, expectedValue: "012345678", }, { writes: []string{"012345", "012345"}, expectedSegs: []string{"012345", "012345"}, expectedValue: "012345012345", }, { writes: []string{"0123456", "0123456"}, expectedSegs: []string{"012345", "601234", "56"}, expectedValue: "01234560123456", }, { writes: []string{"0123456", "0123456"}, seeks: []int{-4, 0}, expectedSegs: []string{"012012", "3456"}, expectedValue: "0120123456", }, { writes: []string{"0123456", "0123456", "abcde"}, seeks: []int{0, -11, 0}, expectedSegs: []string{"012abc", "de1234", "56"}, expectedValue: "012abcde123456", }, { writes: []string{"0123456", "ab"}, seeks: []int{-4, 0}, expectedSegs: []string{"012ab5", "6"}, expectedValue: "012ab56", }, } testSegmentation(t, c, createObj, testCases) } type segmentTest struct { writes []string seeks []int expectedSegs []string expectedValue string } func testSegmentation(t *testing.T, c *swift.Connection, createObj func() swift.LargeObjectFile, testCases []segmentTest) { var err error runTestCase := func(tCase segmentTest) { out := createObj() defer func() { err = c.LargeObjectDelete(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } }() for i, data := range tCase.writes { _, err = fmt.Fprint(out, data) if err != nil { t.Error(err) } if i < len(tCase.seeks)-1 { _, err = out.Seek(int64(tCase.seeks[i]), os.SEEK_CUR) if err != nil { t.Error(err) } } } err = out.Close() if err != nil { t.Error(err) } contents, err := c.ObjectGetString(CONTAINER, OBJECT) if err != nil { t.Error(err) } if contents != tCase.expectedValue { t.Errorf("Contents wrong, expected %q, got: %q", tCase.expectedValue, contents) } container, objects, err := c.LargeObjectGetSegments(CONTAINER, OBJECT) if err != nil { t.Error(err) } if container != SEGMENTS_CONTAINER { t.Errorf("Segments container wrong, expected %q, got: %q", SEGMENTS_CONTAINER, container) } _, headers, err := c.Object(CONTAINER, OBJECT) if err != nil { t.Fatal(err) } if headers.IsLargeObjectSLO() { var info swift.SwiftInfo info, err = c.QueryInfo() if err != nil { t.Fatal(err) } if info.SLOMinSegmentSize() > 4 { t.Log("Skipping checking segments because SLO min segment size imposed by server is larger than wanted for tests.") return } } var segContents []string for _, obj := range objects { var value string value, err = c.ObjectGetString(SEGMENTS_CONTAINER, obj.Name) if err != nil { t.Error(err) } segContents = append(segContents, value) } if !reflect.DeepEqual(segContents, tCase.expectedSegs) { t.Errorf("Segments wrong, expected %#v, got: %#v", tCase.expectedSegs, segContents) } } for _, tCase := range testCases { runTestCase(tCase) } } func TestContainerDelete(t *testing.T) { c, rollback := makeConnectionWithContainer(t) defer rollback() err := c.ContainerDelete(CONTAINER) if err != nil { t.Fatal(err) } err = c.ContainerDelete(CONTAINER) if err != swift.ContainerNotFound { t.Fatal("Expecting container not found", err) } _, _, err = c.Container(CONTAINER) if err != swift.ContainerNotFound { t.Fatal("Expecting container not found", err) } } func TestUnAuthenticate(t *testing.T) { c, rollback := makeConnectionAuth(t) defer rollback() c.UnAuthenticate() if c.Authenticated() { t.Fatal("Shouldn't be authenticated") } // Test re-authenticate err := c.Authenticate() if err != nil { t.Fatal("ReAuth failed", err) } if !c.Authenticated() { t.Fatal("Not authenticated") } }