Skip to main content
البروتوكول الأصلي هو بروتوكول ثنائي موجَّه بالاتصال تتخاطب به عملاء وخوادم ClickHouse عبر TCP. وهو ينقل استعلامات SQL، وبيانات النتائج، وحمولات INSERT، وبيانات القياس عن بُعد الخاصة بالتنفيذ، وإشارات الأخطاء. وهو البروتوكول الذي يستند إليه عميل سطر الأوامر وC++ ومعظم برامج التشغيل الأصلية من جهات خارجية. تغطي هذه الصفحة البروتوكول نفسه: تأطير الحزم، وآلة حالات الاتصال، والتفاوض على الإصدار، وجسم كل رسالة لا تنتمي إلى Block. أما البايتات داخل حزم عائلة Data (أي Block، وأعمدته، وترميزات كل نوع) فهي موضوع منفصل، موثَّق في مواصفة Native Format.
مواصفة مرافقةتمثل هذه الصفحة أحد جزأَي هذه المجموعة، وتُنشر مع المواصفة المرافقة Native Format. وتقسم المواصفتان العمل بوضوح: تختص هذه الصفحة بطبقة الحزم والنقل، بينما تختص مواصفة Native Format بالبايتات داخل حزم عائلة Data.
هناك بعض الخصائص التي تنطبق على امتداد البروتوكول. فالبروتوكول ثنائي وموضعي: لا توجد وسوم للحقول إلا داخل BlockInfo، لذا فإن أي بايت يوضع في غير موضعه يفقد المزامنة مع كل ما يليه. وهو بروتوكول ذو حالة، ويعالج كل اتصال TCP استعلامًا واحدًا في كل مرة — ولا توجد فيه آلية تعدد الإرسال. أما الأعداد الصحيحة ثابتة العرض فتُخزَّن بترتيب little-endian.

نظرة عامة

الخاصيةالقيمة
النقلTCP، مع إمكانية تغليفه باستخدام TLS
ترتيب البايتاتالأقل أهمية أولًا للأعداد الصحيحة ثابتة العرض
الترميزثنائي وموضعي (من دون وسوم حقول باستثناء BlockInfo)
نموذج الاتصالذو حالة، واستعلام واحد في كل مرة، من دون تعدد الإرسال
الإصداريُتفاوض عليه عند المصافحة؛ وتُقيَّد الميزات الفردية بحسب الإصدار
تنسيق البياناتNative Format لجميع البيانات الجدولية
تبدأ كل رسالة برمز نوع حزمة VarUInt، يتبعه جسم يعتمد شكله على ذلك الرمز وعلى إصدار البروتوكول المتفاوض عليه. يمر الاتصال عبر ثلاث مراحل — مصافحة لمرة واحدة، ثم أي عدد من عمليات تبادل Ping أو Query، ثم الإغلاق: ينقل بروتوكول TCP الأصلي دائمًا البيانات الجدولية بتنسيق Native، بغضّ النظر عن أي عبارة FORMAT في SQL. أمّا إعادة تنسيقها إلى RowBinary وCSV وJSON وما إلى ذلك، فهي من مهام العميل، وتتم بعد فك ترميز كتل Native. (أمّا واجهة HTTP فلها مسار برمجي مختلف يلتزم فعلًا بعبارة FORMAT؛ لكن HTTP خارج نطاق هذا المحتوى هنا.)

الأمان

أمان النقل (TLS)

يقع TLS في طبقة النقل، أسفل البروتوكول. وعند تفعيله، يُشفَّر تيار TCP بالكامل، وتظل رسائل البروتوكول متطابقة تمامًا، بايتًا ببايت، سواء استُخدم TLS أم لا.

المصادقة

تحدث المصادقة أثناء المصافحة، في رسالة ClientHello. يُنقل الحقلان user وpassword كسلاسل نصية غير مشفّرة، لذا فإن التشفير على مستوى النقل (TLS) هو ما يحمي بيانات الاعتماد أثناء انتقالها. تتوفر مصادقة التحدي والاستجابة عبر SSH بدءًا من إصدار البروتوكول 54466 — راجع مصادقة التحدي والاستجابة عبر SSH.

السر بين الخوادم

لتنفيذ الاستعلامات الموزعة، تتحقق الخوادم من هوية بعضها بعضًا عبر إثبات معرفة سر مشترك — من دون إرسال السر نفسه عبر الشبكة. تتضمن كل Query قيمة auth_hash بطول 32 بايت من نوع SHA-256 في الحقل 4 من Query، وتُحتسب استنادًا إلى salt وnonce والسر المُعدّ والاستعلام، ثم يعيد الخادم المستقبِل احتسابها ويقارنها. ولا يُفعَّل ذلك إلا مع الميزة INTERSERVER_SECRET ‏(v54441). وترسل البرامج العميلة الخارجية دائمًا سلسلة فارغة هنا. راجع المصادقة بين الخوادم.

الإصدارات وبوابات الميزات

التفاوض على الإصدار

يُعلن كلٌّ من العميل والخادم عن الحد الأقصى لإصدار البروتوكول الذي يدعمه أثناء المصافحة. ويكون الإصدار المتفَق عليه هو الأصغر بين الاثنين:
negotiated_version = min(client_version, server_version)
تستخدم كل رسالة بعد ذلك الإصدار المتفق عليه لتحديد الحقول الموجودة في البيانات المنقولة.

بوابات الميزات

تُحدَّد كل ميزة بإصدار البروتوكول الذي أدخلها، وتكون مفعّلة عندما يكون الإصدار المتفاوض عليه أكبر من هذا الرقم أو مساويًا له.
عندما تكون الميزة مفعّلة، يجب أن تكون حقولها موجودة في التمثيل الثنائي المنقول. يعتمد البروتوكول ترتيبًا موضعيًا صارمًا، لذا فإن حذف حقل خاضع لبوابات الميزات يفسد تدفق البايتات لكل حقل يليه.

جدول الميزات

