ShareX/ShareX.UploadersLib/FileUploaders/AzureStorage.cs

212 lines
8.7 KiB
C#
Raw Normal View History

#region License Information (GPL v3)
/*
ShareX - A program that allows you to take screenshots and share any file type
Copyright (c) 2007-2017 ShareX Team
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Optionally you can also view the license at <http://www.gnu.org/licenses/>.
*/
#endregion License Information (GPL v3)
using ShareX.UploadersLib.Properties;
using ShareX.HelpersLib;
2017-01-27 00:41:49 +13:00
using System;
using System.Collections.Specialized;
2017-01-27 00:41:49 +13:00
using System.Drawing;
using System.Globalization;
2017-01-27 00:41:49 +13:00
using System.IO;
using System.Security.Cryptography;
2017-01-27 00:41:49 +13:00
using System.Text;
using System.Windows.Forms;
namespace ShareX.UploadersLib.FileUploaders
{
2017-03-11 00:28:23 +13:00
public class AzureStorageUploaderService : FileUploaderService
{
public override FileDestination EnumValue { get; } = FileDestination.AzureStorage;
public override Image ServiceImage => Resources.AzureStorage;
public override bool CheckConfig(UploadersConfig config)
{
return !string.IsNullOrEmpty(config.AzureStorageAccountName) &&
!string.IsNullOrEmpty(config.AzureStorageAccountAccessKey) &&
!string.IsNullOrEmpty(config.AzureStorageContainer);
}
public override GenericUploader CreateUploader(UploadersConfig config, TaskReferenceHelper taskInfo)
{
return new AzureStorage(config.AzureStorageAccountName, config.AzureStorageAccountAccessKey, config.AzureStorageContainer);
}
public override TabPage GetUploadersConfigTabPage(UploadersConfigForm form) => form.tpAzureStorage;
}
2017-01-27 00:41:49 +13:00
public sealed class AzureStorage : FileUploader
{
private const string apiVersion = "2016-05-31";
2017-03-11 00:28:23 +13:00
public string AzureStorageAccountName { get; private set; }
public string AzureStorageAccountAccessKey { get; private set; }
public string AzureStorageContainer { get; private set; }
public AzureStorage(string azureStorageAccountName, string azureStorageAccessKey, string azureStorageContainer)
2017-01-27 00:41:49 +13:00
{
2017-03-11 00:28:23 +13:00
AzureStorageAccountName = azureStorageAccountName;
AzureStorageAccountAccessKey = azureStorageAccessKey;
AzureStorageContainer = azureStorageContainer;
2017-01-27 00:41:49 +13:00
}
public override UploadResult Upload(Stream stream, string fileName)
{
2017-03-11 00:28:23 +13:00
if (string.IsNullOrEmpty(AzureStorageAccountName)) Errors.Add("'Account Name' must not be empty");
if (string.IsNullOrEmpty(AzureStorageAccountAccessKey)) Errors.Add("'Access key' must not be empty");
if (string.IsNullOrEmpty(AzureStorageContainer)) Errors.Add("'Container' must not be empty");
2017-01-27 00:41:49 +13:00
if (IsError)
{
return null;
}
CreateContainerIfNotExists();
2017-03-11 00:28:23 +13:00
string date = DateTime.UtcNow.ToString("R", CultureInfo.InvariantCulture);
string url = $"https://{AzureStorageAccountName}.blob.core.windows.net/{AzureStorageContainer}/{fileName}";
string contentType = Helpers.GetMimeType(fileName);
2017-03-11 00:28:23 +13:00
NameValueCollection requestHeaders = new NameValueCollection();
requestHeaders["x-ms-date"] = date;
requestHeaders["x-ms-version"] = apiVersion;
2017-03-11 00:28:23 +13:00
requestHeaders["x-ms-blob-type"] = "BlockBlob";
2017-03-11 00:28:23 +13:00
string canonicalizedHeaders = $"x-ms-blob-type:BlockBlob\nx-ms-date:{date}\nx-ms-version:{apiVersion}\n";
string canonicalizedResource = $"/{AzureStorageAccountName}/{AzureStorageContainer}/{fileName}";
string stringToSign = GenerateStringToSign(canonicalizedHeaders, canonicalizedResource, stream.Length.ToString(), contentType);
2017-03-11 00:28:23 +13:00
requestHeaders["Authorization"] = $"SharedKey {AzureStorageAccountName}:{stringToSign}";
2017-03-11 00:28:23 +13:00
NameValueCollection responseHeaders = SendRequestGetHeaders(HttpMethod.PUT, url, stream, contentType, null, requestHeaders, null);
if (responseHeaders != null)
{
2017-03-11 00:28:23 +13:00
return new UploadResult { IsSuccess = true, URL = url };
}
else
{
Errors.Add("Upload failed.");
return null;
}
}
private void CreateContainerIfNotExists()
{
2017-03-11 00:28:23 +13:00
string date = DateTime.UtcNow.ToString("R", CultureInfo.InvariantCulture);
string url = $"https://{AzureStorageAccountName}.blob.core.windows.net/{AzureStorageContainer}?restype=container";
2017-03-11 00:28:23 +13:00
NameValueCollection requestHeaders = new NameValueCollection();
requestHeaders["Content-Length"] = "0";
requestHeaders["x-ms-date"] = date;
requestHeaders["x-ms-version"] = apiVersion;
2017-03-11 00:28:23 +13:00
string canonicalizedHeaders = $"x-ms-date:{date}\nx-ms-version:{apiVersion}\n";
string canonicalizedResource = $"/{AzureStorageAccountName}/{AzureStorageContainer}\nrestype:container";
string stringToSign = GenerateStringToSign(canonicalizedHeaders, canonicalizedResource);
2017-03-11 00:28:23 +13:00
requestHeaders["Authorization"] = $"SharedKey {AzureStorageAccountName}:{stringToSign}";
2017-03-11 00:28:23 +13:00
NameValueCollection responseHeaders = SendRequestGetHeaders(HttpMethod.PUT, url, null, null, null, requestHeaders, null);
if (responseHeaders != null)
{
SetContainerACL();
}
else
{
2017-03-11 00:28:23 +13:00
if (Errors.Count > 0)
{
if (Errors[0].Contains("409"))
2017-03-11 00:28:23 +13:00
{
SetContainerACL();
2017-03-11 00:28:23 +13:00
}
else
{
Errors.Add("Upload to Azure storage failed.");
}
}
}
}
private void SetContainerACL()
{
2017-03-11 00:28:23 +13:00
string date = DateTime.UtcNow.ToString("R", CultureInfo.InvariantCulture);
string url = "https://{azureStorageAccountName}.blob.core.windows.net/{azureStorageContainer}?restype=container&comp=acl";
NameValueCollection requestHeaders = new NameValueCollection();
requestHeaders["Content-Length"] = "0";
requestHeaders["x-ms-date"] = date;
requestHeaders["x-ms-version"] = apiVersion;
requestHeaders["x-ms-blob-public-access"] = "container";
2017-03-11 00:28:23 +13:00
string canonicalizedHeaders = $"x-ms-blob-public-access:container\nx-ms-date:{date}\nx-ms-version:{apiVersion}\n";
string canonicalizedResource = $"/{AzureStorageAccountName}/{AzureStorageContainer}\ncomp:acl\nrestype:container";
string stringToSign = GenerateStringToSign(canonicalizedHeaders, canonicalizedResource);
2017-03-11 00:28:23 +13:00
requestHeaders["Authorization"] = $"SharedKey {AzureStorageAccountName}:{stringToSign}";
2017-03-11 00:28:23 +13:00
NameValueCollection responseHeaders = SendRequestGetHeaders(HttpMethod.PUT, url, null, null, null, requestHeaders, null);
if (responseHeaders == null)
2017-01-27 23:32:04 +13:00
{
2017-03-11 00:28:23 +13:00
Errors.Add("There was an issue with setting ACL on the container.");
2017-01-27 23:32:04 +13:00
}
}
private string GenerateStringToSign(string canonicalizedHeaders, string canonicalizedResource, string contentLength = "", string contentType = "")
{
2017-03-11 00:28:23 +13:00
string stringToSign = "PUT" + "\n" +
"\n" +
"\n" +
2017-03-11 00:28:23 +13:00
(contentLength ?? "") + "\n" +
"\n" +
2017-03-11 00:28:23 +13:00
(contentType ?? "") + "\n" +
"\n" +
"\n" +
"\n" +
"\n" +
"\n" +
"\n" +
canonicalizedHeaders +
canonicalizedResource;
2017-03-11 00:28:23 +13:00
return HashRequest(stringToSign);
2017-01-27 00:41:49 +13:00
}
2017-03-11 00:28:23 +13:00
private string HashRequest(string stringToSign)
2017-01-27 00:41:49 +13:00
{
2017-03-11 00:28:23 +13:00
string hashedString;
2017-01-27 00:41:49 +13:00
2017-03-11 00:28:23 +13:00
using (HashAlgorithm hashAlgorithm = new HMACSHA256(Convert.FromBase64String(AzureStorageAccountAccessKey)))
{
byte[] messageBuffer = Encoding.UTF8.GetBytes(stringToSign);
hashedString = Convert.ToBase64String(hashAlgorithm.ComputeHash(messageBuffer));
}
2017-01-27 00:41:49 +13:00
2017-03-11 00:28:23 +13:00
return hashedString;
}
2017-01-27 00:41:49 +13:00
}
}