Contributor: MARTIN WOODS

{
 WK> I was wondering if anyone has either the layout for a *.MSG
 WK> packet or knows of a unit to generate and process *.MSG packets.
}

unit fidomsg;  { See 2 demo programs attached below !! }
Interface
uses dos;
const
  MsgSize = 32768;
type
AddressType = record
                Zone  : Byte;
                Net   : Word;
                Node  : Word;
                Point : Word;
                Domain: String[15];
              end;

TxtPtrType = ^TxtRecType;
TxtRecType = array[1..MsgSize] of char;

String36    = string[36];
String72    = string[72];
String20    = string[20];
FMsgType =    record
                FromUserName : String36;
                ToUserName   : String36;
                Subject      : String72;
                DateTime     : String20;
                Origin       : AddressType;
                Destination  : AddressType;
                NextReply    : word;
                MsgTxtPtr    : TxtPtrType;
              end;

procedure LoadMsg(var Msg: FMsgType; MsgFilePath : PathStr; var Result: byte);
procedure GetMsgHeap    (var Msg: FMsgType);
procedure DisposeMsgHeap(var Msg: FMsgType);

 Implementation

procedure GetMsgHeap(var Msg: FMsgType);
begin
  New(Msg.MsgTxtPtr);
end;

procedure DisposeMsgHeap(var Msg: FMsgType);
begin
  Dispose(Msg.MsgTxtPtr);
end;

procedure LoadMsg(var Msg: FMsgType; MsgFilePath : PathStr; var Result: byte);

type
  MsgHeaderType =    record
                       HFromUserName : array[1..36] of char;
                       HToUserName   : array[1..36] of char;
                       HSubject      : array[1..72] of char;
                       HDateTime     : array[1..20] of char;
                       HTimesRead    : word;
                       HDestNode     : word;
                       HOrigNode     : word;
                       HCost         : word;
                       HOrigNet      : word;
                       HDestNet      : word;
                       HFiller       : array[1..8] of char;
                       HReplyto      : word;
                       HAttribute    : word;
                       HNextReply    : word;
                      end;
var
  i : word;
  ReadResult : word;
  MsgFile : file;
  MsgHead : MsgHeaderType;
