-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathlightning-publisher.php
executable file
·195 lines (166 loc) · 7.65 KB
/
lightning-publisher.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
<?php
/*
Plugin Name: Lightning Publisher
Version: 0.1.8
Plugin URI: https://github.com/ElementsProject/wordpress-lightning-publisher
Description: Lightning Publisher for WordPress
Author: Blockstream
Author URI: https://blockstream.com
*/
if (!defined('ABSPATH')) exit;
require_once 'vendor/autoload.php';
define('LIGHTNING_PUBLISHER_KEY', hash_hmac('sha256', 'lightning-publisher-token', AUTH_KEY));
class Lightning_Publisher {
public function __construct() {
$this->options = get_option('ln_publisher');
$this->charge = new LightningChargeClient($this->options['server_url'], $this->options['api_token']);
// frontend
add_action('wp_enqueue_scripts', array($this, 'enqueue_script'));
add_filter('the_content', array($this, 'ifpaid_filter'));
// ajax
add_action('wp_ajax_ln_publisher_invoice', array($this, 'ajax_make_invoice'));
add_action('wp_ajax_nopriv_ln_publisher_invoice', array($this, 'ajax_make_invoice'));
add_action('wp_ajax_ln_publisher_token', array($this, 'ajax_make_token'));
add_action('wp_ajax_nopriv_ln_publisher_token', array($this, 'ajax_make_token'));
// admin
add_action('admin_init', array($this, 'admin_init'));
add_action('admin_menu', array($this, 'admin_menu'));
}
/**
* Process [ifpaid] tags in post content
*/
public function ifpaid_filter($content) {
$ifpaid = self::extract_ifpaid_tag($content);
if (!$ifpaid) return $content;
$post_id = get_the_ID();
list($public, $protected) = preg_split('/(<p>)?' . preg_quote($ifpaid->tag, '/') . '(<\/p>)?/', $content, 2);
return self::check_payment($post_id) ? self::format_paid($post_id, $ifpaid, $public, $protected)
: self::format_unpaid($post_id, $ifpaid, $public);
}
/**
* Register scripts and styles
*/
public function enqueue_script() {
wp_enqueue_script('ln-publisher', plugins_url('js/publisher.js', __FILE__), array('jquery'));
wp_enqueue_style('ln-publisher', plugins_url('css/publisher.css', __FILE__));
wp_localize_script('ln-publisher', 'LN_publisher', array(
'ajax_url' => admin_url('admin-ajax.php'),
'charge_url' => !empty($this->options['public_url']) ? $this->options['public_url'] : $this->options['server_url']
));
}
/**
* AJAX endpoint to create new invoices
*/
public function ajax_make_invoice() {
$post_id = (int)$_POST['post_id'];
$ifpaid = self::extract_ifpaid_tag(get_post_field('post_content', $post_id));
if (!$ifpaid) return status_header(404);
$invoice = $this->charge->invoice([
'currency' => $ifpaid->currency,
'amount' => $ifpaid->amount,
'description' => get_bloginfo('name') . ': pay to continue reading ' . get_the_title($post_id),
'metadata' => [ 'source' => 'wordpress-lightning-publisher', 'post_id' => $post_id, 'url' => get_permalink($post_id) ]
]);
wp_send_json($invoice->id, 201);
}
/**
* AJAX endpoint to exchange invoices for HMAC access tokens
* @TODO persist to cookie?
*/
public function ajax_make_token() {
$invoice = $this->charge->fetch($_POST['invoice_id']);
if (!$invoice) return status_header(404);
if ($invoice->status !== 'paid') return status_header(402);
if (!$invoice->metadata->post_id) return status_header(500); // should never actually happen
$post_id = $invoice->metadata->post_id;
$token = self::make_token($post_id);
$url = add_query_arg('publisher_access', $token, get_permalink($post_id));
wp_send_json([ 'post_id' => $post_id, 'token' => $token, 'url' => $url ]);
}
/**
* Create HMAC tokens granting access to $post_id
* @param int $post_id
* @return str base36 token
* @TODO expiry time, link token to invoice
*/
protected static function make_token($post_id) {
return base_convert(hash_hmac('sha256', $post_id, LIGHTNING_PUBLISHER_KEY), 16, 36);
}
/**
* Check whether the current visitor has access to $post_id
* @param int $post_id
* @return bool
*/
protected static function check_payment($post_id) {
return isset($_GET['publisher_access']) && self::make_token($post_id) === $_GET['publisher_access'];
}
/**
* Parse [ifpaid] tags and return as structured data
* Expected format: [ifpaid AMOUNT CURRENCY KEY=VAL]
* @param string $content
* @return array
*/
protected static function extract_ifpaid_tag($content) {
if (!preg_match('/\[ifpaid [\d.]+ [a-z]+.*?\]/i', $content, $m)) return;
$tag = html_entity_decode(str_replace(array('“', '”'), '"', $m[0]));
if (substr($tag, -2, 1) !== ' ') $tag = substr($tag, 0, -1) . ' ]';
$attrs = shortcode_parse_atts($tag);
return (object)[ 'tag' => $m[0], 'amount' => $attrs[1], 'currency' => $attrs[2], 'attrs' => $attrs ];
}
/**
* Format display for paid post
*/
protected static function format_paid($post_id, $ifpaid, $public, $protected) {
$text = isset($ifpaid->attrs['thanks']) ? $ifpaid->attrs['thanks']
: "<p>Thank you for paying! The rest of the post is available below.</p><p>To return to this content later, please add this page to your bookmarks (Ctrl-d).</p>";
return sprintf('%s<div class="ln-publisher-paid" id="paid">%s</div>%s', $public, $text, $protected);
}
/**
* Format display for unpaid post
*/
protected static function format_unpaid($post_id, $ifpaid, $public) {
$attrs = $ifpaid->attrs;
$text = '<p>' . sprintf(!isset($attrs['text']) ? 'To continue reading the rest of this post, please pay <em>%s</em>.' : $attrs['text'], $ifpaid->amount . ' ' . $ifpaid->currency).'</p>';
$button = sprintf('<a class="ln-publisher-btn" href="#" data-publisher-postid="%d">%s</a>', $post_id, !isset($attrs['button']) ? 'Pay to continue reading' : $attrs['button']);
return sprintf('%s<div class="ln-publisher-pay">%s%s</div>', $public, $text, $button);
}
/**
* Admin settings page
*/
public function admin_menu() {
add_options_page('Lightning Publisher Settings', 'Lightning Publisher',
'manage_options', 'ln_publisher', array($this, 'admin_page'));
}
public function admin_init() {
register_setting('ln_publisher', 'ln_publisher');
add_settings_section('ln_publisher_server', 'Lightning Charge Server', null, 'ln_publisher');
add_settings_field('ln_publisher_server_url', 'Server URL', array($this, 'field_server_url'), 'ln_publisher', 'ln_publisher_server');
add_settings_field('ln_publisher_server_public_url', 'Public URL', array($this, 'field_public_url'), 'ln_publisher', 'ln_publisher_server');
add_settings_field('ln_publisher_token', 'API token', array($this, 'field_token'), 'ln_publisher', 'ln_publisher_server');
}
public function admin_page() {
?>
<div class="wrap">
<h1>Lightning Publisher Settings</h1>
<form method="post" action="options.php">
<?php
settings_fields('ln_publisher');
do_settings_sections('ln_publisher');
submit_button();
?>
</form>
</div>
<?php
}
public function field_server_url(){
printf('<input type="text" name="ln_publisher[server_url]" value="%s" />', esc_attr($this->options['server_url']));
}
public function field_public_url(){
printf('<input type="text" name="ln_publisher[public_url]" value="%s" /><br><label>%s</label>', esc_attr($this->options['public_url']),
'URL where Lightning Charge is publicily accessible to users. Optional, defaults to Server URL.');
}
public function field_token(){
printf('<input type="text" name="ln_publisher[api_token]" value="%s" />', esc_attr($this->options['api_token']));
}
}
new Lightning_Publisher();