Version Description
- Documentation changes
Download this release
Release Info
Developer | unbouncewordpress |
Plugin | Unbounce Landing Pages |
Version | 0.1.5 |
Comparing to | |
See all releases |
Version 0.1.5
- UBConfig.php +179 -0
- UBHTTP.php +231 -0
- UBIcon.php +9 -0
- UBLogger.php +118 -0
- UBUtil.php +14 -0
- Unbounce-Page.php +153 -0
- readme.txt +73 -0
UBConfig.php
ADDED
@@ -0,0 +1,179 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
class UBConfig {
|
4 |
+
|
5 |
+
const UB_PLUGIN_NAME = 'ub-wordpress';
|
6 |
+
const UB_ROUTES_CACHE_KEY = 'ub-route-cache';
|
7 |
+
const UB_REMOTE_DEBUG_KEY = 'ub-remote-debug';
|
8 |
+
const UB_CACHE_TIMEOUT_ENV_KEY = 'UB_WP_ROUTES_CACHE_EXP';
|
9 |
+
const UB_USER_AGENT = 'Unbounce WP Plugin 0.1.5';
|
10 |
+
const UB_VERSION = '0.1.5';
|
11 |
+
|
12 |
+
public static function get_page_server_domain() {
|
13 |
+
$domain = getenv('UB_PAGE_SERVER_DOMAIN');
|
14 |
+
return $domain ? $domain : 'wp.unbounce.com';
|
15 |
+
}
|
16 |
+
|
17 |
+
public static function get_remote_log_url() {
|
18 |
+
$url = getenv('UB_REMOTE_LOG_URL');
|
19 |
+
if ($url == null) {
|
20 |
+
return 'https://events-gateway.unbounce.com/events/wordpress_logs';
|
21 |
+
}
|
22 |
+
return $url;
|
23 |
+
}
|
24 |
+
|
25 |
+
public static function debug_loggging_enabled() {
|
26 |
+
return WP_DEBUG || WP_DEBUG_LOG || UBConfig::remote_debug_logging_enabled();
|
27 |
+
}
|
28 |
+
|
29 |
+
public static function remote_debug_logging_enabled() {
|
30 |
+
return get_option(UBConfig::UB_REMOTE_DEBUG_KEY, 0) == 1;
|
31 |
+
}
|
32 |
+
|
33 |
+
public static function fetch_proxyable_url_set($domain, $etag) {
|
34 |
+
if(!$domain) {
|
35 |
+
UBLogger::warning('Domain not provided, not fetching wp-routes.json');
|
36 |
+
return array('FAILURE', null, null, null);
|
37 |
+
}
|
38 |
+
|
39 |
+
$url = 'https://' . UBConfig::get_page_server_domain() . '/wp-routes.json';
|
40 |
+
$curl = curl_init();
|
41 |
+
$curl_options = array(
|
42 |
+
CURLOPT_URL => $url,
|
43 |
+
CURLOPT_CUSTOMREQUEST => "GET",
|
44 |
+
CURLOPT_HEADER => true,
|
45 |
+
CURLOPT_USERAGENT => UBConfig::UB_USER_AGENT,
|
46 |
+
CURLOPT_HTTPHEADER => array('Host: ' . $domain, 'If-None-Match: ' . $etag),
|
47 |
+
CURLOPT_RETURNTRANSFER => true,
|
48 |
+
CURLOPT_FOLLOWLOCATION => false,
|
49 |
+
CURLOPT_TIMEOUT => 5
|
50 |
+
);
|
51 |
+
|
52 |
+
UBLogger::debug("Retrieving routes from '$url', etag: '$etag', host: '$domain'");
|
53 |
+
|
54 |
+
curl_setopt_array($curl, $curl_options);
|
55 |
+
$data = curl_exec($curl);
|
56 |
+
|
57 |
+
$http_code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
|
58 |
+
$curl_error = null;
|
59 |
+
|
60 |
+
// when having an CURL error, http_code is 0
|
61 |
+
if ($http_code == 0) {
|
62 |
+
$curl_error = curl_error($curl);
|
63 |
+
}
|
64 |
+
|
65 |
+
curl_close($curl);
|
66 |
+
|
67 |
+
list($headers, $body) = array_pad(explode("\r\n\r\n", $data, 2), 2, null);
|
68 |
+
|
69 |
+
$matches = array();
|
70 |
+
$does_match = preg_match('/ETag: (\S+)/is', $headers, $matches);
|
71 |
+
if ($does_match) {
|
72 |
+
$etag = $matches[1];
|
73 |
+
}
|
74 |
+
|
75 |
+
$matches = array();
|
76 |
+
$does_match = preg_match('/Cache-Control: max-age=(\S+)/is', $headers, $matches);
|
77 |
+
if ($does_match) {
|
78 |
+
$max_age = $matches[1];
|
79 |
+
}
|
80 |
+
|
81 |
+
if ($http_code == 200) {
|
82 |
+
$json_body = json_decode($body);
|
83 |
+
|
84 |
+
if (is_null($json_body)) {
|
85 |
+
$json_error = json_last_error();
|
86 |
+
UBLogger::warning("An error occurred while processing routes, JSON error: '$json_error'");
|
87 |
+
return array('FAILURE', null, null, null);
|
88 |
+
}
|
89 |
+
else {
|
90 |
+
UBLogger::debug("Retrieved new routes, HTTP code: '$http_code'");
|
91 |
+
return array('NEW', $etag, $max_age, $json_body);
|
92 |
+
}
|
93 |
+
}
|
94 |
+
if ($http_code == 304) {
|
95 |
+
UBLogger::debug("Routes have not changed, HTTP code: '$http_code'");
|
96 |
+
return array('SAME', $etag, $max_age, null);
|
97 |
+
}
|
98 |
+
if ($http_code == 404) {
|
99 |
+
UBLogger::debug("No routes to retrieve, HTTP code: '$http_code'");
|
100 |
+
return array('NONE', null, null, null);
|
101 |
+
}
|
102 |
+
else {
|
103 |
+
UBLogger::warning("An error occurred while retrieving routes; HTTP code: '$http_code'; Error: " . $curl_error);
|
104 |
+
return array('FAILURE', null, null, null);
|
105 |
+
}
|
106 |
+
}
|
107 |
+
|
108 |
+
public static function _read_unbounce_domain_info($cache_getter,
|
109 |
+
$cache_setter,
|
110 |
+
$fetch_proxyable_url_set,
|
111 |
+
$domain,
|
112 |
+
$expire_now=false) {
|
113 |
+
|
114 |
+
$proxyable_url_set = null;
|
115 |
+
|
116 |
+
$cache_max_time_default = 10;
|
117 |
+
|
118 |
+
$domains_info = $cache_getter(UBConfig::UB_ROUTES_CACHE_KEY);
|
119 |
+
$domain_info = UBUtil::array_fetch($domains_info, $domain, array());
|
120 |
+
|
121 |
+
$proxyable_url_set = UBUtil::array_fetch($domain_info, 'proxyable_url_set');
|
122 |
+
$proxyable_url_set_fetched_at = UBUtil::array_fetch($domain_info, 'proxyable_url_set_fetched_at');
|
123 |
+
$proxyable_url_set_cache_timeout = UBUtil::array_fetch($domain_info, 'proxyable_url_set_cache_timeout');
|
124 |
+
$proxyable_url_set_etag = UBUtil::array_fetch($domain_info, 'proxyable_url_set_etag');
|
125 |
+
|
126 |
+
$cache_max_time = is_null($proxyable_url_set_cache_timeout) ? $cache_max_time_default : $proxyable_url_set_cache_timeout;
|
127 |
+
|
128 |
+
$current_time = time();
|
129 |
+
|
130 |
+
if ($expire_now ||
|
131 |
+
is_null($proxyable_url_set) ||
|
132 |
+
($current_time - $proxyable_url_set_fetched_at > $cache_max_time)) {
|
133 |
+
|
134 |
+
$result_array = call_user_func($fetch_proxyable_url_set, $domain, $proxyable_url_set_etag);
|
135 |
+
|
136 |
+
list($routes_status, $etag, $max_age, $proxyable_url_set_new) = $result_array;
|
137 |
+
|
138 |
+
if ($routes_status == 'NEW') {
|
139 |
+
$domain_info['proxyable_url_set'] = $proxyable_url_set_new;
|
140 |
+
$domain_info['proxyable_url_set_etag'] = $etag;
|
141 |
+
$domain_info['proxyable_url_set_cache_timeout'] = $max_age;
|
142 |
+
}
|
143 |
+
elseif ($routes_status == 'SAME') {
|
144 |
+
// Just extend the cache
|
145 |
+
$domain_info['proxyable_url_set_cache_timeout'] = $max_age;
|
146 |
+
}
|
147 |
+
elseif ($routes_status == 'NONE') {
|
148 |
+
$domain_info['proxyable_url_set'] = array();
|
149 |
+
$domain_info['proxyable_url_set_etag'] = null;
|
150 |
+
}
|
151 |
+
elseif ($routes_status == 'FAILURE') {
|
152 |
+
UBLogger::warning('Route fetching failed');
|
153 |
+
}
|
154 |
+
else {
|
155 |
+
UBLogger::warning("Unknown response from route fetcher: '$routes_status'");
|
156 |
+
}
|
157 |
+
|
158 |
+
$domain_info['proxyable_url_set_fetched_at'] = $current_time;
|
159 |
+
$domains_info[$domain] = $domain_info;
|
160 |
+
$cache_setter(UBConfig::UB_ROUTES_CACHE_KEY, $domains_info);
|
161 |
+
}
|
162 |
+
|
163 |
+
|
164 |
+
return UBUtil::array_select_by_key($domain_info,
|
165 |
+
array('proxyable_url_set',
|
166 |
+
'proxyable_url_set_fetched_at'));
|
167 |
+
}
|
168 |
+
|
169 |
+
public static function read_unbounce_domain_info($domain, $expire_now) {
|
170 |
+
return UBConfig::_read_unbounce_domain_info(
|
171 |
+
'get_option',
|
172 |
+
'update_option',
|
173 |
+
'UBConfig::fetch_proxyable_url_set',
|
174 |
+
$domain,
|
175 |
+
$expire_now);
|
176 |
+
}
|
177 |
+
|
178 |
+
}
|
179 |
+
?>
|
UBHTTP.php
ADDED
@@ -0,0 +1,231 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
class UBHTTP {
|
4 |
+
private static $powered_by_header_regex = '/^X-Powered-By: (.+)$/i';
|
5 |
+
private static $form_confirmation_url_regex = '/(.+)\/[a-z]+-form_confirmation\.html/';
|
6 |
+
private static $forward_headers = '/^(Content-Type:|Location:|ETag:|Last-Modified:|Link:|Content-Location:|Set-Cookie:|X-Server-Instance:|X-Unbounce-PageId:|X-Unbounce-Variant:|X-Unbounce-VisitorID:)/i';
|
7 |
+
|
8 |
+
|
9 |
+
public static function is_private_ip_address($ip_address) {
|
10 |
+
return !filter_var($ip_address,
|
11 |
+
FILTER_VALIDATE_IP,
|
12 |
+
FILTER_FLAG_NO_PRIV_RANGE + FILTER_FLAG_NO_RES_RANGE);
|
13 |
+
}
|
14 |
+
|
15 |
+
public static function cookie_string_from_array($cookies) {
|
16 |
+
$join_cookie_values = function ($k, $v) { return $k . '=' . $v; };
|
17 |
+
$cookie_strings = array_map($join_cookie_values,
|
18 |
+
array_keys($cookies),
|
19 |
+
$cookies);
|
20 |
+
return join('; ', $cookie_strings);
|
21 |
+
}
|
22 |
+
|
23 |
+
private static function fetch_header_value_function($regex) {
|
24 |
+
return function ($header_string) use ($regex) {
|
25 |
+
$matches = array();
|
26 |
+
preg_match($regex,
|
27 |
+
$header_string,
|
28 |
+
$matches);
|
29 |
+
return $matches[1];
|
30 |
+
};
|
31 |
+
}
|
32 |
+
|
33 |
+
public static function rewrite_x_powered_by_header($header_string, $existing_headers) {
|
34 |
+
$fetch_powered_by_value = UBHTTP::fetch_header_value_function(UBHTTP::$powered_by_header_regex);
|
35 |
+
|
36 |
+
$existing_powered_by = preg_grep(UBHTTP::$powered_by_header_regex,
|
37 |
+
$existing_headers);
|
38 |
+
|
39 |
+
$existing_powered_by = array_map($fetch_powered_by_value,
|
40 |
+
$existing_powered_by);
|
41 |
+
|
42 |
+
return 'X-Powered-By: ' .
|
43 |
+
join($existing_powered_by, ', ') . ', ' .
|
44 |
+
$fetch_powered_by_value($header_string);
|
45 |
+
}
|
46 |
+
|
47 |
+
public static function get_proxied_for_header($out_headers,
|
48 |
+
$request_headers,
|
49 |
+
$current_ip) {
|
50 |
+
$forwarded_for = UBUtil::array_fetch($request_headers, 'X-Forwarded-For');
|
51 |
+
|
52 |
+
if($forwarded_for !== null && UBHTTP::is_private_ip_address($current_ip)) {
|
53 |
+
$proxied_for = $forwarded_for;
|
54 |
+
} else {
|
55 |
+
$proxied_for = $current_ip;
|
56 |
+
}
|
57 |
+
|
58 |
+
$out_headers[] = 'X-Proxied-For: ' . $proxied_for;
|
59 |
+
return $out_headers;
|
60 |
+
}
|
61 |
+
|
62 |
+
public static function stream_headers_function($existing_headers) {
|
63 |
+
return function ($curl, $header_string) use ($existing_headers) {
|
64 |
+
$http_status_code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
|
65 |
+
|
66 |
+
http_response_code($http_status_code);
|
67 |
+
|
68 |
+
if(preg_match(UBHTTP::$powered_by_header_regex, $header_string) == 1) {
|
69 |
+
$result = UBHTTP::rewrite_x_powered_by_header($header_string, $existing_headers);
|
70 |
+
header($result);
|
71 |
+
|
72 |
+
} elseif (preg_match(UBHTTP::$forward_headers, $header_string)) {
|
73 |
+
// false means don't replace the exsisting header
|
74 |
+
header($header_string, false);
|
75 |
+
}
|
76 |
+
|
77 |
+
// We must show curl that we've processed every byte of the input header
|
78 |
+
return strlen($header_string);
|
79 |
+
};
|
80 |
+
}
|
81 |
+
|
82 |
+
public static function stream_response_function() {
|
83 |
+
return function ($curl, $string) {
|
84 |
+
// Stream the body to the client
|
85 |
+
echo $string;
|
86 |
+
|
87 |
+
// We must show curl that we've processed every byte of the input string
|
88 |
+
return strlen($string);
|
89 |
+
};
|
90 |
+
}
|
91 |
+
|
92 |
+
public static function determine_protocol($server_global, $wp_is_ssl) {
|
93 |
+
$forwarded_proto = UBUtil::array_fetch($server_global, 'HTTP_X_FORWARDED_PROTO');
|
94 |
+
$request_scheme = UBUtil::array_fetch($server_global, 'REQUEST_SCHEME');
|
95 |
+
$script_uri = UBUtil::array_fetch($server_global, 'SCRIPT_URI');
|
96 |
+
$script_uri_scheme = parse_url($script_uri, PHP_URL_SCHEME);
|
97 |
+
$https = UBUtil::array_fetch($server_global, 'HTTPS', 'off');
|
98 |
+
|
99 |
+
// X-Forwarded-Proto should be respected first, as it is what the end
|
100 |
+
// user will see (if Wordpress is behind a load balancer).
|
101 |
+
if(UBHTTP::is_valid_protocol($forwarded_proto)) {
|
102 |
+
return $forwarded_proto . '://';
|
103 |
+
}
|
104 |
+
// Next use REQUEST_SCHEME, if it is available. This is the recommended way
|
105 |
+
// to get the protocol, but it is not available on all hosts.
|
106 |
+
elseif(UBHTTP::is_valid_protocol($request_scheme)) {
|
107 |
+
return $request_scheme . '://';
|
108 |
+
}
|
109 |
+
// Next try to pull it out of the SCRIPT_URI. This is also not always available.
|
110 |
+
elseif(UBHTTP::is_valid_protocol($script_uri_scheme)) {
|
111 |
+
return $script_uri_scheme . '://';
|
112 |
+
}
|
113 |
+
// Wordpress' is_ssl() may return the correct boolean for http/https if
|
114 |
+
// the site was setup properly.
|
115 |
+
elseif($wp_is_ssl || !is_null($https) && $https !== 'off') {
|
116 |
+
return 'https://';
|
117 |
+
}
|
118 |
+
// We default to http as most HTTPS sites will also have HTTP available.
|
119 |
+
else {
|
120 |
+
return 'http://';
|
121 |
+
}
|
122 |
+
}
|
123 |
+
|
124 |
+
private static function is_valid_protocol($protocol) {
|
125 |
+
return $protocol === 'http' || $protocol === 'https';
|
126 |
+
}
|
127 |
+
|
128 |
+
public static function stream_request($method,
|
129 |
+
$target_url,
|
130 |
+
$cookie_string,
|
131 |
+
$headers0,
|
132 |
+
$post_body = null,
|
133 |
+
$user_agent) {
|
134 |
+
|
135 |
+
$existing_headers = headers_list();
|
136 |
+
$request_headers = getallheaders();
|
137 |
+
$remote_ip = $_SERVER['REMOTE_ADDR'];
|
138 |
+
|
139 |
+
$headers = UBHTTP::get_proxied_for_header($headers0,
|
140 |
+
$request_headers,
|
141 |
+
$remote_ip);
|
142 |
+
|
143 |
+
UBLogger::debug_var('target_url', $target_url);
|
144 |
+
|
145 |
+
$stream_headers = UBHTTP::stream_headers_function($existing_headers);
|
146 |
+
$stream_body = UBHTTP::stream_response_function();
|
147 |
+
$curl = curl_init();
|
148 |
+
// http://php.net/manual/en/function.curl-setopt.php
|
149 |
+
$curl_options = array(
|
150 |
+
CURLOPT_URL => $target_url,
|
151 |
+
CURLOPT_POST => $method == "POST",
|
152 |
+
CURLOPT_CUSTOMREQUEST => $method,
|
153 |
+
CURLOPT_USERAGENT => $user_agent,
|
154 |
+
CURLOPT_COOKIE => $cookie_string,
|
155 |
+
CURLOPT_HTTPHEADER => $headers,
|
156 |
+
CURLOPT_HEADERFUNCTION => $stream_headers,
|
157 |
+
CURLOPT_WRITEFUNCTION => $stream_body,
|
158 |
+
CURLOPT_FOLLOWLOCATION => false,
|
159 |
+
CURLOPT_TIMEOUT => 5
|
160 |
+
);
|
161 |
+
|
162 |
+
if ($method == "POST" && $post_body != null) {
|
163 |
+
$curl_options[CURLOPT_POSTFIELDS] = http_build_query($post_body);
|
164 |
+
}
|
165 |
+
|
166 |
+
curl_setopt_array($curl, $curl_options);
|
167 |
+
$resp = curl_exec($curl);
|
168 |
+
if(!$resp){
|
169 |
+
$message = 'Error proxying to "' . $target_url . ", " . $original_target_url
|
170 |
+
. '": "' . curl_error($curl) . '" - Code: ' . curl_errno($curl);
|
171 |
+
UBLogger::warning($message);
|
172 |
+
http_response_code(500);
|
173 |
+
}
|
174 |
+
curl_close($curl);
|
175 |
+
}
|
176 |
+
|
177 |
+
public static function is_extract_url_proxyable($proxyable_url_set,
|
178 |
+
$extract_regex,
|
179 |
+
$match_position,
|
180 |
+
$url) {
|
181 |
+
$matches = array();
|
182 |
+
$does_match = preg_match($extract_regex,
|
183 |
+
$url,
|
184 |
+
$matches);
|
185 |
+
|
186 |
+
return $does_match && in_array($matches[1], $proxyable_url_set);
|
187 |
+
}
|
188 |
+
|
189 |
+
public static function is_confirmation_dialog($proxyable_url_set, $url_without_protocol) {
|
190 |
+
return UBHTTP::is_extract_url_proxyable($proxyable_url_set,
|
191 |
+
UBHTTP::$form_confirmation_url_regex,
|
192 |
+
1,
|
193 |
+
$url_without_protocol);
|
194 |
+
}
|
195 |
+
|
196 |
+
public static function is_tracking_link($proxyable_url_set, $url_without_protocol) {
|
197 |
+
return UBHTTP::is_extract_url_proxyable($proxyable_url_set,
|
198 |
+
"/^(.+)?\/(clkn|clkg)\/?/",
|
199 |
+
1,
|
200 |
+
$url_without_protocol);
|
201 |
+
}
|
202 |
+
|
203 |
+
public static function get_url_purpose($proxyable_url_set, $http_method, $url) {
|
204 |
+
$host = parse_url($url, PHP_URL_HOST);
|
205 |
+
$path = parse_url($url, PHP_URL_PATH);
|
206 |
+
$url_without_protocol = $host . $path;
|
207 |
+
|
208 |
+
if ($http_method == "POST" &&
|
209 |
+
preg_match("/^\/(fsn|fsg|fs)\/?$/", $path)) {
|
210 |
+
|
211 |
+
return "SubmitLead";
|
212 |
+
|
213 |
+
} elseif ($http_method == "GET" &&
|
214 |
+
UBHTTP::is_tracking_link($proxyable_url_set, $url_without_protocol)) {
|
215 |
+
|
216 |
+
return "TrackClick";
|
217 |
+
|
218 |
+
} elseif ($http_method == "GET" &&
|
219 |
+
(in_array($url_without_protocol, $proxyable_url_set) ||
|
220 |
+
UBHTTP::is_confirmation_dialog($proxyable_url_set, $url_without_protocol))) {
|
221 |
+
|
222 |
+
return "ViewLandingPage";
|
223 |
+
|
224 |
+
} else {
|
225 |
+
return null;
|
226 |
+
}
|
227 |
+
}
|
228 |
+
|
229 |
+
}
|
230 |
+
|
231 |
+
?>
|
UBIcon.php
ADDED
@@ -0,0 +1,9 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
class UBIcon {
|
4 |
+
public static function base64_encoded_svg() {
|
5 |
+
return '';
|
6 |
+
}
|
7 |
+
}
|
8 |
+
|
9 |
+
?>
|
UBLogger.php
ADDED
@@ -0,0 +1,118 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
require_once dirname(__FILE__) . '/UBConfig.php';
|
4 |
+
|
5 |
+
class UBLogger {
|
6 |
+
|
7 |
+
// should be called when the plugin is loaded
|
8 |
+
public static function setup_logger() {
|
9 |
+
if(!isset($GLOBALS['wp_log_plugins'])) {
|
10 |
+
$GLOBALS['wp_log_plugins'] = array();
|
11 |
+
}
|
12 |
+
$GLOBALS['wp_log_plugins'][UBConfig::UB_PLUGIN_NAME] = array();
|
13 |
+
$GLOBALS['wp_log_plugins'][UBConfig::UB_PLUGIN_NAME . '-vars'] = array();
|
14 |
+
}
|
15 |
+
|
16 |
+
public static function upload_logs_to_unbounce($url) {
|
17 |
+
if(UBConfig::remote_debug_logging_enabled()) {
|
18 |
+
$datetime = new DateTime('NOW', new DateTimeZone('UTC'));
|
19 |
+
$data = array(
|
20 |
+
'type' => 'WordpressLogV1.0',
|
21 |
+
'messages' => $GLOBALS['wp_log'][UBConfig::UB_PLUGIN_NAME],
|
22 |
+
'vars' => $GLOBALS['wp_log'][UBConfig::UB_PLUGIN_NAME . '-vars'],
|
23 |
+
'id' => uniqid(),
|
24 |
+
'time_sent' => $datetime->format('Y-m-d\TH:i:s.000\Z'),
|
25 |
+
'source' => UBConfig::UB_USER_AGENT . ' ' . gethostname()
|
26 |
+
);
|
27 |
+
$data_string = json_encode($data, JSON_UNESCAPED_SLASHES);
|
28 |
+
|
29 |
+
$curl = curl_init();
|
30 |
+
$curl_options = array(
|
31 |
+
CURLOPT_URL => $url,
|
32 |
+
CURLOPT_CUSTOMREQUEST => 'POST',
|
33 |
+
CURLOPT_USERAGENT => UBConfig::UB_USER_AGENT,
|
34 |
+
CURLOPT_FOLLOWLOCATION => false,
|
35 |
+
CURLOPT_HTTPHEADER => array(
|
36 |
+
'Content-Type: application/json',
|
37 |
+
'Content-Length: ' . strlen($data_string)
|
38 |
+
),
|
39 |
+
CURLOPT_POSTFIELDS => $data_string,
|
40 |
+
CURLOPT_TIMEOUT => 2
|
41 |
+
);
|
42 |
+
curl_setopt_array($curl, $curl_options);
|
43 |
+
$success = curl_exec($curl);
|
44 |
+
$http_code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
|
45 |
+
|
46 |
+
if(!$success) {
|
47 |
+
$message = 'Unable to send log messages to ' . $url . ': "'
|
48 |
+
. curl_error($curl) . '" - HTTP status: ' . curl_errno($curl);
|
49 |
+
UBLogger::warning($message);
|
50 |
+
} elseif($http_code >= 200 && $http_code < 300) {
|
51 |
+
$message = 'Successfully sent log messsages to ' . $url
|
52 |
+
. ' - HTTP status: ' . $http_code;
|
53 |
+
UBLogger::debug($message);
|
54 |
+
} else {
|
55 |
+
$message = 'Unable to send log messages to ' . $url
|
56 |
+
. ' - HTTP status: ' . $http_code;
|
57 |
+
UBLogger::warning($message);
|
58 |
+
}
|
59 |
+
|
60 |
+
curl_close($curl);
|
61 |
+
}
|
62 |
+
}
|
63 |
+
|
64 |
+
public static function format_log_entry($level, $msg) {
|
65 |
+
$msg = is_string($msg) ? $msg : print_r($msg, true);
|
66 |
+
return '[' . UBConfig::UB_PLUGIN_NAME . '] [' . $level . '] ' . $msg;
|
67 |
+
}
|
68 |
+
|
69 |
+
private static function log_wp_log($log_entry) {
|
70 |
+
$GLOBALS['wp_log'][UBConfig::UB_PLUGIN_NAME][] = $log_entry;
|
71 |
+
}
|
72 |
+
|
73 |
+
private static function log_wp_log_var($var, $val) {
|
74 |
+
$GLOBALS['wp_log'][UBConfig::UB_PLUGIN_NAME . '-vars'][$var] = $val;
|
75 |
+
}
|
76 |
+
|
77 |
+
private static function log_error_log($log_entry) {
|
78 |
+
error_log($log_entry);
|
79 |
+
}
|
80 |
+
|
81 |
+
public static function log($level, $msg) {
|
82 |
+
if(UBConfig::debug_loggging_enabled()) {
|
83 |
+
$log_entry = UBLogger::format_log_entry($level, $msg);
|
84 |
+
UBLogger::log_wp_log($log_entry);
|
85 |
+
UBLogger::log_error_log($log_entry);
|
86 |
+
}
|
87 |
+
}
|
88 |
+
|
89 |
+
public static function log_var($level, $var, $val) {
|
90 |
+
if(UBConfig::debug_loggging_enabled()) {
|
91 |
+
UBLogger::log($level, '$' . $var . ': ' . $val);
|
92 |
+
UBLogger::log_wp_log_var($var, $val);
|
93 |
+
}
|
94 |
+
}
|
95 |
+
|
96 |
+
public static function info($msg) {
|
97 |
+
UBLogger::log('INFO', $msg);
|
98 |
+
}
|
99 |
+
|
100 |
+
public static function warning($msg) {
|
101 |
+
UBLogger::log('WARNING', $msg);
|
102 |
+
}
|
103 |
+
|
104 |
+
public static function debug($msg) {
|
105 |
+
UBLogger::log('DEBUG', $msg);
|
106 |
+
}
|
107 |
+
|
108 |
+
public static function debug_var($var, $val) {
|
109 |
+
UBLogger::log_var('DEBUG', $var, $val);
|
110 |
+
}
|
111 |
+
|
112 |
+
public static function config($msg) {
|
113 |
+
UBLogger::log('CONFIG', $msg);
|
114 |
+
}
|
115 |
+
|
116 |
+
}
|
117 |
+
|
118 |
+
?>
|
UBUtil.php
ADDED
@@ -0,0 +1,14 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
class UBUtil {
|
4 |
+
|
5 |
+
public static function array_select_by_key($input, $keep) {
|
6 |
+
return array_intersect_key($input, array_flip($keep));
|
7 |
+
}
|
8 |
+
|
9 |
+
public static function array_fetch($array, $index, $default = null) {
|
10 |
+
return isset($array[$index]) ? $array[$index] : $default;
|
11 |
+
}
|
12 |
+
|
13 |
+
}
|
14 |
+
?>
|
Unbounce-Page.php
ADDED
@@ -0,0 +1,153 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
/*
|
3 |
+
Plugin Name: Unbounce
|
4 |
+
Plugin URI: http://unbounce.com
|
5 |
+
Description: Publish Unbounce Landing Pages to your Wordpress Domain.
|
6 |
+
Version: 0.1.5
|
7 |
+
Author: Unbounce
|
8 |
+
Author URI: http://unbounce.com
|
9 |
+
License: GPLv2
|
10 |
+
*/
|
11 |
+
|
12 |
+
require_once dirname(__FILE__) . '/UBUtil.php';
|
13 |
+
require_once dirname(__FILE__) . '/UBConfig.php';
|
14 |
+
require_once dirname(__FILE__) . '/UBLogger.php';
|
15 |
+
require_once dirname(__FILE__) . '/UBHTTP.php';
|
16 |
+
require_once dirname(__FILE__). '/UBIcon.php';
|
17 |
+
|
18 |
+
register_activation_hook(__FILE__, function() {
|
19 |
+
add_option(UBConfig::UB_ROUTES_CACHE_KEY, array());
|
20 |
+
add_option(UBConfig::UB_REMOTE_DEBUG_KEY, 0);
|
21 |
+
});
|
22 |
+
|
23 |
+
register_deactivation_hook(__FILE__, function() {
|
24 |
+
delete_option(UBConfig::UB_ROUTES_CACHE_KEY);
|
25 |
+
delete_option(UBConfig::UB_REMOTE_DEBUG_KEY);
|
26 |
+
});
|
27 |
+
|
28 |
+
add_action('init', function() {
|
29 |
+
UBLogger::setup_logger();
|
30 |
+
|
31 |
+
$start = microtime(true);
|
32 |
+
|
33 |
+
$ps_domain = UBConfig::get_page_server_domain();
|
34 |
+
$http_method = UBUtil::array_fetch($_SERVER, 'REQUEST_METHOD');
|
35 |
+
$referer = UBUtil::array_fetch($_SERVER, 'HTTP_REFERER');
|
36 |
+
$user_agent = UBUtil::array_fetch($_SERVER, 'HTTP_USER_AGENT');
|
37 |
+
$protocol = UBHTTP::determine_protocol($_SERVER, is_ssl());
|
38 |
+
$domain = UBUtil::array_fetch($_SERVER, 'HTTP_HOST');
|
39 |
+
$current_path = UBUtil::array_fetch($_SERVER, 'REQUEST_URI');
|
40 |
+
|
41 |
+
$raw_url = $protocol . $ps_domain . $current_path;
|
42 |
+
$current_url = trim($protocol . $domain . $current_path, '/');
|
43 |
+
|
44 |
+
$domain_info = UBConfig::read_unbounce_domain_info($domain, false);
|
45 |
+
$proxyable_url_set = UBUtil::array_fetch($domain_info, 'proxyable_url_set', array());
|
46 |
+
|
47 |
+
UBLogger::debug_var('ps_domain', $ps_domain);
|
48 |
+
UBLogger::debug_var('http_method', $http_method);
|
49 |
+
UBLogger::debug_var('referer', $referer);
|
50 |
+
UBLogger::debug_var('user_agent', $user_agent);
|
51 |
+
UBLogger::debug_var('protocol', $protocol);
|
52 |
+
UBLogger::debug_var('domain', $domain);
|
53 |
+
UBLogger::debug_var('current_path', $current_path);
|
54 |
+
UBLogger::debug_var('raw_url', $raw_url);
|
55 |
+
UBLogger::debug_var('current_url ', $current_url );
|
56 |
+
|
57 |
+
////////////////////
|
58 |
+
|
59 |
+
if ($proxyable_url_set == null) {
|
60 |
+
UBLogger::warning("wp-routes.json not found for domain " . $domain);
|
61 |
+
}
|
62 |
+
else {
|
63 |
+
$url_purpose = UBHTTP::get_url_purpose($proxyable_url_set,
|
64 |
+
$http_method,
|
65 |
+
$current_url);
|
66 |
+
if ($url_purpose == null) {
|
67 |
+
UBLogger::debug("ignoring request to URL " . $current_url);
|
68 |
+
}
|
69 |
+
else {
|
70 |
+
UBLogger::debug("perform ''" . $url_purpose . "'' on received URL " . $current_url);
|
71 |
+
|
72 |
+
$cookies_to_forward = UBUtil::array_select_by_key($_COOKIE,
|
73 |
+
array('ubvs', 'ubpv', 'ubvt'));
|
74 |
+
|
75 |
+
$cookie_string = UBHTTP::cookie_string_from_array($cookies_to_forward);
|
76 |
+
|
77 |
+
$req_headers = $referer == null ? array('Host: ' . $domain) : array('Referer: ' . $referer, 'Host: ' . $domain);
|
78 |
+
|
79 |
+
// Make sure we don't get cached by Wordpress hosts like WPEngine
|
80 |
+
header('Cache-Control: max-age=0; private');
|
81 |
+
|
82 |
+
UBHTTP::stream_request($http_method,
|
83 |
+
$raw_url,
|
84 |
+
$cookie_string,
|
85 |
+
$req_headers,
|
86 |
+
$_POST,
|
87 |
+
$user_agent);
|
88 |
+
|
89 |
+
$end = microtime(true);
|
90 |
+
$time_taken = ($end - $start) * 1000;
|
91 |
+
|
92 |
+
UBLogger::debug_var('time_taken', $time_taken);
|
93 |
+
UBLogger::debug("proxying for $current_url done successfuly -- took $time_taken ms");
|
94 |
+
|
95 |
+
exit(0);
|
96 |
+
}
|
97 |
+
}
|
98 |
+
});
|
99 |
+
|
100 |
+
function render_unbounce_pages($domain_info) {
|
101 |
+
echo '<h1>Unbounce Pages</h1>';
|
102 |
+
|
103 |
+
$proxyable_url_set = UBUtil::array_fetch($domain_info, 'proxyable_url_set');
|
104 |
+
if(empty($proxyable_url_set)) {
|
105 |
+
echo '<p class="warning">No URLs have been registered from Unbounce</p>';
|
106 |
+
|
107 |
+
} else {
|
108 |
+
$proxyable_url_set_fetched_at = UBUtil::array_fetch($domain_info, 'proxyable_url_set_fetched_at');
|
109 |
+
|
110 |
+
$list_items = array_map(function($url) { return '<li><a href="//'. $url .'">' . $url . '</a></li>'; },
|
111 |
+
$proxyable_url_set);
|
112 |
+
|
113 |
+
echo '<div class="unbounce-page-list">';
|
114 |
+
echo '<ul>' . join($list_items, "\n") . '</ul>';
|
115 |
+
echo '<p>Last refresh date: <span id="last-cache-fetch" style="font-weight: bold;">' . date('r', $proxyable_url_set_fetched_at) . '</span></p>';
|
116 |
+
echo '</div>';
|
117 |
+
|
118 |
+
}
|
119 |
+
|
120 |
+
$flush_pages_url = admin_url('admin-post.php?action=flush_unbounce_pages');
|
121 |
+
echo "<p><a href='$flush_pages_url'>Refresh Cache</a></p>";
|
122 |
+
echo '<p><a href="https://app.unbounce.com">Go to Unbounce</a></p>';
|
123 |
+
}
|
124 |
+
|
125 |
+
add_action('admin_menu', function() {
|
126 |
+
$print_admin_panel = function() {
|
127 |
+
$domain = UBUtil::array_fetch($_SERVER, 'HTTP_HOST');
|
128 |
+
$domain_info = UBConfig::read_unbounce_domain_info($domain, false);
|
129 |
+
render_unbounce_pages($domain_info);
|
130 |
+
};
|
131 |
+
|
132 |
+
add_menu_page('Unbounce Pages',
|
133 |
+
'Unbounce Pages',
|
134 |
+
'manage_options',
|
135 |
+
'unbounce-pages',
|
136 |
+
$print_admin_panel,
|
137 |
+
UBIcon::base64_encoded_svg());
|
138 |
+
});
|
139 |
+
|
140 |
+
add_action('admin_post_flush_unbounce_pages', function() {
|
141 |
+
$domain = UBUtil::array_fetch($_SERVER, 'HTTP_HOST');
|
142 |
+
// Expire cache and redirect
|
143 |
+
$_domain_info = UBConfig::read_unbounce_domain_info($domain, true);
|
144 |
+
status_header(301);
|
145 |
+
$location = admin_url('admin.php?page=unbounce-pages');
|
146 |
+
header("Location: $location");
|
147 |
+
});
|
148 |
+
|
149 |
+
add_action('shutdown', function() {
|
150 |
+
UBLogger::upload_logs_to_unbounce(UBConfig::get_remote_log_url());
|
151 |
+
});
|
152 |
+
|
153 |
+
?>
|
readme.txt
ADDED
@@ -0,0 +1,73 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
=== Plugin Name ===
|
2 |
+
Contributors: unbouncewordpress
|
3 |
+
Tags: unbounce
|
4 |
+
Requires at least: 4.1.5
|
5 |
+
Tested up to: 4.2.2
|
6 |
+
Stable tag: 0.1.5
|
7 |
+
License: GPLv2 or later
|
8 |
+
License URI: http://www.gnu.org/licenses/gpl-2.0.html
|
9 |
+
|
10 |
+
Publish your Unbounce pages to WordPress!
|
11 |
+
|
12 |
+
== Description ==
|
13 |
+
|
14 |
+
Publish your [Unbounce](http://unbounce.com/ "The Mobile Responsive Landing Page Builder for Marketers") pages to WordPress!
|
15 |
+
|
16 |
+
This Unbounce Plug-In is in Beta Testing, if you are interested trying it out
|
17 |
+
please contact our support team at support@unbounce.com
|
18 |
+
|
19 |
+
== Installation ==
|
20 |
+
|
21 |
+
1. Create a Wordpress domain in [Unbounce](http://unbounce.com/ "The Mobile Responsive Landing Page Builder for Marketers")
|
22 |
+
1. Install this plugin through the WordPress store
|
23 |
+
1. Activate this plugin after installation completes
|
24 |
+
|
25 |
+
OR
|
26 |
+
|
27 |
+
1. Create a Wordpress domain in [Unbounce](http://unbounce.com/ "The Mobile Responsive Landing Page Builder for Marketers")
|
28 |
+
1. Upload the zip file via the 'Plugins' menu in WordPress
|
29 |
+
1. Activate this plugin after installation completes
|
30 |
+
|
31 |
+
== Frequently Asked Questions ==
|
32 |
+
|
33 |
+
= How do I join the beta test for this plugin? =
|
34 |
+
|
35 |
+
If you are interested trying out the plugin, please contact our support team at support@unbounce.com.
|
36 |
+
|
37 |
+
= Do I need an Unbounce account? =
|
38 |
+
|
39 |
+
Yes. You need to sign up for Unbounce in order to publish pages. To publish Unbounce pages to your
|
40 |
+
Unbounce site, you will need to add a Wordpress domain in Unbounce. For example, if you Wordpress
|
41 |
+
site is available at www.example.com, you will need to add www.example.com and publish pages in
|
42 |
+
Unbounce to that domain for them to be visible on your Wordpress site.
|
43 |
+
|
44 |
+
= Do I need to log in to Unbounce? =
|
45 |
+
|
46 |
+
No, the plugin will work without any authentication.
|
47 |
+
|
48 |
+
= Does this plugin fetch any data from Unbounce? =
|
49 |
+
|
50 |
+
Yes, this plugin will pull information from Unbounce's servers regarding which pages you have
|
51 |
+
published from Unbounce to your Wordpress site. Any pages that you have published to your Wordpress
|
52 |
+
site in Unbounce will be fetched from Unbounce's servers and displayed on your Wordpress site.
|
53 |
+
If you have a page published in Unbounce and are using the same URL for a Wordpress Page, the
|
54 |
+
Unbounce page will be displayed, not the Wordpress page.
|
55 |
+
|
56 |
+
= Does this plugin send any data to Unbounce? =
|
57 |
+
|
58 |
+
No, not by default. This plugin as an optional "debug" mode which will send diagnostic information to
|
59 |
+
Unbounce when switched on. This feature is disabled when you install the plugin. An Unbounce Customer
|
60 |
+
Success Coach may request that you turn the debug feature on if you are experiencing issues with the plugin
|
61 |
+
to help track down the issue.
|
62 |
+
|
63 |
+
== Screenshots ==
|
64 |
+
|
65 |
+
== Changelog ==
|
66 |
+
|
67 |
+
= 0.1.5 =
|
68 |
+
* Documentation changes
|
69 |
+
|
70 |
+
= 0.1.1 =
|
71 |
+
* Initial release
|
72 |
+
|
73 |
+
== Upgrade Notice ==
|