2007-12-30 17:41:49 +01:00
|
|
|
/***************************************************************************
|
|
|
|
|
|
|
|
CIndex.c
|
|
|
|
|
2009-08-17 12:41:51 +02:00
|
|
|
(c) 2000-2009 Benoît Minisini <gambas@users.sourceforge.net>
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
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
|
2009-08-17 12:41:51 +02:00
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
2007-12-30 17:41:49 +01:00
|
|
|
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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#define __CINDEX_C
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
|
|
|
|
#include "CIndex.h"
|
|
|
|
|
|
|
|
|
|
|
|
static int valid_index(CINDEX *_object)
|
|
|
|
{
|
|
|
|
return !THIS->table || !THIS->table->conn || !THIS->table->conn->db.handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool exist_index(CTABLE *table, const char *name)
|
|
|
|
{
|
|
|
|
if (!name || !*name)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return table->driver->Index.Exist(&table->conn->db, table->name, (char *)name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_index(CTABLE *table, const char *name, bool must_exist)
|
|
|
|
{
|
|
|
|
bool exist = exist_index(table, name);
|
|
|
|
|
|
|
|
if (must_exist)
|
|
|
|
{
|
|
|
|
if (!exist)
|
|
|
|
{
|
|
|
|
GB.Error("Unknown index: &1.&2", table->name, name);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (exist)
|
|
|
|
{
|
|
|
|
GB.Error("Index already exists: &1.&2", table->name, name);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static CINDEX *make_index(CTABLE *table, const char *name, bool must_exist)
|
|
|
|
{
|
|
|
|
CINDEX *_object;
|
|
|
|
|
|
|
|
if (check_index(table, name, must_exist))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
GB.New(POINTER(&_object), GB.FindClass("Index"), NULL, NULL);
|
|
|
|
THIS->table = table;
|
|
|
|
THIS->driver = table->conn->driver;
|
2010-06-05 01:48:53 +02:00
|
|
|
THIS->name = GB.NewZeroString(name);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
return _object;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void *CINDEX_get(CTABLE *table, const char *name)
|
|
|
|
{
|
|
|
|
CINDEX *index = make_index(table, name, TRUE);
|
|
|
|
table->driver->Index.Info(&table->conn->db, table->name, (char *)name, &index->info);
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int CINDEX_exist(CTABLE *table, const char *name)
|
|
|
|
{
|
|
|
|
return exist_index(table, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CINDEX_list(CTABLE *table, char ***list)
|
|
|
|
{
|
|
|
|
table->driver->Index.List(&table->conn->db, table->name, list);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CINDEX_release(CTABLE *table, void *_object)
|
|
|
|
{
|
|
|
|
THIS->table = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
|
|
|
|
Index
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
BEGIN_METHOD_VOID(CINDEX_free)
|
|
|
|
|
|
|
|
if (!valid_index(THIS))
|
|
|
|
GB.SubCollection.Remove(THIS->table->indexes, THIS->name, 0);
|
|
|
|
|
|
|
|
GB.FreeString(&THIS->name);
|
|
|
|
|
|
|
|
GB.FreeString(&THIS->info.name);
|
|
|
|
GB.FreeString(&THIS->info.fields);
|
|
|
|
THIS->info.unique = FALSE;
|
|
|
|
|
|
|
|
END_METHOD
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_PROPERTY(CINDEX_name)
|
|
|
|
|
|
|
|
GB.ReturnString(THIS->name);
|
|
|
|
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_PROPERTY(CINDEX_fields)
|
|
|
|
|
2009-11-28 02:22:00 +01:00
|
|
|
GB_ARRAY array;
|
|
|
|
char *fields;
|
|
|
|
char *name;
|
|
|
|
|
2010-06-05 01:48:53 +02:00
|
|
|
fields = GB.NewZeroString(THIS->info.fields);
|
2009-11-28 02:22:00 +01:00
|
|
|
GB.Array.New(&array, GB_T_STRING, 0);
|
|
|
|
|
|
|
|
name = strtok(fields, ",");
|
|
|
|
while (name)
|
|
|
|
{
|
2010-06-05 01:48:53 +02:00
|
|
|
*((char **)GB.Array.Add(array)) = GB.NewZeroString(name);
|
2009-11-28 02:22:00 +01:00
|
|
|
name = strtok(NULL, ",");
|
|
|
|
}
|
|
|
|
|
|
|
|
GB.FreeString(&fields);
|
|
|
|
GB.ReturnObject(array);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_PROPERTY(CINDEX_unique)
|
|
|
|
|
|
|
|
GB.ReturnBoolean(THIS->info.unique);
|
|
|
|
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_PROPERTY(CINDEX_primary)
|
|
|
|
|
|
|
|
GB.ReturnBoolean(THIS->info.primary);
|
|
|
|
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_PROPERTY(CINDEX_table)
|
|
|
|
|
|
|
|
GB.ReturnObject(THIS->table);
|
|
|
|
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
|
|
|
|
|
|
GB_DESC CIndexDesc[] =
|
|
|
|
{
|
|
|
|
GB_DECLARE("Index", sizeof(CINDEX)),
|
|
|
|
GB_NOT_CREATABLE(),
|
|
|
|
GB_HOOK_CHECK(valid_index),
|
|
|
|
|
|
|
|
GB_METHOD("_free", NULL, CINDEX_free, NULL),
|
|
|
|
|
|
|
|
GB_PROPERTY_READ("Name", "s", CINDEX_name),
|
|
|
|
GB_PROPERTY_READ("Unique", "b", CINDEX_unique),
|
|
|
|
GB_PROPERTY_READ("Primary", "b", CINDEX_primary),
|
2009-11-28 02:22:00 +01:00
|
|
|
GB_PROPERTY_READ("Fields", "String[]", CINDEX_fields),
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
GB_PROPERTY_READ("Table", "Table", CINDEX_table),
|
|
|
|
|
|
|
|
GB_END_DECLARE
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
|
|
|
|
.TableIndexes
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#undef THIS
|
|
|
|
#define THIS ((GB_SUBCOLLECTION)_object)
|
|
|
|
|
2009-11-28 02:22:00 +01:00
|
|
|
BEGIN_METHOD(CINDEX_add, GB_STRING name; GB_OBJECT fields; GB_BOOLEAN unique)
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
CTABLE *table = GB.SubCollection.Container(THIS);
|
|
|
|
char *name = GB.ToZeroString(ARG(name));
|
|
|
|
DB_INDEX info;
|
2009-11-28 02:22:00 +01:00
|
|
|
int i;
|
|
|
|
GB_ARRAY fields;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2009-11-25 01:26:26 +01:00
|
|
|
if (DB_CheckNameWith(name, "index", "."))
|
2007-12-30 17:41:49 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (check_index(table, name, FALSE))
|
|
|
|
return;
|
|
|
|
|
|
|
|
info.name = name;
|
2009-11-28 02:22:00 +01:00
|
|
|
|
|
|
|
fields = (GB_ARRAY)VARG(fields);
|
|
|
|
q_init();
|
|
|
|
for (i = 0; i < GB.Array.Count(fields); i++)
|
|
|
|
{
|
|
|
|
if (i > 0)
|
|
|
|
q_add(",");
|
|
|
|
|
|
|
|
q_add(table->driver->GetQuote());
|
|
|
|
q_add(*(char **)GB.Array.Get(fields, i));
|
|
|
|
q_add(table->driver->GetQuote());
|
|
|
|
}
|
|
|
|
|
2010-09-16 02:47:07 +02:00
|
|
|
info.fields = q_steal();
|
2007-12-30 17:41:49 +01:00
|
|
|
info.unique = VARGOPT(unique, FALSE);
|
|
|
|
|
|
|
|
table->driver->Index.Create(&table->conn->db, table->name, name, &info);
|
2010-09-16 02:47:07 +02:00
|
|
|
|
|
|
|
GB.FreeString(&info.fields);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
END_METHOD
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_METHOD(CINDEX_remove, GB_STRING name)
|
|
|
|
|
|
|
|
CTABLE *table = GB.SubCollection.Container(THIS);
|
|
|
|
char *name = GB.ToZeroString(ARG(name));
|
|
|
|
|
|
|
|
if (check_index(table, name, TRUE))
|
|
|
|
return;
|
|
|
|
|
|
|
|
table->driver->Index.Delete(&table->conn->db, table->name, name);
|
|
|
|
|
|
|
|
END_METHOD
|
|
|
|
|
|
|
|
|
|
|
|
GB_DESC CTableIndexesDesc[] =
|
|
|
|
{
|
|
|
|
GB_DECLARE(".TableIndexes", 0), GB_INHERITS(".SubCollection"),
|
|
|
|
|
2009-11-28 02:22:00 +01:00
|
|
|
GB_METHOD("Add", NULL, CINDEX_add, "(Name)s(Fields)String[];[(Unique)b]"),
|
2007-12-30 17:41:49 +01:00
|
|
|
GB_METHOD("Remove", NULL, CINDEX_remove, "(Name)s"),
|
|
|
|
|
|
|
|
GB_END_DECLARE
|
|
|
|
};
|
|
|
|
|
|
|
|
|