/* * Copyright (c) 2021, Tim Flynn * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #pragma once #include #include #include #include #include namespace SQL { class Parser { struct Position { size_t line { 0 }; size_t column { 0 }; }; struct Error { String message; Position position; String to_string() const { return String::formatted("{} (line: {}, column: {})", message, position.line, position.column); } }; public: explicit Parser(Lexer lexer); NonnullRefPtr next_statement(); bool has_errors() const { return m_parser_state.m_errors.size(); } const Vector& errors() const { return m_parser_state.m_errors; } protected: NonnullRefPtr parse_expression(); // Protected for unit testing. private: struct ParserState { explicit ParserState(Lexer); Lexer m_lexer; Token m_token; Vector m_errors; }; NonnullRefPtr parse_create_table_statement(); NonnullRefPtr parse_drop_table_statement(); NonnullRefPtr parse_delete_statement(); NonnullRefPtr parse_primary_expression(); NonnullRefPtr parse_secondary_expression(NonnullRefPtr primary); bool match_secondary_expression() const; Optional> parse_literal_value_expression(); Optional> parse_column_name_expression(); Optional> parse_unary_operator_expression(); Optional> parse_binary_operator_expression(NonnullRefPtr lhs); Optional> parse_chained_expression(); Optional> parse_cast_expression(); Optional> parse_case_expression(); Optional> parse_collate_expression(NonnullRefPtr expression); Optional> parse_is_expression(NonnullRefPtr expression); Optional> parse_match_expression(NonnullRefPtr lhs, bool invert_expression); Optional> parse_null_expression(NonnullRefPtr expression, bool invert_expression); Optional> parse_between_expression(NonnullRefPtr expression, bool invert_expression); Optional> parse_in_expression(NonnullRefPtr expression, bool invert_expression); NonnullRefPtr parse_column_definition(); NonnullRefPtr parse_type_name(); NonnullRefPtr parse_signed_number(); NonnullRefPtr parse_common_table_expression(); NonnullRefPtr parse_qualified_table_name(); NonnullRefPtr parse_returning_clause(); Token consume(); Token consume(TokenType type); bool consume_if(TokenType type); bool match(TokenType type) const; void expected(StringView what); void syntax_error(String message); Position position() const; ParserState m_parser_state; }; }