123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2015 Federico Beffa <beffa@fbengineering.ch>
- ;;; Copyright © 2019 Robert Vollmert <rob@vllmrt.net>
- ;;; Copyright © 2021 Xinglu Chen <public@yoctocell.xyz>
- ;;; Copyright © 2021 Sarah Morgensen <iskarian@mgsn.dev>
- ;;;
- ;;; 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-hackage)
- #:use-module (guix import cabal)
- #:use-module (guix import hackage)
- #:use-module (guix tests)
- #:use-module (srfi srfi-64)
- #:use-module (ice-9 match))
- (define test-cabal-1
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (define test-cabal-2
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal {
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- }
- ")
- ;; Check compiler implementation test with and without spaces.
- (define test-cabal-3
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if impl(ghc >= 7.2 && < 7.6)
- Build-depends: ghc-a
- if impl(ghc>=7.2&&<7.6)
- Build-depends: ghc-b
- if impl(ghc == 7.8)
- Build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- ;; Check "-any", "-none" when name is different.
- (define test-cabal-4
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if impl(ghcjs -any)
- Build-depends: ghc-a
- if impl(ghc>=7.2&&<7.6)
- Build-depends: ghc-b
- if impl(ghc == 7.8)
- Build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- ;; Check "-any", "-none".
- (define test-cabal-5
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if impl(ghc == 7.8)
- Build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- if impl(ghc -any)
- Build-depends: mtl >= 2.0 && < 3
- if impl(ghc>=7.2&&<7.6)
- Build-depends: ghc-b
- ")
- ;; Check "custom-setup".
- (define test-cabal-6
- "name: foo
- build-type: Custom
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- custom-setup
- setup-depends: base >= 4.7 && < 5,
- Cabal >= 1.24,
- haskell-gi == 0.21.*
- library
- if impl(ghc>=7.2&&<7.6)
- Build-depends: ghc-b
- if impl(ghc == 7.8)
- Build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- ;; A fragment of a real Cabal file with minor modification to check precedence
- ;; of 'and' over 'or', missing final newline, spaces between keywords and
- ;; parentheses and between key and column.
- (define test-read-cabal-1
- "name: test-me
- library
- -- Choose which library versions to use.
- if flag(base4point8)
- Build-depends: base >= 4.8 && < 5
- else
- if flag(base4)
- Build-depends: base >= 4 && < 4.8
- else
- if flag(base3)
- Build-depends: base >= 3 && < 4
- else
- Build-depends: base < 3
- if flag(base4point8) || flag (base4) && flag(base3)
- Build-depends: random
- Build-depends : containers
- -- Modules that are always built.
- Exposed-Modules:
- Test.QuickCheck.Exception")
- (define test-read-cabal-2
- "name: test-me
- common defaults
- if os(foobar) { cc-options: -DBARBAZ }
- ") ; Intentional newline.
- ;; Test opening bracket on new line.
- (define test-read-cabal-brackets-newline
- "name: test-me
- common defaults
- build-depends:
- { foobar
- , barbaz
- }
- ")
- ;; Test library with (since Cabal 2.0) and without names.
- (define test-read-cabal-library-name
- "name: test-me
- library foobar
- build-depends: foo, bar
- library
- build-depends: bar, baz
- ")
- (test-begin "hackage")
- (define-syntax-rule (define-package-matcher name pattern)
- (define* (name obj)
- (match obj
- (pattern #t)
- (x (pk 'fail x #f)))))
- (define-package-matcher match-ghc-foo
- ('package
- ('name "ghc-foo")
- ('version "1.0.0")
- ('source
- ('origin
- ('method 'url-fetch)
- ('uri ('hackage-uri "foo" 'version))
- ('sha256
- ('base32
- (? string? hash)))))
- ('build-system 'haskell-build-system)
- ('inputs ('list 'ghc-http))
- ('home-page "http://test.org")
- ('synopsis (? string?))
- ('description (? string?))
- ('license 'license:bsd-3)))
- (define* (eval-test-with-cabal test-cabal matcher #:key (cabal-environment '()))
- (define port (open-input-string test-cabal))
- (matcher (hackage->guix-package "foo" #:port port #:cabal-environment cabal-environment)))
- (test-assert "hackage->guix-package test 1"
- (eval-test-with-cabal test-cabal-1 match-ghc-foo))
- (test-assert "hackage->guix-package test 2"
- (eval-test-with-cabal test-cabal-2 match-ghc-foo))
- (test-assert "hackage->guix-package test 3"
- (eval-test-with-cabal test-cabal-3 match-ghc-foo
- #:cabal-environment '(("impl" . "ghc-7.8"))))
- (test-assert "hackage->guix-package test 4"
- (eval-test-with-cabal test-cabal-4 match-ghc-foo
- #:cabal-environment '(("impl" . "ghc-7.8"))))
- (test-assert "hackage->guix-package test 5"
- (eval-test-with-cabal test-cabal-5 match-ghc-foo
- #:cabal-environment '(("impl" . "ghc-7.8"))))
- (define-package-matcher match-ghc-foo-6
- ('package
- ('name "ghc-foo")
- ('version "1.0.0")
- ('source
- ('origin
- ('method 'url-fetch)
- ('uri ('hackage-uri "foo" 'version))
- ('sha256
- ('base32
- (? string? hash)))))
- ('build-system 'haskell-build-system)
- ('inputs ('list 'ghc-b 'ghc-http))
- ('native-inputs ('list 'ghc-haskell-gi))
- ('home-page "http://test.org")
- ('synopsis (? string?))
- ('description (? string?))
- ('license 'license:bsd-3)))
- (test-assert "hackage->guix-package test 6"
- (eval-test-with-cabal test-cabal-6 match-ghc-foo-6))
- ;; Check multi-line layouted description.
- (define test-cabal-multiline-layout
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: first line
- second line
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (test-assert "hackage->guix-package test multiline desc (layout)"
- (eval-test-with-cabal test-cabal-multiline-layout match-ghc-foo))
- ;; Check multi-line braced description.
- (define test-cabal-multiline-braced
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: {
- first line
- second line
- }
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (test-assert "hackage->guix-package test multiline desc (braced)"
- (eval-test-with-cabal test-cabal-multiline-braced match-ghc-foo))
- ;; Check mixed layout. Compare e.g. warp.
- (define test-cabal-mixed-layout
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ghc-options: -Wall
- ")
- ;; Fails: https://debbugs.gnu.org/cgi/bugreport.cgi?bug=35743
- (test-expect-fail 1)
- (test-assert "hackage->guix-package test mixed layout"
- (eval-test-with-cabal test-cabal-mixed-layout match-ghc-foo))
- ;; Check flag executable. Compare e.g. darcs.
- (define test-cabal-flag-executable
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- flag executable
- description: Build executable
- default: True
- executable cabal
- if !flag(executable)
- buildable: False
- else
- buildable: True
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (test-assert "hackage->guix-package test flag executable"
- (eval-test-with-cabal test-cabal-flag-executable match-ghc-foo))
- ;; There is no mandatory space between property name and value.
- (define test-cabal-property-no-space
- "name:foo
- version:1.0.0
- homepage:http://test.org
- synopsis:synopsis
- description:description
- license:BSD3
- common bench-defaults
- ghc-options:-Wall
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (test-assert "hackage->guix-package test properties without space"
- (eval-test-with-cabal test-cabal-property-no-space match-ghc-foo))
- ;; There may be no final newline terminating a property.
- (define test-cabal-no-final-newline
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal
- build-depends: HTTP >= 4000.2.5 && < 4000.3, mtl >= 2.0 && < 3")
- (test-expect-fail 1)
- (test-assert "hackage->guix-package test without final newline"
- (eval-test-with-cabal test-cabal-no-final-newline match-ghc-foo))
- ;; Make sure internal libraries will not be part of the dependencies,
- ;; ignore case.
- (define test-cabal-internal-library-ignored
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- internAl
- library internaL
- build-depends: mtl >= 2.0 && < 3
- ")
- (test-assert "hackage->guix-package test internal libraries are ignored"
- (eval-test-with-cabal test-cabal-internal-library-ignored match-ghc-foo))
- ;; Check if-elif-else statements
- (define test-cabal-if
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if os(first)
- Build-depends: ghc-c
- ")
- (define test-cabal-else
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if os(first)
- Build-depends: ghc-a
- else
- Build-depends: ghc-c
- ")
- (define test-cabal-elif
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if os(first)
- Build-depends: ghc-a
- elif os(second)
- Build-depends: ghc-b
- elif os(guix)
- Build-depends: ghc-c
- elif os(third)
- Build-depends: ghc-d
- else
- Build-depends: ghc-e
- ")
- ;; Try the same with different bracket styles
- (define test-cabal-elif-brackets
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- library
- if os(first) {
- Build-depends: ghc-a
- }
- elif os(second)
- Build-depends: ghc-b
- elif os(guix) { Build-depends: ghc-c }
- elif os(third) {
- Build-depends: ghc-d }
- elif os(fourth)
- {
- Build-depends: ghc-d
- } else
- Build-depends: ghc-e
- ")
- (define-package-matcher match-ghc-elif
- ('package
- ('name "ghc-foo")
- ('version "1.0.0")
- ('source
- ('origin
- ('method 'url-fetch)
- ('uri ('hackage-uri "foo" 'version))
- ('sha256
- ('base32
- (? string? hash)))))
- ('build-system 'haskell-build-system)
- ('inputs ('list 'ghc-c))
- ('home-page "http://test.org")
- ('synopsis (? string?))
- ('description (? string?))
- ('license 'license:bsd-3)))
- (test-assert "hackage->guix-package test lonely if statement"
- (eval-test-with-cabal test-cabal-else match-ghc-elif
- #:cabal-environment '(("os" . "guix"))))
- (test-assert "hackage->guix-package test else statement"
- (eval-test-with-cabal test-cabal-else match-ghc-elif
- #:cabal-environment '(("os" . "guix"))))
- (test-assert "hackage->guix-package test elif statement"
- (eval-test-with-cabal test-cabal-elif match-ghc-elif
- #:cabal-environment '(("os" . "guix"))))
- (test-assert "hackage->guix-package test elif statement with brackets"
- (eval-test-with-cabal test-cabal-elif-brackets match-ghc-elif
- #:cabal-environment '(("os" . "guix"))))
- ;; Check Hackage Cabal revisions.
- (define test-cabal-revision
- "name: foo
- version: 1.0.0
- x-revision: 2
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- executable cabal
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- ")
- (define-package-matcher match-ghc-foo-revision
- ('package
- ('name "ghc-foo")
- ('version "1.0.0")
- ('source
- ('origin
- ('method 'url-fetch)
- ('uri ('hackage-uri "foo" 'version))
- ('sha256
- ('base32
- (? string? hash)))))
- ('build-system 'haskell-build-system)
- ('inputs ('list 'ghc-http))
- ('arguments
- ('quasiquote
- ('#:cabal-revision
- ("2" "0xxd88fb659f0krljidbvvmkh9ppjnx83j0nqzx8whcg4n5qbyng"))))
- ('home-page "http://test.org")
- ('synopsis (? string?))
- ('description (? string?))
- ('license 'license:bsd-3)))
- (test-assert "hackage->guix-package test cabal revision"
- (eval-test-with-cabal test-cabal-revision match-ghc-foo-revision))
- (test-assert "read-cabal test 1"
- (match (call-with-input-string test-read-cabal-1 read-cabal)
- ((("name" ("test-me"))
- ('section 'library #f
- (('if ('flag "base4point8")
- (("build-depends" ("base >= 4.8 && < 5")))
- (('if ('flag "base4")
- (("build-depends" ("base >= 4 && < 4.8")))
- (('if ('flag "base3")
- (("build-depends" ("base >= 3 && < 4")))
- (("build-depends" ("base < 3"))))))))
- ('if ('or ('flag "base4point8")
- ('and ('flag "base4") ('flag "base3")))
- (("build-depends" ("random")))
- ())
- ("build-depends" ("containers"))
- ("exposed-modules" ("Test.QuickCheck.Exception")))))
- #t)
- (x (pk 'fail x #f))))
- (test-assert "read-cabal test: if brackets on the same line"
- (match (call-with-input-string test-read-cabal-2 read-cabal)
- ((("name" ("test-me"))
- ('section 'common "defaults"
- (('if ('os "foobar")
- (("cc-options" ("-DBARBAZ ")))
- ()))))
- #t)
- (x (pk 'fail x #f))))
- (test-expect-fail 1)
- (test-assert "read-cabal test: property brackets on new line"
- (match (call-with-input-string test-read-cabal-brackets-newline read-cabal)
- ((("name" ("test-me"))
- ('section 'common "defaults"
- (("build-depends" ("foobar , barbaz")))))
- #t)
- (x (pk 'fail x #f))))
- (test-assert "read-cabal test: library name"
- (match (call-with-input-string test-read-cabal-library-name read-cabal)
- ((("name" ("test-me"))
- ('section 'library "foobar"
- (("build-depends" ("foo, bar"))))
- ('section 'library #f
- (("build-depends" ("bar, baz")))))
- #t)
- (x (pk 'fail x #f))))
- (define test-cabal-import
- "name: foo
- version: 1.0.0
- homepage: http://test.org
- synopsis: synopsis
- description: description
- license: BSD3
- common commons
- build-depends:
- HTTP >= 4000.2.5 && < 4000.3,
- mtl >= 2.0 && < 3
- executable cabal
- import: commons
- ")
- (define-package-matcher match-ghc-foo-import
- ('package
- ('name "ghc-foo")
- ('version "1.0.0")
- ('source
- ('origin
- ('method 'url-fetch)
- ('uri ('hackage-uri "foo" 'version))
- ('sha256
- ('base32
- (? string? hash)))))
- ('build-system 'haskell-build-system)
- ('inputs ('list 'ghc-http))
- ('home-page "http://test.org")
- ('synopsis (? string?))
- ('description (? string?))
- ('license 'license:bsd-3)))
- (test-assert "hackage->guix-package test cabal import"
- (eval-test-with-cabal test-cabal-import match-ghc-foo-import))
- (test-end "hackage")
|