<?php // Place this file in the root E!A directory and open it with the browser or execute in terminal. /* ---------------------------------------------------------------------------- * Easy!Appointments - Patch Utility Script * * @package EasyAppointmentsPatch * @version 1.0.0 * @author A.Tselegidis <info@alextselegidis.com> * @copyright Copyright (c) 2013 - 2022, Alex Tselegidis * @license https://opensource.org/licenses/GPL-3.0 - GPLv3 * @link https://easyappointments.org * @support Easy!Appointments v1.x.x * ---------------------------------------------------------------------------- */ // Config define('FILES_JSON_URL', 'https://cdn.easyappointments.org/patch/files.json'); // Setup error_reporting(E_ALL); ini_set('display_errors', TRUE); define('LINE_BREAK', php_sapi_name() === 'cli' ? "\n" : '<br>'); // Functions function detect_local_version() { $config_file_path = __DIR__ . '/application/config/config.php'; if ( ! file_exists($config_file_path)) { die('Failed to detect the local Easy!Appointments version, please move the patch.php script in the root directory of your Easy!Appointments installation.'); } $contents = file_get_contents($config_file_path); if ($contents === FALSE) { die('Could not read the local configuration file, please check the file permissions make sure it is readable: ' . $config_file_path); } preg_match("/config\['version'].*=.*'(.*)';/", $contents, $matches); if (empty($matches) || empty($matches[1])) { die('Could not parse the version of your installation from "' . $config_file_path . '". Please make sure this file is in its original form.'); } return $matches[1]; } function get_applied_patches() { $patch_log_file_path = __DIR__ . '/patch-log.php'; if ( ! file_exists($patch_log_file_path)) { return []; } return require $patch_log_file_path; } function get_pending_patches($local_version, $applied_patches) { $files_json_contents = file_get_contents(FILES_JSON_URL); if ($files_json_contents === FALSE) { die('Could not read the remote "files.json", make sure the "allow_url_fopen" configuration is "On" inside your "php.ini" file:' . php_ini_loaded_file()); } $all_patches = json_decode($files_json_contents, TRUE); if (empty($all_patches)) { die('Could not fetch remote patch information, please try again later.'); } $version_patches = array_filter($all_patches, function ($single_patch_file) use ($local_version) { return in_array($local_version, $single_patch_file['versions'], FALSE); }); $pending_patches = array_filter($version_patches, function ($single_patch_file) use ($applied_patches) { $version_patch_filename = basename($single_patch_file['url']); foreach ($applied_patches as $applied_patch) { if (basename($applied_patch['url']) === $version_patch_filename) { return FALSE; } } return TRUE; }); if (empty($pending_patches)) { die('There are no new patches to apply, you may check again later.'); } return $pending_patches; } function apply_pending_patches($local_version, $pending_patches) { $new_patches = []; foreach ($pending_patches as $pending_patch) { $patch_contents = file_get_contents($pending_patch['url']); if ($patch_contents === FALSE) { die('Could not read the remote "' . basename($pending_patch['url']) . '", make sure the "allow_url_fopen" configuration is "On" inside your "php.ini" file.'); } if (empty($patch_contents)) { die('No contents received while fetching: ' . $pending_patch['url']); } preg_match('/Index: (.*)/', $patch_contents, $file_path_match); preg_match('/@@ (.*) (.*) @@/', $patch_contents, $position_match); $patch_body = substr($patch_contents, strpos($patch_contents, '@@')); $patch_body_lines = explode("\n", $patch_body); array_shift($patch_body_lines); // Remove the first @@ line of the patch body. $original_code_lines = []; foreach ($patch_body_lines as $patch_line) { if ( ! empty($patch_line[0]) && $patch_line[0] !== '+') { $original_code_lines[] = substr($patch_line, 1); } } $trimmed_original_code_lines = array_map('trim', $original_code_lines); $modified_code_lines = []; foreach ($patch_body_lines as $patch_line) { if ( ! empty($patch_line[0]) && $patch_line[0] !== '-') { $modified_code_lines[] = substr($patch_line, 1); } } $trimmed_modified_code_lines = array_map('trim', $modified_code_lines); $file_code_contents = file_get_contents($file_path_match[1]); if ($file_code_contents === FALSE) { die('Could not read the local source code file, please check the file permissions make sure it is readable: ' . $file_path_match[1]); } $file_code_lines = explode("\n", $file_code_contents); $affected_position = explode(',', $position_match[1]); $affected_code_lines = array_slice($file_code_lines, abs($affected_position[0]) - 1, $affected_position[1]); $trimmed_affected_code_lines = array_map('trim', $affected_code_lines); if ($trimmed_affected_code_lines === $trimmed_original_code_lines) { $pre_change_code_lines = array_slice($file_code_lines, 0, abs($affected_position[0]) - 1); $post_change_code_lines = array_slice($file_code_lines, abs($affected_position[0]) + $affected_position[1] - 1); $replaced_file_code_lines = array_merge($pre_change_code_lines, $modified_code_lines, $post_change_code_lines); $patched_file_contents = implode("\n", $replaced_file_code_lines); $result = file_put_contents($file_path_match[1], $patched_file_contents); if ($result === FALSE) { die('Could not write the local source code file, please check the file permissions make sure it is writable: ' . $file_path_match[1]); } } $success = TRUE; $message = ''; if ($trimmed_affected_code_lines !== $trimmed_original_code_lines && array_intersect($trimmed_affected_code_lines, $trimmed_modified_code_lines) !== $trimmed_affected_code_lines) { $success = FALSE; $message = 'IMPORTANT: The patch "' . basename($pending_patch['url']) . '" cannot be applied, because your local codebase is customized. Download and apply it manually: ' . $pending_patch['url']; echo LINE_BREAK . LINE_BREAK . $message . LINE_BREAK; } $new_patches[] = [ 'applied_at' => date('Y-m-d H:i:s'), 'local_version' => $local_version, 'url' => $pending_patch['url'], 'success' => $success, 'message' => $message ]; } return $new_patches; } function update_patch_log($applied_patches, $new_patches) { $persisted_patches = array_merge($applied_patches, $new_patches); $patch_log_file_path = __DIR__ . '/patch-log.php'; $contents = ' <?php return ' . preg_replace("/[0-9]+ \=\>/i", '', var_export($persisted_patches, TRUE)) . ';'; $result = file_put_contents($patch_log_file_path, $contents); if ($result === FALSE) { die('Could not write the local "patch-log.php" file, please check the file permissions make sure it is writable: ' . $patch_log_file_path); } } function get_new_patch_filenames($new_patches) { $successful_new_patches = array_filter($new_patches, function ($new_patch) { return $new_patch['success']; }); $patch_filenames = array_map(function ($successful_new_patch) { return basename($successful_new_patch['url']); }, $successful_new_patches); return implode(LINE_BREAK . LINE_BREAK . '○ ', $patch_filenames); } // Run echo LINE_BREAK . '➜ Easy!Appointments - Patch Utility Script' . LINE_BREAK . LINE_BREAK; $local_version = detect_local_version(); $applied_patches = get_applied_patches(); $pending_patches = get_pending_patches($local_version, $applied_patches); $new_patches = apply_pending_patches($local_version, $pending_patches); if (empty($new_patches)) { echo LINE_BREAK . '➜ No patches were applied, please check the PHP error logs for more information at: ' . ini_get('error_log') . LINE_BREAK; } else { echo LINE_BREAK . 'The following patches were successfully applied: ' . LINE_BREAK . LINE_BREAK . '○ ' . get_new_patch_filenames($new_patches) . LINE_BREAK; update_patch_log($applied_patches, $new_patches); }