A paywall plugin
By default article content is behind the paywall but can be manually opted-out on a per-post setting. Pages are public but like article they can be opted-in manually.
If a page/post has manually configured a setting it will take precedence over everything else but categories can also set a default state for all content tagged with the category. This rule is inherited so if you tag a child-term it traverse ancestors until a preference is found. If nothing is found it defaults to the post-type rules where post
is paywalled and page
is not.
Out of the box paywalled content will show the login form but you can override this template in your theme.
Additionally there is a Paywalled content block which can be used to manually tailor which section of a page is behind the paywall while everything outside of it will be public. Note that using this block does NOT mark a page as paywalled, it only adjusts the content IF paywalled.
There's a integration with Yoast to output rich schema data according to Google's specifications. There's also a X-Robots-Tag: noarchive
HTTP header sent.
To allow reverse proxies to differentiate paywelled content there's a Vary: X-Paywall-Accepted
header and a X-Paywall-Access: 0|1
added to the response. For now this is also where bot whitelisting would happen for now.
PAYWALL_JWT_PRIVATE_KEY
should be set to the absolute of path of a private key used to sign JWT payloads.
WP_PAYWALL_JWT_ENABLED
can be set tofalse
to disable adding the JWT auth cookie, for example if you need it regardless if the plugin is enabled or not. Defaults to enabledtrue
if there is aPAYWALL_JWT_PRIVATE_KEY
environment variable set.
/**
* Hook in early to return a header passed on from a reverse proxy.
*/
add_filter('wp-paywall/has-access', function (?bool $hasAccess, ?int $postId) {
$proxyAuth = $_SERVER['X-Proxy-Valid-Auth'] ?? null;
return $proxyAuth === '1';
}, 10, 2);
/**
* Remove default access rules and set your own.
*/
add_filter('wp-paywall/access-rules', function (array $rules, ?int $postId) {
$rules = [];
$rules[] = CustomAccess::class;
return $rules;
}, 10, 2);
/**
* Hook in early to return if paywall is applied or not.
*/
add_filter('wp-paywall/is-applied', function (?bool $isApplied = null, ?int $postId) {
if ($postId === 1234) {
return true;
}
return $isApplied;
}, 10, 2);
Install dependencies
composer install
npm install
Run the tests
npm run lint
composer lint
# Setup WP-ENV
npm -g i @wordpress/env
wp-env start
# Run unit tests
wp-env run tests-cli --env-cwd=wp-content/plugins/wp-paywall ./vendor/bin/phpunit
# With Xdebug
wp-env stop
wp-env start --xdebug
wp-env run tests-cli --env-cwd=wp-content/plugins/wp-paywall ./vendor/bin/phpunit
Build assets
# Minified assets which are to be committed to git
npm run build:production
# Watch for changes and re-compile while developing the plugin
npm run start
wp i18n make-pot . languages/wp-paywall.pot
wp i18n make-mo languages/