/*************************************************************************** CResult.c (c) 2000-2017 BenoƮt Minisini This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ***************************************************************************/ #define __CRESULT_C #include "main.h" #include "deletemap.h" #include "CField.h" #include "CResultField.h" #include "CResult.h" GB_CLASS CLASS_Blob; static CBLOB *make_blob(CRESULT *result, int field); static void set_blob(CBLOB *_object, char *data, int length); static SUBCOLLECTION_DESC _fields_desc = { ".Result.Fields", (void *)CRESULTFIELD_get, (void *)CRESULTFIELD_exist, (void *)NULL, (void *)CRESULTFIELD_release }; //------------------------------------------------------------------------- static int check_result(CRESULT *_object) { return (THIS->conn->db.handle == NULL); } static bool check_available(CRESULT *_object) { if (!THIS->available) { GB.Error("Result is not available"); return TRUE; } else return FALSE; } static void init_buffer(CRESULT *_object) { int i; if (THIS->info.nfield == 0) return; GB.Alloc(POINTER(&THIS->buffer), sizeof(GB_VARIANT_VALUE) * THIS->info.nfield); BARRAY_create(&THIS->changed, THIS->info.nfield); BARRAY_clear_all(THIS->changed, THIS->info.nfield); for (i = 0; i < THIS->info.nfield; i++) THIS->buffer[i].type = GB_T_NULL; } static void void_buffer(CRESULT *_object) { int i; //fprintf(stderr, "void_buffer\n"); if (THIS->info.nfield == 0) return; for (i = 0; i < THIS->info.nfield; i++) GB.StoreVariant(NULL, &THIS->buffer[i]); BARRAY_clear_all(THIS->changed, THIS->info.nfield); } static void release_buffer(CRESULT *_object) { if (THIS->buffer) { void_buffer(THIS); GB.Free(POINTER(&THIS->buffer)); BARRAY_delete(&THIS->changed); } } static bool load_buffer(CRESULT *_object, int vpos) { int i, ind; int pos; int result; if (vpos == THIS->pos) return FALSE; DB_CurrentDatabase = &THIS->conn->db; if (THIS->count < 0 || THIS->conn->db.flags.no_seek) { if (vpos != (THIS->pos + 1)) { GB.Error("Result is forward only"); return TRUE; } } else { if (vpos < 0 || vpos >= THIS->count || THIS->info.nfield == 0) { THIS->pos = -1; THIS->available = FALSE; return TRUE; } } pos = DELETE_MAP_virtual_to_real(THIS->dmap, vpos); //fprintf(stderr, "Result %p: Loading real %ld\n", THIS, pos); void_buffer(THIS); if (THIS->handle) { result = THIS->driver->Result.Fill(&THIS->conn->db, THIS->handle, pos, THIS->buffer, (pos > 0) && (pos == (DELETE_MAP_virtual_to_real(THIS->dmap, THIS->pos) + 1))); if (result == DB_ERROR) return TRUE; else if (result == DB_NO_DATA) { THIS->pos = -1; THIS->available = FALSE; return TRUE; } if (THIS->mode == RESULT_EDIT) { q_init(); for (i = 0; i < THIS->info.nindex; i++) { ind = THIS->info.index[i]; if (i > 0) q_add(" AND "); q_add(THIS->driver->GetQuote()); q_add(THIS->info.field[ind].name); q_add(THIS->driver->GetQuote()); if (THIS->buffer[ind].type == GB_T_NULL) q_add(" IS NULL"); else { q_add(" = "); DB_FormatVariant(THIS->driver, &THIS->buffer[ind], q_add_length); } } GB.FreeString(&THIS->edit); THIS->edit = q_steal(); } } THIS->pos = vpos; THIS->available = TRUE; return FALSE; } static void reload_buffer(CRESULT *_object) { int pos = THIS->pos; THIS->pos = -1; load_buffer(THIS, pos); } static void table_release(DB_INFO *info) { int i; if (info->table) GB.FreeString(&info->table); if (info->field) { for (i = 0; i < info->nfield; i++) CFIELD_free_info(&info->field[i]); GB.Free(POINTER(&info->field)); } if (info->index) GB.Free(POINTER(&info->index)); } static GB_TYPE get_field_type(CRESULT *_object, int field) { GB_TYPE type; if (THIS->info.field) type = THIS->info.field[field].type; else type = THIS->driver->Result.Field.Type(THIS->handle, field); //fprintf(stderr, "get_field_type: %d -> %ld\n", field, type); return type; } CRESULT *DB_MakeResult(CCONNECTION *conn, int mode, char *table_temp, char *query) { CRESULT *_object; DB_RESULT res; char *duplicate; char *token; const char *error = NULL; char *arg; char *table; switch (mode) { case RESULT_FIND: if (conn->driver->Exec(&conn->db, query, &res, "Query failed: &1")) return NULL; break; case RESULT_CREATE: res = NULL; break; case RESULT_EDIT: if (conn->driver->Exec(&conn->db, query, &res, "Query failed: &1")) return NULL; break; case RESULT_DELETE: conn->driver->Exec(&conn->db, query, NULL, "Query failed: &1"); return NULL; } _object = GB.New(GB.FindClass("Result"), NULL, NULL); THIS->conn = conn; GB.Ref(conn); THIS->driver = conn->driver; THIS->available = FALSE; THIS->mode = mode; THIS->handle = res; THIS->pos = -1; THIS->dmap = NULL; // table must be copied because it can be a temporary string! table = GB.NewZeroString(table_temp); switch (mode) { case RESULT_FIND: THIS->driver->Result.Init(THIS->handle, &THIS->info, &THIS->count); break; case RESULT_CREATE: if (THIS->driver->Table.Init(&conn->db, table, &THIS->info)) goto ERROR; THIS->count = 1; break; case RESULT_EDIT: THIS->driver->Result.Init(THIS->handle, &THIS->info, &THIS->count); if (THIS->driver->Table.Init(&conn->db, table, &THIS->info)) goto ERROR; if (THIS->driver->Table.Index(&conn->db, table, &THIS->info)) { error = "Table '&1' has no primary key"; arg = table; goto ERROR; } break; } init_buffer(THIS); load_buffer(THIS, 0); GB.FreeString(&table); return THIS; ERROR: if (!error) { if (strchr(table, (int)',') == NULL) { arg = table; if (!THIS->driver->Table.Exist(&conn->db, table)) error = "Unknown table: &1"; else error = "Cannot read information about table &1"; } else { duplicate = GB.NewZeroString(table); token = strtok(duplicate,","); do { arg = token; if (!THIS->driver->Table.Exist(&conn->db, token)) error = "Unknown table: &1"; else error = "Cannot read information about table '&1'"; } while ((token = strtok(NULL, ".")) != NULL); GB.FreeString(&duplicate); } } GB.Free(POINTER(&_object)); GB.Error(error, arg); GB.FreeString(&table); return NULL; } //------------------------------------------------------------------------- BEGIN_METHOD_VOID(Result_free) release_buffer(THIS); if (THIS->mode != RESULT_CREATE) THIS->driver->Result.Release(THIS->handle, &THIS->info, check_result(THIS)); if (THIS->mode != RESULT_FIND) table_release(&THIS->info); if (THIS->edit) GB.FreeString(&THIS->edit); DELETE_MAP_free(&THIS->dmap); GB.Unref(POINTER(&THIS->conn)); GB.Unref(POINTER(&THIS->fields)); END_METHOD BEGIN_PROPERTY(Result_Count) GB.ReturnInteger(THIS->count); END_PROPERTY BEGIN_PROPERTY(Result_Max) GB.ReturnInteger(THIS->count - 1); END_PROPERTY BEGIN_PROPERTY(Result_Index) GB.ReturnInteger(THIS->pos); END_PROPERTY BEGIN_PROPERTY(Result_Available) GB.ReturnBoolean(THIS->available); END_PROPERTY static void check_blob(CRESULT *_object, int field) { GB_VARIANT val; //fprintf(stderr, "check_blob: %d\n", field); if (THIS->buffer[field].type == GB_T_NULL) { val.type = GB_T_VARIANT; val.value.type = (GB_TYPE)CLASS_Blob; val.value.value._object = make_blob(THIS, field); GB.StoreVariant(&val, &THIS->buffer[field]); } } BEGIN_METHOD(Result_get, GB_STRING field) int index; GB_TYPE type; if (check_available(THIS)) return; index = CRESULTFIELD_find(THIS, GB.ToZeroString(ARG(field)), TRUE); if (index < 0) return; type = get_field_type(THIS, index); if (type == DB_T_BLOB) check_blob(THIS, index); GB.ReturnVariant(&THIS->buffer[index]); END_METHOD BEGIN_METHOD(Result_GetAll, GB_STRING field) int index; int pos; GB_TYPE type, atype; GB_ARRAY result; GB_VARIANT_VALUE *val; index = CRESULTFIELD_find(THIS, GB.ToZeroString(ARG(field)), TRUE); if (index < 0) return; atype = type = get_field_type(THIS, index); if (atype == DB_T_SERIAL) atype = GB_T_LONG; else if (atype == DB_T_BLOB) atype = GB_T_OBJECT; GB.Array.New(POINTER(&result), atype, 0); pos = THIS->pos; load_buffer(THIS, 0); while (THIS->available) { if (type == DB_T_BLOB) check_blob(THIS, index); val = &THIS->buffer[index]; switch (atype) { case GB_T_BOOLEAN: *(char *)GB.Array.Add(result) = val->value._boolean; break; case GB_T_INTEGER: *(int *)GB.Array.Add(result) = val->value._integer; break; case GB_T_LONG: *(int64_t *)GB.Array.Add(result) = val->value._long; break; case GB_T_FLOAT: *(double *)GB.Array.Add(result) = val->value._float; break; case GB_T_DATE: *(GB_DATE_VALUE *)GB.Array.Add(result) = val->value._date; break; case GB_T_STRING: if (val->type == GB_T_CSTRING) *(char **)GB.Array.Add(result) = GB.NewString(val->value._string, strlen(val->value._string)); else *(char **)GB.Array.Add(result) = GB.RefString(val->value._string); break; case GB_T_OBJECT: *(void **)GB.Array.Add(result) = val->value._object; GB.Ref(val->value._object); break; } load_buffer(THIS, THIS->pos + 1); } if (THIS->count >= 0) load_buffer(THIS, pos); GB.ReturnObject(result); END_METHOD BEGIN_METHOD(Result_put, GB_VARIANT value; GB_STRING field) int index; GB_TYPE type; if (check_available(THIS)) return; if (THIS->mode == RESULT_FIND) { GB.Error("Result is read-only"); return; } index = CRESULTFIELD_find(THIS, GB.ToZeroString(ARG(field)), TRUE); if (index < 0) return; type = get_field_type(THIS, index); if (type == DB_T_SERIAL) { //GB.Error("Read-only field"); return; } if (type == DB_T_BLOB) { check_blob(THIS, index); if (VARG(value).type == (GB_TYPE)CLASS_Blob) { CBLOB *src = VARG(value).value._object; set_blob((CBLOB *)THIS->buffer[index].value._object, src->data, src->length); } else { GB_STRING *str = (GB_STRING *)(void *)ARG(value); if (GB.Conv((GB_VALUE *)(void *)ARG(value), GB_T_STRING)) return; set_blob((CBLOB *)THIS->buffer[index].value._object, str->value.addr + str->value.start, str->value.len); } BARRAY_set(THIS->changed, index); return; } if (VARG(value).type != GB_T_NULL && VARG(value).type != type) { if (GB.Conv((GB_VALUE *)(void *)ARG(value), THIS->info.field[index].type)) { GB.Error("Type mismatch"); return; } GB.Conv((GB_VALUE *)(void *)ARG(value), GB_T_VARIANT); } GB.StoreVariant(ARG(value), &THIS->buffer[index]); BARRAY_set(THIS->changed, index); END_METHOD #if 0 BEGIN_METHOD(CRESULT_copy, GB_OBJECT result) CRESULT *result = (CRESULT *)VARG(result); int index; if (THIS->mode == RESULT_FIND) { GB.Error("Result is read-only"); return; } for (index = 0; index < index = find_field(THIS, GB.ToZeroString(ARG(field))); if (index < 0) return; if (VARG(value).type != GB_T_NULL && VARG(value).type != THIS->info.types[index]) /*{ GB.Error("Type mismatch"); return; }*/ { if (GB.Conv((GB_VALUE *)ARG(value), THIS->info.types[index])) return; GB.Conv((GB_VALUE *)ARG(value), GB_T_VARIANT); } GB.StoreVariant(ARG(value), &THIS->buffer[index]); END_METHOD #endif BEGIN_METHOD_VOID(Result_MoveFirst) GB.ReturnBoolean(load_buffer(THIS, 0)); END_METHOD BEGIN_METHOD_VOID(Result_MoveLast) if (THIS->count < 0) GB.Error("Result is forward only"); else GB.ReturnBoolean(load_buffer(THIS, THIS->count - 1)); END_METHOD BEGIN_METHOD_VOID(Result_MovePrevious) GB.ReturnBoolean(load_buffer(THIS, THIS->pos - 1)); END_METHOD BEGIN_METHOD_VOID(Result_MoveNext) GB.ReturnBoolean(load_buffer(THIS, THIS->pos + 1)); END_METHOD BEGIN_METHOD(Result_MoveTo, GB_INTEGER pos) GB.ReturnBoolean(load_buffer(THIS, VARG(pos))); END_METHOD BEGIN_METHOD_VOID(Result_next) int *pos = (int *)GB.GetEnum(); if (!load_buffer(THIS, *pos)) (*pos)++; else GB.StopEnum(); END_METHOD BEGIN_METHOD_VOID(Result_Update) int i, j; bool comma; DB_INFO *info = &THIS->info; int ret_first = -1; int ret_count = 0; GB_VARIANT_VALUE *ret_buffer = NULL; bool err; DB_RESULT res; if (check_available(THIS)) return; DB_CurrentDatabase = &THIS->conn->db; q_init(); switch(THIS->mode) { case RESULT_CREATE: if (BARRAY_is_void(THIS->changed, THIS->info.nfield)) break; q_add("INSERT INTO "); q_add(DB_GetQuotedTable(THIS->driver, DB_CurrentDatabase, info->table, -1)); q_add(" ( "); comma = FALSE; for (i = 0; i < info->nfield; i++) { if (THIS->buffer[i].type == GB_T_NULL) continue; if (!BARRAY_test(THIS->changed, i)) continue; if (comma) q_add(", "); q_add(THIS->driver->GetQuote()); q_add(info->field[i].name); q_add(THIS->driver->GetQuote()); comma = TRUE; } if (!comma) { q_add(THIS->driver->GetQuote()); q_add(info->field[0].name); q_add(THIS->driver->GetQuote()); } q_add(" ) VALUES ( "); comma = FALSE; for (i = 0; i < info->nfield; i++) { if (THIS->buffer[i].type == GB_T_NULL) continue; if (!BARRAY_test(THIS->changed, i)) continue; if (comma) q_add(", "); DB_FormatVariant(THIS->driver, &THIS->buffer[i], q_add_length); comma = TRUE; } if (!comma) { DB_FormatVariant(THIS->driver, &THIS->buffer[0], q_add_length); } q_add(" )"); if (THIS->returning) { comma = FALSE; for (i = 0; i < info->nfield; i++) { if (info->field[i].type == DB_T_SERIAL) { ret_count++; if (comma) q_add(", "); else { q_add(" RETURNING "); ret_first = i; } q_add(THIS->driver->GetQuote()); q_add(info->field[i].name); q_add(THIS->driver->GetQuote()); comma = TRUE; } } GB.Alloc(POINTER(&ret_buffer), sizeof(GB_VARIANT_VALUE) * ret_count); for (i = 0; i < ret_count; i++) ret_buffer[i].type = GB_T_NULL; } if (ret_count) err = THIS->driver->Exec(&THIS->conn->db, q_get(), &res, "Cannot create record: &1"); else err = THIS->driver->Exec(&THIS->conn->db, q_get(), NULL, "Cannot create record: &1"); if (!err) { if (!THIS->returning) void_buffer(THIS); else if (ret_count) { if (THIS->driver->Result.Fill(&THIS->conn->db, res, 0, ret_buffer, FALSE) == DB_OK) { THIS->buffer[ret_first] = ret_buffer[0]; for (i = ret_first + 1, j = 1; i < info->nfield; i++) { if (info->field[i].type == DB_T_SERIAL) THIS->buffer[i] = ret_buffer[j++]; } } THIS->driver->Result.Release(res, NULL, FALSE); } } GB.Free(POINTER(&ret_buffer)); break; case RESULT_EDIT: if (BARRAY_is_void(THIS->changed, THIS->info.nfield)) break; q_add("UPDATE "); q_add(DB_GetQuotedTable(THIS->driver, DB_CurrentDatabase, info->table, -1)); q_add(" SET "); comma = FALSE; for (i = 0; i < info->nfield; i++) { if (!BARRAY_test(THIS->changed, i)) continue; if (comma) q_add(", "); q_add(THIS->driver->GetQuote()); q_add(THIS->info.field[i].name); q_add(THIS->driver->GetQuote()); q_add(" = "); DB_FormatVariant(THIS->driver, &THIS->buffer[i], q_add_length); comma = TRUE; } q_add(" WHERE "); q_add(THIS->edit); THIS->driver->Exec(&THIS->conn->db, q_get(), NULL, "Cannot modify record: &1"); break; default: GB.Error("Result is read-only"); break; } BARRAY_clear_all(THIS->changed, THIS->info.nfield); END_METHOD BEGIN_METHOD(Result_Delete, GB_BOOLEAN keep) DB_INFO *info = &THIS->info; int *pos; void *save_enum; if (check_available(THIS)) return; q_init(); switch(THIS->mode) { case RESULT_CREATE: void_buffer(THIS); break; case RESULT_EDIT: q_add("DELETE FROM "); q_add(DB_GetQuotedTable(THIS->driver, DB_CurrentDatabase, info->table, -1)); q_add(" WHERE "); q_add(THIS->edit); THIS->driver->Exec(&THIS->conn->db, q_get(), NULL, "Cannot delete record: &1"); if (!VARGOPT(keep, FALSE) && THIS->count > 0) { DELETE_MAP_add(&THIS->dmap, THIS->pos); THIS->count--; reload_buffer(THIS); save_enum = GB.BeginEnum(THIS); while (!GB.NextEnum()) { pos = (int *)GB.GetEnum(); if (*pos > THIS->pos) (*pos)--; } GB.EndEnum(save_enum); } break; default: GB.Error("Result is read-only"); break; } END_METHOD BEGIN_PROPERTY(Result_Fields) GB_SubCollectionNew(&THIS->fields, &_fields_desc, THIS); GB.ReturnObject(THIS->fields); END_PROPERTY BEGIN_PROPERTY(Result_Connection) GB.ReturnObject(THIS->conn); END_PROPERTY BEGIN_PROPERTY(Result_Editable) GB.ReturnBoolean(THIS->mode == RESULT_CREATE || THIS->mode == RESULT_EDIT); END_PROPERTY //------------------------------------------------------------------------- GB_DESC CResultDesc[] = { GB_DECLARE("Result", sizeof(CRESULT)), GB_NOT_CREATABLE(), GB_HOOK_CHECK(check_result), GB_METHOD("_free", NULL, Result_free, NULL), GB_PROPERTY_READ("Count", "i", Result_Count), GB_PROPERTY_READ("Length", "i", Result_Count), GB_PROPERTY_READ("Available", "b", Result_Available), GB_PROPERTY_READ("Index", "i", Result_Index), GB_PROPERTY_READ("Max", "i", Result_Max), GB_PROPERTY_READ("Editable", "b", Result_Editable), GB_METHOD("_get", "v", Result_get, "(Field)s"), GB_METHOD("_put", NULL, Result_put, "(Value)v(Field)s"), GB_METHOD("_next", NULL, Result_next, NULL), GB_METHOD("MoveFirst", "b", Result_MoveFirst, NULL), GB_METHOD("MoveLast", "b", Result_MoveLast, NULL), GB_METHOD("MovePrevious", "b", Result_MovePrevious, NULL), GB_METHOD("MoveNext", "b", Result_MoveNext, NULL), GB_METHOD("MoveTo", "b", Result_MoveTo, "(Index)i"), GB_METHOD("Update", NULL, Result_Update, NULL), GB_METHOD("Delete", NULL, Result_Delete, "[(Keep)b]"), GB_METHOD("All", "Array", Result_GetAll, "(Field)s"), GB_PROPERTY_READ("Fields", ".Result.Fields", Result_Fields), GB_PROPERTY_READ("Connection", "Connection", Result_Connection), GB_END_DECLARE }; //------------------------------------------------------------------------- static bool _convert_blob(CBLOB *_object, GB_TYPE type, GB_VALUE *conv) { if (BLOB) { switch (type) { case GB_T_STRING: case GB_T_CSTRING: conv->_string.value.addr = BLOB->data; conv->_string.value.start = 0; conv->_string.value.len = BLOB->length; conv->type = GB_T_CSTRING; return FALSE; default: return TRUE; } } else return TRUE; } /*static int check_blob(CBLOB *_object) { return check_result(BLOB->result) || (BLOB->result->pos != BLOB->pos); }*/ static CBLOB *make_blob(CRESULT *result, int field) { CBLOB *_object; _object = GB.New(CLASS_Blob, NULL, NULL); //BLOB->result = result; //GB.Ref(result); //BLOB->field = field; //BLOB->pos = result->pos; BLOB->data = NULL; BLOB->length = 0; BLOB->constant = TRUE; if (result->handle && result->pos >= 0) { BLOB->constant = FALSE; result->driver->Result.Blob(result->handle, result->pos, field, BLOB); if (BLOB->constant) set_blob(BLOB, BLOB->data, BLOB->length); } //fprintf(stderr, "make_blob: [%d] %d (%d) -> %p\n", result->pos, field, BLOB->length, BLOB); //GB.UnrefKeep(POINTER(&_object), FALSE); return BLOB; } static void set_blob(CBLOB *_object, char *data, int length) { if (!BLOB->constant && BLOB->data) GB.FreeString((char **)&BLOB->data); if (data && length) { BLOB->data = GB.NewString(data, length); BLOB->constant = FALSE; } BLOB->length = length; } //------------------------------------------------------------------------- BEGIN_METHOD_VOID(Blob_init) CLASS_Blob = GB.FindClass("Blob"); END_METHOD BEGIN_METHOD_VOID(Blob_free) //GB.Unref(POINTER(&BLOB->result)); set_blob(BLOB, NULL, 0); END_METHOD /*BEGIN_PROPERTY(CBLOB_result) GB.ReturnObject(BLOB->result); END_PROPERTY*/ BEGIN_PROPERTY(Blob_Data) if (READ_PROPERTY) { if (BLOB->length) GB.ReturnConstString(BLOB->data, BLOB->length); else GB.ReturnVoidString(); } else { set_blob(BLOB, PSTRING(), PLENGTH()); } END_PROPERTY BEGIN_PROPERTY(Blob_Length) GB.ReturnInteger(BLOB->length); END_PROPERTY //------------------------------------------------------------------------- GB_DESC CBlobDesc[] = { GB_DECLARE("Blob", sizeof(CBLOB)), GB_NOT_CREATABLE(), //GB_HOOK_CHECK(check_blob), GB_STATIC_METHOD("_init", NULL, Blob_init, NULL), GB_METHOD("_free", NULL, Blob_free, NULL), //GB_PROPERTY_READ("Result", "Result", CBLOB_result), GB_PROPERTY("Data", "s", Blob_Data), GB_PROPERTY_READ("Length", "i", Blob_Length), GB_INTERFACE("_convert", &_convert_blob), //GB_METHOD("_unknown", "v", CBLOB_unknown, "v"), GB_END_DECLARE };