6

编辑未来的灵魂绊倒这个:

在下面复制我的任何命令都是不明智的,不要执行它们。只需查看 Zeta 的答案,了解我为什么不正确地处理这个问题。

原始问题

我的沙箱似乎被某些包的全球版本所吸引,而不是使用沙箱版本。我已经在 SO 上四处寻找答案,并且许多问题都没有具体说明来解决这个问题,或者略有不同,所以我似乎无法让我的设置运行。这是我所拥有的:

1. Version of cabal is old:

$ cabal --v
cabal-install version 1.16.0.2
using version 1.16.0 of the Cabal library 

哦,天哪,这是旧的,不会运行沙盒。

2. Install cabal with cabal
$ cabal install cabal
Resolving dependencies...
Downloading Cabal-1.22.6.0...

...lots of stuf...

Registering Cabal-1.22.6.0...
Installed Cabal-1.22.6.0

看起来挺好的。

3. Try it out:
$ cabal sandbox init
cabal: unrecognised command: sandbox (try --help)

唔。

$ which cabal
/usr/bin/cabal

啊。

$ ${HOME}/.cabal/bin/cabal --version
cabal-install version 1.22.2.0
using version 1.22.2.0 of the Cabal library 

啊哈。

$ export PATH="${HOME}/.cabal/bin/:$PATH"
$ export PATH="${HOME}/.cabal/libs/:$PATH"

$ cabal --v
cabal-install version 1.22.2.0

伟大的。

4. Try it out again:
$ cabal sandbox init
Writing a default package environment file to
.../cabal.sandbox.config
Creating a new sandbox at .../.cabal-sandbox

$ cabal install Frames
Resolving dependencies...
cabal: Could not resolve dependencies:
trying: Frames-0.1.2.1 (user goal)
next goal: base (dependency of Frames-0.1.2.1)
rejecting: base-4.6.0.1/installed-8aa... (conflict: Frames => base>=4.7 &&
<4.9)
rejecting: base-4.8.2.0, 4.8.1.0, 4.8.0.0, 4.7.0.2, 4.7.0.1, 4.7.0.0, 4.6.0.1,
4.6.0.0, 4.5.1.0, 4.5.0.0, 4.4.1.0, 4.4.0.0, 4.3.1.0, 4.3.0.0, 4.2.0.2,
4.2.0.1, 4.2.0.0, 4.1.0.0, 4.0.0.0, 3.0.3.2, 3.0.3.1 (global constraint
requires installed instance)
Dependency tree exhaustively searched.

Note: when using a sandbox, all packages are required to have consistent
dependencies. Try reinstalling/unregistering the offending packages or
recreating the sandbox.

嗯,所以base不兼容,OK。

$ cabal install base --reinstall
Resolving dependencies...
cabal: Could not resolve dependencies:
next goal: base (user goal)
rejecting: base-4.8.2.0, 4.8.1.0, 4.8.0.0, 4.7.0.2, 4.7.0.1, 4.7.0.0, 4.6.0.1,
4.6.0.0, 4.5.1.0, 4.5.0.0, 4.4.1.0, 4.4.0.0, 4.3.1.0, 4.3.0.0, 4.2.0.2,
4.2.0.1, 4.2.0.0, 4.1.0.0, 4.0.0.0 (only already installed instances can be
used)
rejecting: base-3.0.3.2 (conflict: base => base>=4.0 && <4.3)
rejecting: base-3.0.3.1 (conflict: base => base>=4.0 && <4.2)
Dependency tree exhaustively searched.

Note: when using a sandbox, all packages are required to have consistent
dependencies. Try reinstalling/unregistering the offending packages or
recreating the sandbox.

呵呵,所有依赖项呢?

$ cabal install --upgrade-dependencies
Resolving dependencies...
cabal: Could not resolve dependencies:
trying: Frames-0.1.2.1 (user goal)
next goal: base (dependency of Frames-0.1.2.1)
rejecting: base-4.8.2.0, 4.8.1.0, 4.8.0.0, 4.7.0.2, 4.7.0.1, 4.7.0.0 (global
constraint requires installed instance)
rejecting: base-4.6.0.1/installed-8aa... (conflict: Frames => base>=4.7 &&
<4.9)
rejecting: base-4.6.0.1, 4.6.0.0, 4.5.1.0, 4.5.0.0, 4.4.1.0, 4.4.0.0, 4.3.1.0,
4.3.0.0, 4.2.0.2, 4.2.0.1, 4.2.0.0, 4.1.0.0, 4.0.0.0, 3.0.3.2, 3.0.3.1 (global
constraint requires installed instance)
Dependency tree exhaustively searched.

Note: when using a sandbox, all packages are required to have consistent
dependencies. Try reinstalling/unregistering the offending packages or
recreating the sandbox.

似乎我无法重新安装/注销有问题的软件包。我希望错误消息能给我更多关于如何执行此操作的信息,因为“--reinstall”标志是我最好的选择。

好吧,它提供了两种解决方案,所以让我们尝试另一种,重新创建沙箱。

$ cabal sandbox delete
Deleting the sandbox located at .../.cabal-sandbox

$ cabal sandbox init
Writing a default package environment file to
.../cabal.sandbox.config
Creating a new sandbox at .../.cabal-sandbox

