/[pub]/suikawiki/script/wiki.cgi
Suika

Contents of /suikawiki/script/wiki.cgi

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.7 - (show annotations) (download)
Mon Feb 4 15:27:22 2002 UTC (22 years, 3 months ago) by wakaba
Branch: MAIN
Changes since 1.6: +6 -7 lines
*** empty log message ***

1 #!/usr/bin/perl
2 use lib "../lib";
3 use CGI::Carp 'fatalsToBrowser';
4 use Algorithm::Diff qw(traverse_sequences);
5 # use strict;
6 #
7 # yukiwiki.cgi - Yet another WikiWikiWeb clone.
8 #
9 # Copyright (C) 2000,2001 by Hiroshi Yuki.
10 # <hyuki@hyuki.com>
11 # http://www.hyuki.com/yukiwiki/
12 #
13 # This program is free software; you can redistribute it and/or modify
14 # it under the terms of the GNU General Public License as published by
15 # the Free Software Foundation; either version 2 of the License, or
16 # (at your option) any later version.
17 #
18 # This program is distributed in the hope that it will be useful,
19 # but WITHOUT ANY WARRANTY; without even the implied warranty of
20 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 # GNU General Public License for more details.
22 #
23 # $Id: wiki.cgi,v 1.3 2002/02/04 14:36:18 wakaba Exp $
24 ##############################
25 my $version = "1.6.6";
26 ##############################
27 # 単独テストのときには 1 にする。
28 my $testing = 0;
29 ##############################
30 # 漢字ライブラリ
31 my $jcodelib = 'jcode.pl';
32 ##############################
33 # 保存・表示の漢字コード
34 my $kanjicode = 'euc'; # 'sjis' 'euc'
35 my $charset = 'euc-jisx0213'; # 'Shift_JIS' 'EUC-JP'
36 ##############################
37 # dbmopenが使えるなら1、使えないなら0
38 my $dbmopen = 0;
39 ##############################
40 # データベース名(.pag, .dir, .dbなどは不要)
41 # $dbmopen = 1のときはデータベース名、
42 # $dbmopen = 0のときはディレクトリ名になる。
43 my $dbname = './wikidata';
44 my $diffdbname = './wikidiff';
45 ##############################
46 # 修正者の氏名(自由に変更してください)
47 my $modifier = 'suika';
48 ##############################
49 # 修正者のWebページ(自由に変更してください)
50 my $modifierlink = 'http://suika.fam.cx/';
51 ##############################
52 # このページのURL
53 my $thisurl = 'wiki';
54 ##############################
55 # 開始ページ名
56 my $toppage = 'HomePage';
57 ##############################
58 # 最終更新ページ名
59 my $whatsnew = 'RecentChanges';
60 ##############################
61 # 最終更新に掲載するページ数
62 my $maxnew = 50;
63 ##############################
64 # アイコンファイル名(カラー版)
65 my $iconfile = '';
66 ##############################
67 # アイコンファイル名(モノクロ版)
68 # my $iconfile = '';
69 ##############################
70 # ページを変更したときにtouchするファイル(''なら何もしない)
71 my $touchfile = 'touch.txt';
72 ##############################
73 # プレビュー用の背景色
74 my $preview_color = '#FFCCCC';
75 ##############################
76 # 全ページのスタイル
77 my $style = <<'EOD';
78 pre, dl, ul, ol, p, blockquote { line-height:120%; }
79 a { text-decoration: none; }
80 a:link { color: #0000FF; background-color: #FFFFFF; }
81 a:visited { color: #9900CC; background-color: #FFFFFF; }
82 a:hover { text-decoration: underline; }
83 EOD
84 ##############################
85 # テキスト入力部分の大きさ
86 my $cols = 80;
87 my $rows = 20;
88 ##############################
89 my %form = ();
90 my %database = ();
91 my %diffbase = ();
92 my $diff_text = '';
93 my @diff_added = ();
94 my @diff_deleted = ();
95 my $msgrefA;
96 my $msgrefB;
97 ##############################
98 # 編集不可ページ名一覧
99 my @uneditable = ( $whatsnew );
100 ##############################
101 # リンク用の正規表現
102 # YukiWikiのリンクは2種類ある。
103 #
104 # (1) WikiName (RecentChangesとかFrontPageのようなもの)
105 # (2) BracketName ([[結城浩]]とか[[トラブルシュート]]のようなもの)
106 #
107 # ※シフトJISの2バイト目には ']' が来うるので、
108 # 文字']'を1つ多くとるようにしている。
109 #
110 my $WikiName = '([A-Z][a-z]+([A-Z][a-z]+)+)';
111 my $BracketName = '\[\[([^>\s]+?\]?)\]\]';
112
113 # アイコン部分のタグ
114 my $IconTag = ''; #<<"EOD";
115 #<a href="http://www.hyuki.com/yukiwiki/"><img src="$iconfile"
116 # border="0" width="80" height="80" alt="[YukiWiki]" /></a>
117 #EOD
118
119 require "$jcodelib";
120
121 &init_form($kanjicode);
122
123 if ($testing) {
124 %form = (
125 # 'mycmd' => 'write',
126 'mycmd' => 'read',
127 #'mycmd' => 'search',
128 #'mycmd' => 'edit',
129 'mymsg' => <<"EOD",
130 はじめまして。
131 これからいろいろ書き込みますね。
132 LinkPageも見てください。
133 TestPageはどうでしょうか。
134 どうぞよろしく。
135 http://www.hyuki.com/
136 [[結城浩]]
137 EOD
138 'mypage' => '<結城浩>',
139 'myword' => '結',
140 # '3C8C8B8FE98D5F3E' => '',
141 # 'TestPage' => '',
142 );
143 }
144 &main;
145 exit(0);
146
147 # メイン
148 sub main {
149 &normalize_form;
150 if ($dbmopen) {
151 if (!dbmopen(%database, $dbname, 0666)) {
152 &print_error("(dbmopen) $dbname が作れません。");
153 }
154 } else {
155 if (!tie(%database, "YukiWikiDB", $dbname)) {
156 &print_error("(tie error)");
157 }
158 }
159
160 # myspecial対応
161 foreach (keys %form) {
162 if (/^myspecial_(.*)/) {
163 $form{mycmd} = $1;
164 last;
165 }
166 }
167
168 if ($form{mycmd} eq 'read') {
169 &do_read;
170 } elsif ($form{mycmd} eq 'preview') {
171 &do_preview;
172 } elsif ($form{mycmd} eq 'write') {
173 &do_write;
174 } elsif ($form{mycmd} eq 'edit') {
175 &do_edit;
176 } elsif ($form{mycmd} eq 'reedit') {
177 &do_reedit;
178 } elsif ($form{mycmd} eq 'search') {
179 &do_search;
180 } elsif ($form{mycmd} eq 'list') {
181 &do_list;
182 } elsif ($form{mycmd} eq 'diff') {
183 &do_diff;
184 } else {
185 $form{mypage} = $toppage;
186 &do_read;
187 }
188 if ($dbmopen) {
189 dbmclose(%database);
190 } else {
191 untie(%database);
192 }
193 }
194
195 # ページの表示
196 sub do_read {
197 my $page_name = $form{mypage};
198 my $percent_name = &encode_percent($page_name);
199 &print_header($page_name);
200 print qq|<h1>$IconTag<a href="$thisurl?mycmd=search;myword=$percent_name">$page_name</a></h1>\n|;
201 &print_toolbar($page_name);
202 print &convert_html(&get_page($page_name));
203 &print_footer;
204 }
205
206 # ページの編集
207 sub do_edit {
208 if (not &is_editable($form{mypage})) {
209 # 編集不可ページは表示のみ
210 &do_read;
211 return;
212 }
213 &editpage(&get_page($form{mypage}));
214 }
215
216 # ページの再編集
217 sub do_reedit {
218 if (not &is_editable($form{mypage})) {
219 # 編集不可ページは表示のみ
220 &do_read;
221 } else {
222 &editpage($form{mymsg});
223 }
224 }
225
226 sub editpage {
227 my $page_msg = shift;
228 my $page_name = $form{mypage};
229 my $digest = &calc_message_digest($page_msg);
230 &print_header($page_name);
231 print qq|<h1>$IconTag${page_name}の編集</h1>\n|;
232 &print_toolbar($page_name);
233 $page_msg = &escape($page_msg);
234 print <<"EOD";
235 <form action="$thisurl" method="post">
236 <!--<input type="hidden" name="mycmd" value="preview">-->
237 <input type="hidden" name="mypage" value="$page_name">
238 <input type="hidden" name="mydigest" value="$digest">
239 <textarea cols="$cols" rows="$rows" name="mymsg" wrap="virtual">$page_msg</textarea><br>
240 <input type="submit" name="myspecial_preview" value="確認">
241 <input type="submit" name="myspecial_write" value="確認せず変更">
242 </form>
243 <hr>
244 <h3>テキスト整形のルール</h3>
245
246 <p>通常は入力した文字がそのまま出力されますが、
247 以下のルールに従ってテキスト整形を行うことができます。</p>
248
249 <ul>
250 <li>
251 空行は段落の区切りとなります。
252
253 <li>
254 HTMLのタグは書けません。
255
256 <li>
257 ''強調''のようにシングルクォート二つではさむと、強調になります。
258
259 <li>
260 '''更に強調'''のようにシングルクォート三つではさむと、更に強調になります。
261
262 <li>
263 ----のようにマイナス4つがあると、水平線になります。
264
265 <li>
266 *を行頭に書くと大見出しになります。
267
268 <li>
269 **を行頭に書くと小見出しになります。
270
271 <li>
272 -を行頭に書くと箇条書きになります。- -- --- の3レベルがあります。
273
274 <li>
275 :を行頭に書くと用語と解説文が作れます。
276
277 <pre>
278 :用語1:いろいろ書いた解説文1
279 :用語2:いろいろ書いた解説文2
280 :用語3:いろいろ書いた解説文3
281 </pre>
282
283 <li>
284 http://www.hyuki.com/ のようなURLは自動的にリンクになります。
285
286 <li>
287 YukiWikiのように大文字小文字を混ぜた英文字列を書くと、
288 YukiWikiのページ名になります。
289
290 <li>
291 [[結城浩]]のように二重の大かっこ[[ ]]でくくった文字列を書くと、
292 YukiWikiのページ名になります。
293 大かっこの中にはスペースを含めてはいけません。
294 日本語も使えます。
295
296 <li>
297 行頭がスペースで始まっていると、
298 その段落は整形済み扱われます。
299 プログラムを書き込むときに使うと便利です。
300
301 <li>
302 &gt; を行頭に書くと、
303 引用文が書けます。
304 &gt; の数が多いとインデントが深くなります(3レベルまで)。
305
306 </ul>
307 EOD
308 &print_footer;
309 }
310
311 # ページの検索
312 sub do_search {
313 if ($form{myword}) {
314 &print_header('検索結果');
315 print qq|<h1>$IconTag$form{myword}の検索結果</h1>\n|;
316 &print_toolbar();
317 print qq|<ul>\n|;
318 my $count = 0;
319 foreach my $page_name (sort keys %database) { # sortするのは無謀かな
320 if ($database{$page_name} =~ /\Q$form{'myword'}\E/) {
321 my $encoded = &encode_percent($page_name);
322 print qq|<li><a href="$thisurl?mycmd=read;mypage=$encoded">$page_name</a>\n|;
323 $count++;
324 }
325 }
326 print qq|</ul>\n|;
327 if ($count > 0) {
328 print qq|<p><strong>$form{myword}</strong>を含むページは、上に示す<strong>$count</strong> ページです。</p>\n|;
329 } else {
330 print qq|<p><strong>$form{myword}</strong>を含むページは見つかりません。</p>\n|;
331 }
332 } else {
333 &print_header('単語検索');
334 print qq|<h1>$IconTag単語検索</h1>\n|;
335 &print_toolbar();
336 }
337 print <<"EOD";
338 <p>
339 <form action="$thisurl" method="post">
340 <input type="hidden" name="mycmd" value="search">
341 <input type="text" name="myword" size="20" value="$form{myword}">
342 <input type="submit" value="単語検索">
343 </form>
344 </p>
345 EOD
346 &print_footer;
347 }
348
349 # ページの一覧
350 sub do_list {
351 &print_header('ページ一覧');
352 print qq|<h1>$IconTag ページ一覧</h1>\n|;
353 &print_toolbar();
354 print qq|<ul>\n|;
355 foreach my $page_name (sort keys %database) { # sortするのは無謀かな
356 my $encoded = &encode_percent($page_name);
357 print qq|<li><a href="$thisurl?mycmd=read;mypage=$encoded">$page_name</a>\n|
358 }
359 print qq|</ul>\n|;
360 &print_footer;
361 }
362
363 # プレビュー
364 sub do_preview {
365 my $page_name = $form{mypage};
366 my $escapedmsg = &escape($form{mymsg});
367 &print_header($page_name);
368 print qq|<h1>$IconTag${page_name}のプレビュー</h1>\n|;
369 &print_toolbar($page_name);
370 # local $percent_name = &encode_percent($page_name);
371 print qq|<p>以下のプレビューを確認して、よければページ下部のボタンで更新してください。</p>\n|;
372 if ($form{mymsg}) {
373 print qq|<table width="100%" bgcolor="$preview_color" ><tr><td>\n|;
374 # print &convert_html($escapedmsg);
375 print &convert_html($form{mymsg});
376 print qq|</td></tr></table>\n|;
377 } else {
378 print qq|<p>(ページの内容は空です。更新するとこのページは<strong>削除</strong>されます。)</p>\n|;
379 }
380 &print_preview_buttons($page_name, $escapedmsg, $form{mydigest});
381 &print_footer;
382 }
383
384 sub print_preview_buttons {
385 my ($page_name, $escapedmsg, $digest) = @_;
386 print <<"EOD";
387 <form action="$thisurl" method="post">
388 <textarea cols="$cols" rows="$rows" name="mymsg" wrap="virtual">$escapedmsg</textarea>
389 <br />
390 <input type="hidden" name="mypage" value="$page_name">
391 <input type="hidden" name="mydigest" value="$digest">
392 <input type="submit" name="myspecial_preview" value="再度プレビュー">
393 <input type="submit" name="myspecial_write" value="ページの更新">
394 </form>
395 EOD
396 }
397
398 # 書き込む
399 sub do_write {
400 if (not &is_editable($form{mypage})) {
401 # 編集不可ページは表示のみ
402 &do_read;
403 return;
404 }
405
406 my $page_name = $form{mypage};
407
408 # digestを使って、更新の衝突チェック
409 my $original_digest = &calc_message_digest(&get_page($page_name));
410 if ($form{mydigest} ne $original_digest) {
411 &print_header($page_name);
412 print qq|<h1>$IconTag${page_name}で【更新の衝突】が起きました</h1>\n|;
413 print <<"EOD";
414 <p>あなたがこのページを編集している間に、
415 他の人が同じページを更新してしまったようです。
416 </p><p>
417 以下に、あなたの編集したテキストがありますので、
418 あなたの編集内容が失われないように、
419 いますぐ、メモ帳などにコピー&ペーストしてください。
420 </p><p>
421 コピー&ペーストが済んでから、
422 最新の内容を見て再度編集し直してください。
423 最新の内容は
424 <a target="_blank" href="$thisurl?mycmd=read;mypage=$form{mypage}">$form{mypage}</a>
425 で見ることができます。
426 </p>
427 EOD
428 # &print_toolbar($page_name);
429 &print_preview_buttons($page_name, &escape($form{mymsg}), $form{mydigest});
430 &print_footer;
431 return;
432 }
433
434 # diff生成
435 {
436 &opendiff;
437 my @msg1 = split(/\n/, &get_page($page_name));
438 my @msg2 = split(/\n/, $form{mymsg});
439 $msgrefA = \@msg1;
440 $msgrefB = \@msg2;
441 &diff_check;
442 $diffbase{$form{mypage}} = $diff_text;
443 $diff_text = '';
444 &closediff;
445 }
446
447 &print_header($page_name);
448 &set_page($page_name, $form{mymsg});
449 if ($form{mymsg}) {
450 print qq|<h1>$IconTag${page_name}を更新しました</h1>\n|;
451 &print_toolbar($page_name);
452 print &convert_html(&get_page($page_name));
453 } else {
454 print qq|<h1>$IconTag${page_name}を削除しました</h1>\n|;
455 &print_toolbar($page_name);
456 print qq|<p>${page_name}を削除しました。</p>\n|;
457 }
458 &print_footer;
459 # 更新されたのでタッチしておく。
460 if ($touchfile) {
461 open(FILE, "> $touchfile");
462 print FILE "\n";
463 close(FILE);
464 }
465 }
466
467 # ページの変更点
468 sub do_diff {
469 if (not &is_editable($form{mypage})) {
470 # 編集不可ページは表示のみ
471 &do_read;
472 return;
473 }
474 &opendiff;
475 &print_header($form{mypage} . 'の変更点');
476 print qq|<h1>$IconTag <a href="$thisurl?mycmd=read&mypage=$form{mypage}">$form{mypage}</a>の変更点</h1>\n|;
477 &print_toolbar();
478 $_ = &escape($diffbase{$form{mypage}});
479 print <<"EOD";
480 <ul>
481 <li>追加された行は<ins>青色</ins>です。
482 <li>削除された行は<del>赤色</del>です。
483 <li><a href="$thisurl?mycmd=read;mypage=$form{mypage}">$form{mypage}</a>へ行く。
484 </ul>
485 <hr />
486 EOD
487 print qq|<pre>\n|;
488 foreach (split(/\n/, $_)) {
489 if (/^\+(.*)/) {
490 print qq|<ins>$1</ins>\n|;
491 } elsif (/^\-(.*)/) {
492 print qq|<del>$1</del>\n|;
493 } elsif (/^\=(.*)/) {
494 print qq|$1\n|;
495 } else {
496 print qq|??? $_\n|;
497 }
498 }
499 print qq|</pre>\n|;
500 &print_footer;
501 &closediff;
502 }
503
504 sub opendiff {
505 if ($dbmopen) {
506 if (!dbmopen(%diffbase, $diffdbname, 0666)) {
507 &print_error("(dbmopen) $diffdbname が作れません。");
508 }
509 } else {
510 if (!tie(%diffbase, "YukiWikiDB", $diffdbname)) {
511 &print_error("(tie error)");
512 }
513 }
514 }
515
516 sub closediff {
517 if ($dbmopen) {
518 dbmclose(%diffbase);
519 } else {
520 untie(%diffbase);
521 }
522 }
523
524 # フォームからの情報を連想配列 %form に入れる
525 # &init_form('euc');
526 sub init_form {
527 my ($charcode) = @_;
528 my $query;
529 if ($ENV{REQUEST_METHOD} =~ /^post$/i) {
530 read(STDIN, $query, $ENV{CONTENT_LENGTH});
531 } else {
532 $query = $ENV{QUERY_STRING};
533 }
534 my @assocarray = split(/[&;]/, $query);
535 foreach my $assoc (@assocarray) {
536 my ($property, $value) = split(/=/, $assoc);
537 $value =~ tr/+/ /;
538 $value =~ s/%([A-Fa-f0-9][A-Fa-f0-9])/pack("C", hex($1))/eg;
539 &jcode::convert(\$value, $charcode);
540 $form{$property} = $value;
541 }
542 }
543
544 # エラーページを出力する
545 sub print_error {
546 my ($msg) = @_;
547 &print_header('Error');
548 print "<h1>$IconTag$msg</h1></body></html>";
549 exit(0);
550 }
551
552 sub escape {
553 my ($line) = shift;
554 $line =~ s|<|&lt;|g;
555 $line =~ s|>|&gt;|g;
556 $line =~ s|"|&quot;|g;
557 # $line =~ s|\&|&amp;|g;
558 return $line;
559 }
560
561 sub inline {
562 my ($line) = shift;
563 $line = &escape($line);
564 $line =~ s|'''([^']+?)'''|<strong>$1</strong>|g;
565 $line =~ s|''([^']+?)''|<em>$1</em>|g;
566 $line =~ s!
567 (
568 (?:&lt;(?:mailto|http|https|ftp|urn):[\x21-\x7E]*)&gt;
569 | (?:$WikiName) # LocalLinkLikeThis
570 | (?:$BracketName) # [[日本語リンク]]
571 )
572 !
573 &make_link($1)
574 !gex;
575 return $line;
576 }
577
578 # ページのタイトルからページの内容を得る
579 sub get_page {
580 my $page_name = shift;
581 return $database{$page_name};
582 }
583
584 # ページの内容を与える
585 # &set_page($title, $txt)
586 sub set_page {
587 # ページを更新する
588 my $title = $_[0];
589 $database{$title} = $_[1];
590 # 空ページなら削除する
591 unless ($database{$title}) {
592 delete $database{$title};
593 }
594 # RecentChangesを更新する
595 my $delim = ' - ';
596 my @pages = split(/\n/, $database{$whatsnew});
597 my $datestr = &get_current_datestr;
598 unshift(@pages, qq|-$datestr$delim$title|);
599 # 同一ページの更新は最新のもののみにし、
600 # 存在しないページはスキップする。
601 my %count;
602 my @newpages;
603 foreach my $line (@pages) {
604 my ($prefix, $title) = split(/$delim/, $line);
605 $count{$title}++;
606 if ($count{$title} == 1 and exists($database{$title})) {
607 push(@newpages, qq|$prefix - $title|);
608 }
609 }
610 # ここで本当に更新
611 $database{$whatsnew} = join("\n", splice(@newpages, 0, $maxnew));
612 }
613
614 # ページのヘッダを出力
615 sub print_header {
616 my $title = shift;
617 print <<"EOD";
618 Content-type: text/html
619
620 <html><head>
621 <title>$title</title>
622 <style type="text/css">
623 $style
624 </style>
625 </head>
626 <body>
627 EOD
628 }
629
630 # ツールバーを出力
631 sub print_toolbar {
632 my $page_name = shift;
633 my $percent_name = &encode_percent($page_name);
634 my $editlink = '';
635 if ($page_name ne '' and &is_editable($page_name)) {
636 $editlink = <<"EOD";
637 <a href="$thisurl?mycmd=edit;mypage=$percent_name">編集</a> |
638 <a href="$thisurl?mycmd=diff;mypage=$percent_name">差分</a> |
639 EOD
640 }
641 print <<"EOD";
642 <p>
643 [
644 <a href="$thisurl?mycmd=read;mypage=$toppage">トップ</a> |
645 <a href="$thisurl?mycmd=list">一覧</a> |
646 $editlink
647 <a href="$thisurl?mycmd=search">単語検索</a> |
648 <a href="$thisurl?mycmd=read;mypage=$whatsnew">最終更新</a>
649 ]
650 </p>
651 EOD
652 }
653
654 # ページのフッタを出力
655 sub print_footer {
656 print <<"EOD";
657 <address>
658 <a href="http://www.hyuki.com/yukiwiki/">YukiWiki</a> 1.6.6 Copyright (C) 2000,2001 by <a href="http://www.hyuki.com/">Hiroshi Yuki.</a>
659 + <a href="$modifierlink">$modifier</a> ${version}.
660 [<a href="/" title="このサーバーの首頁">/</a>
661 <a href="/map" title="このサーバーの案内">地図</a>
662 <a href="/search/" title="このサーバーの検索">検索</a>]
663 </address>
664 </body></html>
665 EOD
666 }
667
668 # URLやページの名前からリンクを作る
669 sub make_link {
670 my $name = shift;
671 $name =~ s/^&lt;(.*)&gt;$/$1/;
672 if ($name =~ /^(http|https|ftp).*?(\.png|\.jpeg|\.jpg)?$/) {
673 if ($2) {
674 return qq|<a href="$name"><img border="0" src="$name" /></a>|;
675 } else {
676 return qq|&lt;<a href="$name">$name</a>&gt;|;
677 }
678 } elsif ($name =~ /^mailto:(.*)/) {
679 my $address = $1;
680 return qq|&lt;<a href="$name">$address</a>&gt;|;
681 } elsif ($name =~ /^urn:[0-9A-Za-z_:-]+/) {
682 return qq|&lt;<a href="/uri-res/N2L?${name}">$name</a>&gt;|;
683 } elsif ($database{$name}) {
684 my $percent_name = &encode_percent($name);
685 return qq|<a href="$thisurl?mycmd=read;mypage=$percent_name">$name</a>|;
686 } else {
687 my $percent_name = &encode_percent($name);
688 return qq|$name<a href="$thisurl?mycmd=edit;mypage=$percent_name">?</a>|;
689 }
690 }
691
692 # %xx の形式にエンコードする
693 # これは、
694 # http://www.hyuki.com/yukiwiki/yukiwiki.cgi?mycmd=read&mypage=%3C%8C%8B%8F%E9%8D_%3E
695 # という形式のために使われる。
696 # '<結城浩>' → '%3C%8C%8B%8F%E9%8D_%3E'
697 sub encode_percent {
698 my $name = shift;
699 my $encoded = '';
700 foreach my $ch (split(//, $name)) {
701 if ($ch =~ /[A-Za-z0-9_]/) {
702 $encoded .= $ch;
703 } else {
704 $encoded .= '%' . sprintf("%02X", ord($ch));
705 }
706 }
707 return $encoded;
708 }
709
710 # テキスト本体をHTMLに変換する
711 sub convert_html {
712 my ($txt) = shift;
713 my (@txt) = split(/\n/, $txt);
714 foreach (@txt) {
715 chomp;
716 if (/^\*\*(.*)/) {
717 push(@result, splice(@saved), '<h3>' . &inline($1) . '</h3>');
718 } elsif (/^\*(.*)/) {
719 push(@result, splice(@saved), '<h2>' . &inline($1) . '</h2>');
720 } elsif (/^----/) {
721 push(@result, splice(@saved), '<hr>');
722 } elsif (/^(-{1,3})(.*)/) {
723 &back_push('ul', length($1));
724 push(@result, '<li>' . &inline($2) . '</li>');
725 } elsif (/^:([^:]+):(.*)/) {
726 &back_push('dl', 1);
727 push(@result, '<dt>' . &inline($1) . '</dt>', '<dd>' . &inline($2) . '</dd>');
728 } elsif (/^(>{1,3})(.*)/) {
729 &back_push('blockquote', length($1));
730 push(@result, &inline($2));
731 } elsif (/^\s*$/) {
732 push(@result, splice(@saved));
733 unshift(@saved, "</p>");
734 push(@result, "<p>");
735 } elsif (/^(\s+.*)$/) {
736 &back_push('pre', 1);
737 push(@result, &escape($1)); # Not &inline, but &escape
738 } else {
739 push(@result, &inline($_));
740 }
741 }
742 push(@result, splice(@saved));
743 return join("\n", @result);
744 }
745
746 # &back_push($tag, $count)
747 # $tagのタグを$levelレベルまで詰める。
748 sub back_push {
749 my ($tag, $level) = @_;
750 while (@saved > $level) {
751 push(@result, shift(@saved));
752 }
753 if ($saved[0] ne "</$tag>") {
754 push(@result, splice(@saved));
755 }
756 while (@saved < $level) {
757 unshift(@saved, "</$tag>");
758 push(@result, "<$tag>");
759 }
760 }
761
762 # 編集可能ページか?
763 sub is_editable {
764 my ($pagename) = @_;
765 foreach (@uneditable) {
766 if ($pagename eq $_) {
767 return 0;
768 }
769 }
770 if (&is_valid_name($pagename)) {
771 return 1;
772 }
773 return 0;
774 }
775
776 # Validな名前か?
777 sub is_valid_name {
778 my ($pagename) = @_;
779 if ($pagename =~ /^$WikiName$/) {
780 return 1;
781 } elsif ($pagename =~ /^$BracketName$/) {
782 return 1;
783 } else {
784 return 0;
785 }
786 }
787
788 # 現在時刻の文字列を得る
789 sub get_current_datestr {
790 my (@wdays) = ( "日", "月", "火", "水", "木", "金", "土" );
791 my ($sec, $min, $hour, $mday, $mon, $year, $wday) = localtime(time);
792 return sprintf("%4d-%02d-%02d (%s) %02d:%02d:%02d",
793 $year + 1900, $mon + 1, $mday, $wdays[$wday], $hour, $min, $sec);
794 }
795
796 # URL?SomePageや、
797 # URL?[[結城浩]]の形式だった場合、(not yet)
798 # 強制的にmycmdをreadにして$formの内容を設定する。
799 sub normalize_form {
800 foreach my $key (keys %form) {
801 if ($key =~ /^$WikiName$/) {
802 $form{mycmd} = 'read';
803 $form{mypage} = $1;
804 last;
805 } elsif ($key =~ /^$BracketName$/) {
806 $form{mycmd} = 'read';
807 $form{mypage} = $1;
808 last;
809 }
810 }
811 }
812
813 # 変換テストを行なうときのサンプル
814 sub print_sample {
815 my $txt = &convert_html(<<"EOD");
816 *大見出し1
817 **小見出し1-1
818 -項目1
819 -項目2
820 -項目3
821 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
822 段落1段落1段落1段落1段落1段落''強調''1段落1段落1段落1段落1段落1
823 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
824
825 段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
826 段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
827 段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
828 **小見出し1-2
829 :用語1:いろいろ書いた解説文1と''強調単語''
830 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
831 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
832 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
833 :用語2:いろいろ書いた解説文2
834 :用語3:いろいろ書いた解説文3
835 ----
836 *大見出し2
837 **小見出し2-1
838 &lt;http://suika.fam.cx/&gt;
839 **小見出し2-2
840
841 [[結城浩]]
842
843 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
844 段落1段落1段落1段落'''強調'''1段落1段落1段落1段落1段落1段落1段落1
845 段落1段落1段落1段落'''''強い強調'''''1段落1段落1段落1段落1段落1段落1段落1段落1
846 >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
847 >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
848 >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
849
850 レベル0レベル0レベル0レベル0レベル0レベル0
851
852 >レベル1
853 >レベル1
854 >レベル1
855 >>レベル2
856 >>レベル2
857 >>レベル2
858 >>>レベル3
859 -はろ1
860 --はろ2
861 ろろろろ2
862 ---はろ3
863 --はろ2
864 ---はろ3
865 --はろ2
866 ---はろ3
867 >>>レベル3
868 >>>レベル3
869 >>>レベル3
870 EOD
871 print $txt;
872 exit;
873 }
874
875 sub diff_check {
876 traverse_sequences(
877 $msgrefA, $msgrefB,
878 {
879 MATCH => \&df_match,
880 DISCARD_A => \&df_delete,
881 DISCARD_B => \&df_add,
882 }
883 );
884 &diff_flush;
885 }
886
887 sub diff_flush {
888 $diff_text .= join('', map { "-$_\n" } splice(@diff_deleted));
889 $diff_text .= join('', map { "+$_\n" } splice(@diff_added));
890 }
891
892 sub df_match {
893 my ($a, $b) = @_;
894 &diff_flush;
895 $diff_text .= "=$msgrefA->[$a]\n";
896 }
897
898 sub df_delete {
899 my ($a, $b) = @_;
900 push(@diff_deleted, $msgrefA->[$a]);
901 }
902
903 sub df_add {
904 my ($a, $b) = @_;
905 push(@diff_added, $msgrefB->[$b]);
906 }
907
908 sub calc_message_digest { # You have to use MD5...
909 my $text = shift;
910 my @text = split(//, $text);
911 my $len = length($text);
912 my $checksum = 0;
913 foreach (@text) {
914 $checksum += ord($_);
915 $checksum = ($checksum * 2) % 65536 + (($checksum & 32768) ? 1 : 0); # 16bit rotate
916 }
917 return "$len:$checksum";
918 }
919
920 # Definition of YukiWikiDB
921 package YukiWikiDB;
922
923 my $debug = 1;
924
925 # Constructor
926 sub new {
927 return shift->TIEHASH(@_);
928 }
929
930 # tying
931 sub TIEHASH {
932 my ($class, $dbname) = @_;
933 my $self = {
934 dir => $dbname,
935 keys => [],
936 };
937 if (not -d $self->{dir}) {
938 if (!mkdir($self->{dir}, 0777)) {
939 print "mkdir(" . $self->{dir} . ") fail\n" if ($debug);
940 return undef;
941 }
942 }
943 return bless($self, $class);
944 }
945
946 # Store
947 sub STORE {
948 my ($self, $key, $val) = @_;
949 my $file = &make_filename($self, $key);
950 if (open(FILE,"> $file")) {
951 binmode(FILE);
952 print FILE $val;
953 close(FILE);
954 return $self->{$key} = $val;
955 } else {
956 print "$file create error.";
957 }
958 }
959
960 # Fetch
961 sub FETCH {
962 my ($self, $key) = @_;
963 my $file = &make_filename($self, $key);
964 if (open(FILE, $file)) {
965 local $/;
966 $self->{$key} = <FILE>;
967 close(FILE);
968 }
969 return $self->{$key};
970 }
971
972 # Exists
973 sub EXISTS {
974 my ($self, $key) = @_;
975 my $file = &make_filename($self, $key);
976 return -e($file);
977 }
978
979 # Delete
980 sub DELETE {
981 my ($self, $key) = @_;
982 my $file = &make_filename($self, $key);
983 unlink $file;
984 return delete $self->{$key};
985 }
986
987 sub FIRSTKEY {
988 my ($self) = @_;
989 opendir(DIR, $self->{dir}) or die $self->{dir};
990 @{$self->{keys}} = grep /\.txt$/, readdir(DIR);
991 foreach my $name (@{$self->{keys}}) {
992 $name =~ s/\.txt$//;
993 $name =~ s/[0-9A-F][0-9A-F]/pack("C", hex($&))/eg;
994 }
995 return shift @{$self->{keys}};
996 }
997
998 sub NEXTKEY {
999 my ($self) = @_;
1000 return shift @{$self->{keys}};
1001 }
1002
1003 sub make_filename {
1004 my ($self, $key) = @_;
1005 my $enkey = '';
1006 foreach my $ch (split(//, $key)) {
1007 $enkey .= sprintf("%02X", ord($ch));
1008 }
1009 return $self->{dir} . "/$enkey.txt";
1010 }
1011 __END__
1012
1013 =head1 NAME
1014
1015 YukiWiki - 自由にページを追加・削除・編集できるWebページ構築CGI
1016
1017 Copyright (C) 2000,2001 by Hiroshi Yuki.
1018 結城浩 <hyuki@hyuki.com>
1019 http://www.hyuki.com/
1020 http://www.hyuki.com/yukiwiki/
1021
1022 =head1 SYNOPSIS
1023
1024 http://www.hyuki.com/yukiwiki/yukiwiki.cgi
1025
1026 =head1 DESCRIPTION
1027
1028 YukiWiki(結城ウィキィ)は参加者が自由にページを追加・削除・編集できる
1029 不思議なWebページ群を作るCGIです。
1030 Webで動作する掲示板とちょっと似ていますが、
1031 Web掲示板が単にメッセージを追加するだけなのに対して、
1032 YukiWikiは、Webページ全体を自由に変更することができます。
1033
1034 YukiWikiは、Cunningham & CunninghamのWikiWikiWebの
1035 仕様を参考にして独自に作られました。
1036
1037 YukiWikiはPerlで書かれたCGIスクリプトとして実現されていますので、
1038 Perlが動作するWebサーバならば比較的容易に設置できます。
1039
1040 あとはdbmopenが使える環境ならば設置できます(Version 1.5.0以降ならdbmopenが使えなくても設置できます)。
1041
1042
1043 YukiWikiはフリーソフトです。
1044 ご自由にお使いください。
1045
1046 =head1 設置方法
1047
1048 =head2 入手
1049
1050 YukiWikiの最新版は、
1051 http://www.hyuki.com/yukiwiki/
1052 から入手できます。
1053
1054 =head2 ファイル一覧
1055
1056 readme.txt ドキュメント
1057 yukiwiki.cgi YukiWiki本体
1058 yukiwiki.gif ロゴ(カラー版)
1059 yukimono.gif ロゴ(モノクロ版)
1060 jcode.pl 漢字コードライブラリ
1061
1062 =head2 インストール
1063
1064 =over 4
1065
1066 =item 1.
1067
1068 アーカイブを解く。
1069
1070 =item 2.
1071
1072 yukiwiki.cgiのはじめの方にある設定を確認します。
1073 通常は何もしなくてよいが、
1074 はじめは$touchfileを''にした方がよいでしょう。
1075
1076 =item 3.
1077
1078 yukiwiki.cgiとjcode.plを同じところに設置します。
1079
1080 =item 4.
1081
1082 サイズ0のyukiwiki.dbというファイルを設置します。
1083 (Perlシステムによってはyukiwiki.pag, yukiwiki.dir)
1084
1085 =item 5.
1086
1087 yukiwiki.cgiにブラウザからアクセスします。
1088
1089 =back
1090
1091 =head2 パーミッション
1092
1093 ファイル パーミッション 転送モード
1094 yukiwiki.cgi 755 ASCII
1095 yukiwiki.gif 644 BINARY
1096 yukimono.gif 644 BINARY
1097 jcode.pl 644 ASCII
1098
1099 $dbmopen = 1; にした場合:
1100 yukiwiki.db 666 BINARY
1101 (yukiwiki.pag, yukiwiki.dirの場合もあり)
1102
1103 $dbmopen = 0; にした場合: (カレントディレクトリを777にしておく)
1104 . 777 (転送しない)
1105
1106 =head1 データのバックアップ方法
1107
1108 $dbmopen = 1;の場合は、
1109 データはすべてyukiwiki.db(.dir, .pag)に入る。
1110 これをバックアップすればよい。
1111
1112 $dbmopen = 0;の場合は、
1113 yukiwikiというディレクトリができるので、
1114 これ以下をバックアップすればよい。
1115
1116 =head1 新しいページの作り方
1117
1118 =over 4
1119
1120 =item 1.
1121
1122 まず、適当なページ(例えばFrontPage)を選び、
1123 ページの下にある「編集」リンクをたどります。
1124
1125 =item 2.
1126
1127 するとテキスト入力ができる状態になるので、
1128 そこにNewPageのような単語
1129 (大文字小文字混在している英文字列)
1130 を書いて「保存」します。
1131
1132 =item 3.
1133
1134 保存すると、FrontPageのページが書き換わり、
1135 あなたが書いたNewPageという文字列の後ろに ? というリンクが表示されます。
1136 この ? はそのページがまだ存在しないことを示す印です。
1137
1138 =item 4.
1139
1140 その ? をクリックすると新しいページNewPageができますので、
1141 あなたの好きな文章をその新しいページに書いて保存します。
1142
1143 =item 5.
1144
1145 NewPageページができるとFrontPageの ? は消えて、リンクとなります。
1146
1147 =back
1148
1149 =head1 テキスト整形のルール
1150
1151 =over 4
1152
1153 =item *
1154
1155 連続した複数行はフィルされて表示されます。
1156
1157 =item *
1158
1159 空行は段落C<< <p> >>の区切りとなります。
1160
1161 =item *
1162
1163 HTMLのタグは書けません。
1164
1165 =item *
1166
1167 B<''ボールド''>のようにシングルクォート二つではさむと、
1168 ボールドC<< <b> >>になります。
1169
1170 =item *
1171
1172 B<'''イタリック'''>のようにシングルクォート三つではさむと、
1173 イタリックC<< <i> >>になります。
1174
1175 =item *
1176
1177 B<---->のようにマイナス4つがあると、
1178 水平線C<< <hr> >>になります。
1179
1180 =item *
1181
1182 行をB<*>ではじめると、
1183 大見出しC<< <h2> >>になります。
1184
1185 =item *
1186
1187 行をB<**>ではじめると、
1188 小見出しC<< <h3> >>になります。
1189
1190 =item *
1191
1192 行をマイナス-ではじめると、
1193 箇条書きC<< <ul> >>になります。
1194 マイナスの数が増えるとレベルが下がります(3レベルまで)
1195
1196 -項目1
1197 --項目1-1
1198 --項目1-2
1199 -項目2
1200 -項目3
1201 --項目3-1
1202 ---項目3-1-1
1203 ---項目3-1-2
1204 --項目3-2
1205
1206 =item *
1207
1208 コロンを使うと、
1209 用語と解説文のリストC<< <dl> >>が書けます。
1210
1211 :用語1:いろいろ書いた解説文1
1212 :用語2:いろいろ書いた解説文2
1213 :用語3:いろいろ書いた解説文3
1214
1215 =item *
1216
1217 リンク
1218
1219 =over 4
1220
1221 =item *
1222
1223 LinkToSomePageやFrontPageのように、
1224 英単語の最初の一文字を大文字にしたものが
1225 二つ以上連続したものはYukiWikiのページ名となり、
1226 それが文章中に含まれるとリンクになります。
1227
1228 =item *
1229
1230 http://www.hyuki.com/ のようなURLは自動的にリンクになります。
1231
1232 =item *
1233
1234 二重の大かっこ[[ ]]でくくった文字列も、
1235 YukiWikiのページ名になります。
1236 大かっこの中にはスペースを含めてはいけません。
1237 日本語も使えます。
1238
1239 =back
1240
1241 =item *
1242
1243 行頭がスペースやタブで始まっていると、
1244 それは整形済みの段落C<< <pre> >>として扱われます。
1245 プログラムの表示などに使うと便利です。
1246
1247
1248 =item *
1249
1250 行を > ではじめると、
1251 引用文C<< <blockquote> >>が書けます。
1252 >の数が多いとインデントが深くなります(3レベルまで)。
1253
1254 >文章
1255 >文章
1256 >>さらなる引用
1257 >文章
1258
1259 =back
1260
1261 =head1 更新履歴
1262
1263 =over 4
1264
1265 =item *
1266
1267 2001年10月20日、Version 1.6.6。
1268
1269 更新の衝突対策。
1270 元ページの簡単なチェックサムを取っておき、
1271 更新前にチェックサムを比較する。
1272 修正個所はdigestという文字列を検索すれば分かる。
1273 本来はMD5などでちゃんとやった方がいいのだけれど。
1274
1275 衝突時に表示されるメッセージなどは「極悪」さんのページを参考にした。
1276
1277 =item *
1278
1279 2001年10月17日、Version 1.6.5。
1280
1281 プレビュー画面で、更新ボタンを押したときに送信される
1282 メッセージの内容をinput要素のtype="hidden"を使って埋め込むのをやめる。
1283 代わりに、textarea要素を使う。
1284 再プレビュー用にmyspecial_を導入。でもきれいな対策ではない。
1285
1286 =item *
1287
1288 2001年8月30日、Version 1.6.4。
1289
1290 URLでダイレクトにページ名を指定しても、
1291 $WikiNameと$BracketName以外のページを作れないようにした。
1292 (is_valid_nameとis_editable参照)。
1293
1294 =item *
1295
1296 2001年8月30日、Version 1.6.3。
1297
1298 RecentChangesを編集・再編集不可とした。
1299 編集不可ページは@uneditableにページ名を入れる。
1300
1301 =item *
1302
1303 2001年2月25日、Version 1.6.1, 1.6.2。
1304
1305 差分機能のバグ修正。
1306 do_previewで'>'が扱えないバグを修正
1307 (ユーザからの指摘)。
1308
1309 =item *
1310
1311 2001年2月22日、Version 1.6.0。
1312 差分機能を実装した。
1313
1314 =item *
1315
1316 2001年2月19日、Version 1.5.4。
1317 画像ファイルへのリンクは画像にしてみた。
1318
1319 =item *
1320
1321 2001年2月19日、Version 1.5.3。
1322 RecentChangesの中に削除したページがあるのをやめた。
1323 use strict;で引っかかる部分を少し整理(完全ではない)。
1324
1325 =item *
1326
1327 2001年2月16日、Version 1.5.2。
1328 textareaに表示およびプレビューする前に < や > を &lt; や &gt; に変換した
1329 (do_preview, editpage, print_preview_buttons)。
1330
1331 =item *
1332
1333 2000年12月27日、Version 1.5.1。
1334 プレビュー画面を整理した。
1335
1336 =item *
1337
1338 2000年12月22日、Version 1.5.0。
1339 全体的にずいぶん書き直した。
1340 一覧を別途作成するようにした(do_list)。
1341 書き込む前に確認画面を出すようにした(do_preview)。
1342 テキストの書き方を編集画面に入れた(do_edit, do_reedit)。
1343 WhatsNew→RecentChanges、TopPage→FrontPageに変更した。
1344
1345 =item *
1346
1347 2000年12月20日、Version 1.1.0。
1348 tieを利用して、dbmopenが使えない場合でも動作するように修正。
1349 利用者の1人である「極悪」さんから
1350 送っていただいたコードを元にしています。
1351
1352 =item *
1353
1354 2000年9月5日、Version 1.0.2。
1355 <body color=...> → <body bgcolor=...>
1356 利用者からの指摘による。感謝。
1357
1358 =item *
1359
1360 2000年8月6日、Version 1.0.1を公開。
1361 C MAGAZINE(ソフトバンクパブリッシング)
1362 2000年10月号連載記事向け公開版。
1363 [[ ]] の最後が「望」のようにシフトJISで
1364 0x5Dになる場合の回避を行なった。
1365
1366 =item *
1367
1368 2000年8月5日、Version 1.0.0を公開。
1369
1370 =item *
1371
1372 2000年7月23日、Version 0.82を公開。
1373 編集時のリンクミス。
1374 <textarea>の属性変更。
1375
1376 =item *
1377
1378 2000年7月22日、Version 0.81を公開。
1379 ロゴを組み込む。
1380
1381 =item *
1382
1383 2000年7月21日、Version 0.80を公開。
1384 PODをCGI中に書き込む。
1385
1386 =item *
1387
1388 2000年7月19日、Version 0.70を公開。
1389 '''イタリック'''や、--、---、>>、>>>などを実装。
1390
1391 =item *
1392
1393 2000年7月18日、Version 0.60を公開。
1394 *太字*を''太字''に変更
1395
1396 =item *
1397
1398 2000年7月17日、Version 0.50を公開。
1399
1400 =item *
1401
1402 2000年7月17日、さらにいろいろ追加する。
1403
1404 =item *
1405
1406 2000年7月16日、いろいろ追加。
1407
1408 =item *
1409
1410 2000年7月15日、公開。
1411
1412 =back
1413
1414 =head1 TODO
1415
1416 - テキスト表示モード
1417 - Charsetを明示。
1418 - textarea中の閉じタグ対応
1419 - メニューの英語表記付記
1420 - プレビューのボタンで、mymsgをinputのvalueに入れているが、改行をそのままvalueにいれてはいけないのではないか。
1421 - 「再編集」の機能はブラウザの back で充分ではないか。プレビューはもっとシンプルに。
1422 - ページタイトル(Wikiname)が検索にかかるようにする。
1423 - InterWiki風の機能「URLを隠しつつ、リンクを張る」
1424
1425 =head1 作者
1426
1427 Copyright (C) 2000 by Hiroshi Yuki.
1428 結城浩 <hyuki@hyuki.com>
1429 http://www.hyuki.com/
1430 http://www.hyuki.com/yukiwiki/
1431
1432 質問、意見、バグ報告は hyuki@hyuki.com にメールしてください。
1433
1434 =head1 配布条件
1435
1436 YukiWikiは、
1437 GNU General Public Licenseにて公開します。
1438
1439 YukiWikiはフリーソフトです。
1440 ご自由にお使いください。
1441 自分好みのYukiWikiが作れるようにシンプルにしてあります。
1442
1443 =head1 謝辞
1444
1445 本家のWikiWikiを作ったCunningham & Cunningham, Inc.に
1446 感謝します。
1447
1448 YukiWikiを楽しんで使ってくださる
1449 ネット上の方々に感謝します。
1450
1451 YukiWikiのロゴをデザインしてくださった橋本礼奈さん
1452 http://city.hokkai.or.jp/~reina/
1453 に感謝します。
1454
1455 tieを使った版の元になるコードを送ってくださった
1456 「極悪」さんに感謝します。
1457
1458 =head1 参照リンク
1459
1460 =over 4
1461
1462 =item *
1463
1464 YukiWikiホームページ
1465 http://www.hyuki.com/yukiwiki/
1466
1467 =item *
1468
1469 本家のWikiWiki
1470 http://c2.com/cgi/wiki?WikiWikiWeb
1471
1472 =item *
1473
1474 本家のWikiWikiの作者(Cunningham & Cunningham, Inc.)
1475 http://c2.com/
1476
1477 =item *
1478
1479 YukiWikiのロゴデザインをしてくださった橋本礼奈さんのページ
1480 http://city.hokkai.or.jp/~reina/
1481
1482 =back
1483
1484 =cut

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24