Skip to content

Commit

Permalink
feat: implement ionos s3 storage
Browse files Browse the repository at this point in the history
  • Loading branch information
NikolaVetnic committed Jun 21, 2024
1 parent 13cae23 commit 8a3330a
Show file tree
Hide file tree
Showing 8 changed files with 372 additions and 10 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,11 @@

<ItemGroup>
<PackageReference Include="Autofac" Version="8.0.0" />
<PackageReference Include="AWSSDK.S3" Version="3.7.309.4" />
<PackageReference Include="Azure.Messaging.ServiceBus" Version="7.17.5" />
<PackageReference Include="Azure.Storage.Blobs" Version="12.19.1" />
<PackageReference Include="Google.Cloud.PubSub.V1" Version="3.12.0"/>
<PackageReference Include="Google.Cloud.Storage.V1" Version="4.10.0"/>
<PackageReference Include="Google.Cloud.PubSub.V1" Version="3.12.0" />
<PackageReference Include="Google.Cloud.Storage.V1" Version="4.10.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="8.0.4" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Abstractions" Version="8.0.4" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Relational" Version="8.0.4" />
Expand All @@ -15,7 +16,7 @@
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="8.0.0" />
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="8.0.2" />
<PackageReference Include="Polly" Version="8.3.1"/>
<PackageReference Include="Polly" Version="8.3.1" />
<PackageReference Include="RabbitMQ.Client" Version="6.8.1" />
<PackageReference Include="System.Interactive.Async" Version="6.0.1" />
</ItemGroup>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
using System.ComponentModel.DataAnnotations;
using Backbone.BuildingBlocks.Application.Abstractions.Infrastructure.Persistence.BlobStorage;
using Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.AzureStorageAccount;
using Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.GoogleCloudStorage;
using Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.Ionos;
using Backbone.Tooling.Extensions;
using Microsoft.Extensions.DependencyInjection;

