That said, most of the functions that interact with the file system are still pretty straightforward. I’ll discuss the standard functions here and point out the ones that suffer from nonportability between implementations. In the next chapter you’ll develop a pathname portability library to smooth over some of those nonportability issues.

    To test whether a file exists in the file system corresponding to a pathname designator—a pathname, namestring, or file stream—you can use the function **PROBE-FILE**. If the file named by the pathname designator exists, **PROBE-FILE** returns the file’s truename, a pathname with any file system-level translations such as resolving symbolic links performed. Otherwise, it returns **NIL**. However, not all implementations support using this function to test whether a directory exists. Also, Common Lisp doesn’t provide a portable way to test whether a given file that exists is a regular file or a directory. In the next chapter you’ll wrap **PROBE-FILE** with a new function, file-exists-p, that can both test whether a directory exists and tell you whether a given name is the name of a file or directory.

    **DELETE-FILE** and **RENAME-FILE** do what their names suggest. takes a pathname designator and deletes the named file, returning true if it succeeds. Otherwise it signals a **FILE-ERROR**.11

    **RENAME-FILE** takes two pathname designators and renames the file named by the first name to the second name.

    Note that if you pass **ENSURE-DIRECTORIES-EXIST** a directory name, it should be in directory form, or the leaf directory won’t be created.

    The functions **FILE-WRITE-DATE** and **FILE-AUTHOR** both take a pathname designator. **FILE-WRITE-DATE** returns the time in number of seconds since midnight January 1, 1900, Greenwich mean time (GMT), that the file was last written, and **FILE-AUTHOR** returns, on Unix and Windows, the file owner.12

      A related function that also takes an open file stream as its argument is **FILE-POSITION**. When called with just a stream, this function returns the current position in the file—the number of elements that have been read from or written to the stream. When called with two arguments, the stream and a position designator, it sets the position of the stream to the designated position. The position designator must be the keyword :start, the keyword :end, or a non-negative integer. The two keywords set the position of the stream to the start or end of the file while an integer moves to the indicated position in the file. With a binary stream the position is simply a byte offset into the file. However, for character streams things are a bit more complicated because of character-encoding issues. Your best bet, if you need to jump around within a file of textual data, is to only ever pass, as a second argument to the two-argument version of **FILE-POSITION**, a value previously returned by the one-argument version of **FILE-POSITION** with the same stream argument.