2018-10-13 06:39:23 +00:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
|
|
|
|
;;; Copyright © 2018 Ricardo Wurmus <rekado@elephly.net>
|
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix 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 of the License, or (at
|
|
|
|
|
;;; your option) any later version.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix 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.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (test-channels)
|
|
|
|
|
#:use-module (guix channels)
|
2019-01-17 15:57:53 +00:00
|
|
|
|
#:use-module (guix profiles)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
#:use-module ((guix build syscalls) #:select (mkdtemp!))
|
|
|
|
|
#:use-module (guix tests)
|
2019-01-17 15:57:53 +00:00
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module ((guix grafts) #:select (%graft?))
|
|
|
|
|
#:use-module (guix derivations)
|
2019-01-18 09:01:37 +00:00
|
|
|
|
#:use-module (guix sets)
|
2019-01-17 15:57:53 +00:00
|
|
|
|
#:use-module (guix gexp)
|
2019-07-16 22:04:41 +00:00
|
|
|
|
#:use-module ((guix utils)
|
|
|
|
|
#:select (error-location? error-location location-line))
|
2019-09-14 21:16:54 +00:00
|
|
|
|
#:use-module ((guix build utils) #:select (which))
|
|
|
|
|
#:use-module (git)
|
|
|
|
|
#:use-module (guix git)
|
|
|
|
|
#:use-module (guix tests git)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
#:use-module (srfi srfi-1)
|
2019-01-17 15:57:53 +00:00
|
|
|
|
#:use-module (srfi srfi-26)
|
2019-07-16 22:04:41 +00:00
|
|
|
|
#:use-module (srfi srfi-34)
|
|
|
|
|
#:use-module (srfi srfi-35)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
#:use-module (srfi srfi-64)
|
2020-05-20 20:15:54 +00:00
|
|
|
|
#:use-module (ice-9 control)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
#:use-module (ice-9 match))
|
|
|
|
|
|
|
|
|
|
(test-begin "channels")
|
|
|
|
|
|
|
|
|
|
(define* (make-instance #:key
|
|
|
|
|
(name 'fake)
|
|
|
|
|
(commit "cafebabe")
|
|
|
|
|
(spec #f))
|
|
|
|
|
(define instance-dir (mkdtemp! "/tmp/checkout.XXXXXX"))
|
2019-07-16 22:41:10 +00:00
|
|
|
|
(when spec
|
|
|
|
|
(call-with-output-file (string-append instance-dir "/.guix-channel")
|
|
|
|
|
(lambda (port) (write spec port))))
|
2019-01-17 15:57:53 +00:00
|
|
|
|
(checkout->channel-instance instance-dir
|
|
|
|
|
#:commit commit
|
|
|
|
|
#:name name))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
|
|
|
|
|
(define instance--boring (make-instance))
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(define instance--unsupported-version
|
|
|
|
|
(make-instance #:spec
|
|
|
|
|
'(channel (version 42) (dependencies whatever))))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(define instance--no-deps
|
|
|
|
|
(make-instance #:spec
|
2019-07-16 22:41:10 +00:00
|
|
|
|
'(channel (version 0))))
|
|
|
|
|
(define instance--sub-directory
|
|
|
|
|
(make-instance #:spec
|
|
|
|
|
'(channel (version 0) (directory "modules"))))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(define instance--simple
|
|
|
|
|
(make-instance #:spec
|
|
|
|
|
'(channel
|
|
|
|
|
(version 0)
|
|
|
|
|
(dependencies
|
|
|
|
|
(channel
|
|
|
|
|
(name test-channel)
|
|
|
|
|
(url "https://example.com/test-channel"))))))
|
|
|
|
|
(define instance--with-dupes
|
|
|
|
|
(make-instance #:spec
|
|
|
|
|
'(channel
|
|
|
|
|
(version 0)
|
|
|
|
|
(dependencies
|
|
|
|
|
(channel
|
|
|
|
|
(name test-channel)
|
|
|
|
|
(url "https://example.com/test-channel"))
|
|
|
|
|
(channel
|
|
|
|
|
(name test-channel)
|
|
|
|
|
(url "https://example.com/test-channel")
|
|
|
|
|
(commit "abc1234"))
|
|
|
|
|
(channel
|
|
|
|
|
(name test-channel)
|
|
|
|
|
(url "https://example.com/test-channel-elsewhere"))))))
|
|
|
|
|
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(define channel-instance-metadata
|
|
|
|
|
(@@ (guix channels) channel-instance-metadata))
|
2019-07-16 22:41:10 +00:00
|
|
|
|
(define channel-metadata-directory
|
|
|
|
|
(@@ (guix channels) channel-metadata-directory))
|
|
|
|
|
(define channel-metadata-dependencies
|
|
|
|
|
(@@ (guix channels) channel-metadata-dependencies))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
|
|
|
|
|
|
2019-07-16 22:41:10 +00:00
|
|
|
|
(test-equal "channel-instance-metadata returns default if .guix-channel does not exist"
|
|
|
|
|
'("/" ())
|
|
|
|
|
(let ((metadata (channel-instance-metadata instance--boring)))
|
|
|
|
|
(list (channel-metadata-directory metadata)
|
|
|
|
|
(channel-metadata-dependencies metadata))))
|
|
|
|
|
|
|
|
|
|
(test-equal "channel-instance-metadata and default dependencies"
|
|
|
|
|
'()
|
|
|
|
|
(channel-metadata-dependencies (channel-instance-metadata instance--no-deps)))
|
|
|
|
|
|
|
|
|
|
(test-equal "channel-instance-metadata and directory"
|
|
|
|
|
"/modules"
|
|
|
|
|
(channel-metadata-directory
|
|
|
|
|
(channel-instance-metadata instance--sub-directory)))
|
2019-07-16 22:04:41 +00:00
|
|
|
|
|
|
|
|
|
(test-equal "channel-instance-metadata rejects unsupported version"
|
|
|
|
|
1 ;line number in the generated '.guix-channel'
|
|
|
|
|
(guard (c ((and (message-condition? c) (error-location? c))
|
|
|
|
|
(location-line (error-location c))))
|
|
|
|
|
(channel-instance-metadata instance--unsupported-version)))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(test-assert "channel-instance-metadata returns <channel-metadata>"
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(every (@@ (guix channels) channel-metadata?)
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(map channel-instance-metadata
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(list instance--no-deps
|
|
|
|
|
instance--simple
|
|
|
|
|
instance--with-dupes))))
|
|
|
|
|
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(test-assert "channel-instance-metadata dependencies are channels"
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(let ((deps ((@@ (guix channels) channel-metadata-dependencies)
|
2019-07-16 22:04:41 +00:00
|
|
|
|
(channel-instance-metadata instance--simple))))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(match deps
|
|
|
|
|
(((? channel? dep)) #t)
|
|
|
|
|
(_ #f))))
|
|
|
|
|
|
|
|
|
|
(test-assert "latest-channel-instances includes channel dependencies"
|
|
|
|
|
(let* ((channel (channel
|
|
|
|
|
(name 'test)
|
|
|
|
|
(url "test")))
|
|
|
|
|
(test-dir (channel-instance-checkout instance--simple)))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(mock ((guix git) update-cached-checkout
|
2020-05-20 15:57:54 +00:00
|
|
|
|
(lambda* (url #:key ref starting-commit)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(match url
|
2020-05-20 15:57:54 +00:00
|
|
|
|
("test" (values test-dir "caf3cabba9e" #f))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(_ (values (channel-instance-checkout instance--no-deps)
|
2020-05-20 15:57:54 +00:00
|
|
|
|
"abcde1234" #f)))))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(with-store store
|
|
|
|
|
(let ((instances (latest-channel-instances store (list channel))))
|
|
|
|
|
(and (eq? 2 (length instances))
|
|
|
|
|
(lset= eq?
|
|
|
|
|
'(test test-channel)
|
|
|
|
|
(map (compose channel-name channel-instance-channel)
|
|
|
|
|
instances))))))))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
|
|
|
|
|
(test-assert "latest-channel-instances excludes duplicate channel dependencies"
|
|
|
|
|
(let* ((channel (channel
|
|
|
|
|
(name 'test)
|
|
|
|
|
(url "test")))
|
|
|
|
|
(test-dir (channel-instance-checkout instance--with-dupes)))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(mock ((guix git) update-cached-checkout
|
2020-05-20 15:57:54 +00:00
|
|
|
|
(lambda* (url #:key ref starting-commit)
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(match url
|
2020-05-20 15:57:54 +00:00
|
|
|
|
("test" (values test-dir "caf3cabba9e" #f))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(_ (values (channel-instance-checkout instance--no-deps)
|
2020-05-20 15:57:54 +00:00
|
|
|
|
"abcde1234" #f)))))
|
2020-05-06 20:45:31 +00:00
|
|
|
|
(with-store store
|
|
|
|
|
(let ((instances (latest-channel-instances store (list channel))))
|
|
|
|
|
(and (= 2 (length instances))
|
|
|
|
|
(lset= eq?
|
|
|
|
|
'(test test-channel)
|
|
|
|
|
(map (compose channel-name channel-instance-channel)
|
|
|
|
|
instances))
|
|
|
|
|
;; only the most specific channel dependency should remain,
|
|
|
|
|
;; i.e. the one with a specified commit.
|
|
|
|
|
(find (lambda (instance)
|
|
|
|
|
(and (eq? (channel-name
|
|
|
|
|
(channel-instance-channel instance))
|
|
|
|
|
'test-channel)
|
|
|
|
|
(string=? (channel-commit
|
|
|
|
|
(channel-instance-channel instance))
|
|
|
|
|
"abc1234")))
|
|
|
|
|
instances)))))))
|
2018-10-13 06:39:23 +00:00
|
|
|
|
|
2020-05-20 20:15:54 +00:00
|
|
|
|
(unless (which (git-command)) (test-skip 1))
|
|
|
|
|
(test-equal "latest-channel-instances #:validate-pull"
|
|
|
|
|
'descendant
|
|
|
|
|
|
|
|
|
|
;; Make sure the #:validate-pull procedure receives the right values.
|
|
|
|
|
(let/ec return
|
|
|
|
|
(with-temporary-git-repository directory
|
|
|
|
|
'((add "a.txt" "A")
|
|
|
|
|
(commit "first commit")
|
|
|
|
|
(add "b.scm" "#t")
|
|
|
|
|
(commit "second commit"))
|
|
|
|
|
(with-repository directory repository
|
|
|
|
|
(let* ((commit1 (find-commit repository "first"))
|
|
|
|
|
(commit2 (find-commit repository "second"))
|
|
|
|
|
(spec (channel (url (string-append "file://" directory))
|
|
|
|
|
(name 'foo)))
|
|
|
|
|
(new (channel (inherit spec)
|
|
|
|
|
(commit (oid->string (commit-id commit2)))))
|
|
|
|
|
(old (channel (inherit spec)
|
|
|
|
|
(commit (oid->string (commit-id commit1))))))
|
|
|
|
|
(define (validate-pull channel current instance relation)
|
|
|
|
|
(return (and (eq? channel old)
|
|
|
|
|
(string=? (oid->string (commit-id commit2))
|
|
|
|
|
current)
|
|
|
|
|
(string=? (oid->string (commit-id commit1))
|
|
|
|
|
(channel-instance-commit instance))
|
|
|
|
|
relation)))
|
|
|
|
|
|
|
|
|
|
(with-store store
|
|
|
|
|
;; Attempt a downgrade from NEW to OLD.
|
|
|
|
|
(latest-channel-instances store (list old)
|
|
|
|
|
#:current-channels (list new)
|
|
|
|
|
#:validate-pull validate-pull)))))))
|
|
|
|
|
|
2019-01-17 15:57:53 +00:00
|
|
|
|
(test-assert "channel-instances->manifest"
|
|
|
|
|
;; Compute the manifest for a graph of instances and make sure we get a
|
|
|
|
|
;; derivation graph that mirrors the instance graph. This test also ensures
|
|
|
|
|
;; we don't try to access Git repositores at all at this stage.
|
|
|
|
|
(let* ((spec (lambda deps
|
|
|
|
|
`(channel (version 0)
|
|
|
|
|
(dependencies
|
|
|
|
|
,@(map (lambda (dep)
|
|
|
|
|
`(channel
|
|
|
|
|
(name ,dep)
|
|
|
|
|
(url "http://example.org")))
|
|
|
|
|
deps)))))
|
|
|
|
|
(guix (make-instance #:name 'guix))
|
|
|
|
|
(instance0 (make-instance #:name 'a))
|
|
|
|
|
(instance1 (make-instance #:name 'b #:spec (spec 'a)))
|
|
|
|
|
(instance2 (make-instance #:name 'c #:spec (spec 'b)))
|
|
|
|
|
(instance3 (make-instance #:name 'd #:spec (spec 'c 'a))))
|
|
|
|
|
(%graft? #f) ;don't try to build stuff
|
|
|
|
|
|
|
|
|
|
;; Create 'build-self.scm' so that GUIX is recognized as the 'guix' channel.
|
|
|
|
|
(let ((source (channel-instance-checkout guix)))
|
|
|
|
|
(mkdir (string-append source "/build-aux"))
|
|
|
|
|
(call-with-output-file (string-append source
|
|
|
|
|
"/build-aux/build-self.scm")
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(write '(begin
|
|
|
|
|
(use-modules (guix) (gnu packages bootstrap))
|
|
|
|
|
|
|
|
|
|
(lambda _
|
|
|
|
|
(package->derivation %bootstrap-guile)))
|
|
|
|
|
port))))
|
|
|
|
|
|
|
|
|
|
(with-store store
|
|
|
|
|
(let ()
|
|
|
|
|
(define manifest
|
|
|
|
|
(run-with-store store
|
|
|
|
|
(channel-instances->manifest (list guix
|
|
|
|
|
instance0 instance1
|
|
|
|
|
instance2 instance3))))
|
|
|
|
|
|
|
|
|
|
(define entries
|
|
|
|
|
(manifest-entries manifest))
|
|
|
|
|
|
|
|
|
|
(define (depends? drv in out)
|
2019-01-18 09:01:37 +00:00
|
|
|
|
;; Return true if DRV depends (directly or indirectly) on all of IN
|
|
|
|
|
;; and none of OUT.
|
|
|
|
|
(let ((set (list->set
|
|
|
|
|
(requisites store
|
|
|
|
|
(list (derivation-file-name drv)))))
|
2019-01-17 15:57:53 +00:00
|
|
|
|
(in (map derivation-file-name in))
|
|
|
|
|
(out (map derivation-file-name out)))
|
2019-01-18 09:01:37 +00:00
|
|
|
|
(and (every (cut set-contains? set <>) in)
|
|
|
|
|
(not (any (cut set-contains? set <>) out)))))
|
2019-01-17 15:57:53 +00:00
|
|
|
|
|
|
|
|
|
(define (lookup name)
|
|
|
|
|
(run-with-store store
|
|
|
|
|
(lower-object
|
|
|
|
|
(manifest-entry-item
|
|
|
|
|
(manifest-lookup manifest
|
|
|
|
|
(manifest-pattern (name name)))))))
|
|
|
|
|
|
|
|
|
|
(let ((drv-guix (lookup "guix"))
|
|
|
|
|
(drv0 (lookup "a"))
|
|
|
|
|
(drv1 (lookup "b"))
|
|
|
|
|
(drv2 (lookup "c"))
|
|
|
|
|
(drv3 (lookup "d")))
|
|
|
|
|
(and (depends? drv-guix '() (list drv0 drv1 drv2 drv3))
|
|
|
|
|
(depends? drv0
|
|
|
|
|
(list) (list drv1 drv2 drv3))
|
|
|
|
|
(depends? drv1
|
|
|
|
|
(list drv0) (list drv2 drv3))
|
|
|
|
|
(depends? drv2
|
2019-01-18 09:01:37 +00:00
|
|
|
|
(list drv1) (list drv3))
|
2019-01-17 15:57:53 +00:00
|
|
|
|
(depends? drv3
|
2019-01-18 09:01:37 +00:00
|
|
|
|
(list drv2 drv0) (list))))))))
|
2019-01-17 15:57:53 +00:00
|
|
|
|
|
2019-09-14 21:16:54 +00:00
|
|
|
|
(unless (which (git-command)) (test-skip 1))
|
|
|
|
|
(test-equal "channel-news, no news"
|
|
|
|
|
'()
|
|
|
|
|
(with-temporary-git-repository directory
|
|
|
|
|
'((add "a.txt" "A")
|
|
|
|
|
(commit "the commit"))
|
|
|
|
|
(with-repository directory repository
|
|
|
|
|
(let ((channel (channel (url (string-append "file://" directory))
|
|
|
|
|
(name 'foo)))
|
|
|
|
|
(latest (reference-name->oid repository "HEAD")))
|
|
|
|
|
(channel-news-for-commit channel (oid->string latest))))))
|
|
|
|
|
|
|
|
|
|
(unless (which (git-command)) (test-skip 1))
|
|
|
|
|
(test-assert "channel-news, one entry"
|
|
|
|
|
(with-temporary-git-repository directory
|
|
|
|
|
`((add ".guix-channel"
|
|
|
|
|
,(object->string
|
|
|
|
|
'(channel (version 0)
|
|
|
|
|
(news-file "news.scm"))))
|
|
|
|
|
(commit "first commit")
|
|
|
|
|
(add "src/a.txt" "A")
|
|
|
|
|
(commit "second commit")
|
2019-09-21 19:29:30 +00:00
|
|
|
|
(tag "tag-for-first-news-entry")
|
2019-09-14 21:16:54 +00:00
|
|
|
|
(add "news.scm"
|
|
|
|
|
,(lambda (repository)
|
|
|
|
|
(let ((previous
|
|
|
|
|
(reference-name->oid repository "HEAD")))
|
|
|
|
|
(object->string
|
|
|
|
|
`(channel-news
|
|
|
|
|
(version 0)
|
|
|
|
|
(entry (commit ,(oid->string previous))
|
|
|
|
|
(title (en "New file!")
|
|
|
|
|
(eo "Nova dosiero!"))
|
|
|
|
|
(body (en "Yeah, a.txt."))))))))
|
|
|
|
|
(commit "third commit")
|
|
|
|
|
(add "src/b.txt" "B")
|
|
|
|
|
(commit "fourth commit")
|
|
|
|
|
(add "news.scm"
|
|
|
|
|
,(lambda (repository)
|
|
|
|
|
(let ((second
|
|
|
|
|
(commit-id
|
|
|
|
|
(find-commit repository "second commit")))
|
|
|
|
|
(previous
|
|
|
|
|
(reference-name->oid repository "HEAD")))
|
|
|
|
|
(object->string
|
|
|
|
|
`(channel-news
|
|
|
|
|
(version 0)
|
|
|
|
|
(entry (commit ,(oid->string previous))
|
|
|
|
|
(title (en "Another file!"))
|
|
|
|
|
(body (en "Yeah, b.txt.")))
|
2019-09-21 19:29:30 +00:00
|
|
|
|
(entry (tag "tag-for-first-news-entry")
|
2019-09-14 21:16:54 +00:00
|
|
|
|
(title (en "Old news.")
|
|
|
|
|
(eo "Malnovaĵoj."))
|
|
|
|
|
(body (en "For a.txt"))))))))
|
|
|
|
|
(commit "fifth commit"))
|
|
|
|
|
(with-repository directory repository
|
|
|
|
|
(define (find-commit* message)
|
|
|
|
|
(oid->string (commit-id (find-commit repository message))))
|
|
|
|
|
|
|
|
|
|
(let ((channel (channel (url (string-append "file://" directory))
|
|
|
|
|
(name 'foo)))
|
|
|
|
|
(commit1 (find-commit* "first commit"))
|
|
|
|
|
(commit2 (find-commit* "second commit"))
|
|
|
|
|
(commit3 (find-commit* "third commit"))
|
|
|
|
|
(commit4 (find-commit* "fourth commit"))
|
|
|
|
|
(commit5 (find-commit* "fifth commit")))
|
|
|
|
|
;; First try fetching all the news up to a given commit.
|
|
|
|
|
(and (null? (channel-news-for-commit channel commit2))
|
|
|
|
|
(lset= string=?
|
|
|
|
|
(map channel-news-entry-commit
|
|
|
|
|
(channel-news-for-commit channel commit5))
|
|
|
|
|
(list commit2 commit4))
|
|
|
|
|
(lset= equal?
|
|
|
|
|
(map channel-news-entry-title
|
|
|
|
|
(channel-news-for-commit channel commit5))
|
|
|
|
|
'((("en" . "Another file!"))
|
|
|
|
|
(("en" . "Old news.") ("eo" . "Malnovaĵoj."))))
|
|
|
|
|
(lset= string=?
|
|
|
|
|
(map channel-news-entry-commit
|
|
|
|
|
(channel-news-for-commit channel commit3))
|
|
|
|
|
(list commit2))
|
|
|
|
|
|
|
|
|
|
;; Now fetch news entries that apply to a commit range.
|
|
|
|
|
(lset= string=?
|
|
|
|
|
(map channel-news-entry-commit
|
|
|
|
|
(channel-news-for-commit channel commit3 commit1))
|
|
|
|
|
(list commit2))
|
|
|
|
|
(lset= string=?
|
|
|
|
|
(map channel-news-entry-commit
|
|
|
|
|
(channel-news-for-commit channel commit5 commit3))
|
|
|
|
|
(list commit4))
|
|
|
|
|
(lset= string=?
|
|
|
|
|
(map channel-news-entry-commit
|
|
|
|
|
(channel-news-for-commit channel commit5 commit1))
|
2019-09-21 19:29:30 +00:00
|
|
|
|
(list commit4 commit2))
|
|
|
|
|
(lset= equal?
|
|
|
|
|
(map channel-news-entry-tag
|
|
|
|
|
(channel-news-for-commit channel commit5 commit1))
|
|
|
|
|
'(#f "tag-for-first-news-entry")))))))
|
2019-09-14 21:16:54 +00:00
|
|
|
|
|
2018-10-13 06:39:23 +00:00
|
|
|
|
(test-end "channels")
|