From 1e7fe3de7021427f8326a5f7710347035f7a63c5 Mon Sep 17 00:00:00 2001 From: Eric Wong Date: Sat, 5 Sep 2015 02:52:47 +0000 Subject: view: preliminary HTML search interface This hopefully makes it easier to find things without resorting to proprietary external services. --- lib/PublicInbox/Feed.pm | 20 ++++++++-- lib/PublicInbox/Search.pm | 3 +- lib/PublicInbox/SearchView.pm | 92 +++++++++++++++++++++++++++++++++++++++++++ lib/PublicInbox/WWW.pm | 9 ++++- 4 files changed, 118 insertions(+), 6 deletions(-) create mode 100644 lib/PublicInbox/SearchView.pm (limited to 'lib/PublicInbox') diff --git a/lib/PublicInbox/Feed.pm b/lib/PublicInbox/Feed.pm index 8fcb8d82..dbb80cad 100644 --- a/lib/PublicInbox/Feed.pm +++ b/lib/PublicInbox/Feed.pm @@ -123,19 +123,31 @@ sub emit_html_index { my $title = $feed_opts->{description} || ''; $title = PublicInbox::Hval->new_oneline($title)->as_html; my $atom_url = $feed_opts->{atomurl}; - my ($footer, $param, $last, $srch); + my ($footer, $param, $last); my $state = { ctx => $ctx, seen => {}, anchor_idx => 0 }; + my $srch = $ctx->{srch}; + + my $top = "$title (Atom feed)"; + + if ($srch) { + $top = qq{$top} . + qq{ } . + qq{} . + qq{} . + PublicInbox::View::PRE_WRAP; + } else { + $top = PublicInbox::View::PRE_WRAP . $top . "\n"; + } $fh->write("$title" . "" . - '' . PublicInbox::View::PRE_WRAP . - "$title (Atom feed)\n"); + "$top"); # if the 'r' query parameter is given, it is a legacy permalink # which we must continue supporting: my $cgi = $ctx->{cgi}; - if ($cgi && !$cgi->param('r') && ($srch = $ctx->{srch})) { + if ($cgi && !$cgi->param('r') && $srch) { $state->{srch} = $srch; $last = PublicInbox::View::emit_index_topics($state, $fh); $param = 'o'; diff --git a/lib/PublicInbox/Search.pm b/lib/PublicInbox/Search.pm index eb49f724..fd79b63f 100644 --- a/lib/PublicInbox/Search.pm +++ b/lib/PublicInbox/Search.pm @@ -80,7 +80,7 @@ sub query { $opts ||= {}; unless ($query_string eq '') { $query = $self->qp->parse_query($query_string, QP_FLAGS); - $opts->{relevance} = 1; + $opts->{relevance} = 1 unless exists $opts->{relevance}; } $self->do_enquire($query, $opts); @@ -124,6 +124,7 @@ sub do_enquire { my $offset = $opts->{offset} || 0; my $limit = $opts->{limit} || 50; my $mset = $enquire->get_mset($offset, $limit); + return $mset if $opts->{mset}; my @msgs = map { PublicInbox::SearchMsg->load_doc($_->get_document); } $mset->items; diff --git a/lib/PublicInbox/SearchView.pm b/lib/PublicInbox/SearchView.pm new file mode 100644 index 00000000..6113bbf1 --- /dev/null +++ b/lib/PublicInbox/SearchView.pm @@ -0,0 +1,92 @@ +# Copyright (C) 2015 all contributors +# License: AGPLv3 or later (https://www.gnu.org/licenses/agpl-3.0.txt) +package PublicInbox::SearchView; +use strict; +use warnings; +use PublicInbox::SearchMsg; +use PublicInbox::Hval; +use PublicInbox::View; +use POSIX qw/strftime/; + +sub sres_top_html { + my ($ctx, $q) = @_; + my $cgi = $ctx->{cgi}; + # $q ||= $cgi->param('q'); + my $o = int($cgi->param('o') || 0); + my $r = $cgi->param('r'); + $r = (defined $r && $r ne '0'); + my $opts = { offset => $o, mset => 1, relevance => $r }; + my $mset = $ctx->{srch}->query($q, $opts); + my $total = $mset->get_matches_estimated; + my $query = PublicInbox::Hval->new_oneline($q); + my $qh = $query->as_html; + my $res = "$qh - search results" . + qq{} . + qq{}; + + $res .= qq{} if $r; + + $res .= qq{} . + PublicInbox::View::PRE_WRAP; + + my $foot = $ctx->{footer}; + if ($total == 0) { + $foot ||= ''; + $res .= "\n\n[No results found]
$foot";
+	} else {
+		$q = $query->as_href;
+		$q =~ s/%20/+/g; # improve URL readability
+		my $qp = "?q=$q";
+		$qp .= "&o=$o" if $o;
+
+		$res .= "Search results ordered by [";
+		if ($r) {
+			$res .= qq{date|relevance};
+		} else {
+			$qp .= '&r';
+			$res .= qq{date|relevance};
+		}
+		$res .= "]\n\n";
+
+		my $pad = length("$total");
+		my $pfx = ' ' x $pad;
+		foreach my $m ($mset->items) {
+			my $rank = sprintf("%${pad}d", $m->get_rank + 1);
+			my $pct = $m->get_percent;
+			my $smsg = $m->get_document;
+			$smsg = PublicInbox::SearchMsg->load_doc($smsg);
+			my $s = PublicInbox::Hval->new_oneline($smsg->subject);
+			my $f = $smsg->from_name;
+			$f = PublicInbox::Hval->new_oneline($f)->as_html;
+			my $d = strftime('%Y-%m-%d %H:%M', gmtime($smsg->ts));
+			my $mid = $smsg->mid;
+			$mid = PublicInbox::Hval->new_msgid($mid)->as_href;
+			$res .= qq{$rank. }.
+				$s->as_html . "\n";
+			$res .= "$pfx  - by $f @ $d UTC [$pct%]\n\n";
+		}
+		my $nr = scalar $mset->items;
+		my $end = $o + $nr;
+		my $beg = $o + 1;
+		$res .= "
";
+		$res .= "Results $beg-$end of $total.";
+		if ($nr < $total) {
+			$o = $o + $nr;
+			$qp = "q=$q&o=$o";
+			$qp .= "&r" if $r;
+			$res .= qq{ more}
+		}
+		$res .= "\n\n".$foot if $foot;
+	}
+
+	$res .= "
"; + [200, ['Content-Type'=>'text/html; charset=UTF-8'], [$res]]; +} + +sub sres_top_atom { +} + +sub sres_top_thread { +} + +1; diff --git a/lib/PublicInbox/WWW.pm b/lib/PublicInbox/WWW.pm index 5584e49b..2718854c 100644 --- a/lib/PublicInbox/WWW.pm +++ b/lib/PublicInbox/WWW.pm @@ -67,6 +67,7 @@ sub preload { eval { require PublicInbox::Search; + require PublicInbox::SearchView; require PublicInbox::Mbox; require IO::Compress::Gzip; }; @@ -128,8 +129,14 @@ sub get_index { my ($ctx) = @_; require PublicInbox::Feed; my $srch = searcher($ctx); + my $q = $ctx->{cgi}->param('q'); footer($ctx); - PublicInbox::Feed::generate_html_index($ctx); + if (defined $q) { + require PublicInbox::SearchView; + PublicInbox::SearchView::sres_top_html($ctx, $q); + } else { + PublicInbox::Feed::generate_html_index($ctx); + } } # just returns a string ref for the blob in the current ctx -- cgit v1.2.3-24-ge0c7