如何在Go中进行预签名的POST上传到AWS S3?

huangapple go评论97阅读模式
英文:

How to do a Pre-signed POST upload to AWS S3 in Go?

问题

我将为您翻译以下内容:

我想要使用Go语言进行预签名 POST 请求以将文件上传到 AWS S3 存储桶,请问如何实现?

请注意,这与使用 PUT 进行预签名上传不同。

英文:

I would like to do a pre-signed POST to upload files to an AWS S3 bucket - how would this be done in Go?

Please note that this is not the same as Pre-signed upload with PUT.

答案1

得分: 6

为了帮助他人,我将回答这个问题并提供一些代码,以帮助其他可能遇到同样问题的人。

在这里可以找到一个示例的 Google App Engine 网页应用程序,用于渲染预签名的 POST 表单:https://github.com/murrekatt/go-aws-s3-presigned-post-app-engine

还有一个我创建的用于在 Go 中执行预签名 POST 的小型库:https://github.com/murrekatt/go-s3presigned-post

简而言之,要将预签名的 POST 请求发送到一个公开读取的 Amazon S3 存储桶,你需要执行以下步骤:

1. 配置 S3 存储桶只允许公开下载。

以下是一个允许仅公开读取的示例存储桶策略:

  1. {
  2. "Version": "2012-10-17",
  3. "Id": "akjsdhakshfjlashdf",
  4. "Statement": [
  5. {
  6. "Sid": "kjahsdkajhsdkjasda",
  7. "Effect": "Allow",
  8. "Principal": {
  9. "AWS": "*"
  10. },
  11. "Action": "s3:GetObject",
  12. "Resource": "arn:aws:s3:::BUCKETNAMEHERE/*"
  13. }
  14. ]
  15. }

2. 创建一个允许上传的 HTTP POST 策略。

参考 AWS S3 文档:http://docs.aws.amazon.com/AmazonS3/latest/API/sigv4-HTTPPOSTConstructPolicy.html

以下是一个带有过期时间、上传到特定存储桶并允许公开读取访问的 POST 策略模板示例:

  1. {
  2. "expiration": "%s",
  3. "conditions": [
  4. {"bucket": "%s"},
  5. ["starts-with", "$key", "%s"],
  6. {"acl": "public-read"},
  7. {"x-amz-credential": "%s"},
  8. {"x-amz-algorithm": "AWS4-HMAC-SHA256"},
  9. {"x-amz-date": "%s"}
  10. ]
  11. }

3. 使用 S3 存储桶所有者的凭证生成并签署策略。

参考 AWS 文档:http://docs.aws.amazon.com/general/latest/gr/sigv4-calculate-signature.html

  • 填写正确的过期时间、存储桶、键、凭证和日期值。
  • 对策略进行 base64 编码。
  • 使用 HMAC-SHA256 对策略进行签名。
  • 对签名进行十六进制编码。

4. 构建并发送多部分表单数据的 POST 请求。

参考 AWS S3 文档:http://docs.aws.amazon.com/AmazonS3/latest/API/sigv4-post-example.html

现在,你可以按照上述链接中描述的方式生成一个 HTML 表单,并自动获取正确的多部分表单数据请求。

我想在 Go 中手动完成这个过程,以下是如何实现的:

无论哪种方式,你都需要提供在步骤 2 和 3 中创建的 POST 策略中指定的所有部分。除了必填字段(不在策略中)外,请求中不能有其他字段。

字段的顺序也是指定的,所有字段都是 HTTP POST 请求中的多部分字段。

  1. func Upload(url string, fields Fields) error {
  2. var b bytes.Buffer
  3. w := multipart.NewWriter(&b)
  4. for _, f := range fields {
  5. fw, err := w.CreateFormField(f.Key)
  6. if err != nil {
  7. return err
  8. }
  9. if _, err := fw.Write([]byte(f.Value)); err != nil {
  10. return err
  11. }
  12. }
  13. w.Close()
  14. req, err := http.NewRequest("POST", url, &b)
  15. if err != nil {
  16. return err
  17. }
  18. req.Header.Set("Content-Type", w.FormDataContentType())
  19. client := &http.Client{}
  20. res, err := client.Do(req)
  21. if err != nil {
  22. return err
  23. }
  24. if res.StatusCode != http.StatusOK {
  25. err = fmt.Errorf("bad status: %s", res.Status)
  26. }
  27. return nil
  28. }

