2023-10-13 15:58:08 +02:00
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
|
|
|
|
#include "error.h"
|
|
|
|
#include "string.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
string unquote_string(string text) {
|
|
|
|
size_t original_text_size = text.size();
|
|
|
|
if (original_text_size == 0) {
|
2023-10-27 13:37:15 +02:00
|
|
|
throw ParsingException("Encountered an unexpected empty string input");
|
2023-10-13 15:58:08 +02:00
|
|
|
}
|
|
|
|
const char first_char = text.front();
|
|
|
|
const char last_char = text.back();
|
|
|
|
if (first_char == '\'' && last_char == '\'') {
|
|
|
|
text = text.substr(1, original_text_size - 2);
|
|
|
|
boost::replace_all(text, "''", "'");
|
|
|
|
boost::replace_all(text, "\\'", "'");
|
|
|
|
} else if (first_char == '"' && last_char == '"') {
|
|
|
|
text = text.substr(1, original_text_size - 2);
|
|
|
|
boost::replace_all(text, "\"\"", "\"");
|
|
|
|
boost::replace_all(text, "\\\"", "\"");
|
|
|
|
} else if (first_char == '`' && last_char == '`') {
|
|
|
|
text = text.substr(1, original_text_size - 2);
|
|
|
|
boost::replace_all(text, "``", "`");
|
|
|
|
boost::replace_all(text, "\\`", "`");
|
|
|
|
} else if (first_char == '{' && last_char == '}') {
|
|
|
|
text = text.substr(1, original_text_size - 2);
|
|
|
|
boost::replace_all(text, "{{", "{");
|
|
|
|
boost::replace_all(text, "\\{", "{");
|
|
|
|
} else {
|
2023-10-27 13:37:15 +02:00
|
|
|
throw SyntaxException("Invalid string literal, must start and end with the same quote type: " + text);
|
2023-10-13 15:58:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Copied from clickhouse_driver/util/escape.py
|
|
|
|
boost::replace_all(text, "\\a", "\a");
|
|
|
|
boost::replace_all(text, "\\b", "\b");
|
|
|
|
boost::replace_all(text, "\\f", "\f");
|
|
|
|
boost::replace_all(text, "\\n", "\n");
|
|
|
|
boost::replace_all(text, "\\r", "\r");
|
|
|
|
boost::replace_all(text, "\\t", "\t");
|
|
|
|
boost::replace_all(text, "\\v", "\v");
|
|
|
|
boost::replace_all(text, "\\0", ""); // NUL characters are ignored
|
|
|
|
boost::replace_all(text, "\\\\", "\\");
|
|
|
|
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
string unquote_string_terminal(antlr4::tree::TerminalNode* node) {
|
|
|
|
string text = node->getText();
|
|
|
|
try {
|
|
|
|
return unquote_string(text);
|
2023-10-27 13:37:15 +02:00
|
|
|
} catch (SyntaxException& e) {
|
|
|
|
throw SyntaxException(e.what(), node->getSymbol()->getStartIndex(), node->getSymbol()->getStopIndex() + 1);
|
|
|
|
} catch (ParsingException& e) {
|
|
|
|
throw ParsingException(e.what(), node->getSymbol()->getStartIndex(), node->getSymbol()->getStopIndex() + 1);
|
2023-10-13 15:58:08 +02:00
|
|
|
}
|
|
|
|
}
|