-- Copyright (C) 1983, 1985  by Xerox Corporation. All rights reserved. 
-- GateFTPServer.mesa, HGM, 25-Jun-85  4:32:14
-- Please don't forget to update the herald....

DIRECTORY
  Ascii USING [CR],
  CmFile USING [Handle, TableError],
  Event USING [aboutToSwap],
  EventTypes USING [aboutToBoot, aboutToBootPhysicalVolume],
  Format USING [StringProc],
  Heap USING [systemZone],
  Process USING [SecondsToTicks],
  Profile USING [GetDefaultRegistry],
  Put USING [Text],
  STP USING [GetProperty],
  STPExtra USING [Action, Denied, ApprovalProc, Server],
  Stream USING [Delete, Handle],
  String USING [AppendChar, AppendString, AppendNumber, CopyToNewString, Equivalent],
  StringLookUp USING [noMatch, TableDesc],
  Supervisor USING [
    AddDependency, AgentProcedure, CreateSubsystem, RemoveDependency,
    SubsystemHandle],
  Time USING [AppendCurrent],
  Token USING [Item, Octal],
  Tool USING [Create, MakeSWsProc, MakeMsgSW],
  ToolWindow USING [TransitionProcType],
  Window USING [Handle],

  Indirect USING [Close, GetParmFileName, NextValue, OpenSection],
  Password USING [Check, Encrypted, Status, ValidMemberOfGroup],
  PupDefs USING [AppendHostName],
  PupStream USING [
    CreatePupByteStreamListener, DestroyPupListener, PupAddress,
    PupPackageDestroy, PupPackageMake, PupListener, RejectThisRequest,
    SecondsToTocks];

