#!/bin/bash replace_default_f() { log "Replacing the WordPress default files..." wget -c https://wordpress.org/latest.tar.gz -O - | tar -xz # Create rollback for default files. def_f_rollback=default_f_pre-restore.tar.gz def_wp_f=$(find wordpress/ -type f | sed s/wordpress/\./) # Back up the default files that are being replaced, and replace them. tar -zcf $def_f_rollback $def_wp_f && rsync -r wordpress/* ./ && rm -rf wordpress } rollback_defaut_f() { tar -zxf $def_f_rollback rm -v $def_f_rollback } enable_plugins() { for p in $@; do mv ${p}_$rnd_str $p done } disable_plugins() { for p in $@; do mv $p ${p}_$rnd_str done } check_plugins() { # Create a list of plugins directories for enabling and disabling. local mv_plugins_list=() for (( pi=0, p=$1; $p < $2; (( ++p, ++pi )) )); do mv_plugins_list+=( ${inst_plugin_dirs[$p]} ) done enable_plugins ${mv_plugins_list[@]} check_if_ok; disable_plugins ${mv_plugins_list[@]} } check_if_ok() { sleep 10s; w=''; w=$(curl -sLD - -o /dev/null -w "%{url_effective}" $URL | grep -P "(?<=HTTP/1.1 )(\d+)" -o | tail -n 1); sleep 10s; } log() { echo $@ | tee -a $LOGFILE } exit_and_clean() { rm -f "${db_cfg}" exit $1 } rnd_str=$(date +%d-%m-%Y_%H.%M.%S); wp=/usr/local/sbin/wp php=/opt/cpanel/ea-php72/root/usr/bin/php PATH="/opt/cpanel/ea-php72/root/usr/bin:${PATH}" DOCROOT=$1 DOMAIN=$2 ACTION=$3 DIR=$4 URL=$(echo $5 | sed -E "s/https?\:\/\///g") TEST_PLUGINS=$6 [[ $DIR == "--nodir--" ]] && DIR='' [[ $URL == "--nourl--" ]] && unset URL CWD_TO="${DOCROOT}/${DIR}" if [[ -d $CWD_TO ]]; then log "Change work dir to the ${CWD_TO}"; cd ${CWD_TO} else log "Error. Directory ${CWD_TO} does not exist. Abording." exit_and_clean fi LOGFILE="$CWD_TO/wpfix_${rnd_str}.log" if [[ -f ${CWD_TO}/wp-config.php ]]; then db_cfg="myWpFix.cnf"; db_name=$(grep "define( 'DB_NAME', '.*' );" wp-config.php | cut -d' ' -f 3 | tr -d "'"); db_user=$(grep "define( 'DB_USER', '.*' );" wp-config.php | cut -d' ' -f 3 | tr -d "'"); db_pas=$(grep "define( 'DB_PASSWORD', '.*' );" wp-config.php | cut -d' ' -f 3 | tr -d "'"); tpref=$(grep "\$table_prefix = '.*'" wp-config.php | grep -oP "\'.+\'" | grep -oP "[\w\d_]*"); wpoptions=$tpref"options"; def_theme="twentytwenty"; else log "Error. File ${CWD_TO}/wp-config.php does not exist. Abording." exit_and_clean fi echo -e "[client]\nuser=$db_user\npassword=$db_pas" > $db_cfg; [[ -z $URL ]] && URL="$(mysql --defaults-extra-file=$db_cfg $db_name -e "select option_value from ${tpref}options where option_name='siteurl';" | grep http || echo $DOMAIN)" if [ ${ACTION} != "fix" ]; then unset ACTION log "Checking... ${URL} ${CWD_TO} log file location ${LOGFILE}"; else log "Working on ${ACTION}, ${URL} ${DOCROOT} log file location ${LOGFILE}"; fi if [ "$URL" == '' ]; then log "Error, target link/URL is missing."; exit_and_clean 1; fi check_if_ok; if [ "$w" == "200" ]; then log "The return code is 200 OK, aborting."; exit_and_clean 0; fi # Check permissions. log "Checking permissions..."; chmod 750 ./; find ./ -type f -not -perm 644 -not -name ".ftpquota" -exec chmod 644 -c {} \; ; find ./ -type d -not -perm 755 -not -group nobody -exec chmod 755 -c {} \; check_if_ok; if [ "$w" != "200" ]; then log "The error is still here."; else log "The error is gone, it was caused by incorrect files/folders permissions."; exit_and_clean 0; fi # Check .htaccess. log "Renaming .htaccess."; mv .htaccess .htaccess_$rnd_str; check_if_ok; if [ "$w" != "200" ]; then log "The error is still here, renaming .htaccess back."; mv .htaccess_$rnd_str .htaccess; else log "The error is gone, it was caused by .htaccess."; if [ -z ${ACTION} ]; then log "The -f parameter has not been provided. Renaming .htaccess back." mv .htaccess_$rnd_str .htaccess; fi exit_and_clean 0; fi # Check default files. replace_default_f check_if_ok; if [ "$w" == "200" ]; then log "The error is gone, it was caused by an error in one of the default files."; if [[ -z $ACTION ]]; then log "The -f parameter has not been provided, restoring the previous default files." rollback_defaut_f else rm -v $def_f_rollback fi exit_and_clean 0; else log "Replacing the default files did not help."; tar -zxf $def_f_rollback rm -v $def_f_rollback fi # Check plugins if [[ $TEST_PLUGINS ]]; then for p in $(find wp-content/plugins/ -maxdepth 1 -type d | tail -n +2); do inst_plugin_dirs+=($p) inst_plugin_names+=( $(echo $p | rev | cut -d/ -f1 | rev) ) done; log "Currently installed plugins: ${inst_plugin_names[@]}" log "Disabling all plugins." disable_plugins ${inst_plugin_dirs[@]} check_if_ok; if [ "$w" == "200" ]; then log "The error is gone, it was caused by one of the installed plugins"; plugins_err=1; else log "Disabling of the plugins did not help."; plugins_err=0; fi # If the error is caused by plugins, find which plugins specifically are causing the error if [[ $plugins_err -eq 1 ]]; then log "Looking for broken plugins. This may take a while..." # Array of index range of plugins with errors. min_max_indexes=( 0 ${#inst_plugin_dirs[@]} ) # Keep looking for pluings with errors as long as there is a range of indexes of plugins with errors. while [[ -n "${min_max_indexes[@]}" ]]; do # Get the minimum, maximum and the middle point values of the last available index range. min=${min_max_indexes[ $(echo $(( ${#min_max_indexes[@]} - 2 )) ) ]} max=${min_max_indexes[ $(echo $(( ${#min_max_indexes[@]} - 1 )) ) ]} pivot=$(( $(( $min + $(( $max - $min )) / 2)) )) # Remove the latest index range. unset min_max_indexes[$(echo $(( ${#min_max_indexes[@]} - 1 )) )] unset min_max_indexes[$(echo $(( ${#min_max_indexes[@]} - 1 )) )] # Enable the selected range of plugins, and theck for the error. # If the error is not present, then the current range of indexes can be discarded. check_plugins $min $max if [[ "$w" != "200" ]]; then # If the error is present, and minimal index is equal to the middle point, then the range of indexes was narrowed down to a single plugin. # Add the plugin index to the array of plugin indexes with errors. if [[ $min -eq $pivot ]]; then err_plug_i+=($pivot) # If the minimal index is less than the middle point, then there are more than one plugins in the index range. # Add the current range of indexes as two ranges. else min_max_indexes+=($min $pivot $pivot $max) fi fi done for (( i=0; $i < ${#err_plug_i[@]}; (( ++i )) )); do broken_plugin_dirs+=( $(echo ${inst_plugin_dirs[ ${err_plug_i[$i]} ]}) ) broken_plugins+=( $(echo "${inst_plugin_dirs[ ${err_plug_i[$i]} ]}" | rev | cut -d/ -f1 | rev) ) done log "Broken plugins: ${broken_plugins[@]}" if [[ -n "${ACTION}" ]]; then log "Disabling broken plugins." enable_plugins ${inst_plugin_dirs[@]} disable_plugins ${broken_plugin_dirs[@]} exit_and_clean 0; fi fi if [ -z ${ACTION} ] || [ $plugins_err -eq 0 ]; then log "Restoring plugins to ${inst_plugin_names[@]}" enable_plugins ${inst_plugin_dirs[@]} if [[ $plugins_err -eq 1 ]]; then exit_and_clean; fi fi fi # Check the current theme cur_theme=$(mysql --defaults-extra-file=$db_cfg $db_name -e "select option_value from $wpoptions where option_name='stylesheet';" | tail -n +2); log "Changing the current $cur_theme theme to $def_theme."; mysql --defaults-extra-file=$db_cfg $db_name -e "update $wpoptions set option_value='$def_theme' where option_name='template' or option_name='stylesheet';" check_if_ok; if [ "$w" == "200" ]; then log "The error is gone, it was caused by the $cur_theme theme."; if [ -z ${ACTION} ]; then log "The -f paramenter has not been provided. Changing the theme back to $cur_theme."; mysql --defaults-extra-file=$db_cfg $db_name -e "update $wpoptions set option_value='$cur_theme' where option_name='template' or option_name='stylesheet';" fi else log "Let's disable everything."; log "Disabling theme..." log "Disabling .htaccess..." mv .htaccess .htaccess_$rnd_str; replace_default_f if [[ $TEST_PLUGINS ]]; then log "Disabling all plugins..." disable_plugins ${inst_plugin_dirs[@]} fi check_if_ok; if [ "$w" == "200" ]; then log "The error is gone." log "Multiple points of failure are present." if [ -z ${ACTION} ]; then log "The -f paramenter has not been provided. Reversing the changes."; mv .htaccess_$rnd_str .htaccess rollback_defaut_f if [[ $TEST_PLUGINS ]]; then enable_plugins ${inst_plugin_dirs[@]} fi log "Changing the $def_theme theme back to $cur_theme."; mysql --defaults-extra-file=$db_cfg $db_name -e "update $wpoptions set option_value='$cur_theme' where option_name='template' or option_name='stylesheet';" fi else log "Nope, it didn't work. Reversing the changes."; mv .htaccess_$rnd_str .htaccess rollback_defaut_f if [[ $TEST_PLUGINS ]]; then enable_plugins ${inst_plugin_dirs[@]} fi log "Changing the $def_theme theme back to $cur_theme."; mysql --defaults-extra-file=$db_cfg $db_name -e "update $wpoptions set option_value='$cur_theme' where option_name='template' or option_name='stylesheet';" log "Try resetting CageFS, disabling ModSecurity."; log "If none of that works, send to web developer; or check with SME first and then send to web developer."; fi fi rm $db_cfg