/* Copyright 2017 Google Inc 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. */ using Google; using Google.Apis.Http; using Google.Apis.Logging; using Google.Apis.Requests; using Google.Apis.Services; using Google.Apis.Storage.v1; using Google.Apis.Storage.v1.Data; using Google.Apis.Tests.Mocks; using Google.Apis.Upload; using IntegrationTests.Utils; using System; using System.Collections.Generic; using System.IO; using System.Text; using Xunit; namespace IntegrationTests { public class StorageApiTests { [Fact] public void PrettyPrinting() { StorageService client = new StorageService(new BaseClientService.Initializer { HttpClientInitializer = Helper.GetServiceCredential().CreateScoped(StorageService.Scope.DevstorageFullControl), ApplicationName = "IntegrationTest" }); var memLog = new MemoryLogger(LogLevel.All, clock: new MockClock(DateTime.UtcNow)); client.HttpClient.MessageHandler.InstanceLogger = memLog; client.HttpClient.MessageHandler.LogEvents = ConfigurableMessageHandler.LogEventType.ResponseBody; var req = client.Buckets.List(Helper.GetProjectId()).Configure(req => req.PrettyPrint = true); var resp = req.Execute(); var entry = Assert.Single(memLog.LogEntries); // The JSON is pretty-printed, so contains many space characters Assert.Contains(" ", entry.Split(new[] { "Body: " }, StringSplitOptions.None)[1]); req.PrettyPrint = null; resp = req.Execute(); Assert.Equal(2, memLog.LogEntries.Count); entry = memLog.LogEntries[1]; // The JSON is pretty-printed by default (PrettyPrint set to null), so contains many space characters Assert.Contains(" ", entry.Split(new[] { "Body: " }, StringSplitOptions.None)[1]); req.PrettyPrint = false; resp = req.Execute(); Assert.Equal(3, memLog.LogEntries.Count); entry = memLog.LogEntries[2]; // The JSON is not pretty-printed, so contains no space characters Assert.DoesNotContain(" ", entry.Split(new[] { "Body: " }, StringSplitOptions.None)[1]); } private const string TestBucketPrefix = "testbucket"; private Bucket CreateTestBucket(StorageService client) { var name = $"{TestBucketPrefix}-{Guid.NewGuid()}"; return client.Buckets.Insert(new Bucket { Name = name }, Helper.GetProjectId()).Execute(); } private void DeleteTestBuckets(StorageService client) { foreach (var bucket in client.Buckets.List(Helper.GetProjectId()).Execute().Items) { if (bucket.Name.StartsWith(TestBucketPrefix, StringComparison.Ordinal)) { try { client.Buckets.Delete(bucket.Name).Execute(); } catch (Exception e) { Console.WriteLine($"DeleteTestBuckets exception: {e}"); } } } } // TODO: Sort out an ETagAction test that works. [Fact(Skip = "GCS etag If-Match/If-None-Match support is currently broken.")] public void ETags() { StorageService client = new StorageService(new BaseClientService.Initializer { HttpClientInitializer = Helper.GetServiceCredential().CreateScoped(StorageService.Scope.DevstorageFullControl), ApplicationName = "IntegrationTest", }); var bucket = CreateTestBucket(client); try { // Set a label with If-None-Match header. This should not change any labels bucket.Labels = new Dictionary { ["label"] = "1" }; var req1 = client.Buckets.Patch(bucket, bucket.Name) .Configure(req => req.ETagAction = Google.Apis.ETagAction.IfNoneMatch); var ex1 = Assert.Throws(() => req1.Execute()); Assert.Contains("[Precondition Failed", ex1.Message); // Check that labels have not been added, and etag is the same var bucket1 = client.Buckets.Get(bucket.Name).Execute(); Assert.Empty(bucket1.Labels.EmptyIfNull()); Assert.Equal(bucket.ETag, bucket1.ETag); // Set a label with If-Match header. This should change labels var bucket2 = client.Buckets.Patch(bucket, bucket.Name) .Configure(req => req.ETagAction = Google.Apis.ETagAction.IfMatch) .Execute(); // Check that labels have been added, and etag is different Assert.NotEmpty(bucket2.Labels.EmptyIfNull()); Assert.NotEqual(bucket.ETag, bucket2.ETag); // Set a label with If-Match header, with an out of date etag. This should not change any labels var req3 = client.Buckets.Patch(bucket, bucket.Name) .Configure(req => req.ETagAction = Google.Apis.ETagAction.IfMatch); var ex3 = Assert.Throws(() => req3.Execute()); Assert.Contains("[Precondition Failed", ex3.Message); } finally { DeleteTestBuckets(client); } } [Fact] public void UriEscaping() { StorageService client = new StorageService(new BaseClientService.Initializer { HttpClientInitializer = Helper.GetServiceCredential().CreateScoped(StorageService.Scope.DevstorageFullControl), ApplicationName = "IntegrationTest", }); var bucket = CreateTestBucket(client); try { const string name = "foo [baz] "; var obj = new Google.Apis.Storage.v1.Data.Object { Name = name }; var data = "Hello test!"; // Upload to GCS using a name with special characters var stUpload = new MemoryStream(Encoding.ASCII.GetBytes(data)); var progress = client.Objects.Insert(obj, bucket.Name, stUpload, "text/plain").Upload(); Assert.Equal(UploadStatus.Completed, progress.Status); Assert.Equal(stUpload.Length, progress.BytesSent); // Download the same object and verify var stDownload = new MemoryStream(); client.Objects.Get(bucket.Name, name).Download(stDownload); Assert.Equal(stUpload.ToArray(), stDownload.ToArray()); } finally { DeleteTestBuckets(client); } } } }