@@ -132,13 +132,67 @@ public static function add_hooks( $compat ) {
132
132
}
133
133
134
134
/**
135
- * For each provider, include it and then instantiate it .
135
+ * Delete all plugin data on uninstall .
136
136
*
137
- * @since 0.1-dev
137
+ * @return void
138
+ */
139
+ public static function uninstall () {
140
+ // Keep this updated as user meta keys are added or removed.
141
+ $ user_meta_keys = array (
142
+ self ::PROVIDER_USER_META_KEY ,
143
+ self ::ENABLED_PROVIDERS_USER_META_KEY ,
144
+ self ::USER_META_NONCE_KEY ,
145
+ self ::USER_RATE_LIMIT_KEY ,
146
+ self ::USER_FAILED_LOGIN_ATTEMPTS_KEY ,
147
+ self ::USER_PASSWORD_WAS_RESET_KEY ,
148
+ );
149
+
150
+ $ option_keys = array ();
151
+
152
+ foreach ( self ::get_providers_classes () as $ provider_class ) {
153
+ // Merge with provider-specific user meta keys.
154
+ if ( method_exists ( $ provider_class , 'uninstall_user_meta_keys ' ) ) {
155
+ try {
156
+ $ user_meta_keys = array_merge (
157
+ $ user_meta_keys ,
158
+ call_user_func ( array ( $ provider_class , 'uninstall_user_meta_keys ' ) )
159
+ );
160
+ } catch ( Exception $ e ) {
161
+ // Do nothing.
162
+ }
163
+ }
164
+
165
+ // Merge with provider-specific option keys.
166
+ if ( method_exists ( $ provider_class , 'uninstall_options ' ) ) {
167
+ try {
168
+ $ option_keys = array_merge (
169
+ $ option_keys ,
170
+ call_user_func ( array ( $ provider_class , 'uninstall_options ' ) )
171
+ );
172
+ } catch ( Exception $ e ) {
173
+ // Do nothing.
174
+ }
175
+ }
176
+ }
177
+
178
+ // Delete options first since that is faster.
179
+ if ( ! empty ( $ option_keys ) ) {
180
+ foreach ( $ option_keys as $ option_key ) {
181
+ delete_option ( $ option_key );
182
+ }
183
+ }
184
+
185
+ foreach ( $ user_meta_keys as $ meta_key ) {
186
+ delete_metadata ( 'user ' , null , $ meta_key , null , true );
187
+ }
188
+ }
189
+
190
+ /**
191
+ * Get the registered providers of which some might not be enabled.
138
192
*
139
- * @return array
193
+ * @return array List of provider keys and paths to class files.
140
194
*/
141
- public static function get_providers () {
195
+ public static function get_providers_registered () {
142
196
$ providers = array (
143
197
'Two_Factor_Email ' => TWO_FACTOR_DIR . 'providers/class-two-factor-email.php ' ,
144
198
'Two_Factor_Totp ' => TWO_FACTOR_DIR . 'providers/class-two-factor-totp.php ' ,
@@ -150,29 +204,29 @@ public static function get_providers() {
150
204
/**
151
205
* Filter the supplied providers.
152
206
*
153
- * This lets third-parties either remove providers (such as Email), or
154
- * add their own providers (such as text message or Clef).
155
- *
156
207
* @param array $providers A key-value array where the key is the class name, and
157
208
* the value is the path to the file containing the class.
158
209
*/
159
- $ providers = apply_filters ( 'two_factor_providers ' , $ providers );
210
+ $ additional_providers = apply_filters ( 'two_factor_providers ' , $ providers );
160
211
161
- // FIDO U2F is PHP 5.3+ only.
162
- if ( isset ( $ providers ['Two_Factor_FIDO_U2F ' ] ) && version_compare ( PHP_VERSION , '5.3.0 ' , '< ' ) ) {
163
- unset( $ providers ['Two_Factor_FIDO_U2F ' ] );
164
- trigger_error ( // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
165
- sprintf (
166
- /* translators: %s: version number */
167
- __ ( 'FIDO U2F is not available because you are using PHP %s. (Requires 5.3 or greater) ' , 'two-factor ' ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
168
- PHP_VERSION
169
- )
170
- );
212
+ // Merge them with the default providers.
213
+ if ( ! empty ( $ additional_providers ) ) {
214
+ return array_merge ( $ providers , $ additional_providers );
171
215
}
172
216
173
- /**
174
- * For each filtered provider,
175
- */
217
+ return $ providers ;
218
+ }
219
+
220
+ /**
221
+ * Get the classnames for all registered providers.
222
+ *
223
+ * Note some of these providers might not be enabled.
224
+ *
225
+ * @return array List of provider keys and classnames.
226
+ */
227
+ private static function get_providers_classes () {
228
+ $ providers = self ::get_providers_registered ();
229
+
176
230
foreach ( $ providers as $ provider_key => $ path ) {
177
231
require_once $ path ;
178
232
@@ -189,9 +243,56 @@ public static function get_providers() {
189
243
/**
190
244
* Confirm that it's been successfully included before instantiating.
191
245
*/
192
- if ( class_exists ( $ class ) ) {
246
+ if ( method_exists ( $ class , 'get_instance ' ) ) {
247
+ $ providers [ $ provider_key ] = $ class ;
248
+ } else {
249
+ unset( $ providers [ $ provider_key ] );
250
+ }
251
+ }
252
+
253
+ return $ providers ;
254
+ }
255
+
256
+ /**
257
+ * Get all enabled two-factor providers.
258
+ *
259
+ * @since 0.1-dev
260
+ *
261
+ * @return array
262
+ */
263
+ public static function get_providers () {
264
+ $ providers = self ::get_providers_registered ();
265
+
266
+ /**
267
+ * Filter the supplied providers.
268
+ *
269
+ * This lets third-parties either remove providers (such as Email), or
270
+ * add their own providers (such as text message or Clef).
271
+ *
272
+ * @param array $providers A key-value array where the key is the class name, and
273
+ * the value is the path to the file containing the class.
274
+ */
275
+ $ providers = apply_filters ( 'two_factor_providers ' , $ providers );
276
+
277
+ // FIDO U2F is PHP 5.3+ only.
278
+ if ( isset ( $ providers ['Two_Factor_FIDO_U2F ' ] ) && version_compare ( PHP_VERSION , '5.3.0 ' , '< ' ) ) {
279
+ unset( $ providers ['Two_Factor_FIDO_U2F ' ] );
280
+ trigger_error ( // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
281
+ sprintf (
282
+ /* translators: %s: version number */
283
+ __ ( 'FIDO U2F is not available because you are using PHP %s. (Requires 5.3 or greater) ' , 'two-factor ' ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
284
+ PHP_VERSION
285
+ )
286
+ );
287
+ }
288
+
289
+ // Map provider keys to classes so that we can instantiate them.
290
+ $ providers = array_intersect_key ( self ::get_providers_classes (), $ providers );
291
+
292
+ foreach ( $ providers as $ provider_key => $ provider_class ) {
293
+ if ( method_exists ( $ provider_class , 'get_instance ' ) ) {
193
294
try {
194
- $ providers [ $ provider_key ] = call_user_func ( array ( $ class , 'get_instance ' ) );
295
+ $ providers [ $ provider_key ] = call_user_func ( array ( $ provider_class , 'get_instance ' ) );
195
296
} catch ( Exception $ e ) {
196
297
unset( $ providers [ $ provider_key ] );
197
298
}
0 commit comments