begin
  assign(MsgFile,MsgFilePath);
  {$I-}
  reset(MsgFile,1);
  {$I+}
  Result := IoResult;
  if result>0 then exit;
  fillchar(MsgHead,SizeOf(MsgHead),#00);
  fillchar(Msg.MsgTxtPtr^,MsgSize,#00);
  BlockRead(MsgFile,MsgHead,Sizeof(MsgHead));           {Read Header Info}
  BlockRead(MsgFile,Msg.MsgTxtPtr^,MsgSize,ReadResult); {Read Msg Text}
  If ReadResult = MsgSize then
  begin
    result := 255; {Msg > MsgSize}
    exit;
  end;
  with Msg, MsgHead do
  begin
    for i := 1 to 36 do
    begin
      if HFromUserName[i] = #00 then
      begin;
        FromUserName[0] := chr(i-1);
        i := 36;
      end;
      FromUserName[i] := HFromUserName[i];
    end;
    for i := 1 to 36 do
    begin
      if HToUserName[i]   = #00 then
      begin
        ToUserName[0] := chr(i-1);
        i := 36;
      end;
      ToUserName[i] := HToUserName[i];
    end;
    for i := 1 to 72 do
    begin
      if HSubject[i] = #00 then
      begin
        Subject[0] := chr(i-1);
        i := 72;
      end;
      Subject[i] := HSubject[i];
    end;
    for i := 1 to 20 do
    begin
      if HDateTime[i] = #00 then
      begin
        DateTime[0] := chr(i-1);
        i := 20;
      end;
      DateTime[i] := HDateTime[i];
    end;
    Destination.Zone := 1;
    Destination.Node := HDestNode;
    Destination.Net  := HDestNet;
    Destination.Point := 0;
    Origin.Zone := 1;
    Origin.Node      := HOrigNode;
    Origin.Net       := HOrigNet;
    Origin.Point := 0;
    NextReply        := HNextReply;
  end;
  close(MsgFile);
end;
end.

{ --------------------   DEMO PROGRAM --------------------- }

program DELMSGBY; { A program to kill all FIDOnet messages by a
                    certain person }

{$M 16384,0,65536}

uses dos,fidomsg;

var foo      :byte;
    nametodel:string;
    msg      :FMsgType;
    s        :searchrec;

function upstr(st:string):string;                { string processor that   }
var a:string;                                    { makes all uppercase and }
begin                                            { removes spaces          }
   a:='';
   for foo:=1 to length(st) do
   begin
      If st[foo]<>#32 then a:=a+upcase(st[foo]);
   end;
   upstr:=a;
end;

begin
   if paramcount<1 then          { If they don't know how to use this, then }
   begin
      writeln;
      writeln(' Usage: DELMSGBY [firstname] [lastname]');   { Tell them     }
      writeln;
   end
   else                       { Otherwise, they DO know how to use this, so }
   begin
      nametodel:='';
      for foo:=1 to paramcount do          { Get the name they don't like}         nametodel:=nametodel+' '+paramstr(foo);
      findfirst('*.MSG',Anyfile,s);        { And search all .MSG files for it}      while (DosError=0) do                { If a file is found then}      begin
         GetMsgHeap(msg);                     { Make space on the heap for it}         loadmsg(msg,fexpand(s.name),foo);    { Load it }
         If (upstr(msg.FromUserName)=upstr(nametodel)) then
         begin                           { If the message if from the bad guy}            swapvectors;                 {     then delete it. I used EXEC so}            exec(getenv('COMSPEC'),' /C '+'Del '+fexpand(s.name)); { you can}            swapvectors;                 { easily move, or rename it.}            writeln('Deleting '+fexpand(s.name)+'. It''s Contaminated!');
         end;
         DisposeMsgHeap(msg);            { Done w/ that message, so take back}         findnext(s);                    { the heap space. Then find another}      end;                               { Message to check. }
   end;
end.

{ ---------------------------   DEMO PROGRAM ----------------------------}

{this is a stand alone *.msg reader}
uses dos,crt;
Type FidoHeader=record {structure of the Message Header}
        WhoTheMessageIsFrom,
        WhoTheMessageItTo   : Array[1..36] of Char; {ASCIIZ Strings}
        MessageSubject      : Array[1..72] of Char;
        MessageDate         : Array[1..20] of Char;
                {The Message Date is an ASCIIZ string following this
                format: DD MMM YY  HH:MM:SS-20 Characters Total
                Example: 01 Jun 94 20:00:00 is June 1st 1994 at 8:00PM
                But SeaDog uses a slightly different version and you
                might want to account for that, unfortunately I can't
                remember the exact format, also SLMAIL for SearchLight
                BBS only puts one space between the year and the hour
                even though it's supposed to be 2, I'm surprised this
                hasn't thrown mailers of other BBS programs}
        TimesTheMessageWasRead,
        DestinationNode,
        OriginalNode,
        CostofTheMessage,
        OriginalNet,
        DestinationNet      : Integer;
                {Note: TimesTheMessageWasRead & CostofTheMessage are
                usually ignored when being exported from the BBS and can
                be ignored when importing into a BBS}
        DateWritten,
        DateArrived         : LongInt;
                {I'm not sure how the dates are stored in here, but
                they're usually ignored}
        MessageToWhichThisRepliesTo: Integer;{Irrevelant over a network}
        Arrtibutes          : Word;
                {Bit Field:
                    Bit 0 Private Message
                        1 Crashmail
                        2 Message Was Read
                        3 Message Was Sent
                        4 File Attatched, Filename in subject
                        5 Forwarded Message
                        6 Orphan Message ???
                        7 Kill After Its Sent (I think)
                        8 Message Originated Here (local)
                        9 Hold
                        10 Reserved
                        11 File Request, Filenames in Subject
                        12 Return Receipt Requested
                        13 This message is a Return Receipt
                        14 Audit Trail Requested
                        15 Update Request }
        UnReply             : Integer; {I have No Idea}
End;

Type FidoMsg=record
   msgchar : char;
end;

{The Message Text follows terminated by either a Null (#0) or to Cr's #13#13.
Also all paragraphs are supposed to end with a Hard CR (#141) and you can
ignoreany #13 and reformat the text for your own program, also any lines
starting with^A (#1) should not be imported into the BBS, they are control
lines... thecontents of these lines varies so you'll have to find out that on
your own }
var
  header : fidoheader;
  headerf: file of fidoheader;
  MsgTxt : FidoMsg;
  MsgTxtf: file of FidoMsg;
  DirInfo: SearchRec;
  ch,cx : char;
  cr,count : shortint;
  i:byte;
  l : string;
  s : string;
  howlong : byte;
begin
  FindFirst('*.MSG', Archive, DirInfo);
  while DosError = 0 do
  begin
    window(1,1,80,25);
    clrscr;
    textcolor(lightgreen);
    WriteLn(DirInfo.Name);
    textcolor(green);
    assign(headerf,DirInfo.Name);
    reset(headerf);
    read(headerf,header);
    with header do
    begin
        Writeln('From:  ',WhoTheMessageIsFrom);
        Writeln('To  :  ',WhoTheMessageItTo);
        Writeln('Subj:  ',MessageSubject);
        Writeln('Date:  ',MessageDate);
    end;
    textcolor(white);
Writeln('ÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍ
Í ÍÍÍÍÍÍ');    window(1,wherey,80,25);
    textcolor(cyan);
    close(headerf);
    assign(MsgTxtF,DirInfo.Name);
    reset(MsgTxtF);
    seek(MsgTxtF,sizeof(header));
    cr := 0;
    count := 0;
    l := '';
    repeat
      read(MsgTxtF,MsgTxt);
      ch := MsgTxt.msgchar;
      if not (ch in [#10,#13]) then
      begin
        l := l + ch;
        howlong := length(l);
      end;
      if keypressed then
      begin
        cx := readkey;
        if cx = #27 then halt;
      end;
      if length(l) > 78 then
      begin
        count := length(l);
        while (count > 60) and (l[count] <> ' ') do dec(count);
        writeln(l,copy(l,1,count));
        delete(l,count,length(l));
      end;
      if ch = #13 then
      begin
        writeln(l);
        l := '';
        howlong := 0;
      end;
      if pos('these things?',l) > 0 then
      begin
        write
      end;
      if wherey > 15 then
      begin
        textcolor(12);
        writeln;

        write('Press enter: ');
        readln;

        clrscr;
        textcolor(cyan);
      end;

    until eof(MsgTxtF) or (ioresult > 0);
    if l > '' then
    begin
      writeln(l);
      l := '';
    end;
    textcolor(11);
    write('End of Msg: ');
    textcolor(7);
    cx := readkey;
    if cx = #27 then halt;
    clrscr;
    FindNext(DirInfo);
  end;
  textcolor(7);
end.

end.