: str_replace(): Passing null to parameter #2 ($replace) of type array|string is deprecated in
require_once(dirname(__FILE__) . '/wfUtils.php');
//Possible responses from `addIssue`
const ISSUE_UPDATED = 'u';
const ISSUE_DUPLICATE = 'd';
const ISSUE_IGNOREP = 'ip';
const ISSUE_IGNOREC = 'ic';
//Possible status message states
const STATUS_NONE = 'n'; //Default state before running
const STATUS_SKIPPED = 's'; //The scan job was skipped because it didn't need to run
const STATUS_IGNORED = 'i'; //The scan job found an issue, but it matched an entry in the ignore list
const STATUS_PROBLEM = 'p'; //The scan job found an issue
const STATUS_SECURE = 'r'; //The scan job found no issues
const STATUS_FAILED = 'f'; //The scan job failed
const STATUS_SUCCESS = 'c'; //The scan job succeeded
const STATUS_PAIDONLY = 'x';
//Possible scan failure types
const SCAN_FAILED_GENERAL = 'general';
const SCAN_FAILED_TIMEOUT = 'timeout';
const SCAN_FAILED_DURATION_REACHED = 'duration';
const SCAN_FAILED_VERSION_CHANGE = 'versionchange';
const SCAN_FAILED_FORK_FAILED = 'forkfailed';
const SCAN_FAILED_CALLBACK_TEST_FAILED = 'callbackfailed';
const SCAN_FAILED_START_TIMEOUT = 'starttimeout';
const SCAN_FAILED_API_SSL_UNAVAILABLE = 'sslunavailable';
const SCAN_FAILED_API_CALL_FAILED = 'apifailed';
const SCAN_FAILED_API_INVALID_RESPONSE = 'apiinvalid';
const SCAN_FAILED_API_ERROR_RESPONSE = 'apierror';
const SEVERITY_MEDIUM = 50;
const SEVERITY_HIGH = 75;
const SEVERITY_CRITICAL = 100;
const SCAN_STATUS_UPDATE_INTERVAL = 10; //Seconds
//Properties that are serialized on sleep:
private $updateCalled = false;
private $issuesTable = '';
private $pendingIssuesTable = '';
private $newIssues = array();
public $totalIgnoredIssues = 0;
private $totalIssuesBySeverity = array();
public static $issueSeverities = array(
'checkGSB' => wfIssues::SEVERITY_CRITICAL,
'checkSpamIP' => wfIssues::SEVERITY_HIGH,
'spamvertizeCheck' => wfIssues::SEVERITY_CRITICAL,
'commentBadURL' => wfIssues::SEVERITY_LOW,
'postBadTitle' => wfIssues::SEVERITY_HIGH,
'postBadURL' => wfIssues::SEVERITY_HIGH,
'file' => wfIssues::SEVERITY_CRITICAL,
'timelimit' => wfIssues::SEVERITY_HIGH,
'checkHowGetIPs' => wfIssues::SEVERITY_HIGH,
'diskSpace' => wfIssues::SEVERITY_HIGH,
'wafStatus' => wfIssues::SEVERITY_CRITICAL,
'configReadable' => wfIssues::SEVERITY_CRITICAL,
'wfPluginVulnerable' => wfIssues::SEVERITY_HIGH,
'coreUnknown' => wfIssues::SEVERITY_HIGH,
'easyPasswordWeak' => wfIssues::SEVERITY_HIGH,
'knownfile' => wfIssues::SEVERITY_HIGH,
'optionBadURL' => wfIssues::SEVERITY_HIGH,
'publiclyAccessible' => wfIssues::SEVERITY_HIGH,
'suspiciousAdminUsers' => wfIssues::SEVERITY_HIGH,
'wfPluginAbandoned' => wfIssues::SEVERITY_MEDIUM,
'wfPluginRemoved' => wfIssues::SEVERITY_CRITICAL,
'wfPluginUpgrade' => wfIssues::SEVERITY_MEDIUM,
'wfThemeUpgrade' => wfIssues::SEVERITY_MEDIUM,
'wfUpgradeError' => wfIssues::SEVERITY_MEDIUM,
'wfUpgrade' => wfIssues::SEVERITY_HIGH,
'wpscan_directoryList' => wfIssues::SEVERITY_HIGH,
'wpscan_fullPathDiscl' => wfIssues::SEVERITY_HIGH,
public static function validIssueTypes() {
return array('checkHowGetIPs', 'checkSpamIP', 'commentBadURL', 'configReadable', 'coreUnknown', 'database', 'diskSpace', 'wafStatus', 'easyPassword', 'file', 'geoipSupport', 'knownfile', 'optionBadURL', 'postBadTitle', 'postBadURL', 'publiclyAccessible', 'spamvertizeCheck', 'suspiciousAdminUsers', 'timelimit', 'wfPluginAbandoned', 'wfPluginRemoved', 'wfPluginUpgrade', 'wfPluginVulnerable', 'wfThemeUpgrade', 'wfUpgradeError', 'wfUpgrade', 'wpscan_directoryList', 'wpscan_fullPathDiscl', 'skippedPaths');
public static function statusPrep(){
wfConfig::set_ser('wfStatusStartMsgs', array());
wordfence::status(10, 'info', "SUM_PREP:Preparing a new scan.");
wfIssues::updateScanStillRunning();
public static function statusStart($message) {
$statusStartMsgs = wfConfig::get_ser('wfStatusStartMsgs', array());
$statusStartMsgs[] = $message;
wfConfig::set_ser('wfStatusStartMsgs', $statusStartMsgs);
wordfence::status(10, 'info', 'SUM_START:' . $message);
wfIssues::updateScanStillRunning();
return count($statusStartMsgs) - 1;
public static function statusEnd($index, $state) {
$statusStartMsgs = wfConfig::get_ser('wfStatusStartMsgs', array());
if ($state == self::STATUS_SKIPPED) {
wordfence::status(10, 'info', 'SUM_ENDSKIPPED:' . $statusStartMsgs[$index]);
else if ($state == self::STATUS_IGNORED) {
wordfence::status(10, 'info', 'SUM_ENDIGNORED:' . $statusStartMsgs[$index]);
else if ($state == self::STATUS_PROBLEM) {
wordfence::status(10, 'info', 'SUM_ENDBAD:' . $statusStartMsgs[$index]);
else if ($state == self::STATUS_SECURE) {
wordfence::status(10, 'info', 'SUM_ENDOK:' . $statusStartMsgs[$index]);
else if ($state == self::STATUS_FAILED) {
wordfence::status(10, 'info', 'SUM_ENDFAILED:' . $statusStartMsgs[$index]);
else if ($state == self::STATUS_SUCCESS) {
wordfence::status(10, 'info', 'SUM_ENDSUCCESS:' . $statusStartMsgs[$index]);
wfIssues::updateScanStillRunning();
$statusStartMsgs[$index] = '';
wfConfig::set_ser('wfStatusStartMsgs', $statusStartMsgs);
public static function statusEndErr() {
$statusStartMsgs = wfConfig::get_ser('wfStatusStartMsgs', array());
for ($i = 0; $i < count($statusStartMsgs); $i++) {
if (empty($statusStartMsgs[$i]) === false) {
wordfence::status(10, 'info', 'SUM_ENDERR:' . $statusStartMsgs[$i]);
$statusStartMsgs[$i] = '';
wfIssues::updateScanStillRunning();
public static function statusPaidOnly($message) {
wordfence::status(10, 'info', "SUM_PAIDONLY:" . $message);
wfIssues::updateScanStillRunning();
public static function statusDisabled($message) {
wordfence::status(10, 'info', "SUM_DISABLED:" . $message);
wfIssues::updateScanStillRunning();
public static function updateScanStillRunning($running = true) {
if ($timestamp - $lastUpdate < self::SCAN_STATUS_UPDATE_INTERVAL)
$lastUpdate = $timestamp;
wfConfig::set('wf_scanLastStatusTime', $timestamp);
* Returns false if the scan has not been detected as failed. If it has, returns a constant corresponding to the reason.
public static function hasScanFailed() {
$lastStatusUpdate = self::lastScanStatusUpdate();
if ($lastStatusUpdate !== false && wfScanner::shared()->isRunning()) {
$threshold = WORDFENCE_SCAN_FAILURE_THRESHOLD;
if (time() - $lastStatusUpdate > $threshold) {
return self::SCAN_FAILED_TIMEOUT;
$scanStartAttempt = wfConfig::get('scanStartAttempt', 0);
if ($scanStartAttempt && time() - $scanStartAttempt > WORDFENCE_SCAN_START_FAILURE_THRESHOLD) {
return self::SCAN_FAILED_START_TIMEOUT;
$recordedFailure = wfConfig::get('lastScanFailureType');
switch ($recordedFailure) {
case self::SCAN_FAILED_GENERAL:
case self::SCAN_FAILED_DURATION_REACHED:
case self::SCAN_FAILED_VERSION_CHANGE:
case self::SCAN_FAILED_FORK_FAILED:
case self::SCAN_FAILED_CALLBACK_TEST_FAILED:
case self::SCAN_FAILED_API_SSL_UNAVAILABLE:
case self::SCAN_FAILED_API_CALL_FAILED:
case self::SCAN_FAILED_API_INVALID_RESPONSE:
case self::SCAN_FAILED_API_ERROR_RESPONSE:
* Returns false if the scan has not been detected as timed out. If it has, it returns the timestamp of the last status update.
public static function lastScanStatusUpdate() {
if (wfConfig::get('wf_scanLastStatusTime', 0) === 0) {
$threshold = WORDFENCE_SCAN_FAILURE_THRESHOLD;
return (time() > wfConfig::get('wf_scanLastStatusTime', 0) + $threshold) ? wfConfig::get('wf_scanLastStatusTime', 0) : false;
* Returns the singleton wfIssues.
public static function shared() {
$_issues = new wfIssues();
public function __sleep(){ //Same order here as vars above
return array('updateCalled', 'issuesTable', 'pendingIssuesTable', 'maxIssues', 'newIssues', 'totalIssues', 'totalIgnoredIssues', 'totalIssuesBySeverity');
public function __construct(){
$this->issuesTable = wfDB::networkTable('wfIssues');
$this->pendingIssuesTable = wfDB::networkTable('wfPendingIssues');
$this->maxIssues = wfConfig::get('scan_maxIssues', 0);
public function __wakeup(){
public function addIssue($type, $severity, $ignoreP, $ignoreC, $shortMsg, $longMsg, $templateData, $alreadyHashed = false) {
return $this->_addIssue('issue', $type, $severity, $ignoreP, $ignoreC, $shortMsg, $longMsg, $templateData, $alreadyHashed);
public function addPendingIssue($type, $severity, $ignoreP, $ignoreC, $shortMsg, $longMsg, $templateData) {
return $this->_addIssue('pending', $type, $severity, $ignoreP, $ignoreC, $shortMsg, $longMsg, $templateData);
* @param string $group The issue type (e.g., issue or pending
* @param string $ignoreP string to compare against for permanent ignores
* @param string $ignoreC string to compare against for ignoring until something changes
* @param string $shortMsg
* @param array $templateData
* @param bool $alreadyHashed If true, don't re-hash $ignoreP and $ignoreC
* @return string One of the ISSUE_ constants
private function _addIssue($group, $type, $severity, $ignoreP, $ignoreC, $shortMsg, $longMsg, $templateData, $alreadyHashed = false) {
if ($group == 'pending') {
$table = $this->pendingIssuesTable;
$table = $this->issuesTable;
$ignoreP = md5($ignoreP);
$ignoreC = md5($ignoreC);
$results = $this->getDB()->querySelect("SELECT id, status, ignoreP, ignoreC FROM {$table} WHERE (ignoreP = '%s' OR ignoreC = '%s')", $ignoreP, $ignoreC);
foreach ($results as $row) {
if ($row['status'] == 'new' && ($row['ignoreC'] == $ignoreC || $row['ignoreP'] == $ignoreP)) {
if ($type != 'file' && $type != 'database') { //Filter out duplicate new issues except for infected files because we want to see all infections even if file contents are identical
return self::ISSUE_DUPLICATE;
if ($row['status'] == 'ignoreP' && $row['ignoreP'] == $ignoreP) { $this->totalIgnoredIssues++; return self::ISSUE_IGNOREP; } //Always ignore
else if ($row['status'] == 'ignoreC' && $row['ignoreC'] == $ignoreC) { $this->totalIgnoredIssues++; return self::ISSUE_IGNOREC; } //Unchanged, ignore
else if ($row['status'] == 'ignoreC') {
$updateID = $row['id']; //Re-use the existing issue row
if ($group != 'pending') {
if (!array_key_exists($severity, $this->totalIssuesBySeverity)) {
$this->totalIssuesBySeverity[$severity] = 0;
$this->totalIssuesBySeverity[$severity]++;
if (empty($this->maxIssues) || $this->totalIssues <= $this->maxIssues)
$this->newIssues[] = array(
'tmplData' => $templateData
if ($group !== 'pending' && wfCentral::isConnected()) {
wfCentral::sendIssue(array(
$this->getDB()->queryWrite(
"UPDATE {$table} SET lastUpdated = UNIX_TIMESTAMP(), status = '%s', type = '%s', severity = %d, ignoreP = '%s', ignoreC = '%s', shortMsg = '%s', longMsg = '%s', data = '%s' WHERE id = %d",
serialize($templateData),
return self::ISSUE_UPDATED;
$this->getDB()->queryWrite("INSERT INTO {$table} (time, lastUpdated, status, type, severity, ignoreP, ignoreC, shortMsg, longMsg, data) VALUES (UNIX_TIMESTAMP(), UNIX_TIMESTAMP(), '%s', '%s', %d, '%s', '%s', '%s', '%s', '%s')",
serialize($templateData));
if ($group !== 'pending' && wfCentral::isConnected()) {
wfCentral::sendIssue(array(
'id' => $wpdb->insert_id,
return self::ISSUE_ADDED;
public function deleteIgnored(){
if (wfCentral::isConnected()) {
$result = $this->getDB()->querySelect("SELECT id from " . $this->issuesTable . " where status='ignoreP' or status='ignoreC'");
foreach ($result as $row) {
wfCentral::deleteIssues($issues);
$this->getDB()->queryWrite("delete from " . $this->issuesTable . " where status='ignoreP' or status='ignoreC'");
public function deleteNew($types = null) {
if (wfCentral::isConnected()) {
wfCentral::deleteNewIssues();
$this->getDB()->queryWrite("DELETE FROM {$this->issuesTable} WHERE status = 'new'");
if (wfCentral::isConnected()) {
wfCentral::deleteIssueTypes($types, 'new');
$query = "DELETE FROM {$this->issuesTable} WHERE status = 'new' AND type IN (" . implode(',', array_fill(0, count($types), "'%s'")) . ")";
array_unshift($types, $query);
call_user_func_array(array($this->getDB(), 'queryWrite'), $types);
public function ignoreAllNew(){
if (wfCentral::isConnected()) {
$issues = $this->getDB()->querySelect('SELECT * FROM ' . $this->issuesTable . ' WHERE status=\'new\'');
wfCentral::sendIssues($issues);
$this->getDB()->queryWrite("update " . $this->issuesTable . " set status='ignoreC' where status='new'");
public function emailNewIssues($timeLimitReached = false, $scanController = false){
$level = wfConfig::getAlertLevel();
$emails = wfConfig::getAlertEmails();
$shortSiteURL = preg_replace('/^https?:\/\//i', '', site_url());
$subject = "[Wordfence Alert] Problems found on $shortSiteURL";
if(sizeof($emails) < 1){ return; }
if($level < 1){ return; }
foreach ($this->totalIssuesBySeverity as $issueSeverity => $totalIssuesBySeverity) {
if ($issueSeverity >= $level && $totalIssuesBySeverity > 0) {
$emailedIssues = wfConfig::get_ser('emailedIssuesList', array());
if(! is_array($emailedIssues)){
$emailedIssues = array();
$overflowCount = $this->totalIssues - count($this->newIssues);
$previousIssues = array();
foreach($this->newIssues as $newIssue){
foreach($emailedIssues as $emailedIssue){
if($newIssue['ignoreP'] == $emailedIssue['ignoreP'] || $newIssue['ignoreC'] == $emailedIssue['ignoreC']){
$previousIssues[] = $newIssue;
$finalIssues[] = $newIssue;
if(sizeof($finalIssues) < 1){ return; }
$this->newIssues = array();
foreach($finalIssues as $i){
$emailedIssues[] = array( 'ignoreC' => $i['ignoreC'], 'ignoreP' => $i['ignoreP'] );
if (!array_key_exists($i['severity'], $totals)) {
$totals[$i['severity']] = 0;
$totals[$i['severity']]++;
wfConfig::set_ser('emailedIssuesList', $emailedIssues, false, wfConfig::DONT_AUTOLOAD);
foreach ($totals as $issueSeverity => $totalIssuesBySeverity) {
if ($issueSeverity >= $level && $totalIssuesBySeverity > 0) {
$content = wfUtils::tmpl('email_newIssues.php', array(
'isPaid' => wfConfig::get('isPaid'),
'issues' => $finalIssues,
'previousIssues' => $previousIssues,
'issuesNotShown' => $overflowCount,
'adminURL' => get_admin_url(),
'timeLimitReached' => $timeLimitReached,
'scanController' => ($scanController ? $scanController : wfScanner::shared()),
foreach ($emails as $email) {
$uniqueContent = str_replace('<!-- ##UNSUBSCRIBE## -->', wp_kses(sprintf(__('No longer an administrator for this site? <a href="%s" target="_blank">Click here</a> to stop receiving security alerts.', 'wordfence'), wfUtils::getSiteBaseURL() . '?_wfsf=removeAlertEmail&jwt=' . wfUtils::generateJWT(array('email' => $email))), array('a'=>array('href'=>array(), 'target'=>array()))), $content);
wp_mail($email, $subject, $uniqueContent, 'Content-type: text/html');
public function clearEmailedStatus($issues) {