以上是你要翻译的内容。

英文:

So in order to help others I will answer the question myself and provide some code to help others who might have the same problem.

Example web app for Google App Engine rendering a pre-signed POST form can be found here.

And a small library I created doing the pre-signed POST in Go.

In short, doing a presigned POST to a public-read Amazon S3 bucket you need to:

1. Configure the S3 bucket to only allow public download.

Example bucket policy that allow only public read.

  1. {
  2. "Version": "2012-10-17",
  3. "Id": "akjsdhakshfjlashdf",
  4. "Statement": [
  5. {
  6. "Sid": "kjahsdkajhsdkjasda",
  7. "Effect": "Allow",
  8. "Principal": {
  9. "AWS": "*"
  10. },
  11. "Action": "s3:GetObject",
  12. "Resource": "arn:aws:s3:::BUCKETNAMEHERE/*"
  13. }
  14. ]
  15. }

2. Create a policy for the HTTP POST that allows the upload.

AWS S3 docs

Example POST policy template with expiration to upload a specific key, into a specific bucket and allow public-read access.

  1. { "expiration": "%s",
  2. "conditions": [
  3. {"bucket": "%s"},
  4. ["starts-with", "$key", "%s"],
  5. {"acl": "public-read"},
  6. {"x-amz-credential": "%s"},
  7. {"x-amz-algorithm": "AWS4-HMAC-SHA256"},
  8. {"x-amz-date": "%s" }
  9. ]
  10. }

3. Generate and sign the policy using the S3 bucket owner's credentials.

AWS docs

  • Fill in the correct values for expiration, bucket, key, credentials and date.
  • base64 encode the policy.
  • HMAC-SHA256 the policy to get a signature.
  • hex encode the signature.

4. Construct and POST the multipart form data

AWS S3 docs

Now either you would generate an HTML form and automatically get the correct multipart form data request like described in the above link.

I wanted to do this by hand in Go so here's how to do that.

Either way you need to provide all the parts that are specified in the POST policy you created in steps 2 and 3. You can also not have additional fields in the request except for the mandatory ones (not in the policy).

The order of the fields is also specified and all of them are multipart fields in the HTTP POST request.

  1. func Upload(url string, fields Fields) error {
  2. var b bytes.Buffer
  3. w := multipart.NewWriter(&b)
  4. for _, f := range fields {
  5. fw, err := w.CreateFormField(f.Key)
  6. if err != nil {
  7. return err
  8. }
  9. if _, err := fw.Write([]byte(f.Value)); err != nil {
  10. return err
  11. }
  12. }
  13. w.Close()
  14. req, err := http.NewRequest("POST", url, &b)
  15. if err != nil {
  16. return err
  17. }
  18. req.Header.Set("Content-Type", w.FormDataContentType())
  19. client := &http.Client{}
  20. res, err := client.Do(req)
  21. if err != nil {
  22. return err
  23. }
  24. if res.StatusCode != http.StatusOK {
  25. err = fmt.Errorf("bad status: %s", res.Status)
  26. }
  27. return nil
  28. }

答案2

得分: 4

