]> git.deb.at Git - deb/packages.git/blob - cgi-bin/dispatcher.pl
Fix some issues in contents search:
[deb/packages.git] / cgi-bin / dispatcher.pl
1 #!/usr/bin/perl -T
2 # $Id: search_packages.pl 91 2006-02-10 22:18:31Z jeroen $
3 # dispatcher.pl -- CGI interface for packages.debian.org
4 #
5 # Copyright (C) 2004-2006 Frank Lichtenheld
6 #
7 # use is allowed under the terms of the GNU Public License (GPL)                              
8 # see http://www.fsf.org/copyleft/gpl.html for a copy of the license
9
10 use strict;
11 use warnings;
12
13 use lib '../lib';
14 use CGI;
15 use POSIX;
16 use URI::Escape;
17 use HTML::Entities;
18 use DB_File;
19 use Benchmark ':hireswallclock';
20 use I18N::AcceptLanguage;
21 use Locale::gettext;
22
23 use Deb::Versions;
24 use Packages::Config qw( $DBDIR $ROOT @SUITES @SECTIONS @ARCHIVES @ARCHITECTURES @LANGUAGES $LOCALES );
25 use Packages::CGI;
26 use Packages::DB;
27 use Packages::Search qw( :all );
28 use Packages::HTML ();
29 use Packages::Sections;
30 use Packages::I18N::Locale;
31
32 use Packages::DoSearch;
33 use Packages::DoSearchContents;
34 use Packages::DoShow;
35 use Packages::DoIndex;
36 use Packages::DoNewPkg;
37 use Packages::DoDownload;
38 use Packages::DoFilelist;
39
40 &Packages::CGI::reset;
41
42 # clean up env
43 $ENV{PATH} = "/bin:/usr/bin";
44 delete $ENV{'LANGUAGE'};
45 delete $ENV{'LANG'};
46 delete $ENV{'LC_ALL'};
47 delete $ENV{'LC_MESSAGES'};
48
49 # Read in all the variables set by the form
50 my $input;
51 if ($ARGV[0] && ($ARGV[0] eq 'php')) {
52         $input = new CGI(\*STDIN);
53 } else {
54         $input = new CGI;
55 }
56
57 my $pet0 = new Benchmark;
58 my $tet0 = new Benchmark;
59 my $debug = DEBUG && $input->param("debug");
60 $debug = 0 if !defined($debug) || $debug !~ /^\d+$/o;
61 $Packages::CGI::debug = $debug;
62
63 &Packages::Config::init( '../' );
64 &Packages::DB::init();
65
66 my $acc = I18N::AcceptLanguage->new();
67 my $http_lang = $acc->accepts( $input->http("Accept-Language"),
68                                \@LANGUAGES ) || 'en';
69 debug( "LANGUAGES=@LANGUAGES header=".
70        ($input->http("Accept-Language")||'').
71        " http_lang=$http_lang", 2 ) if DEBUG;
72 bindtextdomain ( 'pdo', $LOCALES );
73 textdomain( 'pdo' );
74
75 my $what_to_do = 'show';
76 my $source = 0;
77 if (my $path = $input->path_info() || $input->param('PATH_INFO')) {
78     my @components = grep { $_ } map { lc $_ } split /\/+/, $path;
79
80     push @components, 'index' if $path =~ m,/$,;
81
82     my %LANGUAGES = map { $_ => 1 } @LANGUAGES;
83     if (@components > 0 and $LANGUAGES{$components[0]}) {
84         $input->param( 'lang', shift(@components) );
85     }
86     if (@components > 0 and $components[0] eq 'source') {
87         shift @components;
88         $input->param( 'source', 1 );
89     }
90     if (@components > 0 and $components[0] eq 'search') {
91         shift @components;
92         $what_to_do = 'search';
93         # Done
94         fatal_error( _g( "search doesn't take any more path elements" ) )
95             if @components;
96     } elsif (@components == 0) {
97         fatal_error( _g( "We're supposed to display the homepage here, instead of getting dispatch.pl" ) );
98     } elsif (@components == 1) {
99         $what_to_do = 'search';
100     } else {
101
102         for ($components[-1]) {
103             /^(index|allpackages|newpkg|changelog|copyright|download|filelist)$/ && do {
104                 pop @components;
105                 $what_to_do = $1;
106                 last;
107             };
108         }
109
110         my %SUITES = map { $_ => 1 } @SUITES;
111         my %SUITES_ALIAS = ( woody => 'oldstable',
112                              sarge => 'stable',
113                              etch => 'testing',
114                              sid => 'unstable', );
115         my %SECTIONS = map { $_ => 1 } @SECTIONS;
116         my %ARCHIVES = map { $_ => 1 } @ARCHIVES;
117         my %ARCHITECTURES = map { $_ => 1 } (@ARCHITECTURES, 'all');
118         my %params_set;
119         sub set_param_once {
120             my ($cgi, $params_set, $key, $val) = @_;
121             if ($params_set->{$key}++) {
122                 fatal_error( sprintf( _g( "%s set more than once in path" ), $key ) );
123             } else {
124                 $cgi->param( $key, $val );
125             }
126         }
127
128         my (@pkg, $need_pkg);
129         foreach (reverse @components) {
130             $need_pkg = !@pkg
131                 && ($what_to_do !~ /^(index|allpackages|newpkg)$/);
132             if (!$need_pkg && $SUITES{$_}) {
133                 set_param_once( $input, \%params_set, 'suite', $_);
134             } elsif (!$need_pkg && (my $s = $SUITES_ALIAS{$_})) {
135                 set_param_once( $input, \%params_set, 'suite', $s);
136             } elsif (!$need_pkg && $SECTIONS{$_}) {
137                 set_param_once( $input, \%params_set, 'section', $_);
138             } elsif (!$need_pkg && $ARCHIVES{$_}) {
139                 set_param_once( $input, \%params_set, 'archive', $_);
140             } elsif (!$need_pkg && $sections_descs{$_}) {
141                 set_param_once( $input, \%params_set, 'subsection', $_);
142             } elsif (!$need_pkg && ($_ eq 'source')) {
143                 set_param_once( $input, \%params_set, 'source', 1);
144             } elsif ($ARCHITECTURES{$_}) {
145                 set_param_once( $input, \%params_set, 'arch', $_);
146             } else {
147                 push @pkg, $_;
148             }
149         }
150         @components = @pkg;
151
152         if (@components > 1) {
153             fatal_error( sprintf( _g( "two or more packages specified (%s)" ), "@components" ) );
154         }
155     } # else if (@components == 1)
156     
157     if (@components) {
158         $input->param( 'keywords', $components[0] );
159         $input->param( 'package', $components[0] );
160     }
161 }
162
163 my ( $pkg, @suites, @sections, @subsections, @archives, @archs );
164
165 my %params_def = ( keywords => { default => undef,
166                                  match => '^\s*([-+\@\s\w\/.:]+)\s*$',
167                              },
168                    package => { default => undef,
169                                 match => '^([\w.+-]+)$',
170                                 var => \$pkg },
171                    suite => { default => 'default', match => '^([\w-]+)$',
172                               array => ',', var => \@suites,
173                               replace => { all => \@SUITES,
174                                            default => \@SUITES } },
175                    archive => { default => ($what_to_do eq 'search') ?
176                                     'all' : 'default',
177                                     match => '^([\w-]+)$',
178                                     array => ',', var => \@archives,
179                                     replace => { all => \@ARCHIVES,
180                                                  default => \@ARCHIVES} },
181                    exact => { default => 0, match => '^(\w+)$',  },
182                    lang => { default => $http_lang, match => '^(\w+)$',  },
183                    source => { default => 0, match => '^(\d+)$',  },
184                    debug => { default => 0, match => '^(\d+)$',  },
185                    searchon => { default => 'names', match => '^(\w+)$', },
186                    section => { default => 'all', match => '^([\w-]+)$',
187                                 alias => 'release', array => ',',
188                                 var => \@sections,
189                                 replace => { all => \@SECTIONS } },
190                    subsection => { default => 'default', match => '^([\w-]+)$',
191                                    array => ',', var => \@subsections,
192                                    replace => { default => [] } },
193                    arch => { default => 'any', match => '^([\w-]+)$',
194                              array => ',', var => \@archs, replace =>
195                              { any => \@ARCHITECTURES } },
196                    format => { default => 'html', match => '^([\w.]+)$',  },
197                    mode => { default => undef, match => '^(\w+)$',  },
198                    );
199 my %opts;
200 my %params = Packages::CGI::parse_params( $input, \%params_def, \%opts );
201 Packages::CGI::init_url( $input, \%params, \%opts );
202
203 my $locale = get_locale($opts{lang});
204 my $charset = get_charset($opts{lang});
205 setlocale ( LC_ALL, $locale )
206     or do { debug( "couldn't set locale $locale, using default" ) if DEBUG;
207             setlocale( LC_ALL, get_locale() )
208                 or do {
209                     debug( "couldn't set default locale either" ) if DEBUG;
210                     setlocale( LC_ALL, "C" );
211                 };
212         };
213 debug( "locale=$locale charset=$charset", 2 ) if DEBUG;
214
215 $opts{h_suites} = { map { $_ => 1 } @suites };
216 $opts{h_sections} = { map { $_ => 1 } @sections };
217 $opts{h_archives} = { map { $_ => 1 } @archives };
218 $opts{h_archs} = { map { $_ => 1 } @archs };
219
220 if ((($opts{searchon} eq 'names') && $opts{source}) ||
221     ($opts{searchon} eq 'sourcenames')) {
222     $opts{source} = 1;
223     $opts{searchon} = 'names',
224     $opts{searchon_form} = 'sourcenames';
225 } else {
226     $opts{searchon_form} = $opts{searchon};
227 }
228 if ($opts{searchon} eq 'contents' or $opts{searchon} eq 'filenames') {
229     $what_to_do = 'search_contents';
230 }
231
232 my $pet1 = new Benchmark;
233 my $petd = timediff($pet1, $pet0);
234 debug( "Parameter evaluation took ".timestr($petd) ) if DEBUG;
235
236 my (%html_header, $menu, $page_content);
237 unless (@Packages::CGI::fatal_errors) {
238     no strict 'refs';
239     &{"do_$what_to_do"}( \%params, \%opts, \%html_header,
240                          \$menu, \$page_content );
241 } else {
242     %html_header = ( title => _g('Error'),
243                      lang => $opts{lang},
244                      print_title => 1,
245                      print_search_field => 'packages',
246                      search_field_values => { 
247                          keywords => _g('search for a package'),
248                          searchon => 'default',
249                          arch => 'any',
250                          suite => 'all',
251                          section => 'all',
252                          exact => 1,
253                          debug => $debug,
254                      },
255                      );
256 }
257
258 print $input->header( -charset => $charset );
259
260 print Packages::HTML::header( %html_header );
261
262 print $menu||'';
263 print_errors();
264 print_hints();
265 print_msgs();
266 print_debug() if DEBUG;
267 print_notes();
268
269 unless (@Packages::CGI::fatal_errors) {
270     print $page_content;
271 }
272
273 my $tet1 = new Benchmark;
274 my $tetd = timediff($tet1, $tet0);
275 print "Total page evaluation took ".timestr($tetd)."<br>"
276     if DEBUG;
277
278 my $trailer = Packages::HTML::trailer( $ROOT );
279 $trailer =~ s/LAST_MODIFIED_DATE/gmtime()/e; #FIXME
280 print $trailer;
281
282 # vim: ts=8 sw=4
283