216 lines
7 KiB
Go
216 lines
7 KiB
Go
// Package edgegrid allows you to sign http.Request's using the Akamai OPEN Edgegrid Signing Scheme
|
|
package edgegrid
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/hmac"
|
|
"crypto/sha256"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"os"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
"unicode"
|
|
|
|
"github.com/google/uuid"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
const defaultSection = "DEFAULT"
|
|
|
|
// AddRequestHeader sets the Authorization header to use Akamai Open API
|
|
func AddRequestHeader(config Config, req *http.Request) *http.Request {
|
|
if config.Debug {
|
|
log.SetLevel(log.DebugLevel)
|
|
}
|
|
timestamp := makeEdgeTimeStamp()
|
|
nonce := createNonce()
|
|
|
|
if req.Header.Get("Content-Type") == "" {
|
|
req.Header.Set("Content-Type", "application/json")
|
|
}
|
|
|
|
_, AkamaiCliEnvOK := os.LookupEnv("AKAMAI_CLI")
|
|
AkamaiCliVersionEnv, AkamaiCliVersionEnvOK := os.LookupEnv("AKAMAI_CLI_VERSION")
|
|
AkamaiCliCommandEnv, AkamaiCliCommandEnvOK := os.LookupEnv("AKAMAI_CLI_COMMAND")
|
|
AkamaiCliCommandVersionEnv, AkamaiCliCommandVersionEnvOK := os.LookupEnv("AKAMAI_CLI_COMMAND_VERSION")
|
|
|
|
if AkamaiCliEnvOK && AkamaiCliVersionEnvOK {
|
|
if req.Header.Get("User-Agent") != "" {
|
|
req.Header.Set("User-Agent", req.Header.Get("User-Agent")+" AkamaiCLI/"+AkamaiCliVersionEnv)
|
|
} else {
|
|
req.Header.Set("User-Agent", "AkamaiCLI/"+AkamaiCliVersionEnv)
|
|
}
|
|
}
|
|
if AkamaiCliCommandEnvOK && AkamaiCliCommandVersionEnvOK {
|
|
if req.Header.Get("User-Agent") != "" {
|
|
req.Header.Set("User-Agent", req.Header.Get("User-Agent")+" AkamaiCLI-"+AkamaiCliCommandEnv+"/"+AkamaiCliCommandVersionEnv)
|
|
} else {
|
|
req.Header.Set("User-Agent", "AkamaiCLI-"+AkamaiCliCommandEnv+"/"+AkamaiCliCommandVersionEnv)
|
|
}
|
|
}
|
|
|
|
req.Header.Set("Authorization", createAuthHeader(config, req, timestamp, nonce))
|
|
return req
|
|
}
|
|
|
|
// Must be assigned the UTC time when the request is signed.
|
|
// Format of “yyyyMMddTHH:mm:ss+0000”
|
|
func makeEdgeTimeStamp() string {
|
|
local := time.FixedZone("GMT", 0)
|
|
t := time.Now().In(local)
|
|
return fmt.Sprintf("%d%02d%02dT%02d:%02d:%02d+0000",
|
|
t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), t.Second())
|
|
}
|
|
|
|
// Must be assigned a nonce (number used once) for the request.
|
|
// It is a random string used to detect replayed request messages.
|
|
// A GUID is recommended.
|
|
func createNonce() string {
|
|
uuid, err := uuid.NewRandom()
|
|
if err != nil {
|
|
log.Errorf(errorMap[ErrUUIDGenerateFailed], err)
|
|
return ""
|
|
}
|
|
return uuid.String()
|
|
}
|
|
|
|
func stringMinifier(in string) (out string) {
|
|
white := false
|
|
for _, c := range in {
|
|
if unicode.IsSpace(c) {
|
|
if !white {
|
|
out = out + " "
|
|
}
|
|
white = true
|
|
} else {
|
|
out = out + string(c)
|
|
white = false
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func concatPathQuery(path, query string) string {
|
|
if query == "" {
|
|
return path
|
|
}
|
|
return fmt.Sprintf("%s?%s", path, query)
|
|
}
|
|
|
|
// createSignature is the base64-encoding of the SHA–256 HMAC of the data to sign with the signing key.
|
|
func createSignature(message string, secret string) string {
|
|
key := []byte(secret)
|
|
h := hmac.New(sha256.New, key)
|
|
h.Write([]byte(message))
|
|
return base64.StdEncoding.EncodeToString(h.Sum(nil))
|
|
}
|
|
|
|
func createHash(data string) string {
|
|
h := sha256.Sum256([]byte(data))
|
|
return base64.StdEncoding.EncodeToString(h[:])
|
|
}
|
|
|
|
func canonicalizeHeaders(config Config, req *http.Request) string {
|
|
var unsortedHeader []string
|
|
var sortedHeader []string
|
|
for k := range req.Header {
|
|
unsortedHeader = append(unsortedHeader, k)
|
|
}
|
|
sort.Strings(unsortedHeader)
|
|
for _, k := range unsortedHeader {
|
|
for _, sign := range config.HeaderToSign {
|
|
if sign == k {
|
|
v := strings.TrimSpace(req.Header.Get(k))
|
|
sortedHeader = append(sortedHeader, fmt.Sprintf("%s:%s", strings.ToLower(k), strings.ToLower(stringMinifier(v))))
|
|
}
|
|
}
|
|
}
|
|
return strings.Join(sortedHeader, "\t")
|
|
|
|
}
|
|
|
|
// signingKey is derived from the client secret.
|
|
// The signing key is computed as the base64 encoding of the SHA–256 HMAC of the timestamp string
|
|
// (the field value included in the HTTP authorization header described above) with the client secret as the key.
|
|
func signingKey(config Config, timestamp string) string {
|
|
key := createSignature(timestamp, config.ClientSecret)
|
|
return key
|
|
}
|
|
|
|
// The content hash is the base64-encoded SHA–256 hash of the POST body.
|
|
// For any other request methods, this field is empty. But the tac separator (\t) must be included.
|
|
// The size of the POST body must be less than or equal to the value specified by the service.
|
|
// Any request that does not meet this criteria SHOULD be rejected during the signing process,
|
|
// as the request will be rejected by EdgeGrid.
|
|
func createContentHash(config Config, req *http.Request) string {
|
|
var (
|
|
contentHash string
|
|
preparedBody string
|
|
bodyBytes []byte
|
|
)
|
|
if req.Body != nil {
|
|
bodyBytes, _ = ioutil.ReadAll(req.Body)
|
|
req.Body = ioutil.NopCloser(bytes.NewBuffer(bodyBytes))
|
|
preparedBody = string(bodyBytes)
|
|
}
|
|
|
|
log.Debugf("Body is %s", preparedBody)
|
|
if req.Method == "POST" && len(preparedBody) > 0 {
|
|
log.Debugf("Signing content: %s", preparedBody)
|
|
if len(preparedBody) > config.MaxBody {
|
|
log.Debugf("Data length %d is larger than maximum %d",
|
|
len(preparedBody), config.MaxBody)
|
|
|
|
preparedBody = preparedBody[0:config.MaxBody]
|
|
log.Debugf("Data truncated to %d for computing the hash", len(preparedBody))
|
|
}
|
|
contentHash = createHash(preparedBody)
|
|
}
|
|
log.Debugf("Content hash is '%s'", contentHash)
|
|
return contentHash
|
|
}
|
|
|
|
// The data to sign includes the information from the HTTP request that is relevant to ensuring that the request is authentic.
|
|
// This data set comprised of the request data combined with the authorization header value (excluding the signature field,
|
|
// but including the ; right before the signature field).
|
|
func signingData(config Config, req *http.Request, authHeader string) string {
|
|
|
|
dataSign := []string{
|
|
req.Method,
|
|
req.URL.Scheme,
|
|
req.URL.Host,
|
|
concatPathQuery(req.URL.Path, req.URL.RawQuery),
|
|
canonicalizeHeaders(config, req),
|
|
createContentHash(config, req),
|
|
authHeader,
|
|
}
|
|
log.Debugf("Data to sign %s", strings.Join(dataSign, "\t"))
|
|
return strings.Join(dataSign, "\t")
|
|
}
|
|
|
|
func signingRequest(config Config, req *http.Request, authHeader string, timestamp string) string {
|
|
return createSignature(signingData(config, req, authHeader),
|
|
signingKey(config, timestamp))
|
|
}
|
|
|
|
// The Authorization header starts with the signing algorithm moniker (name of the algorithm) used to sign the request.
|
|
// The moniker below identifies EdgeGrid V1, hash message authentication code, SHA–256 as the hash standard.
|
|
// This moniker is then followed by a space and an ordered list of name value pairs with each field separated by a semicolon.
|
|
func createAuthHeader(config Config, req *http.Request, timestamp string, nonce string) string {
|
|
authHeader := fmt.Sprintf("EG1-HMAC-SHA256 client_token=%s;access_token=%s;timestamp=%s;nonce=%s;",
|
|
config.ClientToken,
|
|
config.AccessToken,
|
|
timestamp,
|
|
nonce,
|
|
)
|
|
log.Debugf("Unsigned authorization header: '%s'", authHeader)
|
|
|
|
signedAuthHeader := fmt.Sprintf("%ssignature=%s", authHeader, signingRequest(config, req, authHeader, timestamp))
|
|
|
|
log.Debugf("Signed authorization header: '%s'", signedAuthHeader)
|
|
return signedAuthHeader
|
|
}
|