2021-09-26 14:00:51 +10:00
|
|
|
using System;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Azure.Storage;
|
|
|
|
using Azure.Storage.Blobs;
|
2021-09-26 14:07:50 +10:00
|
|
|
using Azure.Storage.Blobs.Models;
|
2021-09-26 14:00:51 +10:00
|
|
|
using Azure.Storage.Sas;
|
|
|
|
using CommandLine;
|
|
|
|
using Microsoft.Azure.Storage;
|
2021-09-30 19:52:25 +10:00
|
|
|
using Shamir.Abstractions;
|
2021-09-26 14:00:51 +10:00
|
|
|
|
2021-09-30 19:52:25 +10:00
|
|
|
namespace Shamir.Commands.Azure
|
2021-09-26 14:00:51 +10:00
|
|
|
{
|
2021-09-30 19:52:25 +10:00
|
|
|
public sealed class StorageGetUrlOptions
|
2021-09-26 14:00:51 +10:00
|
|
|
{
|
|
|
|
[Option("connection-string", Required = false, HelpText = "Azure Storage connection string for the Storage Account backing the CDN.")]
|
|
|
|
public string? ConnectionString { get; set; }
|
|
|
|
|
|
|
|
[Option('h', "host", Required = false, HelpText = "Hostname to generate the SAS token for.")]
|
|
|
|
public string? HostName { get; set; }
|
|
|
|
|
|
|
|
[Option('d', "days", Default = 7, HelpText = "Number of days that the SAS token should remain valid for.")]
|
|
|
|
public int ValidityPeriodDays { get; set; }
|
|
|
|
|
|
|
|
[Value(0, MetaName = "path", Required = true, HelpText = "Path to enumerate, starting with the Azure Storage container name.")]
|
|
|
|
public string? Path { get; set; }
|
|
|
|
}
|
|
|
|
|
2021-09-26 14:07:50 +10:00
|
|
|
public sealed class StorageGetUrlCommand : ParsedArgumentsCommand<StorageGetUrlOptions>
|
2021-09-26 14:00:51 +10:00
|
|
|
{
|
2021-09-26 14:07:50 +10:00
|
|
|
public override string Name => "get-url";
|
2021-09-26 14:00:51 +10:00
|
|
|
|
2021-09-26 14:07:50 +10:00
|
|
|
public override string Description => "Get the URL for a file in Storage, with a SAS token if required.";
|
2021-09-26 14:00:51 +10:00
|
|
|
|
2021-09-26 14:07:50 +10:00
|
|
|
public override async ValueTask<int> ExecuteAsync(IServiceProvider serviceProvider, StorageGetUrlOptions options)
|
2021-09-26 14:00:51 +10:00
|
|
|
{
|
|
|
|
var connectionString = options.ConnectionString ?? Environment.GetEnvironmentVariable("AZURE_CONNECTION_STRING");
|
|
|
|
var account = CloudStorageAccount.Parse(connectionString);
|
|
|
|
|
|
|
|
Debug.Assert(options.Path != null, "Path should be set.");
|
|
|
|
|
|
|
|
var delimiterIndex = options.Path.IndexOf('/');
|
|
|
|
var (containerName, path) = delimiterIndex > 0
|
|
|
|
? (options.Path[..delimiterIndex], options.Path[(delimiterIndex + 1)..])
|
|
|
|
: (options.Path, string.Empty);
|
|
|
|
|
|
|
|
var uri = new UriBuilder
|
|
|
|
{
|
|
|
|
Scheme = "https",
|
|
|
|
Host = options.HostName ?? account.BlobStorageUri.PrimaryUri.Host,
|
|
|
|
Path = options.Path,
|
|
|
|
};
|
|
|
|
|
2021-09-26 14:07:50 +10:00
|
|
|
var client = new BlobServiceClient(connectionString);
|
|
|
|
var containerPolicy = await client.GetBlobContainerClient(containerName).GetAccessPolicyAsync();
|
|
|
|
if (containerPolicy.Value.BlobPublicAccess == PublicAccessType.None)
|
|
|
|
{
|
|
|
|
var builder = new BlobSasBuilder(BlobSasPermissions.Read, DateTimeOffset.UtcNow.AddDays(options.ValidityPeriodDays));
|
|
|
|
builder.BlobContainerName = containerName;
|
|
|
|
builder.BlobName = path;
|
|
|
|
builder.Protocol = SasProtocol.Https;
|
|
|
|
builder.Resource = "b";
|
|
|
|
|
|
|
|
var key = new StorageSharedKeyCredential(account.Credentials.AccountName, account.Credentials.ExportBase64EncodedKey());
|
|
|
|
var parameters = builder.ToSasQueryParameters(key);
|
|
|
|
|
|
|
|
uri.Query = parameters.ToString();
|
|
|
|
}
|
|
|
|
|
2021-09-30 19:52:25 +10:00
|
|
|
Console.WriteLine(uri.Uri.AbsoluteUri);
|
2021-09-26 14:00:51 +10:00
|
|
|
|
2021-09-26 14:07:50 +10:00
|
|
|
return 0;
|
2021-09-26 14:00:51 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|