forked from beaulebens/keyring
-
Notifications
You must be signed in to change notification settings - Fork 0
/
keyring.php
432 lines (364 loc) · 13.7 KB
/
keyring.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
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
<?php
/*
Plugin Name: Keyring
Plugin URI: http://dentedreality.com.au/projects/wp-keyring/
Description: Keyring helps you manage your keys. It provides a generic, very hookable framework for connecting to remote systems and managing your access tokens, username/password combos etc for those services. On its own it doesn't do much, but it enables other plugins to do things that require authorization to act on your behalf.
Version: 3.0
Author: Beau Lebens
Author URI: http://dentedreality.com.au
License: GPL v2 or newer <https://www.gnu.org/licenses/gpl.txt>
*/
// Define this in your wp-config (and set to true) to enable debugging
defined( 'KEYRING__DEBUG_MODE' ) or define( 'KEYRING__DEBUG_MODE', false );
// The name of a class which extends Keyring_Store to handle storage/manipulation of tokens.
// Optionally define this in your wp-config.php or some other global config file.
defined( 'KEYRING__TOKEN_STORE' ) or define( 'KEYRING__TOKEN_STORE', 'Keyring_SingleStore' );
// Keyring can be run in "headless" mode, which just avoids creating any UI, and leaves
// that up to you. Defaults to off (provides its own basic UI).
defined( 'KEYRING__HEADLESS_MODE' ) or define( 'KEYRING__HEADLESS_MODE', false );
// Debug/messaging levels. Don't mess with these
define( 'KEYRING__DEBUG_NOTICE', 1 );
define( 'KEYRING__DEBUG_WARN', 2 );
define( 'KEYRING__DEBUG_ERROR', 3 );
// Indicates Keyring is installed/active so that other plugins can detect it
define( 'KEYRING__VERSION', '3.0' );
/**
* Core Keyring class that handles UI and the general flow of requesting access tokens etc
* to manage access to remote services.
*
* @package Keyring
*/
class Keyring {
protected $registered_services = array();
protected $store = false;
protected $errors = array();
protected $messages = array();
protected $token_store = '';
var $admin_page = 'keyring';
function __construct() {
if ( ! KEYRING__HEADLESS_MODE ) {
require_once dirname( __FILE__ ) . '/admin-ui.php';
Keyring_Admin_UI::init();
add_filter(
'keyring_admin_url',
function( $url, $params ) {
$url = admin_url( 'tools.php?page=' . Keyring::init()->admin_page );
return add_query_arg( $params, $url );
},
10,
2
);
}
// This is used internally to create URLs, and also to know when to
// attach handers. @see admin_url() and request_handlers()
$this->admin_page = apply_filters( 'keyring_admin_page', 'keyring' );
}
static function init( $force_load = false ) {
static $instance = false;
if ( ! $instance ) {
if ( ! KEYRING__HEADLESS_MODE ) {
load_plugin_textdomain( 'keyring', false, dirname( plugin_basename( __FILE__ ) ) . '/languages/' );
}
$instance = new Keyring;
// Keyring is being loaded 'late', so we need to do some extra set-up
if ( did_action( 'init' ) || $force_load ) {
$instance->plugins_loaded();
do_action( 'keyring_load_services' );
}
} else {
if ( $force_load ) {
$instance->plugins_loaded();
do_action( 'keyring_load_services' );
}
}
return $instance;
}
static function plugins_loaded() {
// Load stores early so we can confirm they're loaded correctly
require_once dirname( __FILE__ ) . '/store.php';
do_action( 'keyring_load_token_stores' );
$keyring = Keyring::init();
$keyring->token_store = apply_filters( 'keyring_token_store', defined( 'KEYRING__TOKEN_STORE' ) ? KEYRING__TOKEN_STORE : false );
if ( ! class_exists( $keyring->token_store ) || ! in_array( 'Keyring_Store', class_parents( $keyring->token_store ), true ) ) {
/* translators: file name */
wp_die( sprintf( __( 'Invalid <code>KEYRING__TOKEN_STORE</code> specified. Please make sure <code>KEYRING__TOKEN_STORE</code> is set to a valid classname for handling token storage in <code>%s</code> (or <code>wp-config.php</code>)', 'keyring' ), __FILE__ ) );
}
// Load base token and service definitions + core services
require_once dirname( __FILE__ ) . '/token.php';
require_once dirname( __FILE__ ) . '/service.php'; // Triggers a load of all core + extended service definitions
// Initiate Keyring
add_action( 'init', array( 'Keyring', 'init' ), 1 );
// Load external Services (plugins etc should hook to this to define new ones/extensions)
add_action(
'init',
function() {
do_action( 'keyring_load_services' );
},
2
);
/**
* And trigger request handlers, which plugins and extended Services use to handle UI,
* redirects, errors etc.
* @see ::request_handlers()
*/
add_action( 'admin_init', array( 'Keyring', 'request_handlers' ), 100 );
}
/**
* Core request handler which is the crux of everything. An action is called
* here for almost everything Keyring does, so you can use it to intercept
* almost everything. Based entirely on $_REQUEST[page|action|service]
*/
static function request_handlers() {
global $current_user;
Keyring_Util::debug( "request_handers" );
$_REQUEST = Keyring_Util::cleanup_request_parameters( $_REQUEST );
Keyring_Util::debug( $_REQUEST );
if ( ! empty( $_REQUEST['state'] ) ) {
Keyring_Util::unpack_state_parameters( $_REQUEST['state'] );
}
if ( defined( 'KEYRING__FORCE_USER' ) && KEYRING__FORCE_USER && in_array( $_REQUEST['action'], array( 'request', 'verify' ), true ) ) {
global $current_user;
$real_user = $current_user->ID;
wp_set_current_user( KEYRING__FORCE_USER );
}
if (
! empty( $_REQUEST['action'] )
&&
in_array( $_REQUEST['action'], apply_filters( 'keyring_core_actions', array( 'request', 'verify', 'created', 'delete', 'manage' ) ), true )
&&
! empty( $_REQUEST['service'] )
&&
in_array( $_REQUEST['service'], array_keys( Keyring::get_registered_services() ), true )
) {
// We have an action here to allow us to do things pre-authorization, just in case
Keyring_Util::debug( "pre_keyring_{$_REQUEST['service']}_{$_REQUEST['action']}", $_REQUEST );
do_action( "pre_keyring_{$_REQUEST['service']}_{$_REQUEST['action']}", $_REQUEST );
// Core nonce check required for everything. "keyring-ACTION" is the kr_nonce format
if ( ! isset( $_REQUEST['kr_nonce'] ) || ! wp_verify_nonce( $_REQUEST['kr_nonce'], 'keyring-' . $_REQUEST['action'] ) ) {
Keyring::error( __( 'Invalid/missing Keyring core nonce. All core actions require a valid nonce.', 'keyring' ) );
exit;
}
Keyring_Util::debug( "keyring_{$_REQUEST['service']}_{$_REQUEST['action']}" );
Keyring_Util::debug( $_GET );
do_action( "keyring_{$_REQUEST['service']}_{$_REQUEST['action']}", $_REQUEST );
if ( 'delete' === $_REQUEST['action'] ) {
do_action( 'keyring_connection_deleted', $_REQUEST['service'], $_REQUEST );
}
}
if ( defined( 'KEYRING__FORCE_USER' ) && KEYRING__FORCE_USER && in_array( $_REQUEST['action'], array( 'request', 'verify' ), true ) ) {
wp_set_current_user( $real_user );
}
}
static function register_service( Keyring_Service $service ) {
Keyring::init()->registered_services[ $service->get_name() ] = $service;
return true;
}
static function get_registered_services() {
return Keyring::init()->registered_services;
}
static function get_service_by_name( $name ) {
$keyring = Keyring::init();
if ( ! isset( $keyring->registered_services[ $name ] ) ) {
return null;
}
return $keyring->registered_services[ $name ];
}
static function get_token_store() {
$keyring = Keyring::init();
if ( ! $keyring->store ) {
$keyring->store = call_user_func( array( $keyring->token_store, 'init' ) );
}
return $keyring->store;
}
static function message( $str ) {
$keyring = Keyring::init();
$keyring->messages[] = $str;
}
/**
* Generic error handler/trigger.
* @param String $str Informational message (user-readable)
* @param array $info Additional information relating to the error.
* @param boolean $die If we should immediately die (default) or continue
*/
static function error( $str, $info = array(), $die = true ) {
$keyring = Keyring::init();
$keyring->errors[] = $str;
do_action( 'keyring_error', $str, $info );
if ( $die ) {
wp_die( $str, __( 'Keyring Error', 'keyring' ) );
exit;
}
}
function has_errors() {
return count( $this->errors );
}
function has_messages() {
return count( $this->messages );
}
function get_messages() {
return $this->messages;
}
function get_errors() {
return $this->errors;
}
}
class Keyring_Util {
static function debug( $str, $level = KEYRING__DEBUG_NOTICE ) {
if ( ! KEYRING__DEBUG_MODE ) {
return;
}
if ( is_object( $str ) || is_array( $str ) ) {
$str = print_r( $str, true );
}
switch ( $level ) {
case KEYRING__DEBUG_WARN:
echo "<div style='border:solid 1px #000; padding: 5px; background: #eee;'>Keyring Warning: $str</div>";
break;
case KEYRING__DEBUG_ERROR:
wp_die( '<h1>Keyring Error:</h1>' . '<p>' . $str . '</p>' );
exit;
}
error_log( "Keyring: $str" );
}
/**
* @deprecated No longer used internally.
* @return boolean
*/
static function is_service( $service ) {
if ( is_object( $service ) && is_subclass_of( $service, 'Keyring_Service' ) ) {
return true;
}
return false;
}
static function has_custom_ui( $service, $action ) {
return has_action( "keyring_{$service}_{$action}_ui" );
}
/**
* Get a URL to the Keyring admin UI, works kinda like WP's admin_url()
*
* @param string $service Shortname of a specific service.
* @return URL to Keyring admin UI (main listing, or specific service verify process)
*/
static function admin_url( $service = false, $params = array() ) {
$url = admin_url();
if ( $service ) {
$params['service'] = $service;
}
if ( count( $params ) ) {
$url = add_query_arg( $params, $url );
}
return apply_filters( 'keyring_admin_url', $url, $params );
}
static function connect_to( $service, $for ) {
Keyring_Util::debug( 'Connect to: ' . $service );
// Redirect into Keyring's auth handler if a valid service is provided
$kr_nonce = wp_create_nonce( 'keyring-request' );
$request_nonce = wp_create_nonce( 'keyring-request-' . $service );
wp_safe_redirect(
Keyring_Util::admin_url(
$service,
array(
'action' => 'request',
'kr_nonce' => $kr_nonce,
'nonce' => $request_nonce,
'for' => $for,
)
)
);
exit;
}
static function token_select_box( $tokens, $name, $create = false ) {
?><select name="<?php echo esc_attr( $name ); ?>" id="<?php echo esc_attr( $name ); ?>">
<?php if ( $create ) : ?>
<option value="new"><?php _e( 'Create a new connection…', 'keyring' ); ?></option>
<?php endif; ?>
<?php foreach ( (array) $tokens as $token ) : ?>
<option value="<?php echo $token->get_uniq_id(); ?>"><?php echo $token->get_display(); ?></option>
<?php endforeach; ?>
</select>
<?php
}
static function is_error( $obj ) {
return is_a( $obj, 'Keyring_Error' );
}
/**
* For some services the request params are JSON encoded into the state param. This
* method JSON decodes these and adds them back to the _REQUEST global
*
* @param string $state The JSON encoded $_REQUEST['state'] param
* @return void
*/
static function unpack_state_parameters( $state ) {
$unpacked_state = json_decode( base64_decode( $state ), true );
if ( ! empty( $unpacked_state['hash'] ) ) {
$validated_parameters = Keyring_Util::get_validated_parameters( $unpacked_state );
if ( ! $validated_parameters ) {
Keyring::error( __( 'Invalid data returned by the service. Please try again.', 'keyring' ) );
exit;
}
foreach ( $validated_parameters as $key => $value ) {
$_REQUEST[ $key ] = $value;
}
$_GET['state'] = $validated_parameters['state'];
}
}
/**
* Get an sha256 hash of a JSON encoded array of parameters
*
* @param string $encoded_parameters A JSON encoded string of a parameter array.
* @return string An sha256 hash
*/
static function get_parameter_hash( $encoded_parameters ) {
return hash_hmac( 'sha256', $encoded_parameters, NONCE_KEY );
}
/**
* Get a based 64 and JSON encoded representation of an array of parameters which includes
* a hash of the original params so they can be verified in a service callback
*
* @param array $parameters An array of query parameters.
* @return string A base64 encoded copy of the JSON encoded paramaters
*/
static function get_hashed_parameters( $parameters ) {
$parameters['hash'] = self::get_parameter_hash( wp_json_encode( $parameters ) );
return base64_encode( wp_json_encode( $parameters ) );
}
/**
* Validates that a hash of the parameter array matches the included hash parameter
*
* @param array $parameters An array of query parameters.
* @return array An array of the parameters minus the hash
*/
static function get_validated_parameters( $parameters ) {
if ( empty( $parameters['hash'] ) ) {
return false;
}
$return_hash = $parameters['hash'];
unset( $parameters['hash'] );
$hash = self::get_parameter_hash( wp_json_encode( $parameters ) );
if ( ! hash_equals( $hash, $return_hash ) ) {
return false;
}
return $parameters;
}
static function cleanup_request_parameters( $params ) {
Keyring_Util::debug( "cleanup_request_parameters" );
Keyring_Util::debug( $params );
if ( is_array( $params ) ) {
foreach ( $params as $key => $val ) {
// Some services double-encode things (Strava), so we need to clean up parameter names before using
if ( substr( $key, 0, 4 ) == 'amp;' ) {
$params[ substr( $key, 4 ) ] = $val;
unset( $params[ $key ] );
}
}
}
return $params;
}
}
/**
* Stub implementation of an error object. May at some point get custom, but
* treat it like a normal WP_Error for now.
*/
class Keyring_Error extends WP_Error { }
// This is the main hook that kicks off everything. Needs to be early so we have time to load everything.
add_action( 'plugins_loaded', array( 'Keyring', 'plugins_loaded' ) );