8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 02:03:03 +01:00
firebird-mirror/src/dsql/preparse.cpp

533 lines
12 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: Dynamic SQL runtime support
2003-10-05 08:37:26 +02:00
* MODULE: preparse.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Dynamic SQL pre parser / parser on client side.
* This module will probably change to a YACC parser.
*
* The contents of this file are subject to the Interbase Public
* License Version 1.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy
* of the License at http://www.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#include "firebird.h"
2001-05-23 15:26:42 +02:00
#include <stdlib.h>
#include <string.h>
#include "../jrd/common.h"
2003-11-08 00:27:24 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../dsql/chars.h"
#include "../dsql/sqlda.h"
2001-05-23 15:26:42 +02:00
#include "../dsql/prepa_proto.h"
#include "../dsql/utld_proto.h"
#include "../jrd/gds_proto.h"
#include "../common/classes/ClumpletWriter.h"
#include "../common/StatusArg.h"
2001-05-23 15:26:42 +02:00
2003-09-28 02:36:28 +02:00
enum pp_vals {
PP_CREATE = 0,
PP_DATABASE = 1,
PP_SCHEMA = 2,
PP_PAGE_SIZE = 3,
PP_USER = 4,
PP_PASSWORD = 5,
PP_PAGESIZE = 6,
PP_LENGTH = 7,
PP_PAGES = 8,
PP_PAGE = 9,
PP_SET = 10,
PP_NAMES = 11
};
2001-05-23 15:26:42 +02:00
const size_t MAX_TOKEN_SIZE = 1024;
static void generate_error(ISC_STATUS*, const Firebird::string&, SSHORT, SSHORT);
static SSHORT get_next_token(const SCHAR**, const SCHAR*, Firebird::string&);
static SSHORT get_token(ISC_STATUS*, SSHORT, bool, const SCHAR**,
const SCHAR* const, Firebird::string&);
2001-05-23 15:26:42 +02:00
struct pp_table {
2001-05-23 15:26:42 +02:00
SCHAR symbol[10];
USHORT length;
2001-05-23 15:26:42 +02:00
SSHORT code;
};
2001-05-23 15:26:42 +02:00
static const pp_table pp_symbols[] = {
{"CREATE", 6, PP_CREATE},
{"DATABASE", 8, PP_DATABASE},
{"SCHEMA", 6, PP_SCHEMA},
{"PAGE_SIZE", 9, PP_PAGE_SIZE},
{"USER", 4, PP_USER},
{"PASSWORD", 8, PP_PASSWORD},
{"PAGESIZE", 8, PP_PAGESIZE},
{"LENGTH", 6, PP_LENGTH},
{"PAGES", 5, PP_PAGES},
{"PAGE", 4, PP_PAGE},
{"SET", 3, PP_SET},
{"NAMES", 5, PP_NAMES},
{"", 0, 0}
2001-05-23 15:26:42 +02:00
};
// define the tokens
2001-05-23 15:26:42 +02:00
2003-09-28 02:36:28 +02:00
enum token_vals {
NO_MORE_TOKENS = -1,
TOKEN_TOO_LONG = -2,
UNEXPECTED_END_OF_COMMAND = -3,
UNEXPECTED_TOKEN = -4,
STRING = 257,
NUMERIC = 258,
SYMBOL = 259
};
2001-05-23 15:26:42 +02:00
using namespace Firebird;
/**
PREPARSE_execute
@brief
@param user_status
@param db_handle
@param trans_handle
@param stmt_length
@param stmt
@param stmt_eaten
@param dialect
**/
bool PREPARSE_execute(
2003-10-05 08:37:26 +02:00
ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* trans_handle,
USHORT stmt_length,
const SCHAR* stmt,
bool* stmt_eaten,
USHORT dialect)
2001-05-23 15:26:42 +02:00
{
// no use creating separate pool for a couple of strings
ContextPoolHolder context(getDefaultMemoryPool());
2001-05-23 15:26:42 +02:00
try
2003-09-04 23:26:15 +02:00
{
if (!stmt)
{
Arg::Gds(isc_command_end_err).raise();
}
if (!stmt_length)
stmt_length = strlen(stmt);
const char* const stmt_end = stmt + stmt_length;
string token;
if (get_token(user_status, SYMBOL, false, &stmt, stmt_end, token) ||
token.length() != pp_symbols[PP_CREATE].length ||
token != pp_symbols[PP_CREATE].symbol)
{
return false;
}
2001-05-23 15:26:42 +02:00
if (get_token(user_status, SYMBOL, false, &stmt, stmt_end, token) ||
(token.length() != pp_symbols[PP_DATABASE].length &&
token.length() != pp_symbols[PP_SCHEMA].length) ||
(token != pp_symbols[PP_DATABASE].symbol &&
token != pp_symbols[PP_SCHEMA].symbol))
{
return false;
}
2001-05-23 15:26:42 +02:00
if (get_token(user_status, STRING, false, &stmt, stmt_end, token))
{
return true;
}
2001-05-23 15:26:42 +02:00
PathName file_name(token.ToPathName());
*stmt_eaten = false;
ClumpletWriter dpb(ClumpletReader::Tagged, MAX_DPB_SIZE, isc_dpb_version1);
2001-05-23 15:26:42 +02:00
dpb.insertByte(isc_dpb_overwrite, 0);
dpb.insertInt(isc_dpb_sql_dialect, dialect);
2001-05-23 15:26:42 +02:00
SLONG page_size = 0;
bool matched;
do {
const SSHORT result = get_next_token(&stmt, stmt_end, token);
if (result == NO_MORE_TOKENS) {
*stmt_eaten = true;
break;
}
else if (result < 0)
break;
2001-05-23 15:26:42 +02:00
matched = false;
for (int i = 3; pp_symbols[i].length && !matched; i++) {
if (token.length() == pp_symbols[i].length &&
token == pp_symbols[i].symbol)
{
bool get_out = false;
// CVC: What's strange, this routine doesn't check token.length()
// but it proceeds blindly, trying to exhaust the token itself.
switch (pp_symbols[i].code) {
case PP_PAGE_SIZE:
case PP_PAGESIZE:
if (get_token(user_status, '=', true, &stmt, stmt_end, token) ||
get_token(user_status, NUMERIC, false, &stmt, stmt_end, token))
{
get_out = true;
break;
}
page_size = atol(token.c_str());
dpb.insertInt(isc_dpb_page_size, page_size);
matched = true;
2001-05-23 15:26:42 +02:00
break;
case PP_USER:
if (get_token(user_status, STRING, false, &stmt, stmt_end, token))
{
get_out = true;
break;
}
2001-05-23 15:26:42 +02:00
dpb.insertString(isc_dpb_user_name, token);
matched = true;
2001-05-23 15:26:42 +02:00
break;
case PP_PASSWORD:
if (get_token(user_status, STRING, false, &stmt, stmt_end, token))
{
get_out = true;
break;
}
2001-05-23 15:26:42 +02:00
dpb.insertString(isc_dpb_password, token);
matched = true;
2001-05-23 15:26:42 +02:00
break;
case PP_SET:
if (get_token(user_status, SYMBOL, false, &stmt, stmt_end, token) ||
token.length() != pp_symbols[PP_NAMES].length ||
token != pp_symbols[PP_NAMES].symbol ||
get_token(user_status, STRING, false, &stmt, stmt_end, token))
{
get_out = true;
break;
}
dpb.insertString(isc_dpb_lc_ctype, token);
matched = true;
break;
2001-05-23 15:26:42 +02:00
case PP_LENGTH:
// Skip a token for value
2001-05-23 15:26:42 +02:00
if (get_token(user_status, '=', true, &stmt, stmt_end, token) ||
get_token(user_status, NUMERIC, false, &stmt, stmt_end, token))
{
get_out = true;
break;
}
2001-05-23 15:26:42 +02:00
matched = true;
break;
2001-05-23 15:26:42 +02:00
case PP_PAGE:
case PP_PAGES:
matched = true;
break;
}
2001-05-23 15:26:42 +02:00
if (get_out) {
return true;
}
2001-05-23 15:26:42 +02:00
}
}
} while (matched);
2001-05-23 15:26:42 +02:00
// This code is because 3.3 server does not recognize isc_dpb_overwrite.
FB_API_HANDLE temp_db_handle = 0;
if (!isc_attach_database(user_status, 0, file_name.c_str(),
&temp_db_handle, dpb.getBufferLength(),
reinterpret_cast<const ISC_SCHAR*>(dpb.getBuffer())) ||
user_status[1] != isc_io_error)
{
if (!user_status[1]) {
// Swallow status from detach.
ISC_STATUS_ARRAY temp_status;
isc_detach_database(temp_status, &temp_db_handle);
}
if (!user_status[1] || user_status[1] == isc_bad_db_format) {
user_status[0] = isc_arg_gds;
user_status[1] = isc_io_error;
user_status[2] = isc_arg_string;
user_status[3] = (ISC_STATUS) "open";
user_status[4] = isc_arg_string;
user_status[5] = (ISC_STATUS) file_name.c_str();
user_status[6] = isc_arg_gds;
user_status[7] = isc_db_or_file_exists;
user_status[8] = isc_arg_end;
UTLD_save_status_strings(user_status);
}
return true;
2001-05-23 15:26:42 +02:00
}
isc_create_database(user_status, 0, file_name.c_str(),
(db_handle), dpb.getBufferLength(),
reinterpret_cast<const ISC_SCHAR*>(dpb.getBuffer()),
0);
}
catch (const Exception& ex)
{
ex.stuff_exception(user_status);
return true;
2001-05-23 15:26:42 +02:00
}
return true;
2001-05-23 15:26:42 +02:00
}
/**
generate_error
@brief
@param user_status
@param token
@param error
@param result
**/
2008-09-10 04:21:13 +02:00
static void generate_error(ISC_STATUS* user_status,
const string& token, SSHORT error, SSHORT result)
2001-05-23 15:26:42 +02:00
{
string err_string;
2001-05-23 15:26:42 +02:00
2003-11-08 00:27:24 +01:00
user_status[0] = isc_arg_gds;
user_status[1] = isc_sqlerr;
user_status[2] = isc_arg_number;
2001-05-23 15:26:42 +02:00
user_status[3] = -104;
2003-11-08 00:27:24 +01:00
user_status[4] = isc_arg_gds;
2001-05-23 15:26:42 +02:00
switch (error) {
case UNEXPECTED_END_OF_COMMAND:
2003-11-08 00:27:24 +01:00
user_status[5] = isc_command_end_err;
user_status[6] = isc_arg_end;
2001-05-23 15:26:42 +02:00
break;
case UNEXPECTED_TOKEN:
case TOKEN_TOO_LONG:
if (result) {
err_string.assign(1, (TEXT) result);
err_string += token;
err_string += (TEXT) result;
2001-05-23 15:26:42 +02:00
}
else
err_string = token;
2003-11-08 00:27:24 +01:00
user_status[5] = isc_token_err;
user_status[6] = isc_arg_gds;
user_status[7] = isc_random;
user_status[8] = isc_arg_string;
user_status[9] = (ISC_STATUS)(err_string.c_str());
2003-11-08 00:27:24 +01:00
user_status[10] = isc_arg_end;
2001-05-23 15:26:42 +02:00
UTLD_save_status_strings(user_status);
break;
}
}
/**
get_next_token
@brief
@param stmt
@param stmt_end
@param token
**/
2008-09-10 04:21:13 +02:00
static SSHORT get_next_token(const SCHAR** stmt,
const SCHAR* stmt_end,
string& token)
2001-05-23 15:26:42 +02:00
{
2008-06-05 13:02:42 +02:00
UCHAR c, char_class = 0;
2001-05-23 15:26:42 +02:00
token.erase();
const SCHAR* s = *stmt;
2001-05-23 15:26:42 +02:00
for (;;) {
if (s >= stmt_end) {
*stmt = s;
return NO_MORE_TOKENS;
}
c = *s++;
if (c == '/' && s < stmt_end && *s == '*') {
s++;
while (s < stmt_end) {
c = *s++;
if (c == '*' && s < stmt_end && *s == '/')
break;
}
s++;
continue;
}
// CVC: Dmitry told me to leave this in peace, but if somebody wants
// to experiment ignoring single line comments, here's an idea.
if (c == '-' && s < stmt_end && *s == '-')
{
s++;
while (s < stmt_end)
{
c = *s++;
if (c == '\n')
break;
}
continue;
}
// CVC: End modification.
char_class = classes(c);
if (!(char_class & CHR_WHITE))
2001-05-23 15:26:42 +02:00
break;
}
/* At this point c contains character and class contains character class.
s is pointing to next character. */
const SCHAR* const start_of_token = s - 1;
2001-05-23 15:26:42 +02:00
/* In here we handle only 4 cases, STRING, INTEGER, arbitrary
SYMBOL and single character punctuation. */
if (char_class & CHR_QUOTE) {
for (;;) {
2001-05-23 15:26:42 +02:00
if (s >= stmt_end)
return UNEXPECTED_END_OF_COMMAND;
/* *s is quote - if next != quote we're at the end */
if ((*s == c) && ((++s == stmt_end) || (*s != c)))
break;
token += *s++;
2001-05-23 15:26:42 +02:00
}
*stmt = s;
if (token.length() > MAX_TOKEN_SIZE) {
2004-12-11 01:08:55 +01:00
// '=' used as then there is no place for null termination
token.erase(MAX_TOKEN_SIZE);
2004-12-11 01:08:55 +01:00
return TOKEN_TOO_LONG;
}
2001-05-23 15:26:42 +02:00
return STRING;
}
/* Is it an integer? */
if (char_class & CHR_DIGIT) {
for (; s < stmt_end && (classes(c = *s) & CHR_DIGIT); ++s); // empty body
2007-04-11 18:05:40 +02:00
fb_assert(s >= start_of_token);
const size_t length = (s - start_of_token);
2001-05-23 15:26:42 +02:00
*stmt = s;
2004-12-11 01:08:55 +01:00
if (length > MAX_TOKEN_SIZE) {
token.assign(start_of_token, MAX_TOKEN_SIZE);
2004-12-11 01:08:55 +01:00
return TOKEN_TOO_LONG;
}
token.assign(start_of_token, length);
2001-05-23 15:26:42 +02:00
return NUMERIC;
}
// Is is a symbol
2001-05-23 15:26:42 +02:00
if (char_class & CHR_LETTER) {
token += UPPER(c);
for (; s < stmt_end && (classes(*s) & CHR_IDENT); s++) {
token += UPPER(*s);
2001-05-23 15:26:42 +02:00
}
*stmt = s;
if (token.length() > MAX_TOKEN_SIZE) {
token.erase(MAX_TOKEN_SIZE);
2004-12-11 01:08:55 +01:00
return TOKEN_TOO_LONG;
}
2001-05-23 15:26:42 +02:00
return SYMBOL;
}
// What remains at this point for us is the single character punctuation.
2001-05-23 15:26:42 +02:00
*stmt = s;
return (c == ';' ? NO_MORE_TOKENS : c);
}
/**
get_token
@brief
@param status
@param token_type
@param optional
@param stmt
@param stmt_end
@param token
**/
static SSHORT get_token(ISC_STATUS* status,
2001-05-23 15:26:42 +02:00
SSHORT token_type,
2003-09-04 23:26:15 +02:00
bool optional,
const SCHAR** stmt,
const SCHAR* const stmt_end,
string& token)
2001-05-23 15:26:42 +02:00
{
const SCHAR* temp_stmt = *stmt;
const SSHORT result = get_next_token(&temp_stmt, stmt_end, token);
2001-05-23 15:26:42 +02:00
switch (result) {
case NO_MORE_TOKENS:
*stmt = temp_stmt;
generate_error(status, token, UNEXPECTED_END_OF_COMMAND, 0);
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
case UNEXPECTED_END_OF_COMMAND:
case TOKEN_TOO_LONG:
*stmt = temp_stmt;
// generate error here
2001-05-23 15:26:42 +02:00
generate_error(status, token, result, 0);
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
// Some token was found
2001-05-23 15:26:42 +02:00
if (result == token_type) {
*stmt = temp_stmt;
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
if (optional)
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
/* generate error here and return failure; */
*stmt = temp_stmt;
generate_error(status, token, UNEXPECTED_TOKEN,
(result == STRING) ? *(temp_stmt - 1) : 0);
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}