#pragma once #include #include #include #include #include #include #include #include #include namespace DB { class LineReader { public: struct Suggest { using Words = std::vector; using Callback = std::function; /// Get vector for the matched range of words if any. replxx::Replxx::completions_t getCompletions(const String & prefix, size_t prefix_length, const char * word_break_characters); void addWords(Words && new_words); void setCompletionsCallback(Callback && callback) { custom_completions_callback = callback; } private: Words words TSA_GUARDED_BY(mutex); Words words_no_case TSA_GUARDED_BY(mutex); Callback custom_completions_callback = nullptr; std::mutex mutex; }; using Patterns = std::vector; LineReader( const String & history_file_path, bool multiline, Patterns extenders, Patterns delimiters, std::istream & input_stream_ = std::cin, std::ostream & output_stream_ = std::cout, int in_fd_ = STDIN_FILENO); virtual ~LineReader() = default; /// Reads the whole line until delimiter (in multiline mode) or until the last line without extender. /// If resulting line is empty, it means the user interrupted the input. /// Non-empty line is appended to history - without duplication. /// Typical delimiter is ';' (semicolon) and typical extender is '\' (backslash). String readLine(const String & first_prompt, const String & second_prompt); /// When bracketed paste mode is set, pasted text is bracketed with control sequences so /// that the program can differentiate pasted text from typed-in text. This helps /// clickhouse-client so that without -m flag, one can still paste multiline queries, and /// possibly get better pasting performance. See https://cirw.in/blog/bracketed-paste for /// more details. /// These methods (if implemented) emit the control characters immediately, without waiting /// for the next readLine() call. virtual void enableBracketedPaste() {} virtual void disableBracketedPaste() {} bool hasInputData() const; protected: enum InputStatus { ABORT = 0, RESET_LINE, INPUT_LINE, }; const String history_file_path; String input; bool multiline; Patterns extenders; Patterns delimiters; String prev_line; virtual InputStatus readOneLine(const String & prompt); virtual void addToHistory(const String &) {} std::istream & input_stream; std::ostream & output_stream; int in_fd; }; }