Expand All @@ -9,6 +12,7 @@ public static class BlobStorageServiceCollectionExtensions
{
public const string AZURE_CLOUD_PROVIDER = "Azure";
public const string GOOGLE_CLOUD_PROVIDER = "GoogleCloud";
public const string IONOS_CLOUD_PROVIDER = "Ionos";

public static void AddBlobStorage(this IServiceCollection services, Action<BlobStorageOptions> setupOptions)
{
Expand All @@ -31,6 +35,19 @@ public static void AddBlobStorage(this IServiceCollection services, BlobStorageO
googleCloudStorageOptions.GcpAuthJson = options.ConnectionInfo;
googleCloudStorageOptions.BucketName = options.Container;

Check warning on line 36 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/BlobStorageServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Possible null reference assignment.

Check warning on line 36 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/BlobStorageServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Possible null reference assignment.
});
else if (options.CloudProvider == IONOS_CLOUD_PROVIDER)
{
services.Configure<IonosS3Options>(opt =>
{
opt.ServiceUrl = options.IonosS3Config.ServiceUrl;

Check warning on line 42 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/BlobStorageServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Dereference of a possibly null reference.

Check warning on line 42 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/BlobStorageServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Dereference of a possibly null reference.
opt.AccessKey = options.IonosS3Config.AccessKey;
opt.SecretKey = options.IonosS3Config.SecretKey;
opt.BucketName = options.IonosS3Config.BucketName;
});

services.AddSingleton<IonosS3ClientFactory>();
services.AddScoped<IBlobStorage, IonosS3BlobStorage>();
}
else if (options.CloudProvider.IsNullOrEmpty())
throw new NotSupportedException("No cloud provider was specified.");
else
Expand All @@ -41,9 +58,29 @@ public static void AddBlobStorage(this IServiceCollection services, BlobStorageO

public class BlobStorageOptions
{
public string CloudProvider { get; set; } = null!;
[Required]
[MinLength(1)]
[RegularExpression("Azure|GoogleCloud|Ionos")]
public string CloudProvider { get; set; } = string.Empty;

public string? ConnectionInfo { get; set; } = string.Empty;
public string? Container { get; set; } = string.Empty;

public IonosS3Config? IonosS3Config { get; set; } = new IonosS3Config();
}

public class IonosS3Config
{
[Required]
public string ServiceUrl { get; set; } = string.Empty;

public string Container { get; set; } = null!;
[Required]
public string AccessKey { get; set; } = string.Empty;

public string? ConnectionInfo { get; set; } = null;
[Required]
public string SecretKey { get; set; } = string.Empty;

[Required]
public string BucketName { get; set; } = string.Empty;
}

Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
using Amazon.S3;
using Amazon.S3.Model;
using Amazon.S3.Transfer;
using Microsoft.Extensions.Logging;
using System.Net;
using Backbone.BuildingBlocks.Application.Abstractions.Exceptions;
using Backbone.BuildingBlocks.Application.Abstractions.Infrastructure.Persistence.BlobStorage;
using Microsoft.Extensions.Options;

namespace Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.Ionos;
public class IonosS3BlobStorage : IBlobStorage, IDisposable
{
private readonly IAmazonS3 _s3Client;
private readonly List<ChangedBlob> _changedBlobs;
private readonly IList<RemovedBlob> _removedBlobs;
private readonly string _bucketName;
private readonly IonosS3Options _config;

Check warning on line 17 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/Ionos/IonosS3BlobStorage.cs

View workflow job for this annotation

GitHub Actions / Run Tests

The field 'IonosS3BlobStorage._config' is never used

Check warning on line 17 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/Ionos/IonosS3BlobStorage.cs

View workflow job for this annotation

GitHub Actions / Run Tests

The field 'IonosS3BlobStorage._config' is never used
private readonly ILogger<IonosS3BlobStorage> _logger;

public IonosS3BlobStorage(IOptions<IonosS3Options> config, ILogger<IonosS3BlobStorage> logger)

Check warning on line 20 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/Ionos/IonosS3BlobStorage.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Non-nullable field '_config' must contain a non-null value when exiting constructor. Consider declaring the field as nullable.

Check warning on line 20 in BuildingBlocks/src/BuildingBlocks.Infrastructure/Persistence/BlobStorage/Ionos/IonosS3BlobStorage.cs

View workflow job for this annotation

GitHub Actions / Run Tests

Non-nullable field '_config' must contain a non-null value when exiting constructor. Consider declaring the field as nullable.
{
var s3Config = new AmazonS3Config
{
ServiceURL = config.Value.ServiceUrl,
ForcePathStyle = true
};

_s3Client = new AmazonS3Client(config.Value.AccessKey, config.Value.SecretKey, s3Config);
_changedBlobs = new List<ChangedBlob>();
_removedBlobs = new List<RemovedBlob>();
_bucketName = config.Value.BucketName;
_logger = logger;
}

public void Add(string folder, string id, byte[] content)
{
_changedBlobs.Add(new ChangedBlob(folder, id, content));
}

public void Remove(string folder, string id)
{
_removedBlobs.Add(new RemovedBlob(folder, id));
}

public void Dispose()
{
_changedBlobs.Clear();
_removedBlobs.Clear();
}

public async Task<byte[]> FindAsync(string folder, string id)
{
_logger.LogTrace("Reading blob with key '{blobId}'...", id);

try
{
var request = new GetObjectRequest
{
BucketName = _bucketName,
Key = $"{folder}/{id}"
};

using var response = await _s3Client.GetObjectAsync(request);
using var memoryStream = new MemoryStream();
await response.ResponseStream.CopyToAsync(memoryStream);

_logger.LogTrace("Found blob with key '{blobId}'.", id);
return memoryStream.ToArray();
}
catch (AmazonS3Exception ex) when (ex.StatusCode == HttpStatusCode.NotFound)
{
_logger.LogError("A blob with key '{blobId}' was not found.", id);
throw new NotFoundException("Blob", ex);
}
catch (Exception ex)
{
_logger.LogError(ex, "Error downloading blob with key '{blobId}'.", id);
throw;
}
}

public Task<IAsyncEnumerable<string>> FindAllAsync(string folder, string? prefix = null)
{
return Task.FromResult(FindAllBlobsAsync(folder, prefix));
}

private async IAsyncEnumerable<string> FindAllBlobsAsync(string folder, string? prefix)
{
_logger.LogTrace("Listing all blobs...");

var request = new ListObjectsV2Request
{
BucketName = _bucketName,
Prefix = prefix != null ? $"{folder}/{prefix}" : folder
};

ListObjectsV2Response response;
do
{
response = await _s3Client.ListObjectsV2Async(request);

foreach (var obj in response.S3Objects)
{
yield return obj.Key;
}

request.ContinuationToken = response.NextContinuationToken;
} while (response.IsTruncated);

_logger.LogTrace("Found all blobs.");
}

public async Task SaveAsync()
{
await UploadChangedBlobs();
await DeleteRemovedBlobs();
}

private async Task UploadChangedBlobs()
{
_logger.LogTrace("Uploading '{changedBlobsCount}' changed blobs...", _changedBlobs.Count);

var changedBlobs = new List<ChangedBlob>(_changedBlobs);

foreach (var blob in changedBlobs)
{
await EnsureKeyDoesNotExist(blob.Folder, blob.Name);

using var memoryStream = new MemoryStream(blob.Content);

try
{
_logger.LogTrace("Uploading blob with key '{blobName}'...", blob.Name);

var request = new TransferUtilityUploadRequest
{
InputStream = memoryStream,
Key = $"{blob.Folder}/{blob.Name}",
BucketName = _bucketName
};

var transferUtility = new TransferUtility(_s3Client);
await transferUtility.UploadAsync(request);

_logger.LogTrace("Upload of blob with key '{blobName}' was successful.", blob.Name);
}
catch (Exception ex)
{
_logger.LogError(ex, "Error uploading blob with key '{blobName}'.", blob.Name);
throw;
}
finally
{
_changedBlobs.Remove(blob);
}
}
}

private async Task EnsureKeyDoesNotExist(string folder, string key)
{
try
{
var request = new GetObjectRequest
{
BucketName = _bucketName,
Key = $"{folder}/{key}"
};

await _s3Client.GetObjectAsync(request);

_logger.LogError("A blob with key '{blobName}' already exists.", key);
throw new BlobAlreadyExistsException(key);
}
catch (AmazonS3Exception ex) when (ex.StatusCode == HttpStatusCode.NotFound)
{
return;
}
}

private async Task DeleteRemovedBlobs()
{
_logger.LogTrace("Deleting '{removedBlobsCount}' blobs...", _removedBlobs.Count);

var blobsToDelete = new List<RemovedBlob>(_removedBlobs);

foreach (var blob in blobsToDelete)
{
try
{
var request = new DeleteObjectRequest
{
BucketName = _bucketName,
Key = $"{blob.Folder}/{blob.Name}"
};

await _s3Client.DeleteObjectAsync(request);

_removedBlobs.Remove(blob);
}
catch (AmazonS3Exception ex) when (ex.StatusCode == HttpStatusCode.NotFound)
{
_logger.LogError("A blob with key '{blobId}' was not found.", blob.Name);
throw new NotFoundException($"Blob with key '{blob.Name}' was not found.", ex);
}
catch (Exception ex)
{
_logger.LogError(ex, "Error deleting blob with key '{blobName}'.", blob.Name);
throw;
}
}

_logger.LogTrace("Deletion successful.");
}

private record ChangedBlob(string Folder, string Name, byte[] Content);

private record RemovedBlob(string Folder, string Name);
}

public class IonosS3Config
{
public required string ServiceUrl { get; set; }
public required string AccessKey { get; set; }
public required string SecretKey { get; set; }
public required string BucketName { get; set; }
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
using Amazon.S3;
using Microsoft.Extensions.Options;

namespace Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.Ionos;
public class IonosS3ClientFactory
{
private readonly IonosS3Options _options;

public IonosS3ClientFactory(IOptions<IonosS3Options> options)
{
_options = options.Value;
}

public IAmazonS3 CreateClient()
{
var config = new AmazonS3Config
{
ServiceURL = _options.ServiceUrl,
ForcePathStyle = true
};

return new AmazonS3Client(_options.AccessKey, _options.SecretKey, config);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
using Backbone.BuildingBlocks.Application.Abstractions.Infrastructure.Persistence.BlobStorage;
using Microsoft.Extensions.DependencyInjection;

namespace Backbone.BuildingBlocks.Infrastructure.Persistence.BlobStorage.Ionos;
public static class IonosS3ServiceCollectionExtensions
{
public static void AddIonosS3(this IServiceCollection services,
Action<IonosS3Options> setupOptions)
{
var options = new IonosS3Options();
setupOptions.Invoke(options);

services.AddIonosS3(options);
}

public static void AddIonosS3(this IServiceCollection services, IonosS3Options options)
{
services.Configure<IonosS3Options>(opt =>
{
opt.ServiceUrl = options.ServiceUrl;
opt.AccessKey = options.AccessKey;
opt.SecretKey = options.SecretKey;
opt.BucketName = options.BucketName;
});

services.AddSingleton<IonosS3ClientFactory>();
services.AddScoped<IBlobStorage, IonosS3BlobStorage>();
}
}

public class IonosS3Options
{
public string ServiceUrl { get; set; } = string.Empty;
public string AccessKey { get; set; } = string.Empty;
public string SecretKey { get; set; } = string.Empty;
public string BucketName { get; set; } = string.Empty;
}
Loading

0 comments on commit 8a3330a

Please sign in to comment.