forked from clousale/amazon-sp-api-php
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSignature.php
141 lines (125 loc) · 4.85 KB
/
Signature.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
<?php
namespace SellerLegend\AmazonSellingPartnerAPI;
use Exception;
class Signature {
/**
* ## Signature Version 4 signing process.
*
* Signature Version 4 is the process to add authentication information to AWS requests sent by HTTP.
* For security, most requests to AWS must be signed with an access key,
* which consists of an access key ID and secret access key.
* These two keys are commonly referred to as your security credentials.
* For details on how to obtain credentials for your account
*
* REF : https://docs.aws.amazon.com/general/latest/gr/signature-version-4.html
*
* PS : Guzzle Request instance doesn't allow us to update headers after instantiation
* So, we need to sign the header separately and return Authentication header back as a result.
*
* Special thanks go to github user crazyfactory / z3niths who made a better implementation of the signature method
*
* @param $host
* @param $method
* @param string $uri
* @param string $queryString
* @param array $data
*
* @throws Exception
*
* @author crazyfactory https://github.com/crazyfactory
*/
public static function calculateSignature(
Configuration $config,
string $host,
string $method,
$uri = '',
$queryString = '',
$data = []
): array {
return self::calculateSignatureForService(
$host,
$method,
$uri,
$queryString,
$data,
'execute-api',
$config->getAccessKey(),
$config->getSecretKey(),
$config->getRegion(),
$config->getAccessToken(),
$config->getSecurityToken(),
$config->getUserAgent()
);
}
public static function calculateSignatureForService(
string $host,
string $method,
$uri,
$queryString,
$data,
string $service,
string $accessKey,
string $secretKey,
string $region,
$accessToken,
$securityToken,
$userAgent
): array {
$terminationString = 'aws4_request';
$algorithm = 'AWS4-HMAC-SHA256';
$amzdate = gmdate('Ymd\THis\Z');
$date = substr($amzdate, 0, 8);
// Prepare payload
if (is_array($data)) {
$param = json_encode($data);
if ('[]' == $param) {
$requestPayload = '';
} else {
$requestPayload = $param;
}
} else {
$requestPayload = $data;
}
// Hashed payload
$hashedPayload = hash('sha256', $requestPayload);
//Compute Canonical Headers
$canonicalHeaders = [
'host' => $host,
'user-agent' => $userAgent,
];
// Check and attach access token to request header.
if (!is_null($accessToken)) {
$canonicalHeaders['x-amz-access-token'] = $accessToken;
}
$canonicalHeaders['x-amz-date'] = $amzdate;
// Check and attach STS token to request header.
if (!is_null($securityToken)) {
$canonicalHeaders['x-amz-security-token'] = $securityToken;
}
$canonicalHeadersStr = '';
foreach ($canonicalHeaders as $h => $v) {
$canonicalHeadersStr .= $h . ':' . $v . "\n";
}
$signedHeadersStr = join(';', array_keys($canonicalHeaders));
//Prepare credentials scope
$credentialScope = $date . '/' . $region . '/' . $service . '/' . $terminationString;
//prepare canonical request
$canonicalRequest = $method . "\n" . $uri . "\n" . $queryString . "\n" . $canonicalHeadersStr . "\n" . $signedHeadersStr . "\n" . $hashedPayload;
//Prepare the string to sign
$stringToSign = $algorithm . "\n" . $amzdate . "\n" . $credentialScope . "\n" . hash('sha256', $canonicalRequest);
//Start signing locker process
//Reference : https://docs.aws.amazon.com/general/latest/gr/signature-version-4.html
$kSecret = 'AWS4' . $secretKey;
$kDate = hash_hmac('sha256', $date, $kSecret, true);
$kRegion = hash_hmac('sha256', $region, $kDate, true);
$kService = hash_hmac('sha256', $service, $kRegion, true);
$kSigning = hash_hmac('sha256', $terminationString, $kService, true);
//Compute the signature
$signature = trim(hash_hmac('sha256', $stringToSign, $kSigning));
//Finalize the authorization structure
$authorizationHeader = $algorithm . " Credential={$accessKey}/{$credentialScope}, SignedHeaders={$signedHeadersStr}, Signature={$signature}";
return array_merge($canonicalHeaders, [
'Authorization' => $authorizationHeader,
]);
}
}