bs-sql-common

6.0.0 • Public • Published

NPM Build Status Coverage Status

bs-sql-common

A common interface for SQL-based Node.js drivers.

Why?

To provide a common interface for MySQL, PostgreSQL and sqlite implementations.

Version 3

A rewrite of the entire package to expose it as a Functor that can accept any module which implements the Queryable interface.

  • Use Belt.Result for responses so to better integrate with then BuckleScript ecosystem.

  • Provide response decoding and inspection functions so that the user has a consistent view into responses from any library.

  • Provide an ID type that properly encodes large integers as strings.

  • Provide batch inserts and queries

Status

The standard things are there and this library is being used live within several production projects.

  • Query parameter substitution
  • Named parameters
  • Promise based interface.
  • Connection pooling
  • Custom Streams

Installation

Inside of a BuckleScript project:

yarn install --save bs-sql-common

Then add bs-sql-common to your bs-dependencies in your bsconfig.json

{
  "bs-dependencies": [ "bs-sql-common" ]
}

Then add a bs-sql-common compatible package to your repository or create your own. All of the examples use the bs-mysql2 package, here are the requirements to use that package:

yarn install --save bs-mysql2Â
{
  "bs-dependencies": [ "bs-sql-common", "bs-mysql2" ]
}
module Sql = SqlCommon.Make(MySql2)
 
let db = Sql.Connection.connect
  ~host="127.0.0.1"
  ~port=3306
  ~user="root"
  ()
 
Sql.query ~db ~sql:"SHOW DATABASES" (fun res ->
  match res with
  | Belt.Result.Error e -> raise e
  | Belt.Result.Ok select ->
    select
    |. Sql.Response.Select.flatMap (Json.Decode.dict Json.Decode.string)
    |. Belt.Array.map (fun x -> Js.dict.unsafeGet x "Database")
    |. Expect.expect
    |> Expect.toContain @@ "test"
)

Usage

Note: All of the examples use the bs-mysql2 package as the connection provider. Any other provider should have the same behavior with differing connection creation requirements.

Create a connection and customized module

The following connection and module will be use within the rest of the examples.

module Sql = SqlCommon.Make(MySql2);
 
let db = Sql.Connection.connect(~host="127.0.0.1", ~port=3306, ~user="root", ());

Assume the following statement occurs at the end of each example.

Sql.Connection.close(conn);

Standard Callback Interface

Standard Query Method

Sql.query(~db, ~sql="SHOW DATABASES",
  fun
  | Belt.Result.Error e => Js.log2("ERROR: ", e)
  | Belt.Result.Ok select =>
    select
    |. Sql.Response.Select.rows
    |. Js.log2("RESPONSE ROWS: ", _)
);
 
Sql.mutate(
  ~db,
  ~sql="INSERT INTO test (foo) VALUES (?)",
  ~params=Sql.Params.positional(Json.Encode.([|string("bar")|] |. array)),
  (res) =>
    fun
    | Belt.Result.Error => Js.log2("ERROR: ", e)
    | Belt.Result.Ok mutation =>
      mutation
      |. Sql.Response.Mutation.insertId
      |. Js.log2("INSERT ID: ", _)
);

Prepared Statements - Named Placeholders

let json = Sql.Params.named(
  Json.Encode.(object_([
  ("x", int(1)),
  ("y", int(2)),
  ]))
));
 
let decoder = Json.Encode.array(Json.Encode.int)
 
Sql.query(~db, ~sql:"SELECT :x + :y AS z", ~params, (res) =>
  switch res {
  | Belt.Result.Error => Js.log2("ERROR: ", e)
  | Belt.Result.Ok select =>
    select
    |. Sql.Response.flatMap(decoder)
    |. Js.log2("DECODED ROWS: ", _)
  }
);
 
Sql.mutate(~db, ~sql:"INSERT INTO test (foo, bar) VALUES (:x, :y)", ~params, (res) =>
  switch res {
  | Belt.Result.Error => Js.log2("ERROR: ", e)
  | Belt.Result.Ok mutation =>
    mutation
    |. Sql.Response.Mutation.insertId
    |. Js.log2("INSERT ID: ", _)
  }
);

Prepared Statements - Positional Placeholders

