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

Contents of /suikawiki/script/wiki.cgi

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.6 - (show annotations) (download)
Mon Feb 4 15:23:58 2002 UTC (22 years, 3 months ago) by wakaba
Branch: MAIN
Changes since 1.5: +1 -1 lines
bug fix of <urn> format.

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; # Direct http://...
569 |
570 ($WikiName) # LocalLinkLikeThis
571 |
572 ($BracketName) # [[日本語リンク]]
573 )
574 !
575 &make_link($1)
576 !gex;
577 return $line;
578 }
579
580 # ページのタイトルからページの内容を得る
581 sub get_page {
582 my $page_name = shift;
583 return $database{$page_name};
584 }
585
586 # ページの内容を与える
587 # &set_page($title, $txt)
588 sub set_page {
589 # ページを更新する
590 my $title = $_[0];
591 $database{$title} = $_[1];
592 # 空ページなら削除する
593 unless ($database{$title}) {
594 delete $database{$title};
595 }
596 # RecentChangesを更新する
597 my $delim = ' - ';
598 my @pages = split(/\n/, $database{$whatsnew});
599 my $datestr = &get_current_datestr;
600 unshift(@pages, qq|-$datestr$delim$title|);
601 # 同一ページの更新は最新のもののみにし、
602 # 存在しないページはスキップする。
603 my %count;
604 my @newpages;
605 foreach my $line (@pages) {
606 my ($prefix, $title) = split(/$delim/, $line);
607 $count{$title}++;
608 if ($count{$title} == 1 and exists($database{$title})) {
609 push(@newpages, qq|$prefix - $title|);
610 }
611 }
612 # ここで本当に更新
613 $database{$whatsnew} = join("\n", splice(@newpages, 0, $maxnew));
614 }
615
616 # ページのヘッダを出力
617 sub print_header {
618 my $title = shift;
619 print <<"EOD";
620 Content-type: text/html
621
622 <html><head>
623 <title>$title</title>
624 <style type="text/css">
625 $style
626 </style>
627 </head>
628 <body>
629 EOD
630 }
631
632 # ツールバーを出力
633 sub print_toolbar {
634 my $page_name = shift;
635 my $percent_name = &encode_percent($page_name);
636 my $editlink = '';
637 if ($page_name ne '' and &is_editable($page_name)) {
638 $editlink = <<"EOD";
639 <a href="$thisurl?mycmd=edit;mypage=$percent_name">編集</a> |
640 <a href="$thisurl?mycmd=diff;mypage=$percent_name">差分</a> |
641 EOD
642 }
643 print <<"EOD";
644 <p>
645 [
646 <a href="$thisurl?mycmd=read;mypage=$toppage">トップ</a> |
647 <a href="$thisurl?mycmd=list">一覧</a> |
648 $editlink
649 <a href="$thisurl?mycmd=search">単語検索</a> |
650 <a href="$thisurl?mycmd=read;mypage=$whatsnew">最終更新</a>
651 ]
652 </p>
653 EOD
654 }
655
656 # ページのフッタを出力
657 sub print_footer {
658 print <<"EOD";
659 <address>
660 <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>
661 + <a href="$modifierlink">$modifier</a> ${version}.
662 [<a href="/" title="このサーバーの首頁">/</a>
663 <a href="/map" title="このサーバーの案内">地図</a>
664 <a href="/search/" title="このサーバーの検索">検索</a>]
665 </address>
666 </body></html>
667 EOD
668 }
669
670 # URLやページの名前からリンクを作る
671 sub make_link {
672 my $name = shift;
673 if ($name =~ /^(http|https|ftp).*?(\.png|\.jpeg|\.jpg)?$/) {
674 if ($2) {
675 return qq|<a href="$name"><img border="0" src="$name" /></a>|;
676 } else {
677 return qq|&lt;<a href="$name">$name</a>&gt;|;
678 }
679 } elsif ($name =~ /^mailto:(.*)/) {
680 my $address = $1;
681 return qq|&lt;<a href="$name">$address</a>&gt;|;
682 } elsif ($name =~ /^urn:[0-9A-Za-z_:-]+/) {
683 return qq|&lt;<a href="/uri-res/N2L?${name}">$name</a>&gt;|;
684 } elsif ($database{$name}) {
685 my $percent_name = &encode_percent($name);
686 return qq|<a href="$thisurl?mycmd=read;mypage=$percent_name">$name</a>|;
687 } else {
688 my $percent_name = &encode_percent($name);
689 return qq|$name<a href="$thisurl?mycmd=edit;mypage=$percent_name">?</a>|;
690 }
691 }
692
693 # %xx の形式にエンコードする
694 # これは、
695 # http://www.hyuki.com/yukiwiki/yukiwiki.cgi?mycmd=read&mypage=%3C%8C%8B%8F%E9%8D_%3E
696 # という形式のために使われる。
697 # '<結城浩>' → '%3C%8C%8B%8F%E9%8D_%3E'
698 sub encode_percent {
699 my $name = shift;
700 my $encoded = '';
701 foreach my $ch (split(//, $name)) {
702 if ($ch =~ /[A-Za-z0-9_]/) {
703 $encoded .= $ch;
704 } else {
705 $encoded .= '%' . sprintf("%02X", ord($ch));
706 }
707 }
708 return $encoded;
709 }
710
711 # テキスト本体をHTMLに変換する
712 sub convert_html {
713 my ($txt) = shift;
714 my (@txt) = split(/\n/, $txt);
715 foreach (@txt) {
716 chomp;
717 if (/^\*\*(.*)/) {
718 push(@result, splice(@saved), '<h3>' . &inline($1) . '</h3>');
719 } elsif (/^\*(.*)/) {
720 push(@result, splice(@saved), '<h2>' . &inline($1) . '</h2>');
721 } elsif (/^----/) {
722 push(@result, splice(@saved), '<hr>');
723 } elsif (/^(-{1,3})(.*)/) {
724 &back_push('ul', length($1));
725 push(@result, '<li>' . &inline($2) . '</li>');
726 } elsif (/^:([^:]+):(.*)/) {
727 &back_push('dl', 1);
728 push(@result, '<dt>' . &inline($1) . '</dt>', '<dd>' . &inline($2) . '</dd>');
729 } elsif (/^(>{1,3})(.*)/) {
730 &back_push('blockquote', length($1));
731 push(@result, &inline($2));
732 } elsif (/^\s*$/) {
733 push(@result, splice(@saved));
734 unshift(@saved, "</p>");
735 push(@result, "<p>");
736 } elsif (/^(\s+.*)$/) {
737 &back_push('pre', 1);
738 push(@result, &escape($1)); # Not &inline, but &escape
739 } else {
740 push(@result, &inline($_));
741 }
742 }
743 push(@result, splice(@saved));
744 return join("\n", @result);
745 }
746
747 # &back_push($tag, $count)
748 # $tagのタグを$levelレベルまで詰める。
749 sub back_push {
750 my ($tag, $level) = @_;
751 while (@saved > $level) {
752 push(@result, shift(@saved));
753 }
754 if ($saved[0] ne "</$tag>") {
755 push(@result, splice(@saved));
756 }
757 while (@saved < $level) {
758 unshift(@saved, "</$tag>");
759 push(@result, "<$tag>");
760 }
761 }
762
763 # 編集可能ページか?
764 sub is_editable {
765 my ($pagename) = @_;
766 foreach (@uneditable) {
767 if ($pagename eq $_) {
768 return 0;
769 }
770 }
771 if (&is_valid_name($pagename)) {
772 return 1;
773 }
774 return 0;
775 }
776
777 # Validな名前か?
778 sub is_valid_name {
779 my ($pagename) = @_;
780 if ($pagename =~ /^$WikiName$/) {
781 return 1;
782 } elsif ($pagename =~ /^$BracketName$/) {
783 return 1;
784 } else {
785 return 0;
786 }
787 }
788
789 # 現在時刻の文字列を得る
790 sub get_current_datestr {
791 my (@wdays) = ( "日", "月", "火", "水", "木", "金", "土" );
792 my ($sec, $min, $hour, $mday, $mon, $year, $wday) = localtime(time);
793 return sprintf("%4d-%02d-%02d (%s) %02d:%02d:%02d",
794 $year + 1900, $mon + 1, $mday, $wdays[$wday], $hour, $min, $sec);
795 }
796
797 # URL?SomePageや、
798 # URL?[[結城浩]]の形式だった場合、(not yet)
799 # 強制的にmycmdをreadにして$formの内容を設定する。
800 sub normalize_form {
801 foreach my $key (keys %form) {
802 if ($key =~ /^$WikiName$/) {
803 $form{mycmd} = 'read';
804 $form{mypage} = $1;
805 last;
806 } elsif ($key =~ /^$BracketName$/) {
807 $form{mycmd} = 'read';
808 $form{mypage} = $1;
809 last;
810 }
811 }
812 }
813
814 # 変換テストを行なうときのサンプル
815 sub print_sample {
816 my $txt = &convert_html(<<"EOD");
817 *大見出し1
818 **小見出し1-1
819 -項目1
820 -項目2
821 -項目3
822 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
823 段落1段落1段落1段落1段落1段落''強調''1段落1段落1段落1段落1段落1
824 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
825
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 段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
829 **小見出し1-2
830 :用語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 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
834 :用語2:いろいろ書いた解説文2
835 :用語3:いろいろ書いた解説文3
836 ----
837 *大見出し2
838 **小見出し2-1
839 &lt;http://suika.fam.cx/&gt;
840 **小見出し2-2
841
842 [[結城浩]]
843
844 段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
845 段落1段落1段落1段落'''強調'''1段落1段落1段落1段落1段落1段落1段落1
846 段落1段落1段落1段落'''''強い強調'''''1段落1段落1段落1段落1段落1段落1段落1段落1
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 >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
850
851 レベル0レベル0レベル0レベル0レベル0レベル0
852
853 >レベル1
854 >レベル1
855 >レベル1
856 >>レベル2
857 >>レベル2
858 >>レベル2
859 >>>レベル3
860 -はろ1
861 --はろ2
862 ろろろろ2
863 ---はろ3
864 --はろ2
865 ---はろ3
866 --はろ2
867 ---はろ3
868 >>>レベル3
869 >>>レベル3
870 >>>レベル3
871 EOD
872 print $txt;
873 exit;
874 }
875
876 sub diff_check {
877 traverse_sequences(
878 $msgrefA, $msgrefB,
879 {
880 MATCH => \&df_match,
881 DISCARD_A => \&df_delete,
882 DISCARD_B => \&df_add,
883 }
884 );
885 &diff_flush;
886 }
887
888 sub diff_flush {
889 $diff_text .= join('', map { "-$_\n" } splice(@diff_deleted));
890 $diff_text .= join('', map { "+$_\n" } splice(@diff_added));
891 }
892
893 sub df_match {
894 my ($a, $b) = @_;
895 &diff_flush;
896 $diff_text .= "=$msgrefA->[$a]\n";
897 }
898
899 sub df_delete {
900 my ($a, $b) = @_;
901 push(@diff_deleted, $msgrefA->[$a]);
902 }
903
904 sub df_add {
905 my ($a, $b) = @_;
906 push(@diff_added, $msgrefB->[$b]);
907 }
908
909 sub calc_message_digest { # You have to use MD5...
910 my $text = shift;
911 my @text = split(//, $text);
912 my $len = length($text);
913 my $checksum = 0;
914 foreach (@text) {
915 $checksum += ord($_);
916 $checksum = ($checksum * 2) % 65536 + (($checksum & 32768) ? 1 : 0); # 16bit rotate
917 }
918 return "$len:$checksum";
919 }
920
921 # Definition of YukiWikiDB
922 package YukiWikiDB;
923
924 my $debug = 1;
925
926 # Constructor
927 sub new {
928 return shift->TIEHASH(@_);
929 }
930
931 # tying
932 sub TIEHASH {
933 my ($class, $dbname) = @_;
934 my $self = {
935 dir => $dbname,
936 keys => [],
937 };
938 if (not -d $self->{dir}) {
939 if (!mkdir($self->{dir}, 0777)) {
940 print "mkdir(" . $self->{dir} . ") fail\n" if ($debug);
941 return undef;
942 }
943 }
944 return bless($self, $class);
945 }
946
947 # Store
948 sub STORE {
949 my ($self, $key, $val) = @_;
950 my $file = &make_filename($self, $key);
951 if (open(FILE,"> $file")) {
952 binmode(FILE);
953 print FILE $val;
954 close(FILE);
955 return $self->{$key} = $val;
956 } else {
957 print "$file create error.";
958 }
959 }
960
961 # Fetch
962 sub FETCH {
963 my ($self, $key) = @_;
964 my $file = &make_filename($self, $key);
965 if (open(FILE, $file)) {
966 local $/;
967 $self->{$key} = <FILE>;
968 close(FILE);
969 }
970 return $self->{$key};
971 }
972
973 # Exists
974 sub EXISTS {
975 my ($self, $key) = @_;
976 my $file = &make_filename($self, $key);
977 return -e($file);
978 }
979
980 # Delete
981 sub DELETE {
982 my ($self, $key) = @_;
983 my $file = &make_filename($self, $key);
984 unlink $file;
985 return delete $self->{$key};
986 }
987
988 sub FIRSTKEY {
989 my ($self) = @_;
990 opendir(DIR, $self->{dir}) or die $self->{dir};
991 @{$self->{keys}} = grep /\.txt$/, readdir(DIR);
992 foreach my $name (@{$self->{keys}}) {
993 $name =~ s/\.txt$//;
994 $name =~ s/[0-9A-F][0-9A-F]/pack("C", hex($&))/eg;
995 }
996 return shift @{$self->{keys}};
997 }
998
999 sub NEXTKEY {
1000 my ($self) = @_;
1001 return shift @{$self->{keys}};
1002 }
1003
1004 sub make_filename {
1005 my ($self, $key) = @_;
1006 my $enkey = '';
1007 foreach my $ch (split(//, $key)) {
1008 $enkey .= sprintf("%02X", ord($ch));
1009 }
1010 return $self->{dir} . "/$enkey.txt";
1011 }
1012 __END__
1013
1014 =head1 NAME
1015
1016 YukiWiki - 自由にページを追加・削除・編集できるWebページ構築CGI
1017
1018 Copyright (C) 2000,2001 by Hiroshi Yuki.
1019 結城浩 <hyuki@hyuki.com>
1020 http://www.hyuki.com/
1021 http://www.hyuki.com/yukiwiki/
1022
1023 =head1 SYNOPSIS
1024
1025 http://www.hyuki.com/yukiwiki/yukiwiki.cgi
1026
1027 =head1 DESCRIPTION
1028
1029 YukiWiki(結城ウィキィ)は参加者が自由にページを追加・削除・編集できる
1030 不思議なWebページ群を作るCGIです。
1031 Webで動作する掲示板とちょっと似ていますが、
1032 Web掲示板が単にメッセージを追加するだけなのに対して、
1033 YukiWikiは、Webページ全体を自由に変更することができます。
1034
1035 YukiWikiは、Cunningham & CunninghamのWikiWikiWebの
1036 仕様を参考にして独自に作られました。
1037
1038 YukiWikiはPerlで書かれたCGIスクリプトとして実現されていますので、
1039 Perlが動作するWebサーバならば比較的容易に設置できます。
1040
1041 あとはdbmopenが使える環境ならば設置できます(Version 1.5.0以降ならdbmopenが使えなくても設置できます)。
1042
1043
1044 YukiWikiはフリーソフトです。
1045 ご自由にお使いください。
1046
1047 =head1 設置方法
1048
1049 =head2 入手
1050
1051 YukiWikiの最新版は、
1052 http://www.hyuki.com/yukiwiki/
1053 から入手できます。
1054
1055 =head2 ファイル一覧
1056
1057 readme.txt ドキュメント
1058 yukiwiki.cgi YukiWiki本体
1059 yukiwiki.gif ロゴ(カラー版)
1060 yukimono.gif ロゴ(モノクロ版)
1061 jcode.pl 漢字コードライブラリ
1062
1063 =head2 インストール
1064
1065 =over 4
1066
1067 =item 1.
1068
1069 アーカイブを解く。
1070
1071 =item 2.
1072
1073 yukiwiki.cgiのはじめの方にある設定を確認します。
1074 通常は何もしなくてよいが、
1075 はじめは$touchfileを''にした方がよいでしょう。
1076
1077 =item 3.
1078
1079 yukiwiki.cgiとjcode.plを同じところに設置します。
1080
1081 =item 4.
1082
1083 サイズ0のyukiwiki.dbというファイルを設置します。
1084 (Perlシステムによってはyukiwiki.pag, yukiwiki.dir)
1085
1086 =item 5.
1087
1088 yukiwiki.cgiにブラウザからアクセスします。
1089
1090 =back
1091
1092 =head2 パーミッション
1093
1094 ファイル パーミッション 転送モード
1095 yukiwiki.cgi 755 ASCII
1096 yukiwiki.gif 644 BINARY
1097 yukimono.gif 644 BINARY
1098 jcode.pl 644 ASCII
1099
1100 $dbmopen = 1; にした場合:
1101 yukiwiki.db 666 BINARY
1102 (yukiwiki.pag, yukiwiki.dirの場合もあり)
1103
1104 $dbmopen = 0; にした場合: (カレントディレクトリを777にしておく)
1105 . 777 (転送しない)
1106
1107 =head1 データのバックアップ方法
1108
1109 $dbmopen = 1;の場合は、
1110 データはすべてyukiwiki.db(.dir, .pag)に入る。
1111 これをバックアップすればよい。
1112
1113 $dbmopen = 0;の場合は、
1114 yukiwikiというディレクトリができるので、
1115 これ以下をバックアップすればよい。
1116
1117 =head1 新しいページの作り方
1118
1119 =over 4
1120
1121 =item 1.
1122
1123 まず、適当なページ(例えばFrontPage)を選び、
1124 ページの下にある「編集」リンクをたどります。
1125
1126 =item 2.
1127
1128 するとテキスト入力ができる状態になるので、
1129 そこにNewPageのような単語
1130 (大文字小文字混在している英文字列)
1131 を書いて「保存」します。
1132
1133 =item 3.
1134
1135 保存すると、FrontPageのページが書き換わり、
1136 あなたが書いたNewPageという文字列の後ろに ? というリンクが表示されます。
1137 この ? はそのページがまだ存在しないことを示す印です。
1138
1139 =item 4.
1140
1141 その ? をクリックすると新しいページNewPageができますので、
1142 あなたの好きな文章をその新しいページに書いて保存します。
1143
1144 =item 5.
1145
1146 NewPageページができるとFrontPageの ? は消えて、リンクとなります。
1147
1148 =back
1149
1150 =head1 テキスト整形のルール
1151
1152 =over 4
1153
1154 =item *
1155
1156 連続した複数行はフィルされて表示されます。
1157
1158 =item *
1159
1160 空行は段落C<< <p> >>の区切りとなります。
1161
1162 =item *
1163
1164 HTMLのタグは書けません。
1165
1166 =item *
1167
1168 B<''ボールド''>のようにシングルクォート二つではさむと、
1169 ボールドC<< <b> >>になります。
1170
1171 =item *
1172
1173 B<'''イタリック'''>のようにシングルクォート三つではさむと、
1174 イタリックC<< <i> >>になります。
1175
1176 =item *
1177
1178 B<---->のようにマイナス4つがあると、
1179 水平線C<< <hr> >>になります。
1180
1181 =item *
1182
1183 行をB<*>ではじめると、
1184 大見出しC<< <h2> >>になります。
1185
1186 =item *
1187
1188 行をB<**>ではじめると、
1189 小見出しC<< <h3> >>になります。
1190
1191 =item *
1192
1193 行をマイナス-ではじめると、
1194 箇条書きC<< <ul> >>になります。
1195 マイナスの数が増えるとレベルが下がります(3レベルまで)
1196
1197 -項目1
1198 --項目1-1
1199 --項目1-2
1200 -項目2
1201 -項目3
1202 --項目3-1
1203 ---項目3-1-1
1204 ---項目3-1-2
1205 --項目3-2
1206
1207 =item *
1208
1209 コロンを使うと、
1210 用語と解説文のリストC<< <dl> >>が書けます。
1211
1212 :用語1:いろいろ書いた解説文1
1213 :用語2:いろいろ書いた解説文2
1214 :用語3:いろいろ書いた解説文3
1215
1216 =item *
1217
1218 リンク
1219
1220 =over 4
1221
1222 =item *
1223
1224 LinkToSomePageやFrontPageのように、
1225 英単語の最初の一文字を大文字にしたものが
1226 二つ以上連続したものはYukiWikiのページ名となり、
1227 それが文章中に含まれるとリンクになります。
1228
1229 =item *
1230
1231 http://www.hyuki.com/ のようなURLは自動的にリンクになります。
1232
1233 =item *
1234
1235 二重の大かっこ[[ ]]でくくった文字列も、
1236 YukiWikiのページ名になります。
1237 大かっこの中にはスペースを含めてはいけません。
1238 日本語も使えます。
1239
1240 =back
1241
1242 =item *
1243
1244 行頭がスペースやタブで始まっていると、
1245 それは整形済みの段落C<< <pre> >>として扱われます。
1246 プログラムの表示などに使うと便利です。
1247
1248
1249 =item *
1250
1251 行を > ではじめると、
1252 引用文C<< <blockquote> >>が書けます。
1253 >の数が多いとインデントが深くなります(3レベルまで)。
1254
1255 >文章
1256 >文章
1257 >>さらなる引用
1258 >文章
1259
1260 =back
1261
1262 =head1 更新履歴
1263
1264 =over 4
1265
1266 =item *
1267
1268 2001年10月20日、Version 1.6.6。
1269
1270 更新の衝突対策。
1271 元ページの簡単なチェックサムを取っておき、
1272 更新前にチェックサムを比較する。
1273 修正個所はdigestという文字列を検索すれば分かる。
1274 本来はMD5などでちゃんとやった方がいいのだけれど。
1275
1276 衝突時に表示されるメッセージなどは「極悪」さんのページを参考にした。
1277
1278 =item *
1279
1280 2001年10月17日、Version 1.6.5。
1281
1282 プレビュー画面で、更新ボタンを押したときに送信される
1283 メッセージの内容をinput要素のtype="hidden"を使って埋め込むのをやめる。
1284 代わりに、textarea要素を使う。
1285 再プレビュー用にmyspecial_を導入。でもきれいな対策ではない。
1286
1287 =item *
1288
1289 2001年8月30日、Version 1.6.4。
1290
1291 URLでダイレクトにページ名を指定しても、
1292 $WikiNameと$BracketName以外のページを作れないようにした。
1293 (is_valid_nameとis_editable参照)。
1294
1295 =item *
1296
1297 2001年8月30日、Version 1.6.3。
1298
1299 RecentChangesを編集・再編集不可とした。
1300 編集不可ページは@uneditableにページ名を入れる。
1301
1302 =item *
1303
1304 2001年2月25日、Version 1.6.1, 1.6.2。
1305
1306 差分機能のバグ修正。
1307 do_previewで'>'が扱えないバグを修正
1308 (ユーザからの指摘)。
1309
1310 =item *
1311
1312 2001年2月22日、Version 1.6.0。
1313 差分機能を実装した。
1314
1315 =item *
1316
1317 2001年2月19日、Version 1.5.4。
1318 画像ファイルへのリンクは画像にしてみた。
1319
1320 =item *
1321
1322 2001年2月19日、Version 1.5.3。
1323 RecentChangesの中に削除したページがあるのをやめた。
1324 use strict;で引っかかる部分を少し整理(完全ではない)。
1325
1326 =item *
1327
1328 2001年2月16日、Version 1.5.2。
1329 textareaに表示およびプレビューする前に < や > を &lt; や &gt; に変換した
1330 (do_preview, editpage, print_preview_buttons)。
1331
1332 =item *
1333
1334 2000年12月27日、Version 1.5.1。
1335 プレビュー画面を整理した。
1336
1337 =item *
1338
1339 2000年12月22日、Version 1.5.0。
1340 全体的にずいぶん書き直した。
1341 一覧を別途作成するようにした(do_list)。
1342 書き込む前に確認画面を出すようにした(do_preview)。
1343 テキストの書き方を編集画面に入れた(do_edit, do_reedit)。
1344 WhatsNew→RecentChanges、TopPage→FrontPageに変更した。
1345
1346 =item *
1347
1348 2000年12月20日、Version 1.1.0。
1349 tieを利用して、dbmopenが使えない場合でも動作するように修正。
1350 利用者の1人である「極悪」さんから
1351 送っていただいたコードを元にしています。
1352
1353 =item *
1354
1355 2000年9月5日、Version 1.0.2。
1356 <body color=...> → <body bgcolor=...>
1357 利用者からの指摘による。感謝。
1358
1359 =item *
1360
1361 2000年8月6日、Version 1.0.1を公開。
1362 C MAGAZINE(ソフトバンクパブリッシング)
1363 2000年10月号連載記事向け公開版。
1364 [[ ]] の最後が「望」のようにシフトJISで
1365 0x5Dになる場合の回避を行なった。
1366
1367 =item *
1368
1369 2000年8月5日、Version 1.0.0を公開。
1370
1371 =item *
1372
1373 2000年7月23日、Version 0.82を公開。
1374 編集時のリンクミス。
1375 <textarea>の属性変更。
1376
1377 =item *
1378
1379 2000年7月22日、Version 0.81を公開。
1380 ロゴを組み込む。
1381
1382 =item *
1383
1384 2000年7月21日、Version 0.80を公開。
1385 PODをCGI中に書き込む。
1386
1387 =item *
1388
1389 2000年7月19日、Version 0.70を公開。
1390 '''イタリック'''や、--、---、>>、>>>などを実装。
1391
1392 =item *
1393
1394 2000年7月18日、Version 0.60を公開。
1395 *太字*を''太字''に変更
1396
1397 =item *
1398
1399 2000年7月17日、Version 0.50を公開。
1400
1401 =item *
1402
1403 2000年7月17日、さらにいろいろ追加する。
1404
1405 =item *
1406
1407 2000年7月16日、いろいろ追加。
1408
1409 =item *
1410
1411 2000年7月15日、公開。
1412
1413 =back
1414
1415 =head1 TODO
1416
1417 - テキスト表示モード
1418 - Charsetを明示。
1419 - textarea中の閉じタグ対応
1420 - メニューの英語表記付記
1421 - プレビューのボタンで、mymsgをinputのvalueに入れているが、改行をそのままvalueにいれてはいけないのではないか。
1422 - 「再編集」の機能はブラウザの back で充分ではないか。プレビューはもっとシンプルに。
1423 - ページタイトル(Wikiname)が検索にかかるようにする。
1424 - InterWiki風の機能「URLを隠しつつ、リンクを張る」
1425
1426 =head1 作者
1427
1428 Copyright (C) 2000 by Hiroshi Yuki.
1429 結城浩 <hyuki@hyuki.com>
1430 http://www.hyuki.com/
1431 http://www.hyuki.com/yukiwiki/
1432
1433 質問、意見、バグ報告は hyuki@hyuki.com にメールしてください。
1434
1435 =head1 配布条件
1436
1437 YukiWikiは、
1438 GNU General Public Licenseにて公開します。
1439
1440 YukiWikiはフリーソフトです。
1441 ご自由にお使いください。
1442 自分好みのYukiWikiが作れるようにシンプルにしてあります。
1443
1444 =head1 謝辞
1445
1446 本家のWikiWikiを作ったCunningham & Cunningham, Inc.に
1447 感謝します。
1448
1449 YukiWikiを楽しんで使ってくださる
1450 ネット上の方々に感謝します。
1451
1452 YukiWikiのロゴをデザインしてくださった橋本礼奈さん
1453 http://city.hokkai.or.jp/~reina/
1454 に感謝します。
1455
1456 tieを使った版の元になるコードを送ってくださった
1457 「極悪」さんに感謝します。
1458
1459 =head1 参照リンク
1460
1461 =over 4
1462
1463 =item *
1464
1465 YukiWikiホームページ
1466 http://www.hyuki.com/yukiwiki/
1467
1468 =item *
1469
1470 本家のWikiWiki
1471 http://c2.com/cgi/wiki?WikiWikiWeb
1472
1473 =item *
1474
1475 本家のWikiWikiの作者(Cunningham & Cunningham, Inc.)
1476 http://c2.com/
1477
1478 =item *
1479
1480 YukiWikiのロゴデザインをしてくださった橋本礼奈さんのページ
1481 http://city.hokkai.or.jp/~reina/
1482
1483 =back
1484
1485 =cut

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24