这是一个来自https://github.com/minio/minio-go的替代方法,你可能会喜欢它,因为它提供了一种完全以编程方式生成预签名 POST 策略的方法。

  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "time"
  6. "github.com/minio/minio-go"
  7. )
  8. func main() {
  9. policy := minio.NewPostPolicy()
  10. policy.SetKey("myobject")
  11. policy.SetBucket("mybucket")
  12. policy.SetExpires(time.Now().UTC().AddDate(0, 0, 10)) // expires in 10 days
  13. config := minio.Config{
  14. AccessKeyID: "YOUR-ACCESS-KEY-HERE",
  15. SecretAccessKey: "YOUR-PASSWORD-HERE",
  16. Endpoint: "https://s3.amazonaws.com",
  17. }
  18. s3Client, err := minio.New(config)
  19. if err != nil {
  20. log.Fatalln(err)
  21. }
  22. m, err := s3Client.PresignedPostPolicy(policy)
  23. if err != nil {
  24. fmt.Println(err)
  25. return
  26. }
  27. fmt.Printf("curl ")
  28. for k, v := range m {
  29. fmt.Printf("-F %s=%s ", k, v)
  30. }
  31. fmt.Printf("-F file=@/etc/bashrc ")
  32. fmt.Printf(config.Endpoint + "/mybucket\n")
  33. }

步骤1:

  1. policy := minio.NewPostPolicy()
  2. policy.SetKey("myobject")
  3. policy.SetBucket("mybucket")
  4. policy.SetExpires(time.Now().UTC().AddDate(0, 0, 10)) // expires in 10 days

实例化一个新的策略结构,该策略结构实现了以下方法:

  1. func NewPostPolicy() *PostPolicy
  2. func (p *PostPolicy) SetBucket(bucket string) error
  3. func (p *PostPolicy) SetContentLength(min, max int) error
  4. func (p *PostPolicy) SetContentType(contentType string) error
  5. func (p *PostPolicy) SetExpires(t time.Time) error
  6. func (p *PostPolicy) SetKey(key string) error
  7. func (p *PostPolicy) SetKeyStartsWith(keyStartsWith string) error
  8. func (p PostPolicy) String() string

步骤2:

  1. m, err := s3Client.PresignedPostPolicy(policy)
  2. if err != nil {
  3. fmt.Println(err)
  4. return
  5. }

现在,PresignedPostPolicy() 接受 PostPolicy 结构,并返回一个“键/值”映射,可以在 HTML 表单或 curl 命令中使用该映射来上传数据到 S3。

英文:

Here is an alternative approach from https://github.com/minio/minio-go
that you might like for a full programmatic way of generating presigned post policy.

  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "time"
  6. "github.com/minio/minio-go"
  7. )
  8. func main() {
  9. policy := minio.NewPostPolicy()
  10. policy.SetKey("myobject")
  11. policy.SetBucket("mybucket")
  12. policy.SetExpires(time.Now().UTC().AddDate(0, 0, 10)) // expires in 10 days
  13. config := minio.Config{
  14. AccessKeyID: "YOUR-ACCESS-KEY-HERE",
  15. SecretAccessKey: "YOUR-PASSWORD-HERE",
  16. Endpoint: "https://s3.amazonaws.com",
  17. }
  18. s3Client, err := minio.New(config)
  19. if err != nil {
  20. log.Fatalln(err)
  21. }
  22. m, err := s3Client.PresignedPostPolicy(policy)
  23. if err != nil {
  24. fmt.Println(err)
  25. return
  26. }
  27. fmt.Printf("curl ")
  28. for k, v := range m {
  29. fmt.Printf("-F %s=%s ", k, v)
  30. }
  31. fmt.Printf("-F file=@/etc/bashrc ")
  32. fmt.Printf(config.Endpoint + "/mybucket\n")
  33. }

Step 1:

  1. policy := minio.NewPostPolicy()
  2. policy.SetKey("myobject")
  3. policy.SetBucket("mybucket")
  4. policy.SetExpires(time.Now().UTC().AddDate(0, 0, 10)) // expires in 10 days

Instantiate a new policy structure, this policy structure implements following methods.

  1. func NewPostPolicy() *PostPolicy
  2. func (p *PostPolicy) SetBucket(bucket string) error
  3. func (p *PostPolicy) SetContentLength(min, max int) error
  4. func (p *PostPolicy) SetContentType(contentType string) error
  5. func (p *PostPolicy) SetExpires(t time.Time) error
  6. func (p *PostPolicy) SetKey(key string) error
  7. func (p *PostPolicy) SetKeyStartsWith(keyStartsWith string) error
  8. func (p PostPolicy) String() string

