source: trunk/wiki.cgi @ 687

Last change on this file since 687 was 687, checked in by Earle Martin, 16 years ago

new format=raw output option

  • Property svn:eol-style set to native
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 12.6 KB
Line 
1#!/usr/local/bin/perl
2
3use strict;
4use warnings;
5
6use vars qw( $VERSION );
7$VERSION = '0.50';
8
9use CGI qw/:standard/;
10use CGI::Carp qw(croak);
11use CGI::Wiki;
12use Geography::NationalGrid;
13use Geography::NationalGrid::GB;
14use OpenGuides;
15use OpenGuides::CGI;
16use OpenGuides::Config;
17use OpenGuides::RDF;
18use OpenGuides::Utils;
19use OpenGuides::Template;
20use Time::Piece;
21use URI::Escape;
22
23my $config_file = $ENV{OPENGUIDES_CONFIG_FILE} || "wiki.conf";
24my $config = OpenGuides::Config->new( file => $config_file );
25
26my $script_name = $config->script_name;
27my $script_url  = $config->script_url;
28
29my ($guide, $wiki, $formatter, $q);
30eval {
31    $guide = OpenGuides->new( config => $config );
32    $wiki = $guide->wiki;
33    $formatter = $wiki->formatter;
34
35    # Get CGI object, find out what to do.
36    $q = CGI->new;
37
38    # Note $q->param('keywords') gives you the entire param string.
39    # We need this to do URLs like foo.com/wiki.cgi?This_Page
40    my $node = $q->param('id') || $q->param('title') || $q->param('keywords') || "";
41    $node = $formatter->node_param_to_node_name( $node );
42
43    my $action       = $q->param('action')  || 'display';
44    my $commit       = $q->param('Save')    || 0;
45    my $preview      = $q->param('preview') || 0;
46    my $search_terms = $q->param('terms')   || $q->param('search') || '';
47    my $format       = $q->param('format')  || '';
48    my $oldid        = $q->param('oldid')   || '';
49
50    # Alternative method of calling search, supported by usemod.
51    $action = 'search' if $q->param("search");
52
53    if ($commit) {
54        $guide->commit_node(
55                             id      => $node,
56                             cgi_obj => $q,
57                           );
58    } elsif ($preview) {
59        preview_node($node);
60    } elsif ($action eq 'edit') {
61        edit_node($node);
62    } elsif ($action eq 'search') {
63        do_search($search_terms);
64    } elsif ($action eq 'show_backlinks') {
65        $guide->show_backlinks( id => $node );
66    } elsif ($action eq 'show_wanted_pages') {
67        show_wanted_pages();
68    } elsif ($action eq 'index') {
69        $guide->show_index(
70                            type   => $q->param("index_type") || "Full",
71                            value  => $q->param("index_value") || "",
72                            format => $format,
73                          );
74    } elsif ($action eq 'random') {
75        my @nodes = $wiki->list_all_nodes();
76        $node = $nodes[int(rand(scalar(@nodes) + 1)) + 1];
77        print $guide->redirect_to_node($node);
78        exit 0;
79    } elsif ($action eq 'find_within_distance') {
80        $guide->find_within_distance(
81                                      id => $node,
82                                      metres => $q->param("distance_in_metres")
83                                    );
84    } elsif ( $action eq 'delete'
85              and ( lc($config->enable_page_deletion) eq "y"
86                    or $config->enable_page_deletion eq "1" )
87            ) {
88        $guide->delete_node(
89                             id       => $node,
90                             version  => $q->param("version") || "",
91                             password => $q->param("password") || "",
92                           );
93    } elsif ($action eq 'userstats') {
94        show_userstats(
95                        username => $q->param("username") || "",
96                        host     => $q->param("host") || "",
97                      );
98    } elsif ($action eq 'list_all_versions') {
99        $guide->list_all_versions( id => $node );
100    } elsif ($action eq 'rss') {
101        my $feed = $q->param("feed");
102        if ( !defined $feed or $feed eq "recent_changes" ) {
103            my %args = map { $_ => ( $q->param($_) || "" ) }
104                       qw( feed items days ignore_minor_edits username
105                           category locale );
106            $guide->display_rss( %args );
107        } elsif ( $feed eq "chef_dan" ) {
108            display_node_rdf( node => $node );
109        } else {
110            croak "Unknown RSS feed type '$feed'";
111        }
112    } else { # Default is to display a node.
113        if ( $format and $format eq "rdf" ) {
114            display_node_rdf( node => $node );
115        } elsif ( $format and $format eq 'raw' ) {
116            $guide->display_node(
117                                  id       => $node,
118                                  format   => 'raw',
119                                );
120        } else {
121            my $version = $q->param("version");
122        my $other_ver = $q->param("diffversion");
123            if ( $other_ver ) {
124                $guide->display_diffs(
125                                       id            => $node,
126                                       version       => $version,
127                                       other_version => $other_ver,
128                                     );
129            } else {
130                my $redirect;
131               
132                if ((defined $q->param("redirect")) && ($q->param("redirect") == 0)) {
133                  $redirect = 0;
134                } else {
135                  $redirect = 1;               
136                }
137               
138                $guide->display_node(
139                                      id       => $node,
140                                      version  => $version,
141                                      oldid    => $oldid,
142                                      redirect => $redirect,
143                                    );
144            }
145        }
146    }
147};
148
149if ($@) {
150    my $error = $@;
151    warn $error;
152    print $q->header;
153    my $contact_email = $config->contact_email;
154    print qq(<html><head><title>ERROR</title></head><body>
155             <p>Sorry!  Something went wrong.  Please contact the
156             Wiki administrator at
157             <a href="mailto:$contact_email">$contact_email</a> and quote
158             the following error message:</p><blockquote>)
159      . $q->escapeHTML($error)
160      . qq(</blockquote><p><a href="$script_name">Return to the Wiki home page</a>
161           </body></html>);
162}
163exit 0;
164
165############################ subroutines ###################################
166
167sub show_userstats {
168    my %args = @_;
169    my ($username, $host) = @args{ qw( username host ) };
170    croak "No username or host supplied to show_userstats"
171        unless $username or $host;
172    my %criteria = ( last_n_changes => 5 );
173    $criteria{metadata_was} = $username ? { username => $username }
174                                        : { host     => $host };
175    my @nodes = $wiki->list_recent_changes( %criteria );
176    @nodes = map { {name          => $q->escapeHTML($_->{name}),
177            last_modified => $q->escapeHTML($_->{last_modified}),
178            comment       => $q->escapeHTML($_->{metadata}{comment}[0]),
179            url           => "$script_name?"
180          . $q->escape($formatter->node_name_to_node_param($_->{name})) }
181                       } @nodes;
182    my %tt_vars = ( last_five_nodes => \@nodes,
183            username        => $username,
184            username_param  => $wiki->formatter->node_name_to_node_param($username),
185                    host            => $host,
186                  );
187    process_template("userstats.tt", "", \%tt_vars);
188}
189
190sub preview_node {
191    my $node = shift;
192    my $content  = $q->param('content');
193    $content     =~ s/\r\n/\n/gs;
194    my $checksum = $q->param('checksum');
195
196    my %tt_metadata_vars = OpenGuides::Template->extract_metadata_vars(
197                                               wiki                 => $wiki,
198                           config               => $config,
199                           cgi_obj              => $q,
200                                               set_coord_field_vars => 1,
201    );
202    foreach my $var ( qw( username comment edit_type ) ) {
203        $tt_metadata_vars{$var} = $q->escapeHTML($q->param($var));
204    }
205
206    if ($wiki->verify_checksum($node, $checksum)) {
207        my %tt_vars = (
208            %tt_metadata_vars,
209            content                => $q->escapeHTML($content),
210            preview_html           => $wiki->format($content),
211            preview_above_edit_box => get_cookie( "preview_above_edit_box" ),
212            checksum               => $q->escapeHTML($checksum)
213    );
214        process_template("edit_form.tt", $node, \%tt_vars);
215    } else {
216        my %node_data = $wiki->retrieve_node($node);
217        my %tt_vars = ( checksum       => $node_data{checksum},
218                        new_content    => $content,
219                        stored_content => $node_data{content} );
220        foreach my $mdvar ( keys %tt_metadata_vars ) {
221            if ($mdvar eq "locales") {
222                $tt_vars{"stored_$mdvar"} = $node_data{metadata}{locale};
223                $tt_vars{"new_$mdvar"}    = $tt_metadata_vars{locale};
224            } elsif ($mdvar eq "categories") {
225                $tt_vars{"stored_$mdvar"} = $node_data{metadata}{category};
226                $tt_vars{"new_$mdvar"}    = $tt_metadata_vars{category};
227            } elsif ($mdvar eq "username" or $mdvar eq "comment"
228                      or $mdvar eq "edit_type" ) {
229                $tt_vars{$mdvar} = $tt_metadata_vars{$mdvar};
230            } else {
231                $tt_vars{"stored_$mdvar"} = $node_data{metadata}{$mdvar}[0];
232                $tt_vars{"new_$mdvar"}    = $tt_metadata_vars{$mdvar};
233            }
234        }
235        process_template("edit_conflict.tt", $node, \%tt_vars);
236    }
237}
238
239sub edit_node {
240    my $node = shift;
241    my %node_data = $wiki->retrieve_node($node);
242    my ($content, $checksum) = @node_data{ qw( content checksum ) };
243    my $username = get_cookie( "username" );
244    my $edit_type = get_cookie( "default_edit_type" ) eq "normal" ?
245                        "Normal edit" : "Minor tidying";
246
247    my %metadata_vars = OpenGuides::Template->extract_metadata_vars(
248                             wiki     => $wiki,
249                             config   => $config,
250                 metadata => $node_data{metadata} );
251
252    my %tt_vars = ( content         => $q->escapeHTML($content),
253                    checksum        => $q->escapeHTML($checksum),
254                    %metadata_vars,
255            username        => $username,
256                    edit_type       => $edit_type,
257                    deter_robots    => 1,
258    );
259
260    process_template("edit_form.tt", $node, \%tt_vars);
261}
262
263sub get_cookie {
264    my $pref_name = shift or return "";
265    my %cookie_data = OpenGuides::CGI->get_prefs_from_cookie(config=>$config);
266    return $cookie_data{$pref_name};
267}
268
269sub display_node_rdf {
270    my %args = @_;
271    my $rdf_writer = OpenGuides::RDF->new( wiki      => $wiki,
272                       config => $config );
273    print "Content-type: text/plain\n\n";
274    print $rdf_writer->emit_rdfxml( node => $args{node} );
275    exit 0;
276}
277
278sub process_template {
279    my ($template, $node, $vars, $conf, $omit_header) = @_;
280
281    my %output_conf = ( wiki     => $wiki,
282            config   => $config,
283                        node     => $node,
284            template => $template,
285            vars     => $vars
286    );
287    $output_conf{content_type} = "" if $omit_header; # defaults otherwise
288    print OpenGuides::Template->output( %output_conf );
289}
290
291
292sub do_search {
293    my $terms = shift;
294    my %finds = $wiki->search_nodes($terms);
295#    my @sorted = sort { $finds{$a} cmp $finds{$b} } keys %finds;
296    my @sorted = sort keys %finds;
297    my @results = map {
298        { url   => $q->escape($formatter->node_name_to_node_param($_)),
299      title => $q->escapeHTML($_)
300        }             } @sorted;
301    my %tt_vars = ( results      => \@results,
302                    num_results  => scalar @results,
303                    not_editable => 1,
304                    search_terms => $q->escapeHTML($terms) );
305    process_template("search_results.tt", "", \%tt_vars);
306}
307
308sub show_wanted_pages {
309    my @dangling = $wiki->list_dangling_links;
310    my @wanted;
311    my %backlinks_count;
312    foreach my $node_name (@dangling) {
313        $backlinks_count{$node_name} = scalar($wiki->list_backlinks( node => $node_name ));
314    }
315    foreach my $node_name (sort { $backlinks_count{$b} <=> $backlinks_count{$a} } @dangling) {
316        my $node_param =
317         uri_escape($formatter->node_name_to_node_param($node_name));
318        push @wanted, {
319            name          => $q->escapeHTML($node_name),
320            edit_link     => $script_url . uri_escape($script_name)
321                           . "?action=edit;id=$node_param",
322            backlink_link => $script_url . uri_escape($script_name)
323                    . "?action=show_backlinks;id=$node_param",
324            backlinks_count => $backlinks_count{$node_name}
325        };
326    }
327    process_template( "wanted_pages.tt",
328                      "",
329                      { not_editable  => 1,
330                        not_deletable => 1,
331                        deter_robots  => 1,
332                        wanted        => \@wanted } );
333}
334
Note: See TracBrowser for help on using the repository browser.