2004-07-16 00:15:55 +02:00
|
|
|
/* Copyright (C) 2004 MySQL AB
|
|
|
|
|
|
|
|
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 of the License, 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "mysql_priv.h"
|
|
|
|
#include "sql_acl.h"
|
|
|
|
#include "sql_select.h"
|
|
|
|
#include "parse_file.h"
|
2004-07-16 14:20:51 +02:00
|
|
|
#include "sp.h"
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
static int mysql_register_view(THD *thd, TABLE_LIST *view,
|
|
|
|
enum_view_create_mode mode);
|
|
|
|
|
|
|
|
const char *sql_updatable_view_key_names[]= { "NO", "YES", "LIMIT1", NullS };
|
|
|
|
TYPELIB sql_updatable_view_key_typelib=
|
|
|
|
{
|
|
|
|
array_elements(sql_updatable_view_key_names)-1, "",
|
|
|
|
sql_updatable_view_key_names
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Creating/altering VIEW procedure
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_create_view()
|
|
|
|
thd - thread handler
|
|
|
|
mode - VIEW_CREATE_NEW, VIEW_ALTER, VIEW_CREATE_OR_REPLACE
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
0 OK
|
|
|
|
-1 Error
|
|
|
|
1 Error and error message given
|
|
|
|
*/
|
2004-09-11 22:52:55 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
int mysql_create_view(THD *thd,
|
|
|
|
enum_view_create_mode mode)
|
|
|
|
{
|
|
|
|
LEX *lex= thd->lex;
|
|
|
|
bool link_to_local;
|
|
|
|
/* first table in list is target VIEW name => cut off it */
|
|
|
|
TABLE_LIST *view= lex->unlink_first_table(&link_to_local);
|
|
|
|
TABLE_LIST *tables= lex->query_tables;
|
2004-07-16 22:44:14 +02:00
|
|
|
TABLE_LIST *tbl;
|
2004-08-24 18:50:16 +02:00
|
|
|
SELECT_LEX *select_lex= &lex->select_lex, *sl;
|
2004-07-16 00:15:55 +02:00
|
|
|
SELECT_LEX_UNIT *unit= &lex->unit;
|
|
|
|
int res= 0;
|
|
|
|
DBUG_ENTER("mysql_create_view");
|
|
|
|
|
2004-09-02 11:09:26 +02:00
|
|
|
if (lex->proc_list.first ||
|
|
|
|
lex->result)
|
|
|
|
{
|
|
|
|
my_error(ER_VIEW_SELECT_CLAUSE, MYF(0), (lex->result ?
|
|
|
|
"INTO" :
|
|
|
|
"PROCEDURE"));
|
|
|
|
res= -1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
if (lex->derived_tables ||
|
2004-07-16 00:15:55 +02:00
|
|
|
lex->variables_used || lex->param_list.elements)
|
|
|
|
{
|
2004-09-02 11:09:26 +02:00
|
|
|
my_error((lex->derived_tables ?
|
|
|
|
ER_VIEW_SELECT_DERIVED :
|
|
|
|
ER_VIEW_SELECT_VARIABLE), MYF(0));
|
2004-07-16 00:15:55 +02:00
|
|
|
res= -1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
if (check_access(thd, CREATE_VIEW_ACL, view->db, &view->grant.privilege,
|
|
|
|
0, 0) ||
|
|
|
|
grant_option && check_grant(thd, CREATE_VIEW_ACL, view, 0, 1, 0))
|
|
|
|
DBUG_RETURN(1);
|
2004-08-24 18:50:16 +02:00
|
|
|
for (sl= select_lex; sl; sl= sl->next_select())
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-08-24 18:50:16 +02:00
|
|
|
for (tbl= sl->get_table_list(); tbl; tbl= tbl->next_local)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-08-24 18:50:16 +02:00
|
|
|
/*
|
|
|
|
Ensure that we have some privilage on this table, more strict check
|
|
|
|
will be done on column level after preparation,
|
|
|
|
*/
|
2004-09-03 21:38:45 +02:00
|
|
|
if (check_some_access(thd, VIEW_ANY_ACL, tbl))
|
2004-08-24 18:50:16 +02:00
|
|
|
{
|
|
|
|
my_printf_error(ER_TABLEACCESS_DENIED_ERROR,
|
|
|
|
ER(ER_TABLEACCESS_DENIED_ERROR),
|
|
|
|
MYF(0),
|
|
|
|
"ANY",
|
|
|
|
thd->priv_user,
|
|
|
|
thd->host_or_ip,
|
|
|
|
tbl->real_name);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
/* mark this table as table which will be checked after preparation */
|
|
|
|
tbl->table_in_first_from_clause= 1;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
2004-08-24 18:50:16 +02:00
|
|
|
/*
|
|
|
|
We need to check only SELECT_ACL for all normal fields, fields
|
2004-09-03 21:38:45 +02:00
|
|
|
where we need any privilege will be marked later
|
2004-08-24 18:50:16 +02:00
|
|
|
*/
|
|
|
|
tbl->grant.want_privilege= SELECT_ACL;
|
|
|
|
/*
|
|
|
|
Make sure that all rights are loaded to table 'grant' field.
|
|
|
|
|
|
|
|
tbl->real_name will be correct name of table because VIEWs are
|
|
|
|
not opened yet.
|
|
|
|
*/
|
|
|
|
fill_effective_table_privileges(thd, &tbl->grant, tbl->db,
|
|
|
|
tbl->real_name);
|
|
|
|
}
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (&lex->select_lex != lex->all_selects_list)
|
|
|
|
{
|
|
|
|
/* check tables of subqueries */
|
2004-07-16 22:44:14 +02:00
|
|
|
for (tbl= tables; tbl; tbl= tbl->next_global)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
if (!tbl->table_in_first_from_clause)
|
|
|
|
{
|
|
|
|
if (check_access(thd, SELECT_ACL, tbl->db,
|
|
|
|
&tbl->grant.privilege, 0, 0) ||
|
|
|
|
grant_option && check_grant(thd, SELECT_ACL, tbl, 0, 1, 0))
|
|
|
|
{
|
|
|
|
res= 1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
Mark fields for special privilege check (any privilege)
|
|
|
|
*/
|
2004-08-24 18:50:16 +02:00
|
|
|
for (sl= select_lex; sl; sl= sl->next_select())
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-08-24 18:50:16 +02:00
|
|
|
List_iterator_fast<Item> it(sl->item_list);
|
2004-07-16 00:15:55 +02:00
|
|
|
Item *item;
|
|
|
|
while ((item= it++))
|
|
|
|
{
|
2004-09-16 22:45:20 +02:00
|
|
|
Item_field *field;
|
|
|
|
if ((field= item->filed_for_view_update()))
|
|
|
|
field->any_privileges= 1;
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if ((res= open_and_lock_tables(thd, tables)))
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
|
|
|
|
/* check that tables are not temporary */
|
2004-07-16 22:44:14 +02:00
|
|
|
for (tbl= tables; tbl; tbl= tbl->next_global)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
if (tbl->table->tmp_table != NO_TMP_TABLE && !tbl->view)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
my_error(ER_VIEW_SELECT_TMPTABLE, MYF(0), tbl->alias);
|
|
|
|
res= -1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copy privileges of underlaying VIEWs which was filled by
|
|
|
|
fill_effective_table_privileges
|
|
|
|
(they was not copied in derived tables processing)
|
|
|
|
*/
|
|
|
|
tbl->table->grant.privilege= tbl->grant.privilege;
|
|
|
|
}
|
|
|
|
|
|
|
|
// prepare select to resolve all fields
|
|
|
|
lex->view_prepare_mode= 1;
|
2004-08-23 11:38:55 +02:00
|
|
|
if (unit->prepare(thd, 0, 0))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
some errors from prepare are reported to user, if is not then
|
|
|
|
it will be checked after err: label
|
|
|
|
*/
|
|
|
|
res= 1;
|
2004-07-16 00:15:55 +02:00
|
|
|
goto err;
|
2004-08-23 11:38:55 +02:00
|
|
|
}
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
/* view list (list of view fields names) */
|
|
|
|
if (lex->view_list.elements)
|
|
|
|
{
|
|
|
|
List_iterator_fast<Item> it(select_lex->item_list);
|
|
|
|
List_iterator_fast<LEX_STRING> nm(lex->view_list);
|
|
|
|
Item *item;
|
|
|
|
LEX_STRING *name;
|
2004-09-03 20:43:04 +02:00
|
|
|
|
|
|
|
if (lex->view_list.elements != select_lex->item_list.elements)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
my_message(ER_VIEW_WRONG_LIST, ER(ER_VIEW_WRONG_LIST), MYF(0));
|
|
|
|
goto err;
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
2004-09-03 20:43:04 +02:00
|
|
|
while ((item= it++, name= nm++))
|
|
|
|
item->set_name(name->str, name->length, system_charset_info);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
|
2004-07-21 11:14:45 +02:00
|
|
|
/* Test absence of duplicates names */
|
|
|
|
{
|
|
|
|
Item *item;
|
|
|
|
List_iterator_fast<Item> it(select_lex->item_list);
|
|
|
|
it++;
|
2004-09-03 21:38:45 +02:00
|
|
|
while ((item= it++))
|
2004-07-21 11:14:45 +02:00
|
|
|
{
|
|
|
|
Item *check;
|
|
|
|
List_iterator_fast<Item> itc(select_lex->item_list);
|
2004-09-03 21:38:45 +02:00
|
|
|
while ((check= itc++) && check != item)
|
2004-07-21 11:14:45 +02:00
|
|
|
{
|
|
|
|
if (strcmp(item->name, check->name) == 0)
|
|
|
|
{
|
|
|
|
my_error(ER_DUP_FIELDNAME, MYF(0), item->name);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
/*
|
|
|
|
Compare/check grants on view with grants of underlaying tables
|
|
|
|
*/
|
2004-08-24 18:50:16 +02:00
|
|
|
for (sl= select_lex; sl; sl= sl->next_select())
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
char *db= view->db ? view->db : thd->db;
|
2004-08-24 18:50:16 +02:00
|
|
|
List_iterator_fast<Item> it(sl->item_list);
|
2004-07-16 00:15:55 +02:00
|
|
|
Item *item;
|
|
|
|
fill_effective_table_privileges(thd, &view->grant, db,
|
|
|
|
view->real_name);
|
2004-09-03 20:43:04 +02:00
|
|
|
while ((item= it++))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-16 22:45:20 +02:00
|
|
|
Item_field *fld;
|
2004-07-16 00:15:55 +02:00
|
|
|
uint priv= (get_column_grant(thd, &view->grant, db,
|
|
|
|
view->real_name, item->name) &
|
|
|
|
VIEW_ANY_ACL);
|
2004-09-16 22:45:20 +02:00
|
|
|
if ((fld= item->filed_for_view_update()))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
/*
|
2004-09-16 23:16:57 +02:00
|
|
|
Do we have more privilegeson view field then underlying table field
|
2004-07-16 00:15:55 +02:00
|
|
|
*/
|
2004-09-16 23:16:57 +02:00
|
|
|
if ((~fld->have_privileges & priv))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
/* VIEW column has more privileges */
|
|
|
|
my_printf_error(ER_COLUMNACCESS_DENIED_ERROR,
|
|
|
|
ER(ER_COLUMNACCESS_DENIED_ERROR),
|
|
|
|
MYF(0),
|
|
|
|
"create view",
|
|
|
|
thd->priv_user,
|
|
|
|
thd->host_or_ip,
|
|
|
|
item->name,
|
|
|
|
view->real_name);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-09-09 05:59:26 +02:00
|
|
|
if (wait_if_global_read_lock(thd, 0, 0))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-09 05:59:26 +02:00
|
|
|
res= -1;
|
2004-07-16 00:15:55 +02:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_open));
|
2004-09-03 20:43:04 +02:00
|
|
|
res= mysql_register_view(thd, view, mode);
|
2004-07-16 00:15:55 +02:00
|
|
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
|
|
|
start_waiting_global_read_lock(thd);
|
2004-09-03 20:43:04 +02:00
|
|
|
if (res)
|
|
|
|
goto err;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
send_ok(thd);
|
|
|
|
lex->link_first_table_back(view, link_to_local);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err:
|
|
|
|
thd->proc_info= "end";
|
|
|
|
lex->link_first_table_back(view, link_to_local);
|
|
|
|
unit->cleanup();
|
|
|
|
if (thd->net.report_error)
|
|
|
|
res= -1;
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-01 21:48:59 +02:00
|
|
|
/* index of revision number in following table */
|
|
|
|
static const int revision_number_position= 5;
|
|
|
|
/* index of last required parameter for making view */
|
2004-09-03 20:38:01 +02:00
|
|
|
static const int required_view_parameters= 7;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
/*
|
|
|
|
table of VIEW .frm field descriptors
|
|
|
|
|
|
|
|
Note that one should NOT change the order for this, as it's used by
|
|
|
|
parse()
|
|
|
|
*/
|
2004-07-16 00:15:55 +02:00
|
|
|
static File_option view_parameters[]=
|
2004-09-03 20:43:04 +02:00
|
|
|
{{{(char*) "query", 5}, offsetof(TABLE_LIST, query),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_STRING},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*) "md5", 3}, offsetof(TABLE_LIST, md5),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_STRING},
|
2004-09-03 21:38:45 +02:00
|
|
|
{{(char*) "updatable", 9}, offsetof(TABLE_LIST, updatable_view),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_ULONGLONG},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*) "algorithm", 9}, offsetof(TABLE_LIST, algorithm),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_ULONGLONG},
|
2004-09-05 21:14:27 +02:00
|
|
|
{{"with_check_option", 17}, offsetof(TABLE_LIST, with_check),
|
2004-09-03 14:18:40 +02:00
|
|
|
FILE_OPTIONS_ULONGLONG},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*) "revision", 8}, offsetof(TABLE_LIST, revision),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_REV},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*) "timestamp", 9}, offsetof(TABLE_LIST, timestamp),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_TIMESTAMP},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*)"create-version", 14},offsetof(TABLE_LIST, file_version),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_ULONGLONG},
|
2004-09-03 20:43:04 +02:00
|
|
|
{{(char*) "source", 6}, offsetof(TABLE_LIST, source),
|
2004-07-16 00:15:55 +02:00
|
|
|
FILE_OPTIONS_ESTRING},
|
|
|
|
{{NULL, 0}, 0,
|
|
|
|
FILE_OPTIONS_STRING}
|
|
|
|
};
|
|
|
|
|
|
|
|
static LEX_STRING view_file_type[]= {{(char*)"VIEW", 4}};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Register VIEW (write .frm & process .frm's history backups)
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_register_view()
|
|
|
|
thd - thread handler
|
|
|
|
view - view description
|
|
|
|
mode - VIEW_CREATE_NEW, VIEW_ALTER, VIEW_CREATE_OR_REPLACE
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK
|
|
|
|
-1 Error
|
|
|
|
1 Error and error message given
|
|
|
|
*/
|
2004-09-11 22:52:55 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
static int mysql_register_view(THD *thd, TABLE_LIST *view,
|
|
|
|
enum_view_create_mode mode)
|
|
|
|
{
|
|
|
|
char buff[4096];
|
|
|
|
String str(buff,(uint32) sizeof(buff), system_charset_info);
|
|
|
|
char md5[33];
|
|
|
|
bool can_be_merged;
|
|
|
|
char dir_buff[FN_REFLEN], file_buff[FN_REFLEN];
|
|
|
|
LEX_STRING dir, file;
|
|
|
|
DBUG_ENTER("mysql_register_view");
|
|
|
|
|
|
|
|
// print query
|
|
|
|
str.length(0);
|
2004-07-20 17:51:02 +02:00
|
|
|
{
|
|
|
|
ulong sql_mode= thd->variables.sql_mode & MODE_ANSI_QUOTES;
|
|
|
|
thd->variables.sql_mode&= ~MODE_ANSI_QUOTES;
|
|
|
|
thd->lex->unit.print(&str);
|
|
|
|
thd->variables.sql_mode|= sql_mode;
|
|
|
|
}
|
2004-07-16 00:15:55 +02:00
|
|
|
str.append('\0');
|
|
|
|
DBUG_PRINT("VIEW", ("View: %s", str.ptr()));
|
|
|
|
|
|
|
|
// print file name
|
|
|
|
(void) my_snprintf(dir_buff, FN_REFLEN, "%s/%s/",
|
|
|
|
mysql_data_home, view->db);
|
|
|
|
unpack_filename(dir_buff, dir_buff);
|
|
|
|
dir.str= dir_buff;
|
|
|
|
dir.length= strlen(dir_buff);
|
|
|
|
|
|
|
|
file.str= file_buff;
|
2004-09-03 20:43:04 +02:00
|
|
|
file.length= (strxnmov(file_buff, FN_REFLEN, view->real_name, reg_ext,
|
|
|
|
NullS) - file_buff);
|
2004-07-16 00:15:55 +02:00
|
|
|
/* init timestamp */
|
2004-09-03 20:43:04 +02:00
|
|
|
if (!view->timestamp.str)
|
2004-07-16 00:15:55 +02:00
|
|
|
view->timestamp.str= view->timestamp_buffer;
|
|
|
|
|
|
|
|
// check old .frm
|
|
|
|
{
|
|
|
|
char path_buff[FN_REFLEN];
|
|
|
|
LEX_STRING path;
|
2004-09-03 20:43:04 +02:00
|
|
|
File_parser *parser;
|
2004-09-29 15:35:01 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
path.str= path_buff;
|
|
|
|
fn_format(path_buff, file.str, dir.str, 0, MY_UNPACK_FILENAME);
|
|
|
|
path.length= strlen(path_buff);
|
|
|
|
|
|
|
|
if (!access(path.str, F_OK))
|
|
|
|
{
|
|
|
|
if (mode == VIEW_CREATE_NEW)
|
|
|
|
{
|
|
|
|
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), view->alias);
|
2004-09-03 14:18:40 +02:00
|
|
|
DBUG_RETURN(-1);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
if (!(parser= sql_parse_prepare(&path, &thd->mem_root, 0)))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
|
|
|
if (!parser->ok() ||
|
|
|
|
strncmp("VIEW", parser->type()->str, parser->type()->length))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
my_error(ER_WRONG_OBJECT, MYF(0), (view->db ? view->db : thd->db),
|
|
|
|
view->real_name, "VIEW");
|
2004-09-05 21:14:27 +02:00
|
|
|
DBUG_RETURN(-1);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
2004-09-03 20:43:04 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
read revision number
|
2004-09-11 22:52:55 +02:00
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
TODO: read dependense list, too, to process cascade/restrict
|
|
|
|
TODO: special cascade/restrict procedure for alter?
|
|
|
|
*/
|
|
|
|
if (parser->parse((gptr)view, &thd->mem_root,
|
|
|
|
view_parameters + revision_number_position, 1))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-05 21:14:27 +02:00
|
|
|
DBUG_RETURN(thd->net.report_error? -1 : 0);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (mode == VIEW_ALTER)
|
|
|
|
{
|
|
|
|
my_error(ER_NO_SUCH_TABLE, MYF(0), view->db, view->alias);
|
2004-09-03 14:18:40 +02:00
|
|
|
DBUG_RETURN(-1);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// fill structure
|
|
|
|
view->query.str= (char*)str.ptr();
|
|
|
|
view->query.length= str.length()-1; // we do not need last \0
|
|
|
|
view->source.str= thd->query;
|
|
|
|
view->source.length= thd->query_length;
|
|
|
|
view->file_version= 1;
|
|
|
|
view->calc_md5(md5);
|
|
|
|
view->md5.str= md5;
|
|
|
|
view->md5.length= 32;
|
|
|
|
can_be_merged= thd->lex->can_be_merged();
|
|
|
|
if (thd->lex->create_view_algorithm == VIEW_ALGORITHM_MERGE &&
|
|
|
|
!thd->lex->can_be_merged())
|
|
|
|
{
|
|
|
|
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN, ER_WARN_VIEW_MERGE,
|
|
|
|
ER(ER_WARN_VIEW_MERGE));
|
|
|
|
thd->lex->create_view_algorithm= VIEW_ALGORITHM_UNDEFINED;
|
|
|
|
}
|
|
|
|
view->algorithm= thd->lex->create_view_algorithm;
|
2004-09-03 14:18:40 +02:00
|
|
|
view->with_check= thd->lex->create_view_check;
|
2004-07-22 16:52:04 +02:00
|
|
|
if ((view->updatable_view= (can_be_merged &&
|
2004-09-03 21:38:45 +02:00
|
|
|
view->algorithm != VIEW_ALGORITHM_TMPTABLE)))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
// TODO: change here when we will support UNIONs
|
|
|
|
for (TABLE_LIST *tbl= (TABLE_LIST *)thd->lex->select_lex.table_list.first;
|
|
|
|
tbl;
|
|
|
|
tbl= tbl->next_local)
|
|
|
|
{
|
2004-09-03 21:38:45 +02:00
|
|
|
if (tbl->view && !tbl->updatable_view)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-07-22 16:52:04 +02:00
|
|
|
view->updatable_view= 0;
|
2004-07-16 00:15:55 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-09-03 14:18:40 +02:00
|
|
|
|
|
|
|
if (view->with_check != VIEW_CHECK_NONE &&
|
|
|
|
!view->updatable_view)
|
|
|
|
{
|
2004-09-29 15:35:01 +02:00
|
|
|
my_error(ER_VIEW_NONUPD_CHECK, MYF(0), view->db, view->real_name);
|
2004-09-03 14:18:40 +02:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
if (sql_create_definition_file(&dir, &file, view_file_type,
|
|
|
|
(gptr)view, view_parameters, 3))
|
|
|
|
{
|
2004-09-03 14:18:40 +02:00
|
|
|
DBUG_RETURN(thd->net.report_error? -1 : 1);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
read VIEW .frm and create structures
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_make_view()
|
|
|
|
parser - parser object;
|
|
|
|
table - TABLE_LIST structure for filling
|
2004-07-21 03:26:20 +02:00
|
|
|
|
|
|
|
RETURN
|
2004-09-03 20:43:04 +02:00
|
|
|
0 ok
|
|
|
|
1 error
|
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
*/
|
2004-09-03 20:43:04 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
my_bool
|
|
|
|
mysql_make_view(File_parser *parser, TABLE_LIST *table)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("mysql_make_view");
|
|
|
|
|
|
|
|
if (table->view)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info",
|
2004-09-03 20:43:04 +02:00
|
|
|
("VIEW %s.%s is already processed on previous PS/SP execution",
|
2004-07-16 00:15:55 +02:00
|
|
|
table->view_db.str, table->view_name.str));
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-07-16 22:44:14 +02:00
|
|
|
SELECT_LEX *end;
|
2004-07-16 00:15:55 +02:00
|
|
|
THD *thd= current_thd;
|
|
|
|
LEX *old_lex= thd->lex, *lex;
|
|
|
|
int res= 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
For now we assume that tables will not be changed during PS life (it
|
|
|
|
will be TRUE as far as we make new table cache).
|
|
|
|
*/
|
|
|
|
Item_arena *arena= thd->current_arena, backup;
|
2004-09-10 01:22:44 +02:00
|
|
|
if (arena->is_conventional())
|
2004-09-09 05:59:26 +02:00
|
|
|
arena= 0;
|
|
|
|
else
|
2004-07-16 00:15:55 +02:00
|
|
|
thd->set_n_backup_item_arena(arena, &backup);
|
|
|
|
|
|
|
|
/* init timestamp */
|
2004-09-03 20:43:04 +02:00
|
|
|
if (!table->timestamp.str)
|
2004-07-16 00:15:55 +02:00
|
|
|
table->timestamp.str= table->timestamp_buffer;
|
|
|
|
/*
|
|
|
|
TODO: when VIEWs will be stored in cache, table mem_root should
|
|
|
|
be used here
|
|
|
|
*/
|
2004-09-01 21:48:59 +02:00
|
|
|
if (parser->parse((gptr)table, &thd->mem_root, view_parameters,
|
2004-09-03 20:38:01 +02:00
|
|
|
required_view_parameters))
|
2004-07-16 00:15:55 +02:00
|
|
|
goto err;
|
|
|
|
|
|
|
|
/*
|
|
|
|
Save VIEW parameters, which will be wiped out by derived table
|
|
|
|
processing
|
|
|
|
*/
|
|
|
|
table->view_db.str= table->db;
|
|
|
|
table->view_db.length= table->db_length;
|
|
|
|
table->view_name.str= table->real_name;
|
|
|
|
table->view_name.length= table->real_name_length;
|
|
|
|
|
2004-07-16 14:20:51 +02:00
|
|
|
/*TODO: md5 test here and warning if it is differ */
|
2004-07-16 00:15:55 +02:00
|
|
|
|
2004-07-16 14:20:51 +02:00
|
|
|
/*
|
|
|
|
TODO: TABLE mem root should be used here when VIEW will be stored in
|
|
|
|
TABLE cache
|
|
|
|
|
|
|
|
now Lex placed in statement memory
|
|
|
|
*/
|
|
|
|
table->view= lex= thd->lex= (LEX*) new(&thd->mem_root) st_lex_local;
|
2004-07-22 00:26:33 +02:00
|
|
|
mysql_init_query(thd, (uchar*)table->query.str, table->query.length, TRUE);
|
2004-07-16 00:15:55 +02:00
|
|
|
lex->select_lex.select_number= ++thd->select_number;
|
|
|
|
old_lex->derived_tables|= DERIVED_VIEW;
|
|
|
|
{
|
|
|
|
ulong options= thd->options;
|
|
|
|
/* switch off modes which can prevent normal parsing of VIEW
|
|
|
|
- MODE_REAL_AS_FLOAT affect only CREATE TABLE parsing
|
|
|
|
+ MODE_PIPES_AS_CONCAT affect expression parsing
|
|
|
|
+ MODE_ANSI_QUOTES affect expression parsing
|
|
|
|
+ MODE_IGNORE_SPACE affect expression parsing
|
|
|
|
- MODE_NOT_USED not used :)
|
|
|
|
* MODE_ONLY_FULL_GROUP_BY affect execution
|
|
|
|
* MODE_NO_UNSIGNED_SUBTRACTION affect execution
|
|
|
|
- MODE_NO_DIR_IN_CREATE affect table creation only
|
|
|
|
- MODE_POSTGRESQL compounded from other modes
|
|
|
|
- MODE_ORACLE compounded from other modes
|
|
|
|
- MODE_MSSQL compounded from other modes
|
|
|
|
- MODE_DB2 compounded from other modes
|
|
|
|
- MODE_MAXDB affect only CREATE TABLE parsing
|
|
|
|
- MODE_NO_KEY_OPTIONS affect only SHOW
|
|
|
|
- MODE_NO_TABLE_OPTIONS affect only SHOW
|
|
|
|
- MODE_NO_FIELD_OPTIONS affect only SHOW
|
|
|
|
- MODE_MYSQL323 affect only SHOW
|
|
|
|
- MODE_MYSQL40 affect only SHOW
|
|
|
|
- MODE_ANSI compounded from other modes
|
|
|
|
(+ transaction mode)
|
|
|
|
? MODE_NO_AUTO_VALUE_ON_ZERO affect UPDATEs
|
|
|
|
+ MODE_NO_BACKSLASH_ESCAPES affect expression parsing
|
|
|
|
*/
|
|
|
|
thd->options&= ~(MODE_PIPES_AS_CONCAT | MODE_ANSI_QUOTES |
|
|
|
|
MODE_IGNORE_SPACE | MODE_NO_BACKSLASH_ESCAPES);
|
2004-09-01 21:48:59 +02:00
|
|
|
CHARSET_INFO *save_cs= thd->variables.character_set_client;
|
2004-09-03 20:38:01 +02:00
|
|
|
thd->variables.character_set_client= system_charset_info;
|
2004-07-16 00:15:55 +02:00
|
|
|
res= yyparse((void *)thd);
|
2004-09-01 21:48:59 +02:00
|
|
|
thd->variables.character_set_client= save_cs;
|
2004-07-16 00:15:55 +02:00
|
|
|
thd->options= options;
|
|
|
|
}
|
|
|
|
if (!res && !thd->is_fatal_error)
|
|
|
|
{
|
2004-07-21 03:26:20 +02:00
|
|
|
TABLE_LIST *top_view= (table->belong_to_view ?
|
|
|
|
table->belong_to_view :
|
|
|
|
table);
|
2004-09-11 22:52:55 +02:00
|
|
|
TABLE_LIST *view_tables= lex->query_tables;
|
|
|
|
TABLE_LIST *view_tables_tail= 0;
|
2004-07-16 14:20:51 +02:00
|
|
|
|
2004-09-01 18:00:41 +02:00
|
|
|
if (lex->spfuns.records)
|
|
|
|
{
|
|
|
|
/* move SP to main LEX */
|
|
|
|
sp_merge_funs(old_lex, lex);
|
2004-09-11 22:52:55 +02:00
|
|
|
/* open mysq.proc for functions which are not in cache */
|
2004-09-01 18:00:41 +02:00
|
|
|
if (old_lex->proc_table == 0 &&
|
|
|
|
(old_lex->proc_table=
|
|
|
|
(TABLE_LIST*)thd->calloc(sizeof(TABLE_LIST))) != 0)
|
|
|
|
{
|
|
|
|
TABLE_LIST *table= old_lex->proc_table;
|
|
|
|
table->db= (char*)"mysql";
|
|
|
|
table->db_length= 5;
|
|
|
|
table->real_name= table->alias= (char*)"proc";
|
|
|
|
table->real_name_length= 4;
|
|
|
|
table->cacheable_table= 1;
|
2004-09-11 22:52:55 +02:00
|
|
|
old_lex->add_to_query_tables(table);
|
2004-09-01 18:00:41 +02:00
|
|
|
}
|
|
|
|
}
|
2004-09-11 22:52:55 +02:00
|
|
|
/* cleanup LEX */
|
2004-09-04 22:05:12 +02:00
|
|
|
if (lex->spfuns.array.buffer)
|
|
|
|
hash_free(&lex->spfuns);
|
2004-07-16 14:20:51 +02:00
|
|
|
|
2004-09-11 22:52:55 +02:00
|
|
|
/*
|
|
|
|
mark to avoid temporary table using and put view reference and find
|
|
|
|
last view table
|
|
|
|
*/
|
|
|
|
for (TABLE_LIST *tbl= view_tables;
|
|
|
|
tbl;
|
|
|
|
tbl= (view_tables_tail= tbl)->next_global)
|
2004-07-21 03:26:20 +02:00
|
|
|
{
|
2004-07-16 00:15:55 +02:00
|
|
|
tbl->skip_temporary= 1;
|
2004-07-21 03:26:20 +02:00
|
|
|
tbl->belong_to_view= top_view;
|
|
|
|
}
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
check rights to run commands (EXPLAIN SELECT & SHOW CREATE) which show
|
|
|
|
underlaying tables
|
|
|
|
*/
|
2004-09-17 21:23:59 +02:00
|
|
|
if ((old_lex->sql_command == SQLCOM_SELECT && old_lex->describe))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-11 22:52:55 +02:00
|
|
|
if (check_table_access(thd, SELECT_ACL, view_tables, 1) &&
|
2004-09-17 21:23:59 +02:00
|
|
|
check_table_access(thd, SHOW_VIEW_ACL, table, 1))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
my_error(ER_VIEW_NO_EXPLAIN, MYF(0));
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
}
|
2004-09-17 21:23:59 +02:00
|
|
|
else if (old_lex->sql_command == SQLCOM_SHOW_CREATE)
|
|
|
|
{
|
|
|
|
if (check_table_access(thd, SHOW_VIEW_ACL, table, 0))
|
|
|
|
goto err;
|
|
|
|
}
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
/* move SQL_NO_CACHE & Co to whole query */
|
|
|
|
old_lex->safe_to_cache_query= (old_lex->safe_to_cache_query &&
|
|
|
|
lex->safe_to_cache_query);
|
|
|
|
/* move SQL_CACHE to whole query */
|
|
|
|
if (lex->select_lex.options & OPTION_TO_QUERY_CACHE)
|
|
|
|
old_lex->select_lex.options|= OPTION_TO_QUERY_CACHE;
|
|
|
|
|
2004-09-11 22:52:55 +02:00
|
|
|
/*
|
|
|
|
Put tables of VIEW after VIEW TABLE_LIST
|
|
|
|
|
|
|
|
NOTE: It is important for UPDATE/INSERT/DELETE checks to have this
|
|
|
|
tables just after VIEW instead of tail of list, to be able check that
|
|
|
|
table is unique. Also we store old next table for the same purpose.
|
|
|
|
*/
|
|
|
|
table->old_next= table->next_global;
|
|
|
|
if (view_tables)
|
|
|
|
{
|
|
|
|
if (table->next_global)
|
|
|
|
{
|
|
|
|
table->next_global->prev_global= &view_tables_tail->next_global;
|
|
|
|
view_tables_tail->next_global= table->old_next;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lex->query_tables_last= &view_tables_tail->next_global;
|
|
|
|
}
|
|
|
|
view_tables->prev_global= &table->next_global;
|
|
|
|
table->next_global= view_tables;
|
|
|
|
}
|
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
/*
|
|
|
|
check MERGE algorithm ability
|
|
|
|
- algorithm is not explicit TEMPORARY TABLE
|
|
|
|
- VIEW SELECT allow marging
|
|
|
|
- VIEW used in subquery or command support MERGE algorithm
|
|
|
|
*/
|
2004-09-03 20:43:04 +02:00
|
|
|
if (table->algorithm != VIEW_ALGORITHM_TMPTABLE &&
|
2004-07-16 00:15:55 +02:00
|
|
|
lex->can_be_merged() &&
|
|
|
|
(table->select_lex->master_unit() != &old_lex->unit ||
|
2004-08-24 21:51:23 +02:00
|
|
|
old_lex->can_use_merged()) &&
|
|
|
|
!old_lex->can_not_use_merged())
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
TODO: support multi tables substitutions
|
|
|
|
*/
|
|
|
|
/* lex should contain at least one table */
|
2004-09-11 22:52:55 +02:00
|
|
|
DBUG_ASSERT(view_tables != 0);
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
table->effective_algorithm= VIEW_ALGORITHM_MERGE;
|
2004-08-24 21:51:23 +02:00
|
|
|
DBUG_PRINT("info", ("algorithm: MERGE"));
|
2004-07-22 16:52:04 +02:00
|
|
|
table->updatable= (table->updatable_view != 0);
|
2004-07-16 00:15:55 +02:00
|
|
|
|
2004-09-11 22:52:55 +02:00
|
|
|
table->ancestor= view_tables;
|
|
|
|
/*
|
|
|
|
next table should include SELECT_LEX under this table SELECT_LEX
|
|
|
|
|
|
|
|
TODO: ehere should be loop for multi tables substitution
|
|
|
|
*/
|
2004-07-16 00:15:55 +02:00
|
|
|
table->ancestor->select_lex= table->select_lex;
|
|
|
|
/*
|
|
|
|
move lock type (TODO: should we issue error in case of TMPTABLE
|
|
|
|
algorithm and non-read locking)?
|
|
|
|
*/
|
2004-09-11 22:52:55 +02:00
|
|
|
view_tables->lock_type= table->lock_type;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
/* Store WHERE clause for postprocessing in setup_ancestor */
|
|
|
|
table->where= lex->select_lex.where;
|
|
|
|
|
|
|
|
/*
|
|
|
|
This SELECT_LEX will be linked in global SELECT_LEX list
|
|
|
|
to make it processed by mysql_handle_derived(),
|
|
|
|
but it will not be included to SELECT_LEX tree, because it
|
|
|
|
will not be executed
|
|
|
|
*/
|
|
|
|
goto ok;
|
|
|
|
}
|
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
table->effective_algorithm= VIEW_ALGORITHM_TMPTABLE;
|
2004-08-24 21:51:23 +02:00
|
|
|
DBUG_PRINT("info", ("algorithm: TEMPORARY TABLE"));
|
2004-07-20 07:48:28 +02:00
|
|
|
lex->select_lex.linkage= DERIVED_TABLE_TYPE;
|
2004-07-22 16:52:04 +02:00
|
|
|
table->updatable= 0;
|
2004-09-03 14:18:40 +02:00
|
|
|
table->with_check= VIEW_CHECK_NONE;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
|
|
|
/* SELECT tree link */
|
|
|
|
lex->unit.include_down(table->select_lex);
|
|
|
|
lex->unit.slave= &lex->select_lex; // fix include_down initialisation
|
|
|
|
|
|
|
|
table->derived= &lex->unit;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ok:
|
|
|
|
if (arena)
|
|
|
|
thd->restore_backup_item_arena(arena, &backup);
|
|
|
|
/* global SELECT list linking */
|
|
|
|
end= &lex->select_lex; // primary SELECT_LEX is always last
|
|
|
|
end->link_next= old_lex->all_selects_list;
|
|
|
|
old_lex->all_selects_list->link_prev= &end->link_next;
|
|
|
|
old_lex->all_selects_list= lex->all_selects_list;
|
|
|
|
lex->all_selects_list->link_prev=
|
|
|
|
(st_select_lex_node**)&old_lex->all_selects_list;
|
|
|
|
|
|
|
|
thd->lex= old_lex;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
err:
|
|
|
|
if (arena)
|
|
|
|
thd->restore_backup_item_arena(arena, &backup);
|
|
|
|
table->view= 0; // now it is not VIEW placeholder
|
|
|
|
thd->lex= old_lex;
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
drop view
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_drop_view()
|
|
|
|
thd - thread handler
|
|
|
|
views - views to delete
|
|
|
|
drop_mode - cascade/check
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
0 OK
|
|
|
|
-1 Error
|
|
|
|
1 Error and error message given
|
|
|
|
*/
|
2004-09-03 20:43:04 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
int mysql_drop_view(THD *thd, TABLE_LIST *views, enum_drop_mode drop_mode)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("mysql_drop_view");
|
|
|
|
char path[FN_REFLEN];
|
|
|
|
TABLE_LIST *view;
|
|
|
|
bool type= 0;
|
|
|
|
|
|
|
|
for (view= views; view; view= view->next_local)
|
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
strxnmov(path, FN_REFLEN, mysql_data_home, "/", view->db, "/",
|
|
|
|
view->real_name, reg_ext, NullS);
|
2004-07-16 00:15:55 +02:00
|
|
|
(void) unpack_filename(path, path);
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_open));
|
|
|
|
if (access(path, F_OK) || (type= (mysql_frm_type(path) != FRMTYPE_VIEW)))
|
|
|
|
{
|
|
|
|
char name[FN_REFLEN];
|
|
|
|
my_snprintf(name, sizeof(name), "%s.%s", view->db, view->real_name);
|
|
|
|
if (thd->lex->drop_if_exists)
|
|
|
|
{
|
|
|
|
push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_NOTE,
|
|
|
|
ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
|
|
|
|
name);
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (type)
|
|
|
|
my_error(ER_WRONG_OBJECT, MYF(0), view->db, view->real_name, "VIEW");
|
|
|
|
else
|
|
|
|
my_error(ER_BAD_TABLE_ERROR, MYF(0), name);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
if (my_delete(path, MYF(MY_WME)))
|
|
|
|
goto err;
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
|
|
|
}
|
|
|
|
send_ok(thd);
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
err:
|
|
|
|
VOID(pthread_mutex_unlock(&LOCK_open));
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Check type of .frm if we are not going to parse it
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_frm_type()
|
|
|
|
path path to file
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FRMTYPE_ERROR error
|
|
|
|
FRMTYPE_TABLE table
|
|
|
|
FRMTYPE_VIEW view
|
|
|
|
*/
|
|
|
|
|
|
|
|
frm_type_enum mysql_frm_type(char *path)
|
|
|
|
{
|
|
|
|
File file;
|
|
|
|
char header[10]; //"TYPE=VIEW\n" it is 10 characters
|
2004-09-03 20:43:04 +02:00
|
|
|
int length;
|
2004-07-16 00:15:55 +02:00
|
|
|
DBUG_ENTER("mysql_frm_type");
|
|
|
|
|
|
|
|
if ((file= my_open(path, O_RDONLY | O_SHARE, MYF(MY_WME))) < 0)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(FRMTYPE_ERROR);
|
|
|
|
}
|
2004-09-03 21:38:45 +02:00
|
|
|
length= my_read(file, (byte*) header, 10, MYF(MY_WME));
|
2004-07-16 00:15:55 +02:00
|
|
|
my_close(file, MYF(MY_WME));
|
2004-09-03 20:43:04 +02:00
|
|
|
if (length == (int) MY_FILE_ERROR)
|
|
|
|
DBUG_RETURN(FRMTYPE_ERROR);
|
|
|
|
if (!strncmp(header, "TYPE=VIEW\n", 10))
|
|
|
|
DBUG_RETURN(FRMTYPE_VIEW);
|
|
|
|
DBUG_RETURN(FRMTYPE_TABLE); // Is probably a .frm table
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
check of key (primary or unique) presence in updatable view
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
check_key_in_view()
|
|
|
|
thd thread handler
|
|
|
|
view view for check with opened table
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK
|
|
|
|
TRUE view do not contain key or all fields
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool check_key_in_view(THD *thd, TABLE_LIST *view)
|
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
TABLE *table;
|
|
|
|
Item **trans;
|
|
|
|
KEY *key_info, *key_info_end;
|
|
|
|
uint i, elements_in_view;
|
2004-07-16 00:15:55 +02:00
|
|
|
DBUG_ENTER("check_key_in_view");
|
2004-09-03 20:43:04 +02:00
|
|
|
|
2004-07-16 00:15:55 +02:00
|
|
|
if (!view->view)
|
|
|
|
DBUG_RETURN(FALSE); /* it is normal table */
|
2004-09-03 20:43:04 +02:00
|
|
|
table= view->table;
|
|
|
|
trans= view->field_translation;
|
|
|
|
key_info_end= (key_info= table->key_info)+ table->keys;
|
2004-07-16 00:15:55 +02:00
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
elements_in_view= view->view->select_lex.item_list.elements;
|
2004-07-16 00:15:55 +02:00
|
|
|
DBUG_ASSERT(view->table != 0 && view->field_translation != 0);
|
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
/* Loop over all keys to see if a unique-not-null key is used */
|
|
|
|
for (;key_info != key_info_end ; key_info++)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
if ((key_info->flags & (HA_NOSAME | HA_NULL_PART_KEY)) == HA_NOSAME)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
KEY_PART_INFO *key_part= key_info->key_part;
|
2004-09-03 20:43:04 +02:00
|
|
|
KEY_PART_INFO *key_part_end= key_part + key_info->key_parts;
|
|
|
|
|
|
|
|
/* check that all key parts are used */
|
|
|
|
for (;;)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
uint k;
|
|
|
|
for (k= 0; k < elements_in_view; k++)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-16 22:45:20 +02:00
|
|
|
Item_field *field;
|
|
|
|
if ((field= trans[k]->filed_for_view_update()) &&
|
|
|
|
field->field == key_part->field)
|
2004-07-16 00:15:55 +02:00
|
|
|
break;
|
|
|
|
}
|
2004-09-03 20:43:04 +02:00
|
|
|
if (k == elements_in_view)
|
|
|
|
break; // Key is not possible
|
|
|
|
if (++key_part == key_part_end)
|
|
|
|
DBUG_RETURN(FALSE); // Found usable key
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
DBUG_PRINT("info", ("checking if all fields of table are used"));
|
2004-07-16 00:15:55 +02:00
|
|
|
/* check all fields presence */
|
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
Field **field_ptr;
|
|
|
|
for (field_ptr= table->field; *field_ptr; field_ptr++)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
for (i= 0; i < elements_in_view; i++)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-16 22:45:20 +02:00
|
|
|
Item_field *field;
|
|
|
|
if ((field= trans[i]->filed_for_view_update()) &&
|
|
|
|
field->field == *field_ptr)
|
2004-07-16 00:15:55 +02:00
|
|
|
break;
|
|
|
|
}
|
2004-09-03 20:43:04 +02:00
|
|
|
if (i == elements_in_view) // If field didn't exists
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
ulong mode= thd->variables.sql_updatable_view_key;
|
2004-09-03 20:43:04 +02:00
|
|
|
/*
|
|
|
|
0 == NO ; Don't give any errors
|
|
|
|
1 == YES ; Give always an error
|
|
|
|
2 == LIMIT1 ; Give an error if this is used with LIMIT 1
|
|
|
|
This is used to protect against gui programs that
|
|
|
|
uses LIMIT 1 to update just the current row. This
|
|
|
|
doesn't work reliable if the view doesn't have a
|
|
|
|
unique key or if the view doesn't use all fields in
|
|
|
|
table.
|
|
|
|
*/
|
2004-07-16 00:15:55 +02:00
|
|
|
if (mode == 1 ||
|
|
|
|
(mode == 2 &&
|
2004-09-03 20:43:04 +02:00
|
|
|
thd->lex->unit.global_parameters->select_limit == 1))
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
2004-09-03 20:43:04 +02:00
|
|
|
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_NOTE,
|
|
|
|
ER_WARN_VIEW_WITHOUT_KEY, ER(ER_WARN_VIEW_WITHOUT_KEY));
|
|
|
|
DBUG_RETURN(FALSE);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
insert fields from VIEW (MERGE algorithm) into given list
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
insert_view_fields()
|
|
|
|
list list for insertion
|
|
|
|
view view for processing
|
|
|
|
*/
|
|
|
|
|
|
|
|
void insert_view_fields(List<Item> *list, TABLE_LIST *view)
|
|
|
|
{
|
2004-09-03 20:43:04 +02:00
|
|
|
uint elements_in_view= view->view->select_lex.item_list.elements;
|
|
|
|
Item **trans;
|
2004-07-16 00:15:55 +02:00
|
|
|
DBUG_ENTER("insert_view_fields");
|
2004-09-03 20:43:04 +02:00
|
|
|
|
|
|
|
if (!(trans= view->field_translation))
|
2004-07-16 00:15:55 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
|
2004-09-03 20:43:04 +02:00
|
|
|
for (uint i= 0; i < elements_in_view; i++)
|
2004-07-16 00:15:55 +02:00
|
|
|
{
|
2004-09-16 22:45:20 +02:00
|
|
|
Item_field *fld;
|
|
|
|
if ((fld= trans[i]->filed_for_view_update()))
|
|
|
|
list->push_back(fld);
|
2004-07-16 00:15:55 +02:00
|
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|