Step 2:

  1. m, err := s3Client.PresignedPostPolicy(policy)
  2. if err != nil {
  3. fmt.Println(err)
  4. return
  5. }

Now PresignedPostPolicy() takes the PostPolicy structure and returns back a map of "key/values" which can be used in your HTML form or curl command to upload data to s3.

答案3

得分: 2

一瞥之下,似乎POST方法需要使用附加的策略和签名,用于基于浏览器的上传。详细信息请参阅AWS文档

具体来说,您需要生成策略并对其进行签名,然后将它们包含在HTML表单中,以及POST请求中的其他必需信息。或者让浏览器为您完成这些步骤。

在HTML表单的POST上传中,您只需对策略字符串进行签名。要发布的最终URL可能会根据表单内容而变化:https://bucket.s3.amazonaws.com/<depends-on-form-content>。因此,您无法预先签名该URL,因为您不知道它是什么。

这与将文件PUT到已签名URL的情况不同。您可以签名该URL,因为您知道完整的URL:https://bucket.s3.amazonaws.com/known-key

您可以构建一个带有适当策略和参数的POST请求,并通过POST方式上传。但是,您需要知道表单的内容才能预先知道URL。在这种情况下,您可以直接使用预签名的PUT URL。

至少乍看之下是这样的...

英文:

At a glance it looks like POST works with an attached policy and signature -- designed for browser based uploads. See the AWS Docs for details.

Specifically, you need to generate a policy and sign that -- then include them in the HTML form, and thereby the POST request -- along with the rest of the required information. Or let the browser do it for you.

In the case of HTML form POST uploads you are only signing the policy string. The final URL to be posted to can vary based on the form contents: https://bucket.s3.amazonaws.com/<depends-on-form-content>. So you can't presign that URL because you don't know what it is.

This is different than a signed URL to which you PUT a file. You can sign that because you know the full URL: https://bucket.s3.amazonaws.com/known-key

You could build a POST request with the appropriate policy and parameters and upload via POST that way. However, you would need to know the contents of the form to know the URL beforehand. In which case you may as well use a presigned PUT URL.

At least that is how it appears at a glance...

答案4

得分: 1

在尝试使用@murrekatt提供的解决方案时,遇到了"InvalidAccessKeyId"错误。

后来我发现这个问题是因为我在lambda函数内部生成了预签名的POST请求,但没有在表单数据和策略中包含"x-amz-security-token"。

