-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathstop-emails.php
306 lines (273 loc) · 7.95 KB
/
stop-emails.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
<?php
/**
* Plugin Name: Stop Emails
* Plugin URI: http://salferrarello.com/stop-emails-wordpress-plugin/
* Description: Stops outgoing emails generated by WordPress core, plugins, and themes. All emails sent following the best practice of using wp_mail() will be stopped. If code uses the PHP mail() function directly, this plugin will not stop the email.
* Version: 1.2.1
* Author: Sal Ferrarello
* Author URI: http://salferrarello.com/
* Text Domain: stop-emails
* Domain Path: /languages
*
* @package stop-emails
*/
// If this file is called directly, abort.
if ( ! defined( 'WPINC' ) ) {
die;
}
register_deactivation_hook( __FILE__, array( 'Fe_Stop_Emails', 'on_deactivation' ) );
if ( Fe_Stop_Emails::is_pre_wp_5_5() ) {
// Legacy: Load PHPMailer class, so we can subclass it.
require_once ABSPATH . WPINC . '/class-phpmailer.php';
class Fe_Stop_Emails_PHPMailer extends PHPMailer {}
} else {
// Modern: Load PHPMailer class, so we can subclass it.
require_once ABSPATH . WPINC . '/PHPMailer/PHPMailer.php';
require_once ABSPATH . WPINC . '/PHPMailer/Exception.php';
class Fe_Stop_Emails_PHPMailer extends PHPMailer\PHPMailer\PHPMailer {}
}
/**
* Subclass of PHPMailer to prevent Sending.
*
* This subclass of PHPMailer replaces the send() method
* with a method that does not send.
* This subclass is based on the WP Core MockPHPMailer
* found in phpunit/includes/mock-mailer.php
*
* @since 0.8.0
* @see PHPMailer
*/
class Fe_Stop_Emails_Fake_PHPMailer extends Fe_Stop_Emails_PHPMailer {
/**
* Mock sent email.
*
* @since 0.8.0
* @var array of email components (e.g. to, cc, etc.)
*/
var $mock_sent = array();
/**
* Replacement send() method that does not send.
*
* Unlike the PHPMailer send method,
* this method never calls the method postSend(),
* which is where the email is actually sent
*
* @since 0.8.0
* @return bool
*/
function send() {
try {
if ( ! $this->preSend() ) {
return false;
}
$mock_email = array(
'to' => $this->to,
'cc' => $this->cc,
'bcc' => $this->bcc,
'header' => $this->MIMEHeader,
'body' => $this->MIMEBody,
);
$this->mock_sent[] = $mock_email;
// Hook to allow logging.
do_action( 'fe_stop_emails_log', $mock_email );
return true;
} catch ( phpmailerException $e ) {
return false;
}
}
}
/**
* Stop Emails Plugin Class.
*
* Prevents emails from being sent and provides basic logging.
* Replaces PHPMailer global instance $phpmailer with an instance
* of the subclass Fe_Stop_Emails_Fake_PHPMailer
*
* @since 0.8.0
*/
class Fe_Stop_Emails {
/**
* Constuctor to setup plugin.
*
* @since 0.8.0
*/
public function __construct() {
$this->add_hooks();
$this->settings_page();
}
/**
* Add hooks.
*
* @since 0.8.0
*/
public function add_hooks() {
add_action( 'plugins_loaded', array( $this, 'replace_phpmailer' ) );
add_action( 'fe_stop_emails_log', array( $this, 'log_to_php_error_log' ) );
add_action( 'admin_notices', array( $this, 'warning' ) );
add_action( 'init', array( $this, 'load_textdomain' ) );
/**
* Force BuddyPress to use wp_mail() rather than its own BP_PHPMailer class
*/
add_filter( 'bp_email_use_wp_mail', '__return_true' );
}
/**
* Replace the global $phpmailer with fake phpmailer.
*
* @since 0.8.0
*
* @return Fe_Stop_Emails_Fake_PHPMailer instance, the object that replaced
* the global $phpmailer
*/
public function replace_phpmailer() {
global $phpmailer;
return $this->replace_w_fake_phpmailer( $phpmailer );
}
/**
* Replace the parameter object with an instance of
* Fe_Stop_Emails_Fake_PHPMailer.
*
* @since 0.8.0
*
* @param PHPMailer $obj WordPress PHPMailer object.
* @return Fe_Stop_Emails_Fake_PHPMailer $obj
*/
public function replace_w_fake_phpmailer( &$obj = null ) {
$obj = new Fe_Stop_Emails_Fake_PHPMailer;
return $obj;
}
/**
* Should emails be logged to the PHP error log.
*
* @since 0.8.0
*
* @return bool
*/
public function should_emails_be_logged_to_php_error_log() {
$options = get_option( 'fe_stop_emails_options' );
if ( $options && isset( $options['log-email'] ) ) {
// Use value from options.
$log_to_error_log = $options['log-email'];
} else {
// Default value.
$log_to_error_log = 0;
}
$log_to_error_log = apply_filters( 'fe_stop_emails_log_email', $log_to_error_log );
return (bool) $log_to_error_log;
}
/**
* Hooked function for email logging.
*
* Checks if email should be logged and logs it if necessary
*
* @param array $mock_email represents the email that was stopped.
* @since 0.8.0
*/
public function log_to_php_error_log( $mock_email ) {
if ( $this->should_emails_be_logged_to_php_error_log() ) {
$text = $this->mock_email_to_text( $mock_email );
error_log( $text );
}
}
/**
* Convert mock email to text.
*
* @since 0.8.0
*
* @param Fe_Stop_Emails_Fake_PHPMailer $mock_email represents the email that was stopped.
* @return string, text version of email
*/
public function mock_email_to_text( $mock_email ) {
return print_r( $mock_email, true );
}
/**
* Display Warning that emails are being stopped.
*
* Display admin notice warning that emails are being
* stopped, additionally if emails are being logged
* in the PHP error_log, it is noted that emails are
* being logged.
*
* @since 0.8.0
*/
public function warning() {
echo "\n<div class='error'><p>";
echo '<strong>';
if ( $this->should_emails_be_logged_to_php_error_log() ) {
esc_html_e( 'Logging Disabled Emails', 'stop-emails' );
} else {
esc_html_e( 'Emails Disabled', 'stop-emails' );
}
echo ': ';
echo '</strong>';
esc_html_e( 'The Stop Emails plugin is currently active, which will prevent any emails from being sent. ', 'stop-emails' );
esc_html_e( 'To send emails, disable the plugin.', 'stop-emails' );
echo '</p></div>';
}
/**
* Create Settings Page.
*
* The settings page is created in lib/admin-settings.php.
* We include a check that this file exists, so we can
* run this plugin with only this primary file; this
* allows using this single file as an "mu-plugins" plugin.
*
* @since 0.8.0
*/
public function settings_page() {
$plugin_dir_path = plugin_dir_path( __FILE__ );
$plugin_basename = plugin_basename( __FILE__ );
if ( file_exists( "{$plugin_dir_path}lib/admin-settings.php" ) ) {
// Create admin settings screen.
require_once( "{$plugin_dir_path}lib/admin-settings.php" );
// Add Settings link on Plugin Page.
add_filter( "plugin_action_links_$plugin_basename", array( $this, 'settings_link_on_plugin_page' ) );
}
}
/**
* Add a settings link to links for this plugin on the plugin page.
*
* Add to the $links array, an element that contains the html markup
* for the settings page for this link.
*
* @since 0.8.0
* @param array of string $links each of which is the markup for a link.
* @return array of strings, each of which is the markup for a link
* with additional link
*/
public function settings_link_on_plugin_page( $links ) {
$links[] = '<a href="' .
admin_url( 'options-general.php?page=fe_stop_emails' ) .
'">' . __( 'Settings' ) . '</a>';
return $links;
}
/**
* Load textdomain for translations.
*
* @since 0.8.0
*/
public function load_textdomain() {
$domain = 'stop-emails';
$plugin_rel_path = dirname( plugin_basename( __FILE__ ) ) . '/languages';
load_plugin_textdomain( $domain, false, $plugin_rel_path );
}
/**
* On plugin deactivation clean up.
*
* Remove the plugin option, where settings are stored
*
* @since 0.8.0
*/
public static function on_deactivation() {
delete_option( 'fe_stop_emails_options' );
}
/**
* WordPress core version is less than 5.5
*
* @since 1.3.0
* @return bool WordPress core version is less than 5.5
*/
public static function is_pre_wp_5_5() {
return version_compare( get_bloginfo( 'version' ), '5.5-alpha', '<' );
}
}
new Fe_Stop_Emails;