-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathwp-cli-rename-db-prefix.php
319 lines (263 loc) · 8.18 KB
/
wp-cli-rename-db-prefix.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
<?php
/*
Plugin Name: WP-CLI Rename Database Prefix
Plugin URI: https://github.com/iandunn/wp-cli-rename-db-prefix
Description: A WP-CLI command to rename WordPress' database prefix.
Version: 0.1
Author: Ian Dunn
Author URI: http://iandunn.name
License: GPLv2
*/
if ( ! defined( 'WP_CLI' ) ) {
return;
}
/*
* TODO
*
* change invocation to `wp db rename-prefix <new>`
*
* Write unit tests
* can be in tests dir intead of features?
* prune unused stuff that `scaffold package-tests` added
*
* Add MultiSite support
*
*/
class WP_CLI_Rename_DB_Prefix extends \WP_CLI_Command {
public $old_prefix;
public $new_prefix;
public $is_dry_run = false;
public $is_prompt = true;
public $is_config_update = true;
/**
* Rename WordPress' database prefix.
*
* You will be prompted for confirmation before the command makes any changes.
*
* ## OPTIONS
*
* <new_prefix>
* : The new database prefix
*
* [--dry-run]
* : Preview which data would be updated.
* default: false
*
* [--confirm]
* : Ask for confirmation.
* default: true
*
* [--config-update]
* : updates wp-config.php.
* default: true
*
* ## EXAMPLES
*
* wp rename-db-prefix foo_
*
* @param array $args
* @param array $assoc_args
*/
public function __invoke( $args, $assoc_args ) {
global $wpdb;
$this->is_dry_run = \WP_CLI\Utils\get_flag_value( $assoc_args, 'dry-run', false );
$this->is_prompt = \WP_CLI\Utils\get_flag_value( $assoc_args, 'confirm', true );
$this->is_config_update = \WP_CLI\Utils\get_flag_value( $assoc_args, 'config-update', true );
wp_debug_mode(); // re-set `display_errors` after WP-CLI overrides it, see https://github.com/wp-cli/wp-cli/issues/706#issuecomment-203610437
$wpdb->show_errors( WP_DEBUG ); // This makes it easier to catch errors while developing this command, but we don't need to show them to users
$this->old_prefix = $wpdb->base_prefix;
$this->new_prefix = $args[0];
if ( is_multisite() ) {
\WP_CLI::error( "This command doesn't support MultiSite yet." );
}
$this->confirm();
try {
\WP_CLI::line();
$this->update_wp_config();
$this->rename_wordpress_tables();
$this->update_blog_options_tables();
$this->update_options_table();
$this->update_usermeta_table();
// todo set global $table_prefix to new one now, or earlier in process, to avoid errors during shutdown, etc?
\WP_CLI::success( 'Successfully renamed database prefix.' );
} catch ( Exception $exception ) {
\WP_CLI::error( $exception->getMessage(), false );
\WP_CLI::error( "You should check your site to see if it's broken. If it is, you can fix it by restoring your `wp-config.php` file and your database from backups." );
}
}
/**
* Confirm that the user wants to rename the prefix
*/
protected function confirm() {
\WP_CLI::line();
if ( $this->is_dry_run ) {
\WP_CLI::line( 'Running in dry run mode.' );
return;
}
if ( ! $this->is_prompt ) {
return;
}
\WP_CLI::warning( "Use this at your own risk. If something goes wrong, it could break your site. Before running this, make sure to back up your `wp-config.php` file and run `wp db export`." );
\WP_CLI::confirm( sprintf(
"\nAre you sure you want to rename %s's database prefix from `%s` to `%s`?",
parse_url( site_url(), PHP_URL_HOST ),
$this->old_prefix,
$this->new_prefix
) );
}
/**
* Update the prefix in `wp-config.php`
*
* @throws Exception
*/
protected function update_wp_config() {
if ( $this->is_dry_run ) {
return;
}
if ( ! $this->is_config_update ) {
\WP_CLI::line( 'Skipping wp-config.php update as requested.' );
return;
}
$wp_config_path = \WP_CLI\Utils\locate_wp_config(); // we know this is valid, because wp-cli won't run if it's not
$wp_config_contents = file_get_contents( $wp_config_path );
$search_pattern = '/(\$table_prefix\s*=\s*)([\'"]).+?\\2(\s*;)/';
$replace_pattern = "\${1}'{$this->new_prefix}'\${3}";
$wp_config_contents = preg_replace( $search_pattern, $replace_pattern, $wp_config_contents, - 1, $number_replacements );
if ( 0 === $number_replacements ) {
throw new Exception( "Failed to replace `\$table_prefix` in `wp-config.php`." );
}
if ( ! file_put_contents( $wp_config_path, $wp_config_contents ) ) {
throw new Exception( "Failed to update updated `wp-config.php` file." );
}
}
/**
* Rename all of WordPress' database tables
*
* @throws Exception
*/
protected function rename_wordpress_tables() {
global $wpdb;
$show_table_query = sprintf(
'SHOW TABLES LIKE "%s%%";',
$wpdb->esc_like( $this->old_prefix )
);
$tables = $wpdb->get_results( $show_table_query, ARRAY_N );
if ( ! $tables ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error );
}
foreach ( $tables as $table ) {
$table = substr( $table[0], strlen( $this->old_prefix ) );
$rename_query = sprintf(
"RENAME TABLE `%s` TO `%s`;",
$this->old_prefix . $table,
$this->new_prefix . $table
);
if ( $this->is_dry_run ) {
\WP_CLI::line( $rename_query );
continue;
}
if ( false === $wpdb->query( $rename_query ) ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error );
}
}
}
/**
* Update rows in all of the site `options` tables
*
* @throws Exception
*/
protected function update_blog_options_tables() {
global $wpdb;
if ( ! is_multisite() ) {
return;
}
throw new Exception( 'Not done yet' );
// todo this hasn't been tested at all
// todo should this really go after update_options_table, and reuse the same query?
// todo is this running on the root site twice b/c update_options_table() hits that too? should call either that or this, based on is_multisite() ?
$sites = get_sites( array( 'number' => false ) );
//blogs = $wpdb->get_col( "SELECT blog_id FROM `" . $this->new_prefix . "blogs` WHERE public = '1' AND archived = '0' AND mature = '0' AND spam = '0' ORDER BY blog_id DESC" );
if ( ! $sites ) {
throw new Exception( 'Failed to get all sites.' ); // todo test
}
foreach ( $sites as $site ) {
$update_query = $wpdb->prepare( "
UPDATE `{$this->new_prefix}{$site->blog_id}_options`
SET option_name = %s
WHERE option_name = %s
LIMIT 1;",
$this->new_prefix . $site->blog_id . '_user_roles',
$this->old_prefix . $site->blog_id . '_user_roles'
);
if ( $this->is_dry_run ) {
\WP_CLI::line( $update_query );
continue;
}
if ( ! $wpdb->query( $update_query ) ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error ); // todo test
}
}
}
/**
* Update rows in the `options` table
*
* @throws Exception
*/
protected function update_options_table() {
global $wpdb;
$update_query = $wpdb->prepare( "
UPDATE `{$this->new_prefix}options`
SET option_name = %s
WHERE option_name = %s
LIMIT 1;",
$this->new_prefix . 'user_roles',
$this->old_prefix . 'user_roles'
);
if ( $this->is_dry_run ) {
\WP_CLI::line( $update_query );
return;
}
if ( ! $wpdb->query( $update_query ) ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error );
}
}
/**
* Update rows in the `usermeta` table
*
* @throws Exception
*/
protected function update_usermeta_table() {
global $wpdb;
if ( $this->is_dry_run ) {
$rows = $wpdb->get_results( "SELECT meta_key FROM `{$this->old_prefix}usermeta`;" );
} else {
$rows = $wpdb->get_results( "SELECT meta_key FROM `{$this->new_prefix}usermeta`;" );
}
if ( ! $rows ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error );
}
foreach ( $rows as $row ) {
$meta_key_prefix = substr( $row->meta_key, 0, strlen( $this->old_prefix ) );
if ( $meta_key_prefix !== $this->old_prefix ) {
continue;
}
$new_key = $this->new_prefix . substr( $row->meta_key, strlen( $this->old_prefix ) );
$update_query = $wpdb->prepare( "
UPDATE `{$this->new_prefix}usermeta`
SET meta_key=%s
WHERE meta_key=%s
LIMIT 1;",
$new_key,
$row->meta_key
);
if ( $this->is_dry_run ) {
\WP_CLI::line( $update_query );
continue;
}
if ( ! $wpdb->query( $update_query ) ) {
throw new Exception( 'MySQL error: ' . $wpdb->last_error );
}
}
}
}
\WP_CLI::add_command( 'rename-db-prefix', 'WP_CLI_Rename_DB_Prefix' );