GateFTPServer: MONITOR
  IMPORTS
    CmFile, Event, Heap, Process, Profile, Put,
    STP, STPExtra, Stream, String, Supervisor, Time,
    Token, Tool, Indirect, Password, PupDefs, PupStream =
  BEGIN

  herald: LONG STRING = "FTP Server of 25-Jun-85  4:32:03";

  myGroup: LONG STRING ← NIL;
  pupListener: PupStream.PupListener ← NIL;

  parmFileName: LONG STRING ← Indirect.GetParmFileName[];

  firstPassword: LONG POINTER TO PasswordBlock ← NIL;
  PasswordBlock: TYPE = RECORD [
    next: LONG POINTER TO PasswordBlock,
    user: LONG STRING,
    encrypted: Password.Encrypted];

  maxServers: CARDINAL = 2;
  servers: CARDINAL ← 0;
  scratchFileNumber: CARDINAL ← 0;

  tool, msg: Window.Handle ← NIL;
  broom: Supervisor.SubsystemHandle = Supervisor.CreateSubsystem[Broom];
  useCount: CARDINAL ← 0;
  pleaseStop, running: BOOLEAN ← FALSE;

  z: UNCOUNTED ZONE = Heap.systemZone;

  FTPServerOn: PUBLIC ENTRY PROCEDURE =
    BEGIN
    IF (useCount ← useCount + 1) = 1 THEN
      BEGIN
      Supervisor.AddDependency[client: broom, implementor: Event.aboutToSwap];
      running ← TRUE;
      Starter[];
      END;
    UpdatePicture[];
    END;

  Starter: PROCEDURE =
    BEGIN
    pleaseStop ← FALSE;
    Announce["Starting "L, herald];
    FindParameters[];
    CreateListener[];
    END;

  FindParameters: PROCEDURE =
    BEGIN
    cmFile: CmFile.Handle;
    Option: TYPE = MACHINE DEPENDENT{
      myGroup(0), password, noMatch(StringLookUp.noMatch)};
    DefinedOption: TYPE = Option [myGroup..password];
    CheckType: PROCEDURE [h: CmFile.Handle, table: StringLookUp.TableDesc]
      RETURNS [index: CARDINAL] = Indirect.NextValue;
    MyNextValue: PROCEDURE [
      h: CmFile.Handle,
      table: LONG DESCRIPTOR FOR ARRAY DefinedOption OF LONG STRING]
      RETURNS [index: Option] = LOOPHOLE[CheckType];
    optionTable: ARRAY DefinedOption OF LONG STRING ← [
      myGroup: "My Group"L,
      password: "Password"L];
    myGroup ← String.CopyToNewString[parmFileName, z];
    FOR i: CARDINAL DECREASING IN [0..myGroup.length) DO
      IF myGroup[i] = '. THEN myGroup.length ← i; ENDLOOP;
    BEGIN
    builtin: ARRAY [0..8) OF WORD = [
      172427B, 11553B, 61543B, 113154B, 161500B, 47164B, 173546B, 120250B];
    firstPassword ← z.NEW[PasswordBlock];
    firstPassword↑ ← [NIL, String.CopyToNewString["Magic"L, z], LOOPHOLE[builtin]];
    END;
    cmFile ← Indirect.OpenSection["FTPServer"L];
    IF cmFile = NIL THEN RETURN;  -- Nothing needed, so don't complain.
    DO
      option: Option;
      option ← MyNextValue[cmFile, DESCRIPTOR[optionTable] !
        CmFile.TableError =>
          BEGIN
	  IF name[0] # '; THEN Message["Unrecognized parameter: ", name];
	  RETRY;
	  END];
      SELECT option FROM
        noMatch => EXIT;
        myGroup =>
          BEGIN
          z.FREE[@myGroup];
          myGroup ← Token.Item[cmFile, FALSE];
          Message["My Group for FTP passwords is "L, myGroup];
          END;
        password =>
          BEGIN
          user: LONG STRING;
          pwd: STRING = [8*8];
          e: Password.Encrypted;
          p: POINTER ← @e;
          temp: LONG POINTER TO PasswordBlock;
          user ← Token.Item[cmFile];
          FOR i: CARDINAL IN [0..SIZE[Password.Encrypted]) DO
            (p + i)↑ ← Token.Octal[cmFile]; ENDLOOP;
          FOR i: CARDINAL IN [0..SIZE[Password.Encrypted]) DO
            IF i # 0 THEN String.AppendChar[pwd, ' ];
            String.AppendNumber[pwd, (p + i)↑, 8];
            ENDLOOP;
          temp ← z.NEW[PasswordBlock ← [firstPassword, user, e]];
          firstPassword ← temp;
          Message["FTP's Password for "L, user, " is "L, pwd];
          END;
        ENDCASE => ERROR;
      ENDLOOP;
    Indirect.Close[cmFile];
    END;

  ForgetParameters: PROCEDURE =
    BEGIN
    UNTIL firstPassword = NIL DO
      temp: LONG POINTER TO PasswordBlock ← firstPassword;
      firstPassword ← firstPassword.next;
      z.FREE[@temp.user];
      z.FREE[@temp];
      ENDLOOP;
    z.FREE[@myGroup];
    END;

  FTPServerOff: PUBLIC ENTRY PROCEDURE =
    BEGIN
    IF useCount # 0 AND (useCount ← useCount - 1) = 0 THEN
      BEGIN
      running ← FALSE;
      Stopper[];
      Supervisor.RemoveDependency[client: broom, implementor: Event.aboutToSwap];
      END;
    END;

  StopperLocked: ENTRY PROCEDURE = INLINE {Stopper[]};
  Stopper: INTERNAL PROCEDURE =
    BEGIN
    pause: CONDITION ← [timeout: Process.SecondsToTicks[1]];
    pleaseStop ← TRUE;
    UNTIL servers = 0 DO WAIT pause; ENDLOOP;
    DestroyListener[];
    ForgetParameters[];
    Announce["Killed "L, herald];
    END;

  UpdatePicture: PROCEDURE = BEGIN END;

  CreateListener: PROCEDURE =
    BEGIN
    IF pupListener # NIL THEN RETURN;
    [] ← PupStream.PupPackageMake[];
    pupListener ← PupStream.CreatePupByteStreamListener[
      local: [0, 3], proc: CreateSTPServer, ticks: PupStream.SecondsToTocks[3*60],
      filter: CheckConnectionRequest];
    END;

  DestroyListener: PROCEDURE =
    BEGIN
    IF pupListener # NIL THEN
      BEGIN
      PupStream.DestroyPupListener[pupListener];
      PupStream.PupPackageDestroy[];
      pupListener ← NIL;
      END;
    END;

  CheckConnectionRequest: ENTRY PROCEDURE [address: PupStream.PupAddress] =
    BEGIN ENABLE UNWIND => NULL;
    IF pleaseStop THEN
      ERROR PupStream.RejectThisRequest["Sorry, we are trying to go away"];
    IF servers = maxServers THEN
      ERROR PupStream.RejectThisRequest["Sorry, we are full now"];
    servers ← servers + 1;
    END;

  DecrementConnections: ENTRY PROCEDURE =
    BEGIN
    servers ← servers - 1;
    END;
    
  CreateSTPServer: PROCEDURE [byteStream: Stream.Handle, address: PupStream.PupAddress] =
    BEGIN  -- this guy gets FORKed for every accepted RFC
    hostName: STRING = [50];
    buffer: STRING = [200];
    Write: Format.StringProc =
      BEGIN
      FOR i: CARDINAL IN [0..s.length) DO
        char: CHARACTER ← s[i];
	SELECT TRUE FROM
	  buffer.length = buffer.maxlength =>
	    BEGIN
	    Announce[buffer, NIL];
	    buffer.length ← 0;
	    String.AppendChar[buffer, char];
	    END;
	  char = Ascii.CR =>
	    BEGIN
	    Announce[buffer, NIL];
	    buffer.length ← 0;
	    END;
	  ENDCASE => String.AppendChar[buffer, char];
        ENDLOOP;
      END;
    Approval: STPExtra.ApprovalProc =
      BEGIN
      user: LONG STRING = STP.GetProperty[stp, userName];
      password: LONG STRING = STP.GetProperty[stp, userPassword];
