OSDN Git Service

636edfcb70439da0515ec778f53d0915b2a1bc5d
[pukiwiki/pukiwiki.git] / plugin / amazon.inc.php
1 <?php
2 // PukiWiki - Yet another WikiWikiWeb clone.
3 // amazon.inc.php
4 //
5 // Amazon plugin: Book-review maker via amazon.com/amazon.jp
6 //
7 // Copyright:
8 //      2004-2016 PukiWiki Development Team
9 //      2003 閑舎 <raku@rakunet.org> (Original author)
10 //
11 // License: GNU/GPL
12 //
13 // ChangeLog:
14 // * 2004/04/03 PukiWiki Developer Team (arino <arino@users.osdn.me>)
15 //        - replace plugin_amazon_get_page().
16 //        - PLUGIN_AMAZON_XML 'xml.amazon.com' -> 'xml.amazon.co.jp'
17 // * 0.6  URL が存在しない場合、No image を表示、画像配置など修正。
18 //        インラインプラグインの呼び出し方を修正。
19 //        ASIN 番号部分をチェックする。
20 //        画像、タイトルのキャッシュによる速度の大幅アップ。
21 // * 0.7  ブックレビュー生成のデバッグ、認証問題の一応のクリア。
22 // * 0.8  amazon 全商品の画像を表示。
23 //        アソシエイト ID に対応。
24 // * 0.9  RedHat9+php4.3.2+apache2.0.46 で画像が途中までしか読み込まれない問題に対処。
25 //        日本語ページの下にブックレビューを作ろうとすると文字化けして作れない問題の解決。
26 //        書籍でなく CD など、ASIN 部分が長くてもタイトルをうまく拾うようにする。
27 //        写影のみ取り込むのでなければ、B000002G6J.01 と書かず B000002G6J と書いても写影が出るようにする。
28 //        ASIN に対応するキャッシュ画像/キャッシュタイトルをそれぞれ削除する機能追加。
29 //        proxy 対応(試験的)。
30 //        proxy 実装の過程で一般ユーザのための AID はなくとも自動生成されることがわかり、削除した。
31 // * 1.0  ブックレビューでなく、レビューとする。
32 //        画像のキャッシュを削除する期限を設ける。
33 //        タイトル、写影を Web Services の XML アクセスの方法によって get することで時間を短縮する。
34 //        レビューページ生成のタイミングについて注を入れる。
35 // * 1.1  編集制限をかけている場合、部外者がレビューを作ろうとして、ページはできないが ASIN4774110655.tit などのキャッシュができるのを解決。
36 //        画像の最後が 01 の場合、image を削除すると noimage.jpg となってしまうバグを修正。
37 //        1.0 で導入した XML アクセスは高速だが、返す画像情報がウソなので、09 がだめなら 01 をトライする、で暫定的に解決。
38 //
39 // Caution!:
40 // * 著作権が関連する為、www.amazon.co.jp のアソシエイトプログラムを確認の上ご利用下さい。
41 // * レビューは、amazon プラグインが呼び出す編集画面はもう出来て PukiWiki に登録されているので、
42 //   中止するなら全文を削除してページの更新ボタンを押すこと。
43 // * 下の PLUGIN_AMAZON_AID、PROXY サーバの部分、expire の部分を適当に編集して使用してください(他はそのままでも Ok)。
44 //
45 // Thanks to: Reimy and PukiWiki Developers Team
46 //
47
48 /////////////////////////////////////////////////
49 // Settings
50
51 // Amazon associate ID
52 //define('PLUGIN_AMAZON_AID',''); // None
53 define('PLUGIN_AMAZON_AID','');
54
55 // Expire caches per ? days
56 define('PLUGIN_AMAZON_EXPIRE_IMAGECACHE',   1);
57 define('PLUGIN_AMAZON_EXPIRE_TITLECACHE', 356);
58
59 // Alternative image for 'Image not found'
60 define('PLUGIN_AMAZON_NO_IMAGE', IMAGE_DIR . 'noimage.png');
61
62 // URI prefixes
63 switch(LANG){
64 case 'ja':
65         // Amazon shop
66         define('PLUGIN_AMAZON_SHOP_URI', 'http://www.amazon.co.jp/exec/obidos/ASIN/');
67
68         // Amazon information inquiry (dev-t = default value in the manual)
69         define('PLUGIN_AMAZON_XML', 'http://xml.amazon.co.jp/onca/xml3?t=webservices-20&' .
70                 'dev-t=GTYDRES564THU&type=lite&page=1&f=xml&locale=jp&AsinSearch=');
71         break;
72 default:
73         // Amazon shop
74         define('PLUGIN_AMAZON_SHOP_URI', 'http://www.amazon.com/exec/obidos/ASIN/');
75
76         // Amazon information inquiry (dev-t = default value in the manual)
77         define('PLUGIN_AMAZON_XML', 'http://xml.amazon.com/onca/xml3?t=webservices-20&' .
78                 'dev-t=GTYDRES564THU&type=lite&page=1&f=xml&locale=us&AsinSearch=');
79         break;
80 }
81
82 /////////////////////////////////////////////////
83
84 function plugin_amazon_init()
85 {
86         global $amazon_aid, $amazon_body;
87
88         if (PLUGIN_AMAZON_AID == '') {
89                 $amazon_aid = '';
90         } else {
91                 $amazon_aid = PLUGIN_AMAZON_AID . '/';
92         }
93         $amazon_body = <<<EOD
94 -作者: [[ここ編集のこと]]
95 -評者: お名前
96 -日付: &date;
97 **お薦め対象
98 [[ここ編集のこと]]
99
100 #amazon(,clear)
101 **感想
102 [[ここ編集のこと]]
103
104 // まず、このレビューを止める場合、全文を削除し、ページの[更新ボタン]を押してください!(PukiWiki にはもう登録されています)
105 // 続けるなら、上の、[[ここ編集のこと]]部分を括弧を含めて削除し、書き直してください。
106 // お名前、部分はご自分の名前に変更してください。私だと、閑舎、です。
107 // **お薦め対象、より上は、新しい行を追加しないでください。目次作成に使用するので。
108 // //で始まるコメント行は、最終的に全部カットしてください。目次が正常に作成できない可能性があります。
109 #comment
110 EOD;
111 }
112
113 function plugin_amazon_convert()
114 {
115         global $script, $vars, $asin, $asin_all;
116
117         if (func_num_args() > 3) {
118                 if (PKWK_READONLY) return ''; // Show nothing
119
120                 return '#amazon([ASIN-number][,left|,right]' .
121                         '[,book-title|,image|,delimage|,deltitle|,delete])';
122
123         } else if (func_num_args() == 0) {
124                 // レビュー作成
125                 if (PKWK_READONLY) return ''; // Show nothing
126
127                 $s_page = htmlsc($vars['page']);
128                 if ($s_page == '') $s_page = isset($vars['refer']) ? $vars['refer'] : '';
129                 $ret = <<<EOD
130 <form action="$script" method="post">
131  <div>
132   <input type="hidden" name="plugin" value="amazon" />
133   <input type="hidden" name="refer" value="$s_page" />
134   ASIN:
135   <input type="text" name="asin" size="30" value="" />
136   <input type="submit" value="レビュー編集" /> (ISBN 10 桁 or ASIN 12 桁)
137  </div>
138 </form>
139 EOD;
140                 return $ret;
141         }
142
143         $aryargs = func_get_args();
144
145         $align = strtolower($aryargs[1]);
146         if ($align == 'clear') return '<div style="clear:both"></div>'; // 改行挿入
147         if ($align != 'left') $align = 'right'; // 配置決定
148
149         $asin_all = htmlsc($aryargs[0]);  // for XSS
150         if (is_asin() == FALSE && $align != 'clear') return FALSE;
151
152         if ($aryargs[2] != '') {
153                 // タイトル指定
154                 $title = $alt = htmlsc($aryargs[2]); // for XSS
155                 if ($alt == 'image') {
156                         $alt = plugin_amazon_get_asin_title();
157                         if ($alt == '') return FALSE;
158                         $title = '';
159                 } else if ($alt == 'delimage') {
160                         if (unlink(CACHE_DIR . 'ASIN' . $asin . '.jpg')) {
161                                 return 'Image of ' . $asin . ' deleted...';
162                         } else {
163                                 return 'Image of ' . $asin . ' NOT DELETED...';
164                         }
165                 } elseif ($alt == 'deltitle') {
166                         if (unlink(CACHE_DIR . 'ASIN' . $asin . '.tit')) {
167                                 return 'Title of ' . $asin . ' deleted...';
168                         } else {
169                                 return 'Title of ' . $asin . ' NOT DELETED...';
170                         }
171                 } elseif ($alt == 'delete') {
172                         if ((unlink(CACHE_DIR . 'ASIN' . $asin . '.jpg') &&
173                              unlink(CACHE_DIR . 'ASIN' . $asin . '.tit'))) {
174                                 return 'Title and Image of ' . $asin . ' deleted...';
175                         } else {
176                                 return 'Title and Image of ' . $asin . ' NOT DELETED...';
177                         }
178                 }
179         } else {
180                 // タイトル自動取得
181                 $alt = $title = plugin_amazon_get_asin_title();
182                 if ($alt == '') return FALSE;
183         }
184
185         return plugin_amazon_print_object($align, $alt, $title);
186 }
187
188 function plugin_amazon_action()
189 {
190         global $vars, $script, $edit_auth, $edit_auth_users;
191         global $amazon_body, $asin, $asin_all;
192
193         if (PKWK_READONLY) die_message('PKWK_READONLY prohibits editing');
194
195         $s_page   = isset($vars['refer']) ? $vars['refer'] : '';
196         $asin_all = isset($vars['asin']) ?
197                 htmlsc(rawurlencode(strip_bracket($vars['asin']))) : '';
198
199         if (! is_asin()) {
200                 $retvars['msg']   = 'ブックレビュー編集';
201                 $retvars['refer'] = & $s_page;
202                 $retvars['body']  = plugin_amazon_convert();
203                 return $retvars;
204
205         } else {
206                 $r_page     = $s_page . '/' . $asin;
207                 $r_page_url = rawurlencode($r_page);
208                 $auth_user = isset($_SERVER['PHP_AUTH_USER']) ? $_SERVER['PHP_AUTH_USER'] : '';
209
210                 pkwk_headers_sent();
211                 if ($edit_auth && ($auth_user == '' || ! isset($edit_auth_users[$auth_user]) ||
212                     $edit_auth_users[$auth_user] != $_SERVER['PHP_AUTH_PW'])) {
213                         // Edit-auth failed. Just look the page
214                         header('Location: ' . get_script_uri() . '?' . $r_page_url);
215                 } else {
216                         $title = plugin_amazon_get_asin_title();
217                         if ($title == '' || preg_match('#^/#', $s_page)) {
218                                 // Invalid page name
219                                 header('Location: ' . get_script_uri() . '?' . pagename_urlencode($s_page));
220                         } else {
221                                 $body = '#amazon(' . $asin_all . ',,image)' . "\n" .
222                                         '*' . $title . "\n" . $amazon_body;
223                                 plugin_amazon_review_save($r_page, $body);
224                                 header('Location: ' . get_script_uri() .
225                                         '?cmd=edit&page=' . $r_page_url);
226                         }
227                 }
228                 exit;
229         }
230 }
231
232 function plugin_amazon_inline()
233 {
234         global $amazon_aid, $asin, $asin_all;
235
236         list($asin_all) = func_get_args();
237
238         $asin_all = htmlsc($asin_all); // for XSS
239         if (! is_asin()) return FALSE;
240
241         $title = plugin_amazon_get_asin_title();
242         if ($title == '') {
243                 return FALSE;
244         } else {
245                 return '<a href="' . PLUGIN_AMAZON_SHOP_URI .
246                         $asin . '/' . $amazon_aid . 'ref=nosim">' . $title . '</a>' . "\n";
247         }
248 }
249
250 function plugin_amazon_print_object($align, $alt, $title)
251 {
252         global $amazon_aid;
253         global $asin, $asin_ext, $asin_all;
254
255         $url      = plugin_amazon_cache_image_fetch(CACHE_DIR);
256         $url_shop = PLUGIN_AMAZON_SHOP_URI . $asin . '/' . $amazon_aid . 'ref=nosim';
257         $center   = 'text-align:center';
258
259         if ($title == '') {
260                 // Show image only
261                 $div  = '<div style="float:' . $align . ';margin:16px 16px 16px 16px;' . $center . '">' . "\n";
262                 $div .= ' <a href="' . $url_shop . '"><img src="' . $url . '" alt="' . $alt . '" /></a>' . "\n";
263                 $div .= '</div>' . "\n";
264
265         } else {
266                 // Show image and title
267                 $div  = '<div style="float:' . $align . ';padding:.5em 1.5em .5em 1.5em;' . $center . '">' . "\n";
268                 $div .= ' <table style="width:110px;border:0;' . $center . '">' . "\n";
269                 $div .= '  <tr><td style="' . $center . '">' . "\n";
270                 $div .= '   <a href="' . $url_shop . '"><img src="' . $url . '" alt="' . $alt  .'" /></a></td></tr>' . "\n";
271                 $div .= '  <tr><td style="' . $center . '"><a href="' . $url_shop . '">' . $title . '</a></td></tr>' . "\n";
272                 $div .= ' </table>' . "\n";
273                 $div .= '</div>' . "\n";
274         }
275         return $div;
276 }
277
278 function plugin_amazon_get_asin_title()
279 {
280         global $asin, $asin_ext, $asin_all;
281
282         if ($asin_all == '') return '';
283
284         $nocache = $nocachable = 0;
285
286         $url = PLUGIN_AMAZON_XML . $asin;
287
288         if (file_exists(CACHE_DIR) === FALSE || is_writable(CACHE_DIR) === FALSE) $nocachable = 1; // キャッシュ不可の場合
289
290         if (($title = plugin_amazon_cache_title_fetch(CACHE_DIR)) == FALSE) {
291                 $nocache = 1; // キャッシュ見つからず
292                 $body    = plugin_amazon_get_page($url); // しかたないので取りにいく
293                 $tmpary  = array();
294                 $body    = mb_convert_encoding($body, SOURCE_ENCODING, 'UTF-8');
295                 preg_match('/<ProductName>([^<]*)</', $body, $tmpary);
296                 $title     = trim($tmpary[1]);
297 //              $tmpary[1] = '';
298 //              preg_match('#<ImageUrlMedium>http://images-jp.amazon.com/images/P/[^.]+\.(..)\.#',
299 //                      $body, $tmpary);
300 //              if ($tmpary[1] != '') {
301 //                      $asin_ext = $tmpary[1];
302 //                      $asin_all = $asin . $asin_ext;
303 //              }
304         }
305
306         if ($title == '') {
307                 return '';
308         } else {
309                 if ($nocache == 1 && $nocachable != 1)
310                         plugin_amazon_cache_title_save($title, CACHE_DIR);
311                 return $title;
312         }
313 }
314
315 // タイトルキャッシュがあるか調べる
316 function plugin_amazon_cache_title_fetch($dir)
317 {
318         global $asin, $asin_ext, $asin_all;
319
320         $filename = $dir . 'ASIN' . $asin . '.tit';
321
322         $get_tit = 0;
323         if (! is_readable($filename)) {
324                 $get_tit = 1;
325         } elseif (PLUGIN_AMAZON_EXPIRE_TITLECACHE * 3600 * 24 < time() - filemtime($filename)) {
326                 $get_tit = 1;
327         }
328
329         if ($get_tit) return FALSE;
330
331         if (($fp = @fopen($filename, 'r')) === FALSE) return FALSE;
332         $title = fgets($fp, 4096);
333 //      $tmp_ext = fgets($fp, 4096);
334 //      if ($tmp_ext != '') $asin_ext = $tmp_ext;
335         fclose($fp);
336
337         if (strlen($title) > 0) {
338                 return $title;
339         } else {
340                 return FALSE;
341         }
342 }
343
344 // 画像キャッシュがあるか調べる
345 function plugin_amazon_cache_image_fetch($dir)
346 {
347         global $asin, $asin_ext, $asin_all;
348
349         $filename = $dir . 'ASIN' . $asin . '.jpg';
350
351         $get_img = 0;
352         if (! is_readable($filename)) {
353                 $get_img = 1;
354         } elseif (PLUGIN_AMAZON_EXPIRE_IMAGECACHE * 3600 * 24 < time() - filemtime($filename)) {
355                 $get_img = 1;
356         }
357
358         if ($get_img) {
359                 $url = 'http://images-jp.amazon.com/images/P/' . $asin . '.' . $asin_ext . '.MZZZZZZZ.jpg';
360                 if (! is_url($url)) return FALSE;
361
362                 $body = plugin_amazon_get_page($url);
363                 if ($body != '') {
364                         $tmpfile = $dir . 'ASIN' . $asin . '.jpg.0';
365                         $fp = fopen($tmpfile, 'wb');
366                         fwrite($fp, $body);
367                         fclose($fp);
368                         $size = getimagesize($tmpfile);
369                         unlink($tmpfile);
370                 }
371                 if ($body == '' || $size[1] <= 1) { // 通常は1が返るが念のため0の場合も(reimy)
372                         // キャッシュを PLUGIN_AMAZON_NO_IMAGE のコピーとする
373                         if ($asin_ext == '09') {
374                                 $url = 'http://images-jp.amazon.com/images/P/' . $asin . '.01.MZZZZZZZ.jpg';
375                                 $body = plugin_amazon_get_page($url);
376                                 if ($body != '') {
377                                         $tmpfile = $dir . 'ASIN' . $asin . '.jpg.0';
378                                         $fp = fopen($tmpfile, 'wb');
379                                         fwrite($fp, $body);
380                                         fclose($fp);
381                                         $size = getimagesize($tmpfile);
382                                         unlink($tmpfile);
383                                 }
384                         }
385                         if ($body == '' || $size[1] <= 1) {
386                                 $fp = fopen(PLUGIN_AMAZON_NO_IMAGE, 'rb');
387                                 if (! $fp) return FALSE;
388                                 
389                                 $body = '';
390                                 while (! feof($fp)) $body .= fread($fp, 4096);
391                                 fclose ($fp);
392                         }
393                 }
394                 plugin_amazon_cache_image_save($body, CACHE_DIR);
395         }
396         return $filename;
397 }
398
399 // Save title cache
400 function plugin_amazon_cache_title_save($data, $dir)
401 {
402         global $asin, $asin_ext, $asin_all;
403
404         $filename = $dir . 'ASIN' . $asin . '.tit';
405         $fp = fopen($filename, 'w');
406         fwrite($fp, $data);
407         fclose($fp);
408
409         return $filename;
410 }
411
412 // Save image cache
413 function plugin_amazon_cache_image_save($data, $dir)
414 {
415         global $asin, $asin_ext, $asin_all;
416
417         $filename = $dir . 'ASIN' . $asin . '.jpg';
418         $fp = fopen($filename, 'wb');
419         fwrite($fp, $data);
420         fclose($fp);
421
422         return $filename;
423 }
424
425 // Save book data
426 function plugin_amazon_review_save($page, $data)
427 {
428         global $asin, $asin_ext, $asin_all;
429
430         $filename = DATA_DIR . encode($page) . '.txt';
431         if (! is_readable($filename)) {
432                 $fp = fopen($filename, 'w');
433                 fwrite($fp, $data);
434                 fclose($fp);
435                 return TRUE;
436         } else {
437                 return FALSE;
438         }
439 }
440
441 function plugin_amazon_get_page($url)
442 {
443         $data = pkwk_http_request($url);
444         return ($data['rc'] == 200) ? $data['data'] : '';
445 }
446
447 // is ASIN?
448 function is_asin()
449 {
450         global $asin, $asin_ext, $asin_all;
451
452         $tmpary = array();
453         if (preg_match('/^([A-Z0-9]{10}).?([0-9][0-9])?$/', $asin_all, $tmpary) == FALSE) {
454                 return FALSE;
455         } else {
456                 $asin     = $tmpary[1];
457                 $asin_ext = isset($tmpary[2]) ? $tmpary[2] : '';
458                 if ($asin_ext == '') $asin_ext = '09';
459                 $asin_all = $asin . $asin_ext;
460                 return TRUE;
461         }
462 }