$ cabal install --upgrade-dependencies
Resolving dependencies...
cabal: Could not resolve dependencies:
trying: Frames-0.1.2.1 (user goal)
next goal: base (dependency of Frames-0.1.2.1)
rejecting: base-4.8.2.0, 4.8.1.0, 4.8.0.0, 4.7.0.2, 4.7.0.1, 4.7.0.0 (global
constraint requires installed instance)
rejecting: base-4.6.0.1/installed-8aa... (conflict: Frames => base>=4.7 &&
<4.9)
rejecting: base-4.6.0.1, 4.6.0.0, 4.5.1.0, 4.5.0.0, 4.4.1.0, 4.4.0.0, 4.3.1.0,
4.3.0.0, 4.2.0.2, 4.2.0.1, 4.2.0.0, 4.1.0.0, 4.0.0.0, 3.0.3.2, 3.0.3.1 (global
constraint requires installed instance)
Dependency tree exhaustively searched.

Note: when using a sandbox, all packages are required to have consistent
dependencies. Try reinstalling/unregistering the offending packages or
recreating the sandbox.

这也没有帮助。新鲜出主意。如何使用沙盒?我在这里读到的是有一个全球性的东西导致了问题。沙盒不应该被沙盒设计所困扰,对吧?

如果它有助于上下文,我只是尝试运行 Frames 的演示:

https://github.com/acowley/Frames

...这是:

$ cabal sandbox init
$ cabal install --dependencies-only -f demos # <-- here

我看到这一步有一个已解决的问题,但这似乎是一个不同的问题。

4

1 回答 1

11

看台后(或:基地有什么特别之处)

的版本与baseGHC 版本紧密耦合。例如,base-4.6.0.1随 GHC 7.6.3 一起提供。您不能将其他版本的 base 与不兼容的 GHC 版本一起使用。为了使用base-4.7.0.x,您需要 GHC 7.8.x。

因此,您不能将本地沙箱用于base. 它是(极少数)无法被沙盒化的软件包之一。

沙盒需要强大的基础

如果我们采用沙盒的形象化概念,我们很快就会发现沙盒需要某种基础。正是如此base。其他一切都可以存在于沙盒中,并且应该可以工作:

如何使用沙盒?

您正确使用了沙箱,但没有安装正确的编译器/基础。

我不明白为什么阴谋集团让我将自己升级到一个如此不一致的状态,我什么都做不了。

同样,Cabal 并没有处于不一致的状态。Frames将最小基本版本限制为4.7,这通常意味着它使用 GHC 7.8 引入的 GHC 功能。即使您能够安装base-4.7.x.y,您仍然会错过那些 GHC 功能。

但是,不建议将最新版本的 Cabal 与旧版本的 GHC 一起使用。

尝试其他下限

但是,有时库的下限过于严格。也许Frames 确实适用于 GHC 7.6。您可以在本地检查:

$ cabal unpack Frames
$ cd Frames-*
$ vim Frames.cabal # change the min base to something lower
$ cabal sandbox init
$ cabal build

如果这可行,则 Frames 的下限base实际上太高了,您可以在库上提交拉取请求。

在旧发行版中获取新内容

现在有四种方法。按难度递减顺序:

  1. 从GHC 下载页面手动安装 GHC 和 Cabal (不推荐,仅供有经验的用户使用)。
  2. 安装 GHC 和 Cabal 的最小版本,请参阅MinGHC
  3. 手动安装最新版本的Haskell 平台(不要忘记事先删除旧版本;使用 shell 脚本,而不是apt-getLTS 发行版上的包)。
  4. 安装堆栈并使用stack setup(最简单的一个)。

我将专注于 MinGHC 和 Stack 之一,因为它们提供了简单的方法来安装多个 GHC 版本。

MinGHC 安装(特定于 Linux)

基本上按照本网站的说明进行操作。在 Ubuntu 上,您必须将第三方源添加到您的apt源并安装特定版本的 Cabal 和 GHC。请注意,这会将 GHC 和 Cabal 安装到/opt/,因此您必须在路径中添加一些目录:

sudo apt-get update
sudo apt-get install -y software-properties-common
sudo add-apt-repository -y ppa:hvr/ghc
sudo apt-get update
sudo apt-get install -y cabal-install-1.22 ghc-7.10.3
cat >> ~/.bashrc <<EOF
export PATH="\$HOME/.cabal/bin:/opt/cabal/1.20/bin:/opt/ghc/7.10.3/bin:\$PATH"
EOF
export PATH=~/.cabal/bin:/opt/cabal/1.22/bin:/opt/ghc/7.10.3/bin:$PATH

这使您可以很容易地安装其他版本的 cabal 和 GHC

sudo apt-get install -y cabal-install-1.xx ghc-7.yy.z

然后改变你的PATH变量。之后,您可以使用cabal sandbox *命令。

我还建议您阅读斯蒂芬迪尔关于阴谋集团的笔记。它们提供了对一些常用功能的深刻见解。

使用堆栈

再次,按照本网站上的说明进行操作。基本相同:您添加外部资源并告诉您的发行版信任这些资源:

sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-keys 575159689BEFB442
echo 'deb http://download.fpcomplete.com/ubuntu trusty main' | sudo tee /etc/apt/sources.list.d/fpco.list
sudo apt-get update && sudo apt-get install stack -y

之后,您必须stack init在已经存在的 Cabal 项目中使用,或者stack new安装一个新项目,然后stack setup安装一个相当新的 GHC 版本:

$ stack new my-project
$ cd my-project
$ stack setup
# Loooooooooooooooooooooooong log, installs GHC in background

请注意,stack以“理智”的方式安装所有依赖项。它使用内部 LTS 系统,您不必使用 cabal 沙箱。但是,您不能再使用 Cabal 命令,而必须使用stack buildor stack ghc

如果您不使用包,例如

stack ghc --package Frames -- -O2 --make -rtsopts File1.hs File2.hs

是可能的。但是,记录堆栈的每个功能都会打破此答案的范围并很快过时,因此请查看文档

于 2016-01-25T09:52:41.113 回答