server/src/Passwd.m3


 Copyright 1999-2003 John D. Polstra.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgment:
 *      This product includes software developed by John D. Polstra.
 * 4. The name of the author may not be used to endorse or promote products
 *    derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 

MODULE Passwd;

IMPORT
  ErrMsg, FileRd, Fmt, Logger, OSError, Pathname, Rd, Text, Thread,
  TokScan;

REVEAL
  DB = BRANDED OBJECT
    path: Pathname.T;
    rd: Rd.T;
    lineNum := 0;
    startIndex: CARDINAL;
    startLineNum: CARDINAL;
    realm: TEXT;
    privateKey: TEXT;
  END;

PROCEDURE Close(db: DB)
  RAISES {Error, Thread.Alerted} =
  BEGIN
    TRY
      Rd.Close(db.rd);
    EXCEPT Rd.Failure(l) =>
      RAISE Error("Cannot close \"" & db.path & "\": " & ErrMsg.StrError(l));
    END;
  END Close;

PROCEDURE Err(db: DB; msg: TEXT)
  RAISES {Error} =
  BEGIN
    RAISE Error(db.path & ":" & Fmt.Int(db.lineNum) & ": " & msg);
  END Err;

PROCEDURE GetPrivateKey(db: DB): TEXT =
  BEGIN
    RETURN db.privateKey;
  END GetPrivateKey;

PROCEDURE GetRealm(db: DB): TEXT =
  BEGIN
    RETURN db.realm;
  END GetRealm;

PROCEDURE GetRec(db: DB): TokScan.T
  RAISES {Error, Rd.EndOfFile, Thread.Alerted} =
  VAR
    line: TEXT;
  BEGIN
    TRY
      REPEAT
	INC(db.lineNum);
	line := TokScan.Trim(Rd.GetLine(db.rd));
      UNTIL Text.Length(line) > 0 AND Text.GetChar(line, 0) # '#';
    EXCEPT Rd.Failure(l) =>
      Err(db, "read error: " & ErrMsg.StrError(l));
    END;
    RETURN TokScan.New(line, SET OF CHAR{':'}, emptyTokens := TRUE);
  END GetRec;

PROCEDURE Lookup(db: DB;
                 client: TEXT;
		 logger: Logger.T := NIL): T
  RAISES {Error, Thread.Alerted} =
  VAR
    startingPoint: CARDINAL;
    ts: TokScan.T;
    found: BOOLEAN;
    ent: T;
  BEGIN
    TRY
      startingPoint := Rd.Index(db.rd);
      ent := NEW(T);
      REPEAT
	TRY
	  ts := GetRec(db);
	  ent.client := ts.getToken("client name");
	  ent.sharedSecret := ts.getToken("shared secret");
	  ent.class := ts.getToken("class");
	  ent.comment := ts.getToken("comment");
	  found := TokScan.EqualFolded(ent.client, client);
	EXCEPT
	| Rd.EndOfFile =>
	    Rd.Seek(db.rd, db.startIndex);
	    db.lineNum := db.startLineNum;
	| TokScan.Error(msg) =>
	    IF logger # NIL THEN
	      Logger.Warning(logger, db.path & ":" & Fmt.Int(db.lineNum) &
		": " & msg);
	    END;
	END;
      UNTIL found OR Rd.Index(db.rd) = startingPoint;
      IF NOT found THEN RETURN NIL END;
    EXCEPT
    | Rd.Failure(l) =>
	RAISE Error("Cannot rewind \"" & db.path & "\": " &
	  ErrMsg.StrError(l));
    END;
    RETURN ent;
  END Lookup;

PROCEDURE Open(path: Pathname.T): DB
  RAISES {OSError.E, Error, Thread.Alerted} =
  VAR
    db: DB;
    ts: TokScan.T;
  BEGIN
    TRY
      db := NEW(DB, path := path, rd := FileRd.Open(path));
      ts := GetRec(db);
      db.startIndex := Rd.Index(db.rd);
      db.startLineNum := db.lineNum;
      db.realm := ts.getToken("server name");
      db.privateKey := ts.getToken("server private key");
      ts.getEnd("end of server record");
    EXCEPT
    | Rd.EndOfFile =>
	RAISE Error(path & ": no server record in file");
    | TokScan.Error(msg) =>
	Err(db, msg);
    END;
    RETURN db;
  END Open;

BEGIN
END Passwd.