-
Notifications
You must be signed in to change notification settings - Fork 89
/
Copy pathCdn_RackSpace_Api_CloudFilesCdn.php
284 lines (253 loc) · 8.15 KB
/
Cdn_RackSpace_Api_CloudFilesCdn.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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
<?php
/**
* File: Cdn_RackSpace_Api_CloudFilesCdn.php
*
* @package W3TC
*/
namespace W3TC;
/**
* Class Cdn_RackSpace_Api_CloudFilesCdn
*
* phpcs:disable PSR2.Classes.PropertyDeclaration.Underscore
* phpcs:disable PSR2.Methods.MethodDeclaration.Underscore
* phpcs:disable WordPress.PHP.NoSilencedErrors.Discouraged
*/
class Cdn_RackSpace_Api_CloudFilesCdn {
/**
* Access token.
*
* @var string
*/
private $_access_token;
/**
* Access region descriptor.
*
* @var string
*/
private $_access_region_descriptor;
/**
* New access required flag.
*
* @var bool
*/
private $_new_access_required;
/**
* Constructor for the Rackspace Cloud Files CDN API class.
*
* Initializes the object with configuration parameters such as access token,
* access region descriptor, and a callback for renewing access.
*
* @param array $config {
* Configuration parameters for the API.
*
* @type string $access_token The access token for API authentication.
* @type array $access_region_descriptor Region-specific API endpoint details.
* @type callable $new_access_required Callback function to handle access renewal.
* }
*
* @return void
*/
public function __construct( $config = array() ) {
$this->_access_token = $config['access_token'];
$this->_access_region_descriptor = $config['access_region_descriptor'];
$this->_new_access_required = $config['new_access_required'];
}
/**
* Retrieves the list of containers from the Cloud Files service.
*
* Uses a GET request to fetch the list of containers associated with the account.
*
* @return array|string|WP_Error The response data or error from the API.
*/
public function containers() {
return $this->_wp_remote_get( '' );
}
/**
* Retrieves metadata for a specific container.
*
* Sends a HEAD request to fetch details about a specified container, such as size and object count.
*
* @param string $container The name of the container to fetch metadata for.
*
* @return array|string|WP_Error The response headers or error from the API.
*/
public function container_get( $container ) {
return $this->_wp_remote_head( '/' . $container );
}
/**
* Enables CDN for a specific container.
*
* Sends a PUT request to enable the Content Delivery Network (CDN) for a specified container.
*
* @param string $container The name of the container to enable CDN for.
*
* @return mixed The API response on success, or an error on failure.
*/
public function container_cdn_enable( $container ) {
return $this->_wp_remote_put(
'/' . $container,
array(
'X-Cdn-Enabled' => 'True',
)
);
}
/**
* Sends a GET request to the Rackspace API.
*
* Retrieves data from the specified URI. If authentication is required, it attempts
* to renew the access token and re-execute the request.
*
* @param string $uri The endpoint URI for the GET request.
*
* @return array|string|WP_Error The response data or error from the API.
*/
private function _wp_remote_get( $uri ) {
if ( ! empty( $this->_access_region_descriptor['object-cdn.publicURL'] ) ) {
$url_base = $this->_access_region_descriptor['object-cdn.publicURL'];
$result = wp_remote_get(
$url_base . $uri . '?format=json',
array(
'headers' => 'X-Auth-Token: ' . $this->_access_token,
// phpcs:ignore Squiz.PHP.CommentedOutCode.Found
// 'sslcertificates' => dirname( __FILE__ ) . '/Cdn_RackSpace_Api_CaCert.pem'
)
);
$r = self::_decode_response_json( $result );
if ( ! $r['auth_required'] ) {
return $r['response_json'];
}
}
$new_object = call_user_func( $this->_new_access_required );
return $new_object->_wp_remote_get( $uri );
}
/**
* Sends a HEAD request to the Rackspace API.
*
* Retrieves headers from the specified URI. If authentication is required, it attempts
* to renew the access token and re-execute the request.
*
* @param string $uri The endpoint URI for the HEAD request.
* @param string $method The HTTP method to use (default: 'GET').
*
* @return array|string|WP_Error The response headers or error from the API.
*/
private function _wp_remote_head( $uri, $method = 'GET' ) {
if ( ! empty( $this->_access_region_descriptor['object-cdn.publicURL'] ) ) {
$url_base = $this->_access_region_descriptor['object-cdn.publicURL'];
$result = wp_remote_get(
$url_base . $uri . '?format=json',
array(
'headers' => 'X-Auth-Token: ' . $this->_access_token,
// phpcs:ignore Squiz.PHP.CommentedOutCode.Found
// 'sslcertificates' => dirname( __FILE__ ) . '/Cdn_RackSpace_Api_CaCert.pem',
'method' => 'HEAD',
)
);
$r = self::_decode_response( $result );
if ( ! $r['auth_required'] ) {
return $result['headers'];
}
}
$new_object = call_user_func( $this->_new_access_required );
return $new_object->_wp_remote_head( $uri, $body );
}
/**
* Sends a PUT request to the Rackspace API.
*
* Updates data at the specified URI. If authentication is required, it attempts
* to renew the access token and re-execute the request.
*
* @param string $uri The endpoint URI for the PUT request.
* @param array $body The body of the PUT request.
* @param array $headers The headers for the PUT request.
*
* @return void
*/
private function _wp_remote_put( $uri, $body = array(), $headers = array() ) {
if ( ! empty( $this->_access_region_descriptor['object-cdn.publicURL'] ) ) {
$url_base = $this->_access_region_descriptor['object-cdn.publicURL'];
$headers['X-Auth-Token'] = $this->_access_token;
$result = wp_remote_post(
$url_base . $uri,
array(
'headers' => $headers,
'body' => $body,
// phpcs:ignore Squiz.PHP.CommentedOutCode.Found
// 'sslcertificates' => dirname( __FILE__ ) . '/Cdn_RackSpace_Api_CaCert.pem',
'method' => 'PUT',
)
);
$r = self::_decode_response( $result );
if ( ! $r['auth_required'] ) {
return;
}
}
$new_object = call_user_func( $this->_new_access_required );
return $new_object->_wp_remote_put( $uri, $body, $headers );
}
/**
* Decodes a JSON response from the Rackspace API.
*
* Validates the response, checks for errors, and decodes the JSON content.
*
* @param array $result The result array returned by a WordPress HTTP request.
*
* @return array {
* Decoded response data.
*
* @type array $response_json The JSON-decoded response data.
* @type bool $auth_required Whether re-authentication is required.
* }
*
* @throws \Exception If the response indicates an error or invalid JSON.
*/
private static function _decode_response_json( $result ) {
if ( is_wp_error( $result ) ) {
throw new \Exception( 'Failed to reach API endpoint' );
}
if ( empty( $result['body'] ) ) {
$response_json = array();
} else {
$response_json = @json_decode( $result['body'], true );
if ( is_null( $response_json ) ) {
throw new \Exception( 'Failed to reach API endpoint, got unexpected response ' . $result['body'] );
}
}
if ( ! in_array( (int) $result['response']['code'], array( 200, 201, 202, 204 ), true ) ) {
throw new \Exception( $result['body'] );
}
return array(
'response_json' => $response_json,
'auth_required' => false,
);
}
/**
* Decodes a response from the Rackspace API.
*
* Validates the response and checks if re-authentication is required.
*
* @param array $result The result array returned by a WordPress HTTP request.
*
* @return array {
* Response metadata.
*
* @type bool $auth_required Whether re-authentication is required.
* }
*
* @throws \Exception If the response indicates an error.
*/
private static function _decode_response( $result ) {
if ( is_wp_error( $result ) ) {
throw new \Exception( 'Failed to reach API endpoint' );
}
if ( ! in_array( (int) $result['response']['code'], array( 200, 201, 202, 204 ), true ) ) {
if ( 'Unauthorized' === $result['response']['message'] ) {
return array(
'auth_required' => true,
);
}
throw new \Exception( 'Failed to reach API endpoint, got unexpected response ' . $result['response']['message'] );
}
return array( 'auth_required' => false );
}
}