1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
|
import socket
import getpass
from time import time
from . import defines
from . import errors
from .opentelemetry import OpenTelemetryTraceContext
from .varint import write_varint
from .writer import write_binary_str, write_binary_uint8, \
write_binary_uint64, write_binary_uint128
class ClientInfo(object):
class Interface(object):
TCP = 1
HTTP = 2
class QueryKind(object):
# Uninitialized object.
NO_QUERY = 0
INITIAL_QUERY = 1
# Query that was initiated by another query for distributed query
# execution.
SECONDARY_QUERY = 2
client_version_major = defines.CLIENT_VERSION_MAJOR
client_version_minor = defines.CLIENT_VERSION_MINOR
client_version_patch = defines.CLIENT_VERSION_PATCH
client_revision = defines.CLIENT_REVISION
interface = Interface.TCP
initial_user = ''
initial_query_id = ''
initial_address = '0.0.0.0:0'
def __init__(self, client_name, context):
self.query_kind = ClientInfo.QueryKind.NO_QUERY
try:
self.os_user = getpass.getuser()
except KeyError:
self.os_user = ''
self.client_hostname = socket.gethostname()
self.client_name = client_name
self.client_trace_context = OpenTelemetryTraceContext(
context.client_settings['opentelemetry_traceparent'],
context.client_settings['opentelemetry_tracestate']
)
self.quota_key = context.client_settings['quota_key']
self.distributed_depth = 0
self.initial_query_start_time_microseconds = int(time() * 1000000)
super(ClientInfo, self).__init__()
@property
def empty(self):
return self.query_kind == ClientInfo.QueryKind.NO_QUERY
def write(self, server_revision, fout):
revision = server_revision
if server_revision < defines.DBMS_MIN_REVISION_WITH_CLIENT_INFO:
raise errors.LogicalError('Method ClientInfo.write is called '
'for unsupported server revision')
write_binary_uint8(self.query_kind, fout)
if self.empty:
return
write_binary_str(self.initial_user, fout)
write_binary_str(self.initial_query_id, fout)
write_binary_str(self.initial_address, fout)
if (
revision >=
defines.DBMS_MIN_PROTOCOL_VERSION_WITH_INITIAL_QUERY_START_TIME
):
write_binary_uint64(
self.initial_query_start_time_microseconds, fout
)
write_binary_uint8(self.interface, fout)
write_binary_str(self.os_user, fout)
write_binary_str(self.client_hostname, fout)
write_binary_str(self.client_name, fout)
write_varint(self.client_version_major, fout)
write_varint(self.client_version_minor, fout)
write_varint(self.client_revision, fout)
if revision >= defines.DBMS_MIN_REVISION_WITH_QUOTA_KEY_IN_CLIENT_INFO:
write_binary_str(self.quota_key, fout)
if revision >= \
defines.DBMS_MIN_PROTOCOL_VERSION_WITH_DISTRIBUTED_DEPTH:
write_varint(self.distributed_depth, fout)
if revision >= defines.DBMS_MIN_REVISION_WITH_VERSION_PATCH:
write_varint(self.client_version_patch, fout)
if revision >= defines.DBMS_MIN_REVISION_WITH_OPENTELEMETRY:
if self.client_trace_context.trace_id is not None:
# Have OpenTelemetry header.
write_binary_uint8(1, fout)
write_binary_uint128(self.client_trace_context.trace_id, fout)
write_binary_uint64(self.client_trace_context.span_id, fout)
write_binary_str(self.client_trace_context.tracestate, fout)
write_binary_uint8(self.client_trace_context.trace_flags, fout)
else:
# Don't have OpenTelemetry header.
write_binary_uint8(0, fout)
if revision >= defines.DBMS_MIN_REVISION_WITH_PARALLEL_REPLICAS:
write_varint(0, fout) # collaborate_with_initiator
write_varint(0, fout) # count_participating_replicas
write_varint(0, fout) # number_of_current_replica
|