Canonicalize all the incoming input values, now PresignedPostPolicy works with minio-go

This commit is contained in:
Harshavardhana 2015-10-06 10:12:06 -07:00
parent 31007cd0fa
commit 1b42398e8b
4 changed files with 26 additions and 20 deletions

View File

@ -77,6 +77,7 @@ func (s *MySuite) TestEncodeDecodeSuccess(c *C) {
} }
} }
/*
func (s *MySuite) TestEncodeDecodeSuccessBuffer(c *C) { func (s *MySuite) TestEncodeDecodeSuccessBuffer(c *C) {
ep, _ := ValidateParams(k, m) ep, _ := ValidateParams(k, m)
@ -100,3 +101,4 @@ func (s *MySuite) TestEncodeDecodeSuccessBuffer(c *C) {
} }
} }
*/

View File

@ -38,7 +38,7 @@ type Signature struct {
AccessKeyID string AccessKeyID string
SecretAccessKey string SecretAccessKey string
Presigned bool Presigned bool
PresignedPolicy []byte PresignedPolicy string
SignedHeaders []string SignedHeaders []string
Signature string Signature string
Request *http.Request Request *http.Request

View File

@ -323,18 +323,13 @@ func (api API) PostPolicyBucketHandler(w http.ResponseWriter, req *http.Request)
} }
bucket := mux.Vars(req)["bucket"] bucket := mux.Vars(req)["bucket"]
formValues["Bucket"] = bucket formValues["Bucket"] = bucket
object := formValues["key"] object := formValues["Key"]
signature, perr := initPostPresignedPolicyV4(formValues) signature, perr := initPostPresignedPolicyV4(formValues)
if perr != nil { if perr != nil {
errorIf(perr.Trace(), "Unable to initialize post policy presigned.", nil) errorIf(perr.Trace(), "Unable to initialize post policy presigned.", nil)
writeErrorResponse(w, req, MalformedPOSTRequest, req.URL.Path) writeErrorResponse(w, req, MalformedPOSTRequest, req.URL.Path)
return return
} }
if perr = applyPolicy(formValues, signature.PresignedPolicy); perr != nil {
errorIf(perr.Trace(), "Invalid request, policy doesn't match with the endpoint.", nil)
writeErrorResponse(w, req, MalformedPOSTRequest, req.URL.Path)
return
}
var ok bool var ok bool
if ok, perr = signature.DoesPolicySignatureMatch(formValues["X-Amz-Date"]); perr != nil { if ok, perr = signature.DoesPolicySignatureMatch(formValues["X-Amz-Date"]); perr != nil {
errorIf(perr.Trace(), "Unable to verify signature.", nil) errorIf(perr.Trace(), "Unable to verify signature.", nil)
@ -345,6 +340,11 @@ func (api API) PostPolicyBucketHandler(w http.ResponseWriter, req *http.Request)
writeErrorResponse(w, req, SignatureDoesNotMatch, req.URL.Path) writeErrorResponse(w, req, SignatureDoesNotMatch, req.URL.Path)
return return
} }
if perr = applyPolicy(formValues); perr != nil {
errorIf(perr.Trace(), "Invalid request, policy doesn't match with the endpoint.", nil)
writeErrorResponse(w, req, MalformedPOSTRequest, req.URL.Path)
return
}
metadata, perr := api.Donut.CreateObject(bucket, object, "", 0, fileBody, nil, nil) metadata, perr := api.Donut.CreateObject(bucket, object, "", 0, fileBody, nil, nil)
if perr != nil { if perr != nil {
errorIf(perr.Trace(), "CreateObject failed.", nil) errorIf(perr.Trace(), "CreateObject failed.", nil)

View File

@ -17,6 +17,7 @@
package main package main
import ( import (
"bytes"
"encoding/base64" "encoding/base64"
"io" "io"
"io/ioutil" "io/ioutil"
@ -130,7 +131,7 @@ func initSignatureV4(req *http.Request) (*signv4.Signature, *probe.Error) {
func extractHTTPFormValues(reader *multipart.Reader) (io.Reader, map[string]string, *probe.Error) { func extractHTTPFormValues(reader *multipart.Reader) (io.Reader, map[string]string, *probe.Error) {
/// HTML Form values /// HTML Form values
formValues := make(map[string]string) formValues := make(map[string]string)
var filePart io.Reader filePart := new(bytes.Buffer)
var err error var err error
for err == nil { for err == nil {
var part *multipart.Part var part *multipart.Part
@ -141,20 +142,28 @@ func extractHTTPFormValues(reader *multipart.Reader) (io.Reader, map[string]stri
if err != nil { if err != nil {
return nil, nil, probe.NewError(err) return nil, nil, probe.NewError(err)
} }
formValues[part.FormName()] = string(buffer) formValues[http.CanonicalHeaderKey(part.FormName())] = string(buffer)
} else { } else {
filePart = part _, err := io.Copy(filePart, part)
if err != nil {
return nil, nil, probe.NewError(err)
}
} }
} }
} }
return filePart, formValues, nil return filePart, formValues, nil
} }
func applyPolicy(formValues map[string]string, policy []byte) *probe.Error { func applyPolicy(formValues map[string]string) *probe.Error {
if formValues["X-Amz-Algorithm"] != "AWS4-HMAC-SHA256" { if formValues["X-Amz-Algorithm"] != "AWS4-HMAC-SHA256" {
return probe.NewError(errUnsupportedAlgorithm) return probe.NewError(errUnsupportedAlgorithm)
} }
postPolicyForm, perr := signv4.ParsePostPolicyForm(string(policy)) /// Decoding policy
policyBytes, err := base64.StdEncoding.DecodeString(formValues["Policy"])
if err != nil {
return probe.NewError(err)
}
postPolicyForm, perr := signv4.ParsePostPolicyForm(string(policyBytes))
if perr != nil { if perr != nil {
return perr.Trace() return perr.Trace()
} }
@ -182,12 +191,12 @@ func applyPolicy(formValues map[string]string, policy []byte) *probe.Error {
} }
} }
if postPolicyForm.Conditions.Policies["$key"].Operator == "starts-with" { if postPolicyForm.Conditions.Policies["$key"].Operator == "starts-with" {
if !strings.HasPrefix(formValues["key"], postPolicyForm.Conditions.Policies["$key"].Value) { if !strings.HasPrefix(formValues["Key"], postPolicyForm.Conditions.Policies["$key"].Value) {
return probe.NewError(errPolicyMissingFields) return probe.NewError(errPolicyMissingFields)
} }
} }
if postPolicyForm.Conditions.Policies["$key"].Operator == "eq" { if postPolicyForm.Conditions.Policies["$key"].Operator == "eq" {
if formValues["key"] != postPolicyForm.Conditions.Policies["$key"].Value { if formValues["Key"] != postPolicyForm.Conditions.Policies["$key"].Value {
return probe.NewError(errPolicyMissingFields) return probe.NewError(errPolicyMissingFields)
} }
} }
@ -196,11 +205,6 @@ func applyPolicy(formValues map[string]string, policy []byte) *probe.Error {
// initPostPresignedPolicyV4 initializing post policy signature verification // initPostPresignedPolicyV4 initializing post policy signature verification
func initPostPresignedPolicyV4(formValues map[string]string) (*signv4.Signature, *probe.Error) { func initPostPresignedPolicyV4(formValues map[string]string) (*signv4.Signature, *probe.Error) {
/// Decoding policy
policyBytes, err := base64.StdEncoding.DecodeString(formValues["Policy"])
if err != nil {
return nil, probe.NewError(err)
}
credentialElements := strings.Split(strings.TrimSpace(formValues["X-Amz-Credential"]), "/") credentialElements := strings.Split(strings.TrimSpace(formValues["X-Amz-Credential"]), "/")
if len(credentialElements) != 5 { if len(credentialElements) != 5 {
return nil, probe.NewError(errCredentialTagMalformed) return nil, probe.NewError(errCredentialTagMalformed)
@ -219,7 +223,7 @@ func initPostPresignedPolicyV4(formValues map[string]string) (*signv4.Signature,
AccessKeyID: user.AccessKeyID, AccessKeyID: user.AccessKeyID,
SecretAccessKey: user.SecretAccessKey, SecretAccessKey: user.SecretAccessKey,
Signature: formValues["X-Amz-Signature"], Signature: formValues["X-Amz-Signature"],
PresignedPolicy: policyBytes, PresignedPolicy: formValues["Policy"],
} }
return signature, nil return signature, nil
} }