I have created a database engine in which I can create and modify tables, and add them to a database. For parsing the SQL queries, I have implemented the Boost.Spirit library us
Note: I opted to invent a sample grammar here for demonstration purposes, since your question doesn't show yours. Using the approach recommended here, it should not be hard to code a function to execute your queries after parsing.
I would really suggest building a parse tree.
I would recommend attribute propagation in preference to semantic actions. See e.g.
Attribute propagation rules are very flexible in Spirit. The default exposed attributes types are well documented right with each Parser's documentation
E.g. -qi::char_
would result in boost::optional
and qi::double_ | qi::int_
would result in boost::variant
.
You will probably want to accumulate the parsed elements in a AST datatype of your own invention, e.g.:
struct SelectStatement
{
std::vector columns, fromtables;
std::string whereclause; // TODO model as a vector :)
friend std::ostream& operator<<(std::ostream& os, SelectStatement const& ss)
{
return os << "SELECT [" << ss.columns.size() << " columns] from [" << ss.fromtables.size() << " tables]\nWHERE " + ss.whereclause;
}
};
You could adapt this to Spirits attribute propagation machinery by adapting the struct as a Fusion sequence:
BOOST_FUSION_ADAPT_STRUCT(SelectStatement,
(std::vector, columns)
(std::vector, fromtables)
(std::string, whereclause)
)
Now you could parse the following rule into that type:
sqlident = lexeme [ alpha >> *alnum ]; // table or column name
columns = no_case [ "select" ] >> (sqlident % ',');
tables = no_case [ "from" ] >> (sqlident % ',');
start = columns >> tables
>> no_case [ "where" ]
>> lexeme [ +(char_ - ';') ]
>> ';';
You can see this code running live here: http://liveworkspace.org/code/0b525234dbce22cbd8becd69f84065c1
Full demo code:
// #define BOOST_SPIRIT_DEBUG
#include
#include
namespace qi = boost::spirit::qi;
struct SelectStatement
{
std::vector columns, fromtables;
std::string whereclause; // TODO model as a vector :)
friend std::ostream& operator<<(std::ostream& os, SelectStatement const& ss)
{
return os << "SELECT [" << ss.columns.size() << " columns] from [" << ss.fromtables.size() << " tables]\nWHERE " + ss.whereclause;
}
};
BOOST_FUSION_ADAPT_STRUCT(SelectStatement,
(std::vector, columns)
(std::vector, fromtables)
(std::string, whereclause)
)
template
struct parser : qi::grammar
{
parser() : parser::base_type(start)
{
using namespace qi;
sqlident = lexeme [ alpha >> *alnum ]; // table or column name
columns = no_case [ "select" ] >> (sqlident % ',');
tables = no_case [ "from" ] >> (sqlident % ',');
start = columns >> tables
>> no_case [ "where" ]
>> lexeme [ +(char_ - ';') ]
>> ';';
BOOST_SPIRIT_DEBUG_NODE(start);
BOOST_SPIRIT_DEBUG_NODE(sqlident);
BOOST_SPIRIT_DEBUG_NODE(columns);
BOOST_SPIRIT_DEBUG_NODE(tables);
}
private:
qi::rule sqlident;
qi::rule(), Skipper> columns , tables;
qi::rule start;
};
template
bool doParse(const C& input, const Skipper& skipper)
{
auto f(std::begin(input)), l(std::end(input));
parser p;
SelectStatement query;
try
{
bool ok = qi::phrase_parse(f,l,p,skipper,query);
if (ok)
{
std::cout << "parse success\n";
std::cout << "query: " << query << "\n";
}
else std::cerr << "parse failed: '" << std::string(f,l) << "'\n";
if (f!=l) std::cerr << "trailing unparsed: '" << std::string(f,l) << "'\n";
return ok;
} catch(const qi::expectation_failure& e)
{
std::string frag(e.first, e.last);
std::cerr << e.what() << "'" << frag << "'\n";
}
return false;
}
int main()
{
const std::string input = "select id, name, price from books, authors where books.author_id = authors.id;";
bool ok = doParse(input, qi::space);
return ok? 0 : 255;
}
Will print output:
parse success
query: SELECT [3 columns] from [2 tables]
WHERE books.author_id = authors.id