summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/jetpack/vendor/automattic/jetpack-backup/src/class-helper-script-manager.php')
-rw-r--r--plugins/jetpack/vendor/automattic/jetpack-backup/src/class-helper-script-manager.php347
1 files changed, 347 insertions, 0 deletions
diff --git a/plugins/jetpack/vendor/automattic/jetpack-backup/src/class-helper-script-manager.php b/plugins/jetpack/vendor/automattic/jetpack-backup/src/class-helper-script-manager.php
new file mode 100644
index 00000000..58c54eb4
--- /dev/null
+++ b/plugins/jetpack/vendor/automattic/jetpack-backup/src/class-helper-script-manager.php
@@ -0,0 +1,347 @@
+<?php
+/**
+ * The Jetpack Backup Helper Script Manager class.
+ *
+ * @package automattic/jetpack-backup
+ */
+
+namespace Automattic\Jetpack\Backup;
+
+/**
+ * Helper_Script_Manager manages installation, deletion and cleanup of Helper Scripts
+ * to assist with backing up Jetpack Sites.
+ */
+class Helper_Script_Manager {
+
+ const TEMP_DIRECTORY = 'jetpack-temp';
+ const HELPER_HEADER = "<?php /* Jetpack Backup Helper Script */\n";
+ const EXPIRY_TIME = 8 * 3600; // 8 hours
+ const MAX_FILESIZE = 1024 * 1024; // 1 MiB
+
+ const README_LINES = array(
+ 'These files have been put on your server by Jetpack to assist with backups and restores of your site content. They are cleaned up automatically when we no longer need them.',
+ 'If you no longer have Jetpack connected to your site, you can delete them manually.',
+ 'If you have questions or need assistance, please contact Jetpack Support at https://jetpack.com/support/',
+ 'If you like to build amazing things with WordPress, you should visit automattic.com/jobs and apply to join the fun – mention this file when you apply!;',
+ );
+
+ const INDEX_FILE = '<?php // Silence is golden';
+
+ /**
+ * Installs a Helper Script, and returns its filesystem path and access url.
+ *
+ * @access public
+ * @static
+ *
+ * @param string $script_body Helper Script file contents.
+ * @return array|WP_Error Either an array containing the path and url of the helper script, or an error.
+ */
+ public static function install_helper_script( $script_body ) {
+ // Check that the script body contains the correct header.
+ if ( strncmp( $script_body, self::HELPER_HEADER, strlen( self::HELPER_HEADER ) ) !== 0 ) {
+ return new \WP_Error( 'invalid_helper', 'Invalid Helper Script header' );
+ }
+
+ // Refuse to install a Helper Script that is too large.
+ if ( strlen( $script_body ) > self::MAX_FILESIZE ) {
+ return new \WP_Error( 'invalid_helper', 'Invalid Helper Script size' );
+ }
+
+ // Replace '[wp_path]' in the Helper Script with the WordPress installation location. Allows the Helper Script to find WordPress.
+ $script_body = str_replace( '[wp_path]', addslashes( ABSPATH ), $script_body );
+
+ // Create a jetpack-temp directory for the Helper Script.
+ $temp_directory = self::create_temp_directory();
+ if ( \is_wp_error( $temp_directory ) ) {
+ return $temp_directory;
+ }
+
+ // Generate a random filename, avoid clashes.
+ $max_attempts = 5;
+ for ( $attempt = 0; $attempt < $max_attempts; $attempt++ ) {
+ $file_key = wp_generate_password( 10, false );
+ $file_name = 'jp-helper-' . $file_key . '.php';
+ $file_path = trailingslashit( $temp_directory['path'] ) . $file_name;
+
+ if ( ! file_exists( $file_path ) ) {
+ // Attempt to write helper script.
+ if ( ! self::put_contents( $file_path, $script_body ) ) {
+ if ( file_exists( $file_path ) ) {
+ unlink( $file_path );
+ }
+
+ continue;
+ }
+
+ // Always schedule a cleanup run shortly after EXPIRY_TIME.
+ \wp_schedule_single_event( time() + self::EXPIRY_TIME + 60, 'jetpack_backup_cleanup_helper_scripts' );
+
+ // Success! Figure out the URL and return the path and URL.
+ return array(
+ 'path' => $file_path,
+ 'url' => trailingslashit( $temp_directory['url'] ) . $file_name,
+ );
+ }
+ }
+
+ return new \WP_Error( 'install_faied', 'Failed to install Helper Script' );
+ }
+
+ /**
+ * Given a path, verify it looks like a helper script and then delete it if so.
+ *
+ * @access public
+ * @static
+ *
+ * @param string $path Path to Helper Script to delete.
+ * @return boolean True if the file is deleted (or does not exist).
+ */
+ public static function delete_helper_script( $path ) {
+ if ( ! file_exists( $path ) ) {
+ return true;
+ }
+
+ // Check this file looks like a JPR helper script.
+ if ( ! self::verify_file_header( $path, self::HELPER_HEADER ) ) {
+ return false;
+ }
+
+ return unlink( $path );
+ }
+
+ /**
+ * Search for Helper Scripts that are suspiciously old, and clean them out.
+ *
+ * @access public
+ * @static
+ */
+ public static function cleanup_expired_helper_scripts() {
+ self::cleanup_helper_scripts( time() - self::EXPIRY_TIME );
+ }
+
+ /**
+ * Search for and delete all Helper Scripts. Used during uninstallation.
+ *
+ * @access public
+ * @static
+ */
+ public static function delete_all_helper_scripts() {
+ self::cleanup_helper_scripts( null );
+ }
+
+ /**
+ * Search for and delete Helper Scripts. If an $expiry_time is specified, only delete Helper Scripts
+ * with an mtime older than $expiry_time. Otherwise, delete them all.
+ *
+ * @access public
+ * @static
+ *
+ * @param int|null $expiry_time If specified, only delete scripts older than $expiry_time.
+ */
+ public static function cleanup_helper_scripts( $expiry_time = null ) {
+ foreach ( self::get_install_locations() as $directory => $url ) {
+ $temp_dir = trailingslashit( $directory ) . self::TEMP_DIRECTORY;
+
+ if ( is_dir( $temp_dir ) ) {
+ // Find expired helper scripts and delete them.
+ $helper_scripts = glob( trailingslashit( $temp_dir ) . 'jp-helper-*.php' );
+ if ( is_array( $helper_scripts ) ) {
+ foreach ( $helper_scripts as $filename ) {
+ if ( null === $expiry_time || filemtime( $filename ) < $expiry_time ) {
+ self::delete_helper_script( $filename );
+ }
+ }
+ }
+
+ // Delete the directory if it's empty now.
+ self::delete_empty_helper_directory( $temp_dir );
+ }
+ }
+ }
+
+ /**
+ * Delete a helper script directory if it's empty
+ *
+ * @access public
+ * @static
+ *
+ * @param string $dir Path to Helper Script directory.
+ * @return boolean True if the directory is deleted
+ */
+ private static function delete_empty_helper_directory( $dir ) {
+ if ( ! is_dir( $dir ) ) {
+ return false;
+ }
+
+ // Tally the files in the target directory, and reject if there are too many.
+ $glob_path = trailingslashit( $dir ) . '*';
+ $dir_contents = glob( $glob_path );
+ if ( count( $dir_contents ) > 2 ) {
+ return false;
+ }
+
+ // Check that the only remaining files are a README and index.php generated by this system.
+ $allowed_files = array(
+ 'README' => self::README_LINES[0],
+ 'index.php' => self::INDEX_FILE,
+ );
+
+ foreach ( $dir_contents as $path ) {
+ $basename = basename( $path );
+ if ( ! isset( $allowed_files[ $basename ] ) ) {
+ return false;
+ }
+
+ // Verify the file starts with the expected contents.
+ if ( ! self::verify_file_header( $path, $allowed_files[ $basename ] ) ) {
+ return false;
+ }
+
+ if ( ! unlink( $path ) ) {
+ return false;
+ }
+ }
+
+ // If the directory is now empty, delete it.
+ if ( count( glob( $glob_path ) ) === 0 ) {
+ return rmdir( $dir );
+ }
+
+ return false;
+ }
+
+ /**
+ * Find an appropriate location for a jetpack-temp folder, and create one
+ *
+ * @access public
+ * @static
+ *
+ * @return WP_Error|array Array containing the url and path of the temp directory if successful, WP_Error if not.
+ */
+ private static function create_temp_directory() {
+ foreach ( self::get_install_locations() as $directory => $url ) {
+ // Check if the install location is writeable.
+ if ( ! is_writeable( $directory ) ) {
+ continue;
+ }
+
+ // Create if one doesn't already exist.
+ $temp_dir = trailingslashit( $directory ) . self::TEMP_DIRECTORY;
+ if ( ! is_dir( $temp_dir ) ) {
+ if ( ! mkdir( $temp_dir ) ) {
+ continue;
+ }
+
+ // Temp directory created. Drop a README and index.php file in there.
+ self::write_supplementary_temp_files( $temp_dir );
+ }
+
+ return array(
+ 'path' => trailingslashit( $directory ) . self::TEMP_DIRECTORY,
+ 'url' => trailingslashit( $url ) . self::TEMP_DIRECTORY,
+ );
+ }
+
+ return new \WP_Error( 'temp_directory', 'Failed to create jetpack-temp directory' );
+ }
+
+ /**
+ * Write out an index.php file and a README file for a new jetpack-temp directory.
+ *
+ * @access public
+ * @static
+ *
+ * @param string $dir Path to Helper Script directory.
+ */
+ private static function write_supplementary_temp_files( $dir ) {
+ $readme_path = trailingslashit( $dir ) . 'README';
+ self::put_contents( $readme_path, implode( "\n\n", self::README_LINES ) );
+
+ $index_path = trailingslashit( $dir ) . 'index.php';
+ self::put_contents( $index_path, self::INDEX_FILE );
+ }
+
+ /**
+ * Write a file to the specified location with the specified contents.
+ *
+ * @access private
+ * @static
+ *
+ * @param string $file_path Path to write to.
+ * @param string $contents File contents to write.
+ * @return boolean True if successfully written.
+ */
+ private static function put_contents( $file_path, $contents ) {
+ global $wp_filesystem;
+
+ if ( ! function_exists( '\\WP_Filesystem' ) ) {
+ require_once ABSPATH . 'wp-admin/includes/file.php';
+ }
+
+ if ( ! \WP_Filesystem() ) {
+ return false;
+ }
+
+ return $wp_filesystem->put_contents( $file_path, $contents );
+ }
+
+ /**
+ * Checks that a file exists, is readable, and has the expected header.
+ *
+ * @access private
+ * @static
+ *
+ * @param string $file_path File to verify.
+ * @param string $expected_header Header that the file should have.
+ * @return boolean True if the file exists, is readable, and the header matches.
+ */
+ private static function verify_file_header( $file_path, $expected_header ) {
+ global $wp_filesystem;
+
+ if ( ! function_exists( '\\WP_Filesystem' ) ) {
+ require_once ABSPATH . 'wp-admin/includes/file.php';
+ }
+
+ if ( ! \WP_Filesystem() ) {
+ return false;
+ }
+
+ // Verify the file exists and is readable.
+ if ( ! $wp_filesystem->exists( $file_path ) || ! $wp_filesystem->is_readable( $file_path ) ) {
+ return false;
+ }
+
+ // Verify that the file isn't too big or small.
+ $file_size = $wp_filesystem->size( $file_path );
+ if ( $file_size < strlen( $expected_header ) || $file_size > self::MAX_FILESIZE ) {
+ return false;
+ }
+
+ // Read the file and verify its header.
+ $contents = $wp_filesystem->get_contents( $file_path );
+ return ( strncmp( $contents, $expected_header, strlen( $expected_header ) ) === 0 );
+ }
+
+ /**
+ * Gets an associative array of possible places to install a jetpack-temp directory, along with the URL to access each.
+ *
+ * @access private
+ * @static
+ *
+ * @return array Array, with keys specifying the full path of install locations, and values with the equivalent URL.
+ */
+ public static function get_install_locations() {
+ // Include WordPress root and wp-content.
+ $install_locations = array(
+ \ABSPATH => \get_site_url(),
+ \WP_CONTENT_DIR => \WP_CONTENT_URL,
+ );
+
+ // Include uploads folder.
+ $upload_dir_info = \wp_upload_dir();
+ $install_locations[ $upload_dir_info['basedir'] ] = $upload_dir_info['baseurl'];
+
+ return $install_locations;
+ }
+
+}