الميزةالإصداريؤثر علىالأثر على صيغة النقل
BLOCK_INFOallBlockيضيف البادئة BlockInfo (is_overflows, bucket_number) إلى كل Block.
CLIENT_INFO54032Queryيضيف كتلة ClientInfo إلى body الخاص بـ Query.
TIMEZONE54058ServerHelloيضيف الحقل timezone إلى ServerHello.
QUOTA_KEY_IN_CLIENT_INFO54060ClientInfoيضيف الحقل quota_key إلى ClientInfo.
DISPLAY_NAME54372ServerHelloيضيف الحقل display_name إلى ServerHello.
VERSION_PATCH54401ServerHello, ClientInfoيضيف الحقل version_patch إلى كليهما.
SERVER_LOGS54406Logيُرسل Server حزم Log عندما يكون send_logs_level مضبوطًا.
COLUMN_DEFAULTS_METADATA54410TableColumnsقد يرسل Server الحزمة TableColumns (النوع 11) مع البيانات الوصفية للقيم الافتراضية للأعمدة قبل كتلة schema الخاصة بـ INSERT/input. لا تُرسل هذه الحزمة إلا إذا كان الإصدار المتفاوض عليه ≥ 54410 و كان input_format_defaults_for_omitted_fields مفعّلًا. وأدنى من هذا الإصدار، لا تُرسل الحزمة مطلقًا؛ ويجب ألا ينتظرها clients.
WRITE_CLIENT_INFO54420Progressيضيف wrote_rows وwrote_bytes إلى Progress. (على الرغم من الاسم، فإن هذا لا يتحكم في كتلة ClientInfo — فهذا هو CLIENT_INFO (v54032).)
SETTINGS_SERIALIZED_AS_STRINGS54429Query (settings encoding)يغيّر كيفية ترميز settings list الموجودة دائمًا؛ ولا يتحكم في ما إذا كانت settings تُرسل أم لا. في v54429+ تُكتب كل setting بالشكل (name, flags, value-as-string)؛ أما الأقران الأقدم فيكتبون (name, type-specific-binary-value) بدون flags. انظر Setting.
INTERSERVER_SECRET54441Queryيضيف الحقل auth_hash بين الخوادم إلى Query — وهو SHA-256 مملّح على secret الخاص بالعنقود، وليس secret الخام. يرسل clients الخارجيون سلسلة فارغة. انظر Inter-server authentication.
OPEN_TELEMETRY54442ClientInfoيضيف trace context الخاص بـ OpenTelemetry إلى ClientInfo.
DISTRIBUTED_DEPTH54448ClientInfoيضيف الحقل distributed_depth إلى ClientInfo.
INITIAL_QUERY_START_TIME54449ClientInfoيضيف الحقل initial_time (Int64، Fixed-width).
PROFILE_EVENTS54451ProfileEventsيُرسل Server حزم ProfileEvents أثناء تنفيذ query.
PARALLEL_REPLICAS54453ClientInfoيضيف حقول تنسيق replicas المتوازية إلى ClientInfo.
CUSTOM_SERIALIZATION54454Block (Column)يضيف البايت has_custom_serialization بعد type string الخاص بكل عمود.
ADDENDUM54458Handshakeيرسل Client ملحقًا (quota_key) بعد تبادل handshake.
PARAMETERS54459Queryيضيف قائمة Parameters إلى body الخاص بـ Query.
SERVER_QUERY_TIME_IN_PROGRESS54460Progressيضيف الحقل elapsed_ns إلى Progress.
PASSWORD_COMPLEXITY_RULES54461ServerHelloيضيف قائمة بأنماط regex الخاصة بسياسة password ورسائل human-readable إلى ServerHello.
INTERSERVER_SECRET_V254462ServerHelloيضيف قيمة nonce من النوع UInt64 بطول 8 بايت إلى ServerHello. تُستخدم لتوقيع query بين الخوادم؛ أما clients الخارجيون فيفكون ترميزها ويتجاهلونها.
TOTAL_BYTES_IN_PROGRESS54463Progressيضيف الحقل total_bytes_to_read (VarUInt) إلى Progress، بين total_rows وwrote_rows.
TIMEZONE_UPDATES54464TimezoneUpdateيضيف حزمة Server باسم TimezoneUpdate (النوع 17). الـ Body: قيمة String واحدة تحمل session timezone. لا تُرسل إلا من مهيّئ table function input، مباشرة بعد كتلة schema الخاصة بالإدخال، لكي يحلل client الصفوف التي يرسلها باستخدام session_timezone الخاص بـ Server. انظر TimezoneUpdate.
SPARSE_SERIALIZATION54465Block (Column)قد يضبط Server القيمة has_custom_serialization = 1 ويُرسل عمودًا مُرمّزًا بترميز sparse. صيغة النقل: نوع بطول 1 بايت (0x01 = SPARSE)، ثم stream من الإزاحات VarUInt ينتهي بـ EOG، ثم القيم غير الافتراضية مُرمّزة بكثافة في النوع الداخلي. انظر kind_stack and sparse encoding.
SSH_AUTHENTICATION54466Auth flowيضيف authentication بأسلوب challenge-response عبر SSH. التفعيل اختياري: يرسل client قيمة user بالشكل " SSH KEY AUTHENTICATION " + <real_user> مع password فارغ لتفعيله. انظر SSH challenge-response authentication.
TABLE_READ_ONLY_CHECK54467TablesStatusResponseيضيف العلامة is_readonly إلى row الخاصة بكل table في TablesStatusResponse. لا يرى clients الخارجيون الذين لا يرسلون TablesStatusRequest أي تغيير في صيغة النقل.
SYSTEM_KEYWORDS_TABLE54468system tablesيملأ Server الجدول system.keywords لكي يتمكن clickhouse-client الرسمي من الإكمال التلقائي للكلمات المفتاحية. لا يوجد أي تغيير في صيغة النقل الخاصة بـ native protocol.
ROWS_BEFORE_AGGREGATION54469ProfileInfoيضيف applied_aggregation (Bool) وrows_before_aggregation (VarUInt) إلى ProfileInfo، بهذا الترتيب في النهاية.
CHUNKED_PROTOCOL54470Connection framingيغلّف chunk framing لكل حزمة كل packet body. يتم التفاوض عليه في Addendum. يحمل ServerHello تفضيل Server لكل اتجاه، بينما يحمل Addendum الاختيار النهائي للـ client. انظر chunked framing.
VERSIONED_PARALLEL_REPLICAS_PROTOCOL54471ServerHello, Addendumيتبادل الطرفان إصدار بروتوكول تنسيق النسخ المتماثلة المتوازية من نوع VarUInt. يقع حقل ServerHello مباشرة بعد protocol_version (قبل timezone). ويُلحَق حقل Addendum بعد سلاسل بروتوكول التجزئة. القيمة الحالية: 7 (DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION).
INTERSERVER_EXTERNALLY_GRANTED_ROLES54472Queryيضيف الحقل String external_roles إلى body الخاص بـ Query، بين مُنهِي settings وتجزئة السر بين الخوادم. ترسل clients الخارجية قائمة roles فارغة (بايت واحد 0x00، أي VarUInt 0 داخل غلاف String).
V2_DYNAMIC_AND_JSON_SERIALIZATION54473Column bodyقد يُخرج server تسلسل V2 لأنواع الأعمدة Dynamic وJSON — ما يحدد إصدار state_prefix المستخدم. راجع الأنواع المُصدّرة حسب الإصدار.
SERVER_SETTINGS54474ServerHelloيبث server إعداداته غير الافتراضية في صورة قائمة في ذيل ServerHello، بعد nonce. الصيغة: ثلاثيات (key, flags, value) تنتهي بمفتاح فارغ — وهي نفسها settings list في Query packet.
QUERY_AND_LINE_NUMBERS54475ClientInfoيضيف script_query_number (VarUInt) وscript_line_number (VarUInt) في ذيل ClientInfo. يستخدمه clickhouse-client لإسناد أخطاء البرنامج النصي متعدد العبارات؛ وترسل clients الخارجية 0, 0.
JWT_IN_INTERSERVER54476ClientInfoيضيف مؤشر وجود JWT من نوع UInt8 مع String jwt اختياري في ذيل ClientInfo. ترسل clients الخارجية (من دون JWT) البايت 0x00. (يُكتب DBMS_MIN_REVISON_WITH_JWT_IN_INTERSERVER في C++ — لاحظ الخطأ الإملائي في اسم الثابت.)
QUERY_PLAN_SERIALIZATION54477ServerHello, QueryPlan packetيُلحِق ServerHello القيمة VarUInt query_plan_serialization_version بعد server settings. ويُقدّم أيضًا ClientPacket::QueryPlan (الرمز 13) لتسليم خطط query المبنية مسبقًا بين الخوادم — ولا ترسلها clients الخارجية مطلقًا.
PARALLEL_BLOCK_MARSHALLING54478Block (Column)قد يغلّف server الأعمدة داخل ColumnBLOB (مضغوطة ضمنيًا) للمعالجة المتوازية. ويُشترط لذلك أن يكون الضغط مفعّلًا في query وأن يكون rows > 1؛ وإلا فتُستخدم صيغة column wire العادية. clients التي لا تفعّل الضغط مطلقًا على Query packets الصادرة لا ترى أي تغيير في wire.
VERSIONED_CLUSTER_FUNCTION_PROTOCOL54479ServerHelloيضيف VarUInt cluster_function_protocol_version في ذيل ServerHello. يُستخدم مع *Cluster table functions (s3Cluster، إلخ). تفك clients الخارجية ترميزه ثم تتجاهله.
OUT_OF_ORDER_BUCKETS_IN_AGGREGATION54480BlockInfoيضيف الحقل 3 (out_of_order_buckets: Vec<Int32>) إلى stream الموسوم بالحقول في BlockInfo. ويُفك ترميزه بالشكل [VarUInt count][Int32]*count. لا تُصدر clients الخارجية هذا بنفسها؛ ويقرأ مفكك الترميز أي قائمة غير فارغة يرسلها server.
COMPRESSED_LOGS_PROFILE_EVENTS_COLUMNS54481Log, ProfileEvents, TableColumnsقد يغلّف server أجسام حزم Log وProfileEvents وTableColumns داخل إطار الضغط. في هذا الإصدار تمر الأجسام الثلاثة كلها عبر مسار الإخراج الاختياري المضغوط نفسه، ولا يصبح إطار ضغط فعليًا إلا عندما تكون قيمة compression = true في query. clients التي لا تفعّل الضغط مطلقًا على Query packets الصادرة لا ترى أي تغيير في wire.
REPLICATED_SERIALIZATION54482Block (Column)قد يُخرج server أعمدة ذات kind_stack 0x04 = REPLICATED — وهي صيغة مدمجة على نمط القاموس للقيم المتكررة — راجع kind_stack والترميز المتناثر. قبل هذا الإصدار كان الكاتب يوسّع هذه الأعمدة قبل الإرسال. ويُفك الترميز عبر lookup على الفهرس (elements[indexes[i]] لكل row)؛ مع دعم الأنواع الورقية بالإضافة إلى القيم الداخلية لـ Nullable/Array/Tuple/Map/Nested/LowCardinality.
NULLABLE_SPARSE_SERIALIZATION54483Block (Column)يدمج التسلسل المتناثر مع Nullable(T). قبل هذا الإصدار كان الكاتب يوسّع sparse لأعمدة Nullable قبل الإرسال؛ أما في v54483+ فتصير بيانات wire sparse-over-Nullable. راجع kind_stack والترميز المتناثر.
PROGRESS_IN_ASYNC_INSERT54484Progress (INSERT)في INSERT غير متزامن (async_insert = 1)، بعد تفريغ الإدراج يرسل server حزمة Progress إضافية، ثم ProfileEvents الخاصة بالإدراج، قبل EndOfStream. ويعتمد ذلك على أن يكون الإصدار المتفاوض عليه ≥ 54484؛ وأدنى من ذلك لا يرسل server هذا Progress اللاحق. تظل صيغة wire الخاصة بـ Progress بلا تغيير — والجديد هو الإرسال فقط. عمليًا، تحمل هذه الزيادة الزمن المنقضي؛ وتُبلَّغ عدادات الصفوف المكتوبة عبر ProfileEvents المصاحبة. ولا يحتاج client الذي يستنزف بالفعل Progress المتداخلة إلى أي تغيير في الصيغة، بل فقط إلى تقبّل حزمة إضافية.
CLIENT_AGENT_IN_CLIENT_INFO54485ClientInfoيضيف client_agent من النوع String في نهاية ClientInfo. يكتشف client القياسي تلقائيًا معرّف agent من بيئته (مثل claude-code أو cursor أو gemini-cli أو قيمة المتغير AGENT)؛ أما client الخارجي الذي لا يكتشف شيئًا فيرسل سلسلة فارغة. ويصبح هذا مطلوبًا عندما يكون الإصدار المتفاوض عليه ≥ 54485 — إذ يؤدي حذفه إلى فقدان تزامن بقية Query packet.

غلاف الحزمة

تشترك كل رسالة متبادلة عبر الشبكة في البنية الخارجية نفسها، في كلا الاتجاهين:
[VarUInt: packet_type_code]    always encoded as VarUInt
[message body]                 format depends on packet_type_code
توجد جداول أنواع الحزم الكاملة في مرجع نوع الحزمة. نوع الحزمة هو VarUInt، وليس بايتًا ثابت العرض. بالنسبة إلى القيم الأقل من 128، ينتج VarUInt البايت المفرد نفسه، لكن يجب على التطبيقات استخدام ترميز VarUInt حتى تظل متوافقة إذا وصلت أنواع الحزم المستقبلية إلى 128 أو أكثر. يوثّق مرجع الرسائل جسم كل حزمة فقط — أي البايتات التي تأتي بعد رمز نوع الحزمة. يبدأ ترقيم الحقول من 1 مع أول حقل في الجسم.

التأطير المُجزّأ (v54470+)

عند التفاوض على الميزة CHUNKED_PROTOCOL (راجع المصافحة)، تُغلَّف كل حزمة على مستوى النقل بتأطير مُجزّأ. ويكون هذا التغليف منفصلًا لكل اتجاه: إذ يجري التفاوض على client→server و server→client كلٌّ على حدة، وقد ينتهي كل منهما إلى وضع مختلف (مُجزّأ أو غير مؤطَّر). البنية على مستوى النقل لكل حزمة:
<chunk>...   one or more chunks; their payloads concatenated form the whole packet
[u32 LE = 0] zero-size terminator marking end of packet
تنسيق wire لكل chunk:
[u32 LE: chunk_size]   chunk_size in [1, UINT32_MAX]
[chunk_size bytes]     packet bytes (see note below)
نوع الحزمة VarUInt يكون داخل التدفق المُجزّأ: فهو البايت الأول من حمولة الحزمة (أي البايت الأول من أول جزء)، وليس بايتًا منفصلًا يُرسَل مسبقًا قبل التأطير. حمولة الجزء لكل حزمة هي [VarUInt packet_type_code][message body] كاملةً كما ترد في غلاف الحزمة. وأي عميل يترك نوع الحزمة خارج التدفق المُجزّأ يجعل الطرف الآخر يقرأ بايت النوع هذا على أنه البايت الأول من حجم الجزء u32، ما يؤدي إلى فقدان تزامن الاتصال. قد تُقسَّم الحزمة الواحدة على عدة أجزاء إذا امتلأ المخزن المؤقت الخاص بالكاتب أثناء كتابة الحزمة؛ ويمكن أن يقع هذا الانقسام في أي موضع، بما في ذلك داخل VarUInt الخاص بنوع الحزمة. يقوم القارئ بضم حمولات الأجزاء، ويتعامل مع الصفر اللاحق المكوَّن من 4 بايتات على أنه حد شفاف للحزمة — فيستهلكه، لكنه لا يمرّره إلى أي مكوّن يقرأ أجسام الحزم. الحزم التي لا تحتوي على جسم تُغلَّف أيضًا: فالحزمة أحادية البايت مثل Ping أو Pong تصبح [u32 size = 1][0x04][u32 0] بمجرد الاتفاق على التجزئة. وأي وصف من نوع “بايت واحد على مستوى النقل” في موضع آخر من هذه الصفحة يشير إلى الصيغة السابقة للتجزئة. التفاوض. يحمل كلٌّ من ServerHello وAddendum حقلي String، واحدًا لكل اتجاه، بقيم مأخوذة من {"chunked", "notchunked", "chunked_optional", "notchunked_optional"}:
  • chunked / notchunked وضعان صارمان: ذلك الجانب يشترط هذا الوضع تحديدًا.
  • صيغ _optional مرنة: فهي تقبل أيًّا من الوضعين الذي يختاره الطرف الآخر.