所以这是我根据@murrekatt和boto3库的帮助编写的代码:

  1. import (
  2. "crypto/hmac"
  3. "crypto/sha256"
  4. "encoding/base64"
  5. "encoding/hex"
  6. "encoding/json"
  7. "fmt"
  8. "time"
  9. "github.com/aws/aws-sdk-go-v2/aws"
  10. )
  11. type PresignedPOST struct {
  12. URL string `json:"url"`
  13. Key string `json:"key"`
  14. Policy string `json:"policy"`
  15. Credential string `json:"credential"`
  16. SecurityToken string `json:"securityToken,omitempty"`
  17. Signature string `json:"signature"`
  18. Date string `json:"date"`
  19. }
  20. func NewPresignedPost(input *NewPresignedPostInput) (*PresignedPOST, error) {
  21. // 过期时间
  22. expirationTime := time.Now().Add(time.Second * time.Duration(input.ExpiresIn)).UTC()
  23. dateString := expirationTime.Format("20060102")
  24. // 凭证字符串
  25. creds := fmt.Sprintf("%s/%s/%s/s3/aws4_request", input.Credentials.AccessKeyID, dateString, input.Region)
  26. // 策略
  27. policyDoc, err := createPolicyDocument(expirationTime, input.Bucket, input.Key, creds, &input.Credentials.SessionToken, input.Conditions)
  28. if err != nil {
  29. return nil, err
  30. }
  31. // 创建签名
  32. signature := createSignature(input.Credentials.SecretAccessKey, input.Region, dateString, policyDoc)
  33. // URL
  34. url := fmt.Sprintf("https://%s.s3.amazonaws.com/", input.Bucket)
  35. // 过期时间
  36. dateTimeString := expirationTime.Format("20060102T150405Z")
  37. // POST
  38. post := &PresignedPOST{
  39. Key: input.Key,
  40. Policy: policyDoc,
  41. Signature: signature,
  42. URL: url,
  43. Credential: creds,
  44. SecurityToken: input.Credentials.SessionToken,
  45. Date: dateTimeString,
  46. }
  47. return post, nil
  48. }
  49. type NewPresignedPostInput struct {
  50. // Key 名称
  51. Key string
  52. // 凭证
  53. Credentials aws.Credentials
  54. // 区域
  55. Region string
  56. // 预签名的 POST 请求的存储桶名称
  57. Bucket string
  58. // 过期时间 - 预签名的 POST 请求的有效时间(秒)
  59. ExpiresIn int64
  60. // 包含在策略中的条件列表。每个元素可以是列表或结构。
  61. // 例如:
  62. // [
  63. // {"acl": "public-read"}, ["content-length-range", 2, 5], ["starts-with", "$success_action_redirect", ""]
  64. // ]
  65. Conditions []interface{}
  66. }
  67. // 辅助函数
  68. func createPolicyDocument(expirationTime time.Time, bucket string, key string, credentialString string, securityToken *string, extraConditions []interface{}) (string, error) {
  69. doc := map[string]interface{}{}
  70. doc["expiration"] = expirationTime.Format("2006-01-02T15:04:05.000Z")
  71. // 条件
  72. conditions := []interface{}{}
  73. conditions = append(conditions, map[string]string{
  74. "bucket": bucket,
  75. })
  76. conditions = append(conditions, []string{
  77. "starts-with", "$key", key,
  78. })
  79. conditions = append(conditions, map[string]string{
  80. "x-amz-credential": credentialString,
  81. })
  82. if securityToken != nil {
  83. conditions = append(conditions, map[string]string{
  84. "x-amz-security-token": *securityToken,
  85. })
  86. }
  87. conditions = append(conditions, map[string]string{
  88. "x-amz-algorithm": "AWS4-HMAC-SHA256",
  89. })
  90. conditions = append(conditions, map[string]string{
  91. "x-amz-date": expirationTime.Format("20060102T150405Z"),
  92. })
  93. // 其他条件
  94. conditions = append(conditions, extraConditions...)
  95. doc["conditions"] = conditions
  96. // base64 编码的 JSON 字符串
  97. jsonBytes, err := json.Marshal(doc)
  98. if err != nil {
  99. return "", err
  100. }
  101. return base64.StdEncoding.EncodeToString(jsonBytes), nil
  102. }
  103. func createSignature(secretKey string, region string, dateString string, stringToSign string) string {
  104. // 辅助函数,用于生成 HMAC-SHA256
  105. makeHmac := func(key []byte, data []byte) []byte {
  106. hash := hmac.New(sha256.New, key)
  107. hash.Write(data)
  108. return hash.Sum(nil)
  109. }
  110. h1 := makeHmac([]byte("AWS4"+secretKey), []byte(dateString))
  111. h2 := makeHmac(h1, []byte(region))
  112. h3 := makeHmac(h2, []byte("s3"))
  113. h4 := makeHmac(h3, []byte("aws4_request"))
  114. signature := makeHmac(h4, []byte(stringToSign))
  115. return hex.EncodeToString(signature)
  116. }

用法

  1. // 凭证
  2. conf, _ := config.LoadDefaultConfig(c.Context)
  3. awsCreds, _ := conf.Credentials.Retrieve(c.Context)
  4. // 生成预签名的 POST 请求
  5. post, err := s3util.NewPresignedPost(&s3util.NewPresignedPostInput{
  6. Key: <file-name>,
  7. Credentials: awsCreds,
  8. Region: <region>,
  9. Bucket: <bucket-name>,
  10. ExpiresIn: <expiration>,
  11. Conditions: []interface{}{
  12. []interface{}{"content-length-range", 1, <size-limit>},
  13. },
  14. })

