diff options
author | Adam Chlipala <adamc@hcoop.net> | 2008-09-02 14:40:57 -0400 |
---|---|---|
committer | Adam Chlipala <adamc@hcoop.net> | 2008-09-02 14:40:57 -0400 |
commit | 2a572eeba9d521400872e6588b3f709818a5d412 (patch) | |
tree | 2b719447badc70a7b852f0091f68fad273d9e9c5 /src | |
parent | 3b770e100b11cbcfc19af6f810962975e9221d9f (diff) |
Validating schema of a live database
Diffstat (limited to 'src')
-rw-r--r-- | src/cjr_print.sml | 228 | ||||
-rw-r--r-- | src/expl_env.sml | 1 | ||||
-rw-r--r-- | src/list_util.sig | 1 | ||||
-rw-r--r-- | src/list_util.sml | 7 | ||||
-rw-r--r-- | src/print.sig | 2 | ||||
-rw-r--r-- | src/print.sml | 13 |
6 files changed, 233 insertions, 19 deletions
diff --git a/src/cjr_print.sml b/src/cjr_print.sml index ff2cada1..41ecb7bc 100644 --- a/src/cjr_print.sml +++ b/src/cjr_print.sml @@ -692,9 +692,10 @@ fun p_decl env (dAll as (d, _) : decl) = string x, string " */", newline] - | DDatabase s => box [string "void lw_db_init(lw_context ctx) {", + | DDatabase s => box [string "static void lw_db_validate(lw_context);", newline, - string "PGresult *res;", + newline, + string "void lw_db_init(lw_context ctx) {", newline, string "PGconn *conn = PQconnectdb(\"", string (String.toString s), @@ -720,6 +721,8 @@ fun p_decl env (dAll as (d, _) : decl) = newline, string "lw_set_db(ctx, conn);", newline, + string "lw_db_validate(ctx);", + newline, string "}", newline, newline, @@ -735,6 +738,17 @@ datatype 'a search = | NotFound | Error +fun p_sqltype' env (tAll as (t, loc)) = + case t of + TFfi ("Basis", "int") => "int8" + | TFfi ("Basis", "float") => "float8" + | TFfi ("Basis", "string") => "text" + | TFfi ("Basis", "bool") => "bool" + | _ => (ErrorMsg.errorAt loc "Don't know SQL equivalent of type"; + Print.eprefaces' [("Type", p_typ env tAll)]; + "ERROR") + +fun p_sqltype env t = string (p_sqltype' env t) fun p_file env (ds, ps) = let @@ -1204,6 +1218,195 @@ fun p_file env (ds, ps) = end val pds' = map p_page ps + + val tables = List.mapPartial (fn (DTable (s, xts), _) => SOME (s, xts) + | _ => NONE) ds + + val validate = + box [string "static void lw_db_validate(lw_context ctx) {", + newline, + string "PGconn *conn = lw_get_db(ctx);", + newline, + string "PGresult *res;", + newline, + newline, + p_list_sep newline + (fn (s, xts) => + let + val q = "SELECT COUNT(*) FROM pg_class WHERE relname = '" + ^ s ^ "'" + + val q' = String.concat ["SELECT COUNT(*) FROM pg_attribute WHERE attrelid = (SELECT oid FROM pg_class WHERE relname = '", + s, + "') AND (", + String.concatWith " OR " + (map (fn (x, t) => + String.concat ["(attname = 'lw_", + CharVector.map + Char.toLower x, + "' AND atttypid = (SELECT oid FROM pg_type", + " WHERE typname = '", + p_sqltype' env t, + "'))"]) xts), + ")"] + + val q'' = String.concat ["SELECT COUNT(*) FROM pg_attribute WHERE attrelid = (SELECT oid FROM pg_class WHERE relname = '", + s, + "') AND attnum >= 0"] + in + box [string "res = PQexec(conn, \"", + string q, + string "\");", + newline, + newline, + string "if (res == NULL) {", + newline, + box [string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Out of memory allocating query result.\");", + newline], + string "}", + newline, + newline, + string "if (PQresultStatus(res) != PGRES_TUPLES_OK) {", + newline, + box [string "char msg[1024];", + newline, + string "strncpy(msg, PQerrorMessage(conn), 1024);", + newline, + string "msg[1023] = 0;", + newline, + string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Query failed:\\n", + string q, + string "\\n%s\", msg);", + newline], + string "}", + newline, + newline, + string "if (strcmp(PQgetvalue(res, 0, 0), \"1\")) {", + newline, + box [string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Table '", + string s, + string "' does not exist.\");", + newline], + string "}", + newline, + newline, + string "PQclear(res);", + newline, + + string "res = PQexec(conn, \"", + string q', + string "\");", + newline, + newline, + string "if (res == NULL) {", + newline, + box [string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Out of memory allocating query result.\");", + newline], + string "}", + newline, + newline, + string "if (PQresultStatus(res) != PGRES_TUPLES_OK) {", + newline, + box [string "char msg[1024];", + newline, + string "strncpy(msg, PQerrorMessage(conn), 1024);", + newline, + string "msg[1023] = 0;", + newline, + string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Query failed:\\n", + string q', + string "\\n%s\", msg);", + newline], + string "}", + newline, + newline, + string "if (strcmp(PQgetvalue(res, 0, 0), \"", + string (Int.toString (length xts)), + string "\")) {", + newline, + box [string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Table '", + string s, + string "' has the wrong column types.\");", + newline], + string "}", + newline, + newline, + string "PQclear(res);", + newline, + newline, + + string "res = PQexec(conn, \"", + string q'', + string "\");", + newline, + newline, + string "if (res == NULL) {", + newline, + box [string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Out of memory allocating query result.\");", + newline], + string "}", + newline, + newline, + string "if (PQresultStatus(res) != PGRES_TUPLES_OK) {", + newline, + box [string "char msg[1024];", + newline, + string "strncpy(msg, PQerrorMessage(conn), 1024);", + newline, + string "msg[1023] = 0;", + newline, + string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Query failed:\\n", + string q'', + string "\\n%s\", msg);", + newline], + string "}", + newline, + newline, + string "if (strcmp(PQgetvalue(res, 0, 0), \"", + string (Int.toString (length xts)), + string "\")) {", + newline, + box [string "PQclear(res);", + newline, + string "PQfinish(conn);", + newline, + string "lw_error(ctx, FATAL, \"Table '", + string s, + string "' has extra columns.\");", + newline], + string "}", + newline, + newline, + string "PQclear(res);", + newline] + end) tables, + string "}"] in box [string "#include <stdio.h>", newline, @@ -1235,23 +1438,12 @@ fun p_file env (ds, ps) = p_list_sep newline (fn x => x) pds', newline, string "}", + newline, + newline, + validate, newline] end -fun p_sqltype env (tAll as (t, loc)) = - let - val s = case t of - TFfi ("Basis", "int") => "int8" - | TFfi ("Basis", "float") => "float8" - | TFfi ("Basis", "string") => "text" - | TFfi ("Basis", "bool") => "bool" - | _ => (ErrorMsg.errorAt loc "Don't know SQL equivalent of type"; - Print.eprefaces' [("Type", p_typ env tAll)]; - "ERROR") - in - string s - end - fun p_sql env (ds, _) = let val (pps, _) = ListUtil.foldlMap @@ -1264,9 +1456,7 @@ fun p_sql env (ds, _) = string "(", p_list (fn (x, t) => box [string "lw_", - string x, - space, - string ":", + string (CharVector.map Char.toLower x), space, p_sqltype env t, space, diff --git a/src/expl_env.sml b/src/expl_env.sml index b2146474..fdb4d995 100644 --- a/src/expl_env.sml +++ b/src/expl_env.sml @@ -288,6 +288,7 @@ fun declBinds env (d, loc) = in pushENamed env x n t end + | DDatabase _ => env fun sgiBinds env (sgi, loc) = case sgi of diff --git a/src/list_util.sig b/src/list_util.sig index 33474b01..3340fffc 100644 --- a/src/list_util.sig +++ b/src/list_util.sig @@ -42,5 +42,6 @@ signature LIST_UTIL = sig val mapi : (int * 'a -> 'b) -> 'a list -> 'b list val foldli : (int * 'a * 'b -> 'b) -> 'b -> 'a list -> 'b + val foldri : (int * 'a * 'b -> 'b) -> 'b -> 'a list -> 'b end diff --git a/src/list_util.sml b/src/list_util.sml index 616efcf0..f9826ab4 100644 --- a/src/list_util.sml +++ b/src/list_util.sml @@ -156,4 +156,11 @@ fun foldli f = m 0 end +fun foldri f i ls = + let + val len = length ls + in + foldli (fn (n, x, s) => f (len - n - 1, x, s)) i (rev ls) + end + end diff --git a/src/print.sig b/src/print.sig index 07b8c1e2..83c84405 100644 --- a/src/print.sig +++ b/src/print.sig @@ -42,6 +42,8 @@ signature PRINT = sig val p_list_sep : PD.pp_desc -> 'a printer -> 'a list printer val p_list : 'a printer -> 'a list printer + val p_list_sepi : PD.pp_desc -> (int -> 'a printer) -> 'a list printer + val fprint : PD.PPS.stream -> PD.pp_desc -> unit val print : PD.pp_desc -> unit val eprint : PD.pp_desc -> unit diff --git a/src/print.sml b/src/print.sml index 3917a998..7329c44d 100644 --- a/src/print.sml +++ b/src/print.sml @@ -59,6 +59,19 @@ fun p_list_sep sep f ls = end fun p_list f = p_list_sep (box [PD.string ",", space]) f +fun p_list_sepi sep f ls = + case ls of + [] => PD.string "" + | [x] => f 0 x + | x :: rest => + let + val tokens = ListUtil.foldri (fn (n, x, tokens) => + sep :: PD.cut :: f (n + 1) x :: tokens) + [] rest + in + box (f 0 x :: tokens) + end + fun fprint f d = (PD.description (f, d); PD.PPS.flushStream f) val print = fprint out |