تُحتسب القيمة المتفق عليها لكل اتجاه على أساس ثنائي:
تفضيل الخادمتفضيل العميلالمتفق عليه
*_optionalأي شيءاتبع CLIENT (وفق قيمة starts_with("chunked") لديه)
أي شيء*_optionalاتبع SERVER
chunked strictchunked strictchunked
notchunked strictnotchunked strictnotchunked
strict mismatchstrict mismatchخطأ في البروتوكول — يجب قطع الاتصال
على جانب العميل، يُفاوَض تفضيل SEND الخاص بالعميل مع تفضيل RECV الخاص بالخادم، والعكس صحيح. التوقيت. تنتقل سلاسل التفاوض عبر قناة غير مؤطَّرة: ClientHello → ServerHello (تفضيلات الخادم) → Addendum (القيم التي تفاوض عليها العميل). ويُطبَّق التحول إلى التأطير على كل بايت يُرسَل بعد تفريغ Addendum. أما Addendum نفسه وClientHello وServerHello فتبقى دائمًا غير مؤطَّرة.

دورة حياة الاتصال

في أي وقت، يكون الاتصال في واحدة فقط من أربع حالات: HANDSHAKE أو READY أو READING_RESPONSE أو يكون قد انتهى. وبما أن البروتوكول لا يدعم تعدد الإرسال، فإن العميل الذي يرسل طلبًا جديدًا قبل قراءة الاستجابة السابقة بالكامل سيتسبب في تداخل البايتات أثناء النقل وإفساد التدفق.

الحالات

يسير المسار المثالي مباشرةً إلى الأسفل — HANDSHAKE → READY → READING_RESPONSE → READY — مع الحلقة الذاتية لـ Ping/Pong، فيما تصبّ جميع حواف الفشل في المصب الوحيد Terminated.
StateDescription
HANDSHAKEالحالة الأولية بعد فتح اتصال TCP. لا تكون صالحة في هذه المرحلة إلا رسائل المصافحة. تنتقل إلى READY عند النجاح، أو تُنهى عند الفشل.
READYخامل. يمكن للعميل إرسال Ping أو استعلام أو إغلاق الاتصال. وقد يبقى الاتصال في READY إلى أجل غير مسمى (وفقًا لـ idle_connection_timeout، انظر حدود الاتصال).
READING_RESPONSEيُنتقل إلى هذه الحالة عندما يرسل العميل استعلامًا. ويجب على العميل استنفاد تدفق استجابة الخادم بالكامل قبل العودة إلى READY. وحزمة العميل→الخادم الوحيدة المسموح بها هنا هي Cancel (غير موضحة في هذه الصفحة).
Terminatedلم تعد قابلة للاستخدام. يجب على العميل فتح اتصال TCP جديد وبدء المصافحة من جديد.

مرحلة المصافحة

تتم المصادقة والتفاوض على إصدار البروتوكول. يحدث ذلك مرة واحدة فقط لكل اتصال، قبل أي شيء آخر. فُتح اتصال TCP للتو، ولم تُتبادل أي رسائل بعد. ويكون التسلسل كالتالي:
  1. يرسل العميل ClientHello مع أعلى إصدار بروتوكول يدعمه.
  2. يقرأ العميل الاستجابة ويعالجها وفقًا لنوع الحزمة:
    نوع الحزمةالإجراء
    Hello (0)فك ترميز ServerHello. احسب negotiated_version = min(client_ver, server_ver). ثم انتقل إلى الخطوة 3.
    Exception (2)فك ترميز Exception. أعده كخطأ وأنهِ الاتصال.
    anything elseانتهاك للبروتوكول. أنهِ الاتصال.
  3. إذا كانت قيمة negotiated_version ≥ 54458 (ميزة ADDENDUM)، يرسل العميل Addendum. يستند هذا القرار إلى الإصدار المتفق عليه، لا إلى الإصدار الذي أعلنه العميل.
عند النجاح، ينتقل الاتصال إلى READY؛ وعند حدوث أي خطأ، ينتهي.

مرحلة Ping

فحص حيوية على مستوى التطبيق، مستقل عن TCP keepalive. تؤكد دورة Ping/Pong الناجحة ذهابًا وإيابًا أن اتصال TCP حيّ في كلا الاتجاهين وأن الخادم يستجيب. تكون Ping عديمة الحالة وغير مرتبطة بأي استعلام، لذا فإن عمليات Ping المتتالية مستقلة بعضها عن بعض. بدءًا من جاهز، يكون التدفق كما يلي:
  1. يرسل العميل Ping.
  2. يقرأ العميل الاستجابة:
    نوع الحزمةالإجراء
    Pong (4)تم تأكيد أن الاتصال ما يزال حيًا. عُد إلى جاهز.
    Exception (2)فك ترميز Exception وأرجِعه كخطأ.
    أي شيء آخرمخالفة للبروتوكول.

مرحلة الاستعلام

يرسل العميل عبارة SQL، ثم يعيد الخادم بث كتل النتائج وبيانات القياس عن بُعد الخاصة بالتنفيذ. وتكون الاستجابة تسلسلاً من الحزم ينتهي بحزمة واحدة فقط، إما EndOfStream أو Exception. بدءًا من جاهز، يكون التدفق كما يلي: عند حدوث خطأ في أي مرحلة، يرسل الخادم Exception بدلًا من EndOfStream، مما يُنهي الاستعلام.
  1. يرسل العميل Query مع query_id فريد (عادةً ما يكون UUID).
  2. يرسل العميل أي جداول خارجية، ثم وسم Data فارغًا. تحتوي حزمة Data الفارغة على table_name = "" و num_columns = 0 و num_rows = 0. ولا يبدأ الخادم تنفيذ الاستعلام حتى يتلقى هذا الوسم.
  3. ينتقل العميل إلى READING_RESPONSE ويُفرغ مخزن الكتابة المؤقت لديه.
  4. يقرأ العميل حزم الاستجابة في حلقة، ويوجّه المعالجة حسب النوع:
    Packet typeالإجراء
    Data (1)فك ترميز block. تمثل أول حزمة Data ترويسة schema، أما الحزم اللاحقة فهي blocks نتائج (تُجمَّع)، ويمثل block الفارغ وسمًا فاصلًا. لا يعني num_rows == 0 عدم انتهاء الاستعلام.
    Progress (3)مقاييس التنفيذ. كل حزمة تمثل زيادة منذ الحزمة السابقة — وتُجمَّع محليًا.
    EndOfStream (5)اكتمل الاستعلام. اخرج من الحلقة وارجع إلى جاهز.
    ProfileInfo (6)بيانات profiling بعد التنفيذ.
    Totals (7)block إجماليات التجميع (بنفس wire format الخاص بـ Data).
    Extremes (8)block القيم الدنيا/العظمى (بنفس wire format الخاص بـ Data).
    Log (10)سطر من server log.
    TableColumns (11)metadata القيم الافتراضية للأعمدة.
    ProfileEvents (14)عدادات الأداء.
    Exception (2)فك الترميز وإرجاعه كخطأ. اخرج من الحلقة وارجع إلى جاهز.
    anything elseغير متوقع أثناء Query phase. أنهِ connection.
عند EndOfStream أو Exception تمت معالجته، تعود connection إلى جاهز. أما مخالفة protocol أو خطأ I/O فيُنهيانها.
غالبًا ما تُربك حالة num_rows == 0 عمليات التنفيذ الجديدة. فالـ block ذو الصفوف الصفرية هو وسم فاصل أو ترويسة schema، وليس إشارة إلى نهاية التدفق. ولا يُنهي الاستجابة إلا EndOfStream أو Exception.

مرحلة INSERT

مرحلة INSERT هي مرحلة الاستعلام مع تبادلين إضافيين. يرسل العميل عبارة INSERT؛ ويرد الخادم بـ كتلة مخطط تصف الجدول الهدف؛ ثم يرسل العميل حزم Data التي تحتوي على الصفوف، ثم وسم Data الفارغ؛ ويُنهي الخادم العملية بـ EndOfStream أو Exception. بدءًا من الحالة جاهز، تكون SQL عبارة INSERT بالصيغة INSERT INTO <table> [(<cols>)] VALUES — من دون قيمة حرفية مضمنة من الشكل VALUES (...)، لأن بيانات الصفوف تتدفق عبر حزم Data. التدفق:
  1. يرسل العميل Query مع ضبط body على عبارة INSERT في SQL.
  2. يرسل العميل أي external tables إن وُجدت (وهذا نادر مع INSERT). وعلى خلاف مرحلة الاستعلام، فهو لا يرسل هنا وسم Data فارغًا. تُرسَل حزمة INSERT Query مع وجود بيانات قيد الانتظار، لذلك يُؤجَّل block الفارغ الذي يدل على نهاية البيانات إلى الخطوة 5؛ لأن إرساله قبل schema block سيجعل الخادم يقرأه على أنه نهاية تدفق الصفوف، فيُنهي INSERT بلا أي rows، ثم يُحلّل أول حزمة صفوف فعلية على أنها حزمة شاردة على المستوى الأعلى.
  3. يستنزف العميل حزم metadata ‏(TableColumns وProgress وProfileInfo وLog وProfileEvents) إلى أن يقرأ حزمة schema Data — وهي Block تضم 0 rows ولكن ببنية columns كاملة (الأسماء والأنواع). ويُعد schema block بمثابة العقد: يجب أن تطابق rows التي يرسلها العميل بعد ذلك بنية هذه columns.
  4. يرسل العميل data block واحدة أو أكثر. ولكل block يكتب VarUInt(ClientPacket::Data = 2)، ثم String("") لاسم external-table الفارغ، ثم الـ Block. يجب أن تتوافق column types مع columns الخاصة بـ schema block بحسب الموضع.
  5. يرسل العميل محدِّد نهاية الإدخال: حزمة Data تحتوي على Block فارغ (0 columns, 0 rows).
  6. يستنزف العميل تدفق الاستجابة إلى أن يصل إلى EndOfStream (نجاح) أو Exception (فشل).
INSERT غير المتزامن (v54484+). عندما يحمل الاستعلام async_insert = 1، يضع الخادم rows في queue ثم ينفّذ flush لها كجزء من batch. وعند الإصدار المتفاوض عليه ≥ 54484 (PROGRESS_IN_ASYNC_INSERT)، ما إن يكتمل flush حتى يُصدر الخادم حزمة Progress إضافية، تتبعها مباشرة ProfileEvents الخاصة بعملية insert، ثم EndOfStream. أما في الإصدارات الأقدم من 54484، فيتجاوز الخادم حزمة Progress اللاحقة هذه. وهذه الحزمة هي Progress عادية؛ ولأن الخادم يعيد ضبط query pipeline قبل احتساب counts الخاصة بالكتابة، فإن الزيادة لا تتضمن عمليًا سوى الزمن المنقضي، بينما تصل إلى العميل إحصاءات rows والبايتات المكتوبة عبر ProfileEvents المصاحبة. وأي عميل يستنزف بالفعل حزم Progress المتداخلة في الخطوة 6 لا يحتاج إلا إلى قبول حزمة إضافية واحدة. يعود connection إلى جاهز عند EndOfStream أو عند Exception تمت معالجته. أما مخالفات protocol وأخطاء I/O فتُنهيه.

