2012-08-01 17:27:34 +03:00
|
|
|
/* Copyright (c) 2010, Oracle and/or its affiliates.
|
|
|
|
|
|
|
|
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; version 2 of the License.
|
|
|
|
|
|
|
|
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,
|
2019-05-11 19:25:02 +03:00
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA */
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
|
2017-06-18 06:42:16 +03:00
|
|
|
#include "mariadb.h"
|
2012-08-01 17:27:34 +03:00
|
|
|
#include <ctype.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "sql_bootstrap.h"
|
2021-03-03 10:24:16 +01:00
|
|
|
#include <string>
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2021-03-03 10:24:16 +01:00
|
|
|
static bool is_end_of_query(const char *line, size_t len,
|
|
|
|
const std::string& delimiter)
|
|
|
|
{
|
|
|
|
if (delimiter.length() > len)
|
|
|
|
return false;
|
|
|
|
return !strcmp(line + len-delimiter.length(),delimiter.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string delimiter= ";";
|
|
|
|
extern "C" int read_bootstrap_query(char *query, int *query_length,
|
|
|
|
fgets_input_t input, fgets_fn_t fgets_fn,
|
|
|
|
int preserve_delimiter, int *error)
|
2012-08-01 17:27:34 +03:00
|
|
|
{
|
|
|
|
char line_buffer[MAX_BOOTSTRAP_LINE_SIZE];
|
|
|
|
const char *line;
|
2017-09-28 10:38:02 +00:00
|
|
|
size_t len;
|
|
|
|
size_t query_len= 0;
|
2012-12-16 21:11:24 +01:00
|
|
|
int fgets_error= 0;
|
|
|
|
*error= 0;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2020-06-08 15:05:57 +03:00
|
|
|
*query_length= 0;
|
2012-08-01 17:27:34 +03:00
|
|
|
for ( ; ; )
|
|
|
|
{
|
2012-12-16 21:11:24 +01:00
|
|
|
line= (*fgets_fn)(line_buffer, sizeof(line_buffer), input, &fgets_error);
|
|
|
|
|
|
|
|
if (error)
|
|
|
|
*error= fgets_error;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2012-12-16 21:11:24 +01:00
|
|
|
if (fgets_error != 0)
|
|
|
|
return READ_BOOTSTRAP_ERROR;
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
if (line == NULL)
|
2012-12-16 21:11:24 +01:00
|
|
|
return (query_len == 0) ? READ_BOOTSTRAP_EOF : READ_BOOTSTRAP_ERROR;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
len= strlen(line);
|
|
|
|
|
|
|
|
/*
|
|
|
|
Remove trailing whitespace characters.
|
|
|
|
This assumes:
|
|
|
|
- no multibyte encoded character can be found at the very end of a line,
|
|
|
|
- whitespace characters from the "C" locale only.
|
|
|
|
which is sufficient for the kind of queries found
|
|
|
|
in the bootstrap scripts.
|
|
|
|
*/
|
|
|
|
while (len && (isspace(line[len - 1])))
|
|
|
|
len--;
|
|
|
|
/*
|
|
|
|
Cleanly end the string, so we don't have to test len > x
|
|
|
|
all the time before reading line[x], in the code below.
|
|
|
|
*/
|
|
|
|
line_buffer[len]= '\0';
|
|
|
|
|
|
|
|
/* Skip blank lines */
|
|
|
|
if (len == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Skip # comments */
|
|
|
|
if (line[0] == '#')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Skip -- comments */
|
|
|
|
if ((line[0] == '-') && (line[1] == '-'))
|
|
|
|
continue;
|
|
|
|
|
2021-03-03 10:24:16 +01:00
|
|
|
size_t i=0;
|
|
|
|
while (line[i] == ' ')
|
|
|
|
i++;
|
|
|
|
|
|
|
|
/* Skip -- comments */
|
|
|
|
if (line[i] == '-' && line[i+1] == '-')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strncmp(line, "DELIMITER", 9) == 0)
|
|
|
|
{
|
|
|
|
const char *p= strrchr(line,' ');
|
|
|
|
if (!p || !p[1])
|
|
|
|
{
|
|
|
|
/* Invalid DELIMITER specifier */
|
|
|
|
return READ_BOOTSTRAP_ERROR;
|
|
|
|
}
|
|
|
|
delimiter.assign(p+1);
|
|
|
|
if (preserve_delimiter)
|
|
|
|
{
|
|
|
|
memcpy(query,line,len);
|
|
|
|
query[len]=0;
|
|
|
|
*query_length = (int)len;
|
|
|
|
return READ_BOOTSTRAP_SUCCESS;
|
|
|
|
}
|
2012-08-01 17:27:34 +03:00
|
|
|
continue;
|
2021-03-03 10:24:16 +01:00
|
|
|
}
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2012-12-16 21:11:24 +01:00
|
|
|
/* Append the current line to a multi line query. If the new line will make
|
|
|
|
the query too long, preserve the partial line to provide context for the
|
|
|
|
error message.
|
|
|
|
*/
|
2012-08-01 17:27:34 +03:00
|
|
|
if (query_len + len + 1 >= MAX_BOOTSTRAP_QUERY_SIZE)
|
2012-12-16 21:11:24 +01:00
|
|
|
{
|
2018-01-12 18:25:02 +00:00
|
|
|
size_t new_len= MAX_BOOTSTRAP_QUERY_SIZE - query_len - 1;
|
2012-12-16 21:11:24 +01:00
|
|
|
if ((new_len > 0) && (query_len < MAX_BOOTSTRAP_QUERY_SIZE))
|
|
|
|
{
|
|
|
|
memcpy(query + query_len, line, new_len);
|
|
|
|
query_len+= new_len;
|
|
|
|
}
|
|
|
|
query[query_len]= '\0';
|
2018-01-12 18:25:02 +00:00
|
|
|
*query_length= (int)query_len;
|
2012-12-16 21:11:24 +01:00
|
|
|
return READ_BOOTSTRAP_QUERY_SIZE;
|
|
|
|
}
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
if (query_len != 0)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Append a \n to the current line, if any,
|
|
|
|
to preserve the intended presentation.
|
|
|
|
*/
|
2012-12-16 21:11:24 +01:00
|
|
|
query[query_len++]= '\n';
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
memcpy(query + query_len, line, len);
|
|
|
|
query_len+= len;
|
|
|
|
|
2021-03-03 10:24:16 +01:00
|
|
|
if (is_end_of_query(line, len, delimiter))
|
2012-08-01 17:27:34 +03:00
|
|
|
{
|
|
|
|
/*
|
2021-03-03 10:24:16 +01:00
|
|
|
The last line is terminated by delimiter
|
2012-08-01 17:27:34 +03:00
|
|
|
Return the query found.
|
|
|
|
*/
|
2021-03-03 10:24:16 +01:00
|
|
|
if (!preserve_delimiter)
|
|
|
|
{
|
|
|
|
query_len-= delimiter.length();
|
|
|
|
query[query_len++]= ';';
|
|
|
|
}
|
|
|
|
query[query_len]= 0;
|
2018-01-12 18:25:02 +00:00
|
|
|
*query_length= (int)query_len;
|
2012-12-16 21:11:24 +01:00
|
|
|
return READ_BOOTSTRAP_SUCCESS;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|