Signs and encrypts emails using PGP/GPG keys or X.509 certificates. Provides OpenPGP and S/MIME functions via WordPress plugin API.
WP PGP Encrypted Emails can automatically sign and encrypt any email that WordPress sends to your site’s admin email address or your users’s email addresses. You give it a copy of the recipient’s OpenPGP public key and/or their S/MIME certificate, and it does the rest. You can even automatically generate an OpenPGP signing keypair for your site to use.
Encrypting outgoing emails protects your user’s privacy by ensuring that emails intended for them can be read only by them, and them alone. Moreover, signing those emails helps your users verify that email they receive purporting to be from your site was actually sent by your server, and not some imposter. If you’re a plugin or theme developer, you can encrypt and/or sign arbitrary data using this plugin’s OpenPGP and S/MIME APIs, which are both built with familiar, standard WordPress filter hooks. This enables you to develop highly secure communication and publishing tools fully integrated with your WordPress install. See the README.markdown
file for details on cryptographic implementation and API usage.
Donations for this and my other free software plugins make up a chunk of my income. If you continue to enjoy this plugin, please consider making a donation. 🙂 Thank you for your support!
Plugin features:
Subject
lines).The plugin works transparently for all email your site generates, and will also sign and encrypt outgoing email generated by other plugins (such as contact form plugins) or the built-in WordPress notification emails. All you have to do is add one or more OpenPGP keys or an S/MIME certificate to the Email Encryption screen (WordPress Admin Dashboard → Settings → Email Encryption). Each user can opt to also remove envelope information such as email subject lines, which encryption schemes cannot protect. With this plugin, there’s no longer any need to pay for the “pro” version of your favorite contact form plugin to get the benefit of email privacy.
Each of your site’s users can supply their own, personal OpenPGP public key and/or X.509 S/MIME certificate for their own email address to have WordPress automatically encrypt any email destined for them. (They merely need to update their user profile.) They can choose which encryption method to use. Once set up, all future emails WordPress sends to that user will be encrypted using the standards-based OpenPGP or S/MIME technologies.
The OpenPGP-encrypted emails can be decrypted by any OpenPGP-compatible mail client, such as MacGPG (macOS), GPG4Win (Windows), Enigmail (cross-platform), OpenKeychain (Android), or iPGMail (iPhone/iOS). For more information on reading encrypted emails, generating keys, and other uses for OpenPGP-compatible encryption, consult any (or all!) of the following guides:
The S/MIME-encrypted emails can be decrypted by any S/MIME-compatible mail client. These include Apple’s Mail on macOS and iOS for iPhone and iPad, Microsoft Outlook, Claws Mail for GNU/Linux, and more.
For developers, WP PGP Encrypted Emails provides an easy to use API to both OpenPGP and S/MIME encryption, decryption, and integrity validation operations through the familiar WordPress plugin API so you can use this plugin’s simple filter hooks to build custom OpenPGP- or S/MIME-based encryption functionality into your own plugins and themes.
Security Disclaimer
Security is a process, not a product. Using WP PGP Encrypted Emails does not guarantee that your site’s outgoing messages are invulnerable to every attacker, in every possible scenario, at all times. No single security measure, in isolation, can do that.
Do not rely solely on this plugin for the security or privacy of your webserver. See the Frequently Asked Questions for more security advice and for more information about the rationale for this plugin.
If you like this plugin, please consider making a donation for your use of the plugin or, better yet, contributing directly to my Cyberbusking fund. Your support is appreciated!
Theme authors can use the following code snippets to integrate a WordPress theme with this plugin.
<?php print admin_url( 'admin-ajax.php?action=download_pgp_signing_public_key' ); ?>
This plugin offers additional functionality intended for other plugin developers or theme authors to make use of. This functionality is documented here.
Gets the user’s preferred encryption method (either pgp
or smime
), if they have provided both an OpenPGP public key and an S/MIME certificate.
WP_User
$user
– The WordPress user object. Defaults to the current user.Gets the user’s saved OpenPGP public key from their WordPress profile data, immediately usable in other openpgp_*
filters.
WP_User
$user
– The WordPress user object. Defaults to the current user.Gets an ASCII-armored representation of an OpenPGP data structure (like a key, or an encrypted message).
string
$data
– The data to be armored.string
$marker
– The marker of the block (the text that follows -----BEGIN
). Defaults to MESSAGE
, but you should set this to a more appropriate value. If you are armoring a PGP public key, for instance, set this to PGP PUBLIC KEY BLOCK
.string[]
$headers
– An array of strings to apply as headers to the ASCII-armored block, usually used to insert comments or identify the OpenPGP client used. Defaults to array()
(no headers).Example: ASCII-armor a binary public key.
$ascii_key = apply_filters('openpgp_enarmor', $public_key, 'PGP PUBLIC KEY BLOCK');
Gets a binary OpenPGP public key for use in later PGP operations from an ASCII-armored representation of that key.
string
$key
– The ASCII-armored PGP public key block.Example: Get a key saved as an ASCII string in the WordPress database option my_plugin_pgp_public_key
.
$key = apply_filters('openpgp_key', get_option('my_plugin_pgp_public_key'));
Clearsigns a message using a given private key.
string
$data
– The message data to sign.OpenPGP_SecretKeyPacket
$signing_key
– The signing key to use, obtained by passing the ASCII-armored private key through the openpgp_key
filter.Example: Sign a short string.
$message = 'This is a message to sign.'; $signing_key = apply_filters('openpgp_key', $ascii_key); $signed_message = apply_filters('openpgp_sign', $message, $signing_key); // $signed_message is now a clearsigned message
Encrypts data to one or more PGP public keys or passphrases.
string
$data
– Data to encrypt.array|string
$keys
– Passphrases or keys to use to encrypt the data.Example: Encrypt the content of a blog post.
// First, get the PGP public key(s) of the recipient(s) $ascii_key = '-----BEGIN PGP PUBLIC KEY BLOCK----- [...snipped for length...] -----END PGP PUBLIC KEY BLOCK-----'; $encryption_key = apply_filters('openpgp_key', $ascii_key); $encrypted_post = apply_filters('openpgp_encrypt', $post->post_content, $encryption_key); // Now you can safely send or display $encrypted_post anywhere you like and only // those who control the corresponding private key(s) can decrypt it.
Signs a message (arbitrary data) with the given private key.
Note that if your plugin uses the built-in WordPress core wp_mail()
function and this plugin is active, your plugin’s outgoing emails are already automatically signed so you do not need to do anything. This filter is intended for use by plugin developers who want to create custom, trusted communiques between WordPress and some other system.
string
$data
– The data to sign.OpenPGP_SecretKeyPacket
$privatekey
– The private key used for signing the message. The default is to use the private key automatically generated during plugin activation. The automatically generated keypair is intended to be a low-trust, single-purpose keypair for your website itself, so you probably do not need or want to use this argument yourself.Example: Send a signed, encrypted JSON payload to a remote, insecure server.
$comment_data = get_comment(2); // get a WP_Comment object with comment ID 2 // Create JSON payload $json = array('success' => true, 'action' => 'new_comment', 'data' => $comment_data); $url = 'http://insecure.example.com/'; $response = wp_safe_remote_post($url, array( ));
A convenience filter that applies openpgp_sign
and then openpgp_encrypt
to the result.
string
$data
– The data to sign and encrypt.string
$signing_key
– The signing key to use.array|string
$recipient_keys_and_passphrases
– Public key(s) of the recipient(s), or passphrases to encrypt to.Gets the user’s saved S/MIME public certificate from their WordPress profile data, immediately usable in other smime_*
filters.
WP_User
$user
– The WordPress user object. Defaults to the current user.Gets a PHP resource handle to an X.509 Certificate.
mixed
$cert
– The certificate, either as a string to a file, or raw PEM-encoded certificate data.Encodes (“exports”) a given X.509 certificate as PEM format.
resource
$cert
Encrypts a message as an S/MIME email given a public certificate.
string
$message
– The message contents to encrypt.string|string[]
$headers
– The message headers for the encrypted part.resource|array
$certificates
– The recipient’s certificate, or an array of recipient certificates.This filter returns an array with two keys, headers
and message
, wherein the message is encrypted.
Example: send an encrypted email via wp_mail()
. (You do not need to do this if the recipient is registered as your site’s user, because this plugin does that automatically. Only do this if you need to send S/MIME encrypted email to an address not stored in WordPress’s own database.)
$cert = apply_filters( 'smime_certificate', get_option( 'my_plugin_smime_certificate' ) ); $body = 'This is a test email message body.'; $head = array( 'From' => get_option( 'admin_email' ), ); $smime_data = apply_filters( 'smime_encrypt', $body, $head, $cert ); if ( $smime_data ) { wp_mail( '[email protected]', 'Test message.', $smime_data['message'], // message is sent encrypted $smime_data['headers'] ); }
WP PGP Encrypted Emails can be installed automatically from the WordPress plugin repository by searching for “PGP Encrypted Emails” in the “Add new plugin” screen of your WordPress admin site and clicking the “Install now” button.
Minimum requirements:
The plugin will automatically de-activate itself, or automatically disable certain features, if these requirements are not met. If you do not see a given feature in the user interface, ensure your server (and your web hosting provider) meet the above requirements!
WP PGP Encrypted Emails can also be installed manually by following these instructions:
wp-pgp-encrypted-emails
folder to the /wp-content/plugins/
directory of your WordPress installation.Once activated, each user who wants to receive encrypted emails must add their OpenPGP public key or S/MIME public certificate to their profile, and optionally choose the type of encryption they prefer messages addressed to them will use.
See the screenshots for a brief walkthrough of how to configure WP PGP Encrypted Emails after it is installed.
If a user does not have an OpenPGP public key already, they need to use an OpenPGP-compatible client to generate one themselves. (Learn more about OpenPGP-compatible clients.) Users will need to have the private key corresponding to the public key saved in their WordPress profile in order to be able to decrypt the email they receive. To learn more about generating keys and decrypting email, consult one (or more) of the following guides:
I have also found the following articles useful, but can not personally vouch for their accuracy:
If you found a good guide to using PGP/GPG that I haven’t listed here, please share it in the WP PGP Encrypted Emails plugin support forum.
Similarly, if a user doesn’t already have one, they will need to obtain an S/MIME certificate from a Certificate Authority (such as a public CA or their employer), or generate a self-signed one themselves. Learn more about getting an S/MIME certificate.
To view a list of known issues with this plugin or to report a bug, please use the project’s issue tracker.
Paste the plain text version of your OpenPGP public key into the "PGP Public Key" field in your profile, then click "Save changes" at the bottom of the page. (There is a similar field for the WordPress admin email in the General Settings screen accessible to Administrator users.)
If the plugin detects a problem with your OpenPGP public key, you will get a notice like the one shown here.
Authors who add an OpenPGP public key to their profile also let readers leave semi-private comments on their posts. These are comments that are automatically encrypted to the author's public key upon submission. Commenters who want to send a "Private" comment simply write their comment normally and ensure the encryption checkbox is enabled when they submit their comment.
Administrators can generate an OpenPGP signing keypair with which to automatically sign outgoing emails. This helps recipients verify that email they receive actually came from your website. Admins can regenerate the keypair automatically by clicking the "Regenerate keypair" button, or they can manually paste an ASCII-armored keypair for the site to use. For security, the private key part of the site's signing key will only be transmitted over a secure (HTTPS) connection, so you will see a prompt to switch to a secure connection if you try to view it insecurely. You can still (re)generate a keypair, including the private key part, over an insecure connection because the key is generated on the server itself.
An OpenPGP-compatible client is simply an app that can read, write, and verify messages encrypted or signed using PGP technology. There are great, free apps for every major platform, including Windows, Mac, Linux, Android, iPhones, iPads, and more. Which app you choose depends largely on which device you already have, and then a bit about your personal tastes.
Since there are so many OpenPGP-compatible apps to choose from, I recommend sticking to the ones listed on the PRISM-Break.org website. (Note that PRISM-Break calls it “GPG” instead of “PGP,” but the two terms are generally synonymous.) Once you choose an OpenPGP-compatible app for your platform, consider seeking out its help and support documentation to get started using it, or check out some of the generic PGP/GPG guides listed at the end of this plugin’s Installation page.
If you have registered a WordPress user account with the same email address as your site’s admin email address (in the Settings → General screen), then WP PGP Encrypted Emails will first check for a public key or certificate in the administrative Email Encryption settings (Settings → Email Encryption) before looking at your user’s profile settings. If an encryption key or certificate exists in the administrative settings, that key will be used instead of the key or certificate in the user’s profile. To resolve this issue, either ensure that you enter encryption keys in only one location (the administrative screen or the user’s profile), or ensure these keys are the same.
If you still cannot decrypt messages, make absolutely certain you have the matching private key or certificate corresponding to the public key or certificate that was used to encrypt the message. Unfortunately, it is highly infeasible that anyone on Earth will be able to decrypt messages encrypted to a public key or certificate that you do not have the associated private key for. That is, of course, the whole point of this software.
If you have received a “Private” comment, you will need to use an OpenPGP-compatible PGP client to decrypt and read it. There are many free apps that do this. Which one you choose depends on what kind of computer you are already using. If you use Windows, I suggest installing and using GPG4Win since it provides the most features. For Mac OS X users, I suggest MacGPG for the same reason, and Linux users should check their distro’s package repository for compatible options. (For Ubuntu users, the Seahorse-Nautilus plugin is popular.)
I might also add support for an in-browser client based on OpenPGP.js at some point, but for now you will still need an external program to read encrypted comments. Please consider donating to help resource me work on this if that is a feature you’d like to see.
Make sure the emails the other plugin sends are being addressed to an email account that WordPress knows about and that WordPress knows which OpenPGP public key or S/MIME certificate to use when encrypting email destined for that address.
More specifically, this means the TO:
field of the outgoing email needs to match either your WordPress’s “admin email” address or the email address of one of your WordPress user accounts, and you need to provide the OpenPGP public key or S/MIME public certificate you want WordPress to use when encrypting the message and sending email to that address. In many contact form plugins, you can supply an arbitrary email address to send those emails to, but if that email address is not the address of a user on your site, WP PGP Encrypted Emails won’t know which OpenPGP public key or S/MIME public certificate to use for encryption.
As a workaround, simply create an unprivileged (“Subscriber” role) new WordPress user account with that email address and enter the OpenPGP public key or S/MIME certificate in that user’s profile. (Either accept the automatically generated password, or supply a new very strong passphrase, since you will not need to remember it because you will never need to log in with that user account.)
Issues with character sets, accented characters, different human languages, or content types not appearing correctly are almost always the result of a misconfiguration in the email-sending plugin you are using. Many contact form plugins, for example, allow you to supply custom email headers. WP PGP Encrypted Emails takes great care not to corrupt the email message sent by the underlying plugin that generated the email in the first place. However, this also means that if you do not set up your contact form or email-sending plugin correctly, this plugin won’t fix the error.
Most often, this is simply a matter of setting the correct Content-Type header in your contact form or email-sending plugin’s settings.
Against the NSA? No, probably not. Against a nosy co-worker? Yes, probably.
The “realness” of security cannot “really” be measured in abstract, imprecise terms, but rather only based on what real threats you are likely to face and what risks you are vulnerable to if those threats materialize in reality. You have a much better sense of the answers to these things than I do for your situation, because I am not you. Security professionals call this process “threat modeling,” and if you are “really” concerned for your security (I encourage you to be!) then learning how to conduct a threat assessment for yourself is a good idea. Learn more about threat modeling from the EFF’s Surveillance Self-Defense Guide.
TL;DR: Don’t let perfect be the enemy of good.
First of all, not everyone’s security needs are the same. (See “threat modeling,” discussed in the previous question.)
Read Bruce Schneier’s “Lessons from the Sony Hack” for a brief, real-life case study in understanding this important nuance between opportunistic and focused attackers.
Further, security is largely a matter of operational practice, not theoretics. If you never use PGP/GPG because the only tools you have access to are not perfect, then you will not have the experience you need to know how to use PGP/GPG when you actually get access to it, because you never even practiced using it. By way of analogy, if you want to learn swordfighting but all you have is sticks you picked up in the forest, you are better off picking up those sticks and practicing with them than waiting and not practicing at all until you get your hands on steel swords.
Yes. You can use any OpenPGP public key you generate from any OpenPGP-compatible client.
When generating an OpenPGP signing keypair for your WordPress site, this plugin will create a 2,048-bit RSA OpenPGP keypair. This is considered “okay” (but not “great”) by 2018 standards. Unfortunately, many hosts will not allow the plugin to create a stronger keypair because of the computation required. Then again, this key is used only for signing, not encryption. Your own OpenPGP public key is always used for encryption, and you are of course encouraged to make that key as strong as you want.
If you want to use a stronger signing keypair, you can generate one yourself (offline), though you will need to load the key into your WordPress database yourself to use it with this plugin. I consider this extra step “paranoid,” but you are of course welcome to be as careful as you feel is appropriate. 🙂
phpseclib
is updated to version 2.0.31 and addresses a moderate severity security vulnerability.Content-Type: text/html
headers that were also S/MIME encrypted now render properly. Props @githubuserx.Content-Type
mail headers are retained.woocommerce-functions.php
file to their theme. Please only do this if you know what you are doing.smime_pem_to_der
. smime_certificate
, smime_certificate_pem_encode
and smime_encrypt
. See S/MIME API for details.S
for signing and C
for certification in GnuPG).openpgp-php
library and dependencies.wp_mail()
with an array no longer cause PHP warnings.print admin_url('admin-ajax.php?action=download_pgp_signing_public_key')
openpgp_enarmor
filter for ASCII-armoring arbitrary OpenPGP data.openpgp_sign
filter for (clear)signing an arbitrary message.openpgp_key
and openpgp_encrypt
so plugin developers and theme authors can encrypt arbitrary data, too.Fatal error
in cases where the public key is set to false
.