مرجع الرسائل

تُدرج الحقول وفق ترتيبها على wire. ويستخدم العمود Type ما يلي:
  • VarUInt — عدد صحيح غير موقّع بطول متغيّر (انظر VarUInt).
  • String — بايتات مسبوقة بـ VarUInt (انظر String).
  • UInt8 وInt32 وما إلى ذلك — أعداد صحيحة ثابتة العرض بترتيب little-endian.
  • Bool — بايت واحد، 0x00 أو 0x01.
يوضح العمود Role الجهة التي تستخدم كل حقل:
  • client — يضبطه العملاء الخارجيون.
  • inter-server — يكون ذا معنى فقط في الاتصال بين الخوادم؛ ويكتب العملاء الخارجيون قيمة افتراضية.
  • universal — يستخدمه الطرفان.
توثّق هذه الجداول جزء body من كل حزمة فقط، بعد رمز نوع الحزمة.

ClientHello (نوع الحزمة 0)

العميل → الخادم. أول رسالة بعد فتح اتصال TCP.
#الحقلالنوعالدورالوصف
1client_nameStringمشتركمعرّف العميل (مثل "clickhouse-client")
2version_majorVarUIntمشتركالإصدار الرئيسي للعميل
3version_minorVarUIntمشتركالإصدار الثانوي للعميل
4protocol_versionVarUIntمشتركأعلى إصدار بروتوكول يدعمه العميل
5databaseStringمشتركاسم قاعدة البيانات الافتراضية
6userStringمشتركاسم المستخدم للمصادقة
7passwordStringمشترككلمة المرور (بنص صريح)

ServerHello (نوع الحزمة 0)

Server → Client. الرد على ClientHello عند نجاح المصادقة.
#FieldTypeRoleConditionDescription
1server_nameStringمشتركدائمًامعرّف الخادم
2version_majorVarUIntمشتركدائمًاالإصدار الرئيسي للخادم
3version_minorVarUIntمشتركدائمًاالإصدار الثانوي للخادم
4protocol_versionVarUIntمشتركدائمًاإصدار البروتوكول الخاص بالخادم
4aparallel_replicas_protocol_versionVarUIntمشتركVERSIONED_PARALLEL_REPLICAS_PROTOCOL (v54471)إصدار بروتوكول coordination للنسخ المتماثلة المتوازية في الخادم. موضعه على wire: مباشرة بعد protocol_version، وقبل timezone. القيمة الحالية: 7.
5timezoneStringمشتركTIMEZONE (v54058)timezone الخاص بالخادم (على سبيل المثال، "UTC")
6display_nameStringمشتركDISPLAY_NAME (v54372)اسم الخادم بصيغة human-readable
7version_patchVarUIntمشتركVERSION_PATCH (v54401)إصدار التصحيح للخادم
8proto_send_chunked_srvStringمشتركCHUNKED_PROTOCOL (v54470)تفضيل الخادم لـ chunking الصادر. إحدى القيم التالية: "chunked" أو "notchunked" أو "chunked_optional" أو "notchunked_optional". راجع التأطير المُجزّأ. يوضع على wire قبل password_complexity_rules رغم أن بوابة الإصدار الخاصة به أعلى.
9proto_recv_chunked_srvStringمشتركCHUNKED_PROTOCOL (v54470)تفضيل الخادم لـ chunking الوارد. نفس مجموعة القيم كما في الحقل 8.
10password_complexity_rulesRule[]مشتركPASSWORD_COMPLEXITY_RULES (v54461)سياسة password الخاصة بالخادم. VarUInt count متبوعًا بـ count × Rule. انظر أدناه.
11nonceUInt64بين الخوادمINTERSERVER_SECRET_V2 (v54462)قيمة nonce عشوائية بطول 8 بايت بتنسيق LE. تستخدمها آلية الخادم لتوقيع query بين الخوادم. يجب على clients الخارجيين فك ترميزها (للحفاظ على محاذاة stream) ويُستحسن تجاهل القيمة.
12server_settingsSetting[]مشتركSERVER_SETTINGS (v54474)بث settings غير الافتراضية من الخادم. التنسيق: صفر أو أكثر من الثلاثيات (String key, VarUInt flags, String value)، وتنتهي بـ key فارغ. مماثلة لـ قائمة الإعدادات في حزمة Query.
13query_plan_serialization_versionVarUIntمشتركQUERY_PLAN_SERIALIZATION (v54477)serialization version لخطة query التي يدعمها الخادم. يفك clients الخارجيون ترميزه ويتجاهلونه.
14cluster_function_protocol_versionVarUIntمشتركVERSIONED_CLUSTER_FUNCTION_PROTOCOL (v54479)إصدار البروتوكول لدالة table‏ *Cluster في الخادم. يفك clients الخارجيون ترميزه ويتجاهلونه.
Rule — عنصر من password_complexity_rules:
#FieldTypeDescription
1patternStringpattern للتعبير النمطي الذي يجب أن تطابقه password المتوافقة.
2messageStringشرح بصيغة human-readable يُعرض عندما تفشل password في استيفاء هذه القاعدة.
تعكس القائمة configuration الخاصة بمشغّل الخادم لسياسة password، وهي استرشادية بحتة — إذ لا يفرض الخادم هذه القواعد أثناء handshake. يمكن لـ client الذي يوفّر وظيفة تغيير/تعيين password استخدام هذه القواعد للإشارة إلى الأخطاء قبل إرسال password غير متوافقة إلى الخادم.
للحد من استخدام الموارد في مواجهة خادم عدائي أو سيئ الإعداد، اجعل الحد الأقصى للقيمة المفكوك ترميزها count هو 256 إدخالًا، ولكل من pattern وmessage من نوع String هو 4096 بايت. وتُعد قيمة count البالغة 0 (من دون أزواج تالية) الحالة الشائعة للخوادم التي لم تُضبط لها سياسة password.

ملحق (من دون نوع حزمة)

العميل → الخادم، ويكون مفعّلًا بواسطة ADDENDUM ‏(v54458). يُرسل مباشرةً بعد اكتمال تبادل المصافحة. وليس نوع حزمة مستقلًا — إذ تُرسل الحقول على الـ wire كما هي، من دون بايت بادئة لنوع الحزمة.
#الحقلالنوعالدورالشرطالوصف
1quota_keyStringمشتركدائمًامفتاح QUOTA للموارد لآليات QUOTA المقيّدة بمفتاح على جانب الخادم. يرسل العملاء الذين لا يستخدمون QUOTA مقيّدة بمفتاح سلسلة فارغة.
2proto_send_chunkedStringمشتركCHUNKED_PROTOCOL (v54470)إعداد chunking الصادر المتفاوض عليه من العميل: "chunked" أو "notchunked". يُحتسب بالاستناد إلى proto_recv_chunked_srv من ServerHello.
3proto_recv_chunkedStringمشتركCHUNKED_PROTOCOL (v54470)إعداد chunking الوارد المتفاوض عليه من العميل. يُحتسب بالاستناد إلى proto_send_chunked_srv.
4parallel_replicas_protocol_versionVarUIntمشتركVERSIONED_PARALLEL_REPLICAS_PROTOCOL (v54471)إصدار protocol الخاص بـ coordination لـ parallel-replicas الذي يدعمه العميل. ينبغي للعملاء الخارجيين الذين لا يشاركون في distributed queries أن يرسلوا مع ذلك إصدارًا صالحًا (الحالي 7) لكي ينجح فحص التوافق في الخادم.
يُطبَّق التحول إلى التأطير المقطّع بعد flush هذا الملحق — أما الملحق نفسه فغير مؤطَّر.

Ping (نوع الحزمة 4)

العميل → الخادم. بلا متن — تتكوّن الحزمة من بايت واحد 0x04 قبل التأطير بالمقاطع؛ وعند التفاوض على استخدام التقسيم إلى مقاطع، يصبح هذا البايت حمولةً من بايت واحد لمقطع واحد (انظر التأطير المُجزّأ).

Pong (نوع الحزمة 4)

الخادم → العميل. من دون محتوى — تتكوّن الحزمة من بايت واحد 0x04 قبل التأطير المُجزّأ؛ وعند الاتفاق على استخدام التجزئة، يصبح هذا البايت حمولةً من بايت واحد لجزء واحد (انظر التأطير المُجزّأ).

Exception (نوع الحزمة 2)

الخادم → العميل. تُرسل عندما يصادف الخادم خطأً أثناء أي مرحلة.
#الحقلالنوعالدورالوصف
1codeInt32مشتركرمز الخطأ
2nameStringمشتركفئة Exception (مثل "DB::Exception")
3messageStringمشتركرسالة خطأ مقروءة للبشر
4stack_traceStringمشتركتتبّع المكدس على جهة الخادم
5has_nested (obsolete)Boolمشتركبايت توافق قديم. يكتبه الخادم دائمًا بالقيمة false

Query (نوع الحزمة 1)

العميل → الخادم.
#الحقلالنوعالدورالشرطالوصف
1query_idStringمشتركدائمًامعرّف استعلام فريد (UUID)
2client_infoClientInfoمشتركCLIENT_INFO (v54032)انظر ClientInfo
3settingsSetting[]مشتركدائمًاانظر Setting. موجود دائمًا (وينتهي بمفتاح فارغ)؛ وما يخضع لتقييد الإصدار هو الترميز الخاص بكل إعداد فقط — راجع ملاحظة الترميز في Setting. يجب على العميل ألّا يحذف هذا الحقل للإصدارات المتفاوض عليها الأقل من 54429.
3aexternal_rolesStringمشتركINTERSERVER_EXTERNALLY_GRANTED_ROLES (v54472)قائمة مُسلسلة بأسماء الأدوار الممنوحة خارجيًا. القائمة الفارغة = البايت 0x00 (VarUInt 0) داخل غلاف String ([VarUInt 1][0x00] في التمثيل المنقول). يرسل العملاء الخارجيون دائمًا قائمة فارغة.
4auth_hashStringبين الخوادمINTERSERVER_SECRET (v54441)تجزئة المصادقة بين الخوادم — وليست قيمة secret الخام الخاصة بالـ cluster. انظر المصادقة بين الخوادم أدناه. يرسل العملاء الخارجيون (وأي InitialQuery) سلسلة فارغة.
5stageVarUIntمشتركدائمًامرحلة معالجة الاستعلام. 0 = FetchColumns، 1 = WithMergeableState، 2 = Complete، 3 = WithMergeableStateAfterAggregation، 4 = WithMergeableStateAfterAggregationAndLimit، 7 = QueryPlan. تظهر القيمتان 3/4 في distributed queries؛ وترافق القيمة 7 خطة استعلام مُسلسلة. يرسل العملاء الخارجيون عادةً 2.
6compressionVarUIntمشتركدائمًا0 = معطّل، 1 = مفعّل
7query_bodyStringمشتركدائمًانص SQL
8parametersParameter[]clientPARAMETERS (v54459)انظر Parameter. وينتهي بمفتاح فارغ.

ClientInfo (مضمّن في Query)