let params = Sql.Params.positional(
  Json.Encode.(array(int, [|5,6|]))
));
 
Sql.query(~db, ~sql:"SELECT 1 + ? + ? AS result", ~params, (res) =>
  switch res {
  | Belt.Result.Error => Js.log2("ERROR: ", e)
  | Belt.Result.Ok select =>
    select
    |. Sql.Response.rows
    |. Js.log2("RAW ROWS: ", _)
  }
);
 
Sql.mutate(~db, ~sql:"INSERT INTO test (foo, bar) VALUES (?, ?)", ~params, (res) =>
  switch res {
  | Belt.Result.Error => Js.log2("ERROR: ", e)
  | Belt.Result.Ok mutation =>
    mutation
    |. Sql.Response.Mutation.insertId
    |. Js.log2("INSERT ID: ", _)
  }
);

Promise Interface

let params = Sql.Params.positional(
  Json.Encode.(array(int, [|"%schema"|]))
));
 
Sql.query(~db, ~params, ~sql="SELECT ? AS search")
|> Js.Promise.then_(select =>
  select
  |. Sql.Response.rows
  |. Js.log2("RAW ROWS: ", _)
  |. ignore
)
|> Js.Promise.catch(err =>
  Js.log2("Failure!!!", err)
  |. ignore
)

Sql.Id

module Id: sig
  type t = Driver.Id.t
 
  val fromJson : Js.Json.t -> Driver.Id.t
 
  val toJson : Driver.Id.t -> Js.Json.t
 
  val toString : Driver.Id.t -> string
end

Sql.Response

module Response: sig
  module Mutation: sig
    val insertId : Driver.Mutation.t -> Id.t option
 
    val affectedRows: Driver.Mutation.t -> int
  end
 
  module Select: sig
    module Meta : sig
      val schema : Driver.Select.Meta.t -> string
 
      val name : Driver.Select.Meta.t -> string
 
      val table : Driver.Select.Meta.t -> string
    end
 
    val meta : Driver.Select.t -> Driver.Select.Meta.t array
 
    val concat : Driver.Select.t -> Driver.Select.t -> Driver.Select.t
 
    val count : Driver.Select.t -> int
 
    val flatMap :
      Driver.Select.t ->
      (Js.Json.t -> Driver.Select.Meta.t array -> 'a) ->
      'a array
 
    val flatMap : Driver.Select.t -> (Js.Json.t -> 'a) -> 'a array
 
    val rows : Driver.Select.t -> Js.Json.t array
  end
end

Queryable Interface

module type Queryable = sig
  module Connection : sig
    type t
 
    val connect :
      ?host:string ->
      ?port:int ->
      ?user:string ->
      ?password:string ->
      ?database:string ->
      unit -> t
 
    val close : t -> unit
  end
 
  module Exn : sig
    val fromJs : Js.Json.t -> exn
  end
 
  module Id : sig
    type t
 
    val fromJson : Js.Json.t -> t
 
    val toJson : t -> Js.Json.t
 
    val toString : t -> string
  end
 
  module Mutation : sig
    type t
 
    val insertId : t -> Id.t option
 
    val affectedRows : t -> int
  end
 
  module Params : sig
    type t
 
    val named : Js.Json.t -> t
 
    val positional : Js.Json.t -> t
  end
 
  module Select : sig
    type t
 
    module Meta : sig
      type t
 
      val schema : t -> string
 
      val name : t -> string
 
      val table : t -> string
    end
 
    val meta : t -> Meta.t array
 
    val concat : t -> t -> t
 
    val count : t -> int
 
    val flatMapWithMeta : t -> (Js.Json.t -> Meta.t array -> 'a) -> 'a array
 
    val flatMap : t -> (Js.Json.t -> 'a) -> 'a array
 
    val rows : t -> Js.Json.t array
  end
 
  type response =
    [
    | `Error of exn
    | `Mutation of Mutation.t
    | `Select of Select.t
    ]
 
  type callback = response -> unit
 
  val execute : Connection.t -> string -> Params.t option -> callback -> unit
end

Package Sidebar

Install

npm i bs-sql-common

Weekly Downloads

31

Version

6.0.0

License

MIT

Unpacked Size

58 kB

Total Files

34

Last publish

Collaborators

  • scull7