Home My Page Projects Code Snippets Project Openings SML/NJ
Summary Activity Forums Tracker Lists Tasks Docs Surveys News SCM Files

SCM Repository

[smlnj] View of /eXene/releases/release-110.74/graph-util/xauth.sml
ViewVC logotype

View of /eXene/releases/release-110.74/graph-util/xauth.sml

Parent Directory Parent Directory | Revision Log Revision Log


Revision 3683 - (download) (annotate)
Sun Jan 22 05:04:52 2012 UTC (7 years, 7 months ago) by dbm
File size: 9090 byte(s)
Tagging 110.74 release of eXene
(* xauth.sml
 *
 * COPYRIGHT (c) 1996 AT&T Research.
 *
 * Support for X11 authentication.  The authentication file, which is
 * specified by the XAUTHORITY variable (default $HOME/.Xauthority),
 * consists of a sequence of entries with the following format:
 *
 *      2 bytes         Family value (second byte is as in protocol HOST)
 *      2 bytes         address length (always MSB first)
 *      A bytes         host address (as in protocol HOST)
 *      2 bytes         display "number" length (always MSB first)
 *      S bytes         display "number" string
 *      2 bytes         name length (always MSB first)
 *      N bytes         authorization name string
 *      2 bytes         data length (always MSB first)
 *      D bytes         authorization data string
 *
 * This implementation is partially based on code contributed by Juergen Buntrock.
 *)

