OCaml-MariaDB is a library containing
Ctypes-based bindings for MariaDB's
libmysqlclient. The library provides access to the traditional MySQL blocking
API via the Mariadb.Blocking module, as well as the MariaDB nonblocking API,
via Mariadb.Nonblocking, which is designed mainly for use with OCaml's monadic
concurrent programming libraries such as Lwt and
Async.
Only the prepared-statement APIs are exposed by OCaml-MariaDB, as these functions provide typed query parameters and database field access.
OCaml-MariaDB requires MariaDB's client library version 5.5.21 or greater and the C connector library version 2.1.0 or greater. If your distribution doesn't yet package those versions, the easiest way to get them is to configure the appropriate repository and install the development packages from it. For example, on Debian or Ubuntu, run
# apt-get install libmariadbclient-devTo install OCaml-MariaDB via OPAM simply type
$ opam install mariadbTo install it manually, type
$ ocaml setup.ml -configure
$ ocaml setup.ml -build
$ ocaml setup.ml -install
If you want to build the Lwt and/or Async example programs, pass respectively
`--enable-lwt` and `--enable-async` to the `configure` command above.
OCaml-Mariadb itself has no dependency on either of those libraries.OCaml-MariaDB's API should be familiar to those who have used other MySQL libraries in OCaml or other languages before. A query must be initially prepared, resulting in a prepared statement, which can then be executed when given an appropriate set of parameters. Statement execution leads to a query result which can then be used to fetch rows, one at a time.
A simple example is given below.
open Printf
module M = Mariadb.Blocking
let or_die = function
  | Ok x -> x
  | Error (num, msg) -> failwith (sprintf "error #%d: %s" num msg)
let main () =
  let mariadb =
    Mariadb.connect
      ~host:"localhost"
      ~user:"myuser"
      ~pass:"secret" |> or_die in
  let query = "SELECT * FROM mysql.users WHERE Host LIKE ? LIMIT ?" in
  let stmt = M.prepare mariadb query |> or_die in
  let res = M.Stmt.execute stmt [| `String "%"; `Int 10 |] |> or_die in
  printf "number of rows: %d\n%!" (M.Res.num_rows res);
  print_rows res; (* see below *)
  M.Stmt.close stmt |> or_die;
  M.close mariadb
let () = main ()Usage of the nonblocking API is very similar to the blocking one, but designed
to support OCaml's popular monadic concurrency libraries (though usage of a
monadic library is not mandatory -- see the examples directory for an example
using Unix.select).
To use the nonblocking API, a module of type Mariadb.Nonblocking.Wait must be
provided to the functor Mariadb.Nonblocking.Make. This module must contain
an asynchronous I/O type definition along with the usual bind and return
monadic operations, as well as a wait function the specifies how to wait for
the MariaDB socket to become readable and/or writable.
The signature is as follows.
module type Wait = sig
  module IO : sig
    type 'a future
    val (>>=) : 'a future -> ('a -> 'b future) -> 'b future
    val return : 'a -> 'a future
  end
  val wait : t -> Mariadb.Nonblocking.Status.t
          -> Mariadb.Nonblocking.Status.t IO.future
end
The wait function receives a status parameter that specifies which socket
events are to be waited for (which can be checked via Status.read,
Status.write and Status.timeout -- see the ocamldoc for more details).
It must then return a new status, specifying which of those events have actually
occurred.
A simple example of the nonblocking library usage is given below. Full examples
of wait modules for Lwt and Async can be found in the examples directory.
module M = Mariadb.Nonblocking.Make(struct
  module IO = struct
    type 'a future = ...
    let (>>=) m f = ...
    let return x = ...
  end
  let wait mariadb status =
    ...
end)
let main () =
  Mariadb.connect
    ~host:"localhost"
    ~user:"myuser"
    ~pass:"secret" >>= or_die
  >>= fun mariadb ->
  let query = "SELECT * FROM mysql.users WHERE Host LIKE ? LIMIT ?" in
  M.prepare mariadb query >>= or_die
  >>= fun stmt ->
  M.Stmt.execute stmt [| `String "%"; `Int 10 |] >>= or_die
  >>= fun res ->
  print_rows res >>= fun () -> (* see below *)
  M.Stmt.close stmt >>= or_die >>= fun () ->
  M.close mariadbRows can be fetched using the Res.fetch function. This function takes a
module as its first parameter that defines the data structure in which the
row is to be fetched.
For example,
M.Res.fetch (module M.Row.Array) resreturns the row as an array of Field.t values. The following built-in
modules are provided with OCaml-MariaDB, but the user is free to implement
one if so desired, in which case it must conform to the Row.S module type
(see the ocamldoc for details).
- Row.Array: fetch row as- Field.t array;
- Row.Map: fetch row as a map of column name (- string) to- Field.t;
- Row.Hashtbl: fetch row as a- (string, Field.t) Hashtbl.t.
The fetch function returns a row option result, where row represents the
row type given by the module argument, and result is a wrapper for the
Pervasives.result type that carries an error in the Error case. In the
Ok case, fetch returns Some row, containing the next available row, or
None, in which case no more rows are available in the result.
A database field is represented by the Field.t type, and its value by the
Field.value type, which can be obtained by calling the Field.value
function.
type value =
  [ `Int of int
  | `Float of float
  | `String of string
  | `Bytes of bytes
  | `Time of Time.t
  ]The Field.value function can also return ```Null`` in case the field is an
SQL NULL.
Since the type of a field is in most cases known beforehand, as the user must be aware of the table definition in order to query it, helper functions are provided to extract the OCaml types directly from a field:
val int : Field.t -> int
val float : Field.t -> float
val string : Field.t -> string
val bytes : Field.t -> bytes
val time : Field.t -> Time.tThese functions will raise an exception if the field value is not of the expected type, but as noted above this shouldn't be a problem.
For nullable fields, the following analogous functions are also provided:
val int_opt : Field.t -> int option
val float_opt : Field.t -> float option
val string_opt : Field.t -> string option
val bytes_opt : Field.t -> bytes option
val time_opt : Field.t -> Time.t optionThese functions return None if the field value is ```Null``, or Some v
otherwise.