ref: babf901b4a508c3ec5d1f89655f10377bbdf9637
dir: /appl/wm/sendmail.b/
implement WmSendmail;
include "sys.m";
sys: Sys;
include "draw.m";
draw: Draw;
Context: import draw;
include "tk.m";
tk: Tk;
Toplevel: import tk;
include "tkclient.m";
tkclient: Tkclient;
include "dialog.m";
dialog: Dialog;
include "selectfile.m";
selectfile: Selectfile;
WmSendmail: module
{
init: fn(ctxt: ref Draw->Context, args: list of string);
};
srv: Sys->Connection;
main: ref Toplevel;
ctxt: ref Context;
username: string;
mail_cfg := array[] of {
"frame .top",
"label .top.l -bitmap email.bit",
"frame .top.con",
"frame .top.con.b",
"button .top.con.b.con -bitmap mailcon -command {send msg connect}",
"bind .top.con.b.con <Enter> +{.top.status configure -text {connect/disconnect to mail server}}",
"button .top.con.b.send -bitmap maildeliver -command {send msg send}",
"bind .top.con.b.send <Enter> +{.top.status configure -text {deliver mail}}",
"button .top.con.b.nocc -bitmap mailnocc -command {.hdr.e.cc delete 0 end}",
"bind .top.con.b.nocc <Enter> +{.top.status configure -text {no carbon copy}}",
"button .top.con.b.new -bitmap mailnew -command {send msg new}",
"bind .top.con.b.new <Enter> +{.top.status configure -text {start a new message}}",
"button .top.con.b.save -bitmap mailsave -command {send msg save}",
"bind .top.con.b.save <Enter> +{.top.status configure -text {save message}}",
"pack .top.con.b.con .top.con.b.send .top.con.b.nocc .top.con.b.new .top.con.b.save -padx 2 -side left",
"label .top.status -text {not connected ...} -anchor w",
"pack .top.l -side left",
"pack .top.con -side left -padx 10",
"pack .top.con.b .top.status -in .top.con -fill x -expand 1",
"frame .hdr",
"frame .hdr.l",
"frame .hdr.e",
"label .hdr.l.mt -text {Mail To:}",
"label .hdr.l.cc -text {Mail CC:}",
"label .hdr.l.sb -text {Subject:}",
"pack .hdr.l.mt .hdr.l.cc .hdr.l.sb -fill y -expand 1",
"entry .hdr.e.mt -bg white",
"entry .hdr.e.cc -bg white",
"entry .hdr.e.sb -bg white",
"bind .hdr.e.mt <Key-\n> {}",
"bind .hdr.e.cc <Key-\n> {}",
"bind .hdr.e.sb <Key-\n> {}",
"pack .hdr.e.mt .hdr.e.cc .hdr.e.sb -fill x -expand 1",
"pack .hdr.l -side left -fill y",
"pack .hdr.e -side left -fill x -expand 1",
"frame .body",
"scrollbar .body.scroll -command {.body.t yview}",
"text .body.t -width 15c -height 7c -yscrollcommand {.body.scroll set} -bg white",
"pack .body.t -side left -expand 1 -fill both",
"pack .body.scroll -side left -fill y",
"pack .top -anchor w -padx 5",
"pack .hdr -fill x -anchor w -padx 5 -pady 5",
"pack .body -expand 1 -fill both -padx 5 -pady 5",
"pack .b -padx 5 -pady 5 -fill x",
"pack propagate . 0",
"update"
};
con_cfg := array[] of {
"frame .b",
"button .b.ok -text {Connect} -command {send cmd ok}",
"button .b.can -text {Cancel} -command {send cmd can}",
"pack .b.ok .b.can -side left -fill x -padx 10 -pady 10 -expand 1",
"frame .l",
"label .l.h -text {Mail Server:} -anchor w",
"label .l.u -text {User Name:} -anchor w",
"pack .l.h .l.u -fill both -expand 1",
"frame .e",
"entry .e.h -width 30w",
"entry .e.u -width 30w",
"pack .e.h .e.u -fill x",
"frame .f -borderwidth 2 -relief raised",
"pack .l .e -fill both -expand 1 -side left -in .f",
"bind .e.h <Key-\n> {send cmd ok}",
"bind .e.u <Key-\n> {send cmd ok}",
};
con_pack := array[] of {
"pack .f",
"pack .b -fill x -expand 1",
"focus .e.u",
"update",
};
new_cmd := array[] of {
".hdr.e.mt delete 0 end",
".hdr.e.cc delete 0 end",
".hdr.e.sb delete 0 end",
".body.t delete 1.0 end",
".body.t see 1.0",
"update"
};
init(xctxt: ref Context, argv: list of string)
{
sys = load Sys Sys->PATH;
if (xctxt == nil) {
sys->fprint(sys->fildes(2), "sendmail: no window context\n");
raise "fail:bad context";
}
draw = load Draw Draw->PATH;
tk = load Tk Tk->PATH;
tkclient = load Tkclient Tkclient->PATH;
dialog = load Dialog Dialog->PATH;
selectfile = load Selectfile Selectfile->PATH;
ctxt = xctxt;
tkclient->init();
dialog->init();
selectfile->init();
tkargs := "";
argv = tl argv;
if(argv != nil) {
tkargs = hd argv;
argv = tl argv;
}
titlectl: chan of string;
(main, titlectl) = tkclient->toplevel(ctxt, tkargs,
"MailStop: Sender", Tkclient->Appl);
msg := chan of string;
tk->namechan(main, msg, "msg");
for (c:=0; c<len mail_cfg; c++)
tk->cmd(main, mail_cfg[c]);
tkclient->onscreen(main, nil);
tkclient->startinput(main, "kbd"::"ptr"::nil);
if(argv != nil)
fromreadmail(hd argv);
for(;;) alt {
s := <-main.ctxt.kbd =>
tk->keyboard(main, s);
s := <-main.ctxt.ptr =>
tk->pointer(main, *s);
s := <-main.ctxt.ctl or
s = <-main.wreq or
s = <-titlectl =>
if(s == "exit") {
if(srv.dfd == nil)
return;
status("Closing connection...");
smtpcmd("QUIT");
return;
}
tkclient->wmctl(main, s);
cmd := <-msg =>
case cmd {
"connect" =>
if(srv.dfd == nil) {
connect(main, 1);
fixbutton();
break;
}
disconnect();
"save" =>
save();
"send" =>
sendmail();
"new" =>
for (c=0; c<len new_cmd; c++)
tk->cmd(main, new_cmd[c]);
}
}
}
fixbutton()
{
s := "Connect";
if(srv.dfd != nil)
s = "Disconnect";
tk->cmd(main, ".top.con configure -text "+s+"; update");
}
sendmail()
{
if(srv.dfd == nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"You must be connected to deliver mail",
0, "Continue"::nil);
return;
}
mto := tk->cmd(main, ".hdr.e.mt get");
if(mto == "") {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"You must fill in the \"Mail To\" entry",
0, "Continue (nothing sent)"::nil);
return;
}
if(tk->cmd(main, ".body.t index end") == "1.0") {
opt := "Cancel" :: "Send anyway" :: nil;
if(dialog->prompt(ctxt, main.image, "warning -fg yellow", "Send",
"The body of the mail is empty", 0, opt) == 0)
return;
}
(err, s) := smtpcmd("MAIL FROM:<"+username+">");
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Failed to specify FROM correctly:\n"+err,
0, "Continue (nothing sent)"::nil);
return;
}
status(s);
(err, s) = smtpcmd("RCPT TO:<"+mto+">");
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Failed to specify TO correctly:\n"+err,
0, "Continue (nothing sent)"::nil);
return;
}
status(s);
cc := tk->cmd(main, ".hdr.e.cc get");
if(cc != nil) {
(nil, l) := sys->tokenize(cc, "\t ,");
while(l != nil) {
copy := hd l;
(err, s) = smtpcmd("RCPT TO:<"+copy+">");
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Carbon copy to "+copy+"failed:\n"+err,
0, "Continue (nothing sent)"::nil);
}
}
}
(err, s) = smtpcmd("DATA");
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Failed to enter DATA mode:\n"+err,
0, "Continue (nothing sent)"::nil);
return;
}
sub := tk->cmd(main, ".hdr.e.sb get");
if(sub != nil)
sys->fprint(srv.dfd, "Subject: %s\n", sub);
b := array of byte tk->cmd(main, ".body.t get 1.0 end");
n := sys->write(srv.dfd, b, len b);
b = nil;
if(n < 0) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Error writing server:\n"+sys->sprint("%r"),
0, "Abort (partial send)"::nil);
return;
}
(err, s) = smtpcmd("\r\n.");
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Send",
"Failed to terminate message:\n"+err,
0, "Abort (partial send)"::nil);
return;
}
status(s);
}
save()
{
mto := tk->cmd(main, ".hdr.e.to get");
if(mto == "") {
dialog->prompt(ctxt, main.image, "error -fg red", "Save",
"No message to save",
0, "Dismiss"::nil);
return;
}
pat := list of {
"*.letter (Saved mail)",
"* (All files)"
};
fname: string;
fd: ref Sys->FD;
for(;;) {
fname = selectfile->filename(ctxt, main.image, "Save in Mailbox", pat,
"/usr/"+rf("/dev/user")+"/mail");
if(fname == nil)
return;
fd = sys->create(fname, sys->OWRITE, 8r660);
if(fd != nil)
break;
r := dialog->prompt(ctxt, main.image, "error -fg red", "Save",
"Failed to create "+sys->sprint("%s\n%r", fname),
0, "Retry"::"Cancel"::nil);
if(r > 0)
return;
}
r := sys->fprint(srv.dfd, "Mail To: %s\n", mto);
cc := tk->cmd(main, ".hdr.e.cc get");
if(cc != nil)
r += sys->fprint(srv.dfd, "Mail CC: %s\n", cc);
sb := tk->cmd(main, ".hdr.e.sb get");
if(sb != nil)
r += sys->fprint(srv.dfd, "Subject: %s\n\n", sb);
s := tk->cmd(main, ".body.t get 1.0 end");
b := array of byte s;
n := sys->write(fd, b, len b);
if(n < 0) {
dialog->prompt(ctxt, main.image, "error -fg red", "Save",
"Error writing file "+sys->sprint("%s\n%r", fname),
0, "Continue"::nil);
return;
}
status("wrote "+string(n+r)+" bytes.");
}
status(msg: string)
{
tk->cmd(main, ".top.status configure -text {"+msg+"}; update");
}
disconnect()
{
(err, s) := smtpcmd("QUIT");
srv.dfd = nil;
fixbutton();
if(err != nil) {
dialog->prompt(ctxt, main.image, "error -fg red", "Disconnect",
"Server problem:\n"+err,
0, "Dismiss"::nil);
return;
}
status(s);
}
connect(parent: ref Toplevel, interactive: int)
{
(t, conctl) := tkclient->toplevel(ctxt, postposn(parent),
"Connection Parameters", 0);
cmd := chan of string;
tk->namechan(t, cmd, "cmd");
for (c:=0; c<len con_cfg; c++)
tk->cmd(t, con_cfg[c]);
username = rf("/dev/user");
s := rf("/usr/"+username+"/mail/smtpserver");
if(s != "")
tk->cmd(t, ".e.h insert 0 '"+s);
s = rf("/usr/"+username+"/mail/domain");
if(s != nil)
username += "@"+s;
u := tk->cmd(t, ".e.u get");
if(u == "")
tk->cmd(t, ".e.u insert 0 '"+username);
if(interactive == 0 && checkthendial(t) != 0)
return;
for (c=0; c<len con_pack; c++)
tk->cmd(t, con_pack[c]);
tkclient->onscreen(t, nil);
tkclient->startinput(t, "kbd"::"ptr"::nil);
for(;;) alt {
ss := <-t.ctxt.kbd =>
tk->keyboard(t, ss);
ss := <-t.ctxt.ptr =>
tk->pointer(t, *ss);
ss := <-t.ctxt.ctl or
ss = <-t.wreq or
ss = <-conctl =>
if (ss == "exit")
return;
tkclient->wmctl(t, ss);
s = <-cmd =>
if(s == "can")
return;
if(checkthendial(t) != 0)
return;
status("not connected");
}
srv.dfd = nil;
}
checkthendial(t: ref Toplevel): int
{
server := tk->cmd(t, ".e.h get");
if(server == "") {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"You must supply a server address",
0, "Continue"::nil);
return 0;
}
user := tk->cmd(t, ".e.u get");
if(user == "") {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"You must supply a user name",
0, "Continue"::nil);
return 0;
}
if(dom(user) == "") {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"The user name must contain an '@'",
0, "Continue"::nil);
return 0;
}
return dialer(t, server, user);
}
dialer(t: ref Toplevel, server, user: string): int
{
ok: int;
status("dialing server...");
(ok, srv) = sys->dial(netmkaddr(server, nil, "25"), nil);
if(ok < 0) {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"The following error occurred while\n"+
"dialing the server: "+sys->sprint("%r"),
0, "Continue"::nil);
return 0;
}
status("connected...");
(err, s) := smtpresp();
if(err != nil) {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"An error occurred during sign on.\n"+err,
0, "Continue"::nil);
return 0;
}
status(s);
(err, s) = smtpcmd("HELO "+dom(user));
if(err != nil) {
dialog->prompt(ctxt, t.image, "error -fg red", "Connect",
"An error occurred during login.\n"+err,
0, "Continue"::nil);
return 0;
}
status("ready to send...");
return 1;
}
rf(file: string): string
{
fd := sys->open(file, sys->OREAD);
if(fd == nil)
return "";
buf := array[128] of byte;
n := sys->read(fd, buf, len buf);
if(n < 0)
return "";
return string buf[0:n];
}
postposn(parent: ref Toplevel): string
{
x := int tk->cmd(parent, ".top.con cget -actx");
y := int tk->cmd(parent, ".top.con cget -acty");
h := int tk->cmd(parent, ".top.con cget -height");
return "-x "+string(x-2)+" -y "+string(y+h+2);
}
dom(name: string): string
{
for(i := 0; i < len name; i++)
if(name[i] == '@')
return name[i+1:];
return nil;
}
fromreadmail(hdr: string)
{
(nil, l) := sys->tokenize(hdr, "\n");
while(l != nil) {
s := hd l;
l = tl l;
n := match(s, "subject: ");
if(n != nil) {
tk->cmd(main, ".hdr.e.sb insert end '"+n);
continue;
}
n = match(s, "cc: ");
if(n != nil) {
tk->cmd(main, ".hdr.e.cc insert end '"+n);
continue;
}
n = match(s, "from: ");
if(n != nil) {
n = extract(n);
tk->cmd(main, ".hdr.e.mt insert end '"+n);
}
}
connect(main, 0);
}
extract(name: string): string
{
for(i := 0; i < len name; i++) {
if(name[i] == '<') {
for(j := i+1; j < len name; j++)
if(name[j] == '>')
break;
return name[i+1:j];
}
}
for(i = 0; i < len name; i++)
if(name[i] == ' ')
break;
return name[0:i];
}
lower(c: int): int
{
if(c >= 'A' && c <= 'Z')
c = 'a' + (c - 'A');
return c;
}
match(text, pat: string): string
{
for(i := 0; i < len pat; i++) {
c := text[i];
p := pat[i];
if(c != p && lower(c) != p)
return "";
}
return text[i:];
}
#
# Talk SMTP
#
smtpcmd(cmd: string): (string, string)
{
cmd += "\r\n";
# sys->print("->%s", cmd);
b := array of byte cmd;
l := len b;
n := sys->write(srv.dfd, b, l);
if(n != l)
return ("send to server:"+sys->sprint("%r"), nil);
return smtpresp();
}
smtpresp(): (string, string)
{
s := "";
i := 0;
lastc := 0;
for(;;) {
c := smtpgetc();
if(c == -1)
return ("read from server:"+sys->sprint("%r"), nil);
if(lastc == '\r' && c == '\n')
break;
s[i++] = c;
lastc = c;
}
# sys->print("<-%s\n", s);
if(i < 3)
return ("short read from server", nil);
s = s[0:i-1];
case s[0] {
'1' or '2' or '3' =>
i = 3;
while(s[i] == ' ' && i < len s)
i++;
return (nil, s[i:]);
'4'or '5' =>
i = 3;
while(s[i] == ' ' && i < len s)
i++;
return (s[i:], nil);
* =>
return ("invalid server response", nil);
}
}
Iob: adt
{
nbyte: int;
posn: int;
buf: array of byte;
};
smtpbuf: Iob;
smtpgetc(): int
{
if(smtpbuf.nbyte > 0) {
smtpbuf.nbyte--;
return int smtpbuf.buf[smtpbuf.posn++];
}
if(smtpbuf.buf == nil)
smtpbuf.buf = array[512] of byte;
smtpbuf.posn = 0;
n := sys->read(srv.dfd, smtpbuf.buf, len smtpbuf.buf);
if(n < 0)
return -1;
smtpbuf.nbyte = n-1;
return int smtpbuf.buf[smtpbuf.posn++];
}
netmkaddr(addr, net, svc: string): string
{
if(net == nil)
net = "net";
(n, l) := sys->tokenize(addr, "!");
if(n <= 1){
if(svc== nil)
return sys->sprint("%s!%s", net, addr);
return sys->sprint("%s!%s!%s", net, addr, svc);
}
if(svc == nil || n > 2)
return addr;
return sys->sprint("%s!%s", addr, svc);
}