adam@1656:  (* Copyright (c) 2009-2010, Adam Chlipala
adamc@885:  * All rights reserved.
adamc@885:  *
adamc@885:  * Redistribution and use in source and binary forms, with or without
adamc@885:  * modification, are permitted provided that the following conditions are met:
adamc@885:  *
adamc@885:  * - Redistributions of source code must retain the above copyright notice,
adamc@885:  *   this list of conditions and the following disclaimer.
adamc@885:  * - Redistributions in binary form must reproduce the above copyright notice,
adamc@885:  *   this list of conditions and the following disclaimer in the documentation
adamc@885:  *   and/or other materials provided with the distribution.
adamc@885:  * - The names of contributors may not be used to endorse or promote products
adamc@885:  *   derived from this software without specific prior written permission.
adamc@885:  *
adamc@885:  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
adamc@885:  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
adamc@885:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
adamc@885:  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
adam@1682:  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
adamc@885:  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
adamc@885:  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
adamc@885:  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
adamc@885:  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
adamc@885:  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
adamc@885:  * POSSIBILITY OF SUCH DAMAGE.
adamc@885:  *)
adamc@885: 
adamc@885: structure SQLite :> SQLITE = struct
adamc@885: 
adamc@885: open Settings
adamc@885: open Print.PD
adamc@885: open Print
adamc@885: 
adamc@885: fun p_sql_type t =
adamc@885:     case t of
adamc@885:         Int => "integer"
adamc@885:       | Float => "real"
adamc@885:       | String => "text"
adamc@1014:       | Char => "text"
adamc@885:       | Bool => "integer"
adamc@887:       | Time => "text"
adamc@885:       | Blob => "blob"
adamc@885:       | Channel => "integer"
adamc@885:       | Client => "integer"
adamc@885:       | Nullable t => p_sql_type t
adamc@885: 
adamc@885: val ident = String.translate (fn #"'" => "PRIME"
adamc@885:                                | ch => str ch)
adamc@885: 
adamc@885: fun checkRel (table, checkNullable) (s, xts) =
adamc@885:     let
adamc@885:         val q = "SELECT COUNT(*) FROM sqlite_master WHERE type = '" ^ table ^ "' AND name = '"
adamc@885:                 ^ s ^ "'"
adamc@885:     in
adamc@885:         box [string "if (sqlite3_prepare_v2(conn->conn, \"",
adamc@885:              string q,
adamc@885:              string "\", -1, &stmt, NULL) != SQLITE_OK) {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3_close(conn->conn);",
adamc@885:                   newline,
adamc@1266:                   string "uw_error(ctx, FATAL, \"Query preparation failed:<br />",
adamc@885:                   string q,
adamc@885:                   string "\");",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@885:              string "while ((res = sqlite3_step(stmt)) == SQLITE_BUSY)",
adamc@885:              newline,
adamc@885:              box [string "sleep(1);",
adamc@885:                   newline],
adamc@885:              newline,
adamc@885:              string "if (res == SQLITE_DONE) {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3_finalize(stmt);",
adamc@885:                   newline,
adamc@885:                   string "sqlite3_close(conn->conn);",
adamc@885:                   newline,
adamc@1266:                   string "uw_error(ctx, FATAL, \"No row returned:<br />",
adamc@885:                   string q,
adamc@885:                   string "\");",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885:              string "if (res != SQLITE_ROW) {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3_finalize(stmt);",
adamc@885:                   newline,
adamc@885:                   string "sqlite3_close(conn->conn);",
adamc@885:                   newline,
adamc@1266:                   string "uw_error(ctx, FATAL, \"Error getting row:<br />",
adamc@885:                   string q,
adamc@885:                   string "\");",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@885:              string "if (sqlite3_column_count(stmt) != 1) {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3_finalize(stmt);",
adamc@885:                   newline,
adamc@885:                   string "sqlite3_close(conn->conn);",
adamc@885:                   newline,
adamc@1266:                   string "uw_error(ctx, FATAL, \"Bad column count:<br />",
adamc@885:                   string q,
adamc@885:                   string "\");",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@885:              string "if (sqlite3_column_int(stmt, 0) != 1) {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3_finalize(stmt);",
adamc@885:                   newline,
adamc@885:                   string "sqlite3_close(conn->conn);",
adamc@885:                   newline,
adamc@885:                   string "uw_error(ctx, FATAL, \"Table '",
adamc@885:                   string s,
adamc@885:                   string "' does not exist.\");",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885:              string "sqlite3_finalize(stmt);",
adamc@885:              newline]
adamc@885:     end
adamc@885: 
adamc@885: fun init {dbstring, prepared = ss, tables, views, sequences} =
adamc@885:     let
adamc@885:         val db = ref dbstring
adamc@885:     in
adamc@885:         app (fn s =>
adamc@885:                 case String.fields (fn ch => ch = #"=") s of
adamc@885:                     [name, value] =>
adamc@885:                     (case name of
adamc@885:                          "dbname" => db := value
adamc@885:                        | _ => ())
adamc@885:                   | _ => ()) (String.tokens Char.isSpace dbstring);
adamc@885: 
adamc@885:         box [string "typedef struct {",
adamc@885:              newline,
adamc@885:              box [string "sqlite3 *conn;",
adamc@885:                   newline,
adamc@885:                   p_list_sepi (box [])
adamc@885:                               (fn i => fn _ =>
adamc@885:                                           box [string "sqlite3_stmt *p",
adamc@885:                                                string (Int.toString i),
adamc@885:                                                string ";",
adamc@885:                                                newline])
adamc@885:                               ss],
adamc@885:              string "} uw_conn;",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@1094:              string "static void uw_client_init(void) {",
adamc@885:              newline,
adamc@885:              box [string "uw_sqlfmtInt = \"%lld%n\";",
adamc@885:                   newline,
adamc@885:                   string "uw_sqlfmtFloat = \"%g%n\";",
adamc@885:                   newline,
adamc@885:                   string "uw_Estrings = 0;",
adamc@885:                   newline,
adamc@885:                   string "uw_sqlsuffixString = \"\";",
adamc@885:                   newline,
adamc@1011:                   string "uw_sqlsuffixChar = \"\";",
adamc@1011:                   newline,
adamc@885:                   string "uw_sqlsuffixBlob = \"\";",
adamc@885:                   newline,
adamc@885:                   string "uw_sqlfmtUint4 = \"%u%n\";",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@885:              if #persistent (currentProtocol ()) then
adamc@885:                  box [string "static void uw_db_validate(uw_context ctx) {",
adamc@885:                       newline,
adamc@885:                       string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:                       newline,
adamc@885:                       string "sqlite3_stmt *stmt;",
adamc@885:                       newline,
adamc@885:                       string "int res;",
adamc@885:                       newline,
adamc@885:                       newline,
adamc@885:                       p_list_sep newline (checkRel ("table", true)) tables,
adamc@885:                       p_list_sep newline (fn name => checkRel ("table", true)
adamc@885:                                                               (name, [("id", Settings.Client)])) sequences,
adamc@885:                       p_list_sep newline (checkRel ("view", false)) views,
adamc@885:                       string "}",
adamc@885:                       newline,
adamc@885:                       newline,
adamc@885: 
adamc@885:                       string "static void uw_db_prepare(uw_context ctx) {",
adamc@885:                       newline,
adamc@885:                       string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:                       newline,
adamc@885:                       newline,
adamc@885: 
adamc@885:                       p_list_sepi newline (fn i => fn (s, n) =>
adamc@885:                                                       let
adamc@885:                                                           fun uhoh this s args =
adamc@885:                                                               box [p_list_sepi (box [])
adamc@885:                                                                                (fn j => fn () =>
adamc@885:                                                                                            box [string
adamc@885:                                                                                                     "sqlite3_finalize(conn->p",
adamc@885:                                                                                                 string (Int.toString j),
adamc@885:                                                                                                 string ");",
adamc@885:                                                                                                 newline])
adamc@885:                                                                                (List.tabulate (i, fn _ => ())),
adamc@885:                                                                    box (if this then
adamc@885:                                                                             [string
adamc@885:                                                                                  "sqlite3_finalize(conn->p",
adamc@885:                                                                              string (Int.toString i),
adamc@885:                                                                              string ");",
adamc@885:                                                                              newline]
adamc@885:                                                                         else
adamc@885:                                                                             []),
adamc@885:                                                                    string "sqlite3_close(conn->conn);",
adamc@885:                                                                    newline,
adamc@885:                                                                    string "uw_error(ctx, FATAL, \"",
adamc@885:                                                                    string s,
adamc@885:                                                                    string "\"",
adamc@885:                                                                    p_list_sep (box []) (fn s => box [string ", ",
adamc@885:                                                                                                      string s]) args,
adamc@885:                                                                    string ");",
adamc@885:                                                                    newline]
adamc@885:                                                       in
adamc@885:                                                           box [string "if (sqlite3_prepare_v2(conn->conn, \"",
adam@1656:                                                                string (Prim.toCString s),
adamc@885:                                                                string "\", -1, &conn->p",
adamc@885:                                                                string (Int.toString i),
adamc@885:                                                                string ", NULL) != SQLITE_OK) {",
adamc@885:                                                                newline,
adamc@886:                                                                box [string "char msg[1024];",
adamc@886:                                                                     newline,
adamc@886:                                                                     string "strncpy(msg, sqlite3_errmsg(conn->conn), 1024);",
adamc@886:                                                                     newline,
adamc@886:                                                                     string "msg[1023] = 0;",
adamc@886:                                                                     newline,
adamc@886:                                                                     uhoh false ("Error preparing statement: "
adam@1656:                                                                                 ^ Prim.toCString s ^ "<br />%s") ["msg"]],
adamc@885:                                                                string "}",
adamc@885:                                                                newline]
adamc@885:                                                       end)
adamc@885:                                   ss,
adamc@885: 
adamc@885:                       string "}"]
adamc@885:              else
adamc@885:                  box [string "static void uw_db_prepare(uw_context ctx) { }",
adamc@885:                       newline,
adamc@885:                       string "static void uw_db_validate(uw_context ctx) { }"],
adamc@885:              newline,
adamc@885:              newline,
adam@1682: 
adamc@1094:              string "static void uw_db_init(uw_context ctx) {",
adamc@885:              newline,
adamc@885:              string "sqlite3 *sqlite;",
adamc@885:              newline,
adamc@1115:              string "sqlite3_stmt *stmt;",
adamc@1115:              newline,
adamc@885:              string "uw_conn *conn;",
adamc@885:              newline,
adamc@885:              newline,
adamc@885:              string "if (sqlite3_open(\"",
adamc@885:              string (!db),
adamc@885:              string "\", &sqlite) != SQLITE_OK) uw_error(ctx, FATAL, ",
adamc@885:              string "\"Can't open SQLite database.\");",
adamc@885:              newline,
adamc@885:              newline,
adamc@1115:              string "if (uw_database_max < SIZE_MAX) {",
adamc@1115:              newline,
adamc@1115:              box [string "char buf[100];",
adamc@1115:                   newline,
adamc@1115:                   newline,
adamc@1115: 
adamc@1115:                   string "sprintf(buf, \"PRAGMA max_page_count = %llu\", (unsigned long long)(uw_database_max / 1024));",
adamc@1115:                   newline,
adamc@1115:                   newline,
adamc@1115: 
adamc@1115:                   string "if (sqlite3_prepare_v2(sqlite, buf, -1, &stmt, NULL) != SQLITE_OK) {",
adamc@1115:                   newline,
adamc@1115:                   box [string "sqlite3_close(sqlite);",
adamc@1115:                        newline,
adamc@1115:                        string "uw_error(ctx, FATAL, \"Can't prepare max_page_count query for SQLite database\");",
adamc@1115:                        newline],
adamc@1115:                   string "}",
adamc@1115:                   newline,
adamc@1115:                   newline,
adamc@1115: 
adamc@1115:                   string "if (sqlite3_step(stmt) != SQLITE_ROW) {",
adamc@1115:                   newline,
adamc@1115:                   box [string "sqlite3_finalize(stmt);",
adamc@1115:                        newline,
adamc@1115:                        string "sqlite3_close(sqlite);",
adamc@1115:                        newline,
adamc@1115:                        string "uw_error(ctx, FATAL, \"Can't set max_page_count parameter for SQLite database\");",
adamc@1115:                        newline],
adamc@1115:                   string "}",
adamc@1115:                   newline,
adamc@1115:                   newline,
adamc@1115: 
adamc@1115:                   string "sqlite3_finalize(stmt);",
adamc@1115:                   newline],
adamc@1115:              string "}",
adamc@1115:              newline,
adamc@1115:              newline,
adam@1682: 
adamc@885:              string "conn = calloc(1, sizeof(uw_conn));",
adamc@885:              newline,
adamc@885:              string "conn->conn = sqlite;",
adamc@885:              newline,
adamc@885:              string "uw_set_db(ctx, conn);",
adamc@885:              newline,
adamc@885:              string "uw_db_validate(ctx);",
adamc@885:              newline,
adamc@885:              string "uw_db_prepare(ctx);",
adamc@885:              newline,
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@1094:              string "static void uw_db_close(uw_context ctx) {",
adamc@885:              newline,
adamc@885:              string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:              newline,
adamc@885:              p_list_sepi (box [])
adamc@885:                          (fn i => fn _ =>
adamc@885:                                      box [string "if (conn->p",
adamc@885:                                           string (Int.toString i),
adamc@885:                                           string ") sqlite3_finalize(conn->p",
adamc@885:                                           string (Int.toString i),
adamc@885:                                           string ");",
adamc@885:                                           newline])
adamc@885:                          ss,
adamc@885:              string "sqlite3_close(conn->conn);",
adamc@885:              newline,
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@1094:              string "static int uw_db_begin(uw_context ctx) {",
adamc@885:              newline,
adamc@885:              string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:              newline,
adamc@885:              newline,
adamc@885:              string "if (sqlite3_exec(conn->conn, \"BEGIN\", NULL, NULL, NULL) == SQLITE_OK)",
adamc@885:              newline,
adamc@885:              box [string "return 0;",
adamc@885:                   newline],
adamc@885:              string "else {",
adamc@885:              newline,
adamc@1266:              box [string "fprintf(stderr, \"Begin error: %s<br />\", sqlite3_errmsg(conn->conn));",
adamc@885:                   newline,
adamc@885:                   string "return 1;",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              string "}",
adamc@885:              newline,
adamc@1094:              string "static int uw_db_commit(uw_context ctx) {",
adamc@885:              newline,
adamc@885:              string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:              newline,
adamc@885:              string "if (sqlite3_exec(conn->conn, \"COMMIT\", NULL, NULL, NULL) == SQLITE_OK)",
adamc@885:              newline,
adamc@885:              box [string "return 0;",
adamc@885:                   newline],
adamc@885:              string "else {",
adamc@885:              newline,
adamc@1266:              box [string "fprintf(stderr, \"Commit error: %s<br />\", sqlite3_errmsg(conn->conn));",
adamc@885:                   newline,
adamc@885:                   string "return 1;",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline,
adamc@885: 
adamc@1094:              string "static int uw_db_rollback(uw_context ctx) {",
adamc@885:              newline,
adamc@885:              string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:              newline,
adamc@885:              string "if (sqlite3_exec(conn->conn, \"ROLLBACK\", NULL, NULL, NULL) == SQLITE_OK)",
adamc@885:              newline,
adamc@885:              box [string "return 0;",
adamc@885:                   newline],
adamc@885:              string "else {",
adamc@885:              newline,
adamc@1266:              box [string "fprintf(stderr, \"Rollback error: %s<br />\", sqlite3_errmsg(conn->conn));",
adamc@885:                   newline,
adamc@885:                   string "return 1;",
adamc@885:                   newline],
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              string "}",
adamc@885:              newline,
adamc@885:              newline]
adamc@885:     end
adamc@885: 
adam@1352: val fmt = "\"%Y-%m-%d %H:%M:%S\""
adam@1352: 
adamc@885: fun p_getcol {loc, wontLeakStrings, col = i, typ = t} =
adamc@885:     let
adamc@885:         fun p_unsql t =
adamc@885:             case t of
adamc@885:                 Int => box [string "sqlite3_column_int64(stmt, ", string (Int.toString i), string ")"]
adamc@885:               | Float => box [string "sqlite3_column_double(stmt, ", string (Int.toString i), string ")"]
adamc@885:               | String =>
adamc@885:                 if wontLeakStrings then
adamc@1014:                     box [string "(uw_Basis_string)sqlite3_column_text(stmt, ", string (Int.toString i), string ")"]
adamc@885:                 else
adamc@1014:                     box [string "uw_strdup(ctx, (uw_Basis_string)sqlite3_column_text(stmt, ", string (Int.toString i), string "))"]
adamc@1014:               | Char => box [string "sqlite3_column_text(stmt, ", string (Int.toString i), string ")[0]"]
adamc@885:               | Bool => box [string "(uw_Basis_bool)sqlite3_column_int(stmt, ", string (Int.toString i), string ")"]
adam@1352:               | Time => box [string "uw_Basis_stringToTimef_error(ctx, ",
adam@1352:                              string fmt,
adam@1352:                              string ", (uw_Basis_string)sqlite3_column_text(stmt, ",
adam@1352:                              string (Int.toString i),
adam@1352:                              string "))"]
adamc@885:               | Blob => box [string "({",
adamc@885:                              newline,
adamc@890:                              string "char *data = (char *)sqlite3_column_blob(stmt, ",
adamc@885:                              string (Int.toString i),
adamc@885:                              string ");",
adamc@885:                              newline,
adamc@890:                              string "int len = sqlite3_column_bytes(stmt, ",
adamc@885:                              string (Int.toString i),
adamc@890:                              string ");",
adamc@890:                              newline,
adamc@890:                              string "uw_Basis_blob b = {len, uw_memdup(ctx, data, len)};",
adamc@885:                              newline,
adamc@885:                              string "b;",
adamc@885:                              newline,
adamc@885:                              string "})"]
adamc@886:               | Channel => box [string "({",
adamc@886:                                 newline,
adamc@886:                                 string "sqlite3_int64 n = sqlite3_column_int64(stmt, ",
adamc@886:                                 string (Int.toString i),
adamc@886:                                 string ");",
adamc@886:                                 newline,
adamc@886:                                 string "uw_Basis_channel ch = {n >> 32, n & 0xFFFFFFFF};",
adamc@886:                                 newline,
adamc@886:                                 string "ch;",
adamc@886:                                 newline,
adamc@886:                                 string "})"]
adamc@885:               | Client => box [string "sqlite3_column_int(stmt, ", string (Int.toString i), string ")"]
adamc@885: 
adamc@885:               | Nullable _ => raise Fail "Postgres: Recursive Nullable"
adamc@885: 
adamc@885:         fun getter t =
adamc@885:             case t of
adamc@885:                 Nullable t =>
adamc@885:                 box [string "(sqlite3_column_type(stmt, ",
adamc@885:                      string (Int.toString i),
adamc@885:                      string ") == SQLITE_NULL ? NULL : ",
adamc@885:                      case t of
adamc@885:                          String => getter t
adamc@885:                        | _ => box [string "({",
adamc@885:                                    newline,
adamc@885:                                    string (p_sql_ctype t),
adamc@885:                                    space,
adamc@885:                                    string "*tmp = uw_malloc(ctx, sizeof(",
adamc@885:                                    string (p_sql_ctype t),
adamc@885:                                    string "));",
adamc@885:                                    newline,
adamc@885:                                    string "*tmp = ",
adamc@885:                                    getter t,
adamc@885:                                    string ";",
adamc@885:                                    newline,
adamc@885:                                    string "tmp;",
adamc@885:                                    newline,
adamc@885:                                    string "})"],
adamc@885:                      string ")"]
adamc@885:               | _ =>
adamc@885:                 box [string "(sqlite3_column_type(stmt, ",
adamc@885:                      string (Int.toString i),
adamc@885:                      string ") == SQLITE_NULL ? ",
adamc@885:                      box [string "({",
adamc@885:                           string (p_sql_ctype t),
adamc@885:                           space,
adamc@885:                           string "tmp;",
adamc@885:                           newline,
adamc@885:                           string "uw_error(ctx, FATAL, \"",
adamc@885:                           string (ErrorMsg.spanToString loc),
adamc@885:                           string ": Unexpectedly NULL field #",
adamc@885:                           string (Int.toString i),
adamc@885:                           string "\");",
adamc@885:                           newline,
adamc@885:                           string "tmp;",
adamc@885:                           newline,
adamc@885:                           string "})"],
adamc@885:                      string " : ",
adamc@885:                      p_unsql t,
adamc@885:                      string ")"]
adamc@885:     in
adamc@885:         getter t
adamc@885:     end
adamc@885: 
adamc@885: fun queryCommon {loc, query, cols, doCols} =
adamc@885:     box [string "int r;",
adamc@885:          newline,
adamc@885: 
adamc@885:          string "sqlite3_reset(stmt);",
adamc@885:          newline,
adamc@885: 
adamc@885:          string "uw_end_region(ctx);",
adamc@885:          newline,
adamc@885:          string "while ((r = sqlite3_step(stmt)) == SQLITE_ROW) {",
adamc@885:          newline,
adamc@885:          doCols p_getcol,
adamc@885:          string "}",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adamc@885:          string "if (r == SQLITE_BUSY) {",
adamc@885:          box [string "sleep(1);",
adamc@885:               newline,
adamc@885:               string "uw_error(ctx, UNLIMITED_RETRY, \"Database is busy\");",
adamc@885:               newline],
adamc@885:          string "}",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adamc@885:          string "if (r != SQLITE_DONE) uw_error(ctx, FATAL, \"",
adamc@885:          string (ErrorMsg.spanToString loc),
adamc@1266:          string ": query step failed: %s<br />%s\", ",
adamc@885:          query,
adamc@885:          string ", sqlite3_errmsg(conn->conn));",
adamc@885:          newline,
adamc@885:          newline]
adamc@885: 
adamc@885: fun query {loc, cols, doCols} =
adamc@885:     box [string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:          newline,
adamc@886:          string "sqlite3_stmt *stmt;",
adamc@885:          newline,
adamc@885:          newline,
adamc@1266:          string "if (sqlite3_prepare_v2(conn->conn, query, -1, &stmt, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"Error preparing statement: %s<br />%s\", sqlite3_errmsg(conn->conn), query);",
adamc@885:          newline,
adamc@885:          newline,
adamc@885:          string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_finalize, stmt);",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adamc@885:          queryCommon {loc = loc, cols = cols, doCols = doCols, query = string "query"},
adamc@885: 
adamc@885:          string "uw_pop_cleanup(ctx);",
adamc@885:          newline]
adamc@885: 
adamc@1014: val p_pre_inputs =
adamc@1014:     p_list_sepi (box [])
adamc@1014:                 (fn i => fn t =>
adamc@1014:                             case t of
adamc@1014:                                 Char => box [string "char arg",
adamc@1014:                                              string (Int.toString (i + 1)),
adamc@1014:                                              string "s = {arg",
adamc@1014:                                              string (Int.toString (i + 1)),
adamc@1014:                                              string ", 0};",
adamc@1014:                                              newline]
adamc@1014:                               | _ => box [])
adamc@1014: 
adamc@885: fun p_inputs loc =
adamc@885:     p_list_sepi (box [])
adamc@885:                 (fn i => fn t =>
adamc@885:                             let
adamc@885:                                 fun bind (t, arg) =
adamc@885:                                     case t of
adamc@885:                                         Int => box [string "sqlite3_bind_int64(stmt, ",
adamc@885:                                                     string (Int.toString (i + 1)),
adamc@885:                                                     string ", ",
adamc@885:                                                     arg,
adamc@885:                                                     string ")"]
adamc@885:                                       | Float => box [string "sqlite3_bind_double(stmt, ",
adamc@885:                                                       string (Int.toString (i + 1)),
adamc@885:                                                       string ", ",
adamc@885:                                                       arg,
adamc@1014:                                                       string ")"]
adamc@885:                                       | String => box [string "sqlite3_bind_text(stmt, ",
adamc@885:                                                        string (Int.toString (i + 1)),
adamc@885:                                                        string ", ",
adamc@885:                                                        arg,
adamc@885:                                                        string ", -1, SQLITE_TRANSIENT)"]
adamc@1014:                                       | Char => box [string "sqlite3_bind_text(stmt, ",
adamc@1014:                                                      string (Int.toString (i + 1)),
adamc@1014:                                                      string ", ",
adamc@1014:                                                      arg,
adamc@1014:                                                      string "s, -1, SQLITE_TRANSIENT)"]
adamc@885:                                       | Bool => box [string "sqlite3_bind_int(stmt, ",
adamc@885:                                                      string (Int.toString (i + 1)),
adamc@885:                                                      string ", ",
adamc@885:                                                      arg,
adamc@885:                                                      string ")"]
adamc@887:                                       | Time => box [string "sqlite3_bind_text(stmt, ",
adamc@885:                                                      string (Int.toString (i + 1)),
adam@1359:                                                      string ", uw_Basis_timef(ctx, ",
adam@1352:                                                      string fmt,
adam@1352:                                                      string ", ",
adamc@885:                                                      arg,
adamc@887:                                                      string "), -1, SQLITE_TRANSIENT)"]
adamc@885:                                       | Blob => box [string "sqlite3_bind_blob(stmt, ",
adamc@885:                                                      string (Int.toString (i + 1)),
adamc@885:                                                      string ", ",
adamc@885:                                                      arg,
adamc@885:                                                      string ".data, ",
adamc@885:                                                      arg,
adamc@890:                                                      string ".size, SQLITE_TRANSIENT)"]
adamc@886:                                       | Channel => box [string "sqlite3_bind_int64(stmt, ",
adamc@885:                                                         string (Int.toString (i + 1)),
adamc@886:                                                         string ", ((sqlite3_int64)",
adamc@885:                                                         arg,
adamc@886:                                                         string ".cli << 32) | ",
adamc@886:                                                         arg,
adamc@886:                                                         string ".chn)"]
adamc@885:                                       | Client => box [string "sqlite3_bind_int(stmt, ",
adamc@885:                                                        string (Int.toString (i + 1)),
adamc@885:                                                        string ", ",
adamc@885:                                                        arg,
adamc@885:                                                        string ")"]
adamc@885:                                       | Nullable t => box [string "(",
adamc@885:                                                            arg,
adamc@885:                                                            string " == NULL ? sqlite3_bind_null(stmt, ",
adamc@885:                                                            string (Int.toString (i + 1)),
adamc@885:                                                            string ") : ",
adamc@885:                                                            bind (t, case t of
adamc@885:                                                                         String => arg
adamc@885:                                                                       | _ => box [string "(*", arg, string ")"]),
adamc@885:                                                            string ")"]
adamc@885:                             in
adamc@885:                                 box [string "if (",
adamc@885:                                      bind (t, box [string "arg", string (Int.toString (i + 1))]),
adamc@885:                                      string " != SQLITE_OK) uw_error(ctx, FATAL, \"",
adamc@885:                                      string (ErrorMsg.spanToString loc),
adamc@885:                                      string ": Error binding parameter #",
adamc@885:                                      string (Int.toString (i + 1)),
adamc@885:                                      string ": %s\", sqlite3_errmsg(conn->conn));",
adamc@885:                                      newline]
adamc@885:                             end)
adamc@885: 
adamc@885: fun queryPrepared {loc, id, query, inputs, cols, doCols, nested} =
adamc@885:     box [string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:          newline,
adamc@1014:          p_pre_inputs inputs,
adamc@885:          if nested then
adamc@885:              box [string "sqlite3_stmt *stmt;",
adamc@885:                   newline]
adamc@885:          else
adamc@885:              box [string "sqlite3_stmt *stmt = conn->p",
adamc@885:                   string (Int.toString id),
adamc@885:                   string ";",
adamc@885:                   newline,
adamc@885:                   newline,
adamc@885: 
adamc@885:                   string "if (stmt == NULL) {",
adamc@885:                   newline],
adamc@885: 
adamc@885:          string "if (sqlite3_prepare_v2(conn->conn, \"",
adam@1656:          string (Prim.toCString query),
adamc@885:          string "\", -1, &stmt, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"Error preparing statement: ",
adam@1656:          string (Prim.toCString query),
adamc@1266:          string "<br />%s\", sqlite3_errmsg(conn->conn));",
adamc@885:          newline,
adamc@885:          if nested then
adamc@885:              box [string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_finalize, stmt);",
adamc@885:                   newline]
adamc@885:          else
adamc@885:              box [string "conn->p",
adamc@885:                   string (Int.toString id),
adamc@885:                   string " = stmt;",
adamc@885:                   newline,
adamc@885:                   string "}",
adamc@885:                   newline,
adamc@885:                   newline,
adamc@885:                   string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_clear_bindings, stmt);",
adamc@885:                   newline,
adamc@885:                   string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_reset, stmt);",
adamc@885:                   newline],
adamc@885:          newline,
adamc@885: 
adamc@885:          p_inputs loc inputs,
adamc@885:          newline,
adamc@885: 
adamc@885:          queryCommon {loc = loc, cols = cols, doCols = doCols, query = box [string "\"",
adam@1656:                                                                             string (Prim.toCString query),
adamc@885:                                                                             string "\""]},
adamc@885: 
adamc@885:          string "uw_pop_cleanup(ctx);",
adamc@885:          newline,
adamc@885:          if nested then
adamc@885:              box []
adamc@885:          else
adamc@885:              box [string "uw_pop_cleanup(ctx);",
adamc@885:                   newline]]
adamc@885: 
adam@1293: fun dmlCommon {loc, dml, mode} =
adamc@885:     box [string "int r;",
adamc@885:          newline,
adamc@885: 
adamc@885:          string "if ((r = sqlite3_step(stmt)) == SQLITE_BUSY) {",
adamc@885:          box [string "sleep(1);",
adamc@885:               newline,
adamc@885:               string "uw_error(ctx, UNLIMITED_RETRY, \"Database is busy\");",
adamc@885:               newline],
adamc@885:          string "}",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adam@1293:          string "if (r != SQLITE_DONE) ",
adam@1293:          case mode of
adam@1293:              Settings.Error => box [string "uw_error(ctx, FATAL, \"",
adam@1293:                                     string (ErrorMsg.spanToString loc),
adam@1293:                                     string ": DML step failed: %s<br />%s\", ",
adam@1293:                                     dml,
adam@1293:                                     string ", sqlite3_errmsg(conn->conn));"]
adam@1295:            | Settings.None => string "uw_set_error_message(ctx, sqlite3_errmsg(conn->conn));",
adamc@885:          newline]
adamc@885: 
adam@1293: fun dml (loc, mode) =
adamc@885:     box [string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:          newline,
adamc@886:          string "sqlite3_stmt *stmt;",
adamc@885:          newline,
adamc@885:          newline,
adamc@1266:          string "if (sqlite3_prepare_v2(conn->conn, dml, -1, &stmt, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"Error preparing statement: %s<br />%s\", dml, sqlite3_errmsg(conn->conn));",
adamc@885:          newline,
adamc@885:          newline,
adamc@885:          string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_finalize, stmt);",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adam@1293:          dmlCommon {loc = loc, dml = string "dml", mode = mode},
adamc@885: 
adamc@885:          string "uw_pop_cleanup(ctx);",
adamc@885:          newline]
adamc@885: 
adam@1293: fun dmlPrepared {loc, id, dml, inputs, mode = mode} =
adamc@885:     box [string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:          newline,
adamc@1014:          p_pre_inputs inputs,
adamc@885:          string "sqlite3_stmt *stmt = conn->p",
adamc@885:          string (Int.toString id),
adamc@885:          string ";",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adamc@885:          string "if (stmt == NULL) {",
adamc@885:          newline,
adamc@885:          box [string "if (sqlite3_prepare_v2(conn->conn, \"",
adam@1656:               string (Prim.toCString dml),
adamc@885:               string "\", -1, &stmt, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"Error preparing statement: ",
adam@1656:               string (Prim.toCString dml),
adamc@1266:               string "<br />%s\", sqlite3_errmsg(conn->conn));",
adamc@885:               newline,
adamc@885:               string "conn->p",
adamc@885:               string (Int.toString id),
adamc@885:               string " = stmt;",
adamc@885:               newline],
adamc@885:          string "}",
adamc@885:          newline,
adamc@885: 
adamc@885:          string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_clear_bindings, stmt);",
adamc@885:          newline,
adamc@885:          string "uw_push_cleanup(ctx, (void (*)(void *))sqlite3_reset, stmt);",
adamc@885:          newline,
adamc@885: 
adamc@885:          p_inputs loc inputs,
adamc@885:          newline,
adamc@885: 
adamc@885:          dmlCommon {loc = loc, dml = box [string "\"",
adam@1656:                                           string (Prim.toCString dml),
adam@1293:                                           string "\""], mode = mode},
adamc@885: 
adamc@885:          string "uw_pop_cleanup(ctx);",
adamc@885:          newline,
adamc@885:          string "uw_pop_cleanup(ctx);",
adamc@885:          newline]
adamc@885: 
adamc@885: fun nextval {loc, seqE, seqName} =
adamc@885:     box [string "uw_conn *conn = uw_get_db(ctx);",
adamc@885:          newline,
adamc@885:          string "char *insert = ",
adamc@885:          case seqName of
adamc@886:              SOME s => string ("\"INSERT INTO " ^ s ^ " VALUES (NULL)\"")
adamc@885:            | NONE => box [string "uw_Basis_strcat(ctx, \"INSERT INTO \", uw_Basis_strcat(ctx, ",
adamc@885:                           seqE,
adamc@885:                           string ", \" VALUES ()\"))"],
adamc@885:          string ";",
adamc@885:          newline,
adamc@885:          string "char *delete = ",
adamc@885:          case seqName of
adamc@885:              SOME s => string ("\"DELETE FROM " ^ s ^ "\"")
adamc@885:            | NONE => box [string "uw_Basis_strcat(ctx, \"DELETE FROM \", ",
adamc@885:                           seqE,
adamc@885:                           string ")"],
adamc@885:          string ";",
adamc@885:          newline,
adamc@885:          newline,
adamc@885: 
adamc@886:          string "if (sqlite3_exec(conn->conn, insert, NULL, NULL, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"'nextval' INSERT failed: %s\", sqlite3_errmsg(conn->conn));",
adamc@885:          newline,
adamc@885:          string "n = sqlite3_last_insert_rowid(conn->conn);",
adamc@885:          newline,
adamc@886:          string "if (sqlite3_exec(conn->conn, delete, NULL, NULL, NULL) != SQLITE_OK) uw_error(ctx, FATAL, \"'nextval' DELETE failed: %s\", sqlite3_errmsg(conn->conn));",
adamc@885:          newline]
adamc@885: 
adamc@885: fun nextvalPrepared _ = raise Fail "SQLite.nextvalPrepared called"
adamc@1073: fun setval _ = raise Fail "SQLite.setval called"
adamc@885: 
adamc@885: fun sqlifyString s = "'" ^ String.translate (fn #"'" => "''"
adam@1285:                                               | #"\000" => ""
adam@1285:                                               | ch => str ch)
adam@1285:                                             s ^ "'"
adamc@885: 
adamc@885: fun p_cast (s, _) = s
adamc@885: 
adamc@885: fun p_blank _ = "?"
adamc@885: 
adamc@885: val () = addDbms {name = "sqlite",
adam@1682:                   randomFunction = "RANDOM",
adam@1464:                   header = Config.sqheader,
adamc@885:                   link = "-lsqlite3",
adamc@885:                   init = init,
adamc@885:                   p_sql_type = p_sql_type,
adamc@885:                   query = query,
adamc@885:                   queryPrepared = queryPrepared,
adamc@885:                   dml = dml,
adamc@885:                   dmlPrepared = dmlPrepared,
adamc@885:                   nextval = nextval,
adamc@885:                   nextvalPrepared = nextvalPrepared,
adamc@1073:                   setval = setval,
adamc@885:                   sqlifyString = sqlifyString,
adamc@885:                   p_cast = p_cast,
adamc@885:                   p_blank = p_blank,
adamc@885:                   supportsDeleteAs = false,
adamc@886:                   supportsUpdateAs = false,
adamc@886:                   createSequence = fn s => "CREATE TABLE " ^ s ^ " (id INTEGER PRIMARY KEY AUTOINCREMENT)",
adamc@885:                   textKeysNeedLengths = false,
adamc@885:                   supportsNextval = false,
adamc@885:                   supportsNestedPrepared = false,
adamc@890:                   sqlPrefix = "",
adamc@1014:                   supportsOctetLength = false,
adamc@1014:                   trueString = "1",
adamc@1196:                   falseString = "0",
adamc@1196:                   onlyUnion = false,
adamc@1196:                   nestedRelops = false}
adamc@885: 
adamc@885: end