chiark / gitweb /
safely.lisp: More CLisp fixing.
[lisp] / safely.lisp
1 ;;; -*-lisp-*-
2 ;;;
3 ;;; $Id$
4 ;;;
5 ;;; Safely modify collections of files
6 ;;;
7 ;;; (c) 2005 Straylight/Edgeware
8 ;;;
9
10 ;;;----- Licensing notice ---------------------------------------------------
11 ;;;
12 ;;; This program is free software; you can redistribute it and/or modify
13 ;;; it under the terms of the GNU General Public License as published by
14 ;;; the Free Software Foundation; either version 2 of the License, or
15 ;;; (at your option) any later version.
16 ;;;
17 ;;; This program is distributed in the hope that it will be useful,
18 ;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
19 ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20 ;;; GNU General Public License for more details.
21 ;;;
22 ;;; You should have received a copy of the GNU General Public License
23 ;;; along with this program; if not, write to the Free Software Foundation,
24 ;;; Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
25
26 (defpackage #:safely
27   (:use #:common-lisp #:mdw.base)
28   (:export #:safely #:safely-close #:safely-delete-file
29            #:safely-open-output-stream #:safely-bail #:safely-commit
30            #:safely-writing))
31 (in-package #:safely)
32
33 #+(or cmu sbcl)
34 (eval-when (:compile-toplevel :execute)
35   (import #+cmu '(ext:unix-namestring unix:unix-link)
36           #+sbcl '(sb-int:unix-namestring)))
37
38 (defstruct (safely (:predicate safelyp))
39   "Stores information about how to commit or undo safe writes."
40   (streams nil)
41   (trail nil))
42
43 (defun safely-close (safe stream)
44   "Make sure that STREAM is closed when SAFE is finished."
45   (push stream (safely-streams safe)))
46
47 (defun safely-delete-file (safe file)
48   "Delete FILE when SAFE is committed."
49   (push `(:delete ,file) (safely-trail safe)))
50
51 (defun generate-fresh-file-name (base tag &optional func)
52   "Return a fresh file name constructed from BASE (a filespec) and TAG (some
53    short descriptive string).  The generated name has the same directory and
54    type as the BASE name, but a different name.
55
56    If FUNC is non-nil, then it is a function to call on the generated file
57    name: generate-fresh-file-name runs in a loop, calling FUNC with generated
58    file names until FUNC returns non-nil, at which point generate-fresh-
59    file-name returns two values: the generated name, and the result of FUNC.
60    generate-fresh-file-name catches errors of type file-error from FUNC, and
61    just tries again with a new name.
62
63    If FUNC is nil, it's treated the same as a function which always returns
64    t.
65
66    This is inspired by a similar facility in scsh."
67   (let ((base (pathname base)))
68     (dotimes (i 256
69               (error "Gave up trying to find a temporary ~A file for ~S."
70                      tag base))
71       (let* ((new (merge-pathnames
72                    (make-pathname
73                     :name (format nil "~A-~A-~X"
74                                   (pathname-name base)
75                                   tag
76                                   (random most-positive-fixnum)))
77                    base))
78              (ret (and (not (probe-file new))
79                        (if func
80                            (handler-case (funcall func new)
81                              (file-error (cond)
82                                (unless (pathname-match-p
83                                         (file-error-pathname cond)
84                                         new)
85                                  (error cond))
86                                nil))
87                            t))))
88         (when ret
89           (return (values new ret)))))))
90
91 (defun safely-open-output-stream (safe file &rest open-args)
92   "Create an output stream which will be named FILE when SAFE is committed.
93    Other OPEN-ARGS are passed to open."
94   (multiple-value-bind
95       (name stream)
96       (generate-fresh-file-name file "new"
97                                 (lambda (name)
98                                   (apply #'open name
99                                          :direction :output
100                                          :if-exists nil
101                                          open-args)))
102     (safely-close safe stream)
103     (push `(:shunt ,name ,file)
104           (safely-trail safe))
105     stream))
106
107 (declaim (inline rename))
108 (defun rename (old new)
109   #-clisp (rename-file old new)
110   #+clisp (posix:copy-file old new :method :rename))
111
112 (defun delete-file-without-moaning (file)
113   "Delete the FILE, ignoring errors."
114   (handler-case (delete-file file)
115     (file-error () nil)))
116
117 (defun rename-file-without-moaning (old new)
118   "Rename OLD to NEW, ignoring errors, and without doing any stupid name
119    mangling."
120   (handler-case (rename old new)
121     (file-error () nil)))
122
123 (defun safely-unwind (trail)
124   "Roll back the TRAIL of operations."
125   (dolist (job trail)
126     (ecase (car job)
127       (:shunt (destructuring-bind (new file) (cdr job)
128                 (declare (ignore file))
129                 (delete-file-without-moaning new)))
130       (:delete)
131       (:rmtmp (destructuring-bind (file) (cdr job)
132                 (delete-file-without-moaning file)))
133       (:revert (destructuring-bind (old new) (cdr job)
134                  (rename-file-without-moaning old new))))))
135
136 (defun safely-reset (safe)
137   "Reset SAFE to its initial state."
138   (setf (safely-streams safe) nil)
139   (setf (safely-trail safe) nil))
140
141 (defun safely-bail (safe)
142   "Abort the operations in SAFE, unwinding all the things that have been
143    done.  Streams are closed, new files are removed."
144   (dolist (stream (safely-streams safe))
145     (close stream :abort t))
146   (safely-unwind (safely-trail safe))
147   (safely-reset safe))
148
149 #+sbcl
150 (defun unix-link (from to)
151   (sb-unix::int-syscall ("link" sb-alien:c-string sb-alien:c-string)
152                         from to))
153
154 (defun safe-copy (file tag)
155   "Make a copy of the FILE.  Return the new name."
156
157   #+(or cmu sbcl)
158   ;; Use link(2) where available.
159   (generate-fresh-file-name file tag
160                             (lambda (name)
161                               (let ((from (unix-namestring file t))
162                                     (to (unix-namestring name nil)))
163                                 (and from to
164                                      (unix-link from to)))))
165
166   #+clisp
167   (generate-fresh-file-name file tag
168                             (lambda (name)
169                               (posix:copy-file (namestring file)
170                                                (namestring name)
171                                                :method :hardlink
172                                                :if-exists nil)))
173
174
175
176   #-(or cmu sbcl)
177   ;; Otherwise just copy the file contents and hope for the best.
178   (with-open-file (input file :element-type :default)
179     (multiple-value-bind
180         (copy output)
181         (generate-fresh-file-name file tag
182                                   (lambda (name)
183                                     (open name
184                                           :direction :output
185                                           :if-exists nil
186                                           :element-type :default)))
187       (unwind-protect
188            (progn
189              (let ((buffer (make-array 8192
190                                        :element-type (stream-element-type
191                                                       input))))
192                (loop
193                  (let ((read (read-sequence buffer input)))
194                    (when (plusp read)
195                      (write-sequence buffer output :end read))
196                    (when (< read (length buffer))
197                      (return copy))))))
198         (close output)))))
199
200 (defun safely-commit (safe)
201   "Commit SAFE.  The files deleted by safely-delete-file are deleted; the
202    files created by safely-open-output-stream are renamed over the old
203    versions, if any.  If a problem occurs during this stage, everything is
204    rewound and no changes are made."
205   (let ((trail (safely-trail safe))
206         (revert nil)
207         (cleanup nil))
208     (unwind-protect
209         (progn
210           (dolist (stream (safely-streams safe))
211             (close stream))
212           (loop
213             (unless trail
214               (return))
215               (let ((job (pop trail)))
216                 (ecase (car job)
217                   (:shunt (destructuring-bind (tag new file) job
218                             (declare (ignore tag))
219                             (push `(:rmtmp ,new) revert)
220                             (if (probe-file file)
221                                 (let ((old (safe-copy file "old")))
222                                   (push `(:rmtmp ,old) cleanup)
223                                   (push `(:revert ,old ,file) revert))
224                                 (push `(:rmtmp ,file) revert))
225                             (rename new file)))
226                   (:delete (destructuring-bind (tag file) job
227                              (declare (ignore tag))
228                              (let ((old (safe-copy file "delete")))
229                                (push `(:revert ,old ,file) revert)
230                                (push `(:rmtmp ,old) cleanup)
231                                (delete-file file)))))))
232           (setf revert nil))
233       (safely-unwind trail)
234       (safely-unwind revert)
235       (safely-unwind cleanup)
236       (safely-reset safe))))
237
238 (defmacro safely ((safe &key) &body body)
239   "Do stuff within the BODY safely.  If BODY completes without errors, the
240    SAFE is committed; otherwise it's bailed."
241   `(let ((,safe (make-safely)))
242      (unwind-protect
243          (progn
244            ,@body
245            (safely-commit ,safe)
246            (setf ,safe nil))
247        (when ,safe
248          (safely-bail ,safe)))))
249
250 (defmacro safely-writing ((stream file &rest open-args) &body body)
251   "Simple macro for writing a single file safely.  STREAM is opened onto a
252    temporary file, and if BODY completes, it is renamed to FILE."
253   (with-gensyms safe
254     `(safely (,safe)
255        (let ((,stream (safely-open-output-stream ,safe ,file ,@open-args)))
256          ,@body))))
257
258 ;;;----- That's all, folks --------------------------------------------------