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

Contents of /suikawiki/script/wiki.cgi

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.4 - (hide annotations) (download)
Mon Feb 4 15:03:23 2002 UTC (22 years, 3 months ago) by wakaba
Branch: MAIN
Changes since 1.3: +4 -3 lines
don't use preview

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

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24