PluginProbe ʕ •ᴥ•ʔ
Wordfence Security – Firewall, Malware Scan, and Login Security / 7.2.5
Wordfence Security – Firewall, Malware Scan, and Login Security v7.2.5
8.2.2 8.2.1 8.2.0 3.7.1 3.7.2 3.8.1 3.8.2 3.8.3 3.8.4 3.8.5 3.8.6 3.8.7 3.8.8 3.8.9 3.9.1 4.0.1 4.0.2 4.0.3 5.0.1 5.0.2 5.0.3 5.0.4 5.0.5 5.0.6 5.0.7 5.0.8 5.0.9 5.1.1 5.1.2 5.1.4 5.1.5 5.1.6 5.1.7 5.1.8 5.1.9 5.2.1 5.2.2 5.2.3 5.2.4 5.2.5 5.2.6 5.2.7 5.2.8 5.2.9 5.3.1 5.3.10 5.3.11 5.3.12 5.3.2 5.3.3 5.3.4 5.3.5 5.3.6 5.3.7 5.3.8 5.3.9 6.0.1 6.0.10 6.0.11 6.0.12 6.0.14 6.0.15 6.0.16 6.0.17 6.0.18 6.0.19 6.0.2 6.0.20 6.0.21 6.0.22 6.0.23 6.0.24 6.0.25 6.0.3 6.0.4 6.0.5 6.0.6 6.0.7 6.0.8 6.0.9 6.1.1 6.1.10 6.1.11 6.1.12 6.1.14 6.1.15 6.1.16 6.1.17 6.1.2 6.1.3 6.1.4 6.1.5 6.1.6 6.1.7 6.1.8 6.1.9 6.2.0 6.2.1 6.2.10 6.2.2 6.2.3 6.2.4 6.2.5 6.2.6 6.2.7 6.2.8 6.2.9 6.3.0 6.3.1 6.3.10 6.3.11 6.3.12 6.3.14 6.3.15 6.3.16 6.3.17 6.3.18 6.3.19 6.3.2 6.3.20 6.3.21 6.3.22 6.3.3 6.3.4 6.3.5 6.3.6 6.3.7 6.3.8 6.3.9 7.0.1 7.0.2 7.0.3 7.0.4 7.0.5 7.1.0 7.1.1 7.1.10 7.1.11 7.1.12 7.1.14 7.1.15 7.1.16 7.1.17 7.1.18 7.1.19 7.1.2 7.1.20 7.1.3 7.1.4 7.1.5 7.1.6 7.1.7 7.1.8 7.1.9 7.10.0 7.10.1 7.10.2 7.10.3 7.10.4 7.10.5 7.10.6 7.10.7 7.11.0 7.11.1 7.11.2 7.11.3 7.11.4 7.11.5 7.11.6 7.11.7 7.2.1 7.2.2 7.2.3 7.2.4 7.2.5 7.3.1 7.3.2 7.3.3 7.3.4 7.3.5 7.3.6 7.4.0 7.4.1 7.4.10 7.4.11 7.4.12 7.4.14 7.4.2 7.4.3 trunk 7.4.4 1.1 7.4.5 1.2 7.4.6 1.3 7.4.7 1.3.1 7.4.8 1.3.2 7.4.9 1.3.3 7.5.0 1.4.2 7.5.1 1.4.3 7.5.10 1.4.4 7.5.11 1.4.5 7.5.2 1.4.6 7.5.3 1.4.7 7.5.4 1.4.8 7.5.5 1.5.1 7.5.6 1.5.2 7.5.7 1.5.3 7.5.8 1.5.4 7.5.9 1.5.5 7.6.0 1.5.6 7.6.1 2.0.1 7.6.2 2.0.2 7.7.0 2.0.3 7.7.1 2.0.5 7.8.0 2.0.6 7.8.1 2.0.7 7.8.2 2.1.0 7.9.0 2.1.1 7.9.1 2.1.2 7.9.2 2.1.3 7.9.3 2.1.4 8.0.0 2.1.5 8.0.1 3.0.2 8.0.2 3.0.3 8.0.3 3.0.4 8.0.4 3.0.5 8.0.5 3.0.6 8.1.0 3.0.7 8.1.1 3.0.8 8.1.2 3.0.9 8.1.3 3.1.0 8.1.4 3.1.1 v1.4.1 3.1.2 3.1.4 3.1.6 3.2.1 3.2.3 3.2.4 3.2.5 3.2.6 3.2.7 3.3.2 3.3.3 3.3.4 3.3.5 3.3.6 3.3.7 3.4.1 3.4.4 3.4.5 3.5.1 3.5.2 3.6.1 3.6.3 3.6.4 3.6.5 3.6.6 3.6.7 3.6.8 3.6.9
wordfence / lib / wfAPI.php
wordfence / lib Last commit date
Diff 8 years ago dashboard 7 years ago rest-api 7 years ago .htaccess 7 years ago Diff.php 14 years ago GeoLite2-Country.mmdb 7 years ago IPTraf.php 8 years ago IPTrafList.php 7 years ago compat.php 8 years ago conntest.php 7 years ago cronview.php 8 years ago dbview.php 8 years ago diffResult.php 8 years ago email_genericAlert.php 7 years ago email_newIssues.php 7 years ago email_unlockRequest.php 8 years ago email_unsubscribeRequest.php 7 years ago flags.php 7 years ago live_activity.php 8 years ago menu_dashboard.php 7 years ago menu_dashboard_options.php 7 years ago menu_firewall.php 7 years ago menu_firewall_blocking.php 7 years ago menu_firewall_blocking_options.php 8 years ago menu_firewall_waf.php 7 years ago menu_firewall_waf_options.php 7 years ago menu_options.php 7 years ago menu_scanner.php 7 years ago menu_scanner_credentials.php 8 years ago menu_scanner_options.php 8 years ago menu_support.php 7 years ago menu_tools.php 7 years ago menu_tools_diagnostic.php 7 years ago menu_tools_importExport.php 7 years ago menu_tools_livetraffic.php 7 years ago menu_tools_twoFactor.php 8 years ago menu_tools_whois.php 8 years ago menu_wordfence_central.php 7 years ago sysinfo.php 8 years ago unknownFiles.php 8 years ago viewFullActivityLog.php 8 years ago wf503.php 7 years ago wfAPI.php 7 years ago wfActivityReport.php 7 years ago wfAdminNoticeQueue.php 8 years ago wfArray.php 7 years ago wfBrowscap.php 8 years ago wfBrowscapCache.php 7 years ago wfBulkCountries.php 7 years ago wfCache.php 9 years ago wfCentralAPI.php 7 years ago wfConfig.php 7 years ago wfCrawl.php 8 years ago wfCredentialsController.php 7 years ago wfCrypt.php 8 years ago wfDB.php 7 years ago wfDashboard.php 7 years ago wfDateLocalization.php 8 years ago wfDiagnostic.php 7 years ago wfDict.php 8 years ago wfDirectoryIterator.php 7 years ago wfHelperBin.php 11 years ago wfHelperString.php 11 years ago wfIPWhitelist.php 7 years ago wfImportExportController.php 7 years ago wfIssues.php 7 years ago wfJWT.php 7 years ago wfLockedOut.php 7 years ago wfLog.php 7 years ago wfMD5BloomFilter.php 8 years ago wfNotification.php 8 years ago wfOnboardingController.php 7 years ago wfPersistenceController.php 8 years ago wfRESTAPI.php 7 years ago wfScan.php 7 years ago wfScanEngine.php 7 years ago wfSchema.php 7 years ago wfStyle.php 7 years ago wfSupportController.php 7 years ago wfUnlockMsg.php 7 years ago wfUpdateCheck.php 8 years ago wfUtils.php 7 years ago wfVersionCheckController.php 8 years ago wfView.php 10 years ago wfViewResult.php 8 years ago wordfenceClass.php 7 years ago wordfenceConstants.php 7 years ago wordfenceHash.php 7 years ago wordfenceScanner.php 7 years ago wordfenceURLHoover.php 7 years ago
wfAPI.php
242 lines
1 <?php
2 require_once('wordfenceConstants.php');
3 require_once('wordfenceClass.php');
4
5 class wfAPI {
6 const KEY_TYPE_FREE = 'free';
7 const KEY_TYPE_PAID_CURRENT = 'paid-current';
8 const KEY_TYPE_PAID_EXPIRED = 'paid-expired';
9
10 public $lastHTTPStatus = '';
11 public $lastCurlErrorNo = '';
12 private $curlContent = 0;
13 private $APIKey = '';
14 private $wordpressVersion = '';
15
16 public function __construct($apiKey, $wordpressVersion) {
17 $this->APIKey = $apiKey;
18 $this->wordpressVersion = $wordpressVersion;
19 }
20
21 public function getStaticURL($url) { // In the form '/something.bin' without quotes
22 return $this->getURL(rtrim($this->getAPIURL(), '/') . '/' . ltrim($url, '/'));
23 }
24
25 public function call($action, $getParams = array(), $postParams = array(), $forceSSL = false, $timeout = 900) {
26 $apiURL = $this->getAPIURL();
27 //Sanity check. Developer should call wfAPI::SSLEnabled() to check if SSL is enabled before forcing SSL and return a user friendly msg if it's not.
28 if ($forceSSL && (!preg_match('/^https:/i', $apiURL))) {
29 //User's should never see this message unless we aren't calling SSLEnabled() to check if SSL is enabled before using call() with forceSSL
30 throw new wfAPICallSSLUnavailableException("SSL is not supported by your web server and is required to use this function. Please ask your hosting provider or site admin to install cURL with openSSL to use this feature.");
31 }
32 $json = $this->getURL(rtrim($apiURL, '/') . '/v' . WORDFENCE_API_VERSION . '/?' . $this->makeAPIQueryString() . '&' . self::buildQuery(
33 array_merge(
34 array('action' => $action),
35 $getParams
36 )), $postParams, $timeout);
37 if (!$json) {
38 throw new wfAPICallInvalidResponseException("We received an empty data response from the Wordfence scanning servers when calling the '$action' function.");
39 }
40
41 $dat = json_decode($json, true);
42 if (isset($dat['_isPaidKey'])) {
43 wfConfig::set('keyExpDays', $dat['_keyExpDays']);
44 if ($dat['_keyExpDays'] > -1) {
45 wfConfig::set('isPaid', 1);
46 }
47 else if ($dat['_keyExpDays'] < 0) {
48 wfConfig::set('isPaid', '');
49 }
50
51 if (!isset($dat['errorMsg'])) {
52 if ($dat['_keyExpDays'] > -1) {
53 wfConfig::set('keyType', self::KEY_TYPE_PAID_CURRENT);
54 }
55 else if ($dat['_keyExpDays'] < 0) {
56 wfConfig::set('keyType', self::KEY_TYPE_PAID_EXPIRED);
57 }
58
59 if (isset($dat['_autoRenew'])) { wfConfig::set('premiumAutoRenew', wfUtils::truthyToInt($dat['_autoRenew'])); } else { wfConfig::remove('premiumAutoRenew'); }
60 if (isset($dat['_nextRenewAttempt'])) { wfConfig::set('premiumNextRenew', time() + $dat['_nextRenewAttempt'] * 86400); } else { wfConfig::remove('premiumNextRenew'); }
61 if (isset($dat['_paymentExpiring'])) { wfConfig::set('premiumPaymentExpiring', wfUtils::truthyToInt($dat['_paymentExpiring'])); } else { wfConfig::remove('premiumPaymentExpiring'); }
62 if (isset($dat['_paymentExpired'])) { wfConfig::set('premiumPaymentExpired', wfUtils::truthyToInt($dat['_paymentExpired'])); } else { wfConfig::remove('premiumPaymentExpired'); }
63 if (isset($dat['_paymentMissing'])) { wfConfig::set('premiumPaymentMissing', wfUtils::truthyToInt($dat['_paymentMissing'])); } else { wfConfig::remove('premiumPaymentMissing'); }
64 if (isset($dat['_paymentHold'])) { wfConfig::set('premiumPaymentHold', wfUtils::truthyToInt($dat['_paymentHold'])); } else { wfConfig::remove('premiumPaymentHold'); }
65 }
66 }
67
68 $hasKeyConflict = false;
69 if (isset($dat['_hasKeyConflict'])) {
70 $hasKeyConflict = ($dat['_hasKeyConflict'] == 1);
71 if ($hasKeyConflict) {
72 new wfNotification(null, wfNotification::PRIORITY_HIGH_CRITICAL, '<a href="' . wfUtils::wpAdminURL('admin.php?page=Wordfence&subpage=global_options') . '">The Wordfence license you\'re using does not match this site\'s address. Premium features are disabled.</a>', 'wfplugin_keyconflict', null, array(array('link' => 'https://www.wordfence.com/manage-wordfence-api-keys/', 'label' => 'Manage Keys')));
73 wfConfig::set('hasKeyConflict', 1);
74 }
75 }
76
77 $keyNoLongerValid = false;
78 if (isset($dat['_keyNoLongerValid'])) {
79 $keyNoLongerValid = ($dat['_keyNoLongerValid'] == 1);
80 if ($keyNoLongerValid) {
81 wordfence::ajax_downgradeLicense_callback();
82 }
83 }
84
85 if (!$hasKeyConflict) {
86 wfConfig::remove('hasKeyConflict');
87 $n = wfNotification::getNotificationForCategory('wfplugin_keyconflict');
88 if ($n !== null) {
89 wordfence::status(1, 'info', 'Idle');
90 $n->markAsRead();
91 }
92 }
93
94 if (isset($dat['_touppChanged'])) {
95 wfConfig::set('touppPromptNeeded', wfUtils::truthyToBoolean($dat['_touppChanged']));
96 }
97
98 if (!is_array($dat)) {
99 throw new wfAPICallInvalidResponseException("We received a data structure that is not the expected array when contacting the Wordfence scanning servers and calling the '$action' function.");
100 }
101 if (is_array($dat) && isset($dat['errorMsg'])) {
102 throw new wfAPICallErrorResponseException($dat['errorMsg']);
103 }
104 return $dat;
105 }
106
107 protected function getURL($url, $postParams = array(), $timeout = 900) {
108 wordfence::status(4, 'info', "Calling Wordfence API v" . WORDFENCE_API_VERSION . ":" . $url);
109
110 if (!function_exists('wp_remote_post')) {
111 require_once ABSPATH . WPINC . 'http.php';
112 }
113
114 $ssl_verify = (bool) wfConfig::get('ssl_verify');
115 $args = array(
116 'timeout' => $timeout,
117 'user-agent' => "Wordfence.com UA " . (defined('WORDFENCE_VERSION') ? WORDFENCE_VERSION : '[Unknown version]'),
118 'body' => $postParams,
119 'sslverify' => $ssl_verify,
120 'headers' => array('Referer' => false),
121 );
122 if (!$ssl_verify) {
123 // Some versions of cURL will complain that SSL verification is disabled but the CA bundle was supplied.
124 $args['sslcertificates'] = false;
125 }
126
127 $response = wp_remote_post($url, $args);
128
129 $this->lastHTTPStatus = (int) wp_remote_retrieve_response_code($response);
130
131 if (is_wp_error($response)) {
132 $error_message = $response->get_error_message();
133 throw new wfAPICallFailedException("There was an " . ($error_message ? '' : 'unknown ') . "error connecting to the Wordfence scanning servers" . ($error_message ? ": $error_message" : '.'));
134 }
135
136 $dateHeader = @$response['headers']['date'];
137 if (!empty($dateHeader) && (time() - wfConfig::get('timeoffset_wf_updated', 0) > 3600)) {
138 if (function_exists('date_create_from_format')) {
139 $dt = DateTime::createFromFormat('D, j M Y G:i:s O', $dateHeader);
140 $timestamp = $dt->getTimestamp();
141 }
142 else {
143 $timestamp = strtotime($dateHeader);
144 }
145 $offset = $timestamp - time();
146 wfConfig::set('timeoffset_wf', $offset);
147 wfConfig::set('timeoffset_wf_updated', time());
148 }
149
150 if (!empty($response['response']['code'])) {
151 $this->lastHTTPStatus = (int) $response['response']['code'];
152 }
153
154 if (200 != $this->lastHTTPStatus) {
155 throw new wfAPICallFailedException("The Wordfence scanning servers are currently unavailable. This may be for maintenance or a temporary outage. If this still occurs in an hour, please contact support. [$this->lastHTTPStatus]");
156 }
157
158 $content = wp_remote_retrieve_body($response);
159 return $content;
160 }
161
162 public function binCall($func, $postData) {
163 $url = rtrim($this->getAPIURL(), '/') . '/v' . WORDFENCE_API_VERSION . '/?' . $this->makeAPIQueryString() . '&action=' . $func;
164
165 $data = $this->getURL($url, $postData);
166
167 if (preg_match('/\{.*errorMsg/', $data)) {
168 $jdat = @json_decode($data, true);
169 if (is_array($jdat) && $jdat['errorMsg']) {
170 throw new Exception($jdat['errorMsg']);
171 }
172 }
173 return array('code' => $this->lastHTTPStatus, 'data' => $data);
174 }
175
176 public function makeAPIQueryString() {
177 $cv = null;
178 $cs = null;
179 if (function_exists('curl_version')) {
180 $curl = curl_version();
181 $cv = $curl['version'];
182 $cs = $curl['ssl_version'];
183 }
184
185 $values = array(
186 'wp' => $this->wordpressVersion,
187 'wf' => WORDFENCE_VERSION,
188 'ms' => (is_multisite() ? get_blog_count() : false),
189 'h' => wfUtils::wpHomeURL(),
190 'sslv' => function_exists('openssl_verify') && defined('OPENSSL_VERSION_NUMBER') ? OPENSSL_VERSION_NUMBER : null,
191 'pv' => phpversion(),
192 'pt' => php_sapi_name(),
193 'cv' => $cv,
194 'cs' => $cs,
195 'sv' => (isset($_SERVER['SERVER_SOFTWARE']) ? $_SERVER['SERVER_SOFTWARE'] : null),
196 'dv' => wfConfig::get('dbVersion', null),
197 );
198
199 return self::buildQuery(array(
200 'k' => $this->APIKey,
201 's' => wfUtils::base64url_encode(json_encode($values)),
202 'betaFeed' => (int) wfConfig::get('betaThreatDefenseFeed'),
203 ));
204 }
205
206 private function buildQuery($data) {
207 if (version_compare(phpversion(), '5.1.2', '>=')) {
208 return http_build_query($data, '', '&'); //arg_separator parameter was only added in PHP 5.1.2. We do this because some PHP.ini's have arg_separator.output set to '&amp;'
209 } else {
210 return http_build_query($data);
211 }
212 }
213
214 private function getAPIURL() {
215 return self::SSLEnabled() ? WORDFENCE_API_URL_SEC : WORDFENCE_API_URL_NONSEC;
216 }
217
218 public static function SSLEnabled() {
219 if (!function_exists('wp_http_supports')) {
220 require_once ABSPATH . WPINC . 'http.php';
221 }
222 return wp_http_supports(array('ssl'));
223 }
224
225 public function getTextImageURL($text) {
226 $apiURL = $this->getAPIURL();
227 return rtrim($apiURL, '/') . '/v' . WORDFENCE_API_VERSION . '/?' . $this->makeAPIQueryString() . '&' . self::buildQuery(array('action' => 'image', 'txt' => base64_encode($text)));
228 }
229 }
230
231 class wfAPICallSSLUnavailableException extends Exception {
232 }
233
234 class wfAPICallFailedException extends Exception {
235 }
236
237 class wfAPICallInvalidResponseException extends Exception {
238 }
239
240 class wfAPICallErrorResponseException extends Exception {
241 }
242