OSDN Git Service

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