XDG Basedir Library
William Hatch <william@hatch.uno>
| (require basedir) | package: basedir | 
1 XDG Basedir Specification
This library provides functions for easily accessing configuration and data files according to the XDG Base Directory Specification.
The point of the XDG base directory is to allow programs to put configuration and data files on the filesystem in a consistent and user-configurable way. It solves the problem of having many different configuration and data files and directories (dotfiles) in your home directory.
An important point of the XDG specification is that it specifies a single directory for writable configuration files ($XDG_CONFIG_HOME) and a list of directories ($XDG_CONFIG_DIRS) for non-writable configuration files. If a program can modify its own configuration, it should do so only using the writable directory, and not change the others. Similarly $XDG_DATA_HOME and $XDG_DATA_DIRS exist for the same purpose for data files. There is only one directory for cache files ($XDG_CACHE_HOME), since the whole point of cache files is that they are for the program to write.
Having multiple configuration or data file directories is useful. Consider the common scenario of keeping a synchronized directory of configuration data (with git, dropbox, or the like), as well as local configuration for each machine, and potentially different directories for public and private synchronized configuration. So programs should read all available configuration files and compose the configurations, where possible.
- $XDG_CONFIG_HOME = $HOME/.config/ (on Windows: %LOCALAPPDATA%) 
- $XDG_CONFIG_DIRS = /etc/xdg/ (on Windows: %APPDATA%) 
- $XDG_DATA_HOME = $HOME/.local/share/ (on Windows: %LOCALAPPDATA%) 
- $XDG_DATA_DIRS = /usr/local/share/:/usr/share/ (on Windows: %APPDATA%) 
- $XDG_CACHE_HOME = $HOME/.cache (on Windows: %TEMP%) 
- $XDG_RUNTIME_DIR is a little more complicated. On Unixy systems it checks whether /run/user/$(id -u) exists and uses it if it does (systems with systemd and its PAM module have these directories created automatically for this purpose). Otherwise it falls back to /tmp/user-$USER. On Windows it currently defaults to %TEMP%, because I am unaware of a standard directory with semantics more like XDG_RUNTIME_DIR is supposed to have. 
Some words of warning about the runtime dir: The runtime dir should be used for storing ephemeral things like pipes and sockets or other objects restricted to the current run of the program. It is likely created when a user logs in and deleted when the user logs out. It probably lives in a RAM-based file system. The spec says that it may be cleaned of old files occasionally, and to keep your files from being cleaned up you should either update their timestamps regularly (every 6 hours is what they consider regularly) or set the sticky bit on the files. The directory should be readable only by the owner.
To learn more about the XDG basedir specification, visit https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
2 Reference
parameter
(current-basedir-program-name name) → void? name : path-string? 
This parameter is provided for convenience, so that you can set your program’s XDG path name once and not put it in every call to other basedir functions.
The default value of this parameter is NOT guaranteed to remain stable. A program that wants to have configuration files should be setting its name so the user can see it.
procedure
(list-config-files file-name [ #:program program-name #:only-existing? only-existing?]) → (listof path?) file-name : path-string? program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-data-files file-name [ #:program program-name #:only-existing? only-existing?]) → (listof path?) file-name : path-string? program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-cache-files file-name [ #:program program-name #:only-existing? only-existing?]) → (listof path?) file-name : path-string? program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-runtime-files file-name [ #:program program-name #:only-existing? only-existing?]) → (listof path?) file-name : path-string? program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
For example, let’s say you write a program named "foo" and your main config file is named "foorc". Bobby (a Debian user) has $XDG_CONFIG_HOME unset (IE at its default value) and $XDG_CONFIG_DIRS set to /home/bobby/config-git:/home/bobby/config-git-private:/home/bobby/local-config and the files /home/bobby/.config/foo/foorc, /home/bobby/config-git/foo/foorc, and /home/bobby/local-config/foo/foorc exist (but not /home/bobby/config-git-private/foo/foorc), then (list-config-files "foorc" #:program "foo") will return a list of paths for those three files that exist on the path. The foo program should, if possible, use the pieces of configuration found in each of the files.
Also note that file-name may actually be a path (eg. subdir/filename), in case you want further nesting in your configuration or data directory.
Since there is only one cache directory and one runtime directory, list-cache-files and list-runtime-files will return either a list with one element or an empty list.
procedure
(writable-config-file file-name [ #:program program-name]) → path? file-name : path-string? program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-data-file file-name [ #:program program-name]) → path? file-name : path-string? program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-cache-file file-name [ #:program program-name]) → path? file-name : path-string? program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-runtime-file file-name [ #:program program-name]) → path? file-name : path-string? program-name : path-string? = (current-basedir-program-name) 
Following the example from above with Bobby and the foo program, while several "foorc" files exist, the only one that should be written to is /home/bobby/.config/foo/foorc. That path would be returned by (writable-config-file "foorc" #:program "foo")
procedure
(list-config-dirs [ #:program program-name #:only-existing? only-existing?]) → (listof path?) program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-data-dirs [ #:program program-name #:only-existing? only-existing?]) → (listof path?) program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-cache-dirs [ #:program program-name #:only-existing? only-existing?]) → (listof path?) program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(list-runtime-dirs [ #:program program-name #:only-existing? only-existing?]) → (listof path?) program-name : path-string? = (current-basedir-program-name) only-existing? : any/c = #t 
procedure
(writable-config-dir [#:program program-name]) → path?
program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-data-dir [#:program program-name]) → path?
program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-cache-dir [#:program program-name]) → path?
program-name : path-string? = (current-basedir-program-name) 
procedure
(writable-runtime-dir [#:program program-name]) → path?
program-name : path-string? = (current-basedir-program-name) 
3 Code and License
The code is available on github.
This library is distributed under the MIT license and the Apache version 2.0 license, at your option.