structure XAuth : X_AUTH =
  struct

    structure EXB = EXeneBase

    val w8vextract = Word8VectorSlice.vector o Word8VectorSlice.slice

    val get8 = Word8.toInt o Word8Vector.sub
  (* this version of get16 handles unaligned data *)
    fun get16 (s, i) = let
          val s = w8vextract (s, i, SOME 2)
          in
            LargeWord.toInt(PackWord16Big.subVec(s, 0))
          end
    fun getData (s, i, n) = w8vextract (s, i, SOME n)
    fun getString (s, i, n) =
        Byte.unpackStringVec (Word8VectorSlice.slice (s, i, SOME n))
    (* added ddeboer: ip addresses are encoded as raw bytes not a string. *)
    fun getAddressString (s, i, n) =
        case (n,(Word8VectorSlice.foldr (fn (e,a) => 
            (Int.toString (Char.ord (Byte.byteToChar e)))::a) []
                (Word8VectorSlice.slice (s, i, SOME n)))) of
           (4,[a,b,c,d]) => a^"."^b^"."^c^"."^d
          | _ => ""
        
  (* the different family codes (from X.h and xc/lib/Xau/Xauth.h) *)
    val familyInternet  = 0
    val familyDECnet    = 1
    val familyChaos = 2
    val familyLocal = 256
    val familyWild  = 65535

  (* return the default name of the authentication file (either
   * specified by the XAUTHORITY environment variable, or the
   * file $HOME/.Xauthority.  If neither XAUTHORITY or HOME 
   * are defined, then ".Xauthority" is returned.
   *)
    fun authFileName () = (case (OS.Process.getEnv "XAUTHORITY")
       of (SOME fname) => fname
        | NONE => (case (OS.Process.getEnv "HOME")
         of (SOME path) => path ^ "/.Xauthority"
          | NONE => ".Xauthority"
        (* end case *))
      (* end case *))

  (* read the entire contents of a file *)
    fun readFile file = let
      val instrm = BinIO.openIn file
      val contents = BinIO.inputAll instrm
      in
        BinIO.closeIn instrm;
        contents
      end

  (* extract an authentication entry from a data string *)
    fun extractAuth contents = let
      val len = Word8Vector.length contents
      fun getLen start = get16(contents, start-2)
      fun extract offset = if (offset < len)
        then let
          val addrStart = 4 + offset
          val addrLen = getLen addrStart
          val dpyStart = addrStart + addrLen + 2
          val dpyLen = getLen dpyStart
          val nameStart = dpyStart + dpyLen + 2
          val nameLen = getLen nameStart
          val dataStart = nameStart + nameLen + 2
          val dataLen = getLen dataStart
          val next = dataStart + dataLen
          (* added following line, Feb 2005, ddeboer *)
          val family = get16 (contents, offset)
          in
            SOME(EXB.AUTH{
            (* modified by ddeboer: 
             * entries of family=familyInternet are stored as 4-byte ip addresses.
            original:
            family = get16 (contents, offset),
            addr = getString (contents, addrStart, addrLen),*) 
            family = family,
            addr = (if (family = familyInternet) 
                then (getAddressString (contents,addrStart,addrLen))
                (* (let val str = getAddressString(contents,addrStart,addrLen) in
                    (case (NetHostDB.fromString str) of
                        NONE => ""
                      | SOME ia => (case (NetHostDB.getByAddr ia) of 
                                        NONE => "" 
                                      | SOME e => (NetHostDB.name e))) end) *)
                else (getString (contents, addrStart, addrLen)) ),
            (* end modification *)
            dpy = getString (contents, dpyStart, dpyLen),
            name = getString (contents, nameStart, nameLen),
            data = getData (contents, dataStart, dataLen)
              }, next)
          end
        else NONE
      in
        extract
      end

  (* searches the default authentication file for the first entry that
   * matches the family, network address and display number.  If no
   * such match is found, then NONE is returned.  The * value familyWild
   * matches anything, as do the empty strings when given for addr or dpy.
   *)
    fun getAuthByAddr {family, dpy, addr} = let
      val extractAuth = extractAuth (readFile (authFileName()))
      (* hack by ddeboer, Feb 2005 - this is surely not the right way to do this...?? 
         if family is internet and address is localhost, change to the local hostname
         and familyLocal. 
      val (family,addr) = if (((addr="localhost")) andalso family=familyInternet) 
                            then (familyLocal,NetHostDB.getHostName())
                            else (family,addr) *)
      (* end hack *)
      fun cmpStr ("", _) = true
        | cmpStr (_, "") = true
        | cmpStr (a, b) = (a = b)
      fun chkAuth (EXB.AUTH{family=f, dpy=d, addr=a, ...}) = (
       (* tracing added ddeboer, Jan 2005. 
       (TextIO.print ("chkAuth seeking family="^(Int.toString(family))^", dpy="
            ^dpy^", addr="^addr^"; examining addr="^a^",dpy="^d^"\n"));*)
        ((family = familyWild) orelse (f = familyWild) orelse (family = f))
        andalso cmpStr(dpy, d)
        andalso cmpStr(addr, a))
      fun look offset = (case (extractAuth offset)
         of NONE => NONE
          | (SOME(auth, next)) =>
              if (chkAuth auth) then (SOME auth) else look next
        (* end case *))
      in
        (* look 0 *)
        (* modified ddeboer, Jan 2005 for testing. *)
        (let val rv = look 0 in 
            (case rv of (SOME (EXB.AUTH{addr,dpy,name,data,...})) =>
                ((*TextIO.print ("getAuthByAddr returns SOME EXB.AUTH{addr="^
                    addr^",dpy="^dpy^",name="^name^",...}\n")*))
                | NONE => ((*TextIO.print ("getAuthByAddr returns NONE (addr="^
                    addr^",dpy="^dpy^")\n")*))); rv end)
      end
        handle _ => NONE

  (* this similar to getAuthByAddr, except that a list of acceptable
   * authentication methods is specified by the list authNames.  It
   * returns the matching authentication info that matches the earliest
   * name on the list.  NONE is returned if no match is found.
   *)
    fun getBestAuthByAddr {family, addr, dpy, authNames} = let
      val extractAuth = extractAuth (readFile (authFileName()))
      (* hack by ddeboer, Feb 2005 - this is surely not the right way to do this...?? 
         if family is internet and address is localhost, change to the local hostname
         and familyLocal. 
      val (family,addr) = if (((addr="localhost") ) andalso family=familyInternet) 
                            then (familyLocal,NetHostDB.getHostName())
                            else (family,addr)*)
      (* end hack *)
      fun cmpStr ("", _) = true
        | cmpStr (_, "") = true
        | cmpStr (a, b) = (a = b)
      fun chkAuth (EXB.AUTH{family=f, dpy=d, addr=a, ...}) = (
        ((family = familyWild) orelse (f = familyWild) orelse (family = f))
        andalso cmpStr(dpy, d)
        andalso cmpStr(addr, a))
      fun look (offset, bestRank, best) = (case (extractAuth offset)
         of NONE => best
          | (SOME(auth as EXB.AUTH{name, ...}, next)) =>
              if (chkAuth auth)
                then let
              fun chkName ([], _) = look (next, bestRank, best)
                | chkName (n::r, rank) =
                    if (rank < bestRank)
                  then if (name = n)
                    then look (next, rank, SOME auth)
                    else chkName (r, rank+1)
                  else look (next, bestRank, best)
              in
                chkName (authNames, 0)
              end
                else look (next, bestRank, best)
        (* end case *))
      in
        look (0, length authNames, NONE)
      end
        handle _ => NONE

  (* read the specified authentication file and return a list of
   * entries that satisfy the given predicate.
   *)
    fun readAuthFile checkAuth file = let
      val extractAuth = extractAuth (readFile file)
      fun filter (offset, l) = (case (extractAuth offset)
         of NONE => rev l
          | (SOME(auth, next)) => if (checkAuth auth)
              then filter (next, auth::l)
              else filter (next, l)
        (* end case *))
      in
        filter (0, [])
      end

  end;


root@smlnj-gforge.cs.uchicago.edu
ViewVC Help
Powered by ViewVC 1.0.0