IF FALSE THEN -- Barf.  Property list has been flused for retrieve and delete
      IF user = NIL THEN ERROR STPExtra.Denied["Name missing"L];
      SELECT action FROM
        retrieve => AnnounceFor["Retrieving "L, name, user, hostName];
        store =>
          BEGIN
	  AnnounceFor["Storing "L, name, user, hostName];
          InspectCredentials[user, password];
	  END;
        delete =>
          BEGIN
	  AnnounceFor["Deleting "L, name, user, hostName];
          InspectCredentials[user, password];
	  END;
        rename =>
          BEGIN
	  AnnounceFor["Renaming "L, name, user, hostName];
          InspectCredentials[user, password];
	  END;
        list => AnnounceFor["Listing "L, name, user, hostName];
        ENDCASE => ERROR;
      END;
    PupDefs.AppendHostName[hostName, address];
    STPExtra.Server[byteStream, hostName, Write, Approval ! ABORTED => CONTINUE];
    Stream.Delete[byteStream];
    DecrementConnections[];
    END;

  InspectCredentials: PROCEDURE [user, password: LONG STRING] =
    BEGIN
    registry: STRING = [100];
    CopyRegistry: PROCEDURE [s: LONG STRING] = BEGIN String.AppendString[registry, s]; END;
    Profile.GetDefaultRegistry[CopyRegistry];
    IF user = NIL OR password = NIL THEN
      ERROR STPExtra.Denied["Name or Password missing"L];
    -- check passwords from parameter file
    FOR finger: LONG POINTER TO PasswordBlock ← firstPassword, finger.next UNTIL
      finger = NIL DO
      IF String.Equivalent[finger.user, user] THEN
        BEGIN
        matched: BOOLEAN ← Password.Check[password, finger.encrypted];
        IF matched THEN RETURN
        ELSE ERROR STPExtra.Denied["Password rejected by info from parameter file"L];
        END;
      ENDLOOP;
    -- try Grapevine if that fails
    IF myGroup = NIL THEN
      ERROR STPExtra.Denied["Don't know our Grapevine group"L]
    ELSE
      BEGIN
      luser: LONG STRING ← MaybeAppend[user, registry];
      machine: LONG STRING ← MaybeAppend[myGroup, "internet"L];
      status: Password.Status;
      status ← Password.ValidMemberOfGroup[luser, password, machine];
      z.FREE[@machine];
      z.FREE[@luser];
      SELECT status FROM
        yes => NULL;
        nil => ERROR STPExtra.Denied["Confusion about NIL"L];
        allDown =>
          ERROR STPExtra.Denied["All Grapevine servers appear to be down"L];
        notFound =>
          ERROR STPExtra.Denied["Grapevine doesn't like your name"L];
        badPwd =>
          ERROR STPExtra.Denied["Grapevine doesn't like your password"L];
        group =>
          ERROR STPExtra.Denied["Grapevine thinks you are a group"L];
        no =>
          ERROR STPExtra.Denied["You are not in the appropriate group"L];
        notGroup =>
          ERROR STPExtra.Denied["Grapevine doesn't recognize this machine's group"L];
        error =>
          ERROR STPExtra.Denied["Error from GrapevineUser package"L];
        ENDCASE => ERROR;
      END;
    END;

  MaybeAppend: PROCEDURE [string, tail: LONG STRING] RETURNS [new: LONG STRING] =
    BEGIN
    IF tail = NIL THEN RETURN[String.CopyToNewString[string, z]];
    FOR i: CARDINAL IN [0..string.length) DO
      IF string[i] = '. THEN RETURN[String.CopyToNewString[string, z]]; ENDLOOP;
    new ← String.CopyToNewString[string, z, 1 + tail.length];
    String.AppendChar[new, '.];
    String.AppendString[new, tail];
    END;


  Announce: PROCEDURE [one, two: LONG STRING] =
    BEGIN OPEN String;
    text: STRING = [200];
    Time.AppendCurrent[text];
    AppendString[text, "  "L];
    AppendString[text, one];
    AppendString[text, two];
    LogString[text];
    END;

  AnnounceFor: PROCEDURE [one, two: LONG STRING, who, where: LONG STRING] =
    BEGIN OPEN String;
    text: STRING = [200];
    Time.AppendCurrent[text];
    AppendString[text, "  "L];
    AppendString[text, one];
    AppendString[text, two];
    AppendString[text, " for "L];
    AppendString[text, who];
    AppendString[text, " on "L];
    AppendString[text, where];
    LogString[text];
    END;

  Message: PROCEDURE [one, two, three, four: LONG STRING ← NIL] =
    BEGIN
    text: STRING = [100];
    String.AppendString[text, one];
    IF two # NIL THEN String.AppendString[text, two];
    IF three # NIL THEN String.AppendString[text, three];
    IF four # NIL THEN String.AppendString[text, four];
    LogString[text];
    END;

  LogString: PROCEDURE [text: LONG STRING] =
    BEGIN
    String.AppendChar[text, '.];
    String.AppendChar[text, Ascii.CR];
    Put.Text[NIL, text];
    IF msg # NIL THEN Put.Text[msg, text];
    END;

  MakeSWs: Tool.MakeSWsProc =
    BEGIN msg ← Tool.MakeMsgSW[window: window, lines: 5]; END;

  ClientTransition: ToolWindow.TransitionProcType =
    BEGIN IF new = inactive THEN msg ← NIL; END;

  restarting: BOOLEAN ← FALSE;
  Restart: ENTRY PROCEDURE = BEGIN Stopper[]; restarting ← FALSE; Starter[]; END;

  Broom: Supervisor.AgentProcedure =
    BEGIN
    SELECT event FROM
      EventTypes.aboutToBoot, EventTypes.aboutToBootPhysicalVolume =>
        IF running THEN StopperLocked[];
      ENDCASE => NULL;
    END;

  -- Initialization
  tool ← Tool.Create[
    name: herald, makeSWsProc: MakeSWs, clientTransition: ClientTransition];
  FTPServerOn[];
  END.