mirror of
git://git.9front.org/plan9front/plan9front
synced 2025-01-12 11:10:06 +00:00
add newt(1): nntp client for use with nntpfs(4)
This commit is contained in:
parent
5d3d085492
commit
8347075fd9
2 changed files with 378 additions and 0 deletions
246
rc/bin/newt
Executable file
246
rc/bin/newt
Executable file
|
@ -0,0 +1,246 @@
|
|||
#!/bin/rc
|
||||
# bloated, featureful usenet reader for use with nntpfs
|
||||
rfork en
|
||||
ramfs
|
||||
argv0=$0
|
||||
if(~ $#editor 0)
|
||||
editor=hold
|
||||
group=alt/test
|
||||
maxposts=30
|
||||
mnt=/mnt/news
|
||||
if(~ $#newtname 0)
|
||||
newtname=newt@dont-email.me
|
||||
fn enterpost{
|
||||
{
|
||||
echo From: $"newtname
|
||||
echo Newsgroups: `{echo $group | sed 's/\//\./g'}
|
||||
echo Subject: $"subject
|
||||
echo
|
||||
} >/tmp/post
|
||||
eval $editor /tmp/post
|
||||
cat /tmp/post >$mnt/$group/post
|
||||
}
|
||||
fn f { du -a $* | sed 's/^.* //g' }
|
||||
fn fmtd{
|
||||
date=`{cat}
|
||||
if(! ~ $date(1) [0-9]*)
|
||||
date=`{nshift $date}
|
||||
da=$date(1)
|
||||
switch($date(2)){
|
||||
case Jan; mo=1
|
||||
case Feb; mo=2
|
||||
case Mar; mo=3
|
||||
case Apr; mo=4
|
||||
case May; mo=5
|
||||
case Jun; mo=6
|
||||
case Jul; mo=7
|
||||
case Aug; mo=8
|
||||
case Sep; mo=9
|
||||
case Oct; mo=10
|
||||
case Nov; mo=11
|
||||
case Dec; mo=12
|
||||
}
|
||||
if(! ~ $date(3) `{date | awk '{print $6;}'})
|
||||
ti=$date(3)
|
||||
if not
|
||||
ti=`{echo $date(4) | awk '{print substr($0,0,5);}'}
|
||||
echo $mo/$da $ti
|
||||
}
|
||||
fn geth{
|
||||
for(i in $*){
|
||||
from=`{awk -F ' ' '{print $3;}' $i/xover}
|
||||
if(! ~ $#from 0 && ! ~ $#from 1){
|
||||
nfrom=`{
|
||||
for(i in $from){
|
||||
if(~ $i *@*)
|
||||
echo $i | sed 's/(<|>)//g'
|
||||
}
|
||||
}
|
||||
if(! ~ $#nfrom 0)
|
||||
from=$nfrom
|
||||
}
|
||||
if(! ~ $#from 0){
|
||||
date=`{awk -F ' ' '{print $4;}' $i/xover >[2]/dev/null | fmtd}
|
||||
awk -v date'='$"date -v from'='$from(1) -F ' ' \
|
||||
'{print " " $1 " " date " " from " " substr($2,0,50);}' $i/xover >[2]/dev/null
|
||||
}
|
||||
if not
|
||||
echo ' '$"i' nil nil nil'
|
||||
}
|
||||
}
|
||||
fn getposts{ ls | grep -e '^[0-9]+$' | sort -n | tail -$maxposts }
|
||||
fn k{
|
||||
kmnt=`{echo $mnt | sed 's/\//\\\//g'}
|
||||
f $mnt/$* |
|
||||
grep -v -e '\/([0-9]+(\/|$)|post$)' |
|
||||
sed 's/^'$"kmnt'\// g /g' |
|
||||
sort
|
||||
}
|
||||
fn nshift{ shift; echo $* }
|
||||
fn printhelp{
|
||||
echo '[0-9]+ print specified message
|
||||
b back
|
||||
e enter message
|
||||
f jump to first message
|
||||
g ... go to specified group
|
||||
h print message headlines
|
||||
help print this help message
|
||||
k ... list sub-groups under specified group
|
||||
l jump to last message
|
||||
n next
|
||||
p print message with minimal headers
|
||||
P print message with full headers
|
||||
q quit
|
||||
r reply to message
|
||||
y synchronize message list with server
|
||||
" print message in quoted form, suitable for reply
|
||||
|cmd pipe message body to a command
|
||||
||cmd pipe raw message to a command
|
||||
? print debug information'
|
||||
}
|
||||
fn printp{
|
||||
if(test -d $mnt/$group/$1){
|
||||
grep -e '(^From|^Newsgroups|^Subject|^Date)' $1/header
|
||||
echo
|
||||
cat $1/body
|
||||
}
|
||||
echo
|
||||
prompt=$group/$1
|
||||
}
|
||||
fn printpp{
|
||||
if(test -d $mnt/$group/$1){
|
||||
cat $1/article
|
||||
}
|
||||
echo
|
||||
prompt=$group/$1
|
||||
}
|
||||
fn usage{
|
||||
echo usage: $argv0 '[ -f newsgroup ] [ -m mountpoint ] [ -p maxposts ]' >[1=2]
|
||||
exit usage
|
||||
}
|
||||
while(~ $1 -*){
|
||||
switch($1){
|
||||
case -f
|
||||
group=`{echo $2 | sed 's/\./\//g'}
|
||||
shift
|
||||
case -m
|
||||
mnt=$2
|
||||
shift
|
||||
case -p
|
||||
maxposts=$2
|
||||
shift
|
||||
case *
|
||||
usage
|
||||
}
|
||||
shift
|
||||
}
|
||||
if(! ~ $#* 0)
|
||||
usage
|
||||
prompt=$group
|
||||
if(! test -d $mnt/$group){
|
||||
echo !$mnt/$group does not exist >[1=2]
|
||||
exit
|
||||
}
|
||||
builtin cd $mnt/$group
|
||||
go=()
|
||||
posts=`{getposts}
|
||||
geth $posts >/tmp/h
|
||||
post=$posts(1)
|
||||
echo $#posts messages
|
||||
while(){
|
||||
echo -n $"prompt': '
|
||||
cmd=`{read}
|
||||
switch($cmd){
|
||||
case [0-9]*
|
||||
post=$cmd(1)
|
||||
printp $post
|
||||
case b
|
||||
if(! ~ $post $posts(1)){
|
||||
post=`{echo $post^-1 | bc}
|
||||
printp $post
|
||||
}
|
||||
case e
|
||||
enterpost
|
||||
case f
|
||||
post=$posts(1)
|
||||
printp $post
|
||||
case g' '*
|
||||
ngroup=`{nshift $cmd | sed 's/\./\//g'}
|
||||
if(test -d $mnt/$ngroup){
|
||||
if(grep -s -e '^[0-9]+$' <{ls -p $mnt/$ngroup}){
|
||||
group=$ngroup
|
||||
builtin cd $mnt/$group
|
||||
go=()
|
||||
posts=`{getposts}
|
||||
geth $posts >/tmp/h
|
||||
post=$posts(1)
|
||||
prompt=$group
|
||||
echo $#posts messages
|
||||
}
|
||||
if not
|
||||
echo !$ngroup contains no messages
|
||||
}
|
||||
if not
|
||||
echo !$ngroup does not exist
|
||||
case h
|
||||
cat /tmp/h
|
||||
case help
|
||||
printhelp
|
||||
case k
|
||||
k $group
|
||||
case k' '*
|
||||
k `{nshift $cmd | sed 's/\./\//g'}
|
||||
case l
|
||||
post=$posts($#posts)
|
||||
printp $post
|
||||
case p
|
||||
printp $post
|
||||
case p' '*
|
||||
post=`{nshift $cmd}
|
||||
printp $post
|
||||
case P
|
||||
printpp $post
|
||||
case P' '*
|
||||
post=`{nshift $cmd}
|
||||
printpp $post
|
||||
case q
|
||||
exit
|
||||
case r
|
||||
if(test -f $mnt/$group/$post/header){
|
||||
subject='Re: '^`{grep -e '^Subject: ' $mnt/$group/$post/header | sed 's/^Subject: //g'}
|
||||
enterpost
|
||||
}
|
||||
if not
|
||||
echo !message missing
|
||||
case y
|
||||
posts=`{getposts}
|
||||
geth $posts >/tmp/h
|
||||
echo $#posts messages
|
||||
case '"'
|
||||
printp $post | sed 1d | sed 's/^/> /g' | sed 's/^> >/>>/g'
|
||||
case '||'*
|
||||
cmd=`{echo $"cmd | sed 's/^\|\|//g'}
|
||||
cat $mnt/$group/$post/article | eval $cmd
|
||||
case '|'*
|
||||
cmd=`{echo $"cmd | sed 's/^\|//g'}
|
||||
cat $mnt/$group/$post/body | eval $cmd
|
||||
case '?'
|
||||
echo mnt: $mnt
|
||||
echo group: $group
|
||||
echo maxposts: $maxposts
|
||||
echo posts: $posts
|
||||
echo post: $post
|
||||
case n *
|
||||
if(~ $post $posts(1) && ~ $#go 0){
|
||||
go=1
|
||||
printp $post
|
||||
}
|
||||
if not if(! ~ $post $posts($#posts)){
|
||||
go=1
|
||||
post=`{echo $post^+1 | bc}
|
||||
if(test $post -gt $posts($#posts))
|
||||
post=$posts($#posts)
|
||||
printp $post
|
||||
}
|
||||
}
|
||||
}
|
132
sys/man/1/newt
Normal file
132
sys/man/1/newt
Normal file
|
@ -0,0 +1,132 @@
|
|||
.TH NEWT 1
|
||||
.SH NAME
|
||||
newt \- network news transport protocol (NNTP) client
|
||||
.SH SYNOPSIS
|
||||
.B newt
|
||||
[
|
||||
.B -f
|
||||
.I newsgroup
|
||||
] [
|
||||
.B -m
|
||||
.I mountpoint
|
||||
] [
|
||||
.B -p
|
||||
.I maxposts
|
||||
]
|
||||
.SH DESCRIPTION
|
||||
.I Newt
|
||||
provides an interactive, text-based interface to NNTP
|
||||
articles served by
|
||||
.IR nntpfs (4) .
|
||||
.PP
|
||||
There are a number of options:
|
||||
.TP
|
||||
.B -f
|
||||
Load the specified newsgroup. Default is
|
||||
.B alt.test.
|
||||
.TP
|
||||
.B -m
|
||||
Directory where
|
||||
.I nntpfs
|
||||
is mounted. Default is
|
||||
.B /mnt/news.
|
||||
.TP
|
||||
.B -p
|
||||
Number of posts to display, up to and including the most recent post.
|
||||
.PP
|
||||
.I Newt
|
||||
starts by reading the list of messages in the
|
||||
.I newsgroup,
|
||||
printing out the number of messages, and then prompting for commands
|
||||
from standard input. The prompt itself presents the name of the group
|
||||
and the message number in the form of a file system path relative to the
|
||||
.I mountpoint.
|
||||
.PP
|
||||
The commands are:
|
||||
.PP
|
||||
.TP
|
||||
.I number
|
||||
Print message
|
||||
.I number.
|
||||
.TP
|
||||
.B b
|
||||
Print the previous message.
|
||||
.TP
|
||||
.B e
|
||||
Enter a new message, honoring the environment variable
|
||||
.I editor.
|
||||
Default is
|
||||
.IR hold (1) .
|
||||
.TP
|
||||
.B f
|
||||
Jump to the first message in the group.
|
||||
.TP
|
||||
.BI g " newsgroup
|
||||
Change to the specified
|
||||
.I newsgroup.
|
||||
The name of a group may be provided in dotted
|
||||
(\fIalt.test\fR) or path (\fIalt/test\fR) format.
|
||||
.TP
|
||||
.B h
|
||||
Print the disposition, date, sender and subject line
|
||||
of all messages in the group. These lines are suitable
|
||||
for selecting and sending to the prompt, in order to
|
||||
print messages either singly or in aggregate.
|
||||
.TP
|
||||
.B help
|
||||
Print a summary of the available commands.
|
||||
.TP
|
||||
.BI k " [newsgroup]
|
||||
Without an argument,
|
||||
.I k
|
||||
walks the directories under the current group
|
||||
and prints commands suitable for changing to each
|
||||
available sub-group. When provided with an argument,
|
||||
it instead walks the directories under the group specified
|
||||
by the argument.
|
||||
.TP
|
||||
.B l
|
||||
Jump to the last message in the group.
|
||||
.TP
|
||||
.B n, or enter key
|
||||
Print the next message.
|
||||
.TP
|
||||
.B p
|
||||
Print the current message with minimal headers.
|
||||
.TP
|
||||
.B P
|
||||
Print the raw message with full headers.
|
||||
.TP
|
||||
.B q
|
||||
Quit.
|
||||
.TP
|
||||
.B r
|
||||
Reply to the current message.
|
||||
.TP
|
||||
.B y
|
||||
Synchronize message list with the server.
|
||||
.TP
|
||||
.BI | command
|
||||
Run the
|
||||
.I command
|
||||
with the message body as standard input.
|
||||
.TP
|
||||
.BI || command
|
||||
Run the
|
||||
.I command
|
||||
with the whole message as standard input.
|
||||
.TP
|
||||
\fB"\fP
|
||||
Print the current message in quoted form, suitable for reply.
|
||||
.SH SOURCE
|
||||
.B /rc/bin/newt
|
||||
.SH "SEE ALSO"
|
||||
.IR nntpfs (4)
|
||||
.SH BUGS
|
||||
The list of available newsgroups offered by a given server
|
||||
may run to many megabytes in size. This complicates
|
||||
walking the list over a slow Internet connection, and renders
|
||||
searching all but infeasible.
|
||||
.SH HISTORY
|
||||
.I Newt
|
||||
first appeared in 9front (April, 2014).
|
Loading…
Reference in a new issue