# Servant-util-beam-pg This package contains backend implementation for API combinators of `servant-util` package via 'beam-postgres'. ## Build Instructions Run `stack build servant-util-beam-pg` to build everything. ## Usage ### Sorting We will demonstrate this on the [previously shown example](/servant-util/README.md#sorting): ```haskell type instance SortingParamBaseOf Book = ["name" ?: Isbn, "author" ?: Text] type instance SortingParamProvidedOf Book = ["isbn" ?: Asc Isbn] type GetBooks :> SortingParamsOf Book = Get '[JSON] [Book] ``` Let's assume that `Book` definition is extended to a Beam table (this is not necessary, but simplifies our example): ```haskell data BookT f = Book { isbn :: C f Isbn , bookName :: C f Text , author :: C f Text } deriving (Generic) type Book = BookT Identity ``` Implementation will look like ```haskell import Servant.Util (SortingSpecOf, HNil, (.*.)) import Servant.Util.Beam.Postgres (fieldSort) -- some code getBooks :: _ => SortingSpecOf Book -> m [Book] getBooks sorting = runSelect . select $ sortBy_ sorting sortingApp $ all_ (books booksSchema) where sortingApp Book{..} = fieldSort @"name" bookName .*. fieldSort @"author" author .*. fieldSort @"isbn" isbn .*. HNil ``` Function `sortingApp` specifies how to correlate user-provided specification with fields of our table. For each field which we allow to sort on, we specify a Beam field from the table. If one of the fields lacks such specification in `sortingApp` definition or order of fields is incorrect then compile error is raised. The same happens when field types in API and schema definition mismatch. Annotating `fieldSort` calls with a field name is fully optional but may save you in case when several fields of the same type participate in sorting; this also improves readability. ### Filtering Filtering is implemented pretty much like sorting. Let's consider the [previously shown example](/servant-util/README.md#filtering): ```haskell type instance SortingParamTypesOf Book = ["isbn" ?: Word64, "name" ?: Text, "hasLongName" ?: Bool] type GetBooks :> SortingParamsOf Book = Get '[JSON] [Book] ``` Implementation can look like ```haskell import Servant.Util (FilteringSpecOf, HNil, (.*.)) import Servant.Util.Beam.Postgres (filterOn, manualFilter, matches_) -- some code getBooks :: _ => FilteringSpecOf Book -> m [Book] getBooks filters = runSelect . select $ do book <- all_ (books booksSchema) guard_ $ matches_ filters (filteringApp book) return book where filteringApp Book{..} = filterOn @"isbn" isbn .*. filterOn @"name" bookName .*. manualFilter @"hasLongName" (\hasLongName -> hasLongName ==. (charLength_ bookName >=. 10)) .*. HNil ``` Again, for each possible user-provided filter you have to provide an implementation. Automatic filters require only a corresponding field of response, they will apply the required operation to it themselves. On the other hand, manual filters carry the value that user provides, and you implement the predicate on this value. Again, type annotations in `filterOn` and `manualFilter` calls are optional. ### Pagination Pagination can be applied simply with `paginate_` function. ## For Contributors Please see [CONTRIBUTING.md](/.github/CONTRIBUTING.md) for more information. ## About Serokell Servant-util is maintained and funded with :heart: by [Serokell](https://serokell.io/). The names and logo for Serokell are trademark of Serokell OÜ. We love open source software! See [our other projects](https://serokell.io/community?utm_source=github) or [hire us](https://serokell.io/hire-us?utm_source=github) to design, develop and grow your idea!