PageRenderTime 336ms CodeModel.GetById 24ms app.highlight 278ms RepoModel.GetById 1ms app.codeStats 1ms

/index.php

https://gitlab.com/Blueprint-Marketing/Search-Replace-DB
PHP | 1222 lines | 751 code | 271 blank | 200 comment | 77 complexity | 9357c81b3bb553d77290929601f88844 MD5 | raw file
   1<?php
   2
   3/**
   4 *
   5 * Safe Search and Replace on Database with Serialized Data v3.0.0
   6 *
   7 * This script is to solve the problem of doing database search and replace when
   8 * some data is stored within PHP serialized arrays or objects.
   9 *
  10 * For more information, see
  11 * http://interconnectit.com/124/search-and-replace-for-wordpress-databases/
  12 *
  13 * To contribute go to
  14 * http://github.com/interconnectit/search-replace-db
  15 *
  16 * To use, load the script on your server and point your web browser to it.
  17 * In some situations, consider using the command line interface version.
  18 *
  19 * BIG WARNING!  Take a backup first, and carefully test the results of this
  20 * code. If you don't, and you vape your data then you only have yourself to
  21 * blame. Seriously.  And if your English is bad and you don't fully
  22 * understand the instructions then STOP. Right there. Yes. Before you do any
  23 * damage.
  24 *
  25 * USE OF THIS SCRIPT IS ENTIRELY AT YOUR OWN RISK. I/We accept no liability
  26 * from its use.
  27 *
  28 * First Written 2009-05-25 by David Coveney of Interconnect IT Ltd (UK)
  29 * http://www.davidcoveney.com or http://interconnectit.com
  30 * and released under the GPL v3
  31 * ie, do what ever you want with the code, and we take no responsibility for it
  32 * OK? If you don't wish to take responsibility, hire us at Interconnect IT Ltd
  33 * on +44 (0)151 331 5140 and we will do the work for you at our hourly rate,
  34 * minimum 1hr
  35 *
  36 * License: GPL v3
  37 * License URL: http://www.gnu.org/copyleft/gpl.html
  38 *
  39 *
  40 * Version 3.0.0:
  41 * 		* Major overhaul
  42 * 		* Multibyte string replacements
  43 * 		* UI completely redesigned
  44 * 		* Removed all links from script until 'delete' has been clicked to avoid
  45 * 		  security risk from our access logs
  46 * 		* Search replace functionality moved to it's own separate class
  47 * 		* Replacements done table by table to avoid timeouts
  48 * 		* Convert tables to InnoDB
  49 * 		* Convert tables to utf8_unicode_ci
  50 * 		* Use PDO if available
  51 * 		* Preview/view changes
  52 * 		* Optionally use preg_replace()
  53 * 		* Scripts bootstraps WordPress/Drupal to avoid issues with unknown
  54 * 		  serialised objects/classes
  55 * 		* Added marketing stuff to deleted screen (sorry but we're running a
  56 * 		  business!)
  57 *
  58 * Version 2.2.0:
  59 * 		* Added remove script patch from David Anderson (wordshell.net)
  60 * 		* Added ability to replace strings with nothing
  61 *		* Copy changes
  62 * 		* Added code to recursive_unserialize_replace to deal with objects not
  63 * 		just arrays. This was submitted by Tina Matter.
  64 * 		ToDo: Test object handling. Not sure how it will cope with object in the
  65 * 		db created with classes that don't exist in anything but the base PHP.
  66 *
  67 * Version 2.1.0:
  68 *              - Changed to version 2.1.0
  69 *		* Following change by Sergei Biryukov - merged in and tested by Dave Coveney
  70 *              - Added Charset Support (tested with UTF-8, not tested on other charsets)
  71 *		* Following changes implemented by James Whitehead with thanks to all the commenters and feedback given!
  72 * 		- Removed PHP warnings if you go to step 3+ without DB details.
  73 * 		- Added options to skip changing the guid column. If there are other
  74 * 		columns that need excluding you can add them to the $exclude_cols global
  75 * 		array. May choose to add another option to the table select page to let
  76 * 		you add to this array from the front end.
  77 * 		- Minor tweak to label styling.
  78 * 		- Added comments to each of the functions.
  79 * 		- Removed a dead param from icit_srdb_replacer
  80 * Version 2.0.0:
  81 * 		- returned to using unserialize function to check if string is
  82 * 		serialized or not
  83 * 		- marked is_serialized_string function as deprecated
  84 * 		- changed form order to improve usability and make use on multisites a
  85 * 		bit less scary
  86 * 		- changed to version 2, as really should have done when the UI was
  87 * 		introduced
  88 * 		- added a recursive array walker to deal with serialized strings being
  89 * 		stored in serialized strings. Yes, really.
  90 * 		- changes by James R Whitehead (kudos for recursive walker) and David
  91 * 		Coveney 2011-08-26
  92 *  Version 1.0.2:
  93 *  	- typos corrected, button text tweak - David Coveney / Robert O'Rourke
  94 *  Version 1.0.1
  95 *  	- styling and form added by James R Whitehead.
  96 *
  97 *  Credits:  moz667 at gmail dot com for his recursive_array_replace posted at
  98 *            uk.php.net which saved me a little time - a perfect sample for me
  99 *            and seems to work in all cases.
 100 *
 101 */
 102
 103// always good here
 104header( 'HTTP/1.1 200 OK' );
 105header('Cache-Control: no-cache, no-store, must-revalidate'); // HTTP 1.1.
 106header('Pragma: no-cache'); // HTTP 1.0.
 107header('Expires: 0'); // Proxies.
 108
 109require_once( 'srdb.class.php' );
 110
 111class icit_srdb_ui extends icit_srdb {
 112
 113	/**
 114	 * @var string Root path of the CMS
 115	 */
 116	public $path;
 117
 118	public $is_wordpress = false;
 119	public $is_drupal = false;
 120
 121	public function __construct() {
 122
 123		// php 5.4 date timezone requirement, shouldn't affect anything
 124		date_default_timezone_set( 'Europe/London' );
 125
 126		// prevent fatals from hiding the UI
 127		register_shutdown_function( array( $this, 'fatal_handler' ) );
 128
 129		// flag to bootstrap WP or Drupal
 130		$bootstrap = true; // isset( $_GET[ 'bootstrap' ] );
 131
 132		// discover environment
 133		if ( $bootstrap && $this->is_wordpress() ) {
 134
 135			// prevent warnings if the charset and collate aren't defined
 136			if ( !defined( 'DB_CHARSET') ) {
 137				define( 'DB_CHARSET', 'utf8' );
 138			}
 139			if ( !defined( 'DB_COLLATE') ) {
 140				define( 'DB_COLLATE', '' );
 141			}
 142
 143			// populate db details
 144			$name 		= DB_NAME;
 145			$user 		= DB_USER;
 146			$pass 		= DB_PASSWORD;
 147			$host 		= DB_HOST;
 148			$charset 	= DB_CHARSET;
 149			$collate 	= DB_COLLATE;
 150
 151			$this->response( $name, $user, $pass, $host, $charset, $collate );
 152
 153		} elseif( $bootstrap && $this->is_drupal() ) {
 154
 155			$database = Database::getConnection();
 156			$database_opts = $database->getConnectionOptions();
 157
 158			// populate db details
 159			$name 		= $database_opts[ 'database' ];
 160			$user 		= $database_opts[ 'username' ];
 161			$pass 		= $database_opts[ 'password' ];
 162			$host 		= $database_opts[ 'host' ];
 163			$charset 	= 'utf8';
 164			$collate 	= '';
 165
 166		    $this->response( $name, $user, $pass, $host, $charset, $collate );
 167
 168		} else {
 169
 170			$this->response();
 171
 172		}
 173
 174	}
 175
 176
 177	public function response( $name = '', $user = '', $pass = '', $host = '127.0.0.1', $charset = 'utf8', $collate = '' ) {
 178
 179		// always override with post data
 180		if ( isset( $_POST[ 'name' ] ) ) {
 181			$name = $_POST[ 'name' ];	// your database
 182			$user = $_POST[ 'user' ];	// your db userid
 183			$pass = $_POST[ 'pass' ];	// your db password
 184			$host = $_POST[ 'host' ]; // normally localhost, but not necessarily.
 185			$charset = 'utf8'; // isset( $_POST[ 'char' ] ) ? stripcslashes( $_POST[ 'char' ] ) : '';	// your db charset
 186			$collate = '';
 187		}
 188
 189		// Search replace details
 190		$search = isset( $_POST[ 'search' ] ) ? $_POST[ 'search' ] : '';
 191		$replace = isset( $_POST[ 'replace' ] ) ? $_POST[ 'replace' ] : '';
 192
 193		// regex options
 194		$regex 	 = isset( $_POST[ 'regex' ] );
 195		$regex_i = isset( $_POST[ 'regex_i' ] );
 196		$regex_m = isset( $_POST[ 'regex_m' ] );
 197		$regex_s = isset( $_POST[ 'regex_s' ] );
 198		$regex_x = isset( $_POST[ 'regex_x' ] );
 199
 200		// Tables to scanned
 201		$tables = isset( $_POST[ 'tables' ] ) && is_array( $_POST[ 'tables' ] ) ? $_POST[ 'tables' ] : array( );
 202		if ( isset( $_POST[ 'use_tables' ] ) && $_POST[ 'use_tables' ] == 'all' )
 203			$tables = array();
 204
 205		// exclude / include columns
 206		$exclude_cols = isset( $_POST[ 'exclude_cols' ] ) ? $_POST[ 'exclude_cols' ] : array();
 207		$include_cols = isset( $_POST[ 'include_cols' ] ) ? $_POST[ 'include_cols' ] : array();
 208
 209		foreach( array( 'exclude_cols', 'include_cols' ) as $maybe_string_arg ) {
 210			if ( is_string( $$maybe_string_arg ) )
 211				$$maybe_string_arg = array_filter( array_map( 'trim', explode( ',', $$maybe_string_arg ) ) );
 212		}
 213
 214		// update class vars
 215		$vars = array(
 216			'name', 'user', 'pass', 'host',
 217			'charset', 'collate', 'tables',
 218			'search', 'replace',
 219			'exclude_cols', 'include_cols',
 220			'regex', 'regex_i', 'regex_m', 'regex_s', 'regex_x'
 221		);
 222
 223		foreach( $vars as $var ) {
 224			if ( isset( $$var ) )
 225				$this->set( $var, $$var );
 226		}
 227
 228		// are doing something?
 229		$show = '';
 230		if ( isset( $_POST[ 'submit' ] ) ) {
 231			if ( is_array( $_POST[ 'submit' ] ) )
 232				$show = key( $_POST[ 'submit' ] );
 233			if ( is_string( $_POST[ 'submit' ] ) )
 234				$show = preg_replace( '/submit\[([a-z0-9]+)\]/', '$1', $_POST[ 'submit' ] );
 235		}
 236
 237		// is it an AJAX call
 238		$ajax = isset( $_POST[ 'ajax' ] );
 239
 240		// body callback
 241		$html = 'ui';
 242
 243		switch( $show ) {
 244
 245			// remove search replace
 246			case 'delete':
 247
 248				// determine if it's the folder of compiled version
 249				if ( basename( __FILE__ ) == 'index.php' )
 250					$path = str_replace( basename( __FILE__ ), '', __FILE__ );
 251				else
 252					$path = __FILE__;
 253
 254				if ( $this->delete_script( $path ) ) {
 255					if ( is_file( __FILE__ ) && file_exists( __FILE__ ) )
 256						$this->add_error( 'Could not delete the search replace script. You will have to delete it manually', 'delete' );
 257					else
 258						$this->add_error( 'Search/Replace has been successfully removed from your server', 'delete' );
 259				} else {
 260					$this->add_error( 'Could not delete the search replace script automatically. You will have to delete it manually, sorry!', 'delete' );
 261				}
 262
 263				$html = 'deleted';
 264
 265				break;
 266
 267			case 'liverun':
 268
 269				// bsy-web, 20130621: Check live run was explicitly clicked and only set false then
 270				$this->set( 'dry_run', false );
 271
 272			case 'dryrun':
 273
 274				// build regex string
 275				// non UI implements can just pass in complete regex string
 276				if ( $this->regex ) {
 277					$mods = '';
 278					if ( $this->regex_i ) $mods .= 'i';
 279					if ( $this->regex_s ) $mods .= 's';
 280					if ( $this->regex_m ) $mods .= 'm';
 281					if ( $this->regex_x ) $mods .= 'x';
 282					$this->search = '/' . $this->search . '/' . $mods;
 283				}
 284
 285				// call search replace class
 286				$parent = parent::__construct( array(
 287					'name' => $this->get( 'name' ),
 288					'user' => $this->get( 'user' ),
 289					'pass' => $this->get( 'pass' ),
 290					'host' => $this->get( 'host' ),
 291					'search' => $this->get( 'search' ),
 292					'replace' => $this->get( 'replace' ),
 293					'tables' => $this->get( 'tables' ),
 294					'dry_run' => $this->get( 'dry_run' ),
 295					'regex' => $this->get( 'regex' ),
 296					'exclude_cols' => $this->get( 'exclude_cols' ),
 297					'include_cols' => $this->get( 'include_cols' )
 298				) );
 299
 300				break;
 301
 302			case 'innodb':
 303
 304				// call search replace class to alter engine
 305				$parent = parent::__construct( array(
 306					'name' => $this->get( 'name' ),
 307					'user' => $this->get( 'user' ),
 308					'pass' => $this->get( 'pass' ),
 309					'host' => $this->get( 'host' ),
 310					'tables' => $this->get( 'tables' ),
 311					'alter_engine' => 'InnoDB',
 312				) );
 313
 314				break;
 315
 316			case 'utf8':
 317
 318				// call search replace class to alter engine
 319				$parent = parent::__construct( array(
 320					'name' => $this->get( 'name' ),
 321					'user' => $this->get( 'user' ),
 322					'pass' => $this->get( 'pass' ),
 323					'host' => $this->get( 'host' ),
 324					'tables' => $this->get( 'tables' ),
 325					'alter_collation' => 'utf8_unicode_ci',
 326				) );
 327
 328				break;
 329
 330			case 'update':
 331			default:
 332
 333				// get tables or error messages
 334				$this->db_setup();
 335
 336				if ( $this->db_valid() ) {
 337
 338					// get engines
 339					$this->set( 'engines', $this->get_engines() );
 340
 341					// get tables
 342					$this->set( 'all_tables', $this->get_tables() );
 343
 344				}
 345
 346				break;
 347		}
 348
 349		$info = array(
 350			'table_select' => $this->table_select( false ),
 351			'engines' => $this->get( 'engines' )
 352		);
 353
 354		// set header again before output in case WP does it's thing
 355		header( 'HTTP/1.1 200 OK' );
 356
 357		if ( ! $ajax ) {
 358			$this->html( $html );
 359		} else {
 360
 361			// return json version of results
 362			header( 'Content-Type: application/json' );
 363
 364			echo json_encode( array(
 365				'errors' => $this->get( 'errors' ),
 366				'report' => $this->get( 'report' ),
 367				'info' 	 => $info
 368			) );
 369
 370			exit;
 371
 372		}
 373
 374	}
 375
 376
 377	public function exceptions( $exception ) {
 378		$this->add_error( '<p class="exception">' . $exception->getMessage() . '</p>' );
 379	}
 380
 381	public function errors( $no, $message, $file, $line ) {
 382		$this->add_error( '<p class="error">' . "<strong>{$no}:</strong> {$message} in {$file} on line {$line}" . '</p>', 'results' );
 383	}
 384
 385	public function fatal_handler() {
 386	    $error = error_get_last();
 387
 388		if( $error !== NULL ) {
 389			$errno   = $error["type"];
 390			$errfile = $error["file"];
 391			$errline = $error["line"];
 392			$errstr  = $error["message"];
 393
 394			if ( $errno == 1 ) {
 395				header( 'HTTP/1.1 200 OK' );
 396				$this->add_error( '<p class="error">Could not bootstrap environment.<br /> ' . "Fatal error in {$errfile}, line {$errline}. {$errstr}" . '</p>', 'environment' );
 397				$this->response();
 398			}
 399		}
 400	}
 401
 402
 403	/**
 404	 * http://stackoverflow.com/questions/3349753/delete-directory-with-files-in-it
 405	 *
 406	 * @param string $path directory/file path
 407	 *
 408	 * @return void
 409	 */
 410	public function delete_script( $path ) {
 411		return is_file( $path ) ?
 412				@unlink( $path ) :
 413				array_map( array( $this, __FUNCTION__ ), glob( $path . '/*' ) ) == @rmdir( $path );
 414	}
 415
 416
 417	/**
 418	 * Attempts to detect a WordPress installation and bootstraps the environment with it
 419	 *
 420	 * @return bool    Whether it is a WP install and we have database credentials
 421	 */
 422	public function is_wordpress() {
 423
 424		$path_mod = '';
 425		$depth = 0;
 426		$max_depth = 4;
 427		$bootstrap_file = 'wp-blog-header.php';
 428
 429		while( ! file_exists( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" ) ) {
 430			$path_mod .= '/..';
 431			if ( $depth++ >= $max_depth )
 432				break;
 433		}
 434
 435		if ( file_exists( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" ) ) {
 436
 437			// store WP path
 438			$this->path = dirname( __FILE__ ) . $path_mod;
 439
 440			// just in case we're white screening
 441			try {
 442				// need to make as many of the globals available as possible or things can break
 443				// (globals suck)
 444				global $wp, $wpdb, $wp_query, $wp_the_query, $wp_version,
 445					$wp_db_version, $tinymce_version, $manifest_version,
 446					$required_php_version, $required_mysql_version,
 447					$post, $posts, $wp_locale, $authordata, $more, $numpages,
 448					$currentday, $currentmonth, $page, $pages, $multipage,
 449					$wp_rewrite, $wp_filesystem, $blog_id, $request,
 450					$wp_styles, $wp_taxonomies, $wp_post_types, $wp_filter,
 451					$wp_object_cache, $query_string, $single, $post_type,
 452					$is_iphone, $is_chrome, $is_safari, $is_NS4, $is_opera,
 453					$is_macIE, $is_winIE, $is_gecko, $is_lynx, $is_IE,
 454					$is_apache, $is_iis7, $is_IIS;
 455
 456				// prevent multisite redirect
 457				define( 'WP_INSTALLING', true );
 458
 459				// prevent super/total cache
 460				define( 'DONOTCACHEDB', true );
 461				define( 'DONOTCACHEPAGE', true );
 462				define( 'DONOTCACHEOBJECT', true );
 463				define( 'DONOTCDN', true );
 464				define( 'DONOTMINIFY', true );
 465
 466				// cancel batcache
 467				if ( function_exists( 'batcache_cancel' ) )
 468					batcache_cancel();
 469
 470				// bootstrap WordPress
 471				require( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" );
 472
 473				$this->set( 'path', ABSPATH );
 474
 475				$this->set( 'is_wordpress', true );
 476
 477				return true;
 478
 479			} catch( Exception $error ) {
 480
 481				// try and get database values using regex approach
 482				$db_details = $this->define_find( $this->path . '/wp-config.php' );
 483
 484				if ( $db_details ) {
 485
 486					define( 'DB_NAME', $db_details[ 'name' ] );
 487					define( 'DB_USER', $db_details[ 'user' ] );
 488					define( 'DB_PASSWORD', $db_details[ 'pass' ] );
 489					define( 'DB_HOST', $db_details[ 'host' ] );
 490					define( 'DB_CHARSET', $db_details[ 'char' ] );
 491					define( 'DB_COLLATE', $db_details[ 'coll' ] );
 492
 493					// additional error message
 494					$this->add_error( 'WordPress detected but could not bootstrap environment. There might be a PHP error, possibly caused by changes to the database', 'db' );
 495
 496				}
 497
 498				if ( $db_details )
 499					return true;
 500
 501			}
 502
 503		}
 504
 505		return false;
 506	}
 507
 508
 509	public function is_drupal() {
 510
 511		$path_mod = '';
 512		$depth = 0;
 513		$max_depth = 4;
 514		$bootstrap_file = 'includes/bootstrap.inc';
 515
 516		while( ! file_exists( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" ) ) {
 517			$path_mod .= '/..';
 518			if ( $depth++ >= $max_depth )
 519				break;
 520		}
 521
 522		if ( file_exists( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" ) ) {
 523
 524			try {
 525				// require the bootstrap include
 526				require_once( dirname( __FILE__ ) . "{$path_mod}/{$bootstrap_file}" );
 527
 528				// define drupal root
 529				if ( ! defined( 'DRUPAL_ROOT' ) )
 530					define( 'DRUPAL_ROOT', dirname( __FILE__ ) . $path_mod );
 531
 532				// load drupal
 533				drupal_bootstrap( DRUPAL_BOOTSTRAP_FULL );
 534
 535				// confirm environment
 536				$this->set( 'is_drupal', true );
 537
 538				return true;
 539
 540			} catch( Exception $error ) {
 541
 542				$this->add_error( 'Drupal detected but could not bootstrap environment. There might be a PHP error, possibly caused by changes to the database', 'db' );
 543
 544			}
 545
 546		}
 547
 548		return false;
 549	}
 550
 551
 552	/**
 553	 * Search through the file name passed for a set of defines used to set up
 554	 * WordPress db access.
 555	 *
 556	 * @param string $filename The file name we need to scan for the defines.
 557	 *
 558	 * @return array    List of db connection details.
 559	 */
 560	public function define_find( $filename = 'wp-config.php' ) {
 561
 562		if ( $filename == 'wp-config.php' ) {
 563			$filename = dirname( __FILE__ ) . '/' . basename( $filename );
 564
 565			// look up one directory if config file doesn't exist in current directory
 566			if ( ! file_exists( $filename ) )
 567				$filename = dirname( __FILE__ ) . '/../' . basename( $filename );
 568		}
 569
 570		if ( file_exists( $filename ) && is_file( $filename ) && is_readable( $filename ) ) {
 571			$file = @fopen( $filename, 'r' );
 572			$file_content = fread( $file, filesize( $filename ) );
 573			@fclose( $file );
 574		}
 575
 576		preg_match_all( '/define\s*?\(\s*?([\'"])(DB_NAME|DB_USER|DB_PASSWORD|DB_HOST|DB_CHARSET|DB_COLLATE)\1\s*?,\s*?([\'"])([^\3]*?)\3\s*?\)\s*?;/si', $file_content, $defines );
 577
 578		if ( ( isset( $defines[ 2 ] ) && ! empty( $defines[ 2 ] ) ) && ( isset( $defines[ 4 ] ) && ! empty( $defines[ 4 ] ) ) ) {
 579			foreach( $defines[ 2 ] as $key => $define ) {
 580
 581				switch( $define ) {
 582					case 'DB_NAME':
 583						$name = $defines[ 4 ][ $key ];
 584						break;
 585					case 'DB_USER':
 586						$user = $defines[ 4 ][ $key ];
 587						break;
 588					case 'DB_PASSWORD':
 589						$pass = $defines[ 4 ][ $key ];
 590						break;
 591					case 'DB_HOST':
 592						$host = $defines[ 4 ][ $key ];
 593						break;
 594					case 'DB_CHARSET':
 595						$char = $defines[ 4 ][ $key ];
 596						break;
 597					case 'DB_COLLATE':
 598						$coll = $defines[ 4 ][ $key ];
 599						break;
 600				}
 601			}
 602		}
 603
 604		return array(
 605			'host' => $host,
 606			'name' => $name,
 607			'user' => $user,
 608			'pass' => $pass,
 609			'char' => $char,
 610			'coll' => $coll
 611		);
 612	}
 613
 614
 615	/**
 616	 * Display the current url
 617	 *
 618	 */
 619	public function self_link() {
 620		return 'http://' . $_SERVER[ 'HTTP_HOST' ] . rtrim( $_SERVER[ 'REQUEST_URI' ], '/' );
 621	}
 622
 623
 624	/**
 625	* Simple html escaping
 626	*
 627	* @param string $string Thing that needs escaping
 628	* @param bool $echo   Do we echo or return?
 629	*
 630	* @return string    Escaped string.
 631	*/
 632	public function esc_html_attr( $string = '', $echo = false ) {
 633		$output = htmlentities( $string, ENT_QUOTES, 'UTF-8' );
 634		if ( $echo )
 635			echo $output;
 636		else
 637			return $output;
 638	}
 639
 640	public function checked( $value, $value2, $echo = true ) {
 641		$output = $value == $value2 ? ' checked="checked"' : '';
 642		if ( $echo )
 643			echo $output;
 644		return $output;
 645	}
 646
 647	public function selected( $value, $value2, $echo = true ) {
 648		$output = $value == $value2 ? ' selected="selected"' : '';
 649		if ( $echo )
 650			echo $output;
 651		return $output;
 652	}
 653
 654
 655	public function get_errors( $type ) {
 656		if ( ! isset( $this->errors[ $type ] ) || ! count( $this->errors[ $type ] ) )
 657			return;
 658
 659		echo '<div class="errors">';
 660		foreach( $this->errors[ $type ] as $error ) {
 661			if ( $error instanceof Exception )
 662				echo '<p class="exception">' . $error->getMessage() . '</p>';
 663			elseif ( is_string( $error ) )
 664				echo $error;
 665		}
 666		echo '</div>';
 667	}
 668
 669
 670	public function get_report( $table = null ) {
 671
 672		$report = $this->get( 'report' );
 673
 674		if ( empty( $report ) )
 675			return;
 676
 677		$dry_run = $this->get( 'dry_run' );
 678		$search = $this->get( 'search' );
 679		$replace = $this->get( 'replace' );
 680
 681		// Calc the time taken.
 682		$time = array_sum( explode( ' ', $report[ 'end' ] ) ) - array_sum( explode( ' ', $report[ 'start' ] ) );
 683
 684		$srch_rplc_input_phrase = $dry_run ?
 685			'searching for <strong>"' . $search . '"</strong> (to be replaced by <strong>"' . $replace . '"</strong>)' :
 686			'replacing <strong>"' . $search . '"</strong> with <strong>"' . $replace . '"</strong>';
 687
 688		echo '
 689		<div class="report">';
 690
 691		echo '
 692			<h2>Report</h2>';
 693
 694		echo '
 695			<p>';
 696		printf(
 697			'In the process of %s we scanned <strong>%d</strong> tables with a total of
 698			<strong>%d</strong> rows, <strong>%d</strong> cells %s changed.
 699			<strong>%d</strong> db updates were actually performed.
 700			It all took <strong>%f</strong> seconds.',
 701			$srch_rplc_input_phrase,
 702			$report[ 'tables' ],
 703			$report[ 'rows' ],
 704			$report[ 'change' ],
 705			$dry_run ? 'would have been' : 'were',
 706			$report[ 'updates' ],
 707			$time
 708		);
 709		echo '
 710			</p>';
 711
 712		echo '
 713			<table class="table-reports">
 714				<thead>
 715					<tr>
 716						<th>Table</th>
 717						<th>Rows</th>
 718						<th>Cells changed</th>
 719						<th>Updates</th>
 720						<th>Seconds</th>
 721					</tr>
 722				</thead>
 723				<tbody>';
 724		foreach( $report[ 'table_reports' ] as $table => $t_report ) {
 725
 726			$t_time = array_sum( explode( ' ', $t_report[ 'end' ] ) ) - array_sum( explode( ' ', $t_report[ 'start' ] ) );
 727
 728			echo '
 729					<tr>';
 730			printf( '
 731						<th>%s:</th>
 732						<td>%d</td>
 733						<td>%d</td>
 734						<td>%d</td>
 735						<td>%f</td>',
 736				$table,
 737				$t_report[ 'rows' ],
 738				$t_report[ 'change' ],
 739				$t_report[ 'updates' ],
 740				$t_time
 741			);
 742			echo '
 743					</tr>';
 744
 745		}
 746		echo '
 747				</tbody>
 748			</table>';
 749
 750		echo '
 751		</div>';
 752
 753	}
 754
 755
 756	public function table_select( $echo = true ) {
 757
 758		$table_select = '';
 759
 760		if ( ! empty( $this->all_tables ) ) {
 761			$table_select .= '<select name="tables[]" multiple="multiple">';
 762			foreach( $this->all_tables as $table ) {
 763				$size = $table[ 'Data_length' ] / 1000;
 764				$size_unit = 'kb';
 765				if ( $size > 1000 ) {
 766					$size = $size / 1000;
 767					$size_unit = 'Mb';
 768				}
 769				if ( $size > 1000 ) {
 770					$size = $size / 1000;
 771					$size_unit = 'Gb';
 772				}
 773				$size = number_format( $size, 2 ) . $size_unit;
 774				$rows = $table[ 'Rows' ] > 1 ? 'rows' : 'row';
 775
 776				$table_select .= sprintf( '<option value="%s" %s>%s</option>',
 777							 $this->esc_html_attr( $table[ 0 ], false ),
 778							 $this->selected( true, in_array( $table[ 0 ], $this->tables ), false ),
 779							 "{$table[0]}: {$table['Engine']}, rows: {$table['Rows']}, size: {$size}, collation: {$table['Collation']}, character_set: {$table['Character_set']}"
 780						);
 781			}
 782			$table_select .= '</select>';
 783		}
 784
 785		if ( $echo )
 786			echo $table_select;
 787		return $table_select;
 788	}
 789
 790
 791	public function ui() {
 792
 793		// Warn if we're running in safe mode as we'll probably time out.
 794		if ( ini_get( 'safe_mode' ) ) {
 795			?>
 796			<div class="special-errors">
 797				<h4>Warning</h4>
 798				<?php echo printf( '<p>Safe mode is on so you may run into problems if it takes longer than %s seconds to process your request.</p>', ini_get( 'max_execution_time' ) ); ?>
 799			</div>
 800			<?php
 801		}
 802
 803		?>
 804		<form action="" method="post">
 805
 806			<!-- 1. search/replace -->
 807			<fieldset class="row row-search">
 808
 809				<h1>search<span>/</span>replace</h1>
 810
 811				<?php $this->get_errors( 'search' ); ?>
 812
 813				<div class="fields fields-large">
 814					<label for="search"><span class="label-text">replace</span> <span class="hide-if-regex-off regex-left">/</span><input id="search" type="text" placeholder="search for&hellip;" value="<?php $this->esc_html_attr( $this->search, true ); ?>" name="search" /><span class="hide-if-regex-off regex-right">/</span></label>
 815					<label for="replace"><span class="label-text">with</span> <input id="replace" type="text" placeholder="replace with&hellip;" value="<?php $this->esc_html_attr( $this->replace, true ); ?>" name="replace" /></label>
 816					<label for="regex" class="field-advanced"><input id="regex" type="checkbox" name="regex" value="1" <?php $this->checked( true, $this->regex ); ?> /> use regex</label>
 817				</div>
 818
 819				<div class="fields field-advanced hide-if-regex-off">
 820					<label for="regex_i" class="field field-advanced"><input type="checkbox" name="regex_i" id="regex_i" value="1" <?php $this->checked( true, $this->regex_i ); ?> /> <abbr title="case insensitive">i</abbr></abbr></label>
 821					<label for="regex_m" class="field field-advanced"><input type="checkbox" name="regex_m" id="regex_m" value="1" <?php $this->checked( true, $this->regex_m ); ?> /> <abbr title="multiline">m</abbr></label>
 822					<label for="regex_s" class="field field-advanced"><input type="checkbox" name="regex_s" id="regex_s" value="1" <?php $this->checked( true, $this->regex_s ); ?> /> <abbr title="dot also matches newlines">s</abbr></label>
 823					<label for="regex_x" class="field field-advanced"><input type="checkbox" name="regex_x" id="regex_x" value="1" <?php $this->checked( true, $this->regex_x ); ?> /> <abbr title="extended mode">x</abbr></label>
 824				</div>
 825
 826			</fieldset>
 827
 828			<!-- 2. db details -->
 829			<fieldset class="row row-db">
 830
 831				<h1>db details</h1>
 832
 833				<?php $this->get_errors( 'environment' ); ?>
 834
 835				<?php $this->get_errors( 'db' ); ?>
 836
 837				<div class="fields fields-small">
 838
 839					<div class="field field-short">
 840						<label for="name">name</label>
 841						<input id="name" name="name" type="text" value="<?php $this->esc_html_attr( $this->name, true ); ?>" />
 842					</div>
 843
 844					<div class="field field-short">
 845						<label for="user">user</label>
 846						<input id="user" name="user" type="text" value="<?php $this->esc_html_attr( $this->user, true ); ?>" />
 847					</div>
 848
 849					<div class="field field-short">
 850						<label for="pass">pass</label>
 851						<input id="pass" name="pass" type="text" value="<?php $this->esc_html_attr( $this->pass, true ); ?>" />
 852					</div>
 853
 854					<div class="field field-short">
 855						<label for="host">host</label>
 856						<input id="host" name="host" type="text" value="<?php $this->esc_html_attr( $this->host, true ); ?>" />
 857					</div>
 858
 859				</div>
 860
 861			</fieldset>
 862
 863			<!-- 3. tables -->
 864			<fieldset class="row row-tables">
 865
 866				<h1>tables</h1>
 867
 868				<?php $this->get_errors( 'tables' ); ?>
 869
 870				<div class="fields">
 871
 872					<div class="field radio">
 873						<label for="all_tables">
 874							<input id="all_tables" name="use_tables" value="all" type="radio" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> <?php $this->checked( true, empty( $this->tables ) ); ?> />
 875							all tables
 876						</label>
 877					</div>
 878
 879					<div class="field radio">
 880						<label for="subset_tables">
 881							<input id="subset_tables" name="use_tables" value="subset" type="radio" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> <?php $this->checked( false, empty( $this->tables ) ); ?> />
 882							select tables
 883						</label>
 884					</div>
 885
 886					<div class="field table-select hide-if-js"><?php $this->table_select(); ?></div>
 887
 888				</div>
 889
 890				<div class="fields field-advanced">
 891
 892					<div class="field field-advanced field-medium">
 893						<label for="exclude_cols">columns to exclude (optional, comma separated)</label>
 894						<input id="exclude_cols" type="text" name="exclude_cols" value="<?php $this->esc_html_attr( implode( ',', $this->get( 'exclude_cols' ) ) ) ?>" placeholder="eg. guid" />
 895					</div>
 896					<div class="field field-advanced field-medium">
 897						<label for="include_cols">columns to include only (optional, comma separated)</label>
 898						<input id="include_cols" type="text" name="include_cols" value="<?php $this->esc_html_attr( implode( ',', $this->get( 'include_cols' ) ) ) ?>" placeholder="eg. post_content, post_excerpt" />
 899					</div>
 900
 901				</div>
 902
 903			</fieldset>
 904
 905			<!-- 4. results -->
 906			<fieldset class="row row-results">
 907
 908				<h1>actions</h1>
 909
 910				<?php $this->get_errors( 'results' ); ?>
 911
 912				<div class="fields">
 913
 914					<span class="submit-group">
 915						<input type="submit" name="submit[update]" value="update details" />
 916
 917						<input type="submit" name="submit[dryrun]" value="dry run" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> class="db-required" />
 918
 919						<input type="submit" name="submit[liverun]" value="live run" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> class="db-required" />
 920
 921						<span class="separator">/</span>
 922					</span>
 923
 924					<span class="submit-group">
 925						<?php if ( in_array( 'InnoDB', $this->get( 'engines' ) ) ) { ?>
 926						<input type="submit" name="submit[innodb]" value="convert to innodb" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> class="db-required secondary field-advanced" />
 927						<?php } ?>
 928
 929						<input type="submit" name="submit[utf8]" value="convert to utf8 unicode" <?php if ( ! $this->db_valid() ) echo 'disabled="disabled"'; ?> class="db-required secondary field-advanced" />
 930
 931					</span>
 932
 933				</div>
 934
 935				<?php $this->get_report(); ?>
 936
 937			</fieldset>
 938
 939
 940			<!-- 5. branding -->
 941			<section class="row row-delete">
 942
 943				<h1>delete</h1>
 944
 945				<div class="fields">
 946					<p>
 947						<input type="submit" name="submit[delete]" value="delete me" />
 948						Once you&rsquo;re done click the <strong>delete me</strong> button to secure your server
 949					</p>
 950				</div>
 951
 952			</section>
 953
 954		</form>
 955
 956		<section class="help">
 957
 958			<h1 class="branding">interconnect/it</h1>
 959
 960			<h2>Safe Search and Replace on Database with Serialized Data v3.0.0</h2>
 961
 962			<p>This developer/sysadmin tool carries out search/replace functions on MySQL DBs and can handle serialised PHP Arrays and Objects.</p>
 963
 964			<p><strong class="red">WARNINGS!</strong>
 965			Ensure data is backed up.
 966			We take no responsibility for any damage caused by this script or its misuse.
 967			DB Connection Settings are auto-filled when WordPress or Drupal is detected but can be confused by commented out settings so CHECK!
 968			There is NO UNDO!
 969			Be careful running this script on a production server.</p>
 970
 971			<h3>Don't Forget to Remove Me!</h3>
 972
 973			<p>Delete this utility from your
 974			server after use by clicking the 'delete me' button. It represents a major security threat to your database if
 975			maliciously used.</p>
 976
 977			<p>If you have feedback or want to contribute to this script click the delete button to find out how.</p>
 978
 979			<p><em>We don't put links on the search replace UI itself to avoid seeing URLs for the script in our access logs.</em></p>
 980
 981			<h3>Again, use Of This Script Is Entirely At Your Own Risk</h3>
 982
 983			<p>The easiest and safest way to use this script is to copy your site's files and DB to a new location.
 984			You then, if required, fix up your .htaccess and wp-config.php appropriately.  Once
 985			done, run this script, select your tables (in most cases all of them) and then
 986			enter the search replace strings.  You can press back in your browser to do
 987			this several times, as may be required in some cases.</p>
 988
 989		</section>
 990
 991		<?php
 992	}
 993
 994	public function deleted() {
 995
 996		// obligatory marketing!
 997		// seriously though it's good stuff
 998		?>
 999
1000		<!-- 1. branding -->
1001		<section class="row row-branding">
1002
1003			<h1><a href="http://interconnectit.com/" target="_blank">interconnect<span>/</span><strong>it</strong></a></h1>
1004
1005			<?php $this->get_errors( 'delete' ); ?>
1006
1007			<div class="content">
1008				<p>Thanks for using our search/replace tool! We&rsquo;d really appreciate it if you took a
1009					minute to join our mailing list and check out some of our other products.</p>
1010			</div>
1011
1012		</section>
1013
1014		<!-- 2. subscribe -->
1015		<section class="row row-subscribe">
1016
1017			<h1>newsletter</h1>
1018
1019			<form action="http://interconnectit.us2.list-manage.com/subscribe/post" method="POST" class="fields fields-small">
1020				<input type="hidden" name="u" value="08ec797202866aded7b2619b2">
1021				<input type="hidden" name="id" value="538abe0a97">
1022
1023				<div id="mergeTable" class="mergeTable">
1024
1025					<div class="mergeRow dojoDndItem mergeRow-email field field-short" id="mergeRow-0">
1026						<label for="MERGE0"><strong>email address</strong> <span class="asterisk">*</span></label>
1027						<input type="email" autocapitalize="off" autocorrect="off" name="MERGE0" id="MERGE0" size="25" value="">
1028					</div>
1029
1030					<div class="mergeRow dojoDndItem mergeRow-text field field-short" id="mergeRow-1">
1031						<label for="MERGE1">first name</label>
1032						<input type="text" name="MERGE1" id="MERGE1" size="25" value="">
1033					</div>
1034
1035					<div class="mergeRow dojoDndItem mergeRow-text field field-short" id="mergeRow-2">
1036						<label for="MERGE2">last name</label>
1037						<input type="text" name="MERGE2" id="MERGE2" size="25" value="">
1038					</div>
1039
1040					<div class="submit_container field field-short">
1041						<br />
1042						<input type="submit" name="submit" value="subscribe">
1043					</div>
1044
1045				</div>
1046			</form>
1047
1048		</section>
1049
1050		<!-- 3. contribute -->
1051		<section class="row row-contribute">
1052
1053			<h1>contribute</h1>
1054
1055			<div class="content">
1056
1057				<p>Got suggestions? Found a bug? Want to contribute code? <a href="https://github.com/interconnectit/search-replace-db">Join us on Github!</a></p>
1058
1059			</div>
1060
1061		</section>
1062
1063		<section class="row row-blog">
1064
1065			<h1>blogs</h1>
1066
1067			<div class="content">
1068				<p><a href="http://interconnectit.com/blog/" target="_blank">We couldn't load our blog feed for some reason so here's a link instead!</a></p>
1069			</div>
1070
1071		</section>
1072
1073		<!-- 5. products -->
1074		<section class="row row-products">
1075
1076			<h1>products</h1>
1077
1078			<div class="content">
1079				<p><a href="http://interconnectit.com/products/" target="_blank">We couldn't load our product feed for some reason so here's a link instead!</a></p>
1080			</div>
1081
1082		</section>
1083
1084
1085
1086		<?php
1087
1088	}
1089
1090	public function html( $body ) {
1091
1092		// html classes
1093		$classes = array( 'no-js' );
1094		$classes[] = $this->regex ? 'regex-on' : 'regex-off';
1095
1096		?><!DOCTYPE html>
1097<html class="<?php echo implode( ' ', $classes ); ?>">
1098	<head>
1099		<script>var h = document.getElementsByTagName('html')[0];h.className = h.className.replace('no-js', 'js');</script>
1100
1101		<title>interconnect/it : search replace db</title>
1102
1103		<?php $this->css(); ?>
1104		<?php $this->js(); ?>
1105
1106	</head>
1107	<body>
1108
1109		<?php $this->$body(); ?>
1110
1111
1112	</body>
1113</html>
1114		<?php
1115	}
1116
1117
1118
1119	public function css() {
1120		?>
1121		<style type="text/css">
1122* { margin: 0; padding: 0; }
1123
1124::-webkit-input-placeholder { /* WebKit browsers */
1125    color:    #999;
1126}
1127:-moz-placeholder { /* Mozilla Firefox 4 to 18 */
1128    color:    #999;
1129}
1130::-moz-placeholder { /* Mozilla Firefox 19+ */
1131    color:    #999;
1132}
1133:-ms-input-placeholder { /* Internet Explorer 10+ */
1134    color:    #999;
1135}
1136
1137.js .hide-if-js {
1138	display: none;
1139}
1140.no-js .hide-if-nojs {
1141	display: none;
1142}
1143
1144.regex-off .hide-if-regex-off {
1145	display: none;
1146}
1147.regex-on .hide-if-regex-on {
1148	display: none;
1149}
1150
1151html {
1152	background: #fff;
1153	font-size: 10px;
1154	border-top: 20px solid #de1301;
1155}
1156
1157body {
1158	font-family: 'Gill Sans MT', 'Gill Sans', Calibri, sans-serif;
1159	font-size: 1.6rem;
1160}
1161
1162h2,
1163h3 {
1164	text-transform: uppercase;
1165	font-weight: normal;
1166	margin: 2.0rem 0 1.0rem;
1167}
1168
1169label {
1170	cursor: pointer;
1171}
1172
1173/*.row {
1174	background-color: rgba( 210, 0, 0, 1 );
1175	padding: 20px 40px;
1176	border: 0;
1177	overflow: hidden;
1178}
1179.row + .row {
1180	background-color: rgba( 210, 0, 0, .8 );
1181}
1182.row + .row + .row {
1183	background-color: rgba( 210, 0, 0, .6 );
1184}
1185.row + .row + .row + .row {
1186	background-color: rgba( 210, 0, 0, .4 );
1187}
1188.row + .row + .row + .row + .row {
1189	background-color: rgba( 210, 0, 0, .2 );
1190}*/
1191
1192.row {
1193	background-color: rgba( 210, 210, 210, 1 );
1194	padding: 20px 40px;
1195	border: 0;
1196	overflow: hidden;
1197}
1198.row + .row {
1199	background-color: rgba( 210, 210, 210, .8 );
1200}
1201.row + .row + .row {
1202	background-color: rgba( 210, 210, 210, .6 );
1203}
1204.row + .row + .row + .row {
1205	background-color: rgba( 210, 210, 210, .4 );
1206}
1207.row + .row + .row + .row + .row {
1208	background-color: rgba( 210, 210, 210, .2 );
1209}
1210
1211.row h1 {
1212	display: block;
1213	font-size: 4.0rem;
1214	font-weight: normal;
1215	margin: 15px 0 20px;
1216	float: left;
1217}
1218.row h1,
1219.branding {
1220	width: 260px;
1221	background:
1222		url(