#pragma once #include namespace DB { /// Client-server protocol. /// /// Client opens a connection and sends Hello packet. /// If client version is incompatible, the server can terminate the connection. /// Server responds with Hello packet. /// If server version is incompatible, the client can terminate the connection. /// /// The main loop follows: /// /// 1. The client sends Query packet. /// /// Starting from version 50263 immediately after sending the Query packet the client starts /// transfer of external (temporary) table (external storages) - one or several Data packets. /// End of transmission is marked by an empty block. /// At present, non-empty tables can be sent only along with SELECT query. /// /// If the query is an INSERT (and thus requires data transfer from client), then the server transmits /// Data packet containing empty block that describes the table structure. /// Then the client sends one or several Data packets - data for insertion. /// End of data is marked by the transmission of empty block. /// Then the server sends EndOfStream packet. /// /// If the query is a SELECT or a query of other type, then the server transmits packets of /// one of the following types: /// - Data - data corresponding to one block of query results. /// - Progress - query execution progress. /// - Exception - error description. /// - EndOfStream - the end of data transmission. /// /// The client should read packets until EndOfStream or Exception. /// /// The client can also send Cancel packet - a request to cancel the query. /// In this case the server can stop executing the query and return incomplete data, /// but the client must still read until EndOfStream packet. /// /// Also if there is profiling info and the client revision is recent enough, the server can /// send one of the following packets before EndOfStream: /// - Totals - a block with total values /// - ProfileInfo - serialized BlockStreamProfileInfo structure. /// /// If a query returns data, the server sends an empty header block containing /// the description of resulting columns before executing the query. /// Using this block the client can initialise the output formatter and display the prefix of resulting table /// beforehand. namespace Protocol { /// Packet types that server transmits. namespace Server { enum Enum { Hello = 0, /// Name, version, revision. Data = 1, /// A block of data (compressed or not). Exception = 2, /// The exception during query execution. Progress = 3, /// Query execution progress: rows read, bytes read. Pong = 4, /// Ping response EndOfStream = 5, /// All packets were transmitted ProfileInfo = 6, /// Packet with profiling info. Totals = 7, /// A block with totals (compressed or not). Extremes = 8, /// A block with minimums and maximums (compressed or not). TablesStatusResponse = 9, /// A response to TablesStatus request. Log = 10 /// System logs of the query execution }; /// NOTE: If the type of packet argument would be Enum, the comparison packet >= 0 && packet < 10 /// would always be true because of compiler optimisation. That would lead to out-of-bounds error /// if the packet is invalid. /// See https://www.securecoding.cert.org/confluence/display/cplusplus/INT36-CPP.+Do+not+use+out-of-range+enumeration+values inline const char * toString(UInt64 packet) { static const char * data[] = { "Hello", "Data", "Exception", "Progress", "Pong", "EndOfStream", "ProfileInfo", "Totals", "Extremes", "TablesStatusResponse", "Log" }; return packet < 11 ? data[packet] : "Unknown packet"; } } /// Packet types that client transmits. namespace Client { enum Enum { Hello = 0, /// Name, version, revision, default DB Query = 1, /// Query id, query settings, stage up to which the query must be executed, /// whether the compression must be used, /// query text (without data for INSERTs). Data = 2, /// A block of data (compressed or not). Cancel = 3, /// Cancel the query execution. Ping = 4, /// Check that connection to the server is alive. TablesStatusRequest = 5, /// Check status of tables on the server. KeepAlive = 6 /// Keep the connection alive }; inline const char * toString(UInt64 packet) { static const char * data[] = { "Hello", "Query", "Data", "Cancel", "Ping", "TablesStatusRequest" }; return packet < 6 ? data[packet] : "Unknown packet"; } } /// Whether the compression must be used. enum class Compression { Disable = 0, Enable = 1, }; /// Whether the ssl must be used. enum class Secure { Disable = 0, Enable = 1, }; } }