Client → Server، وهو مضمّن في جسم Query (الحقل 2). ومقيّد بـ CLIENT_INFO ‏(v54032). (بعض الحقول داخل ClientInfo مقيّدة بإصدارات أحدث، كما هو مذكور لكل حقل أدناه.)
#الحقلالنوعالدورالشرطالوصف
1query_kindUInt8مشتركدائمًا0 = NoQuery، 1 = InitialQuery، 2 = SecondaryQuery. يرسل العملاء الخارجيون 1.
2initial_userStringمشتركدائمًاالمستخدم الذي بدأ الاستعلام
3initial_query_idStringمشتركدائمًامعرّف الاستعلام الأصلي
4initial_addressStringمشتركدائمًاعنوان socket للعميل المصدر بصيغة host:port
5initial_timeInt64clientINITIAL_QUERY_START_TIME (v54449)وقت بدء الاستعلام (بالميكروثانية). Fixed-width بحجم 8 بايت، وليس VarUInt
6query_interfaceUInt8مشتركدائمًا1 = TCP، 2 = HTTP
7os_userStringclientإذا كانت interface = TCPاسم مستخدم نظام التشغيل
8client_hostnameStringclientإذا كانت interface = TCPhostname لجهاز العميل
9client_nameStringclientإذا كانت interface = TCPاسم تطبيق العميل
10version_majorVarUIntمشتركإذا كانت interface = TCPClient major version
11version_minorVarUIntمشتركإذا كانت interface = TCPClient minor version
12protocol_versionVarUIntمشتركإذا كانت interface = TCPإصدار protocol الخاص بـ TCP للعميل المصدر نفسه (DBMS_TCP_PROTOCOL_VERSIONوليس الإصدار الذي جرى التفاوض عليه. يحدد revision الخاص بالنظير فقط الحقول الموجودة؛ أما هذه القيمة فهي الإصدار المضمَّن وقت الترجمة لدى initiator، لذلك عند استخدام عميل أحدث للتواصل مع server أقدم قد تكون أعلى من negotiated/server revision.
13quota_keyStringمشتركQUOTA_KEY_IN_CLIENT_INFO (v54060)quota key للموارد من أجل server-side keyed quotas. يرسل العملاء الذين لا يستخدمون حصة مقيّدة بمفتاح سلسلة فارغة.
14distributed_depthVarUIntبين الخوادمDISTRIBUTED_DEPTH (v54448)عمق تداخل distributed query. يرسل العملاء الخارجيون 0.
15version_patchVarUIntمشتركVERSION_PATCH (v54401), TCP onlyإصدار التصحيح للعميل
16open_telemetry(below)clientOPEN_TELEMETRY (v54442)سياق التتبّع. يرسل العملاء الذين لا يستخدمون tracing القيمة 0.
17collaborate_with_initiatorVarUIntبين الخوادمPARALLEL_REPLICAS (v54453)Bool بصيغة VarUInt. يرسل العملاء الخارجيون 0.
18count_participating_replicasVarUIntبين الخوادمPARALLEL_REPLICAS (v54453)يرسل العملاء الخارجيون 0.
19number_of_current_replicaVarUIntبين الخوادمPARALLEL_REPLICAS (v54453)يرسل العملاء الخارجيون 0.
20script_query_numberVarUIntclientQUERY_AND_LINE_NUMBERS (v54475)موضع statement مفهرس بدءًا من 1 داخل برنامج نصي متعدد العبارات. يرسل العملاء الخارجيون 0.
21script_line_numberVarUIntclientQUERY_AND_LINE_NUMBERS (v54475)رقم السطر مفهرسًا بدءًا من 1 داخل برنامج نصي المصدر. يرسل العملاء الخارجيون 0.
22jwt_presentUInt8بين الخوادمJWT_IN_INTERSERVER (v54476)0 = لا يوجد JWT؛ 1 = يتبع ذلك JWT. يرسل العملاء الخارجيون الذين لا يستخدمون مصادقة JWT القيمة 0.
23jwtStringبين الخوادمJWT_IN_INTERSERVER (v54476), if jwt_present=1Bearer token لـ JWT، ولا يكون موجودًا إلا عندما تكون قيمة الحقل 22 هي 1.
24client_agentStringclientCLIENT_AGENT_IN_CLIENT_INFO (v54485)حقل ختامي. معرّف أداة/agent العميل، ويُكتشف تلقائيًا من البيئة (مثل claude-code أو cursor أو gemini-cli أو متغير البيئة AGENT). يرسل العملاء الخارجيون الذين لم يُكتشف لهم agent سلسلة فارغة. يوجد في مسار Query العادي بمجرد أن يكون الإصدار المتفاوض عليه ≥ 54485 (ويُرسل على جميع interfaces، وليس على TCP فقط).
التخطيط المعتمد على الواجهة (الحقول 7–12)الحقول 7–12 أعلاه تمثّل فرع TCP. عندما لا تكون قيمة query_interface (الحقل 6) هي TCP، تُستبدل هذه الحقول بتخطيط wire مختلف — وليست مجرد حقول اختيارية محذوفة، لذا يجب على وحدة فك الترميز أن تتفرع بناءً على الحقل 6.
  • query_interface = 2 (HTTP): تُكتب بدلًا من ذلك معلومات طلب HTTP المُمرَّر من الخادم — http_method (UInt8)، وhttp_user_agent (String)، ثم forwarded_for (String، ويخضع لـ X_FORWARDED_FOR_IN_CLIENT_INFO v54443) وhttp_referer (String، ويخضع لـ REFERER_IN_CLIENT_INFO v54447). ولا تظهر حقول os_user/client_hostname/client_name/version_*/protocol_version.
  • أي واجهة أخرى: لا تُكتب أي من حقول TCP (7–12)، ولا أي من حقول HTTP؛ ويستمر التدفّق مباشرةً مع quota_key.
بعد هذا التفرع، يعود التخطيط ليلتقي مجددًا: يأتي quota_key (الحقل 13) وdistributed_depth (الحقل 14) في جميع الواجهات، ثم يُكتب version_patch (الحقل 15) فقط في حالة TCP.تبرز أهمية هذا التفرع أساسًا في حركة المرور بين الخوادم، حيث يمرّر الخادم المُبادِر استعلامًا وصل أصلًا عبر HTTP. وأي وحدة فك ترميز تقرأ دائمًا حقول TCP ستُسيء تفسير مثل هذه الحزم — فتتعامل مع http_method أو http_user_agent على أنه quota_key.
ترميز OpenTelemetry (الحقل 16):
[UInt8: has_trace]              0 = no trace data follows, 1 = trace data follows
If has_trace == 1:
  [16 bytes: trace_id]          byte-swapped per-8-bytes
  [8 bytes:  span_id]           byte-swapped
  [String:   trace_state]       W3C trace state
  [UInt8:    trace_flags]       W3C trace flags

المصادقة بين الخوادم

الحقل 4 من الاستعلام (auth_hash) ليس السر المشترك للعنقود على مستوى النقل. إذ إن إرسال السر بصيغته الخام سيؤدي إلى فشل المصادقة وكشفه في الوقت نفسه. وبدلًا من ذلك، يثبت الخادم الذي يعمل كعميل بين الخوادم أنه يعرف السر باستخدام تجزئة SHA-256 مملّحة:
  1. الدخول إلى وضع inter-server. يشير الخادم المتصل إلى ذلك داخل ClientHello: تكون قيمة الحقل user هي وسم inter-server ويكون password فارغًا. ثم يُلحق سلسلتين إضافيتين — اسم العنقود وsalt مُولَّد حديثًا بطول 32 بايت (encodeSHA256 لقيمة عشوائية) — مباشرة بعد حقلي user/password، ضمن حزمة ClientHello نفسها. يقرأ الخادم هاتين السلسلتين قبل أن يرسل ServerHello، لذلك يجب على العميل كتابتهما مسبقًا؛ إذ إن انتظار ServerHello أولًا يسبب حالة deadlock، لأن الخادم يكون متوقفًا أثناء قراءتهما.
  2. الحصول على nonce. يحمل ServerHello قيمة nonce من النوع UInt64 بطول 8 بايت عند التفاوض على INTERSERVER_SECRET_V2 (v54462).
  3. حساب قيمة التجزئة. لكل حزمة Query غير InitialQuery، يكتب العميل encodeSHA256(salt + nonce + cluster_secret + query + query_id + initial_user + external_roles) في الحقل 4 — أي ناتج digest بطول 32 بايت. (تكون nonce بصيغة سلسلة عشرية، ولا تكون موجودة إلا عند التفاوض على إصدار ≥ v54462؛ ولا يُلحَق external_roles إلا عند التفاوض على INTERSERVER_EXTERNALLY_GRANTED_ROLES (v54472).) أما في حالة InitialQuery، أو عند عدم تهيئة أي سر للعنقود، فيكتب العميل سلسلة فارغة بدلًا من ذلك.
  4. التحقق. يقرأ الخادم الحقل 4 بحد أقصى 32 بايت ويعيد حساب عملية الربط نفسها باستخدام نسخته الخاصة من السر المشترك للعنقود؛ ويُرفض الاتصال إذا اختلفت قيمتا digest.
العملاء الخارجيون (غير العاملين بين الخوادم) لا يدخلون هذا الوضع مطلقًا، ويرسلون دائمًا auth_hash فارغًا.

الإعداد

يُرمَّز هذا مضمّنًا داخل قائمة الإعدادات في جسم Query (حزمة Query، الحقل 3). تكون القائمة موجودة دائمًا، بغضّ النظر عن الإصدار المتفاوض عليه، وتنتهي بإعداد ذي مفتاح فارغ — أي VarUInt 0 واحد، من دون أي flags أو قيمة بعده. ويعتمد ترميز كل إعداد فقط على الإصدار المتفاوض عليه، وتتحكم فيه SETTINGS_SERIALIZED_AS_STRINGS ‏(v54429). v54429+ (STRINGS_WITH_FLAGS) — يكون كل إعداد هو الثلاثي الموضّح هنا:
#الحقلالنوعالدورالوصف
1keyStringمشتركاسم الإعداد. الفارغ = نهاية القائمة.
2flagsVarUIntمشتركعلامات بت لبيانات التعريف؛ انظر أدناه.
3valueStringمشتركقيمة الإعداد كسلسلة نصية
يغيب الحقلان 2 و3 عندما يكون key فارغًا. قبل 54429 (BINARY) — يكون كل إعداد بالشكل [String key][type-specific binary value]: لا يُكتب الحقل flags إطلاقًا، وتُرمَّز القيمة بصيغتها الثنائية الأصلية الخاصة بالإعداد (على سبيل المثال، عدد صحيح ثابت العرض أو سلسلة مسبوقة بالطول) بدلًا من سلسلة عشرية/نصية. وتظل القائمة منتهيةً بـ key فارغ. يجب على العميل الذي يستهدف إصدارًا متفاوضًا عليه أقل من 54429 أن يقرأ هذه الصيغة الثنائية ويكتبها، لا الثلاثي أعلاه. (الإعدادات المخصّصة المعرّفة من قبل المستخدم هي الاستثناء: فهي تتضمن دائمًا flags وقيمة نصية، في كلا الترميزين.) يحتوي الحقل flags على:
  • 0x01مهم: يؤثر الإعداد في نتائج الاستعلام، ويجب ألا تتجاهله النظراء الأقدم بصمت.
  • 0x02مخصّص: إعداد مخصّص معرّف من قبل المستخدم.
  • 0x0c — حقل tier من 2 بت، وليس علامة مستقلة: 0x00 = Production، 0x04 = Obsolete، 0x08 = Experimental، 0x0c = Beta. اقرأ البتين معًا (flags & 0x0c) — لأن اختبارًا ساذجًا مثل flags & 0x04 سيُصنّف Beta (0x0c) خطأً على أنها Obsolete.
  • 0x80HotReload (إعادة تحميل config من دون إعادة تشغيل؛ معرّف في تعداد العلامات، ويظهر أساسًا في إعدادات coordination).

المعلَمة

معلمات الاستعلام، للاستعلامات ذات المعلَمات مثل SELECT {x:UInt64}. تُرمَّز بالطريقة نفسها تمامًا مثل إعداد مع ضبط العلامة Custom (0x02)، وتُنهى بمفتاح فارغ بالطريقة نفسها.
#الحقلالنوعالدورالوصف
1keyStringالعميلاسم المعلَمة. فارغ = نهاية القائمة.
2flagsVarUIntالعميلدائمًا 0x02 (Custom)
3valueStringالعميلقيمة المعلَمة كسلسلة. راجع الملاحظة أدناه بشأن علامات الاقتباس.
قيمة المعلَمة هي تمثيل SQL للقيمة، وليست قيمة حرفية خام. يجب تمرير المعلَمات من النوع String بعد إحاطتها مسبقًا بعلامات اقتباس مفردة (على سبيل المثال، القيمة لـ {name:String} هي 'Alice' وليست Alice)؛ وإلا فسيرفض محلل القيم في الخادم تحليلها.

Data (نوع الحزمة 1 server→client، ونوع الحزمة 2 client→server)

في كلا الاتجاهين. تحمل كتل النتائج، وبيانات INSERT، والجداول الخارجية، وعلامات نهاية البيانات. تنسيق النقل متماثلة — إذ يتضمن كلا الاتجاهين بادئة table_name قبل Block. والاختلاف الوحيد هو بايت نوع الحزمة.
[VarUInt: packet_type]     1 (server→client) or 2 (client→server)
[String:  table_name]      External table name; empty in most cases
[Block]                    See the Native Format spec for the Block layout
الحقلالنوعالدورالوصف
table_nameStringمشتركاسم الجدول الخارجي. وتكون القيمة الفارغة ("") هي الحالة الشائعة — للجدول الرئيسي، ونتائج الاستعلام، وتدفق صفوف INSERT. ولا تُعد القيمة الفارغة table_name وحدها وسم نهاية البيانات (إذ إن حزم صفوف INSERT العادية تحمل أيضًا "").
جسم Blockراجع بنية Block والعمود.
وسم نهاية البيانات هو حزمة يكون فيها Block فارغًا — 0 أعمدة و0 صفوف — بغض النظر عن table_name. ويتعامل الخادم مع حزمة Data من العميل على أنها المُنهِي فقط عندما تكون كتلة البيانات المفككة فارغة (block.empty())؛ أما الحزمة التي فيها table_name = "" وBlock غير فارغ، فهي حزمة صفوف عادية وليست مُنهِيًا. لذا فإن تدفق صفوف INSERT هو تسلسل من كتل Data غير الفارغة، تتبعها كتلة Data فارغة واحدة تُنهيه. تُوثَّق متغيرات Block وما تعنيه ضمن متغيرات Block.

Progress (نوع الحزمة 3)

الخادم → العميل. تُرسَل دوريًا أثناء تنفيذ الاستعلام. جميع الحقول من نوع VarUInt، وتحمل كل حزمة الزيادات منذ حزمة Progress السابقة، لا الإجماليات التراكمية. قبل الإرسال، يقرأ الخادم عدّاداته ويُعيد ضبطها ذريًا إلى الصفر، ويحسب elapsed_ns على أنه فرق الزمن منذ آخر إرسال. لذلك يجب على العميل تجميع الحزم المتعاقبة محليًا للحصول على الإجماليات الجارية — فالتعامل مع الحزمة على أنها قيمة مطلقة يجعل عرض التقدّم يرتد إلى الخلف أو يقلّل العدد عند وصول أكثر من حزمة واحدة.
#الحقلالنوعالدورالشرطالوصف
1rowsVarUIntمشتركدائمًاالصفوف المقروءة منذ الحزمة السابقة (أضِفها إلى الإجمالي الجاري)
2bytesVarUIntمشتركدائمًاالبايتات المقروءة منذ الحزمة السابقة (أضِفها إلى الإجمالي الجاري)
3total_rowsVarUIntمشتركدائمًازيادة في العدد الإجمالي التقديري للصفوف المطلوب قراءتها؛ تُجمَّع تراكميًا (وقد تكون 0 في حزمة معيّنة)
4total_bytesVarUIntمشتركTOTAL_BYTES_IN_PROGRESS (v54463)زيادة في العدد الإجمالي التقديري للبايتات المطلوب قراءتها؛ تُجمَّع تراكميًا. وتأتي على السلك بين total_rows وwrote_rows.
5wrote_rowsVarUIntمشتركWRITE_CLIENT_INFO (v54420)الصفوف المكتوبة منذ الحزمة السابقة (لأجل INSERT)؛ تُجمَّع تراكميًا
6wrote_bytesVarUIntمشتركWRITE_CLIENT_INFO (v54420)البايتات المكتوبة منذ الحزمة السابقة (لأجل INSERT)؛ تُجمَّع تراكميًا
7elapsed_nsVarUIntمشتركSERVER_QUERY_TIME_IN_PROGRESS (v54460)عدد النانوثواني المنقضي منذ الحزمة السابقة (فرق زمني، وليس زمن الاستعلام الإجمالي)؛ يُجمَّع تراكميًا

ProfileInfo (نوع الحزمة 6)

الخادم → العميل. يُرسل مرة واحدة لكل استعلام، قرب نهاية التنفيذ.
#الحقلالنوعالدورالشرطالوصف
1rowsVarUIntمشتركدائمًاإجمالي الصفوف المُعالجة
2blocksVarUIntمشتركدائمًاإجمالي الكتل المُعالجة
3bytesVarUIntمشتركدائمًاإجمالي البايتات المُعالجة
4applied_limitBoolمشتركدائمًاما إذا كانت عبارة LIMIT قد طُبِّقت
5rows_before_limitVarUIntمشتركدائمًاعدد الصفوف قبل LIMIT
6obsoleteBoolمشتركدائمًابايت توافقية متقادم. يكتب الخادم دائمًا true هنا، ويتجاهله العميل عند القراءة؛ وهو ليس علامةً على أنه تم احتساب “rows_before_limit”. حالة الحد ذات المعنى هي الحقل 4 (applied_limit) مع الحقل 5. اقرأه وتجاهله.
7applied_aggregationBoolمشتركROWS_BEFORE_AGGREGATION (v54469)ما إذا كان GROUP BY قد طُبِّق
8rows_before_aggregationVarUIntمشتركROWS_BEFORE_AGGREGATION (v54469)عدد الصفوف قبل التجميع

الإجماليات (نوع الحزمة 7)

الخادوم → العميل. يُرسَل للاستعلامات التي تتضمن WITH TOTALS. تنسيق النقل على مستوى النقل مطابق تمامًا لـ Data: سلسلة table_name (وتكون فارغة دائمًا) تليها كتلة. والاختلاف الوحيد هو بايت نوع الحزمة.
[VarUInt: 7]                packet type
[String:  table_name]       always empty
[Block]                     see the Native Format spec

القيم القصوى (نوع الحزمة 8)

الخادم → العميل. تُرسَل عندما يكون الإعداد extremes مفعّلًا. تنسيق النقل مطابق تمامًا لـ Data. تحتوي الكتلة على صفّين بالضبط: يحتوي الصف 0 على الحد الأدنى لكل عمود، ويحتوي الصف 1 على الحد الأقصى.
[VarUInt: 8]                packet type
[String:  table_name]       always empty
[Block]                     num_rows = 2

Log (نوع الحزمة 10)

الخادم → العميل. تُرسَل هذه الحزمة عندما تكون للاستعلام قائمة انتظار سجلات نشطة (إعداد send_logs_level؛ راجع بث السجلات). تنسيق الغلاف والمحتوى مطابق لتنسيق Data. تحتوي الكتلة على قيمة ثابتة num_columns = 8 ومخطط مُعرّف مسبقًا. ويمثّل كل سطر سجل صفًا واحدًا عبر الأعمدة الثمانية جميعها، وقد تتضمن حزمة Log واحدة صفوفًا عديدة.
[VarUInt: 10]               packet type
[String:  table_name]       always empty
[Block]                     num_columns = 8, num_rows = number of log lines
الأعمدة الثمانية، بهذا الترتيب تمامًا:
#NameTypeDescription
1event_timeDateTimeالطابع الزمني للحدث (بالثواني منذ epoch)
2event_time_microsecondsUInt32مكوّن الميكروثانية
3host_nameStringاسم مضيف الخادم الذي يُصدر السجل
4query_idStringمعرّف الاستعلام الذي ينتمي إليه السجل
5thread_idUInt64معرّف خيط نظام التشغيل
6priorityInt8مستوى السجل (أولوية Poco: 1 = Fatal، … 8 = Trace)
7sourceStringاسم المُسجِّل
8textStringنص رسالة السجل

ProfileEvents (نوع الحزمة 14)

الخادم → العميل. يحمل عدّادات أداء لكل استعلام. له نفس تنسيق الغلاف والمحتوى كما في Data. تحتوي الكتلة على قيمة ثابتة لـ num_columns = 6 ومخطط محدد مسبقًا. ويمثل كل حدث صفًا واحدًا.
[VarUInt: 14]               packet type
[String:  table_name]       always empty
[Block]                     num_columns = 6, num_rows = number of events
الأعمدة الستة:
#الاسمالنوعالوصف
1host_nameStringاسم مضيف الخادم
2current_timeDateTimeالطابع الزمني للحدث
3thread_idUInt64معرّف الخيط
4typeEnum8نوع الحدث: 1 = زيادة (counter)، 2 = Gauge. ويكون التخزين الأساسي بايتًا موقّعًا واحدًا.
5nameStringاسم الحدث (مثل: "Query"، "NetworkReceiveBytes")
6valueInt64قيمة العداد أو قراءة Gauge
نوع العنصر في العمود value ليس ثابتًا عبر الحزم — إذ ترسل الخوادم الأقدم UInt64، بينما ترسل الأحدث Int64. اقرأ سلسلة نوع العمود من ترويسة block بدلًا من افتراض عرض واحد.

TableColumns (نوع الحزمة 11)

الخادم → العميل، ويكون إرسالها مشروطًا بـ COLUMN_DEFAULTS_METADATA ‏(v54410). يرسلها الخادم قبل كتلة المخطط الخاصة بـ INSERT لنقل البيانات الوصفية للقيم الافتراضية للأعمدة، ولكن فقط عندما يكون الإصدار المتفاوض عليه ≥ 54410 و يكون الإعداد input_format_defaults_for_omitted_fields مفعّلًا. في الإصدارات الأقدم من 54410، لا تُرسل هذه الحزمة مطلقًا، لذلك يجب على العميل الأقدم ألا ينتظرها — إذ تأتي كتلة المخطط Data مباشرةً. ينبغي أن يكون عميل v54410+ مستعدًا لأيٍّ من الترتيبين: TableColumns اختيارية، ثم كتلة المخطط.
#FieldTypeRoleDescription
1external_tableStringuniversalاسم الجدول الخارجي. فارغ = الجدول الرئيسي.
2columns_descriptionStringuniversalتعريفات الأعمدة النصية، مثل "id Int32, name String DEFAULT ''". نص حرّ — حلّله كسلسلة نصية.
جسم مضغوط عند v54481+عند الإصدار المتفاوض عليه ≥ 54481 (COMPRESSED_LOGS_PROFILE_EVENTS_COLUMNS)، يكتب الخادم كلا الحقلين عبر مسار الإخراج نفسه القابل للضغط اختياريًا، لذا عندما يكون compression = true في query، يكون جسم TableColumns بالكامل (external_table + columns_description) داخل إطار الضغط؛ ويقرأه العميل عبر stream فك الضغط المطابقة. وعندما لا تكون هناك عملية ضغط في query، يكون الجسم على wire غير مضغوط تمامًا كما يوضحه الجدول أعلاه. وهذا مهم لاستجابات كتلة المخطط الخاصة بـ INSERT: فالعميل الذي يبدّل معالجة الضغط لكل من Log وProfileEvents دون TableColumns سيُسيء قراءة الاستجابة عند تفعيل ضغط query.

TimezoneUpdate (نوع الحزمة 17)

الخادم → العميل، ويخضع لـ TIMEZONE_UPDATES ‏(v54464). تُرسَل في موضع واحد فقط: مُهيِّئ دالة الجدول input (استعلام بالصيغة INSERT INTO <table> SELECT ... FROM input('<structure>')، حيث تتدفق الصفوف من العميل). مباشرةً بعد أن يرسل الخادم كتلة Data الخاصة بمخطط الإدخال (انظر مرحلة INSERT)، يُرسل TimezoneUpdate حاملةً قيمة session_timezone الحالية في سياق الاستعلام، لكي يفسِّر العميل الصفوف التي يوشك على إرسالها باستخدام المنطقة الزمنية نفسها. لا يرسل الخادم هذه الحزمة عند أي تغييرات لاحقة على SET session_timezone أثناء الاستعلام، ولا لإبلاغ العميل بكيفية تنسيق كتل النتائج اللاحقة.
#FieldTypeRoleDescription
1timezoneStringمشتركالمنطقة الزمنية الافتراضية الجديدة للجلسة (مثل "UTC" و"Europe/Berlin").
تصل الحزمة مرة واحدة، مباشرةً بعد كتلة مخطط الإدخال وقبل أن يبدأ العميل في إرسال كتل الصفوف. ويجب على decoder الذي يتجاهل TimezoneUpdate أن يستهلك أيضًا قيمة String اللاحقة للحفاظ على محاذاة wire.

مصادقة SSH بالتحدي والاستجابة (أنواع الحزم 11 و12 و18)

تخضع لـ SSH_AUTHENTICATION ‏(v54466)، ولا تُفعَّل إلا عند الاختيار الصريح. يدخل الاتصال في مسار SSH عندما يرسل ClientHello القيمة user = " SSH KEY AUTHENTICATION " + <real_user> (مع المسافات البادئة واللاحقة) وpassword = "". يقرأ الخادم البادئة، ويزيلها لاستعادة اسم المستخدم الحقيقي، ثم ينتقل إلى نمط التحدي والاستجابة.
PacketCodeDirectionBody
SSHChallengeRequest11Client → Server(لا يوجد جسم)
SSHChallenge18Server → ClientString challenge — بايتات عشوائية؛ أحد مكوّنات السلسلة التي يجري توقيعها (انظر أدناه)
SSHChallengeResponse12Client → ServerString signature — توقيع SSH على الربط المعرّف أدناه، وليس على challenge الخام
يعمل هذا المسار بدلًا من password authentication، ويحدث تبادل التحدي والاستجابة قبل ServerHello — إذ يؤجل الخادم رد Hello الخاص به حتى تنجح المصادقة:
  1. يرسل العميل ClientHello مع بادئة وسم SSH وكلمة مرور فارغة.
  2. يرسل العميل SSHChallengeRequest (الحزمة 11). لم يرسل الخادم ServerHello بعد — إذ يعالج المصادقة أولًا ويتوقف هنا منتظرًا هذه الحزمة.
  3. يرد الخادم بـ SSHChallenge متضمنةً بايتات عشوائية (الحزمة 18).
  4. ينشئ العميل السلسلة المطلوب توقيعها ويوقّع هذه السلسلة، لا challenge الخام، ثم يرسل SSHChallengeResponse (الحزمة 12) مع التوقيع. الرسالة الموقَّعة هي ربط بايت-ببايت، من دون فواصل، لأربعة أجزاء بهذا الترتيب الدقيق:
    to_sign = decimal(protocol_version) + default_database + user + challenge
    
    PartSource
    decimal(protocol_version)إصدار البروتوكول الخاص بالعميل على هيئة سلسلة ASCII عشرية (مثل "54466") — رقم الإصدار كسلسلة، وليس VarUInt أو عددًا صحيحًا ثابت العرض. يتحقق الخادم باستخدام إصدار البروتوكول نفسه الذي استلمه في ClientHello.
    default_databaseالحقل database من ClientHello (سلسلة فارغة إذا لم تكن هناك قيمة).
    userاسم المستخدم الحقيقي بعد إزالة بادئة الوسم " SSH KEY AUTHENTICATION " — وهو الاسم نفسه الذي يستعيده الخادم بعد إزالة البادئة.
    challengeبايتات challenge الخام من حزمة SSHChallenge.
  5. يتحقق الخادم من التوقيع باستخدام public key المسجل للمستخدم، مع إعادة تكوين السلسلة نفسها decimal(protocol_version) + default_database + user + challenge. عند النجاح، يرسل ServerHello — وهو الرد نفسه كما في مسار كلمة المرور — وتستمر المصافحة بشكل طبيعي (Addendum، إلخ)؛ وعند الفشل، يعيد Exception وينهي الاتصال. سيفشل العميل الذي يوقّع بايتات التحدي الخام فقط في المصادقة.
هذا عكس ما يحدث في مصافحة كلمة المرور، إذ يأتي ServerHello مباشرةً بعد ClientHello. أما في مصادقة SSH، فيُحجب ServerHello إلى ما بعد التحقق من التوقيع، لذا يتداخل تبادل التحدي والاستجابة الخاص بـ SSH ضمن المصافحة قبل ظهور أي ServerHello.
العملاء الخارجيون الذين لا يستخدمون مصادقة SSH لا يرون الحزم 11 أو 12 أو 18 مطلقًا — إذ لا تظهر على wire إلا إذا اختار المستخدم ذلك صراحةً عبر بادئة اسم المستخدم.

مرجع أنواع الحزم

العميل → الخادم

CodeNameBody formatDescription
0HelloClientHelloبدء المصافحة
1QueryQueryطلب تنفيذ الاستعلام
2DataDataكتلة بيانات (بيانات INSERT، والجداول الخارجية، ووسم نهاية البيانات)
3Cancel(من دون محتوى)إلغاء الاستعلام الجاري
4PingPingفحص الحيوية
5TablesStatusRequestغير محددالتحقق من حالة الجدول
6KeepAliveغير محددإبقاء الاتصال حيًّا
7Scalarغير محددكتلة بيانات scalar
8IgnoredPartUUIDsغير محددالأجزاء المطلوب استبعادها من الاستعلام
9ReadTaskResponseغير محدداستجابة مهمة قراءة cluster في S3
10MergeTreeReadTaskResponseغير محدداستجابة مهمة القراءة المتوازية
11SSHChallengeRequestمصادقة SSHطلب challenge لمصادقة SSH
12SSHChallengeResponseمصادقة SSHاستجابة challenge لمصادقة SSH
13QueryPlanغير محددخطة الاستعلام

الخادم → العميل

الرمزالاسمتنسيق المحتوىالوصف
0HelloServerHelloاستجابة المصافحة
1DataDataكتلة بيانات النتيجة
2ExceptionExceptionخطأ
3ProgressProgressتقدّم تنفيذ الاستعلام
4PongPongاستجابة فحص الحيوية
5EndOfStream(بدون محتوى)اكتمل الاستعلام
6ProfileInfoProfileInfoبيانات توصيف الأداء بعد التنفيذ
7TotalsTotalsصف GROUP BY WITH TOTALS
8ExtremesExtremesالقيم الصغرى/الكبرى (كتلة من صفّين)
9TablesStatusResponseغير محدداستجابة حالة الجدول
10LogLogأسطر سجل تنفيذ الاستعلام
11TableColumnsTableColumnsأوصاف الأعمدة للقيم الافتراضية
12PartUUIDsغير محددمعرّفات الأجزاء الفريدة
13ReadTaskRequestغير محددطلب مهمة قراءة للعنقود
14ProfileEventsProfileEventsعدّادات الأداء
15MergeTreeAllRangesAnnouncementغير محددتهيئة القراءة المتوازية
16MergeTreeReadTaskRequestغير محددإسناد مهمة القراءة المتوازية
17TimezoneUpdateTimezoneUpdateتحديث المنطقة الزمنية للخادم
18SSHChallengeSSH authتحدّي مصادقة SSH

التهيئة

يغطي هذا القسم الإعدادات القابلة للضبط التي تُحدِّد سلوك اتصالات البروتوكول الأصلي: تعكس القيم الافتراضية أدناه إصدارًا حديثًا من الخادم؛ وقد تختلف بين الإصدارات وعمليات النشر.

إعدادات طبقة النقل

خيارات المقبس

الخيارالقيمة الافتراضيةالجهةالوصف
TCP_NODELAYمفعّلكلا الطرفينخوارزمية Nagle معطّلة. تُرسَل الحزم الصغيرة فورًا.
SO_KEEPALIVEمفعّل (العميل)، الإعداد الافتراضي لنظام التشغيل (الخادم)غير متماثلفحوصات keepalive لـ TCP على مستوى النواة. يفعّل العميل هذا الخيار صراحةً عندما تكون قيمة tcp_keep_alive_timeout > 0. ويرث الخادم الإعداد الافتراضي لنظام التشغيل.
SO_RCVBUF / SO_SNDBUFالإعدادات الافتراضية لنظام التشغيلأحجام المخزن المؤقت للمقبس. لا يضبط البروتوكول هذه القيم.

المهلات الزمنية

SettingDefaultUnitSideDescription
connect_timeout10ثوانٍالعميلمهلة إنشاء اتصال TCP الأولي.
handshake_timeout_ms10000مللي ثانيةالعميلمهلة استلام ServerHello أثناء المصافحة.
send_timeout300ثوانٍكلاهماإذا تعذر إرسال أي بايتات خلال هذه المدة، فسيُطلق الاتصال استثناءً.
receive_timeout300ثوانٍكلاهماإذا تعذر قراءة أي بايتات خلال هذه المدة، فسيُطلق الاتصال استثناءً.
tcp_keep_alive_timeout290ثوانٍالعميلمدة الخمول قبل أن يرسل نظام التشغيل أول مسبار TCP keepalive.
receive_data_timeout_ms2000مللي ثانيةالعميلمهلة استلام أول حزمة Data من نسخة متماثلة.
connect_timeout_with_failover_ms1000مللي ثانيةالعميلمهلة الاتصال لكل محاولة عند التكرار على النسخ المتماثلة.
connect_timeout_with_failover_secure_ms1000مللي ثانيةالعميلمهلة الاتصال لكل محاولة عند التكرار على النسخ المتماثلة عبر TLS.
hedged_connection_timeout_ms50مللي ثانيةالعميلمهلة الاتصال لكل محاولة للطلبات الاحتياطية.
poll_interval10ثوانٍالخادممعدل حلقة تحقّق الخادم من الاتصالات الخاملة وعمليات الإيقاف.
تتداخل المهلات الزمنية على النحو التالي:
tcp_keep_alive_timeout (290s)
      < receive_timeout (300s)
      < idle_connection_timeout (3600s)
      < tcp_close_connection_after_queries_seconds (0 = unlimited by default)
تعمل آلية keepalive في نظام التشغيل أولًا، وقد تكتشف الأطراف النظيرة المتوقفة بصمت على مستوى النواة. وتُعد مهلة الاستقبال في التطبيق خط الدفاع التالي. أما مهلة الخمول فهي الملاذ الأخير للتخلص من الاتصالات غير المستخدمة منذ وقت طويل.

حدود الاتصالات

الإعدادالافتراضيالوحدةالجانبالوصف
max_connections4096عددالخادمالحد الأقصى لعدد اتصالات TCP المتزامنة.
idle_connection_timeout3600ثوانٍالخادمالحد الأقصى للمدة التي يمكن أن يظل فيها الاتصال الخامل مفتوحًا.
tcp_close_connection_after_queries_num0 (غير محدود)عددالخادمالحد الأقصى لعدد الاستعلامات لكل اتصال قبل إغلاقه قسرًا.
tcp_close_connection_after_queries_seconds0 (غير محدود)ثوانٍالخادمالحد الأقصى لإجمالي عمر الاتصال بغض النظر عن النشاط.
يمكن للاتصال الذي يرسل استعلامات بانتظام أن يبقى مفتوحًا إلى أجل غير مسمى. ولا تُغلق بعد ساعة إلا الاتصالات الخاملة، ولا يوجد حد أقصى افتراضي لمدة الاتصال.

إعدادات طبقة التطبيق

تُنقل هذه الإعدادات مع كل استعلام ضمن قائمة إعدادات حزمة Query. وهي تغيّر ما يرسله الخادم على الـwire، أو كيفية تأطيره.

الضغط

الإعدادالافتراضيالوحدةالوصف
network_compression_method"LZ4"سلسلة نصيةترميز الضغط المستخدم عند ضبط العلامة compression في حزمة Query. القيم: "LZ4", "LZ4HC", "ZSTD", "NONE".
network_zstd_compression_level11–15مستوى ZSTD عندما يكون network_compression_method == "ZSTD".
تؤدي العلامة compression في حزمة Query (الحقل 6) إلى تفعيل الضغط أو تعطيله؛ وتحدد هذه الإعدادات ترميز الضغط المستخدم عند تفعيله.

بث السجلات

الإعدادالافتراضيالوحدةالوصف
send_logs_level"fatal"stringالحد الأدنى لمستوى السجل. القيم: "none"، "fatal"، "error"، "warning"، "information"، "debug"، "trace".
send_logs_source_regexp""stringمرشح Regex على مصدر المُسجِّل. فارغ = تمرّ جميع المصادر.
يؤدي ضبط send_logs_level على أي قيمة غير "none" إلى جعل الخادم يرسل حزم Log أثناء تنفيذ الاستعلام.

الإبلاغ عن التقدّم

الإعدادالافتراضيالوحدةالوصف
interactive_delay100000ميكروثانيةالحد الأدنى المستهدف للفاصل الزمني بين حزم Progress المتعاقبة.
هذا حد أدنى مستهدف، وليس حدًا أقصى صارمًا: قد يرسل الخادوم حزم Progress بوتيرة أبطأ عندما لا يولّد الاستعلام عملاً بالسرعة الكافية.

غلاف النتيجة

الإعدادالافتراضيالوحدةالوصف
extremesfalseboolعندما تكون القيمة true، يرسل الخادم حزمة Extremes تتضمن قيم الحد الأدنى/الحد الأقصى لكل عمود.
max_result_rows0 (غير محدود)countحد أقصى لعدد الصفوف المرسلة. يتحكم result_overflow_mode في هذا السلوك.
max_result_bytes0 (غير محدود)uncompressed bytesحد أقصى لحجم البيانات غير المضغوطة بالبايت. يتحكم result_overflow_mode في هذا السلوك.
result_overflow_mode"throw"stringتنهي "throw" الدفق مع Exception؛ بينما ترسل "break" نتائج جزئية ثم EndOfStream.

INSERT غير المتزامن

الإعدادالافتراضيالوحدةالوصف
async_inserttrueمنطقيعندما تكون القيمة true، تُوضَع بيانات INSERT في قائمة انتظار على الخادم وتُجمَّع على دفعات.
wait_for_async_inserttrueمنطقيعندما تكون القيمة true (مع تفعيل async_insert)، يؤخّر الخادم الاستجابة حتى تُفرَّغ البيانات الموجودة في قائمة الانتظار.
wait_for_async_insert_timeout120ثانيةالحد الأقصى للوقت الذي ينتظره الخادم لإتمام التفريغ قبل إرجاع الاستجابة.

التتبّع الموزّع

الإعدادالافتراضيالوحدةالوصف
opentelemetry_start_trace_probability0.0احتمال 0–1احتمال قيام الخادم بإرفاق سياق OpenTelemetry ببيانات القياس عن بُعد الخاصة بالاستجابة.

إعدادات خارج النطاق

يُظَنّ أحيانًا، خطأً، أن هذه الإعدادات تخص مستوى البروتوكول، لكنها تتحكم في تنفيذ SQL أو التخزين أو استخدام CPU، لا في السلوك على مستوى النقل عبر السلك. لذلك لا يحتاج تنفيذ البروتوكول إلى التعامل معها بشكل خاص.
  • max_threads — التوازي داخل تنفيذ الاستعلام.
  • max_memory_usage — الحد الأقصى للذاكرة لكل استعلام.
  • max_block_size, preferred_block_size_bytes — تحديد حجم الكتل داخليًا أثناء معالجة الاستعلام؛ أما كتل النقل عبر السلك فهي مستقلة عن هذه الإعدادات.
  • compile_expressions — الترجمة الفورية JIT؛ تخص CPU فقط.
  • async_insert_max_data_size — مخزن queue المؤقت على جهة الخادم.
  • جميع إعدادات input_format_* وoutput_format_* باستثناء مجموعة input_format_native_* / output_format_native_* — الإعدادات غير native تختار تنسيقات أخرى أو تضبطها (على سبيل المثال عبر HTTP) ولا تغيّر كتل Data في البروتوكول الأصلي.
تمثل إعدادات *_native_* الاستثناء: فهي تغيّر البايتات داخل كتل Data في native TCP، لذا يجب على أي تنفيذ للبروتوكول أن يراعيها. يبدّل output_format_native_encode_types_in_binary_format حقل type في العمود من سلسلة نصية إلى ترميز نوع ثنائي، ويُخرج output_format_native_write_json_as_string أعمدة JSON على هيئة String، ويختار output_format_native_use_flattened_dynamic_and_json_serialization تخطيط Dynamic/JSON من نوع FLATTENED. ولأن هذه الإعدادات تؤثر في جسم الكتلة لا في غلاف الحزمة، فهي محددة في مواصفة Native Format — راجع بنية الأعمدة على السلك والأنواع ذات الإصدارات.

المسرد

Cancel — حزمة يبدأها العميل (النوع 3) لإلغاء query قيد التنفيذ. لا تُعرض تفاصيلها في هذه الصفحة. وسم نهاية بيانات العميل — حزمة Data فارغة (0 columns, 0 rows) يرسلها العميل لإغلاق stream الإدخال. يختلف موضعها بحسب نوع query:
  • query عادي (SELECT, etc.): تُرسل بعد حزمة Query وأي حزم Data للجداول الخارجية للإشارة إلى “لا مزيد من البيانات الخارجية”. بعدها يبدأ server التنفيذ.
  • INSERT: لا يرسل العميل وسمًا قبل schema. يرسل server أولًا كتلة المخطط، ثم يبث العميل data blocks الخاصة بالصفوف، وبعد ذلك فقط يرسل حزمة Data الفارغة لإنهاء stream الصفوف. إن إرسال وسم فارغ قبل كتلة المخطط سيُفسَّر على أنه نهاية فورية للصفوف، ما يؤدي إلى فقدان البيانات.
Feature — تغيير في تنسيق النقل أُدخل في إصدار محدد من protocol. تكون الميزة نشطة عندما تكون protocol version المتفاوض عليها مساوية لإصدار الميزة أو أعلى منه. راجع الإصدار وبوابات الميزات. Inter-server — تسمية دور لحقل لا يكون ذا معنى إلا في distributed queries بين الخوادم. ويكتب العملاء الخارجيون default value (عادةً سلسلة فارغة أو 0 أو false). الإصدار المتفاوض عليهmin(client_version, server_version)، ويُحتسب أثناء المصافحة. ويحدد الميزات النشطة طوال lifetime الخاص بالاتصال. Packet — رسالة wire: رمز packet type من نوع VarUInt يتبعه body يعتمد تنسيقه على النوع. راجع غلاف الحزمة. رمز نوع الحزمة — قيمة VarUInt الأولى في الحزمة، وهي التي تحدد تنسيقها. القيم من 0 إلى 18 مخصَّصة حاليًا. راجع مرجع أنواع الحزم. تدفق الاستجابة — تسلسل الحزم التي يصدرها server أثناء query. وهو مفتوح الطول، وينتهي بـ EndOfStream واحد فقط (نجاح) أو Exception (فشل). راجع مرحلة query. كتلة المخطط — header block (أي Block يحتوي على columns لكن 0 rows) يرسله server أثناء مرحلة INSERT للإعلان عن أشكال columns المتوقعة قبل أن يرسل العميل البيانات. قائمة الإعدادات — تسلسل من tuples بالشكل (key, flags, value) داخل body الخاص بـ Query، وينتهي بمفتاح فارغ. يحمل configuration على مستوى التطبيق لكل query. راجع Setting. Stage — حقل VarUInt في حزمة Query (الحقل 5) يحدد إلى أي مدى ينفّذ server الـ query. يرسل العملاء الخارجيون عادةً 2 (Complete)، بينما تستخدم distributed queries وquery plans المُسلسلة القيم الأعلى. راجع الحقل 5 في Query للاطلاع على المجموعة الكاملة من قيم wire. Terminator — حزمة تُنهي stream. تنتهي استجابة Query عند EndOfStream (نجاح) أو Exception (فشل). وينتهي stream الإدخال الخاص بالعميل عند وسم Data الفارغ.
Last modified on June 25, 2026