perm filename CSRPT2.SAI[USE,CSR]6  blob 
sn#664823 filedate 1982-06-29 generic text, type C, neo UTF8
COMMENT ⊗   VALID 00021 PAGES
C REC  PAGE   DESCRIPTION
C00001 00001
C00004 00002	Declarations
C00013 00003	I/O preparation: ttin,lookupfail,enterfail,renamefail,inuse
C00017 00004	Procedures for input: inscan, resp, ynresp, cresp
C00021 00005	Forms: textinfail, invout
C00029 00006	Procedures onhandin, onhandout, new!report for onhand file input/output
C00034 00007	Procedure rdaddr, buildtree, and addfilin for address file input
C00041 00008	Procedures wraddr, untree, addfilout for address file output
C00045 00009	binary search tree maintenance routines: search,insert,delete
C00050 00010	Procedures to access the address file: unpack,display,find
C00054 00011	Sub-procedures for update actions: zipcheck,gethash
C00058 00012	procedures for update actions: look,ins,mfy,del,update,shorten
C00064 00013	The procedure which records orders received
C00073 00014	The `receive' procedure, which handles virtual money
C00079 00015	Procedures for making labels: lab,emitlab,endlab
C00083 00016	The MAIL procedure and its subprocedures abst,invo,inv,status,scanorders
C00103 00017	The `send' procedure, for isolated orders
C00122 00018	The `adjust' procedure, to adjust the inventory
C00127 00019	The president (chief executive)
C00136 00020	The program starts here (sets string constants, including HELPs)
C00147 00021	Set breaks, open channels, call main procedure, end gracefully
C00151 ENDMK
C⊗;
comment Declarations;
begin "report"
      comment This is the CS report system coded by D. Knuth, October 1976,
    	modified and extended since then by Jim Davidson
	Cheshire labels fiddled by Les Earnest, March 1980;
EXTERNAL PROCEDURE BAIL;
require 200 system!pdl;
require 30000 string!space;
require "<>" delimiters;
define # = <;comment>;
define crlf = <('15&'12)>;
define crlf2 = <('15&'12&'12)>;
define icr = <'15>;
define ialt = <'175>;
define iff = <'14>;
define asize = <1900> # maximum number of addresses in the mailing list;
define logasize = <11> # must be equal to 1 + floor(lg asize);
define bsize = <1600> # maximum number of reports in ONHAND.DSK file;
define csize = <1000> # maximum number of customers for a single mailing;
define cvc(i) = <(if i<10 then "0"+i else ("A"-10)+i)> # encodes
			small integer as a single character -- digit or letter;
define ugetf(val,chan,add) = <val ← code('073000000000+'40000000*chan,add)>
	# used to skip to the end of the LOG file when necessary;
define btt = <1> # breaktable for ttin;
define bfflf = <2> # break on formfeed (end page) or linefeed (end line);
define blf = <3> # break on linefeed;
define babs = <4> # break on |;
define bast = <5> # break on *;
define bff = <6> # break on formfeed;
define bpar = <7> # break on );
define bsp = <8> # break on space (or tab);
define bch = <9> # break on non-blank;
define blfdollar = <10> # break on lf or "$";
define bvar = <11> # variable break table, set dynamically;
boolean eof # end of file indicator;
boolean logused, repeating # are we using input from a file, after a system crash? ;
integer brchar # break character;
integer flag # input-output flag;
integer linecnt, charcnt # the numbers of lines and chars not yet saved on the LOG file;
integer ichan # channel for input;
comment there used to be two channels here, ichan & iichan, for character input
	and address file input, respectively.  The change of 12/11/77 obviated this;
integer ochan, oochan, ooochan # channels for output;
comment ochan is used for invoices, ONHAND.DSK file, ORDERS file.
	oochan gets the `compressed' version of the invoices.
	ooochan is used to write out the new version of the address file;
integer lchan # channel for mailing label output;
integer bchan # used for the LOG file of commands;
integer c,d,i,j,k,m,p,q,t, chr # miscellaneous temporary integers, pointers, etc.;
string s,st,stt,str # miscellaneous temporary strings;
string typein # input returned by ttin, ends with cr;
integer scale # scale factor returned by inscan;
integer nl # number of lines returned by unpack;
string array lne[0:6] # individual lines of an address;
integer array mtape[0:1] # used to call the update MTAPE UUO, to reset pointers;
	
integer array llink,rlink,key,balance[0:asize] # binary search tree structure;
	comment the address file is organized as a binary tree.
	key[p] is the hashcode, in binary form, of the addressee whose
	serial number is p. balance[p] is the number of pennies he owes.
	Unused positions of the table are doubly-linked into an AVAIL list
	whose head is at position 0. Such entries have key=0;
string array nmline,lines[0:asize] # nmline[p] is line 1 of an address,
	ending with crlf. lines[p] contains the rest of the address information,
	as follows: Let s=lines[p], then
		s[1 to 1] is the mailing category ("C","F","A","N","M", or"B")
		s[2 to 6] is the zip code or country, or IDMAI, AUTOM, etc.
		s[7 to 18] is the activity code for last 12 mailings
			(0,1,...,9,A,B,... for 0, 1, ..., 9, 10, 11, ...
			orders, or Z if there were back-orders)
		s[19 to ∞] is lines 2,3, etc. of the address, including
			carriage returns and line feeds but not US zip code;
integer troot # the root of the main binary search tree (contains the 
		addresses on the mailing list);
integer recd # total money (in cents) received in today's transactions;
integer fixd # total accounting adjustments in today's transactions;
integer chgd # total money charged to accounts in today's transactions;
integer calrecd # total amount of receipts from California residents;
boolean mailed # has MAHC_A¬YeKC⊃rAgKPAk`A=kiakPAi↑A	JAga=←YKH|v~∃E=←YKC8AgK]⊃KH@F↓QCfAM≥λA¬YeKC⊃rAgKPAk`A=kiakPAi↑A	JAga=←YKH|v~∃E=←YKC8ACMG!C]OK⊂@FAg!←kYH↓β		
%_]	',AEJA]eSii∃\A←kPACMi∃dAae=GKgg%]N}v4∀@~∃MieS] date,invoicedate # today's date in form dd MON 19yy (for invoice, MONdd19yy);
string mon # month whose orders are being processed (3-letter abbr);
string lf,ff,tab,nulls,cshelp,findhelp1,findhelp2,codehelp,updhelp,ordhelp1,ordhelp2,
	yesnohelp,blanks,mailhelp,acthelp,sendhelp,onhandhelp,splithelp,rhelp,
	namehelp1, namehelp2, onhand!filled!msg;
comment constant strings, see page 16;
string array canned[0:31] # text used to write invoices;
preload!with "JAN","FEB","MAR","APR","MAY","JUN",
	"JUL","AUG","SEP","OCT","NOV","DEC"; string array months[1:12];
preload!with "Y","N"; string array yesnoopts[1:2];
preload!with ""; string array nullopt[1:1];
preload!with "UPD","ORD","REC","MAI","SEN","ADJ"; string array csopts[1:6];
preload!with "C","F","N","M","A","B"; string array codeopts[1:6];
preload!with "INS","DEL","MOD","LOOK"; string array updopts[1:4];
preload!with "AVE","CHE";string array labelopts[1:2];
preload!with "ADD","CHA"; string array ordopts[1:2];
preload!with "ABS", "REP"; string array mailopts[1:2];
comment I/O preparation: ttin,lookupfail,enterfail,renamefail,inuse;
procedure ttin;
begin comment sets typein to the line typed in and echoes it also on
	the PRINT file, then gets rid of leading blanks;
