2022-02-25 19:54:06 +03:00
< ? 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 ()
{
2022-02-25 20:32:24 +03:00
$patch_log_file_path = __DIR__ . '/patch-log.php' ;
2022-02-25 19:54:06 +03:00
2022-02-25 20:32:24 +03:00
if ( ! file_exists ( $patch_log_file_path ))
2022-02-25 19:54:06 +03:00
{
return [];
}
2022-02-25 20:32:24 +03:00
return require $patch_log_file_path ;
2022-02-25 19:54:06 +03:00
}
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 ;
}
2022-02-25 20:32:24 +03:00
function update_patch_log ( $applied_patches , $new_patches )
2022-02-25 19:54:06 +03:00
{
$persisted_patches = array_merge ( $applied_patches , $new_patches );
2022-02-25 20:32:24 +03:00
$patch_log_file_path = __DIR__ . '/patch-log.php' ;
$contents = '
< ? php
return ' . preg_replace("/[0-9]+ \=\>/i", ' ', var_export($persisted_patches, TRUE)) . ' ; ' ;
2022-02-25 19:54:06 +03:00
2022-02-25 20:32:24 +03:00
$result = file_put_contents ( $patch_log_file_path , $contents );
2022-02-25 19:54:06 +03:00
if ( $result === FALSE )
{
2022-02-25 20:32:24 +03:00
die ( 'Could not write the local "patch-log.php" file, please check the file permissions make sure it is writable: ' . $patch_log_file_path );
2022-02-25 19:54:06 +03:00
}
}
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 ;
2022-02-25 20:32:24 +03:00
update_patch_log ( $applied_patches , $new_patches );
2022-02-25 19:54:06 +03:00
}