Version Notes
Plugin corresponds to version 6.0.2 of the Bazaarvoice platform
Download this release
Release Info
Developer | BV DTS |
Extension | Bazaarvoice_Connector |
Version | 6.3.7 |
Comparing to | |
See all releases |
Code changes from version 6.3.6 to 6.3.7
- app/code/local/Bazaarvoice/Connector/Model/ProductFeed/Product.php +1 -1
- app/code/local/Bazaarvoice/Connector/etc/config.xml +1 -1
- docs/Integrations_Magento_for_Bazaarvoice_v6.3.6.pdf +0 -0
- docs/Integrations_Magento_for_Bazaarvoice_v6.3.7.pdf +0 -0
- lib/Bazaarvoice/BVFooter.php +127 -129
- lib/Bazaarvoice/BVUtility.php +212 -254
- lib/Bazaarvoice/bvseosdk.php +1042 -1048
- package.xml +4 -4
app/code/local/Bazaarvoice/Connector/Model/ProductFeed/Product.php
CHANGED
@@ -323,7 +323,7 @@ class Bazaarvoice_Connector_Model_ProductFeed_Product extends Mage_Core_Model_Ab
|
|
323 |
$settingCode = strtolower($customAttribute);
|
324 |
$attributeCode = Mage::getStoreConfig("bazaarvoice/bv_config/product_feed_{$settingCode}_attribute_code");
|
325 |
if ($attributeCode && $productDefault->getData($attributeCode)) {
|
326 |
-
$ioObject->streamWrite(' <'.$customAttribute.'s><'.$customAttribute.'
|
327 |
}
|
328 |
}
|
329 |
|
323 |
$settingCode = strtolower($customAttribute);
|
324 |
$attributeCode = Mage::getStoreConfig("bazaarvoice/bv_config/product_feed_{$settingCode}_attribute_code");
|
325 |
if ($attributeCode && $productDefault->getData($attributeCode)) {
|
326 |
+
$ioObject->streamWrite(' <'.$customAttribute.'s><'.$customAttribute.'><![CDATA[' . $productDefault->getData($attributeCode) . ']]></'.$customAttribute.'></'.$customAttribute."s>\n");
|
327 |
}
|
328 |
}
|
329 |
|
app/code/local/Bazaarvoice/Connector/etc/config.xml
CHANGED
@@ -8,7 +8,7 @@
|
|
8 |
<config>
|
9 |
<modules>
|
10 |
<Bazaarvoice_Connector>
|
11 |
-
<version>6.3.
|
12 |
<depends>
|
13 |
<!-- no dependencies -->
|
14 |
</depends>
|
8 |
<config>
|
9 |
<modules>
|
10 |
<Bazaarvoice_Connector>
|
11 |
+
<version>6.3.7</version>
|
12 |
<depends>
|
13 |
<!-- no dependencies -->
|
14 |
</depends>
|
docs/Integrations_Magento_for_Bazaarvoice_v6.3.6.pdf
DELETED
Binary file
|
docs/Integrations_Magento_for_Bazaarvoice_v6.3.7.pdf
ADDED
Binary file
|
lib/Bazaarvoice/BVFooter.php
CHANGED
@@ -1,129 +1,127 @@
|
|
1 |
-
<?php
|
2 |
-
|
3 |
-
/**
|
4 |
-
* BV PHP SEO SDK Footer
|
5 |
-
*/
|
6 |
-
class BVFooter {
|
7 |
-
const VERSION = '3.2.
|
8 |
-
|
9 |
-
private $base;
|
10 |
-
private $
|
11 |
-
private $
|
12 |
-
|
13 |
-
|
14 |
-
|
15 |
-
*
|
16 |
-
*
|
17 |
-
* @
|
18 |
-
* @param
|
19 |
-
* @param string ($
|
20 |
-
* @
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
$this->
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
*
|
32 |
-
*
|
33 |
-
*
|
34 |
-
*
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
$
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
$exec_time =
|
45 |
-
}
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
$
|
50 |
-
|
51 |
-
$footer
|
52 |
-
$footer .= "\n" . ' <li data-bvseo="
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
$footer
|
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 |
-
$subject_id = $this->base->config['
|
85 |
-
}
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
$footer
|
90 |
-
|
91 |
-
$footer .= "\n" . ' <li data-bvseo="
|
92 |
-
$
|
93 |
-
|
94 |
-
|
95 |
-
$footer .= "\n" . ' <li data-bvseo="
|
96 |
-
$footer .= "\n" . ' <li data-bvseo="
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
$footer .= "\n" . ' <li data-bvseo="
|
101 |
-
$footer .= "\n" . ' <li data-bvseo="
|
102 |
-
$footer .= "\n" . ' <li data-bvseo="
|
103 |
-
$footer .= "\n" . ' <li data-bvseo="
|
104 |
-
$footer .= "\n" . ' <li data-bvseo="
|
105 |
-
$footer .= "\n" . ' <li data-bvseo="
|
106 |
-
$footer .= "\n" . ' <li data-bvseo="
|
107 |
-
$footer .= "\n" . ' <li data-bvseo="
|
108 |
-
$footer .= "\n" . ' <li data-bvseo="
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
$footer .= "\n" . ' <li data-bvseo="
|
114 |
-
$footer .= "\n" . ' <li data-bvseo="
|
115 |
-
$footer .= "\n" . ' <li data-bvseo="
|
116 |
-
$footer .= "\n" . ' <li data-bvseo="
|
117 |
-
$footer .= "\n" . ' <li data-bvseo="
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
123 |
-
|
124 |
-
$footer
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
}
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* BV PHP SEO SDK Footer
|
5 |
+
*/
|
6 |
+
class BVFooter {
|
7 |
+
const VERSION = '3.2.1';
|
8 |
+
|
9 |
+
private $base;
|
10 |
+
private $access_method;
|
11 |
+
private $msg;
|
12 |
+
|
13 |
+
/**
|
14 |
+
* BVFooter Class Constructor
|
15 |
+
*
|
16 |
+
* @access public
|
17 |
+
* @param array ($base) - base class parameters
|
18 |
+
* @param string ($access_method) - access method
|
19 |
+
* @param string ($msg) - build message
|
20 |
+
* @return object
|
21 |
+
*/
|
22 |
+
public function __construct($base, $access_method, $msg) {
|
23 |
+
$this->base = $base;
|
24 |
+
$this->access_method = $access_method;
|
25 |
+
$this->msg = $msg;
|
26 |
+
}
|
27 |
+
|
28 |
+
/**
|
29 |
+
* buildSDKFooter
|
30 |
+
*
|
31 |
+
* Returns hidden SDK footer.
|
32 |
+
*
|
33 |
+
* @access public
|
34 |
+
* @return string Html formatted footer.
|
35 |
+
*/
|
36 |
+
public function buildSDKFooter() {
|
37 |
+
$method_type = !empty($this->base->config['internal_file_path']) ? 'LOCAL' : 'CLOUD';
|
38 |
+
$access_method = $this->access_method;
|
39 |
+
$time_end = microtime(true);
|
40 |
+
|
41 |
+
if (!empty($this->base->start_time)) {
|
42 |
+
$exec_time = round(($time_end - $this->base->start_time) * 1000, 2);
|
43 |
+
} else {
|
44 |
+
$exec_time = 0;
|
45 |
+
}
|
46 |
+
$content_type = mb_strtoupper($this->base->config['content_type']);
|
47 |
+
$subject_type = mb_strtoupper($this->base->config['subject_type']);
|
48 |
+
|
49 |
+
$footer = "\n" . '<ul id="BVSEOSDK_meta" style="display:none !important;">';
|
50 |
+
$footer .= "\n" . ' <li data-bvseo="sdk">bvseo_sdk, p_sdk, ' . self::VERSION . '</li>';
|
51 |
+
$footer .= "\n" . ' <li data-bvseo="sp_mt">' . $method_type . ', ' . $access_method . ', ' . $exec_time . 'ms</li>';
|
52 |
+
$footer .= "\n" . ' <li data-bvseo="ct_st">' . $content_type . ', ' . $subject_type . '</li>';
|
53 |
+
if (!empty($this->msg)) {
|
54 |
+
$footer .= "\n" . ' <li data-bvseo="ms">bvseo-msg: ' . $this->msg . '</li>';
|
55 |
+
}
|
56 |
+
$footer .= "\n" . '</ul>';
|
57 |
+
|
58 |
+
return $footer;
|
59 |
+
}
|
60 |
+
|
61 |
+
/**
|
62 |
+
* buildSDKDebugFooter
|
63 |
+
*
|
64 |
+
* Returns hidden SDK debug footer.
|
65 |
+
*
|
66 |
+
* @access public
|
67 |
+
* @return string Html formatted debug footer.
|
68 |
+
*/
|
69 |
+
public function buildSDKDebugFooter() {
|
70 |
+
$staging = !empty($this->base->config['staging']) ? 'TRUE' : 'FALSE';
|
71 |
+
$testing = !empty($this->base->config['testing']) ? 'TRUE' : 'FALSE';
|
72 |
+
$sdk_enabled = !empty($this->base->config['seo_sdk_enabled']) ? 'TRUE' : 'FALSE';
|
73 |
+
$ssl_enabled = !empty($this->base->config['ssl_enabled']) ? 'TRUE' : 'FALSE';
|
74 |
+
$proxy_host = !empty($this->base->config['proxy_host']) ? $this->base->config['proxy_host'] : 'none';
|
75 |
+
$proxy_port = !empty($this->base->config['proxy_port']) ? $this->base->config['proxy_port'] : '0';
|
76 |
+
$local_seo_file_root = (!empty($this->base->config['load_seo_files_locally'])) ? $this->base->config['local_seo_file_root'] : 'FALSE';
|
77 |
+
$content_type = mb_strtoupper($this->base->config['content_type']);
|
78 |
+
$subject_type = mb_strtoupper($this->base->config['subject_type']);
|
79 |
+
if (!empty($this->base->config['page_params']['subject_id'])
|
80 |
+
&& !empty($this->base->config['page_params']['content_type'])
|
81 |
+
&& $this->base->config['page_params']['content_type'] == $this->base->config['content_type']) {
|
82 |
+
$subject_id = $this->base->config['page_params']['subject_id'];
|
83 |
+
} else {
|
84 |
+
$subject_id = $this->base->config['subject_id'];
|
85 |
+
}
|
86 |
+
|
87 |
+
$footer = "\n" . '<ul id="BVSEOSDK_DEBUG" style="display:none;">';
|
88 |
+
|
89 |
+
$footer .= "\n" . ' <li data-bvseo="staging">' . $staging . '</li>';
|
90 |
+
$footer .= "\n" . ' <li data-bvseo="testing">' . $testing . '</li>';
|
91 |
+
$footer .= "\n" . ' <li data-bvseo="seo.sdk.enabled">' . $sdk_enabled . '</li>';
|
92 |
+
if (!isset($this->base->config['subject_type']) || $this->base->config['subject_type'] != 'seller') {
|
93 |
+
$footer .= "\n" . ' <li data-bvseo="stagingS3Hostname">' . $this->base->bv_config['seo-domain']['staging'] . '</li>';
|
94 |
+
$footer .= "\n" . ' <li data-bvseo="productionS3Hostname">' . $this->base->bv_config['seo-domain']['production'] . '</li>';
|
95 |
+
$footer .= "\n" . ' <li data-bvseo="testingStagingS3Hostname">' . $this->base->bv_config['seo-domain']['testing_staging'] . '</li>';
|
96 |
+
$footer .= "\n" . ' <li data-bvseo="testingProductionS3Hostname">' . $this->base->bv_config['seo-domain']['testing_production'] . '</li>';
|
97 |
+
}
|
98 |
+
$footer .= "\n" . ' <li data-bvseo="proxyHost">' . $proxy_host . '</li>';
|
99 |
+
$footer .= "\n" . ' <li data-bvseo="proxyPort">' . $proxy_port . '</li>';
|
100 |
+
$footer .= "\n" . ' <li data-bvseo="seo.sdk.execution.timeout.bot">' . $this->base->config['execution_timeout_bot'] . '</li>';
|
101 |
+
$footer .= "\n" . ' <li data-bvseo="seo.sdk.execution.timeout">' . $this->base->config['execution_timeout'] . '</li>';
|
102 |
+
$footer .= "\n" . ' <li data-bvseo="localSEOFileRoot">' . $local_seo_file_root . '</li>';
|
103 |
+
$footer .= "\n" . ' <li data-bvseo="cloudKey">' . $this->base->config['cloud_key'] . '</li>';
|
104 |
+
$footer .= "\n" . ' <li data-bvseo="bv.root.folder">' . $this->base->config['bv_root_folder'] . '</li>';
|
105 |
+
$footer .= "\n" . ' <li data-bvseo="seo.sdk.charset">' . $this->base->config['charset'] . '</li>';
|
106 |
+
$footer .= "\n" . ' <li data-bvseo="seo.sdk.ssl.enabled">' . $ssl_enabled . '</li>';
|
107 |
+
$footer .= "\n" . ' <li data-bvseo="crawlerAgentPattern">' . $this->base->config['crawler_agent_pattern'] . '</li>';
|
108 |
+
$footer .= "\n" . ' <li data-bvseo="subjectID">' . urlencode($subject_id) . '</li>';
|
109 |
+
|
110 |
+
|
111 |
+
$footer .= "\n" . ' <li data-bvseo="en">' . $sdk_enabled . '</li>';
|
112 |
+
$footer .= "\n" . ' <li data-bvseo="pn">bvseo-' . $this->base->config['page'] . '</li>';
|
113 |
+
$footer .= "\n" . ' <li data-bvseo="userAgent">' . $_SERVER['HTTP_USER_AGENT'] . '</li>';
|
114 |
+
$footer .= "\n" . ' <li data-bvseo="pageURI">' . $this->base->config['page_url'] . '</li>';
|
115 |
+
$footer .= "\n" . ' <li data-bvseo="baseURI">' . $this->base->config['base_url'] . '</li>';
|
116 |
+
$footer .= "\n" . ' <li data-bvseo="contentType">' . $content_type . '</li>';
|
117 |
+
$footer .= "\n" . ' <li data-bvseo="subjectType">' . $subject_type . '</li>';
|
118 |
+
if (!empty($this->base->seo_url)) {
|
119 |
+
$footer .= "\n" . ' <li data-bvseo="contentURL">' . $this->base->seo_url . '</li>';
|
120 |
+
}
|
121 |
+
|
122 |
+
$footer .= "\n" . '</ul>';
|
123 |
+
|
124 |
+
return $footer;
|
125 |
+
}
|
126 |
+
|
127 |
+
}
|
|
|
|
lib/Bazaarvoice/BVUtility.php
CHANGED
@@ -1,254 +1,212 @@
|
|
1 |
-
<?php
|
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 |
-
* @param
|
45 |
-
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
if (
|
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 |
-
* going to see that in practice. Attention is given to that here for the
|
214 |
-
* sake of completeness.
|
215 |
-
*
|
216 |
-
* @access public
|
217 |
-
* @param string ($url) - The URL.
|
218 |
-
* @param string ($paramName) - Name of the parameter to be removed.
|
219 |
-
* @return string - The updated URL.
|
220 |
-
*/
|
221 |
-
public static function removeUrlParam($url, $paramName) {
|
222 |
-
// The ereg POSIX regex functions are all greedy all the time, which makes
|
223 |
-
// this harder than it has to be.
|
224 |
-
//
|
225 |
-
// Big assumption: our seo link values will never contain the % character.
|
226 |
-
//
|
227 |
-
// http://example.com/product/123?bvstate=pg:4/ct:r&a=b
|
228 |
-
$url = mb_ereg_replace($paramName . '=[^&#%]*&', '', $url);
|
229 |
-
// http://example.com/product/123?bvstate=pg:4/ct:r&a=b
|
230 |
-
// http://example.com/product/123?#!bvstate=pg:4/ct:r&a=b
|
231 |
-
// http://example.com/product/123?_escaped_fragment_=bvstate=pg:4/ct:r%26a=b
|
232 |
-
$url = mb_ereg_replace($paramName . '=[^&#%]*(&|%26)', '', $url);
|
233 |
-
// http://example.com/product/123?bvstate=pg:4/ct:r#!x/y/z
|
234 |
-
$url = mb_ereg_replace($paramName . '=[^&#]*#', '#', $url);
|
235 |
-
// This one last as it will break everything if we haven't already dealt
|
236 |
-
// with all of the cases, since .* is always greedy in POSIX regex.
|
237 |
-
// http://example.com/product/123?bvstate=pg:4/ct:r
|
238 |
-
$url = mb_ereg_replace($paramName . '=[^&#%]*$', '', $url);
|
239 |
-
return $url;
|
240 |
-
}
|
241 |
-
|
242 |
-
/**
|
243 |
-
* A multibyte-safe trim.
|
244 |
-
* (http://stackoverflow.com/questions/10066647/multibyte-trim-in-php/10067670#10067670)
|
245 |
-
*
|
246 |
-
* @access public
|
247 |
-
* @param string ($str) - The string that will be trimmed.
|
248 |
-
* @return string - The trimmed string.
|
249 |
-
*/
|
250 |
-
public static function mb_trim($str) {
|
251 |
-
return mb_ereg_replace('(^\s+)|(\s+$)', '', $str);
|
252 |
-
}
|
253 |
-
|
254 |
-
}
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* BV PHP SEO SDK Utilities.
|
5 |
+
*/
|
6 |
+
class BVUtility {
|
7 |
+
public static $supportedContentTypes = array(
|
8 |
+
'r' => 'REVIEWS',
|
9 |
+
'q' => 'QUESTIONS',
|
10 |
+
's' => 'STORIES',
|
11 |
+
'u' => 'UNIVERSAL',
|
12 |
+
'sp'=> 'SPOTLIGHTS'
|
13 |
+
);
|
14 |
+
private static $supportedSubjectTypes = array(
|
15 |
+
'p' => 'PRODUCT',
|
16 |
+
'c' => 'CATEGORY',
|
17 |
+
'e' => 'ENTRY',
|
18 |
+
'd' => 'DETAIL',
|
19 |
+
's' => 'SELLER'
|
20 |
+
);
|
21 |
+
|
22 |
+
/**
|
23 |
+
* Parse the provided "bvstate" parameter value.
|
24 |
+
*
|
25 |
+
* @access public
|
26 |
+
* @param string $bvstate - Value of the bvstate parameter.
|
27 |
+
* @return array - parsed "bvstate" parameters.
|
28 |
+
*/
|
29 |
+
public static function getBVStateHash($bvstate) {
|
30 |
+
$bvStateHash = array();
|
31 |
+
$bvp = mb_split("/", $bvstate);
|
32 |
+
foreach ($bvp as $param) {
|
33 |
+
$key = static::mb_trim(mb_substr($param, 0, mb_strpos($param, ':')));
|
34 |
+
$bvStateHash[$key] = static::mb_trim(mb_substr($param, mb_strpos($param, ':') + 1));
|
35 |
+
}
|
36 |
+
return $bvStateHash;
|
37 |
+
}
|
38 |
+
|
39 |
+
/**
|
40 |
+
* Checks content type or subject type is supported.
|
41 |
+
* If type is not supported throw exception.
|
42 |
+
*
|
43 |
+
* @access public
|
44 |
+
* @param string ($type) - content type or subject type which have to be checked.
|
45 |
+
* @param string ($typeType) - default 'ct', mark of type 'ct' - content type, 'st' - subject type
|
46 |
+
* @return boolean True if type is correct and no exception was thrown.
|
47 |
+
*/
|
48 |
+
public static function checkType($type, $typeType = 'ct') {
|
49 |
+
if ($typeType == 'st') {
|
50 |
+
$typeName = 'subject type';
|
51 |
+
$typeArray = static::$supportedSubjectTypes;
|
52 |
+
} else {
|
53 |
+
$typeName = 'content type';
|
54 |
+
$typeArray = static::$supportedContentTypes;
|
55 |
+
}
|
56 |
+
if (!array_key_exists(mb_strtolower($type), $typeArray)) {
|
57 |
+
foreach ($typeArray as $key => $value) {
|
58 |
+
$supportList[] = $key . '=' . $value;
|
59 |
+
}
|
60 |
+
throw new Exception('Obtained not supported ' . $typeName
|
61 |
+
. '. BV Class supports following ' . $typeName . ': '
|
62 |
+
. implode(', ', $supportList));
|
63 |
+
}
|
64 |
+
|
65 |
+
return true;
|
66 |
+
}
|
67 |
+
|
68 |
+
/**
|
69 |
+
* Generates an array of parameters from the bvstate parameter value.
|
70 |
+
*
|
71 |
+
* @access public
|
72 |
+
* @param string $bvstate - "bvstate" parameter value.
|
73 |
+
* @return array - array of parameters that are ready to use in script.
|
74 |
+
*/
|
75 |
+
public static function getBVStateParams($bvstate) {
|
76 |
+
$bvStateHash = self::getBVStateHash($bvstate);
|
77 |
+
$params = array();
|
78 |
+
|
79 |
+
// If the content type 'ct' parameter is not present, then ignore bvstate.
|
80 |
+
if (empty($bvStateHash['ct'])) {
|
81 |
+
return $params;
|
82 |
+
}
|
83 |
+
|
84 |
+
if (!empty($bvStateHash)) {
|
85 |
+
if (!empty($bvStateHash['id'])) {
|
86 |
+
$params['subject_id'] = $bvStateHash['id'];
|
87 |
+
}
|
88 |
+
if (!empty($bvStateHash['pg'])) {
|
89 |
+
$params['page'] = $bvStateHash['pg'];
|
90 |
+
}
|
91 |
+
if (!empty($bvStateHash['ct'])) {
|
92 |
+
$cType = $bvStateHash['ct'];
|
93 |
+
self::checkType($cType, 'ct');
|
94 |
+
$params['content_type'] = mb_strtolower(self::$supportedContentTypes[$cType]);
|
95 |
+
}
|
96 |
+
if (!empty($bvStateHash['st'])) {
|
97 |
+
$sType = $bvStateHash['st'];
|
98 |
+
self::checkType($sType, 'st');
|
99 |
+
$params['subject_type'] = mb_strtolower(self::$supportedSubjectTypes[$sType]);
|
100 |
+
}
|
101 |
+
if (!empty($bvStateHash['reveal'])) {
|
102 |
+
$params['bvreveal'] = $bvStateHash['reveal'];
|
103 |
+
}
|
104 |
+
}
|
105 |
+
|
106 |
+
if (!empty($params)) {
|
107 |
+
// This acts as a flag to tell us that a useful bvstate value was in fact
|
108 |
+
// extracted from the URL.
|
109 |
+
$params['base_url_bvstate'] = TRUE;
|
110 |
+
}
|
111 |
+
if (empty($params['page'])) {
|
112 |
+
$params['page'] = '1';
|
113 |
+
}
|
114 |
+
|
115 |
+
return $params;
|
116 |
+
}
|
117 |
+
|
118 |
+
/**
|
119 |
+
* Parse name=value parameters from the URL query string, fragment, and
|
120 |
+
* _escaped_fragment_.
|
121 |
+
*
|
122 |
+
* @access public
|
123 |
+
* @param string ($url) - The URL.
|
124 |
+
* @return array - An array of parameters values indexed by parameter names.
|
125 |
+
*/
|
126 |
+
public static function parseUrlParameters($url) {
|
127 |
+
// Why are we doing things in this devious way? The answer is to be as
|
128 |
+
// multibyte-supportive as possible. Most of the URL-parsing tools in the
|
129 |
+
// toolbox appear to be only varying degrees of multibyte-supportive; good
|
130 |
+
// for UTF-8 but not so great if you venture beyond that.
|
131 |
+
|
132 |
+
// Break down the URL into a mix of things, some of which are name=value
|
133 |
+
// pairs.
|
134 |
+
$params = array();
|
135 |
+
$chunks = mb_split('\?|&|&|#!|#|_escaped_fragment_=|%26', $url);
|
136 |
+
foreach ($chunks as $chunk) {
|
137 |
+
// If this is name=value, then there will be two items.
|
138 |
+
$values = mb_split('=', $chunk);
|
139 |
+
if (sizeof($values) == 2) {
|
140 |
+
// Since we're moving left to right in the URL, and we want query string
|
141 |
+
// to win over fragment if there are the same parameters in both, then
|
142 |
+
// only add if not already there.
|
143 |
+
if (!isset($params[$values[0]])) {
|
144 |
+
$params[$values[0]] = $values[1];
|
145 |
+
}
|
146 |
+
}
|
147 |
+
}
|
148 |
+
return $params;
|
149 |
+
}
|
150 |
+
|
151 |
+
/**
|
152 |
+
* Remove a parameter from the provided URL.
|
153 |
+
*
|
154 |
+
* This will remove the named parameter wherever it occurs as name=value in
|
155 |
+
* the URL via a simple regex replacement. This is crude but the most
|
156 |
+
* straightforward way of going about this in PHP.
|
157 |
+
*
|
158 |
+
* If there is a query string delimeter following the name=value parameter
|
159 |
+
* then that will also be removed.
|
160 |
+
*
|
161 |
+
* E.g. we're expecting to remove the bvstate from URLs such as:
|
162 |
+
*
|
163 |
+
* http://example.com/product/123?bvstate=pg:4/ct:r
|
164 |
+
* http://example.com/product/123#!bvstate=pg:4/ct:r
|
165 |
+
*
|
166 |
+
* This will only be used for Bazaarvoice SEO parameters, so apologies in
|
167 |
+
* advance to the one person in the universe for whom bvstate=xyz is a vital
|
168 |
+
* part of the URL path.
|
169 |
+
*
|
170 |
+
* Note that the fragment isn't passed to the server, so we're not really
|
171 |
+
* going to see that in practice. Attention is given to that here for the
|
172 |
+
* sake of completeness.
|
173 |
+
*
|
174 |
+
* @access public
|
175 |
+
* @param string ($url) - The URL.
|
176 |
+
* @param string ($paramName) - Name of the parameter to be removed.
|
177 |
+
* @return string - The updated URL.
|
178 |
+
*/
|
179 |
+
public static function removeUrlParam($url, $paramName) {
|
180 |
+
// The ereg POSIX regex functions are all greedy all the time, which makes
|
181 |
+
// this harder than it has to be.
|
182 |
+
//
|
183 |
+
// Big assumption: our seo link values will never contain the % character.
|
184 |
+
//
|
185 |
+
// http://example.com/product/123?bvstate=pg:4/ct:r&a=b
|
186 |
+
$url = mb_ereg_replace($paramName . '=[^&#%]*&', '', $url);
|
187 |
+
// http://example.com/product/123?bvstate=pg:4/ct:r&a=b
|
188 |
+
// http://example.com/product/123?#!bvstate=pg:4/ct:r&a=b
|
189 |
+
// http://example.com/product/123?_escaped_fragment_=bvstate=pg:4/ct:r%26a=b
|
190 |
+
$url = mb_ereg_replace($paramName . '=[^&#%]*(&|%26)', '', $url);
|
191 |
+
// http://example.com/product/123?bvstate=pg:4/ct:r#!x/y/z
|
192 |
+
$url = mb_ereg_replace($paramName . '=[^&#]*#', '#', $url);
|
193 |
+
// This one last as it will break everything if we haven't already dealt
|
194 |
+
// with all of the cases, since .* is always greedy in POSIX regex.
|
195 |
+
// http://example.com/product/123?bvstate=pg:4/ct:r
|
196 |
+
$url = mb_ereg_replace($paramName . '=[^&#%]*$', '', $url);
|
197 |
+
return $url;
|
198 |
+
}
|
199 |
+
|
200 |
+
/**
|
201 |
+
* A multibyte-safe trim.
|
202 |
+
* (http://stackoverflow.com/questions/10066647/multibyte-trim-in-php/10067670#10067670)
|
203 |
+
*
|
204 |
+
* @access public
|
205 |
+
* @param string ($str) - The string that will be trimmed.
|
206 |
+
* @return string - The trimmed string.
|
207 |
+
*/
|
208 |
+
public static function mb_trim($str) {
|
209 |
+
return mb_ereg_replace('(^\s+)|(\s+$)', '', $str);
|
210 |
+
}
|
211 |
+
|
212 |
+
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
lib/Bazaarvoice/bvseosdk.php
CHANGED
@@ -1,1048 +1,1042 @@
|
|
1 |
-
<?php
|
2 |
-
|
3 |
-
/**
|
4 |
-
* BV PHP SEO SDK
|
5 |
-
*
|
6 |
-
* Base code to power either SEO or SEO and display. This SDK
|
7 |
-
* is provided as is and Bazaarvoice, Inc. is not responsible
|
8 |
-
* for future maintenance or support. You are free to modify
|
9 |
-
* this SDK as needed to suit your needs.
|
10 |
-
*
|
11 |
-
* This SDK was built with the following assumptions:
|
12 |
-
* - you are running PHP 5 or greater
|
13 |
-
* - you have the curl library installed
|
14 |
-
* - every request has the user agent header
|
15 |
-
* in it (if using a CDN like Akamai additional configuration
|
16 |
-
* maybe required).
|
17 |
-
*
|
18 |
-
*/
|
19 |
-
/**
|
20 |
-
* Example usage:
|
21 |
-
*
|
22 |
-
* require(bvsdk.php);
|
23 |
-
*
|
24 |
-
* $bv = new BV(array(
|
25 |
-
* 'bv_root_folder' => '1234-en_US',
|
26 |
-
* 'subject_id' => 'XXYYY',
|
27 |
-
* 'cloud_key' => 'company-cdfa682b84bef44672efed074093ccd3',
|
28 |
-
* 'staging' => FALSE
|
29 |
-
* ));
|
30 |
-
*
|
31 |
-
*/
|
32 |
-
require_once 'BVUtility.php';
|
33 |
-
require_once 'BVFooter.php';
|
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 |
-
if
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
//
|
133 |
-
|
134 |
-
|
135 |
-
//
|
136 |
-
//
|
137 |
-
$this->config['base_url'] =
|
138 |
-
|
139 |
-
|
140 |
-
|
141 |
-
|
142 |
-
|
143 |
-
$this->
|
144 |
-
$this->
|
145 |
-
|
146 |
-
|
147 |
-
$this->
|
148 |
-
|
149 |
-
$this->
|
150 |
-
$this->
|
151 |
-
|
152 |
-
|
153 |
-
|
154 |
-
|
155 |
-
|
156 |
-
|
157 |
-
|
158 |
-
|
159 |
-
|
160 |
-
|
161 |
-
|
162 |
-
|
163 |
-
|
164 |
-
|
165 |
-
|
166 |
-
case '
|
167 |
-
break;
|
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 |
-
$this->
|
219 |
-
|
220 |
-
|
221 |
-
|
222 |
-
|
223 |
-
$this->bv_config['
|
224 |
-
$this->bv_config['
|
225 |
-
|
226 |
-
|
227 |
-
$this->bv_config['srd-
|
228 |
-
|
229 |
-
$this->
|
230 |
-
|
231 |
-
|
232 |
-
|
233 |
-
|
234 |
-
|
235 |
-
|
236 |
-
|
237 |
-
|
238 |
-
$
|
239 |
-
|
240 |
-
|
241 |
-
|
242 |
-
|
243 |
-
|
244 |
-
|
245 |
-
|
246 |
-
|
247 |
-
|
248 |
-
|
249 |
-
|
250 |
-
|
251 |
-
if (
|
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 |
-
|
307 |
-
|
308 |
-
|
309 |
-
|
310 |
-
|
311 |
-
|
312 |
-
|
313 |
-
|
314 |
-
|
315 |
-
|
316 |
-
|
317 |
-
//
|
318 |
-
$
|
319 |
-
|
320 |
-
//
|
321 |
-
//
|
322 |
-
|
323 |
-
|
324 |
-
|
325 |
-
|
326 |
-
|
327 |
-
|
328 |
-
|
329 |
-
$this->
|
330 |
-
|
331 |
-
|
332 |
-
|
333 |
-
$
|
334 |
-
|
335 |
-
|
336 |
-
|
337 |
-
|
338 |
-
|
339 |
-
|
340 |
-
|
341 |
-
|
342 |
-
|
343 |
-
|
344 |
-
|
345 |
-
|
346 |
-
|
347 |
-
|
348 |
-
|
349 |
-
|
350 |
-
|
351 |
-
|
352 |
-
|
353 |
-
|
354 |
-
|
355 |
-
|
356 |
-
|
357 |
-
|
358 |
-
$
|
359 |
-
|
360 |
-
|
361 |
-
|
362 |
-
|
363 |
-
|
364 |
-
|
365 |
-
|
366 |
-
|
367 |
-
|
368 |
-
|
369 |
-
|
370 |
-
|
371 |
-
|
372 |
-
|
373 |
-
|
374 |
-
|
375 |
-
|
376 |
-
|
377 |
-
|
378 |
-
|
379 |
-
|
380 |
-
|
381 |
-
|
382 |
-
|
383 |
-
|
384 |
-
|
385 |
-
|
386 |
-
|
387 |
-
|
388 |
-
|
389 |
-
|
390 |
-
|
391 |
-
$payload =
|
392 |
-
|
393 |
-
|
394 |
-
|
395 |
-
|
396 |
-
|
397 |
-
|
398 |
-
|
399 |
-
|
400 |
-
|
401 |
-
*
|
402 |
-
*
|
403 |
-
*
|
404 |
-
*
|
405 |
-
*
|
406 |
-
|
407 |
-
|
408 |
-
|
409 |
-
|
410 |
-
|
411 |
-
|
412 |
-
|
413 |
-
$this->
|
414 |
-
|
415 |
-
|
416 |
-
|
417 |
-
|
418 |
-
|
419 |
-
|
420 |
-
|
421 |
-
|
422 |
-
|
423 |
-
$this->
|
424 |
-
}
|
425 |
-
|
426 |
-
|
427 |
-
|
428 |
-
|
429 |
-
|
430 |
-
|
431 |
-
|
432 |
-
|
433 |
-
|
434 |
-
|
435 |
-
|
436 |
-
|
437 |
-
|
438 |
-
|
439 |
-
|
440 |
-
|
441 |
-
|
442 |
-
|
443 |
-
|
444 |
-
*
|
445 |
-
|
446 |
-
|
447 |
-
|
448 |
-
|
449 |
-
|
450 |
-
|
451 |
-
|
452 |
-
|
453 |
-
|
454 |
-
|
455 |
-
|
456 |
-
|
457 |
-
|
458 |
-
|
459 |
-
|
460 |
-
|
461 |
-
|
462 |
-
|
463 |
-
|
464 |
-
|
465 |
-
|
466 |
-
|
467 |
-
|
468 |
-
|
469 |
-
|
470 |
-
|
471 |
-
|
472 |
-
|
473 |
-
$this->config['bvreveal']
|
474 |
-
|
475 |
-
|
476 |
-
|
477 |
-
|
478 |
-
|
479 |
-
|
480 |
-
$this->config['
|
481 |
-
|
482 |
-
|
483 |
-
|
484 |
-
|
485 |
-
|
486 |
-
|
487 |
-
|
488 |
-
|
489 |
-
|
490 |
-
|
491 |
-
|
492 |
-
|
493 |
-
|
494 |
-
|
495 |
-
|
496 |
-
|
497 |
-
|
498 |
-
|
499 |
-
|
500 |
-
|
501 |
-
|
502 |
-
|
503 |
-
|
504 |
-
|
505 |
-
|
506 |
-
|
507 |
-
|
508 |
-
|
509 |
-
|
510 |
-
|
511 |
-
|
512 |
-
|
513 |
-
|
514 |
-
|
515 |
-
|
516 |
-
|
517 |
-
|
518 |
-
|
519 |
-
|
520 |
-
|
521 |
-
//
|
522 |
-
//
|
523 |
-
//
|
524 |
-
//
|
525 |
-
//
|
526 |
-
|
527 |
-
|
528 |
-
|
529 |
-
|
530 |
-
|
531 |
-
|
532 |
-
|
533 |
-
|
534 |
-
|
535 |
-
$
|
536 |
-
|
537 |
-
$
|
538 |
-
}
|
539 |
-
|
540 |
-
|
541 |
-
|
542 |
-
|
543 |
-
|
544 |
-
|
545 |
-
|
546 |
-
|
547 |
-
|
548 |
-
|
549 |
-
|
550 |
-
*
|
551 |
-
|
552 |
-
|
553 |
-
|
554 |
-
|
555 |
-
|
556 |
-
|
557 |
-
|
558 |
-
|
559 |
-
|
560 |
-
|
561 |
-
|
562 |
-
|
563 |
-
if ($this->config['staging']) {
|
564 |
-
$env_selector = '
|
565 |
-
} else {
|
566 |
-
$env_selector = '
|
567 |
-
}
|
568 |
-
}
|
569 |
-
|
570 |
-
|
571 |
-
|
572 |
-
|
573 |
-
|
574 |
-
|
575 |
-
|
576 |
-
|
577 |
-
|
578 |
-
|
579 |
-
|
580 |
-
|
581 |
-
|
582 |
-
|
583 |
-
|
584 |
-
|
585 |
-
|
586 |
-
$
|
587 |
-
|
588 |
-
|
589 |
-
|
590 |
-
|
591 |
-
|
592 |
-
|
593 |
-
|
594 |
-
$this->config['
|
595 |
-
|
596 |
-
|
597 |
-
)
|
598 |
-
|
599 |
-
|
600 |
-
$url_parts[] = $this->config['
|
601 |
-
}
|
602 |
-
|
603 |
-
if
|
604 |
-
|
605 |
-
|
606 |
-
|
607 |
-
|
608 |
-
|
609 |
-
|
610 |
-
|
611 |
-
|
612 |
-
|
613 |
-
|
614 |
-
|
615 |
-
|
616 |
-
|
617 |
-
|
618 |
-
|
619 |
-
|
620 |
-
|
621 |
-
|
622 |
-
|
623 |
-
|
624 |
-
|
625 |
-
|
626 |
-
|
627 |
-
|
628 |
-
|
629 |
-
|
630 |
-
|
631 |
-
|
632 |
-
|
633 |
-
|
634 |
-
|
635 |
-
*
|
636 |
-
*
|
637 |
-
|
638 |
-
|
639 |
-
|
640 |
-
|
641 |
-
|
642 |
-
|
643 |
-
|
644 |
-
|
645 |
-
|
646 |
-
|
647 |
-
|
648 |
-
|
649 |
-
|
650 |
-
|
651 |
-
|
652 |
-
|
653 |
-
|
654 |
-
|
655 |
-
|
656 |
-
|
657 |
-
|
658 |
-
|
659 |
-
|
660 |
-
|
661 |
-
|
662 |
-
|
663 |
-
|
664 |
-
|
665 |
-
|
666 |
-
|
667 |
-
|
668 |
-
|
669 |
-
|
670 |
-
|
671 |
-
|
672 |
-
*
|
673 |
-
*
|
674 |
-
|
675 |
-
|
676 |
-
|
677 |
-
|
678 |
-
|
679 |
-
|
680 |
-
|
681 |
-
|
682 |
-
|
683 |
-
|
684 |
-
//
|
685 |
-
|
686 |
-
//
|
687 |
-
|
688 |
-
|
689 |
-
|
690 |
-
//
|
691 |
-
curl_setopt($ch,
|
692 |
-
//
|
693 |
-
curl_setopt($ch,
|
694 |
-
//
|
695 |
-
curl_setopt($ch,
|
696 |
-
//
|
697 |
-
curl_setopt($ch,
|
698 |
-
//
|
699 |
-
|
700 |
-
|
701 |
-
|
702 |
-
|
703 |
-
|
704 |
-
|
705 |
-
|
706 |
-
|
707 |
-
|
708 |
-
|
709 |
-
|
710 |
-
|
711 |
-
|
712 |
-
|
713 |
-
|
714 |
-
|
715 |
-
|
716 |
-
|
717 |
-
|
718 |
-
|
719 |
-
|
720 |
-
|
721 |
-
|
722 |
-
)
|
723 |
-
|
724 |
-
|
725 |
-
|
726 |
-
|
727 |
-
|
728 |
-
|
729 |
-
|
730 |
-
|
731 |
-
|
732 |
-
|
733 |
-
if
|
734 |
-
|
735 |
-
|
736 |
-
|
737 |
-
|
738 |
-
|
739 |
-
|
740 |
-
|
741 |
-
|
742 |
-
|
743 |
-
|
744 |
-
|
745 |
-
*
|
746 |
-
*
|
747 |
-
|
748 |
-
|
749 |
-
|
750 |
-
|
751 |
-
|
752 |
-
|
753 |
-
|
754 |
-
|
755 |
-
|
756 |
-
|
757 |
-
//
|
758 |
-
//
|
759 |
-
|
760 |
-
|
761 |
-
|
762 |
-
//
|
763 |
-
//
|
764 |
-
|
765 |
-
|
766 |
-
|
767 |
-
|
768 |
-
|
769 |
-
|
770 |
-
|
771 |
-
|
772 |
-
|
773 |
-
|
774 |
-
|
775 |
-
|
776 |
-
|
777 |
-
}
|
778 |
-
|
779 |
-
|
780 |
-
|
781 |
-
|
782 |
-
|
783 |
-
|
784 |
-
|
785 |
-
|
786 |
-
|
787 |
-
|
788 |
-
|
789 |
-
|
790 |
-
|
791 |
-
|
792 |
-
|
793 |
-
|
794 |
-
|
795 |
-
|
796 |
-
|
797 |
-
|
798 |
-
|
799 |
-
|
800 |
-
|
801 |
-
|
802 |
-
|
803 |
-
|
804 |
-
|
805 |
-
|
806 |
-
|
807 |
-
|
808 |
-
|
809 |
-
|
810 |
-
|
811 |
-
|
812 |
-
|
813 |
-
|
814 |
-
|
815 |
-
|
816 |
-
|
817 |
-
|
818 |
-
$
|
819 |
-
|
820 |
-
|
821 |
-
|
822 |
-
|
823 |
-
|
824 |
-
|
825 |
-
$
|
826 |
-
|
827 |
-
|
828 |
-
|
829 |
-
|
830 |
-
|
831 |
-
|
832 |
-
|
833 |
-
|
834 |
-
|
835 |
-
|
836 |
-
|
837 |
-
|
838 |
-
|
839 |
-
|
840 |
-
|
841 |
-
|
842 |
-
|
843 |
-
|
844 |
-
|
845 |
-
|
846 |
-
|
847 |
-
|
848 |
-
//
|
849 |
-
|
850 |
-
|
851 |
-
//
|
852 |
-
//
|
853 |
-
|
854 |
-
|
855 |
-
|
856 |
-
|
857 |
-
|
858 |
-
|
859 |
-
|
860 |
-
|
861 |
-
|
862 |
-
|
863 |
-
|
864 |
-
|
865 |
-
|
866 |
-
|
867 |
-
|
868 |
-
|
869 |
-
|
870 |
-
|
871 |
-
|
872 |
-
|
873 |
-
|
874 |
-
|
875 |
-
|
876 |
-
|
877 |
-
|
878 |
-
|
879 |
-
|
880 |
-
|
881 |
-
|
882 |
-
|
883 |
-
|
884 |
-
|
885 |
-
|
886 |
-
|
887 |
-
|
888 |
-
|
889 |
-
|
890 |
-
|
891 |
-
|
892 |
-
|
893 |
-
|
894 |
-
|
895 |
-
|
896 |
-
|
897 |
-
|
898 |
-
|
899 |
-
|
900 |
-
|
901 |
-
|
902 |
-
|
903 |
-
|
904 |
-
//
|
905 |
-
|
906 |
-
|
907 |
-
|
908 |
-
|
909 |
-
|
910 |
-
|
911 |
-
|
912 |
-
|
913 |
-
|
914 |
-
|
915 |
-
|
916 |
-
|
917 |
-
|
918 |
-
|
919 |
-
$
|
920 |
-
|
921 |
-
|
922 |
-
|
923 |
-
|
924 |
-
|
925 |
-
|
926 |
-
|
927 |
-
|
928 |
-
|
929 |
-
|
930 |
-
|
931 |
-
|
932 |
-
|
933 |
-
|
934 |
-
|
935 |
-
|
936 |
-
|
937 |
-
|
938 |
-
|
939 |
-
|
940 |
-
|
941 |
-
|
942 |
-
|
943 |
-
|
944 |
-
|
945 |
-
|
946 |
-
|
947 |
-
|
948 |
-
//
|
949 |
-
|
950 |
-
|
951 |
-
//
|
952 |
-
//
|
953 |
-
|
954 |
-
|
955 |
-
|
956 |
-
|
957 |
-
//
|
958 |
-
|
959 |
-
|
960 |
-
|
961 |
-
|
962 |
-
|
963 |
-
|
964 |
-
|
965 |
-
|
966 |
-
|
967 |
-
|
968 |
-
|
969 |
-
|
970 |
-
|
971 |
-
|
972 |
-
|
973 |
-
|
974 |
-
|
975 |
-
|
976 |
-
|
977 |
-
|
978 |
-
|
979 |
-
|
980 |
-
|
981 |
-
|
982 |
-
|
983 |
-
|
984 |
-
|
985 |
-
|
986 |
-
|
987 |
-
|
988 |
-
|
989 |
-
|
990 |
-
|
991 |
-
|
992 |
-
|
993 |
-
|
994 |
-
|
995 |
-
|
996 |
-
|
997 |
-
|
998 |
-
|
999 |
-
|
1000 |
-
//
|
1001 |
-
|
1002 |
-
|
1003 |
-
//
|
1004 |
-
//
|
1005 |
-
|
1006 |
-
|
1007 |
-
|
1008 |
-
|
1009 |
-
|
1010 |
-
|
1011 |
-
|
1012 |
-
|
1013 |
-
|
1014 |
-
|
1015 |
-
|
1016 |
-
|
1017 |
-
|
1018 |
-
|
1019 |
-
//
|
1020 |
-
|
1021 |
-
|
1022 |
-
|
1023 |
-
|
1024 |
-
|
1025 |
-
//
|
1026 |
-
|
1027 |
-
|
1028 |
-
//
|
1029 |
-
//
|
1030 |
-
|
1031 |
-
|
1032 |
-
|
1033 |
-
|
1034 |
-
|
1035 |
-
|
1036 |
-
|
1037 |
-
|
1038 |
-
|
1039 |
-
|
1040 |
-
|
1041 |
-
|
1042 |
-
|
1043 |
-
|
1044 |
-
}
|
1045 |
-
// end of Spotlights class
|
1046 |
-
|
1047 |
-
|
1048 |
-
// end of bvseosdk.php
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* BV PHP SEO SDK
|
5 |
+
*
|
6 |
+
* Base code to power either SEO or SEO and display. This SDK
|
7 |
+
* is provided as is and Bazaarvoice, Inc. is not responsible
|
8 |
+
* for future maintenance or support. You are free to modify
|
9 |
+
* this SDK as needed to suit your needs.
|
10 |
+
*
|
11 |
+
* This SDK was built with the following assumptions:
|
12 |
+
* - you are running PHP 5 or greater
|
13 |
+
* - you have the curl library installed
|
14 |
+
* - every request has the user agent header
|
15 |
+
* in it (if using a CDN like Akamai additional configuration
|
16 |
+
* maybe required).
|
17 |
+
*
|
18 |
+
*/
|
19 |
+
/**
|
20 |
+
* Example usage:
|
21 |
+
*
|
22 |
+
* require(bvsdk.php);
|
23 |
+
*
|
24 |
+
* $bv = new BV(array(
|
25 |
+
* 'bv_root_folder' => '1234-en_US',
|
26 |
+
* 'subject_id' => 'XXYYY',
|
27 |
+
* 'cloud_key' => 'company-cdfa682b84bef44672efed074093ccd3',
|
28 |
+
* 'staging' => FALSE
|
29 |
+
* ));
|
30 |
+
*
|
31 |
+
*/
|
32 |
+
require_once 'BVUtility.php';
|
33 |
+
require_once 'BVFooter.php';
|
34 |
+
|
35 |
+
// Default charset will be used in case charset parameter is not properly configured by user.
|
36 |
+
define('DEFAULT_CHARSET', 'UTF-8');
|
37 |
+
|
38 |
+
// ------------------------------------------------------------------------
|
39 |
+
|
40 |
+
/**
|
41 |
+
* BV Class
|
42 |
+
*
|
43 |
+
* When you instantiate the BV class, pass it's constructor an array
|
44 |
+
* containing the following key value pairs.
|
45 |
+
*
|
46 |
+
* Required fields:
|
47 |
+
* bv_root_folder (string)
|
48 |
+
* subject_id (string)
|
49 |
+
* cloud_key (string)
|
50 |
+
*
|
51 |
+
* Optional fields
|
52 |
+
* base_url (string) (defaults to detecting the base_url automatically)
|
53 |
+
* page_url (string) (defaults to empty, to provide query parameters )
|
54 |
+
* staging (boolean) (defaults to false, need to put true for testing with staging data)
|
55 |
+
* testing (boolean) (defaults to false, need to put true for testing with testing data)
|
56 |
+
* content_type (string) (defaults to reviews, you can pass content type here if needed)
|
57 |
+
* subject_type (string) (defaults to product, you can pass subject type here if needed)
|
58 |
+
* content_sub_type (string) (defaults to stories, for stories you can pass either STORIES_LIST or STORIES_GRID content type)
|
59 |
+
* execution_timeout (int) (in milliseconds) (defaults to 500ms, to set period of time before the BVSEO injection times out for user agents that do not match the criteria set in CRAWLER_AGENT_PATTERN)
|
60 |
+
* execution_timeout_bot (int) (in milliseconds) (defaults to 2000ms, to set period of time before the BVSEO injection times out for user agents that match the criteria set in CRAWLER_AGENT_PATTERN)
|
61 |
+
* charset (string) (defaults to UTF-8, to set alternate character for SDK output)
|
62 |
+
* crawler_agent_pattern (string) (defaults to msnbot|googlebot|teoma|bingbot|yandexbot|yahoo)
|
63 |
+
*/
|
64 |
+
class BV {
|
65 |
+
|
66 |
+
/**
|
67 |
+
* BV Class Constructor
|
68 |
+
*
|
69 |
+
* The constructor takes in all the arguments via a single array.
|
70 |
+
*
|
71 |
+
* @access public
|
72 |
+
* @param array
|
73 |
+
* @return object
|
74 |
+
*/
|
75 |
+
public function __construct($params = array()) {
|
76 |
+
|
77 |
+
$this->validateParameters($params);
|
78 |
+
|
79 |
+
// config array, defaults are defined here.
|
80 |
+
$this->config = array(
|
81 |
+
'staging' => FALSE,
|
82 |
+
'testing' => FALSE,
|
83 |
+
'content_type' => isset($params['content_type']) ? $params['content_type'] : 'reviews',
|
84 |
+
'subject_type' => isset($params['subject_type']) ? $params['subject_type'] : 'product',
|
85 |
+
'page_url' => isset($params['page_url']) ? $params['page_url'] : '',
|
86 |
+
'base_url' => isset($params['base_url']) ? $params['base_url'] : '',
|
87 |
+
'include_display_integration_code' => FALSE,
|
88 |
+
'client_name' => $params['bv_root_folder'],
|
89 |
+
'local_seo_file_root' => '',
|
90 |
+
'load_seo_files_locally' => FALSE,
|
91 |
+
// used in regex to determine if request is a bot or not
|
92 |
+
'crawler_agent_pattern' => 'msnbot|google|teoma|bingbot|yandexbot|yahoo',
|
93 |
+
'ssl_enabled' => FALSE,
|
94 |
+
'proxy_host' => '',
|
95 |
+
'proxy_port' => '',
|
96 |
+
'charset' => 'UTF-8',
|
97 |
+
'seo_sdk_enabled' => TRUE,
|
98 |
+
'execution_timeout' => 500,
|
99 |
+
'execution_timeout_bot' => 2000,
|
100 |
+
'bvreveal' => isset($params['bvreveal']) ? $params['bvreveal'] : '',
|
101 |
+
'page' => 1,
|
102 |
+
'page_params' => array()
|
103 |
+
);
|
104 |
+
|
105 |
+
// Merge passed in params with defaults for config.
|
106 |
+
$this->config = array_merge($this->config, $params);
|
107 |
+
|
108 |
+
// Obtain all the name=value parameters from either the page URL passed in,
|
109 |
+
// or from the actual page URL as seen by PHP. Parameter values from the
|
110 |
+
// actual URL override those from the URL passed in, as that is usually a
|
111 |
+
// trucated URL where present at all.
|
112 |
+
//
|
113 |
+
// Note that we're taking parameters from query string, fragment, or
|
114 |
+
// _escaped_fragment_. (Though fragment is not passed to the server, so
|
115 |
+
// we won't actually see that in practice).
|
116 |
+
//
|
117 |
+
// We're after bvrrp, bvqap, bvsyp, and bvstate, but sweep up everything
|
118 |
+
// while we're here.
|
119 |
+
if (isset($params['page_url'])) {
|
120 |
+
$this->config['bv_page_data'] = BVUtility::parseUrlParameters($params['page_url']);
|
121 |
+
}
|
122 |
+
|
123 |
+
// Extract bvstate if present and parse that into a set of useful values.
|
124 |
+
if (isset($this->config['bv_page_data']['bvstate'])) {
|
125 |
+
$this->config['page_params'] = BVUtility::getBVStateParams($this->config['bv_page_data']['bvstate']);
|
126 |
+
}
|
127 |
+
|
128 |
+
// Remove any trailing URL delimeters from the base URL. E.g.:
|
129 |
+
// http://example.com?
|
130 |
+
// http://example.com?a=b&
|
131 |
+
// http://example.com?a=b&_escaped_fragment_=x/y/z?r=s%26
|
132 |
+
//
|
133 |
+
$this->config['base_url'] = mb_ereg_replace('(&|\?|%26)$', '', $this->config['base_url']);
|
134 |
+
|
135 |
+
// Get rid of all the other things we care about from the base URL, so that
|
136 |
+
// we don't double up the parameters.
|
137 |
+
$this->config['base_url'] = BVUtility::removeUrlParam($this->config['base_url'], 'bvstate');
|
138 |
+
$this->config['base_url'] = BVUtility::removeUrlParam($this->config['base_url'], 'bvrrp');
|
139 |
+
$this->config['base_url'] = BVUtility::removeUrlParam($this->config['base_url'], 'bvqap');
|
140 |
+
$this->config['base_url'] = BVUtility::removeUrlParam($this->config['base_url'], 'bvsyp');
|
141 |
+
|
142 |
+
// Create the processor objects.
|
143 |
+
$this->reviews = new Reviews($this->config);
|
144 |
+
$this->questions = new Questions($this->config);
|
145 |
+
$this->stories = new Stories($this->config);
|
146 |
+
$this->spotlights = new Spotlights($this->config);
|
147 |
+
$this->sellerratings = new SellerRatings($this->config);
|
148 |
+
|
149 |
+
// Assign one to $this->SEO based on the content type.
|
150 |
+
$ct = isset($this->config['page_params']['content_type']) ? $this->config['page_params']['content_type'] : $this->config['content_type'];
|
151 |
+
if (isset($ct)) {
|
152 |
+
switch ($ct) {
|
153 |
+
case 'reviews': {
|
154 |
+
$st = isset($this->config['page_params']['subject_type']) ? $this->config['page_params']['subject_type'] : $this->config['subject_type'];
|
155 |
+
if (isset($st) && $st == 'seller') {
|
156 |
+
$this->SEO = $this->sellerratings;
|
157 |
+
} else {
|
158 |
+
$this->SEO = $this->reviews;
|
159 |
+
}
|
160 |
+
break;
|
161 |
+
}
|
162 |
+
case 'questions': $this->SEO = $this->questions;
|
163 |
+
break;
|
164 |
+
case 'stories': $this->SEO = $this->stories;
|
165 |
+
break;
|
166 |
+
case 'spotlights': $this->SEO = $this->spotlights;
|
167 |
+
break;
|
168 |
+
default:
|
169 |
+
throw new Exception('Invalid content_type value provided: ' . $this->config['content_type']);
|
170 |
+
}
|
171 |
+
}
|
172 |
+
}
|
173 |
+
|
174 |
+
protected function validateParameters($params) {
|
175 |
+
if (!is_array($params)) {
|
176 |
+
throw new Exception(
|
177 |
+
'BV class constructor argument $params must be an array.'
|
178 |
+
);
|
179 |
+
}
|
180 |
+
|
181 |
+
// check to make sure we have the required parameters.
|
182 |
+
if (empty($params['bv_root_folder'])) {
|
183 |
+
throw new Exception(
|
184 |
+
'BV class constructor argument $params is missing required bv_root_folder key. An ' .
|
185 |
+
'array containing bv_root_folder (string) is expected.'
|
186 |
+
);
|
187 |
+
}
|
188 |
+
|
189 |
+
if (empty($params['subject_id'])) {
|
190 |
+
throw new Exception(
|
191 |
+
'BV class constructor argument $params is missing required subject_id key. An ' .
|
192 |
+
'array containing subject_id (string) is expected.'
|
193 |
+
);
|
194 |
+
}
|
195 |
+
}
|
196 |
+
}
|
197 |
+
// end of BV class
|
198 |
+
|
199 |
+
/**
|
200 |
+
* Base Class containing most shared functionality. So when we add support for
|
201 |
+
* questions and answers it should be minimal changes. Just need to create an
|
202 |
+
* answers class which inherits from Base.
|
203 |
+
*
|
204 |
+
* Configuration array is required for creation class object.
|
205 |
+
*
|
206 |
+
*/
|
207 |
+
class Base {
|
208 |
+
private $msg = '';
|
209 |
+
|
210 |
+
public function __construct($params = array()) {
|
211 |
+
|
212 |
+
$this->validateParams($params);
|
213 |
+
|
214 |
+
$this->config = $params;
|
215 |
+
|
216 |
+
// setup bv (internal) defaults
|
217 |
+
$this->bv_config['seo-domain']['staging'] = 'seo-stg.bazaarvoice.com';
|
218 |
+
$this->bv_config['seo-domain']['production'] = 'seo.bazaarvoice.com';
|
219 |
+
$this->bv_config['seo-domain']['testing_staging'] = 'seo-qa-stg.bazaarvoice.com';
|
220 |
+
$this->bv_config['seo-domain']['testing_production'] = 'seo-qa.bazaarvoice.com';
|
221 |
+
|
222 |
+
// seller rating display is a special snowflake
|
223 |
+
$this->bv_config['srd-domain'] = 'srd.bazaarvoice.com';
|
224 |
+
$this->bv_config['srd-prefix-staging'] = 'stg';
|
225 |
+
$this->bv_config['srd-prefix-production'] = 'prod';
|
226 |
+
$this->bv_config['srd-prefix-testing_staging'] = 'qa-stg';
|
227 |
+
$this->bv_config['srd-prefix-testing_production'] = 'qa';
|
228 |
+
|
229 |
+
$this->config['latency_timeout'] = $this->_isBot()
|
230 |
+
? $this->config['execution_timeout_bot']
|
231 |
+
: $this->config['execution_timeout'];
|
232 |
+
|
233 |
+
// set up combined user agent to be passed to cloud storage (if needed)
|
234 |
+
$this->config['user_agent'] = "bv_php_sdk/3.2.1;" . $_SERVER['HTTP_USER_AGENT'];
|
235 |
+
}
|
236 |
+
|
237 |
+
protected function validateParams($params) {
|
238 |
+
if (!is_array($params)) {
|
239 |
+
throw new Exception('BV Base Class missing config array.');
|
240 |
+
}
|
241 |
+
}
|
242 |
+
|
243 |
+
/**
|
244 |
+
* A check on the bvstate parameter content type value.
|
245 |
+
*/
|
246 |
+
protected function _checkBVStateContentType() {
|
247 |
+
if (empty($this->config['page_params']['content_type'])) {
|
248 |
+
return TRUE;
|
249 |
+
}
|
250 |
+
|
251 |
+
if (
|
252 |
+
!empty($this->config['page_params']['content_type']) &&
|
253 |
+
$this->config['page_params']['content_type'] == $this->config['content_type']
|
254 |
+
) {
|
255 |
+
return TRUE;
|
256 |
+
}
|
257 |
+
|
258 |
+
return FALSE;
|
259 |
+
}
|
260 |
+
|
261 |
+
/**
|
262 |
+
* Function for collecting messages.
|
263 |
+
*/
|
264 |
+
protected function _setBuildMessage($msg) {
|
265 |
+
$msg = rtrim($msg, ";");
|
266 |
+
$this->msg .= ' ' . $msg . ';';
|
267 |
+
}
|
268 |
+
|
269 |
+
/**
|
270 |
+
* Is this SDK enabled?
|
271 |
+
*
|
272 |
+
* Return true if either seo_sdk_enabled is set truthy or bvreveal flags are
|
273 |
+
* set.
|
274 |
+
*/
|
275 |
+
private function _isSdkEnabled() {
|
276 |
+
return $this->config['seo_sdk_enabled'] || $this->_getBVReveal();
|
277 |
+
}
|
278 |
+
|
279 |
+
/**
|
280 |
+
* Check if charset is correct, if not set to default
|
281 |
+
*/
|
282 |
+
private function _checkCharset($seo_content) {
|
283 |
+
if (isset($this->config['charset'])) {
|
284 |
+
$supportedCharsets = mb_list_encodings();
|
285 |
+
if (!in_array($this->config['charset'], $supportedCharsets)) {
|
286 |
+
$this->config['charset'] = DEFAULT_CHARSET;
|
287 |
+
$this->_setBuildMessage("Charset is not configured properly. "
|
288 |
+
. "BV-SEO-SDK will load default charset and continue.");
|
289 |
+
}
|
290 |
+
} else {
|
291 |
+
$this->config['charset'] = DEFAULT_CHARSET;
|
292 |
+
}
|
293 |
+
}
|
294 |
+
|
295 |
+
/**
|
296 |
+
* Return encoded content with set charset
|
297 |
+
*/
|
298 |
+
private function _charsetEncode($seo_content) {
|
299 |
+
if (isset($this->config['charset'])) {
|
300 |
+
$enc = mb_detect_encoding($seo_content);
|
301 |
+
$seo_content = mb_convert_encoding($seo_content, $this->config['charset'], $enc);
|
302 |
+
}
|
303 |
+
|
304 |
+
return $seo_content;
|
305 |
+
}
|
306 |
+
|
307 |
+
/**
|
308 |
+
* Return full SEO content.
|
309 |
+
*/
|
310 |
+
private function _getFullSeoContents() {
|
311 |
+
$seo_content = '';
|
312 |
+
|
313 |
+
// get the page number of SEO content to load
|
314 |
+
$page_number = $this->_getPageNumber();
|
315 |
+
|
316 |
+
// build the URL to access the SEO content for
|
317 |
+
// this product / page combination
|
318 |
+
$this->seo_url = $this->_buildSeoUrl($page_number);
|
319 |
+
|
320 |
+
// make call to get SEO payload from cloud unless seo_sdk_enabled is false
|
321 |
+
// make call if bvreveal param in query string is set to 'debug'
|
322 |
+
if ($this->_isSdkEnabled()) {
|
323 |
+
$seo_content = $this->_fetchSeoContent($this->seo_url);
|
324 |
+
|
325 |
+
$this->_checkCharset($seo_content);
|
326 |
+
$seo_content = $this->_charsetEncode($seo_content);
|
327 |
+
|
328 |
+
// replace tokens for pagination URLs with page_url
|
329 |
+
$seo_content = $this->_replaceTokens($seo_content);
|
330 |
+
}
|
331 |
+
// show footer even if seo_sdk_enabled flag is false
|
332 |
+
else {
|
333 |
+
$this->_setBuildMessage(
|
334 |
+
'SEO SDK is disabled. Enable by setting seo.sdk.enabled to true.'
|
335 |
+
);
|
336 |
+
}
|
337 |
+
|
338 |
+
$payload = $seo_content;
|
339 |
+
|
340 |
+
return $payload;
|
341 |
+
}
|
342 |
+
|
343 |
+
/**
|
344 |
+
* Remove predefined section from a string.
|
345 |
+
*/
|
346 |
+
private function _replaceSection($str, $search_str_begin, $search_str_end) {
|
347 |
+
$result = $str;
|
348 |
+
$start_index = mb_strrpos($str, $search_str_begin);
|
349 |
+
|
350 |
+
if ($start_index !== false) {
|
351 |
+
$end_index = mb_strrpos($str, $search_str_end);
|
352 |
+
|
353 |
+
if ($end_index !== false) {
|
354 |
+
$end_index += mb_strlen($search_str_end);
|
355 |
+
$str_begin = mb_substr($str, 0, $start_index);
|
356 |
+
$str_end = mb_substr($str, $end_index);
|
357 |
+
|
358 |
+
$result = $str_begin . $str_end;
|
359 |
+
}
|
360 |
+
}
|
361 |
+
|
362 |
+
return $result;
|
363 |
+
}
|
364 |
+
|
365 |
+
/**
|
366 |
+
* Get only aggregate rating from SEO content.
|
367 |
+
*/
|
368 |
+
protected function _renderAggregateRating() {
|
369 |
+
$payload = $this->_renderSEO('getAggregateRating');
|
370 |
+
|
371 |
+
// remove reviews section from full_contents
|
372 |
+
$payload = $this->_replaceSection($payload, '<!--begin-reviews-->', '<!--end-reviews-->');
|
373 |
+
|
374 |
+
// remove pagination section from full contents
|
375 |
+
$payload = $this->_replaceSection($payload, '<!--begin-pagination-->', '<!--end-pagination-->');
|
376 |
+
|
377 |
+
return $payload;
|
378 |
+
}
|
379 |
+
|
380 |
+
/**
|
381 |
+
* Get only reviews from SEO content.
|
382 |
+
*/
|
383 |
+
protected function _renderReviews() {
|
384 |
+
$payload = $this->_renderSEO('getReviews');
|
385 |
+
|
386 |
+
// remove aggregate rating section from full_contents
|
387 |
+
$payload = $this->_replaceSection($payload, '<!--begin-aggregate-rating-->', '<!--end-aggregate-rating-->');
|
388 |
+
|
389 |
+
// Remove schema.org product text from reviews if it exists
|
390 |
+
$schema_org_text = "itemscope itemtype=\"http://schema.org/Product\"";
|
391 |
+
$payload = mb_ereg_replace($schema_org_text, '', $payload);
|
392 |
+
|
393 |
+
return $payload;
|
394 |
+
}
|
395 |
+
|
396 |
+
/**
|
397 |
+
* Render SEO
|
398 |
+
*
|
399 |
+
* Method used to do all the work to fetch, parse, and then return
|
400 |
+
* the SEO payload. This is set as protected so classes inheriting
|
401 |
+
* from the base class can invoke it or replace it if needed.
|
402 |
+
*
|
403 |
+
* @access protected
|
404 |
+
* @param $access_method
|
405 |
+
* @return string
|
406 |
+
*/
|
407 |
+
protected function _renderSEO($access_method) {
|
408 |
+
$payload = '';
|
409 |
+
$this->start_time = microtime(1);
|
410 |
+
|
411 |
+
$isBot = $this->_isBot();
|
412 |
+
|
413 |
+
if (!$isBot && $this->config['latency_timeout'] == 0) {
|
414 |
+
$this->_setBuildMessage("EXECUTION_TIMEOUT is set to 0 ms; JavaScript-only Display.");
|
415 |
+
} else {
|
416 |
+
|
417 |
+
if ($isBot && $this->config['latency_timeout'] < 100) {
|
418 |
+
$this->config['latency_timeout'] = 100;
|
419 |
+
$this->_setBuildMessage("EXECUTION_TIMEOUT_BOT is less than the minimum value allowed. Minimum value of 100ms used.");
|
420 |
+
}
|
421 |
+
|
422 |
+
try {
|
423 |
+
$payload = $this->_getFullSeoContents($access_method);
|
424 |
+
} catch (Exception $e) {
|
425 |
+
$this->_setBuildMessage($e->getMessage());
|
426 |
+
}
|
427 |
+
}
|
428 |
+
|
429 |
+
$payload .= $this->_buildComment($access_method);
|
430 |
+
return $payload;
|
431 |
+
}
|
432 |
+
|
433 |
+
// -------------------------------------------------------------------
|
434 |
+
// Private methods. Internal workings of SDK.
|
435 |
+
//--------------------------------------------------------------------
|
436 |
+
|
437 |
+
/**
|
438 |
+
* isBot
|
439 |
+
*
|
440 |
+
* Helper method to determine if current request is a bot or not. Will
|
441 |
+
* use the configured regex string which can be overridden with params.
|
442 |
+
*
|
443 |
+
* @access private
|
444 |
+
* @return bool
|
445 |
+
*/
|
446 |
+
private function _isBot() {
|
447 |
+
$bvreveal = $this->_getBVReveal();
|
448 |
+
if ($bvreveal) {
|
449 |
+
return TRUE;
|
450 |
+
}
|
451 |
+
|
452 |
+
// search the user agent string for an indication if this is a search bot or not
|
453 |
+
return mb_eregi('(' . $this->config['crawler_agent_pattern'] . ')', $_SERVER['HTTP_USER_AGENT']);
|
454 |
+
}
|
455 |
+
|
456 |
+
/**
|
457 |
+
* getBVReveal
|
458 |
+
*
|
459 |
+
* Return true if bvreveal flags are set, either via reveal:debug in the
|
460 |
+
* bvstate query parameter or in the old bvreveal query parameter, or is
|
461 |
+
* passed in via the configuration of the main class.
|
462 |
+
*/
|
463 |
+
private function _getBVReveal() {
|
464 |
+
// Passed in as configuration override?
|
465 |
+
if (
|
466 |
+
!empty($this->config['bvreveal']) &&
|
467 |
+
$this->config['bvreveal'] == 'debug'
|
468 |
+
) {
|
469 |
+
return TRUE;
|
470 |
+
}
|
471 |
+
// Set via bvstate query parameter?
|
472 |
+
else if (
|
473 |
+
!empty($this->config['page_params']['bvreveal']) &&
|
474 |
+
$this->config['page_params']['bvreveal'] == 'debug'
|
475 |
+
) {
|
476 |
+
return TRUE;
|
477 |
+
}
|
478 |
+
// Set via bvreveal query parameter?
|
479 |
+
else if (
|
480 |
+
!empty($this->config['bv_page_data']['bvreveal']) &&
|
481 |
+
$this->config['bv_page_data']['bvreveal'] == 'debug'
|
482 |
+
) {
|
483 |
+
return TRUE;
|
484 |
+
} else {
|
485 |
+
return FALSE;
|
486 |
+
}
|
487 |
+
}
|
488 |
+
|
489 |
+
/**
|
490 |
+
* getPageNumber
|
491 |
+
*
|
492 |
+
* Helper method to pull from the URL the page of SEO we need to view.
|
493 |
+
*
|
494 |
+
* @access private
|
495 |
+
* @return int
|
496 |
+
*/
|
497 |
+
private function _getPageNumber() {
|
498 |
+
$page_number = 1;
|
499 |
+
|
500 |
+
// Override from config.
|
501 |
+
if (isset($this->config['page']) && $this->config['page'] != $page_number) {
|
502 |
+
$page_number = (int) $this->config['page'];
|
503 |
+
}
|
504 |
+
// Check the bvstate parameter if one was found and successfully parsed.
|
505 |
+
else if (isset($this->config['page_params']['base_url_bvstate'])) {
|
506 |
+
// We only apply the bvstate page number parameter if the content type
|
507 |
+
// specified matches the content type being generated here. E.g. if
|
508 |
+
// someone calls up a page with bvstate=ct:r/pg:2 and loads stories rather
|
509 |
+
// than reviews, show page 1 for stories. Only show page 2 if they are in
|
510 |
+
// fact displaying review content.
|
511 |
+
if ($this->config['content_type'] == $this->config['page_params']['content_type']) {
|
512 |
+
$page_number = $this->config['page_params']['page'];
|
513 |
+
}
|
514 |
+
}
|
515 |
+
// other implementations use the bvrrp, bvqap, or bvsyp parameter
|
516 |
+
// ?bvrrp=1234-en_us/reviews/product/2/ASF234.htm
|
517 |
+
//
|
518 |
+
// Note that unlike bvstate, we don't actually check for the content type
|
519 |
+
// to match the parameter type for the legacy page parameters bvrrp, bvqap,
|
520 |
+
// and bvsyp. This is consistent with the behavior of the other SDKs, even
|
521 |
+
// if it doesn't really make much sense.
|
522 |
+
//
|
523 |
+
// Note that there is a bug in the SEO-CPS content generation where it uses
|
524 |
+
// the bvrrp parameter in place of bvqap, so this may all be sort of
|
525 |
+
// deliberate, if not sensible.
|
526 |
+
else if (isset($this->config['bv_page_data']['bvrrp'])) {
|
527 |
+
$bvparam = $this->config['bv_page_data']['bvrrp'];
|
528 |
+
} else if (isset($this->config['bv_page_data']['bvqap'])) {
|
529 |
+
$bvparam = $this->config['bv_page_data']['bvqap'];
|
530 |
+
} else if (isset($this->config['bv_page_data']['bvsyp'])) {
|
531 |
+
$bvparam = $this->config['bv_page_data']['bvsyp'];
|
532 |
+
}
|
533 |
+
|
534 |
+
if (!empty($bvparam)) {
|
535 |
+
$match = array();
|
536 |
+
mb_ereg('\/(\d+)\/', $bvparam, $match);
|
537 |
+
$page_number = max(1, (int) $match[1]);
|
538 |
+
}
|
539 |
+
|
540 |
+
return $page_number;
|
541 |
+
}
|
542 |
+
|
543 |
+
/**
|
544 |
+
* buildSeoUrl
|
545 |
+
*
|
546 |
+
* Helper method to that builds the URL to the SEO payload
|
547 |
+
*
|
548 |
+
* @access private
|
549 |
+
* @param int (page number)
|
550 |
+
* @return string
|
551 |
+
*/
|
552 |
+
private function _buildSeoUrl($page_number) {
|
553 |
+
$primary_selector = 'seo-domain';
|
554 |
+
|
555 |
+
// calculate, which environment should we be using
|
556 |
+
if ($this->config['testing']) {
|
557 |
+
if ($this->config['staging']) {
|
558 |
+
$env_selector = 'testing_staging';
|
559 |
+
} else {
|
560 |
+
$env_selector = 'testing_production';
|
561 |
+
}
|
562 |
+
} else {
|
563 |
+
if ($this->config['staging']) {
|
564 |
+
$env_selector = 'staging';
|
565 |
+
} else {
|
566 |
+
$env_selector = 'production';
|
567 |
+
}
|
568 |
+
}
|
569 |
+
|
570 |
+
$url_scheme = $this->config['ssl_enabled'] ? 'https://' : 'http://';
|
571 |
+
|
572 |
+
if ($this->config['content_type'] == 'reviews' &&
|
573 |
+
$this->config['subject_type'] == 'seller') {
|
574 |
+
// when content type is reviews and subject type is seller,
|
575 |
+
// we're dealing with seller rating, so use different primary selector
|
576 |
+
$primary_selector = 'srd-domain';
|
577 |
+
// for seller rating we use different selector for prefix
|
578 |
+
$hostname = $this->bv_config[$primary_selector] . '/' . $this->bv_config['srd-prefix-' . $env_selector];
|
579 |
+
} else {
|
580 |
+
$hostname = $this->bv_config[$primary_selector][$env_selector];
|
581 |
+
};
|
582 |
+
|
583 |
+
// dictates order of URL
|
584 |
+
$url_parts = array(
|
585 |
+
$url_scheme . $hostname,
|
586 |
+
$this->config['cloud_key'],
|
587 |
+
$this->config['bv_root_folder'],
|
588 |
+
$this->config['content_type'],
|
589 |
+
$this->config['subject_type'],
|
590 |
+
$page_number
|
591 |
+
);
|
592 |
+
|
593 |
+
if (isset($this->config['content_sub_type']) && !empty($this->config['content_sub_type'])) {
|
594 |
+
$url_parts[] = $this->config['content_sub_type'];
|
595 |
+
}
|
596 |
+
|
597 |
+
if (!empty($this->config['page_params']['subject_id']) && $this->_checkBVStateContentType()) {
|
598 |
+
$url_parts[] = urlencode($this->config['page_params']['subject_id']) . '.htm';
|
599 |
+
} else {
|
600 |
+
$url_parts[] = urlencode($this->config['subject_id']) . '.htm';
|
601 |
+
}
|
602 |
+
|
603 |
+
// if our SEO content source is a file path
|
604 |
+
// we need to remove the first two sections
|
605 |
+
// and prepend the passed in file path
|
606 |
+
if (!empty($this->config['load_seo_files_locally']) && !empty($this->config['local_seo_file_root'])) {
|
607 |
+
unset($url_parts[0]);
|
608 |
+
unset($url_parts[1]);
|
609 |
+
|
610 |
+
return $this->config['local_seo_file_root'] . implode("/", $url_parts);
|
611 |
+
}
|
612 |
+
|
613 |
+
// implode will convert array to a string with / in between each value in array
|
614 |
+
return implode("/", $url_parts);
|
615 |
+
}
|
616 |
+
|
617 |
+
/**
|
618 |
+
* Return a SEO content from local or distant sourse.
|
619 |
+
*/
|
620 |
+
private function _fetchSeoContent($resource) {
|
621 |
+
if ($this->config['load_seo_files_locally']) {
|
622 |
+
return $this->_fetchFileContent($resource);
|
623 |
+
} else {
|
624 |
+
return $this->_fetchCloudContent($resource);
|
625 |
+
}
|
626 |
+
}
|
627 |
+
|
628 |
+
/**
|
629 |
+
* fetchFileContent
|
630 |
+
*
|
631 |
+
* Helper method that will take in a file path and return it's payload while
|
632 |
+
* handling the possible errors or exceptions that can happen.
|
633 |
+
*
|
634 |
+
* @access private
|
635 |
+
* @param string (valid file path)
|
636 |
+
* @return string (content of file)
|
637 |
+
*/
|
638 |
+
private function _fetchFileContent($path) {
|
639 |
+
$file = @file_get_contents($path);
|
640 |
+
if ($file === FALSE) {
|
641 |
+
$this->_setBuildMessage('Trying to get content from "' . $path
|
642 |
+
. '". The resource file is currently unavailable');
|
643 |
+
} else {
|
644 |
+
$this->_setBuildMessage('Local file content was uploaded');
|
645 |
+
}
|
646 |
+
return $file;
|
647 |
+
}
|
648 |
+
|
649 |
+
public function curlExecute($ch) {
|
650 |
+
return curl_exec($ch);
|
651 |
+
}
|
652 |
+
|
653 |
+
public function curlInfo($ch) {
|
654 |
+
return curl_getinfo($ch);
|
655 |
+
}
|
656 |
+
|
657 |
+
public function curlErrorNo($ch) {
|
658 |
+
return curl_errno($ch);
|
659 |
+
}
|
660 |
+
|
661 |
+
public function curlError($ch) {
|
662 |
+
return curl_error($ch);
|
663 |
+
}
|
664 |
+
|
665 |
+
/**
|
666 |
+
* fetchCloudContent
|
667 |
+
*
|
668 |
+
* Helper method that will take in a URL and return it's payload while
|
669 |
+
* handling the possible errors or exceptions that can happen.
|
670 |
+
*
|
671 |
+
* @access private
|
672 |
+
* @param string (valid url)
|
673 |
+
* @return string
|
674 |
+
*/
|
675 |
+
private function _fetchCloudContent($url) {
|
676 |
+
|
677 |
+
// is cURL installed yet?
|
678 |
+
// if ( ! function_exists('curl_init')){
|
679 |
+
// return '<!-- curl library is not installed -->';
|
680 |
+
// }
|
681 |
+
// create a new cURL resource handle
|
682 |
+
$ch = curl_init();
|
683 |
+
|
684 |
+
// Set URL to download
|
685 |
+
curl_setopt($ch, CURLOPT_URL, $url);
|
686 |
+
// Set a referer as coming from the current page url
|
687 |
+
curl_setopt($ch, CURLOPT_REFERER, $this->config['page_url']);
|
688 |
+
// Include header in result? (0 = yes, 1 = no)
|
689 |
+
curl_setopt($ch, CURLOPT_HEADER, 0);
|
690 |
+
// Should cURL return or print out the data? (true = return, false = print)
|
691 |
+
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
692 |
+
// Timeout in seconds
|
693 |
+
curl_setopt($ch, CURLOPT_TIMEOUT_MS, $this->config['latency_timeout']);
|
694 |
+
// Enable decoding of the response
|
695 |
+
curl_setopt($ch, CURLOPT_ENCODING, 'gzip,deflate');
|
696 |
+
// Enable following of redirects
|
697 |
+
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
|
698 |
+
// set user agent if needed
|
699 |
+
if ($this->config['user_agent'] != '') {
|
700 |
+
curl_setopt($ch, CURLOPT_USERAGENT, $this->config['user_agent']);
|
701 |
+
}
|
702 |
+
|
703 |
+
if ($this->config['proxy_host'] != '') {
|
704 |
+
curl_setopt($ch, CURLOPT_PROXY, $this->config['proxy_host']);
|
705 |
+
curl_setopt($ch, CURLOPT_PROXYPORT, $this->config['proxy_port']);
|
706 |
+
}
|
707 |
+
|
708 |
+
// make the request to the given URL and then store the response,
|
709 |
+
// request info, and error number
|
710 |
+
// so we can use them later
|
711 |
+
$request = array(
|
712 |
+
'response' => $this->curlExecute($ch),
|
713 |
+
'info' => $this->curlInfo($ch),
|
714 |
+
'error_number' => $this->curlErrorNo($ch),
|
715 |
+
'error_message' => $this->curlError($ch)
|
716 |
+
);
|
717 |
+
|
718 |
+
// Close the cURL resource, and free system resources
|
719 |
+
curl_close($ch);
|
720 |
+
|
721 |
+
// see if we got any errors with the connection
|
722 |
+
if ($request['error_number'] != 0) {
|
723 |
+
$this->_setBuildMessage('Error - ' . $request['error_message']);
|
724 |
+
}
|
725 |
+
|
726 |
+
// see if we got a status code of something other than 200
|
727 |
+
if ($request['info']['http_code'] != 200) {
|
728 |
+
$this->_setBuildMessage('HTTP status code of '
|
729 |
+
. $request['info']['http_code'] . ' was returned');
|
730 |
+
return '';
|
731 |
+
}
|
732 |
+
|
733 |
+
// if we are here we got a response so let's return it
|
734 |
+
$this->response_time = round($request['info']['total_time'] * 1000);
|
735 |
+
return $request['response'];
|
736 |
+
}
|
737 |
+
|
738 |
+
/**
|
739 |
+
* replaceTokens
|
740 |
+
*
|
741 |
+
* After we have an SEO payload we need to replace the {INSERT_PAGE_URI}
|
742 |
+
* tokens with the current page url so pagination works.
|
743 |
+
*
|
744 |
+
* @access private
|
745 |
+
* @param string (valid url)
|
746 |
+
* @return string
|
747 |
+
*/
|
748 |
+
private function _replaceTokens($content) {
|
749 |
+
$page_url_query_prefix = '';
|
750 |
+
|
751 |
+
// Attach a suitable ending to the base URL if it doesn't already end with
|
752 |
+
// either ? or &. This is complicated by the _escaped_fragment_ case.
|
753 |
+
//
|
754 |
+
// We're assuming that the base URL can't have a fragment or be a hashbang
|
755 |
+
// URL - that just won't work in conjunction with the assumption that we
|
756 |
+
// always postfix the SEO query parameters to the end of the URL.
|
757 |
+
//
|
758 |
+
// If the base url ends with an empty _escaped_fragment_ property.
|
759 |
+
if (mb_ereg('_escaped_fragment_=$', $this->config['base_url'])) {
|
760 |
+
// Append nothing for this annoying edge case.
|
761 |
+
}
|
762 |
+
// Otherwise if there is something in the _escaped_fragment_ then append
|
763 |
+
// the escaped ampersand.
|
764 |
+
else if (mb_ereg('_escaped_fragment_=.+$', $this->config['base_url'])) {
|
765 |
+
$page_url_query_prefix = '%26';
|
766 |
+
}
|
767 |
+
// Otherwise we're back to thinking about query strings.
|
768 |
+
else if (!mb_ereg('[\?&]$', $this->config['base_url'])) {
|
769 |
+
if(mb_ereg('\?', $this->config['base_url'])) {
|
770 |
+
$page_url_query_prefix = '&';
|
771 |
+
} else {
|
772 |
+
$page_url_query_prefix = '?';
|
773 |
+
}
|
774 |
+
}
|
775 |
+
|
776 |
+
$content = mb_ereg_replace(
|
777 |
+
'{INSERT_PAGE_URI}',
|
778 |
+
// Make sure someone doesn't sneak in "><script>...<script> in the URL
|
779 |
+
// contents.
|
780 |
+
htmlspecialchars(
|
781 |
+
$this->config['base_url'] . $page_url_query_prefix,
|
782 |
+
ENT_QUOTES | ENT_HTML5,
|
783 |
+
$this->config['charset'],
|
784 |
+
// Don't double-encode.
|
785 |
+
false
|
786 |
+
),
|
787 |
+
$content
|
788 |
+
);
|
789 |
+
|
790 |
+
return $content;
|
791 |
+
}
|
792 |
+
|
793 |
+
/**
|
794 |
+
* Return hidden metadata for adding to SEO content.
|
795 |
+
*/
|
796 |
+
private function _buildComment($access_method) {
|
797 |
+
$bvf = new BVFooter($this, $access_method, $this->msg);
|
798 |
+
$footer = $bvf->buildSDKFooter();
|
799 |
+
$reveal = $this->_getBVReveal();
|
800 |
+
if ($reveal) {
|
801 |
+
$footer .= $bvf->buildSDKDebugFooter();
|
802 |
+
}
|
803 |
+
return $footer;
|
804 |
+
}
|
805 |
+
|
806 |
+
public function getBVMessages() {
|
807 |
+
return $this->msg;
|
808 |
+
}
|
809 |
+
|
810 |
+
public function getContent() {
|
811 |
+
$this->_setBuildMessage('Content Type "' . $this->config['content_type'] . '" is not supported by getContent().');
|
812 |
+
$pay_load = $this->_buildComment('', 'getContent');
|
813 |
+
|
814 |
+
return $pay_load;
|
815 |
+
}
|
816 |
+
|
817 |
+
public function getAggregateRating() {
|
818 |
+
$this->_setBuildMessage('Content Type "' . $this->config['content_type'] . '" is not supported by getAggregateRating().');
|
819 |
+
$pay_load = $this->_buildComment('', 'getAggregateRating');
|
820 |
+
|
821 |
+
return $pay_load;
|
822 |
+
}
|
823 |
+
|
824 |
+
public function getReviews() {
|
825 |
+
$this->_setBuildMessage('Content Type "' . $this->config['content_type'] . '" is not supported by getReviews().');
|
826 |
+
$pay_load = $this->_buildComment('', 'getReviews');
|
827 |
+
|
828 |
+
return $pay_load;
|
829 |
+
}
|
830 |
+
|
831 |
+
}
|
832 |
+
// end of Base class
|
833 |
+
|
834 |
+
/**
|
835 |
+
* Reviews Class
|
836 |
+
*
|
837 |
+
* Base class extention for work with "reviews" content type.
|
838 |
+
*/
|
839 |
+
class Reviews extends Base {
|
840 |
+
|
841 |
+
function __construct($params = array()) {
|
842 |
+
// call Base Class constructor
|
843 |
+
parent::__construct($params);
|
844 |
+
|
845 |
+
// since we are in the reviews class
|
846 |
+
// we need to set the content_type config
|
847 |
+
// to reviews so we get reviews in our
|
848 |
+
// SEO request
|
849 |
+
$this->config['content_type'] = 'reviews';
|
850 |
+
|
851 |
+
// for reviews subject type will always
|
852 |
+
// need to be product
|
853 |
+
$this->config['subject_type'] = 'product';
|
854 |
+
}
|
855 |
+
|
856 |
+
public function getAggregateRating() {
|
857 |
+
return $this->_renderAggregateRating();
|
858 |
+
}
|
859 |
+
|
860 |
+
public function getReviews() {
|
861 |
+
return $this->_renderReviews();
|
862 |
+
}
|
863 |
+
|
864 |
+
public function getContent() {
|
865 |
+
$payload = $this->_renderSEO('getContent');
|