From c07d6f7a820ef8f9877bd94a78e204d85cef0e4f Mon Sep 17 00:00:00 2001 From: Jordan Lewis Date: Fri, 8 Jun 2018 17:44:49 -0400 Subject: [PATCH] storage: return byte batches in ScanResponse Previously, the MVCCScan API call completely deserialized the batch it got from C++ into a slice of roachpb.KeyValue, and sent that slice of roachpb.KeyValue over gRPC via ScanResponse. This is needlessly expensive for several reasons. - gRPC must re-serialize what we sent it to a flat byte stream. But, we already had a flat byte stream to begin with, before inflating it into KeyValues. In effect, we're doing pointless deserialization and reserialization. - We needed to dynamically allocate a slice of roachpb.KeyValue on every scan request, in buildScanResponse. This was the second largest cause of allocations in our system, beside the first copy from C++ to Go. But, it's pointless, since we're just going to throw that slice away again when we either serialize to the network or iterate over it and inflate the KeyValues into rows later down the pipe. Now, MVCCScan can optionally skip this inflation and return the raw write batch that it got from C++. The txnKVFetcher and rowFetcher are modified to use this option. They now deserialize keys from the write batch as necessary. This results in a large decrease in the number of allocations performed per scan. When going over the network, only 1 object has to be marshalled and demarshalled (the batch) instead of the number of returned keys. Also, we don't have to allocate the initial slice of []KeyValue, or any of the slices within Key or Value, to return data. Release note: None --- docs/generated/settings/settings.html | 2 +- pkg/kv/dist_sender.go | 13 + pkg/roachpb/api.go | 2 + pkg/roachpb/api.pb.go | 895 +++++++++++------- pkg/roachpb/api.proto | 35 + pkg/server/updates_test.go | 2 +- pkg/settings/cluster/cockroach_versions.go | 6 + .../testdata/logic_test/crdb_internal | 4 +- pkg/sql/sqlbase/errors.go | 13 +- pkg/sql/sqlbase/fk.go | 13 +- pkg/sql/sqlbase/kvfetcher.go | 29 +- pkg/sql/sqlbase/rowfetcher.go | 38 +- pkg/storage/batcheval/cmd_reverse_scan.go | 36 +- pkg/storage/batcheval/cmd_scan.go | 37 +- pkg/storage/engine/mvcc.go | 215 +++-- pkg/storage/engine/rocksdb.go | 28 +- 16 files changed, 917 insertions(+), 451 deletions(-) diff --git a/docs/generated/settings/settings.html b/docs/generated/settings/settings.html index 90ebcf30ab11..bfedd988bffb 100644 --- a/docs/generated/settings/settings.html +++ b/docs/generated/settings/settings.html @@ -68,6 +68,6 @@ trace.debug.enablebooleanfalseif set, traces for recent requests can be seen in the /debug page trace.lightstep.tokenstringif set, traces go to Lightstep using this token trace.zipkin.collectorstringif set, traces go to the given Zipkin instance (example: '127.0.0.1:9411'); ignored if trace.lightstep.token is set. -versioncustom validation2.0-9set the active cluster version in the format '.'. +versioncustom validation2.0-10set the active cluster version in the format '.'. diff --git a/pkg/kv/dist_sender.go b/pkg/kv/dist_sender.go index 249c23626c40..983211b62412 100644 --- a/pkg/kv/dist_sender.go +++ b/pkg/kv/dist_sender.go @@ -516,6 +516,19 @@ func (ds *DistSender) initAndVerifyBatch( } } + // Make sure that MVCCScan requests aren't in batch form if our cluster + // version is too old. + // TODO(jordan): delete this stanza after 2.1 is released. + if !ds.st.Version.IsMinSupported(cluster.VersionBatchResponse) { + for i := range ba.Requests { + switch req := ba.Requests[i].GetInner().(type) { + case *roachpb.ScanRequest: + req.ScanFormat = roachpb.KEY_VALUES + case *roachpb.ReverseScanRequest: + req.ScanFormat = roachpb.KEY_VALUES + } + } + } return nil } diff --git a/pkg/roachpb/api.go b/pkg/roachpb/api.go index 901037e58fea..8511ca3663c1 100644 --- a/pkg/roachpb/api.go +++ b/pkg/roachpb/api.go @@ -256,6 +256,7 @@ func (sr *ScanResponse) combine(c combinable) error { if sr != nil { sr.Rows = append(sr.Rows, otherSR.Rows...) sr.IntentRows = append(sr.IntentRows, otherSR.IntentRows...) + sr.BatchResponse = append(sr.BatchResponse, otherSR.BatchResponse...) if err := sr.ResponseHeader.combine(otherSR.Header()); err != nil { return err } @@ -271,6 +272,7 @@ func (sr *ReverseScanResponse) combine(c combinable) error { if sr != nil { sr.Rows = append(sr.Rows, otherSR.Rows...) sr.IntentRows = append(sr.IntentRows, otherSR.IntentRows...) + sr.BatchResponse = append(sr.BatchResponse, otherSR.BatchResponse...) if err := sr.ResponseHeader.combine(otherSR.Header()); err != nil { return err } diff --git a/pkg/roachpb/api.pb.go b/pkg/roachpb/api.pb.go index c388732fb468..4cff8ba99791 100644 --- a/pkg/roachpb/api.pb.go +++ b/pkg/roachpb/api.pb.go @@ -248,6 +248,32 @@ func (x ReadConsistencyType) String() string { } func (ReadConsistencyType) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{0} } +// ScanFormat is an enumeration of the available response formats for MVCCScan +// operations. +type ScanFormat int32 + +const ( + // The standard MVCCScan format: a slice of KeyValue messages. + KEY_VALUES ScanFormat = 0 + // The batch_response format: a byte slice of alternating keys and values, + // each prefixed by their length as a varint. + BATCH_RESPONSE ScanFormat = 1 +) + +var ScanFormat_name = map[int32]string{ + 0: "KEY_VALUES", + 1: "BATCH_RESPONSE", +} +var ScanFormat_value = map[string]int32{ + "KEY_VALUES": 0, + "BATCH_RESPONSE": 1, +} + +func (x ScanFormat) String() string { + return proto.EnumName(ScanFormat_name, int32(x)) +} +func (ScanFormat) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{1} } + // TxnPushType determines what action to take when pushing a transaction. type PushTxnType int32 @@ -279,7 +305,7 @@ var PushTxnType_value = map[string]int32{ func (x PushTxnType) String() string { return proto.EnumName(PushTxnType_name, int32(x)) } -func (PushTxnType) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{1} } +func (PushTxnType) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{2} } type ExportStorageProvider int32 @@ -312,7 +338,7 @@ var ExportStorageProvider_value = map[string]int32{ func (x ExportStorageProvider) String() string { return proto.EnumName(ExportStorageProvider_name, int32(x)) } -func (ExportStorageProvider) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{2} } +func (ExportStorageProvider) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{3} } type MVCCFilter int32 @@ -333,7 +359,7 @@ var MVCCFilter_value = map[string]int32{ func (x MVCCFilter) String() string { return proto.EnumName(MVCCFilter_name, int32(x)) } -func (MVCCFilter) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{3} } +func (MVCCFilter) EnumDescriptor() ([]byte, []int) { return fileDescriptorApi, []int{4} } type ResponseHeader_ResumeReason int32 @@ -759,6 +785,10 @@ func (*ScanOptions) Descriptor() ([]byte, []int) { return fileDescriptorApi, []i // number of results (unbounded if zero). type ScanRequest struct { RequestHeader `protobuf:"bytes,1,opt,name=header,embedded=header" json:"header"` + // The desired format for the response. If set to BATCH_RESPONSE, the server + // will set the batch_response field in the ScanResponse instead of the rows + // field. + ScanFormat ScanFormat `protobuf:"varint,4,opt,name=scan_format,json=scanFormat,proto3,enum=cockroach.roachpb.ScanFormat" json:"scan_format,omitempty"` } func (m *ScanRequest) Reset() { *m = ScanRequest{} } @@ -775,6 +805,11 @@ type ScanResponse struct { // consistency level. These rows do not count against the MaxSpanRequestKeys // count. IntentRows []KeyValue `protobuf:"bytes,3,rep,name=intent_rows,json=intentRows" json:"intent_rows"` + // If set, the results of the scan in batch format - the key/value pairs are + // a buffer of varint-prefixed slices, alternating from key to value. There + // are num_keys pairs, as defined by the ResponseHeader. If set, rows will not + // be set and vice versa. + BatchResponse []byte `protobuf:"bytes,4,opt,name=batch_response,json=batchResponse,proto3" json:"batch_response,omitempty"` } func (m *ScanResponse) Reset() { *m = ScanResponse{} } @@ -787,6 +822,10 @@ func (*ScanResponse) Descriptor() ([]byte, []int) { return fileDescriptorApi, [] // number of results (unbounded if zero). type ReverseScanRequest struct { RequestHeader `protobuf:"bytes,1,opt,name=header,embedded=header" json:"header"` + // The desired format for the response. If set to BATCH_RESPONSE, the server + // will set the batch_response field in the ScanResponse instead of the rows + // field. + ScanFormat ScanFormat `protobuf:"varint,4,opt,name=scan_format,json=scanFormat,proto3,enum=cockroach.roachpb.ScanFormat" json:"scan_format,omitempty"` } func (m *ReverseScanRequest) Reset() { *m = ReverseScanRequest{} } @@ -803,6 +842,11 @@ type ReverseScanResponse struct { // consistency level. These rows do not count against the MaxSpanRequestKeys // count. IntentRows []KeyValue `protobuf:"bytes,3,rep,name=intent_rows,json=intentRows" json:"intent_rows"` + // If set, the results of the scan in batch format - the key/value pairs are + // a buffer of varint-prefixed slices, alternating from key to value. There + // are num_keys pairs, as defined by the ResponseHeader. If set, rows will not + // be set and vice versa. + BatchResponse []byte `protobuf:"bytes,4,opt,name=batch_response,json=batchResponse,proto3" json:"batch_response,omitempty"` } func (m *ReverseScanResponse) Reset() { *m = ReverseScanResponse{} } @@ -4631,6 +4675,7 @@ func init() { proto.RegisterType((*BatchResponse)(nil), "cockroach.roachpb.BatchResponse") proto.RegisterType((*BatchResponse_Header)(nil), "cockroach.roachpb.BatchResponse.Header") proto.RegisterEnum("cockroach.roachpb.ReadConsistencyType", ReadConsistencyType_name, ReadConsistencyType_value) + proto.RegisterEnum("cockroach.roachpb.ScanFormat", ScanFormat_name, ScanFormat_value) proto.RegisterEnum("cockroach.roachpb.PushTxnType", PushTxnType_name, PushTxnType_value) proto.RegisterEnum("cockroach.roachpb.ExportStorageProvider", ExportStorageProvider_name, ExportStorageProvider_value) proto.RegisterEnum("cockroach.roachpb.MVCCFilter", MVCCFilter_name, MVCCFilter_value) @@ -4944,6 +4989,9 @@ func (this *ScanRequest) Equal(that interface{}) bool { if !this.RequestHeader.Equal(&that1.RequestHeader) { return false } + if this.ScanFormat != that1.ScanFormat { + return false + } return true } func (this *ReverseScanRequest) Equal(that interface{}) bool { @@ -4968,6 +5016,9 @@ func (this *ReverseScanRequest) Equal(that interface{}) bool { if !this.RequestHeader.Equal(&that1.RequestHeader) { return false } + if this.ScanFormat != that1.ScanFormat { + return false + } return true } func (this *CheckConsistencyRequest) Equal(that interface{}) bool { @@ -6971,6 +7022,11 @@ func (m *ScanRequest) MarshalTo(dAtA []byte) (int, error) { return 0, err } i += n27 + if m.ScanFormat != 0 { + dAtA[i] = 0x20 + i++ + i = encodeVarintApi(dAtA, i, uint64(m.ScanFormat)) + } return i, nil } @@ -7021,6 +7077,12 @@ func (m *ScanResponse) MarshalTo(dAtA []byte) (int, error) { i += n } } + if len(m.BatchResponse) > 0 { + dAtA[i] = 0x22 + i++ + i = encodeVarintApi(dAtA, i, uint64(len(m.BatchResponse))) + i += copy(dAtA[i:], m.BatchResponse) + } return i, nil } @@ -7047,6 +7109,11 @@ func (m *ReverseScanRequest) MarshalTo(dAtA []byte) (int, error) { return 0, err } i += n29 + if m.ScanFormat != 0 { + dAtA[i] = 0x20 + i++ + i = encodeVarintApi(dAtA, i, uint64(m.ScanFormat)) + } return i, nil } @@ -7097,6 +7164,12 @@ func (m *ReverseScanResponse) MarshalTo(dAtA []byte) (int, error) { i += n } } + if len(m.BatchResponse) > 0 { + dAtA[i] = 0x22 + i++ + i = encodeVarintApi(dAtA, i, uint64(len(m.BatchResponse))) + i += copy(dAtA[i:], m.BatchResponse) + } return i, nil } @@ -11418,6 +11491,9 @@ func (m *ScanRequest) Size() (n int) { _ = l l = m.RequestHeader.Size() n += 1 + l + sovApi(uint64(l)) + if m.ScanFormat != 0 { + n += 1 + sovApi(uint64(m.ScanFormat)) + } return n } @@ -11438,6 +11514,10 @@ func (m *ScanResponse) Size() (n int) { n += 1 + l + sovApi(uint64(l)) } } + l = len(m.BatchResponse) + if l > 0 { + n += 1 + l + sovApi(uint64(l)) + } return n } @@ -11446,6 +11526,9 @@ func (m *ReverseScanRequest) Size() (n int) { _ = l l = m.RequestHeader.Size() n += 1 + l + sovApi(uint64(l)) + if m.ScanFormat != 0 { + n += 1 + sovApi(uint64(m.ScanFormat)) + } return n } @@ -11466,6 +11549,10 @@ func (m *ReverseScanResponse) Size() (n int) { n += 1 + l + sovApi(uint64(l)) } } + l = len(m.BatchResponse) + if l > 0 { + n += 1 + l + sovApi(uint64(l)) + } return n } @@ -15426,6 +15513,25 @@ func (m *ScanRequest) Unmarshal(dAtA []byte) error { return err } iNdEx = postIndex + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field ScanFormat", wireType) + } + m.ScanFormat = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowApi + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.ScanFormat |= (ScanFormat(b) & 0x7F) << shift + if b < 0x80 { + break + } + } default: iNdEx = preIndex skippy, err := skipApi(dAtA[iNdEx:]) @@ -15568,6 +15674,37 @@ func (m *ScanResponse) Unmarshal(dAtA []byte) error { return err } iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field BatchResponse", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowApi + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= (int(b) & 0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return ErrInvalidLengthApi + } + postIndex := iNdEx + byteLen + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.BatchResponse = append(m.BatchResponse[:0], dAtA[iNdEx:postIndex]...) + if m.BatchResponse == nil { + m.BatchResponse = []byte{} + } + iNdEx = postIndex default: iNdEx = preIndex skippy, err := skipApi(dAtA[iNdEx:]) @@ -15648,6 +15785,25 @@ func (m *ReverseScanRequest) Unmarshal(dAtA []byte) error { return err } iNdEx = postIndex + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field ScanFormat", wireType) + } + m.ScanFormat = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowApi + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.ScanFormat |= (ScanFormat(b) & 0x7F) << shift + if b < 0x80 { + break + } + } default: iNdEx = preIndex skippy, err := skipApi(dAtA[iNdEx:]) @@ -15790,6 +15946,37 @@ func (m *ReverseScanResponse) Unmarshal(dAtA []byte) error { return err } iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field BatchResponse", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowApi + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= (int(b) & 0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return ErrInvalidLengthApi + } + postIndex := iNdEx + byteLen + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.BatchResponse = append(m.BatchResponse[:0], dAtA[iNdEx:postIndex]...) + if m.BatchResponse == nil { + m.BatchResponse = []byte{} + } + iNdEx = postIndex default: iNdEx = preIndex skippy, err := skipApi(dAtA[iNdEx:]) @@ -27459,352 +27646,358 @@ var ( func init() { proto.RegisterFile("roachpb/api.proto", fileDescriptorApi) } var fileDescriptorApi = []byte{ - // 5552 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xdc, 0x5c, 0x4d, 0x6c, 0x23, 0xc9, - 0x75, 0x56, 0x93, 0x94, 0x44, 0x3e, 0x8a, 0x54, 0xab, 0xe6, 0x8f, 0xa3, 0x99, 0x1d, 0x69, 0xb8, - 0xf3, 0xbf, 0xb6, 0x26, 0xa3, 0xf1, 0xc2, 0xce, 0xae, 0xff, 0x44, 0x4a, 0x2b, 0x72, 0xb4, 0x92, - 0x66, 0x8b, 0xd4, 0xac, 0x67, 0x1d, 0xbb, 0xdd, 0xea, 0x2e, 0x51, 0xed, 0x21, 0xbb, 0x39, 0xdd, - 0xcd, 0x91, 0x34, 0x40, 0x90, 0x00, 0x09, 0xe0, 0xc0, 0x30, 0x8c, 0x00, 0x09, 0x82, 0x00, 0x41, - 0x10, 0x03, 0x39, 0xe4, 0x14, 0x27, 0x01, 0x02, 0x04, 0x08, 0x10, 0xfb, 0x92, 0x83, 0x0f, 0x41, - 0xe0, 0x04, 0xb0, 0x11, 0x04, 0x88, 0x90, 0x28, 0x39, 0x38, 0xb9, 0xe6, 0xb6, 0xa7, 0xa0, 0x7e, - 0xfa, 0x87, 0x64, 0x37, 0xc9, 0x19, 0xf7, 0x22, 0x4e, 0x4e, 0x64, 0xbd, 0xaa, 0xf7, 0xaa, 0xea, - 0xd5, 0xab, 0x57, 0x5f, 0x55, 0xbd, 0x6a, 0x58, 0xb0, 0x2d, 0x55, 0x3b, 0xec, 0xee, 0xdf, 0x57, - 0xbb, 0xc6, 0x4a, 0xd7, 0xb6, 0x5c, 0x0b, 0x2d, 0x68, 0x96, 0xf6, 0x8c, 0x91, 0x57, 0x44, 0xe6, - 0x22, 0xf2, 0x4a, 0xe9, 0xaa, 0xab, 0xf2, 0x62, 0x8b, 0xe7, 0x3d, 0x1a, 0xb1, 0x6d, 0xcb, 0x76, - 0x04, 0xf5, 0xa2, 0x47, 0xed, 0x10, 0x57, 0x0d, 0x95, 0x2e, 0x3b, 0xae, 0x65, 0xab, 0x2d, 0x72, - 0x9f, 0x98, 0x2d, 0xc3, 0xf4, 0x7e, 0x68, 0xb9, 0x17, 0x9a, 0x26, 0xca, 0xbc, 0x39, 0xaa, 0xcc, - 0x43, 0x51, 0xa8, 0xd4, 0x73, 0x8d, 0xf6, 0xfd, 0xc3, 0xb6, 0x76, 0xdf, 0x35, 0x3a, 0xc4, 0x71, - 0xd5, 0x4e, 0x57, 0xe4, 0x2c, 0xb3, 0x1c, 0xd7, 0x56, 0x35, 0xc3, 0x6c, 0xdd, 0xb7, 0x89, 0x66, - 0xd9, 0x3a, 0xd1, 0x15, 0xa7, 0xab, 0x9a, 0x5e, 0x93, 0x5b, 0x56, 0xcb, 0x62, 0x7f, 0xef, 0xd3, - 0x7f, 0x9c, 0x5a, 0xfe, 0x35, 0xc8, 0x61, 0xd5, 0x6c, 0x91, 0xba, 0x79, 0x60, 0xa1, 0xcf, 0x43, - 0x46, 0x27, 0x8e, 0x56, 0x92, 0x96, 0xa5, 0x3b, 0xf9, 0xd5, 0xf2, 0xca, 0x90, 0x2e, 0x56, 0x58, - 0xd9, 0x75, 0xe2, 0x68, 0xb6, 0xd1, 0x75, 0x2d, 0xbb, 0x92, 0xf9, 0xd1, 0xe9, 0xd2, 0x14, 0x66, - 0x5c, 0xe8, 0x33, 0x30, 0xdd, 0x26, 0xaa, 0x43, 0x4a, 0x29, 0xc6, 0x5e, 0x8a, 0x60, 0x7f, 0x9f, - 0xe6, 0x0b, 0x26, 0x5e, 0xb8, 0xfc, 0x12, 0x0a, 0x98, 0x3c, 0xef, 0x11, 0xc7, 0xad, 0x11, 0x55, - 0x27, 0x36, 0xba, 0x0c, 0xe9, 0x67, 0xe4, 0xa4, 0x94, 0x5e, 0x96, 0xee, 0xcc, 0x55, 0x66, 0x3f, - 0x3e, 0x5d, 0x4a, 0x6f, 0x91, 0x13, 0x4c, 0x69, 0x68, 0x19, 0x66, 0x89, 0xa9, 0x2b, 0x34, 0x3b, - 0xd3, 0x9f, 0x3d, 0x43, 0x4c, 0x7d, 0x8b, 0x9c, 0xa0, 0x45, 0xc8, 0x3a, 0x54, 0x9a, 0xa9, 0x91, - 0xd2, 0xf4, 0xb2, 0x74, 0x67, 0x1a, 0xfb, 0xe9, 0x77, 0x32, 0x3f, 0xfb, 0xde, 0x92, 0xf4, 0x28, - 0x93, 0x95, 0xe4, 0xd4, 0xa3, 0x4c, 0x36, 0x25, 0xa7, 0xcb, 0xdf, 0x49, 0x43, 0x11, 0x13, 0xa7, - 0x6b, 0x99, 0x0e, 0x11, 0xb5, 0xff, 0x12, 0xa4, 0xdd, 0x63, 0x93, 0xd5, 0x9e, 0x5f, 0xbd, 0x16, - 0xd1, 0x85, 0xa6, 0xad, 0x9a, 0x8e, 0xaa, 0xb9, 0x86, 0x65, 0x62, 0x5a, 0x14, 0x7d, 0x0e, 0xf2, - 0x36, 0x71, 0x7a, 0x1d, 0xc2, 0x94, 0xcd, 0x1a, 0x96, 0x5f, 0xbd, 0x14, 0xc1, 0xd9, 0xe8, 0xaa, - 0x26, 0x06, 0x5e, 0x96, 0xfe, 0x47, 0x97, 0x21, 0x6b, 0xf6, 0x3a, 0xb4, 0x3b, 0x0e, 0x6b, 0x6c, - 0x1a, 0xcf, 0x9a, 0xbd, 0xce, 0x16, 0x39, 0x71, 0x50, 0x15, 0xf2, 0x36, 0x55, 0xb5, 0x62, 0x98, - 0x07, 0x96, 0x53, 0x9a, 0x59, 0x4e, 0xdf, 0xc9, 0xaf, 0x5e, 0x8d, 0x1b, 0x10, 0x3a, 0x78, 0x42, - 0xab, 0x60, 0x7b, 0x04, 0x07, 0x35, 0xa0, 0x20, 0x5a, 0x66, 0x13, 0xd5, 0xb1, 0xcc, 0xd2, 0xec, - 0xb2, 0x74, 0xa7, 0xb8, 0xba, 0x12, 0x25, 0xa6, 0x4f, 0x0b, 0x34, 0xd9, 0xeb, 0x10, 0xcc, 0xb8, - 0xf0, 0x9c, 0x1d, 0x4a, 0x95, 0x9f, 0xc2, 0x5c, 0x38, 0x17, 0x21, 0x28, 0xe2, 0x8d, 0xc6, 0xde, - 0xf6, 0x86, 0xb2, 0xb7, 0xb3, 0xb5, 0xb3, 0xfb, 0xe1, 0x8e, 0x3c, 0x85, 0xce, 0x83, 0x2c, 0x68, - 0x5b, 0x1b, 0x4f, 0x95, 0xf7, 0xeb, 0xdb, 0xf5, 0xa6, 0x2c, 0xa1, 0xcb, 0x70, 0x41, 0x50, 0xf1, - 0xda, 0xce, 0xe6, 0x86, 0x52, 0xd9, 0xdd, 0xdb, 0x59, 0x5f, 0xc3, 0x4f, 0xe5, 0xd4, 0x62, 0xe6, - 0xb7, 0xfe, 0xf8, 0xda, 0x54, 0xf9, 0x09, 0xc0, 0x26, 0x71, 0x85, 0x35, 0xa0, 0x0a, 0xcc, 0x1c, - 0xb2, 0xd6, 0x08, 0x73, 0x5c, 0x8e, 0x6c, 0x76, 0xc8, 0x72, 0x2a, 0x59, 0xaa, 0x81, 0x1f, 0x9f, - 0x2e, 0x49, 0x58, 0x70, 0xf2, 0x21, 0x2f, 0xff, 0x50, 0x82, 0x3c, 0x13, 0xcc, 0xfb, 0x88, 0xaa, - 0x03, 0x92, 0xaf, 0x8f, 0x55, 0xc8, 0xb0, 0x68, 0xb4, 0x02, 0xd3, 0x2f, 0xd4, 0x76, 0x6f, 0x94, - 0xb5, 0x3f, 0xa1, 0xf9, 0x98, 0x17, 0x43, 0xef, 0xc2, 0x9c, 0x61, 0xba, 0xc4, 0x74, 0x15, 0xce, - 0x96, 0x1e, 0xc3, 0x96, 0xe7, 0xa5, 0x59, 0xa2, 0xfc, 0xd7, 0x12, 0xc0, 0xe3, 0x5e, 0x92, 0xaa, - 0xa1, 0xb3, 0x75, 0xa2, 0xf6, 0x7b, 0xb3, 0x95, 0xf7, 0xe2, 0x22, 0xcc, 0x18, 0x66, 0xdb, 0x30, - 0x79, 0xfb, 0xb3, 0x58, 0xa4, 0xd0, 0x79, 0x98, 0xde, 0x6f, 0x1b, 0xa6, 0xce, 0xcc, 0x3f, 0x8b, - 0x79, 0x42, 0xa8, 0x1f, 0x43, 0x9e, 0xb5, 0x3d, 0x41, 0xed, 0x97, 0xff, 0x43, 0x82, 0x0b, 0x55, - 0xcb, 0xd4, 0x0d, 0x3a, 0x0f, 0xd5, 0xf6, 0x2f, 0x84, 0x6e, 0xde, 0x86, 0x1c, 0x39, 0xee, 0x4e, - 0x38, 0xbc, 0x59, 0x72, 0xdc, 0x65, 0xff, 0x46, 0xaa, 0xee, 0x6b, 0x70, 0x71, 0xb0, 0x97, 0x49, - 0x6a, 0xf1, 0x1f, 0x25, 0x28, 0xd6, 0x4d, 0xc3, 0xfd, 0x85, 0x50, 0x9f, 0xaf, 0x87, 0x74, 0x48, - 0x0f, 0xe8, 0x1e, 0xc8, 0x07, 0xaa, 0xd1, 0xde, 0x35, 0x9b, 0x56, 0x67, 0xdf, 0x71, 0x2d, 0x93, - 0x38, 0x42, 0x51, 0x43, 0x74, 0xa1, 0xb3, 0x27, 0x30, 0xef, 0xf7, 0x29, 0x49, 0x65, 0xbd, 0x04, - 0xb9, 0x6e, 0x6a, 0x36, 0xe9, 0x10, 0x33, 0x51, 0x6d, 0x5d, 0x85, 0x9c, 0xe1, 0xc9, 0x65, 0x1a, - 0x4b, 0xe3, 0x80, 0x20, 0xfa, 0xd4, 0x83, 0x85, 0x50, 0xdd, 0x49, 0xba, 0xb1, 0x2b, 0x90, 0x33, - 0xc9, 0x91, 0x12, 0x8c, 0x57, 0x1a, 0x67, 0x4d, 0x72, 0xc4, 0xdd, 0xce, 0x53, 0x28, 0xac, 0x93, - 0x36, 0x71, 0x49, 0xf2, 0x3e, 0x79, 0x0f, 0x8a, 0x9e, 0xe8, 0x24, 0x07, 0xe9, 0x0f, 0x25, 0x40, - 0x42, 0x2e, 0x5d, 0x07, 0x93, 0x1c, 0xa7, 0x25, 0xba, 0xce, 0xbb, 0x3d, 0xdb, 0xe4, 0x0b, 0x36, - 0xb7, 0x52, 0xe0, 0x24, 0xb6, 0x66, 0x07, 0xbe, 0x31, 0x13, 0xf6, 0x8d, 0x3e, 0xee, 0xa0, 0x88, - 0xe3, 0x08, 0xce, 0xf5, 0x35, 0x2f, 0xd9, 0xa1, 0xcc, 0xb0, 0x96, 0xa5, 0x96, 0xd3, 0x61, 0x68, - 0xc4, 0x88, 0xe5, 0xaf, 0xc1, 0x42, 0xb5, 0x4d, 0x54, 0x3b, 0x69, 0xb5, 0x88, 0xe1, 0x7c, 0x0a, - 0x28, 0x2c, 0x3e, 0xc9, 0x21, 0x35, 0x20, 0xdf, 0xd0, 0x54, 0x73, 0xb7, 0x4b, 0x9d, 0xa0, 0x83, - 0x1e, 0xc2, 0x45, 0xc7, 0xb5, 0xba, 0x8a, 0xea, 0x2a, 0x1c, 0x21, 0xed, 0x5b, 0x3d, 0x53, 0x57, - 0xed, 0x13, 0x56, 0x47, 0x16, 0x9f, 0xa3, 0xb9, 0x6b, 0x2e, 0x6b, 0x48, 0x45, 0x64, 0xd1, 0xb1, - 0xeb, 0x18, 0xa6, 0x42, 0x81, 0x4c, 0xdb, 0x75, 0x84, 0x9d, 0x43, 0xc7, 0x30, 0x31, 0xa7, 0x88, - 0x5e, 0x68, 0xbc, 0xaa, 0xc4, 0xd5, 0xc3, 0x07, 0xff, 0x51, 0x26, 0x9b, 0x96, 0x33, 0xe5, 0xbf, - 0x97, 0x60, 0x8e, 0xd7, 0x92, 0xe4, 0xe0, 0xbf, 0x0d, 0x19, 0xdb, 0x3a, 0xe2, 0x83, 0x9f, 0x5f, - 0xbd, 0x12, 0x21, 0x62, 0x8b, 0x9c, 0x84, 0xbd, 0x2e, 0x2b, 0x8e, 0x2a, 0x20, 0x70, 0x86, 0xc2, - 0xb8, 0xd3, 0x93, 0x72, 0x03, 0xe7, 0xc2, 0xd6, 0x91, 0x53, 0xfe, 0x26, 0x20, 0x4c, 0x5e, 0x10, - 0xdb, 0x21, 0x9f, 0xbc, 0xf2, 0x7e, 0x22, 0xc1, 0xb9, 0xbe, 0xca, 0xfe, 0x9f, 0xe8, 0xf0, 0xd7, - 0x25, 0xb8, 0x54, 0x3d, 0x24, 0xda, 0xb3, 0xaa, 0x65, 0x3a, 0x86, 0xe3, 0x12, 0x53, 0x3b, 0x49, - 0xd2, 0x79, 0x5d, 0x81, 0xdc, 0x91, 0xe1, 0x1e, 0x2a, 0xba, 0x71, 0x70, 0xc0, 0xcc, 0x3f, 0x8b, - 0xb3, 0x94, 0xb0, 0x6e, 0x1c, 0x1c, 0x08, 0xe3, 0x57, 0xa0, 0x34, 0xdc, 0x82, 0x64, 0x17, 0xd0, - 0x0b, 0x98, 0x68, 0x56, 0xa7, 0xdb, 0x73, 0x49, 0xc3, 0x55, 0x5d, 0x27, 0xc9, 0x0e, 0x5e, 0x82, - 0x59, 0xdd, 0x3e, 0x51, 0xec, 0x9e, 0x29, 0xba, 0x37, 0xa3, 0xdb, 0x27, 0xb8, 0x67, 0x8a, 0xce, - 0xfd, 0x95, 0x04, 0x17, 0x07, 0x2b, 0x4f, 0xd2, 0x74, 0xbe, 0x02, 0x79, 0x55, 0xa7, 0x1b, 0x6e, - 0x9d, 0xb4, 0x5d, 0x55, 0x00, 0x9f, 0x07, 0x21, 0x49, 0x62, 0x77, 0xbf, 0xc2, 0xb7, 0xf5, 0x2b, - 0xde, 0xee, 0x7e, 0x65, 0xfb, 0x49, 0xb5, 0xca, 0xda, 0xb3, 0x4e, 0x19, 0x3d, 0xcb, 0x60, 0xb2, - 0x18, 0xa5, 0xac, 0xc1, 0xa5, 0x0a, 0x69, 0x19, 0x66, 0x78, 0xdf, 0x99, 0xb8, 0xfb, 0x56, 0xa0, - 0x34, 0x5c, 0x49, 0x92, 0x63, 0xff, 0x0f, 0x69, 0xb8, 0xb0, 0x61, 0xea, 0x9f, 0x4c, 0x27, 0xe8, - 0xca, 0xab, 0x59, 0x9d, 0x8e, 0xe1, 0x7a, 0x63, 0xcf, 0x53, 0xe8, 0x97, 0x21, 0xab, 0x13, 0x55, - 0xf7, 0xf7, 0x2b, 0xf9, 0xd5, 0x37, 0x42, 0xd2, 0x7b, 0xae, 0xd1, 0x5e, 0x39, 0x6c, 0x6b, 0x2b, - 0x4d, 0xef, 0x2c, 0x05, 0xfb, 0xc5, 0xd1, 0x37, 0xe0, 0x12, 0x9d, 0x9e, 0xb6, 0xa9, 0xb6, 0x15, - 0x2e, 0x4d, 0x71, 0x6d, 0xa3, 0xd5, 0x22, 0xb6, 0xd8, 0xe1, 0xdf, 0x89, 0x68, 0x67, 0x5d, 0x70, - 0x54, 0x19, 0x43, 0x93, 0x97, 0xc7, 0x17, 0x8c, 0x28, 0x32, 0xfa, 0xb2, 0xbf, 0x21, 0x74, 0xba, - 0xaa, 0xe9, 0x94, 0xa6, 0x99, 0xdf, 0x88, 0x3b, 0x38, 0x10, 0x96, 0x21, 0x3c, 0x0d, 0xa5, 0x38, - 0xe8, 0x3e, 0x45, 0x24, 0xcf, 0x7b, 0x86, 0x4d, 0x94, 0x07, 0x5d, 0xad, 0x34, 0x43, 0xfb, 0x5e, - 0x29, 0x9e, 0x9d, 0x2e, 0x01, 0xe6, 0xe4, 0x07, 0x8f, 0xab, 0x14, 0xa1, 0xf0, 0xff, 0x5d, 0x0d, - 0xdd, 0x01, 0xd9, 0xb4, 0x14, 0x9b, 0x1c, 0xd8, 0xc4, 0x39, 0x14, 0xd5, 0x66, 0x99, 0xc6, 0x8a, - 0xa6, 0x85, 0x39, 0x99, 0x8b, 0xbe, 0x08, 0x33, 0x5d, 0xcb, 0x70, 0x2c, 0xb3, 0x94, 0xe3, 0x1a, - 0xe5, 0x29, 0xdf, 0x23, 0xcf, 0xca, 0xd9, 0xf2, 0xef, 0x48, 0x70, 0x71, 0x70, 0x4c, 0x93, 0x9c, - 0x53, 0x77, 0x40, 0xb6, 0x4c, 0xa2, 0x74, 0x0f, 0x55, 0x87, 0x88, 0x31, 0x10, 0xc8, 0xaa, 0x68, - 0x99, 0xe4, 0x31, 0x25, 0x73, 0x8d, 0xf6, 0xad, 0x10, 0xbf, 0x21, 0xc1, 0xc2, 0x9a, 0xde, 0x31, - 0xcc, 0x46, 0xb7, 0x6d, 0x24, 0x0a, 0xd4, 0x6f, 0x40, 0xce, 0xa1, 0x32, 0xd9, 0xf9, 0x53, 0xaa, - 0xff, 0xfc, 0x29, 0xcb, 0x72, 0xb6, 0xc8, 0x49, 0x80, 0x87, 0xc2, 0x8d, 0x48, 0x72, 0x2a, 0x7d, - 0x4d, 0xf4, 0x6f, 0x9b, 0xd8, 0x9f, 0x10, 0x92, 0x0b, 0x8b, 0x4f, 0xb2, 0xe5, 0xdf, 0x96, 0xe0, - 0x32, 0x93, 0xcd, 0x4c, 0xe6, 0x80, 0xd8, 0xec, 0x38, 0x30, 0xc9, 0x21, 0x7a, 0x13, 0x66, 0x5c, - 0xd5, 0x6e, 0x11, 0xee, 0x08, 0xa6, 0x2b, 0xf9, 0x8f, 0x4f, 0x97, 0x66, 0x1b, 0xae, 0x65, 0x93, - 0xfa, 0x3a, 0x16, 0x59, 0xa2, 0x9f, 0x2a, 0x2c, 0x46, 0xb5, 0x25, 0xc9, 0xfe, 0xfe, 0xb7, 0x24, - 0xea, 0xa8, 0x1e, 0x72, 0x58, 0xdc, 0x6d, 0x1b, 0x9a, 0x9a, 0xe8, 0xb2, 0xb7, 0x01, 0x79, 0x8d, - 0x09, 0x57, 0xdc, 0x93, 0x2e, 0xdf, 0xc0, 0x15, 0x57, 0x6f, 0x44, 0x0a, 0x62, 0x95, 0xf3, 0x96, - 0x34, 0x4f, 0xba, 0x04, 0x83, 0xe6, 0xff, 0x47, 0xeb, 0x30, 0xcb, 0x95, 0xe3, 0xc1, 0x97, 0x11, - 0x22, 0xe8, 0x44, 0x6f, 0xb2, 0xc2, 0xc2, 0x27, 0x79, 0xac, 0x42, 0xb1, 0xfb, 0x70, 0x25, 0xb2, - 0xd3, 0x49, 0x6a, 0xf6, 0xf7, 0x24, 0x38, 0x57, 0x23, 0xaa, 0xed, 0xee, 0x13, 0xd5, 0x6d, 0x1e, - 0x27, 0xba, 0x98, 0xbc, 0x0d, 0x69, 0xd3, 0x3a, 0x12, 0x4b, 0xf8, 0xe8, 0xf5, 0x42, 0x28, 0x80, - 0x96, 0x17, 0x9d, 0xff, 0x2a, 0x9c, 0xef, 0x6f, 0x57, 0x92, 0xbd, 0xfe, 0x8b, 0x34, 0xe4, 0x36, - 0xab, 0x49, 0xf6, 0xf5, 0xf3, 0x62, 0xcb, 0xc8, 0x07, 0x3d, 0xea, 0xc0, 0xdf, 0xaf, 0x6f, 0x65, - 0xb3, 0xba, 0x45, 0x4e, 0x3c, 0xe0, 0x4b, 0xb9, 0xd0, 0x1a, 0xe4, 0xdc, 0x43, 0xba, 0x66, 0x58, - 0x6d, 0x5d, 0xac, 0x8a, 0x13, 0xe9, 0x2b, 0xe0, 0x42, 0x6d, 0xb8, 0xe0, 0x1e, 0x9b, 0x6c, 0x29, - 0x52, 0x5a, 0x9a, 0x12, 0x88, 0x9b, 0x9e, 0x44, 0xdc, 0x22, 0x15, 0x77, 0x76, 0xba, 0x84, 0x9a, - 0xc7, 0x26, 0x5d, 0xb7, 0x36, 0xab, 0x4d, 0x4f, 0x00, 0x46, 0xae, 0xa0, 0x69, 0x3e, 0x6d, 0xf1, - 0x19, 0x4c, 0xb3, 0x5e, 0x78, 0x77, 0x0c, 0x52, 0xc4, 0x1d, 0x03, 0xed, 0x94, 0x57, 0xc1, 0xab, - 0x18, 0x41, 0xc0, 0xc5, 0x4d, 0x41, 0x18, 0xc4, 0x07, 0x00, 0x54, 0x85, 0x49, 0x9a, 0xc1, 0x0f, - 0xd2, 0x50, 0x7c, 0xdc, 0x73, 0x0e, 0x13, 0xb6, 0xfb, 0x2a, 0x40, 0xb7, 0xe7, 0x1c, 0x12, 0x5b, - 0x71, 0x8f, 0x4d, 0xd1, 0xf3, 0x31, 0x17, 0x20, 0x5e, 0xd7, 0x39, 0x5f, 0xf3, 0xd8, 0x44, 0xbb, - 0x42, 0x08, 0x51, 0x82, 0x5b, 0x94, 0x7b, 0x13, 0xc0, 0xe0, 0xe6, 0xb1, 0xb9, 0x4d, 0x7c, 0xfc, - 0xcb, 0x05, 0x12, 0x2a, 0xf0, 0xf3, 0x30, 0x4b, 0x13, 0x8a, 0x6b, 0xbd, 0x8a, 0x85, 0xcd, 0x50, - 0x9e, 0xa6, 0xe5, 0xcd, 0xe5, 0xe9, 0x57, 0x9b, 0xcb, 0xe8, 0x5d, 0xc8, 0xf1, 0x4a, 0xa9, 0x4f, - 0x9d, 0x61, 0x3e, 0x35, 0x4a, 0x13, 0x62, 0x10, 0x98, 0x37, 0xcd, 0xb2, 0x1a, 0xa9, 0x2f, 0x3d, - 0x0f, 0xd3, 0x07, 0x96, 0xad, 0x11, 0x76, 0xdb, 0x92, 0xc5, 0x3c, 0xe1, 0x03, 0xa7, 0xac, 0x9c, - 0x2b, 0xff, 0x81, 0x04, 0xf3, 0xfe, 0x00, 0x26, 0x89, 0x98, 0xaa, 0x7d, 0xda, 0x7f, 0xf5, 0x21, - 0xa4, 0x1a, 0x2f, 0x7f, 0x3f, 0x05, 0xf3, 0x1f, 0xf4, 0x88, 0x7d, 0x92, 0xb0, 0x7d, 0x55, 0xf8, - 0xcd, 0x5a, 0xea, 0x35, 0x6d, 0x82, 0xdd, 0xb5, 0xdd, 0x82, 0xf9, 0x23, 0xd5, 0x70, 0x95, 0x03, - 0xcb, 0x56, 0x7a, 0x5d, 0x5d, 0x75, 0xbd, 0x7b, 0x88, 0x02, 0x25, 0xbf, 0x67, 0xd9, 0x7b, 0x8c, - 0x88, 0x08, 0xa0, 0x67, 0xa6, 0x75, 0x64, 0x2a, 0x94, 0x6c, 0x98, 0x2d, 0xaa, 0x0f, 0xa7, 0x94, - 0x61, 0x07, 0x63, 0x9f, 0xfd, 0xe7, 0xd3, 0xa5, 0x87, 0x2d, 0xc3, 0x3d, 0xec, 0xed, 0xaf, 0x68, - 0x56, 0xe7, 0xbe, 0xdf, 0x10, 0x7d, 0x3f, 0xf8, 0x7f, 0xbf, 0xfb, 0xac, 0x75, 0x9f, 0x5d, 0xa9, - 0xf6, 0x7a, 0x86, 0xbe, 0xb2, 0xb7, 0x57, 0x5f, 0xc7, 0x32, 0x13, 0xf9, 0x21, 0x97, 0xd8, 0x3c, - 0x36, 0xbd, 0x05, 0xef, 0x63, 0x09, 0xe4, 0x40, 0x61, 0x49, 0x8e, 0xe7, 0x06, 0xe4, 0x9f, 0xf7, - 0x88, 0x6d, 0x10, 0xfd, 0x95, 0x07, 0x14, 0x04, 0x23, 0x9d, 0x43, 0x1f, 0xc1, 0x5c, 0x9f, 0x1e, - 0xd2, 0x3f, 0x9f, 0x1e, 0xf2, 0x47, 0x81, 0x0a, 0xca, 0x3f, 0x48, 0x01, 0x62, 0x9d, 0xaf, 0xf3, - 0xc3, 0x8c, 0x5f, 0x30, 0x83, 0x79, 0x02, 0x60, 0x1c, 0x28, 0x1d, 0xc3, 0x71, 0x0c, 0xb3, 0xc5, - 0x6c, 0xa5, 0xb8, 0xfa, 0xd9, 0x88, 0xb6, 0x0c, 0x77, 0x61, 0xa5, 0x7e, 0xb0, 0xcd, 0xd9, 0x2a, - 0xe4, 0x50, 0x7d, 0x61, 0x58, 0x36, 0xce, 0x19, 0x1e, 0xa9, 0x5c, 0x81, 0x85, 0xa1, 0x7c, 0x54, - 0x04, 0x58, 0xdf, 0x55, 0x76, 0x76, 0x9b, 0xb5, 0xfa, 0xce, 0xa6, 0x3c, 0x85, 0x64, 0x98, 0xc3, - 0x1b, 0xcd, 0x3d, 0xbc, 0xa3, 0x6c, 0x60, 0xbc, 0x8b, 0x65, 0x09, 0xe5, 0x61, 0xf6, 0x31, 0xde, - 0x78, 0xb2, 0xb1, 0xd3, 0x94, 0x53, 0xc2, 0x7a, 0x7e, 0x15, 0xce, 0xf5, 0x55, 0x9e, 0xa4, 0xfd, - 0x5c, 0x87, 0xb9, 0x03, 0xab, 0x67, 0xea, 0x0a, 0xdf, 0x35, 0x8a, 0xdd, 0x71, 0x9e, 0xd1, 0x78, - 0x7d, 0xe5, 0x6f, 0xa5, 0xe0, 0x3c, 0x26, 0x8e, 0xd5, 0x7e, 0x41, 0x92, 0x1f, 0xc1, 0x5d, 0x10, - 0x67, 0x5c, 0xca, 0xcf, 0x33, 0x90, 0x39, 0x2e, 0x83, 0xaf, 0x06, 0x33, 0x8e, 0xab, 0xba, 0x3d, - 0x47, 0x0c, 0xe5, 0x8d, 0xd1, 0x73, 0xa1, 0xc1, 0xca, 0x62, 0xc1, 0x13, 0xda, 0xd4, 0x66, 0x86, - 0x37, 0xb5, 0xe5, 0x5f, 0x81, 0x0b, 0x03, 0x8a, 0x48, 0x72, 0xd1, 0xfe, 0x69, 0x0a, 0x2e, 0xf7, - 0x8b, 0x4f, 0xfa, 0x7e, 0xe2, 0xff, 0x86, 0xb2, 0x51, 0x0d, 0x0a, 0x1d, 0xc3, 0x54, 0x02, 0x8c, - 0xf5, 0x0a, 0x8b, 0xf3, 0x1c, 0xdd, 0xaf, 0xf5, 0xc3, 0x2c, 0xba, 0x8f, 0x8b, 0xd2, 0x6b, 0x92, - 0x63, 0xf7, 0x5d, 0x09, 0xe6, 0x92, 0xde, 0x6d, 0xbf, 0xde, 0x25, 0xa9, 0xe8, 0x73, 0x13, 0x0a, - 0x9f, 0xc0, 0xf6, 0xfc, 0x4f, 0x24, 0x40, 0x4d, 0xbb, 0x67, 0x6a, 0xaa, 0x4b, 0xde, 0xb7, 0x5a, - 0x49, 0x76, 0xf6, 0x3c, 0x4c, 0x1b, 0xa6, 0x4e, 0x8e, 0x59, 0x67, 0x33, 0x98, 0x27, 0xd0, 0x03, - 0xc8, 0x8a, 0x20, 0x17, 0x7e, 0xe9, 0x9b, 0xae, 0x5c, 0x3c, 0x3b, 0x5d, 0x9a, 0xe5, 0x21, 0x2d, - 0xeb, 0x1f, 0x07, 0x7f, 0xf1, 0x2c, 0x8f, 0x6a, 0xf1, 0xae, 0xc5, 0x3f, 0x82, 0x73, 0x7d, 0x0d, - 0x4d, 0x52, 0x0b, 0xdf, 0x4f, 0xc1, 0x39, 0xd1, 0x9d, 0xc4, 0x8f, 0x27, 0x5e, 0x2b, 0x42, 0x0a, - 0x7d, 0x01, 0xa0, 0x6b, 0x93, 0x17, 0x0a, 0x67, 0x4d, 0x4f, 0xc4, 0x9a, 0xa3, 0x1c, 0x8c, 0x80, - 0xbe, 0x02, 0xf3, 0x74, 0xc2, 0x75, 0x6d, 0xab, 0x6b, 0x39, 0x14, 0x49, 0x38, 0x93, 0x21, 0xe9, - 0x85, 0xb3, 0xd3, 0xa5, 0xc2, 0xb6, 0x61, 0x3e, 0x16, 0x8c, 0xcd, 0x06, 0xa6, 0x33, 0xd7, 0x4f, - 0x7a, 0xf0, 0xe7, 0x27, 0x12, 0x9c, 0xff, 0xc4, 0x0e, 0x74, 0xfe, 0x37, 0x34, 0xe6, 0xaf, 0x07, - 0x32, 0x4b, 0xd6, 0xcd, 0x03, 0x2b, 0xf9, 0x63, 0xb6, 0xef, 0x4a, 0xb0, 0x10, 0x12, 0x9f, 0xe4, - 0xaa, 0xff, 0x7a, 0x71, 0x78, 0x5f, 0xa5, 0x38, 0x20, 0x6c, 0xf6, 0x49, 0x4e, 0xaa, 0xdf, 0x4c, - 0xc1, 0xc5, 0x2a, 0xbf, 0x7c, 0x61, 0x77, 0x4c, 0x4e, 0xaf, 0x93, 0xa4, 0x95, 0x94, 0x60, 0xf6, - 0x05, 0xb1, 0x1d, 0xc3, 0xe2, 0xeb, 0x5e, 0x01, 0x7b, 0x49, 0xf4, 0x4d, 0xc8, 0x6b, 0xa2, 0x42, - 0xcf, 0xcb, 0xcc, 0x55, 0xea, 0x54, 0xc0, 0x6b, 0xa2, 0xdf, 0xb3, 0xd3, 0x25, 0xf0, 0xba, 0x50, - 0x5f, 0xc7, 0xe0, 0x49, 0xaf, 0xeb, 0x2c, 0xf6, 0xd0, 0x54, 0xbb, 0xce, 0xa1, 0xe5, 0x9d, 0x53, - 0xfb, 0x69, 0x31, 0xe8, 0x5f, 0x87, 0x4b, 0x43, 0x5a, 0x48, 0x52, 0xcd, 0x7f, 0x37, 0x0b, 0x85, - 0x8d, 0xe3, 0xae, 0x65, 0xbb, 0x0d, 0xbe, 0xd8, 0xa3, 0x75, 0xc8, 0x76, 0x6d, 0xeb, 0x85, 0xe1, - 0x09, 0x2e, 0x46, 0xde, 0x5b, 0xf4, 0xf1, 0x3c, 0x16, 0xe5, 0xb1, 0xcf, 0x89, 0x30, 0xe4, 0xde, - 0xb7, 0x34, 0xb5, 0xfd, 0x9e, 0xd1, 0xf6, 0xac, 0x6a, 0x65, 0x9c, 0x98, 0x15, 0x9f, 0xe3, 0xb1, - 0xea, 0x1e, 0x7a, 0x93, 0xcc, 0x27, 0xa2, 0x4d, 0xc8, 0xd6, 0x5c, 0xb7, 0x4b, 0x33, 0xc5, 0x0c, - 0xbd, 0x39, 0x56, 0x24, 0x65, 0x10, 0x92, 0x7c, 0x66, 0x84, 0x61, 0x61, 0xd3, 0xb2, 0x5a, 0x6d, - 0x52, 0x6d, 0x5b, 0x3d, 0xbd, 0x6a, 0x99, 0x07, 0x46, 0x4b, 0x78, 0xb8, 0x1b, 0x63, 0x25, 0x6e, - 0x56, 0x1b, 0x78, 0x98, 0x1d, 0x7d, 0x09, 0xb2, 0x8d, 0x87, 0x42, 0x14, 0xc7, 0x27, 0x6f, 0x8e, - 0x15, 0xd5, 0x78, 0x88, 0x7d, 0x26, 0x54, 0x83, 0xfc, 0xda, 0xcb, 0x9e, 0x4d, 0x84, 0x8c, 0x19, - 0x26, 0xe3, 0xd6, 0x58, 0x19, 0x8c, 0x07, 0x87, 0x59, 0x17, 0xef, 0x42, 0xa1, 0x4f, 0x93, 0x08, - 0x41, 0xa6, 0x4b, 0x95, 0x46, 0x87, 0x33, 0x87, 0xd9, 0x7f, 0x6e, 0x5e, 0x8b, 0xb7, 0x20, 0x43, - 0xb5, 0x42, 0xa7, 0xc3, 0xbe, 0xea, 0x90, 0x3d, 0xdb, 0x10, 0x85, 0xbc, 0xa4, 0x28, 0xf7, 0xb7, - 0x12, 0xa4, 0x1a, 0x0f, 0x29, 0x42, 0xdb, 0xef, 0x69, 0xcf, 0x88, 0x2b, 0x4a, 0x89, 0x14, 0x43, - 0x6e, 0x36, 0x39, 0x30, 0xf8, 0x6a, 0x9d, 0xc3, 0x22, 0x85, 0xde, 0x00, 0x50, 0x35, 0x8d, 0x38, - 0x8e, 0xe2, 0xc5, 0xe7, 0xe6, 0x70, 0x8e, 0x53, 0xb6, 0xc8, 0x09, 0x65, 0x73, 0x88, 0x66, 0x13, - 0x6e, 0xfc, 0x39, 0x2c, 0x52, 0x94, 0xcd, 0x25, 0x9d, 0xae, 0xe2, 0x5a, 0xcf, 0x88, 0xc9, 0xb4, - 0x99, 0xc3, 0x39, 0x4a, 0x69, 0x52, 0x02, 0x9d, 0x35, 0xc4, 0xd4, 0xbb, 0x96, 0x61, 0xba, 0x4c, - 0x4d, 0x39, 0xec, 0xa7, 0xa9, 0x48, 0x9b, 0xb4, 0x0c, 0x11, 0xb9, 0x9a, 0xc3, 0x22, 0x25, 0xba, - 0xb1, 0x0b, 0xe9, 0xcd, 0x6a, 0xe3, 0x95, 0xbb, 0x81, 0x20, 0xa3, 0xf6, 0x84, 0xd1, 0xe5, 0x30, - 0xfb, 0x2f, 0x04, 0x7e, 0x4b, 0x82, 0x69, 0xa6, 0x7a, 0x74, 0x15, 0x72, 0x9a, 0x65, 0xba, 0xaa, - 0x61, 0x8a, 0x79, 0x93, 0xc3, 0x01, 0x21, 0x56, 0xf2, 0x75, 0x98, 0x53, 0x35, 0xcd, 0xea, 0x99, - 0xae, 0x62, 0xaa, 0x1d, 0x22, 0x6a, 0xc8, 0x0b, 0xda, 0x8e, 0xda, 0x21, 0x68, 0x09, 0xbc, 0xa4, - 0x1f, 0xc5, 0x9c, 0xc3, 0x20, 0x48, 0xfe, 0xf5, 0x91, 0x70, 0x17, 0x7f, 0x2a, 0xc1, 0xc2, 0x87, - 0xb6, 0xe1, 0x92, 0x8a, 0xea, 0x6a, 0x87, 0x49, 0x3a, 0xcc, 0x77, 0x20, 0xa7, 0xab, 0xae, 0xca, - 0x23, 0x96, 0x53, 0x23, 0x23, 0x96, 0xbd, 0xf9, 0x46, 0xcb, 0xb3, 0xa8, 0x65, 0x04, 0x19, 0xfa, - 0x9f, 0xfb, 0x52, 0xcc, 0xfe, 0x07, 0x57, 0x47, 0xe1, 0xe6, 0x26, 0xe9, 0xd9, 0xfe, 0x25, 0xe5, - 0x79, 0xb6, 0x24, 0xd5, 0xf0, 0x65, 0x98, 0x15, 0xbb, 0x22, 0xa1, 0x84, 0xe5, 0x71, 0x33, 0xd4, - 0xbb, 0xf2, 0x10, 0x6c, 0xa8, 0x02, 0xe0, 0xb8, 0xaa, 0xed, 0xb2, 0xfd, 0xcc, 0x44, 0x77, 0xcc, - 0x9e, 0x27, 0x64, 0x6c, 0x94, 0x8a, 0x76, 0x20, 0xdf, 0x79, 0xa1, 0x69, 0xca, 0x81, 0xd1, 0x76, - 0xc5, 0xf5, 0x72, 0xb1, 0x4f, 0x88, 0xd7, 0x92, 0xed, 0x27, 0xd5, 0xea, 0x7b, 0xac, 0x10, 0xbf, - 0xe5, 0x0d, 0xd2, 0x18, 0xa8, 0x04, 0xfe, 0x1f, 0x7d, 0x0a, 0x44, 0x54, 0x9a, 0xe2, 0x38, 0x2e, - 0x9b, 0x70, 0xd9, 0x4a, 0xe1, 0xec, 0x74, 0x29, 0x87, 0x19, 0xb5, 0xd1, 0x68, 0xe2, 0x1c, 0x2f, - 0xd0, 0x70, 0xbc, 0x95, 0xe9, 0x3b, 0x12, 0x14, 0x2a, 0xbd, 0xf6, 0xb3, 0xdd, 0x6e, 0xa3, 0xd7, - 0xe9, 0xa8, 0xf6, 0x09, 0xba, 0xe2, 0x99, 0x88, 0xf1, 0x92, 0x30, 0x15, 0xa7, 0x85, 0x0d, 0x18, - 0x2f, 0x09, 0xb5, 0x01, 0x11, 0x29, 0x43, 0xe9, 0x3c, 0x0c, 0xe6, 0x4d, 0x28, 0x30, 0x58, 0xaf, - 0x10, 0xd3, 0xb5, 0x0d, 0xc2, 0x77, 0x8d, 0x69, 0x3c, 0xc7, 0x88, 0x1b, 0x9c, 0x86, 0x6e, 0x42, - 0xd1, 0x39, 0x71, 0x5c, 0xd2, 0x51, 0xf8, 0x13, 0x05, 0x8e, 0x45, 0xd3, 0xb8, 0xc0, 0xa9, 0x98, - 0x13, 0xcb, 0x7f, 0x9e, 0x86, 0xa2, 0x37, 0xdc, 0x49, 0x42, 0xa3, 0x0a, 0x4c, 0x1f, 0x18, 0x6d, - 0xe2, 0x85, 0xf8, 0xc4, 0x3b, 0x64, 0x4f, 0xd2, 0x0a, 0x75, 0xbb, 0x1e, 0x50, 0x62, 0xac, 0x49, - 0x0c, 0xf9, 0xe2, 0x8f, 0x25, 0xc8, 0xb0, 0x55, 0xf0, 0x01, 0x64, 0xd8, 0x1c, 0x94, 0x26, 0x99, - 0x83, 0xac, 0xa8, 0xef, 0xff, 0x53, 0x81, 0xff, 0x67, 0xbe, 0xf7, 0x50, 0x7d, 0xfb, 0xc1, 0x2a, - 0x1b, 0xee, 0x39, 0x2c, 0x52, 0xa8, 0x02, 0x59, 0xc2, 0xfa, 0x43, 0x74, 0xb1, 0x06, 0x45, 0x59, - 0x78, 0xdf, 0xc0, 0x7b, 0xf3, 0xdd, 0xe3, 0x43, 0x97, 0x21, 0x4d, 0xed, 0x68, 0x96, 0xdf, 0x95, - 0x9c, 0x9d, 0x2e, 0xa5, 0xa9, 0x05, 0x51, 0x1a, 0xbf, 0x71, 0x7f, 0x94, 0xc9, 0x66, 0xe4, 0xe9, - 0xf2, 0x5f, 0x66, 0xa0, 0x50, 0xef, 0x24, 0x3d, 0x43, 0xd7, 0xfa, 0x07, 0x2c, 0x0a, 0x22, 0xf4, - 0x55, 0x1a, 0x31, 0x5e, 0x7d, 0xbe, 0x2e, 0xfd, 0x6a, 0xbe, 0xae, 0x4e, 0x17, 0x20, 0xf1, 0x3e, - 0x83, 0xd6, 0xff, 0xd6, 0xd8, 0xfa, 0x9b, 0xea, 0x7e, 0x9b, 0x60, 0xca, 0xe3, 0x5d, 0x45, 0x70, - 0x01, 0xe8, 0x8b, 0x6c, 0x9d, 0xe3, 0x46, 0x33, 0x33, 0xb9, 0xd1, 0xcc, 0x12, 0x53, 0x67, 0x26, - 0x73, 0x2c, 0x2c, 0xe6, 0x73, 0x90, 0xd6, 0x8d, 0x51, 0x2a, 0x8d, 0xf2, 0x57, 0x94, 0x65, 0x8c, - 0xe1, 0x64, 0xc2, 0x86, 0xe3, 0x5f, 0x57, 0xa4, 0xe5, 0xcc, 0xe2, 0x2e, 0x40, 0xd0, 0x2b, 0xb4, - 0x0c, 0x33, 0x56, 0x5b, 0xa7, 0x60, 0x9a, 0x36, 0xa1, 0x50, 0xc9, 0x9d, 0x9d, 0x2e, 0x4d, 0xef, - 0xb6, 0xf5, 0xfa, 0x3a, 0x9e, 0xb6, 0xda, 0x7a, 0x5d, 0x67, 0xcf, 0x5a, 0xc8, 0x91, 0xc2, 0x5e, - 0x12, 0xb1, 0x30, 0x09, 0x3c, 0x6b, 0x92, 0xa3, 0x75, 0xe2, 0x68, 0xe1, 0xd5, 0x4d, 0x98, 0xcd, - 0x1f, 0x49, 0x50, 0xf4, 0x34, 0x98, 0xec, 0x4c, 0xcf, 0x1a, 0x1d, 0x61, 0xf9, 0xe9, 0x57, 0xb3, - 0x7c, 0x8f, 0x4f, 0x84, 0xec, 0x7e, 0x5b, 0x82, 0x73, 0x3c, 0x96, 0x43, 0x53, 0x5d, 0xea, 0x6b, - 0x13, 0x34, 0xef, 0xbb, 0x20, 0xdb, 0xaa, 0xa9, 0x5b, 0x1d, 0xe3, 0x25, 0xe1, 0xbb, 0x55, 0x47, - 0x1c, 0xd2, 0xce, 0xfb, 0x74, 0xb6, 0x1d, 0xf3, 0x36, 0xdb, 0xff, 0x25, 0xc1, 0xf9, 0xfe, 0xc6, - 0x24, 0xa9, 0xb4, 0x2d, 0x98, 0x61, 0x07, 0x2d, 0xde, 0x74, 0xfb, 0x74, 0x84, 0x90, 0xa8, 0xda, - 0xf9, 0x2b, 0x24, 0xdf, 0xe0, 0x99, 0x88, 0xc5, 0x2f, 0xc3, 0x34, 0x23, 0xbf, 0x86, 0x8f, 0x13, - 0x9a, 0x7f, 0x0e, 0x0b, 0x6b, 0xba, 0xde, 0x68, 0x08, 0xeb, 0x4b, 0x4e, 0xed, 0x1e, 0x84, 0x49, - 0x45, 0x41, 0x98, 0x70, 0x95, 0x49, 0x42, 0x98, 0x2e, 0x14, 0x45, 0x88, 0x55, 0xc2, 0x27, 0x6b, - 0x47, 0x14, 0x73, 0x09, 0xb3, 0xe1, 0x89, 0xe0, 0x25, 0x84, 0x5f, 0x63, 0x92, 0x3d, 0xe9, 0xc1, - 0x39, 0x4f, 0x6e, 0xd2, 0x87, 0xd8, 0xa3, 0xba, 0xc3, 0x4e, 0x28, 0xc2, 0xd5, 0x26, 0xd9, 0xa7, - 0x3f, 0x93, 0x60, 0x71, 0x93, 0xb8, 0x0d, 0xb1, 0x61, 0x7f, 0xcf, 0xb2, 0x13, 0x3f, 0xf1, 0xdd, - 0x04, 0x68, 0x93, 0x03, 0x11, 0xb6, 0x2e, 0x00, 0xe7, 0xe4, 0x6f, 0x2c, 0x73, 0x94, 0x97, 0x65, - 0x09, 0x75, 0xfc, 0x54, 0x82, 0x2b, 0x91, 0x2d, 0x4e, 0xd2, 0x23, 0x44, 0xcc, 0x14, 0xf4, 0x75, - 0x60, 0x68, 0x53, 0x71, 0x5c, 0xd5, 0x75, 0x84, 0x73, 0xfd, 0xd4, 0xab, 0x84, 0xba, 0x56, 0x16, - 0x44, 0xdc, 0x46, 0xce, 0x27, 0xe1, 0x1c, 0x15, 0xc9, 0xfe, 0x96, 0xbf, 0x7b, 0x09, 0xe6, 0x84, - 0x2e, 0xf7, 0x4c, 0xc3, 0x32, 0xd1, 0x03, 0x48, 0xb7, 0xc4, 0xf6, 0x2e, 0x1f, 0x09, 0x8c, 0x83, - 0x57, 0x83, 0xb5, 0x29, 0x4c, 0xcb, 0x52, 0x96, 0x6e, 0xcf, 0x8d, 0x88, 0xdf, 0x08, 0xee, 0xee, - 0xc3, 0x2c, 0xdd, 0x9e, 0x8b, 0x1a, 0x30, 0xaf, 0x05, 0x6f, 0xad, 0x14, 0xca, 0x9e, 0x8e, 0x8d, - 0xf4, 0x8c, 0x7c, 0x7b, 0x56, 0x9b, 0xc2, 0x45, 0xad, 0x2f, 0x03, 0x55, 0xc3, 0x8f, 0x7b, 0x32, - 0xb1, 0x27, 0x09, 0x83, 0x0f, 0x8b, 0x6a, 0x53, 0xa1, 0x37, 0x40, 0xe8, 0x1d, 0x98, 0xd1, 0xd9, - 0xa3, 0x11, 0x71, 0x16, 0x11, 0x65, 0x7c, 0x7d, 0xef, 0x74, 0x6a, 0x53, 0x58, 0x70, 0xa0, 0x47, - 0x30, 0xc7, 0xff, 0x09, 0xb3, 0x9b, 0x89, 0x3d, 0x6a, 0x19, 0x7e, 0x36, 0x53, 0x9b, 0xc2, 0x79, - 0x3d, 0xa0, 0xa2, 0xcf, 0x40, 0xc6, 0xd1, 0x54, 0xbe, 0x19, 0x8f, 0xbe, 0x87, 0x0e, 0x3d, 0x00, - 0xa8, 0x51, 0x5f, 0xae, 0xa9, 0x26, 0x7a, 0x0a, 0x0b, 0xfb, 0xa4, 0x65, 0x98, 0x8a, 0x1b, 0xdc, - 0x15, 0xb1, 0xa8, 0xd3, 0xfe, 0xeb, 0x29, 0x7f, 0x49, 0x8e, 0x0e, 0x76, 0xae, 0x4d, 0x61, 0x79, - 0x7f, 0x20, 0x8b, 0x0e, 0x19, 0xc3, 0x54, 0x21, 0xc1, 0xb9, 0xd8, 0x21, 0x8b, 0x0c, 0x3f, 0xa6, - 0x43, 0x46, 0xfa, 0x32, 0xd0, 0x26, 0xe4, 0x55, 0xba, 0xc6, 0x29, 0x2c, 0xa4, 0xb3, 0x04, 0xb1, - 0x27, 0x49, 0x43, 0x51, 0xa6, 0x35, 0x16, 0xb9, 0xed, 0x11, 0x03, 0x41, 0x1d, 0x3a, 0x2f, 0x4b, - 0xf9, 0xd1, 0x82, 0xc2, 0xee, 0xc6, 0x17, 0xc4, 0x88, 0x68, 0x1b, 0x0a, 0x87, 0x5e, 0x50, 0x19, - 0xbb, 0xdc, 0x9b, 0x8b, 0x3d, 0x4e, 0x8a, 0x08, 0x8a, 0xab, 0x4d, 0xe1, 0xb9, 0xc3, 0x10, 0x19, - 0xad, 0x40, 0xaa, 0xa5, 0x95, 0x0a, 0x4c, 0xc6, 0xd5, 0x51, 0x21, 0x5f, 0xb5, 0x29, 0x9c, 0x6a, - 0x69, 0x14, 0xb9, 0xf2, 0x68, 0x98, 0x63, 0xb3, 0x54, 0x8c, 0x75, 0x25, 0xfd, 0x11, 0x49, 0xb5, - 0x29, 0xcc, 0xe2, 0x76, 0x68, 0x7d, 0x8f, 0xa1, 0x68, 0xf3, 0x1b, 0x3a, 0xef, 0x1e, 0x5a, 0x66, - 0x52, 0x6e, 0x47, 0x3b, 0xa4, 0xa1, 0xab, 0xe8, 0xda, 0x14, 0x2e, 0xd8, 0x61, 0x3a, 0xfa, 0x06, - 0x9c, 0xef, 0x97, 0x28, 0x8c, 0x7b, 0x61, 0xc8, 0x17, 0x45, 0xcb, 0xed, 0xb7, 0x71, 0x64, 0x0f, - 0x65, 0xa2, 0xcf, 0xc2, 0x34, 0x1f, 0x35, 0xc4, 0x44, 0x2e, 0x45, 0xed, 0xc6, 0xfb, 0x07, 0x8c, - 0x97, 0xa7, 0xf3, 0xcd, 0x15, 0x57, 0x53, 0x4a, 0xdb, 0x6a, 0x95, 0xce, 0xc5, 0xce, 0xb7, 0xe1, - 0xab, 0x36, 0x3a, 0xdf, 0xdc, 0x80, 0x4a, 0xc7, 0xdd, 0xe6, 0x39, 0xe2, 0x26, 0xe3, 0x7c, 0xec, - 0xb8, 0x47, 0xdc, 0x58, 0xd1, 0x71, 0xb7, 0x43, 0x64, 0xda, 0x34, 0x9b, 0x3f, 0x9d, 0x51, 0xd8, - 0x34, 0xbe, 0x10, 0xdb, 0xb4, 0xe1, 0xe7, 0x3c, 0xb4, 0x69, 0x76, 0x40, 0x45, 0x4f, 0x40, 0x16, - 0x8f, 0x29, 0x14, 0xef, 0x04, 0xbc, 0x74, 0x91, 0xc9, 0xbb, 0x1b, 0xe9, 0x2d, 0xa3, 0x8e, 0xfe, - 0x6b, 0x53, 0x78, 0x5e, 0xeb, 0xcf, 0xa1, 0xce, 0x82, 0xc9, 0x53, 0xb4, 0xe0, 0x15, 0x4a, 0xa9, - 0x14, 0xeb, 0x2c, 0x62, 0x9e, 0xcc, 0x50, 0x67, 0xa1, 0x0d, 0x64, 0x51, 0x33, 0x36, 0x4c, 0xc3, - 0x65, 0x8e, 0x7d, 0x31, 0xd6, 0x8c, 0xfb, 0x9f, 0xc3, 0x52, 0x33, 0x36, 0x38, 0x85, 0x9a, 0xb1, - 0x2b, 0xae, 0xb9, 0xc4, 0x70, 0x5c, 0x8d, 0x35, 0xe3, 0xa8, 0xfb, 0x30, 0x6a, 0xc6, 0x6e, 0x98, - 0x4e, 0xcd, 0x98, 0x3b, 0x88, 0x01, 0xb9, 0x6f, 0xc4, 0x9a, 0x71, 0x6c, 0xf4, 0x34, 0x35, 0x63, - 0x75, 0x28, 0x13, 0xad, 0x53, 0xc8, 0xa1, 0x3a, 0xfc, 0x33, 0x02, 0xa5, 0x6b, 0xb1, 0xeb, 0xcf, - 0xe0, 0x45, 0x57, 0x8d, 0xe1, 0x0d, 0x41, 0xa3, 0x8e, 0x8c, 0xe1, 0x30, 0x65, 0x5f, 0x75, 0xb5, - 0xc3, 0xd2, 0x52, 0xac, 0x23, 0x1b, 0x3a, 0xac, 0xa4, 0x8e, 0xec, 0xc8, 0x27, 0xd2, 0x85, 0x8c, - 0x1f, 0x2b, 0x94, 0x96, 0xc7, 0x6c, 0x5f, 0x43, 0x0b, 0x19, 0xe7, 0x40, 0x6b, 0x90, 0x7b, 0xde, - 0x23, 0xf6, 0x09, 0x73, 0x43, 0xd7, 0x63, 0xc1, 0xd3, 0x40, 0xe4, 0x5a, 0x6d, 0x0a, 0x67, 0x9f, - 0x0b, 0x12, 0xad, 0x9e, 0xef, 0xed, 0x4a, 0xe5, 0xd8, 0xea, 0xfb, 0x76, 0xf3, 0xb4, 0x7a, 0xce, - 0x81, 0x34, 0xb8, 0xc0, 0xc7, 0x4a, 0x84, 0x5b, 0xdb, 0x22, 0xae, 0xb9, 0xf4, 0x26, 0x13, 0x15, - 0xbb, 0x53, 0x8a, 0x0c, 0xfd, 0xae, 0x4d, 0xe1, 0x73, 0xea, 0x70, 0x2e, 0x9d, 0xf0, 0x62, 0xe9, - 0xe1, 0xfb, 0xab, 0xd2, 0x8d, 0xd8, 0x09, 0x1f, 0xb1, 0x23, 0xa5, 0x13, 0x5e, 0x0d, 0x91, 0xf9, - 0x02, 0xa4, 0x2b, 0x8e, 0xe3, 0xd2, 0xdd, 0x4c, 0xe9, 0xe6, 0x88, 0x05, 0x68, 0x60, 0x97, 0xc5, - 0x17, 0x20, 0xbd, 0xc1, 0x39, 0xa9, 0x20, 0xad, 0x4d, 0x54, 0x5b, 0xb8, 0xd9, 0x5b, 0xb1, 0x82, - 0x86, 0x9e, 0x98, 0x52, 0x41, 0x9a, 0x4f, 0xa4, 0x0b, 0xb6, 0xed, 0xbd, 0xc2, 0x12, 0xf8, 0xf1, - 0x76, 0xec, 0x82, 0x1d, 0xf9, 0x58, 0x8c, 0x2e, 0xd8, 0x76, 0x5f, 0x06, 0xfa, 0x02, 0xcc, 0x8a, - 0x27, 0x2d, 0xa5, 0x3b, 0x23, 0x90, 0x6e, 0x78, 0xeb, 0x45, 0xe7, 0xb5, 0xe0, 0xe1, 0x5e, 0x96, - 0xbf, 0x88, 0xe1, 0xdd, 0xbb, 0x3b, 0xc2, 0xcb, 0x0e, 0xed, 0x7a, 0xb8, 0x97, 0x0d, 0xc8, 0xd4, - 0xcb, 0x72, 0x3b, 0x15, 0x6b, 0xdd, 0xbd, 0x58, 0x2f, 0x3b, 0x1c, 0x73, 0x46, 0xbd, 0xec, 0xf3, - 0x80, 0x8a, 0x74, 0xb8, 0xd8, 0x22, 0xae, 0xe2, 0xdd, 0x22, 0xb2, 0xb0, 0x47, 0xbe, 0x2c, 0xbd, - 0x15, 0x6b, 0x75, 0xf1, 0x9b, 0x18, 0x6a, 0x75, 0xad, 0xe1, 0xdc, 0xca, 0xac, 0x88, 0x44, 0x79, - 0x94, 0xc9, 0xce, 0xcb, 0xf2, 0xa3, 0x4c, 0xf6, 0x92, 0x5c, 0x7a, 0x94, 0xc9, 0x5e, 0x96, 0x17, - 0x1f, 0x65, 0xb2, 0x57, 0xe4, 0xab, 0xe5, 0xff, 0xbc, 0x08, 0x05, 0x6f, 0xa7, 0xc0, 0x11, 0xf9, - 0x6a, 0x18, 0x91, 0x5f, 0x8b, 0x43, 0xe4, 0x62, 0x6f, 0x21, 0x20, 0xf9, 0x6a, 0x18, 0x92, 0x5f, - 0x8b, 0x83, 0xe4, 0x01, 0x0f, 0xc5, 0xe4, 0xcd, 0x38, 0x4c, 0x7e, 0x77, 0x02, 0x4c, 0xee, 0x8b, - 0x1a, 0x04, 0xe5, 0xeb, 0xc3, 0xa0, 0xfc, 0xc6, 0x68, 0x50, 0xee, 0x8b, 0x0a, 0xa1, 0xf2, 0x77, - 0x07, 0x50, 0xf9, 0xf5, 0x11, 0xa8, 0xdc, 0xe7, 0xf7, 0x60, 0xf9, 0x56, 0x24, 0x2c, 0xbf, 0x35, - 0x0e, 0x96, 0xfb, 0x72, 0xfa, 0x70, 0xf9, 0xdb, 0x7d, 0xb8, 0x7c, 0x29, 0x16, 0x97, 0xfb, 0xdc, - 0x1c, 0x98, 0x7f, 0x14, 0x0f, 0xcc, 0xdf, 0x9a, 0x08, 0x98, 0xfb, 0xf2, 0x86, 0x91, 0x79, 0x33, - 0x0e, 0x99, 0xdf, 0x9d, 0x00, 0x99, 0x07, 0x03, 0x37, 0x00, 0xcd, 0x6b, 0x51, 0xd0, 0xfc, 0xe6, - 0x18, 0x68, 0xee, 0x4b, 0x0b, 0x63, 0xf3, 0x5a, 0x14, 0x36, 0xbf, 0x39, 0x06, 0x9b, 0x0f, 0x48, - 0xe2, 0xe0, 0x7c, 0x27, 0x1a, 0x9c, 0xdf, 0x1e, 0x0b, 0xce, 0x7d, 0x69, 0xfd, 0xe8, 0xfc, 0x7e, - 0x08, 0x9d, 0xbf, 0x11, 0x83, 0xce, 0x7d, 0x56, 0x0a, 0xcf, 0xbf, 0x34, 0x04, 0xcf, 0xcb, 0xa3, - 0xe0, 0xb9, 0xcf, 0xeb, 0xe3, 0xf3, 0x0f, 0x62, 0xf0, 0xf9, 0x9d, 0xf1, 0xf8, 0xdc, 0x17, 0x36, - 0x00, 0xd0, 0xd5, 0x91, 0x00, 0xfd, 0xd3, 0x13, 0x02, 0x74, 0x5f, 0x7a, 0x14, 0x42, 0xff, 0x5c, - 0x3f, 0x42, 0x5f, 0x8e, 0x47, 0xe8, 0xbe, 0x18, 0x01, 0xd1, 0xb7, 0x22, 0x21, 0xfa, 0xad, 0x71, - 0x10, 0x3d, 0x98, 0x7b, 0x61, 0x8c, 0xbe, 0x13, 0x8d, 0xd1, 0x6f, 0x8f, 0xc5, 0xe8, 0xc1, 0xf0, - 0xf7, 0x81, 0xf4, 0xad, 0x48, 0x90, 0x7e, 0x6b, 0x1c, 0x48, 0x0f, 0x1a, 0x17, 0x46, 0xe9, 0x1f, - 0xc6, 0xa2, 0xf4, 0x7b, 0x93, 0xa0, 0x74, 0x5f, 0xe8, 0x10, 0x4c, 0xff, 0x28, 0x1e, 0xa6, 0xbf, - 0x35, 0x11, 0x4c, 0x0f, 0x5c, 0xc7, 0x10, 0x4e, 0xff, 0xd2, 0x10, 0x4e, 0x2f, 0x8f, 0xc2, 0xe9, - 0x81, 0x3d, 0x7b, 0x40, 0x5d, 0x1d, 0x09, 0xab, 0x3f, 0x3d, 0x21, 0xac, 0x0e, 0x8c, 0x2f, 0x02, - 0x57, 0x6f, 0x44, 0xe0, 0xea, 0x1b, 0xa3, 0x71, 0x75, 0xb0, 0x84, 0x04, 0xc0, 0xba, 0x16, 0x05, - 0xac, 0x6f, 0x8e, 0x01, 0xd6, 0x81, 0x17, 0x0a, 0x21, 0xeb, 0x77, 0x07, 0x90, 0xf5, 0xf5, 0xb1, - 0x37, 0x9b, 0x21, 0x68, 0x5d, 0x19, 0x86, 0xd6, 0x6f, 0x8e, 0x84, 0xd6, 0xbe, 0x84, 0x00, 0x5b, - 0xbf, 0x3b, 0x80, 0xad, 0xaf, 0x8f, 0xc0, 0xd6, 0x41, 0x03, 0x04, 0xb8, 0xd6, 0x47, 0x83, 0xeb, - 0x95, 0x49, 0xc1, 0xb5, 0x2f, 0x38, 0x12, 0x5d, 0xef, 0x44, 0xa3, 0xeb, 0xdb, 0x13, 0x5e, 0x72, - 0x0c, 0xc1, 0xeb, 0x5a, 0x14, 0xbc, 0xbe, 0x39, 0x06, 0x5e, 0x87, 0xd7, 0x10, 0x1f, 0x5f, 0xd7, - 0xa2, 0xf0, 0xf5, 0xcd, 0x31, 0xf8, 0x3a, 0x90, 0x14, 0x02, 0xd8, 0xcd, 0x38, 0x80, 0x7d, 0x77, - 0x02, 0x80, 0x1d, 0xac, 0xbb, 0x03, 0x08, 0xfb, 0x8b, 0x83, 0x08, 0xbb, 0x3c, 0x0a, 0x61, 0x07, - 0x33, 0xd2, 0x83, 0xd8, 0x3b, 0xd1, 0x10, 0xfb, 0xf6, 0x58, 0x88, 0x1d, 0x76, 0x92, 0x21, 0x8c, - 0xbd, 0x15, 0x89, 0xb1, 0x6f, 0x8d, 0xc3, 0xd8, 0x81, 0x93, 0x0c, 0x83, 0x6c, 0x32, 0x06, 0x64, - 0xaf, 0x4c, 0x0a, 0xb2, 0x03, 0xeb, 0x7b, 0x1d, 0x94, 0xfd, 0x28, 0x93, 0xbd, 0x2a, 0xbf, 0x51, - 0xfe, 0x9b, 0x69, 0x98, 0xa9, 0x79, 0x77, 0xdf, 0xa1, 0x97, 0x88, 0xd2, 0xeb, 0xbc, 0x44, 0x44, - 0xeb, 0x74, 0xe0, 0x98, 0xf9, 0x0b, 0xdc, 0x3d, 0xe2, 0x5d, 0xef, 0xd0, 0x8d, 0x83, 0xc7, 0xfa, - 0x1a, 0x71, 0xda, 0xe8, 0x6d, 0x28, 0xf4, 0x1c, 0x62, 0x2b, 0x5d, 0xdb, 0xb0, 0x6c, 0xc3, 0xe5, - 0x91, 0x4e, 0x52, 0x45, 0xfe, 0xf8, 0x74, 0x69, 0x6e, 0xcf, 0x21, 0xf6, 0x63, 0x41, 0xc7, 0x73, - 0xbd, 0x50, 0xca, 0xfb, 0xfa, 0xe2, 0xf4, 0xe4, 0x5f, 0x5f, 0xfc, 0x00, 0x64, 0x9b, 0xa8, 0x7a, - 0xdf, 0x42, 0xc4, 0x5f, 0xec, 0x45, 0xaf, 0x99, 0xaa, 0x1e, 0x5a, 0x6b, 0xd8, 0xcb, 0xbd, 0x79, - 0xbb, 0x9f, 0x88, 0x1e, 0xc0, 0x85, 0x8e, 0x7a, 0xcc, 0xdf, 0xa4, 0x7a, 0x6b, 0x3b, 0x8b, 0x01, - 0xc8, 0xb2, 0x50, 0x15, 0xd4, 0x51, 0x8f, 0xd9, 0xa7, 0x1c, 0x79, 0x16, 0xfb, 0xf4, 0xd3, 0x4d, - 0x28, 0xea, 0x86, 0xe3, 0x1a, 0xa6, 0xe6, 0x7d, 0xcd, 0x81, 0x7f, 0x36, 0xa1, 0xe0, 0x51, 0xf9, - 0x57, 0x15, 0xee, 0xc1, 0x82, 0x88, 0xcc, 0x09, 0x3e, 0xee, 0xc8, 0x50, 0x6c, 0x96, 0xb6, 0x82, - 0x66, 0x04, 0xdf, 0xe2, 0xac, 0xc2, 0x7c, 0x4b, 0x75, 0xc9, 0x91, 0x7a, 0xa2, 0x98, 0x96, 0xce, - 0x74, 0x9f, 0x67, 0x6f, 0xda, 0xaf, 0x9c, 0x9d, 0x2e, 0x15, 0x36, 0x79, 0xd6, 0x8e, 0xa5, 0xf3, - 0x11, 0x98, 0xe1, 0xff, 0x70, 0xa1, 0x15, 0xca, 0xd0, 0xd1, 0x1a, 0xcc, 0x51, 0x14, 0xa1, 0x58, - 0xfc, 0xe3, 0x49, 0x02, 0x9b, 0xc6, 0x9d, 0xdc, 0x8b, 0x4f, 0x2c, 0xe1, 0xbc, 0x13, 0xfa, 0xde, - 0xd2, 0x6d, 0x98, 0x57, 0x9d, 0x13, 0x53, 0x63, 0x1a, 0x26, 0xa6, 0xd3, 0x73, 0x18, 0x38, 0xcd, - 0xe2, 0x22, 0x23, 0x57, 0x3d, 0xaa, 0xf8, 0x28, 0xc4, 0xef, 0x4a, 0x30, 0xd7, 0x17, 0xae, 0xf6, - 0xee, 0xc0, 0x35, 0xd4, 0xe5, 0x68, 0x60, 0x1c, 0x17, 0xfe, 0x91, 0x15, 0x23, 0xe0, 0x5d, 0x49, - 0x2f, 0xc5, 0x03, 0x2b, 0xb6, 0x35, 0xf5, 0x2e, 0xf1, 0x3d, 0xb6, 0x77, 0x32, 0xbf, 0xff, 0xbd, - 0xa5, 0xa9, 0xf2, 0xcf, 0xd2, 0x50, 0xe8, 0x0f, 0x4b, 0xab, 0x0f, 0xb4, 0x2b, 0xca, 0x19, 0xf5, - 0x71, 0xc4, 0xb7, 0x72, 0x1d, 0x72, 0xb6, 0x28, 0xe4, 0x35, 0x73, 0x79, 0xc4, 0x65, 0x5b, 0xb8, - 0x9d, 0x01, 0xe3, 0xe2, 0x0f, 0x53, 0xfe, 0xcc, 0x5f, 0x81, 0x69, 0xf6, 0x5d, 0x59, 0xd1, 0xb4, - 0xa8, 0x08, 0xee, 0x0d, 0x9a, 0x8f, 0x79, 0x31, 0xea, 0x29, 0x9a, 0xaf, 0xf5, 0x66, 0xd9, 0x27, - 0xbc, 0xc6, 0x77, 0x4f, 0x5f, 0xf3, 0x6d, 0x2d, 0xbb, 0x66, 0x6b, 0xb7, 0x89, 0xe6, 0x8a, 0xcf, - 0xd3, 0x7a, 0x5f, 0x37, 0xbd, 0x31, 0x28, 0x42, 0x7c, 0xcc, 0x76, 0x05, 0x8b, 0x8f, 0xd9, 0x86, - 0xa2, 0x04, 0x8a, 0xbe, 0x08, 0x36, 0xb1, 0x78, 0x2c, 0x09, 0x1f, 0xea, 0x7b, 0x0d, 0x38, 0x17, - 0x31, 0xcd, 0x51, 0x11, 0xa0, 0xba, 0xbb, 0xd3, 0xa8, 0x37, 0x9a, 0x1b, 0x3b, 0x4d, 0xef, 0x1b, - 0xa5, 0x6b, 0xeb, 0xca, 0xde, 0x4e, 0x75, 0x77, 0x7b, 0xbb, 0xde, 0x6c, 0x6e, 0xac, 0xcb, 0x12, - 0x92, 0x61, 0xae, 0xbe, 0x13, 0x2a, 0x27, 0x3e, 0x4d, 0x7a, 0xef, 0x43, 0xc8, 0x87, 0x5e, 0xfb, - 0x22, 0x04, 0xc5, 0xc7, 0x7b, 0x8d, 0x9a, 0xd2, 0xac, 0x6f, 0x6f, 0x34, 0x9a, 0x6b, 0xdb, 0x8f, - 0xe5, 0x29, 0x5a, 0x01, 0xa3, 0xad, 0x55, 0x76, 0x71, 0x53, 0x96, 0xfc, 0x74, 0x73, 0x77, 0xaf, - 0x5a, 0x93, 0x53, 0x7e, 0xfa, 0x83, 0xbd, 0x0d, 0xfc, 0x54, 0x4e, 0x0b, 0xc1, 0x2a, 0x5c, 0x88, - 0x8c, 0xbe, 0x46, 0x79, 0x98, 0xdd, 0x33, 0xd9, 0x8b, 0x53, 0x79, 0x0a, 0x15, 0x42, 0x01, 0xd8, - 0xb2, 0x84, 0xb2, 0x3c, 0xd0, 0x57, 0x4e, 0xa1, 0x19, 0x48, 0x35, 0x1e, 0xca, 0x69, 0x34, 0x0f, - 0xf9, 0x50, 0x14, 0xb3, 0x9c, 0x41, 0x39, 0x11, 0xca, 0x2a, 0x4f, 0xdf, 0xbb, 0x0e, 0xa1, 0x48, - 0x41, 0x04, 0x30, 0xf3, 0xbe, 0xea, 0x12, 0xc7, 0x95, 0xa7, 0xd0, 0x2c, 0xa4, 0xd7, 0xda, 0x6d, - 0x59, 0x5a, 0xfd, 0x0a, 0x64, 0xbd, 0x6f, 0xd7, 0xa0, 0xf7, 0x61, 0x9a, 0x63, 0xca, 0xa5, 0xf8, - 0x19, 0xc1, 0xe6, 0xd6, 0xe2, 0xf2, 0xb8, 0x29, 0x53, 0x9e, 0xaa, 0x5c, 0xff, 0xd1, 0xbf, 0x5d, - 0x9b, 0xfa, 0xd1, 0xd9, 0x35, 0xe9, 0xc7, 0x67, 0xd7, 0xa4, 0x7f, 0x3a, 0xbb, 0x26, 0xfd, 0xeb, - 0xd9, 0x35, 0xe9, 0xb7, 0xff, 0xfd, 0xda, 0xd4, 0x47, 0xb3, 0x82, 0x65, 0x7f, 0x86, 0x7d, 0x89, - 0xf8, 0xe1, 0xff, 0x04, 0x00, 0x00, 0xff, 0xff, 0xce, 0x1c, 0xb4, 0x9c, 0x8e, 0x59, 0x00, 0x00, + // 5637 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x5c, 0x5d, 0x6c, 0x23, 0x4b, + 0x56, 0x76, 0xdb, 0x4e, 0x62, 0x1f, 0xc7, 0x4e, 0xa7, 0xe6, 0xcf, 0x93, 0x99, 0x3b, 0xc9, 0xf8, + 0xce, 0xff, 0xdd, 0xcd, 0x30, 0x99, 0xbd, 0xec, 0x72, 0xef, 0xfe, 0xc5, 0x4e, 0x6e, 0x9c, 0xc9, + 0xe4, 0xe7, 0x96, 0x9d, 0xb9, 0x3b, 0x77, 0xd9, 0xed, 0xed, 0x74, 0x57, 0x9c, 0x66, 0xec, 0x6e, + 0x4f, 0x77, 0x7b, 0x92, 0x8c, 0x84, 0x40, 0x02, 0x69, 0xd1, 0x6a, 0xb5, 0x42, 0x02, 0x21, 0x24, + 0x84, 0x58, 0x69, 0x1f, 0x90, 0x90, 0x58, 0x40, 0x42, 0x42, 0x42, 0x62, 0xf7, 0x85, 0x87, 0x7d, + 0xe0, 0x61, 0x41, 0x62, 0x85, 0x90, 0x88, 0x20, 0xf0, 0xb0, 0xf0, 0x8a, 0x78, 0xb9, 0xe2, 0x01, + 0xd5, 0x4f, 0xff, 0x38, 0xee, 0xb6, 0x3d, 0xb3, 0x7d, 0xc5, 0x45, 0x3c, 0xd9, 0x7d, 0xaa, 0xce, + 0xa9, 0xaa, 0x53, 0xa7, 0x4e, 0x7d, 0x55, 0x75, 0xaa, 0x60, 0xd6, 0xb6, 0x54, 0xed, 0xa0, 0xbb, + 0x77, 0x5f, 0xed, 0x1a, 0x8b, 0x5d, 0xdb, 0x72, 0x2d, 0x34, 0xab, 0x59, 0xda, 0x33, 0x46, 0x5e, + 0x14, 0x89, 0x73, 0xc8, 0xcb, 0xa5, 0xab, 0xae, 0xca, 0xb3, 0xcd, 0x9d, 0xf7, 0x68, 0xc4, 0xb6, + 0x2d, 0xdb, 0x11, 0xd4, 0x8b, 0x1e, 0xb5, 0x43, 0x5c, 0x35, 0x94, 0xbb, 0xe2, 0xb8, 0x96, 0xad, + 0xb6, 0xc8, 0x7d, 0x62, 0xb6, 0x0c, 0xd3, 0xfb, 0xa1, 0xf9, 0x5e, 0x68, 0x9a, 0xc8, 0xf3, 0xe6, + 0xb0, 0x3c, 0x0f, 0x45, 0xa6, 0x72, 0xcf, 0x35, 0xda, 0xf7, 0x0f, 0xda, 0xda, 0x7d, 0xd7, 0xe8, + 0x10, 0xc7, 0x55, 0x3b, 0x5d, 0x91, 0xb2, 0xc0, 0x52, 0x5c, 0x5b, 0xd5, 0x0c, 0xb3, 0x75, 0xdf, + 0x26, 0x9a, 0x65, 0xeb, 0x44, 0x57, 0x9c, 0xae, 0x6a, 0x7a, 0x55, 0x6e, 0x59, 0x2d, 0x8b, 0xfd, + 0xbd, 0x4f, 0xff, 0x71, 0x6a, 0xe5, 0x57, 0x20, 0x8f, 0x55, 0xb3, 0x45, 0xd6, 0xcd, 0x7d, 0x0b, + 0x7d, 0x1e, 0xb2, 0x3a, 0x71, 0xb4, 0xb2, 0xb4, 0x20, 0xdd, 0x29, 0x2c, 0x55, 0x16, 0x07, 0x74, + 0xb1, 0xc8, 0xf2, 0xae, 0x10, 0x47, 0xb3, 0x8d, 0xae, 0x6b, 0xd9, 0xd5, 0xec, 0x8f, 0x4e, 0xe6, + 0x53, 0x98, 0x71, 0xa1, 0xcf, 0xc0, 0x44, 0x9b, 0xa8, 0x0e, 0x29, 0xa7, 0x19, 0x7b, 0x39, 0x82, + 0xfd, 0x31, 0x4d, 0x17, 0x4c, 0x3c, 0x73, 0xe5, 0x25, 0x14, 0x31, 0x79, 0xde, 0x23, 0x8e, 0x5b, + 0x27, 0xaa, 0x4e, 0x6c, 0x74, 0x19, 0x32, 0xcf, 0xc8, 0x71, 0x39, 0xb3, 0x20, 0xdd, 0x99, 0xae, + 0x4e, 0x7d, 0x74, 0x32, 0x9f, 0xd9, 0x20, 0xc7, 0x98, 0xd2, 0xd0, 0x02, 0x4c, 0x11, 0x53, 0x57, + 0x68, 0x72, 0xb6, 0x3f, 0x79, 0x92, 0x98, 0xfa, 0x06, 0x39, 0x46, 0x73, 0x90, 0x73, 0xa8, 0x34, + 0x53, 0x23, 0xe5, 0x89, 0x05, 0xe9, 0xce, 0x04, 0xf6, 0xbf, 0xdf, 0xc9, 0xfe, 0xf4, 0xbb, 0xf3, + 0xd2, 0xa3, 0x6c, 0x4e, 0x92, 0xd3, 0x8f, 0xb2, 0xb9, 0xb4, 0x9c, 0xa9, 0x7c, 0x3b, 0x03, 0x25, + 0x4c, 0x9c, 0xae, 0x65, 0x3a, 0x44, 0x94, 0xfe, 0x73, 0x90, 0x71, 0x8f, 0x4c, 0x56, 0x7a, 0x61, + 0xe9, 0x5a, 0x44, 0x13, 0x9a, 0xb6, 0x6a, 0x3a, 0xaa, 0xe6, 0x1a, 0x96, 0x89, 0x69, 0x56, 0xf4, + 0x39, 0x28, 0xd8, 0xc4, 0xe9, 0x75, 0x08, 0x53, 0x36, 0xab, 0x58, 0x61, 0xe9, 0x52, 0x04, 0x67, + 0xa3, 0xab, 0x9a, 0x18, 0x78, 0x5e, 0xfa, 0x1f, 0x5d, 0x86, 0x9c, 0xd9, 0xeb, 0xd0, 0xe6, 0x38, + 0xac, 0xb2, 0x19, 0x3c, 0x65, 0xf6, 0x3a, 0x1b, 0xe4, 0xd8, 0x41, 0x35, 0x28, 0xd8, 0x54, 0xd5, + 0x8a, 0x61, 0xee, 0x5b, 0x4e, 0x79, 0x72, 0x21, 0x73, 0xa7, 0xb0, 0x74, 0x35, 0xae, 0x43, 0x68, + 0xe7, 0x09, 0xad, 0x82, 0xed, 0x11, 0x1c, 0xd4, 0x80, 0xa2, 0xa8, 0x99, 0x4d, 0x54, 0xc7, 0x32, + 0xcb, 0x53, 0x0b, 0xd2, 0x9d, 0xd2, 0xd2, 0x62, 0x94, 0x98, 0x3e, 0x2d, 0xd0, 0xcf, 0x5e, 0x87, + 0x60, 0xc6, 0x85, 0xa7, 0xed, 0xd0, 0x57, 0xe5, 0x29, 0x4c, 0x87, 0x53, 0x11, 0x82, 0x12, 0x5e, + 0x6d, 0xec, 0x6e, 0xae, 0x2a, 0xbb, 0x5b, 0x1b, 0x5b, 0xdb, 0x1f, 0x6c, 0xc9, 0x29, 0x74, 0x1e, + 0x64, 0x41, 0xdb, 0x58, 0x7d, 0xaa, 0x3c, 0x5e, 0xdf, 0x5c, 0x6f, 0xca, 0x12, 0xba, 0x0c, 0x17, + 0x04, 0x15, 0x2f, 0x6f, 0xad, 0xad, 0x2a, 0xd5, 0xed, 0xdd, 0xad, 0x95, 0x65, 0xfc, 0x54, 0x4e, + 0xcf, 0x65, 0x7f, 0xe3, 0x7b, 0xd7, 0x52, 0x95, 0x27, 0x00, 0x6b, 0xc4, 0x15, 0xd6, 0x80, 0xaa, + 0x30, 0x79, 0xc0, 0x6a, 0x23, 0xcc, 0x71, 0x21, 0xb2, 0xda, 0x21, 0xcb, 0xa9, 0xe6, 0xa8, 0x06, + 0x7e, 0x7c, 0x32, 0x2f, 0x61, 0xc1, 0xc9, 0xbb, 0xbc, 0xf2, 0x43, 0x09, 0x0a, 0x4c, 0x30, 0x6f, + 0x23, 0xaa, 0x9d, 0x91, 0x7c, 0x7d, 0xa4, 0x42, 0x06, 0x45, 0xa3, 0x45, 0x98, 0x78, 0xa1, 0xb6, + 0x7b, 0xc3, 0xac, 0xfd, 0x09, 0x4d, 0xc7, 0x3c, 0x1b, 0x7a, 0x17, 0xa6, 0x0d, 0xd3, 0x25, 0xa6, + 0xab, 0x70, 0xb6, 0xcc, 0x08, 0xb6, 0x02, 0xcf, 0xcd, 0x3e, 0x2a, 0x7f, 0x29, 0x01, 0xec, 0xf4, + 0x92, 0x54, 0x0d, 0x1d, 0xad, 0x63, 0xd5, 0xdf, 0x1b, 0xad, 0xbc, 0x15, 0x17, 0x61, 0xd2, 0x30, + 0xdb, 0x86, 0xc9, 0xeb, 0x9f, 0xc3, 0xe2, 0x0b, 0x9d, 0x87, 0x89, 0xbd, 0xb6, 0x61, 0xea, 0xcc, + 0xfc, 0x73, 0x98, 0x7f, 0x08, 0xf5, 0x63, 0x28, 0xb0, 0xba, 0x27, 0xa8, 0xfd, 0xca, 0xbf, 0x49, + 0x70, 0xa1, 0x66, 0x99, 0xba, 0x41, 0xc7, 0xa1, 0xda, 0xfe, 0x44, 0xe8, 0xe6, 0x6d, 0xc8, 0x93, + 0xa3, 0xee, 0x98, 0xdd, 0x9b, 0x23, 0x47, 0x5d, 0xf6, 0x6f, 0xa8, 0xea, 0xbe, 0x06, 0x17, 0xcf, + 0xb6, 0x32, 0x49, 0x2d, 0xfe, 0x9d, 0x04, 0xa5, 0x75, 0xd3, 0x70, 0x3f, 0x11, 0xea, 0xf3, 0xf5, + 0x90, 0x09, 0xe9, 0x01, 0xdd, 0x03, 0x79, 0x5f, 0x35, 0xda, 0xdb, 0x66, 0xd3, 0xea, 0xec, 0x39, + 0xae, 0x65, 0x12, 0x47, 0x28, 0x6a, 0x80, 0x2e, 0x74, 0xf6, 0x04, 0x66, 0xfc, 0x36, 0x25, 0xa9, + 0xac, 0x97, 0x20, 0xaf, 0x9b, 0x9a, 0x4d, 0x3a, 0xc4, 0x4c, 0x54, 0x5b, 0x57, 0x21, 0x6f, 0x78, + 0x72, 0x99, 0xc6, 0x32, 0x38, 0x20, 0x88, 0x36, 0xf5, 0x60, 0x36, 0x54, 0x76, 0x92, 0x6e, 0xec, + 0x0a, 0xe4, 0x4d, 0x72, 0xa8, 0x04, 0xfd, 0x95, 0xc1, 0x39, 0x93, 0x1c, 0x72, 0xb7, 0xf3, 0x14, + 0x8a, 0x2b, 0xa4, 0x4d, 0x5c, 0x92, 0xbc, 0x4f, 0xde, 0x85, 0x92, 0x27, 0x3a, 0xc9, 0x4e, 0xfa, + 0x7d, 0x09, 0x90, 0x90, 0x4b, 0xe7, 0xc1, 0x24, 0xfb, 0x69, 0x9e, 0xce, 0xf3, 0x6e, 0xcf, 0x36, + 0xf9, 0x84, 0xcd, 0xad, 0x14, 0x38, 0x89, 0xcd, 0xd9, 0x81, 0x6f, 0xcc, 0x86, 0x7d, 0xa3, 0x8f, + 0x3b, 0x28, 0xe2, 0x38, 0x84, 0x73, 0x7d, 0xd5, 0x4b, 0xb6, 0x2b, 0xb3, 0xac, 0x66, 0xe9, 0x85, + 0x4c, 0x18, 0x1a, 0x31, 0x62, 0xe5, 0x6b, 0x30, 0x5b, 0x6b, 0x13, 0xd5, 0x4e, 0x5a, 0x2d, 0xa2, + 0x3b, 0x9f, 0x02, 0x0a, 0x8b, 0x4f, 0xb2, 0x4b, 0x0d, 0x28, 0x34, 0x34, 0xd5, 0xdc, 0xee, 0x52, + 0x27, 0xe8, 0xa0, 0x87, 0x70, 0xd1, 0x71, 0xad, 0xae, 0xa2, 0xba, 0x0a, 0x47, 0x48, 0x7b, 0x56, + 0xcf, 0xd4, 0x55, 0xfb, 0x98, 0x95, 0x91, 0xc3, 0xe7, 0x68, 0xea, 0xb2, 0xcb, 0x2a, 0x52, 0x15, + 0x49, 0xb4, 0xef, 0x3a, 0x86, 0xa9, 0x50, 0x20, 0xd3, 0x76, 0x1d, 0x61, 0xe7, 0xd0, 0x31, 0x4c, + 0xcc, 0x29, 0xa2, 0x15, 0xdf, 0x93, 0x78, 0x59, 0x49, 0x9a, 0xcd, 0x17, 0xa1, 0xe0, 0x68, 0xaa, + 0xa9, 0xec, 0x5b, 0x76, 0x47, 0x75, 0x99, 0x69, 0x94, 0x96, 0xde, 0x88, 0x82, 0x87, 0x9a, 0x6a, + 0xbe, 0xc7, 0x32, 0x61, 0x70, 0xfc, 0xff, 0x61, 0xeb, 0x79, 0x94, 0xcd, 0x65, 0xe4, 0x6c, 0xe5, + 0xbf, 0x24, 0x98, 0xe6, 0xb5, 0x4c, 0xd2, 0x7a, 0xde, 0x86, 0xac, 0x6d, 0x1d, 0x72, 0xeb, 0x29, + 0x2c, 0x5d, 0x89, 0x10, 0xb1, 0x41, 0x8e, 0xc3, 0x6e, 0x9b, 0x65, 0x47, 0x55, 0x10, 0x40, 0x45, + 0x61, 0xdc, 0x99, 0x71, 0xb9, 0x81, 0x73, 0x61, 0x2a, 0xe3, 0x26, 0x94, 0xf6, 0x54, 0x57, 0x3b, + 0xa0, 0xfd, 0xc3, 0x2a, 0xc9, 0xd1, 0x3d, 0x2e, 0x32, 0xaa, 0x57, 0xf3, 0xca, 0x1f, 0x49, 0x80, + 0x30, 0x79, 0x41, 0x6c, 0x87, 0x7c, 0xf2, 0x3b, 0xe9, 0xbf, 0x25, 0x38, 0xd7, 0x57, 0xd9, 0xff, + 0x5f, 0x7d, 0xf5, 0xab, 0x12, 0x5c, 0xaa, 0x1d, 0x10, 0xed, 0x59, 0xcd, 0x32, 0x1d, 0xc3, 0x71, + 0x89, 0xa9, 0x1d, 0x27, 0xd9, 0x61, 0x57, 0x20, 0x7f, 0x68, 0xb8, 0x07, 0x8a, 0x6e, 0xec, 0xef, + 0xb3, 0xe1, 0x9c, 0xc3, 0x39, 0x4a, 0x58, 0x31, 0xf6, 0xf7, 0xc5, 0x60, 0x56, 0xa0, 0x3c, 0x58, + 0x83, 0x64, 0x01, 0xc1, 0x05, 0x4c, 0x34, 0xab, 0xd3, 0xed, 0xb9, 0xa4, 0xe1, 0xaa, 0xae, 0x93, + 0x64, 0x03, 0x2f, 0xc1, 0x94, 0x6e, 0x1f, 0x2b, 0x76, 0xcf, 0x14, 0xcd, 0x9b, 0xd4, 0xed, 0x63, + 0xdc, 0x33, 0x45, 0xe3, 0xfe, 0x42, 0x82, 0x8b, 0x67, 0x0b, 0x4f, 0xd2, 0xc2, 0xbe, 0x02, 0x05, + 0x55, 0xd7, 0x89, 0xae, 0xe8, 0xa4, 0xed, 0xaa, 0x02, 0xc8, 0x3d, 0x08, 0x49, 0x12, 0xbb, 0x15, + 0x8b, 0x7c, 0x9b, 0x62, 0xd1, 0xdb, 0xad, 0x58, 0xdc, 0x7c, 0x52, 0xab, 0xb1, 0xfa, 0xac, 0x50, + 0x46, 0xcf, 0x80, 0x98, 0x2c, 0x46, 0xa9, 0x68, 0x70, 0xa9, 0x4a, 0x5a, 0x86, 0x19, 0x5e, 0x47, + 0x27, 0x3e, 0x1d, 0x29, 0x50, 0x1e, 0x2c, 0x24, 0xc9, 0xbe, 0xff, 0xdb, 0x0c, 0x5c, 0x58, 0x35, + 0xf5, 0x8f, 0xa7, 0x11, 0x14, 0x49, 0x68, 0x56, 0xa7, 0x63, 0xb8, 0x5e, 0xdf, 0xf3, 0x2f, 0xf4, + 0x0b, 0x90, 0xd3, 0x89, 0xaa, 0xfb, 0xeb, 0xaf, 0x42, 0x9f, 0x8f, 0xea, 0xb9, 0x46, 0x7b, 0xf1, + 0xa0, 0xad, 0x2d, 0x36, 0xbd, 0xbd, 0x21, 0xec, 0x67, 0x47, 0xdf, 0x80, 0x4b, 0x74, 0x14, 0xdb, + 0xa6, 0xda, 0x56, 0xb8, 0x34, 0xc5, 0xb5, 0x8d, 0x56, 0x8b, 0xd8, 0x62, 0xc7, 0xe2, 0x4e, 0x44, + 0x3d, 0xd7, 0x05, 0x47, 0x8d, 0x31, 0x34, 0x79, 0x7e, 0x7c, 0xc1, 0x88, 0x22, 0xa3, 0x2f, 0xfb, + 0x0b, 0x5c, 0xa7, 0xab, 0x9a, 0x4e, 0x79, 0x82, 0xb9, 0x97, 0xb8, 0x8d, 0x10, 0x61, 0x19, 0xc2, + 0x21, 0x51, 0x8a, 0x83, 0xee, 0x53, 0x84, 0xf5, 0xbc, 0x67, 0xd8, 0x44, 0x79, 0xd0, 0xd5, 0xca, + 0x93, 0xb4, 0xed, 0xd5, 0xd2, 0xe9, 0xc9, 0x3c, 0x60, 0x4e, 0x7e, 0xb0, 0x53, 0xa3, 0x88, 0x8b, + 0xff, 0xef, 0x6a, 0xe8, 0x0e, 0xc8, 0xa6, 0xa5, 0xd8, 0x64, 0xdf, 0x26, 0xce, 0x81, 0x28, 0x36, + 0xc7, 0x34, 0x56, 0x32, 0x2d, 0xcc, 0xc9, 0x5c, 0xf4, 0x45, 0x98, 0xec, 0x5a, 0x86, 0x63, 0x99, + 0xe5, 0x3c, 0xd7, 0x28, 0xff, 0xf2, 0x1d, 0xf7, 0x94, 0x9c, 0xab, 0xfc, 0x96, 0x04, 0x17, 0xcf, + 0xf6, 0x69, 0x92, 0x63, 0xea, 0x0e, 0xc8, 0x96, 0x49, 0x94, 0xee, 0x81, 0xea, 0x10, 0xd1, 0x07, + 0x02, 0x29, 0x96, 0x2c, 0x93, 0xec, 0x50, 0x32, 0xd7, 0x68, 0xdf, 0x44, 0xf2, 0x6b, 0x12, 0xcc, + 0x2e, 0xeb, 0x1d, 0xc3, 0x6c, 0x74, 0xdb, 0x46, 0xa2, 0x0b, 0x8f, 0x1b, 0x90, 0x77, 0xa8, 0x4c, + 0xb6, 0x9f, 0x96, 0xee, 0xdf, 0x4f, 0xcb, 0xb1, 0x94, 0x0d, 0x72, 0x1c, 0xe0, 0xbb, 0x70, 0x25, + 0x92, 0x1c, 0x4a, 0x5f, 0x13, 0xed, 0xdb, 0x24, 0xf6, 0xc7, 0x84, 0x4c, 0xc3, 0xe2, 0x93, 0xac, + 0xf9, 0xb7, 0x24, 0xb8, 0xcc, 0x64, 0x33, 0x93, 0xd9, 0x27, 0x36, 0xdb, 0xde, 0x4c, 0xb2, 0x8b, + 0xde, 0x84, 0x49, 0x57, 0xb5, 0x5b, 0x84, 0x3b, 0x82, 0x89, 0x6a, 0xe1, 0xa3, 0x93, 0xf9, 0xa9, + 0x86, 0x6b, 0xd9, 0x64, 0x7d, 0x05, 0x8b, 0x24, 0xd1, 0x4e, 0x15, 0xe6, 0xa2, 0xea, 0x92, 0x64, + 0x7b, 0xff, 0x53, 0x12, 0x65, 0xd4, 0x0e, 0x38, 0xcc, 0xef, 0xb6, 0x0d, 0x4d, 0x4d, 0x74, 0xda, + 0x5b, 0x85, 0x82, 0xc6, 0x84, 0x2b, 0xee, 0x71, 0x97, 0x2f, 0x48, 0x4b, 0x4b, 0x37, 0x22, 0x05, + 0xb1, 0xc2, 0x79, 0x4d, 0x9a, 0xc7, 0x5d, 0x82, 0x41, 0xf3, 0xff, 0xa3, 0x15, 0x98, 0xe2, 0xca, + 0xf1, 0x50, 0xce, 0x10, 0x11, 0x74, 0xa0, 0x37, 0x59, 0x66, 0xe1, 0x93, 0x3c, 0x56, 0xa1, 0xd8, + 0x3d, 0xb8, 0x12, 0xd9, 0xe8, 0x24, 0x35, 0xfb, 0x3b, 0x12, 0x9c, 0xab, 0x13, 0xd5, 0x76, 0xf7, + 0x88, 0xea, 0x36, 0x8f, 0x12, 0x9d, 0x4c, 0xde, 0x86, 0x8c, 0x69, 0x1d, 0x8a, 0x29, 0x7c, 0xf8, + 0x7c, 0x21, 0x14, 0x40, 0xf3, 0x8b, 0xc6, 0x7f, 0x15, 0xce, 0xf7, 0xd7, 0x2b, 0xc9, 0x56, 0xff, + 0x59, 0x06, 0xf2, 0x6b, 0xb5, 0x24, 0xdb, 0xfa, 0x79, 0xb1, 0x04, 0xe6, 0x9d, 0x1e, 0x75, 0x80, + 0xe1, 0x97, 0xb7, 0xb8, 0x56, 0xdb, 0x20, 0xc7, 0x1e, 0x3e, 0xa6, 0x5c, 0x68, 0x19, 0xf2, 0xee, + 0x01, 0x9d, 0x33, 0xac, 0xb6, 0x2e, 0x66, 0xc5, 0xb1, 0xf4, 0x15, 0x70, 0xa1, 0x36, 0x5c, 0x70, + 0x8f, 0x4c, 0x36, 0x15, 0x29, 0x2d, 0x4d, 0x09, 0xc4, 0x4d, 0x8c, 0x23, 0x6e, 0x8e, 0x8a, 0x3b, + 0x3d, 0x99, 0x47, 0xcd, 0x23, 0x93, 0xce, 0x5b, 0x6b, 0xb5, 0xa6, 0x27, 0x00, 0x23, 0x57, 0xd0, + 0x34, 0x9f, 0x36, 0xf7, 0x0c, 0x26, 0x58, 0x2b, 0xbc, 0x33, 0x13, 0x29, 0xe2, 0xcc, 0x84, 0x36, + 0xca, 0x2b, 0xe0, 0x55, 0x8c, 0x20, 0xe0, 0xe2, 0xa6, 0x20, 0x0c, 0xe2, 0x7d, 0x00, 0xaa, 0xc2, + 0x24, 0xcd, 0xe0, 0x07, 0x19, 0x28, 0xed, 0xf4, 0x9c, 0x83, 0x84, 0xed, 0xbe, 0x06, 0xd0, 0xed, + 0x39, 0x07, 0xc4, 0x56, 0xdc, 0x23, 0x53, 0xb4, 0x7c, 0xc4, 0x81, 0x8e, 0xd7, 0x74, 0xce, 0xd7, + 0x3c, 0x32, 0xd1, 0xb6, 0x10, 0x42, 0x94, 0xe0, 0x54, 0xe8, 0xde, 0x18, 0x30, 0xb8, 0x79, 0x64, + 0x6e, 0x12, 0x1f, 0xff, 0x72, 0x81, 0x84, 0x0a, 0xfc, 0x3c, 0x4c, 0xd1, 0x0f, 0xc5, 0xb5, 0x5e, + 0xc5, 0xc2, 0x26, 0x29, 0x4f, 0xd3, 0xf2, 0xc6, 0xf2, 0xc4, 0xab, 0x8d, 0x65, 0xf4, 0x2e, 0xe4, + 0x79, 0xa1, 0xd4, 0xa7, 0x4e, 0x32, 0x9f, 0x1a, 0xa5, 0x09, 0xd1, 0x09, 0xcc, 0x9b, 0xe6, 0x58, + 0x89, 0xd4, 0x97, 0x9e, 0x87, 0x89, 0x7d, 0xcb, 0xd6, 0x08, 0x3b, 0x3d, 0xca, 0x61, 0xfe, 0xe1, + 0x03, 0xa7, 0x9c, 0x9c, 0xaf, 0xfc, 0x9e, 0x04, 0x33, 0x7e, 0x07, 0x26, 0x89, 0x98, 0x6a, 0x7d, + 0xda, 0x7f, 0xf5, 0x2e, 0xa4, 0x1a, 0xaf, 0x7c, 0x3f, 0x0d, 0x33, 0xef, 0xf7, 0x88, 0x7d, 0x9c, + 0xb0, 0x7d, 0x55, 0xf9, 0x49, 0x61, 0xfa, 0x35, 0x6d, 0x82, 0x9d, 0x1d, 0xde, 0x82, 0x99, 0x43, + 0xd5, 0x70, 0x95, 0x7d, 0xcb, 0x56, 0x7a, 0x5d, 0x5d, 0x75, 0xbd, 0x73, 0x95, 0x22, 0x25, 0xbf, + 0x67, 0xd9, 0xbb, 0x8c, 0x88, 0x08, 0xa0, 0x67, 0xa6, 0x75, 0x68, 0x2a, 0x94, 0x6c, 0x98, 0x2d, + 0xaa, 0x0f, 0xa7, 0x9c, 0x65, 0x1b, 0x7d, 0x9f, 0xfd, 0xc7, 0x93, 0xf9, 0x87, 0x2d, 0xc3, 0x3d, + 0xe8, 0xed, 0x2d, 0x6a, 0x56, 0xe7, 0xbe, 0x5f, 0x11, 0x7d, 0x2f, 0xf8, 0x7f, 0xbf, 0xfb, 0xac, + 0x75, 0x9f, 0x1d, 0x11, 0xf7, 0x7a, 0x86, 0xbe, 0xb8, 0xbb, 0xbb, 0xbe, 0x82, 0x65, 0x26, 0xf2, + 0x03, 0x2e, 0xb1, 0x79, 0x64, 0x7a, 0x13, 0xde, 0x47, 0x12, 0xc8, 0x81, 0xc2, 0x92, 0xec, 0xcf, + 0x55, 0x28, 0x3c, 0xef, 0x11, 0xdb, 0x20, 0xfa, 0x2b, 0x77, 0x28, 0x08, 0x46, 0x3a, 0x86, 0x3e, + 0x84, 0xe9, 0x3e, 0x3d, 0x64, 0x7e, 0x36, 0x3d, 0x14, 0x0e, 0x03, 0x15, 0x54, 0x7e, 0x90, 0x06, + 0xc4, 0x1a, 0xbf, 0xce, 0xf7, 0x3c, 0x3e, 0x61, 0x06, 0xf3, 0x04, 0xc0, 0xd8, 0x57, 0x3a, 0x86, + 0xe3, 0x18, 0x66, 0x8b, 0xd9, 0x4a, 0x69, 0xe9, 0xb3, 0x11, 0x75, 0x19, 0x6c, 0xc2, 0xe2, 0xfa, + 0xfe, 0x26, 0x67, 0xab, 0x92, 0x03, 0xf5, 0x85, 0x61, 0xd9, 0x38, 0x6f, 0x78, 0xa4, 0x4a, 0x15, + 0x66, 0x07, 0xd2, 0x51, 0x09, 0x60, 0x65, 0x5b, 0xd9, 0xda, 0x6e, 0xd6, 0xd7, 0xb7, 0xd6, 0xe4, + 0x14, 0x92, 0x61, 0x1a, 0xaf, 0x36, 0x77, 0xf1, 0x96, 0xb2, 0x8a, 0xf1, 0x36, 0x96, 0x25, 0x54, + 0x80, 0xa9, 0x1d, 0xbc, 0xfa, 0x64, 0x75, 0xab, 0x29, 0xa7, 0x85, 0xf5, 0xfc, 0x32, 0x9c, 0xeb, + 0x2b, 0x3c, 0x49, 0xfb, 0xb9, 0x0e, 0xd3, 0xfb, 0x56, 0xcf, 0xd4, 0x15, 0xbe, 0x6a, 0x14, 0xab, + 0xe3, 0x02, 0xa3, 0xf1, 0xf2, 0x2a, 0xdf, 0x4c, 0xc3, 0x79, 0x4c, 0x1c, 0xab, 0xfd, 0x82, 0x24, + 0xdf, 0x83, 0xdb, 0x20, 0xb6, 0xc2, 0x94, 0x9f, 0xa5, 0x23, 0xf3, 0x5c, 0x06, 0x9f, 0x0d, 0x26, + 0x1d, 0x57, 0x75, 0x7b, 0x8e, 0xe8, 0xca, 0x1b, 0xc3, 0xc7, 0x42, 0x83, 0xe5, 0xc5, 0x82, 0x27, + 0xb4, 0xa8, 0xcd, 0x0e, 0x2e, 0x6a, 0x2b, 0xbf, 0x08, 0x17, 0xce, 0x28, 0x22, 0xc9, 0x49, 0xfb, + 0x27, 0x69, 0xb8, 0xdc, 0x2f, 0x3e, 0xe9, 0xf3, 0x96, 0xff, 0x1b, 0xca, 0x46, 0x75, 0x28, 0x76, + 0x0c, 0x53, 0x09, 0x30, 0xd6, 0x2b, 0x4c, 0xce, 0xd3, 0x74, 0xbd, 0xd6, 0x0f, 0xb3, 0xe8, 0x3a, + 0x2e, 0x4a, 0xaf, 0x49, 0xf6, 0xdd, 0x77, 0x24, 0x98, 0x4e, 0x7a, 0xb5, 0xfd, 0x7a, 0x87, 0xbe, + 0xa2, 0xcd, 0x4d, 0x28, 0x7e, 0x0c, 0xcb, 0xf3, 0x3f, 0x94, 0x00, 0x35, 0xed, 0x9e, 0xa9, 0xa9, + 0x2e, 0x79, 0x6c, 0xb5, 0x92, 0x6c, 0xec, 0x79, 0x98, 0x30, 0x4c, 0x9d, 0x1c, 0xb1, 0xc6, 0x66, + 0x31, 0xff, 0x40, 0x0f, 0x20, 0x27, 0x82, 0x76, 0xf8, 0x21, 0x76, 0xa6, 0x7a, 0xf1, 0xf4, 0x64, + 0x7e, 0x8a, 0x87, 0xe8, 0xac, 0x7c, 0x14, 0xfc, 0xc5, 0x53, 0x3c, 0x4a, 0xc7, 0x3b, 0xe6, 0xff, + 0x10, 0xce, 0xf5, 0x55, 0x34, 0x49, 0x2d, 0x7c, 0x3f, 0x0d, 0xe7, 0x44, 0x73, 0x12, 0xdf, 0x9e, + 0x78, 0xad, 0x88, 0x2f, 0xf4, 0x05, 0x80, 0xae, 0x4d, 0x5e, 0x28, 0x9c, 0x35, 0x33, 0x16, 0x6b, + 0x9e, 0x72, 0x30, 0x02, 0xfa, 0x0a, 0xcc, 0xd0, 0x01, 0xd7, 0xb5, 0xad, 0xae, 0xe5, 0x50, 0x24, + 0xe1, 0x8c, 0x87, 0xa4, 0x67, 0x4f, 0x4f, 0xe6, 0x8b, 0x9b, 0x86, 0xb9, 0x23, 0x18, 0x9b, 0x0d, + 0x4c, 0x47, 0xae, 0xff, 0xe9, 0xc1, 0x9f, 0xbf, 0x97, 0xe0, 0xfc, 0xc7, 0xb6, 0xa1, 0xf3, 0xbf, + 0xa1, 0x31, 0x7f, 0x3e, 0x90, 0xd9, 0xe7, 0xba, 0xb9, 0x6f, 0x25, 0xbf, 0xcd, 0xf6, 0x1d, 0x09, + 0x66, 0x43, 0xe2, 0x93, 0x9c, 0xf5, 0x5f, 0x2f, 0xae, 0xf0, 0xab, 0x14, 0x07, 0x84, 0xcd, 0x3e, + 0xc9, 0x41, 0xf5, 0xeb, 0x69, 0xb8, 0x58, 0xe3, 0x87, 0x2f, 0xec, 0x8c, 0xc9, 0xe9, 0x75, 0x92, + 0xb4, 0x92, 0x32, 0x4c, 0xbd, 0x20, 0xb6, 0x63, 0x58, 0x7c, 0xde, 0x2b, 0x62, 0xef, 0x13, 0xfd, + 0x12, 0x14, 0x34, 0x51, 0xa0, 0xe7, 0x65, 0xa6, 0xab, 0xeb, 0x54, 0xc0, 0x6b, 0xa2, 0xdf, 0xd3, + 0x93, 0x79, 0xf0, 0x9a, 0xb0, 0xbe, 0x82, 0xc1, 0x93, 0xbe, 0xae, 0xb3, 0x58, 0x4a, 0x53, 0xed, + 0x3a, 0x07, 0x96, 0xb7, 0x4f, 0xed, 0x7f, 0x8b, 0x4e, 0xff, 0x3a, 0x5c, 0x1a, 0xd0, 0x42, 0x92, + 0x6a, 0xfe, 0x9b, 0x29, 0x28, 0xae, 0x1e, 0x75, 0x2d, 0xdb, 0x6d, 0xf0, 0xc9, 0x1e, 0xad, 0x40, + 0xae, 0x6b, 0x5b, 0x2f, 0x0c, 0x4f, 0x70, 0x29, 0xf2, 0xdc, 0xa2, 0x8f, 0x67, 0x47, 0xe4, 0xc7, + 0x3e, 0x27, 0xc2, 0x90, 0x7f, 0x6c, 0x69, 0x6a, 0xfb, 0x3d, 0xa3, 0xed, 0x59, 0xd5, 0xe2, 0x28, + 0x31, 0x8b, 0x3e, 0xc7, 0x8e, 0xea, 0x1e, 0x78, 0x83, 0xcc, 0x27, 0xa2, 0x35, 0xc8, 0xd5, 0x5d, + 0xb7, 0x4b, 0x13, 0xc5, 0x08, 0xbd, 0x39, 0x52, 0x24, 0x65, 0x10, 0x92, 0x7c, 0x66, 0x84, 0x61, + 0x76, 0xcd, 0xb2, 0x5a, 0x6d, 0x52, 0x6b, 0x5b, 0x3d, 0xbd, 0x66, 0x99, 0xfb, 0x46, 0x4b, 0x78, + 0xb8, 0x1b, 0x23, 0x25, 0xae, 0xd5, 0x1a, 0x78, 0x90, 0x1d, 0x7d, 0x09, 0x72, 0x8d, 0x87, 0x42, + 0x14, 0xc7, 0x27, 0x6f, 0x8e, 0x14, 0xd5, 0x78, 0x88, 0x7d, 0x26, 0x54, 0x87, 0xc2, 0xf2, 0xcb, + 0x9e, 0x4d, 0x84, 0x8c, 0x49, 0x26, 0xe3, 0xd6, 0x48, 0x19, 0x8c, 0x07, 0x87, 0x59, 0xe7, 0xee, + 0x42, 0xb1, 0x4f, 0x93, 0x08, 0x41, 0xb6, 0x4b, 0x95, 0x46, 0xbb, 0x33, 0x8f, 0xd9, 0x7f, 0x6e, + 0x5e, 0x73, 0xb7, 0x20, 0x4b, 0xb5, 0x42, 0x87, 0xc3, 0x9e, 0xea, 0x90, 0x5d, 0xdb, 0x10, 0x99, + 0xbc, 0x4f, 0x91, 0xef, 0xaf, 0x25, 0x48, 0x37, 0x1e, 0x52, 0x84, 0xb6, 0xd7, 0xd3, 0x9e, 0x11, + 0x57, 0xe4, 0x12, 0x5f, 0x0c, 0xb9, 0xd9, 0x64, 0xdf, 0xe0, 0xb3, 0x75, 0x1e, 0x8b, 0x2f, 0xf4, + 0x06, 0x80, 0xaa, 0x69, 0xc4, 0x71, 0x14, 0x2f, 0xde, 0x38, 0x8f, 0xf3, 0x9c, 0xb2, 0x41, 0x8e, + 0x29, 0x9b, 0x43, 0x34, 0x9b, 0x70, 0xe3, 0xcf, 0x63, 0xf1, 0x45, 0xd9, 0x5c, 0xd2, 0xe9, 0x2a, + 0xae, 0xf5, 0x8c, 0x98, 0x4c, 0x9b, 0x79, 0x9c, 0xa7, 0x94, 0x26, 0x25, 0xd0, 0x51, 0x43, 0x4c, + 0xbd, 0x6b, 0x19, 0xa6, 0xcb, 0xd4, 0x94, 0xc7, 0xfe, 0x37, 0x15, 0x69, 0x93, 0x96, 0x21, 0x22, + 0x71, 0xf3, 0x58, 0x7c, 0x89, 0x66, 0x6c, 0x43, 0x66, 0xad, 0xd6, 0x78, 0xe5, 0x66, 0x20, 0xc8, + 0xaa, 0x3d, 0x61, 0x74, 0x79, 0xcc, 0xfe, 0x0b, 0x81, 0xdf, 0x94, 0x60, 0x82, 0xa9, 0x1e, 0x5d, + 0x85, 0xbc, 0x66, 0x99, 0xae, 0x6a, 0x98, 0x62, 0xdc, 0xe4, 0x71, 0x40, 0x88, 0x95, 0x7c, 0x1d, + 0xa6, 0x55, 0x4d, 0xb3, 0x7a, 0xa6, 0xab, 0x98, 0x6a, 0x87, 0x88, 0x12, 0x0a, 0x82, 0xb6, 0xa5, + 0x76, 0x08, 0x9a, 0x07, 0xef, 0xd3, 0x8f, 0xca, 0xce, 0x63, 0x10, 0x24, 0xff, 0xf8, 0x48, 0xb8, + 0x8b, 0x3f, 0x96, 0x60, 0xf6, 0x03, 0xdb, 0x70, 0x49, 0x95, 0xc7, 0x0a, 0x24, 0xe7, 0x30, 0xdf, + 0x81, 0xbc, 0xae, 0xba, 0x2a, 0x8f, 0xc0, 0x4e, 0x0f, 0x8d, 0xc0, 0xf6, 0xc6, 0x1b, 0xcd, 0xcf, + 0xa2, 0xb0, 0x11, 0x64, 0xe9, 0x7f, 0xee, 0x4b, 0x31, 0xfb, 0x1f, 0x1c, 0x1d, 0x85, 0xab, 0x9b, + 0xa4, 0x67, 0xfb, 0xa7, 0xb4, 0xe7, 0xd9, 0x92, 0x54, 0xc3, 0x97, 0x61, 0x4a, 0xac, 0x8a, 0x84, + 0x12, 0x16, 0x46, 0x8d, 0x50, 0xef, 0xc8, 0x43, 0xb0, 0xa1, 0x2a, 0x80, 0xe3, 0xaa, 0xb6, 0xcb, + 0xd6, 0x33, 0x63, 0x9d, 0x31, 0x7b, 0x9e, 0x90, 0xb1, 0x51, 0x2a, 0xda, 0x82, 0x42, 0xe7, 0x85, + 0xa6, 0x29, 0xfb, 0x46, 0xdb, 0x15, 0xc7, 0xcb, 0xd1, 0xc1, 0x34, 0x9b, 0x4f, 0x6a, 0xb5, 0xf7, + 0x58, 0x26, 0x7e, 0xca, 0x1b, 0x7c, 0x63, 0xa0, 0x12, 0xf8, 0x7f, 0xf4, 0x29, 0x10, 0x51, 0x76, + 0x8a, 0xe3, 0xb8, 0x6c, 0xc0, 0xe5, 0xaa, 0xc5, 0xd3, 0x93, 0xf9, 0x3c, 0x66, 0xd4, 0x46, 0xa3, + 0x89, 0xf3, 0x3c, 0x43, 0xc3, 0xf1, 0x66, 0xa6, 0x6f, 0x4b, 0x50, 0xac, 0xf6, 0xda, 0xcf, 0xb6, + 0xbb, 0x8d, 0x5e, 0xa7, 0xa3, 0xda, 0xc7, 0xe8, 0x8a, 0x67, 0x22, 0xc6, 0x4b, 0xc2, 0x54, 0x9c, + 0x11, 0x36, 0x60, 0xbc, 0x24, 0xd4, 0x06, 0x44, 0x40, 0x0d, 0xa5, 0xf3, 0x68, 0x99, 0x37, 0xa1, + 0xc8, 0x60, 0xbd, 0x42, 0x4c, 0xd7, 0x36, 0x08, 0x5f, 0x35, 0x66, 0xf0, 0x34, 0x23, 0xae, 0x72, + 0x1a, 0xba, 0x09, 0x25, 0xe7, 0xd8, 0x71, 0x49, 0x47, 0xe1, 0x57, 0x2e, 0x38, 0x16, 0xcd, 0xe0, + 0x22, 0xa7, 0x62, 0x4e, 0xac, 0xfc, 0x69, 0x06, 0x4a, 0x5e, 0x77, 0x27, 0x09, 0x8d, 0xaa, 0x30, + 0xb1, 0x6f, 0xb4, 0x89, 0x17, 0x09, 0x14, 0xef, 0x90, 0x3d, 0x49, 0x8b, 0xd4, 0xed, 0x7a, 0x40, + 0x89, 0xb1, 0x26, 0xd1, 0xe5, 0x73, 0x3f, 0x96, 0x20, 0xcb, 0x66, 0xc1, 0x07, 0x90, 0x65, 0x63, + 0x50, 0x1a, 0x67, 0x0c, 0xb2, 0xac, 0xbe, 0xff, 0x4f, 0x07, 0xfe, 0x9f, 0xf9, 0xde, 0x03, 0xf5, + 0xed, 0x07, 0x4b, 0xac, 0xbb, 0xa7, 0xb1, 0xf8, 0x42, 0x55, 0xc8, 0x11, 0xd6, 0x1e, 0xa2, 0x8b, + 0x39, 0x28, 0xca, 0xc2, 0xfb, 0x3a, 0xde, 0x1b, 0xef, 0x1e, 0x1f, 0xba, 0x0c, 0x19, 0x6a, 0x47, + 0x53, 0xfc, 0xac, 0xe4, 0xf4, 0x64, 0x3e, 0x43, 0x2d, 0x88, 0xd2, 0xf8, 0x89, 0xfb, 0xa3, 0x6c, + 0x2e, 0x2b, 0x4f, 0x54, 0xfe, 0x3c, 0x0b, 0xc5, 0xf5, 0x4e, 0xd2, 0x23, 0x74, 0xb9, 0xbf, 0xc3, + 0xa2, 0x20, 0x42, 0x5f, 0xa1, 0x11, 0xfd, 0xd5, 0xe7, 0xeb, 0x32, 0xaf, 0xe6, 0xeb, 0xd6, 0xe9, + 0x04, 0x24, 0xee, 0x9b, 0xd0, 0xf2, 0xdf, 0x1a, 0x59, 0x7e, 0x53, 0xdd, 0x6b, 0x13, 0x4c, 0x79, + 0xbc, 0xa3, 0x08, 0x2e, 0x00, 0x7d, 0x91, 0xcd, 0x73, 0xdc, 0x68, 0x26, 0xc7, 0x37, 0x9a, 0x29, + 0x62, 0xea, 0xcc, 0x64, 0x8e, 0x84, 0xc5, 0x7c, 0x0e, 0x32, 0xba, 0x31, 0x4c, 0xa5, 0x51, 0xfe, + 0x8a, 0xb2, 0x8c, 0x30, 0x9c, 0x6c, 0xd8, 0x70, 0xfc, 0xe3, 0x8a, 0x8c, 0x9c, 0x9d, 0xdb, 0x06, + 0x08, 0x5a, 0x85, 0x16, 0x60, 0xd2, 0x6a, 0xeb, 0x14, 0x4c, 0xd3, 0x2a, 0x14, 0xab, 0xf9, 0xd3, + 0x93, 0xf9, 0x89, 0xed, 0xb6, 0xbe, 0xbe, 0x82, 0x27, 0xac, 0xb6, 0xbe, 0xae, 0xb3, 0x6b, 0x3a, + 0xe4, 0x50, 0x61, 0x37, 0xa3, 0x58, 0x98, 0x04, 0x9e, 0x32, 0xc9, 0xe1, 0x0a, 0x71, 0xb4, 0xf0, + 0xec, 0x26, 0xcc, 0xe6, 0x0f, 0x24, 0x28, 0x79, 0x1a, 0x4c, 0x76, 0xa4, 0xe7, 0x8c, 0x8e, 0xb0, + 0xfc, 0xcc, 0xab, 0x59, 0xbe, 0xc7, 0x27, 0x42, 0x90, 0xbf, 0x25, 0xc1, 0x39, 0x1e, 0xcb, 0xa1, + 0xa9, 0x2e, 0xf5, 0xb5, 0x09, 0x9a, 0xf7, 0x5d, 0x90, 0x6d, 0xd5, 0xd4, 0xad, 0x8e, 0xf1, 0x92, + 0xf0, 0xd5, 0xaa, 0x23, 0x36, 0x69, 0x67, 0x7c, 0x3a, 0x5b, 0x8e, 0x79, 0x8b, 0xed, 0xff, 0x90, + 0xe0, 0x7c, 0x7f, 0x65, 0x92, 0x54, 0xda, 0x06, 0x4c, 0xb2, 0x8d, 0x16, 0x6f, 0xb8, 0x7d, 0x3a, + 0x42, 0x48, 0x54, 0xe9, 0xfc, 0x56, 0x95, 0x6f, 0xf0, 0x4c, 0xc4, 0xdc, 0x97, 0x61, 0x82, 0x91, + 0x5f, 0xc3, 0xc7, 0x09, 0xcd, 0x3f, 0x87, 0xd9, 0x65, 0x5d, 0x6f, 0x34, 0x84, 0xf5, 0x25, 0xa7, + 0x76, 0x0f, 0xc2, 0xa4, 0xa3, 0x20, 0x4c, 0xb8, 0xc8, 0x24, 0x21, 0x4c, 0x17, 0x4a, 0x22, 0xc4, + 0x2a, 0xe1, 0x9d, 0xb5, 0x43, 0x8a, 0xb9, 0x84, 0xd9, 0xf0, 0x8f, 0xe0, 0x66, 0x87, 0x5f, 0x62, + 0x92, 0x2d, 0xe9, 0xc1, 0x39, 0x4f, 0x6e, 0xd2, 0x9b, 0xd8, 0xc3, 0x9a, 0xc3, 0x76, 0x28, 0xc2, + 0xc5, 0x26, 0xd9, 0xa6, 0x3f, 0x91, 0x60, 0x6e, 0x8d, 0xb8, 0x0d, 0xb1, 0x60, 0x7f, 0xcf, 0xb2, + 0x13, 0xdf, 0xf1, 0x5d, 0x03, 0x68, 0x93, 0x7d, 0x11, 0x86, 0x2f, 0x00, 0xe7, 0xf8, 0x77, 0x46, + 0xf3, 0x94, 0x97, 0x25, 0x09, 0x75, 0xfc, 0x44, 0x82, 0x2b, 0x91, 0x35, 0x4e, 0xd2, 0x23, 0x44, + 0x8c, 0x14, 0xf4, 0x75, 0x60, 0x68, 0x53, 0x71, 0x5c, 0xd5, 0x75, 0x84, 0x73, 0xfd, 0xd4, 0xab, + 0x84, 0xba, 0x56, 0x67, 0x45, 0xdc, 0x46, 0xde, 0x27, 0xe1, 0x3c, 0x15, 0xc9, 0xfe, 0x56, 0xbe, + 0x73, 0x09, 0xa6, 0x85, 0x2e, 0x77, 0x4d, 0xc3, 0x32, 0xd1, 0x03, 0xc8, 0xb4, 0xc4, 0xf2, 0xae, + 0x10, 0x09, 0x8c, 0x83, 0x5b, 0x90, 0xf5, 0x14, 0xa6, 0x79, 0x29, 0x4b, 0xb7, 0xe7, 0x46, 0xc4, + 0x6f, 0x04, 0x67, 0xf7, 0x61, 0x96, 0x6e, 0xcf, 0x45, 0x0d, 0x98, 0xd1, 0x82, 0xbb, 0x63, 0x0a, + 0x65, 0xcf, 0xc4, 0x46, 0x7a, 0x46, 0xde, 0xa5, 0xab, 0xa7, 0x70, 0x49, 0xeb, 0x4b, 0x40, 0xb5, + 0xf0, 0x65, 0xa5, 0x6c, 0xec, 0x4e, 0xc2, 0xd9, 0x8b, 0x52, 0xf5, 0x54, 0xe8, 0x4e, 0x13, 0x7a, + 0x07, 0x26, 0x75, 0x76, 0x09, 0x46, 0xec, 0x45, 0x44, 0x19, 0x5f, 0xdf, 0xbd, 0xa3, 0x7a, 0x0a, + 0x0b, 0x0e, 0xf4, 0x08, 0xa6, 0xf9, 0x3f, 0x61, 0x76, 0x93, 0xb1, 0x5b, 0x2d, 0x83, 0xd7, 0x80, + 0xea, 0x29, 0x5c, 0xd0, 0x03, 0x2a, 0xfa, 0x0c, 0x64, 0x1d, 0x4d, 0xe5, 0x8b, 0xf1, 0xe8, 0x73, + 0xe8, 0xd0, 0x3d, 0x83, 0x3a, 0xf5, 0xe5, 0x9a, 0x6a, 0xa2, 0xa7, 0x30, 0xbb, 0x47, 0x5a, 0x86, + 0xa9, 0xb8, 0xc1, 0x59, 0x11, 0x8b, 0x3a, 0xed, 0x3f, 0x9e, 0xf2, 0xa7, 0xe4, 0xe8, 0x60, 0xe7, + 0x7a, 0x0a, 0xcb, 0x7b, 0x67, 0x92, 0x68, 0x97, 0x31, 0x4c, 0x15, 0x12, 0x9c, 0x8f, 0xed, 0xb2, + 0xc8, 0xf0, 0x63, 0xda, 0x65, 0xa4, 0x2f, 0x01, 0xad, 0x41, 0x41, 0xa5, 0x73, 0x9c, 0xc2, 0x42, + 0x3a, 0xcb, 0x10, 0xbb, 0x93, 0x34, 0x10, 0x65, 0x5a, 0x67, 0x91, 0xdb, 0x1e, 0x31, 0x10, 0xd4, + 0xa1, 0xe3, 0xb2, 0x5c, 0x18, 0x2e, 0x28, 0xec, 0x6e, 0x7c, 0x41, 0x8c, 0x88, 0x36, 0xa1, 0x78, + 0xe0, 0x05, 0x95, 0xb1, 0xc3, 0xbd, 0xe9, 0xd8, 0xed, 0xa4, 0x88, 0xa0, 0xb8, 0x7a, 0x0a, 0x4f, + 0x1f, 0x84, 0xc8, 0x68, 0x11, 0xd2, 0x2d, 0xad, 0x5c, 0x64, 0x32, 0xae, 0x0e, 0x0b, 0xf9, 0xaa, + 0xa7, 0x70, 0xba, 0xa5, 0x51, 0xe4, 0xca, 0xa3, 0x61, 0x8e, 0xcc, 0x72, 0x29, 0xd6, 0x95, 0xf4, + 0x47, 0x24, 0xd5, 0x53, 0x98, 0xc5, 0xed, 0xd0, 0xf2, 0x76, 0xa0, 0x64, 0xf3, 0x13, 0x3a, 0xef, + 0x1c, 0x5a, 0x66, 0x52, 0x6e, 0x47, 0x3b, 0xa4, 0x81, 0xa3, 0xe8, 0x7a, 0x0a, 0x17, 0xed, 0x30, + 0x1d, 0x7d, 0x03, 0xce, 0xf7, 0x4b, 0x14, 0xc6, 0x3d, 0x3b, 0xe0, 0x8b, 0xa2, 0xe5, 0xf6, 0xdb, + 0x38, 0xb2, 0x07, 0x12, 0xd1, 0x67, 0x61, 0x82, 0xf7, 0x1a, 0x62, 0x22, 0xe7, 0xa3, 0x56, 0xe3, + 0xfd, 0x1d, 0xc6, 0xf3, 0xd3, 0xf1, 0xe6, 0x8a, 0xa3, 0x29, 0xa5, 0x6d, 0xb5, 0xca, 0xe7, 0x62, + 0xc7, 0xdb, 0xe0, 0x51, 0x1b, 0x1d, 0x6f, 0x6e, 0x40, 0xa5, 0xfd, 0x6e, 0xf3, 0x14, 0x71, 0x92, + 0x71, 0x3e, 0xb6, 0xdf, 0x23, 0x4e, 0xac, 0x68, 0xbf, 0xdb, 0x21, 0x32, 0xad, 0x9a, 0xcd, 0x6f, + 0xd8, 0x28, 0x6c, 0x18, 0x5f, 0x88, 0xad, 0xda, 0xe0, 0xad, 0x21, 0x5a, 0x35, 0x3b, 0xa0, 0xa2, + 0x27, 0x20, 0x8b, 0xcb, 0x14, 0x8a, 0xb7, 0x03, 0x5e, 0xbe, 0xc8, 0xe4, 0xdd, 0x8d, 0xf4, 0x96, + 0x51, 0x5b, 0xff, 0xf5, 0x14, 0x9e, 0xd1, 0xfa, 0x53, 0xa8, 0xb3, 0x60, 0xf2, 0x14, 0x2d, 0xb8, + 0x85, 0x52, 0x2e, 0xc7, 0x3a, 0x8b, 0x98, 0x2b, 0x33, 0xd4, 0x59, 0x68, 0x67, 0x92, 0xa8, 0x19, + 0x1b, 0xa6, 0xe1, 0x32, 0xc7, 0x3e, 0x17, 0x6b, 0xc6, 0xfd, 0xd7, 0x7b, 0xa9, 0x19, 0x1b, 0x9c, + 0x42, 0xcd, 0xd8, 0x15, 0xc7, 0x5c, 0xa2, 0x3b, 0xae, 0xc6, 0x9a, 0x71, 0xd4, 0x79, 0x18, 0x35, + 0x63, 0x37, 0x4c, 0xa7, 0x66, 0xcc, 0x1d, 0xc4, 0x19, 0xb9, 0x6f, 0xc4, 0x9a, 0x71, 0x6c, 0xf4, + 0x34, 0x35, 0x63, 0x75, 0x20, 0x11, 0xad, 0x50, 0xc8, 0xa1, 0x3a, 0xfc, 0x59, 0x84, 0xf2, 0xb5, + 0xd8, 0xf9, 0xe7, 0xec, 0x41, 0x57, 0x9d, 0xe1, 0x0d, 0x41, 0xa3, 0x8e, 0x8c, 0xe1, 0x30, 0x85, + 0xdd, 0x59, 0x2a, 0xcf, 0xc7, 0x3a, 0xb2, 0x81, 0xcd, 0x4a, 0xea, 0xc8, 0x0e, 0x7d, 0x22, 0x9d, + 0xc8, 0xf8, 0xb6, 0x42, 0x79, 0x61, 0xc4, 0xf2, 0x35, 0x34, 0x91, 0x71, 0x0e, 0xb4, 0x0c, 0xf9, + 0xe7, 0x3d, 0x62, 0x1f, 0x33, 0x37, 0x74, 0x3d, 0x16, 0x3c, 0x9d, 0x89, 0x5c, 0xab, 0xa7, 0x70, + 0xee, 0xb9, 0x20, 0xd1, 0xe2, 0xf9, 0xda, 0xae, 0x5c, 0x89, 0x2d, 0xbe, 0x6f, 0x35, 0x4f, 0x8b, + 0xe7, 0x1c, 0x48, 0x83, 0x0b, 0xbc, 0xaf, 0x44, 0xb8, 0xb5, 0x2d, 0xe2, 0x9a, 0xcb, 0x6f, 0x32, + 0x51, 0xb1, 0x2b, 0xa5, 0xc8, 0xd0, 0xef, 0x7a, 0x0a, 0x9f, 0x53, 0x07, 0x53, 0xe9, 0x80, 0x17, + 0x53, 0x0f, 0x5f, 0x5f, 0x95, 0x6f, 0xc4, 0x0e, 0xf8, 0x88, 0x15, 0x29, 0x1d, 0xf0, 0x6a, 0x88, + 0xcc, 0x27, 0x20, 0x5d, 0x71, 0x1c, 0x97, 0xae, 0x66, 0xca, 0x37, 0x87, 0x4c, 0x40, 0x67, 0x56, + 0x59, 0x7c, 0x02, 0xd2, 0x1b, 0x9c, 0x93, 0x0a, 0xd2, 0xda, 0x44, 0xb5, 0x85, 0x9b, 0xbd, 0x15, + 0x2b, 0x68, 0xe0, 0xca, 0x2c, 0x15, 0xa4, 0xf9, 0x44, 0x3a, 0x61, 0xdb, 0xde, 0x2d, 0x2c, 0x81, + 0x1f, 0x6f, 0xc7, 0x4e, 0xd8, 0x91, 0x97, 0xc5, 0xe8, 0x84, 0x6d, 0xf7, 0x25, 0xa0, 0x2f, 0xc0, + 0x94, 0xb8, 0xd2, 0x52, 0xbe, 0x33, 0x04, 0xe9, 0x86, 0x97, 0x5e, 0x74, 0x5c, 0x0b, 0x1e, 0xee, + 0x65, 0xf9, 0x8d, 0x18, 0xde, 0xbc, 0xbb, 0x43, 0xbc, 0xec, 0xc0, 0xaa, 0x87, 0x7b, 0xd9, 0x80, + 0x4c, 0xbd, 0x2c, 0xb7, 0x53, 0x31, 0xd7, 0xdd, 0x8b, 0xf5, 0xb2, 0x83, 0x31, 0x67, 0xd4, 0xcb, + 0x3e, 0x0f, 0xa8, 0x48, 0x87, 0x8b, 0x2d, 0xe2, 0x2a, 0xde, 0x29, 0x22, 0x0b, 0x7b, 0xe4, 0xd3, + 0xd2, 0x5b, 0xb1, 0x56, 0x17, 0xbf, 0x88, 0xa1, 0x56, 0xd7, 0x1a, 0x4c, 0xad, 0x4e, 0x89, 0x48, + 0x94, 0x47, 0xd9, 0xdc, 0x8c, 0x2c, 0x3f, 0xca, 0xe6, 0x2e, 0xc9, 0xe5, 0x47, 0xd9, 0xdc, 0x65, + 0x79, 0xee, 0x51, 0x36, 0x77, 0x45, 0xbe, 0x5a, 0xf9, 0xf7, 0x8b, 0x50, 0xf4, 0x56, 0x0a, 0x1c, + 0x91, 0x2f, 0x85, 0x11, 0xf9, 0xb5, 0x38, 0x44, 0x2e, 0xd6, 0x16, 0x02, 0x92, 0x2f, 0x85, 0x21, + 0xf9, 0xb5, 0x38, 0x48, 0x1e, 0xf0, 0x50, 0x4c, 0xde, 0x8c, 0xc3, 0xe4, 0x77, 0xc7, 0xc0, 0xe4, + 0xbe, 0xa8, 0xb3, 0xa0, 0x7c, 0x65, 0x10, 0x94, 0xdf, 0x18, 0x0e, 0xca, 0x7d, 0x51, 0x21, 0x54, + 0xfe, 0xee, 0x19, 0x54, 0x7e, 0x7d, 0x08, 0x2a, 0xf7, 0xf9, 0x3d, 0x58, 0xbe, 0x11, 0x09, 0xcb, + 0x6f, 0x8d, 0x82, 0xe5, 0xbe, 0x9c, 0x3e, 0x5c, 0xfe, 0x76, 0x1f, 0x2e, 0x9f, 0x8f, 0xc5, 0xe5, + 0x3e, 0x37, 0x07, 0xe6, 0x1f, 0xc6, 0x03, 0xf3, 0xb7, 0xc6, 0x02, 0xe6, 0xbe, 0xbc, 0x41, 0x64, + 0xde, 0x8c, 0x43, 0xe6, 0x77, 0xc7, 0x40, 0xe6, 0x41, 0xc7, 0x9d, 0x81, 0xe6, 0xf5, 0x28, 0x68, + 0x7e, 0x73, 0x04, 0x34, 0xf7, 0xa5, 0x85, 0xb1, 0x79, 0x3d, 0x0a, 0x9b, 0xdf, 0x1c, 0x81, 0xcd, + 0xcf, 0x48, 0xe2, 0xe0, 0x7c, 0x2b, 0x1a, 0x9c, 0xdf, 0x1e, 0x09, 0xce, 0x7d, 0x69, 0xfd, 0xe8, + 0xfc, 0x7e, 0x08, 0x9d, 0xbf, 0x11, 0x83, 0xce, 0x7d, 0x56, 0x0a, 0xcf, 0xbf, 0x34, 0x00, 0xcf, + 0x2b, 0xc3, 0xe0, 0xb9, 0xcf, 0xeb, 0xe3, 0xf3, 0xf7, 0x63, 0xf0, 0xf9, 0x9d, 0xd1, 0xf8, 0xdc, + 0x17, 0x76, 0x06, 0xa0, 0xab, 0x43, 0x01, 0xfa, 0xa7, 0xc7, 0x04, 0xe8, 0xbe, 0xf4, 0x28, 0x84, + 0xfe, 0xb9, 0x7e, 0x84, 0xbe, 0x10, 0x8f, 0xd0, 0x7d, 0x31, 0x02, 0xa2, 0x6f, 0x44, 0x42, 0xf4, + 0x5b, 0xa3, 0x20, 0x7a, 0x30, 0xf6, 0xc2, 0x18, 0x7d, 0x2b, 0x1a, 0xa3, 0xdf, 0x1e, 0x89, 0xd1, + 0x83, 0xee, 0xef, 0x03, 0xe9, 0x1b, 0x91, 0x20, 0xfd, 0xd6, 0x28, 0x90, 0x1e, 0x54, 0x2e, 0x8c, + 0xd2, 0x3f, 0x88, 0x45, 0xe9, 0xf7, 0xc6, 0x41, 0xe9, 0xbe, 0xd0, 0x01, 0x98, 0xfe, 0x61, 0x3c, + 0x4c, 0x7f, 0x6b, 0x2c, 0x98, 0x1e, 0xb8, 0x8e, 0x01, 0x9c, 0xfe, 0xa5, 0x01, 0x9c, 0x5e, 0x19, + 0x86, 0xd3, 0x03, 0x7b, 0xf6, 0x80, 0xba, 0x3a, 0x14, 0x56, 0x7f, 0x7a, 0x4c, 0x58, 0x1d, 0x18, + 0x5f, 0x04, 0xae, 0x5e, 0x8d, 0xc0, 0xd5, 0x37, 0x86, 0xe3, 0xea, 0x60, 0x0a, 0x09, 0x80, 0x75, + 0x3d, 0x0a, 0x58, 0xdf, 0x1c, 0x01, 0xac, 0x03, 0x2f, 0x14, 0x42, 0xd6, 0xef, 0x9e, 0x41, 0xd6, + 0xd7, 0x47, 0x9e, 0x6c, 0x86, 0xa0, 0x75, 0x75, 0x10, 0x5a, 0xbf, 0x39, 0x14, 0x5a, 0xfb, 0x12, + 0x02, 0x6c, 0xfd, 0xee, 0x19, 0x6c, 0x7d, 0x7d, 0x08, 0xb6, 0x0e, 0x2a, 0x20, 0xc0, 0xb5, 0x3e, + 0x1c, 0x5c, 0x2f, 0x8e, 0x0b, 0xae, 0x7d, 0xc1, 0x91, 0xe8, 0x7a, 0x2b, 0x1a, 0x5d, 0xdf, 0x1e, + 0xf3, 0x90, 0x63, 0x00, 0x5e, 0xd7, 0xa3, 0xe0, 0xf5, 0xcd, 0x11, 0xf0, 0x3a, 0x3c, 0x87, 0xf8, + 0xf8, 0xba, 0x1e, 0x85, 0xaf, 0x6f, 0x8e, 0xc0, 0xd7, 0x81, 0xa4, 0x10, 0xc0, 0x6e, 0xc6, 0x01, + 0xec, 0xbb, 0x63, 0x00, 0xec, 0x60, 0xde, 0x3d, 0x83, 0xb0, 0xbf, 0x78, 0x16, 0x61, 0x57, 0x86, + 0x21, 0xec, 0x60, 0x44, 0x7a, 0x10, 0x7b, 0x2b, 0x1a, 0x62, 0xdf, 0x1e, 0x09, 0xb1, 0xc3, 0x4e, + 0x32, 0x84, 0xb1, 0x37, 0x22, 0x31, 0xf6, 0xad, 0x51, 0x18, 0x3b, 0x70, 0x92, 0x61, 0x90, 0x4d, + 0x46, 0x80, 0xec, 0xc5, 0x71, 0x41, 0x76, 0x60, 0x7d, 0xaf, 0x83, 0xb2, 0x1f, 0x65, 0x73, 0x57, + 0xe5, 0x37, 0x2a, 0x7f, 0x35, 0x01, 0x93, 0x75, 0xef, 0xec, 0x3b, 0x74, 0x13, 0x51, 0x7a, 0x9d, + 0x9b, 0x88, 0x68, 0x85, 0x76, 0x1c, 0x33, 0x7f, 0x81, 0xbb, 0x87, 0xdc, 0xeb, 0x1d, 0x38, 0x71, + 0xf0, 0x58, 0x5f, 0x23, 0x4e, 0x1b, 0xbd, 0x0d, 0xc5, 0x9e, 0x43, 0x6c, 0xa5, 0x6b, 0x1b, 0x96, + 0x6d, 0xb8, 0x3c, 0xd2, 0x49, 0xaa, 0xca, 0x1f, 0x9d, 0xcc, 0x4f, 0xef, 0x3a, 0xc4, 0xde, 0x11, + 0x74, 0x3c, 0xdd, 0x0b, 0x7d, 0x79, 0xaf, 0x49, 0x4e, 0x8c, 0xff, 0x9a, 0xe4, 0xfb, 0x20, 0xdb, + 0x44, 0xd5, 0xfb, 0x26, 0x22, 0x7e, 0x63, 0x2f, 0x7a, 0xce, 0x54, 0xf5, 0xd0, 0x5c, 0xc3, 0x6e, + 0xee, 0xcd, 0xd8, 0xfd, 0x44, 0xf4, 0x00, 0x2e, 0x74, 0xd4, 0x23, 0x7e, 0x27, 0xd5, 0x9b, 0xdb, + 0x59, 0x0c, 0x40, 0x8e, 0x85, 0xaa, 0xa0, 0x8e, 0x7a, 0xc4, 0x9e, 0xa6, 0xe4, 0x49, 0xec, 0x29, + 0xab, 0x9b, 0x50, 0xd2, 0x0d, 0xc7, 0x35, 0x4c, 0xcd, 0x7b, 0xcd, 0x81, 0x3f, 0x9b, 0x50, 0xf4, + 0xa8, 0xfc, 0x55, 0x85, 0x7b, 0x30, 0x2b, 0x22, 0x73, 0x82, 0xc7, 0x2a, 0x19, 0x8a, 0xcd, 0xd1, + 0x5a, 0xd0, 0x84, 0xe0, 0x6d, 0xd1, 0x1a, 0xcc, 0xb4, 0x54, 0x97, 0x1c, 0xaa, 0xc7, 0x8a, 0x69, + 0xe9, 0x4c, 0xf7, 0x05, 0x76, 0xa7, 0xfd, 0xca, 0xe9, 0xc9, 0x7c, 0x71, 0x8d, 0x27, 0x6d, 0x59, + 0x3a, 0xef, 0x81, 0x49, 0xfe, 0x0f, 0x17, 0x5b, 0xa1, 0x04, 0x1d, 0x2d, 0xc3, 0x34, 0x7b, 0xa7, + 0xc7, 0xe2, 0x8f, 0x41, 0x09, 0x6c, 0x1a, 0xb7, 0x73, 0x2f, 0x9e, 0x8c, 0xc2, 0xec, 0x6d, 0x1f, + 0xef, 0xfd, 0xa8, 0xdb, 0x30, 0xa3, 0x3a, 0xc7, 0xa6, 0xc6, 0x34, 0x4c, 0x4c, 0xa7, 0xe7, 0x30, + 0x70, 0x9a, 0xc3, 0x25, 0x46, 0xae, 0x79, 0x54, 0xf1, 0x28, 0xc4, 0x6f, 0x4b, 0x30, 0xdd, 0x17, + 0xae, 0xf6, 0xee, 0x99, 0x63, 0xa8, 0xcb, 0xd1, 0xc0, 0x38, 0x2e, 0xfc, 0x23, 0x27, 0x7a, 0xc0, + 0x3b, 0x92, 0x9e, 0x8f, 0x07, 0x56, 0x6c, 0x69, 0xea, 0x1d, 0xe2, 0x7b, 0x6c, 0xef, 0x64, 0x7f, + 0xf7, 0xbb, 0xf3, 0xa9, 0xca, 0x4f, 0x33, 0x50, 0xec, 0x0f, 0x4b, 0x5b, 0x3f, 0x53, 0xaf, 0x28, + 0x67, 0xd4, 0xc7, 0x11, 0x5f, 0xcb, 0x15, 0xc8, 0x7b, 0xaf, 0xfb, 0x78, 0xd5, 0x5c, 0x18, 0x72, + 0xd8, 0x16, 0xae, 0x67, 0xc0, 0x38, 0xf7, 0xc3, 0xb4, 0x3f, 0xf2, 0x17, 0x61, 0x82, 0xbd, 0x93, + 0x2b, 0xaa, 0x16, 0x15, 0xc1, 0xbd, 0x4a, 0xd3, 0x31, 0xcf, 0x46, 0x3d, 0x45, 0xf3, 0xb5, 0xee, + 0x2c, 0xfb, 0x84, 0xd7, 0x78, 0xc7, 0xf5, 0x35, 0xef, 0xd6, 0xb2, 0x63, 0xb6, 0x76, 0x9b, 0x68, + 0xae, 0x78, 0x6e, 0xd7, 0x7b, 0xad, 0xf5, 0xc6, 0x59, 0x11, 0xe2, 0x71, 0xde, 0x45, 0x2c, 0x1e, + 0xe7, 0x0d, 0x45, 0x09, 0x94, 0x7c, 0x11, 0x6c, 0x60, 0xf1, 0x58, 0x12, 0xde, 0xd5, 0xf7, 0x1a, + 0x70, 0x2e, 0x62, 0x98, 0xa3, 0x12, 0x40, 0x6d, 0x7b, 0xab, 0xb1, 0xde, 0x68, 0xae, 0x6e, 0x35, + 0xbd, 0x37, 0x57, 0x97, 0x57, 0x94, 0xdd, 0xad, 0xda, 0xf6, 0xe6, 0xe6, 0x7a, 0xb3, 0xb9, 0xba, + 0x22, 0x4b, 0x48, 0x86, 0xe9, 0xf5, 0xad, 0x50, 0x3e, 0xf1, 0xd4, 0xea, 0xbd, 0x9f, 0x07, 0x08, + 0x9e, 0xb2, 0xa2, 0xb2, 0x36, 0x56, 0x9f, 0x2a, 0x4f, 0x96, 0x1f, 0xef, 0xae, 0x36, 0xe4, 0x14, + 0x42, 0x50, 0xaa, 0x2e, 0x37, 0x6b, 0x75, 0x05, 0xaf, 0x36, 0x76, 0xb6, 0xb7, 0x1a, 0xab, 0xb2, + 0x24, 0xf8, 0x3e, 0x80, 0x42, 0xe8, 0x96, 0x30, 0xcd, 0xb8, 0xb3, 0xdb, 0xa8, 0x2b, 0xcd, 0xf5, + 0xcd, 0xd5, 0x46, 0x73, 0x79, 0x73, 0x47, 0x4e, 0x51, 0x61, 0x8c, 0xb6, 0x5c, 0xdd, 0xc6, 0x4d, + 0x59, 0xf2, 0xbf, 0x9b, 0xdb, 0xbb, 0xb5, 0xba, 0x9c, 0xf6, 0xbf, 0xdf, 0xdf, 0x5d, 0xc5, 0x4f, + 0xe5, 0x8c, 0x10, 0xac, 0xc2, 0x85, 0xc8, 0xa8, 0x6d, 0x54, 0x80, 0xa9, 0x5d, 0x93, 0xdd, 0x54, + 0x95, 0x53, 0xa8, 0x18, 0x0a, 0xdc, 0x96, 0x25, 0x94, 0xe3, 0x01, 0xc2, 0x72, 0x1a, 0x4d, 0x42, + 0xba, 0xf1, 0x50, 0xce, 0xa0, 0x19, 0x28, 0x84, 0xa2, 0x9f, 0xe5, 0x2c, 0xca, 0x8b, 0x10, 0x58, + 0x79, 0xe2, 0xde, 0x75, 0x08, 0x45, 0x18, 0x22, 0x80, 0xc9, 0xc7, 0xaa, 0x4b, 0x1c, 0x57, 0x4e, + 0xa1, 0x29, 0xc8, 0x2c, 0xb7, 0xdb, 0xb2, 0xb4, 0xf4, 0x15, 0xc8, 0x79, 0x6f, 0xde, 0xa0, 0xc7, + 0x30, 0xc1, 0xb1, 0xe8, 0x7c, 0xfc, 0x48, 0x62, 0x63, 0x72, 0x6e, 0x61, 0xd4, 0x50, 0xab, 0xa4, + 0xaa, 0xd7, 0x7f, 0xf4, 0x2f, 0xd7, 0x52, 0x3f, 0x3a, 0xbd, 0x26, 0xfd, 0xf8, 0xf4, 0x9a, 0xf4, + 0x0f, 0xa7, 0xd7, 0xa4, 0x7f, 0x3e, 0xbd, 0x26, 0xfd, 0xe6, 0xbf, 0x5e, 0x4b, 0x7d, 0x38, 0x25, + 0x58, 0xf6, 0x26, 0xd9, 0x8b, 0xcc, 0x0f, 0xff, 0x27, 0x00, 0x00, 0xff, 0xff, 0x29, 0x1d, 0xbb, + 0x57, 0x96, 0x5a, 0x00, 0x00, } diff --git a/pkg/roachpb/api.proto b/pkg/roachpb/api.proto index 972be3476209..520a10c4c7b0 100644 --- a/pkg/roachpb/api.proto +++ b/pkg/roachpb/api.proto @@ -334,6 +334,19 @@ message ScanOptions { int64 min_results = 2; } +// ScanFormat is an enumeration of the available response formats for MVCCScan +// operations. +enum ScanFormat { + option (gogoproto.goproto_enum_prefix) = false; + + // The standard MVCCScan format: a slice of KeyValue messages. + KEY_VALUES = 0; + // The batch_response format: a byte slice of alternating keys and values, + // each prefixed by their length as a varint. + BATCH_RESPONSE = 1; +} + + // A ScanRequest is the argument to the Scan() method. It specifies the // start and end keys for an ascending scan of [start,end) and the maximum // number of results (unbounded if zero). @@ -343,6 +356,11 @@ message ScanRequest { reserved 2, 3; RequestHeader header = 1 [(gogoproto.nullable) = false, (gogoproto.embed) = true]; + + // The desired format for the response. If set to BATCH_RESPONSE, the server + // will set the batch_response field in the ScanResponse instead of the rows + // field. + ScanFormat scan_format = 4; } // A ScanResponse is the return value from the Scan() method. @@ -354,6 +372,12 @@ message ScanResponse { // consistency level. These rows do not count against the MaxSpanRequestKeys // count. repeated KeyValue intent_rows = 3 [(gogoproto.nullable) = false]; + + // If set, the results of the scan in batch format - the key/value pairs are + // a buffer of varint-prefixed slices, alternating from key to value. There + // are num_keys pairs, as defined by the ResponseHeader. If set, rows will not + // be set and vice versa. + bytes batch_response = 4; } // A ReverseScanRequest is the argument to the ReverseScan() method. It specifies the @@ -365,6 +389,11 @@ message ReverseScanRequest { reserved 2, 3; RequestHeader header = 1 [(gogoproto.nullable) = false, (gogoproto.embed) = true]; + + // The desired format for the response. If set to BATCH_RESPONSE, the server + // will set the batch_response field in the ScanResponse instead of the rows + // field. + ScanFormat scan_format = 4; } // A ReverseScanResponse is the return value from the ReverseScan() method. @@ -376,6 +405,12 @@ message ReverseScanResponse { // consistency level. These rows do not count against the MaxSpanRequestKeys // count. repeated KeyValue intent_rows = 3 [(gogoproto.nullable) = false]; + + // If set, the results of the scan in batch format - the key/value pairs are + // a buffer of varint-prefixed slices, alternating from key to value. There + // are num_keys pairs, as defined by the ResponseHeader. If set, rows will not + // be set and vice versa. + bytes batch_response = 4; } // A CheckConsistencyRequest is the argument to the CheckConsistency() method. diff --git a/pkg/server/updates_test.go b/pkg/server/updates_test.go index 509bf2687072..34510b64d1d9 100644 --- a/pkg/server/updates_test.go +++ b/pkg/server/updates_test.go @@ -442,7 +442,7 @@ func TestReportUsage(t *testing.T) { "diagnostics.reporting.send_crash_reports": "false", "server.time_until_store_dead": "1m30s", "trace.debug.enable": "false", - "version": "2.0-9", + "version": "2.0-10", "cluster.secret": "", } { if got, ok := r.last.AlteredSettings[key]; !ok { diff --git a/pkg/settings/cluster/cockroach_versions.go b/pkg/settings/cluster/cockroach_versions.go index 0fe6112b2d62..c34ef2aed287 100644 --- a/pkg/settings/cluster/cockroach_versions.go +++ b/pkg/settings/cluster/cockroach_versions.go @@ -66,6 +66,7 @@ const ( VersionColumnarTimeSeries VersionTxnCoordMetaInvalidField VersionAsyncConsensus + VersionBatchResponse // Add new versions here (step one of two). @@ -253,6 +254,11 @@ var versionsSingleton = keyedVersions([]keyedVersion{ Key: VersionAsyncConsensus, Version: roachpb.Version{Major: 2, Minor: 0, Unstable: 9}, }, + { + // VersionBatchResponse is https://github.com/cockroachdb/cockroach/pull/26553. + Key: VersionBatchResponse, + Version: roachpb.Version{Major: 2, Minor: 0, Unstable: 10}, + }, // Add new versions here (step two of two). diff --git a/pkg/sql/logictest/testdata/logic_test/crdb_internal b/pkg/sql/logictest/testdata/logic_test/crdb_internal index 49b2b5b489c4..99a8b373b011 100644 --- a/pkg/sql/logictest/testdata/logic_test/crdb_internal +++ b/pkg/sql/logictest/testdata/logic_test/crdb_internal @@ -244,7 +244,7 @@ select crdb_internal.set_vmodule('') query T select crdb_internal.node_executable_version() ---- -2.0-9 +2.0-10 query ITTT colnames select node_id, component, field, regexp_replace(regexp_replace(value, '^\d+$', ''), e':\\d+', ':') as value from crdb_internal.node_runtime_info @@ -332,4 +332,4 @@ select * from crdb_internal.gossip_alerts query T select crdb_internal.node_executable_version() ---- -2.0-9 +2.0-10 diff --git a/pkg/sql/sqlbase/errors.go b/pkg/sql/sqlbase/errors.go index 18d6ed82f425..5908bf1b69e7 100644 --- a/pkg/sql/sqlbase/errors.go +++ b/pkg/sql/sqlbase/errors.go @@ -255,12 +255,19 @@ type singleKVFetcher struct { // nextBatch implements the kvFetcher interface. func (f *singleKVFetcher) nextBatch( _ context.Context, -) (ok bool, kvs []roachpb.KeyValue, isNewScan bool, err error) { +) ( + ok bool, + kvs []roachpb.KeyValue, + batchResponse []byte, + numKvs int64, + maybeNewSpan bool, + err error, +) { if f.done { - return false, nil, true, nil + return false, nil, nil, 0, true, nil } f.done = true - return true, f.kvs[:], true, nil + return true, f.kvs[:], nil, 0, true, nil } // getRangesInfo implements the kvFetcher interface. diff --git a/pkg/sql/sqlbase/fk.go b/pkg/sql/sqlbase/fk.go index b0bb41ef9d30..825c286a493a 100644 --- a/pkg/sql/sqlbase/fk.go +++ b/pkg/sql/sqlbase/fk.go @@ -274,13 +274,20 @@ type SpanKVFetcher struct { // nextBatch implements the kvFetcher interface. func (f *SpanKVFetcher) nextBatch( _ context.Context, -) (ok bool, kvs []roachpb.KeyValue, isNewSpan bool, err error) { +) ( + ok bool, + kvs []roachpb.KeyValue, + batchResponse []byte, + numKvs int64, + maybeNewSpan bool, + err error, +) { if len(f.KVs) == 0 { - return false, nil, true, nil + return false, nil, nil, 0, true, nil } res := f.KVs f.KVs = nil - return true, res, true, nil + return true, res, nil, 0, true, nil } // getRangesInfo implements the kvFetcher interface. diff --git a/pkg/sql/sqlbase/kvfetcher.go b/pkg/sql/sqlbase/kvfetcher.go index 6acda2ba9d6d..84fde68dd4f7 100644 --- a/pkg/sql/sqlbase/kvfetcher.go +++ b/pkg/sql/sqlbase/kvfetcher.go @@ -282,12 +282,14 @@ func (f *txnKVFetcher) fetch(ctx context.Context) error { if f.reverse { scans := make([]roachpb.ReverseScanRequest, len(f.spans)) for i := range f.spans { + scans[i].ScanFormat = roachpb.BATCH_RESPONSE scans[i].SetSpan(f.spans[i]) ba.Requests[i].MustSetInner(&scans[i]) } } else { scans := make([]roachpb.ScanRequest, len(f.spans)) for i := range f.spans { + scans[i].ScanFormat = roachpb.BATCH_RESPONSE scans[i].SetSpan(f.spans[i]) ba.Requests[i].MustSetInner(&scans[i]) } @@ -355,22 +357,29 @@ func (f *txnKVFetcher) fetch(ctx context.Context) error { return nil } -func (f *txnKVFetcher) batchIsLimited(batchSize int) bool { +func (f *txnKVFetcher) batchIsLimited(batchSize int64) bool { if f.useBatchLimit { // f.batchIdx - 1 refers to the most recent fetch. - return int64(batchSize) == f.getBatchSizeForIdx(f.batchIdx-1) + return batchSize == f.getBatchSizeForIdx(f.batchIdx-1) } return false } // nextBatch returns the next batch of key/value pairs. If there are none -// available, a fetch is initiated. +// available, a fetch is initiated. When there are no more keys, returns false. // ok returns whether or not there are more kv pairs to be fetched. // maybeNewSpan returns true if it was possible that the kv pairs returned were // from a new span. func (f *txnKVFetcher) nextBatch( ctx context.Context, -) (ok bool, kvs []roachpb.KeyValue, maybeNewSpan bool, err error) { +) ( + ok bool, + kvs []roachpb.KeyValue, + batchResponse []byte, + numKvs int64, + maybeNewSpan bool, + err error, +) { if len(f.responses) > 0 { reply := f.responses[0].GetInner() f.responses = f.responses[1:] @@ -378,18 +387,18 @@ func (f *txnKVFetcher) nextBatch( maybeNewSpan = !f.lastBatchLimited switch t := reply.(type) { case *roachpb.ScanResponse: - f.lastBatchLimited = f.batchIsLimited(len(t.Rows)) - return true, t.Rows, maybeNewSpan, nil + f.lastBatchLimited = f.batchIsLimited(t.NumKeys) + return true, t.Rows, t.BatchResponse, t.NumKeys, maybeNewSpan, nil case *roachpb.ReverseScanResponse: - f.lastBatchLimited = f.batchIsLimited(len(t.Rows)) - return true, t.Rows, maybeNewSpan, nil + f.lastBatchLimited = f.batchIsLimited(t.NumKeys) + return true, t.Rows, t.BatchResponse, t.NumKeys, maybeNewSpan, nil } } if f.fetchEnd { - return false, nil, false, nil + return false, nil, nil, 0, false, nil } if err := f.fetch(ctx); err != nil { - return false, nil, false, err + return false, nil, nil, 0, false, err } return f.nextBatch(ctx) } diff --git a/pkg/sql/sqlbase/rowfetcher.go b/pkg/sql/sqlbase/rowfetcher.go index 5ec5e3784a9e..a9abed19ebcf 100644 --- a/pkg/sql/sqlbase/rowfetcher.go +++ b/pkg/sql/sqlbase/rowfetcher.go @@ -27,6 +27,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/roachpb" "github.com/cockroachdb/cockroach/pkg/sql/scrub" "github.com/cockroachdb/cockroach/pkg/sql/sem/tree" + "github.com/cockroachdb/cockroach/pkg/storage/engine" "github.com/cockroachdb/cockroach/pkg/util" "github.com/cockroachdb/cockroach/pkg/util/encoding" "github.com/cockroachdb/cockroach/pkg/util/hlc" @@ -38,7 +39,12 @@ import ( const debugRowFetch = false type kvFetcher interface { - nextBatch(ctx context.Context) (bool, []roachpb.KeyValue, bool, error) + // nextBatch returns the next batch of rows. Returns false in the first + // parameter if there are no more keys in the scan. May return either a slice + // of KeyValues or a batchResponse, numKvs pair, depending on the server + // version - both must be handled by calling code. maybeNewSpan is true if + // if it was possible that the kv pairs returned were from a new span. + nextBatch(ctx context.Context) (ok bool, kvs []roachpb.KeyValue, batchResponse []byte, numKvs int64, maybeNewSpan bool, err error) getRangesInfo() []roachpb.RangeInfo } @@ -218,6 +224,9 @@ type RowFetcher struct { kvs []roachpb.KeyValue + batchResponse []byte + batchNumKvs int64 + // isCheck indicates whether or not we are running checks for k/v // correctness. It is set only during SCRUB commands. isCheck bool @@ -431,6 +440,8 @@ func (rf *RowFetcher) StartScanFrom(ctx context.Context, f kvFetcher) error { rf.indexKey = nil rf.kvFetcher = f rf.kvs = nil + rf.batchNumKvs = 0 + rf.batchResponse = nil // Retrieve the first key. _, err := rf.NextKey(ctx) return err @@ -451,7 +462,30 @@ func (rf *RowFetcher) nextKV( rf.maybeNewSpan = false return true, kv, newSpan, nil } - ok, rf.kvs, rf.maybeNewSpan, err = rf.kvFetcher.nextBatch(ctx) + if rf.batchNumKvs > 0 { + rf.batchNumKvs-- + var key engine.MVCCKey + var rawBytes []byte + var err error + newSpan = rf.maybeNewSpan + rf.maybeNewSpan = false + key, rawBytes, rf.batchResponse, err = engine.MVCCScanDecodeKeyValue(rf.batchResponse) + if err != nil { + return false, kv, false, err + } + return true, roachpb.KeyValue{ + Key: key.Key, + Value: roachpb.Value{ + RawBytes: rawBytes, + }, + }, newSpan, nil + } + + var numKeys int64 + ok, rf.kvs, rf.batchResponse, numKeys, rf.maybeNewSpan, err = rf.kvFetcher.nextBatch(ctx) + if rf.batchResponse != nil { + rf.batchNumKvs = numKeys + } if err != nil { return ok, kv, false, err } diff --git a/pkg/storage/batcheval/cmd_reverse_scan.go b/pkg/storage/batcheval/cmd_reverse_scan.go index 651ba17dad4e..6edf696073d6 100644 --- a/pkg/storage/batcheval/cmd_reverse_scan.go +++ b/pkg/storage/batcheval/cmd_reverse_scan.go @@ -17,6 +17,8 @@ package batcheval import ( "context" + "fmt" + "github.com/cockroachdb/cockroach/pkg/roachpb" "github.com/cockroachdb/cockroach/pkg/storage/batcheval/result" "github.com/cockroachdb/cockroach/pkg/storage/engine" @@ -37,18 +39,40 @@ func ReverseScan( h := cArgs.Header reply := resp.(*roachpb.ReverseScanResponse) - rows, resumeSpan, intents, err := engine.MVCCReverseScan(ctx, batch, args.Key, args.EndKey, - cArgs.MaxKeys, h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) - if err != nil { - return result.Result{}, err + var err error + var intents []roachpb.Intent + var resumeSpan *roachpb.Span + + switch args.ScanFormat { + case roachpb.BATCH_RESPONSE: + var kvData []byte + var numKvs int64 + kvData, numKvs, resumeSpan, intents, err = engine.MVCCReverseScanToBytes( + ctx, batch, args.Key, args.EndKey, cArgs.MaxKeys, + h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) + if err != nil { + return result.Result{}, err + } + reply.NumKeys = numKvs + reply.BatchResponse = kvData + case roachpb.KEY_VALUES: + var rows []roachpb.KeyValue + rows, resumeSpan, intents, err = engine.MVCCReverseScan(ctx, batch, args.Key, args.EndKey, + cArgs.MaxKeys, h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) + if err != nil { + return result.Result{}, err + } + reply.NumKeys = int64(len(rows)) + reply.Rows = rows + default: + panic(fmt.Sprintf("Unknown scanFormat %d", args.ScanFormat)) } - reply.NumKeys = int64(len(rows)) if resumeSpan != nil { reply.ResumeSpan = resumeSpan reply.ResumeReason = roachpb.RESUME_KEY_LIMIT } - reply.Rows = rows + if h.ReadConsistency == roachpb.READ_UNCOMMITTED { reply.IntentRows, err = CollectIntentRows(ctx, batch, cArgs, intents) } diff --git a/pkg/storage/batcheval/cmd_scan.go b/pkg/storage/batcheval/cmd_scan.go index 20e2d93c2c0a..a373bb164471 100644 --- a/pkg/storage/batcheval/cmd_scan.go +++ b/pkg/storage/batcheval/cmd_scan.go @@ -17,6 +17,8 @@ package batcheval import ( "context" + "fmt" + "github.com/cockroachdb/cockroach/pkg/roachpb" "github.com/cockroachdb/cockroach/pkg/storage/batcheval/result" "github.com/cockroachdb/cockroach/pkg/storage/engine" @@ -37,20 +39,43 @@ func Scan( h := cArgs.Header reply := resp.(*roachpb.ScanResponse) - rows, resumeSpan, intents, err := engine.MVCCScan(ctx, batch, args.Key, args.EndKey, - cArgs.MaxKeys, h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) - if err != nil { - return result.Result{}, err + var err error + var intents []roachpb.Intent + var resumeSpan *roachpb.Span + + switch args.ScanFormat { + case roachpb.BATCH_RESPONSE: + var kvData []byte + var numKvs int64 + kvData, numKvs, resumeSpan, intents, err = engine.MVCCScanToBytes( + ctx, batch, args.Key, args.EndKey, cArgs.MaxKeys, + h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) + if err != nil { + return result.Result{}, err + } + reply.NumKeys = numKvs + reply.BatchResponse = kvData + case roachpb.KEY_VALUES: + var rows []roachpb.KeyValue + rows, resumeSpan, intents, err = engine.MVCCScan(ctx, batch, args.Key, args.EndKey, + cArgs.MaxKeys, h.Timestamp, h.ReadConsistency == roachpb.CONSISTENT, h.Txn) + if err != nil { + return result.Result{}, err + } + reply.NumKeys = int64(len(rows)) + reply.Rows = rows + default: + panic(fmt.Sprintf("Unknown scanFormat %d", args.ScanFormat)) } - reply.NumKeys = int64(len(rows)) if resumeSpan != nil { reply.ResumeSpan = resumeSpan reply.ResumeReason = roachpb.RESUME_KEY_LIMIT } - reply.Rows = rows + if h.ReadConsistency == roachpb.READ_UNCOMMITTED { reply.IntentRows, err = CollectIntentRows(ctx, batch, cArgs, intents) } return result.FromIntents(intents, args), err + } diff --git a/pkg/storage/engine/mvcc.go b/pkg/storage/engine/mvcc.go index 84dc2024c909..b4acd998d34a 100644 --- a/pkg/storage/engine/mvcc.go +++ b/pkg/storage/engine/mvcc.go @@ -1639,9 +1639,6 @@ func MVCCDeleteRange( return keys, resumeSpan, int64(len(kvs)), err } -// mvccScanInternal scans the key range [key,endKey) up to some maximum number -// of results. Specify reverse=true to scan in descending instead of ascending -// order. If iter is not specified, a new iterator is created from engine. func mvccScanInternal( ctx context.Context, engine Reader, @@ -1654,12 +1651,12 @@ func mvccScanInternal( tombstones bool, txn *roachpb.Transaction, reverse bool, -) ([]roachpb.KeyValue, *roachpb.Span, []roachpb.Intent, error) { +) ([]byte, int64, *roachpb.Span, []roachpb.Intent, error) { if max == 0 { - return nil, &roachpb.Span{Key: key, EndKey: endKey}, nil, nil + return nil, 0, &roachpb.Span{Key: key, EndKey: endKey}, nil, nil } if len(endKey) == 0 { - return nil, nil, nil, emptyKeyError() + return nil, 0, nil, nil, emptyKeyError() } var ownIter bool @@ -1687,25 +1684,89 @@ func mvccScanInternal( iter.Close() } if err != nil { - return nil, nil, nil, err + return nil, 0, nil, nil, err } - kvs, resumeKey, intents, err := buildScanResults(kvData, numKvs, intentData, max, consistent) - var resumeSpan *roachpb.Span - if resumeKey != nil { - // NB: we copy the resume key here to ensure that it doesn't point to the - // same shared buffer as the main results. Higher levels of the code may - // cache the resume key and we don't want them pinning excessive amounts of - // memory. - if reverse { - resumeKey = resumeKey[:len(resumeKey):len(resumeKey)] - resumeSpan = &roachpb.Span{Key: key, EndKey: resumeKey.Next()} - } else { - resumeKey = append(roachpb.Key(nil), resumeKey...) - resumeSpan = &roachpb.Span{Key: resumeKey, EndKey: endKey} + intents, err := buildScanIntents(intentData) + if err != nil { + return nil, 0, nil, nil, err + } + if consistent && len(intents) > 0 { + // When encountering intents during a consistent scan we still need to + // return the resume key. + resumeKey, _, err := buildScanResumeKey(kvData, numKvs, max) + if err != nil { + return nil, 0, nil, nil, err } + return nil, 0, resumeKeyToSpan(key, endKey, resumeKey, reverse), nil, &roachpb.WriteIntentError{Intents: intents} + } + + return kvData, numKvs, nil, intents, nil +} + +// mvccScanToBytes scans the key range [key,endKey) up to some maximum number +// of results. Specify reverse=true to scan in descending instead of ascending +// order. If iter is not specified, a new iterator is created from engine. It +// returns the found keys and values in batch format. +func mvccScanToBytes( + ctx context.Context, + engine Reader, + iter Iterator, + key, + endKey roachpb.Key, + max int64, + timestamp hlc.Timestamp, + consistent bool, + tombstones bool, + txn *roachpb.Transaction, + reverse bool, +) ([]byte, int64, *roachpb.Span, []roachpb.Intent, error) { + kvData, numKvs, resumeSpan, intents, err := mvccScanInternal(ctx, engine, iter, key, endKey, max, timestamp, consistent, tombstones, txn, reverse) + if err != nil || resumeSpan != nil { + return nil, 0, resumeSpan, intents, err + } + if len(kvData) == 0 || numKvs == 0 { + return nil, 0, nil, intents, nil + } + + resumeKey, offset, err := buildScanResumeKey(kvData, numKvs, max) + if err != nil { + return nil, 0, nil, nil, nil + } + if resumeKey != nil { + kvData = kvData[:offset] + numKvs = max + } + return kvData, numKvs, resumeKeyToSpan(key, endKey, resumeKey, reverse), intents, err +} + +// mvccScanToKvs scans the key range [key,endKey) up to some maximum number +// of results. Specify reverse=true to scan in descending instead of ascending +// order. If iter is not specified, a new iterator is created from engine. It +// returns the found keys and values in a slice of roachpb.KeyValue. +func mvccScanToKvs( + ctx context.Context, + engine Reader, + iter Iterator, + key, + endKey roachpb.Key, + max int64, + timestamp hlc.Timestamp, + consistent bool, + tombstones bool, + txn *roachpb.Transaction, + reverse bool, +) ([]roachpb.KeyValue, *roachpb.Span, []roachpb.Intent, error) { + kvData, numKvs, resumeSpan, intents, err := mvccScanInternal(ctx, engine, iter, key, endKey, max, timestamp, consistent, tombstones, txn, reverse) + if err != nil || resumeSpan != nil { + return nil, resumeSpan, intents, err + } + if len(kvData) == 0 || numKvs == 0 { + return nil, nil, intents, nil } - return kvs, resumeSpan, intents, err + + kvs, resumeKey, err := buildScanResults(kvData, numKvs, max) + return kvs, resumeKeyToSpan(key, endKey, resumeKey, reverse), intents, err } func buildScanIntents(data []byte) ([]roachpb.Intent, error) { @@ -1741,51 +1802,51 @@ func buildScanIntents(data []byte) ([]roachpb.Intent, error) { return intents, nil } -func buildScanResumeKey(kvData []byte, numKvs int64, max int64) ([]byte, error) { - if len(kvData) == 0 { - return nil, nil +// resumeKeyToSpan creates a resumeSpan suitable for returning in a ScanResponse +// out of a resumeKey (the last key in a batch that exceeded a limit). +func resumeKeyToSpan(key, endKey, resumeKey roachpb.Key, reverse bool) (resumeSpan *roachpb.Span) { + if resumeKey != nil { + // NB: we copy the resume key here to ensure that it doesn't point to the + // same shared buffer as the main results. Higher levels of the code may + // cache the resume key and we don't want them pinning excessive amounts of + // memory. + if reverse { + resumeKey = resumeKey[:len(resumeKey):len(resumeKey)] + resumeSpan = &roachpb.Span{Key: key, EndKey: resumeKey.Next()} + } else { + resumeKey = append(roachpb.Key(nil), resumeKey...) + resumeSpan = &roachpb.Span{Key: resumeKey, EndKey: endKey} + } + } + return resumeSpan +} + +func buildScanResumeKey(kvData []byte, numKvs int64, max int64) ([]byte, int, error) { + kvLen := len(kvData) + if kvLen == 0 { + return nil, 0, nil } if numKvs <= max { - return nil, nil + return nil, 0, nil } var err error for i := int64(0); i < max; i++ { - _, _, kvData, err = mvccScanDecodeKeyValue(kvData) + kvData, err = mvccScanSkipKeyValue(kvData) if err != nil { - return nil, err + return nil, 0, err } } - key, _, _, err := mvccScanDecodeKeyValue(kvData) + offset := kvLen - len(kvData) + key, _, _, err := MVCCScanDecodeKeyValue(kvData) if err != nil { - return nil, err + return nil, 0, err } - return key.Key, nil + return key.Key, offset, nil } func buildScanResults( - kvData []byte, numKvs int64, intentData []byte, max int64, consistent bool, -) ([]roachpb.KeyValue, roachpb.Key, []roachpb.Intent, error) { - intents, err := buildScanIntents(intentData) - if err != nil { - return nil, nil, nil, err - } - if consistent && len(intents) > 0 { - // When encountering intents during a consistent scan we still need to - // return the resume key. - resumeKey, err := buildScanResumeKey(kvData, numKvs, max) - if err != nil { - return nil, nil, nil, err - } - return nil, resumeKey, nil, &roachpb.WriteIntentError{Intents: intents} - } - if len(kvData) == 0 { - return nil, nil, intents, nil - } - - if numKvs == 0 { - return nil, nil, intents, nil - } - + kvData []byte, numKvs int64, max int64, +) ([]roachpb.KeyValue, roachpb.Key, error) { // Iterator.MVCCScan will return up to max+1 results. The extra result is // returned so that we can generate the resumeKey in the same manner as the // old Go version of MVCCScan. @@ -1800,10 +1861,11 @@ func buildScanResults( var key MVCCKey var rawBytes []byte + var err error for i := range kvs { - key, rawBytes, kvData, err = mvccScanDecodeKeyValue(kvData) + key, rawBytes, kvData, err = MVCCScanDecodeKeyValue(kvData) if err != nil { - return nil, nil, nil, err + return nil, nil, err } kvs[i].Key = key.Key kvs[i].Value.RawBytes = rawBytes @@ -1812,14 +1874,14 @@ func buildScanResults( var resumeKey roachpb.Key if numKvs > max { - key, _, _, err = mvccScanDecodeKeyValue(kvData) + key, _, _, err = MVCCScanDecodeKeyValue(kvData) if err != nil { - return nil, nil, nil, err + return nil, nil, err } resumeKey = key.Key } - return kvs, resumeKey, intents, nil + return kvs, resumeKey, nil } // MVCCScan scans the key range [key,endKey) key up to some maximum number of @@ -1837,7 +1899,22 @@ func MVCCScan( consistent bool, txn *roachpb.Transaction, ) ([]roachpb.KeyValue, *roachpb.Span, []roachpb.Intent, error) { - return mvccScanInternal(ctx, engine, nil, key, endKey, max, timestamp, + return mvccScanToKvs(ctx, engine, nil, key, endKey, max, timestamp, + consistent, false /* tombstones */, txn, false /* reverse */) +} + +// MVCCScanToBytes is like MVCCScan, but it returns the results in a byte array. +func MVCCScanToBytes( + ctx context.Context, + engine Reader, + key, + endKey roachpb.Key, + max int64, + timestamp hlc.Timestamp, + consistent bool, + txn *roachpb.Transaction, +) ([]byte, int64, *roachpb.Span, []roachpb.Intent, error) { + return mvccScanToBytes(ctx, engine, nil, key, endKey, max, timestamp, consistent, false /* tombstones */, txn, false /* reverse */) } @@ -1854,7 +1931,23 @@ func MVCCReverseScan( consistent bool, txn *roachpb.Transaction, ) ([]roachpb.KeyValue, *roachpb.Span, []roachpb.Intent, error) { - return mvccScanInternal(ctx, engine, nil, key, endKey, max, timestamp, + return mvccScanToKvs(ctx, engine, nil, key, endKey, max, timestamp, + consistent, false /* tombstones */, txn, true /* reverse */) +} + +// MVCCReverseScanToBytes is like MVCCReverseScan, but it returns the results +// in a byte array. +func MVCCReverseScanToBytes( + ctx context.Context, + engine Reader, + key, + endKey roachpb.Key, + max int64, + timestamp hlc.Timestamp, + consistent bool, + txn *roachpb.Transaction, +) ([]byte, int64, *roachpb.Span, []roachpb.Intent, error) { + return mvccScanToBytes(ctx, engine, nil, key, endKey, max, timestamp, consistent, false /* tombstones */, txn, true /* reverse */) } @@ -1901,7 +1994,7 @@ func MVCCIterateUsingIter( for { const maxKeysPerScan = 1000 - kvs, resume, newIntents, err := mvccScanInternal( + kvs, resume, newIntents, err := mvccScanToKvs( ctx, engine, iter, startKey, endKey, maxKeysPerScan, timestamp, consistent, tombstones, txn, reverse) if err != nil { switch tErr := err.(type) { diff --git a/pkg/storage/engine/rocksdb.go b/pkg/storage/engine/rocksdb.go index aafbdd5f01f1..e3b881a89eb1 100644 --- a/pkg/storage/engine/rocksdb.go +++ b/pkg/storage/engine/rocksdb.go @@ -2113,7 +2113,7 @@ func (r *rocksDBIterator) MVCCGet( // Extract the value from the batch data. repr := copyFromSliceVector(state.data.bufs, state.data.len) - mvccKey, rawValue, _, err := mvccScanDecodeKeyValue(repr) + mvccKey, rawValue, _, err := MVCCScanDecodeKeyValue(repr) if err != nil { return nil, nil, err } @@ -2762,10 +2762,28 @@ func unlockFile(lock C.DBFileLock) error { return statusToError(C.DBUnlockFile(lock)) } -// Decode a key/value pair returned in an MVCCScan "batch" (this is not the -// RocksDB batch repr format), returning both the key/value and the suffix of -// data remaining in the batch. -func mvccScanDecodeKeyValue(repr []byte) (key MVCCKey, value []byte, orepr []byte, err error) { +// mvccScanSkipKeyValue is like MVCCScanDecodeKeyValue, but doesn't bother +// actually decoding the kvs. Instead, it skips the kv and returns the rest of +// the byte buffer. +func mvccScanSkipKeyValue(repr []byte) ([]byte, error) { + if len(repr) < 8 { + return repr, errors.Errorf("unexpected batch EOF") + } + v := binary.LittleEndian.Uint64(repr) + keySize := v >> 32 + valSize := v & ((1 << 32) - 1) + if (keySize + valSize) > uint64(len(repr)) { + return nil, fmt.Errorf("expected %d bytes, but only %d remaining", + keySize+valSize, len(repr)) + } + repr = repr[8+keySize+valSize:] + return repr, nil +} + +// MVCCScanDecodeKeyValue decodes a key/value pair returned in an MVCCScan +// "batch" (this is not the RocksDB batch repr format), returning both the +// key/value and the suffix of data remaining in the batch. +func MVCCScanDecodeKeyValue(repr []byte) (key MVCCKey, value []byte, orepr []byte, err error) { if len(repr) < 8 { return key, nil, repr, errors.Errorf("unexpected batch EOF") }