;;; no-littering.el --- help keeping ~/.emacs.d clean
;; Copyright (C) 2016-2018 Jonas Bernoulli
;; Author: Jonas Bernoulli <jonas@bernoul.li>
;; Homepage: https://github.com/emacscollective/no-littering
;; Package-Requires: ((cl-lib "0.5"))
;; This file is not part of GNU Emacs.
;; This file is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 3, or (at your option)
;; any later version.
;; This file is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; For a full copy of the GNU General Public License
;; see https://www.gnu.org/licenses.
;;; Commentary:
;; Help keeping ~/.emacs.d clean.
;; The default paths used to store configuration files and persistent
;; data are not consistent across Emacs packages. This isn't just a
;; problem with third-party packages but even with built-in packages.
;; Some packages put these files directly in `user-emacs-directory'
;; or $HOME or in a subdirectory of either of the two or elsewhere.
;; Furthermore sometimes file names are used that don't provide any
;; insight into what package might have created them.
;; This package sets out to fix this by changing the values of path
;; variables to put configuration files in `no-littering-etc-directory'
;; (defaulting to "~/.emacs.d/etc/") and persistent data files in
;; `no-littering-var-directory' (defaulting to "~/.emacs.d/var/"), and
;; by using descriptive file names and subdirectories when appropriate.
;; This is similar to a color-theme; a "path-theme" if you will.
;; We still have a long way to go until most built-in and many third-
;; party path variables are properly "themed". Like a color-theme,
;; this package depends on user contributions to accomplish decent
;; coverage. Pull requests are highly welcome (but please follow the
;; conventions described below and in the pull request template).
;; Usage:
;; Load the feature `no-littering' as early as possible in your init
;; file. Make sure you load it at least before you change any path
;; variables using some other method.
;;
;; (require 'no-littering)
;; If you would like to use base directories different from what
;; `no-littering' uses by default, then you have to set the respective
;; variables before loading the feature.
;;
;; (setq no-littering-etc-directory
;; (expand-file-name "config/" user-emacs-directory))
;; (setq no-littering-var-directory
;; (expand-file-name "data/" user-emacs-directory))
;; (require 'no-littering)
;; Conventions:
;; * A) File names
;;
;; 1. File names are based on the name of the respective Emacs Lisp
;; variables and the name of the respective Emacs package.
;;
;; 2. The name of the respective Emacs package should serve as the
;; prefix of the file name, unless the file is in a subdirectory in
;; which case the name of the subdirectory serves as the prefix.
;;
;; 3. If the name of the package and the prefix of the variable do not
;; match, then we prefer the name of the package.
;;
;; 4. If the name of a path variable ends with `-file`, `-default-file`,
;; `-directory`, `-default-directory`, or something similar, then that
;; suffix is usually dropped from the file name.
;;
;; 5. If applicable, the appropriate extension is added to the file name
;; so that files are visited using the appropriate major-modes and
;; also to provide a hint about the kind of data stored in the file.
;; E.g. if a file contains an S-expression, then the suffix should be
;; `*.el`.
;; * B) File location and subdirectories
;;
;; 1. If a package has only one data file, then that is usually placed in
;; `no-littering-var-directory` itself. Likewise if a package has
;; only one config file, then that is placed in
;; `no-littering-etc-directory` itself.
;;
;; 2. If a package has multiple data (or config files), then those files
;; are placed in a subdirectory of `no-littering-var-directory` (or
;; `no-littering-etc-directory`).
;;
;; 3. If a subdirectory is used for a package's data (or config) file
;; variables, then the name of the directory should match the name of
;; the package in most cases. The subdirectory name may serve as the
;; package prefix of the file name.
;;
;; 4. If a package provides a "framework" for other packages to use,
;; then we may reuse its directories for other packages that make use
;; of that framework or otherwise "extend" the "main package".
;; E.g. we place all `helm` related files in `helm/`.
;;
;; 5. If a package only defines a single variable that specifies a data
;; (or config) directory, then the directory name should
;; nevertheless be just the package name. E.g. the path used for
;; `sx-cache-directory` from the `sx` package is `sx/cache/`, not
;; `sx-cache/`.
;;
;; 6. However if the name of the directory variable implies that the
;; package won't ever define any data (or config) files that won't be
;; placed in that directory, then we use a top-level directory. E.g.
;; when the name of the variable is `<package>-directory`, in which
;; case we would use just `<package>/` as the path.
;; * C) Ordering and alignment
;;
;; 1. The code that sets the values of themed variables is split into two
;; groups. The first group sets the value of variables that belong to
;; packages that are part of Emacs, and the second group is used for
;; variables that are defined by packages that are not part of Emacs.
;;
;; 2. Each of these lists is sorted alphabetically (usually by variable
;; name). Please keep it that way.
;;
;; 3. We attempt to align the value forms inside different `setq` forms.
;; If the symbol part for a particular variable is too long to allow
;; doing so, then don't worry about it and just break the alignment.
;; If it turns out that this happens very often, then we will adjust
;; the alignment eventually.
;; * D) Commit messages
;;
;; 1. Please theme each package using a separate commit and use commit
;; messages of the form "<package>: theme <variable".
;;
;; 2. If a package has several path variables, then you should theme them
;; all in one commit.
;;
;; 3. If the variable names do not fit nicely on the summary line, then
;; use a message such as:
;;
;; foo: theme variables
;;
;; Theme `foo-config-file', `foo-cache-directory',
;; and `foo-persistent-file'.
;; 4. When appropriate add statements like the following to the commit
;; message:
;;
;; - This file is used to store an s-expression.
;; - This file is used to store raw text.
;; - This is the only configuration/data file of the package.
;; - This package does/doesn't take care of creating the containing
;; directory if necessary. (If the package does not do it, then you
;; should also fix that and submit an upstream pull request.)
;;
;; 5. If you are uncertain, then be explicit about it by adding a comment
;; to the pull-request.
;;; Code:
(require 'cl-lib)
(defvar no-littering-etc-directory
(expand-file-name (convert-standard-filename "etc/") user-emacs-directory)
"The directory where packages place their configuration files.
This variable has to be set before `no-littering' is loaded.")
(defvar no-littering-var-directory
(expand-file-name (convert-standard-filename "var/") user-emacs-directory)
"The directory where packages place their persistent data files.
This variable has to be set before `no-littering' is loaded.")
(defun no-littering-expand-etc-file-name (file)
"Expand filename FILE relative to `no-littering-etc-directory'."
(expand-file-name (convert-standard-filename file)
no-littering-etc-directory))
(defun no-littering-expand-var-file-name (file)
"Expand filename FILE relative to `no-littering-var-directory'."
(expand-file-name (convert-standard-filename file)
no-littering-var-directory))
(cl-letf (((symbol-function 'etc)
(symbol-function #'no-littering-expand-etc-file-name))
((symbol-function 'var)
(symbol-function #'no-littering-expand-var-file-name)))
(with-no-warnings ; many of these variables haven't been defined yet
;;; Built-in packages
(setq abbrev-file-name (var "abbrev.el"))
(setq auto-insert-directory (etc "auto-insert/"))
(setq auto-save-list-file-prefix (var "auto-save/sessions/"))
(setq backup-directory-alist (list (cons "." (var "backup/"))))
(setq bookmark-default-file (var "bookmark-default.el"))
(eval-after-load 'desktop
`(make-directory ,(var "desktop/") t))
(setq desktop-dirname (var "desktop/"))
(setq desktop-path (list desktop-dirname))
(setq diary-file (var "diary"))
(setq epkg-repository (var "epkgs/"))
(setq eshell-directory-name (var "eshell/"))
(eval-after-load 'eww
`(make-directory ,(var "eww/") t))
(setq eww-bookmarks-directory (var "eww/"))
(setq gamegrid-user-score-file-directory (var "gamegrid-user-score/"))
(setq ido-save-directory-list-file (var "ido-save-directory-list.el"))
(setq image-dired-db-file (var "image-dired/db.el"))
(setq image-dired-dir (var "image-dired/"))
(setq image-dired-gallery-dir (var "image-dired/gallery/"))
(setq image-dired-temp-image-file (var "image-dired/temp-image"))
(setq image-dired-temp-rotate-image-file (var "image-dired/temp-rotate-image"))
(setq kkc-init-file-name (var "kkc-init.el"))
(eval-after-load 'newsticker
`(make-directory ,(var "newsticker/") t))
(setq newsticker-cache-filename (var "newsticker/cache.el"))
(setq newsticker-dir (var "newsticker/data/"))
(setq nsm-settings-file (var "nsm-settings.el"))
(eval-after-load 'org
`(make-directory ,(var "org/") t))
(setq omnisharp-cache-directory (var "omnisharp/cache"))
(setq org-clock-persist-file (var "org/clock-persist.el"))
(setq org-id-locations-file (var "org/id-locations.el"))
(setq org-publish-timestamp-directory (var "org/timestamps/"))
(setq org-registry-file (var "org/registry.el"))
(setq recentf-save-file (var "recentf-save.el"))
(setq remember-data-file (var "remember/data"))
(setq remember-data-directory (var "remember/data.d/"))
(setq save-place-file (var "save-place.el"))
(setq savehist-file (var "savehist.el"))
(setq semanticdb-default-save-directory (var "semantic/"))
(setq shared-game-score-directory (var "shared-game-score/"))
(setq tramp-auto-save-directory (var "tramp/auto-save/"))
(setq tramp-persistency-file-name (var "tramp/persistency.el"))
(setq url-cache-directory (var "url/cache/"))
(setq url-configuration-directory (var "url/configuration/"))
;;; Third-party packages
(setq ac-comphist-file (var "ac-comphist.el"))
(setq amx-save-file (var "amx-save.el"))
(setq anaconda-mode-installation-directory (var "anaconda-mode/"))
(setq async-byte-compile-log-file (var "async-bytecomp.log"))
(eval-after-load 'bbdb
`(make-directory ,(var "bbdb/") t))
(setq bbdb-file (var "bbdb/bbdb.el"))
(setq bbdb-vcard-directory (var "bbdb/vcard/"))
(setq bm-repository-file (var "bm-repository.el"))
(eval-after-load 'bookmark+-1
`(make-directory ,(var "bmkp/") t))
(setq bmkp-current-bookmark-file (var "bmkp/current-bookmark.el"))
(setq bmkp-last-bookmark-file (var "bmkp/last-bookmark.el"))
(eval-after-load 'company-statistics
`(make-directory ,(var "company/") t))
(setq company-statistics-file (var "company/statistics.el"))
(setq dired-recent-directories-file (var "dired-recent-directories.el"))
(setq elbank-data-file (var "elbank-data.el"))
(eval-after-load 'elfeed
`(make-directory ,(var "elfeed/") t))
(setq elfeed-db-directory (var "elfeed/db/"))
(setq elfeed-enclosure-default-dir (var "elfeed/enclosures/"))
(eval-after-load 'x-win
(let ((session-dir (var "emacs-session/")))
`(progn
(make-directory ,session-dir t)
(defun emacs-session-filename (session-id)
"Construct a filename to save the session in based on SESSION-ID.
This function overrides the one on `x-win' to use `no-littering'
directories."
(expand-file-name session-id ,session-dir)))))
(setq emms-directory (var "emms/"))
(eval-after-load 'emojify
`(make-directory ,(var "emojify/") t))
(setq emojify-emojis-dir (var "emojify/"))
(setq forge-database-file (var "forge/database.sqlite"))
(setq forge-post-directory (var "forge/posts/"))
(setq geben-temporary-file-directory (var "geben/"))
(eval-after-load 'geiser
`(make-directory ,(var "geiser/") t))
(setq geiser-repl-history-filename (var "geiser/repl-history"))
(setq hackernews-visited-links-file (var "hackernews/visited-links.el"))
(eval-after-load 'helm
`(make-directory ,(var "helm/") t))
(setq helm-adaptive-history-file (var "helm/adaptive-history.el"))
(setq helm-backup-path (var "helm/backup/"))
(setq helm-github-stars-cache-file (var "helm/github-stars-cache.el"))
(setq historian-save-file (var "historian-save.el"))
(setq indium-workspace-file (var "indium/workspaces.el"))
(setq irfc-directory (var "irfc/"))
(setq irony-user-dir (var "irony/"))
(setq jabber-avatar-cache-directory (var "jabber/avatar-cache"))
(eval-after-load 'jabber
`(make-directory ,(var "jabber/avatar-cache/") t))
(setq jabber-history-dir (var "jabber/history"))
(eval-after-load 'jabber
`(make-directory ,(var "jabber/history/") t))
(setq keyfreq-file (var "keyfreq.el"))
(setq keyfreq-file-lock (var "keyfreq.lock"))
(eval-after-load 'lookup
`(make-directory ,(etc "lookup/") t))
(setq litable-list-file (var "litable-list.el"))
(setq lookup-init-directory (etc "lookup/"))
(setq magithub-dir (var "magithub/"))
(setq magithub-cache-file (var "magithub/cache.el"))
(setq mc/list-file (var "mc-list.el"))
(setq meghanada-server-install-dir (var "meghanada/"))
(setq multi-compile-history-file (var "multi-compile-history.el"))
;; The value of this variable MUST NOT end with ".el" but the
;; actual file name MUST end with ".el". Use "git blame" for
;; more information.
(setq notmuch-init-file (etc "notmuch-init"))
(setq nov-save-place-file (var "nov-save-place.el"))
(setq org-gcal-dir (var "org/gcal/"))
(eval-after-load 'org-caldav
`(make-directory ,(var "org/caldav/save") t))
(setq org-caldav-backup-file (var "org/caldav/backup.org"))
(setq org-caldav-save-directory (var "org/caldav/save"))
(setq pandoc-data-dir (etc "pandoc-mode/"))
(setq pcache-directory (var "pcache/"))
(setq persistent-scratch-save-file (var "persistent-scratch.el"))
(setq persp-save-dir (var "persp-mode/"))
(eval-after-load 'projectile
`(make-directory ,(var "projectile/") t))
(setq projectile-cache-file (var "projectile/cache.el"))
(setq projectile-known-projects-file (var "projectile/known-projects.el"))
(setq purpose-default-layout-file (etc "window-purpose/default-layout.el"))
(setq purpose-layout-dirs (list (etc "window-purpose/layouts/")))
(setq pyim-dcache-directory (var "pyim/dcache/"))
(setq quack-dir (var "quack/"))
(setq request-storage-directory (var "request/storage/"))
(setq rmh-elfeed-org-files (list (var "elfeed/rmh-elfeed.org")))
(setq runner-init-file (var "runner-init.el"))
(setq save-kill-file-name (var "save-kill.el"))
(setq save-visited-files-location (var "save-visited-files-location"))
(setq smex-save-file (var "smex-save.el"))
(setq speed-type-gb-dir (var "speed-type/"))
(eval-after-load 'sx
`(make-directory ,(var "sx/cache/") t))
(setq sx-cache-directory (var "sx/cache/"))
(setq tldr-directory-path (var "tldr/"))
(setq transient-values-file (etc "transient/values.el"))
(setq treemacs-persist-file (var "treemacs-persist.org"))
(setq undo-tree-history-directory-alist (list (cons "." (var "undo-tree-hist/"))))
(setq user-emacs-ensime-directory (var "ensime/"))
(setq vimish-fold-dir (var "vimish-fold/"))
(eval-after-load 'xkcd
`(make-directory ,(var "xkcd/") t))
(setq xkcd-cache-dir (var "xkcd/"))
(eval-after-load 'yasnippet
`(make-directory ,(etc "yasnippet/snippets/") t))
(setq yas-snippet-dirs (list (etc "yasnippet/snippets/")))
))
;;; _
(provide 'no-littering)
;; Local Variables:
;; indent-tabs-mode: nil
;; End:
;;; no-littering.el ends here