然后在前端,使用返回的 JSON 数据作为 POST 表单数据

  1. key: <key>
  2. X-Amz-Credential: <credential>
  3. X-Amz-Security-Token: <securityToken> // 如果提供了
  4. X-Amz-Algorithm: AWS4-HMAC-SHA256
  5. X-Amz-Date: <date>
  6. Policy: <policy>
  7. X-Amz-Signature: <signature>
  8. file: <file>
英文:

Came across this problem and faced the "InvalidAccessKeyId" error when attempting to use the solution provided by @murrekatt.

Later I found out this issue was because I was generating the presigned POST inside a lambda and not including x-amz-security-token in the form data & policy.

So here's what I wrote with the help from @murrekatt and the boto3 library:

  1. import (
  2. &quot;crypto/hmac&quot;
  3. &quot;crypto/sha256&quot;
  4. &quot;encoding/base64&quot;
  5. &quot;encoding/hex&quot;
  6. &quot;encoding/json&quot;
  7. &quot;fmt&quot;
  8. &quot;time&quot;
  9. &quot;github.com/aws/aws-sdk-go-v2/aws&quot;
  10. )
  11. type PresignedPOST struct {
  12. URL string `json:&quot;url&quot;`
  13. Key string `json:&quot;key&quot;`
  14. Policy string `json:&quot;policy&quot;`
  15. Credential string `json:&quot;credential&quot;`
  16. SecurityToken string `json:&quot;securityToken,omitempty&quot;`
  17. Signature string `json:&quot;signature&quot;`
  18. Date string `json:&quot;date&quot;`
  19. }
  20. func NewPresignedPost(input *NewPresignedPostInput) (*PresignedPOST, error) {
  21. // expiration time
  22. expirationTime := time.Now().Add(time.Second * time.Duration(input.ExpiresIn)).UTC()
  23. dateString := expirationTime.Format(&quot;20060102&quot;)
  24. // credentials string
  25. creds := fmt.Sprintf(&quot;%s/%s/%s/s3/aws4_request&quot;, input.Credentials.AccessKeyID, dateString, input.Region)
  26. // policy
  27. policyDoc, err := createPolicyDocument(expirationTime, input.Bucket, input.Key, creds, &amp;input.Credentials.SessionToken, input.Conditions)
  28. if err != nil {
  29. return nil, err
  30. }
  31. // create signature
  32. signature := createSignature(input.Credentials.SecretAccessKey, input.Region, dateString, policyDoc)
  33. // url
  34. url := fmt.Sprintf(&quot;https://%s.s3.amazonaws.com/&quot;, input.Bucket)
  35. // expiration time
  36. dateTimeString := expirationTime.Format(&quot;20060102T150405Z&quot;)
  37. // post
  38. post := &amp;PresignedPOST{
  39. Key: input.Key,
  40. Policy: policyDoc,
  41. Signature: signature,
  42. URL: url,
  43. Credential: creds,
  44. SecurityToken: input.Credentials.SessionToken,
  45. Date: dateTimeString,
  46. }
  47. return post, nil
  48. }
  49. type NewPresignedPostInput struct {
  50. // Key name
  51. Key string
  52. // Creds
  53. Credentials aws.Credentials
  54. // Region
  55. Region string
  56. // The name of the bucket to presign the post to
  57. Bucket string
  58. // Expiration - The number of seconds the presigned post is valid for.
  59. ExpiresIn int64
  60. // A list of conditions to include in the policy. Each element can be either a list or a structure.
  61. // For example:
  62. // [
  63. // {&quot;acl&quot;: &quot;public-read&quot;}, [&quot;content-length-range&quot;, 2, 5], [&quot;starts-with&quot;, &quot;$success_action_redirect&quot;, &quot;&quot;]
  64. // ]
  65. Conditions []interface{}
  66. }
  67. // helpers
  68. func createPolicyDocument(expirationTime time.Time, bucket string, key string, credentialString string, securityToken *string, extraConditions []interface{}) (string, error) {
  69. doc := map[string]interface{}{}
  70. doc[&quot;expiration&quot;] = expirationTime.Format(&quot;2006-01-02T15:04:05.000Z&quot;)
  71. // conditions
  72. conditions := []interface{}{}
  73. conditions = append(conditions, map[string]string{
  74. &quot;bucket&quot;: bucket,
  75. })
  76. conditions = append(conditions, []string{
  77. &quot;starts-with&quot;, &quot;$key&quot;, key,
  78. })
  79. conditions = append(conditions, map[string]string{
  80. &quot;x-amz-credential&quot;: credentialString,
  81. })
  82. if securityToken != nil {
  83. conditions = append(conditions, map[string]string{
  84. &quot;x-amz-security-token&quot;: *securityToken,
  85. })
  86. }
  87. conditions = append(conditions, map[string]string{
  88. &quot;x-amz-algorithm&quot;: &quot;AWS4-HMAC-SHA256&quot;,
  89. })
  90. conditions = append(conditions, map[string]string{
  91. &quot;x-amz-date&quot;: expirationTime.Format(&quot;20060102T150405Z&quot;),
  92. })
  93. // other conditions
  94. conditions = append(conditions, extraConditions...)
  95. doc[&quot;conditions&quot;] = conditions
  96. // base64 encoded json string
  97. jsonBytes, err := json.Marshal(doc)
  98. if err != nil {
  99. return &quot;&quot;, err
  100. }
  101. return base64.StdEncoding.EncodeToString(jsonBytes), nil
  102. }
  103. func createSignature(secretKey string, region string, dateString string, stringToSign string) string {
  104. // Helper to make the HMAC-SHA256.
  105. makeHmac := func(key []byte, data []byte) []byte {
  106. hash := hmac.New(sha256.New, key)
  107. hash.Write(data)
  108. return hash.Sum(nil)
  109. }
  110. h1 := makeHmac([]byte(&quot;AWS4&quot;+secretKey), []byte(dateString))
  111. h2 := makeHmac(h1, []byte(region))
  112. h3 := makeHmac(h2, []byte(&quot;s3&quot;))
  113. h4 := makeHmac(h3, []byte(&quot;aws4_request&quot;))
  114. signature := makeHmac(h4, []byte(stringToSign))
  115. return hex.EncodeToString(signature)
  116. }

Usage

  1. // credentials
  2. conf, _ := config.LoadDefaultConfig(c.Context)
  3. awsCreds, _ := conf.Credentials.Retrieve(c.Context)
  4. // generate presigned post
  5. post, err := s3util.NewPresignedPost(&amp;s3util.NewPresignedPostInput{
  6. Key: &lt;file-name&gt;,
  7. Credentials: awsCreds,
  8. Region: &lt;region&gt;,
  9. Bucket: &lt;bucket-name&gt;,
  10. ExpiresIn: &lt;expiration&gt;,
  11. Conditions: []interface{}{
  12. []interface{}{&quot;content-length-range&quot;, 1, &lt;size-limit&gt;},
  13. },
  14. })

Then on the frontend, use the returned json in a POST form data

  1. key: &lt;key&gt;
  2. X-Amz-Credential: &lt;credential&gt;
  3. X-Amz-Security-Token: &lt;securityToken&gt; // if provided
  4. X-Amz-Algorithm: AWS4-HMAC-SHA256
  5. X-Amz-Date: &lt;date&gt;
  6. Policy: &lt;policy&gt;
  7. X-Amz-Signature: &lt;signature&gt;
  8. file: &lt;file&gt;

huangapple
  • 本文由 发表于 2015年9月3日 22:00:13
  • 转载请务必保留本文链接:https://go.coder-hub.com/32377782.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定