mirror of
https://github.com/ShareX/ShareX.git
synced 2024-07-01 04:30:49 +12:00
Fixed a bug where the upload policy was hard coded to 2015, implemented some suggestions and bugfixes from Jaex. Tidied code.
This commit is contained in:
parent
8744352d01
commit
bf6d6d2097
|
@ -39,49 +39,52 @@ namespace UploadersLib.FileUploaders
|
|||
{
|
||||
public sealed class AmazonS3 : FileUploader
|
||||
{
|
||||
const string FILE_INPUT_NAME = "file";
|
||||
|
||||
public AmazonS3Settings AccessKeys { get; set; }
|
||||
private AmazonS3Settings S3Settings { get; set; }
|
||||
|
||||
public AmazonS3(AmazonS3Settings accessKeys)
|
||||
{
|
||||
AccessKeys = accessKeys;
|
||||
S3Settings = accessKeys;
|
||||
}
|
||||
|
||||
private string GetObjectStorageClass()
|
||||
{
|
||||
return AccessKeys.UseReducedRedundancyStorage ? "REDUCED_REDUNDANCY" : "STANDARD";
|
||||
return S3Settings.UseReducedRedundancyStorage ? "REDUCED_REDUNDANCY" : "STANDARD";
|
||||
}
|
||||
|
||||
private string GetObjectKey(string fileName)
|
||||
{
|
||||
var parser = new NameParser(NameParserType.FileName);
|
||||
return string.Format("{0}{1}", parser.Parse(AccessKeys.ObjectPrefix), fileName);
|
||||
return string.Format("{0}{1}", parser.Parse(S3Settings.ObjectPrefix), fileName);
|
||||
}
|
||||
|
||||
// Helper class to construct the S3 policy document (below)
|
||||
private class S3PolicyCondition : Dictionary<string, string>
|
||||
{
|
||||
public S3PolicyCondition(string key, string value)
|
||||
{
|
||||
Add(key, value);
|
||||
}
|
||||
}
|
||||
|
||||
private string GetPolicyDocument(string fileName)
|
||||
{
|
||||
var mimeType = Helpers.GetMimeType(fileName);
|
||||
var objectKey = GetObjectKey(fileName);
|
||||
var objectStorageClass = GetObjectStorageClass();
|
||||
var policyDocument = new {
|
||||
expiration = DateTime.UtcNow.AddDays(2).ToString("o"), // The policy is valid for 2 days
|
||||
conditions = new List<S3PolicyCondition> {
|
||||
new S3PolicyCondition("acl", "public-read"),
|
||||
new S3PolicyCondition("bucket", S3Settings.Bucket),
|
||||
new S3PolicyCondition("Content-Type", Helpers.GetMimeType(fileName)),
|
||||
new S3PolicyCondition("key", GetObjectKey(fileName)),
|
||||
new S3PolicyCondition("x-amz-storage-class", GetObjectStorageClass())
|
||||
}
|
||||
};
|
||||
|
||||
var policyDocument = string.Format(@"{{
|
||||
'expiration': '2015-12-01T12:00:00.000Z',
|
||||
'conditions': [
|
||||
{{'acl': 'public-read' }},
|
||||
{{'bucket': '{0}' }},
|
||||
{{'Content-Type': '{1}'}},
|
||||
{{'key': '{2}'}},
|
||||
{{'x-amz-storage-class': '{3}'}},
|
||||
]
|
||||
}}", AccessKeys.Bucket, mimeType, objectKey, objectStorageClass);
|
||||
|
||||
return Regex.Replace(policyDocument, @"\s", "");
|
||||
return JsonConvert.SerializeObject(policyDocument);
|
||||
}
|
||||
|
||||
private string GetEndpoint()
|
||||
{
|
||||
return string.Format("{0}{1}", AccessKeys.Endpoint, AccessKeys.Bucket);
|
||||
return string.Format("{0}{1}", S3Settings.Endpoint, S3Settings.Bucket);
|
||||
}
|
||||
|
||||
// http://codeonaboat.wordpress.com/2011/04/22/uploading-a-file-to-amazon-s3-using-an-asp-net-mvc-application-directly-from-the-users-browser/
|
||||
|
@ -90,9 +93,14 @@ private string CreateSignature(string secretKey, byte[] policyBytes)
|
|||
var encoding = new ASCIIEncoding();
|
||||
var base64Policy = Convert.ToBase64String(policyBytes);
|
||||
var secretKeyBytes = encoding.GetBytes(secretKey);
|
||||
var hmacsha1 = new HMACSHA1(secretKeyBytes);
|
||||
var base64PolicyBytes = encoding.GetBytes(base64Policy);
|
||||
var signatureBytes = hmacsha1.ComputeHash(base64PolicyBytes);
|
||||
|
||||
byte[] signatureBytes;
|
||||
using (var hmacsha1 = new HMACSHA1(secretKeyBytes))
|
||||
{
|
||||
var base64PolicyBytes = encoding.GetBytes(base64Policy);
|
||||
signatureBytes = hmacsha1.ComputeHash(base64PolicyBytes);
|
||||
}
|
||||
|
||||
return Convert.ToBase64String(signatureBytes);
|
||||
}
|
||||
|
||||
|
@ -100,13 +108,13 @@ private string CreateSignature(string secretKey, byte[] policyBytes)
|
|||
{
|
||||
var policyDocument = GetPolicyDocument(fileName);
|
||||
var policyBytes = Encoding.ASCII.GetBytes(policyDocument);
|
||||
var signature = CreateSignature(AccessKeys.SecretAccessKey, policyBytes);
|
||||
var signature = CreateSignature(S3Settings.SecretAccessKey, policyBytes);
|
||||
|
||||
var parameters = new Dictionary<string, string>();
|
||||
parameters.Add("key", objectKey);
|
||||
parameters.Add("acl", "public-read");
|
||||
parameters.Add("content-type", Helpers.GetMimeType(fileName));
|
||||
parameters.Add("AWSAccessKeyId", AccessKeys.AccessKeyID);
|
||||
parameters.Add("AWSAccessKeyId", S3Settings.AccessKeyID);
|
||||
parameters.Add("policy", Convert.ToBase64String(policyBytes));
|
||||
parameters.Add("signature", signature);
|
||||
parameters.Add("x-amz-storage-class", GetObjectStorageClass());
|
||||
|
@ -115,20 +123,20 @@ private string CreateSignature(string secretKey, byte[] policyBytes)
|
|||
|
||||
public override UploadResult Upload(Stream stream, string fileName)
|
||||
{
|
||||
if (string.IsNullOrEmpty(AccessKeys.AccessKeyID)) throw new Exception("'Access Key' must not be empty.");
|
||||
if (string.IsNullOrEmpty(AccessKeys.SecretAccessKey)) throw new Exception("'Secret Access Key' must not be empty.");
|
||||
if (string.IsNullOrEmpty(AccessKeys.Endpoint)) throw new Exception("'Endpoint' must not be emoty.");
|
||||
if (string.IsNullOrEmpty(AccessKeys.Bucket)) throw new Exception("'Bucket' must not be empty.");
|
||||
if (string.IsNullOrEmpty(S3Settings.AccessKeyID)) throw new Exception("'Access Key' must not be empty.");
|
||||
if (string.IsNullOrEmpty(S3Settings.SecretAccessKey)) throw new Exception("'Secret Access Key' must not be empty.");
|
||||
if (string.IsNullOrEmpty(S3Settings.Endpoint)) throw new Exception("'Endpoint' must not be emoty.");
|
||||
if (string.IsNullOrEmpty(S3Settings.Bucket)) throw new Exception("'Bucket' must not be empty.");
|
||||
|
||||
var objectKey = GetObjectKey(fileName);
|
||||
|
||||
var uploadResult = UploadData(stream, GetEndpoint(), fileName, FILE_INPUT_NAME, GetParameters(fileName, objectKey), null, null, ResponseType.Headers);
|
||||
var uploadResult = UploadData(stream, GetEndpoint(), fileName, arguments: GetParameters(fileName, objectKey), responseType: ResponseType.Headers);
|
||||
|
||||
if (uploadResult.IsSuccess)
|
||||
{
|
||||
if (AccessKeys.UseCustomCNAME)
|
||||
if (S3Settings.UseCustomCNAME)
|
||||
{
|
||||
uploadResult.URL = string.Format("http://{0}/{1}", AccessKeys.Bucket, objectKey);
|
||||
uploadResult.URL = string.Format("http://{0}/{1}", S3Settings.Bucket, objectKey);
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -150,10 +158,4 @@ public class AmazonS3Settings
|
|||
public string Bucket { get; set; }
|
||||
public string Endpoint { get; set; }
|
||||
}
|
||||
|
||||
public class AmazonS3Region
|
||||
{
|
||||
public string Name { get; set; }
|
||||
public string Endpoint { get; set; }
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue