[Haskell-cafe] Re: Configuring cabal dependencies at install-time

Edward Kmett ekmett at gmail.com
Tue Apr 7 10:10:49 EDT 2009


This has been a lot on my mind lately as my current library provides
additional functionality to data types from a wide array of other packages.
I face a version of Wadler's expression problem.

I provide a set of classes for injecting into monoids/seminearrings/etc. to
allow for quick reductions over different data structures. The problem is
that of course the interfaces are fairly general so whole swathes of types
(including every applicative functor!) qualifies for certain operations.

Perhaps the ultimate answer would be to push more of the instances down into
the source packages. I can do this with some of the monoid instances, but
convincing folks of the utility of the fact that their particular
applicative forms a right-seminearring when it contains a monoid is another
matter entirely.

The problem is there is no path to get there from here. Getting another
library to depend on mine, they have to pick up the brittle dependency set I
have now. Splitting my package into smaller packages fails because I need to
keep the instances for 3rd party data types packed with the class
definitions to avoid orphan instances and poor API design. So the option to
split things into the equivalent of 'buster-ui', 'buster-network' and so
forth largely fails on that design criterion. I can do that for new monoids,
rings and so forth that I define that purely layer on top of the base
functionality I provide, but not for ones that provide additional instances
for 3rd party data types.

I can keep adding libraries as dependencies like I am doing now, but that
means that my library continues to accrete content at an alarming rate and
more importantly every one introduces a greater possibility of build issues,
because I can only operate in an environment where every one of my
dependencies can install.

This further exacerbates the problem that no one would want to add all of my
pedantic instances because to do so they would have to inject a huge brittle
dependency into their package.

The only other alternative that I seem to have at this point in the cabal
packaging system is to create a series of flags for optional functionality.
This solves _my_ problem, in particular it lets me install on a much broader
base of environments, but now the order in which my package was installed
with respect to its dependencies matters. In particular, clients of the
library won't know if they have access to half of the instances, and so are
stuck limiting themselves to working either on a particular computer, or
using the intersection of the functionality I can provide.

Perhaps, what I would ideally like to have would be some kind of 'augments'
or 'codependencies' clause in the cabal file inside of flags and build
targets that indicates packages that should force my package to
reinstall after a package matching the version range inside the
codependencies clause is installed or at least prompt indicatig that new
functionality would be available and what packages you should reinstall.

This would let me have my cake and eat it too. I could provide a wide array
of instances for different stock data types, and I could know that if
someone depends on both, say,  'monoids' and 'parsec 3' that the parsec
instances will be present and usable in my package.

Most importantly, it would allow me to fix my 'expression problem'. Others
could introduce dependencies on the easier to install library allowing me to
shrink the library and I would be able to install in more environments.

-Edward Kmett

On Tue, Apr 7, 2009 at 9:20 AM, John Dorsey <haskell at colquitt.org> wrote:

> John Lato wrote:
> > I think that the proper solution is to break up libraries into
> > separate packages as Jeff suggests (buster, buster-ui, etc.), but then
> > the total packages on hackage would explode.  I don't feel great about
>
> I thought about this a while back and came to the conclusion that the
> package count should only grow by a small contant factor due to this,
> and that's a lot better than dealing with hairy and problematic
> dependencies.
>
> It should usually be:
>
>  libfoo
>  libfoo-blarg
>  libfoo-xyzzy
>  etc.
>
> and more rarely:
>
>  libbar-with-xyzzy
>  libbar-no-xyzzy
>  etc.
>
> each providing libbar.  Although I don't remember whether Cabal has
> 'provides'.  The latter case could explode exponentially for weird
> packages that have several soft dependencies that can't be managed in
> the plugin manner, but I can't see that being a real issue.
>
> This looks manageable to me, but I'm no packaging guru.  I guess it's a
> little harder for authors/maintainers of packages that look like leaves
> in the dependency tree, which could be bad.  Am I missing something bad?
>
> Regards,
> John
>
> _______________________________________________
> Haskell-Cafe mailing list
> Haskell-Cafe at haskell.org
> http://www.haskell.org/mailman/listinfo/haskell-cafe
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: http://www.haskell.org/pipermail/haskell-cafe/attachments/20090407/aea7d1f2/attachment.htm


More information about the Haskell-Cafe mailing list