integer i;
label retry;
retry:
if ¬repeating then begin "interactive"
    typein←inchwl&icr; comment alternate for  ttyin(btt,brchar);
    setprint(null,"I"); print(typein,lf); setprint(null,"C");
    out(bchan,typein&lf);
    linecnt←linecnt+1;
    charcnt←charcnt+length(typein)+1;
    if linecnt≥15 ∨ charcnt≥540 then begin 
	    comment flush the buffer, to ensure saving of these lines;
	    out(bchan,nulls[(charcnt+1) to 640]);
	    comment update pointer and header information for log file
		(doesn't happen automatically in read-alter mode);
	    i ← code ((('072000+(bchan lsh 5)) lsh 18),mtape[0]);
	    linecnt←charcnt←0;
    end;
end "interactive"
else begin "input from file"
    typein←input(bchan,blf) # get the next line from the file;
    if eof then begin
	comment have come to the end of the file. revert to TTY input;
	repeating ← FALSE;
	ugetf(i,bchan,0) # skip to end of file, before starting output;
	print(crlf2,"	SWITCHING BACK TO TERMINAL INPUT.",crlf2);
	goto retry;
    end;
    print(typein);
    typein←typein[1 to ∞-1] # strip off lf;
end "input from file";
while typein = " " do i←lop(typein);
end;
boolean procedure lookupfail(integer chan; string file);
begin close(chan); lookup(chan,file,flag);
if flag then print(crlf,"Whoa, I can't find ",file,", so I'm stuck.",crlf);
return(flag);
end;
boolean procedure enterfail(integer chan; string file);
begin close(chan); enter(chan,file,flag);
if flag then print(crlf,"Whoa, system error trying to enter ",file,
	", so I'm stuck.",crlf);
return(flag);
end;
boolean procedure renamefail(integer chan; string file; integer pro; reference integer flag);
begin comment the file open on CHAN is renamed to FILE, with protection PRO;
rename(chan,file,pro,flag);
if flag then print(crlf,"Whoa, error in RENAME of ", file,
			   ", so the files might have strange names.",crlf);
return(flag);
end;
boolean procedure inuse(string file);
comment a kludge -- checks to see if file is in use. 'oochan' is used as a temp channel;
begin lookup(oochan,file,flag); enter(oochan,file,flag);
if (flag land '000000777777)=3 then
     print(crlf,"Someone else is using the program, so I'm stuck.",crlf);
close(oochan);
return(flag);
end;
comment Procedures for input: inscan, resp, ynresp, cresp;
integer procedure inscan;
begin comment returns integer contents of typein, ignoring nondigits;
comment sets brchar to last nondigit, scale to no. of digits after ".";
integer t,d; 
t←scale←0; brchar←0;
while typein≠icr do
	begin d←lop(typein);
	if d≥"0" and d≤"9" then
		begin t←10*t+d-"0";
		if brchar="." then scale←scale+1;
		end
	else brchar←d;
	end;
return(t);
end;
integer procedure resp(string q; reference string h; string array opts);
begin comment q is the question asked of the user, h is the HELP string, and opts
	lists the initial characters of allowable responses;
comment the output is 0 if the response was <cr>, otherwise it is the
	index of the option typed;
integer i;
while true do
	begin print(q); ttin;
	if typein=icr then return(0);
	if typein≠"?" then
		begin if equ(typein[1 to 4],"HELP") then
			begin print(crlf,h,crlf,crlf);continue;
			end;
		for i←1 step 1 until arrinfo(opts,2) do
			if equ(typein[1 to length(opts[i])],opts[i])
			then return(i);
		end;
	print("?The responses I can understand at this point are:",crlf);
	for i←1 step 1 until arrinfo(opts,2) do
		print(opts[i],"...,");
	print(crlf,"or <cr> (to get out of this loop),",crlf);
	print("or HELP<cr> (for more information).",crlf);
	end;
end;
integer procedure ynresp(string q);
return(resp(q&" (Y or N) ",yesnohelp,yesnoopts));
integer procedure cresp(string q);
begin comment q asks for a response in dollars and cents;
comment this procedure returns the amount in cents, or -1 if response is just <cr>;
comment also brchar is set to the last nondigit typed;
integer c;
while true do
	begin print(q,"$"); ttin;
	if typein=icr then return(-1);
	c←inscan;
	if scale=2 then return(c);
	print("?I wanted you to type a dollars-and-cents number like 3.14<cr>",
	"----",crlf,"Please try again, or just type <cr> to get out of this.",crlf);
	end;
end;
comment Forms: textinfail, invout;
boolean procedure textinfail;
begin comment the canned text for invoices is read into memory;
if lookupfail(ichan,"FORM.DAT") then return(true);
do st←input(ichan,bfflf) until equ(st[1 to 7],"INVOICE") ∨ eof # bypass directory;
if eof then
	begin print("Whoa, file FORM.DAT has been clobbered, so I'm stuck.",crlf);
	return(true);
	end;
for i←0 step 1 until 31 do canned[i]←input(ichan,bast);
comment for the desired form of FORM.DAT, see the example in the
	user manual and/or the procedure invout below;
return(false);
end;
procedure invout(reference string send,sorry,name,addrlabel;
	integer oldbal,charges, handling; boolean Calif);
begin comment outputs an invoice to ochan, and a compressed version to oochan;
   string procedure short (value string reptlist);
   comment returns a compressed version of the report list, with the names deleted;
   begin "short"
	   string shortlist, name;
	   shortlist←null;
	   name←scan(reptlist,bpar,brchar);
	   while brchar do
		   if ¬equ(name[5 to 12],"Shipping") then begin
		       shortlist←shortlist&name&crlf;
		       scan(reptlist,blf,brchar);
		       name←scan(reptlist,bpar,brchar);
		   end else scan(name,blf,brchar);
	   return(shortlist);
   end "short";
integer newbal; string str;
out(ochan,canned[0]&date&canned[1]&name&canned[2]) # heading, salutation, ret. addr;
out(oochan,date&crlf2&(if ¬equ(addrlabel[45 to 50],"      ")
			 then addrlabel[1 to 50] else addrlabel)&crlf2);
			 comment what's this for?;
if sorry≠0 then begin "sorry list" 
	out(ochan,canned[6]&crlf);
	if equ(sorry[1 to 2],crlf) then out(ochan,canned[18])
	    # crlf at the beginning means that one order had to be split -- say so;
    	out(ochan,sorry&crlf&canned[7]&crlf);
	out(oochan,canned[6]&crlf&short(sorry)&crlf); end "sorry list";
newbal←oldbal+charges; str←cvf(abs(newbal)/100) # notice that str is always positive;
if send≠0 then begin "send list"
	comment modification here September 1979. Check to see if there's any
	    charge, and if not, print 'packing slip' instead of 'invoice';
	out(ochan,canned[3]&(if newbal>0 then canned[4] else canned[5]));
	out(ochan,crlf&send&crlf);
	if Calif and charges>0 then out(ochan,canned[17]&crlf);
	out(oochan,canned[3]&(if newbal>0 then canned[4] else canned[5])&crlf&short(send)&crlf);
	end;
if charges>0 then comment print account status, dep on old and current balance;
	begin if oldbal=0 then out(ochan,canned[10]&str&canned[11])
	else if newbal>0 then out(ochan,canned[12]&str&canned[13])
	else if newbal<0 then out(ochan,canned[14]&str&canned[15])
	else out(ochan,canned[16])
	end
else if oldbal>0 then out(ochan,canned[8]&str&canned[9]);
out(oochan,"balance= "&cvf(newbal/100)&crlf2&"*******************************"&crlf);
if newbal>0 then begin "actual invoice"
    comment if there's actually any money owing, print a real invoice;
    comment 19: INVOICE ↔ invoiceno:. 20: ↔ DATE:. 21 : ↔ customer no.:
	22 : ↔ P.O. no.: ↔. 23 : (header) Descr , Price ↔. 24 : ↔ sub total
	25 : ↔ previously due. 26: previous credit. 27 : ↔ shipping. 28: ↔ taxes
	29 : ↔ total due. 30,31 : ↔ all the rest;
    string rept;
    out(ochan,crlf&canned[19]&invoicedate);
    out(ochan,canned[20]&date&canned[21]);
    comment print reference no., if applicable.  Get it from the addrlabel.
	    Only '9999's won't have a reference no (hash code);
    if addrlabel[45 to 45]="#" then out(ochan,addrlabel[46 to 50]);
    out(ochan,canned[22]&addrlabel&crlf);
    if charges>0 then out(ochan,canned[23]);
    comment now print out the list of reports in SEND, one at a time, with
	    the price right-justified on each line;
    brchar←1;
    while brchar≠0 do begin "one line"
	    rept←scan(send,blfdollar,brchar);
	    if brchar="$" then begin
		    comment print the cost, after right-justifying;
		    if equ(rept[1 to 4],"   (") then comment multiple copies;
			rept←rept&scan(send,blfdollar,brchar);
		    out(ochan,(rept&blanks)[1 to 74]&" $"&scan(send,blf,brchar))
	    end else
		comment no price included: free, or specified on next line;
		out(ochan,rept&lf);
    end "one line";
    comment have printed the whole list -- now print the totals;
    setformat(6,2);
    comment handling charges (if any) have already been added into CHARGES.
 	For printing, subtract them out again, to be printed separately;
    out(ochan,canned[24]&cvf((charges-handling)/100));
    comment next two lines print previous balance (due, or credit);
    out(ochan,if oldbal≥0 then canned[25] else canned[26]);
    out(ochan,cvf(if oldbal≠0 then oldbal/100 else 0.0));
    if handling>0 then out(ochan,canned[27]&cvf(handling/100));
    out(ochan,canned[28]&(if Calif then "Incl." else "N.A."));
    out(ochan,canned[29]&cvf(newbal/100)&canned[30]&canned[31]);
    setformat(0,2);
end "actual invoice"
else out(ochan,crlf&lf&lf&addrlabel&ff);
end;
comment Procedures onhandin, onhandout, new!report for onhand file input/output;
boolean procedure onhandin (string array oldrep,title; integer array onhandh,onhandm,cost;
			reference integer imax);
comment reads from the ONHAND.DSK file, and sets up the arrays and count;
begin "read onhand"
string st;
if lookupfail(ichan,"ONHAND.DSK") then return (false);
imax←-1;
for i←0 step 1 until bsize-1 do
	begin do st←input(ichan,bfflf) until lop(st)="*" or eof;
	if eof then done;
	oldrep[i]←scan(st,babs,brchar);
	if equ(st[1 to 4],"SAME") then title[i]←"" else
		begin title[i]←scan(st,babs,brchar);
		typein←scan(st,babs,brchar)&icr; onhandh[i]←inscan;
		typein←scan(st,babs,brchar)&icr; onhandm[i]←inscan;
		if st=0 then
			begin print("Bad entry in ONHAND.DSK file for ",
			oldrep[i],crlf); st←"$.00"&icr;
			end;
		typein←st; cost[i]←inscan;
		end;
	imax←i;
	end;
print("I have found ",imax+1," records about old reports in file ONHAND.DSK.",crlf);
close(ichan);
return (true);
end "read onhand";
boolean procedure onhandout(string array oldrep,title; integer array onhandh,onhandm,cost;
			value integer imax);
begin "write onhand"
if enterfail(ochan,"ONHAND.DSK") then return (false);
for i←0 step 1 until imax do
	begin out(ochan,"*"&oldrep[i]&"|");
	if title[i]=0 then out(ochan,"SAME"&crlf)
	else out(ochan,title[i]&"|"&cvs(onhandh[i])&"|"&
		cvs(onhandm[i])&"|$"&cvf(cost[i]/100)&crlf);
	if i mod 55 = 54 then out(ochan,ff);
	end;
close(ochan);
return (true);
end "write onhand";
boolean procedure new!report(string name; string array oldrep, title;
			 integer array onhandh, onhandm, cost; reference integer imax);
comment Adds a new report title to the ONHAND.DSK file;
begin "new report"
    k←imax+1; oldrep[k]←st;
    title[k]←name;
    cost[k]←cresp("What is the price of hardcopy? ");
    if cost[k]<0 then return (false);
    print("How many hard copies are on hand? "); ttin;
    if typein=icr then return(false) else onhandh[k]←inscan;
    print("How many microfiche copies are on hand? ");
    ttin; if typein=icr then return(false) else onhandm[k]←inscan;
    imax←k;
    comment if this is has another name (e.g., an AIM number), write a SAME line;
    j←resp("Does it have an alternate number? (Y or N) ",namehelp1,yesnoopts);
    if j=1 then begin
	if resp("What is the alternate number? ",namehelp2,nullopt)=0 then return(false)
	else begin imax←imax+1; oldrep[imax]←typein[1 to ∞-1]; title[imax]←""; end;
    end;
    return (true);
end "new report";
comment Procedure rdaddr, buildtree, and addfilin for address file input;
integer prevk # previous key read by rdaddr;
integer procedure rdaddr;
begin "rdaddr"
comment reads and stores the next valid address from ADDFIL.DSK,
	returning the serial number, or 0 if the end of file is sensed;
comment during this procedure, st represents the file line most recently
	read but not yet digested;
comment The ADDFIL.DSK contains up to twenty entries per page.  Each entry begins
	with a line in the format
		*CZZZZZ|AAAAAAAAAAAA#HHHHHSSSSS$BAL<crlf>
	where C=category, ZZZZZ=zipcode, AAAAAAAAAAAA=activity codes,
	HHHHH=hashcode, SSSSS=serial number, BAL=dollar balance due
	(preceded by - if negative). Then comes 2 to 5 lines of the
	address, each of which should be at most 34 characters wide
	in most cases;
comment most of this code is devoted to simple error checking;
comment when we enter this routine each time (except the first), st is already set
	to the first line of the next entry.  This is because the system has already
	read that line, to detect the end of the previous entry;
string ent,name,addr; integer loc,k; label start;
key[0]←1; nmline[0]←"Listhead"&crlf;
start:
comment if the file was in E editor format, pass over the index page, and page markers;
while st ≠ "*" do
	if eof then return(0) else st←input(ichan,bfflf);
ent←st;name←input(ichan,bfflf);
addr←input(ichan,bfflf);
st←addr[1 to 1];
comment now ent,name,addr are the first three address lines;
comment this next loop glues the remainder of the address onto addr;
while st ≠ "*" and st ≠ "#" do
	begin if length(st)>2 then addr←addr&st # I'm not sure what this checks for.
	    I think that 2 is from the length of crlf, but there's also the fact 
	    that st was set to a single character before entering here;
	if eof then done;
	st←input(ichan,bfflf);
	end;
loc←cvd(ent[27 to 31]) # loc  now contains the serial number;
if loc>asize then
	begin print(crlf,"ADDFIL.DSK error, serial number too big...
	    the following name has been deleted from the file:",crlf,name,crlf,
	    "since it had a serial number of ",loc,".",crlf,
	    "...The rest of the deleted file entry was:",crlf,ent,addr);
	go to start;
	end;
comment the key array contains the hash codes;
if key[loc]≠0 then
	begin print(crlf,"ADDFIL.DSK error, two people with same serial number...
	    the following name has been deleted from the file:",crlf,name,
	    "since it had the same serial number as:",crlf,nmline[loc],
	    "...The rest of the deleted file entry was:",crlf,ent,addr); 
	go to start;
	end;
comment convert the hash key to internal format (integer);
k←cvasc(ent[22 to 26]);
if k ≤ prevk then
	begin print(crlf,"ADDFIL.DSK error, hash codes not increasing...
	    the following name has been deleted from the file:",crlf,name,
	    "since its hash code was not greater than the preceding one.",crlf,
	    "...The rest of the deleted file entry was:",crlf,ent,addr);
	go to start;
	end;
key[loc]←k; prevk←k;
rlink[llink[loc]]←rlink[loc];llink[rlink[loc]]←llink[loc] # remove from AVAIL;
nmline[loc]←name;    
lines[loc]←ent[2 to 7]&ent[9 to 20]&addr # zip, activity codes;
typein←ent[33 to ∞]; balance[loc]←inscan;
if ent[33 to 33]="-" then balance[loc]←-balance[loc];
i←i+1; 
return(loc);
end "rdaddr";
recursive integer procedure buildtree(integer m);
begin "buildtree" comment builds a somewhat balanced binary search tree of up to
		2↑m-1 nodes, returning a pointer to the root;
integer root,subtree;
if m=0 then return(0) 
else begin 
    subtree←buildtree(m-1);
    root←rdaddr;
    if root=0 then return(subtree) comment end of file reached;
    else begin 
	llink[root]←subtree;
	rlink[root]←buildtree(m-1);
	return(root);
    end;
end;
end "buildtree";
procedure addfilin;
begin "addfilin"
comment inputs the address file, assuming that it is on ichan;
for i←1 step 1 until asize-1 do
	begin key[i]←0; llink[i]←i-1; rlink[i]←i+1;
	end;
key[0]←0;llink[0]←asize;rlink[0]←1;
key[asize]←0;llink[asize]←asize-1;rlink[asize]←0;
st←""; prevk←'400000000000;
i←0;
troot←buildtree(logasize);
print(crlf,"The address file contains a total of ",i," entries.",crlf);
end "addfilin";
comment Procedures wraddr, untree, addfilout for address file output;
integer totbal # total balance (owed to us) from all accounts in the file;
Integer cbal # total credit (prepayment by users);
integer kf,kn,km,ka,kb # total number of entries of various categories;
procedure wraddr(integer p);
begin comment appends the address for serial number p to current output page,
	and outputs if the page is full);
comment also gathers statistics about the file;
comment assumes that ooochan is attached to the file ADDFIL.TMP;
string s,t;
t←lines[p];
out(ooochan,"*"); out(ooochan,t[1 to 6]); out(ooochan,"|"); out(ooochan,t[7 to 18]);
out(ooochan,"#"); out(ooochan,cvstr(key[p]));
setformat(5,2); out(ooochan,cvs(p)); setformat(0,2); comment for serial number;
out(ooochan,"$"); out(ooochan,cvf(balance[p]/100)); out(ooochan,crlf);
out(ooochan,nmline[p]); out(ooochan,t[19 to ∞]); comment output address;
if balance[p]>0 then totbal←totbal+balance[p] else cbal←cbal+balance[p];
k←k+1;
if t≠"C" then
	begin if t="F" then kf←kf+1
	else if t="A" then ka←ka+1
	else if t="B" then kb←kb+1
	else if t="N" then kn←kn+1
	else if t="M" then km←km+1;
	end;
if k mod 20 = 0 then out(ooochan,ff);
end;
recursive procedure untree(integer p);
begin comment outputs the binary search tree rooted at p in order by key;
if p≠0 then
	begin untree(llink[p]);
	wraddr(p);
	untree(rlink[p]);
	end;
end;
procedure addfilout;
begin comment outputs the entire address file to ooochan;
k←kf←ka←kn←km←totbal←cbal←0;
untree(troot);
print(crlf,"The address file now contains a total of ",k," entries,
including the following special categories:
F = ",kf," A = ",ka," B = ",kb," N = ",kn," M = ",km,",",crlf,
"The total balance outstanding is $",cvf(totbal/100),",",crlf,
"and the total credit prepaid by customers is $",cvf(abs(cbal)/100),".",crlf);
if asize-k<50 then print("I am currently programmed to handle up to ",
	asize," entries maximum.",crlf);
close(ooochan);
end;
comment binary search tree maintenance routines: search,insert,delete;
integer lp # last position unsuccessfully probed in search routine;
integer procedure search(integer k);
begin comment binary search, returns serial number of addressee having key 
	(hash code) k, or 0 if not in the table;
comment this assumes that the binary tree is sorted by hash code.  Note that
    nothing in the construction of the tree causes it to be sorted -- we
    rely on the fact that it was written out in sorted order, and that insertions
    preserve the sortedness;
integer p;
p←troot; lp←0; key[0]←k;
while k≠key[p] do
	begin lp←p;
	if k<key[p] then p←llink[p] else p←rlink[p];
	end;
return(p);
end;
integer procedure insert(reference string name,ent; string hash; integer bal);
begin comment inserts new address file entry into an available place
	and returns the value of this place (i.e. the serial number);
comment assume that the pointer lp has already been set to the closest leaf, through
	the call to`SEARCH' from inside `GETHASH';
comment the entry is inserted in a place which preserves the ordering;
integer p,k;
k←cvasc(hash);
p←rlink[0] # get available location;
if p=0 then
	begin print("The mailing list is now completely full, so I can't ",
	    "insert the entry for the",crlf," following name: ",name,
	    "To increase the table size one may simply recompile CSREPT",crlf,
	    "with asize and logasize defined larger. 
	    (But do we really want such a big mailing list?)",crlf);
	return(0);
	end;
rlink[0]←rlink[p]; llink[rlink[p]]←0 # remove from AVAIL list;
nmline[p]←name; lines[p]←ent; key[p]←k; balance[p]←bal;
llink[p]←rlink[p]←0;
fixd←fixd-bal;
if k<key[lp] then llink[lp]←p else rlink[lp]←p;
return(p);
end;
procedure delete(integer k);
begin comment deletes entry with key k from its place in the address file,
	using the standard algorithm;
integer p,q,r;
p←search(k);
if p=0 then
	begin print("Hmm... Something went wrong, I just attempted to ",
	"delete a nonexistent key.",crlf); return;
	end;
comment new delete p from its subtree, yielding a subtree with root q;
if llink[p]=0 then q←rlink[p]
else if rlink[p]=0 then q←llink[p]
else	begin q←rlink[p];
	if llink[q]=0 then llink[q]←llink[p]
	else	begin do q←llink[r←q] until llink[q]=0;
		llink[r]←rlink[q]; llink[q]←llink[p]; rlink[q]←rlink[p];
		end;
	end;
comment now adjust the upper part of the tree and the AVAIL list;
if lp=0 then troot←q
else if k<key[lp] then llink[lp]←q else rlink[lp]←q;
q←rlink[0]; rlink[p]←q; llink[q]←p; llink[p]←0; rlink[0]←p;
comment the next insert will go into location p again (this property
	is used in the update "mfy" routine);
key[p]←0; nmline[p]←lines[p]←"";
fixd←fixd+balance[p]; balance[p]←0;
end;
comment Procedures to access the address file: unpack,display,find;
procedure unpack(integer p);
begin comment takes entry from address file position p and stores it
	in lne[0], lne[1], ..., lne[nl];
string ent,zip;
lne[1]←nmline[p] # customer name;
lne[0]←lines[p][1 to 18] # includes category, zip or country, activity codes;
ent←lines[p][19 to ∞] # includes remainder of address;
for j←2 step 1 until 6 do
	begin lne[j]←scan(ent,blf,brchar);
	if ent=0 then
		begin nl←j; done;
		end;
	end;
zip←lne[0][2 to 6];
if zip≤"9" then lne[nl]←lne[nl][1 to ∞-2]&"  "&zip&crlf;
end;
procedure display(integer p);
begin comment types an address entry;
string ent,s;
unpack(p);
if equ(lne[0][2 to 6],"IDMAI") then lne[nl]←lne[nl][0 to ∞-2]&" IDMAIL"&crlf;
for j←1 step 1 until nl do
	print("LINE ",j,": ",lne[j]);
print("hashcode=#",cvstr(key[p]),",   category=",lne[0][1 to 1],
	",   serial=",p,
	if equ(lne[0][2 to 6],"IDMAI") then ",  IDMAIL," else ",",
	crlf,"ordering history=",lne[0][7 to 18],
	",   current balance=$",cvf(balance[p]/100),crlf);
end;
integer procedure find(string s);
begin comment interactive specification of a table entry,
	where s is part of the prompting message;
integer k,p,c,d;
while true do
	begin if resp("Type hashcode "&s&": #",findhelp1,nullopt) = 0 then return(0);
	if (p←search(cvasc(typein[1 to 5])))≠0 then
	case ynresp("Is the name "&nmline[p][1 to ∞-2]&"?") of
		begin continue; return(p); ;
		end
	else	begin if resp("Sorry, that hashcode isn't in the file."
		&" What is the name? ",findhelp2,nullopt)=0 then continue;
		typein←typein[1 to ∞-1]; d←length(typein);
		c←typein; setbreak(bvar,c,null,"IR");
		print("Here are all the entries matching that name:",crlf);
		for i ← 1 step 1 until asize do if key[i]≠0 then
			begin stt←nmline[i];
			while true do
				begin scan(stt,bvar,brchar);
				if brchar=0 then done;
				if equ(stt[1 to d],typein) then
					begin print("#",cvstr(key[i]),": ",nmline[i]);
					      done;
					end else k←lop(stt);
				end;
			end;
		end
	end
end;
comment Sub-procedures for update actions: zipcheck,gethash;
string zip,hash # returned by zipcheck and gethash;
boolean procedure zipcheck (boolean newzip);
begin "zipcheck"
comment before writing an address into the file, we need to check its
	zip code for validity: the first three characters of the hash and the
	zip should agree;
comment this procedure set zip to the desired zip code and sets typein to
	the classification category, or returns false if the user wishes
	to flush the address;
comment it also removes the zip code from the end of the address, if it is
	a real zip;
integer i,k;
stt←lne[nl][1 to ∞-2]&"    "; k←length(stt)-5;
while k>0 and stt[k to k]=" " do k←k-1;
comment find the characters after the rightmost blank, and set zip to the first five;
while k>0 and stt[k to k]≠" " do k←k-1;
zip←stt[k+1 to k+5];
while k>0 and stt[k to k]=" " do k←k-1;
if newzip then
	print("I deduce that the ZIP code or country is ",zip,";",crlf,
 		" if not, please reject this and try again.",crlf);
case resp("Type the classification (C,F,N,M,A, or B), or type <cr> to reject "
	&"this entry: ",codehelp,codeopts) of begin
	return(false);
	comment checks for American ZIP code (sometimes finds false hits),
		and removes them from end of address, to keep them on
		the first line of the entry;
	comment this has to be the last part of zipcheck, since other routines
	    assume that the classfication remains in typein;
	if zip≤"9" or equ(zip,"IDMAI") then lne[nl]←stt[1 to k]&crlf # C;
	if zip≤"9" or equ(zip,"IDMAI") then lne[nl]←stt[1 to k]&crlf # F;
	zip←"ONRXX" # N;
	zip←"DARPA" # M;
	zip←"AUTOM" # A;
	zip←"AUTOF" # B;
	end;
return(true);
end "zipcheck";
procedure gethash;
begin "gethash"
comment sets hash to a hashcode not already in the table,
	beginning with the first three characters of zip;
integer j,k,c,d;
k←length(lne[1]); j←k div 3; k←2*j;
do	begin c←lne[1][j to j]; j←j-1;
	end until (c≥"A" and c≤"Z") or j=0;
if c<"A" or c>"Z" then c←"X";
do	begin d←lne[1][k to k]; k←k-1;
	end until (d≥"A" and d≤"Z") or k=0;
if k=0 then d←"J";
while true do
	begin hash←zip[1 to 3]&c&d; k←cvasc(hash);
	if search(k)=0 then done;
	if d≠"Z" then d←d+1
	else	begin d←"A";
		if c≠"Z" then c←c+1 else c←"A";
		end;
	end # will loop forever if 676 people with same zip[1 to 3];
end "gethash";
comment procedures for update actions: look,ins,mfy,del,update,shorten;
procedure look;
if(p←find("of entry to be displayed"))=0 then return
else display(p);
procedure shorten(integer d);
print("That line was ",d," character", if d=1 then "" else "s",
      " too long for our mailing labels.
       Please shorten it.",crlf);
procedure ins;
begin comment interactive insertion of new address;
integer i,c,p; string ent;
print("Type the new address, two to five lines long:",crlf);
nl←0; for i←1 step 1 until 5 do
	begin label prompt;
prompt:	print("Line ",i,": "); ttin;
	if typein=icr then done;
	if length(typein)>35 then
		begin shorten(length(typein)-35);
		go to prompt;
		end;
	lne[i]←typein&lf; nl←i;
	end;
if nl=0 then return;
if nl=1 then
	begin print("You need another line; try again.",crlf); return;
	end;
if not zipcheck(true) then return;
comment zipcheck sets ZIP and, if necessary, the last line of LNE;
c←lop(typein) # C, F, N, M, A, or B, left over from zipcheck;
gethash;
ent←c&zip&"NNNNNNNNNNN0";
for i←2 step 1 until nl do ent←ent&lne[i];
if(p←insert(lne[1],ent,hash,0))=0 then return;
afchanged←true;
print("OK, I've inserted it; hash code is #",hash,", serial number is ",p,crlf);
end;
procedure mfy;
begin comment interactive modification of an address;
boolean zch # if zipcode could not have changed, avoids a typeout;
string ent;
integer b,j,jmax,p,k;
if (p←find("of entry to be modified"))=0 then return;
display(p);
zch←false;
while true do
    begin jmax←nl+1; if jmax>5 then jmax←5;
    print("Type number of a line to be changed (1 to ",jmax,"),",crlf,              
	    "or <cr> if modifications are complete: "); ttin;
    if typein = icr then done;
    j←typein-"0";
    if j≤0 or j>jmax then print("Invalid line number.",crlf)
    else begin label prompt;
prompt:     print("New line ",j,": "); ttin;
	    if typein=icr then
		begin comment deleting a line. Make sure that it doesn't
		    	reduce the address to one line (causes problems later);
		    if j>2 then begin 
			nl←j-1; zch←true; 
		    end else print("Can't have an address that short. Try again.",crlf);
		    continue;
		end
	    else if length(typein)>35 then
		begin shorten(length(typein)-35);
		go to prompt;
		end;
	    lne[j]←typein&lf;
	    if j≥nl then zch←true;
	    if j>nl then nl←nl+1;
        end;
    end;
if not zipcheck(zch) then return;
afchanged←true;
ent←lop(typein)&zip&lne[0][7 to 18];
for j←2 step 1 until nl do ent←ent&lne[j];
k←key[p];b←balance[p];
if not equ(zip[1 to 3], lne[0][2 to 4]) then
	begin delete(key[p]);
	gethash;
	print("Hashcode changed from #",cvstr(k)," to #",hash,".",crlf);
	insert(lne[1],ent,hash,b) # it goes into location p again but relinked;
end else
begin comment hashcode did not change;
	nmline[p]←lne[1]; lines[p]←ent;
	end;
print("OK, the modification has been made.",crlf);
end;
procedure del;
begin comment interactive deletion of a table entry;
integer j,p;
if(p←find("of entry to be deleted"))=0 then return;
display(p);
j←ynresp("Do you really want to delete this?");
if j≠1 then return else
	begin delete(key[p]);
	afchanged←true;
	print("OK, I did it.",crlf);
	end;
end;
procedure update # main control routine for update loop;
begin comment when debugging, call bail here;
while true do
	case resp("UPDATE: INS, DEL, MOD, or LOOK? ",updhelp,updopts) of
		begin done;ins;del;mfy;look;
		end;
end;
comment The procedure which records orders received;
procedure orders;
begin "orders"
	comment the files ORDERS.XXX, where XXX is a month,
	    consist of a number of lines of the form 
		    #HHHHH,SSSSS:DDDDtabDATEcrlf
	    where HHHHH is the hashcode (ignored in the processing),
	    SSSSS is the serial number right-justified to seven digits,
	    DDDD is a variable-length list of report-order digits 
	    0,...,9,A,B,..., and DATE is the date of recording this order in the file;
integer flag,j,p;
integer imax # number of entries in the file when we start;
   procedure add # used to put new orders into the file;
   begin "add"
       close(ichan); lookup(ichan,"orders."&mon,flag);
       imax←0;
       if enterfail(ochan,"orders."&mon) then return;
       if flag then
	   print("No orders on file for ",mon,", I will create a new file.",crlf)
       else begin 
          print("I will append to existing orders recorded on file ORDERS.",mon,".",crlf);
	   j←0 # count the number of entries we read in;
	   stt←input(ichan,bfflf);
	   if equ(stt[1 to 10],"COMMENT ⊗ ") then
		   begin while brchar≠iff do stt←input(ichan,bff) # skip directory page;
		   stt←input(ichan,bfflf);
		   end;
	   do begin 
		   out(ochan,stt); 
		   j←j+1;
		   stt←input(ichan,bfflf);
		   if stt=ff then stt←input(ichan,bfflf);
		   end 
	   until stt=0;
	   imax←j;
	   print(imax," entries found in ORDERS file.",crlf);
       end;
       j←0 # will be used to count the number of orders added during this session;
       while true do begin "next customer"
	       if(p←find("of person ordering"))=0 then done;
	       if resp("Reports ordered: ",ordhelp2,nullopt)=0 then continue;
	       j←j+1;
	       setformat(7,2);
	       out(ochan,"#"&cvstr(key[p])&","&cvs(p)&":"&typein[1 to ∞-1]
		       &tab&date&crlf);
	       setformat(0,2);
	       end "next customer";
       close(ochan);
       print(j," new orders written onto ORDERS.",mon," for a total of ",imax+j,".",crlf);
   end "add";
   procedure chg # used to modify the order for one of the customers already in the file;
   begin "change"
       label restart;
       integer array sernum[1:csize]; string array tkey,reports,dte[1:csize];
       close(ichan); lookup(ichan,"orders."&mon,flag);
       if flag then begin
	   print("No orders on file for ",mon," so I can't change anything.",crlf);
	   return; end;
       comment allow the user to modify the order file;
       comment first, read in the file, and store it in the arrays;
       stt←input(ichan,bfflf);
       if equ(stt[1 to 10],"COMMENT ⊗ ") then begin
	       while brchar≠iff do stt←input(ichan,bff) # skip directory page;
	       stt←input(ichan,bfflf);
       end;
       for i←1 step 1 until csize do begin "read orders file"
	   if brchar=0 then done;
	   tkey[i]←stt[2 to 6];
	   stt←stt[8 to ∞];
	   sernum[i]←intscan(stt,brchar);
	   reports[i]←scan(stt,bsp,brchar)[2 to ∞];
	   scan(stt,bch,brchar);
	   dte[i]←stt[1 to ∞-2];
	   stt←input(ichan,bfflf);
       end "read orders file";
       if ¬eof then begin
	   print("Whoops. The order file is too big for me to modify.",crlf,
		   "You'll have to use E.",crlf);  return;  end;
       imax←i-1;
       print(imax," entries found in ORDERS file.",crlf);
       comment have read in the file, now let the user modify it;
       if enterfail(ochan,"orders."&mon) then return;
       comment this used to occur earlier, before the file was read in.  The result was
	  that if the file was in fact too big, the routine took the error return,
	  and the subsequent closure of the channel meant that an empty file was written;
       j←0; k←0; comment number of reports changed and deleted, respectively;
       while true do begin "next customer"
		string code; label found;
		if resp("Enter hash code of customer to be modified: #",
			   ordhelp2,nullopt)=0        then done;
   	 	code←typein[1 to 5];
		for i←1 step 1 until imax do 
		    if equ(tkey[i],code) then goto found;
		print("That code is not in the file. Try again.",crlf);
		continue;
	found:	if ynresp("Is the name "&nmline[sernum[i]][1 TO ∞-2]&"?")≠1 then continue;
		print("Current list of reports ordered: ",reports[i],".",crlf);
		if resp("Enter corrected list: ",ordhelp2,nullopt)=0 then begin
			reports[i]←null; k←k+1; end
		else begin reports[i]←typein[1 to ∞-1]; dte[i]←date; j←j+1; end;
       end "next customer";
       print(j," records changed in ORDERS file.",crlf);
       if k≠0 then print(k," records deleted.",crlf);
       comment now write the file back out again;
       setformat(7,2);
       for i←1 step 1 until imax do
	   if reports[i]≠null then
	       out(ochan,"#"&tkey[i]&","&cvs(sernum[i])&":"&reports[i]
		       &tab&dte[i]&crlf);
       setformat(0,2);
       close(ochan);
   end "change";
j←resp("For which month? ",ordhelp1,months);
if j=0 then return else mon←months[j];
while true do
	case resp("ORDER: ADD or CHANGE? ",ordhelp2,ordopts) of begin
		done;
		add;
		chg;
	end;
end "orders";
comment The `receive' procedure, which handles virtual money;
procedure receive;
begin comment interactive processing of receipts;
while true do
	begin label prompt; integer amt;
	p←find("of account to credit (or 99999)");
	afchanged←true;
	if p=0 then done;
prompt:	print("Amount rec'd (or amount + or -, if accounting adjustment)? $");ttin;
	amt←inscan;
	if scale≠2 then
		begin print("Type amount followed by <cr>, e.g., 5.20<cr>,",crlf,
		    "if $5.20 has been received in payment for this account.",crlf,
		    "Type amount followed by -<cr> if the account balance is to",crlf,
		    "decrease by this amount but no payment has been received.",crlf,
		    "Type amount followed by +<cr> if the account balance is to",crlf,
		    "increase by this amount. Just type <cr> to leave the account",crlf,
		    "unchanged. People not on the mailing list have hash code #99999.",crlf);
		go to prompt;
		end;
	if brchar="-" then
		begin fixd←fixd+amt; balance[p]←balance[p]-amt;
		end
	else if brchar="+" then
		begin fixd←fixd-amt; balance[p]←balance[p]+amt;
		end
	else	begin label notax; if brchar≠"." then
			begin print("Incorrect form, try again.",crlf);
			go to prompt;
			end;
		if key[p]=cvasc("99999") then
			begin if ynresp("California resident?")≠1 then go to notax;
			end
		else if key[p]<cvasc("90000") or key[p]≥cvasc("96700") then
			go to notax;
		comment We must pay tax on California residents, the tax was
			included in the purchase price;
		calrecd←calrecd+amt;
notax:		balance[p]←balance[p]-amt; recd←recd+amt;
		end;
	end;
end;
comment Procedures for making labels: lab,emitlab,endlab;
integer ltype # 0 for AVERY labels, 1 for CHESHIRE;
integer lct # mod 3 counter for CHESHIRE label output;
string array blne[1:5] # CHESHIRE label buffer;
string procedure lab(integer p,w; boolean lvunpacked,free);
begin comment makes a 5-line label, w characters wide, for addressee at
	serial number p, either leaving the result in lne[1] thru lne[5]
	(if lvunpacked is true) or delivering it as a string.
	If free=true, the word "(FREE)" is inserted on the second line
	when appropriate;
comment lines are padded to full width;
unpack(p) # puts the address into the array LNE;
for i←1 step 1 until nl do
	begin stt←lne[i][1 to ∞-2]&blanks # pad out each line to full width;
	lne[i]←stt[1 to w-6]&
	(if i=1 then "#"&cvstr(key[p]) comment put in hash code;
	else if free and i=2 and lne[0]≠"C" then "(FREE)"
	else stt[w-5 to w])&crlf;
	end;
for i←nl+1 step 1 until 5 do lne[i]←
	if lvunpacked then blanks[1 to w]&crlf else crlf;
if lvunpacked then return("")
else return(lne[1]&lne[2]&lne[3]&lne[4]&lne[5]&crlf);
comment note that a sixth blank line was returned;
end;
procedure emitlab(integer p,free) # outputs one label;
case ltype of
	begin   out(lchan,lab(p,32,false,free)) # AVERY label;
		begin # CHESHIRE label;
		    lab(p,31,true,free);
		    for i←1 step 1 until 5 do
			case lct of
				begin 
				blne[i]←lne[i][1 to 31] # lct=0;
				blne[i]←blne[i]&"  "&lne[i][1 to 31] # lct=1;
				blne[i]←blne[i]&"   "&lne[i][1 to 31] # lct=2;
				out(lchan,blne[i]&"   "&lne[i][1 to 31]&
					('15&"∪")) # lct=3;
				end;
		    if (lct←(lct+1)mod 4)=0 then out(lchan,'15&"∪");
		end;
	end;
procedure endlab;
begin comment outputs the last labels, if any;
if ltype = 1 and lct≠0 then out(lchan,blne[1]&('15&"∪")&blne[2]&('15&"∪")&
	blne[3]&('15&"∪")&blne[4]&('15&"∪")&blne[5]&('15&"∪"));
close(lchan);
print("The mailing labels have been written onto file LABELS.TMP. 
To print them, see instructions in the user manual; be sure to delete
this file afterwards.",crlf);
end;
comment The MAIL procedure and its subprocedures abst,invo,inv,status,scanorders;
procedure mail;
begin comment takes care of abstract and invoice mailings;
comment The following arrays are allocated only within MAIL;
integer array send,sorry[0:asize] # record of orders that can and can't be filled;
integer array msk[0:42] # the bit corresponding to a report, if that report
	appears on the current month's list (send and sorry use these bit codes);
string array starname,reptname[0:42] # identifies a report;
integer array stock,filled,unf[0:42] # on hand, requests filled, requests unfilled;
integer array reptcost[0:42] # price of report in pennies;
integer imax # maximum report number;
comment The arrays are dimensioned `42' because there is a 7-place gap in the 
	numbering system, between digits and letters.  Actually, only 36 reports can
	be handled in a single mailing. [Actually, it's only 34, because 0
	isn't used for numbering];
comment The 7-place gap caused a screwup on 79/7/31, because it switches the
	odd/even parity for hardcopy/microfiche.  The FOR loops have to be
	changed again to rectify the bug;
   recursive procedure abst(integer p) # emits mailing labels in symmetric
	   order (i.e., in order by hashcode) for tree rooted at p;
   if p≠0 then begin
       abst(llink[p]);
       j←lines[p];
       if j="C" or j="F" then emitlab(p,true);
       abst(rlink[p]);
   end;
   integer procedure status;
   begin comment prints a status report for the user, and returns 0,1,2 according as
       the verdict is to start over, go ahead with shifting, go ahead without shifting;
   print("I have read through all the orders, and here is how things stand:",crlf,
     "(hardcopy)      To be   Unfillable      (microfiche)    To be   Unfillable",crlf,
     "Cost  On hand   sent    requests              On hand   sent    requests",crlf);
   for i←1 step 2 until imax do begin
       print(starname[i]);
       j←cvc(i)-"1"; 
       print("$",cvf(reptcost[j]/100),tab,stock[j],tab,filled[j],tab,unf[j],tab);
       j←cvc(i+1)-"1"; 
       print(tab,tab,stock[j],tab,filled[j],tab,unf[j],crlf);
   end;
   print("Please check this carefully. If there has been some error,",crlf,
     "type <cr> to exit; but if it's all right to go ahead and print the invoices,",crlf,
     "type Y<cr> and I will prepare them: "); ttin;
   if typein = "Y" then begin
       j←resp("OK, I will begin to work on the invoices."&crlf&"Do you want the"&
	      " activity codes to be shifted? (Y or N) ",acthelp,yesnoopts);
       return(j);
   end else begin
       print("OK, I will not print those invoices, please try again.",crlf);
       return(0);
       end;
   end;
   procedure scanorders;
   begin "scanorders" comment read through all orders, and set up the `send' & `sorry' arrays;
   key[0]←0;
   stt←input(ichan,bfflf);
   if equ(stt[1 to 10],"COMMENT ⊗ ") then
	   begin while brchar≠iff do stt←input(ichan,bff) # skip directory page;
	   stt←input(ichan,bfflf);
	   end;
   for i←1 step 1 until asize do send[i]←0;
   while true do
       begin label nextline; integer sendp,sorryp;
       if eof then done;
       if stt="#" then begin
	   p←cvd(stt[8 to 14]) # get the serial number for this order;
	   if p>asize or key[p]=0 then
	   COMMENT potential screwup here--if the customer has been deleted, it
	   is possible that someone has received the same serial number,
	   especially when the file is close to being full.  The effect here is
	   that the system blithely trusts the serial number, and never checks
	   the correctness of the hash code (I think).  The cure is to put in a
	   check for hash code;
	       begin print("I ignored the order ",stt,
		       "since that serial number is no longer in the file.",crlf);
		   go to nextline;
	       end;
	   comment check this customer's balance (filter out people who owe a lot);
	   if balance[p]>4000 ∧ key[p]≠cvasc("99999") then begin "high balance"
	       print("Note: This customer, ",nmline[p][1 to ∞-2],":  ",cvstr(key[p]),
			", currently owes us  $", cvf(balance[p]/100),".",crlf);
	       if ynresp("Do you still want to send him this order?")≠1 then 
		    go to nextline;
	   end "high balance";
	   st←stt[16 to ∞] # actual list of orders for this person;
	   sendp←send[p]; sorryp←sorry[p];
	   while st≥"1" do
	       begin j←lop(st)-"1";
	       comment note that the check for MSK eliminates characters
		  which would fall into the gap bet. nums & letters;
	       if j>42 or msk[j]=0 then
		       begin print("I ignored the invalid report code "&
		       (j+"1")," which appears in the following order:",
		       crlf,stt);
		       end
	       else if((sendp lor sorryp)land msk[j])=0 then
		   comment the check is to see if this rpt has already 
		       been ordered by this user;
		   begin if stock[j]>filled[j] then
			   comment we've got enough -- send one;
			   begin filled[j]←filled[j]+1;
			   sendp←sendp lor msk[j];
			   end
		   else comment none left -- say we're sorry;
			   begin unf[j]←unf[j]+1;
			   sorryp←sorryp lor msk[j];
			   end;
		   end;
	       end;
	   send[p]←sendp;sorry[p]←sorryp;
	   comment store the SEND and SORRY records into the appte place for
	       this user (will be picked up when we print invoices);
	   end;
nextline:stt←input(ichan,bfflf);
       end;
   end "scanorders";
   procedure inv(integer p; boolean shift);
   begin "inv" comment processes (makes up invoices and label for) the addressee
	with serial number p;
   integer reps; reps←0  # used to record the number of diff repts ordered by this
	customer in this mailing. Written out as the last activity code, if
	SHIFT is true. Note that multiple copies must be sent with SEND, not MAIL;
   if lines[p]≠"C" and lines[p]≠"F" and shift then emitlab(p,false)
	comment make up labels for all automatic people: A, B, M, or N;
   else if send[p]≠0∨sorry[p]≠0∨(shift∧balance[p]>0∧equ(lines[p][17 to 18],"00")) then
       comment if this customer has ordered something, or owes us money;
       comment the "00" ensures that customers aren't bugged for money owing
		until they've been quiet for two mailing periods;
       comment Also, we only bug people if this is a 'real' mailing--if shift is true;
       begin comment set up the list of reports that we're going to send;
       string addrlab,sends,sorrys; integer t,j,tbal;
       sends←sorrys←""; tbal←reps←0;
       if send[p]≠0 then
	   comment use the send bit vector, to make up the list of names
		of reports that we're going to send;
	   comment this was changed, July 1979, so that the reports would
		come in order of hardcopy first, followed by fiche.  Previously
		they had been listed in straight numeric order.  The change
		is to make things easier for the people stuffing the envelopes;
	   comment changed again 79/8/1, to correct a bug induced by the 7-place
	       gap in characters.  Note that the gap would always be missed
	       because MSK is 0 for the 7 chars in that range, but the FOR loop
	       steps around it anyway;
	   begin t←send[p];
	   for j ← 0 step 2 until 8, 17 step 2 until 41,
		1 step 2 until 7, 16 step 2 until 42 do
	      if msk[j] land t ≠ 0 then begin
		   comment send this report;
		   reps←reps+1;
		   sends←sends&reptname[j];
		   comment if doesn't get it free, print the cost;
		   if lines[p]≠"C" or reptcost[j]=0 then
		   sends←sends&crlf
		   else    begin tbal←tbal+reptcost[j];
			   sends←sends&"    $"&cvf(reptcost[j]/100)&crlf;
			   end;
		   if(t←t xor msk[j])=0 then done;
		   end;
	   end;
       if sorry[p]≠0 then comment set up the list of reports that we're out of;
	       begin t←sorry[p];
	       for j ← 0 step 2 until 8, 17 step 2 until 41,
		    1 step 2 until 7, 16 step 2 until 42 do
		   if msk[j] land t ≠ 0 then begin
		       comment say that we can't send this one;
		       reps←reps+1;
		       sorrys←sorrys&reptname[j]&crlf;
		       if(t←t xor msk[j])=0 then done;
		       end;
	       end;
       addrlab←lab(p,50,false,false);
       comment actually make up the invoice for this person;
       invout(sends,sorrys,nmline[p],addrlab,balance[p],tbal,0,
	       key[p]≥cvasc("90000") and key[p]<cvasc("96700"));
       emitlab(p,false);
       balance[p]←balance[p]+tbal;
       chgd←chgd+tbal;
       end;
   st←lines[p];
   comment adjust the activity code for this customer, to include this mailing.
	The last column will be filled with cvc(reps);
   if shift then lines[p]←st[1 to 6]&st[8 to 18]&cvc(reps)&st[19 to ∞]
   else if reps > 0 then begin
       comment increase the number of reports ordered in the last period;
       t←st[18 to 18]-"0";
       if t>9 then t←t-7;
       lines[p]←st[1 to 17]&cvc(t+reps)&st[19 to ∞];
       end;
   end; "inv";
   recursive procedure invo(integer p;boolean shift);
   begin comment calls inv for all addresses in p's subtree, symmetric order (inorder);
   if p≠0 then
	   begin invo(llink[p],shift);inv(p,shift);invo(rlink[p],shift);
	   end;
   end;
comment The MAIL procedure really starts here;
if mailed then
	begin print("Sorry, but you can't use MAIL again at this session;",crlf,
	            "   you have to spool the output from this session first.",crlf);
	return;
	end;
if enterfail(lchan,"LABELS.TMP") then return;
if (ltype←resp("MAIL subsystem: AVERY or CHESHIRE labels? ",mailhelp,labelopts)-1)
	< 0 then return;
if ltype=1 then out(lchan,"∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪∪"&
	"∪∪∪∪∪∪∪∪∪");
case resp("MAIL subsystem: Sending abstracts or reports? ",mailhelp,mailopts) of begin
	return;
	begin # abstracts;
	    print("OK, I'm making the labels for you...",crlf);
	    abst(troot);
	end;
	begin # reports;
	    if textinfail then return;
	    j←resp("For which month? ",mailhelp,months);
	    if j=0 then return else mon←months[j];
	    if lookupfail(ichan,"ORDERS."&mon) then return;
	    if enterfail(ochan,"INVOIC.TMP") then return;
	    if enterfail(oochan,"INVOI$.TMP") then return;
	    comment now get report data;
	    print("I need to know some things from that abstract list.",crlf);
	    for j←0 step 1 until 42 do msk[j]←0;
	    imax←0;
	    for i←1 step 2 until 35 do
		    begin label restart;
  restart:          if resp("Please enter STAN- or AIM- or HPP- number of reports "
			      &cvc(i)&" and "&cvc(i+1)&","&crlf&" followed by *AUTHOR,TITLE"&
			      " (or <cr> if done, QUIT<cr> to abort):"&crlf,
			    rhelp,nullopt) = 0 then done;
		    if equ(typein[1 to 4],"QUIT") then return;
		    j←cvc(i)-"1"; k←cvc(i+1)-"1";
		    str←starname[i]←typein&lf;
		    st←scan(str,bast,brchar);
		    reptname[j]←st&"(hardcopy) ";
		    reptname[k]←st&"(microfiche) ";
		    if(reptcost[j]←cresp("What is the cost of hardcopy? "&
			 "(If unavailable, say anything.) "))<0 then go to restart;
		    reptcost[k]←0;
		    for p←j,k do
			    begin print("How many copies of ",reptname[p],
			    crlf,"are available for distribution? "); ttin;
			    if typein=icr then go to restart;
			    stock[p]←inscan;unf[p]←filled[p]←0;
			    end;
		    reptname[j]←reptname[j]&str[1 to ∞-2];
		    reptname[k]←reptname[k]&str[1 to ∞-2];
		    if ynresp("Thanks. Can I assume that the information you just gave "&
			      "for this report"&crlf&"is correct and complete?")≠1 then
			    begin print("Then let's try again.",crlf);
			    go to restart;
			    end
		    else    begin msk[j]←1 lsh(i-1); msk[k]←1 lsh i;
			    imax←i;
			    end;
		    end;
	    print("OK, now I'm looking at the orders...",crlf);
	    scanorders;
	    case status of
		    begin return;invo(troot,true);invo(troot,false);
		    comment note that for each mailing, we tour the entire tree, to
			shift the activity code for each customer;
		    end;
	    close(ochan); afchanged←true;
	    print("The invoices, bills of lading, and dunning letters have ",
		"been written",crlf,"onto file INVOIC.TMP. To print them, do",
		crlf,tab,tab,"XS INVOIC.TMP/NOHEAD",crlf,
		"and after successful completion of that do",crlf,tab,tab,
		"DEL INVOIC.TMP.",crlf,"The files ORDERS.",mon," and INVOI$.TMP ",
		"contain shorter records of",crlf,"the orders requested and ",
		"the invoices actually sent (for your permanent",crlf,
		"records). These should also be XSpooled and then DELeted.",crlf);
	end;
end;
endlab;
mailed←true;
end;
comment The `send' procedure, for isolated orders;
procedure send;
begin "send" 
string array oldrep,title[0:bsize] # contain the abbreviation and title of each rept;
integer array onhandh,onhandm,cost[0:bsize] # quantities on hand, and cost of pc;
comment if the file ONHAND.DSK contains a line like this:
    *CS249|STAN-CS-74-249*KNUTH,HOW NOT TO RUN A COMMITTEE|22|0|$3.50
    then the internal representation has oldrep[i]="CS249", title[i]=
    "STAN...TEE", onhandh[i]=22 (hardcopy on hand), onhandm[i]=0,
    cost[i]=350. If the line on the file is "*AIM123|SAME" then
    oldrep[i]="AIM123", title[i]="", and it means the same as report i-1;
integer imax,tbal,amt,copies; boolean fiche;
integer array bufh,bufm,counth,countm [0:30] # places to update onhandh, onhandm arrays;
comment note the limit of 30 orders at a time per customer;
integer ph,pm # stack pointers for buf and count arrays;
integer handling # amount that we're charging for shipping/handling, if any.
    (printed separately on the invoice);
label restart, finish;
string procedure multiple(integer copies);
        return(crlf&"   ("&cvs(copies)&" copies)");
simple procedure order(integer array buf, count; reference integer ptr; integer quant);
     	comment to add an order to the temporary list (may be cancelled by the user);
if ptr > 30 then begin
	print(crlf,"I can't add any more reports to this order. (size restriction)",crlf,
           "If you still have some you want to send to this customer, just enter",crlf,
	   "his hashcode again, and make another order.",crlf);
	goto finish; end
else begin
	buf[ptr]←k; count[ptr]←quant; ptr←ptr+1;
end;
procedure bulkrate (integer copies);
comment This makes adjustments necessary for volume deals -- reducing
	the price, or adding a handling charge -- and calculates the totals;
begin "bulkrate"
    integer handling1, cost1;
    case ynresp("We are sending "&cvs(copies)&" copies of "&oldrep[k]&"."&crlf&
		    "Do you wish to change the price per copy?") of begin
	    goto restart;
	    cost1←cresp("List price is $"&cvf(cost[k]/100)&"."&crlf&
			    "What is the actual price to be charged? ");
	    cost1←cost[k];
	end;
    if cost1<0 then return;
    amt←copies*cost1;
    tbal←tbal+amt;
    comment cost is printed out here;
    st←st&" @ $"&cvf(cost1/100)&":   $"&cvf(amt/100);
    case ynresp("Do you wish to add a charge for shipping & handling?") of begin
	goto restart;
	begin
	    handling1←cresp("What is the shipping/handling charge? ");
	    if handling1<0 then goto restart
	    else begin
		handling←handling+handling1 # total handling chg for this customre;
		tbal←tbal+handling1;
		st←st&crlf&"   Shipping and handling  $"&cvf(handling1/100);
		comment ship/handle charge (if any) is indicated on a separate line;
	    end;
	end; ;
    end;
end "bulkrate";
if sended then
	begin print("Sorry, but you can't use SEND again at this session;",crlf,
		     "you have to spool the output from this session first.",crlf);
	return;
	end;
if textinfail then return # read canned text for invoice forms;
if enterfail(ochan,"BILLS.TMP") then return;
if enterfail(oochan,"BILLS$.TMP")  then return;
if enterfail(lchan,"SNDLAB.TMP") then return # file for address labels;
if ¬onhandin(oldrep,title,onhandh,onhandm,cost,imax) then return; # read the file;
while true do
	begin "next user" 
	string name,addr,sends,sorrys,thenext;
	integer reps; boolean free;
        if(p←find("(or 99999) for person requesting old reports"))=0 then done;
	afchanged←true;
	handling←0;
	ph←pm←0;
	if key[p]=cvasc("99999") then
		begin print("Type the name and address of customer, ",
		"followed by a blank line:",crlf); ttin;
		if typein=icr then continue else name←typein&lf;
		ttin; if typein=icr then continue else addr←name&typein&lf;
		for i←3 step 1 until 6 do begin
			ttin; if typein=icr then done else addr←addr&typein&lf;
			end;
		while i≤6 do begin addr←addr&crlf; i←i+1; end;
		free←ynresp("Should this customer get the reports free of charge?")=1;
		end
	else free←lines[p]≠"C";
	comment check this customer's balance (filter out people who owe a lot);
	if balance[p]>4000 ∧ key[p]≠cvasc("99999") then begin "high balance"
		print("Note: This customer currently owes us  $", 
			cvf(balance[p]/100),".",crlf);
		if ynresp("Do you still want to send him this order?")≠1 then continue;
	end "high balance";
	thenext←sends←sorrys←""; tbal←reps←0;
	while true do
		begin "next report" 
		label found,notfound;
restart:	if resp("Type short name of "&thenext&"report requested: ",
		 	    sendhelp,nullopt) = 0 then done;
		reps←reps+1;
		st←scan(typein,bast,brchar);
		if brchar≠"*" then begin st←st[1 to ∞-1]; copies←1; end
		else copies ← intscan(typein,brchar);
 		if st[∞ for 1] = "F" then
			begin st←st[1 to ∞-1]; fiche←true; end
		else fiche←false;
		for i←imax step -1 until 0 do if equ(oldrep[i],st) then
			begin k←i; while title[k]=0 do k←k-1;
			go to found;
			end;
notfound:	print("I couldn't find that one in the file.",crlf);
		if resp("Enter its specs in the form STAN- or AIM- or HPP-number "&
		   "followed by *AUTHOR,TITLE:"&crlf,rhelp,nullopt)=0 then continue;
		k←bsize; title[k]←typein[1 to ∞-1];
		if imax≥bsize-2 then begin j←2; print(onhand!filled!msg,crlf); end
		else j←ynresp("Do you want to enter it into the file?");
		case j of begin
		    continue # go on to next report;
		    if ¬new!report(title[k],oldrep,title,onhandh,onhandm,cost,imax)
			then continue  # add this report to the file;
		    begin comment do not enter;
			j←ynresp("Do you have "&(if copies=1 then "a copy"
				   else "enough copies")&" on hand?");
			case j of begin
			    goto restart;
			    onhandh[k]←onhandm[k]←1000;
			    onhandh[k]←onhandm[k]←0;
			    end;
			comment Next line used to just allow (j=1).  But, with the inventory
			    checking disabled, it's possible that the report will be
			    sent even though user say's we're out of it;
			if (not fiche) and (not free) and ((j=1) or (j=2)) then begin
			    cost[k]←cresp("What does a copy cost? ");
			    if cost[k]<0 then goto restart;
			end;
		    end;
		end;
found:		str←title[k]; st←scan(str,bast,brchar);
		comment Some changes were made here, March 1981.  Basically, the
		    inventory checking (to see if we have copies on hand) was disabled,
		    because the inventory had gone out of date.  The user is thus
		    relied on not to try to SEND a report that we're out of;
		if fiche then begin
			amt←0; st←st&"(microfiche) "&str;
			j←onhandm[k]-copies;
			comment This used to be
			    if j≥0 and k<bsize then order(...);
			if k<bsize then order(bufm,countm,pm,copies);
		end else begin
			amt←cost[k]*copies; st←st&"(hardcopy) "&str;
			j←onhandh[k]-copies;
			comment This used to be
			    if j≥0 and k<bsize then order(...);
			if k<bsize then order(bufh,counth,ph,copies);
		end;
		if copies=1 then begin
			if (not free) ∧ (amt>0) then begin
				tbal←tbal+amt;
				st←st&"  $"&cvf(cost[k]/100);
				comment if there's a cost, print it;
			end;
			sends←sends&st&crlf;
		    comment The above block used to be surrounded by
			   if j≥0 then begin .... end else sorrys←sorrys&st&crlf;
		end
		else comment multiple copies requested;
		    comment if there's a cost, BULKRATE will compute it;
		    comment indicate multiple copies just by printing the no. 
			copies on the next line;
		    comment We used to check "j≥0" in the IF statement here;
		    if true then begin comment we can fill the order;
			    st←st&multiple(copies);
			    if ¬free ∧ amt>0 then bulkrate(copies);
			    sends←sends&st&crlf;
		    end else begin "split" 
			comment not enough to fill a complete request for
			    multiple copies. See if we should send what's available;
			integer i;
			j←-j # number of copies we're short;
			if j=copies then i←2 comment no copies left, so don't ask;
			else begin
			    print("The customer has requested "&cvs(copies)&" copies ",
				    "of this report, but we have only "&cvs(copies-j),
				    crlf,"    copies on hand.",crlf);
			    i←resp("Should I send the ones we have? (Y or N) ",
			      splithelp,yesnoopts);
			end;
			case i of begin "send any?"
			    continue;
			    begin comment split the request, and send what we have;
				comment first set the sorry list;
				sorrys←crlf&sorrys&st&multiple(j)&crlf;
				st←st&multiple(copies-j);
				comment now calculate the cost, if any;
				if ¬free ∧ amt>0 then bulkrate(copies-j);
				comment now set the list of reports to send;
				sends←sends&st&crlf;
				if fiche then order(bufm,countm,pm,copies-j)
				else order(bufh,counth,ph,copies-j);
			    end;
			    begin comment don't send anything (wait for more on hand);
				    sorrys←sorrys&st&multiple(copies)&crlf;
			    end;
			end "send any?";
		    end "split";
		thenext←"the next ";
		end "next report";
finish:	if ynresp("Before I make up the invoice, you'd better doublecheck the above."&crlf&
	    (if sends≠0 then "We will be sending"&crlf&sends else "")&
	   (if sorrys≠0 then "We will say that we are unable to send"&crlf&sorrys else "")&
	    "Is it all right to make up the invoice?") ≠ 1   then
		begin print("OK, please try again.",crlf);continue;end;
	print("OK, I am making an invoice for this customer.",crlf);
	while pm>0 do
		begin pm←pm-1; k←bufm[pm];
		onhandm[k]←onhandm[k]-countm[pm];
		comment The next statement was inserted when we disabled invntry checking;
		if onhandm[k]<0 then onhandm[k]←0;
		end;
	while ph>0 do
		begin ph←ph-1; k←bufh[ph];
		onhandh[k]←onhandh[k]-counth[ph];
		comment The next statement was inserted when we disabled invntry checking;
		if onhandh[k]<0 then onhandh[k]←0;
		end;
	comment make up the invoice for this customer;
	if key[p]=cvasc("99999") then 
		comment this seems to be the only case where invout gets hold
			of an address that hasn't been padded out to full width.
			Of course, it also doesn't have a hash code;
		invout(sends,sorrys,name,addr,0,tbal,handling,false)
	else begin 
		addr←lab(p,50,false,false);
		invout(sends,sorrys,nmline[p],addr,balance[p],tbal,handling,
			key[p]≥cvasc("90000")and key[p]<cvasc("96700"));
		addr←lab(p,32,false,false);
		comment There used to be code here to add this set of reports to the
		    customer's activity code for this mailing period:
		         t←lines[p][18 to 18]-"0".  if t>9 then t←t-7.
			 lines[p]←lines[p][1 to 17]&cvc(reps+t)&lines[p][19 to ∞].
		    This was removed Dec/81, since MAIL is no longer used, so we
		    conceptually have one long mailing period;
		end;
	comment make up label for this customer (it will just be printed on paper);
	out(lchan,addr);
	balance[p]←balance[p]+tbal;chgd←chgd+tbal;
	sended←true;
	end "next user";
if sended then begin
	print("I wrote the invoices onto file BILLS.TMP. ",crlf,
	    "To print them, do",crlf,
   	    "	    XS BILLS.TMP/NOHEAD",crlf,
	    "and after successful completion don't forget to DEL BILLS.TMP.",crlf,
	    "The file BILLS$.TMP contains a list of the invoices sent (for your",crlf,
	    "permanent record).  You should XSpool it and then DELete it.",crlf);
	print("The mailing `labels' have been written onto the file SNDLAB.TMP.",crlf,
	    "You should spool them,in the normal way. The paper can then be cut",crlf,
	    "and pasted onto the envelopes. Don't forget to delete the file.",crlf);
end;
close(lchan);
comment now rewrite the ONHAND.DSK file;
if ¬onhandout(oldrep,title,onhandh,onhandm,cost,imax) then return;
end "send";
comment The `adjust' procedure, to adjust the inventory;
procedure adjust;
begin "onhand"
comment The format of the ONHAND.DSK file is described in the `send' procedure;
string thenext;
integer imax;
string array oldrep,title[0:bsize];
integer array onhandh,onhandm,cost[0:bsize];
comment first load the current ONHAND.DSK file, into the arrays;
if ¬onhandin(oldrep,title,onhandh,onhandm,cost,imax) then return;
thenext←"";
comment adjust one report at a time;
while true do
	begin boolean found, fiche;
	found←fiche←false;
	if resp("Type short name of "&thenext&"report to be adjusted: ",
	    onhandhelp,nullopt)=0 then done;
	st←typein[1 to ∞-1];
	if st[∞ for 1]="F" then begin st←st[1 to ∞-1]; fiche←true; end;
	thenext←"the next ";
	for i←imax step -1 until 0 do
		if equ(oldrep[i],st) then begin
			k←i; while title[k]=0 do k←k-1; found←true; done;
		end;
	
	if found then begin
	        comment we've located the entry specified. Now give the user a
		    chance to adjust the quantity;
		print(title[k],crlf);
		if fiche then begin
			print("Number of microfiche copies on hand = ",onhandm[k],crlf,
			      "Enter new quantity of microfiche copies: ");
			ttin; if typein≠icr then onhandm[k]←inscan;
		end else begin
			print("Number of hard copies on hand = ",onhandh[k],crlf,
			      "Enter new quantity of hard copies: ");
			ttin; if typein≠icr then onhandh[k]←inscan;
		end;
	end else begin
		comment adding a new entry to the file;
         	print("I couldn't find that one in the file.",crlf);
		if imax≥bsize-2 then begin j←2; print(onhand!filled!msg,crlf); end
		else j←ynresp("Do you want to enter it into the file?");
		case j of begin
		    done;
		    begin comment adding a new entry to the file;
			if resp("Enter its specs in the form STAN- or AIM-  or HPP- number"
			      &crlf&"followed by *AUTHOR,TITLE:"&crlf,rhelp,nullopt)=0 
			    then continue;
			new!report(typein[1 to ∞-1],oldrep,title,onhandh,onhandm,cost,imax)
				    # add this report to the file;
		     end;
		     continue # don't add an entry -- user changed his mind ;
	 	end;
	end;
end;
comment have made all changes to the ONHAND file. Now write it out;
if ¬onhandout(oldrep,title,onhandh,onhandm,cost,imax) then  return;
end "onhand";
comment The president (chief executive);
procedure the!president;
begin "the!president" comment The main control routine for CSREPORT system functions;
comment a series of revisions made here, Dec 11/77, to provide better backup
	facilities to protect against system glitches (and losing tapes).
	The input file (ADDFIL.DSK) is no longer written directly. Rather, the 
	new mailing list is written into ADDFIL.TMP, then the files are shuffled.
	Thus, the previous version is always around as ADDFIL.BKP;
comment An extra check for another user will happen automatically, since the output
	file (ADDFIL.TMP) will be ENTERed at the start, thus locking out all other
	versions of the program;
if enterfail(ooochan,"ADDFIL.TMP") then begin
	print("Someone else must be using the program.",crlf);
	return;
end;
comment first determine if we're repeating;
backup # get the input buffer again;
while true do begin "rescanning"
    chr←inchrw;
    if chr=lf then begin logused←false; repeating←false; done "rescanning"; end
    else if chr=";" then begin "param" 
	comment found a parameter -- identify it;
	str←inchwl; 
	while str=" " do chr←lop(str);
	if equ(str[1 to 3],"BAC") then begin 
	    logused←true; repeating←true; done "rescanning"; end
	else begin print("Unrecognized parameter on RUN command: ",str,crlf); return; end;
    end "param";
end "rescanning";
comment now set up the LOG file, which records the commands issued;
if ¬repeating then begin "create LOG" 
    comment opening in RA mode won't create the file, so we do it here;
    comment First, check to see that the log file doesn't already exist;
    lookup(bchan,"LOG.TMP",flag);
    if ¬flag then begin "should be in backup"
	print("The file LOG.TMP already exists (possibly as the result of a",crlf,
	    "system crash).  You should either run the program in BACKUP mode,",crlf,
	    "(to redo the transactions recorded in the backup file), or delete",crlf,
	    "the file LOG.TMP before starting up the program.",crlf2);
	return;
    end "should be in backup"
    else begin comment backup doesn't already exist;
	 if enterfail(bchan,"LOG.TMP") then return; close(bchan);
         comment close the channel, and just leave the empty file around;
    end;
end "create LOG";
comment now, open the file in read-alter mode, by doing a LOOKUP then an ENTER;
if lookupfail(bchan,"LOG.TMP") then return # e.g., asked for BACKUP, but file doesn't exist;
enter(bchan,"LOG.TMP",flag);
if flag then begin print("whoa, I can't open the LOG file.",crlf); return; end;
if repeating then begin "check for directory page"
	str←input(bchan,bfflf);
	if equ(str[1 to 10],"COMMENT ⊗ ") then 
		while brchar≠iff do stt←input(bchan,bff) comment skip directory page;
	else useti(bchan,1) # comment reset to start of file;
end "check for directory page";
comment at this point we have entered the output file. Leave it open for writing,
	to ensure that no-one else breaks thru;
comment next, get the input;
if lookupfail(ichan,"ADDFIL.DSK") then return;
print("Hello! Please wait a minute while I read in the address file....",crlf);
addfilin;
comment the channel `ichan' will be used by other parts of the program, for input;
recd←fixd←chgd←calrecd←0;
linecnt←charcnt←0;
while true do begin 
case resp(crlf&"  CSREPORT system:  What can I do for you? ",cshelp,csopts) of
		begin done;
		update;
		orders;
		receive;
		mail;
		send;
		adjust;
		end;
end;
if afchanged and ynresp("May I record all of today's transactions permanently"&
	" on file ADDFIL.DSK?")=1 then
	begin
	addfilout # write everything out to ADDFIL.TMP;
	comment now, shuffle the files, saving a backup;
	comment SAIL requires that we do a lookup before a rename;
	comment new protection set to '277 (delete protect, no read/write by other users);
	comment first, delete the previous backup, if one exists;
	close(ichan); lookup(ichan,"ADDFIL.BKP",flag); if ¬flag then 
		if renamefail(ichan,null,0,flag) then return # delete old version;
	if lookupfail(ichan,"ADDFIL.DSK") then return;
	if renamefail(ichan,"ADDFIL.BKP",'277,flag) then return # old .DSK → .BKP;
	if renamefail(ooochan,"ADDFIL.DSK",'277,flag) then return # .TMP → .DSK;
	end
else print("No changes made to ADDFIL.DSK this time.",crlf);
comment A change made here, April 1980, so that the log file is saved in all cases.
    It used to saved only if we had been running in backup mode (i.e., logused);
begin "save log"
	close(bchan); 
	lookup(bchan,"REPORT.LOG",flag); 
	if ¬flag then
		if renamefail(bchan,null,0,flag) then return # delete prev version;
	if lookupfail(bchan,"LOG.TMP") then return;
	if renamefail(bchan,"REPORT.LOG",'077,flag) then return # LOG.TMP → REPORT.LOG;
end "save log";
	
if recd+calrecd+abs(fixd)+chgd>0 then
print(crlf,"SUMMARY of today's financial transactions:
    $",cvf(recd/100)," received in payments,
    $",cvf(calrecd/100)," of which was from residents of California.
    $",cvf(fixd/100)," was subtracted from accounts due to adjustments or
	    deletions from the mailing list.
    $",cvf(chgd/100)," new charges were sent out on invoices.",crlf);
end "the!president";
comment The program starts here (sets string constants, including HELPs);
tab←'11;
lf←'12;
ff←'14;
setformat(0,2) # format should always be returned to this if changed;
t←call(0,"DATE");
d←t mod 31 + 1;
m←(t←t div 31) mod 12 +1;
date←cvs(d)&" "&months[m]&" "&cvs(1964+ t div 12);
invoicedate←months[m]&cvs(d)&cvs(1964+ t div 12);
mtape[0]← cvsix("GODMOD"); mtape[1]← '17 # set to make the appte call;
nulls←'0; for i←1 step 1 until 10 do nulls←nulls&nulls; 
nulls←nulls[1 to 640];
blanks←"                              ";
blanks←blanks&blanks;
cshelp←"Hello, this is your friendly CSREPORT system.
There are six kinds of things I am programmed to do for you:
	UPDATE	Look at and perhaps change the mailing list.
	ORDER	File away any orders that have been received
		for a given month's list.
	RECEIVE	Record payments received, or adjust accounts.
	MAIL	Prepare mailing labels and/or invoices and
		bills of lading for everyone that has ordered
		reports from a given month's list.
	SEND	Prepare invoices for isolated back-order requests,
		including orders from people not on the mailing list.
	ADJUST  Change the records in the inventory list, to reflect
		changes in the number of copies on hand.
When I ask, `What can I do for you?', just type the first three letters
of one of these functions and hit carriage RETURN <cr>. (You can also
type more than three letters if you want to.) The user's manual, which
contains more information, is file REPORT.TXT[DOC,CSR].";
findhelp1←"I am going to try to identify an addressee for you. Type the
five-character hash code if you know it, or type 99999 if the
addressee is not on our mailing list and not being inserted into it.
If you don't know the hash code, type XXXXX and I will try a name search
of the whole file. If you type just <cr> now, I will go on to something
else. Since hash codes sometimes change, you should doublecheck the
addressee name I find in case it is the wrong person.";
findhelp2←"I am trying to identify an addressee for you. Type the name
or any part of the name, and I will show you all name lines in the file 
which contain that sequence of characters (including blank spaces in
the middle of the sequence, if you use them). Note that I will search
only the first line of each address on the mailing list.";
codehelp←"Type <cr> to reject this entry and flush it; or type C<cr> for
normal entry, F<cr> for the free list, N<cr> for the ONR list, M<cr> for
the ARPA list, A<cr> for the `automatic' list, B<cr> for the `automatic
microfiche' list.";
updhelp←"The UPDATE routine should be used to make all changes to the
address file, since editing with E is risky. To insert a new entry,
type INS and follow instructions. To modify an existing entry, type
MOD and see what happens.  To delete an entry, type DEL (but don't delete
anybody who has orders outstanding on some ORDERS file -- it's best to
delete only after MAILing all orders).  To simply look at an entry,
type LOOK.  Type only <cr> when you want to quit updating.";
ordhelp1←"The ORDERS subsystem is used to record orders received from a
given month's mailing list. The information you enter is recorded in
the file ORDERS.XXX where XXX is JAN,FEB, ..., or DEC.";
ordhelp2←"In ADD mode, you can add new orders to the file. In CHANGE, you
modify a previous entry for a prticular customer, adding or deleting
reports from his list.  You identify a person by his hashcode,
and then say which reports he has ordered.  For example, if he 
wants reports 1,9,A, and G, you can type 19AG or 1AG9, etc.";
mailhelp←"The MAIL subsystem is used to prepare mailing labels and/or
invoices for monthly report distribution.  Two kinds of mailing labels
are presently provided for: AVERY (34 characters wide, one printed
at a time) and CHESHIRE (34 characters wide and three printed at once).
When mailing an abstract list, type ABS<cr> and I will prepare the file
LABELS.TMP containing mailing labels for everyone on the mailing list
except codes N, M, or A.  When mailing invoices and reports, type
REP<cr> and give the necessary information about the relevant month's
reports.  The ORDERS.XXX file for that month will be used to specify
all orders, and the activity records for all customers are shifted
left one position unless you request otherwise.";
acthelp←"Each addressee has activity codes representing the number of orders
he made during the last 12 mailing periods. If you type Y<cr>, the
present mailing is considered a new mailing period.  If you type
N<cr> or <cr>, the present mailing is considered to be combined with
the previous mailing period.";
sendhelp←"Give the short name of a report requested, e.g. CS287 or AIM239 or HPP772,
followed by F if it is microfiche, e.g. CS287F.  But if no more 
reports are requested by this customer, just type <cr>.";
splithelp←"If you type Y, I will send all the copies of this report which we have
on hand, with a note mentioning that we don't have the rest. If you type N
I won't fill any of the order (i.e., as if we were completely out of it),
and I'll send a sorry letter saying so.";
onhandhelp←"The ADJUST routine should be used to make all changes to the
inventory file, since editing with E is risky. To change the quantity
for a report, just enter its short name, e.g. CS287 or AIM 239F
or HPP772. You will be prompted for the new total of copies on hand.
If no more reports are to be updated, just type <cr>.";
namehelp1←"I want to know whether this report can also be referenced by a different
name. For example, some CS Reports also have AI Memo numbers.  If this 
report has such an alternate name, type YES, otherwise type NO or <cr>.";
namehelp2←"Type the alternate name for this report.";
yesnohelp←"Answer YES,SIR<cr> or NO,SIR<cr> or some abbreviation.";
rhelp←"At this point I need to know the names and numbers of the reports,
in order to identify them meaningfully on the invoices to
be written. Here are three examples of the form I want you to type:
STAN-CS-76-562*KNUTH,TRABB PARDO,EARLY DEVEL OF PROG LANGUAGES
AIM-282*TAYLOR,SYNTHESIS OF MANIPULATOR CONTROL PROGRAMS(THESIS)
HPP-77-5*STEFIK,MARTIN,A REVIEW OF KNWLDG-BASED PROBLEM SOLVING
Note that there should be an asterisk (and no space) between the report
number and the author name(s). The title has to be abbreviated so that
everything fits on one line, even when I substitute the word `(microfiche) '
for the asterisk. If the report is a thesis, follow the title by `(THESIS)'.";
onhand!filled!msg←"Whoops! The ONHAND.DSK file is as big as it can get, so I won't
be able to enter this report into the file.  To increase the table size 
one may simply recompile CSREPT with bsize defined larger.";
comment Set breaks, open channels, call main procedure, end gracefully;
setbreak(btt,lf,null,"ISK") # for ttin, translates lower case to upper;
setbreak(bfflf,ff&lf,null,"IA") # for scanning lines of character files;
setbreak(bff,ff,null,"IA") # for scanning pages quickly but carefully;
setbreak(blf,lf,null,"IA") # for scanning lines within a page;
setbreak(babs,"|",null,"IS") # for separating substrings delimited by |;
setbreak(bast,"*",null,"IS") # for separating substrings delimited by *;
setbreak(bpar,")",null,"IA") # for substrings ending with );
setbreak(bsp," "&tab,null,"IS") # for scanning to blank or tab;
setbreak(bch," "&tab,null,"XR") # for scanning past blanks and tabs;
setbreak(blfdollar,"$"&lf,null,"IS") # for scanning for price or eol;
eof←0;
open(ichan←getchan,"DSK",0,19,0,450,brchar,eof) # channel for character and addfil input;
open(ochan←getchan,"DSK",0,0,19,0,0,eof) # channel for character output;
open(oochan←getchan,"DSK",0,0,19,0,0,eof) # for backup invoices;
open(ooochan←getchan,"DSK",0,0,19,0,0,eof) # for addfil output;
open(lchan←getchan,"DSK",0,0,19,0,0,eof) # channel for mailing label output;
open(bchan←getchan,"DSK",0,2,1,450,brchar,eof) # for LOG file;
if ¬inuse("DIALOG.TMP") then begin
setprint("DIALOG.TMP","B");
mailed←sended←afchanged←false;
the!president;
print(crlf,"See you later. Be sure to xspool a copy of DIALOG.TMP,",crlf,
       "which records what happened today.");
setprint(null,"N"); comment closes the dialog file;
close(bchan) # close the LOG file (which is robust in case of system crash);
close(ichan);
close(ochan); close(oochan); close (ooochan); close(lchan) # just in case I forgot;
end;
end "report";