Project

sort_param

0.0
No release in over a year
Sort records using a sort query parameter à la JSON:API format
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
 Dependencies
 Project Readme

SortParam

Gem Version CI codecov Maintainability

Sort records using a query parameter based on JSON API's sort parameter format.

In a nutshell, this gem converts comma-separated sort fields from this:

?sort=+users.first_name,-users.last_name:nulls_last,users.email

to this:

users.first_name asc, users.last_name desc nulls last, users.email asc

or to this:

{"users.first_name"=>{:direction=>:asc}, "users.last_name"=>{:direction=>:desc, :nulls=>:last}, "users.email"=>{:direction=>:asc}}

Features

  • Sort field whitelisting.
  • Supports ORDER BY expression generation for MySQL and PG.
  • Parsing of comma-separated sort fields into hash for any further processing.
  • Specifying NULL sort order.

Installation

Add this line to your application's Gemfile:

gem 'sort_param'

And then execute:

bundle install

Or install it yourself as:

gem install sort_param

Usage

Basic

1. Whitelist/define the sort fields

sort_param = SortParam.define do
               field :first_name, nulls: :first
               field :last_name
             end

This is is equivalent to:

sort_param = SortParam::Definition.new
                                  .field(:first_name, nulls: :first)
                                  .field(:last_name)

field method accepts the column name as the first argument. Any default column configuration such as :nulls(for NULLS FIRST or NULLS LAST sort order) follows the name.

2. Parse sort fields from a parameter

The load! method translates a given sort string/fields parameter to an SQL ORDER BY expression or to a Hash:

I. PostgreSQL example
sort_param.load!("+first_name,-last_name", mode: :pg)

=> "first_name asc nulls first, last_name desc"
II. MySQL example
sort_param.load!("+first_name,-last_name", mode: :mysql)

=> "first_name is not null, first_name asc, last_name desc"
III. Hash example
sort_param.load!("+first_name,-last_name")

=> {"first_name"=>{:nulls=>:first, :direction=>:asc}, "last_name"=>{:direction=>:desc}}

Any other additional column option set in SortParam::Definition or SortParam.define will be included in the column's hash value. For example:

sort_param = SortParam.define do
               field :first_name, foo: :bar, nulls: :first
             end

sort_param.load!("+first_name")
=> {"first_name"=>{:foo=>:bar, :nulls=>:first, :direction=>:asc}}

sort_param.load!("-first_name:nulls_last")
=> {"first_name"=>{:foo=>:bar, :nulls=>:last, :direction=>:desc}}

IV. Example with explicit nulls sort order

Example in PG mode:
sort_param.load!("+first_name:nulls_last,-last_name:nulls_first", mode: :pg)

=> "first_name asc nulls last, last_name desc nulls first"

Ignoring non-whitelisted sort fields instead of raising error

Use #load method instead:

  sort_param = SortParam.define do
                 field :first_name
               end

  sort_param.load("+first_name,+last_name", mode: :pg)
  => "first_name asc"

Output a different field name

Set the :rename field option to a string value or a Proc to output a different field name.

  sort_param = SortParam.define do
                 field :first_name, rename: 'users.name'
                 field :last_name, rename: ->(col) { "users.#{col}" }
               end
  
  sort_param.load!("+first_name", mode: :pg)
  => "users.name asc"
  
  sort_param.load!("+first_name", mode: :mysql)
  => "users.name asc"
  
  sort_param.load!("+first_name")
  => {"users.name"=>{:direction=>:asc}}

  sort_param.load!("+last_name")
  => {"users.last_name"=>{:direction=>:asc}}

Whitelisting multiple fields with the same options at once

Use #fields instead of #field:

  SortParam.define do
    fields :first_name, :last_name, nulls: :first, rename: ->(col) { "users.#{col}" }
    field :email
  end

NOTE: Unlike #field, #fields can only accept a Proc for the :rename option.

Rails example

controller

def index 
  render json: User.all.order(sort_fields)
end

private

def sort_fields
  SortParam.define do
    field :first_name
    field :last_name, nulls: :first
  end.load!(sort_param, mode: :pg)
end

# Fetch the sort fields from :sort query parameter.
# If none is given, default sort by `first_name ASC` and `last_name ASC NULLS FIRST`.
def sort_param
  params[:sort].presence || "+first_name,+last_name"
end

We can DRY this up a bit by creating a concern:

controllers/concerns/has_sort_param.rb

module HasSortParam
  extend ActiveSupport::Concern

  def sort_param(default: nil, &block)
    raise ArgumentError.new('Missing block') unless block_given?

    definition = SortParam.define(&block)
    definition.load!(params[:sort].presence || default, mode: :pg)
  end
end

controller v2

def index 
  render json: User.all.order(sort_fields)
end

private

def sort_fields
  sort_param default: '+first_name,-last_name' do
    field :first_name
    field :last_name, nulls: :first
  end
end

Error

Class Description
SortParam::UnsupportedSortField Raised when loading sort string via #load! and a sort field from the parameter isn't included in the whitelisted sort fields.

Development

After checking out the repo, run bin/setup to install dependencies. Then, run rake spec to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.

To install this gem onto your local machine, run bundle exec rake install.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/jsonb-uy/sort_param.

License

The gem is available as open source under the terms of the MIT License.