mirror of
https://git.savannah.gnu.org/git/guix.git
synced 2025-01-18 21:46:35 +01:00
system: 'init' does not recompute the hash of each store item.
Fixes <https://bugs.gnu.org/44760>. Previously, the 'register-path' call would re-traverse ITEM to compute its nar hash, even though that hash is already known in the initial store. This patch also avoids repeated opening/closing of the database. * guix/store/database.scm (call-with-database): Export. * guix/scripts/system.scm (copy-item): Add 'db' parameter. Call 'sqlite-register' instead of 'register-path'. (copy-closure): Remove redundant call to 'references*'. Call 'call-with-database' and pass the database to 'copy-item'.
This commit is contained in:
parent
0682cc5936
commit
1574bd82bb
3 changed files with 34 additions and 27 deletions
|
@ -121,6 +121,7 @@
|
||||||
(eval . (put 'let-system 'scheme-indent-function 1))
|
(eval . (put 'let-system 'scheme-indent-function 1))
|
||||||
|
|
||||||
(eval . (put 'with-database 'scheme-indent-function 2))
|
(eval . (put 'with-database 'scheme-indent-function 2))
|
||||||
|
(eval . (put 'call-with-database 'scheme-indent-function 1))
|
||||||
(eval . (put 'call-with-transaction 'scheme-indent-function 1))
|
(eval . (put 'call-with-transaction 'scheme-indent-function 1))
|
||||||
(eval . (put 'with-statement 'scheme-indent-function 3))
|
(eval . (put 'with-statement 'scheme-indent-function 3))
|
||||||
(eval . (put 'call-with-retrying-transaction 'scheme-indent-function 1))
|
(eval . (put 'call-with-retrying-transaction 'scheme-indent-function 1))
|
||||||
|
|
|
@ -29,7 +29,9 @@ (define-module (guix scripts system)
|
||||||
#:use-module (guix ui)
|
#:use-module (guix ui)
|
||||||
#:use-module ((guix status) #:select (with-status-verbosity))
|
#:use-module ((guix status) #:select (with-status-verbosity))
|
||||||
#:use-module (guix store)
|
#:use-module (guix store)
|
||||||
#:autoload (guix store database) (register-path)
|
#:autoload (guix base16) (bytevector->base16-string)
|
||||||
|
#:autoload (guix store database)
|
||||||
|
(sqlite-register store-database-file call-with-database)
|
||||||
#:autoload (guix build store-copy) (copy-store-item)
|
#:autoload (guix build store-copy) (copy-store-item)
|
||||||
#:use-module (guix describe)
|
#:use-module (guix describe)
|
||||||
#:use-module (guix grafts)
|
#:use-module (guix grafts)
|
||||||
|
@ -130,12 +132,11 @@ (define topologically-sorted*
|
||||||
(store-lift topologically-sorted))
|
(store-lift topologically-sorted))
|
||||||
|
|
||||||
|
|
||||||
(define* (copy-item item references target
|
(define* (copy-item item info target db
|
||||||
#:key (log-port (current-error-port)))
|
#:key (log-port (current-error-port)))
|
||||||
"Copy ITEM to the store under root directory TARGET and register it with
|
"Copy ITEM to the store under root directory TARGET and populate DB with the
|
||||||
REFERENCES as its set of references."
|
given INFO, a <path-info> record."
|
||||||
(let ((dest (string-append target item))
|
(let ((dest (string-append target item)))
|
||||||
(state (string-append target "/var/guix")))
|
|
||||||
(format log-port "copying '~a'...~%" item)
|
(format log-port "copying '~a'...~%" item)
|
||||||
|
|
||||||
;; Remove DEST if it exists to make sure that (1) we do not fail badly
|
;; Remove DEST if it exists to make sure that (1) we do not fail badly
|
||||||
|
@ -151,41 +152,45 @@ (define* (copy-item item references target
|
||||||
(copy-store-item item target
|
(copy-store-item item target
|
||||||
#:deduplicate? #t)
|
#:deduplicate? #t)
|
||||||
|
|
||||||
;; Register ITEM; as a side-effect, it resets timestamps, etc.
|
(sqlite-register db
|
||||||
;; Explicitly use "TARGET/var/guix" as the state directory, to avoid
|
#:path item
|
||||||
;; reproducing the user's current settings; see
|
#:references (path-info-references info)
|
||||||
;; <http://bugs.gnu.org/18049>.
|
#:deriver (path-info-deriver info)
|
||||||
(unless (register-path item
|
#:hash (string-append
|
||||||
#:prefix target
|
"sha256:"
|
||||||
#:state-directory state
|
(bytevector->base16-string (path-info-hash info)))
|
||||||
#:references references)
|
#:nar-size (path-info-nar-size info))))
|
||||||
(leave (G_ "failed to register '~a' under '~a'~%")
|
|
||||||
item target))))
|
|
||||||
|
|
||||||
(define* (copy-closure item target
|
(define* (copy-closure item target
|
||||||
#:key (log-port (current-error-port)))
|
#:key (log-port (current-error-port)))
|
||||||
"Copy ITEM and all its dependencies to the store under root directory
|
"Copy ITEM and all its dependencies to the store under root directory
|
||||||
TARGET, and register them."
|
TARGET, and register them."
|
||||||
(mlet* %store-monad ((to-copy (topologically-sorted* (list item)))
|
(mlet* %store-monad ((to-copy (topologically-sorted* (list item)))
|
||||||
(refs (mapm %store-monad references* to-copy))
|
(info (mapm %store-monad query-path-info* to-copy))
|
||||||
(info (mapm %store-monad query-path-info*
|
|
||||||
(delete-duplicates
|
|
||||||
(append to-copy (concatenate refs)))))
|
|
||||||
(size -> (reduce + 0 (map path-info-nar-size info))))
|
(size -> (reduce + 0 (map path-info-nar-size info))))
|
||||||
(define progress-bar
|
(define progress-bar
|
||||||
(progress-reporter/bar (length to-copy)
|
(progress-reporter/bar (length to-copy)
|
||||||
(format #f (G_ "copying to '~a'...")
|
(format #f (G_ "copying to '~a'...")
|
||||||
target)))
|
target)))
|
||||||
|
|
||||||
|
(define state
|
||||||
|
(string-append target "/var/guix"))
|
||||||
|
|
||||||
(check-available-space size target)
|
(check-available-space size target)
|
||||||
|
|
||||||
(call-with-progress-reporter progress-bar
|
;; Explicitly use "TARGET/var/guix" as the state directory to avoid
|
||||||
(lambda (report)
|
;; reproducing the user's current settings; see
|
||||||
(let ((void (%make-void-port "w")))
|
;; <http://bugs.gnu.org/18049>.
|
||||||
(for-each (lambda (item refs)
|
(call-with-database (store-database-file #:prefix target
|
||||||
(copy-item item refs target #:log-port void)
|
#:state-directory state)
|
||||||
(report))
|
(lambda (db)
|
||||||
to-copy refs))))
|
(call-with-progress-reporter progress-bar
|
||||||
|
(lambda (report)
|
||||||
|
(let ((void (%make-void-port "w")))
|
||||||
|
(for-each (lambda (item info)
|
||||||
|
(copy-item item info target db #:log-port void)
|
||||||
|
(report))
|
||||||
|
to-copy info))))))
|
||||||
|
|
||||||
(return *unspecified*)))
|
(return *unspecified*)))
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,7 @@ (define-module (guix store database)
|
||||||
#:export (sql-schema
|
#:export (sql-schema
|
||||||
%default-database-file
|
%default-database-file
|
||||||
store-database-file
|
store-database-file
|
||||||
|
call-with-database
|
||||||
with-database
|
with-database
|
||||||
path-id
|
path-id
|
||||||
sqlite-register
|
sqlite-register
|
||||||
|
|
Loading…
Reference in a new issue