I've been working on a personal project in my spare time for…an awfully long time now. Although a huge part of what it does is necessarily server-based, I want the UI of the project to be offline-mobile-first—that is, in addition to mobile first, it must be able to run offline seamlessly, etc.
I don't have the time or the energy or expertise to seriously consider building my own infrastructure for doing such things, so I've elected to go with CouchDB+PouchDB as my data storage solution. I will admit to a tiny bit of concern about PouchDB—not so much whether it's good, but whether it's capable of handling the data storage needs I envision. Still, even if it's not, it provides a starting point, and there are other options (TouchDB, or Couchbase-Lite or whatever it's called right now) that I can consider.
On the server side—in the stuff I am definitely doing in Haskell, as opposed to the client-side where I would like to be able to use Haskell, but might compromise if necessary—I want, first and foremost, a well-maintained database library.
Unfortunately, of the five libraries to interface with CouchDB, four haven't been updated in at least two years, and even the one that has is a major version behind on one of its primary dependencies. So none of these, IMHO, represent a well-maintained option.
I have, to this point, been using couchdb-conduit (with a bunch of patches I've maintained to keep it compatible with current libs), but I've recently run across an issue whose workaround is annoying enough—trying to handle exceptions when calling a routine from within a segment of a conduit—that I think I'm just going to write my own.
So, my first potentially-public Haskell library. It's actually a little intimidating.
My first step, I think, is to identify what I want
- easy access to the CouchDB API It's actually pretty important to me that this mirror the official API—it allows me to refer to it as documentation, it gives me (and others) a good guide to relatively completeness, it makes tracking any changes easier, etc. It gives me a built in structure.
- good type guarantees for correctness Especially when some of these calls end up feeling like log strings of parameters, I want to make sure the compiler will tell me when I leave one out.
- to process streaming outputs in a streaming fashion Most access to individual records and what-not doesn't require actual streaming—really, just incremental processing of what is ultimately going to be one result. But when you want to handle a bunch of records coming out of a view, or you want to hook into the changes feed? Streaming must be available.
- choice in streaming library At this point, I am more acquainted with Conduit, no question, but I would like to make sure not to exclude users of the Pipes library from being able to provide their own streaming option on top of this.
- implicit parameters (host, port, database) most of the time Most of the time, I just want to stuff all the connection stuff into a Reader instance and never have to mention it again…
- explicit parameters when I need it …except sometimes, when I really need to do something odd in the middle of a bunch of other stuff..
- well maintaned Even if I have to do it myself.
So, that's what I want to achieve. I think it is all achievable, but I'm going to start small and try and build up to it. The great thing is that I already have a body of code that's currently using couchdb-conduit that I can use as a development testbed.