diff --git a/Tiltfile b/Tiltfile index e91d7aa4ea3..d6c502733b0 100644 --- a/Tiltfile +++ b/Tiltfile @@ -1,26 +1,30 @@ update_settings(max_parallel_updates=6) docker_build( - 'local:migration', - context='.', + 'local:sysdb-migration', + '.', + only=['go/'], dockerfile='./go/Dockerfile.migration' ) docker_build( - 'local:coordinator', - context='.', + 'local:sysdb', + '.', + only=['go/', 'idl/'], dockerfile='./go/Dockerfile' ) docker_build( - 'local:frontend-server', - context='.', + 'local:frontend-service', + '.', + only=['chromadb/', 'idl/', 'requirements.txt', 'bin/'], dockerfile='./Dockerfile', ) docker_build( - 'local:worker', - context='.', + 'local:query-service', + '.', + only=["rust/", "idl/", "Cargo.toml", "Cargo.lock"], dockerfile='./rust/worker/Dockerfile' ) @@ -40,13 +44,13 @@ k8s_yaml([ # Extra stuff to make debugging and testing easier k8s_yaml([ - 'k8s/test/coordinator_service.yaml', - 'k8s/test/jaeger_service.yaml', - 'k8s/test/logservice_service.yaml', + 'k8s/test/sysdb-service.yaml', + 'k8s/test/jaeger-service.yaml', + 'k8s/test/pulsar-service.yaml', + 'k8s/test/logservice-service.yaml', 'k8s/test/minio.yaml', - 'k8s/test/pulsar_service.yaml', - 'k8s/test/worker_service.yaml', - 'k8s/test/test_memberlist_cr.yaml', + 'k8s/test/query-service-service.yaml', + 'k8s/test/test-memberlist-cr.yaml', ]) # Lots of things assume the cluster is in a basic state. Get it into a basic @@ -60,19 +64,19 @@ k8s_resource( objects=[ 'pod-watcher:Role', 'memberlists.chroma.cluster:CustomResourceDefinition', - 'worker-memberlist:MemberList', + 'query-service-memberlist:MemberList', - 'coordinator-serviceaccount:serviceaccount', - 'coordinator-serviceaccount-rolebinding:RoleBinding', - 'coordinator-worker-memberlist-binding:clusterrolebinding', + 'sysdb-serviceaccount:serviceaccount', + 'sysdb-serviceaccount-rolebinding:RoleBinding', + 'sysdb-query-service-memberlist-binding:clusterrolebinding', 'logservice-serviceaccount:serviceaccount', - 'worker-serviceaccount:serviceaccount', - 'worker-serviceaccount-rolebinding:RoleBinding', - 'worker-memberlist-readerwriter:ClusterRole', - 'worker-worker-memberlist-binding:clusterrolebinding', - 'worker-memberlist-readerwriter-binding:clusterrolebinding', + 'query-service-serviceaccount:serviceaccount', + 'query-service-serviceaccount-rolebinding:RoleBinding', + 'query-service-memberlist-readerwriter:ClusterRole', + 'query-service-query-service-memberlist-binding:clusterrolebinding', + 'query-service-memberlist-readerwriter-binding:clusterrolebinding', 'test-memberlist:MemberList', 'test-memberlist-reader:ClusterRole', @@ -84,17 +88,17 @@ k8s_resource( ) # Production Chroma -k8s_resource('postgres', resource_deps=['k8s_setup'], labels=["infrastructure"]) -k8s_resource('pulsar', resource_deps=['k8s_setup'], labels=["infrastructure"], port_forwards=['6650:6650', '8080:8080']) -k8s_resource('migration', resource_deps=['postgres'], labels=["infrastructure"]) -k8s_resource('logservice', resource_deps=['migration'], labels=["chroma"], port_forwards='50052:50051') -k8s_resource('coordinator', resource_deps=['pulsar', 'migration'], labels=["chroma"], port_forwards='50051:50051') -k8s_resource('frontend-server', resource_deps=['pulsar', 'coordinator', 'logservice'],labels=["chroma"], port_forwards='8000:8000') -k8s_resource('worker', resource_deps=['coordinator', 'pulsar'], labels=["chroma"]) +k8s_resource('postgres', resource_deps=['k8s_setup', 'namespace'], labels=["infrastructure"]) +k8s_resource('pulsar', resource_deps=['k8s_setup', 'namespace'], labels=["infrastructure"], port_forwards=['6650:6650', '8080:8080']) +k8s_resource('sysdb-migration', resource_deps=['postgres', 'namespace'], labels=["infrastructure"]) +k8s_resource('logservice', resource_deps=['sysdb-migration'], labels=["chroma"], port_forwards='50052:50051') +k8s_resource('sysdb', resource_deps=['pulsar', 'sysdb-migration'], labels=["chroma"], port_forwards='50051:50051') +k8s_resource('frontend-service', resource_deps=['pulsar', 'sysdb', 'logservice'],labels=["chroma"], port_forwards='8000:8000') +k8s_resource('query-service', resource_deps=['sysdb', 'pulsar'], labels=["chroma"]) # I have no idea why these need their own lines but the others don't. -k8s_resource(objects=['worker:service'], new_name='worker_service', resource_deps=['worker'], labels=["chroma"]) -k8s_resource(objects=['jaeger-lb:Service'], new_name='jaeger_service', resource_deps=['k8s_setup'], labels=["debug"]) +k8s_resource(objects=['query-service:service'], new_name='query-service-service', resource_deps=['query-service'], labels=["chroma"]) +k8s_resource(objects=['jaeger-lb:Service'], new_name='jaeger-service', resource_deps=['k8s_setup'], labels=["debug"]) # Local S3 k8s_resource('minio-deployment', resource_deps=['k8s_setup'], labels=["debug"], port_forwards='9000:9000') diff --git a/bin/cluster-test.sh b/bin/cluster-test.sh index 375ca464d00..5f86a78e2c6 100755 --- a/bin/cluster-test.sh +++ b/bin/cluster-test.sh @@ -11,10 +11,10 @@ echo "Chroma Server is running at port $CHROMA_SERVER_HOST" echo "Pulsar Broker is running at port $PULSAR_BROKER_URL" echo "Chroma Coordinator is running at port $CHROMA_COORDINATOR_HOST" -kubectl -n chroma port-forward svc/coordinator-lb 50051:50051 & +kubectl -n chroma port-forward svc/sysdb-lb 50051:50051 & kubectl -n chroma port-forward svc/logservice-lb 50052:50051 & kubectl -n chroma port-forward svc/pulsar-lb 6650:6650 & kubectl -n chroma port-forward svc/pulsar-lb 8080:8080 & -kubectl -n chroma port-forward svc/frontend-server 8000:8000 & +kubectl -n chroma port-forward svc/frontend-service 8000:8000 & "$@" diff --git a/chromadb/config.py b/chromadb/config.py index 9f9c2f50e45..d0e6e45a00f 100644 --- a/chromadb/config.py +++ b/chromadb/config.py @@ -112,7 +112,7 @@ class Settings(BaseSettings): # type: ignore chroma_collection_assignment_policy_impl: str = ( "chromadb.ingest.impl.simple_policy.SimpleAssignmentPolicy" ) - worker_memberlist_name: str = "worker-memberlist" + worker_memberlist_name: str = "query-service-memberlist" chroma_coordinator_host = "localhost" chroma_logservice_host = "localhost" @@ -331,11 +331,11 @@ def __init__(self, settings: Settings): and settings["chroma_segment_cache_policy"] != "LRU" ): logger.error( - f"Failed to set chroma_segment_cache_policy: Only LRU is available." + "Failed to set chroma_segment_cache_policy: Only LRU is available." ) if settings["chroma_memory_limit_bytes"] == 0: logger.error( - f"Failed to set chroma_segment_cache_policy: chroma_memory_limit_bytes is require." + "Failed to set chroma_segment_cache_policy: chroma_memory_limit_bytes is require." ) # Apply the nofile limit if set diff --git a/chromadb/logservice/logservice.py b/chromadb/logservice/logservice.py index 9683d170a95..83161d519db 100644 --- a/chromadb/logservice/logservice.py +++ b/chromadb/logservice/logservice.py @@ -166,6 +166,7 @@ def pull_logs( collection_id=str(collection_id), start_from_id=start_id, batch_size=batch_size, + end_timestamp=-1, ) response = self._log_service_stub.PullLogs(request) return response.records # type: ignore diff --git a/chromadb/proto/chroma_pb2.py b/chromadb/proto/chroma_pb2.py index 48b64144192..d54e7c6e22d 100644 --- a/chromadb/proto/chroma_pb2.py +++ b/chromadb/proto/chroma_pb2.py @@ -13,7 +13,7 @@ -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1b\x63hromadb/proto/chroma.proto\x12\x06\x63hroma\"&\n\x06Status\x12\x0e\n\x06reason\x18\x01 \x01(\t\x12\x0c\n\x04\x63ode\x18\x02 \x01(\x05\"U\n\x06Vector\x12\x11\n\tdimension\x18\x01 \x01(\x05\x12\x0e\n\x06vector\x18\x02 \x01(\x0c\x12(\n\x08\x65ncoding\x18\x03 \x01(\x0e\x32\x16.chroma.ScalarEncoding\"\xca\x01\n\x07Segment\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04type\x18\x02 \x01(\t\x12#\n\x05scope\x18\x03 \x01(\x0e\x32\x14.chroma.SegmentScope\x12\x12\n\x05topic\x18\x04 \x01(\tH\x00\x88\x01\x01\x12\x17\n\ncollection\x18\x05 \x01(\tH\x01\x88\x01\x01\x12-\n\x08metadata\x18\x06 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x02\x88\x01\x01\x42\x08\n\x06_topicB\r\n\x0b_collectionB\x0b\n\t_metadata\"\xb9\x01\n\nCollection\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\r\n\x05topic\x18\x03 \x01(\t\x12-\n\x08metadata\x18\x04 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x88\x01\x01\x12\x16\n\tdimension\x18\x05 \x01(\x05H\x01\x88\x01\x01\x12\x0e\n\x06tenant\x18\x06 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x07 \x01(\tB\x0b\n\t_metadataB\x0c\n\n_dimension\"4\n\x08\x44\x61tabase\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x0e\n\x06tenant\x18\x03 \x01(\t\"\x16\n\x06Tenant\x12\x0c\n\x04name\x18\x01 \x01(\t\"b\n\x13UpdateMetadataValue\x12\x16\n\x0cstring_value\x18\x01 \x01(\tH\x00\x12\x13\n\tint_value\x18\x02 \x01(\x03H\x00\x12\x15\n\x0b\x66loat_value\x18\x03 \x01(\x01H\x00\x42\x07\n\x05value\"\x96\x01\n\x0eUpdateMetadata\x12\x36\n\x08metadata\x18\x01 \x03(\x0b\x32$.chroma.UpdateMetadata.MetadataEntry\x1aL\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12*\n\x05value\x18\x02 \x01(\x0b\x32\x1b.chroma.UpdateMetadataValue:\x02\x38\x01\"\xcc\x01\n\x15SubmitEmbeddingRecord\x12\n\n\x02id\x18\x01 \x01(\t\x12#\n\x06vector\x18\x02 \x01(\x0b\x32\x0e.chroma.VectorH\x00\x88\x01\x01\x12-\n\x08metadata\x18\x03 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x01\x88\x01\x01\x12$\n\toperation\x18\x04 \x01(\x0e\x32\x11.chroma.Operation\x12\x15\n\rcollection_id\x18\x05 \x01(\tB\t\n\x07_vectorB\x0b\n\t_metadata\"S\n\x15VectorEmbeddingRecord\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06seq_id\x18\x02 \x01(\x0c\x12\x1e\n\x06vector\x18\x03 \x01(\x0b\x32\x0e.chroma.Vector\"q\n\x11VectorQueryResult\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06seq_id\x18\x02 \x01(\x0c\x12\x10\n\x08\x64istance\x18\x03 \x01(\x02\x12#\n\x06vector\x18\x04 \x01(\x0b\x32\x0e.chroma.VectorH\x00\x88\x01\x01\x42\t\n\x07_vector\"@\n\x12VectorQueryResults\x12*\n\x07results\x18\x01 \x03(\x0b\x32\x19.chroma.VectorQueryResult\"4\n\x11GetVectorsRequest\x12\x0b\n\x03ids\x18\x01 \x03(\t\x12\x12\n\nsegment_id\x18\x02 \x01(\t\"D\n\x12GetVectorsResponse\x12.\n\x07records\x18\x01 \x03(\x0b\x32\x1d.chroma.VectorEmbeddingRecord\"\x86\x01\n\x13QueryVectorsRequest\x12\x1f\n\x07vectors\x18\x01 \x03(\x0b\x32\x0e.chroma.Vector\x12\t\n\x01k\x18\x02 \x01(\x05\x12\x13\n\x0b\x61llowed_ids\x18\x03 \x03(\t\x12\x1a\n\x12include_embeddings\x18\x04 \x01(\x08\x12\x12\n\nsegment_id\x18\x05 \x01(\t\"C\n\x14QueryVectorsResponse\x12+\n\x07results\x18\x01 \x03(\x0b\x32\x1a.chroma.VectorQueryResults*8\n\tOperation\x12\x07\n\x03\x41\x44\x44\x10\x00\x12\n\n\x06UPDATE\x10\x01\x12\n\n\x06UPSERT\x10\x02\x12\n\n\x06\x44\x45LETE\x10\x03*(\n\x0eScalarEncoding\x12\x0b\n\x07\x46LOAT32\x10\x00\x12\t\n\x05INT32\x10\x01*(\n\x0cSegmentScope\x12\n\n\x06VECTOR\x10\x00\x12\x0c\n\x08METADATA\x10\x01\x32\xa2\x01\n\x0cVectorReader\x12\x45\n\nGetVectors\x12\x19.chroma.GetVectorsRequest\x1a\x1a.chroma.GetVectorsResponse\"\x00\x12K\n\x0cQueryVectors\x12\x1b.chroma.QueryVectorsRequest\x1a\x1c.chroma.QueryVectorsResponse\"\x00\x42:Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpbb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1b\x63hromadb/proto/chroma.proto\x12\x06\x63hroma\"&\n\x06Status\x12\x0e\n\x06reason\x18\x01 \x01(\t\x12\x0c\n\x04\x63ode\x18\x02 \x01(\x05\"U\n\x06Vector\x12\x11\n\tdimension\x18\x01 \x01(\x05\x12\x0e\n\x06vector\x18\x02 \x01(\x0c\x12(\n\x08\x65ncoding\x18\x03 \x01(\x0e\x32\x16.chroma.ScalarEncoding\"\x1a\n\tFilePaths\x12\r\n\x05paths\x18\x01 \x03(\t\"\xc3\x02\n\x07Segment\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04type\x18\x02 \x01(\t\x12#\n\x05scope\x18\x03 \x01(\x0e\x32\x14.chroma.SegmentScope\x12\x12\n\x05topic\x18\x04 \x01(\tH\x00\x88\x01\x01\x12\x17\n\ncollection\x18\x05 \x01(\tH\x01\x88\x01\x01\x12-\n\x08metadata\x18\x06 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x02\x88\x01\x01\x12\x32\n\nfile_paths\x18\x07 \x03(\x0b\x32\x1e.chroma.Segment.FilePathsEntry\x1a\x43\n\x0e\x46ilePathsEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12 \n\x05value\x18\x02 \x01(\x0b\x32\x11.chroma.FilePaths:\x02\x38\x01\x42\x08\n\x06_topicB\r\n\x0b_collectionB\x0b\n\t_metadata\"\xdf\x01\n\nCollection\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\r\n\x05topic\x18\x03 \x01(\t\x12-\n\x08metadata\x18\x04 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x88\x01\x01\x12\x16\n\tdimension\x18\x05 \x01(\x05H\x01\x88\x01\x01\x12\x0e\n\x06tenant\x18\x06 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x07 \x01(\t\x12\x13\n\x0blogPosition\x18\x08 \x01(\x03\x12\x0f\n\x07version\x18\t \x01(\x05\x42\x0b\n\t_metadataB\x0c\n\n_dimension\"4\n\x08\x44\x61tabase\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x0e\n\x06tenant\x18\x03 \x01(\t\"\x16\n\x06Tenant\x12\x0c\n\x04name\x18\x01 \x01(\t\"b\n\x13UpdateMetadataValue\x12\x16\n\x0cstring_value\x18\x01 \x01(\tH\x00\x12\x13\n\tint_value\x18\x02 \x01(\x03H\x00\x12\x15\n\x0b\x66loat_value\x18\x03 \x01(\x01H\x00\x42\x07\n\x05value\"\x96\x01\n\x0eUpdateMetadata\x12\x36\n\x08metadata\x18\x01 \x03(\x0b\x32$.chroma.UpdateMetadata.MetadataEntry\x1aL\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12*\n\x05value\x18\x02 \x01(\x0b\x32\x1b.chroma.UpdateMetadataValue:\x02\x38\x01\"\xcc\x01\n\x15SubmitEmbeddingRecord\x12\n\n\x02id\x18\x01 \x01(\t\x12#\n\x06vector\x18\x02 \x01(\x0b\x32\x0e.chroma.VectorH\x00\x88\x01\x01\x12-\n\x08metadata\x18\x03 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x01\x88\x01\x01\x12$\n\toperation\x18\x04 \x01(\x0e\x32\x11.chroma.Operation\x12\x15\n\rcollection_id\x18\x05 \x01(\tB\t\n\x07_vectorB\x0b\n\t_metadata\"S\n\x15VectorEmbeddingRecord\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06seq_id\x18\x02 \x01(\x0c\x12\x1e\n\x06vector\x18\x03 \x01(\x0b\x32\x0e.chroma.Vector\"q\n\x11VectorQueryResult\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06seq_id\x18\x02 \x01(\x0c\x12\x10\n\x08\x64istance\x18\x03 \x01(\x02\x12#\n\x06vector\x18\x04 \x01(\x0b\x32\x0e.chroma.VectorH\x00\x88\x01\x01\x42\t\n\x07_vector\"@\n\x12VectorQueryResults\x12*\n\x07results\x18\x01 \x03(\x0b\x32\x19.chroma.VectorQueryResult\"4\n\x11GetVectorsRequest\x12\x0b\n\x03ids\x18\x01 \x03(\t\x12\x12\n\nsegment_id\x18\x02 \x01(\t\"D\n\x12GetVectorsResponse\x12.\n\x07records\x18\x01 \x03(\x0b\x32\x1d.chroma.VectorEmbeddingRecord\"\x86\x01\n\x13QueryVectorsRequest\x12\x1f\n\x07vectors\x18\x01 \x03(\x0b\x32\x0e.chroma.Vector\x12\t\n\x01k\x18\x02 \x01(\x05\x12\x13\n\x0b\x61llowed_ids\x18\x03 \x03(\t\x12\x1a\n\x12include_embeddings\x18\x04 \x01(\x08\x12\x12\n\nsegment_id\x18\x05 \x01(\t\"C\n\x14QueryVectorsResponse\x12+\n\x07results\x18\x01 \x03(\x0b\x32\x1a.chroma.VectorQueryResults*8\n\tOperation\x12\x07\n\x03\x41\x44\x44\x10\x00\x12\n\n\x06UPDATE\x10\x01\x12\n\n\x06UPSERT\x10\x02\x12\n\n\x06\x44\x45LETE\x10\x03*(\n\x0eScalarEncoding\x12\x0b\n\x07\x46LOAT32\x10\x00\x12\t\n\x05INT32\x10\x01*(\n\x0cSegmentScope\x12\n\n\x06VECTOR\x10\x00\x12\x0c\n\x08METADATA\x10\x01\x32\xa2\x01\n\x0cVectorReader\x12\x45\n\nGetVectors\x12\x19.chroma.GetVectorsRequest\x1a\x1a.chroma.GetVectorsResponse\"\x00\x12K\n\x0cQueryVectors\x12\x1b.chroma.QueryVectorsRequest\x1a\x1c.chroma.QueryVectorsResponse\"\x00\x42:Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpbb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -21,48 +21,54 @@ if _descriptor._USE_C_DESCRIPTORS == False: DESCRIPTOR._options = None DESCRIPTOR._serialized_options = b'Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpb' + _SEGMENT_FILEPATHSENTRY._options = None + _SEGMENT_FILEPATHSENTRY._serialized_options = b'8\001' _UPDATEMETADATA_METADATAENTRY._options = None _UPDATEMETADATA_METADATAENTRY._serialized_options = b'8\001' - _globals['_OPERATION']._serialized_start=1693 - _globals['_OPERATION']._serialized_end=1749 - _globals['_SCALARENCODING']._serialized_start=1751 - _globals['_SCALARENCODING']._serialized_end=1791 - _globals['_SEGMENTSCOPE']._serialized_start=1793 - _globals['_SEGMENTSCOPE']._serialized_end=1833 + _globals['_OPERATION']._serialized_start=1880 + _globals['_OPERATION']._serialized_end=1936 + _globals['_SCALARENCODING']._serialized_start=1938 + _globals['_SCALARENCODING']._serialized_end=1978 + _globals['_SEGMENTSCOPE']._serialized_start=1980 + _globals['_SEGMENTSCOPE']._serialized_end=2020 _globals['_STATUS']._serialized_start=39 _globals['_STATUS']._serialized_end=77 _globals['_VECTOR']._serialized_start=79 _globals['_VECTOR']._serialized_end=164 - _globals['_SEGMENT']._serialized_start=167 - _globals['_SEGMENT']._serialized_end=369 - _globals['_COLLECTION']._serialized_start=372 - _globals['_COLLECTION']._serialized_end=557 - _globals['_DATABASE']._serialized_start=559 - _globals['_DATABASE']._serialized_end=611 - _globals['_TENANT']._serialized_start=613 - _globals['_TENANT']._serialized_end=635 - _globals['_UPDATEMETADATAVALUE']._serialized_start=637 - _globals['_UPDATEMETADATAVALUE']._serialized_end=735 - _globals['_UPDATEMETADATA']._serialized_start=738 - _globals['_UPDATEMETADATA']._serialized_end=888 - _globals['_UPDATEMETADATA_METADATAENTRY']._serialized_start=812 - _globals['_UPDATEMETADATA_METADATAENTRY']._serialized_end=888 - _globals['_SUBMITEMBEDDINGRECORD']._serialized_start=891 - _globals['_SUBMITEMBEDDINGRECORD']._serialized_end=1095 - _globals['_VECTOREMBEDDINGRECORD']._serialized_start=1097 - _globals['_VECTOREMBEDDINGRECORD']._serialized_end=1180 - _globals['_VECTORQUERYRESULT']._serialized_start=1182 - _globals['_VECTORQUERYRESULT']._serialized_end=1295 - _globals['_VECTORQUERYRESULTS']._serialized_start=1297 - _globals['_VECTORQUERYRESULTS']._serialized_end=1361 - _globals['_GETVECTORSREQUEST']._serialized_start=1363 - _globals['_GETVECTORSREQUEST']._serialized_end=1415 - _globals['_GETVECTORSRESPONSE']._serialized_start=1417 - _globals['_GETVECTORSRESPONSE']._serialized_end=1485 - _globals['_QUERYVECTORSREQUEST']._serialized_start=1488 - _globals['_QUERYVECTORSREQUEST']._serialized_end=1622 - _globals['_QUERYVECTORSRESPONSE']._serialized_start=1624 - _globals['_QUERYVECTORSRESPONSE']._serialized_end=1691 - _globals['_VECTORREADER']._serialized_start=1836 - _globals['_VECTORREADER']._serialized_end=1998 + _globals['_FILEPATHS']._serialized_start=166 + _globals['_FILEPATHS']._serialized_end=192 + _globals['_SEGMENT']._serialized_start=195 + _globals['_SEGMENT']._serialized_end=518 + _globals['_SEGMENT_FILEPATHSENTRY']._serialized_start=413 + _globals['_SEGMENT_FILEPATHSENTRY']._serialized_end=480 + _globals['_COLLECTION']._serialized_start=521 + _globals['_COLLECTION']._serialized_end=744 + _globals['_DATABASE']._serialized_start=746 + _globals['_DATABASE']._serialized_end=798 + _globals['_TENANT']._serialized_start=800 + _globals['_TENANT']._serialized_end=822 + _globals['_UPDATEMETADATAVALUE']._serialized_start=824 + _globals['_UPDATEMETADATAVALUE']._serialized_end=922 + _globals['_UPDATEMETADATA']._serialized_start=925 + _globals['_UPDATEMETADATA']._serialized_end=1075 + _globals['_UPDATEMETADATA_METADATAENTRY']._serialized_start=999 + _globals['_UPDATEMETADATA_METADATAENTRY']._serialized_end=1075 + _globals['_SUBMITEMBEDDINGRECORD']._serialized_start=1078 + _globals['_SUBMITEMBEDDINGRECORD']._serialized_end=1282 + _globals['_VECTOREMBEDDINGRECORD']._serialized_start=1284 + _globals['_VECTOREMBEDDINGRECORD']._serialized_end=1367 + _globals['_VECTORQUERYRESULT']._serialized_start=1369 + _globals['_VECTORQUERYRESULT']._serialized_end=1482 + _globals['_VECTORQUERYRESULTS']._serialized_start=1484 + _globals['_VECTORQUERYRESULTS']._serialized_end=1548 + _globals['_GETVECTORSREQUEST']._serialized_start=1550 + _globals['_GETVECTORSREQUEST']._serialized_end=1602 + _globals['_GETVECTORSRESPONSE']._serialized_start=1604 + _globals['_GETVECTORSRESPONSE']._serialized_end=1672 + _globals['_QUERYVECTORSREQUEST']._serialized_start=1675 + _globals['_QUERYVECTORSREQUEST']._serialized_end=1809 + _globals['_QUERYVECTORSRESPONSE']._serialized_start=1811 + _globals['_QUERYVECTORSRESPONSE']._serialized_end=1878 + _globals['_VECTORREADER']._serialized_start=2023 + _globals['_VECTORREADER']._serialized_end=2185 # @@protoc_insertion_point(module_scope) diff --git a/chromadb/proto/chroma_pb2.pyi b/chromadb/proto/chroma_pb2.pyi index 9fb730ca6d9..6e8b267a584 100644 --- a/chromadb/proto/chroma_pb2.pyi +++ b/chromadb/proto/chroma_pb2.pyi @@ -49,24 +49,39 @@ class Vector(_message.Message): encoding: ScalarEncoding def __init__(self, dimension: _Optional[int] = ..., vector: _Optional[bytes] = ..., encoding: _Optional[_Union[ScalarEncoding, str]] = ...) -> None: ... +class FilePaths(_message.Message): + __slots__ = ["paths"] + PATHS_FIELD_NUMBER: _ClassVar[int] + paths: _containers.RepeatedScalarFieldContainer[str] + def __init__(self, paths: _Optional[_Iterable[str]] = ...) -> None: ... + class Segment(_message.Message): - __slots__ = ["id", "type", "scope", "topic", "collection", "metadata"] + __slots__ = ["id", "type", "scope", "topic", "collection", "metadata", "file_paths"] + class FilePathsEntry(_message.Message): + __slots__ = ["key", "value"] + KEY_FIELD_NUMBER: _ClassVar[int] + VALUE_FIELD_NUMBER: _ClassVar[int] + key: str + value: FilePaths + def __init__(self, key: _Optional[str] = ..., value: _Optional[_Union[FilePaths, _Mapping]] = ...) -> None: ... ID_FIELD_NUMBER: _ClassVar[int] TYPE_FIELD_NUMBER: _ClassVar[int] SCOPE_FIELD_NUMBER: _ClassVar[int] TOPIC_FIELD_NUMBER: _ClassVar[int] COLLECTION_FIELD_NUMBER: _ClassVar[int] METADATA_FIELD_NUMBER: _ClassVar[int] + FILE_PATHS_FIELD_NUMBER: _ClassVar[int] id: str type: str scope: SegmentScope topic: str collection: str metadata: UpdateMetadata - def __init__(self, id: _Optional[str] = ..., type: _Optional[str] = ..., scope: _Optional[_Union[SegmentScope, str]] = ..., topic: _Optional[str] = ..., collection: _Optional[str] = ..., metadata: _Optional[_Union[UpdateMetadata, _Mapping]] = ...) -> None: ... + file_paths: _containers.MessageMap[str, FilePaths] + def __init__(self, id: _Optional[str] = ..., type: _Optional[str] = ..., scope: _Optional[_Union[SegmentScope, str]] = ..., topic: _Optional[str] = ..., collection: _Optional[str] = ..., metadata: _Optional[_Union[UpdateMetadata, _Mapping]] = ..., file_paths: _Optional[_Mapping[str, FilePaths]] = ...) -> None: ... class Collection(_message.Message): - __slots__ = ["id", "name", "topic", "metadata", "dimension", "tenant", "database"] + __slots__ = ["id", "name", "topic", "metadata", "dimension", "tenant", "database", "logPosition", "version"] ID_FIELD_NUMBER: _ClassVar[int] NAME_FIELD_NUMBER: _ClassVar[int] TOPIC_FIELD_NUMBER: _ClassVar[int] @@ -74,6 +89,8 @@ class Collection(_message.Message): DIMENSION_FIELD_NUMBER: _ClassVar[int] TENANT_FIELD_NUMBER: _ClassVar[int] DATABASE_FIELD_NUMBER: _ClassVar[int] + LOGPOSITION_FIELD_NUMBER: _ClassVar[int] + VERSION_FIELD_NUMBER: _ClassVar[int] id: str name: str topic: str @@ -81,7 +98,9 @@ class Collection(_message.Message): dimension: int tenant: str database: str - def __init__(self, id: _Optional[str] = ..., name: _Optional[str] = ..., topic: _Optional[str] = ..., metadata: _Optional[_Union[UpdateMetadata, _Mapping]] = ..., dimension: _Optional[int] = ..., tenant: _Optional[str] = ..., database: _Optional[str] = ...) -> None: ... + logPosition: int + version: int + def __init__(self, id: _Optional[str] = ..., name: _Optional[str] = ..., topic: _Optional[str] = ..., metadata: _Optional[_Union[UpdateMetadata, _Mapping]] = ..., dimension: _Optional[int] = ..., tenant: _Optional[str] = ..., database: _Optional[str] = ..., logPosition: _Optional[int] = ..., version: _Optional[int] = ...) -> None: ... class Database(_message.Message): __slots__ = ["id", "name", "tenant"] diff --git a/chromadb/proto/coordinator_pb2.py b/chromadb/proto/coordinator_pb2.py index 301c1c2f4f7..7264a86f038 100644 --- a/chromadb/proto/coordinator_pb2.py +++ b/chromadb/proto/coordinator_pb2.py @@ -15,7 +15,7 @@ from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n chromadb/proto/coordinator.proto\x12\x06\x63hroma\x1a\x1b\x63hromadb/proto/chroma.proto\x1a\x1bgoogle/protobuf/empty.proto\"A\n\x15\x43reateDatabaseRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x0e\n\x06tenant\x18\x03 \x01(\t\"8\n\x16\x43reateDatabaseResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"2\n\x12GetDatabaseRequest\x12\x0c\n\x04name\x18\x01 \x01(\t\x12\x0e\n\x06tenant\x18\x02 \x01(\t\"Y\n\x13GetDatabaseResponse\x12\"\n\x08\x64\x61tabase\x18\x01 \x01(\x0b\x32\x10.chroma.Database\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"#\n\x13\x43reateTenantRequest\x12\x0c\n\x04name\x18\x02 \x01(\t\"6\n\x14\x43reateTenantResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\" \n\x10GetTenantRequest\x12\x0c\n\x04name\x18\x01 \x01(\t\"S\n\x11GetTenantResponse\x12\x1e\n\x06tenant\x18\x01 \x01(\x0b\x32\x0e.chroma.Tenant\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"8\n\x14\x43reateSegmentRequest\x12 \n\x07segment\x18\x01 \x01(\x0b\x32\x0f.chroma.Segment\"7\n\x15\x43reateSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\"\n\x14\x44\x65leteSegmentRequest\x12\n\n\x02id\x18\x01 \x01(\t\"7\n\x15\x44\x65leteSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\xc2\x01\n\x12GetSegmentsRequest\x12\x0f\n\x02id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x11\n\x04type\x18\x02 \x01(\tH\x01\x88\x01\x01\x12(\n\x05scope\x18\x03 \x01(\x0e\x32\x14.chroma.SegmentScopeH\x02\x88\x01\x01\x12\x12\n\x05topic\x18\x04 \x01(\tH\x03\x88\x01\x01\x12\x17\n\ncollection\x18\x05 \x01(\tH\x04\x88\x01\x01\x42\x05\n\x03_idB\x07\n\x05_typeB\x08\n\x06_scopeB\x08\n\x06_topicB\r\n\x0b_collection\"X\n\x13GetSegmentsResponse\x12!\n\x08segments\x18\x01 \x03(\x0b\x32\x0f.chroma.Segment\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"\xfa\x01\n\x14UpdateSegmentRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0f\n\x05topic\x18\x02 \x01(\tH\x00\x12\x15\n\x0breset_topic\x18\x03 \x01(\x08H\x00\x12\x14\n\ncollection\x18\x04 \x01(\tH\x01\x12\x1a\n\x10reset_collection\x18\x05 \x01(\x08H\x01\x12*\n\x08metadata\x18\x06 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x02\x12\x18\n\x0ereset_metadata\x18\x07 \x01(\x08H\x02\x42\x0e\n\x0ctopic_updateB\x13\n\x11\x63ollection_updateB\x11\n\x0fmetadata_update\"7\n\x15UpdateSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\xe5\x01\n\x17\x43reateCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12-\n\x08metadata\x18\x03 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x88\x01\x01\x12\x16\n\tdimension\x18\x04 \x01(\x05H\x01\x88\x01\x01\x12\x1a\n\rget_or_create\x18\x05 \x01(\x08H\x02\x88\x01\x01\x12\x0e\n\x06tenant\x18\x06 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x07 \x01(\tB\x0b\n\t_metadataB\x0c\n\n_dimensionB\x10\n\x0e_get_or_create\"s\n\x18\x43reateCollectionResponse\x12&\n\ncollection\x18\x01 \x01(\x0b\x32\x12.chroma.Collection\x12\x0f\n\x07\x63reated\x18\x02 \x01(\x08\x12\x1e\n\x06status\x18\x03 \x01(\x0b\x32\x0e.chroma.Status\"G\n\x17\x44\x65leteCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06tenant\x18\x02 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x03 \x01(\t\":\n\x18\x44\x65leteCollectionResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\x8b\x01\n\x15GetCollectionsRequest\x12\x0f\n\x02id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x11\n\x04name\x18\x02 \x01(\tH\x01\x88\x01\x01\x12\x12\n\x05topic\x18\x03 \x01(\tH\x02\x88\x01\x01\x12\x0e\n\x06tenant\x18\x04 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x05 \x01(\tB\x05\n\x03_idB\x07\n\x05_nameB\x08\n\x06_topic\"a\n\x16GetCollectionsResponse\x12\'\n\x0b\x63ollections\x18\x01 \x03(\x0b\x32\x12.chroma.Collection\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"\xde\x01\n\x17UpdateCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x12\n\x05topic\x18\x02 \x01(\tH\x01\x88\x01\x01\x12\x11\n\x04name\x18\x03 \x01(\tH\x02\x88\x01\x01\x12\x16\n\tdimension\x18\x04 \x01(\x05H\x03\x88\x01\x01\x12*\n\x08metadata\x18\x05 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x12\x18\n\x0ereset_metadata\x18\x06 \x01(\x08H\x00\x42\x11\n\x0fmetadata_updateB\x08\n\x06_topicB\x07\n\x05_nameB\x0c\n\n_dimension\":\n\x18UpdateCollectionResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"O\n\x0cNotification\x12\n\n\x02id\x18\x01 \x01(\x03\x12\x15\n\rcollection_id\x18\x02 \x01(\t\x12\x0c\n\x04type\x18\x03 \x01(\t\x12\x0e\n\x06status\x18\x04 \x01(\t\"4\n\x12ResetStateResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\":\n%GetLastCompactionTimeForTenantRequest\x12\x11\n\ttenant_id\x18\x01 \x03(\t\"K\n\x18TenantLastCompactionTime\x12\x11\n\ttenant_id\x18\x01 \x01(\t\x12\x1c\n\x14last_compaction_time\x18\x02 \x01(\x03\"o\n&GetLastCompactionTimeForTenantResponse\x12\x45\n\x1btenant_last_compaction_time\x18\x01 \x03(\x0b\x32 .chroma.TenantLastCompactionTime\"n\n%SetLastCompactionTimeForTenantRequest\x12\x45\n\x1btenant_last_compaction_time\x18\x01 \x01(\x0b\x32 .chroma.TenantLastCompactionTime2\x80\n\n\x05SysDB\x12Q\n\x0e\x43reateDatabase\x12\x1d.chroma.CreateDatabaseRequest\x1a\x1e.chroma.CreateDatabaseResponse\"\x00\x12H\n\x0bGetDatabase\x12\x1a.chroma.GetDatabaseRequest\x1a\x1b.chroma.GetDatabaseResponse\"\x00\x12K\n\x0c\x43reateTenant\x12\x1b.chroma.CreateTenantRequest\x1a\x1c.chroma.CreateTenantResponse\"\x00\x12\x42\n\tGetTenant\x12\x18.chroma.GetTenantRequest\x1a\x19.chroma.GetTenantResponse\"\x00\x12N\n\rCreateSegment\x12\x1c.chroma.CreateSegmentRequest\x1a\x1d.chroma.CreateSegmentResponse\"\x00\x12N\n\rDeleteSegment\x12\x1c.chroma.DeleteSegmentRequest\x1a\x1d.chroma.DeleteSegmentResponse\"\x00\x12H\n\x0bGetSegments\x12\x1a.chroma.GetSegmentsRequest\x1a\x1b.chroma.GetSegmentsResponse\"\x00\x12N\n\rUpdateSegment\x12\x1c.chroma.UpdateSegmentRequest\x1a\x1d.chroma.UpdateSegmentResponse\"\x00\x12W\n\x10\x43reateCollection\x12\x1f.chroma.CreateCollectionRequest\x1a .chroma.CreateCollectionResponse\"\x00\x12W\n\x10\x44\x65leteCollection\x12\x1f.chroma.DeleteCollectionRequest\x1a .chroma.DeleteCollectionResponse\"\x00\x12Q\n\x0eGetCollections\x12\x1d.chroma.GetCollectionsRequest\x1a\x1e.chroma.GetCollectionsResponse\"\x00\x12W\n\x10UpdateCollection\x12\x1f.chroma.UpdateCollectionRequest\x1a .chroma.UpdateCollectionResponse\"\x00\x12\x42\n\nResetState\x12\x16.google.protobuf.Empty\x1a\x1a.chroma.ResetStateResponse\"\x00\x12\x81\x01\n\x1eGetLastCompactionTimeForTenant\x12-.chroma.GetLastCompactionTimeForTenantRequest\x1a..chroma.GetLastCompactionTimeForTenantResponse\"\x00\x12i\n\x1eSetLastCompactionTimeForTenant\x12-.chroma.SetLastCompactionTimeForTenantRequest\x1a\x16.google.protobuf.Empty\"\x00\x42:Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpbb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n chromadb/proto/coordinator.proto\x12\x06\x63hroma\x1a\x1b\x63hromadb/proto/chroma.proto\x1a\x1bgoogle/protobuf/empty.proto\"A\n\x15\x43reateDatabaseRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x0e\n\x06tenant\x18\x03 \x01(\t\"8\n\x16\x43reateDatabaseResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"2\n\x12GetDatabaseRequest\x12\x0c\n\x04name\x18\x01 \x01(\t\x12\x0e\n\x06tenant\x18\x02 \x01(\t\"Y\n\x13GetDatabaseResponse\x12\"\n\x08\x64\x61tabase\x18\x01 \x01(\x0b\x32\x10.chroma.Database\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"#\n\x13\x43reateTenantRequest\x12\x0c\n\x04name\x18\x02 \x01(\t\"6\n\x14\x43reateTenantResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\" \n\x10GetTenantRequest\x12\x0c\n\x04name\x18\x01 \x01(\t\"S\n\x11GetTenantResponse\x12\x1e\n\x06tenant\x18\x01 \x01(\x0b\x32\x0e.chroma.Tenant\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"8\n\x14\x43reateSegmentRequest\x12 \n\x07segment\x18\x01 \x01(\x0b\x32\x0f.chroma.Segment\"7\n\x15\x43reateSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\"\n\x14\x44\x65leteSegmentRequest\x12\n\n\x02id\x18\x01 \x01(\t\"7\n\x15\x44\x65leteSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\xc2\x01\n\x12GetSegmentsRequest\x12\x0f\n\x02id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x11\n\x04type\x18\x02 \x01(\tH\x01\x88\x01\x01\x12(\n\x05scope\x18\x03 \x01(\x0e\x32\x14.chroma.SegmentScopeH\x02\x88\x01\x01\x12\x12\n\x05topic\x18\x04 \x01(\tH\x03\x88\x01\x01\x12\x17\n\ncollection\x18\x05 \x01(\tH\x04\x88\x01\x01\x42\x05\n\x03_idB\x07\n\x05_typeB\x08\n\x06_scopeB\x08\n\x06_topicB\r\n\x0b_collection\"X\n\x13GetSegmentsResponse\x12!\n\x08segments\x18\x01 \x03(\x0b\x32\x0f.chroma.Segment\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"\xfa\x01\n\x14UpdateSegmentRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0f\n\x05topic\x18\x02 \x01(\tH\x00\x12\x15\n\x0breset_topic\x18\x03 \x01(\x08H\x00\x12\x14\n\ncollection\x18\x04 \x01(\tH\x01\x12\x1a\n\x10reset_collection\x18\x05 \x01(\x08H\x01\x12*\n\x08metadata\x18\x06 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x02\x12\x18\n\x0ereset_metadata\x18\x07 \x01(\x08H\x02\x42\x0e\n\x0ctopic_updateB\x13\n\x11\x63ollection_updateB\x11\n\x0fmetadata_update\"7\n\x15UpdateSegmentResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\xe5\x01\n\x17\x43reateCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12-\n\x08metadata\x18\x03 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x88\x01\x01\x12\x16\n\tdimension\x18\x04 \x01(\x05H\x01\x88\x01\x01\x12\x1a\n\rget_or_create\x18\x05 \x01(\x08H\x02\x88\x01\x01\x12\x0e\n\x06tenant\x18\x06 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x07 \x01(\tB\x0b\n\t_metadataB\x0c\n\n_dimensionB\x10\n\x0e_get_or_create\"s\n\x18\x43reateCollectionResponse\x12&\n\ncollection\x18\x01 \x01(\x0b\x32\x12.chroma.Collection\x12\x0f\n\x07\x63reated\x18\x02 \x01(\x08\x12\x1e\n\x06status\x18\x03 \x01(\x0b\x32\x0e.chroma.Status\"G\n\x17\x44\x65leteCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06tenant\x18\x02 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x03 \x01(\t\":\n\x18\x44\x65leteCollectionResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"\x8b\x01\n\x15GetCollectionsRequest\x12\x0f\n\x02id\x18\x01 \x01(\tH\x00\x88\x01\x01\x12\x11\n\x04name\x18\x02 \x01(\tH\x01\x88\x01\x01\x12\x12\n\x05topic\x18\x03 \x01(\tH\x02\x88\x01\x01\x12\x0e\n\x06tenant\x18\x04 \x01(\t\x12\x10\n\x08\x64\x61tabase\x18\x05 \x01(\tB\x05\n\x03_idB\x07\n\x05_nameB\x08\n\x06_topic\"a\n\x16GetCollectionsResponse\x12\'\n\x0b\x63ollections\x18\x01 \x03(\x0b\x32\x12.chroma.Collection\x12\x1e\n\x06status\x18\x02 \x01(\x0b\x32\x0e.chroma.Status\"\xde\x01\n\x17UpdateCollectionRequest\x12\n\n\x02id\x18\x01 \x01(\t\x12\x12\n\x05topic\x18\x02 \x01(\tH\x01\x88\x01\x01\x12\x11\n\x04name\x18\x03 \x01(\tH\x02\x88\x01\x01\x12\x16\n\tdimension\x18\x04 \x01(\x05H\x03\x88\x01\x01\x12*\n\x08metadata\x18\x05 \x01(\x0b\x32\x16.chroma.UpdateMetadataH\x00\x12\x18\n\x0ereset_metadata\x18\x06 \x01(\x08H\x00\x42\x11\n\x0fmetadata_updateB\x08\n\x06_topicB\x07\n\x05_nameB\x0c\n\n_dimension\":\n\x18UpdateCollectionResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\"O\n\x0cNotification\x12\n\n\x02id\x18\x01 \x01(\x03\x12\x15\n\rcollection_id\x18\x02 \x01(\t\x12\x0c\n\x04type\x18\x03 \x01(\t\x12\x0e\n\x06status\x18\x04 \x01(\t\"4\n\x12ResetStateResponse\x12\x1e\n\x06status\x18\x01 \x01(\x0b\x32\x0e.chroma.Status\":\n%GetLastCompactionTimeForTenantRequest\x12\x11\n\ttenant_id\x18\x01 \x03(\t\"K\n\x18TenantLastCompactionTime\x12\x11\n\ttenant_id\x18\x01 \x01(\t\x12\x1c\n\x14last_compaction_time\x18\x02 \x01(\x03\"o\n&GetLastCompactionTimeForTenantResponse\x12\x45\n\x1btenant_last_compaction_time\x18\x01 \x03(\x0b\x32 .chroma.TenantLastCompactionTime\"n\n%SetLastCompactionTimeForTenantRequest\x12\x45\n\x1btenant_last_compaction_time\x18\x01 \x01(\x0b\x32 .chroma.TenantLastCompactionTime\"\xbc\x01\n\x1a\x46lushSegmentCompactionInfo\x12\x12\n\nsegment_id\x18\x01 \x01(\t\x12\x45\n\nfile_paths\x18\x02 \x03(\x0b\x32\x31.chroma.FlushSegmentCompactionInfo.FilePathsEntry\x1a\x43\n\x0e\x46ilePathsEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12 \n\x05value\x18\x02 \x01(\x0b\x32\x11.chroma.FilePaths:\x02\x38\x01\"\xc3\x01\n FlushCollectionCompactionRequest\x12\x11\n\ttenant_id\x18\x01 \x01(\t\x12\x15\n\rcollection_id\x18\x02 \x01(\t\x12\x14\n\x0clog_position\x18\x03 \x01(\x03\x12\x1a\n\x12\x63ollection_version\x18\x04 \x01(\x05\x12\x43\n\x17segment_compaction_info\x18\x05 \x03(\x0b\x32\".chroma.FlushSegmentCompactionInfo\"t\n!FlushCollectionCompactionResponse\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x1a\n\x12\x63ollection_version\x18\x02 \x01(\x05\x12\x1c\n\x14last_compaction_time\x18\x03 \x01(\x03\x32\xf4\n\n\x05SysDB\x12Q\n\x0e\x43reateDatabase\x12\x1d.chroma.CreateDatabaseRequest\x1a\x1e.chroma.CreateDatabaseResponse\"\x00\x12H\n\x0bGetDatabase\x12\x1a.chroma.GetDatabaseRequest\x1a\x1b.chroma.GetDatabaseResponse\"\x00\x12K\n\x0c\x43reateTenant\x12\x1b.chroma.CreateTenantRequest\x1a\x1c.chroma.CreateTenantResponse\"\x00\x12\x42\n\tGetTenant\x12\x18.chroma.GetTenantRequest\x1a\x19.chroma.GetTenantResponse\"\x00\x12N\n\rCreateSegment\x12\x1c.chroma.CreateSegmentRequest\x1a\x1d.chroma.CreateSegmentResponse\"\x00\x12N\n\rDeleteSegment\x12\x1c.chroma.DeleteSegmentRequest\x1a\x1d.chroma.DeleteSegmentResponse\"\x00\x12H\n\x0bGetSegments\x12\x1a.chroma.GetSegmentsRequest\x1a\x1b.chroma.GetSegmentsResponse\"\x00\x12N\n\rUpdateSegment\x12\x1c.chroma.UpdateSegmentRequest\x1a\x1d.chroma.UpdateSegmentResponse\"\x00\x12W\n\x10\x43reateCollection\x12\x1f.chroma.CreateCollectionRequest\x1a .chroma.CreateCollectionResponse\"\x00\x12W\n\x10\x44\x65leteCollection\x12\x1f.chroma.DeleteCollectionRequest\x1a .chroma.DeleteCollectionResponse\"\x00\x12Q\n\x0eGetCollections\x12\x1d.chroma.GetCollectionsRequest\x1a\x1e.chroma.GetCollectionsResponse\"\x00\x12W\n\x10UpdateCollection\x12\x1f.chroma.UpdateCollectionRequest\x1a .chroma.UpdateCollectionResponse\"\x00\x12\x42\n\nResetState\x12\x16.google.protobuf.Empty\x1a\x1a.chroma.ResetStateResponse\"\x00\x12\x81\x01\n\x1eGetLastCompactionTimeForTenant\x12-.chroma.GetLastCompactionTimeForTenantRequest\x1a..chroma.GetLastCompactionTimeForTenantResponse\"\x00\x12i\n\x1eSetLastCompactionTimeForTenant\x12-.chroma.SetLastCompactionTimeForTenantRequest\x1a\x16.google.protobuf.Empty\"\x00\x12r\n\x19\x46lushCollectionCompaction\x12(.chroma.FlushCollectionCompactionRequest\x1a).chroma.FlushCollectionCompactionResponse\"\x00\x42:Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpbb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -23,6 +23,8 @@ if _descriptor._USE_C_DESCRIPTORS == False: DESCRIPTOR._options = None DESCRIPTOR._serialized_options = b'Z8github.com/chroma-core/chroma/go/pkg/proto/coordinatorpb' + _FLUSHSEGMENTCOMPACTIONINFO_FILEPATHSENTRY._options = None + _FLUSHSEGMENTCOMPACTIONINFO_FILEPATHSENTRY._serialized_options = b'8\001' _globals['_CREATEDATABASEREQUEST']._serialized_start=102 _globals['_CREATEDATABASEREQUEST']._serialized_end=167 _globals['_CREATEDATABASERESPONSE']._serialized_start=169 @@ -83,6 +85,14 @@ _globals['_GETLASTCOMPACTIONTIMEFORTENANTRESPONSE']._serialized_end=2778 _globals['_SETLASTCOMPACTIONTIMEFORTENANTREQUEST']._serialized_start=2780 _globals['_SETLASTCOMPACTIONTIMEFORTENANTREQUEST']._serialized_end=2890 - _globals['_SYSDB']._serialized_start=2893 - _globals['_SYSDB']._serialized_end=4173 + _globals['_FLUSHSEGMENTCOMPACTIONINFO']._serialized_start=2893 + _globals['_FLUSHSEGMENTCOMPACTIONINFO']._serialized_end=3081 + _globals['_FLUSHSEGMENTCOMPACTIONINFO_FILEPATHSENTRY']._serialized_start=3014 + _globals['_FLUSHSEGMENTCOMPACTIONINFO_FILEPATHSENTRY']._serialized_end=3081 + _globals['_FLUSHCOLLECTIONCOMPACTIONREQUEST']._serialized_start=3084 + _globals['_FLUSHCOLLECTIONCOMPACTIONREQUEST']._serialized_end=3279 + _globals['_FLUSHCOLLECTIONCOMPACTIONRESPONSE']._serialized_start=3281 + _globals['_FLUSHCOLLECTIONCOMPACTIONRESPONSE']._serialized_end=3397 + _globals['_SYSDB']._serialized_start=3400 + _globals['_SYSDB']._serialized_end=4796 # @@protoc_insertion_point(module_scope) diff --git a/chromadb/proto/coordinator_pb2.pyi b/chromadb/proto/coordinator_pb2.pyi index 185a41b901a..6175b63917e 100644 --- a/chromadb/proto/coordinator_pb2.pyi +++ b/chromadb/proto/coordinator_pb2.pyi @@ -266,3 +266,42 @@ class SetLastCompactionTimeForTenantRequest(_message.Message): TENANT_LAST_COMPACTION_TIME_FIELD_NUMBER: _ClassVar[int] tenant_last_compaction_time: TenantLastCompactionTime def __init__(self, tenant_last_compaction_time: _Optional[_Union[TenantLastCompactionTime, _Mapping]] = ...) -> None: ... + +class FlushSegmentCompactionInfo(_message.Message): + __slots__ = ["segment_id", "file_paths"] + class FilePathsEntry(_message.Message): + __slots__ = ["key", "value"] + KEY_FIELD_NUMBER: _ClassVar[int] + VALUE_FIELD_NUMBER: _ClassVar[int] + key: str + value: _chroma_pb2.FilePaths + def __init__(self, key: _Optional[str] = ..., value: _Optional[_Union[_chroma_pb2.FilePaths, _Mapping]] = ...) -> None: ... + SEGMENT_ID_FIELD_NUMBER: _ClassVar[int] + FILE_PATHS_FIELD_NUMBER: _ClassVar[int] + segment_id: str + file_paths: _containers.MessageMap[str, _chroma_pb2.FilePaths] + def __init__(self, segment_id: _Optional[str] = ..., file_paths: _Optional[_Mapping[str, _chroma_pb2.FilePaths]] = ...) -> None: ... + +class FlushCollectionCompactionRequest(_message.Message): + __slots__ = ["tenant_id", "collection_id", "log_position", "collection_version", "segment_compaction_info"] + TENANT_ID_FIELD_NUMBER: _ClassVar[int] + COLLECTION_ID_FIELD_NUMBER: _ClassVar[int] + LOG_POSITION_FIELD_NUMBER: _ClassVar[int] + COLLECTION_VERSION_FIELD_NUMBER: _ClassVar[int] + SEGMENT_COMPACTION_INFO_FIELD_NUMBER: _ClassVar[int] + tenant_id: str + collection_id: str + log_position: int + collection_version: int + segment_compaction_info: _containers.RepeatedCompositeFieldContainer[FlushSegmentCompactionInfo] + def __init__(self, tenant_id: _Optional[str] = ..., collection_id: _Optional[str] = ..., log_position: _Optional[int] = ..., collection_version: _Optional[int] = ..., segment_compaction_info: _Optional[_Iterable[_Union[FlushSegmentCompactionInfo, _Mapping]]] = ...) -> None: ... + +class FlushCollectionCompactionResponse(_message.Message): + __slots__ = ["collection_id", "collection_version", "last_compaction_time"] + COLLECTION_ID_FIELD_NUMBER: _ClassVar[int] + COLLECTION_VERSION_FIELD_NUMBER: _ClassVar[int] + LAST_COMPACTION_TIME_FIELD_NUMBER: _ClassVar[int] + collection_id: str + collection_version: int + last_compaction_time: int + def __init__(self, collection_id: _Optional[str] = ..., collection_version: _Optional[int] = ..., last_compaction_time: _Optional[int] = ...) -> None: ... diff --git a/chromadb/proto/coordinator_pb2_grpc.py b/chromadb/proto/coordinator_pb2_grpc.py index 74bcba4c8d8..92ede663915 100644 --- a/chromadb/proto/coordinator_pb2_grpc.py +++ b/chromadb/proto/coordinator_pb2_grpc.py @@ -16,80 +16,85 @@ def __init__(self, channel): channel: A grpc.Channel. """ self.CreateDatabase = channel.unary_unary( - "/chroma.SysDB/CreateDatabase", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseResponse.FromString, - ) + '/chroma.SysDB/CreateDatabase', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseResponse.FromString, + ) self.GetDatabase = channel.unary_unary( - "/chroma.SysDB/GetDatabase", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseResponse.FromString, - ) + '/chroma.SysDB/GetDatabase', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseResponse.FromString, + ) self.CreateTenant = channel.unary_unary( - "/chroma.SysDB/CreateTenant", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantResponse.FromString, - ) + '/chroma.SysDB/CreateTenant', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantResponse.FromString, + ) self.GetTenant = channel.unary_unary( - "/chroma.SysDB/GetTenant", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantResponse.FromString, - ) + '/chroma.SysDB/GetTenant', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantResponse.FromString, + ) self.CreateSegment = channel.unary_unary( - "/chroma.SysDB/CreateSegment", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentResponse.FromString, - ) + '/chroma.SysDB/CreateSegment', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentResponse.FromString, + ) self.DeleteSegment = channel.unary_unary( - "/chroma.SysDB/DeleteSegment", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentResponse.FromString, - ) + '/chroma.SysDB/DeleteSegment', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentResponse.FromString, + ) self.GetSegments = channel.unary_unary( - "/chroma.SysDB/GetSegments", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsResponse.FromString, - ) + '/chroma.SysDB/GetSegments', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsResponse.FromString, + ) self.UpdateSegment = channel.unary_unary( - "/chroma.SysDB/UpdateSegment", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentResponse.FromString, - ) + '/chroma.SysDB/UpdateSegment', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentResponse.FromString, + ) self.CreateCollection = channel.unary_unary( - "/chroma.SysDB/CreateCollection", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionResponse.FromString, - ) + '/chroma.SysDB/CreateCollection', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionResponse.FromString, + ) self.DeleteCollection = channel.unary_unary( - "/chroma.SysDB/DeleteCollection", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionResponse.FromString, - ) + '/chroma.SysDB/DeleteCollection', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionResponse.FromString, + ) self.GetCollections = channel.unary_unary( - "/chroma.SysDB/GetCollections", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsResponse.FromString, - ) + '/chroma.SysDB/GetCollections', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsResponse.FromString, + ) self.UpdateCollection = channel.unary_unary( - "/chroma.SysDB/UpdateCollection", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionResponse.FromString, - ) + '/chroma.SysDB/UpdateCollection', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionResponse.FromString, + ) self.ResetState = channel.unary_unary( - "/chroma.SysDB/ResetState", - request_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.ResetStateResponse.FromString, - ) + '/chroma.SysDB/ResetState', + request_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.ResetStateResponse.FromString, + ) self.GetLastCompactionTimeForTenant = channel.unary_unary( - "/chroma.SysDB/GetLastCompactionTimeForTenant", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantResponse.FromString, - ) + '/chroma.SysDB/GetLastCompactionTimeForTenant', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantResponse.FromString, + ) self.SetLastCompactionTimeForTenant = channel.unary_unary( - "/chroma.SysDB/SetLastCompactionTimeForTenant", - request_serializer=chromadb_dot_proto_dot_coordinator__pb2.SetLastCompactionTimeForTenantRequest.SerializeToString, - response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString, - ) + '/chroma.SysDB/SetLastCompactionTimeForTenant', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.SetLastCompactionTimeForTenantRequest.SerializeToString, + response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString, + ) + self.FlushCollectionCompaction = channel.unary_unary( + '/chroma.SysDB/FlushCollectionCompaction', + request_serializer=chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionResponse.FromString, + ) class SysDBServicer(object): @@ -98,613 +103,460 @@ class SysDBServicer(object): def CreateDatabase(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetDatabase(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def CreateTenant(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetTenant(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def CreateSegment(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def DeleteSegment(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetSegments(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def UpdateSegment(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def CreateCollection(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def DeleteCollection(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetCollections(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def UpdateCollection(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def ResetState(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetLastCompactionTimeForTenant(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def SetLastCompactionTimeForTenant(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') + + def FlushCollectionCompaction(self, request, context): + """Missing associated documentation comment in .proto file.""" + context.set_code(grpc.StatusCode.UNIMPLEMENTED) + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def add_SysDBServicer_to_server(servicer, server): rpc_method_handlers = { - "CreateDatabase": grpc.unary_unary_rpc_method_handler( - servicer.CreateDatabase, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseResponse.SerializeToString, - ), - "GetDatabase": grpc.unary_unary_rpc_method_handler( - servicer.GetDatabase, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseResponse.SerializeToString, - ), - "CreateTenant": grpc.unary_unary_rpc_method_handler( - servicer.CreateTenant, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantResponse.SerializeToString, - ), - "GetTenant": grpc.unary_unary_rpc_method_handler( - servicer.GetTenant, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantResponse.SerializeToString, - ), - "CreateSegment": grpc.unary_unary_rpc_method_handler( - servicer.CreateSegment, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentResponse.SerializeToString, - ), - "DeleteSegment": grpc.unary_unary_rpc_method_handler( - servicer.DeleteSegment, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentResponse.SerializeToString, - ), - "GetSegments": grpc.unary_unary_rpc_method_handler( - servicer.GetSegments, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsResponse.SerializeToString, - ), - "UpdateSegment": grpc.unary_unary_rpc_method_handler( - servicer.UpdateSegment, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentResponse.SerializeToString, - ), - "CreateCollection": grpc.unary_unary_rpc_method_handler( - servicer.CreateCollection, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionResponse.SerializeToString, - ), - "DeleteCollection": grpc.unary_unary_rpc_method_handler( - servicer.DeleteCollection, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionResponse.SerializeToString, - ), - "GetCollections": grpc.unary_unary_rpc_method_handler( - servicer.GetCollections, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsResponse.SerializeToString, - ), - "UpdateCollection": grpc.unary_unary_rpc_method_handler( - servicer.UpdateCollection, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionResponse.SerializeToString, - ), - "ResetState": grpc.unary_unary_rpc_method_handler( - servicer.ResetState, - request_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.ResetStateResponse.SerializeToString, - ), - "GetLastCompactionTimeForTenant": grpc.unary_unary_rpc_method_handler( - servicer.GetLastCompactionTimeForTenant, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantRequest.FromString, - response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantResponse.SerializeToString, - ), - "SetLastCompactionTimeForTenant": grpc.unary_unary_rpc_method_handler( - servicer.SetLastCompactionTimeForTenant, - request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.SetLastCompactionTimeForTenantRequest.FromString, - response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString, - ), + 'CreateDatabase': grpc.unary_unary_rpc_method_handler( + servicer.CreateDatabase, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseResponse.SerializeToString, + ), + 'GetDatabase': grpc.unary_unary_rpc_method_handler( + servicer.GetDatabase, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseResponse.SerializeToString, + ), + 'CreateTenant': grpc.unary_unary_rpc_method_handler( + servicer.CreateTenant, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateTenantResponse.SerializeToString, + ), + 'GetTenant': grpc.unary_unary_rpc_method_handler( + servicer.GetTenant, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetTenantResponse.SerializeToString, + ), + 'CreateSegment': grpc.unary_unary_rpc_method_handler( + servicer.CreateSegment, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentResponse.SerializeToString, + ), + 'DeleteSegment': grpc.unary_unary_rpc_method_handler( + servicer.DeleteSegment, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentResponse.SerializeToString, + ), + 'GetSegments': grpc.unary_unary_rpc_method_handler( + servicer.GetSegments, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsResponse.SerializeToString, + ), + 'UpdateSegment': grpc.unary_unary_rpc_method_handler( + servicer.UpdateSegment, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentResponse.SerializeToString, + ), + 'CreateCollection': grpc.unary_unary_rpc_method_handler( + servicer.CreateCollection, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionResponse.SerializeToString, + ), + 'DeleteCollection': grpc.unary_unary_rpc_method_handler( + servicer.DeleteCollection, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionResponse.SerializeToString, + ), + 'GetCollections': grpc.unary_unary_rpc_method_handler( + servicer.GetCollections, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsResponse.SerializeToString, + ), + 'UpdateCollection': grpc.unary_unary_rpc_method_handler( + servicer.UpdateCollection, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionResponse.SerializeToString, + ), + 'ResetState': grpc.unary_unary_rpc_method_handler( + servicer.ResetState, + request_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.ResetStateResponse.SerializeToString, + ), + 'GetLastCompactionTimeForTenant': grpc.unary_unary_rpc_method_handler( + servicer.GetLastCompactionTimeForTenant, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantResponse.SerializeToString, + ), + 'SetLastCompactionTimeForTenant': grpc.unary_unary_rpc_method_handler( + servicer.SetLastCompactionTimeForTenant, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.SetLastCompactionTimeForTenantRequest.FromString, + response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString, + ), + 'FlushCollectionCompaction': grpc.unary_unary_rpc_method_handler( + servicer.FlushCollectionCompaction, + request_deserializer=chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionRequest.FromString, + response_serializer=chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionResponse.SerializeToString, + ), } generic_handler = grpc.method_handlers_generic_handler( - "chroma.SysDB", rpc_method_handlers - ) + 'chroma.SysDB', rpc_method_handlers) server.add_generic_rpc_handlers((generic_handler,)) -# This class is part of an EXPERIMENTAL API. + # This class is part of an EXPERIMENTAL API. class SysDB(object): """Missing associated documentation comment in .proto file.""" @staticmethod - def CreateDatabase( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def CreateDatabase(request, target, - "/chroma.SysDB/CreateDatabase", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/CreateDatabase', chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.CreateDatabaseResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetDatabase( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetDatabase(request, target, - "/chroma.SysDB/GetDatabase", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/GetDatabase', chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.GetDatabaseResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def CreateTenant( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def CreateTenant(request, target, - "/chroma.SysDB/CreateTenant", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/CreateTenant', chromadb_dot_proto_dot_coordinator__pb2.CreateTenantRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.CreateTenantResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetTenant( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetTenant(request, target, - "/chroma.SysDB/GetTenant", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/GetTenant', chromadb_dot_proto_dot_coordinator__pb2.GetTenantRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.GetTenantResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def CreateSegment( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def CreateSegment(request, target, - "/chroma.SysDB/CreateSegment", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/CreateSegment', chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.CreateSegmentResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def DeleteSegment( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def DeleteSegment(request, target, - "/chroma.SysDB/DeleteSegment", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/DeleteSegment', chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.DeleteSegmentResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetSegments( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetSegments(request, target, - "/chroma.SysDB/GetSegments", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/GetSegments', chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.GetSegmentsResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def UpdateSegment( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def UpdateSegment(request, target, - "/chroma.SysDB/UpdateSegment", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/UpdateSegment', chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.UpdateSegmentResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def CreateCollection( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def CreateCollection(request, target, - "/chroma.SysDB/CreateCollection", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/CreateCollection', chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.CreateCollectionResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def DeleteCollection( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def DeleteCollection(request, target, - "/chroma.SysDB/DeleteCollection", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/DeleteCollection', chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.DeleteCollectionResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetCollections( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetCollections(request, target, - "/chroma.SysDB/GetCollections", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/GetCollections', chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.GetCollectionsResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def UpdateCollection( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def UpdateCollection(request, target, - "/chroma.SysDB/UpdateCollection", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/UpdateCollection', chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.UpdateCollectionResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def ResetState( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def ResetState(request, target, - "/chroma.SysDB/ResetState", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/ResetState', google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.ResetStateResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetLastCompactionTimeForTenant( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetLastCompactionTimeForTenant(request, target, - "/chroma.SysDB/GetLastCompactionTimeForTenant", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/GetLastCompactionTimeForTenant', chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantRequest.SerializeToString, chromadb_dot_proto_dot_coordinator__pb2.GetLastCompactionTimeForTenantResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def SetLastCompactionTimeForTenant( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def SetLastCompactionTimeForTenant(request, target, - "/chroma.SysDB/SetLastCompactionTimeForTenant", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/SetLastCompactionTimeForTenant', chromadb_dot_proto_dot_coordinator__pb2.SetLastCompactionTimeForTenantRequest.SerializeToString, google_dot_protobuf_dot_empty__pb2.Empty.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) + + @staticmethod + def FlushCollectionCompaction(request, + target, + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.SysDB/FlushCollectionCompaction', + chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionRequest.SerializeToString, + chromadb_dot_proto_dot_coordinator__pb2.FlushCollectionCompactionResponse.FromString, + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) diff --git a/chromadb/proto/logservice_pb2.py b/chromadb/proto/logservice_pb2.py index 5ce9b4c5dcd..39f62f13aa7 100644 --- a/chromadb/proto/logservice_pb2.py +++ b/chromadb/proto/logservice_pb2.py @@ -16,7 +16,7 @@ DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile( - b'\n\x1f\x63hromadb/proto/logservice.proto\x12\x06\x63hroma\x1a\x1b\x63hromadb/proto/chroma.proto"X\n\x0fPushLogsRequest\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12.\n\x07records\x18\x02 \x03(\x0b\x32\x1d.chroma.SubmitEmbeddingRecord"(\n\x10PushLogsResponse\x12\x14\n\x0crecord_count\x18\x01 \x01(\x05"S\n\x0fPullLogsRequest\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x15\n\rstart_from_id\x18\x02 \x01(\x03\x12\x12\n\nbatch_size\x18\x03 \x01(\x05"J\n\tRecordLog\x12\x0e\n\x06log_id\x18\x01 \x01(\x03\x12-\n\x06record\x18\x02 \x01(\x0b\x32\x1d.chroma.SubmitEmbeddingRecord"6\n\x10PullLogsResponse\x12"\n\x07records\x18\x01 \x03(\x0b\x32\x11.chroma.RecordLog"V\n\x0e\x43ollectionInfo\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x14\n\x0c\x66irst_log_id\x18\x02 \x01(\x03\x12\x17\n\x0f\x66irst_log_id_ts\x18\x03 \x01(\x03"&\n$GetAllCollectionInfoToCompactRequest"\\\n%GetAllCollectionInfoToCompactResponse\x12\x33\n\x13\x61ll_collection_info\x18\x01 \x03(\x0b\x32\x16.chroma.CollectionInfo2\x8e\x02\n\nLogService\x12?\n\x08PushLogs\x12\x17.chroma.PushLogsRequest\x1a\x18.chroma.PushLogsResponse"\x00\x12?\n\x08PullLogs\x12\x17.chroma.PullLogsRequest\x1a\x18.chroma.PullLogsResponse"\x00\x12~\n\x1dGetAllCollectionInfoToCompact\x12,.chroma.GetAllCollectionInfoToCompactRequest\x1a-.chroma.GetAllCollectionInfoToCompactResponse"\x00\x42\x39Z7github.com/chroma-core/chroma/go/pkg/proto/logservicepbb\x06proto3' + b'\n\x1f\x63hromadb/proto/logservice.proto\x12\x06\x63hroma\x1a\x1b\x63hromadb/proto/chroma.proto"X\n\x0fPushLogsRequest\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12.\n\x07records\x18\x02 \x03(\x0b\x32\x1d.chroma.SubmitEmbeddingRecord"(\n\x10PushLogsResponse\x12\x14\n\x0crecord_count\x18\x01 \x01(\x05"j\n\x0fPullLogsRequest\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x15\n\rstart_from_id\x18\x02 \x01(\x03\x12\x12\n\nbatch_size\x18\x03 \x01(\x05\x12\x15\n\rend_timestamp\x18\x04 \x01(\x03"J\n\tRecordLog\x12\x0e\n\x06log_id\x18\x01 \x01(\x03\x12-\n\x06record\x18\x02 \x01(\x0b\x32\x1d.chroma.SubmitEmbeddingRecord"6\n\x10PullLogsResponse\x12"\n\x07records\x18\x01 \x03(\x0b\x32\x11.chroma.RecordLog"V\n\x0e\x43ollectionInfo\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x14\n\x0c\x66irst_log_id\x18\x02 \x01(\x03\x12\x17\n\x0f\x66irst_log_id_ts\x18\x03 \x01(\x03"&\n$GetAllCollectionInfoToCompactRequest"\\\n%GetAllCollectionInfoToCompactResponse\x12\x33\n\x13\x61ll_collection_info\x18\x01 \x03(\x0b\x32\x16.chroma.CollectionInfo2\x8e\x02\n\nLogService\x12?\n\x08PushLogs\x12\x17.chroma.PushLogsRequest\x1a\x18.chroma.PushLogsResponse"\x00\x12?\n\x08PullLogs\x12\x17.chroma.PullLogsRequest\x1a\x18.chroma.PullLogsResponse"\x00\x12~\n\x1dGetAllCollectionInfoToCompact\x12,.chroma.GetAllCollectionInfoToCompactRequest\x1a-.chroma.GetAllCollectionInfoToCompactResponse"\x00\x42\x39Z7github.com/chroma-core/chroma/go/pkg/proto/logservicepbb\x06proto3' ) _globals = globals() @@ -34,17 +34,17 @@ _globals["_PUSHLOGSRESPONSE"]._serialized_start = 162 _globals["_PUSHLOGSRESPONSE"]._serialized_end = 202 _globals["_PULLLOGSREQUEST"]._serialized_start = 204 - _globals["_PULLLOGSREQUEST"]._serialized_end = 287 - _globals["_RECORDLOG"]._serialized_start = 289 - _globals["_RECORDLOG"]._serialized_end = 363 - _globals["_PULLLOGSRESPONSE"]._serialized_start = 365 - _globals["_PULLLOGSRESPONSE"]._serialized_end = 419 - _globals["_COLLECTIONINFO"]._serialized_start = 421 - _globals["_COLLECTIONINFO"]._serialized_end = 507 - _globals["_GETALLCOLLECTIONINFOTOCOMPACTREQUEST"]._serialized_start = 509 - _globals["_GETALLCOLLECTIONINFOTOCOMPACTREQUEST"]._serialized_end = 547 - _globals["_GETALLCOLLECTIONINFOTOCOMPACTRESPONSE"]._serialized_start = 549 - _globals["_GETALLCOLLECTIONINFOTOCOMPACTRESPONSE"]._serialized_end = 641 - _globals["_LOGSERVICE"]._serialized_start = 644 - _globals["_LOGSERVICE"]._serialized_end = 914 + _globals["_PULLLOGSREQUEST"]._serialized_end = 310 + _globals["_RECORDLOG"]._serialized_start = 312 + _globals["_RECORDLOG"]._serialized_end = 386 + _globals["_PULLLOGSRESPONSE"]._serialized_start = 388 + _globals["_PULLLOGSRESPONSE"]._serialized_end = 442 + _globals["_COLLECTIONINFO"]._serialized_start = 444 + _globals["_COLLECTIONINFO"]._serialized_end = 530 + _globals["_GETALLCOLLECTIONINFOTOCOMPACTREQUEST"]._serialized_start = 532 + _globals["_GETALLCOLLECTIONINFOTOCOMPACTREQUEST"]._serialized_end = 570 + _globals["_GETALLCOLLECTIONINFOTOCOMPACTRESPONSE"]._serialized_start = 572 + _globals["_GETALLCOLLECTIONINFOTOCOMPACTRESPONSE"]._serialized_end = 664 + _globals["_LOGSERVICE"]._serialized_start = 667 + _globals["_LOGSERVICE"]._serialized_end = 937 # @@protoc_insertion_point(module_scope) diff --git a/chromadb/proto/logservice_pb2.pyi b/chromadb/proto/logservice_pb2.pyi index 62d8d74f3c2..f4be90b553a 100644 --- a/chromadb/proto/logservice_pb2.pyi +++ b/chromadb/proto/logservice_pb2.pyi @@ -35,18 +35,21 @@ class PushLogsResponse(_message.Message): def __init__(self, record_count: _Optional[int] = ...) -> None: ... class PullLogsRequest(_message.Message): - __slots__ = ["collection_id", "start_from_id", "batch_size"] + __slots__ = ["collection_id", "start_from_id", "batch_size", "end_timestamp"] COLLECTION_ID_FIELD_NUMBER: _ClassVar[int] START_FROM_ID_FIELD_NUMBER: _ClassVar[int] BATCH_SIZE_FIELD_NUMBER: _ClassVar[int] + END_TIMESTAMP_FIELD_NUMBER: _ClassVar[int] collection_id: str start_from_id: int batch_size: int + end_timestamp: int def __init__( self, collection_id: _Optional[str] = ..., start_from_id: _Optional[int] = ..., batch_size: _Optional[int] = ..., + end_timestamp: _Optional[int] = ..., ) -> None: ... class RecordLog(_message.Message): diff --git a/chromadb/proto/logservice_pb2_grpc.py b/chromadb/proto/logservice_pb2_grpc.py index 7e4ab6a7c29..ab20441aa9a 100644 --- a/chromadb/proto/logservice_pb2_grpc.py +++ b/chromadb/proto/logservice_pb2_grpc.py @@ -15,20 +15,20 @@ def __init__(self, channel): channel: A grpc.Channel. """ self.PushLogs = channel.unary_unary( - "/chroma.LogService/PushLogs", - request_serializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsResponse.FromString, - ) + '/chroma.LogService/PushLogs', + request_serializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsResponse.FromString, + ) self.PullLogs = channel.unary_unary( - "/chroma.LogService/PullLogs", - request_serializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsResponse.FromString, - ) + '/chroma.LogService/PullLogs', + request_serializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsResponse.FromString, + ) self.GetAllCollectionInfoToCompact = channel.unary_unary( - "/chroma.LogService/GetAllCollectionInfoToCompact", - request_serializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactRequest.SerializeToString, - response_deserializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactResponse.FromString, - ) + '/chroma.LogService/GetAllCollectionInfoToCompact', + request_serializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactRequest.SerializeToString, + response_deserializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactResponse.FromString, + ) class LogServiceServicer(object): @@ -37,133 +37,96 @@ class LogServiceServicer(object): def PushLogs(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def PullLogs(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def GetAllCollectionInfoToCompact(self, request, context): """Missing associated documentation comment in .proto file.""" context.set_code(grpc.StatusCode.UNIMPLEMENTED) - context.set_details("Method not implemented!") - raise NotImplementedError("Method not implemented!") + context.set_details('Method not implemented!') + raise NotImplementedError('Method not implemented!') def add_LogServiceServicer_to_server(servicer, server): rpc_method_handlers = { - "PushLogs": grpc.unary_unary_rpc_method_handler( - servicer.PushLogs, - request_deserializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsRequest.FromString, - response_serializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsResponse.SerializeToString, - ), - "PullLogs": grpc.unary_unary_rpc_method_handler( - servicer.PullLogs, - request_deserializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsRequest.FromString, - response_serializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsResponse.SerializeToString, - ), - "GetAllCollectionInfoToCompact": grpc.unary_unary_rpc_method_handler( - servicer.GetAllCollectionInfoToCompact, - request_deserializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactRequest.FromString, - response_serializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactResponse.SerializeToString, - ), + 'PushLogs': grpc.unary_unary_rpc_method_handler( + servicer.PushLogs, + request_deserializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsRequest.FromString, + response_serializer=chromadb_dot_proto_dot_logservice__pb2.PushLogsResponse.SerializeToString, + ), + 'PullLogs': grpc.unary_unary_rpc_method_handler( + servicer.PullLogs, + request_deserializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsRequest.FromString, + response_serializer=chromadb_dot_proto_dot_logservice__pb2.PullLogsResponse.SerializeToString, + ), + 'GetAllCollectionInfoToCompact': grpc.unary_unary_rpc_method_handler( + servicer.GetAllCollectionInfoToCompact, + request_deserializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactRequest.FromString, + response_serializer=chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactResponse.SerializeToString, + ), } generic_handler = grpc.method_handlers_generic_handler( - "chroma.LogService", rpc_method_handlers - ) + 'chroma.LogService', rpc_method_handlers) server.add_generic_rpc_handlers((generic_handler,)) -# This class is part of an EXPERIMENTAL API. + # This class is part of an EXPERIMENTAL API. class LogService(object): """Missing associated documentation comment in .proto file.""" @staticmethod - def PushLogs( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def PushLogs(request, target, - "/chroma.LogService/PushLogs", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.LogService/PushLogs', chromadb_dot_proto_dot_logservice__pb2.PushLogsRequest.SerializeToString, chromadb_dot_proto_dot_logservice__pb2.PushLogsResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def PullLogs( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def PullLogs(request, target, - "/chroma.LogService/PullLogs", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.LogService/PullLogs', chromadb_dot_proto_dot_logservice__pb2.PullLogsRequest.SerializeToString, chromadb_dot_proto_dot_logservice__pb2.PullLogsResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) @staticmethod - def GetAllCollectionInfoToCompact( - request, - target, - options=(), - channel_credentials=None, - call_credentials=None, - insecure=False, - compression=None, - wait_for_ready=None, - timeout=None, - metadata=None, - ): - return grpc.experimental.unary_unary( - request, + def GetAllCollectionInfoToCompact(request, target, - "/chroma.LogService/GetAllCollectionInfoToCompact", + options=(), + channel_credentials=None, + call_credentials=None, + insecure=False, + compression=None, + wait_for_ready=None, + timeout=None, + metadata=None): + return grpc.experimental.unary_unary(request, target, '/chroma.LogService/GetAllCollectionInfoToCompact', chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactRequest.SerializeToString, chromadb_dot_proto_dot_logservice__pb2.GetAllCollectionInfoToCompactResponse.FromString, - options, - channel_credentials, - insecure, - call_credentials, - compression, - wait_for_ready, - timeout, - metadata, - ) + options, channel_credentials, + insecure, call_credentials, compression, wait_for_ready, timeout, metadata) diff --git a/chromadb/segment/impl/distributed/segment_directory.py b/chromadb/segment/impl/distributed/segment_directory.py index 70b766de675..d0738c054ee 100644 --- a/chromadb/segment/impl/distributed/segment_directory.py +++ b/chromadb/segment/impl/distributed/segment_directory.py @@ -10,6 +10,11 @@ from kubernetes import client, config, watch from kubernetes.client.rest import ApiException import threading +from chromadb.telemetry.opentelemetry import ( + OpenTelemetryGranularity, + add_attributes_to_current_span, + trace_method, +) from chromadb.utils.rendezvous_hash import assign, murmur3hasher @@ -226,6 +231,11 @@ def register_updated_segment_callback( ) -> None: raise NotImplementedError() + @trace_method( + "RendezvousHashSegmentDirectory._update_memberlist", + OpenTelemetryGranularity.ALL, + ) def _update_memberlist(self, memberlist: Memberlist) -> None: with self._curr_memberlist_mutex: + add_attributes_to_current_span({"new_memberlist": memberlist}) self._curr_memberlist = memberlist diff --git a/chromadb/segment/impl/distributed/server.py b/chromadb/segment/impl/distributed/server.py index d9a6c317f7a..32bd1f67cfd 100644 --- a/chromadb/segment/impl/distributed/server.py +++ b/chromadb/segment/impl/distributed/server.py @@ -11,13 +11,8 @@ import chromadb.proto.chroma_pb2 as proto import grpc from concurrent import futures -from chromadb.proto.convert import ( - to_proto_vector_embedding_record -) from chromadb.segment import SegmentImplementation, SegmentType -from chromadb.telemetry.opentelemetry import ( - OpenTelemetryClient -) +from chromadb.telemetry.opentelemetry import OpenTelemetryClient from chromadb.types import EmbeddingRecord from chromadb.segment.distributed import MemberlistProvider, Memberlist from chromadb.utils.rendezvous_hash import assign, murmur3hasher @@ -54,7 +49,7 @@ def __init__(self, system: System) -> None: self._opentelemetry_client = system.require(OpenTelemetryClient) # TODO: add term and epoch to segment server self._memberlist_provider = system.require(MemberlistProvider) - self._memberlist_provider.set_memberlist_name("worker-memberlist") + self._memberlist_provider.set_memberlist_name("query-service-memberlist") self._assignment_policy = system.require(CollectionAssignmentPolicy) self._create_pulsar_topics() self._consumer = system.require(Consumer) diff --git a/chromadb/utils/embedding_functions.py b/chromadb/utils/embedding_functions.py index 4dbc3f20100..4d997339d52 100644 --- a/chromadb/utils/embedding_functions.py +++ b/chromadb/utils/embedding_functions.py @@ -61,7 +61,16 @@ def __init__( model_name: str = "all-MiniLM-L6-v2", device: str = "cpu", normalize_embeddings: bool = False, + **kwargs: Any ): + """Initialize SentenceTransformerEmbeddingFunction. + + Args: + model_name (str, optional): Identifier of the SentenceTransformer model, defaults to "all-MiniLM-L6-v2" + device (str, optional): Device used for computation, defaults to "cpu" + normalize_embeddings (bool, optional): Whether to normalize returned vectors, defaults to False + **kwargs: Additional arguments to pass to the SentenceTransformer model. + """ if model_name not in self.models: try: from sentence_transformers import SentenceTransformer @@ -69,7 +78,7 @@ def __init__( raise ValueError( "The sentence_transformers python package is not installed. Please install it with `pip install sentence_transformers`" ) - self.models[model_name] = SentenceTransformer(model_name, device=device) + self.models[model_name] = SentenceTransformer(model_name, device=device, **kwargs) self._model = self.models[model_name] self._normalize_embeddings = normalize_embeddings diff --git a/go/Dockerfile b/go/Dockerfile index 3bcc97e01b5..ebc9dadb335 100644 --- a/go/Dockerfile +++ b/go/Dockerfile @@ -1,33 +1,19 @@ -FROM golang:1.20-alpine3.18 as build -WORKDIR /src/chroma-coordinator -RUN apk add --no-cache make git build-base bash +FROM golang:bookworm as builder +WORKDIR /build-dir +RUN apt-get update && apt-get install -y make git bash ADD ./go/go.mod ./go.mod ADD ./go/go.sum ./go.sum -ENV PATH=$PATH:/go/bin RUN go mod download ADD ./go/ ./ ENV GOCACHE=/root/.cache/go-build RUN --mount=type=cache,target="/root/.cache/go-build" make -FROM alpine:3.17.3 +FROM debian:bookworm-slim -RUN apk add --no-cache bash bash-completion jq findutils - -# As of 6 Dec 2023, the atlas package isn't in Alpine's main package manager, only -# testing. So we have to add the testing repository to get it. -RUN apk add \ - --no-cache \ - --repository http://dl-cdn.alpinelinux.org/alpine/edge/testing \ - --repository http://dl-cdn.alpinelinux.org/alpine/edge/main \ - atlas - -RUN mkdir /chroma-coordinator -WORKDIR /chroma-coordinator - -COPY --from=build /src/chroma-coordinator/bin/coordinator /chroma-coordinator/bin/coordinator -COPY --from=build /src/chroma-coordinator/bin/logservice /chroma-coordinator/bin/logservice -ENV PATH=$PATH:/chroma-coordinator/bin +COPY --from=builder /build-dir/bin/coordinator . +COPY --from=builder /build-dir/bin/logservice . +ENV PATH=$PATH:./ CMD /bin/bash diff --git a/go/Dockerfile.migration b/go/Dockerfile.migration index b93e9cb01c4..eab472c7324 100644 --- a/go/Dockerfile.migration +++ b/go/Dockerfile.migration @@ -1,4 +1,9 @@ -FROM arigaio/atlas:latest -workdir /app +FROM debian:bookworm-slim + +RUN apt update +RUN apt upgrade -y +RUN apt install -y curl jq +RUN curl -sSf https://atlasgo.sh | sh -s -- --community + COPY ./go/migrations migrations COPY ./go/atlas.hcl atlas.hcl diff --git a/go/Makefile b/go/Makefile index 286fd5388a0..8bd6606d50b 100644 --- a/go/Makefile +++ b/go/Makefile @@ -4,7 +4,8 @@ build: go build -v -o bin/logservice ./cmd/logservice/ test: build - go test -p 1 -cover ./... + go test -race -cover ./... + lint: #brew install golangci-lint diff --git a/go/README.md b/go/README.md index f5f2d25d7f9..0cfe009a428 100644 --- a/go/README.md +++ b/go/README.md @@ -10,6 +10,10 @@ - postgres=# `create role chroma with login password 'chroma';` - postgres=# `alter role chroma with superuser;` - postgres=# `create database chroma;` +- Set postgres ENV Vars + Several tests (such as record_log_service_test.go) require the following environment variables to be set: + - `export POSTGRES_HOST=localhost` + - `export POSTGRES_PORT=5432` - Atlas schema migration - [~/chroma/go]: `atlas migrate diff --env dev` - [~/chroma/go]: `atlas --env dev migrate apply --url "postgres://chroma:chroma@localhost:5432/chroma?sslmode=disable"` diff --git a/go/cmd/coordinator/cmd.go b/go/cmd/coordinator/cmd.go index d549adb7eae..896618a73cc 100644 --- a/go/cmd/coordinator/cmd.go +++ b/go/cmd/coordinator/cmd.go @@ -1,11 +1,12 @@ package main import ( - "github.com/chroma-core/chroma/go/pkg/coordinator/grpc" - "github.com/chroma-core/chroma/go/pkg/grpcutils" "io" "time" + "github.com/chroma-core/chroma/go/pkg/coordinator/grpc" + "github.com/chroma-core/chroma/go/pkg/grpcutils" + "github.com/chroma-core/chroma/go/cmd/flag" "github.com/chroma-core/chroma/go/pkg/utils" "github.com/spf13/cobra" @@ -53,7 +54,8 @@ func init() { // Memberlist Cmd.Flags().StringVar(&conf.KubernetesNamespace, "kubernetes-namespace", "chroma", "Kubernetes namespace") - Cmd.Flags().StringVar(&conf.WorkerMemberlistName, "worker-memberlist-name", "worker-memberlist", "Worker memberlist name") + Cmd.Flags().StringVar(&conf.WorkerMemberlistName, "worker-memberlist-name", "query-service-memberlist", "Worker memberlist name") + Cmd.Flags().StringVar(&conf.WorkerPodLabel, "worker-pod-label", "query-service", "Worker pod label") Cmd.Flags().StringVar(&conf.AssignmentPolicy, "assignment-policy", "rendezvous", "Assignment policy") Cmd.Flags().DurationVar(&conf.WatchInterval, "watch-interval", 60*time.Second, "Watch interval") } diff --git a/go/go.sum b/go/go.sum index 7dddbec0ed6..2e0c9378567 100644 --- a/go/go.sum +++ b/go/go.sum @@ -1,9 +1,5 @@ -ariga.io/atlas-go-sdk v0.1.1-0.20231001054405-7edfcfc14f1c h1:jvi4KB/7DmYYT+Wy2TFImccaBU0+dw7V8Un67NDGuio= -ariga.io/atlas-go-sdk v0.1.1-0.20231001054405-7edfcfc14f1c/go.mod h1:MLvZ9QwZx1KhI6+8XguxHPUPm0/PTTUr46S5GQAe9WI= ariga.io/atlas-go-sdk v0.2.3 h1:DpKruiJ9ElJcNhYxnQM9ddzupHXEYFH0Jx6ZcZ7lKYQ= ariga.io/atlas-go-sdk v0.2.3/go.mod h1:owkEEXw6jqne5KPVDfKsYB7cwMiMk3jtOiAAeKxS/yU= -ariga.io/atlas-provider-gorm v0.1.1 h1:Y0VsZCQkXJRYIJxenn2BM6sW2u9SkTca5mLvJumqrgE= -ariga.io/atlas-provider-gorm v0.1.1/go.mod h1:jb8uYcN+ul8Nf7OVzi5Vd2y+SQXrI4dHYBEUCiCi/6Q= ariga.io/atlas-provider-gorm v0.3.1 h1:+RrnoBwlqMj+B1x/Cf1BfwtZzq6v5vKzHdl2A6nZuBU= ariga.io/atlas-provider-gorm v0.3.1/go.mod h1:NOXGkyHfWFm8vQO7T+je5Zj5DdLZhkzReXGfxnnK4VM= cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= @@ -16,14 +12,20 @@ github.com/AthenZ/athenz v1.10.39/go.mod h1:3Tg8HLsiQZp81BJY58JBeU2BR6B/H4/0MQGf github.com/Azure/azure-sdk-for-go/sdk/azcore v1.4.0/go.mod h1:ON4tFdPTwRcgWEaVDrN3584Ef+b7GgSJaXxe5fW9t4M= github.com/Azure/azure-sdk-for-go/sdk/azcore v1.6.0/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q= github.com/Azure/azure-sdk-for-go/sdk/azcore v1.6.1/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.1 h1:/iHxaJhsFr0+xVFfbMr5vxz848jyiWuIEDhYq3y5odY= github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.1/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.0 h1:vcYCAze6p19qBW7MhZybIsqD8sMV8js0NyQM8JDnVtg= github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.0/go.mod h1:OQeznEEkTZ9OrhHJoDD8ZDq51FHgXjqtP9z6bEwBq9U= github.com/Azure/azure-sdk-for-go/sdk/internal v1.1.2/go.mod h1:eWRD7oawr1Mu1sLCawqVc0CUiF43ia3qQMxLscsKQ9w= github.com/Azure/azure-sdk-for-go/sdk/internal v1.2.0/go.mod h1:eWRD7oawr1Mu1sLCawqVc0CUiF43ia3qQMxLscsKQ9w= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0 h1:sXr+ck84g/ZlZUOZiNELInmMgOsuGwdjjVkEIde0OtY= github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0/go.mod h1:okt5dMMTOFjX/aovMlrjvvXoPMBVSPzk9185BT0+eZM= +github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.0.0 h1:yfJe15aSwEQ6Oo6J+gdfdulPNoZ3TEhmbhLIoxZcA+U= github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.0.0/go.mod h1:Q28U+75mpCaSCDowNEmhIo/rmgdkqmkmzI7N6TGR4UY= +github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v0.8.0 h1:T028gtTPiYt/RMUfs8nVsAL7FDQrfLlrm/NnRG/zcC4= github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v0.8.0/go.mod h1:cw4zVQgBby0Z5f2v0itn6se2dDP17nTjbZFXW5uPyHA= github.com/AzureAD/microsoft-authentication-library-for-go v1.0.0/go.mod h1:kgDmCTgBzIEPFElEF+FK0SdjAor06dRq2Go927dnQ6o= +github.com/AzureAD/microsoft-authentication-library-for-go v1.1.0 h1:HCc0+LpPfpCKs6LGGLAhwBARt9632unrVcI6i8s/8os= github.com/AzureAD/microsoft-authentication-library-for-go v1.1.0/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/DataDog/zstd v1.5.0 h1:+K/VEwIAaPcHiMtQvpLD4lqW7f0Gk3xdYZmI1hD+CXo= @@ -72,6 +74,7 @@ github.com/emicklei/go-restful/v3 v3.9.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry github.com/evanphx/json-patch v4.12.0+incompatible h1:4onqiflcdA9EOZ4RxV643DvftH5pOlLGNtQ5lPWQu84= github.com/evanphx/json-patch v4.12.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= +github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= @@ -96,6 +99,7 @@ github.com/go-sql-driver/mysql v1.7.1 h1:lUIinVbN1DY0xBg0eMOzmmtGoHwWBbvnWubQUrt github.com/go-sql-driver/mysql v1.7.1/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= +github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls= github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2 h1:ZpnhV/YsD2/4cESfV5+Hoeu/iUR3ruzNvZ+yQfO03a0= github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2/go.mod h1:bBOAhwG1umN6/6ZUMtDFBMQR8jRg9O75tm9K00oMsK4= github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= @@ -106,6 +110,7 @@ github.com/golang-jwt/jwt v3.2.1+incompatible h1:73Z+4BJcrTC+KczS6WvTPvRGOp1WmfE github.com/golang-jwt/jwt v3.2.1+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= github.com/golang-jwt/jwt/v4 v4.4.3/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= +github.com/golang-jwt/jwt/v5 v5.0.0 h1:1n1XNM9hk7O9mnQoNBGolZvzebBQ7p93ULHRc28XJUE= github.com/golang-jwt/jwt/v5 v5.0.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 h1:au07oEsX2xN0ktxqI+Sida1w446QrXBRJ0nee3SNZlA= github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0= @@ -142,6 +147,7 @@ github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/ github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1 h1:K6RDEckDVWvDI9JAJYCmNdQXq6neHJOYx3V6jnqNEec= +github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.4.0 h1:MtMxsa51/r9yyhkyLsVeVt0B+BGQZzpQiTQ4eHZ8bc4= github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= @@ -196,11 +202,16 @@ github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORN github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= +github.com/lib/pq v1.10.0/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= +github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw= +github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= github.com/linkedin/goavro/v2 v2.9.8 h1:jN50elxBsGBDGVDEKqUlDuU1cFwJ11K/yrJCBMe/7Wg= github.com/linkedin/goavro/v2 v2.9.8/go.mod h1:UgQUb2N/pmueQYH9bfqFioWxzYCZXSfF8Jw03O5sjqA= github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= @@ -233,15 +244,20 @@ github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRW github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= +github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= +github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= github.com/onsi/ginkgo/v2 v2.9.4 h1:xR7vG4IXt5RWx6FfIjyAtsoMAtnc3C/rFXBBd2AjZwE= +github.com/onsi/ginkgo/v2 v2.9.4/go.mod h1:gCQYp2Q+kSoIj7ykSVb9nskRSsR6PUj4AiLywzIhbKM= github.com/onsi/gomega v1.27.6 h1:ENqfyGeS5AX/rlXDd/ETokDz93u0YufY1Pgxuy/PvWE= +github.com/onsi/gomega v1.27.6/go.mod h1:PIQNjfQwkP3aQAH7lf7j87O/5FiNr+ZR8+ipb+qQlhg= github.com/pierrec/lz4 v2.0.5+incompatible h1:2xWsjqPFWcplujydGg4WmhC/6fZqK42wMM8aXeqhl0I= github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY= +github.com/pingcap/errors v0.11.0 h1:DCJQB8jrHbQ1VVlMFIrbj2ApScNNotVmkSNplu2yUt4= github.com/pingcap/errors v0.11.0/go.mod h1:Oi8TUi2kEtXXLMJk9l1cGmz20kV3TaQ0usTwv5KuLY8= -github.com/pingcap/errors v0.11.5-0.20210425183316-da1aaba5fb63 h1:+FZIDR/D97YOPik4N4lPDaUcLDF/EQPogxtlHB2ZZRM= github.com/pingcap/log v1.1.0 h1:ELiPxACz7vdo1qAvvaWJg1NrYFoY6gqAh/+Uo6aXdD8= github.com/pingcap/log v1.1.0/go.mod h1:DWQW5jICDR7UJh4HtxXSM20Churx4CQL0fwL/SoOSA4= +github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8 h1:KoWmjvw+nsYOo29YJK9vDA65RGE3NrOnUtO7a+RF9HU= github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= @@ -250,6 +266,7 @@ github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINE github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/prashantv/gostub v1.1.0 h1:BTyx3RfQjRHnUWaGF9oQos79AlQ5k8WNktv7VGvVH4g= +github.com/prashantv/gostub v1.1.0/go.mod h1:A5zLQHz7ieHGG7is6LLXLz7I8+3LZzsrV0P1IAHhP5U= github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= @@ -327,6 +344,7 @@ go.uber.org/automaxprocs v1.5.3 h1:kWazyxZUrS3Gs4qUpbwo5kEIMGe/DAvi5Z4tl2NW4j8= go.uber.org/automaxprocs v1.5.3/go.mod h1:eRbA25aqJrxAbsLO0xy5jVwPt7FQnRgjW+efnwa1WM0= go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A= go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= go.uber.org/multierr v1.7.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak= go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= @@ -344,8 +362,6 @@ golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58 golang.org/x/crypto v0.7.0/go.mod h1:pYwdfH91IfpZVANVyUOhSIPZaFoJGxTFbZhFTx+dXZU= golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= -golang.org/x/crypto v0.16.0 h1:mMMrFzRSCF0GvB7Ne27XVtVAaXLrPmgPC7/v0tkwHaY= -golang.org/x/crypto v0.16.0/go.mod h1:gCAAfMLgwOJRpTjQ2zCCt2OcSfYMTeZVSRtQlPC7Nq4= golang.org/x/crypto v0.18.0 h1:PGVlW0xEltQnzFZ55hkuX5+KLyrMYhHld1YHO4AKcdc= golang.org/x/crypto v0.18.0/go.mod h1:R0j02AL6hcrfOiy9T4ZYp/rcWeMxM3L6QYxlOuEG1mg= golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= @@ -353,8 +369,6 @@ golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= -golang.org/x/mod v0.11.0 h1:bUO06HqtnRcc/7l71XBe4WcqTZ+3AH1J59zWDDwLKgU= -golang.org/x/mod v0.11.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/mod v0.15.0 h1:SernR4v+D55NyBH2QiEQrlBAnj1ECL6AGrA5+dPaMY8= golang.org/x/mod v0.15.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= @@ -378,8 +392,6 @@ golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc= golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI= -golang.org/x/net v0.19.0 h1:zTwKpTd2XuCqf8huc7Fo2iSy+4RHPd10s4KzeTnVr1c= -golang.org/x/net v0.19.0/go.mod h1:CfAk/cbD4CthTvqiEl8NpboMuiuOYsAr/7NOjZJtv1U= golang.org/x/net v0.20.0 h1:aCL9BSgETF1k+blQaYUBx9hJ9LOGP3gAVemcZlf1Kpo= golang.org/x/net v0.20.0/go.mod h1:z8BVo6PvndSri0LbOE3hAn0apkU+1YvI6E70E9jsnvY= golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= @@ -427,8 +439,6 @@ golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U= golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= -golang.org/x/term v0.15.0 h1:y/Oo/a/q3IXu26lQgl04j/gjuBDOBlx7X6Om1j2CPW4= -golang.org/x/term v0.15.0/go.mod h1:BDl952bC7+uMoWR75FIrCDx79TPU9oHkTZ9yRbYOrX0= golang.org/x/term v0.16.0 h1:m+B6fahuftsE9qjo0VWp2FW0mB3MTJvR0BaMQrq0pmE= golang.org/x/term v0.16.0/go.mod h1:yn7UURbUtPyrVJPGPq404EukNFxcm/foM+bV/bfcDsY= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -455,8 +465,6 @@ golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roY golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= -golang.org/x/tools v0.10.0 h1:tvDr/iQoUqNdohiYm0LmmKcBk+q86lb9EprIUFhHHGg= -golang.org/x/tools v0.10.0/go.mod h1:UJwyiVBsOA2uwvK/e5OY3GTpDUJriEd+/YlqAwLPmyM= golang.org/x/tools v0.17.0 h1:FvmRgNOcs3kOa+T20R1uhfP9F6HgG2mfxDv1vrx1Htc= golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -467,6 +475,7 @@ google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7 google.golang.org/appengine v1.6.8 h1:IhEN5q69dyKagZPYMSdIjS2HqprW324FRQZJcGqPAsM= google.golang.org/appengine v1.6.8/go.mod h1:1jJ3jBArFh5pcgW8gCtRJnepW8FzD1V44FJffLiz/Ds= google.golang.org/genproto v0.0.0-20231212172506-995d672761c0 h1:YJ5pD9rF8o9Qtta0Cmy9rdBwkSjrTCT6XTiUQVOtIos= +google.golang.org/genproto v0.0.0-20231212172506-995d672761c0/go.mod h1:l/k7rMz0vFTBPy+tFSGvXEd3z+BcoG1k7EHbqm+YBsY= google.golang.org/genproto/googleapis/api v0.0.0-20240102182953-50ed04b92917 h1:rcS6EyEaoCO52hQDupoSfrxI3R6C2Tq741is7X8OvnM= google.golang.org/genproto/googleapis/api v0.0.0-20240102182953-50ed04b92917/go.mod h1:CmlNWB9lSezaYELKS5Ym1r44VrrbPUa7JTvw+6MbpJ0= google.golang.org/genproto/googleapis/rpc v0.0.0-20240102182953-50ed04b92917 h1:6G8oQ016D88m1xAKljMlBOOGWDZkes4kMhgGFlf8WcQ= @@ -497,6 +506,7 @@ gopkg.in/natefinch/lumberjack.v2 v2.2.1 h1:bBRl1b0OH9s/DuPhuXpNl+VtCaJXFZ5/uEFST gopkg.in/natefinch/lumberjack.v2 v2.2.1/go.mod h1:YD8tP3GAjkrDg1eZH7EGmyESg/lsYskCTPBJVb9jqSc= gopkg.in/square/go-jose.v2 v2.4.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= @@ -519,8 +529,6 @@ gorm.io/driver/sqlserver v1.5.2 h1:+o4RQ8w1ohPbADhFqDxeeZnSWjwOcBnxBckjTbcP4wk= gorm.io/driver/sqlserver v1.5.2/go.mod h1:gaKF0MO0cfTq9Q3/XhkowSw4g6nIwHPGAs4hzKCmvBo= gorm.io/gorm v1.25.2-0.20230530020048-26663ab9bf55/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= gorm.io/gorm v1.25.2-0.20230610234218-206613868439/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= -gorm.io/gorm v1.25.5 h1:zR9lOiiYf09VNh5Q1gphfyia1JpiClIWG9hQaxB/mls= -gorm.io/gorm v1.25.5/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= gorm.io/gorm v1.25.7 h1:VsD6acwRjz2zFxGO50gPO6AkNs7KKnvfzUjHQhZDz/A= gorm.io/gorm v1.25.7/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= k8s.io/api v0.28.3 h1:Gj1HtbSdB4P08C8rs9AR94MfSGpRhJgsS+GF9V26xMM= diff --git a/go/migrations/20240309223050.sql b/go/migrations/20240313233558.sql similarity index 96% rename from go/migrations/20240309223050.sql rename to go/migrations/20240313233558.sql index 91cca57c953..d86bcf072c2 100644 --- a/go/migrations/20240309223050.sql +++ b/go/migrations/20240313233558.sql @@ -1,3 +1,5 @@ +CREATE SCHEMA "public"; + -- Create "collection_metadata" table CREATE TABLE "public"."collection_metadata" ( "collection_id" text NOT NULL, @@ -22,6 +24,7 @@ CREATE TABLE "public"."collections" ( "created_at" timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, "updated_at" timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, "log_position" bigint NULL DEFAULT 0, + "version" integer NULL DEFAULT 0, PRIMARY KEY ("id") ); -- Create index "uni_collections_name" to table: "collections" @@ -78,6 +81,7 @@ CREATE TABLE "public"."segments" ( "is_deleted" boolean NULL DEFAULT false, "created_at" timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, "updated_at" timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, + "file_paths" text NULL DEFAULT '{}', PRIMARY KEY ("collection_id", "id") ); -- Create "tenants" table diff --git a/go/migrations/20240321194713.sql b/go/migrations/20240321194713.sql new file mode 100644 index 00000000000..c0a778c530d --- /dev/null +++ b/go/migrations/20240321194713.sql @@ -0,0 +1,14 @@ +INSERT INTO + "public"."tenants" (id, last_compaction_time) +VALUES + ('default_tenant', 0); + +-- The default tenant id is 'default_tenant' others are UUIDs +INSERT INTO + databases (id, name, tenant_id) +VALUES + ( + '00000000-0000-0000-0000-000000000000', + 'default_database', + 'default_tenant' + ); \ No newline at end of file diff --git a/go/migrations/atlas.sum b/go/migrations/atlas.sum index 828fcfc446d..6e102db3e0b 100644 --- a/go/migrations/atlas.sum +++ b/go/migrations/atlas.sum @@ -1,2 +1,3 @@ -h1:w35hwPquwsvenxzG956rH1l7vvSoB2S6XNTGOz2C78w= -20240309223050.sql h1:N3DifBqpCQpbRHqCtOc9sr+Qaq7mZek5Zz59KoFAy8g= +h1:/GQrxINPAbkke6SbB5PIM8idH6RIYDFUvT8K9eBkbHU= +20240313233558.sql h1:shyeY6BuLGJ1Ia/G/hH+NZS6HZqHxhBJ2Pfdoeerz7I= +20240321194713.sql h1:K5CAwiFb9kx+O8E/3Dq2C7jzMa7P+ZvqGL5HtLKe2YU= diff --git a/go/pkg/common/errors.go b/go/pkg/common/errors.go index a5a3119bd1f..209ea7a21af 100644 --- a/go/pkg/common/errors.go +++ b/go/pkg/common/errors.go @@ -20,6 +20,9 @@ var ( ErrCollectionTopicEmpty = errors.New("collection topic is empty") ErrCollectionUniqueConstraintViolation = errors.New("collection unique constraint violation") ErrCollectionDeleteNonExistingCollection = errors.New("delete non existing collection") + ErrCollectionLogPositionStale = errors.New("collection log position Stale") + ErrCollectionVersionStale = errors.New("collection version stale") + ErrCollectionVersionInvalid = errors.New("collection version invalid") // Collection metadata errors ErrUnknownCollectionMetadataType = errors.New("collection metadata value type not supported") @@ -35,7 +38,4 @@ var ( // Segment metadata errors ErrUnknownSegmentMetadataType = errors.New("segment metadata value type not supported") - - // Record Log errors - ErrPushLogs = errors.New("error pushing logs") ) diff --git a/go/pkg/coordinator/apis.go b/go/pkg/coordinator/apis.go index c1e5e9f2231..13f75943c78 100644 --- a/go/pkg/coordinator/apis.go +++ b/go/pkg/coordinator/apis.go @@ -30,6 +30,7 @@ type ICoordinator interface { GetTenant(ctx context.Context, getTenant *model.GetTenant) (*model.Tenant, error) SetTenantLastCompactionTime(ctx context.Context, tenantID string, lastCompactionTime int64) error GetTenantsLastCompactionTime(ctx context.Context, tenantIDs []string) ([]*dbmodel.Tenant, error) + FlushCollectionCompaction(ctx context.Context, flushCollectionCompaction *model.FlushCollectionCompaction) (*model.FlushCollectionInfo, error) } func (s *Coordinator) ResetState(ctx context.Context) error { @@ -69,12 +70,12 @@ func (s *Coordinator) GetTenant(ctx context.Context, getTenant *model.GetTenant) } func (s *Coordinator) CreateCollection(ctx context.Context, createCollection *model.CreateCollection) (*model.Collection, error) { + log.Info("create collection", zap.Any("createCollection", createCollection)) collectionTopic, err := s.assignCollection(createCollection.ID) if err != nil { return nil, err } createCollection.Topic = collectionTopic - log.Info("apis create collection", zap.Any("collection", createCollection)) collection, err := s.catalog.CreateCollection(ctx, createCollection, createCollection.Ts) if err != nil { return nil, err @@ -167,3 +168,7 @@ func (s *Coordinator) SetTenantLastCompactionTime(ctx context.Context, tenantID func (s *Coordinator) GetTenantsLastCompactionTime(ctx context.Context, tenantIDs []string) ([]*dbmodel.Tenant, error) { return s.catalog.GetTenantsLastCompactionTime(ctx, tenantIDs) } + +func (s *Coordinator) FlushCollectionCompaction(ctx context.Context, flushCollectionCompaction *model.FlushCollectionCompaction) (*model.FlushCollectionInfo, error) { + return s.catalog.FlushCollectionCompaction(ctx, flushCollectionCompaction) +} diff --git a/go/pkg/coordinator/apis_test.go b/go/pkg/coordinator/apis_test.go index 24aee2c4a5a..47a8b9b3218 100644 --- a/go/pkg/coordinator/apis_test.go +++ b/go/pkg/coordinator/apis_test.go @@ -2,10 +2,12 @@ package coordinator import ( "context" + "github.com/chroma-core/chroma/go/pkg/metastore/db/dao" "github.com/pingcap/log" "github.com/stretchr/testify/suite" "gorm.io/gorm" "sort" + "strconv" "testing" "github.com/chroma-core/chroma/go/pkg/common" @@ -13,17 +15,20 @@ import ( "github.com/chroma-core/chroma/go/pkg/model" "github.com/chroma-core/chroma/go/pkg/types" "github.com/google/uuid" - "github.com/stretchr/testify/assert" "pgregory.net/rapid" ) type APIsTestSuite struct { suite.Suite - db *gorm.DB - t *testing.T - collectionId1 types.UniqueID - collectionId2 types.UniqueID - records [][]byte + db *gorm.DB + collectionId1 types.UniqueID + collectionId2 types.UniqueID + records [][]byte + tenantName string + databaseName string + databaseId string + sampleCollections []*model.Collection + coordinator *Coordinator } func (suite *APIsTestSuite) SetupSuite() { @@ -33,12 +38,43 @@ func (suite *APIsTestSuite) SetupSuite() { func (suite *APIsTestSuite) SetupTest() { log.Info("setup test") - dbcore.ResetTestTables(suite.db) + suite.tenantName = "tenant_" + suite.T().Name() + suite.databaseName = "database_" + suite.T().Name() + DbId, err := dao.CreateTestTenantAndDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.databaseId = DbId + suite.sampleCollections = SampleCollections(suite.tenantName, suite.databaseName) + for index, collection := range suite.sampleCollections { + collection.ID = types.NewUniqueID() + collection.Name = "collection_" + suite.T().Name() + strconv.Itoa(index) + } + assignmentPolicy := NewMockAssignmentPolicy(suite.sampleCollections) + ctx := context.Background() + c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) + if err != nil { + suite.T().Fatalf("error creating coordinator: %v", err) + } + suite.coordinator = c + for _, collection := range suite.sampleCollections { + _, errCollectionCreation := c.CreateCollection(ctx, &model.CreateCollection{ + ID: collection.ID, + Name: collection.Name, + Topic: collection.Topic, + Metadata: collection.Metadata, + Dimension: collection.Dimension, + TenantID: collection.TenantID, + DatabaseName: collection.DatabaseName, + }) + suite.NoError(errCollectionCreation) + } } func (suite *APIsTestSuite) TearDownTest() { log.Info("teardown test") - dbcore.ResetTestTables(suite.db) + err := dao.CleanUpTestDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + err = dao.CleanUpTestTenant(suite.db, suite.tenantName) + suite.NoError(err) } // TODO: This is not complete yet. We need to add more tests for the other APIs. @@ -180,7 +216,7 @@ func TestAPIs(t *testing.T) { // rapid.Check(t, testSegment) } -func SampleCollections(t *testing.T, tenantID string, databaseName string) []*model.Collection { +func SampleCollections(tenantID string, databaseName string) []*model.Collection { dimension := int32(128) metadata1 := model.NewCollectionMetadata[model.CollectionMetadataValueType]() metadata1.Add("test_str", &model.CollectionMetadataValueStringType{Value: "str1"}) @@ -248,446 +284,411 @@ func (m *MockAssignmentPolicy) AssignCollection(collectionID types.UniqueID) (st } func (suite *APIsTestSuite) TestCreateGetDeleteCollections() { - - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) - ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) - - for _, collection := range sampleCollections { - c.CreateCollection(ctx, &model.CreateCollection{ - ID: collection.ID, - Name: collection.Name, - Topic: collection.Topic, - Metadata: collection.Metadata, - Dimension: collection.Dimension, - TenantID: collection.TenantID, - DatabaseName: collection.DatabaseName, - }) - } - - results, err := c.GetCollections(ctx, types.NilUniqueID(), nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) + results, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) sort.Slice(results, func(i, j int) bool { return results[i].Name < results[j].Name }) - - assert.Equal(suite.t, sampleCollections, results) + suite.Equal(suite.sampleCollections, results) // Duplicate create fails - _, err = c.CreateCollection(ctx, &model.CreateCollection{ - ID: sampleCollections[0].ID, - Name: sampleCollections[0].Name, - TenantID: common.DefaultTenant, - DatabaseName: common.DefaultDatabase, + _, err = suite.coordinator.CreateCollection(ctx, &model.CreateCollection{ + ID: suite.sampleCollections[0].ID, + Name: suite.sampleCollections[0].Name, + TenantID: suite.tenantName, + DatabaseName: suite.databaseName, }) - assert.Error(suite.t, err) + suite.Error(err) // Find by name - for _, collection := range sampleCollections { - result, err := c.GetCollections(ctx, types.NilUniqueID(), &collection.Name, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{collection}, result) + for _, collection := range suite.sampleCollections { + result, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), &collection.Name, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{collection}, result) } // Find by topic - for _, collection := range sampleCollections { - result, err := c.GetCollections(ctx, types.NilUniqueID(), nil, &collection.Topic, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{collection}, result) + for _, collection := range suite.sampleCollections { + result, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, &collection.Topic, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{collection}, result) } // Find by id - for _, collection := range sampleCollections { - result, err := c.GetCollections(ctx, collection.ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{collection}, result) + for _, collection := range suite.sampleCollections { + result, err := suite.coordinator.GetCollections(ctx, collection.ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{collection}, result) } // Find by id and topic (positive case) - for _, collection := range sampleCollections { - result, err := c.GetCollections(ctx, collection.ID, nil, &collection.Topic, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{collection}, result) + for _, collection := range suite.sampleCollections { + result, err := suite.coordinator.GetCollections(ctx, collection.ID, nil, &collection.Topic, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{collection}, result) } // find by id and topic (negative case) - for _, collection := range sampleCollections { + for _, collection := range suite.sampleCollections { otherTopic := "other topic" - result, err := c.GetCollections(ctx, collection.ID, nil, &otherTopic, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Empty(suite.t, result) + result, err := suite.coordinator.GetCollections(ctx, collection.ID, nil, &otherTopic, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Empty(result) } // Delete - c1 := sampleCollections[0] + c1 := suite.sampleCollections[0] deleteCollection := &model.DeleteCollection{ ID: c1.ID, - DatabaseName: common.DefaultDatabase, - TenantID: common.DefaultTenant, + DatabaseName: suite.databaseName, + TenantID: suite.tenantName, } - err = c.DeleteCollection(ctx, deleteCollection) - assert.NoError(suite.t, err) + err = suite.coordinator.DeleteCollection(ctx, deleteCollection) + suite.NoError(err) - results, err = c.GetCollections(ctx, types.NilUniqueID(), nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) + results, err = suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) - assert.NotContains(suite.t, results, c1) - assert.Len(suite.t, results, len(sampleCollections)-1) - assert.ElementsMatch(suite.t, results, sampleCollections[1:]) - byIDResult, err := c.GetCollections(ctx, c1.ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Empty(suite.t, byIDResult) + suite.NotContains(results, c1) + suite.Len(results, len(suite.sampleCollections)-1) + suite.ElementsMatch(results, suite.sampleCollections[1:]) + byIDResult, err := suite.coordinator.GetCollections(ctx, c1.ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Empty(byIDResult) // Duplicate delete throws an exception - err = c.DeleteCollection(ctx, deleteCollection) - assert.Error(suite.t, err) + err = suite.coordinator.DeleteCollection(ctx, deleteCollection) + suite.Error(err) } func (suite *APIsTestSuite) TestUpdateCollections() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) - ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) - coll := &model.Collection{ - Name: sampleCollections[0].Name, - ID: sampleCollections[0].ID, - Topic: sampleCollections[0].Topic, - Metadata: sampleCollections[0].Metadata, - Dimension: sampleCollections[0].Dimension, - TenantID: sampleCollections[0].TenantID, - DatabaseName: sampleCollections[0].DatabaseName, + Name: suite.sampleCollections[0].Name, + ID: suite.sampleCollections[0].ID, + Topic: suite.sampleCollections[0].Topic, + Metadata: suite.sampleCollections[0].Metadata, + Dimension: suite.sampleCollections[0].Dimension, + TenantID: suite.sampleCollections[0].TenantID, + DatabaseName: suite.sampleCollections[0].DatabaseName, } - c.CreateCollection(ctx, &model.CreateCollection{ - ID: coll.ID, - Name: coll.Name, - Topic: coll.Topic, - Metadata: coll.Metadata, - Dimension: coll.Dimension, - TenantID: coll.TenantID, - DatabaseName: coll.DatabaseName, - }) - // Update name coll.Name = "new_name" - result, err := c.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Name: &coll.Name}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, coll, result) - resultList, err := c.GetCollections(ctx, types.NilUniqueID(), &coll.Name, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{coll}, resultList) + result, err := suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Name: &coll.Name}) + suite.NoError(err) + suite.Equal(coll, result) + resultList, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), &coll.Name, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{coll}, resultList) // Update topic coll.Topic = "new_topic" - result, err = c.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Topic: &coll.Topic}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, coll, result) - resultList, err = c.GetCollections(ctx, types.NilUniqueID(), nil, &coll.Topic, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{coll}, resultList) + result, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Topic: &coll.Topic}) + suite.NoError(err) + suite.Equal(coll, result) + resultList, err = suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, &coll.Topic, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{coll}, resultList) // Update dimension newDimension := int32(128) coll.Dimension = &newDimension - result, err = c.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Dimension: coll.Dimension}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, coll, result) - resultList, err = c.GetCollections(ctx, coll.ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{coll}, resultList) + result, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Dimension: coll.Dimension}) + suite.NoError(err) + suite.Equal(coll, result) + resultList, err = suite.coordinator.GetCollections(ctx, coll.ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{coll}, resultList) // Reset the metadata newMetadata := model.NewCollectionMetadata[model.CollectionMetadataValueType]() newMetadata.Add("test_str2", &model.CollectionMetadataValueStringType{Value: "str2"}) coll.Metadata = newMetadata - result, err = c.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Metadata: coll.Metadata}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, coll, result) - resultList, err = c.GetCollections(ctx, coll.ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{coll}, resultList) + result, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Metadata: coll.Metadata}) + suite.NoError(err) + suite.Equal(coll, result) + resultList, err = suite.coordinator.GetCollections(ctx, coll.ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{coll}, resultList) // Delete all metadata keys coll.Metadata = nil - result, err = c.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Metadata: coll.Metadata, ResetMetadata: true}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, coll, result) - resultList, err = c.GetCollections(ctx, coll.ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Collection{coll}, resultList) + result, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ID: coll.ID, Metadata: coll.Metadata, ResetMetadata: true}) + suite.NoError(err) + suite.Equal(coll, result) + resultList, err = suite.coordinator.GetCollections(ctx, coll.ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal([]*model.Collection{coll}, resultList) } func (suite *APIsTestSuite) TestCreateUpdateWithDatabase() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) - _, err = c.CreateDatabase(ctx, &model.CreateDatabase{ - ID: types.MustParse("00000000-d7d7-413b-92e1-731098a6e492").String(), - Name: "new_database", - Tenant: common.DefaultTenant, - }) - assert.NoError(suite.t, err) - - c.CreateCollection(ctx, &model.CreateCollection{ - ID: sampleCollections[0].ID, - Name: sampleCollections[0].Name, - Topic: sampleCollections[0].Topic, - Metadata: sampleCollections[0].Metadata, - Dimension: sampleCollections[0].Dimension, - TenantID: sampleCollections[0].TenantID, - DatabaseName: "new_database", + newDatabaseName := "test_apis_CreateUpdateWithDatabase" + newDatabaseId := uuid.New().String() + _, err := suite.coordinator.CreateDatabase(ctx, &model.CreateDatabase{ + ID: newDatabaseId, + Name: newDatabaseName, + Tenant: suite.tenantName, }) - - c.CreateCollection(ctx, &model.CreateCollection{ - ID: sampleCollections[1].ID, - Name: sampleCollections[1].Name, - Topic: sampleCollections[1].Topic, - Metadata: sampleCollections[1].Metadata, - Dimension: sampleCollections[1].Dimension, - TenantID: sampleCollections[1].TenantID, - DatabaseName: sampleCollections[1].DatabaseName, + suite.NoError(err) + + suite.sampleCollections[0].ID = types.NewUniqueID() + suite.sampleCollections[0].Name = suite.sampleCollections[0].Name + "1" + _, err = suite.coordinator.CreateCollection(ctx, &model.CreateCollection{ + ID: suite.sampleCollections[0].ID, + Name: suite.sampleCollections[0].Name, + Topic: suite.sampleCollections[0].Topic, + Metadata: suite.sampleCollections[0].Metadata, + Dimension: suite.sampleCollections[0].Dimension, + TenantID: suite.sampleCollections[0].TenantID, + DatabaseName: newDatabaseName, }) - + suite.NoError(err) newName1 := "new_name_1" - c.UpdateCollection(ctx, &model.UpdateCollection{ - ID: sampleCollections[1].ID, + _, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ + ID: suite.sampleCollections[1].ID, Name: &newName1, }) - - result, err := c.GetCollections(ctx, sampleCollections[1].ID, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 1, len(result)) - assert.Equal(suite.t, "new_name_1", result[0].Name) + suite.NoError(err) + result, err := suite.coordinator.GetCollections(ctx, suite.sampleCollections[1].ID, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Len(result, 1) + suite.Equal(newName1, result[0].Name) newName0 := "new_name_0" - c.UpdateCollection(ctx, &model.UpdateCollection{ - ID: sampleCollections[0].ID, + _, err = suite.coordinator.UpdateCollection(ctx, &model.UpdateCollection{ + ID: suite.sampleCollections[0].ID, Name: &newName0, }) - result, err = c.GetCollections(ctx, sampleCollections[0].ID, nil, nil, common.DefaultTenant, "new_database") - assert.NoError(suite.t, err) - assert.Equal(suite.t, 1, len(result)) - assert.Equal(suite.t, "new_name_0", result[0].Name) + suite.NoError(err) + //suite.Equal(newName0, collection.Name) + result, err = suite.coordinator.GetCollections(ctx, suite.sampleCollections[0].ID, nil, nil, suite.tenantName, newDatabaseName) + suite.NoError(err) + suite.Len(result, 1) + suite.Equal(newName0, result[0].Name) + + // clean up + err = dao.CleanUpTestDatabase(suite.db, suite.tenantName, newDatabaseName) + suite.NoError(err) } func (suite *APIsTestSuite) TestGetMultipleWithDatabase() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, "new_database") + newDatabaseName := "test_apis_GetMultipleWithDatabase" ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) - _, err = c.CreateDatabase(ctx, &model.CreateDatabase{ - ID: types.MustParse("00000000-d7d7-413b-92e1-731098a6e492").String(), - Name: "new_database", - Tenant: common.DefaultTenant, - }) - assert.NoError(suite.t, err) - for _, collection := range sampleCollections { - c.CreateCollection(ctx, &model.CreateCollection{ + newDatabaseId := uuid.New().String() + _, err := suite.coordinator.CreateDatabase(ctx, &model.CreateDatabase{ + ID: newDatabaseId, + Name: newDatabaseName, + Tenant: suite.tenantName, + }) + suite.NoError(err) + + for index, collection := range suite.sampleCollections { + collection.ID = types.NewUniqueID() + collection.Name = collection.Name + "1" + collection.TenantID = suite.tenantName + collection.DatabaseName = newDatabaseName + _, err := suite.coordinator.CreateCollection(ctx, &model.CreateCollection{ ID: collection.ID, Name: collection.Name, Topic: collection.Topic, Metadata: collection.Metadata, Dimension: collection.Dimension, - TenantID: common.DefaultTenant, - DatabaseName: "new_database", + TenantID: collection.TenantID, + DatabaseName: collection.DatabaseName, }) + suite.NoError(err) + suite.sampleCollections[index] = collection } - result, err := c.GetCollections(ctx, types.NilUniqueID(), nil, nil, common.DefaultTenant, "new_database") - assert.NoError(suite.t, err) - assert.Equal(suite.t, len(sampleCollections), len(result)) + result, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, suite.tenantName, newDatabaseName) + suite.NoError(err) + suite.Equal(len(suite.sampleCollections), len(result)) sort.Slice(result, func(i, j int) bool { return result[i].Name < result[j].Name }) - assert.Equal(suite.t, sampleCollections, result) + suite.Equal(suite.sampleCollections, result) - result, err = c.GetCollections(ctx, types.NilUniqueID(), nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 0, len(result)) + result, err = suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Equal(len(suite.sampleCollections), len(result)) + + // clean up + err = dao.CleanUpTestDatabase(suite.db, suite.tenantName, newDatabaseName) + suite.NoError(err) } func (suite *APIsTestSuite) TestCreateDatabaseWithTenants() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) // Create a new tenant - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: "tenant1", + newTenantName := "tenant1" + _, err := suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: newTenantName, }) - assert.NoError(suite.t, err) + suite.NoError(err) // Create tenant that already exits and expect an error - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: "tenant1", + _, err = suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: newTenantName, }) - assert.Error(suite.t, err) + suite.Error(err) // Create tenant that already exits and expect an error - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: common.DefaultTenant, + _, err = suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: suite.tenantName, }) - assert.Error(suite.t, err) + suite.Error(err) // Create a new database within this tenant and also in the default tenant - _, err = c.CreateDatabase(ctx, &model.CreateDatabase{ + newDatabaseName := "test_apis_CreateDatabaseWithTenants" + _, err = suite.coordinator.CreateDatabase(ctx, &model.CreateDatabase{ ID: types.MustParse("33333333-d7d7-413b-92e1-731098a6e492").String(), - Name: "new_database", - Tenant: "tenant1", + Name: newDatabaseName, + Tenant: newTenantName, }) - assert.NoError(suite.t, err) + suite.NoError(err) - _, err = c.CreateDatabase(ctx, &model.CreateDatabase{ + _, err = suite.coordinator.CreateDatabase(ctx, &model.CreateDatabase{ ID: types.MustParse("44444444-d7d7-413b-92e1-731098a6e492").String(), - Name: "new_database", - Tenant: common.DefaultTenant, + Name: newDatabaseName, + Tenant: suite.tenantName, }) - assert.NoError(suite.t, err) + suite.NoError(err) // Create a new collection in the new tenant - _, err = c.CreateCollection(ctx, &model.CreateCollection{ - ID: sampleCollections[0].ID, - Name: sampleCollections[0].Name, - Topic: sampleCollections[0].Topic, - Metadata: sampleCollections[0].Metadata, - Dimension: sampleCollections[0].Dimension, - TenantID: "tenant1", - DatabaseName: "new_database", + suite.sampleCollections[0].ID = types.NewUniqueID() + suite.sampleCollections[0].Name = suite.sampleCollections[0].Name + "1" + _, err = suite.coordinator.CreateCollection(ctx, &model.CreateCollection{ + ID: suite.sampleCollections[0].ID, + Name: suite.sampleCollections[0].Name, + Topic: suite.sampleCollections[0].Topic, + Metadata: suite.sampleCollections[0].Metadata, + Dimension: suite.sampleCollections[0].Dimension, + TenantID: newTenantName, + DatabaseName: newDatabaseName, }) - assert.NoError(suite.t, err) + suite.NoError(err) // Create a new collection in the default tenant - c.CreateCollection(ctx, &model.CreateCollection{ - ID: sampleCollections[1].ID, - Name: sampleCollections[1].Name, - Topic: sampleCollections[1].Topic, - Metadata: sampleCollections[1].Metadata, - Dimension: sampleCollections[1].Dimension, - TenantID: common.DefaultTenant, - DatabaseName: "new_database", + suite.sampleCollections[1].ID = types.NewUniqueID() + suite.sampleCollections[1].Name = suite.sampleCollections[1].Name + "2" + _, err = suite.coordinator.CreateCollection(ctx, &model.CreateCollection{ + ID: suite.sampleCollections[1].ID, + Name: suite.sampleCollections[1].Name, + Topic: suite.sampleCollections[1].Topic, + Metadata: suite.sampleCollections[1].Metadata, + Dimension: suite.sampleCollections[1].Dimension, + TenantID: suite.tenantName, + DatabaseName: newDatabaseName, }) + suite.NoError(err) // Check that both tenants have the correct collections - expected := []*model.Collection{sampleCollections[0]} - expected[0].TenantID = "tenant1" - expected[0].DatabaseName = "new_database" - result, err := c.GetCollections(ctx, types.NilUniqueID(), nil, nil, "tenant1", "new_database") - assert.NoError(suite.t, err) - assert.Equal(suite.t, 1, len(result)) - assert.Equal(suite.t, expected[0], result[0]) - - expected = []*model.Collection{sampleCollections[1]} - expected[0].TenantID = common.DefaultTenant - expected[0].DatabaseName = "new_database" - result, err = c.GetCollections(ctx, types.NilUniqueID(), nil, nil, common.DefaultTenant, "new_database") - assert.NoError(suite.t, err) - assert.Equal(suite.t, 1, len(result)) - assert.Equal(suite.t, expected[0], result[0]) + expected := []*model.Collection{suite.sampleCollections[0]} + expected[0].TenantID = newTenantName + expected[0].DatabaseName = newDatabaseName + result, err := suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, newTenantName, newDatabaseName) + suite.NoError(err) + suite.Len(result, 1) + suite.Equal(expected[0], result[0]) + + expected = []*model.Collection{suite.sampleCollections[1]} + expected[0].TenantID = suite.tenantName + expected[0].DatabaseName = newDatabaseName + result, err = suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, suite.tenantName, newDatabaseName) + suite.NoError(err) + suite.Len(result, 1) + suite.Equal(expected[0], result[0]) // A new tenant DOES NOT have a default database. This does not error, instead 0 // results are returned - result, err = c.GetCollections(ctx, types.NilUniqueID(), nil, nil, "tenant1", common.DefaultDatabase) - assert.NoError(suite.t, err) - assert.Nil(suite.t, result) + result, err = suite.coordinator.GetCollections(ctx, types.NilUniqueID(), nil, nil, newTenantName, suite.databaseName) + suite.NoError(err) + suite.Nil(result) + + // clean up + err = dao.CleanUpTestTenant(suite.db, newTenantName) + suite.NoError(err) + err = dao.CleanUpTestDatabase(suite.db, suite.tenantName, newDatabaseName) + suite.NoError(err) } func (suite *APIsTestSuite) TestCreateGetDeleteTenants() { ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(nil) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) // Create a new tenant - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: "tenant1", + newTenantName := "tenant1" + _, err := suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: newTenantName, }) - assert.NoError(suite.t, err) + suite.NoError(err) // Create tenant that already exits and expect an error - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: "tenant1", + _, err = suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: newTenantName, }) - assert.Error(suite.t, err) + suite.Error(err) // Create tenant that already exits and expect an error - _, err = c.CreateTenant(ctx, &model.CreateTenant{ - Name: common.DefaultTenant, + _, err = suite.coordinator.CreateTenant(ctx, &model.CreateTenant{ + Name: suite.tenantName, }) - assert.Error(suite.t, err) + suite.Error(err) // Get the tenant and check that it exists - result, err := c.GetTenant(ctx, &model.GetTenant{Name: "tenant1"}) - assert.NoError(suite.t, err) - assert.Equal(suite.t, "tenant1", result.Name) + result, err := suite.coordinator.GetTenant(ctx, &model.GetTenant{Name: newTenantName}) + suite.NoError(err) + suite.Equal(newTenantName, result.Name) // Get a tenant that does not exist and expect an error - _, err = c.GetTenant(ctx, &model.GetTenant{Name: "tenant2"}) - assert.Error(suite.t, err) + _, err = suite.coordinator.GetTenant(ctx, &model.GetTenant{Name: "tenant2"}) + suite.Error(err) // Create a new database within this tenant - _, err = c.CreateDatabase(ctx, &model.CreateDatabase{ + newDatabaseName := "test_apis_CreateGetDeleteTenants" + _, err = suite.coordinator.CreateDatabase(ctx, &model.CreateDatabase{ ID: types.MustParse("33333333-d7d7-413b-92e1-731098a6e492").String(), - Name: "new_database", - Tenant: "tenant1", + Name: newDatabaseName, + Tenant: newTenantName, }) - assert.NoError(suite.t, err) + suite.NoError(err) // Get the database and check that it exists - databaseResult, err := c.GetDatabase(ctx, &model.GetDatabase{ - Name: "new_database", - Tenant: "tenant1", + databaseResult, err := suite.coordinator.GetDatabase(ctx, &model.GetDatabase{ + Name: newDatabaseName, + Tenant: newTenantName, }) - assert.NoError(suite.t, err) - assert.Equal(suite.t, "new_database", databaseResult.Name) - assert.Equal(suite.t, "tenant1", databaseResult.Tenant) + suite.NoError(err) + suite.Equal(newDatabaseName, databaseResult.Name) + suite.Equal(newTenantName, databaseResult.Tenant) // Get a database that does not exist in a tenant that does exist and expect an error - _, err = c.GetDatabase(ctx, &model.GetDatabase{ + _, err = suite.coordinator.GetDatabase(ctx, &model.GetDatabase{ Name: "new_database1", - Tenant: "tenant1", + Tenant: newTenantName, }) - assert.Error(suite.t, err) + suite.Error(err) // Get a database that does not exist in a tenant that does not exist and expect an // error - _, err = c.GetDatabase(ctx, &model.GetDatabase{ + _, err = suite.coordinator.GetDatabase(ctx, &model.GetDatabase{ Name: "new_database1", Tenant: "tenant2", }) - assert.Error(suite.t, err) + suite.Error(err) + + // clean up + err = dao.CleanUpTestTenant(suite.db, newTenantName) + suite.NoError(err) + err = dao.CleanUpTestDatabase(suite.db, suite.tenantName, newDatabaseName) + suite.NoError(err) } -func SampleSegments(t *testing.T, sampleCollections []*model.Collection) []*model.Segment { +func SampleSegments(sampleCollections []*model.Collection) []*model.Segment { metadata1 := model.NewSegmentMetadata[model.SegmentMetadataValueType]() metadata1.Set("test_str", &model.SegmentMetadataValueStringType{Value: "str1"}) metadata1.Set("test_int", &model.SegmentMetadataValueInt64Type{Value: 1}) @@ -713,6 +714,7 @@ func SampleSegments(t *testing.T, sampleCollections []*model.Collection) []*mode Scope: "VECTOR", CollectionID: sampleCollections[0].ID, Metadata: metadata1, + FilePaths: map[string][]string{}, }, { ID: types.MustParse("11111111-d7d7-413b-92e1-731098a6e492"), @@ -721,6 +723,7 @@ func SampleSegments(t *testing.T, sampleCollections []*model.Collection) []*mode Scope: "VECTOR", CollectionID: sampleCollections[1].ID, Metadata: metadata2, + FilePaths: map[string][]string{}, }, { ID: types.MustParse("22222222-d7d7-413b-92e1-731098a6e492"), @@ -729,36 +732,19 @@ func SampleSegments(t *testing.T, sampleCollections []*model.Collection) []*mode Scope: "METADATA", CollectionID: types.NilUniqueID(), Metadata: metadata3, // This segment is not assigned to any collection + FilePaths: map[string][]string{}, }, } return sampleSegments } func (suite *APIsTestSuite) TestCreateGetDeleteSegments() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) - } - c.ResetState(ctx) - - for _, collection := range sampleCollections { - c.CreateCollection(ctx, &model.CreateCollection{ - ID: collection.ID, - Name: collection.Name, - Topic: collection.Topic, - Metadata: collection.Metadata, - Dimension: collection.Dimension, - TenantID: collection.TenantID, - DatabaseName: collection.DatabaseName, - }) - } + c := suite.coordinator - sampleSegments := SampleSegments(suite.t, sampleCollections) + sampleSegments := SampleSegments(suite.sampleCollections) for _, segment := range sampleSegments { - c.CreateSegment(ctx, &model.CreateSegment{ + errSegmentCreation := c.CreateSegment(ctx, &model.CreateSegment{ ID: segment.ID, Type: segment.Type, Topic: segment.Topic, @@ -766,17 +752,23 @@ func (suite *APIsTestSuite) TestCreateGetDeleteSegments() { CollectionID: segment.CollectionID, Metadata: segment.Metadata, }) + suite.NoError(errSegmentCreation) } - results, err := c.GetSegments(ctx, types.NilUniqueID(), nil, nil, nil, types.NilUniqueID()) + var results []*model.Segment + for _, segment := range sampleSegments { + result, err := c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) + results = append(results, result...) + } sort.Slice(results, func(i, j int) bool { return results[i].ID.String() < results[j].ID.String() }) - assert.NoError(suite.t, err) - assert.Equal(suite.t, sampleSegments, results) + suite.Equal(sampleSegments, results) // Duplicate create fails - err = c.CreateSegment(ctx, &model.CreateSegment{ + err := c.CreateSegment(ctx, &model.CreateSegment{ ID: sampleSegments[0].ID, Type: sampleSegments[0].Type, Topic: sampleSegments[0].Topic, @@ -784,67 +776,63 @@ func (suite *APIsTestSuite) TestCreateGetDeleteSegments() { CollectionID: sampleSegments[0].CollectionID, Metadata: sampleSegments[0].Metadata, }) - assert.Error(suite.t, err) + suite.Error(err) // Find by id for _, segment := range sampleSegments { result, err := c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) } // Find by type testTypeA := "test_type_a" result, err := c.GetSegments(ctx, types.NilUniqueID(), &testTypeA, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, sampleSegments[:1], result) + suite.NoError(err) + suite.Equal(sampleSegments[:1], result) testTypeB := "test_type_b" result, err = c.GetSegments(ctx, types.NilUniqueID(), &testTypeB, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.ElementsMatch(suite.t, result, sampleSegments[1:]) + suite.NoError(err) + suite.ElementsMatch(sampleSegments[1:], result) // Find by collection ID - result, err = c.GetSegments(ctx, types.NilUniqueID(), nil, nil, nil, sampleCollections[0].ID) - assert.NoError(suite.t, err) - assert.Equal(suite.t, sampleSegments[:1], result) + result, err = c.GetSegments(ctx, types.NilUniqueID(), nil, nil, nil, suite.sampleCollections[0].ID) + suite.NoError(err) + suite.Equal(sampleSegments[:1], result) // Find by type and collection ID (positive case) - result, err = c.GetSegments(ctx, types.NilUniqueID(), &testTypeA, nil, nil, sampleCollections[0].ID) - assert.NoError(suite.t, err) - assert.Equal(suite.t, sampleSegments[:1], result) + result, err = c.GetSegments(ctx, types.NilUniqueID(), &testTypeA, nil, nil, suite.sampleCollections[0].ID) + suite.NoError(err) + suite.Equal(sampleSegments[:1], result) // Find by type and collection ID (negative case) - result, err = c.GetSegments(ctx, types.NilUniqueID(), &testTypeB, nil, nil, sampleCollections[0].ID) - assert.NoError(suite.t, err) - assert.Empty(suite.t, result) + result, err = c.GetSegments(ctx, types.NilUniqueID(), &testTypeB, nil, nil, suite.sampleCollections[0].ID) + suite.NoError(err) + suite.Empty(result) // Delete s1 := sampleSegments[0] err = c.DeleteSegment(ctx, s1.ID) - assert.NoError(suite.t, err) + suite.NoError(err) results, err = c.GetSegments(ctx, types.NilUniqueID(), nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.NotContains(suite.t, results, s1) - assert.Len(suite.t, results, len(sampleSegments)-1) - assert.ElementsMatch(suite.t, results, sampleSegments[1:]) + suite.NoError(err) + suite.NotContains(results, s1) + suite.Len(results, len(sampleSegments)-1) + suite.ElementsMatch(results, sampleSegments[1:]) // Duplicate delete throws an exception err = c.DeleteSegment(ctx, s1.ID) - assert.Error(suite.t, err) -} + suite.Error(err) -func (suite *APIsTestSuite) TestUpdateSegment() { - sampleCollections := SampleCollections(suite.t, common.DefaultTenant, common.DefaultDatabase) - ctx := context.Background() - assignmentPolicy := NewMockAssignmentPolicy(sampleCollections) - c, err := NewCoordinator(ctx, assignmentPolicy, suite.db, nil, nil) - if err != nil { - suite.t.Fatalf("error creating coordinator: %v", err) + // clean up segments + for _, segment := range sampleSegments { + _ = c.DeleteSegment(ctx, segment.ID) } - c.ResetState(ctx) +} +func (suite *APIsTestSuite) TestUpdateSegment() { testTopic := "test_topic_a" metadata := model.NewSegmentMetadata[model.SegmentMetadataValueType]() @@ -857,25 +845,13 @@ func (suite *APIsTestSuite) TestUpdateSegment() { Type: "test_type_a", Scope: "VECTOR", Topic: &testTopic, - CollectionID: sampleCollections[0].ID, + CollectionID: suite.sampleCollections[0].ID, Metadata: metadata, + FilePaths: map[string][]string{}, } - for _, collection := range sampleCollections { - _, err := c.CreateCollection(ctx, &model.CreateCollection{ - ID: collection.ID, - Name: collection.Name, - Topic: collection.Topic, - Metadata: collection.Metadata, - Dimension: collection.Dimension, - TenantID: collection.TenantID, - DatabaseName: collection.DatabaseName, - }) - - assert.NoError(suite.t, err) - } - - c.CreateSegment(ctx, &model.CreateSegment{ + ctx := context.Background() + errSegmentCreation := suite.coordinator.CreateSegment(ctx, &model.CreateSegment{ ID: segment.ID, Type: segment.Type, Topic: segment.Topic, @@ -883,31 +859,34 @@ func (suite *APIsTestSuite) TestUpdateSegment() { CollectionID: segment.CollectionID, Metadata: segment.Metadata, }) + suite.NoError(errSegmentCreation) // Update topic to new value collectionID := segment.CollectionID.String() newTopic := "new_topic" segment.Topic = &newTopic - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err := suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Topic: segment.Topic, }) - result, err := c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err := suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) // Update topic to None segment.Topic = nil - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err = suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Topic: segment.Topic, ResetTopic: true, }) - result, err = c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err = suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) // TODO: revisit why we need this // Update collection to new value @@ -934,51 +913,54 @@ func (suite *APIsTestSuite) TestUpdateSegment() { // Add a new metadata key segment.Metadata.Set("test_str2", &model.SegmentMetadataValueStringType{Value: "str2"}) - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err = suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Metadata: segment.Metadata}) - result, err = c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err = suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) // Update a metadata key segment.Metadata.Set("test_str", &model.SegmentMetadataValueStringType{Value: "str3"}) - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err = suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Metadata: segment.Metadata}) - result, err = c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err = suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) // Delete a metadata key segment.Metadata.Remove("test_str") newMetadata := model.NewSegmentMetadata[model.SegmentMetadataValueType]() newMetadata.Set("test_str", nil) - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err = suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Metadata: newMetadata}) - result, err = c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err = suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) // Delete all metadata keys segment.Metadata = nil - c.UpdateSegment(ctx, &model.UpdateSegment{ + _, err = suite.coordinator.UpdateSegment(ctx, &model.UpdateSegment{ Collection: &collectionID, ID: segment.ID, Metadata: segment.Metadata, ResetMetadata: true}, ) - result, err = c.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) - assert.NoError(suite.t, err) - assert.Equal(suite.t, []*model.Segment{segment}, result) + suite.NoError(err) + result, err = suite.coordinator.GetSegments(ctx, segment.ID, nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Equal([]*model.Segment{segment}, result) } func TestAPIsTestSuite(t *testing.T) { testSuite := new(APIsTestSuite) - testSuite.t = t suite.Run(t, testSuite) } diff --git a/go/pkg/coordinator/coordinator.go b/go/pkg/coordinator/coordinator.go index 110b641be44..d52aeaf8954 100644 --- a/go/pkg/coordinator/coordinator.go +++ b/go/pkg/coordinator/coordinator.go @@ -38,7 +38,6 @@ func NewCoordinator(ctx context.Context, assignmentPolicy CollectionAssignmentPo txnImpl := dbcore.NewTxImpl() metaDomain := dao.NewMetaDomain() s.catalog = coordinator.NewTableCatalogWithNotification(txnImpl, metaDomain, notificationStore) - return s, nil } diff --git a/go/pkg/coordinator/grpc/collection_service.go b/go/pkg/coordinator/grpc/collection_service.go index aa9b6a7151f..a6b9816ec62 100644 --- a/go/pkg/coordinator/grpc/collection_service.go +++ b/go/pkg/coordinator/grpc/collection_service.go @@ -2,7 +2,9 @@ package grpc import ( "context" + "encoding/json" "errors" + "github.com/chroma-core/chroma/go/pkg/grpcutils" "github.com/chroma-core/chroma/go/pkg/common" "github.com/chroma-core/chroma/go/pkg/model" @@ -211,6 +213,53 @@ func (s *Server) UpdateCollection(ctx context.Context, req *coordinatorpb.Update return res, nil } +func (s *Server) FlushCollectionCompaction(ctx context.Context, req *coordinatorpb.FlushCollectionCompactionRequest) (*coordinatorpb.FlushCollectionCompactionResponse, error) { + blob, err := json.Marshal(req) + if err != nil { + return nil, err + } + log.Info("flush collection compaction", zap.String("request", string(blob))) + collectionID, err := types.ToUniqueID(&req.CollectionId) + err = grpcutils.BuildErrorForUUID(collectionID, "collection", err) + if err != nil { + return nil, err + } + segmentCompactionInfo := make([]*model.FlushSegmentCompaction, 0, len(req.SegmentCompactionInfo)) + for _, flushSegmentCompaction := range req.SegmentCompactionInfo { + segmentID, err := types.ToUniqueID(&flushSegmentCompaction.SegmentId) + err = grpcutils.BuildErrorForUUID(segmentID, "segment", err) + if err != nil { + return nil, err + } + filePaths := make(map[string][]string) + for key, filePath := range flushSegmentCompaction.FilePaths { + filePaths[key] = filePath.Paths + } + segmentCompactionInfo = append(segmentCompactionInfo, &model.FlushSegmentCompaction{ + ID: segmentID, + FilePaths: filePaths, + }) + } + FlushCollectionCompaction := &model.FlushCollectionCompaction{ + ID: collectionID, + TenantID: req.TenantId, + LogPosition: req.LogPosition, + CurrentCollectionVersion: req.CollectionVersion, + FlushSegmentCompactions: segmentCompactionInfo, + } + flushCollectionInfo, err := s.coordinator.FlushCollectionCompaction(ctx, FlushCollectionCompaction) + if err != nil { + log.Error("error FlushCollectionCompaction", zap.Error(err)) + return nil, grpcutils.BuildInternalGrpcError(err.Error()) + } + res := &coordinatorpb.FlushCollectionCompactionResponse{ + CollectionId: flushCollectionInfo.ID, + CollectionVersion: flushCollectionInfo.CollectionVersion, + LastCompactionTime: flushCollectionInfo.TenantLastCompactionTime, + } + return res, nil +} + func failResponseWithError(err error, code int32) *coordinatorpb.Status { return &coordinatorpb.Status{ Reason: err.Error(), diff --git a/go/pkg/coordinator/grpc/collection_service_test.go b/go/pkg/coordinator/grpc/collection_service_test.go index a300d4c9b3a..9e86c8ff4f1 100644 --- a/go/pkg/coordinator/grpc/collection_service_test.go +++ b/go/pkg/coordinator/grpc/collection_service_test.go @@ -2,15 +2,67 @@ package grpc import ( "context" - "github.com/chroma-core/chroma/go/pkg/grpcutils" - "testing" - "github.com/chroma-core/chroma/go/pkg/common" + "github.com/chroma-core/chroma/go/pkg/grpcutils" + "github.com/chroma-core/chroma/go/pkg/metastore/coordinator" + "github.com/chroma-core/chroma/go/pkg/metastore/db/dao" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/chroma-core/chroma/go/pkg/proto/coordinatorpb" + "github.com/pingcap/log" + "github.com/stretchr/testify/suite" + "google.golang.org/genproto/googleapis/rpc/code" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + "gorm.io/gorm" + "k8s.io/apimachinery/pkg/util/rand" "pgregory.net/rapid" + "reflect" + "strconv" + "testing" + "time" ) +type CollectionServiceTestSuite struct { + suite.Suite + catalog *coordinator.Catalog + db *gorm.DB + s *Server + tenantName string + databaseName string + databaseId string +} + +func (suite *CollectionServiceTestSuite) SetupSuite() { + log.Info("setup suite") + suite.db = dbcore.ConfigDatabaseForTesting() + s, err := NewWithGrpcProvider(Config{ + AssignmentPolicy: "simple", + SystemCatalogProvider: "database", + NotificationStoreProvider: "memory", + NotifierProvider: "memory", + Testing: true}, grpcutils.Default, suite.db) + if err != nil { + suite.T().Fatalf("error creating server: %v", err) + } + suite.s = s + txnImpl := dbcore.NewTxImpl() + metaDomain := dao.NewMetaDomain() + suite.catalog = coordinator.NewTableCatalogWithNotification(txnImpl, metaDomain, nil) + suite.tenantName = "tenant_" + suite.T().Name() + suite.databaseName = "database_" + suite.T().Name() + DbId, err := dao.CreateTestTenantAndDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.databaseId = DbId +} + +func (suite *CollectionServiceTestSuite) TearDownSuite() { + log.Info("teardown suite") + err := dao.CleanUpTestDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + err = dao.CleanUpTestTenant(suite.db, suite.tenantName) + suite.NoError(err) +} + // CreateCollection // Collection created successfully are visible to ListCollections // Collection created should have the right metadata, the metadata should be a flat map, with keys as strings and values as strings, ints, or floats @@ -123,3 +175,168 @@ func generateFloat64MetadataValue(t *rapid.T) *coordinatorpb.UpdateMetadataValue func TestCollection(t *testing.T) { // rapid.Check(t, testCollection) } + +func validateDatabase(suite *CollectionServiceTestSuite, collectionId string, collection *coordinatorpb.Collection, filePaths map[string]map[string]*coordinatorpb.FilePaths) { + getCollectionReq := coordinatorpb.GetCollectionsRequest{ + Id: &collectionId, + } + collectionsInDB, err := suite.s.GetCollections(context.Background(), &getCollectionReq) + suite.NoError(err) + suite.Len(collectionsInDB.Collections, 1) + suite.Equal(collection.Id, collection.Id) + suite.Equal(collection.Name, collection.Name) + suite.Equal(collection.Topic, collection.Topic) + suite.Equal(collection.LogPosition, collection.LogPosition) + suite.Equal(collection.Version, collection.Version) + + getSegmentReq := coordinatorpb.GetSegmentsRequest{ + Collection: &collectionId, + } + segments, err := suite.s.GetSegments(context.Background(), &getSegmentReq) + suite.NoError(err) + for _, segment := range segments.Segments { + suite.True(reflect.DeepEqual(filePaths[segment.Id], segment.FilePaths)) + } +} + +func (suite *CollectionServiceTestSuite) TestServer_FlushCollectionCompaction() { + log.Info("TestServer_FlushCollectionCompaction") + // create test collection + collectionName := "collection_service_test_flush_collection_compaction" + collectionTopic := "collection_service_test_flush_collection_compaction_topic" + collectionID, err := dao.CreateTestCollection(suite.db, collectionName, collectionTopic, 128, suite.databaseId) + suite.NoError(err) + + // flush collection compaction + getSegmentReq := coordinatorpb.GetSegmentsRequest{ + Collection: &collectionID, + } + segments, err := suite.s.GetSegments(context.Background(), &getSegmentReq) + suite.NoError(err) + + flushInfo := make([]*coordinatorpb.FlushSegmentCompactionInfo, 0, len(segments.Segments)) + filePaths := make(map[string]map[string]*coordinatorpb.FilePaths, 0) + testFilePathTypes := []string{"TypeA", "TypeB", "TypeC", "TypeD"} + for _, segment := range segments.Segments { + filePaths[segment.Id] = make(map[string]*coordinatorpb.FilePaths, 0) + for i := 0; i < rand.Intn(len(testFilePathTypes)); i++ { + filePathsThisSeg := make([]string, 0) + for j := 0; j < rand.Intn(5); j++ { + filePathsThisSeg = append(filePathsThisSeg, "test_file_path_"+strconv.Itoa(j+1)) + } + filePathTypeI := rand.Intn(len(testFilePathTypes)) + filePaths[segment.Id][testFilePathTypes[filePathTypeI]] = &coordinatorpb.FilePaths{ + Paths: filePathsThisSeg, + } + } + info := &coordinatorpb.FlushSegmentCompactionInfo{ + SegmentId: segment.Id, + FilePaths: filePaths[segment.Id], + } + flushInfo = append(flushInfo, info) + } + + req := &coordinatorpb.FlushCollectionCompactionRequest{ + TenantId: suite.tenantName, + CollectionId: collectionID, + LogPosition: 10, + CollectionVersion: 0, + SegmentCompactionInfo: flushInfo, + } + response, err := suite.s.FlushCollectionCompaction(context.Background(), req) + t1 := time.Now().Unix() + suite.NoError(err) + suite.Equal(collectionID, response.CollectionId) + suite.Equal(int32(1), response.CollectionVersion) + suite.Less(int64(0), response.LastCompactionTime) + suite.LessOrEqual(response.LastCompactionTime, t1) + + // validate database + collection := &coordinatorpb.Collection{ + Id: collectionID, + LogPosition: int64(10), + Version: int32(1), + } + validateDatabase(suite, collectionID, collection, filePaths) + + // flush one segment + filePaths[segments.Segments[0].Id][testFilePathTypes[0]] = &coordinatorpb.FilePaths{ + Paths: []string{"test_file_path_1"}, + } + info := &coordinatorpb.FlushSegmentCompactionInfo{ + SegmentId: segments.Segments[0].Id, + FilePaths: filePaths[segments.Segments[0].Id], + } + req = &coordinatorpb.FlushCollectionCompactionRequest{ + TenantId: suite.tenantName, + CollectionId: collectionID, + LogPosition: 100, + CollectionVersion: 1, + SegmentCompactionInfo: []*coordinatorpb.FlushSegmentCompactionInfo{info}, + } + response, err = suite.s.FlushCollectionCompaction(context.Background(), req) + t2 := time.Now().Unix() + suite.NoError(err) + suite.Equal(collectionID, response.CollectionId) + suite.Equal(int32(2), response.CollectionVersion) + suite.LessOrEqual(t1, response.LastCompactionTime) + suite.LessOrEqual(response.LastCompactionTime, t2) + + // validate database + collection = &coordinatorpb.Collection{ + Id: collectionID, + LogPosition: int64(100), + Version: int32(2), + } + validateDatabase(suite, collectionID, collection, filePaths) + + // test invalid log position + req = &coordinatorpb.FlushCollectionCompactionRequest{ + TenantId: suite.tenantName, + CollectionId: collectionID, + LogPosition: 50, + CollectionVersion: 2, + SegmentCompactionInfo: []*coordinatorpb.FlushSegmentCompactionInfo{info}, + } + response, err = suite.s.FlushCollectionCompaction(context.Background(), req) + suite.Error(err) + suite.Equal(status.Error(codes.Code(code.Code_INTERNAL), common.ErrCollectionLogPositionStale.Error()), err) + // nothing should change in DB + validateDatabase(suite, collectionID, collection, filePaths) + + // test invalid version + req = &coordinatorpb.FlushCollectionCompactionRequest{ + TenantId: suite.tenantName, + CollectionId: collectionID, + LogPosition: 150, + CollectionVersion: 1, + SegmentCompactionInfo: []*coordinatorpb.FlushSegmentCompactionInfo{info}, + } + response, err = suite.s.FlushCollectionCompaction(context.Background(), req) + suite.Error(err) + suite.Equal(status.Error(codes.Code(code.Code_INTERNAL), common.ErrCollectionVersionStale.Error()), err) + // nothing should change in DB + validateDatabase(suite, collectionID, collection, filePaths) + + req = &coordinatorpb.FlushCollectionCompactionRequest{ + TenantId: suite.tenantName, + CollectionId: collectionID, + LogPosition: 150, + CollectionVersion: 5, + SegmentCompactionInfo: []*coordinatorpb.FlushSegmentCompactionInfo{info}, + } + response, err = suite.s.FlushCollectionCompaction(context.Background(), req) + suite.Error(err) + suite.Equal(status.Error(codes.Code(code.Code_INTERNAL), common.ErrCollectionVersionInvalid.Error()), err) + // nothing should change in DB + validateDatabase(suite, collectionID, collection, filePaths) + + // clean up + err = dao.CleanUpTestCollection(suite.db, collectionID) + suite.NoError(err) +} + +func TestCollectionServiceTestSuite(t *testing.T) { + testSuite := new(CollectionServiceTestSuite) + suite.Run(t, testSuite) +} diff --git a/go/pkg/coordinator/grpc/proto_model_convert.go b/go/pkg/coordinator/grpc/proto_model_convert.go index 1f396d20880..61359b2fdc0 100644 --- a/go/pkg/coordinator/grpc/proto_model_convert.go +++ b/go/pkg/coordinator/grpc/proto_model_convert.go @@ -38,12 +38,14 @@ func convertCollectionToProto(collection *model.Collection) *coordinatorpb.Colle } collectionpb := &coordinatorpb.Collection{ - Id: collection.ID.String(), - Name: collection.Name, - Topic: collection.Topic, - Dimension: collection.Dimension, - Tenant: collection.TenantID, - Database: collection.DatabaseName, + Id: collection.ID.String(), + Name: collection.Name, + Topic: collection.Topic, + Dimension: collection.Dimension, + Tenant: collection.TenantID, + Database: collection.DatabaseName, + LogPosition: collection.LogPosition, + Version: collection.Version, } if collection.Metadata == nil { return collectionpb @@ -145,6 +147,12 @@ func convertSegmentToProto(segment *model.Segment) *coordinatorpb.Segment { } scope := coordinatorpb.SegmentScope_value[segment.Scope] segmentSceope := coordinatorpb.SegmentScope(scope) + filePaths := make(map[string]*coordinatorpb.FilePaths) + for t, paths := range segment.FilePaths { + filePaths[t] = &coordinatorpb.FilePaths{ + Paths: paths, + } + } segmentpb := &coordinatorpb.Segment{ Id: segment.ID.String(), Type: segment.Type, @@ -152,6 +160,7 @@ func convertSegmentToProto(segment *model.Segment) *coordinatorpb.Segment { Topic: segment.Topic, Collection: nil, Metadata: nil, + FilePaths: filePaths, } collectionID := segment.CollectionID diff --git a/go/pkg/coordinator/grpc/proto_model_convert_test.go b/go/pkg/coordinator/grpc/proto_model_convert_test.go index e875233aa72..6033fff5a37 100644 --- a/go/pkg/coordinator/grpc/proto_model_convert_test.go +++ b/go/pkg/coordinator/grpc/proto_model_convert_test.go @@ -184,11 +184,12 @@ func TestConvertSegmentToProto(t *testing.T) { // Test case 2: segment is not nil testTopic := "test_topic" segment := &model.Segment{ - ID: types.NewUniqueID(), - Type: "test_type", - Scope: "METADATA", - Topic: &testTopic, - Metadata: nil, + ID: types.NewUniqueID(), + Type: "test_type", + Scope: "METADATA", + Topic: &testTopic, + Metadata: nil, + FilePaths: map[string][]string{}, } segmentpb = convertSegmentToProto(segment) assert.NotNil(t, segmentpb) diff --git a/go/pkg/coordinator/grpc/server.go b/go/pkg/coordinator/grpc/server.go index 0f4841215f8..b531cbc3a44 100644 --- a/go/pkg/coordinator/grpc/server.go +++ b/go/pkg/coordinator/grpc/server.go @@ -3,9 +3,10 @@ package grpc import ( "context" "errors" - "github.com/chroma-core/chroma/go/pkg/grpcutils" "time" + "github.com/chroma-core/chroma/go/pkg/grpcutils" + "github.com/apache/pulsar-client-go/pulsar" "github.com/chroma-core/chroma/go/pkg/coordinator" "github.com/chroma-core/chroma/go/pkg/memberlist_manager" @@ -45,6 +46,7 @@ type Config struct { // Kubernetes config KubernetesNamespace string WorkerMemberlistName string + WorkerPodLabel string // Assignment policy config can be "simple" or "rendezvous" AssignmentPolicy string @@ -172,8 +174,7 @@ func NewWithGrpcProvider(config Config, provider grpcutils.GrpcProvider, db *gor } func createMemberlistManager(config Config) (*memberlist_manager.MemberlistManager, error) { - // TODO: Make this configuration - log.Info("Starting memberlist manager") + log.Info("Creating memberlist manager") memberlist_name := config.WorkerMemberlistName namespace := config.KubernetesNamespace clientset, err := utils.GetKubernetesInterface() @@ -184,7 +185,7 @@ func createMemberlistManager(config Config) (*memberlist_manager.MemberlistManag if err != nil { return nil, err } - nodeWatcher := memberlist_manager.NewKubernetesWatcher(clientset, namespace, "worker", config.WatchInterval) + nodeWatcher := memberlist_manager.NewKubernetesWatcher(clientset, namespace, config.WorkerPodLabel, config.WatchInterval) memberlistStore := memberlist_manager.NewCRMemberlistStore(dynamicClient, namespace, memberlist_name) memberlist_manager := memberlist_manager.NewMemberlistManager(nodeWatcher, memberlistStore) return memberlist_manager, nil diff --git a/go/pkg/coordinator/grpc/tenant_database_service.go b/go/pkg/coordinator/grpc/tenant_database_service.go index 56c5f224218..7ae4445fb08 100644 --- a/go/pkg/coordinator/grpc/tenant_database_service.go +++ b/go/pkg/coordinator/grpc/tenant_database_service.go @@ -98,7 +98,7 @@ func (s *Server) SetLastCompactionTimeForTenant(ctx context.Context, req *coordi err := s.coordinator.SetTenantLastCompactionTime(ctx, req.TenantLastCompactionTime.TenantId, req.TenantLastCompactionTime.LastCompactionTime) if err != nil { log.Error("error SetTenantLastCompactionTime", zap.Any("request", req.TenantLastCompactionTime), zap.Error(err)) - return nil, grpcutils.BuildInternalGrpcError("error SetTenantLastCompactionTime") + return nil, grpcutils.BuildInternalGrpcError(err.Error()) } return &emptypb.Empty{}, nil } @@ -109,7 +109,7 @@ func (s *Server) GetLastCompactionTimeForTenant(ctx context.Context, req *coordi tenants, err := s.coordinator.GetTenantsLastCompactionTime(ctx, tenantIDs) if err != nil { log.Error("error GetLastCompactionTimeForTenant", zap.Any("tenantIDs", tenantIDs), zap.Error(err)) - return nil, grpcutils.BuildInternalGrpcError("error GetTenantsLastCompactionTime") + return nil, grpcutils.BuildInternalGrpcError(err.Error()) } for _, tenant := range tenants { res.TenantLastCompactionTime = append(res.TenantLastCompactionTime, &coordinatorpb.TenantLastCompactionTime{ diff --git a/go/pkg/coordinator/grpc/tenant_database_service_test.go b/go/pkg/coordinator/grpc/tenant_database_service_test.go index 153d721cc27..4f37b060734 100644 --- a/go/pkg/coordinator/grpc/tenant_database_service_test.go +++ b/go/pkg/coordinator/grpc/tenant_database_service_test.go @@ -2,13 +2,13 @@ package grpc import ( "context" + "github.com/chroma-core/chroma/go/pkg/common" "github.com/chroma-core/chroma/go/pkg/grpcutils" "github.com/chroma-core/chroma/go/pkg/metastore/coordinator" "github.com/chroma-core/chroma/go/pkg/metastore/db/dao" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/chroma-core/chroma/go/pkg/model" "github.com/chroma-core/chroma/go/pkg/proto/coordinatorpb" - "github.com/chroma-core/chroma/go/pkg/types" "github.com/pingcap/log" "github.com/stretchr/testify/suite" "google.golang.org/genproto/googleapis/rpc/code" @@ -21,11 +21,9 @@ import ( type TenantDatabaseServiceTestSuite struct { suite.Suite - catalog *coordinator.Catalog - db *gorm.DB - s *Server - t *testing.T - collectionId types.UniqueID + catalog *coordinator.Catalog + db *gorm.DB + s *Server } func (suite *TenantDatabaseServiceTestSuite) SetupSuite() { @@ -33,12 +31,12 @@ func (suite *TenantDatabaseServiceTestSuite) SetupSuite() { suite.db = dbcore.ConfigDatabaseForTesting() s, err := NewWithGrpcProvider(Config{ AssignmentPolicy: "simple", - SystemCatalogProvider: "memory", + SystemCatalogProvider: "database", NotificationStoreProvider: "memory", NotifierProvider: "memory", Testing: true}, grpcutils.Default, suite.db) if err != nil { - suite.t.Fatalf("error creating server: %v", err) + suite.T().Fatalf("error creating server: %v", err) } suite.s = s txnImpl := dbcore.NewTxImpl() @@ -52,8 +50,6 @@ func (suite *TenantDatabaseServiceTestSuite) SetupTest() { func (suite *TenantDatabaseServiceTestSuite) TearDownTest() { log.Info("teardown test") - // TODO: clean up per test when delete is implemented for tenant - dbcore.ResetTestTables(suite.db) } func (suite *TenantDatabaseServiceTestSuite) TestServer_TenantLastCompactionTime() { @@ -66,7 +62,7 @@ func (suite *TenantDatabaseServiceTestSuite) TestServer_TenantLastCompactionTime }, } _, err := suite.s.SetLastCompactionTimeForTenant(context.Background(), request) - suite.Equal(status.Error(codes.Code(code.Code_INTERNAL), "error SetTenantLastCompactionTime"), err) + suite.Equal(status.Error(codes.Code(code.Code_INTERNAL), common.ErrTenantNotFound.Error()), err) // create tenant _, err = suite.catalog.CreateTenant(context.Background(), &model.CreateTenant{ @@ -99,10 +95,13 @@ func (suite *TenantDatabaseServiceTestSuite) TestServer_TenantLastCompactionTime suite.Equal(1, len(tenants.TenantLastCompactionTime)) suite.Equal(tenantId, tenants.TenantLastCompactionTime[0].TenantId) suite.Equal(int64(1), tenants.TenantLastCompactionTime[0].LastCompactionTime) + + // clean up + err = dao.CleanUpTestTenant(suite.db, tenantId) + suite.NoError(err) } func TestTenantDatabaseServiceTestSuite(t *testing.T) { testSuite := new(TenantDatabaseServiceTestSuite) - testSuite.t = t suite.Run(t, testSuite) } diff --git a/go/pkg/grpcutils/response.go b/go/pkg/grpcutils/response.go index 981bdba1011..5a89344eb30 100644 --- a/go/pkg/grpcutils/response.go +++ b/go/pkg/grpcutils/response.go @@ -31,10 +31,10 @@ func BuildInternalGrpcError(msg string) error { return status.Error(codes.Internal, msg) } -func BuildErrorForCollectionId(collectionID types.UniqueID, err error) error { - if err != nil || collectionID == types.NilUniqueID() { - log.Error("collection id format error", zap.String("collection.id", collectionID.String())) - grpcError, err := BuildInvalidArgumentGrpcError("collection_id", "wrong collection_id format") +func BuildErrorForUUID(ID types.UniqueID, name string, err error) error { + if err != nil || ID == types.NilUniqueID() { + log.Error(name+"id format error", zap.String(name+".id", ID.String())) + grpcError, err := BuildInvalidArgumentGrpcError(name+"_id", "wrong "+name+"_id format") if err != nil { log.Error("error building grpc error", zap.Error(err)) return err diff --git a/go/pkg/logservice/apis.go b/go/pkg/logservice/apis.go index 778b0397152..40736ee6cfa 100644 --- a/go/pkg/logservice/apis.go +++ b/go/pkg/logservice/apis.go @@ -2,6 +2,7 @@ package logservice import ( "context" + "github.com/chroma-core/chroma/go/pkg/common" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/types" @@ -11,17 +12,19 @@ type ( IRecordLog interface { common.Component PushLogs(ctx context.Context, collectionID types.UniqueID, recordContent [][]byte) (int, error) - PullLogs(ctx context.Context, collectionID types.UniqueID, id int64, batchSize int) ([]*dbmodel.RecordLog, error) + PullLogs(ctx context.Context, collectionID types.UniqueID, id int64, batchSize int, endTimestamp int64) ([]*dbmodel.RecordLog, error) GetAllCollectionIDsToCompact() ([]*dbmodel.RecordLog, error) } ) +var _ IRecordLog = &RecordLog{} + func (s *RecordLog) PushLogs(ctx context.Context, collectionID types.UniqueID, recordsContent [][]byte) (int, error) { return s.recordLogDb.PushLogs(collectionID, recordsContent) } -func (s *RecordLog) PullLogs(ctx context.Context, collectionID types.UniqueID, id int64, batchSize int) ([]*dbmodel.RecordLog, error) { - return s.recordLogDb.PullLogs(collectionID, id, batchSize) +func (s *RecordLog) PullLogs(ctx context.Context, collectionID types.UniqueID, id int64, batchSize int, endTimestamp int64) ([]*dbmodel.RecordLog, error) { + return s.recordLogDb.PullLogs(collectionID, id, batchSize, endTimestamp) } func (s *RecordLog) GetAllCollectionIDsToCompact() ([]*dbmodel.RecordLog, error) { diff --git a/go/pkg/logservice/grpc/record_log_service.go b/go/pkg/logservice/grpc/record_log_service.go index 1aa88eb956c..40d38a3e57b 100644 --- a/go/pkg/logservice/grpc/record_log_service.go +++ b/go/pkg/logservice/grpc/record_log_service.go @@ -2,6 +2,7 @@ package grpc import ( "context" + "github.com/chroma-core/chroma/go/pkg/grpcutils" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/proto/coordinatorpb" @@ -21,12 +22,14 @@ type CollectionInfo struct { func (s *Server) PushLogs(ctx context.Context, req *logservicepb.PushLogsRequest) (*logservicepb.PushLogsResponse, error) { res := &logservicepb.PushLogsResponse{} collectionID, err := types.ToUniqueID(&req.CollectionId) - err = grpcutils.BuildErrorForCollectionId(collectionID, err) + err = grpcutils.BuildErrorForUUID(collectionID, "collection", err) if err != nil { return nil, err } var recordsContent [][]byte for _, record := range req.Records { + // We remove the collection id for space reasons, as its double stored in the wrapping database RecordLog object. + // PullLogs will rehydrate the collection id from the database. record.CollectionId = "" data, err := proto.Marshal(record) if err != nil { @@ -42,7 +45,7 @@ func (s *Server) PushLogs(ctx context.Context, req *logservicepb.PushLogsRequest recordCount, err := s.logService.PushLogs(ctx, collectionID, recordsContent) if err != nil { log.Error("error pushing logs", zap.Error(err)) - return nil, grpcutils.BuildInternalGrpcError("error pushing logs") + return nil, grpcutils.BuildInternalGrpcError(err.Error()) } res.RecordCount = int32(recordCount) log.Info("PushLogs success", zap.String("collectionID", req.CollectionId), zap.Int("recordCount", recordCount)) @@ -52,15 +55,15 @@ func (s *Server) PushLogs(ctx context.Context, req *logservicepb.PushLogsRequest func (s *Server) PullLogs(ctx context.Context, req *logservicepb.PullLogsRequest) (*logservicepb.PullLogsResponse, error) { res := &logservicepb.PullLogsResponse{} collectionID, err := types.ToUniqueID(&req.CollectionId) - err = grpcutils.BuildErrorForCollectionId(collectionID, err) + err = grpcutils.BuildErrorForUUID(collectionID, "collection", err) if err != nil { return nil, err } records := make([]*logservicepb.RecordLog, 0) - recordLogs, err := s.logService.PullLogs(ctx, collectionID, req.GetStartFromId(), int(req.BatchSize)) + recordLogs, err := s.logService.PullLogs(ctx, collectionID, req.GetStartFromId(), int(req.BatchSize), req.GetEndTimestamp()) if err != nil { log.Error("error pulling logs", zap.Error(err)) - return nil, grpcutils.BuildInternalGrpcError("error pulling logs") + return nil, grpcutils.BuildInternalGrpcError(err.Error()) } for index := range recordLogs { record := &coordinatorpb.SubmitEmbeddingRecord{} @@ -72,6 +75,8 @@ func (s *Server) PullLogs(ctx context.Context, req *logservicepb.PullLogsRequest } return nil, grpcError } + // Here we rehydrate the collection id from the database since in PushLogs we removed it for space reasons. + record.CollectionId = *recordLogs[index].CollectionID recordLog := &logservicepb.RecordLog{ LogId: recordLogs[index].ID, Record: record, @@ -90,7 +95,7 @@ func (s *Server) GetAllCollectionInfoToCompact(ctx context.Context, req *logserv recordLogs, err := s.logService.GetAllCollectionIDsToCompact() if err != nil { log.Error("error getting collection info", zap.Error(err)) - return nil, grpcutils.BuildInternalGrpcError("error getting collection info") + return nil, grpcutils.BuildInternalGrpcError(err.Error()) } for _, recordLog := range recordLogs { collectionInfo := &logservicepb.CollectionInfo{ diff --git a/go/pkg/logservice/grpc/record_log_service_test.go b/go/pkg/logservice/grpc/record_log_service_test.go index 52ac27c0176..3e453e351ed 100644 --- a/go/pkg/logservice/grpc/record_log_service_test.go +++ b/go/pkg/logservice/grpc/record_log_service_test.go @@ -4,6 +4,9 @@ import ( "bytes" "context" "encoding/binary" + "testing" + "time" + "github.com/chroma-core/chroma/go/pkg/logservice/testutils" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" @@ -11,21 +14,17 @@ import ( "github.com/chroma-core/chroma/go/pkg/proto/logservicepb" "github.com/chroma-core/chroma/go/pkg/types" "github.com/pingcap/log" - "github.com/stretchr/testify/assert" "github.com/stretchr/testify/suite" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" "google.golang.org/protobuf/proto" "gorm.io/gorm" - "testing" - "time" ) type RecordLogServiceTestSuite struct { suite.Suite db *gorm.DB s *Server - t *testing.T collectionId types.UniqueID } @@ -38,18 +37,25 @@ func (suite *RecordLogServiceTestSuite) SetupSuite() { StartGrpc: false, }) suite.s = s - suite.db = dbcore.GetDB(context.Background()) - suite.collectionId = types.NewUniqueID() + suite.db = dbcore.ConfigDatabaseForTesting() + recordLogTableExist := suite.db.Migrator().HasTable(&dbmodel.RecordLog{}) + if !recordLogTableExist { + err := suite.db.Migrator().CreateTable(&dbmodel.RecordLog{}) + suite.NoError(err) + } } func (suite *RecordLogServiceTestSuite) SetupTest() { log.Info("setup test") - testutils.SetupTest(suite.db, suite.collectionId) + suite.collectionId = types.NewUniqueID() + err := testutils.CreateCollections(suite.db, suite.collectionId) + suite.NoError(err) } func (suite *RecordLogServiceTestSuite) TearDownTest() { log.Info("teardown test") - testutils.TearDownTest(suite.db) + err := testutils.CleanupCollections(suite.db, suite.collectionId) + suite.NoError(err) } func encodeVector(dimension int32, vector []float32, encoding coordinatorpb.ScalarEncoding) *coordinatorpb.Vector { @@ -101,37 +107,43 @@ func (suite *RecordLogServiceTestSuite) TestServer_PushLogs() { Records: recordsToSubmit, } response, err := suite.s.PushLogs(context.Background(), &pushRequest) - assert.Nil(suite.t, err) - assert.Equal(suite.t, int32(3), response.RecordCount) + suite.NoError(err) + suite.Equal(int32(3), response.RecordCount) var recordLogs []*dbmodel.RecordLog suite.db.Where("collection_id = ?", types.FromUniqueID(suite.collectionId)).Find(&recordLogs) - assert.Len(suite.t, recordLogs, 3) + suite.Len(recordLogs, 3) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID) - assert.Equal(suite.t, suite.collectionId.String(), *recordLogs[index].CollectionID) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.collectionId.String(), *recordLogs[index].CollectionID) record := &coordinatorpb.SubmitEmbeddingRecord{} - if err := proto.Unmarshal(*recordLogs[index].Record, record); err != nil { - panic(err) + if unmarshalErr := proto.Unmarshal(*recordLogs[index].Record, record); err != nil { + suite.NoError(unmarshalErr) } - assert.Equal(suite.t, record.Id, recordsToSubmit[index].Id) - assert.Equal(suite.t, record.Operation, recordsToSubmit[index].Operation) - assert.Equal(suite.t, record.CollectionId, "") - assert.Equal(suite.t, record.Metadata, recordsToSubmit[index].Metadata) - assert.Equal(suite.t, record.Vector.Dimension, recordsToSubmit[index].Vector.Dimension) - assert.Equal(suite.t, record.Vector.Encoding, recordsToSubmit[index].Vector.Encoding) - assert.Equal(suite.t, record.Vector.Vector, recordsToSubmit[index].Vector.Vector) + suite.Equal(recordsToSubmit[index].Id, record.Id) + suite.Equal(recordsToSubmit[index].Operation, record.Operation) + suite.Equal("", record.CollectionId) + suite.Equal(recordsToSubmit[index].Metadata, record.Metadata) + suite.Equal(recordsToSubmit[index].Vector.Dimension, record.Vector.Dimension) + suite.Equal(recordsToSubmit[index].Vector.Encoding, record.Vector.Encoding) + suite.Equal(recordsToSubmit[index].Vector.Vector, record.Vector.Vector) } } func (suite *RecordLogServiceTestSuite) TestServer_PullLogs() { // push some records recordsToSubmit := GetTestEmbeddingRecords(suite.collectionId.String()) + // deep clone the records since PushLogs will mutate the records and we need a source of truth + recordsToSubmit_sot := make([]*coordinatorpb.SubmitEmbeddingRecord, len(recordsToSubmit)) + for i := range recordsToSubmit { + recordsToSubmit_sot[i] = proto.Clone(recordsToSubmit[i]).(*coordinatorpb.SubmitEmbeddingRecord) + } pushRequest := logservicepb.PushLogsRequest{ CollectionId: suite.collectionId.String(), Records: recordsToSubmit, } - suite.s.PushLogs(context.Background(), &pushRequest) + _, err := suite.s.PushLogs(context.Background(), &pushRequest) + suite.NoError(err) // pull the records pullRequest := logservicepb.PullLogsRequest{ @@ -140,17 +152,17 @@ func (suite *RecordLogServiceTestSuite) TestServer_PullLogs() { BatchSize: 10, } pullResponse, err := suite.s.PullLogs(context.Background(), &pullRequest) - assert.Nil(suite.t, err) - assert.Len(suite.t, pullResponse.Records, 3) + suite.NoError(err) + suite.Len(pullResponse.Records, 3) for index := range pullResponse.Records { - assert.Equal(suite.t, int64(index+1), pullResponse.Records[index].LogId) - assert.Equal(suite.t, recordsToSubmit[index].Id, pullResponse.Records[index].Record.Id) - assert.Equal(suite.t, recordsToSubmit[index].Operation, pullResponse.Records[index].Record.Operation) - assert.Equal(suite.t, recordsToSubmit[index].CollectionId, pullResponse.Records[index].Record.CollectionId) - assert.Equal(suite.t, recordsToSubmit[index].Metadata, pullResponse.Records[index].Record.Metadata) - assert.Equal(suite.t, recordsToSubmit[index].Vector.Dimension, pullResponse.Records[index].Record.Vector.Dimension) - assert.Equal(suite.t, recordsToSubmit[index].Vector.Encoding, pullResponse.Records[index].Record.Vector.Encoding) - assert.Equal(suite.t, recordsToSubmit[index].Vector.Vector, pullResponse.Records[index].Record.Vector.Vector) + suite.Equal(int64(index+1), pullResponse.Records[index].LogId) + suite.Equal(recordsToSubmit_sot[index].Id, pullResponse.Records[index].Record.Id) + suite.Equal(recordsToSubmit_sot[index].Operation, pullResponse.Records[index].Record.Operation) + suite.Equal(recordsToSubmit_sot[index].CollectionId, pullResponse.Records[index].Record.CollectionId) + suite.Equal(recordsToSubmit_sot[index].Metadata, pullResponse.Records[index].Record.Metadata) + suite.Equal(recordsToSubmit_sot[index].Vector.Dimension, pullResponse.Records[index].Record.Vector.Dimension) + suite.Equal(recordsToSubmit_sot[index].Vector.Encoding, pullResponse.Records[index].Record.Vector.Encoding) + suite.Equal(recordsToSubmit_sot[index].Vector.Vector, pullResponse.Records[index].Record.Vector.Vector) } } @@ -161,13 +173,12 @@ func (suite *RecordLogServiceTestSuite) TestServer_Bad_CollectionId() { CollectionId: "badId", Records: []*coordinatorpb.SubmitEmbeddingRecord{}, } - pushResponse, err := suite.s.PushLogs(context.Background(), &pushRequest) - assert.Nil(suite.t, pushResponse) - assert.NotNil(suite.t, err) + _, err := suite.s.PushLogs(context.Background(), &pushRequest) + suite.Error(err) st, ok := status.FromError(err) - assert.True(suite.t, ok) - assert.Equal(suite.T(), codes.InvalidArgument, st.Code()) - assert.Equal(suite.T(), "invalid collection_id", st.Message()) + suite.True(ok) + suite.Equal(codes.InvalidArgument, st.Code()) + suite.Equal("invalid collection_id", st.Message()) // pull the records // pull the records @@ -176,13 +187,12 @@ func (suite *RecordLogServiceTestSuite) TestServer_Bad_CollectionId() { StartFromId: 0, BatchSize: 10, } - pullResponse, err := suite.s.PullLogs(context.Background(), &pullRequest) - assert.Nil(suite.t, pullResponse) - assert.NotNil(suite.t, err) + _, err = suite.s.PullLogs(context.Background(), &pullRequest) + suite.Error(err) st, ok = status.FromError(err) - assert.True(suite.t, ok) - assert.Equal(suite.T(), codes.InvalidArgument, st.Code()) - assert.Equal(suite.T(), "invalid collection_id", st.Message()) + suite.True(ok) + suite.Equal(codes.InvalidArgument, st.Code()) + suite.Equal("invalid collection_id", st.Message()) } func (suite *RecordLogServiceTestSuite) TestServer_GetAllCollectionInfoToCompact() { @@ -193,17 +203,18 @@ func (suite *RecordLogServiceTestSuite) TestServer_GetAllCollectionInfoToCompact CollectionId: suite.collectionId.String(), Records: recordsToSubmit, } - suite.s.PushLogs(context.Background(), &pushRequest) + _, err := suite.s.PushLogs(context.Background(), &pushRequest) + suite.NoError(err) // get collection info for compactor request := logservicepb.GetAllCollectionInfoToCompactRequest{} response, err := suite.s.GetAllCollectionInfoToCompact(context.Background(), &request) - assert.Nil(suite.t, err) - assert.Len(suite.t, response.AllCollectionInfo, 1) - assert.Equal(suite.T(), suite.collectionId.String(), response.AllCollectionInfo[0].CollectionId) - assert.Equal(suite.T(), int64(1), response.AllCollectionInfo[0].FirstLogId) - assert.True(suite.T(), response.AllCollectionInfo[0].FirstLogIdTs > startTime) - assert.True(suite.T(), response.AllCollectionInfo[0].FirstLogIdTs < time.Now().UnixNano()) + suite.NoError(err) + suite.Len(response.AllCollectionInfo, 1) + suite.Equal(suite.collectionId.String(), response.AllCollectionInfo[0].CollectionId) + suite.Equal(int64(1), response.AllCollectionInfo[0].FirstLogId) + suite.True(response.AllCollectionInfo[0].FirstLogIdTs > startTime) + suite.True(response.AllCollectionInfo[0].FirstLogIdTs < time.Now().UnixNano()) // move log position testutils.MoveLogPosition(suite.db, suite.collectionId, 2) @@ -211,17 +222,15 @@ func (suite *RecordLogServiceTestSuite) TestServer_GetAllCollectionInfoToCompact // get collection info for compactor request = logservicepb.GetAllCollectionInfoToCompactRequest{} response, err = suite.s.GetAllCollectionInfoToCompact(context.Background(), &request) - assert.Nil(suite.t, err) - assert.Len(suite.t, response.AllCollectionInfo, 1) - assert.Equal(suite.T(), suite.collectionId.String(), response.AllCollectionInfo[0].CollectionId) - assert.Equal(suite.T(), int64(3), response.AllCollectionInfo[0].FirstLogId) - assert.True(suite.T(), response.AllCollectionInfo[0].FirstLogIdTs > startTime) - assert.True(suite.T(), response.AllCollectionInfo[0].FirstLogIdTs < time.Now().UnixNano()) - + suite.NoError(err) + suite.Len(response.AllCollectionInfo, 1) + suite.Equal(suite.collectionId.String(), response.AllCollectionInfo[0].CollectionId) + suite.Equal(int64(3), response.AllCollectionInfo[0].FirstLogId) + suite.True(response.AllCollectionInfo[0].FirstLogIdTs > startTime) + suite.True(response.AllCollectionInfo[0].FirstLogIdTs < time.Now().UnixNano()) } func TestRecordLogServiceTestSuite(t *testing.T) { testSuite := new(RecordLogServiceTestSuite) - testSuite.t = t suite.Run(t, testSuite) } diff --git a/go/pkg/logservice/testutils/record_log_test_util.go b/go/pkg/logservice/testutils/record_log_test_util.go index e70f55747fc..a6f7c3d9aa0 100644 --- a/go/pkg/logservice/testutils/record_log_test_util.go +++ b/go/pkg/logservice/testutils/record_log_test_util.go @@ -1,18 +1,13 @@ package testutils import ( - "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/types" - "github.com/pingcap/log" - "go.uber.org/zap" "gorm.io/gorm" "strconv" ) -func SetupTest(db *gorm.DB, collectionIds ...types.UniqueID) { - dbcore.ResetTestTables(db) - +func CreateCollections(db *gorm.DB, collectionIds ...types.UniqueID) error { // create test collections for index, collectionId := range collectionIds { collectionName := "collection" + strconv.Itoa(index+1) @@ -27,18 +22,27 @@ func SetupTest(db *gorm.DB, collectionIds ...types.UniqueID) { } err := db.Create(collection).Error if err != nil { - log.Error("create collection error", zap.Error(err)) + return err } } + return nil } -func TearDownTest(db *gorm.DB) { - db.Migrator().DropTable(&dbmodel.Segment{}) - db.Migrator().CreateTable(&dbmodel.Segment{}) - db.Migrator().DropTable(&dbmodel.Collection{}) - db.Migrator().CreateTable(&dbmodel.Collection{}) - db.Migrator().DropTable(&dbmodel.RecordLog{}) - db.Migrator().CreateTable(&dbmodel.RecordLog{}) +func CleanupCollections(db *gorm.DB, collectionIds ...types.UniqueID) error { + // delete test collections + for _, collectionId := range collectionIds { + err := db.Where("id = ?", collectionId.String()).Delete(&dbmodel.Collection{}).Error + if err != nil { + return err + } + } + + // cleanup logs + err := db.Where("collection_id in ?", collectionIds).Delete(&dbmodel.RecordLog{}).Error + if err != nil { + return err + } + return nil } func MoveLogPosition(db *gorm.DB, collectionId types.UniqueID, position int64) { diff --git a/go/pkg/memberlist_manager/memberlist_manager.go b/go/pkg/memberlist_manager/memberlist_manager.go index fec3e91d1c5..990d97a056b 100644 --- a/go/pkg/memberlist_manager/memberlist_manager.go +++ b/go/pkg/memberlist_manager/memberlist_manager.go @@ -39,6 +39,7 @@ func NewMemberlistManager(nodeWatcher IWatcher, memberlistStore IMemberlistStore } func (m *MemberlistManager) Start() error { + log.Info("Starting memberlist manager") m.nodeWatcher.RegisterCallback(func(nodeIp string) { m.workqueue.Add(nodeIp) }) diff --git a/go/pkg/memberlist_manager/memberlist_store.go b/go/pkg/memberlist_manager/memberlist_store.go index 0567897f46e..49191359d0c 100644 --- a/go/pkg/memberlist_manager/memberlist_store.go +++ b/go/pkg/memberlist_manager/memberlist_store.go @@ -3,6 +3,8 @@ package memberlist_manager import ( "context" + "github.com/pingcap/log" + "go.uber.org/zap" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" "k8s.io/apimachinery/pkg/runtime/schema" @@ -53,6 +55,7 @@ func (s *CRMemberlistStore) GetMemberlist(ctx context.Context) (return_memberlis func (s *CRMemberlistStore) UpdateMemberlist(ctx context.Context, memberlist *Memberlist, resourceVersion string) error { gvr := getGvr() + log.Info("Updating memberlist store", zap.Any("memberlist", memberlist)) unstructured := memberlistToCr(memberlist, s.coordinatorNamespace, s.memberlistCustomResource, resourceVersion) _, err := s.dynamicClient.Resource(gvr).Namespace("chroma").Update(context.TODO(), unstructured, metav1.UpdateOptions{}) if err != nil { diff --git a/go/pkg/memberlist_manager/node_watcher.go b/go/pkg/memberlist_manager/node_watcher.go index cac27f5466e..d3d2a04944b 100644 --- a/go/pkg/memberlist_manager/node_watcher.go +++ b/go/pkg/memberlist_manager/node_watcher.go @@ -47,6 +47,7 @@ type KubernetesWatcher struct { } func NewKubernetesWatcher(clientset kubernetes.Interface, coordinator_namespace string, pod_label string, resyncPeriod time.Duration) *KubernetesWatcher { + log.Info("Creating new kubernetes watcher", zap.String("namespace", coordinator_namespace), zap.String("pod label", pod_label), zap.Duration("resync period", resyncPeriod)) labelSelector := labels.SelectorFromSet(map[string]string{MemberLabel: pod_label}) factory := informers.NewSharedInformerFactoryWithOptions(clientset, resyncPeriod, informers.WithNamespace(coordinator_namespace), informers.WithTweakListOptions(func(options *metav1.ListOptions) { options.LabelSelector = labelSelector.String() })) podInformer := factory.Core().V1().Pods().Informer() @@ -75,6 +76,7 @@ func (w *KubernetesWatcher) Start() error { log.Error("Error while asserting object to pod") } if err == nil { + log.Info("Kubernetes Pod Added", zap.String("key", key), zap.String("ip", objPod.Status.PodIP)) ip := objPod.Status.PodIP w.mu.Lock() w.ipToKey[ip] = key @@ -91,6 +93,7 @@ func (w *KubernetesWatcher) Start() error { log.Error("Error while asserting object to pod") } if err == nil { + log.Info("Kubernetes Pod Updated", zap.String("key", key), zap.String("ip", objPod.Status.PodIP)) ip := objPod.Status.PodIP w.ipToKey[ip] = key w.notify(ip) @@ -105,6 +108,7 @@ func (w *KubernetesWatcher) Start() error { log.Error("Error while asserting object to pod") } if err == nil { + log.Info("Kubernetes Pod Deleted", zap.String("ip", objPod.Status.PodIP)) ip := objPod.Status.PodIP // The contract for GetStatus is that if the ip is not in this map, then it returns NotReady delete(w.ipToKey, ip) diff --git a/go/pkg/metastore/catalog.go b/go/pkg/metastore/catalog.go index 52d6ac2ca35..15f73bc0d1f 100644 --- a/go/pkg/metastore/catalog.go +++ b/go/pkg/metastore/catalog.go @@ -29,4 +29,5 @@ type Catalog interface { GetAllTenants(ctx context.Context, ts types.Timestamp) ([]*model.Tenant, error) SetTenantLastCompactionTime(ctx context.Context, tenantID string, lastCompactionTime int64) error GetTenantsLastCompactionTime(ctx context.Context, tenantIDs []string) ([]*dbmodel.Tenant, error) + FlushCollectionCompaction(ctx context.Context, flushCollectionCompaction *model.FlushCollectionCompaction) (*model.FlushCollectionInfo, error) } diff --git a/go/pkg/metastore/coordinator/model_db_convert.go b/go/pkg/metastore/coordinator/model_db_convert.go index 2f164be253d..717b713cf19 100644 --- a/go/pkg/metastore/coordinator/model_db_convert.go +++ b/go/pkg/metastore/coordinator/model_db_convert.go @@ -22,6 +22,8 @@ func convertCollectionToModel(collectionAndMetadataList []*dbmodel.CollectionAnd TenantID: collectionAndMetadata.TenantID, DatabaseName: collectionAndMetadata.DatabaseName, Ts: collectionAndMetadata.Collection.Ts, + LogPosition: collectionAndMetadata.Collection.LogPosition, + Version: collectionAndMetadata.Collection.Version, } collection.Metadata = convertCollectionMetadataToModel(collectionAndMetadata.CollectionMetadata) collections = append(collections, collection) diff --git a/go/pkg/metastore/coordinator/table_catalog.go b/go/pkg/metastore/coordinator/table_catalog.go index bed31c51532..e1ae1e53d5c 100644 --- a/go/pkg/metastore/coordinator/table_catalog.go +++ b/go/pkg/metastore/coordinator/table_catalog.go @@ -316,6 +316,7 @@ func (tc *Catalog) GetCollections(ctx context.Context, collectionID types.Unique } func (tc *Catalog) DeleteCollection(ctx context.Context, deleteCollection *model.DeleteCollection) error { + log.Info("deleting collection", zap.Any("deleteCollection", deleteCollection)) return tc.txImpl.Transaction(ctx, func(txCtx context.Context) error { collectionID := deleteCollection.ID collectionAndMetadata, err := tc.metaDomain.CollectionDb(txCtx).GetCollections(types.FromUniqueID(collectionID), nil, nil, deleteCollection.TenantID, deleteCollection.DatabaseName) @@ -351,6 +352,7 @@ func (tc *Catalog) DeleteCollection(ctx context.Context, deleteCollection *model } func (tc *Catalog) UpdateCollection(ctx context.Context, updateCollection *model.UpdateCollection, ts types.Timestamp) (*model.Collection, error) { + log.Info("updating collection", zap.String("collectionId", updateCollection.ID.String())) var result *model.Collection err := tc.txImpl.Transaction(ctx, func(txCtx context.Context) error { @@ -411,7 +413,7 @@ func (tc *Catalog) UpdateCollection(ctx context.Context, updateCollection *model if err != nil { return nil, err } - log.Info("collection updated", zap.Any("collection", result)) + log.Info("collection updated", zap.String("collectionID", result.ID.String())) return result, nil } @@ -473,11 +475,12 @@ func (tc *Catalog) GetSegments(ctx context.Context, segmentID types.UniqueID, se segments := make([]*model.Segment, 0, len(segmentAndMetadataList)) for _, segmentAndMetadata := range segmentAndMetadataList { segment := &model.Segment{ - ID: types.MustParse(segmentAndMetadata.Segment.ID), - Type: segmentAndMetadata.Segment.Type, - Scope: segmentAndMetadata.Segment.Scope, - Topic: segmentAndMetadata.Segment.Topic, - Ts: segmentAndMetadata.Segment.Ts, + ID: types.MustParse(segmentAndMetadata.Segment.ID), + Type: segmentAndMetadata.Segment.Type, + Scope: segmentAndMetadata.Segment.Scope, + Topic: segmentAndMetadata.Segment.Topic, + Ts: segmentAndMetadata.Segment.Ts, + FilePaths: segmentAndMetadata.Segment.FilePaths, } if segmentAndMetadata.Segment.CollectionID != nil { @@ -614,3 +617,41 @@ func (tc *Catalog) GetTenantsLastCompactionTime(ctx context.Context, tenantIDs [ tenants, err := tc.metaDomain.TenantDb(ctx).GetTenantsLastCompactionTime(tenantIDs) return tenants, err } + +func (tc *Catalog) FlushCollectionCompaction(ctx context.Context, flushCollectionCompaction *model.FlushCollectionCompaction) (*model.FlushCollectionInfo, error) { + flushCollectionInfo := &model.FlushCollectionInfo{ + ID: flushCollectionCompaction.ID.String(), + } + + err := tc.txImpl.Transaction(ctx, func(txCtx context.Context) error { + // register files to Segment metadata + err := tc.metaDomain.SegmentDb(txCtx).RegisterFilePaths(flushCollectionCompaction.FlushSegmentCompactions) + if err != nil { + return err + } + + // update collection log position and version + collectionVersion, err := tc.metaDomain.CollectionDb(txCtx).UpdateLogPositionAndVersion(flushCollectionCompaction.ID.String(), flushCollectionCompaction.LogPosition, flushCollectionCompaction.CurrentCollectionVersion) + if err != nil { + return err + } + flushCollectionInfo.CollectionVersion = collectionVersion + + // update tenant last compaction time + // TODO: add a system configuration to disable + // since this might cause resource contention if one tenant has a lot of collection compactions at the same time + lastCompactionTime := time.Now().Unix() + err = tc.metaDomain.TenantDb(txCtx).UpdateTenantLastCompactionTime(flushCollectionCompaction.TenantID, lastCompactionTime) + if err != nil { + return err + } + flushCollectionInfo.TenantLastCompactionTime = lastCompactionTime + + // return nil will commit the transaction + return nil + }) + if err != nil { + return nil, err + } + return flushCollectionInfo, nil +} diff --git a/go/pkg/metastore/db/dao/collection.go b/go/pkg/metastore/db/dao/collection.go index 295046f42f0..f2f381b6b0d 100644 --- a/go/pkg/metastore/db/dao/collection.go +++ b/go/pkg/metastore/db/dao/collection.go @@ -6,6 +6,7 @@ import ( "github.com/chroma-core/chroma/go/pkg/common" "github.com/jackc/pgx/v5/pgconn" "gorm.io/gorm/clause" + "strings" "go.uber.org/zap" "gorm.io/gorm" @@ -25,31 +26,40 @@ func (s *collectionDb) DeleteAll() error { } func (s *collectionDb) GetCollections(id *string, name *string, topic *string, tenantID string, databaseName string) ([]*dbmodel.CollectionAndMetadata, error) { + var getCollectionInput strings.Builder + getCollectionInput.WriteString("GetCollections input: ") + var collections []*dbmodel.CollectionAndMetadata query := s.db.Table("collections"). - Select("collections.id, collections.name, collections.topic, collections.dimension, collections.database_id, databases.name, databases.tenant_id, collection_metadata.key, collection_metadata.str_value, collection_metadata.int_value, collection_metadata.float_value"). + Select("collections.id, collections.log_position, collections.version, collections.name, collections.topic, collections.dimension, collections.database_id, databases.name, databases.tenant_id, collection_metadata.key, collection_metadata.str_value, collection_metadata.int_value, collection_metadata.float_value"). Joins("LEFT JOIN collection_metadata ON collections.id = collection_metadata.collection_id"). Joins("INNER JOIN databases ON collections.database_id = databases.id"). Order("collections.id") if databaseName != "" { query = query.Where("databases.name = ?", databaseName) + getCollectionInput.WriteString("databases.name: " + databaseName + ", ") } if tenantID != "" { query = query.Where("databases.tenant_id = ?", tenantID) + getCollectionInput.WriteString("databases.tenant_id: " + tenantID + ", ") } if id != nil { query = query.Where("collections.id = ?", *id) + getCollectionInput.WriteString("collections.id: " + *id + ", ") } if topic != nil { query = query.Where("collections.topic = ?", *topic) + getCollectionInput.WriteString("collections.topic: " + *topic + ", ") } if name != nil { query = query.Where("collections.name = ?", *name) + getCollectionInput.WriteString("collections.name: " + *name + ", ") } + log.Info(getCollectionInput.String()) rows, err := query.Rows() if err != nil { @@ -64,6 +74,8 @@ func (s *collectionDb) GetCollections(id *string, name *string, topic *string, t for rows.Next() { var ( collectionID string + logPosition int64 + version int32 collectionName string collectionTopic string collectionDimension sql.NullInt32 @@ -76,7 +88,7 @@ func (s *collectionDb) GetCollections(id *string, name *string, topic *string, t floatValue sql.NullFloat64 ) - err := rows.Scan(&collectionID, &collectionName, &collectionTopic, &collectionDimension, &collectionDatabaseID, &databaseName, &databaseTenantID, &key, &strValue, &intValue, &floatValue) + err := rows.Scan(&collectionID, &logPosition, &version, &collectionName, &collectionTopic, &collectionDimension, &collectionDatabaseID, &databaseName, &databaseTenantID, &key, &strValue, &intValue, &floatValue) if err != nil { log.Error("scan collection failed", zap.Error(err)) return nil, err @@ -87,10 +99,12 @@ func (s *collectionDb) GetCollections(id *string, name *string, topic *string, t currentCollection = &dbmodel.CollectionAndMetadata{ Collection: &dbmodel.Collection{ - ID: collectionID, - Name: &collectionName, - Topic: &collectionTopic, - DatabaseID: collectionDatabaseID, + ID: collectionID, + Name: &collectionName, + Topic: &collectionTopic, + DatabaseID: collectionDatabaseID, + LogPosition: logPosition, + Version: version, }, CollectionMetadata: metadata, TenantID: databaseTenantID, @@ -182,6 +196,33 @@ func generateCollectionUpdatesWithoutID(in *dbmodel.Collection) map[string]inter } func (s *collectionDb) Update(in *dbmodel.Collection) error { + log.Info("update collection", zap.Any("collection", in)) updates := generateCollectionUpdatesWithoutID(in) return s.db.Model(&dbmodel.Collection{}).Where("id = ?", in.ID).Updates(updates).Error } + +func (s *collectionDb) UpdateLogPositionAndVersion(collectionID string, logPosition int64, currentCollectionVersion int32) (int32, error) { + log.Info("update log position and version", zap.String("collectionID", collectionID), zap.Int64("logPosition", logPosition), zap.Int32("currentCollectionVersion", currentCollectionVersion)) + var collection dbmodel.Collection + err := s.db.Where("id = ?", collectionID).First(&collection).Error + if err != nil { + return 0, err + } + if collection.LogPosition > logPosition { + return 0, common.ErrCollectionLogPositionStale + } + if collection.Version > currentCollectionVersion { + return 0, common.ErrCollectionVersionStale + } + if collection.Version < currentCollectionVersion { + // this should not happen, potentially a bug + return 0, common.ErrCollectionVersionInvalid + } + + version := currentCollectionVersion + 1 + err = s.db.Model(&dbmodel.Collection{}).Where("id = ?", collectionID).Updates(map[string]interface{}{"log_position": logPosition, "version": version}).Error + if err != nil { + return 0, err + } + return version, nil +} diff --git a/go/pkg/metastore/db/dao/collection_test.go b/go/pkg/metastore/db/dao/collection_test.go index aa40eabf53a..8e86a6203b5 100644 --- a/go/pkg/metastore/db/dao/collection_test.go +++ b/go/pkg/metastore/db/dao/collection_test.go @@ -3,84 +3,137 @@ package dao import ( "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/pingcap/log" - "go.uber.org/zap" + "github.com/stretchr/testify/suite" "testing" - "github.com/chroma-core/chroma/go/pkg/common" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" - "github.com/chroma-core/chroma/go/pkg/types" - "github.com/stretchr/testify/assert" - "gorm.io/driver/sqlite" "gorm.io/gorm" ) -func TestCollectionDb_GetCollections(t *testing.T) { - db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{}) - assert.NoError(t, err) - - err = db.AutoMigrate(&dbmodel.Tenant{}, &dbmodel.Database{}, &dbmodel.Collection{}, &dbmodel.CollectionMetadata{}) - databaseID := dbcore.CreateDefaultTenantAndDatabase(db) - - assert.NoError(t, err) - name := "test_name" - topic := "test_topic" - collection := &dbmodel.Collection{ - ID: types.NewUniqueID().String(), - Name: &name, - Topic: &topic, - DatabaseID: databaseID, +type CollectionDbTestSuite struct { + suite.Suite + db *gorm.DB + collectionDb *collectionDb + tenantName string + databaseName string + databaseId string +} + +func (suite *CollectionDbTestSuite) SetupSuite() { + log.Info("setup suite") + suite.db = dbcore.ConfigDatabaseForTesting() + suite.collectionDb = &collectionDb{ + db: suite.db, } - err = db.Create(collection).Error - assert.NoError(t, err) + suite.tenantName = "test_collection_tenant" + suite.databaseName = "test_collection_database" + DbId, err := CreateTestTenantAndDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.databaseId = DbId +} + +func (suite *CollectionDbTestSuite) TearDownSuite() { + log.Info("teardown suite") + err := CleanUpTestDatabase(suite.db, suite.tenantName, suite.databaseName) + suite.NoError(err) + err = CleanUpTestTenant(suite.db, suite.tenantName) + suite.NoError(err) +} + +func (suite *CollectionDbTestSuite) TestCollectionDb_GetCollections() { + collectionName := "test_collection_get_collections" + collectionTopic := "test_collection_topic" + collectionID, err := CreateTestCollection(suite.db, collectionName, collectionTopic, 128, suite.databaseId) + suite.NoError(err) testKey := "test" testValue := "test" metadata := &dbmodel.CollectionMetadata{ - CollectionID: collection.ID, + CollectionID: collectionID, Key: &testKey, StrValue: &testValue, } - err = db.Create(metadata).Error - assert.NoError(t, err) - - collectionDb := &collectionDb{ - db: db, - } + err = suite.db.Create(metadata).Error + suite.NoError(err) - query := db.Table("collections").Select("collections.id") + query := suite.db.Table("collections").Select("collections.id").Where("collections.id = ?", collectionID) rows, err := query.Rows() - assert.NoError(t, err) + suite.NoError(err) for rows.Next() { - var collectionID string - err = rows.Scan(&collectionID) - assert.NoError(t, err) - log.Info("collectionID", zap.String("collectionID", collectionID)) + var scanedCollectionID string + err = rows.Scan(&scanedCollectionID) + suite.NoError(err) + suite.Equal(collectionID, scanedCollectionID) } - collections, err := collectionDb.GetCollections(nil, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(t, err) - assert.Len(t, collections, 1) - assert.Equal(t, collection.ID, collections[0].Collection.ID) - assert.Equal(t, collection.Name, collections[0].Collection.Name) - assert.Equal(t, collection.Topic, collections[0].Collection.Topic) - assert.Len(t, collections[0].CollectionMetadata, 1) - assert.Equal(t, metadata.Key, collections[0].CollectionMetadata[0].Key) - assert.Equal(t, metadata.StrValue, collections[0].CollectionMetadata[0].StrValue) + collections, err := suite.collectionDb.GetCollections(nil, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Len(collections, 1) + suite.Equal(collectionID, collections[0].Collection.ID) + suite.Equal(collectionName, *collections[0].Collection.Name) + suite.Equal(collectionTopic, *collections[0].Collection.Topic) + suite.Len(collections[0].CollectionMetadata, 1) + suite.Equal(metadata.Key, collections[0].CollectionMetadata[0].Key) + suite.Equal(metadata.StrValue, collections[0].CollectionMetadata[0].StrValue) // Test when filtering by ID - collections, err = collectionDb.GetCollections(nil, nil, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(t, err) - assert.Len(t, collections, 1) - assert.Equal(t, collection.ID, collections[0].Collection.ID) + collections, err = suite.collectionDb.GetCollections(nil, nil, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Len(collections, 1) + suite.Equal(collectionID, collections[0].Collection.ID) // Test when filtering by name - collections, err = collectionDb.GetCollections(nil, collection.Name, nil, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(t, err) - assert.Len(t, collections, 1) - assert.Equal(t, collection.ID, collections[0].Collection.ID) + collections, err = suite.collectionDb.GetCollections(nil, &collectionName, nil, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Len(collections, 1) + suite.Equal(collectionID, collections[0].Collection.ID) // Test when filtering by topic - collections, err = collectionDb.GetCollections(nil, nil, collection.Topic, common.DefaultTenant, common.DefaultDatabase) - assert.NoError(t, err) - assert.Len(t, collections, 1) - assert.Equal(t, collection.ID, collections[0].Collection.ID) + collections, err = suite.collectionDb.GetCollections(nil, nil, &collectionTopic, suite.tenantName, suite.databaseName) + suite.NoError(err) + suite.Len(collections, 1) + suite.Equal(collectionID, collections[0].Collection.ID) + + // clean up + err = CleanUpTestCollection(suite.db, collectionID) + suite.NoError(err) +} + +func (suite *CollectionDbTestSuite) TestCollectionDb_UpdateLogPositionAndVersion() { + collectionName := "test_collection_get_collections" + collectionTopic := "test_topic" + collectionID, err := CreateTestCollection(suite.db, collectionName, collectionTopic, 128, suite.databaseId) + // verify default values + collections, err := suite.collectionDb.GetCollections(&collectionID, nil, nil, "", "") + suite.NoError(err) + suite.Len(collections, 1) + suite.Equal(int64(0), collections[0].Collection.LogPosition) + suite.Equal(int32(0), collections[0].Collection.Version) + + // update log position and version + version, err := suite.collectionDb.UpdateLogPositionAndVersion(collectionID, int64(10), 0) + suite.NoError(err) + suite.Equal(int32(1), version) + collections, err = suite.collectionDb.GetCollections(&collectionID, nil, nil, "", "") + suite.Len(collections, 1) + suite.Equal(int64(10), collections[0].Collection.LogPosition) + suite.Equal(int32(1), collections[0].Collection.Version) + + // invalid log position + _, err = suite.collectionDb.UpdateLogPositionAndVersion(collectionID, int64(5), 0) + suite.Error(err, "collection log position Stale") + + // invalid version + _, err = suite.collectionDb.UpdateLogPositionAndVersion(collectionID, int64(20), 0) + suite.Error(err, "collection version invalid") + _, err = suite.collectionDb.UpdateLogPositionAndVersion(collectionID, int64(20), 3) + suite.Error(err, "collection version invalid") + + //clean up + err = CleanUpTestCollection(suite.db, collectionID) + suite.NoError(err) +} + +func TestCollectionDbTestSuiteSuite(t *testing.T) { + testSuite := new(CollectionDbTestSuite) + suite.Run(t, testSuite) } diff --git a/go/pkg/metastore/db/dao/database.go b/go/pkg/metastore/db/dao/database.go index 7ede1c5bc4f..fb7ffb07a12 100644 --- a/go/pkg/metastore/db/dao/database.go +++ b/go/pkg/metastore/db/dao/database.go @@ -5,6 +5,7 @@ import ( "github.com/pingcap/log" "go.uber.org/zap" "gorm.io/gorm" + "gorm.io/gorm/clause" ) type databaseDb struct { @@ -17,6 +18,12 @@ func (s *databaseDb) DeleteAll() error { return s.db.Where("1 = 1").Delete(&dbmodel.Database{}).Error } +func (s *databaseDb) DeleteByTenantIdAndName(tenantId string, databaseName string) (int, error) { + var databases []dbmodel.Database + err := s.db.Clauses(clause.Returning{}).Where("tenant_id = ?", tenantId).Where("name = ?", databaseName).Delete(&databases).Error + return len(databases), err +} + func (s *databaseDb) GetAllDatabases() ([]*dbmodel.Database, error) { var databases []*dbmodel.Database query := s.db.Table("databases") @@ -44,3 +51,16 @@ func (s *databaseDb) GetDatabases(tenantID string, databaseName string) ([]*dbmo func (s *databaseDb) Insert(database *dbmodel.Database) error { return s.db.Create(database).Error } + +func (s *databaseDb) GetDatabasesByTenantID(tenantID string) ([]*dbmodel.Database, error) { + var databases []*dbmodel.Database + query := s.db.Table("databases"). + Select("databases.id, databases.name, databases.tenant_id"). + Where("databases.tenant_id = ?", tenantID) + + if err := query.Find(&databases).Error; err != nil { + log.Error("GetDatabasesByTenantID", zap.Error(err)) + return nil, err + } + return databases, nil +} diff --git a/go/pkg/metastore/db/dao/record_log.go b/go/pkg/metastore/db/dao/record_log.go index c7a15697df1..aa0c102929c 100644 --- a/go/pkg/metastore/db/dao/record_log.go +++ b/go/pkg/metastore/db/dao/record_log.go @@ -3,18 +3,21 @@ package dao import ( "database/sql" "errors" + "time" + "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/types" "github.com/pingcap/log" "go.uber.org/zap" "gorm.io/gorm" - "time" ) type recordLogDb struct { db *gorm.DB } +var _ dbmodel.IRecordLogDb = &recordLogDb{} + func (s *recordLogDb) PushLogs(collectionID types.UniqueID, recordsContent [][]byte) (int, error) { err := s.db.Transaction(func(tx *gorm.DB) error { var timestamp = time.Now().UnixNano() @@ -58,18 +61,27 @@ func (s *recordLogDb) PushLogs(collectionID types.UniqueID, recordsContent [][]b return len(recordsContent), nil } -func (s *recordLogDb) PullLogs(collectionID types.UniqueID, id int64, batchSize int) ([]*dbmodel.RecordLog, error) { +func (s *recordLogDb) PullLogs(collectionID types.UniqueID, id int64, batchSize int, endTimestamp int64) ([]*dbmodel.RecordLog, error) { var collectionIDStr = types.FromUniqueID(collectionID) log.Info("PullLogs", zap.String("collectionID", *collectionIDStr), zap.Int64("ID", id), - zap.Int("batch_size", batchSize)) + zap.Int("batch_size", batchSize), + zap.Int64("endTimestamp", endTimestamp)) var recordLogs []*dbmodel.RecordLog - result := s.db.Where("collection_id = ? AND id >= ?", collectionIDStr, id).Order("id").Limit(batchSize).Find(&recordLogs) - if result.Error != nil && !errors.Is(result.Error, gorm.ErrRecordNotFound) { - log.Error("PullLogs error", zap.Error(result.Error)) - return nil, result.Error + if endTimestamp > 0 { + result := s.db.Where("collection_id = ? AND id >= ? AND timestamp <= ?", collectionIDStr, id, endTimestamp).Order("id").Limit(batchSize).Find(&recordLogs) + if result.Error != nil && !errors.Is(result.Error, gorm.ErrRecordNotFound) { + log.Error("PullLogs error", zap.Error(result.Error)) + return nil, result.Error + } + } else { + result := s.db.Where("collection_id = ? AND id >= ?", collectionIDStr, id).Order("id").Limit(batchSize).Find(&recordLogs) + if result.Error != nil && !errors.Is(result.Error, gorm.ErrRecordNotFound) { + log.Error("PullLogs error", zap.Error(result.Error)) + return nil, result.Error + } } log.Info("PullLogs", zap.String("collectionID", *collectionIDStr), diff --git a/go/pkg/metastore/db/dao/record_log_test.go b/go/pkg/metastore/db/dao/record_log_test.go index cb1a3ac6a0d..b28f904a6d8 100644 --- a/go/pkg/metastore/db/dao/record_log_test.go +++ b/go/pkg/metastore/db/dao/record_log_test.go @@ -1,22 +1,22 @@ package dao import ( + "testing" + "time" + "github.com/chroma-core/chroma/go/pkg/logservice/testutils" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/types" "github.com/pingcap/log" - "github.com/stretchr/testify/assert" "github.com/stretchr/testify/suite" "gorm.io/gorm" - "testing" ) type RecordLogDbTestSuite struct { suite.Suite db *gorm.DB Db *recordLogDb - t *testing.T collectionId1 types.UniqueID collectionId2 types.UniqueID records [][]byte @@ -28,21 +28,28 @@ func (suite *RecordLogDbTestSuite) SetupSuite() { suite.Db = &recordLogDb{ db: suite.db, } - suite.collectionId1 = types.NewUniqueID() - suite.collectionId2 = types.NewUniqueID() suite.records = make([][]byte, 0, 5) suite.records = append(suite.records, []byte("test1"), []byte("test2"), []byte("test3"), []byte("test4"), []byte("test5")) + recordLogTableExist := suite.db.Migrator().HasTable(&dbmodel.RecordLog{}) + if !recordLogTableExist { + err := suite.db.Migrator().CreateTable(&dbmodel.RecordLog{}) + suite.NoError(err) + } } func (suite *RecordLogDbTestSuite) SetupTest() { log.Info("setup test") - testutils.SetupTest(suite.db, suite.collectionId1, suite.collectionId2) + suite.collectionId1 = types.NewUniqueID() + suite.collectionId2 = types.NewUniqueID() + err := testutils.CreateCollections(suite.db, suite.collectionId1, suite.collectionId2) + suite.NoError(err) } func (suite *RecordLogDbTestSuite) TearDownTest() { log.Info("teardown test") - testutils.TearDownTest(suite.db) + err := testutils.CleanupCollections(suite.db, suite.collectionId1, suite.collectionId2) + suite.NoError(err) } func (suite *RecordLogDbTestSuite) TestRecordLogDb_PushLogs() { @@ -50,133 +57,142 @@ func (suite *RecordLogDbTestSuite) TestRecordLogDb_PushLogs() { // id: 0, // records: test1, test2, test3 count, err := suite.Db.PushLogs(suite.collectionId1, suite.records[:3]) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 3, count) + suite.NoError(err) + suite.Equal(3, count) // verify logs are pushed var recordLogs []*dbmodel.RecordLog suite.db.Where("collection_id = ?", types.FromUniqueID(suite.collectionId1)).Find(&recordLogs) - assert.Len(suite.t, recordLogs, 3) + suite.Len(recordLogs, 3) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID) - assert.Equal(suite.t, suite.records[index], *recordLogs[index].Record) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.records[index], *recordLogs[index].Record) } // run push logs in transaction // id: 1, // records: test4, test5 count, err = suite.Db.PushLogs(suite.collectionId1, suite.records[3:]) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 2, count) + suite.NoError(err) + suite.Equal(2, count) // verify logs are pushed suite.db.Where("collection_id = ?", types.FromUniqueID(suite.collectionId1)).Find(&recordLogs) - assert.Len(suite.t, recordLogs, 5) + suite.Len(recordLogs, 5) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID, "id mismatch for index %d", index) - assert.Equal(suite.t, suite.records[index], *recordLogs[index].Record, "record mismatch for index %d", index) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.records[index], *recordLogs[index].Record) } // run push logs in transaction // id: 0, // records: test1, test2, test3, test4, test5 count, err = suite.Db.PushLogs(suite.collectionId2, suite.records) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 5, count) + suite.NoError(err) + suite.Equal(5, count) // verify logs are pushed suite.db.Where("collection_id = ?", types.FromUniqueID(suite.collectionId2)).Find(&recordLogs) - assert.Len(suite.t, recordLogs, 5) + suite.Len(recordLogs, 5) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID, "id mismatch for index %d", index) - assert.Equal(suite.t, suite.records[index], *recordLogs[index].Record, "record mismatch for index %d", index) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.records[index], *recordLogs[index].Record) } } func (suite *RecordLogDbTestSuite) TestRecordLogDb_PullLogsFromID() { // pull empty logs var recordLogs []*dbmodel.RecordLog - recordLogs, err := suite.Db.PullLogs(suite.collectionId1, 0, 3) - assert.NoError(suite.t, err) - assert.Len(suite.t, recordLogs, 0) + invalidEndTimestamp := int64(-1) + recordLogs, err := suite.Db.PullLogs(suite.collectionId1, 0, 3, invalidEndTimestamp) + suite.NoError(err) + suite.Len(recordLogs, 0) // push some logs count, err := suite.Db.PushLogs(suite.collectionId1, suite.records[:3]) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 3, count) + suite.NoError(err) + suite.Equal(3, count) count, err = suite.Db.PushLogs(suite.collectionId1, suite.records[3:]) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 2, count) + suite.NoError(err) + suite.Equal(2, count) // pull logs from id 0 batch_size 3 - recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 0, 3) - assert.NoError(suite.t, err) - assert.Len(suite.t, recordLogs, 3) + recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 0, 3, invalidEndTimestamp) + suite.NoError(err) + suite.Len(recordLogs, 3) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID, "id mismatch for index %d", index) - assert.Equal(suite.t, suite.records[index], *recordLogs[index].Record, "record mismatch for index %d", index) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.records[index], *recordLogs[index].Record) } // pull logs from id 0 batch_size 6 - recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 0, 6) - assert.NoError(suite.t, err) - assert.Len(suite.t, recordLogs, 5) + recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 0, 6, invalidEndTimestamp) + suite.NoError(err) + suite.Len(recordLogs, 5) for index := range recordLogs { - assert.Equal(suite.t, int64(index+1), recordLogs[index].ID, "id mismatch for index %d", index) - assert.Equal(suite.t, suite.records[index], *recordLogs[index].Record, "record mismatch for index %d", index) + suite.Equal(int64(index+1), recordLogs[index].ID) + suite.Equal(suite.records[index], *recordLogs[index].Record) } // pull logs from id 3 batch_size 4 - recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 3, 4) - assert.NoError(suite.t, err) - assert.Len(suite.t, recordLogs, 3) + recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 3, 4, invalidEndTimestamp) + suite.NoError(err) + suite.Len(recordLogs, 3) + for index := range recordLogs { + suite.Equal(int64(index+3), recordLogs[index].ID) + suite.Equal(suite.records[index+2], *recordLogs[index].Record) + } + + // pull logs from id 3 batch_size 4 endTimestamp Now + recordLogs, err = suite.Db.PullLogs(suite.collectionId1, 3, 4, time.Now().UnixNano()) + suite.NoError(err) + suite.Len(recordLogs, 3) for index := range recordLogs { - assert.Equal(suite.t, int64(index+3), recordLogs[index].ID, "id mismatch for index %d", index) - assert.Equal(suite.t, suite.records[index+2], *recordLogs[index].Record, "record mismatch for index %d", index) + suite.Equal(int64(index+3), recordLogs[index].ID) + suite.Equal(suite.records[index+2], *recordLogs[index].Record) } } func (suite *RecordLogDbTestSuite) TestRecordLogDb_GetAllCollectionsToCompact() { // push some logs count, err := suite.Db.PushLogs(suite.collectionId1, suite.records) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 5, count) + suite.NoError(err) + suite.Equal(5, count) // get all collection ids to compact collectionInfos, err := suite.Db.GetAllCollectionsToCompact() - assert.NoError(suite.t, err) - assert.Len(suite.t, collectionInfos, 1) - assert.Equal(suite.t, suite.collectionId1.String(), *collectionInfos[0].CollectionID) - assert.Equal(suite.t, int64(1), collectionInfos[0].ID) + suite.NoError(err) + suite.Len(collectionInfos, 1) + suite.Equal(suite.collectionId1.String(), *collectionInfos[0].CollectionID) + suite.Equal(int64(1), collectionInfos[0].ID) // move log position testutils.MoveLogPosition(suite.db, suite.collectionId1, 2) // get all collection ids to compact collectionInfos, err = suite.Db.GetAllCollectionsToCompact() - assert.NoError(suite.t, err) - assert.Len(suite.t, collectionInfos, 1) - assert.Equal(suite.t, suite.collectionId1.String(), *collectionInfos[0].CollectionID) - assert.Equal(suite.t, int64(3), collectionInfos[0].ID) + suite.NoError(err) + suite.Len(collectionInfos, 1) + suite.Equal(suite.collectionId1.String(), *collectionInfos[0].CollectionID) + suite.Equal(int64(3), collectionInfos[0].ID) // push some logs count, err = suite.Db.PushLogs(suite.collectionId2, suite.records) - assert.NoError(suite.t, err) - assert.Equal(suite.t, 5, count) + suite.NoError(err) + suite.Equal(5, count) // get all collection ids to compact collectionInfos, err = suite.Db.GetAllCollectionsToCompact() - assert.NoError(suite.t, err) - assert.Len(suite.t, collectionInfos, 2) - assert.Equal(suite.t, suite.collectionId1.String(), *collectionInfos[0].CollectionID) - assert.Equal(suite.t, int64(3), collectionInfos[0].ID) - assert.Equal(suite.t, suite.collectionId2.String(), *collectionInfos[1].CollectionID) - assert.Equal(suite.t, int64(1), collectionInfos[1].ID) + suite.NoError(err) + suite.Len(collectionInfos, 2) + suite.Equal(suite.collectionId1.String(), *collectionInfos[0].CollectionID) + suite.Equal(int64(3), collectionInfos[0].ID) + suite.Equal(suite.collectionId2.String(), *collectionInfos[1].CollectionID) + suite.Equal(int64(1), collectionInfos[1].ID) } func TestRecordLogDbTestSuite(t *testing.T) { testSuite := new(RecordLogDbTestSuite) - testSuite.t = t suite.Run(t, testSuite) } diff --git a/go/pkg/metastore/db/dao/segment.go b/go/pkg/metastore/db/dao/segment.go index 57701aa8066..a69cd13ce6a 100644 --- a/go/pkg/metastore/db/dao/segment.go +++ b/go/pkg/metastore/db/dao/segment.go @@ -2,8 +2,10 @@ package dao import ( "database/sql" + "encoding/json" "errors" "github.com/chroma-core/chroma/go/pkg/common" + "github.com/chroma-core/chroma/go/pkg/model" "github.com/jackc/pgx/v5/pgconn" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" @@ -53,7 +55,7 @@ func (s *segmentDb) GetSegments(id types.UniqueID, segmentType *string, scope *s var segments []*dbmodel.SegmentAndMetadata query := s.db.Table("segments"). - Select("segments.id, segments.collection_id, segments.type, segments.scope, segments.topic, segment_metadata.key, segment_metadata.str_value, segment_metadata.int_value, segment_metadata.float_value"). + Select("segments.id, segments.collection_id, segments.type, segments.scope, segments.topic, segments.file_paths, segment_metadata.key, segment_metadata.str_value, segment_metadata.int_value, segment_metadata.float_value"). Joins("LEFT JOIN segment_metadata ON segments.id = segment_metadata.segment_id"). Order("segments.id") @@ -86,18 +88,19 @@ func (s *segmentDb) GetSegments(id types.UniqueID, segmentType *string, scope *s for rows.Next() { var ( - segmentID string - collectionID sql.NullString - segmentType string - scope string - topic sql.NullString - key sql.NullString - strValue sql.NullString - intValue sql.NullInt64 - floatValue sql.NullFloat64 + segmentID string + collectionID sql.NullString + segmentType string + scope string + topic sql.NullString + filePathsJson string + key sql.NullString + strValue sql.NullString + intValue sql.NullInt64 + floatValue sql.NullFloat64 ) - err := rows.Scan(&segmentID, &collectionID, &segmentType, &scope, &topic, &key, &strValue, &intValue, &floatValue) + err := rows.Scan(&segmentID, &collectionID, &segmentType, &scope, &topic, &filePathsJson, &key, &strValue, &intValue, &floatValue) if err != nil { log.Error("scan segment failed", zap.Error(err)) } @@ -105,11 +108,17 @@ func (s *segmentDb) GetSegments(id types.UniqueID, segmentType *string, scope *s currentSegmentID = segmentID metadata = nil + var filePaths map[string][]string + err := json.Unmarshal([]byte(filePathsJson), &filePaths) + if err != nil { + return nil, err + } currentSegment = &dbmodel.SegmentAndMetadata{ Segment: &dbmodel.Segment{ - ID: segmentID, - Type: segmentType, - Scope: scope, + ID: segmentID, + Type: segmentType, + Scope: scope, + FilePaths: filePaths, }, SegmentMetadata: metadata, } @@ -201,3 +210,22 @@ func (s *segmentDb) Update(in *dbmodel.UpdateSegment) error { Where("collection_id = ?", &in.Collection). Where("id = ?", in.ID).Updates(updates).Error } + +func (s *segmentDb) RegisterFilePaths(flushSegmentCompactions []*model.FlushSegmentCompaction) error { + log.Info("register file paths", zap.Any("flushSegmentCompactions", flushSegmentCompactions)) + for _, flushSegmentCompaction := range flushSegmentCompactions { + filePaths, err := json.Marshal(flushSegmentCompaction.FilePaths) + if err != nil { + log.Error("marshal file paths failed", zap.Error(err)) + return err + } + err = s.db.Model(&dbmodel.Segment{}). + Where("id = ?", flushSegmentCompaction.ID). + Update("file_paths", filePaths).Error + if err != nil { + log.Error("register file path failed", zap.Error(err)) + return err + } + } + return nil +} diff --git a/go/pkg/metastore/db/dao/segment_test.go b/go/pkg/metastore/db/dao/segment_test.go index 3eb527b1da7..7712ccf0bed 100644 --- a/go/pkg/metastore/db/dao/segment_test.go +++ b/go/pkg/metastore/db/dao/segment_test.go @@ -1,25 +1,37 @@ package dao import ( + "github.com/chroma-core/chroma/go/pkg/metastore/db/dbcore" + "github.com/chroma-core/chroma/go/pkg/model" + "github.com/pingcap/log" + "github.com/stretchr/testify/suite" + "k8s.io/apimachinery/pkg/util/rand" + "strconv" "testing" "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" "github.com/chroma-core/chroma/go/pkg/types" - "github.com/stretchr/testify/assert" - "gorm.io/driver/sqlite" "gorm.io/gorm" ) -func TestSegmentDb_GetSegments(t *testing.T) { - db, err := gorm.Open(sqlite.Open(":memory:"), &gorm.Config{}) - assert.NoError(t, err) +type SegmentDbTestSuite struct { + suite.Suite + db *gorm.DB + segmentDb *segmentDb +} - err = db.AutoMigrate(&dbmodel.Segment{}, &dbmodel.SegmentMetadata{}) - assert.NoError(t, err) +func (suite *SegmentDbTestSuite) SetupSuite() { + log.Info("setup suite") + suite.db = dbcore.ConfigDatabaseForTesting() + suite.segmentDb = &segmentDb{ + db: suite.db, + } +} +func (suite *SegmentDbTestSuite) TestSegmentDb_GetSegments() { uniqueID := types.NewUniqueID() collectionID := uniqueID.String() - testTopic := "test_topic" + testTopic := "test_segment_topic" segment := &dbmodel.Segment{ ID: uniqueID.String(), CollectionID: &collectionID, @@ -27,8 +39,8 @@ func TestSegmentDb_GetSegments(t *testing.T) { Scope: "test_scope", Topic: &testTopic, } - err = db.Create(segment).Error - assert.NoError(t, err) + err := suite.db.Create(segment).Error + suite.NoError(err) testKey := "test" testValue := "test" @@ -37,53 +49,110 @@ func TestSegmentDb_GetSegments(t *testing.T) { Key: &testKey, StrValue: &testValue, } - err = db.Create(metadata).Error - assert.NoError(t, err) - - segmentDb := &segmentDb{ - db: db, - } + err = suite.db.Create(metadata).Error + suite.NoError(err) // Test when all parameters are nil - segments, err := segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.NilUniqueID()) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) - assert.Equal(t, segment.CollectionID, segments[0].Segment.CollectionID) - assert.Equal(t, segment.Type, segments[0].Segment.Type) - assert.Equal(t, segment.Scope, segments[0].Segment.Scope) - assert.Equal(t, segment.Topic, segments[0].Segment.Topic) - assert.Len(t, segments[0].SegmentMetadata, 1) - assert.Equal(t, metadata.Key, segments[0].SegmentMetadata[0].Key) - assert.Equal(t, metadata.StrValue, segments[0].SegmentMetadata[0].StrValue) + segments, err := suite.segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) + suite.Equal(segment.CollectionID, segments[0].Segment.CollectionID) + suite.Equal(segment.Type, segments[0].Segment.Type) + suite.Equal(segment.Scope, segments[0].Segment.Scope) + suite.Equal(segment.Topic, segments[0].Segment.Topic) + suite.Len(segments[0].SegmentMetadata, 1) + suite.Equal(metadata.Key, segments[0].SegmentMetadata[0].Key) + suite.Equal(metadata.StrValue, segments[0].SegmentMetadata[0].StrValue) // Test when filtering by ID - segments, err = segmentDb.GetSegments(types.MustParse(segment.ID), nil, nil, nil, types.NilUniqueID()) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) + segments, err = suite.segmentDb.GetSegments(types.MustParse(segment.ID), nil, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) // Test when filtering by type - segments, err = segmentDb.GetSegments(types.NilUniqueID(), &segment.Type, nil, nil, types.NilUniqueID()) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) + segments, err = suite.segmentDb.GetSegments(types.NilUniqueID(), &segment.Type, nil, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) // Test when filtering by scope - segments, err = segmentDb.GetSegments(types.NilUniqueID(), nil, &segment.Scope, nil, types.NilUniqueID()) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) + segments, err = suite.segmentDb.GetSegments(types.NilUniqueID(), nil, &segment.Scope, nil, types.NilUniqueID()) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) // Test when filtering by topic - segments, err = segmentDb.GetSegments(types.NilUniqueID(), nil, nil, segment.Topic, types.NilUniqueID()) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) + segments, err = suite.segmentDb.GetSegments(types.NilUniqueID(), nil, nil, segment.Topic, types.NilUniqueID()) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) // Test when filtering by collection ID - segments, err = segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.MustParse(*segment.CollectionID)) - assert.NoError(t, err) - assert.Len(t, segments, 1) - assert.Equal(t, segment.ID, segments[0].Segment.ID) + segments, err = suite.segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.MustParse(*segment.CollectionID)) + suite.NoError(err) + suite.Len(segments, 1) + suite.Equal(segment.ID, segments[0].Segment.ID) + + // clean up + err = suite.db.Delete(segment).Error + suite.NoError(err) + err = suite.db.Delete(metadata).Error + suite.NoError(err) +} + +func (suite *SegmentDbTestSuite) TestSegmentDb_RegisterFilePath() { + // create a collection for testing + databaseId := types.NewUniqueID().String() + collectionName := "test_segment_register_file_paths" + collectionID, err := CreateTestCollection(suite.db, collectionName, "test_topic", 128, databaseId) + suite.NoError(err) + + segments, err := suite.segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.MustParse(collectionID)) + suite.NoError(err) + + // create entries to flush + segmentsFilePaths := make(map[string]map[string][]string) + flushSegmentCompactions := make([]*model.FlushSegmentCompaction, 0) + testFilePathTypes := []string{"TypeA", "TypeB", "TypeC", "TypeD"} + for _, segment := range segments { + segmentID := segment.Segment.ID + segmentsFilePaths[segmentID] = make(map[string][]string) + for i := 0; i < rand.Intn(len(testFilePathTypes)); i++ { + filePaths := make([]string, 0) + for j := 0; j < rand.Intn(5); j++ { + filePaths = append(filePaths, "test_file_path_"+strconv.Itoa(j+1)) + } + filePathTypeI := rand.Intn(len(testFilePathTypes)) + filePathType := testFilePathTypes[filePathTypeI] + segmentsFilePaths[segmentID][filePathType] = filePaths + } + flushSegmentCompaction := &model.FlushSegmentCompaction{ + ID: types.MustParse(segmentID), + FilePaths: segmentsFilePaths[segmentID], + } + flushSegmentCompactions = append(flushSegmentCompactions, flushSegmentCompaction) + } + + // flush the entries + err = suite.segmentDb.RegisterFilePaths(flushSegmentCompactions) + suite.NoError(err) + + // verify file paths registered + segments, err = suite.segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.MustParse(collectionID)) + suite.NoError(err) + for _, segment := range segments { + suite.Contains(segmentsFilePaths, segment.Segment.ID) + suite.Equal(segmentsFilePaths[segment.Segment.ID], segment.Segment.FilePaths) + } + + // clean up + err = CleanUpTestCollection(suite.db, collectionID) + suite.NoError(err) +} + +func TestSegmentDbTestSuiteSuite(t *testing.T) { + testSuite := new(SegmentDbTestSuite) + suite.Run(t, testSuite) } diff --git a/go/pkg/metastore/db/dao/tenant.go b/go/pkg/metastore/db/dao/tenant.go index adc79c06dfa..fcd73f2cdcb 100644 --- a/go/pkg/metastore/db/dao/tenant.go +++ b/go/pkg/metastore/db/dao/tenant.go @@ -21,6 +21,12 @@ func (s *tenantDb) DeleteAll() error { return s.db.Where("1 = 1").Delete(&dbmodel.Tenant{}).Error } +func (s *tenantDb) DeleteByID(tenantID string) (int, error) { + var tenants []dbmodel.Tenant + err := s.db.Clauses(clause.Returning{}).Where("id = ?", tenantID).Delete(&tenants).Error + return len(tenants), err +} + func (s *tenantDb) GetAllTenants() ([]*dbmodel.Tenant, error) { var tenants []*dbmodel.Tenant @@ -61,6 +67,7 @@ func (s *tenantDb) Insert(tenant *dbmodel.Tenant) error { } func (s *tenantDb) UpdateTenantLastCompactionTime(tenantID string, lastCompactionTime int64) error { + log.Info("UpdateTenantLastCompactionTime", zap.String("tenantID", tenantID), zap.Int64("lastCompactionTime", lastCompactionTime)) var tenants []dbmodel.Tenant result := s.db.Model(&tenants). Clauses(clause.Returning{Columns: []clause.Column{{Name: "id"}}}). @@ -78,6 +85,7 @@ func (s *tenantDb) UpdateTenantLastCompactionTime(tenantID string, lastCompactio } func (s *tenantDb) GetTenantsLastCompactionTime(tenantIDs []string) ([]*dbmodel.Tenant, error) { + log.Info("GetTenantsLastCompactionTime", zap.Any("tenantIDs", tenantIDs)) var tenants []*dbmodel.Tenant result := s.db.Select("id", "last_compaction_time").Find(&tenants, "id IN ?", tenantIDs) diff --git a/go/pkg/metastore/db/dao/tenant_test.go b/go/pkg/metastore/db/dao/tenant_test.go index 5f4e658928a..7bb613ae7df 100644 --- a/go/pkg/metastore/db/dao/tenant_test.go +++ b/go/pkg/metastore/db/dao/tenant_test.go @@ -21,7 +21,6 @@ type TenantDbTestSuite struct { func (suite *TenantDbTestSuite) SetupSuite() { log.Info("setup suite") suite.db = dbcore.ConfigDatabaseForTesting() - dbcore.ResetTestTables(suite.db) suite.Db = &tenantDb{ db: suite.db, } @@ -38,14 +37,15 @@ func (suite *TenantDbTestSuite) TearDownTest() { func (suite *TenantDbTestSuite) TestTenantDb_UpdateTenantLastCompactionTime() { tenantId := "testUpdateTenantLastCompactionTime" var tenant dbmodel.Tenant - suite.Db.Insert(&dbmodel.Tenant{ + err := suite.Db.Insert(&dbmodel.Tenant{ ID: tenantId, LastCompactionTime: 0, }) + suite.Require().NoError(err) suite.db.First(&tenant, "id = ?", tenantId) suite.Require().Equal(int64(0), tenant.LastCompactionTime) - err := suite.Db.UpdateTenantLastCompactionTime(tenantId, 1) + err = suite.Db.UpdateTenantLastCompactionTime(tenantId, 1) suite.Require().NoError(err) suite.db.First(&tenant, "id = ?", tenantId) suite.Require().Equal(int64(1), tenant.LastCompactionTime) @@ -63,10 +63,11 @@ func (suite *TenantDbTestSuite) TestTenantDb_GetTenantsLastCompactionTime() { tenantIds := make([]string, 0) for i := 0; i < 10; i++ { tenantId := "testGetTenantsLastCompactionTime" + strconv.Itoa(i) - suite.Db.Insert(&dbmodel.Tenant{ + err := suite.Db.Insert(&dbmodel.Tenant{ ID: tenantId, LastCompactionTime: int64(i), }) + suite.Require().NoError(err) tenantIds = append(tenantIds, tenantId) } diff --git a/go/pkg/metastore/db/dao/test_utils.go b/go/pkg/metastore/db/dao/test_utils.go new file mode 100644 index 00000000000..6ae3293d1c1 --- /dev/null +++ b/go/pkg/metastore/db/dao/test_utils.go @@ -0,0 +1,184 @@ +package dao + +import ( + "github.com/chroma-core/chroma/go/pkg/metastore/db/dbmodel" + "github.com/chroma-core/chroma/go/pkg/types" + "github.com/pingcap/log" + "go.uber.org/zap" + "gorm.io/gorm" + "time" +) + +const SegmentType = "urn:chroma:segment/vector/hnsw-distributed" + +func GetSegmentScopes() []string { + return []string{"VECTOR", "METADATA"} +} + +func CreateTestTenantAndDatabase(db *gorm.DB, tenant string, database string) (string, error) { + log.Info("create test tenant and database", zap.String("tenant", tenant), zap.String("database", database)) + tenantDb := &tenantDb{ + db: db, + } + databaseDb := &databaseDb{ + db: db, + } + + err := tenantDb.Insert(&dbmodel.Tenant{ + ID: tenant, + LastCompactionTime: time.Now().Unix(), + }) + if err != nil { + return "", err + } + + databaseId := types.NewUniqueID().String() + err = databaseDb.Insert(&dbmodel.Database{ + ID: databaseId, + Name: database, + TenantID: tenant, + }) + if err != nil { + return "", err + } + + return databaseId, nil +} + +func CleanUpTestDatabase(db *gorm.DB, tenantName string, databaseName string) error { + log.Info("clean up test database", zap.String("tenantName", tenantName), zap.String("databaseName", databaseName)) + // clean up collections + collectionDb := &collectionDb{ + db: db, + } + collections, err := collectionDb.GetCollections(nil, nil, nil, tenantName, databaseName) + log.Info("clean up test database", zap.Int("collections", len(collections))) + if err != nil { + return err + } + for _, collection := range collections { + err = CleanUpTestCollection(db, collection.Collection.ID) + if err != nil { + return err + } + } + + // clean up database + databaseDb := &databaseDb{ + db: db, + } + + _, err = databaseDb.DeleteByTenantIdAndName(tenantName, databaseName) + if err != nil { + return err + } + + return nil +} + +func CleanUpTestTenant(db *gorm.DB, tenantName string) error { + log.Info("clean up test tenant", zap.String("tenantName", tenantName)) + tenantDb := &tenantDb{ + db: db, + } + databaseDb := &databaseDb{ + db: db, + } + + // clean up databases + databases, err := databaseDb.GetDatabasesByTenantID(tenantName) + if err != nil { + return err + } + for _, database := range databases { + err = CleanUpTestDatabase(db, tenantName, database.Name) + if err != nil { + return err + } + } + + // clean up tenant + _, err = tenantDb.DeleteByID(tenantName) + if err != nil { + return err + } + return nil +} + +func CreateTestCollection(db *gorm.DB, collectionName string, topic string, dimension int32, databaseID string) (string, error) { + log.Info("create test collection", zap.String("collectionName", collectionName), zap.String("topic", topic), zap.Int32("dimension", dimension), zap.String("databaseID", databaseID)) + collectionDb := &collectionDb{ + db: db, + } + segmentDb := &segmentDb{ + db: db, + } + collectionId := types.NewUniqueID().String() + + err := collectionDb.Insert(&dbmodel.Collection{ + ID: collectionId, + Name: &collectionName, + Topic: &topic, + Dimension: &dimension, + DatabaseID: databaseID, + }) + if err != nil { + return "", err + } + + for _, scope := range GetSegmentScopes() { + segmentId := types.NewUniqueID().String() + err = segmentDb.Insert(&dbmodel.Segment{ + CollectionID: &collectionId, + ID: segmentId, + Type: SegmentType, + Scope: scope, + }) + if err != nil { + return "", err + } + } + + return collectionId, nil +} + +func CleanUpTestCollection(db *gorm.DB, collectionId string) error { + log.Info("clean up collection", zap.String("collectionId", collectionId)) + collectionDb := &collectionDb{ + db: db, + } + collectionMetadataDb := &collectionMetadataDb{ + db: db, + } + segmentDb := &segmentDb{ + db: db, + } + segmentMetadataDb := &segmentMetadataDb{ + db: db, + } + + _, err := collectionMetadataDb.DeleteByCollectionID(collectionId) + if err != nil { + return err + } + _, err = collectionDb.DeleteCollectionByID(collectionId) + if err != nil { + return err + } + segments, err := segmentDb.GetSegments(types.NilUniqueID(), nil, nil, nil, types.MustParse(collectionId)) + if err != nil { + return err + } + for _, segment := range segments { + err = segmentDb.DeleteSegmentByID(segment.Segment.ID) + if err != nil { + return err + } + err = segmentMetadataDb.DeleteBySegmentID(segment.Segment.ID) + if err != nil { + return err + } + } + + return nil +} diff --git a/go/pkg/metastore/db/dbcore/core.go b/go/pkg/metastore/db/dbcore/core.go index 215b3375725..83b47338ae7 100644 --- a/go/pkg/metastore/db/dbcore/core.go +++ b/go/pkg/metastore/db/dbcore/core.go @@ -118,28 +118,69 @@ func GetDB(ctx context.Context) *gorm.DB { return globalDB.WithContext(ctx) } -func ResetTestTables(db *gorm.DB) { - db.Exec("TRUNCATE TABLE tenants, databases, collection_metadata, collections, segment_metadata, segments, notifications") - CreateDefaultTenantAndDatabase(db) -} - func CreateDefaultTenantAndDatabase(db *gorm.DB) string { - db.Model(&dbmodel.Tenant{}).Create(&dbmodel.Tenant{ + defaultTenant := &dbmodel.Tenant{ ID: common.DefaultTenant, LastCompactionTime: time.Now().Unix(), - }) - databaseId := types.NilUniqueID().String() - db.Model(&dbmodel.Database{}).Create(&dbmodel.Database{ - ID: databaseId, - Name: common.DefaultDatabase, - TenantID: common.DefaultTenant, - }) - return databaseId + } + db.Model(&dbmodel.Tenant{}).Where("id = ?", common.DefaultTenant).Save(defaultTenant) + + var database []dbmodel.Database + databaseId := types.NewUniqueID().String() + result := db.Model(&dbmodel.Database{}). + Where("name = ?", common.DefaultDatabase). + Where("tenant_id = ?", common.DefaultTenant). + Find(&database) + if result.Error != nil { + return "" + } + + if result.RowsAffected == 0 { + db.Create(&dbmodel.Database{ + ID: databaseId, + Name: common.DefaultDatabase, + TenantID: common.DefaultTenant, + }) + return databaseId + } + + err := result.Row().Scan(&database) + if err != nil { + return "" + } + return database[0].ID } func CreateTestTables(db *gorm.DB) { log.Info("CreateTestTables") - db.AutoMigrate(&dbmodel.Tenant{}, &dbmodel.Database{}, &dbmodel.CollectionMetadata{}, &dbmodel.Collection{}, &dbmodel.SegmentMetadata{}, &dbmodel.Segment{}, &dbmodel.Notification{}) + tableExist := db.Migrator().HasTable(&dbmodel.Tenant{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.Tenant{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.Database{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.Database{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.CollectionMetadata{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.CollectionMetadata{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.Collection{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.Collection{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.SegmentMetadata{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.SegmentMetadata{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.Segment{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.Segment{}) + } + tableExist = db.Migrator().HasTable(&dbmodel.Notification{}) + if !tableExist { + db.Migrator().CreateTable(&dbmodel.Notification{}) + } // create default tenant and database CreateDefaultTenantAndDatabase(db) diff --git a/go/pkg/metastore/db/dbmodel/collection.go b/go/pkg/metastore/db/dbmodel/collection.go index 4c6af65483c..30a9ab945ac 100644 --- a/go/pkg/metastore/db/dbmodel/collection.go +++ b/go/pkg/metastore/db/dbmodel/collection.go @@ -17,6 +17,7 @@ type Collection struct { CreatedAt time.Time `gorm:"created_at;type:timestamp;not null;default:current_timestamp"` UpdatedAt time.Time `gorm:"updated_at;type:timestamp;not null;default:current_timestamp"` LogPosition int64 `gorm:"log_position;default:0"` + Version int32 `gorm:"version;default:0"` } func (v Collection) TableName() string { @@ -37,4 +38,5 @@ type ICollectionDb interface { Insert(in *Collection) error Update(in *Collection) error DeleteAll() error + UpdateLogPositionAndVersion(collectionID string, logPosition int64, currentCollectionVersion int32) (int32, error) } diff --git a/go/pkg/metastore/db/dbmodel/mocks/ICollectionDb.go b/go/pkg/metastore/db/dbmodel/mocks/ICollectionDb.go index 1a07397926c..b819b0b1889 100644 --- a/go/pkg/metastore/db/dbmodel/mocks/ICollectionDb.go +++ b/go/pkg/metastore/db/dbmodel/mocks/ICollectionDb.go @@ -124,6 +124,34 @@ func (_m *ICollectionDb) Update(in *dbmodel.Collection) error { return r0 } +// UpdateLogPositionAndVersion provides a mock function with given fields: collectionID, logPosition, currentCollectionVersion +func (_m *ICollectionDb) UpdateLogPositionAndVersion(collectionID string, logPosition int64, currentCollectionVersion int32) (int32, error) { + ret := _m.Called(collectionID, logPosition, currentCollectionVersion) + + if len(ret) == 0 { + panic("no return value specified for UpdateLogPositionAndVersion") + } + + var r0 int32 + var r1 error + if rf, ok := ret.Get(0).(func(string, int64, int32) (int32, error)); ok { + return rf(collectionID, logPosition, currentCollectionVersion) + } + if rf, ok := ret.Get(0).(func(string, int64, int32) int32); ok { + r0 = rf(collectionID, logPosition, currentCollectionVersion) + } else { + r0 = ret.Get(0).(int32) + } + + if rf, ok := ret.Get(1).(func(string, int64, int32) error); ok { + r1 = rf(collectionID, logPosition, currentCollectionVersion) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + // NewICollectionDb creates a new instance of ICollectionDb. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. // The first argument is typically a *testing.T value. func NewICollectionDb(t interface { diff --git a/go/pkg/metastore/db/dbmodel/record_log.go b/go/pkg/metastore/db/dbmodel/record_log.go index 72ff7f8f2f6..5bd4da0ca0a 100644 --- a/go/pkg/metastore/db/dbmodel/record_log.go +++ b/go/pkg/metastore/db/dbmodel/record_log.go @@ -18,6 +18,6 @@ func (v RecordLog) TableName() string { //go:generate mockery --name=IRecordLogDb type IRecordLogDb interface { PushLogs(collectionID types.UniqueID, recordsContent [][]byte) (int, error) - PullLogs(collectionID types.UniqueID, id int64, batchSize int) ([]*RecordLog, error) + PullLogs(collectionID types.UniqueID, id int64, batchSize int, endTimestamp int64) ([]*RecordLog, error) GetAllCollectionsToCompact() ([]*RecordLog, error) } diff --git a/go/pkg/metastore/db/dbmodel/segment.go b/go/pkg/metastore/db/dbmodel/segment.go index 0285f32791a..14eaf19ca4c 100644 --- a/go/pkg/metastore/db/dbmodel/segment.go +++ b/go/pkg/metastore/db/dbmodel/segment.go @@ -1,6 +1,7 @@ package dbmodel import ( + "github.com/chroma-core/chroma/go/pkg/model" "time" "github.com/chroma-core/chroma/go/pkg/types" @@ -11,15 +12,16 @@ type Segment struct { This requires us to push down CollectionID from the caller. We don't think there is need to modify CollectionID in the near future. Each Segment should always have a collection as a parent and cannot be modified. */ - CollectionID *string `gorm:"collection_id;primaryKey"` - ID string `gorm:"id;primaryKey"` - Type string `gorm:"type;type:string;not null"` - Scope string `gorm:"scope"` - Topic *string `gorm:"topic"` - Ts types.Timestamp `gorm:"ts;type:bigint;default:0"` - IsDeleted bool `gorm:"is_deleted;type:bool;default:false"` - CreatedAt time.Time `gorm:"created_at;type:timestamp;not null;default:current_timestamp"` - UpdatedAt time.Time `gorm:"updated_at;type:timestamp;not null;default:current_timestamp"` + CollectionID *string `gorm:"collection_id;primaryKey"` + ID string `gorm:"id;primaryKey"` + Type string `gorm:"type;type:string;not null"` + Scope string `gorm:"scope"` + Topic *string `gorm:"topic"` + Ts types.Timestamp `gorm:"ts;type:bigint;default:0"` + IsDeleted bool `gorm:"is_deleted;type:bool;default:false"` + CreatedAt time.Time `gorm:"created_at;type:timestamp;not null;default:current_timestamp"` + UpdatedAt time.Time `gorm:"updated_at;type:timestamp;not null;default:current_timestamp"` + FilePaths map[string][]string `gorm:"file_paths;serializer:json;default:'{}'"` } func (s Segment) TableName() string { @@ -46,4 +48,5 @@ type ISegmentDb interface { Insert(*Segment) error Update(*UpdateSegment) error DeleteAll() error + RegisterFilePaths(flushSegmentCompactions []*model.FlushSegmentCompaction) error } diff --git a/go/pkg/model/collection.go b/go/pkg/model/collection.go index 240d81fa8a2..1340c44df5b 100644 --- a/go/pkg/model/collection.go +++ b/go/pkg/model/collection.go @@ -13,6 +13,8 @@ type Collection struct { TenantID string DatabaseName string Ts types.Timestamp + LogPosition int64 + Version int32 } type CreateCollection struct { @@ -46,6 +48,20 @@ type UpdateCollection struct { Ts types.Timestamp } +type FlushCollectionCompaction struct { + ID types.UniqueID + TenantID string + LogPosition int64 + CurrentCollectionVersion int32 + FlushSegmentCompactions []*FlushSegmentCompaction +} + +type FlushCollectionInfo struct { + ID string + CollectionVersion int32 + TenantLastCompactionTime int64 +} + func FilterCollection(collection *Collection, collectionID types.UniqueID, collectionName *string, collectionTopic *string) bool { if collectionID != types.NilUniqueID() && collectionID != collection.ID { return false diff --git a/go/pkg/model/segment.go b/go/pkg/model/segment.go index 3127f515aaa..07030e77c91 100644 --- a/go/pkg/model/segment.go +++ b/go/pkg/model/segment.go @@ -12,6 +12,7 @@ type Segment struct { CollectionID types.UniqueID Metadata *SegmentMetadata[SegmentMetadataValueType] Ts types.Timestamp + FilePaths map[string][]string } type CreateSegment struct { @@ -43,6 +44,11 @@ type GetSegments struct { CollectionID types.UniqueID } +type FlushSegmentCompaction struct { + ID types.UniqueID + FilePaths map[string][]string +} + func FilterSegments(segment *Segment, segmentID types.UniqueID, segmentType *string, scope *string, topic *string, collectionID types.UniqueID) bool { if segmentID != types.NilUniqueID() && segment.ID != segmentID { return false diff --git a/go/pkg/proto/coordinatorpb/chroma.pb.go b/go/pkg/proto/coordinatorpb/chroma.pb.go index 49b077e803a..62566a9a083 100644 --- a/go/pkg/proto/coordinatorpb/chroma.pb.go +++ b/go/pkg/proto/coordinatorpb/chroma.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go. DO NOT EDIT. // versions: -// protoc-gen-go v1.32.0 -// protoc v4.25.3 +// protoc-gen-go v1.31.0 +// protoc v4.23.4 // source: chromadb/proto/chroma.proto package coordinatorpb @@ -282,6 +282,53 @@ func (x *Vector) GetEncoding() ScalarEncoding { return ScalarEncoding_FLOAT32 } +type FilePaths struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Paths []string `protobuf:"bytes,1,rep,name=paths,proto3" json:"paths,omitempty"` +} + +func (x *FilePaths) Reset() { + *x = FilePaths{} + if protoimpl.UnsafeEnabled { + mi := &file_chromadb_proto_chroma_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *FilePaths) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*FilePaths) ProtoMessage() {} + +func (x *FilePaths) ProtoReflect() protoreflect.Message { + mi := &file_chromadb_proto_chroma_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use FilePaths.ProtoReflect.Descriptor instead. +func (*FilePaths) Descriptor() ([]byte, []int) { + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{2} +} + +func (x *FilePaths) GetPaths() []string { + if x != nil { + return x.Paths + } + return nil +} + type Segment struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache @@ -293,14 +340,15 @@ type Segment struct { Topic *string `protobuf:"bytes,4,opt,name=topic,proto3,oneof" json:"topic,omitempty"` // TODO should channel <> segment binding exist here? // If a segment has a collection, it implies that this segment implements the full // collection and can be used to service queries (for it's given scope.) - Collection *string `protobuf:"bytes,5,opt,name=collection,proto3,oneof" json:"collection,omitempty"` - Metadata *UpdateMetadata `protobuf:"bytes,6,opt,name=metadata,proto3,oneof" json:"metadata,omitempty"` + Collection *string `protobuf:"bytes,5,opt,name=collection,proto3,oneof" json:"collection,omitempty"` + Metadata *UpdateMetadata `protobuf:"bytes,6,opt,name=metadata,proto3,oneof" json:"metadata,omitempty"` + FilePaths map[string]*FilePaths `protobuf:"bytes,7,rep,name=file_paths,json=filePaths,proto3" json:"file_paths,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` } func (x *Segment) Reset() { *x = Segment{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[2] + mi := &file_chromadb_proto_chroma_proto_msgTypes[3] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -313,7 +361,7 @@ func (x *Segment) String() string { func (*Segment) ProtoMessage() {} func (x *Segment) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[2] + mi := &file_chromadb_proto_chroma_proto_msgTypes[3] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -326,7 +374,7 @@ func (x *Segment) ProtoReflect() protoreflect.Message { // Deprecated: Use Segment.ProtoReflect.Descriptor instead. func (*Segment) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{2} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{3} } func (x *Segment) GetId() string { @@ -371,24 +419,33 @@ func (x *Segment) GetMetadata() *UpdateMetadata { return nil } +func (x *Segment) GetFilePaths() map[string]*FilePaths { + if x != nil { + return x.FilePaths + } + return nil +} + type Collection struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache unknownFields protoimpl.UnknownFields - Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` - Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"` - Topic string `protobuf:"bytes,3,opt,name=topic,proto3" json:"topic,omitempty"` - Metadata *UpdateMetadata `protobuf:"bytes,4,opt,name=metadata,proto3,oneof" json:"metadata,omitempty"` - Dimension *int32 `protobuf:"varint,5,opt,name=dimension,proto3,oneof" json:"dimension,omitempty"` - Tenant string `protobuf:"bytes,6,opt,name=tenant,proto3" json:"tenant,omitempty"` - Database string `protobuf:"bytes,7,opt,name=database,proto3" json:"database,omitempty"` + Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` + Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"` + Topic string `protobuf:"bytes,3,opt,name=topic,proto3" json:"topic,omitempty"` + Metadata *UpdateMetadata `protobuf:"bytes,4,opt,name=metadata,proto3,oneof" json:"metadata,omitempty"` + Dimension *int32 `protobuf:"varint,5,opt,name=dimension,proto3,oneof" json:"dimension,omitempty"` + Tenant string `protobuf:"bytes,6,opt,name=tenant,proto3" json:"tenant,omitempty"` + Database string `protobuf:"bytes,7,opt,name=database,proto3" json:"database,omitempty"` + LogPosition int64 `protobuf:"varint,8,opt,name=logPosition,proto3" json:"logPosition,omitempty"` + Version int32 `protobuf:"varint,9,opt,name=version,proto3" json:"version,omitempty"` } func (x *Collection) Reset() { *x = Collection{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[3] + mi := &file_chromadb_proto_chroma_proto_msgTypes[4] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -401,7 +458,7 @@ func (x *Collection) String() string { func (*Collection) ProtoMessage() {} func (x *Collection) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[3] + mi := &file_chromadb_proto_chroma_proto_msgTypes[4] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -414,7 +471,7 @@ func (x *Collection) ProtoReflect() protoreflect.Message { // Deprecated: Use Collection.ProtoReflect.Descriptor instead. func (*Collection) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{3} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{4} } func (x *Collection) GetId() string { @@ -466,6 +523,20 @@ func (x *Collection) GetDatabase() string { return "" } +func (x *Collection) GetLogPosition() int64 { + if x != nil { + return x.LogPosition + } + return 0 +} + +func (x *Collection) GetVersion() int32 { + if x != nil { + return x.Version + } + return 0 +} + type Database struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache @@ -479,7 +550,7 @@ type Database struct { func (x *Database) Reset() { *x = Database{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[4] + mi := &file_chromadb_proto_chroma_proto_msgTypes[5] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -492,7 +563,7 @@ func (x *Database) String() string { func (*Database) ProtoMessage() {} func (x *Database) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[4] + mi := &file_chromadb_proto_chroma_proto_msgTypes[5] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -505,7 +576,7 @@ func (x *Database) ProtoReflect() protoreflect.Message { // Deprecated: Use Database.ProtoReflect.Descriptor instead. func (*Database) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{4} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{5} } func (x *Database) GetId() string { @@ -540,7 +611,7 @@ type Tenant struct { func (x *Tenant) Reset() { *x = Tenant{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[5] + mi := &file_chromadb_proto_chroma_proto_msgTypes[6] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -553,7 +624,7 @@ func (x *Tenant) String() string { func (*Tenant) ProtoMessage() {} func (x *Tenant) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[5] + mi := &file_chromadb_proto_chroma_proto_msgTypes[6] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -566,7 +637,7 @@ func (x *Tenant) ProtoReflect() protoreflect.Message { // Deprecated: Use Tenant.ProtoReflect.Descriptor instead. func (*Tenant) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{5} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{6} } func (x *Tenant) GetName() string { @@ -592,7 +663,7 @@ type UpdateMetadataValue struct { func (x *UpdateMetadataValue) Reset() { *x = UpdateMetadataValue{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[6] + mi := &file_chromadb_proto_chroma_proto_msgTypes[7] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -605,7 +676,7 @@ func (x *UpdateMetadataValue) String() string { func (*UpdateMetadataValue) ProtoMessage() {} func (x *UpdateMetadataValue) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[6] + mi := &file_chromadb_proto_chroma_proto_msgTypes[7] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -618,7 +689,7 @@ func (x *UpdateMetadataValue) ProtoReflect() protoreflect.Message { // Deprecated: Use UpdateMetadataValue.ProtoReflect.Descriptor instead. func (*UpdateMetadataValue) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{6} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{7} } func (m *UpdateMetadataValue) GetValue() isUpdateMetadataValue_Value { @@ -682,7 +753,7 @@ type UpdateMetadata struct { func (x *UpdateMetadata) Reset() { *x = UpdateMetadata{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[7] + mi := &file_chromadb_proto_chroma_proto_msgTypes[8] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -695,7 +766,7 @@ func (x *UpdateMetadata) String() string { func (*UpdateMetadata) ProtoMessage() {} func (x *UpdateMetadata) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[7] + mi := &file_chromadb_proto_chroma_proto_msgTypes[8] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -708,7 +779,7 @@ func (x *UpdateMetadata) ProtoReflect() protoreflect.Message { // Deprecated: Use UpdateMetadata.ProtoReflect.Descriptor instead. func (*UpdateMetadata) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{7} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{8} } func (x *UpdateMetadata) GetMetadata() map[string]*UpdateMetadataValue { @@ -733,7 +804,7 @@ type SubmitEmbeddingRecord struct { func (x *SubmitEmbeddingRecord) Reset() { *x = SubmitEmbeddingRecord{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[8] + mi := &file_chromadb_proto_chroma_proto_msgTypes[9] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -746,7 +817,7 @@ func (x *SubmitEmbeddingRecord) String() string { func (*SubmitEmbeddingRecord) ProtoMessage() {} func (x *SubmitEmbeddingRecord) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[8] + mi := &file_chromadb_proto_chroma_proto_msgTypes[9] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -759,7 +830,7 @@ func (x *SubmitEmbeddingRecord) ProtoReflect() protoreflect.Message { // Deprecated: Use SubmitEmbeddingRecord.ProtoReflect.Descriptor instead. func (*SubmitEmbeddingRecord) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{8} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{9} } func (x *SubmitEmbeddingRecord) GetId() string { @@ -810,7 +881,7 @@ type VectorEmbeddingRecord struct { func (x *VectorEmbeddingRecord) Reset() { *x = VectorEmbeddingRecord{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[9] + mi := &file_chromadb_proto_chroma_proto_msgTypes[10] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -823,7 +894,7 @@ func (x *VectorEmbeddingRecord) String() string { func (*VectorEmbeddingRecord) ProtoMessage() {} func (x *VectorEmbeddingRecord) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[9] + mi := &file_chromadb_proto_chroma_proto_msgTypes[10] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -836,7 +907,7 @@ func (x *VectorEmbeddingRecord) ProtoReflect() protoreflect.Message { // Deprecated: Use VectorEmbeddingRecord.ProtoReflect.Descriptor instead. func (*VectorEmbeddingRecord) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{9} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{10} } func (x *VectorEmbeddingRecord) GetId() string { @@ -874,7 +945,7 @@ type VectorQueryResult struct { func (x *VectorQueryResult) Reset() { *x = VectorQueryResult{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[10] + mi := &file_chromadb_proto_chroma_proto_msgTypes[11] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -887,7 +958,7 @@ func (x *VectorQueryResult) String() string { func (*VectorQueryResult) ProtoMessage() {} func (x *VectorQueryResult) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[10] + mi := &file_chromadb_proto_chroma_proto_msgTypes[11] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -900,7 +971,7 @@ func (x *VectorQueryResult) ProtoReflect() protoreflect.Message { // Deprecated: Use VectorQueryResult.ProtoReflect.Descriptor instead. func (*VectorQueryResult) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{10} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{11} } func (x *VectorQueryResult) GetId() string { @@ -942,7 +1013,7 @@ type VectorQueryResults struct { func (x *VectorQueryResults) Reset() { *x = VectorQueryResults{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[11] + mi := &file_chromadb_proto_chroma_proto_msgTypes[12] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -955,7 +1026,7 @@ func (x *VectorQueryResults) String() string { func (*VectorQueryResults) ProtoMessage() {} func (x *VectorQueryResults) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[11] + mi := &file_chromadb_proto_chroma_proto_msgTypes[12] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -968,7 +1039,7 @@ func (x *VectorQueryResults) ProtoReflect() protoreflect.Message { // Deprecated: Use VectorQueryResults.ProtoReflect.Descriptor instead. func (*VectorQueryResults) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{11} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{12} } func (x *VectorQueryResults) GetResults() []*VectorQueryResult { @@ -990,7 +1061,7 @@ type GetVectorsRequest struct { func (x *GetVectorsRequest) Reset() { *x = GetVectorsRequest{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[12] + mi := &file_chromadb_proto_chroma_proto_msgTypes[13] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -1003,7 +1074,7 @@ func (x *GetVectorsRequest) String() string { func (*GetVectorsRequest) ProtoMessage() {} func (x *GetVectorsRequest) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[12] + mi := &file_chromadb_proto_chroma_proto_msgTypes[13] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -1016,7 +1087,7 @@ func (x *GetVectorsRequest) ProtoReflect() protoreflect.Message { // Deprecated: Use GetVectorsRequest.ProtoReflect.Descriptor instead. func (*GetVectorsRequest) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{12} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{13} } func (x *GetVectorsRequest) GetIds() []string { @@ -1044,7 +1115,7 @@ type GetVectorsResponse struct { func (x *GetVectorsResponse) Reset() { *x = GetVectorsResponse{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[13] + mi := &file_chromadb_proto_chroma_proto_msgTypes[14] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -1057,7 +1128,7 @@ func (x *GetVectorsResponse) String() string { func (*GetVectorsResponse) ProtoMessage() {} func (x *GetVectorsResponse) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[13] + mi := &file_chromadb_proto_chroma_proto_msgTypes[14] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -1070,7 +1141,7 @@ func (x *GetVectorsResponse) ProtoReflect() protoreflect.Message { // Deprecated: Use GetVectorsResponse.ProtoReflect.Descriptor instead. func (*GetVectorsResponse) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{13} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{14} } func (x *GetVectorsResponse) GetRecords() []*VectorEmbeddingRecord { @@ -1095,7 +1166,7 @@ type QueryVectorsRequest struct { func (x *QueryVectorsRequest) Reset() { *x = QueryVectorsRequest{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[14] + mi := &file_chromadb_proto_chroma_proto_msgTypes[15] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -1108,7 +1179,7 @@ func (x *QueryVectorsRequest) String() string { func (*QueryVectorsRequest) ProtoMessage() {} func (x *QueryVectorsRequest) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[14] + mi := &file_chromadb_proto_chroma_proto_msgTypes[15] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -1121,7 +1192,7 @@ func (x *QueryVectorsRequest) ProtoReflect() protoreflect.Message { // Deprecated: Use QueryVectorsRequest.ProtoReflect.Descriptor instead. func (*QueryVectorsRequest) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{14} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{15} } func (x *QueryVectorsRequest) GetVectors() []*Vector { @@ -1170,7 +1241,7 @@ type QueryVectorsResponse struct { func (x *QueryVectorsResponse) Reset() { *x = QueryVectorsResponse{} if protoimpl.UnsafeEnabled { - mi := &file_chromadb_proto_chroma_proto_msgTypes[15] + mi := &file_chromadb_proto_chroma_proto_msgTypes[16] ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) ms.StoreMessageInfo(mi) } @@ -1183,7 +1254,7 @@ func (x *QueryVectorsResponse) String() string { func (*QueryVectorsResponse) ProtoMessage() {} func (x *QueryVectorsResponse) ProtoReflect() protoreflect.Message { - mi := &file_chromadb_proto_chroma_proto_msgTypes[15] + mi := &file_chromadb_proto_chroma_proto_msgTypes[16] if protoimpl.UnsafeEnabled && x != nil { ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) if ms.LoadMessageInfo() == nil { @@ -1196,7 +1267,7 @@ func (x *QueryVectorsResponse) ProtoReflect() protoreflect.Message { // Deprecated: Use QueryVectorsResponse.ProtoReflect.Descriptor instead. func (*QueryVectorsResponse) Descriptor() ([]byte, []int) { - return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{15} + return file_chromadb_proto_chroma_proto_rawDescGZIP(), []int{16} } func (x *QueryVectorsResponse) GetResults() []*VectorQueryResults { @@ -1222,149 +1293,164 @@ var file_chromadb_proto_chroma_proto_rawDesc = []byte{ 0x6e, 0x63, 0x6f, 0x64, 0x69, 0x6e, 0x67, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x63, 0x61, 0x6c, 0x61, 0x72, 0x45, 0x6e, 0x63, 0x6f, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x08, 0x65, 0x6e, 0x63, 0x6f, 0x64, 0x69, 0x6e, 0x67, 0x22, - 0xf8, 0x01, 0x0a, 0x07, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x0e, 0x0a, 0x02, 0x69, - 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x74, - 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x74, 0x79, 0x70, 0x65, 0x12, - 0x2a, 0x0a, 0x05, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x14, - 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x53, - 0x63, 0x6f, 0x70, 0x65, 0x52, 0x05, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x12, 0x19, 0x0a, 0x05, 0x74, - 0x6f, 0x70, 0x69, 0x63, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, 0x05, 0x74, 0x6f, - 0x70, 0x69, 0x63, 0x88, 0x01, 0x01, 0x12, 0x23, 0x0a, 0x0a, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, - 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x05, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, 0x0a, 0x63, 0x6f, - 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x88, 0x01, 0x01, 0x12, 0x37, 0x0a, 0x08, 0x6d, - 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, - 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, - 0x61, 0x64, 0x61, 0x74, 0x61, 0x48, 0x02, 0x52, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, - 0x61, 0x88, 0x01, 0x01, 0x42, 0x08, 0x0a, 0x06, 0x5f, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x42, 0x0d, - 0x0a, 0x0b, 0x5f, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x42, 0x0b, 0x0a, - 0x09, 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x22, 0xf1, 0x01, 0x0a, 0x0a, 0x43, - 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, + 0x21, 0x0a, 0x09, 0x46, 0x69, 0x6c, 0x65, 0x50, 0x61, 0x74, 0x68, 0x73, 0x12, 0x14, 0x0a, 0x05, + 0x70, 0x61, 0x74, 0x68, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, 0x05, 0x70, 0x61, 0x74, + 0x68, 0x73, 0x22, 0x88, 0x03, 0x0a, 0x07, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x0e, + 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, + 0x0a, 0x04, 0x74, 0x79, 0x70, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x74, 0x79, + 0x70, 0x65, 0x12, 0x2a, 0x0a, 0x05, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, + 0x0e, 0x32, 0x14, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x65, 0x67, 0x6d, 0x65, + 0x6e, 0x74, 0x53, 0x63, 0x6f, 0x70, 0x65, 0x52, 0x05, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x12, 0x19, + 0x0a, 0x05, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x48, 0x00, 0x52, + 0x05, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x88, 0x01, 0x01, 0x12, 0x23, 0x0a, 0x0a, 0x63, 0x6f, 0x6c, + 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x05, 0x20, 0x01, 0x28, 0x09, 0x48, 0x01, 0x52, + 0x0a, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x88, 0x01, 0x01, 0x12, 0x37, + 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, + 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, + 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x48, 0x02, 0x52, 0x08, 0x6d, 0x65, 0x74, 0x61, + 0x64, 0x61, 0x74, 0x61, 0x88, 0x01, 0x01, 0x12, 0x3d, 0x0a, 0x0a, 0x66, 0x69, 0x6c, 0x65, 0x5f, + 0x70, 0x61, 0x74, 0x68, 0x73, 0x18, 0x07, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1e, 0x2e, 0x63, 0x68, + 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x2e, 0x46, 0x69, 0x6c, + 0x65, 0x50, 0x61, 0x74, 0x68, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x09, 0x66, 0x69, 0x6c, + 0x65, 0x50, 0x61, 0x74, 0x68, 0x73, 0x1a, 0x4f, 0x0a, 0x0e, 0x46, 0x69, 0x6c, 0x65, 0x50, 0x61, + 0x74, 0x68, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, + 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, + 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2e, 0x46, 0x69, 0x6c, 0x65, 0x50, 0x61, 0x74, 0x68, 0x73, 0x52, 0x05, 0x76, 0x61, + 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x42, 0x08, 0x0a, 0x06, 0x5f, 0x74, 0x6f, 0x70, 0x69, + 0x63, 0x42, 0x0d, 0x0a, 0x0b, 0x5f, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x42, 0x0b, 0x0a, 0x09, 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x22, 0xad, 0x02, + 0x0a, 0x0a, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x0e, 0x0a, 0x02, + 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, + 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, + 0x12, 0x14, 0x0a, 0x05, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, + 0x05, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x12, 0x37, 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, + 0x74, 0x61, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, + 0x48, 0x00, 0x52, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x88, 0x01, 0x01, 0x12, + 0x21, 0x0a, 0x09, 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x05, 0x20, 0x01, + 0x28, 0x05, 0x48, 0x01, 0x52, 0x09, 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x88, + 0x01, 0x01, 0x12, 0x16, 0x0a, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, 0x06, 0x20, 0x01, + 0x28, 0x09, 0x52, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x1a, 0x0a, 0x08, 0x64, 0x61, + 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x64, 0x61, + 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x12, 0x20, 0x0a, 0x0b, 0x6c, 0x6f, 0x67, 0x50, 0x6f, 0x73, + 0x69, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x08, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0b, 0x6c, 0x6f, 0x67, + 0x50, 0x6f, 0x73, 0x69, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x18, 0x0a, 0x07, 0x76, 0x65, 0x72, 0x73, + 0x69, 0x6f, 0x6e, 0x18, 0x09, 0x20, 0x01, 0x28, 0x05, 0x52, 0x07, 0x76, 0x65, 0x72, 0x73, 0x69, + 0x6f, 0x6e, 0x42, 0x0b, 0x0a, 0x09, 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x42, + 0x0c, 0x0a, 0x0a, 0x5f, 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x22, 0x46, 0x0a, + 0x08, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, - 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x14, 0x0a, - 0x05, 0x74, 0x6f, 0x70, 0x69, 0x63, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x74, 0x6f, - 0x70, 0x69, 0x63, 0x12, 0x37, 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, - 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, - 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x48, 0x00, 0x52, - 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x88, 0x01, 0x01, 0x12, 0x21, 0x0a, 0x09, - 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x05, 0x20, 0x01, 0x28, 0x05, 0x48, - 0x01, 0x52, 0x09, 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x88, 0x01, 0x01, 0x12, - 0x16, 0x0a, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, 0x06, 0x20, 0x01, 0x28, 0x09, 0x52, - 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x1a, 0x0a, 0x08, 0x64, 0x61, 0x74, 0x61, 0x62, - 0x61, 0x73, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x64, 0x61, 0x74, 0x61, 0x62, - 0x61, 0x73, 0x65, 0x42, 0x0b, 0x0a, 0x09, 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, - 0x42, 0x0c, 0x0a, 0x0a, 0x5f, 0x64, 0x69, 0x6d, 0x65, 0x6e, 0x73, 0x69, 0x6f, 0x6e, 0x22, 0x46, - 0x0a, 0x08, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, - 0x6d, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x16, - 0x0a, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, - 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x22, 0x1c, 0x0a, 0x06, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, - 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, - 0x6e, 0x61, 0x6d, 0x65, 0x22, 0x85, 0x01, 0x0a, 0x13, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, - 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x12, 0x23, 0x0a, 0x0c, - 0x73, 0x74, 0x72, 0x69, 0x6e, 0x67, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x01, 0x20, 0x01, - 0x28, 0x09, 0x48, 0x00, 0x52, 0x0b, 0x73, 0x74, 0x72, 0x69, 0x6e, 0x67, 0x56, 0x61, 0x6c, 0x75, - 0x65, 0x12, 0x1d, 0x0a, 0x09, 0x69, 0x6e, 0x74, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, - 0x20, 0x01, 0x28, 0x03, 0x48, 0x00, 0x52, 0x08, 0x69, 0x6e, 0x74, 0x56, 0x61, 0x6c, 0x75, 0x65, - 0x12, 0x21, 0x0a, 0x0b, 0x66, 0x6c, 0x6f, 0x61, 0x74, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, - 0x03, 0x20, 0x01, 0x28, 0x01, 0x48, 0x00, 0x52, 0x0a, 0x66, 0x6c, 0x6f, 0x61, 0x74, 0x56, 0x61, - 0x6c, 0x75, 0x65, 0x42, 0x07, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x22, 0xac, 0x01, 0x0a, - 0x0e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x12, - 0x40, 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x01, 0x20, 0x03, 0x28, - 0x0b, 0x32, 0x24, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, - 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x2e, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, - 0x74, 0x61, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, - 0x61, 0x1a, 0x58, 0x0a, 0x0d, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x45, 0x6e, 0x74, - 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, - 0x03, 0x6b, 0x65, 0x79, 0x12, 0x31, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, - 0x01, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, - 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x56, 0x61, 0x6c, 0x75, 0x65, - 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0xfb, 0x01, 0x0a, 0x15, - 0x53, 0x75, 0x62, 0x6d, 0x69, 0x74, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, - 0x65, 0x63, 0x6f, 0x72, 0x64, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, - 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x2b, 0x0a, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x18, - 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x56, - 0x65, 0x63, 0x74, 0x6f, 0x72, 0x48, 0x00, 0x52, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x88, - 0x01, 0x01, 0x12, 0x37, 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x03, - 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, - 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x48, 0x01, 0x52, 0x08, - 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x88, 0x01, 0x01, 0x12, 0x2f, 0x0a, 0x09, 0x6f, - 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x11, - 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x4f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, - 0x6e, 0x52, 0x09, 0x6f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x23, 0x0a, 0x0d, - 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x18, 0x05, 0x20, - 0x01, 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, - 0x64, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x42, 0x0b, 0x0a, 0x09, - 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x22, 0x66, 0x0a, 0x15, 0x56, 0x65, 0x63, - 0x74, 0x6f, 0x72, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, - 0x72, 0x64, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, - 0x69, 0x64, 0x12, 0x15, 0x0a, 0x06, 0x73, 0x65, 0x71, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, - 0x28, 0x0c, 0x52, 0x05, 0x73, 0x65, 0x71, 0x49, 0x64, 0x12, 0x26, 0x0a, 0x06, 0x76, 0x65, 0x63, - 0x74, 0x6f, 0x72, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, 0x6f, - 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, - 0x72, 0x22, 0x8e, 0x01, 0x0a, 0x11, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, - 0x79, 0x52, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, - 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x15, 0x0a, 0x06, 0x73, 0x65, 0x71, 0x5f, 0x69, - 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0c, 0x52, 0x05, 0x73, 0x65, 0x71, 0x49, 0x64, 0x12, 0x1a, - 0x0a, 0x08, 0x64, 0x69, 0x73, 0x74, 0x61, 0x6e, 0x63, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x02, - 0x52, 0x08, 0x64, 0x69, 0x73, 0x74, 0x61, 0x6e, 0x63, 0x65, 0x12, 0x2b, 0x0a, 0x06, 0x76, 0x65, - 0x63, 0x74, 0x6f, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x48, 0x00, 0x52, 0x06, 0x76, 0x65, - 0x63, 0x74, 0x6f, 0x72, 0x88, 0x01, 0x01, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x76, 0x65, 0x63, 0x74, - 0x6f, 0x72, 0x22, 0x49, 0x0a, 0x12, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, - 0x79, 0x52, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x12, 0x33, 0x0a, 0x07, 0x72, 0x65, 0x73, 0x75, - 0x6c, 0x74, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x19, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x16, 0x0a, + 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x74, + 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x22, 0x1c, 0x0a, 0x06, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, + 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, + 0x61, 0x6d, 0x65, 0x22, 0x85, 0x01, 0x0a, 0x13, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, + 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x12, 0x23, 0x0a, 0x0c, 0x73, + 0x74, 0x72, 0x69, 0x6e, 0x67, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x09, 0x48, 0x00, 0x52, 0x0b, 0x73, 0x74, 0x72, 0x69, 0x6e, 0x67, 0x56, 0x61, 0x6c, 0x75, 0x65, + 0x12, 0x1d, 0x0a, 0x09, 0x69, 0x6e, 0x74, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, + 0x01, 0x28, 0x03, 0x48, 0x00, 0x52, 0x08, 0x69, 0x6e, 0x74, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x12, + 0x21, 0x0a, 0x0b, 0x66, 0x6c, 0x6f, 0x61, 0x74, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x03, + 0x20, 0x01, 0x28, 0x01, 0x48, 0x00, 0x52, 0x0a, 0x66, 0x6c, 0x6f, 0x61, 0x74, 0x56, 0x61, 0x6c, + 0x75, 0x65, 0x42, 0x07, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x22, 0xac, 0x01, 0x0a, 0x0e, + 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x12, 0x40, + 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, + 0x32, 0x24, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, + 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x2e, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, + 0x61, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, + 0x1a, 0x58, 0x0a, 0x0d, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x45, 0x6e, 0x74, 0x72, + 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, + 0x6b, 0x65, 0x79, 0x12, 0x31, 0x0a, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, + 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x52, + 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0xfb, 0x01, 0x0a, 0x15, 0x53, + 0x75, 0x62, 0x6d, 0x69, 0x74, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x65, + 0x63, 0x6f, 0x72, 0x64, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, + 0x52, 0x02, 0x69, 0x64, 0x12, 0x2b, 0x0a, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x18, 0x02, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, + 0x63, 0x74, 0x6f, 0x72, 0x48, 0x00, 0x52, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x88, 0x01, + 0x01, 0x12, 0x37, 0x0a, 0x08, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, + 0x61, 0x74, 0x65, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x48, 0x01, 0x52, 0x08, 0x6d, + 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x88, 0x01, 0x01, 0x12, 0x2f, 0x0a, 0x09, 0x6f, 0x70, + 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x11, 0x2e, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x4f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, + 0x52, 0x09, 0x6f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x23, 0x0a, 0x0d, 0x63, + 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x18, 0x05, 0x20, 0x01, + 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, + 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x42, 0x0b, 0x0a, 0x09, 0x5f, + 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x22, 0x66, 0x0a, 0x15, 0x56, 0x65, 0x63, 0x74, + 0x6f, 0x72, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, 0x72, + 0x64, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, + 0x64, 0x12, 0x15, 0x0a, 0x06, 0x73, 0x65, 0x71, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, + 0x0c, 0x52, 0x05, 0x73, 0x65, 0x71, 0x49, 0x64, 0x12, 0x26, 0x0a, 0x06, 0x76, 0x65, 0x63, 0x74, + 0x6f, 0x72, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x06, 0x76, 0x65, 0x63, 0x74, 0x6f, 0x72, + 0x22, 0x8e, 0x01, 0x0a, 0x11, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, 0x79, + 0x52, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, + 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x15, 0x0a, 0x06, 0x73, 0x65, 0x71, 0x5f, 0x69, 0x64, + 0x18, 0x02, 0x20, 0x01, 0x28, 0x0c, 0x52, 0x05, 0x73, 0x65, 0x71, 0x49, 0x64, 0x12, 0x1a, 0x0a, + 0x08, 0x64, 0x69, 0x73, 0x74, 0x61, 0x6e, 0x63, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x02, 0x52, + 0x08, 0x64, 0x69, 0x73, 0x74, 0x61, 0x6e, 0x63, 0x65, 0x12, 0x2b, 0x0a, 0x06, 0x76, 0x65, 0x63, + 0x74, 0x6f, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x48, 0x00, 0x52, 0x06, 0x76, 0x65, 0x63, + 0x74, 0x6f, 0x72, 0x88, 0x01, 0x01, 0x42, 0x09, 0x0a, 0x07, 0x5f, 0x76, 0x65, 0x63, 0x74, 0x6f, + 0x72, 0x22, 0x49, 0x0a, 0x12, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, 0x79, + 0x52, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x12, 0x33, 0x0a, 0x07, 0x72, 0x65, 0x73, 0x75, 0x6c, + 0x74, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x19, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x65, 0x73, + 0x75, 0x6c, 0x74, 0x52, 0x07, 0x72, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x22, 0x44, 0x0a, 0x11, + 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, + 0x74, 0x12, 0x10, 0x0a, 0x03, 0x69, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, 0x03, + 0x69, 0x64, 0x73, 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, 0x69, + 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, + 0x49, 0x64, 0x22, 0x4d, 0x0a, 0x12, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, + 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x37, 0x0a, 0x07, 0x72, 0x65, 0x63, 0x6f, + 0x72, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, + 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x52, 0x07, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, + 0x73, 0x22, 0xbc, 0x01, 0x0a, 0x13, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, + 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x28, 0x0a, 0x07, 0x76, 0x65, 0x63, + 0x74, 0x6f, 0x72, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, 0x72, + 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x07, 0x76, 0x65, 0x63, 0x74, + 0x6f, 0x72, 0x73, 0x12, 0x0c, 0x0a, 0x01, 0x6b, 0x18, 0x02, 0x20, 0x01, 0x28, 0x05, 0x52, 0x01, + 0x6b, 0x12, 0x1f, 0x0a, 0x0b, 0x61, 0x6c, 0x6c, 0x6f, 0x77, 0x65, 0x64, 0x5f, 0x69, 0x64, 0x73, + 0x18, 0x03, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x61, 0x6c, 0x6c, 0x6f, 0x77, 0x65, 0x64, 0x49, + 0x64, 0x73, 0x12, 0x2d, 0x0a, 0x12, 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, 0x65, 0x5f, 0x65, 0x6d, + 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x73, 0x18, 0x04, 0x20, 0x01, 0x28, 0x08, 0x52, 0x11, + 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, 0x65, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, + 0x73, 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, 0x69, 0x64, 0x18, + 0x05, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x49, 0x64, + 0x22, 0x4c, 0x0a, 0x14, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, + 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x34, 0x0a, 0x07, 0x72, 0x65, 0x73, 0x75, + 0x6c, 0x74, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x65, - 0x73, 0x75, 0x6c, 0x74, 0x52, 0x07, 0x72, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x22, 0x44, 0x0a, - 0x11, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, - 0x73, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x69, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, - 0x03, 0x69, 0x64, 0x73, 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, - 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, - 0x74, 0x49, 0x64, 0x22, 0x4d, 0x0a, 0x12, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, - 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x37, 0x0a, 0x07, 0x72, 0x65, 0x63, - 0x6f, 0x72, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1d, 0x2e, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, - 0x69, 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x52, 0x07, 0x72, 0x65, 0x63, 0x6f, 0x72, - 0x64, 0x73, 0x22, 0xbc, 0x01, 0x0a, 0x13, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, - 0x6f, 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x28, 0x0a, 0x07, 0x76, 0x65, - 0x63, 0x74, 0x6f, 0x72, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x0e, 0x2e, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x07, 0x76, 0x65, 0x63, - 0x74, 0x6f, 0x72, 0x73, 0x12, 0x0c, 0x0a, 0x01, 0x6b, 0x18, 0x02, 0x20, 0x01, 0x28, 0x05, 0x52, - 0x01, 0x6b, 0x12, 0x1f, 0x0a, 0x0b, 0x61, 0x6c, 0x6c, 0x6f, 0x77, 0x65, 0x64, 0x5f, 0x69, 0x64, - 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x61, 0x6c, 0x6c, 0x6f, 0x77, 0x65, 0x64, - 0x49, 0x64, 0x73, 0x12, 0x2d, 0x0a, 0x12, 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, 0x65, 0x5f, 0x65, - 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x73, 0x18, 0x04, 0x20, 0x01, 0x28, 0x08, 0x52, - 0x11, 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, 0x65, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, - 0x67, 0x73, 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, 0x69, 0x64, - 0x18, 0x05, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x49, - 0x64, 0x22, 0x4c, 0x0a, 0x14, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, - 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x34, 0x0a, 0x07, 0x72, 0x65, 0x73, - 0x75, 0x6c, 0x74, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2e, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, - 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x52, 0x07, 0x72, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x2a, - 0x38, 0x0a, 0x09, 0x4f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x07, 0x0a, 0x03, - 0x41, 0x44, 0x44, 0x10, 0x00, 0x12, 0x0a, 0x0a, 0x06, 0x55, 0x50, 0x44, 0x41, 0x54, 0x45, 0x10, - 0x01, 0x12, 0x0a, 0x0a, 0x06, 0x55, 0x50, 0x53, 0x45, 0x52, 0x54, 0x10, 0x02, 0x12, 0x0a, 0x0a, - 0x06, 0x44, 0x45, 0x4c, 0x45, 0x54, 0x45, 0x10, 0x03, 0x2a, 0x28, 0x0a, 0x0e, 0x53, 0x63, 0x61, - 0x6c, 0x61, 0x72, 0x45, 0x6e, 0x63, 0x6f, 0x64, 0x69, 0x6e, 0x67, 0x12, 0x0b, 0x0a, 0x07, 0x46, - 0x4c, 0x4f, 0x41, 0x54, 0x33, 0x32, 0x10, 0x00, 0x12, 0x09, 0x0a, 0x05, 0x49, 0x4e, 0x54, 0x33, - 0x32, 0x10, 0x01, 0x2a, 0x28, 0x0a, 0x0c, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x53, 0x63, - 0x6f, 0x70, 0x65, 0x12, 0x0a, 0x0a, 0x06, 0x56, 0x45, 0x43, 0x54, 0x4f, 0x52, 0x10, 0x00, 0x12, - 0x0c, 0x0a, 0x08, 0x4d, 0x45, 0x54, 0x41, 0x44, 0x41, 0x54, 0x41, 0x10, 0x01, 0x32, 0xa2, 0x01, - 0x0a, 0x0c, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x65, 0x61, 0x64, 0x65, 0x72, 0x12, 0x45, - 0x0a, 0x0a, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x12, 0x19, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, - 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1a, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, - 0x2e, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, - 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4b, 0x0a, 0x0c, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, - 0x63, 0x74, 0x6f, 0x72, 0x73, 0x12, 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x51, - 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, - 0x73, 0x74, 0x1a, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x51, 0x75, 0x65, 0x72, - 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, - 0x22, 0x00, 0x42, 0x3a, 0x5a, 0x38, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, - 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2f, 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, - 0x2f, 0x63, 0x6f, 0x6f, 0x72, 0x64, 0x69, 0x6e, 0x61, 0x74, 0x6f, 0x72, 0x70, 0x62, 0x62, 0x06, - 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x73, 0x75, 0x6c, 0x74, 0x73, 0x52, 0x07, 0x72, 0x65, 0x73, 0x75, 0x6c, 0x74, 0x73, 0x2a, 0x38, + 0x0a, 0x09, 0x4f, 0x70, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x07, 0x0a, 0x03, 0x41, + 0x44, 0x44, 0x10, 0x00, 0x12, 0x0a, 0x0a, 0x06, 0x55, 0x50, 0x44, 0x41, 0x54, 0x45, 0x10, 0x01, + 0x12, 0x0a, 0x0a, 0x06, 0x55, 0x50, 0x53, 0x45, 0x52, 0x54, 0x10, 0x02, 0x12, 0x0a, 0x0a, 0x06, + 0x44, 0x45, 0x4c, 0x45, 0x54, 0x45, 0x10, 0x03, 0x2a, 0x28, 0x0a, 0x0e, 0x53, 0x63, 0x61, 0x6c, + 0x61, 0x72, 0x45, 0x6e, 0x63, 0x6f, 0x64, 0x69, 0x6e, 0x67, 0x12, 0x0b, 0x0a, 0x07, 0x46, 0x4c, + 0x4f, 0x41, 0x54, 0x33, 0x32, 0x10, 0x00, 0x12, 0x09, 0x0a, 0x05, 0x49, 0x4e, 0x54, 0x33, 0x32, + 0x10, 0x01, 0x2a, 0x28, 0x0a, 0x0c, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x53, 0x63, 0x6f, + 0x70, 0x65, 0x12, 0x0a, 0x0a, 0x06, 0x56, 0x45, 0x43, 0x54, 0x4f, 0x52, 0x10, 0x00, 0x12, 0x0c, + 0x0a, 0x08, 0x4d, 0x45, 0x54, 0x41, 0x44, 0x41, 0x54, 0x41, 0x10, 0x01, 0x32, 0xa2, 0x01, 0x0a, + 0x0c, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x52, 0x65, 0x61, 0x64, 0x65, 0x72, 0x12, 0x45, 0x0a, + 0x0a, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x12, 0x19, 0x2e, 0x63, 0x68, + 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, + 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1a, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, + 0x47, 0x65, 0x74, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, + 0x73, 0x65, 0x22, 0x00, 0x12, 0x4b, 0x0a, 0x0c, 0x51, 0x75, 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, + 0x74, 0x6f, 0x72, 0x73, 0x12, 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x51, 0x75, + 0x65, 0x72, 0x79, 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, + 0x74, 0x1a, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x51, 0x75, 0x65, 0x72, 0x79, + 0x56, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, + 0x00, 0x42, 0x3a, 0x5a, 0x38, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2f, 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, + 0x63, 0x6f, 0x6f, 0x72, 0x64, 0x69, 0x6e, 0x61, 0x74, 0x6f, 0x72, 0x70, 0x62, 0x62, 0x06, 0x70, + 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( @@ -1380,54 +1466,58 @@ func file_chromadb_proto_chroma_proto_rawDescGZIP() []byte { } var file_chromadb_proto_chroma_proto_enumTypes = make([]protoimpl.EnumInfo, 3) -var file_chromadb_proto_chroma_proto_msgTypes = make([]protoimpl.MessageInfo, 17) +var file_chromadb_proto_chroma_proto_msgTypes = make([]protoimpl.MessageInfo, 19) var file_chromadb_proto_chroma_proto_goTypes = []interface{}{ (Operation)(0), // 0: chroma.Operation (ScalarEncoding)(0), // 1: chroma.ScalarEncoding (SegmentScope)(0), // 2: chroma.SegmentScope (*Status)(nil), // 3: chroma.Status (*Vector)(nil), // 4: chroma.Vector - (*Segment)(nil), // 5: chroma.Segment - (*Collection)(nil), // 6: chroma.Collection - (*Database)(nil), // 7: chroma.Database - (*Tenant)(nil), // 8: chroma.Tenant - (*UpdateMetadataValue)(nil), // 9: chroma.UpdateMetadataValue - (*UpdateMetadata)(nil), // 10: chroma.UpdateMetadata - (*SubmitEmbeddingRecord)(nil), // 11: chroma.SubmitEmbeddingRecord - (*VectorEmbeddingRecord)(nil), // 12: chroma.VectorEmbeddingRecord - (*VectorQueryResult)(nil), // 13: chroma.VectorQueryResult - (*VectorQueryResults)(nil), // 14: chroma.VectorQueryResults - (*GetVectorsRequest)(nil), // 15: chroma.GetVectorsRequest - (*GetVectorsResponse)(nil), // 16: chroma.GetVectorsResponse - (*QueryVectorsRequest)(nil), // 17: chroma.QueryVectorsRequest - (*QueryVectorsResponse)(nil), // 18: chroma.QueryVectorsResponse - nil, // 19: chroma.UpdateMetadata.MetadataEntry + (*FilePaths)(nil), // 5: chroma.FilePaths + (*Segment)(nil), // 6: chroma.Segment + (*Collection)(nil), // 7: chroma.Collection + (*Database)(nil), // 8: chroma.Database + (*Tenant)(nil), // 9: chroma.Tenant + (*UpdateMetadataValue)(nil), // 10: chroma.UpdateMetadataValue + (*UpdateMetadata)(nil), // 11: chroma.UpdateMetadata + (*SubmitEmbeddingRecord)(nil), // 12: chroma.SubmitEmbeddingRecord + (*VectorEmbeddingRecord)(nil), // 13: chroma.VectorEmbeddingRecord + (*VectorQueryResult)(nil), // 14: chroma.VectorQueryResult + (*VectorQueryResults)(nil), // 15: chroma.VectorQueryResults + (*GetVectorsRequest)(nil), // 16: chroma.GetVectorsRequest + (*GetVectorsResponse)(nil), // 17: chroma.GetVectorsResponse + (*QueryVectorsRequest)(nil), // 18: chroma.QueryVectorsRequest + (*QueryVectorsResponse)(nil), // 19: chroma.QueryVectorsResponse + nil, // 20: chroma.Segment.FilePathsEntry + nil, // 21: chroma.UpdateMetadata.MetadataEntry } var file_chromadb_proto_chroma_proto_depIdxs = []int32{ 1, // 0: chroma.Vector.encoding:type_name -> chroma.ScalarEncoding 2, // 1: chroma.Segment.scope:type_name -> chroma.SegmentScope - 10, // 2: chroma.Segment.metadata:type_name -> chroma.UpdateMetadata - 10, // 3: chroma.Collection.metadata:type_name -> chroma.UpdateMetadata - 19, // 4: chroma.UpdateMetadata.metadata:type_name -> chroma.UpdateMetadata.MetadataEntry - 4, // 5: chroma.SubmitEmbeddingRecord.vector:type_name -> chroma.Vector - 10, // 6: chroma.SubmitEmbeddingRecord.metadata:type_name -> chroma.UpdateMetadata - 0, // 7: chroma.SubmitEmbeddingRecord.operation:type_name -> chroma.Operation - 4, // 8: chroma.VectorEmbeddingRecord.vector:type_name -> chroma.Vector - 4, // 9: chroma.VectorQueryResult.vector:type_name -> chroma.Vector - 13, // 10: chroma.VectorQueryResults.results:type_name -> chroma.VectorQueryResult - 12, // 11: chroma.GetVectorsResponse.records:type_name -> chroma.VectorEmbeddingRecord - 4, // 12: chroma.QueryVectorsRequest.vectors:type_name -> chroma.Vector - 14, // 13: chroma.QueryVectorsResponse.results:type_name -> chroma.VectorQueryResults - 9, // 14: chroma.UpdateMetadata.MetadataEntry.value:type_name -> chroma.UpdateMetadataValue - 15, // 15: chroma.VectorReader.GetVectors:input_type -> chroma.GetVectorsRequest - 17, // 16: chroma.VectorReader.QueryVectors:input_type -> chroma.QueryVectorsRequest - 16, // 17: chroma.VectorReader.GetVectors:output_type -> chroma.GetVectorsResponse - 18, // 18: chroma.VectorReader.QueryVectors:output_type -> chroma.QueryVectorsResponse - 17, // [17:19] is the sub-list for method output_type - 15, // [15:17] is the sub-list for method input_type - 15, // [15:15] is the sub-list for extension type_name - 15, // [15:15] is the sub-list for extension extendee - 0, // [0:15] is the sub-list for field type_name + 11, // 2: chroma.Segment.metadata:type_name -> chroma.UpdateMetadata + 20, // 3: chroma.Segment.file_paths:type_name -> chroma.Segment.FilePathsEntry + 11, // 4: chroma.Collection.metadata:type_name -> chroma.UpdateMetadata + 21, // 5: chroma.UpdateMetadata.metadata:type_name -> chroma.UpdateMetadata.MetadataEntry + 4, // 6: chroma.SubmitEmbeddingRecord.vector:type_name -> chroma.Vector + 11, // 7: chroma.SubmitEmbeddingRecord.metadata:type_name -> chroma.UpdateMetadata + 0, // 8: chroma.SubmitEmbeddingRecord.operation:type_name -> chroma.Operation + 4, // 9: chroma.VectorEmbeddingRecord.vector:type_name -> chroma.Vector + 4, // 10: chroma.VectorQueryResult.vector:type_name -> chroma.Vector + 14, // 11: chroma.VectorQueryResults.results:type_name -> chroma.VectorQueryResult + 13, // 12: chroma.GetVectorsResponse.records:type_name -> chroma.VectorEmbeddingRecord + 4, // 13: chroma.QueryVectorsRequest.vectors:type_name -> chroma.Vector + 15, // 14: chroma.QueryVectorsResponse.results:type_name -> chroma.VectorQueryResults + 5, // 15: chroma.Segment.FilePathsEntry.value:type_name -> chroma.FilePaths + 10, // 16: chroma.UpdateMetadata.MetadataEntry.value:type_name -> chroma.UpdateMetadataValue + 16, // 17: chroma.VectorReader.GetVectors:input_type -> chroma.GetVectorsRequest + 18, // 18: chroma.VectorReader.QueryVectors:input_type -> chroma.QueryVectorsRequest + 17, // 19: chroma.VectorReader.GetVectors:output_type -> chroma.GetVectorsResponse + 19, // 20: chroma.VectorReader.QueryVectors:output_type -> chroma.QueryVectorsResponse + 19, // [19:21] is the sub-list for method output_type + 17, // [17:19] is the sub-list for method input_type + 17, // [17:17] is the sub-list for extension type_name + 17, // [17:17] is the sub-list for extension extendee + 0, // [0:17] is the sub-list for field type_name } func init() { file_chromadb_proto_chroma_proto_init() } @@ -1461,7 +1551,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*Segment); i { + switch v := v.(*FilePaths); i { case 0: return &v.state case 1: @@ -1473,7 +1563,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*Collection); i { + switch v := v.(*Segment); i { case 0: return &v.state case 1: @@ -1485,7 +1575,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*Database); i { + switch v := v.(*Collection); i { case 0: return &v.state case 1: @@ -1497,7 +1587,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[5].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*Tenant); i { + switch v := v.(*Database); i { case 0: return &v.state case 1: @@ -1509,7 +1599,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[6].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*UpdateMetadataValue); i { + switch v := v.(*Tenant); i { case 0: return &v.state case 1: @@ -1521,7 +1611,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[7].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*UpdateMetadata); i { + switch v := v.(*UpdateMetadataValue); i { case 0: return &v.state case 1: @@ -1533,7 +1623,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[8].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*SubmitEmbeddingRecord); i { + switch v := v.(*UpdateMetadata); i { case 0: return &v.state case 1: @@ -1545,7 +1635,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[9].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*VectorEmbeddingRecord); i { + switch v := v.(*SubmitEmbeddingRecord); i { case 0: return &v.state case 1: @@ -1557,7 +1647,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[10].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*VectorQueryResult); i { + switch v := v.(*VectorEmbeddingRecord); i { case 0: return &v.state case 1: @@ -1569,7 +1659,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[11].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*VectorQueryResults); i { + switch v := v.(*VectorQueryResult); i { case 0: return &v.state case 1: @@ -1581,7 +1671,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[12].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*GetVectorsRequest); i { + switch v := v.(*VectorQueryResults); i { case 0: return &v.state case 1: @@ -1593,7 +1683,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[13].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*GetVectorsResponse); i { + switch v := v.(*GetVectorsRequest); i { case 0: return &v.state case 1: @@ -1605,7 +1695,7 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[14].Exporter = func(v interface{}, i int) interface{} { - switch v := v.(*QueryVectorsRequest); i { + switch v := v.(*GetVectorsResponse); i { case 0: return &v.state case 1: @@ -1617,6 +1707,18 @@ func file_chromadb_proto_chroma_proto_init() { } } file_chromadb_proto_chroma_proto_msgTypes[15].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*QueryVectorsRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_chromadb_proto_chroma_proto_msgTypes[16].Exporter = func(v interface{}, i int) interface{} { switch v := v.(*QueryVectorsResponse); i { case 0: return &v.state @@ -1629,22 +1731,22 @@ func file_chromadb_proto_chroma_proto_init() { } } } - file_chromadb_proto_chroma_proto_msgTypes[2].OneofWrappers = []interface{}{} file_chromadb_proto_chroma_proto_msgTypes[3].OneofWrappers = []interface{}{} - file_chromadb_proto_chroma_proto_msgTypes[6].OneofWrappers = []interface{}{ + file_chromadb_proto_chroma_proto_msgTypes[4].OneofWrappers = []interface{}{} + file_chromadb_proto_chroma_proto_msgTypes[7].OneofWrappers = []interface{}{ (*UpdateMetadataValue_StringValue)(nil), (*UpdateMetadataValue_IntValue)(nil), (*UpdateMetadataValue_FloatValue)(nil), } - file_chromadb_proto_chroma_proto_msgTypes[8].OneofWrappers = []interface{}{} - file_chromadb_proto_chroma_proto_msgTypes[10].OneofWrappers = []interface{}{} + file_chromadb_proto_chroma_proto_msgTypes[9].OneofWrappers = []interface{}{} + file_chromadb_proto_chroma_proto_msgTypes[11].OneofWrappers = []interface{}{} type x struct{} out := protoimpl.TypeBuilder{ File: protoimpl.DescBuilder{ GoPackagePath: reflect.TypeOf(x{}).PkgPath(), RawDescriptor: file_chromadb_proto_chroma_proto_rawDesc, NumEnums: 3, - NumMessages: 17, + NumMessages: 19, NumExtensions: 0, NumServices: 1, }, diff --git a/go/pkg/proto/coordinatorpb/chroma_grpc.pb.go b/go/pkg/proto/coordinatorpb/chroma_grpc.pb.go index 0b45e03517f..09283123121 100644 --- a/go/pkg/proto/coordinatorpb/chroma_grpc.pb.go +++ b/go/pkg/proto/coordinatorpb/chroma_grpc.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go-grpc. DO NOT EDIT. // versions: -// - protoc-gen-go-grpc v1.2.0 -// - protoc v4.25.3 +// - protoc-gen-go-grpc v1.3.0 +// - protoc v4.23.4 // source: chromadb/proto/chroma.proto package coordinatorpb @@ -18,6 +18,11 @@ import ( // Requires gRPC-Go v1.32.0 or later. const _ = grpc.SupportPackageIsVersion7 +const ( + VectorReader_GetVectors_FullMethodName = "/chroma.VectorReader/GetVectors" + VectorReader_QueryVectors_FullMethodName = "/chroma.VectorReader/QueryVectors" +) + // VectorReaderClient is the client API for VectorReader service. // // For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. @@ -36,7 +41,7 @@ func NewVectorReaderClient(cc grpc.ClientConnInterface) VectorReaderClient { func (c *vectorReaderClient) GetVectors(ctx context.Context, in *GetVectorsRequest, opts ...grpc.CallOption) (*GetVectorsResponse, error) { out := new(GetVectorsResponse) - err := c.cc.Invoke(ctx, "/chroma.VectorReader/GetVectors", in, out, opts...) + err := c.cc.Invoke(ctx, VectorReader_GetVectors_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -45,7 +50,7 @@ func (c *vectorReaderClient) GetVectors(ctx context.Context, in *GetVectorsReque func (c *vectorReaderClient) QueryVectors(ctx context.Context, in *QueryVectorsRequest, opts ...grpc.CallOption) (*QueryVectorsResponse, error) { out := new(QueryVectorsResponse) - err := c.cc.Invoke(ctx, "/chroma.VectorReader/QueryVectors", in, out, opts...) + err := c.cc.Invoke(ctx, VectorReader_QueryVectors_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -94,7 +99,7 @@ func _VectorReader_GetVectors_Handler(srv interface{}, ctx context.Context, dec } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.VectorReader/GetVectors", + FullMethod: VectorReader_GetVectors_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(VectorReaderServer).GetVectors(ctx, req.(*GetVectorsRequest)) @@ -112,7 +117,7 @@ func _VectorReader_QueryVectors_Handler(srv interface{}, ctx context.Context, de } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.VectorReader/QueryVectors", + FullMethod: VectorReader_QueryVectors_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(VectorReaderServer).QueryVectors(ctx, req.(*QueryVectorsRequest)) diff --git a/go/pkg/proto/coordinatorpb/coordinator.pb.go b/go/pkg/proto/coordinatorpb/coordinator.pb.go index 5ca8bce37d4..6995099b9d1 100644 --- a/go/pkg/proto/coordinatorpb/coordinator.pb.go +++ b/go/pkg/proto/coordinatorpb/coordinator.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go. DO NOT EDIT. // versions: -// protoc-gen-go v1.32.0 -// protoc v4.25.3 +// protoc-gen-go v1.31.0 +// protoc v4.23.4 // source: chromadb/proto/coordinator.proto package coordinatorpb @@ -1855,6 +1855,203 @@ func (x *SetLastCompactionTimeForTenantRequest) GetTenantLastCompactionTime() *T return nil } +type FlushSegmentCompactionInfo struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + SegmentId string `protobuf:"bytes,1,opt,name=segment_id,json=segmentId,proto3" json:"segment_id,omitempty"` + FilePaths map[string]*FilePaths `protobuf:"bytes,2,rep,name=file_paths,json=filePaths,proto3" json:"file_paths,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (x *FlushSegmentCompactionInfo) Reset() { + *x = FlushSegmentCompactionInfo{} + if protoimpl.UnsafeEnabled { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[30] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *FlushSegmentCompactionInfo) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*FlushSegmentCompactionInfo) ProtoMessage() {} + +func (x *FlushSegmentCompactionInfo) ProtoReflect() protoreflect.Message { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[30] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use FlushSegmentCompactionInfo.ProtoReflect.Descriptor instead. +func (*FlushSegmentCompactionInfo) Descriptor() ([]byte, []int) { + return file_chromadb_proto_coordinator_proto_rawDescGZIP(), []int{30} +} + +func (x *FlushSegmentCompactionInfo) GetSegmentId() string { + if x != nil { + return x.SegmentId + } + return "" +} + +func (x *FlushSegmentCompactionInfo) GetFilePaths() map[string]*FilePaths { + if x != nil { + return x.FilePaths + } + return nil +} + +type FlushCollectionCompactionRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + TenantId string `protobuf:"bytes,1,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + CollectionId string `protobuf:"bytes,2,opt,name=collection_id,json=collectionId,proto3" json:"collection_id,omitempty"` + LogPosition int64 `protobuf:"varint,3,opt,name=log_position,json=logPosition,proto3" json:"log_position,omitempty"` + CollectionVersion int32 `protobuf:"varint,4,opt,name=collection_version,json=collectionVersion,proto3" json:"collection_version,omitempty"` + SegmentCompactionInfo []*FlushSegmentCompactionInfo `protobuf:"bytes,5,rep,name=segment_compaction_info,json=segmentCompactionInfo,proto3" json:"segment_compaction_info,omitempty"` +} + +func (x *FlushCollectionCompactionRequest) Reset() { + *x = FlushCollectionCompactionRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[31] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *FlushCollectionCompactionRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*FlushCollectionCompactionRequest) ProtoMessage() {} + +func (x *FlushCollectionCompactionRequest) ProtoReflect() protoreflect.Message { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[31] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use FlushCollectionCompactionRequest.ProtoReflect.Descriptor instead. +func (*FlushCollectionCompactionRequest) Descriptor() ([]byte, []int) { + return file_chromadb_proto_coordinator_proto_rawDescGZIP(), []int{31} +} + +func (x *FlushCollectionCompactionRequest) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +func (x *FlushCollectionCompactionRequest) GetCollectionId() string { + if x != nil { + return x.CollectionId + } + return "" +} + +func (x *FlushCollectionCompactionRequest) GetLogPosition() int64 { + if x != nil { + return x.LogPosition + } + return 0 +} + +func (x *FlushCollectionCompactionRequest) GetCollectionVersion() int32 { + if x != nil { + return x.CollectionVersion + } + return 0 +} + +func (x *FlushCollectionCompactionRequest) GetSegmentCompactionInfo() []*FlushSegmentCompactionInfo { + if x != nil { + return x.SegmentCompactionInfo + } + return nil +} + +type FlushCollectionCompactionResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + CollectionId string `protobuf:"bytes,1,opt,name=collection_id,json=collectionId,proto3" json:"collection_id,omitempty"` + CollectionVersion int32 `protobuf:"varint,2,opt,name=collection_version,json=collectionVersion,proto3" json:"collection_version,omitempty"` + LastCompactionTime int64 `protobuf:"varint,3,opt,name=last_compaction_time,json=lastCompactionTime,proto3" json:"last_compaction_time,omitempty"` +} + +func (x *FlushCollectionCompactionResponse) Reset() { + *x = FlushCollectionCompactionResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[32] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *FlushCollectionCompactionResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*FlushCollectionCompactionResponse) ProtoMessage() {} + +func (x *FlushCollectionCompactionResponse) ProtoReflect() protoreflect.Message { + mi := &file_chromadb_proto_coordinator_proto_msgTypes[32] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use FlushCollectionCompactionResponse.ProtoReflect.Descriptor instead. +func (*FlushCollectionCompactionResponse) Descriptor() ([]byte, []int) { + return file_chromadb_proto_coordinator_proto_rawDescGZIP(), []int{32} +} + +func (x *FlushCollectionCompactionResponse) GetCollectionId() string { + if x != nil { + return x.CollectionId + } + return "" +} + +func (x *FlushCollectionCompactionResponse) GetCollectionVersion() int32 { + if x != nil { + return x.CollectionVersion + } + return 0 +} + +func (x *FlushCollectionCompactionResponse) GetLastCompactionTime() int64 { + if x != nil { + return x.LastCompactionTime + } + return 0 +} + var File_chromadb_proto_coordinator_proto protoreflect.FileDescriptor var file_chromadb_proto_coordinator_proto_rawDesc = []byte{ @@ -2078,91 +2275,141 @@ var file_chromadb_proto_coordinator_proto_rawDesc = []byte{ 0x6d, 0x61, 0x2e, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x52, 0x18, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, - 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x32, 0x80, 0x0a, 0x0a, 0x05, 0x53, 0x79, 0x73, 0x44, 0x42, 0x12, - 0x51, 0x0a, 0x0e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, - 0x65, 0x12, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, - 0x65, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, - 0x1a, 0x1e, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, - 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, - 0x22, 0x00, 0x12, 0x48, 0x0a, 0x0b, 0x47, 0x65, 0x74, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, - 0x65, 0x12, 0x1a, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x44, 0x61, - 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1b, 0x2e, - 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, - 0x73, 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4b, 0x0a, 0x0c, - 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x1b, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, 0x6e, 0x61, - 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, - 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, - 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x42, 0x0a, 0x09, 0x47, 0x65, 0x74, - 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, - 0x47, 0x65, 0x74, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, - 0x1a, 0x19, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x54, 0x65, 0x6e, - 0x61, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4e, 0x0a, - 0x0d, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x1c, + 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x22, 0xde, 0x01, 0x0a, 0x1a, 0x46, 0x6c, 0x75, 0x73, 0x68, 0x53, + 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, + 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, + 0x74, 0x49, 0x64, 0x12, 0x50, 0x0a, 0x0a, 0x66, 0x69, 0x6c, 0x65, 0x5f, 0x70, 0x61, 0x74, 0x68, + 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x31, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, + 0x2e, 0x46, 0x6c, 0x75, 0x73, 0x68, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x43, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x2e, 0x46, 0x69, 0x6c, 0x65, + 0x50, 0x61, 0x74, 0x68, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x09, 0x66, 0x69, 0x6c, 0x65, + 0x50, 0x61, 0x74, 0x68, 0x73, 0x1a, 0x4f, 0x0a, 0x0e, 0x46, 0x69, 0x6c, 0x65, 0x50, 0x61, 0x74, + 0x68, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x27, 0x0a, 0x05, 0x76, 0x61, 0x6c, + 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x46, 0x69, 0x6c, 0x65, 0x50, 0x61, 0x74, 0x68, 0x73, 0x52, 0x05, 0x76, 0x61, 0x6c, + 0x75, 0x65, 0x3a, 0x02, 0x38, 0x01, 0x22, 0x92, 0x02, 0x0a, 0x20, 0x46, 0x6c, 0x75, 0x73, 0x68, + 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x1b, 0x0a, 0x09, 0x74, + 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, + 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, 0x64, 0x12, 0x23, 0x0a, 0x0d, 0x63, 0x6f, 0x6c, 0x6c, + 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, + 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, 0x12, 0x21, 0x0a, + 0x0c, 0x6c, 0x6f, 0x67, 0x5f, 0x70, 0x6f, 0x73, 0x69, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x03, 0x52, 0x0b, 0x6c, 0x6f, 0x67, 0x50, 0x6f, 0x73, 0x69, 0x74, 0x69, 0x6f, 0x6e, + 0x12, 0x2d, 0x0a, 0x12, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x76, + 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x04, 0x20, 0x01, 0x28, 0x05, 0x52, 0x11, 0x63, 0x6f, + 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x12, + 0x5a, 0x0a, 0x17, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x5f, 0x63, 0x6f, 0x6d, 0x70, 0x61, + 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x6e, 0x66, 0x6f, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, + 0x32, 0x22, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x46, 0x6c, 0x75, 0x73, 0x68, 0x53, + 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x15, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x43, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x22, 0xa9, 0x01, 0x0a, 0x21, + 0x46, 0x6c, 0x75, 0x73, 0x68, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x43, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, + 0x65, 0x12, 0x23, 0x0a, 0x0d, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, + 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, 0x12, 0x2d, 0x0a, 0x12, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x05, 0x52, 0x11, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x56, 0x65, + 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x12, 0x30, 0x0a, 0x14, 0x6c, 0x61, 0x73, 0x74, 0x5f, 0x63, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x03, 0x52, 0x12, 0x6c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x32, 0xf4, 0x0a, 0x0a, 0x05, 0x53, 0x79, 0x73, 0x44, + 0x42, 0x12, 0x51, 0x0a, 0x0e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x44, 0x61, 0x74, 0x61, 0x62, + 0x61, 0x73, 0x65, 0x12, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, + 0x61, 0x74, 0x65, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x71, 0x75, 0x65, + 0x73, 0x74, 0x1a, 0x1e, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, + 0x74, 0x65, 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, + 0x73, 0x65, 0x22, 0x00, 0x12, 0x48, 0x0a, 0x0b, 0x47, 0x65, 0x74, 0x44, 0x61, 0x74, 0x61, 0x62, + 0x61, 0x73, 0x65, 0x12, 0x1a, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, + 0x44, 0x61, 0x74, 0x61, 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, + 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x44, 0x61, 0x74, 0x61, + 0x62, 0x61, 0x73, 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4b, + 0x0a, 0x0c, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x1b, + 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, + 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1c, 0x2e, 0x63, 0x68, + 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x54, 0x65, 0x6e, 0x61, 0x6e, + 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x42, 0x0a, 0x09, 0x47, + 0x65, 0x74, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x47, 0x65, 0x74, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, + 0x73, 0x74, 0x1a, 0x19, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x54, + 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, + 0x4e, 0x0a, 0x0d, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, + 0x12, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, + 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x53, 0x65, - 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, - 0x65, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4e, 0x0a, - 0x0d, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x1c, + 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, + 0x4e, 0x0a, 0x0d, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, + 0x12, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, + 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x53, 0x65, - 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, - 0x65, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x48, 0x0a, - 0x0b, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x73, 0x12, 0x1a, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, - 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1b, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, - 0x61, 0x2e, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x73, 0x52, 0x65, 0x73, - 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4e, 0x0a, 0x0d, 0x55, 0x70, 0x64, 0x61, 0x74, - 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x1c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, + 0x48, 0x0a, 0x0b, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x73, 0x12, 0x1a, + 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, + 0x6e, 0x74, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1b, 0x2e, 0x63, 0x68, 0x72, + 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x73, 0x52, + 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4e, 0x0a, 0x0d, 0x55, 0x70, 0x64, + 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x12, 0x1c, 0x2e, 0x63, 0x68, 0x72, + 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, + 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, - 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, - 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x52, 0x65, 0x73, - 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x57, 0x0a, 0x10, 0x43, 0x72, 0x65, 0x61, 0x74, - 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x1f, 0x2e, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, - 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x20, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, - 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, - 0x12, 0x57, 0x0a, 0x10, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, - 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x1f, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x44, 0x65, - 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, - 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x20, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x44, - 0x65, 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, - 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x51, 0x0a, 0x0e, 0x47, 0x65, 0x74, - 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x1d, 0x2e, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, - 0x6f, 0x6e, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1e, 0x2e, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, - 0x6e, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x57, 0x0a, 0x10, - 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, - 0x12, 0x1f, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, - 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, - 0x74, 0x1a, 0x20, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, 0x74, - 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, - 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x42, 0x0a, 0x0a, 0x52, 0x65, 0x73, 0x65, 0x74, 0x53, 0x74, - 0x61, 0x74, 0x65, 0x12, 0x16, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, - 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x1a, 0x1a, 0x2e, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x52, 0x65, 0x73, 0x65, 0x74, 0x53, 0x74, 0x61, 0x74, 0x65, 0x52, - 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x81, 0x01, 0x0a, 0x1e, 0x47, 0x65, - 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, - 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x2d, 0x2e, 0x63, - 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, - 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, - 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x2e, 0x2e, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, - 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, - 0x61, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x69, 0x0a, - 0x1e, 0x53, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, - 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, - 0x2d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, - 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, - 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x16, - 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, - 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x22, 0x00, 0x42, 0x3a, 0x5a, 0x38, 0x67, 0x69, 0x74, 0x68, - 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, - 0x72, 0x65, 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2f, 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, - 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x63, 0x6f, 0x6f, 0x72, 0x64, 0x69, 0x6e, 0x61, 0x74, - 0x6f, 0x72, 0x70, 0x62, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, + 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x57, 0x0a, 0x10, 0x43, 0x72, 0x65, + 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x1f, 0x2e, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, + 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x20, + 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x43, 0x6f, + 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, + 0x22, 0x00, 0x12, 0x57, 0x0a, 0x10, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, + 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x1f, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, + 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x20, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, + 0x2e, 0x44, 0x65, 0x6c, 0x65, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x51, 0x0a, 0x0e, 0x47, + 0x65, 0x74, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x1d, 0x2e, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1e, 0x2e, 0x63, + 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x57, + 0x0a, 0x10, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, + 0x6f, 0x6e, 0x12, 0x1f, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, 0x61, + 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, + 0x65, 0x73, 0x74, 0x1a, 0x20, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x55, 0x70, 0x64, + 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, + 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x42, 0x0a, 0x0a, 0x52, 0x65, 0x73, 0x65, 0x74, + 0x53, 0x74, 0x61, 0x74, 0x65, 0x12, 0x16, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, + 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x1a, 0x1a, 0x2e, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x52, 0x65, 0x73, 0x65, 0x74, 0x53, 0x74, 0x61, 0x74, + 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x81, 0x01, 0x0a, 0x1e, + 0x47, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x2d, + 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, + 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x2e, 0x2e, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, + 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, + 0x69, 0x0a, 0x1e, 0x53, 0x65, 0x74, 0x4c, 0x61, 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, + 0x74, 0x12, 0x2d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x65, 0x74, 0x4c, 0x61, + 0x73, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x54, 0x69, 0x6d, 0x65, + 0x46, 0x6f, 0x72, 0x54, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, + 0x1a, 0x16, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, + 0x75, 0x66, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x22, 0x00, 0x12, 0x72, 0x0a, 0x19, 0x46, 0x6c, + 0x75, 0x73, 0x68, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x43, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x12, 0x28, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, + 0x2e, 0x46, 0x6c, 0x75, 0x73, 0x68, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, + 0x74, 0x1a, 0x29, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x46, 0x6c, 0x75, 0x73, 0x68, + 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x3a, + 0x5a, 0x38, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x63, 0x68, 0x72, + 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2f, + 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x63, 0x6f, 0x6f, + 0x72, 0x64, 0x69, 0x6e, 0x61, 0x74, 0x6f, 0x72, 0x70, 0x62, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, + 0x6f, 0x33, } var ( @@ -2177,7 +2424,7 @@ func file_chromadb_proto_coordinator_proto_rawDescGZIP() []byte { return file_chromadb_proto_coordinator_proto_rawDescData } -var file_chromadb_proto_coordinator_proto_msgTypes = make([]protoimpl.MessageInfo, 30) +var file_chromadb_proto_coordinator_proto_msgTypes = make([]protoimpl.MessageInfo, 34) var file_chromadb_proto_coordinator_proto_goTypes = []interface{}{ (*CreateDatabaseRequest)(nil), // 0: chroma.CreateDatabaseRequest (*CreateDatabaseResponse)(nil), // 1: chroma.CreateDatabaseResponse @@ -2209,76 +2456,86 @@ var file_chromadb_proto_coordinator_proto_goTypes = []interface{}{ (*TenantLastCompactionTime)(nil), // 27: chroma.TenantLastCompactionTime (*GetLastCompactionTimeForTenantResponse)(nil), // 28: chroma.GetLastCompactionTimeForTenantResponse (*SetLastCompactionTimeForTenantRequest)(nil), // 29: chroma.SetLastCompactionTimeForTenantRequest - (*Status)(nil), // 30: chroma.Status - (*Database)(nil), // 31: chroma.Database - (*Tenant)(nil), // 32: chroma.Tenant - (*Segment)(nil), // 33: chroma.Segment - (SegmentScope)(0), // 34: chroma.SegmentScope - (*UpdateMetadata)(nil), // 35: chroma.UpdateMetadata - (*Collection)(nil), // 36: chroma.Collection - (*emptypb.Empty)(nil), // 37: google.protobuf.Empty + (*FlushSegmentCompactionInfo)(nil), // 30: chroma.FlushSegmentCompactionInfo + (*FlushCollectionCompactionRequest)(nil), // 31: chroma.FlushCollectionCompactionRequest + (*FlushCollectionCompactionResponse)(nil), // 32: chroma.FlushCollectionCompactionResponse + nil, // 33: chroma.FlushSegmentCompactionInfo.FilePathsEntry + (*Status)(nil), // 34: chroma.Status + (*Database)(nil), // 35: chroma.Database + (*Tenant)(nil), // 36: chroma.Tenant + (*Segment)(nil), // 37: chroma.Segment + (SegmentScope)(0), // 38: chroma.SegmentScope + (*UpdateMetadata)(nil), // 39: chroma.UpdateMetadata + (*Collection)(nil), // 40: chroma.Collection + (*FilePaths)(nil), // 41: chroma.FilePaths + (*emptypb.Empty)(nil), // 42: google.protobuf.Empty } var file_chromadb_proto_coordinator_proto_depIdxs = []int32{ - 30, // 0: chroma.CreateDatabaseResponse.status:type_name -> chroma.Status - 31, // 1: chroma.GetDatabaseResponse.database:type_name -> chroma.Database - 30, // 2: chroma.GetDatabaseResponse.status:type_name -> chroma.Status - 30, // 3: chroma.CreateTenantResponse.status:type_name -> chroma.Status - 32, // 4: chroma.GetTenantResponse.tenant:type_name -> chroma.Tenant - 30, // 5: chroma.GetTenantResponse.status:type_name -> chroma.Status - 33, // 6: chroma.CreateSegmentRequest.segment:type_name -> chroma.Segment - 30, // 7: chroma.CreateSegmentResponse.status:type_name -> chroma.Status - 30, // 8: chroma.DeleteSegmentResponse.status:type_name -> chroma.Status - 34, // 9: chroma.GetSegmentsRequest.scope:type_name -> chroma.SegmentScope - 33, // 10: chroma.GetSegmentsResponse.segments:type_name -> chroma.Segment - 30, // 11: chroma.GetSegmentsResponse.status:type_name -> chroma.Status - 35, // 12: chroma.UpdateSegmentRequest.metadata:type_name -> chroma.UpdateMetadata - 30, // 13: chroma.UpdateSegmentResponse.status:type_name -> chroma.Status - 35, // 14: chroma.CreateCollectionRequest.metadata:type_name -> chroma.UpdateMetadata - 36, // 15: chroma.CreateCollectionResponse.collection:type_name -> chroma.Collection - 30, // 16: chroma.CreateCollectionResponse.status:type_name -> chroma.Status - 30, // 17: chroma.DeleteCollectionResponse.status:type_name -> chroma.Status - 36, // 18: chroma.GetCollectionsResponse.collections:type_name -> chroma.Collection - 30, // 19: chroma.GetCollectionsResponse.status:type_name -> chroma.Status - 35, // 20: chroma.UpdateCollectionRequest.metadata:type_name -> chroma.UpdateMetadata - 30, // 21: chroma.UpdateCollectionResponse.status:type_name -> chroma.Status - 30, // 22: chroma.ResetStateResponse.status:type_name -> chroma.Status + 34, // 0: chroma.CreateDatabaseResponse.status:type_name -> chroma.Status + 35, // 1: chroma.GetDatabaseResponse.database:type_name -> chroma.Database + 34, // 2: chroma.GetDatabaseResponse.status:type_name -> chroma.Status + 34, // 3: chroma.CreateTenantResponse.status:type_name -> chroma.Status + 36, // 4: chroma.GetTenantResponse.tenant:type_name -> chroma.Tenant + 34, // 5: chroma.GetTenantResponse.status:type_name -> chroma.Status + 37, // 6: chroma.CreateSegmentRequest.segment:type_name -> chroma.Segment + 34, // 7: chroma.CreateSegmentResponse.status:type_name -> chroma.Status + 34, // 8: chroma.DeleteSegmentResponse.status:type_name -> chroma.Status + 38, // 9: chroma.GetSegmentsRequest.scope:type_name -> chroma.SegmentScope + 37, // 10: chroma.GetSegmentsResponse.segments:type_name -> chroma.Segment + 34, // 11: chroma.GetSegmentsResponse.status:type_name -> chroma.Status + 39, // 12: chroma.UpdateSegmentRequest.metadata:type_name -> chroma.UpdateMetadata + 34, // 13: chroma.UpdateSegmentResponse.status:type_name -> chroma.Status + 39, // 14: chroma.CreateCollectionRequest.metadata:type_name -> chroma.UpdateMetadata + 40, // 15: chroma.CreateCollectionResponse.collection:type_name -> chroma.Collection + 34, // 16: chroma.CreateCollectionResponse.status:type_name -> chroma.Status + 34, // 17: chroma.DeleteCollectionResponse.status:type_name -> chroma.Status + 40, // 18: chroma.GetCollectionsResponse.collections:type_name -> chroma.Collection + 34, // 19: chroma.GetCollectionsResponse.status:type_name -> chroma.Status + 39, // 20: chroma.UpdateCollectionRequest.metadata:type_name -> chroma.UpdateMetadata + 34, // 21: chroma.UpdateCollectionResponse.status:type_name -> chroma.Status + 34, // 22: chroma.ResetStateResponse.status:type_name -> chroma.Status 27, // 23: chroma.GetLastCompactionTimeForTenantResponse.tenant_last_compaction_time:type_name -> chroma.TenantLastCompactionTime 27, // 24: chroma.SetLastCompactionTimeForTenantRequest.tenant_last_compaction_time:type_name -> chroma.TenantLastCompactionTime - 0, // 25: chroma.SysDB.CreateDatabase:input_type -> chroma.CreateDatabaseRequest - 2, // 26: chroma.SysDB.GetDatabase:input_type -> chroma.GetDatabaseRequest - 4, // 27: chroma.SysDB.CreateTenant:input_type -> chroma.CreateTenantRequest - 6, // 28: chroma.SysDB.GetTenant:input_type -> chroma.GetTenantRequest - 8, // 29: chroma.SysDB.CreateSegment:input_type -> chroma.CreateSegmentRequest - 10, // 30: chroma.SysDB.DeleteSegment:input_type -> chroma.DeleteSegmentRequest - 12, // 31: chroma.SysDB.GetSegments:input_type -> chroma.GetSegmentsRequest - 14, // 32: chroma.SysDB.UpdateSegment:input_type -> chroma.UpdateSegmentRequest - 16, // 33: chroma.SysDB.CreateCollection:input_type -> chroma.CreateCollectionRequest - 18, // 34: chroma.SysDB.DeleteCollection:input_type -> chroma.DeleteCollectionRequest - 20, // 35: chroma.SysDB.GetCollections:input_type -> chroma.GetCollectionsRequest - 22, // 36: chroma.SysDB.UpdateCollection:input_type -> chroma.UpdateCollectionRequest - 37, // 37: chroma.SysDB.ResetState:input_type -> google.protobuf.Empty - 26, // 38: chroma.SysDB.GetLastCompactionTimeForTenant:input_type -> chroma.GetLastCompactionTimeForTenantRequest - 29, // 39: chroma.SysDB.SetLastCompactionTimeForTenant:input_type -> chroma.SetLastCompactionTimeForTenantRequest - 1, // 40: chroma.SysDB.CreateDatabase:output_type -> chroma.CreateDatabaseResponse - 3, // 41: chroma.SysDB.GetDatabase:output_type -> chroma.GetDatabaseResponse - 5, // 42: chroma.SysDB.CreateTenant:output_type -> chroma.CreateTenantResponse - 7, // 43: chroma.SysDB.GetTenant:output_type -> chroma.GetTenantResponse - 9, // 44: chroma.SysDB.CreateSegment:output_type -> chroma.CreateSegmentResponse - 11, // 45: chroma.SysDB.DeleteSegment:output_type -> chroma.DeleteSegmentResponse - 13, // 46: chroma.SysDB.GetSegments:output_type -> chroma.GetSegmentsResponse - 15, // 47: chroma.SysDB.UpdateSegment:output_type -> chroma.UpdateSegmentResponse - 17, // 48: chroma.SysDB.CreateCollection:output_type -> chroma.CreateCollectionResponse - 19, // 49: chroma.SysDB.DeleteCollection:output_type -> chroma.DeleteCollectionResponse - 21, // 50: chroma.SysDB.GetCollections:output_type -> chroma.GetCollectionsResponse - 23, // 51: chroma.SysDB.UpdateCollection:output_type -> chroma.UpdateCollectionResponse - 25, // 52: chroma.SysDB.ResetState:output_type -> chroma.ResetStateResponse - 28, // 53: chroma.SysDB.GetLastCompactionTimeForTenant:output_type -> chroma.GetLastCompactionTimeForTenantResponse - 37, // 54: chroma.SysDB.SetLastCompactionTimeForTenant:output_type -> google.protobuf.Empty - 40, // [40:55] is the sub-list for method output_type - 25, // [25:40] is the sub-list for method input_type - 25, // [25:25] is the sub-list for extension type_name - 25, // [25:25] is the sub-list for extension extendee - 0, // [0:25] is the sub-list for field type_name + 33, // 25: chroma.FlushSegmentCompactionInfo.file_paths:type_name -> chroma.FlushSegmentCompactionInfo.FilePathsEntry + 30, // 26: chroma.FlushCollectionCompactionRequest.segment_compaction_info:type_name -> chroma.FlushSegmentCompactionInfo + 41, // 27: chroma.FlushSegmentCompactionInfo.FilePathsEntry.value:type_name -> chroma.FilePaths + 0, // 28: chroma.SysDB.CreateDatabase:input_type -> chroma.CreateDatabaseRequest + 2, // 29: chroma.SysDB.GetDatabase:input_type -> chroma.GetDatabaseRequest + 4, // 30: chroma.SysDB.CreateTenant:input_type -> chroma.CreateTenantRequest + 6, // 31: chroma.SysDB.GetTenant:input_type -> chroma.GetTenantRequest + 8, // 32: chroma.SysDB.CreateSegment:input_type -> chroma.CreateSegmentRequest + 10, // 33: chroma.SysDB.DeleteSegment:input_type -> chroma.DeleteSegmentRequest + 12, // 34: chroma.SysDB.GetSegments:input_type -> chroma.GetSegmentsRequest + 14, // 35: chroma.SysDB.UpdateSegment:input_type -> chroma.UpdateSegmentRequest + 16, // 36: chroma.SysDB.CreateCollection:input_type -> chroma.CreateCollectionRequest + 18, // 37: chroma.SysDB.DeleteCollection:input_type -> chroma.DeleteCollectionRequest + 20, // 38: chroma.SysDB.GetCollections:input_type -> chroma.GetCollectionsRequest + 22, // 39: chroma.SysDB.UpdateCollection:input_type -> chroma.UpdateCollectionRequest + 42, // 40: chroma.SysDB.ResetState:input_type -> google.protobuf.Empty + 26, // 41: chroma.SysDB.GetLastCompactionTimeForTenant:input_type -> chroma.GetLastCompactionTimeForTenantRequest + 29, // 42: chroma.SysDB.SetLastCompactionTimeForTenant:input_type -> chroma.SetLastCompactionTimeForTenantRequest + 31, // 43: chroma.SysDB.FlushCollectionCompaction:input_type -> chroma.FlushCollectionCompactionRequest + 1, // 44: chroma.SysDB.CreateDatabase:output_type -> chroma.CreateDatabaseResponse + 3, // 45: chroma.SysDB.GetDatabase:output_type -> chroma.GetDatabaseResponse + 5, // 46: chroma.SysDB.CreateTenant:output_type -> chroma.CreateTenantResponse + 7, // 47: chroma.SysDB.GetTenant:output_type -> chroma.GetTenantResponse + 9, // 48: chroma.SysDB.CreateSegment:output_type -> chroma.CreateSegmentResponse + 11, // 49: chroma.SysDB.DeleteSegment:output_type -> chroma.DeleteSegmentResponse + 13, // 50: chroma.SysDB.GetSegments:output_type -> chroma.GetSegmentsResponse + 15, // 51: chroma.SysDB.UpdateSegment:output_type -> chroma.UpdateSegmentResponse + 17, // 52: chroma.SysDB.CreateCollection:output_type -> chroma.CreateCollectionResponse + 19, // 53: chroma.SysDB.DeleteCollection:output_type -> chroma.DeleteCollectionResponse + 21, // 54: chroma.SysDB.GetCollections:output_type -> chroma.GetCollectionsResponse + 23, // 55: chroma.SysDB.UpdateCollection:output_type -> chroma.UpdateCollectionResponse + 25, // 56: chroma.SysDB.ResetState:output_type -> chroma.ResetStateResponse + 28, // 57: chroma.SysDB.GetLastCompactionTimeForTenant:output_type -> chroma.GetLastCompactionTimeForTenantResponse + 42, // 58: chroma.SysDB.SetLastCompactionTimeForTenant:output_type -> google.protobuf.Empty + 32, // 59: chroma.SysDB.FlushCollectionCompaction:output_type -> chroma.FlushCollectionCompactionResponse + 44, // [44:60] is the sub-list for method output_type + 28, // [28:44] is the sub-list for method input_type + 28, // [28:28] is the sub-list for extension type_name + 28, // [28:28] is the sub-list for extension extendee + 0, // [0:28] is the sub-list for field type_name } func init() { file_chromadb_proto_coordinator_proto_init() } @@ -2648,6 +2905,42 @@ func file_chromadb_proto_coordinator_proto_init() { return nil } } + file_chromadb_proto_coordinator_proto_msgTypes[30].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*FlushSegmentCompactionInfo); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_chromadb_proto_coordinator_proto_msgTypes[31].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*FlushCollectionCompactionRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_chromadb_proto_coordinator_proto_msgTypes[32].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*FlushCollectionCompactionResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } } file_chromadb_proto_coordinator_proto_msgTypes[12].OneofWrappers = []interface{}{} file_chromadb_proto_coordinator_proto_msgTypes[14].OneofWrappers = []interface{}{ @@ -2670,7 +2963,7 @@ func file_chromadb_proto_coordinator_proto_init() { GoPackagePath: reflect.TypeOf(x{}).PkgPath(), RawDescriptor: file_chromadb_proto_coordinator_proto_rawDesc, NumEnums: 0, - NumMessages: 30, + NumMessages: 34, NumExtensions: 0, NumServices: 1, }, diff --git a/go/pkg/proto/coordinatorpb/coordinator_grpc.pb.go b/go/pkg/proto/coordinatorpb/coordinator_grpc.pb.go index 755d0190efd..1306dbc1793 100644 --- a/go/pkg/proto/coordinatorpb/coordinator_grpc.pb.go +++ b/go/pkg/proto/coordinatorpb/coordinator_grpc.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go-grpc. DO NOT EDIT. // versions: -// - protoc-gen-go-grpc v1.2.0 -// - protoc v4.25.3 +// - protoc-gen-go-grpc v1.3.0 +// - protoc v4.23.4 // source: chromadb/proto/coordinator.proto package coordinatorpb @@ -19,6 +19,25 @@ import ( // Requires gRPC-Go v1.32.0 or later. const _ = grpc.SupportPackageIsVersion7 +const ( + SysDB_CreateDatabase_FullMethodName = "/chroma.SysDB/CreateDatabase" + SysDB_GetDatabase_FullMethodName = "/chroma.SysDB/GetDatabase" + SysDB_CreateTenant_FullMethodName = "/chroma.SysDB/CreateTenant" + SysDB_GetTenant_FullMethodName = "/chroma.SysDB/GetTenant" + SysDB_CreateSegment_FullMethodName = "/chroma.SysDB/CreateSegment" + SysDB_DeleteSegment_FullMethodName = "/chroma.SysDB/DeleteSegment" + SysDB_GetSegments_FullMethodName = "/chroma.SysDB/GetSegments" + SysDB_UpdateSegment_FullMethodName = "/chroma.SysDB/UpdateSegment" + SysDB_CreateCollection_FullMethodName = "/chroma.SysDB/CreateCollection" + SysDB_DeleteCollection_FullMethodName = "/chroma.SysDB/DeleteCollection" + SysDB_GetCollections_FullMethodName = "/chroma.SysDB/GetCollections" + SysDB_UpdateCollection_FullMethodName = "/chroma.SysDB/UpdateCollection" + SysDB_ResetState_FullMethodName = "/chroma.SysDB/ResetState" + SysDB_GetLastCompactionTimeForTenant_FullMethodName = "/chroma.SysDB/GetLastCompactionTimeForTenant" + SysDB_SetLastCompactionTimeForTenant_FullMethodName = "/chroma.SysDB/SetLastCompactionTimeForTenant" + SysDB_FlushCollectionCompaction_FullMethodName = "/chroma.SysDB/FlushCollectionCompaction" +) + // SysDBClient is the client API for SysDB service. // // For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. @@ -38,6 +57,7 @@ type SysDBClient interface { ResetState(ctx context.Context, in *emptypb.Empty, opts ...grpc.CallOption) (*ResetStateResponse, error) GetLastCompactionTimeForTenant(ctx context.Context, in *GetLastCompactionTimeForTenantRequest, opts ...grpc.CallOption) (*GetLastCompactionTimeForTenantResponse, error) SetLastCompactionTimeForTenant(ctx context.Context, in *SetLastCompactionTimeForTenantRequest, opts ...grpc.CallOption) (*emptypb.Empty, error) + FlushCollectionCompaction(ctx context.Context, in *FlushCollectionCompactionRequest, opts ...grpc.CallOption) (*FlushCollectionCompactionResponse, error) } type sysDBClient struct { @@ -50,7 +70,7 @@ func NewSysDBClient(cc grpc.ClientConnInterface) SysDBClient { func (c *sysDBClient) CreateDatabase(ctx context.Context, in *CreateDatabaseRequest, opts ...grpc.CallOption) (*CreateDatabaseResponse, error) { out := new(CreateDatabaseResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/CreateDatabase", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_CreateDatabase_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -59,7 +79,7 @@ func (c *sysDBClient) CreateDatabase(ctx context.Context, in *CreateDatabaseRequ func (c *sysDBClient) GetDatabase(ctx context.Context, in *GetDatabaseRequest, opts ...grpc.CallOption) (*GetDatabaseResponse, error) { out := new(GetDatabaseResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/GetDatabase", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_GetDatabase_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -68,7 +88,7 @@ func (c *sysDBClient) GetDatabase(ctx context.Context, in *GetDatabaseRequest, o func (c *sysDBClient) CreateTenant(ctx context.Context, in *CreateTenantRequest, opts ...grpc.CallOption) (*CreateTenantResponse, error) { out := new(CreateTenantResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/CreateTenant", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_CreateTenant_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -77,7 +97,7 @@ func (c *sysDBClient) CreateTenant(ctx context.Context, in *CreateTenantRequest, func (c *sysDBClient) GetTenant(ctx context.Context, in *GetTenantRequest, opts ...grpc.CallOption) (*GetTenantResponse, error) { out := new(GetTenantResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/GetTenant", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_GetTenant_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -86,7 +106,7 @@ func (c *sysDBClient) GetTenant(ctx context.Context, in *GetTenantRequest, opts func (c *sysDBClient) CreateSegment(ctx context.Context, in *CreateSegmentRequest, opts ...grpc.CallOption) (*CreateSegmentResponse, error) { out := new(CreateSegmentResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/CreateSegment", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_CreateSegment_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -95,7 +115,7 @@ func (c *sysDBClient) CreateSegment(ctx context.Context, in *CreateSegmentReques func (c *sysDBClient) DeleteSegment(ctx context.Context, in *DeleteSegmentRequest, opts ...grpc.CallOption) (*DeleteSegmentResponse, error) { out := new(DeleteSegmentResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/DeleteSegment", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_DeleteSegment_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -104,7 +124,7 @@ func (c *sysDBClient) DeleteSegment(ctx context.Context, in *DeleteSegmentReques func (c *sysDBClient) GetSegments(ctx context.Context, in *GetSegmentsRequest, opts ...grpc.CallOption) (*GetSegmentsResponse, error) { out := new(GetSegmentsResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/GetSegments", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_GetSegments_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -113,7 +133,7 @@ func (c *sysDBClient) GetSegments(ctx context.Context, in *GetSegmentsRequest, o func (c *sysDBClient) UpdateSegment(ctx context.Context, in *UpdateSegmentRequest, opts ...grpc.CallOption) (*UpdateSegmentResponse, error) { out := new(UpdateSegmentResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/UpdateSegment", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_UpdateSegment_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -122,7 +142,7 @@ func (c *sysDBClient) UpdateSegment(ctx context.Context, in *UpdateSegmentReques func (c *sysDBClient) CreateCollection(ctx context.Context, in *CreateCollectionRequest, opts ...grpc.CallOption) (*CreateCollectionResponse, error) { out := new(CreateCollectionResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/CreateCollection", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_CreateCollection_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -131,7 +151,7 @@ func (c *sysDBClient) CreateCollection(ctx context.Context, in *CreateCollection func (c *sysDBClient) DeleteCollection(ctx context.Context, in *DeleteCollectionRequest, opts ...grpc.CallOption) (*DeleteCollectionResponse, error) { out := new(DeleteCollectionResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/DeleteCollection", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_DeleteCollection_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -140,7 +160,7 @@ func (c *sysDBClient) DeleteCollection(ctx context.Context, in *DeleteCollection func (c *sysDBClient) GetCollections(ctx context.Context, in *GetCollectionsRequest, opts ...grpc.CallOption) (*GetCollectionsResponse, error) { out := new(GetCollectionsResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/GetCollections", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_GetCollections_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -149,7 +169,7 @@ func (c *sysDBClient) GetCollections(ctx context.Context, in *GetCollectionsRequ func (c *sysDBClient) UpdateCollection(ctx context.Context, in *UpdateCollectionRequest, opts ...grpc.CallOption) (*UpdateCollectionResponse, error) { out := new(UpdateCollectionResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/UpdateCollection", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_UpdateCollection_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -158,7 +178,7 @@ func (c *sysDBClient) UpdateCollection(ctx context.Context, in *UpdateCollection func (c *sysDBClient) ResetState(ctx context.Context, in *emptypb.Empty, opts ...grpc.CallOption) (*ResetStateResponse, error) { out := new(ResetStateResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/ResetState", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_ResetState_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -167,7 +187,7 @@ func (c *sysDBClient) ResetState(ctx context.Context, in *emptypb.Empty, opts .. func (c *sysDBClient) GetLastCompactionTimeForTenant(ctx context.Context, in *GetLastCompactionTimeForTenantRequest, opts ...grpc.CallOption) (*GetLastCompactionTimeForTenantResponse, error) { out := new(GetLastCompactionTimeForTenantResponse) - err := c.cc.Invoke(ctx, "/chroma.SysDB/GetLastCompactionTimeForTenant", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_GetLastCompactionTimeForTenant_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -176,7 +196,16 @@ func (c *sysDBClient) GetLastCompactionTimeForTenant(ctx context.Context, in *Ge func (c *sysDBClient) SetLastCompactionTimeForTenant(ctx context.Context, in *SetLastCompactionTimeForTenantRequest, opts ...grpc.CallOption) (*emptypb.Empty, error) { out := new(emptypb.Empty) - err := c.cc.Invoke(ctx, "/chroma.SysDB/SetLastCompactionTimeForTenant", in, out, opts...) + err := c.cc.Invoke(ctx, SysDB_SetLastCompactionTimeForTenant_FullMethodName, in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +func (c *sysDBClient) FlushCollectionCompaction(ctx context.Context, in *FlushCollectionCompactionRequest, opts ...grpc.CallOption) (*FlushCollectionCompactionResponse, error) { + out := new(FlushCollectionCompactionResponse) + err := c.cc.Invoke(ctx, SysDB_FlushCollectionCompaction_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -202,6 +231,7 @@ type SysDBServer interface { ResetState(context.Context, *emptypb.Empty) (*ResetStateResponse, error) GetLastCompactionTimeForTenant(context.Context, *GetLastCompactionTimeForTenantRequest) (*GetLastCompactionTimeForTenantResponse, error) SetLastCompactionTimeForTenant(context.Context, *SetLastCompactionTimeForTenantRequest) (*emptypb.Empty, error) + FlushCollectionCompaction(context.Context, *FlushCollectionCompactionRequest) (*FlushCollectionCompactionResponse, error) mustEmbedUnimplementedSysDBServer() } @@ -254,6 +284,9 @@ func (UnimplementedSysDBServer) GetLastCompactionTimeForTenant(context.Context, func (UnimplementedSysDBServer) SetLastCompactionTimeForTenant(context.Context, *SetLastCompactionTimeForTenantRequest) (*emptypb.Empty, error) { return nil, status.Errorf(codes.Unimplemented, "method SetLastCompactionTimeForTenant not implemented") } +func (UnimplementedSysDBServer) FlushCollectionCompaction(context.Context, *FlushCollectionCompactionRequest) (*FlushCollectionCompactionResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method FlushCollectionCompaction not implemented") +} func (UnimplementedSysDBServer) mustEmbedUnimplementedSysDBServer() {} // UnsafeSysDBServer may be embedded to opt out of forward compatibility for this service. @@ -277,7 +310,7 @@ func _SysDB_CreateDatabase_Handler(srv interface{}, ctx context.Context, dec fun } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/CreateDatabase", + FullMethod: SysDB_CreateDatabase_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).CreateDatabase(ctx, req.(*CreateDatabaseRequest)) @@ -295,7 +328,7 @@ func _SysDB_GetDatabase_Handler(srv interface{}, ctx context.Context, dec func(i } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/GetDatabase", + FullMethod: SysDB_GetDatabase_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).GetDatabase(ctx, req.(*GetDatabaseRequest)) @@ -313,7 +346,7 @@ func _SysDB_CreateTenant_Handler(srv interface{}, ctx context.Context, dec func( } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/CreateTenant", + FullMethod: SysDB_CreateTenant_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).CreateTenant(ctx, req.(*CreateTenantRequest)) @@ -331,7 +364,7 @@ func _SysDB_GetTenant_Handler(srv interface{}, ctx context.Context, dec func(int } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/GetTenant", + FullMethod: SysDB_GetTenant_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).GetTenant(ctx, req.(*GetTenantRequest)) @@ -349,7 +382,7 @@ func _SysDB_CreateSegment_Handler(srv interface{}, ctx context.Context, dec func } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/CreateSegment", + FullMethod: SysDB_CreateSegment_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).CreateSegment(ctx, req.(*CreateSegmentRequest)) @@ -367,7 +400,7 @@ func _SysDB_DeleteSegment_Handler(srv interface{}, ctx context.Context, dec func } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/DeleteSegment", + FullMethod: SysDB_DeleteSegment_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).DeleteSegment(ctx, req.(*DeleteSegmentRequest)) @@ -385,7 +418,7 @@ func _SysDB_GetSegments_Handler(srv interface{}, ctx context.Context, dec func(i } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/GetSegments", + FullMethod: SysDB_GetSegments_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).GetSegments(ctx, req.(*GetSegmentsRequest)) @@ -403,7 +436,7 @@ func _SysDB_UpdateSegment_Handler(srv interface{}, ctx context.Context, dec func } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/UpdateSegment", + FullMethod: SysDB_UpdateSegment_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).UpdateSegment(ctx, req.(*UpdateSegmentRequest)) @@ -421,7 +454,7 @@ func _SysDB_CreateCollection_Handler(srv interface{}, ctx context.Context, dec f } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/CreateCollection", + FullMethod: SysDB_CreateCollection_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).CreateCollection(ctx, req.(*CreateCollectionRequest)) @@ -439,7 +472,7 @@ func _SysDB_DeleteCollection_Handler(srv interface{}, ctx context.Context, dec f } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/DeleteCollection", + FullMethod: SysDB_DeleteCollection_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).DeleteCollection(ctx, req.(*DeleteCollectionRequest)) @@ -457,7 +490,7 @@ func _SysDB_GetCollections_Handler(srv interface{}, ctx context.Context, dec fun } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/GetCollections", + FullMethod: SysDB_GetCollections_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).GetCollections(ctx, req.(*GetCollectionsRequest)) @@ -475,7 +508,7 @@ func _SysDB_UpdateCollection_Handler(srv interface{}, ctx context.Context, dec f } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/UpdateCollection", + FullMethod: SysDB_UpdateCollection_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).UpdateCollection(ctx, req.(*UpdateCollectionRequest)) @@ -493,7 +526,7 @@ func _SysDB_ResetState_Handler(srv interface{}, ctx context.Context, dec func(in } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/ResetState", + FullMethod: SysDB_ResetState_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).ResetState(ctx, req.(*emptypb.Empty)) @@ -511,7 +544,7 @@ func _SysDB_GetLastCompactionTimeForTenant_Handler(srv interface{}, ctx context. } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/GetLastCompactionTimeForTenant", + FullMethod: SysDB_GetLastCompactionTimeForTenant_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).GetLastCompactionTimeForTenant(ctx, req.(*GetLastCompactionTimeForTenantRequest)) @@ -529,7 +562,7 @@ func _SysDB_SetLastCompactionTimeForTenant_Handler(srv interface{}, ctx context. } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.SysDB/SetLastCompactionTimeForTenant", + FullMethod: SysDB_SetLastCompactionTimeForTenant_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(SysDBServer).SetLastCompactionTimeForTenant(ctx, req.(*SetLastCompactionTimeForTenantRequest)) @@ -537,6 +570,24 @@ func _SysDB_SetLastCompactionTimeForTenant_Handler(srv interface{}, ctx context. return interceptor(ctx, in, info, handler) } +func _SysDB_FlushCollectionCompaction_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(FlushCollectionCompactionRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(SysDBServer).FlushCollectionCompaction(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: SysDB_FlushCollectionCompaction_FullMethodName, + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(SysDBServer).FlushCollectionCompaction(ctx, req.(*FlushCollectionCompactionRequest)) + } + return interceptor(ctx, in, info, handler) +} + // SysDB_ServiceDesc is the grpc.ServiceDesc for SysDB service. // It's only intended for direct use with grpc.RegisterService, // and not to be introspected or modified (even as a copy) @@ -604,6 +655,10 @@ var SysDB_ServiceDesc = grpc.ServiceDesc{ MethodName: "SetLastCompactionTimeForTenant", Handler: _SysDB_SetLastCompactionTimeForTenant_Handler, }, + { + MethodName: "FlushCollectionCompaction", + Handler: _SysDB_FlushCollectionCompaction_Handler, + }, }, Streams: []grpc.StreamDesc{}, Metadata: "chromadb/proto/coordinator.proto", diff --git a/go/pkg/proto/logservicepb/logservice.pb.go b/go/pkg/proto/logservicepb/logservice.pb.go index 491d57956d3..2122524d933 100644 --- a/go/pkg/proto/logservicepb/logservice.pb.go +++ b/go/pkg/proto/logservicepb/logservice.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go. DO NOT EDIT. // versions: -// protoc-gen-go v1.32.0 -// protoc v4.25.3 +// protoc-gen-go v1.31.0 +// protoc v4.23.4 // source: chromadb/proto/logservice.proto package logservicepb @@ -131,6 +131,7 @@ type PullLogsRequest struct { CollectionId string `protobuf:"bytes,1,opt,name=collection_id,json=collectionId,proto3" json:"collection_id,omitempty"` StartFromId int64 `protobuf:"varint,2,opt,name=start_from_id,json=startFromId,proto3" json:"start_from_id,omitempty"` BatchSize int32 `protobuf:"varint,3,opt,name=batch_size,json=batchSize,proto3" json:"batch_size,omitempty"` + EndTimestamp int64 `protobuf:"varint,4,opt,name=end_timestamp,json=endTimestamp,proto3" json:"end_timestamp,omitempty"` } func (x *PullLogsRequest) Reset() { @@ -186,6 +187,13 @@ func (x *PullLogsRequest) GetBatchSize() int32 { return 0 } +func (x *PullLogsRequest) GetEndTimestamp() int64 { + if x != nil { + return x.EndTimestamp + } + return 0 +} + type RecordLog struct { state protoimpl.MessageState sizeCache protoimpl.SizeCache @@ -454,64 +462,66 @@ var file_chromadb_proto_logservice_proto_rawDesc = []byte{ 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x73, 0x22, 0x35, 0x0a, 0x10, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x21, 0x0a, 0x0c, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x5f, 0x63, 0x6f, 0x75, 0x6e, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, - 0x05, 0x52, 0x0b, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x22, 0x79, - 0x0a, 0x0f, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, - 0x74, 0x12, 0x23, 0x0a, 0x0d, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, - 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, - 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, 0x12, 0x22, 0x0a, 0x0d, 0x73, 0x74, 0x61, 0x72, 0x74, 0x5f, - 0x66, 0x72, 0x6f, 0x6d, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0b, 0x73, - 0x74, 0x61, 0x72, 0x74, 0x46, 0x72, 0x6f, 0x6d, 0x49, 0x64, 0x12, 0x1d, 0x0a, 0x0a, 0x62, 0x61, - 0x74, 0x63, 0x68, 0x5f, 0x73, 0x69, 0x7a, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x05, 0x52, 0x09, - 0x62, 0x61, 0x74, 0x63, 0x68, 0x53, 0x69, 0x7a, 0x65, 0x22, 0x59, 0x0a, 0x09, 0x52, 0x65, 0x63, - 0x6f, 0x72, 0x64, 0x4c, 0x6f, 0x67, 0x12, 0x15, 0x0a, 0x06, 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x64, - 0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x05, 0x6c, 0x6f, 0x67, 0x49, 0x64, 0x12, 0x35, 0x0a, - 0x06, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1d, 0x2e, - 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x75, 0x62, 0x6d, 0x69, 0x74, 0x45, 0x6d, 0x62, - 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x52, 0x06, 0x72, 0x65, - 0x63, 0x6f, 0x72, 0x64, 0x22, 0x3f, 0x0a, 0x10, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, - 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x2b, 0x0a, 0x07, 0x72, 0x65, 0x63, 0x6f, - 0x72, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x11, 0x2e, 0x63, 0x68, 0x72, 0x6f, - 0x6d, 0x61, 0x2e, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x4c, 0x6f, 0x67, 0x52, 0x07, 0x72, 0x65, - 0x63, 0x6f, 0x72, 0x64, 0x73, 0x22, 0x7e, 0x0a, 0x0e, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, - 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x23, 0x0a, 0x0d, 0x63, 0x6f, 0x6c, 0x6c, 0x65, - 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, - 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, 0x12, 0x20, 0x0a, 0x0c, - 0x66, 0x69, 0x72, 0x73, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, - 0x28, 0x03, 0x52, 0x0a, 0x66, 0x69, 0x72, 0x73, 0x74, 0x4c, 0x6f, 0x67, 0x49, 0x64, 0x12, 0x25, - 0x0a, 0x0f, 0x66, 0x69, 0x72, 0x73, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x64, 0x5f, 0x74, - 0x73, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0c, 0x66, 0x69, 0x72, 0x73, 0x74, 0x4c, 0x6f, - 0x67, 0x49, 0x64, 0x54, 0x73, 0x22, 0x26, 0x0a, 0x24, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, + 0x05, 0x52, 0x0b, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x22, 0x9e, + 0x01, 0x0a, 0x0f, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, + 0x73, 0x74, 0x12, 0x23, 0x0a, 0x0d, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, + 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x64, 0x12, 0x22, 0x0a, 0x0d, 0x73, 0x74, 0x61, 0x72, 0x74, + 0x5f, 0x66, 0x72, 0x6f, 0x6d, 0x5f, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0b, + 0x73, 0x74, 0x61, 0x72, 0x74, 0x46, 0x72, 0x6f, 0x6d, 0x49, 0x64, 0x12, 0x1d, 0x0a, 0x0a, 0x62, + 0x61, 0x74, 0x63, 0x68, 0x5f, 0x73, 0x69, 0x7a, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x05, 0x52, + 0x09, 0x62, 0x61, 0x74, 0x63, 0x68, 0x53, 0x69, 0x7a, 0x65, 0x12, 0x23, 0x0a, 0x0d, 0x65, 0x6e, + 0x64, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x18, 0x04, 0x20, 0x01, 0x28, + 0x03, 0x52, 0x0c, 0x65, 0x6e, 0x64, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x22, + 0x59, 0x0a, 0x09, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x4c, 0x6f, 0x67, 0x12, 0x15, 0x0a, 0x06, + 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x05, 0x6c, 0x6f, + 0x67, 0x49, 0x64, 0x12, 0x35, 0x0a, 0x06, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x18, 0x02, 0x20, + 0x01, 0x28, 0x0b, 0x32, 0x1d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x53, 0x75, 0x62, + 0x6d, 0x69, 0x74, 0x45, 0x6d, 0x62, 0x65, 0x64, 0x64, 0x69, 0x6e, 0x67, 0x52, 0x65, 0x63, 0x6f, + 0x72, 0x64, 0x52, 0x06, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x22, 0x3f, 0x0a, 0x10, 0x50, 0x75, + 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x2b, + 0x0a, 0x07, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, + 0x11, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x52, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x4c, + 0x6f, 0x67, 0x52, 0x07, 0x72, 0x65, 0x63, 0x6f, 0x72, 0x64, 0x73, 0x22, 0x7e, 0x0a, 0x0e, 0x43, + 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x23, 0x0a, + 0x0d, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x64, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x49, 0x64, 0x12, 0x20, 0x0a, 0x0c, 0x66, 0x69, 0x72, 0x73, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x5f, + 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0a, 0x66, 0x69, 0x72, 0x73, 0x74, 0x4c, + 0x6f, 0x67, 0x49, 0x64, 0x12, 0x25, 0x0a, 0x0f, 0x66, 0x69, 0x72, 0x73, 0x74, 0x5f, 0x6c, 0x6f, + 0x67, 0x5f, 0x69, 0x64, 0x5f, 0x74, 0x73, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0c, 0x66, + 0x69, 0x72, 0x73, 0x74, 0x4c, 0x6f, 0x67, 0x49, 0x64, 0x54, 0x73, 0x22, 0x26, 0x0a, 0x24, 0x47, + 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, + 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x71, 0x75, + 0x65, 0x73, 0x74, 0x22, 0x6f, 0x0a, 0x25, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, + 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x46, 0x0a, 0x13, + 0x61, 0x6c, 0x6c, 0x5f, 0x63, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, + 0x6e, 0x66, 0x6f, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2e, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, + 0x6f, 0x52, 0x11, 0x61, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x49, 0x6e, 0x66, 0x6f, 0x32, 0x8e, 0x02, 0x0a, 0x0a, 0x4c, 0x6f, 0x67, 0x53, 0x65, 0x72, 0x76, + 0x69, 0x63, 0x65, 0x12, 0x3f, 0x0a, 0x08, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, 0x73, 0x12, + 0x17, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, + 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, + 0x61, 0x2e, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, + 0x73, 0x65, 0x22, 0x00, 0x12, 0x3f, 0x0a, 0x08, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, + 0x12, 0x17, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, + 0x67, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, + 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, + 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x7e, 0x0a, 0x1d, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, - 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x22, 0x6f, 0x0a, - 0x25, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, - 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, - 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x46, 0x0a, 0x13, 0x61, 0x6c, 0x6c, 0x5f, 0x63, 0x6f, - 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x69, 0x6e, 0x66, 0x6f, 0x18, 0x01, 0x20, - 0x03, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x43, 0x6f, 0x6c, - 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x11, 0x61, 0x6c, 0x6c, - 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x32, 0x8e, - 0x02, 0x0a, 0x0a, 0x4c, 0x6f, 0x67, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x3f, 0x0a, - 0x08, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, 0x73, 0x12, 0x17, 0x2e, 0x63, 0x68, 0x72, 0x6f, - 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x73, 0x68, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, - 0x73, 0x74, 0x1a, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x73, 0x68, - 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x3f, - 0x0a, 0x08, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x12, 0x17, 0x2e, 0x63, 0x68, 0x72, - 0x6f, 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x6c, 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x71, 0x75, - 0x65, 0x73, 0x74, 0x1a, 0x18, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x50, 0x75, 0x6c, - 0x6c, 0x4c, 0x6f, 0x67, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, - 0x7e, 0x0a, 0x1d, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, - 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, - 0x12, 0x2c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, - 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, - 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x2d, - 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, - 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, - 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, - 0x39, 0x5a, 0x37, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x63, 0x68, - 0x72, 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, - 0x2f, 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x6c, 0x6f, - 0x67, 0x73, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x70, 0x62, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, - 0x6f, 0x33, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x12, 0x2c, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, + 0x47, 0x65, 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x49, 0x6e, 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x71, + 0x75, 0x65, 0x73, 0x74, 0x1a, 0x2d, 0x2e, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2e, 0x47, 0x65, + 0x74, 0x41, 0x6c, 0x6c, 0x43, 0x6f, 0x6c, 0x6c, 0x65, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x49, 0x6e, + 0x66, 0x6f, 0x54, 0x6f, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x52, 0x65, 0x73, 0x70, 0x6f, + 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x39, 0x5a, 0x37, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, + 0x63, 0x6f, 0x6d, 0x2f, 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2d, 0x63, 0x6f, 0x72, 0x65, 0x2f, + 0x63, 0x68, 0x72, 0x6f, 0x6d, 0x61, 0x2f, 0x67, 0x6f, 0x2f, 0x70, 0x6b, 0x67, 0x2f, 0x70, 0x72, + 0x6f, 0x74, 0x6f, 0x2f, 0x6c, 0x6f, 0x67, 0x73, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x70, 0x62, + 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( diff --git a/go/pkg/proto/logservicepb/logservice_grpc.pb.go b/go/pkg/proto/logservicepb/logservice_grpc.pb.go index 62d87449a12..7b9895d172a 100644 --- a/go/pkg/proto/logservicepb/logservice_grpc.pb.go +++ b/go/pkg/proto/logservicepb/logservice_grpc.pb.go @@ -1,7 +1,7 @@ // Code generated by protoc-gen-go-grpc. DO NOT EDIT. // versions: -// - protoc-gen-go-grpc v1.2.0 -// - protoc v4.25.3 +// - protoc-gen-go-grpc v1.3.0 +// - protoc v4.23.4 // source: chromadb/proto/logservice.proto package logservicepb @@ -18,6 +18,12 @@ import ( // Requires gRPC-Go v1.32.0 or later. const _ = grpc.SupportPackageIsVersion7 +const ( + LogService_PushLogs_FullMethodName = "/chroma.LogService/PushLogs" + LogService_PullLogs_FullMethodName = "/chroma.LogService/PullLogs" + LogService_GetAllCollectionInfoToCompact_FullMethodName = "/chroma.LogService/GetAllCollectionInfoToCompact" +) + // LogServiceClient is the client API for LogService service. // // For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. @@ -37,7 +43,7 @@ func NewLogServiceClient(cc grpc.ClientConnInterface) LogServiceClient { func (c *logServiceClient) PushLogs(ctx context.Context, in *PushLogsRequest, opts ...grpc.CallOption) (*PushLogsResponse, error) { out := new(PushLogsResponse) - err := c.cc.Invoke(ctx, "/chroma.LogService/PushLogs", in, out, opts...) + err := c.cc.Invoke(ctx, LogService_PushLogs_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -46,7 +52,7 @@ func (c *logServiceClient) PushLogs(ctx context.Context, in *PushLogsRequest, op func (c *logServiceClient) PullLogs(ctx context.Context, in *PullLogsRequest, opts ...grpc.CallOption) (*PullLogsResponse, error) { out := new(PullLogsResponse) - err := c.cc.Invoke(ctx, "/chroma.LogService/PullLogs", in, out, opts...) + err := c.cc.Invoke(ctx, LogService_PullLogs_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -55,7 +61,7 @@ func (c *logServiceClient) PullLogs(ctx context.Context, in *PullLogsRequest, op func (c *logServiceClient) GetAllCollectionInfoToCompact(ctx context.Context, in *GetAllCollectionInfoToCompactRequest, opts ...grpc.CallOption) (*GetAllCollectionInfoToCompactResponse, error) { out := new(GetAllCollectionInfoToCompactResponse) - err := c.cc.Invoke(ctx, "/chroma.LogService/GetAllCollectionInfoToCompact", in, out, opts...) + err := c.cc.Invoke(ctx, LogService_GetAllCollectionInfoToCompact_FullMethodName, in, out, opts...) if err != nil { return nil, err } @@ -108,7 +114,7 @@ func _LogService_PushLogs_Handler(srv interface{}, ctx context.Context, dec func } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.LogService/PushLogs", + FullMethod: LogService_PushLogs_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(LogServiceServer).PushLogs(ctx, req.(*PushLogsRequest)) @@ -126,7 +132,7 @@ func _LogService_PullLogs_Handler(srv interface{}, ctx context.Context, dec func } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.LogService/PullLogs", + FullMethod: LogService_PullLogs_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(LogServiceServer).PullLogs(ctx, req.(*PullLogsRequest)) @@ -144,7 +150,7 @@ func _LogService_GetAllCollectionInfoToCompact_Handler(srv interface{}, ctx cont } info := &grpc.UnaryServerInfo{ Server: srv, - FullMethod: "/chroma.LogService/GetAllCollectionInfoToCompact", + FullMethod: LogService_GetAllCollectionInfoToCompact_FullMethodName, } handler := func(ctx context.Context, req interface{}) (interface{}, error) { return srv.(LogServiceServer).GetAllCollectionInfoToCompact(ctx, req.(*GetAllCollectionInfoToCompactRequest)) diff --git a/idl/chromadb/proto/chroma.proto b/idl/chromadb/proto/chroma.proto index 7a95fbe5c89..44d899e4530 100644 --- a/idl/chromadb/proto/chroma.proto +++ b/idl/chromadb/proto/chroma.proto @@ -34,6 +34,10 @@ enum SegmentScope { METADATA = 1; } +message FilePaths { + repeated string paths = 1; +} + message Segment { string id = 1; string type = 2; @@ -43,6 +47,7 @@ message Segment { // collection and can be used to service queries (for it's given scope.) optional string collection = 5; optional UpdateMetadata metadata = 6; + map file_paths = 7; } message Collection { @@ -53,6 +58,8 @@ message Collection { optional int32 dimension = 5; string tenant = 6; string database = 7; + int64 logPosition = 8; + int32 version = 9; } message Database { diff --git a/idl/chromadb/proto/coordinator.proto b/idl/chromadb/proto/coordinator.proto index 5e31b3273af..3695999ded8 100644 --- a/idl/chromadb/proto/coordinator.proto +++ b/idl/chromadb/proto/coordinator.proto @@ -176,6 +176,25 @@ message SetLastCompactionTimeForTenantRequest { TenantLastCompactionTime tenant_last_compaction_time = 1; } +message FlushSegmentCompactionInfo { + string segment_id = 1; + map file_paths = 2; +} + +message FlushCollectionCompactionRequest { + string tenant_id = 1; + string collection_id = 2; + int64 log_position = 3; + int32 collection_version = 4; + repeated FlushSegmentCompactionInfo segment_compaction_info = 5; +} + +message FlushCollectionCompactionResponse { + string collection_id = 1; + int32 collection_version = 2; + int64 last_compaction_time = 3; +} + service SysDB { rpc CreateDatabase(CreateDatabaseRequest) returns (CreateDatabaseResponse) {} rpc GetDatabase(GetDatabaseRequest) returns (GetDatabaseResponse) {} @@ -192,4 +211,5 @@ service SysDB { rpc ResetState(google.protobuf.Empty) returns (ResetStateResponse) {} rpc GetLastCompactionTimeForTenant(GetLastCompactionTimeForTenantRequest) returns (GetLastCompactionTimeForTenantResponse) {} rpc SetLastCompactionTimeForTenant(SetLastCompactionTimeForTenantRequest) returns (google.protobuf.Empty) {} + rpc FlushCollectionCompaction(FlushCollectionCompactionRequest) returns (FlushCollectionCompactionResponse) {} } diff --git a/idl/chromadb/proto/logservice.proto b/idl/chromadb/proto/logservice.proto index eaa5d0fad8f..e55a8ef331a 100644 --- a/idl/chromadb/proto/logservice.proto +++ b/idl/chromadb/proto/logservice.proto @@ -18,6 +18,7 @@ message PullLogsRequest { string collection_id = 1; int64 start_from_id = 2; int32 batch_size = 3; + int64 end_timestamp = 4; } message RecordLog { diff --git a/k8s/distributed-chroma/templates/frontend-server.yaml b/k8s/distributed-chroma/templates/frontend-service.yaml similarity index 50% rename from k8s/distributed-chroma/templates/frontend-server.yaml rename to k8s/distributed-chroma/templates/frontend-service.yaml index 43e0b063fd0..067fd033dea 100644 --- a/k8s/distributed-chroma/templates/frontend-server.yaml +++ b/k8s/distributed-chroma/templates/frontend-service.yaml @@ -1,21 +1,21 @@ apiVersion: apps/v1 kind: Deployment metadata: - name: frontend-server + name: frontend-service namespace: {{ .Values.namespace }} spec: replicas: 2 selector: matchLabels: - app: frontend-server + app: frontend-service template: metadata: labels: - app: frontend-server + app: frontend-service spec: containers: - - name: frontend-server - image: "{{ .Values.frontend.image.repository }}:{{ .Values.frontend.image.tag }}" + - name: frontend-service + image: "{{ .Values.frontendService.image.repository }}:{{ .Values.frontendService.image.tag }}" imagePullPolicy: IfNotPresent ports: - containerPort: 8000 @@ -24,42 +24,42 @@ spec: mountPath: /test env: - name: IS_PERSISTENT - {{ .Values.frontend.isPersistent }} + {{ .Values.frontendService.isPersistent }} - name: CHROMA_PRODUCER_IMPL - {{ .Values.frontend.producerImpl }} + {{ .Values.frontendService.producerImpl }} - name: CHROMA_CONSUMER_IMPL - {{ .Values.frontend.consumerImpl }} + {{ .Values.frontendService.consumerImpl }} - name: CHROMA_SEGMENT_MANAGER_IMPL - {{ .Values.frontend.segmentManagerImpl }} + {{ .Values.frontendService.segmentManagerImpl }} - name: PULSAR_BROKER_URL - {{ .Values.frontend.pulsarBrokerUrl }} + {{ .Values.frontendService.pulsarBrokerUrl }} - name: PULSAR_BROKER_PORT - {{ .Values.frontend.pulsarBrokerPort }} + {{ .Values.frontendService.pulsarBrokerPort }} - name: PULSAR_ADMIN_PORT - {{ .Values.frontend.pulsarAdminPort }} + {{ .Values.frontendService.pulsarAdminPort }} - name: ALLOW_RESET - {{ .Values.frontend.allowReset }} + {{ .Values.frontendService.allowReset }} - name: CHROMA_SYSDB_IMPL - {{ .Values.frontend.sysdbImpl }} + {{ .Values.frontendService.sysdbImpl }} - name: CHROMA_SERVER_GRPC_PORT - {{ .Values.frontend.serverGrpcPort }} + {{ .Values.frontendService.serverGrpcPort }} - name: CHROMA_COORDINATOR_HOST - {{ .Values.frontend.coordinatorHost }} + {{ .Values.frontendService.coordinatorHost }} - name: CHROMA_SERVER_AUTH_PROVIDER - {{ .Values.frontend.authProvider }} + {{ .Values.frontendService.authProvider }} - name: CHROMA_SERVER_AUTH_CREDENTIALS_PROVIDER - {{ .Values.frontend.authCredentialsProvider }} + {{ .Values.frontendService.authCredentialsProvider }} - name: CHROMA_SERVER_AUTHZ_PROVIDER - {{ .Values.frontend.authzProvider }} + {{ .Values.frontendService.authzProvider }} - name: CHROMA_SERVER_AUTHZ_CONFIG_PROVIDER - {{ .Values.frontend.authzConfigProvider }} + {{ .Values.frontendService.authzConfigProvider }} - name: CHROMA_MEMBERLIST_PROVIDER_IMPL - {{ .Values.frontend.memberlistProviderImpl }} + {{ .Values.frontendService.memberlistProviderImpl }} - name: CHROMA_LOGSERVICE_HOST - {{ .Values.frontend.logServiceHost }} + {{ .Values.frontendService.logServiceHost }} - name: CHROMA_LOGSERVICE_PORT - {{ .Values.frontend.logServicePort }} -{{ .Values.frontend.otherEnvConfig | nindent 12 }} + {{ .Values.frontendService.logServicePort }} +{{ .Values.frontendService.otherEnvConfig | nindent 12 }} volumes: - name: chroma emptyDir: {} @@ -69,7 +69,7 @@ spec: apiVersion: v1 kind: Service metadata: - name: frontend-server + name: frontend-service namespace: {{ .Values.namespace }} spec: ports: @@ -77,5 +77,5 @@ spec: port: 8000 targetPort: 8000 selector: - app: frontend-server + app: frontend-service type: ClusterIP diff --git a/k8s/distributed-chroma/templates/migration.yaml b/k8s/distributed-chroma/templates/migration.yaml deleted file mode 100644 index fcf605e2bea..00000000000 --- a/k8s/distributed-chroma/templates/migration.yaml +++ /dev/null @@ -1,22 +0,0 @@ -apiVersion: batch/v1 -kind: Job -metadata: - name: migration - namespace: {{ .Values.namespace }} -spec: - template: - metadata: - labels: - app: migration - spec: - restartPolicy: OnFailure - containers: - - args: - - 'migrate' - - 'apply' - - '--url' - - 'postgres://chroma:chroma@postgres:5432/chroma?sslmode=disable' - image: "{{ .Values.sysdbMigration.image.repository }}:{{ .Values.sysdbMigration.image.tag }}" - imagePullPolicy: IfNotPresent - name: migration ---- diff --git a/k8s/distributed-chroma/templates/worker.yaml b/k8s/distributed-chroma/templates/query-service.yaml similarity index 67% rename from k8s/distributed-chroma/templates/worker.yaml rename to k8s/distributed-chroma/templates/query-service.yaml index e8cee69ca0f..674ec8d44bb 100644 --- a/k8s/distributed-chroma/templates/worker.yaml +++ b/k8s/distributed-chroma/templates/query-service.yaml @@ -3,51 +3,49 @@ apiVersion: v1 kind: Service metadata: - name: worker + name: query-service namespace: {{ .Values.namespace }} spec: ports: - - name: worker-server-port + - name: query-service-server-port port: 50051 targetPort: 50051 selector: - app: worker-server + app: query-service-server type: ClusterIP --- - apiVersion: apps/v1 kind: Deployment metadata: - name: worker + name: query-service namespace: {{ .Values.namespace }} spec: replicas: 2 selector: matchLabels: - app: worker + app: query-service template: metadata: labels: - app: worker - member-type: worker + app: query-service + member-type: query-service spec: - serviceAccountName: worker-serviceaccount + serviceAccountName: query-service-serviceaccount containers: - - name: worker - image: "{{ .Values.worker.image.repository }}:{{ .Values.worker.image.tag }}" + - name: query-service + image: "{{ .Values.queryService.image.repository }}:{{ .Values.queryService.image.tag }}" imagePullPolicy: IfNotPresent - command: ["cargo", "run"] ports: - containerPort: 50051 volumeMounts: - name: chroma mountPath: /index_data env: - - name: CHROMA_WORKER__PULSAR_URL + - name: CHROMA_query-service__PULSAR_URL value: pulsar://pulsar.chroma:6650 - - name: CHROMA_WORKER__MY_IP + - name: CHROMA_query-service__MY_IP valueFrom: fieldRef: fieldPath: status.podIP @@ -57,7 +55,7 @@ spec: whenUnsatisfiable: ScheduleAnyway labelSelector: matchLabels: - member-type: worker + member-type: query-service volumes: - name: chroma emptyDir: {} @@ -67,7 +65,7 @@ spec: apiVersion: v1 kind: ServiceAccount metadata: - name: worker-serviceaccount + name: query-service-serviceaccount namespace: {{ .Values.namespace }} --- @@ -75,7 +73,7 @@ metadata: apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - name: worker-serviceaccount-rolebinding + name: query-service-serviceaccount-rolebinding namespace: {{ .Values.namespace }} roleRef: apiGroup: rbac.authorization.k8s.io @@ -83,7 +81,7 @@ roleRef: name: pod-watcher subjects: - kind: ServiceAccount - name: worker-serviceaccount + name: query-service-serviceaccount namespace: {{ .Values.namespace }} --- \ No newline at end of file diff --git a/k8s/distributed-chroma/templates/sysdb-migration.yaml b/k8s/distributed-chroma/templates/sysdb-migration.yaml new file mode 100644 index 00000000000..611d38a8d31 --- /dev/null +++ b/k8s/distributed-chroma/templates/sysdb-migration.yaml @@ -0,0 +1,28 @@ +apiVersion: batch/v1 +kind: Job +metadata: + name: sysdb-migration + namespace: {{ .Values.namespace }} +spec: + template: + metadata: + labels: + app: sysdb-migration + spec: + restartPolicy: OnFailure + containers: + - command: + - "/bin/sh" + - "-c" + - "atlas schema clean --auto-approve --url postgres://{{ .Values.sysdbMigration.username }}:{{ .Values.sysdbMigration.password }}@{{ .Values.sysdbMigration.netloc }}:{{ .Values.sysdbMigration.port }}/{{ .Values.sysdbMigration.dbName }}?sslmode={{ .Values.sysdbMigration.sslmode }}; atlas migrate apply --url postgres://{{ .Values.sysdbMigration.username }}:{{ .Values.sysdbMigration.password }}@{{ .Values.sysdbMigration.netloc }}:{{ .Values.sysdbMigration.port }}/{{ .Values.sysdbMigration.dbName }}?sslmode={{ .Values.sysdbMigration.sslmode }}" + image: "{{ .Values.sysdbMigration.image.repository }}:{{ .Values.sysdbMigration.image.tag }}" + imagePullPolicy: IfNotPresent + name: migration + env: + {{ range .Values.sysdb.env }} + - name: {{ .name }} + # TODO properly use flow control here to check which type of value we need. +{{ .value | nindent 14 }} + {{ end }} + +--- diff --git a/k8s/distributed-chroma/templates/coordinator.yaml b/k8s/distributed-chroma/templates/sysdb-service.yaml similarity index 69% rename from k8s/distributed-chroma/templates/coordinator.yaml rename to k8s/distributed-chroma/templates/sysdb-service.yaml index 37a6a2e8987..296d391c485 100644 --- a/k8s/distributed-chroma/templates/coordinator.yaml +++ b/k8s/distributed-chroma/templates/sysdb-service.yaml @@ -1,34 +1,34 @@ apiVersion: apps/v1 kind: Deployment metadata: - name: coordinator + name: sysdb namespace: {{ .Values.namespace }} spec: - replicas: {{ .Values.coordinator.replicaCount }} + replicas: {{ .Values.sysdb.replicaCount }} selector: matchLabels: - app: coordinator + app: sysdb template: metadata: labels: - app: coordinator + app: sysdb spec: - serviceAccountName: coordinator-serviceaccount + serviceAccountName: sysdb-serviceaccount containers: - command: - "/bin/sh" - "-c" # This has to be one line to be passed into the `exec` env correctly. I truly could not tell you why. - - coordinator coordinator {{ range $k, $v := .Values.coordinator.flags }} --{{ $k }}={{ $v }} {{ end }} + - coordinator coordinator {{ range $k, $v := .Values.sysdb.flags }} --{{ $k }}={{ $v }} {{ end }} env: - {{ range .Values.coordinator.env }} + {{ range .Values.sysdb.env }} - name: {{ .name }} # TODO properly use flow control here to check which type of value we need. {{ .value | nindent 14 }} {{ end }} - image: "{{ .Values.coordinator.image.repository }}:{{ .Values.coordinator.image.tag }}" + image: "{{ .Values.sysdb.image.repository }}:{{ .Values.sysdb.image.tag }}" imagePullPolicy: IfNotPresent - name: coordinator + name: sysdb ports: - containerPort: 50051 name: grpc @@ -38,7 +38,7 @@ spec: apiVersion: v1 kind: Service metadata: - name: coordinator + name: sysdb namespace: {{ .Values.namespace }} spec: ports: @@ -46,7 +46,7 @@ spec: port: 50051 targetPort: grpc selector: - app: coordinator + app: sysdb type: ClusterIP --- @@ -54,7 +54,7 @@ spec: apiVersion: v1 kind: ServiceAccount metadata: - name: coordinator-serviceaccount + name: sysdb-serviceaccount namespace: {{ .Values.namespace }} --- @@ -62,7 +62,7 @@ metadata: apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - name: coordinator-serviceaccount-rolebinding + name: sysdb-serviceaccount-rolebinding namespace: {{ .Values.namespace }} roleRef: apiGroup: rbac.authorization.k8s.io @@ -70,7 +70,7 @@ roleRef: name: pod-watcher subjects: - kind: ServiceAccount - name: coordinator-serviceaccount + name: sysdb-serviceaccount namespace: {{ .Values.namespace }} --- diff --git a/k8s/distributed-chroma/templates/worker_memberlist_cr.yaml b/k8s/distributed-chroma/templates/worker-memberlist-cr.yaml similarity index 71% rename from k8s/distributed-chroma/templates/worker_memberlist_cr.yaml rename to k8s/distributed-chroma/templates/worker-memberlist-cr.yaml index 1b022afa2ce..2261057ec5d 100644 --- a/k8s/distributed-chroma/templates/worker_memberlist_cr.yaml +++ b/k8s/distributed-chroma/templates/worker-memberlist-cr.yaml @@ -5,7 +5,7 @@ apiVersion: chroma.cluster/v1 kind: MemberList metadata: - name: worker-memberlist + name: query-service-memberlist namespace: {{ .Values.namespace}} spec: members: @@ -15,7 +15,7 @@ spec: apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRole metadata: - name: worker-memberlist-readerwriter + name: query-service-memberlist-readerwriter rules: - apiGroups: - chroma.cluster @@ -36,14 +36,14 @@ rules: apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRoleBinding metadata: - name: coordinator-worker-memberlist-binding + name: sysdb-query-service-memberlist-binding roleRef: apiGroup: rbac.authorization.k8s.io kind: ClusterRole - name: worker-memberlist-readerwriter + name: query-service-memberlist-readerwriter subjects: - kind: ServiceAccount - name: coordinator-serviceaccount + name: sysdb-serviceaccount namespace: {{ .Values.namespace }} --- @@ -51,16 +51,16 @@ subjects: apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRoleBinding metadata: - # Awkward name, but this lets the worker-serviceaccount read - # the worker-memberlist. - name: worker-worker-memberlist-binding + # Awkward name, but this lets the query-service-serviceaccount read + # the query-service-memberlist. + name: query-service-query-service-memberlist-binding roleRef: apiGroup: rbac.authorization.k8s.io kind: ClusterRole - name: worker-memberlist-readerwriter + name: query-service-memberlist-readerwriter subjects: - kind: ServiceAccount - name: worker-serviceaccount + name: query-service-serviceaccount namespace: {{ .Values.namespace }} --- @@ -68,11 +68,11 @@ subjects: apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRoleBinding metadata: - name: worker-memberlist-readerwriter-binding + name: query-service-memberlist-readerwriter-binding roleRef: apiGroup: rbac.authorization.k8s.io kind: ClusterRole - name: worker-memberlist-readerwriter + name: query-service-memberlist-readerwriter subjects: - kind: ServiceAccount name: default diff --git a/k8s/distributed-chroma/values.yaml b/k8s/distributed-chroma/values.yaml index e30bcb0e58c..37b11cc487e 100644 --- a/k8s/distributed-chroma/values.yaml +++ b/k8s/distributed-chroma/values.yaml @@ -3,10 +3,10 @@ namespace: 'chroma' -frontend: +frontendService: image: repository: 'local' - tag: 'frontend-server' + tag: 'frontend-service' # Sometimes users (and us) want to pass values directly as flags. Sometimes, these are # populated from secrets or configMaps. So we let consumers fill these directly. @@ -22,7 +22,7 @@ frontend: allowReset: 'value: "TRUE"' sysdbImpl: 'value: "chromadb.db.impl.grpc.client.GrpcSysDB"' serverGrpcPort: 'value: "50051"' - coordinatorHost: 'value: "coordinator.chroma"' + coordinatorHost: 'value: "sysdb.chroma"' authProvider: 'value: ""' authCredentialsProvider: 'value: ""' authzProvider: 'value: ""' @@ -32,10 +32,10 @@ frontend: logServicePort: 'value: "50051"' otherEnvConfig: '' -coordinator: +sysdb: image: repository: 'local' - tag: 'coordinator' + tag: 'sysdb' replicaCount: 1 env: flags: @@ -46,16 +46,22 @@ coordinator: logService: image: repository: 'local' - tag: 'coordinator' + tag: 'sysdb' env: flags: -worker: +queryService: image: repository: 'local' - tag: 'worker' + tag: 'query-service' sysdbMigration: image: repository: 'local' - tag: 'migration' + tag: 'sysdb-migration' + username: chroma + password: chroma + netloc: postgres + port: 5432 + dbName: chroma + sslmode: disable diff --git a/k8s/test/jaeger_service.yaml b/k8s/test/jaeger-service.yaml similarity index 100% rename from k8s/test/jaeger_service.yaml rename to k8s/test/jaeger-service.yaml diff --git a/k8s/test/logservice_service.yaml b/k8s/test/logservice-service.yaml similarity index 100% rename from k8s/test/logservice_service.yaml rename to k8s/test/logservice-service.yaml diff --git a/k8s/test/pulsar_service.yaml b/k8s/test/pulsar-service.yaml similarity index 100% rename from k8s/test/pulsar_service.yaml rename to k8s/test/pulsar-service.yaml diff --git a/k8s/test/worker_service.yaml b/k8s/test/query-service-service.yaml similarity index 65% rename from k8s/test/worker_service.yaml rename to k8s/test/query-service-service.yaml index 9fac38d0e1f..6cfdd71677b 100644 --- a/k8s/test/worker_service.yaml +++ b/k8s/test/query-service-service.yaml @@ -1,13 +1,13 @@ apiVersion: v1 kind: Service metadata: - name: worker-lb + name: query-service-lb namespace: chroma spec: ports: - - name: worker-port + - name: query-service-port port: 50052 targetPort: 50051 selector: - app: worker + app: query-service type: LoadBalancer diff --git a/k8s/test/coordinator_service.yaml b/k8s/test/sysdb-service.yaml similarity index 79% rename from k8s/test/coordinator_service.yaml rename to k8s/test/sysdb-service.yaml index 37334b12187..dda139fcb7d 100644 --- a/k8s/test/coordinator_service.yaml +++ b/k8s/test/sysdb-service.yaml @@ -1,7 +1,7 @@ apiVersion: v1 kind: Service metadata: - name: coordinator-lb + name: sysdb-lb namespace: chroma spec: ports: @@ -9,5 +9,5 @@ spec: port: 50051 targetPort: 50051 selector: - app: coordinator + app: sysdb type: LoadBalancer diff --git a/k8s/test/test_memberlist_cr.yaml b/k8s/test/test-memberlist-cr.yaml similarity index 100% rename from k8s/test/test_memberlist_cr.yaml rename to k8s/test/test-memberlist-cr.yaml diff --git a/rust/worker/.dockerignore b/rust/worker/.dockerignore new file mode 100644 index 00000000000..2f0f51534e8 --- /dev/null +++ b/rust/worker/.dockerignore @@ -0,0 +1,3 @@ +.dockerignore +.gitignore +Dockerfile \ No newline at end of file diff --git a/rust/worker/Cargo.toml b/rust/worker/Cargo.toml index 080e17031ff..65ab2a68ff0 100644 --- a/rust/worker/Cargo.toml +++ b/rust/worker/Cargo.toml @@ -4,8 +4,8 @@ version = "0.1.0" edition = "2021" [[bin]] -name = "worker" -path = "src/bin/worker.rs" +name = "query_service" +path = "src/bin/query_service.rs" [dependencies] tonic = "0.10" @@ -46,4 +46,4 @@ proptest-state-machine = "0.1.0" [build-dependencies] tonic-build = "0.10" -cc = "1.0" \ No newline at end of file +cc = "1.0" diff --git a/rust/worker/Dockerfile b/rust/worker/Dockerfile index 6cf34960217..efa9b7d811f 100644 --- a/rust/worker/Dockerfile +++ b/rust/worker/Dockerfile @@ -1,24 +1,40 @@ FROM rust:1.74.1 as builder ARG CHROMA_KUBERNETES_INTEGRATION=0 -ARG RELEASE_MODE=0 ENV CHROMA_KUBERNETES_INTEGRATION $CHROMA_KUBERNETES_INTEGRATION +ARG RELEASE_MODE= + WORKDIR / RUN git clone https://github.com/chroma-core/hnswlib.git +# Cache dependencies by building them without our code first. +# https://dev.to/rogertorres/first-steps-with-docker-rust-30oi +# https://www.reddit.com/r/rust/comments/126xeyx/exploring_the_problem_of_faster_cargo_docker/ WORKDIR /chroma/ - +COPY Cargo.toml Cargo.toml +COPY Cargo.lock Cargo.lock +COPY idl/ idl/ +COPY /rust/worker/Cargo.toml rust/worker/Cargo.toml ENV PROTOC_ZIP=protoc-25.1-linux-x86_64.zip RUN curl -OL https://github.com/protocolbuffers/protobuf/releases/download/v25.1/$PROTOC_ZIP \ && unzip -o $PROTOC_ZIP -d /usr/local bin/protoc \ && unzip -o $PROTOC_ZIP -d /usr/local 'include/*' \ && rm -f $PROTOC_ZIP -COPY . . +# We need to replace the query node's real main() with a dummy at the expected location. +RUN mkdir -p rust/worker/src/bin/ && echo "fn main() {}" > rust/worker/src/bin/query_service.rs +RUN if [ "$RELEASE_MODE" = "1" ]; then cargo build --release; else cargo build; fi +RUN rm -rf rust/ + +COPY rust/ rust/ +RUN touch rust/worker/src/bin/query_service.rs +RUN if [ "$RELEASE_MODE" = "1" ]; then cargo build --release; else cargo build; fi +RUN if [ "$RELEASE_MODE" = "1" ]; then mv target/release/query_service .; else mv target/debug/query_service .; fi -ENV CARGO_TARGET_DIR=/root/.cache/cargo -RUN --mount=type=cache,target=/root/.cache/cargo if [ "$RELEASE_MODE" = "1" ]; then cargo build --release; else cargo build; fi +FROM debian:bookworm-slim -WORKDIR /chroma/rust/worker +COPY --from=builder /chroma/query_service . +COPY --from=builder /chroma/rust/worker/chroma_config.yaml . +RUN apt-get update && apt-get install -y libssl-dev -CMD ["cargo", "run"] +ENTRYPOINT [ "./query_service" ] \ No newline at end of file diff --git a/rust/worker/build.rs b/rust/worker/build.rs index be6fc45b7bc..c98a96de683 100644 --- a/rust/worker/build.rs +++ b/rust/worker/build.rs @@ -1,13 +1,15 @@ fn main() -> Result<(), Box> { // Compile the protobuf files in the chromadb proto directory. - tonic_build::configure().compile( - &[ - "../../idl/chromadb/proto/chroma.proto", - "../../idl/chromadb/proto/coordinator.proto", - "../../idl/chromadb/proto/logservice.proto", - ], - &["../../idl/"], - )?; + tonic_build::configure() + .emit_rerun_if_changed(true) + .compile( + &[ + "../../idl/chromadb/proto/chroma.proto", + "../../idl/chromadb/proto/coordinator.proto", + "../../idl/chromadb/proto/logservice.proto", + ], + &["../../idl/"], + )?; // Compile the hnswlib bindings. cc::Build::new() diff --git a/rust/worker/chroma_config.yaml b/rust/worker/chroma_config.yaml index 32e5165924d..9fab1f154c6 100644 --- a/rust/worker/chroma_config.yaml +++ b/rust/worker/chroma_config.yaml @@ -16,16 +16,24 @@ worker: hasher: Murmur3 memberlist_provider: CustomResource: - memberlist_name: "worker-memberlist" + memberlist_name: "query-service-memberlist" queue_size: 100 ingest: queue_size: 10000 sysdb: Grpc: - host: "coordinator.chroma" + host: "sysdb.chroma" port: 50051 segment_manager: storage_path: "./tmp/segment_manager/" storage: S3: bucket: "chroma-storage" + log: + Grpc: + host: "logservice.chroma" + port: 50051 + dispatcher: + num_worker_threads: 4 + dispatcher_queue_size: 100 + worker_queue_size: 100 diff --git a/rust/worker/src/bin/query_service.rs b/rust/worker/src/bin/query_service.rs new file mode 100644 index 00000000000..f3cfa4c8282 --- /dev/null +++ b/rust/worker/src/bin/query_service.rs @@ -0,0 +1,6 @@ +use worker::query_service_entrypoint; + +#[tokio::main] +async fn main() { + query_service_entrypoint().await; +} diff --git a/rust/worker/src/bin/worker.rs b/rust/worker/src/bin/worker.rs deleted file mode 100644 index 16428d244ff..00000000000 --- a/rust/worker/src/bin/worker.rs +++ /dev/null @@ -1,6 +0,0 @@ -use worker::worker_entrypoint; - -#[tokio::main] -async fn main() { - worker_entrypoint().await; -} diff --git a/rust/worker/src/compactor/scheduler.rs b/rust/worker/src/compactor/scheduler.rs index 8e418dcd99e..bd51ed0320b 100644 --- a/rust/worker/src/compactor/scheduler.rs +++ b/rust/worker/src/compactor/scheduler.rs @@ -134,8 +134,9 @@ impl Scheduler { } } +#[async_trait] impl Component for Scheduler { - fn on_start(&mut self, ctx: &ComponentContext) { + async fn on_start(&mut self, ctx: &ComponentContext) { ctx.scheduler.schedule_interval( ctx.sender.clone(), ScheduleMessage {}, @@ -186,7 +187,7 @@ mod tests { use std::time::Duration; use uuid::Uuid; - #[derive(Clone)] + #[derive(Clone, Debug)] pub(crate) struct TestSysDb { collections: HashMap, } @@ -322,6 +323,8 @@ mod tests { dimension: Some(1), tenant: "tenant_1".to_string(), database: "database_1".to_string(), + log_position: 0, + version: 0, }; let collection_2 = Collection { @@ -332,6 +335,8 @@ mod tests { dimension: Some(1), tenant: "tenant_2".to_string(), database: "database_2".to_string(), + log_position: 0, + version: 0, }; sysdb.add_collection(collection_1); sysdb.add_collection(collection_2); diff --git a/rust/worker/src/config.rs b/rust/worker/src/config.rs index c9c3d82f679..560557fd045 100644 --- a/rust/worker/src/config.rs +++ b/rust/worker/src/config.rs @@ -110,6 +110,7 @@ pub(crate) struct WorkerConfig { pub(crate) segment_manager: crate::segment::config::SegmentManagerConfig, pub(crate) storage: crate::storage::config::StorageConfig, pub(crate) log: crate::log::config::LogConfig, + pub(crate) dispatcher: crate::execution::config::DispatcherConfig, } /// # Description @@ -164,7 +165,11 @@ mod tests { log: Grpc: host: "localhost" - port: 50052 + port: 50051 + dispatcher: + num_worker_threads: 4 + dispatcher_queue_size: 100 + worker_queue_size: 100 "#, ); let config = RootConfig::load(); @@ -212,7 +217,11 @@ mod tests { log: Grpc: host: "localhost" - port: 50052 + port: 50051 + dispatcher: + num_worker_threads: 4 + dispatcher_queue_size: 100 + worker_queue_size: 100 "#, ); @@ -276,7 +285,11 @@ mod tests { log: Grpc: host: "localhost" - port: 50052 + port: 50051 + dispatcher: + num_worker_threads: 4 + dispatcher_queue_size: 100 + worker_queue_size: 100 "#, ); let config = RootConfig::load(); @@ -320,7 +333,11 @@ mod tests { log: Grpc: host: "localhost" - port: 50052 + port: 50051 + dispatcher: + num_worker_threads: 4 + dispatcher_queue_size: 100 + worker_queue_size: 100 "#, ); let config = RootConfig::load(); @@ -333,4 +350,10 @@ mod tests { Ok(()) }); } + + #[test] + fn test_default_config_path() { + // Sanity check that root config loads from default path correctly + let _ = RootConfig::load(); + } } diff --git a/rust/worker/src/distance/mod.rs b/rust/worker/src/distance/mod.rs new file mode 100644 index 00000000000..b5d12e1b8a9 --- /dev/null +++ b/rust/worker/src/distance/mod.rs @@ -0,0 +1,3 @@ +mod types; + +pub use types::*; diff --git a/rust/worker/src/distance/types.rs b/rust/worker/src/distance/types.rs new file mode 100644 index 00000000000..99e0df285c5 --- /dev/null +++ b/rust/worker/src/distance/types.rs @@ -0,0 +1,145 @@ +use crate::errors::{ChromaError, ErrorCodes}; +use thiserror::Error; + +/// The distance function enum. +/// # Description +/// This enum defines the distance functions supported by indices in Chroma. +/// # Variants +/// - `Euclidean` - The Euclidean or l2 norm. +/// - `Cosine` - The cosine distance. Specifically, 1 - cosine. +/// - `InnerProduct` - The inner product. Specifically, 1 - inner product. +/// # Notes +/// See https://docs.trychroma.com/usage-guide#changing-the-distance-function +#[derive(Clone, Debug, PartialEq)] +pub(crate) enum DistanceFunction { + Euclidean, + Cosine, + InnerProduct, +} + +impl DistanceFunction { + // TOOD: Should we error if mismatched dimensions? + pub(crate) fn distance(&self, a: &[f32], b: &[f32]) -> f32 { + // TODO: implement this in SSE/AVX SIMD + // For now we write these as loops since we suspect that will more likely + // lead to the compiler vectorizing the code. (We saw this on + // Apple Silicon Macs who didn't have hand-rolled SIMD instructions in our + // C++ code). + match self { + DistanceFunction::Euclidean => { + let mut sum = 0.0; + for i in 0..a.len() { + sum += (a[i] - b[i]).powi(2); + } + sum + } + DistanceFunction::Cosine => { + // For cosine we just assume the vectors have been normalized, since that + // is what our indices expect. + let mut sum = 0.0; + for i in 0..a.len() { + sum += a[i] * b[i]; + } + 1.0_f32 - sum + } + DistanceFunction::InnerProduct => { + let mut sum = 0.0; + for i in 0..a.len() { + sum += a[i] * b[i]; + } + 1.0_f32 - sum + } + } + } +} + +#[derive(Error, Debug)] +pub(crate) enum DistanceFunctionError { + #[error("Invalid distance function `{0}`")] + InvalidDistanceFunction(String), +} + +impl ChromaError for DistanceFunctionError { + fn code(&self) -> ErrorCodes { + match self { + DistanceFunctionError::InvalidDistanceFunction(_) => ErrorCodes::InvalidArgument, + } + } +} + +impl TryFrom<&str> for DistanceFunction { + type Error = DistanceFunctionError; + + fn try_from(value: &str) -> Result { + match value { + "l2" => Ok(DistanceFunction::Euclidean), + "cosine" => Ok(DistanceFunction::Cosine), + "ip" => Ok(DistanceFunction::InnerProduct), + _ => Err(DistanceFunctionError::InvalidDistanceFunction( + value.to_string(), + )), + } + } +} + +impl Into for DistanceFunction { + fn into(self) -> String { + match self { + DistanceFunction::Euclidean => "l2".to_string(), + DistanceFunction::Cosine => "cosine".to_string(), + DistanceFunction::InnerProduct => "ip".to_string(), + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use std::convert::TryInto; + + #[test] + fn test_distance_function_try_from() { + let distance_function: DistanceFunction = "l2".try_into().unwrap(); + assert_eq!(distance_function, DistanceFunction::Euclidean); + let distance_function: DistanceFunction = "cosine".try_into().unwrap(); + assert_eq!(distance_function, DistanceFunction::Cosine); + let distance_function: DistanceFunction = "ip".try_into().unwrap(); + assert_eq!(distance_function, DistanceFunction::InnerProduct); + } + + #[test] + fn test_distance_function_into() { + let distance_function: String = DistanceFunction::Euclidean.into(); + assert_eq!(distance_function, "l2"); + let distance_function: String = DistanceFunction::Cosine.into(); + assert_eq!(distance_function, "cosine"); + let distance_function: String = DistanceFunction::InnerProduct.into(); + assert_eq!(distance_function, "ip"); + } + + #[test] + fn test_distance_function_l2sqr() { + let a = vec![1.0, 2.0, 3.0]; + let a_mag = (1.0_f32.powi(2) + 2.0_f32.powi(2) + 3.0_f32.powi(2)).sqrt(); + let a_norm = vec![1.0 / a_mag, 2.0 / a_mag, 3.0 / a_mag]; + let b = vec![4.0, 5.0, 6.0]; + let b_mag = (4.0_f32.powi(2) + 5.0_f32.powi(2) + 6.0_f32.powi(2)).sqrt(); + let b_norm = vec![4.0 / b_mag, 5.0 / b_mag, 6.0 / b_mag]; + + let l2_sqr = (1.0 - 4.0_f32).powi(2) + (2.0 - 5.0_f32).powi(2) + (3.0 - 6.0_f32).powi(2); + let inner_product_sim = 1.0_f32 + - a_norm + .iter() + .zip(b_norm.iter()) + .map(|(a, b)| a * b) + .sum::(); + + let distance_function: DistanceFunction = "l2".try_into().unwrap(); + assert_eq!(distance_function.distance(&a, &b), l2_sqr); + let distance_function: DistanceFunction = "ip".try_into().unwrap(); + assert_eq!( + distance_function.distance(&a_norm, &b_norm), + inner_product_sim + ); + } +} diff --git a/rust/worker/src/errors.rs b/rust/worker/src/errors.rs index 18365cb789f..086b938f265 100644 --- a/rust/worker/src/errors.rs +++ b/rust/worker/src/errors.rs @@ -1,7 +1,6 @@ // Defines 17 standard error codes based on the error codes defined in the // gRPC spec. https://grpc.github.io/grpc/core/md_doc_statuscodes.html // Custom errors can use these codes in order to allow for generic handling - use std::error::Error; #[derive(PartialEq, Debug)] @@ -42,6 +41,6 @@ pub(crate) enum ErrorCodes { DataLoss = 15, } -pub(crate) trait ChromaError: Error { +pub(crate) trait ChromaError: Error + Send { fn code(&self) -> ErrorCodes; } diff --git a/rust/worker/src/execution/config.rs b/rust/worker/src/execution/config.rs new file mode 100644 index 00000000000..d8550dc41bc --- /dev/null +++ b/rust/worker/src/execution/config.rs @@ -0,0 +1,8 @@ +use serde::Deserialize; + +#[derive(Deserialize)] +pub(crate) struct DispatcherConfig { + pub(crate) num_worker_threads: usize, + pub(crate) dispatcher_queue_size: usize, + pub(crate) worker_queue_size: usize, +} diff --git a/rust/worker/src/execution/dispatcher.rs b/rust/worker/src/execution/dispatcher.rs new file mode 100644 index 00000000000..b1668b1c60a --- /dev/null +++ b/rust/worker/src/execution/dispatcher.rs @@ -0,0 +1,289 @@ +use super::{operator::TaskMessage, worker_thread::WorkerThread}; +use crate::{ + config::{Configurable, WorkerConfig}, + errors::ChromaError, + system::{Component, ComponentContext, Handler, Receiver, System}, +}; +use async_trait::async_trait; +use std::fmt::Debug; + +/// The dispatcher is responsible for distributing tasks to worker threads. +/// It is a component that receives tasks and distributes them to worker threads. +/** +```plaintext + ┌─────────────────────────────────────────┐ + │ │ + │ │ + │ │ + TaskMessage ───────────►├─────┐ Dispatcher │ + │ ▼ │ + │ ┌┬───────────────────────────────┐ │ + │ │┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼┼│ │ + └────┴──────────────┴─────────────────┴───┘ + ▲ + │ │ + │ │ + TaskRequestMessage │ │ TaskMessage + │ │ + │ │ + ▼ + ┌────────────────┐ ┌────────────────┐ ┌────────────────┐ + │ │ │ │ │ │ + │ │ │ │ │ │ + │ │ │ │ │ │ + │ Worker │ │ Worker │ │ Worker │ + │ │ │ │ │ │ + │ │ │ │ │ │ + │ │ │ │ │ │ + └────────────────┘ └────────────────┘ └────────────────┘ +``` +## Implementation notes +- The dispatcher has a queue of tasks that it distributes to worker threads +- A worker thread sends a TaskRequestMessage to the dispatcher when it is ready for a new task +- If no task is available for the worker thread, the dispatcher will place that worker's reciever + in a queue and send a task to the worker when it recieves another one +- The reason to introduce this abstraction is to allow us to control fairness and dynamically adjust + system utilization. It also makes mechanisms like pausing/stopping work easier. + It would have likely been more performant to use the Tokio MT runtime, but we chose to use + this abstraction to grant us flexibility. We can always switch to Tokio MT later if we need to, + or make this dispatcher much more performant through implementing memory-awareness, task-batches, + coarser work-stealing, and other optimizations. +*/ +#[derive(Debug)] +pub(crate) struct Dispatcher { + task_queue: Vec, + waiters: Vec, + n_worker_threads: usize, + queue_size: usize, + worker_queue_size: usize, +} + +impl Dispatcher { + /// Create a new dispatcher + /// # Parameters + /// - n_worker_threads: The number of worker threads to use + /// - queue_size: The size of the components message queue + /// - worker_queue_size: The size of the worker components queue + pub fn new(n_worker_threads: usize, queue_size: usize, worker_queue_size: usize) -> Self { + Dispatcher { + task_queue: Vec::new(), + waiters: Vec::new(), + n_worker_threads, + queue_size, + worker_queue_size, + } + } + + /// Spawn worker threads + /// # Parameters + /// - system: The system to spawn the worker threads in + /// - self_receiver: The receiver to send tasks to the worker threads, this is a address back to the dispatcher + fn spawn_workers( + &self, + system: &mut System, + self_receiver: Box>, + ) { + for _ in 0..self.n_worker_threads { + let worker = WorkerThread::new(self_receiver.clone(), self.worker_queue_size); + system.start_component(worker); + } + } + + /// Enqueue a task to be processed + /// # Parameters + /// - task: The task to enqueue + async fn enqueue_task(&mut self, task: TaskMessage) { + // If a worker is waiting for a task, send it to the worker in FIFO order + // Otherwise, add it to the task queue + match self.waiters.pop() { + Some(channel) => match channel.reply_to.send(task).await { + Ok(_) => {} + Err(e) => { + println!("Error sending task to worker: {:?}", e); + } + }, + None => { + self.task_queue.push(task); + } + } + } + + /// Handle a work request from a worker thread + /// # Parameters + /// - worker: The request for work + /// If no work is available, the worker will be placed in a queue and a task will be sent to it + /// when one is available + async fn handle_work_request(&mut self, request: TaskRequestMessage) { + match self.task_queue.pop() { + Some(task) => match request.reply_to.send(task).await { + Ok(_) => {} + Err(e) => { + println!("Error sending task to worker: {:?}", e); + } + }, + None => { + self.waiters.push(request); + } + } + } +} + +#[async_trait] +impl Configurable for Dispatcher { + async fn try_from_config(worker_config: &WorkerConfig) -> Result> { + Ok(Dispatcher::new( + worker_config.dispatcher.num_worker_threads, + worker_config.dispatcher.dispatcher_queue_size, + worker_config.dispatcher.worker_queue_size, + )) + } +} + +/// A message that a worker thread sends to the dispatcher to request a task +/// # Members +/// - reply_to: The receiver to send the task to, this is the worker thread +#[derive(Debug)] +pub(super) struct TaskRequestMessage { + reply_to: Box>, +} + +impl TaskRequestMessage { + /// Create a new TaskRequestMessage + /// # Parameters + /// - reply_to: The receiver to send the task to, this is the worker thread + /// that is requesting the task + pub(super) fn new(reply_to: Box>) -> Self { + TaskRequestMessage { reply_to } + } +} + +// ============= Component implementation ============= + +#[async_trait] +impl Component for Dispatcher { + fn queue_size(&self) -> usize { + self.queue_size + } + + async fn on_start(&mut self, ctx: &ComponentContext) { + self.spawn_workers(&mut ctx.system.clone(), ctx.sender.as_receiver()); + } +} + +#[async_trait] +impl Handler for Dispatcher { + async fn handle(&mut self, task: TaskMessage, _ctx: &ComponentContext) { + self.enqueue_task(task).await; + } +} + +// Worker sends a request for task +#[async_trait] +impl Handler for Dispatcher { + async fn handle(&mut self, message: TaskRequestMessage, _ctx: &ComponentContext) { + self.handle_work_request(message).await; + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::{ + execution::operator::{wrap, Operator}, + system::System, + }; + use std::sync::{ + atomic::{AtomicUsize, Ordering}, + Arc, + }; + + // Create a component that will schedule DISPATCH_COUNT invocations of the MockOperator + // on an interval of DISPATCH_FREQUENCY_MS. + // Each invocation will sleep for MOCK_OPERATOR_SLEEP_DURATION_MS to simulate work + // Use THREAD_COUNT worker threads + const MOCK_OPERATOR_SLEEP_DURATION_MS: u64 = 100; + const DISPATCH_FREQUENCY_MS: u64 = 5; + const DISPATCH_COUNT: usize = 50; + const THREAD_COUNT: usize = 4; + + #[derive(Debug)] + struct MockOperator {} + #[async_trait] + impl Operator for MockOperator { + type Error = (); + async fn run(&self, input: &f32) -> Result { + // sleep to simulate work + tokio::time::sleep(tokio::time::Duration::from_millis( + MOCK_OPERATOR_SLEEP_DURATION_MS, + )) + .await; + Ok(input.to_string()) + } + } + + #[derive(Debug)] + struct MockDispatchUser { + pub dispatcher: Box>, + counter: Arc, // We expect to recieve DISPATCH_COUNT messages + } + #[async_trait] + impl Component for MockDispatchUser { + fn queue_size(&self) -> usize { + 1000 + } + + async fn on_start(&mut self, ctx: &ComponentContext) { + // dispatch a new task every DISPATCH_FREQUENCY_MS for DISPATCH_COUNT times + let duration = std::time::Duration::from_millis(DISPATCH_FREQUENCY_MS); + ctx.scheduler.schedule_interval( + ctx.sender.clone(), + (), + duration, + Some(DISPATCH_COUNT), + ctx, + ); + } + } + #[async_trait] + impl Handler> for MockDispatchUser { + async fn handle( + &mut self, + message: Result, + ctx: &ComponentContext, + ) { + self.counter.fetch_add(1, Ordering::SeqCst); + let curr_count = self.counter.load(Ordering::SeqCst); + // Cancel self + if curr_count == DISPATCH_COUNT { + ctx.cancellation_token.cancel(); + } + } + } + + #[async_trait] + impl Handler<()> for MockDispatchUser { + async fn handle(&mut self, message: (), ctx: &ComponentContext) { + let task = wrap(Box::new(MockOperator {}), 42.0, ctx.sender.as_receiver()); + let res = self.dispatcher.send(task).await; + } + } + + #[tokio::test] + async fn test_dispatcher() { + let mut system = System::new(); + let dispatcher = Dispatcher::new(THREAD_COUNT, 1000, 1000); + let dispatcher_handle = system.start_component(dispatcher); + let counter = Arc::new(AtomicUsize::new(0)); + let dispatch_user = MockDispatchUser { + dispatcher: dispatcher_handle.receiver(), + counter: counter.clone(), + }; + let mut dispatch_user_handle = system.start_component(dispatch_user); + // yield to allow the component to process the messages + tokio::task::yield_now().await; + // Join on the dispatch user, since it will kill itself after DISPATCH_COUNT messages + dispatch_user_handle.join().await; + // We should have received DISPATCH_COUNT messages + assert_eq!(counter.load(Ordering::SeqCst), DISPATCH_COUNT); + } +} diff --git a/rust/worker/src/execution/mod.rs b/rust/worker/src/execution/mod.rs new file mode 100644 index 00000000000..0000e23f3a3 --- /dev/null +++ b/rust/worker/src/execution/mod.rs @@ -0,0 +1,6 @@ +pub(crate) mod config; +pub(crate) mod dispatcher; +pub(crate) mod operator; +mod operators; +pub(crate) mod orchestration; +mod worker_thread; diff --git a/rust/worker/src/execution/operator.rs b/rust/worker/src/execution/operator.rs new file mode 100644 index 00000000000..935c01eb16e --- /dev/null +++ b/rust/worker/src/execution/operator.rs @@ -0,0 +1,75 @@ +use crate::system::Receiver; +use async_trait::async_trait; +use std::fmt::Debug; + +/// An operator takes a generic input and returns a generic output. +/// It is a definition of a function. +#[async_trait] +pub(super) trait Operator: Send + Sync + Debug +where + I: Send + Sync, + O: Send + Sync, +{ + type Error; + // It would have been nice to do this with a default trait for result + // but that's not stable in rust yet. + async fn run(&self, input: &I) -> Result; +} + +/// A task is a wrapper around an operator and its input. +/// It is a description of a function to be run. +#[derive(Debug)] +struct Task +where + Input: Send + Sync + Debug, + Output: Send + Sync + Debug, +{ + operator: Box>, + input: Input, + reply_channel: Box>>, +} + +/// A message type used by the dispatcher to send tasks to worker threads. +pub(crate) type TaskMessage = Box; + +/// A task wrapper is a trait that can be used to run a task. We use it to +/// erase the I, O types from the Task struct so that tasks. +#[async_trait] +pub(crate) trait TaskWrapper: Send + Debug { + async fn run(&self); +} + +/// Implement the TaskWrapper trait for every Task. This allows us to +/// erase the I, O types from the Task struct so that tasks can be +/// stored in a homogenous queue regardless of their input and output types. +#[async_trait] +impl TaskWrapper for Task +where + Error: Debug, + Input: Send + Sync + Debug, + Output: Send + Sync + Debug, +{ + async fn run(&self) { + let output = self.operator.run(&self.input).await; + let res = self.reply_channel.send(output).await; + // TODO: if this errors, it means the caller was dropped + } +} + +/// Wrap an operator and its input into a task message. +pub(super) fn wrap( + operator: Box>, + input: Input, + reply_channel: Box>>, +) -> TaskMessage +where + Error: Debug + 'static, + Input: Send + Sync + Debug + 'static, + Output: Send + Sync + Debug + 'static, +{ + Box::new(Task { + operator, + input, + reply_channel, + }) +} diff --git a/rust/worker/src/execution/operators/brute_force_knn.rs b/rust/worker/src/execution/operators/brute_force_knn.rs new file mode 100644 index 00000000000..3a9d05c8426 --- /dev/null +++ b/rust/worker/src/execution/operators/brute_force_knn.rs @@ -0,0 +1,126 @@ +use crate::{distance::DistanceFunction, execution::operator::Operator}; +use async_trait::async_trait; +use std::cmp; + +/// The brute force k-nearest neighbors operator is responsible for computing the k-nearest neighbors +/// of a given query vector against a set of vectors using brute force calculation. +/// # Note +/// - Callers should ensure that the input vectors are normalized if using the cosine similarity metric. +#[derive(Debug)] +pub struct BruteForceKnnOperator {} + +/// The input to the brute force k-nearest neighbors operator. +/// # Parameters +/// * `data` - The vectors to query against. +/// * `query` - The query vector. +/// * `k` - The number of nearest neighbors to find. +/// * `distance_metric` - The distance metric to use. +#[derive(Debug)] +pub struct BruteForceKnnOperatorInput { + pub data: Vec>, + pub query: Vec, + pub k: usize, + pub distance_metric: DistanceFunction, +} + +/// The output of the brute force k-nearest neighbors operator. +/// # Parameters +/// * `indices` - The indices of the nearest neighbors. This is a mask against the `query_vecs` input. +/// One row for each query vector. +/// * `distances` - The distances of the nearest neighbors. +/// One row for each query vector. +#[derive(Debug)] +pub struct BruteForceKnnOperatorOutput { + pub indices: Vec, + pub distances: Vec, +} + +pub type BruteForceKnnOperatorResult = Result; + +#[async_trait] +impl Operator for BruteForceKnnOperator { + type Error = (); + + async fn run(&self, input: &BruteForceKnnOperatorInput) -> BruteForceKnnOperatorResult { + // We could use a heap approach here, but for now we just sort the distances and take the + // first k. + let mut sorted_indices_distances = input + .data + .iter() + .map(|data| input.distance_metric.distance(&input.query, data)) + .enumerate() + .collect::>(); + sorted_indices_distances.sort_by(|a, b| a.1.partial_cmp(&b.1).unwrap()); + let (sorted_indices, sorted_distances) = sorted_indices_distances + .drain(..cmp::min(input.k, input.data.len())) + .unzip(); + + Ok(BruteForceKnnOperatorOutput { + indices: sorted_indices, + distances: sorted_distances, + }) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[tokio::test] + async fn test_brute_force_knn_l2sqr() { + let operator = BruteForceKnnOperator {}; + let input = BruteForceKnnOperatorInput { + data: vec![ + vec![0.0, 0.0, 0.0], + vec![0.0, 1.0, 1.0], + vec![7.0, 8.0, 9.0], + ], + query: vec![0.0, 0.0, 0.0], + k: 2, + distance_metric: DistanceFunction::Euclidean, + }; + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.indices, vec![0, 1]); + let distance_1 = 0.0_f32.powi(2) + 1.0_f32.powi(2) + 1.0_f32.powi(2); + assert_eq!(output.distances, vec![0.0, distance_1]); + } + + #[tokio::test] + async fn test_brute_force_knn_cosine() { + let operator = BruteForceKnnOperator {}; + + let norm_1 = (1.0_f32.powi(2) + 2.0_f32.powi(2) + 3.0_f32.powi(2)).sqrt(); + let data_1 = vec![1.0 / norm_1, 2.0 / norm_1, 3.0 / norm_1]; + + let norm_2 = (0.0_f32.powi(2) + -1.0_f32.powi(2) + 6.0_f32.powi(2)).sqrt(); + let data_2 = vec![0.0 / norm_2, -1.0 / norm_2, 6.0 / norm_2]; + + let input = BruteForceKnnOperatorInput { + data: vec![vec![0.0, 1.0, 0.0], data_1.clone(), data_2.clone()], + query: vec![0.0, 1.0, 0.0], + k: 2, + distance_metric: DistanceFunction::InnerProduct, + }; + let output = operator.run(&input).await.unwrap(); + + assert_eq!(output.indices, vec![0, 1]); + let expected_distance_1 = + 1.0f32 - ((data_1[0] * 0.0) + (data_1[1] * 1.0) + (data_1[2] * 0.0)); + assert_eq!(output.distances, vec![0.0, expected_distance_1]); + } + + #[tokio::test] + async fn test_data_less_than_k() { + // If we have less data than k, we should return all the data, sorted by distance. + let operator = BruteForceKnnOperator {}; + let input = BruteForceKnnOperatorInput { + data: vec![vec![0.0, 0.0, 0.0]], + query: vec![0.0, 0.0, 0.0], + k: 2, + distance_metric: DistanceFunction::Euclidean, + }; + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.indices, vec![0]); + assert_eq!(output.distances, vec![0.0]); + } +} diff --git a/rust/worker/src/execution/operators/mod.rs b/rust/worker/src/execution/operators/mod.rs new file mode 100644 index 00000000000..60198481545 --- /dev/null +++ b/rust/worker/src/execution/operators/mod.rs @@ -0,0 +1,3 @@ +pub(super) mod brute_force_knn; +pub(super) mod normalize_vectors; +pub(super) mod pull_log; diff --git a/rust/worker/src/execution/operators/normalize_vectors.rs b/rust/worker/src/execution/operators/normalize_vectors.rs new file mode 100644 index 00000000000..57607ddd2ee --- /dev/null +++ b/rust/worker/src/execution/operators/normalize_vectors.rs @@ -0,0 +1,81 @@ +use crate::execution::operator::Operator; +use async_trait::async_trait; + +const EPS: f32 = 1e-30; + +#[derive(Debug)] +pub struct NormalizeVectorOperator {} + +pub struct NormalizeVectorOperatorInput { + pub vectors: Vec>, +} + +pub struct NormalizeVectorOperatorOutput { + pub normalized_vectors: Vec>, +} + +#[async_trait] +impl Operator + for NormalizeVectorOperator +{ + type Error = (); + + async fn run( + &self, + input: &NormalizeVectorOperatorInput, + ) -> Result { + // TODO: this should not have to reallocate the vectors. We can optimize this later. + let mut normalized_vectors = Vec::with_capacity(input.vectors.len()); + for vector in &input.vectors { + let mut norm = 0.0; + for x in vector { + norm += x * x; + } + let norm = 1.0 / (norm.sqrt() + EPS); + let normalized_vector = vector.iter().map(|x| x * norm).collect(); + normalized_vectors.push(normalized_vector); + } + Ok(NormalizeVectorOperatorOutput { normalized_vectors }) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + const COMPARE_EPS: f32 = 1e-9; + fn float_eps_eq(a: &[f32], b: &[f32]) -> bool { + a.iter() + .zip(b.iter()) + .all(|(a, b)| (a - b).abs() < COMPARE_EPS) + } + + #[tokio::test] + async fn test_normalize_vector() { + let operator = NormalizeVectorOperator {}; + let input = NormalizeVectorOperatorInput { + vectors: vec![ + vec![1.0, 2.0, 3.0], + vec![4.0, 5.0, 6.0], + vec![7.0, 8.0, 9.0], + ], + }; + + let output = operator.run(&input).await.unwrap(); + let expected_output = NormalizeVectorOperatorOutput { + normalized_vectors: vec![ + vec![0.26726124, 0.5345225, 0.8017837], + vec![0.45584232, 0.5698029, 0.6837635], + vec![0.5025707, 0.5743665, 0.64616233], + ], + }; + + for (a, b) in output + .normalized_vectors + .iter() + .zip(expected_output.normalized_vectors.iter()) + { + assert!(float_eps_eq(a, b), "{:?} != {:?}", a, b); + } + } +} diff --git a/rust/worker/src/execution/operators/pull_log.rs b/rust/worker/src/execution/operators/pull_log.rs new file mode 100644 index 00000000000..ad564662654 --- /dev/null +++ b/rust/worker/src/execution/operators/pull_log.rs @@ -0,0 +1,250 @@ +use crate::{ + execution::operator::Operator, + log::log::{Log, PullLogsError}, + types::EmbeddingRecord, +}; +use async_trait::async_trait; +use uuid::Uuid; + +/// The pull logs operator is responsible for reading logs from the log service. +#[derive(Debug)] +pub struct PullLogsOperator { + client: Box, +} + +impl PullLogsOperator { + /// Create a new pull logs operator. + /// # Parameters + /// * `client` - The log client to use for reading logs. + pub fn new(client: Box) -> Box { + Box::new(PullLogsOperator { client }) + } +} + +/// The input to the pull logs operator. +/// # Parameters +/// * `collection_id` - The collection id to read logs from. +/// * `offset` - The offset to start reading logs from. +/// * `batch_size` - The number of log entries to read. +/// * `num_records` - The maximum number of records to read. +/// * `end_timestamp` - The end timestamp to read logs until. +#[derive(Debug)] +pub struct PullLogsInput { + collection_id: Uuid, + offset: i64, + batch_size: i32, + num_records: Option, + end_timestamp: Option, +} + +impl PullLogsInput { + /// Create a new pull logs input. + /// # Parameters + /// * `collection_id` - The collection id to read logs from. + /// * `offset` - The offset to start reading logs from. + /// * `batch_size` - The number of log entries to read. + /// * `num_records` - The maximum number of records to read. + /// * `end_timestamp` - The end timestamp to read logs until. + pub fn new( + collection_id: Uuid, + offset: i64, + batch_size: i32, + num_records: Option, + end_timestamp: Option, + ) -> Self { + PullLogsInput { + collection_id, + offset, + batch_size, + num_records, + end_timestamp, + } + } +} + +/// The output of the pull logs operator. +#[derive(Debug)] +pub struct PullLogsOutput { + logs: Vec>, +} + +impl PullLogsOutput { + /// Create a new pull logs output. + /// # Parameters + /// * `logs` - The logs that were read. + pub fn new(logs: Vec>) -> Self { + PullLogsOutput { logs } + } + + /// Get the log entries that were read by an invocation of the pull logs operator. + /// # Returns + /// The log entries that were read. + pub fn logs(&self) -> &Vec> { + &self.logs + } +} + +pub type PullLogsResult = Result; + +#[async_trait] +impl Operator for PullLogsOperator { + type Error = PullLogsError; + + async fn run(&self, input: &PullLogsInput) -> PullLogsResult { + // We expect the log to be cheaply cloneable, we need to clone it since we need + // a mutable reference to it. Not necessarily the best, but it works for our needs. + let mut client_clone = self.client.clone(); + let batch_size = input.batch_size; + let mut num_records_read = 0; + let mut offset = input.offset; + let mut result = Vec::new(); + loop { + let logs = client_clone + .read( + input.collection_id.to_string(), + offset, + batch_size, + input.end_timestamp, + ) + .await; + + let mut logs = match logs { + Ok(logs) => logs, + Err(e) => { + return Err(e); + } + }; + + if logs.is_empty() { + break; + } + + num_records_read += logs.len(); + offset += batch_size as i64; + result.append(&mut logs); + + // We used a a timestamp and we didn't get a full batch, so we have retrieved + // the last batch of logs relevant to our query + if input.end_timestamp.is_some() && num_records_read < batch_size as usize { + break; + } + + // We have read all the records up to the size we wanted + if input.num_records.is_some() + && num_records_read >= input.num_records.unwrap() as usize + { + break; + } + } + if input.num_records.is_some() && result.len() > input.num_records.unwrap() as usize { + result.truncate(input.num_records.unwrap() as usize); + } + Ok(PullLogsOutput::new(result)) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::log::log::InMemoryLog; + use crate::log::log::LogRecord; + use crate::types::EmbeddingRecord; + use crate::types::Operation; + use num_bigint::BigInt; + use std::str::FromStr; + use uuid::Uuid; + + #[tokio::test] + async fn test_pull_logs() { + let mut log = Box::new(InMemoryLog::new()); + + let collection_uuid_1 = Uuid::from_str("00000000-0000-0000-0000-000000000001").unwrap(); + let collection_id_1 = collection_uuid_1.to_string(); + log.add_log( + collection_id_1.clone(), + Box::new(LogRecord { + collection_id: collection_id_1.clone(), + log_id: 1, + log_id_ts: 1, + record: Box::new(EmbeddingRecord { + id: "embedding_id_1".to_string(), + seq_id: BigInt::from(1), + embedding: None, + encoding: None, + metadata: None, + operation: Operation::Add, + collection_id: collection_uuid_1, + }), + }), + ); + log.add_log( + collection_id_1.clone(), + Box::new(LogRecord { + collection_id: collection_id_1.clone(), + log_id: 2, + log_id_ts: 2, + record: Box::new(EmbeddingRecord { + id: "embedding_id_2".to_string(), + seq_id: BigInt::from(2), + embedding: None, + encoding: None, + metadata: None, + operation: Operation::Add, + collection_id: collection_uuid_1, + }), + }), + ); + + let operator = PullLogsOperator::new(log); + + // Pull all logs from collection 1 + let input = PullLogsInput::new(collection_uuid_1, 0, 1, None, None); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 2); + + // Pull all logs from collection 1 with a large batch size + let input = PullLogsInput::new(collection_uuid_1, 0, 100, None, None); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 2); + + // Pull logs from collection 1 with a limit + let input = PullLogsInput::new(collection_uuid_1, 0, 1, Some(1), None); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + + // Pull logs from collection 1 with an end timestamp + let input = PullLogsInput::new(collection_uuid_1, 0, 1, None, Some(1)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + + // Pull logs from collection 1 with an end timestamp + let input = PullLogsInput::new(collection_uuid_1, 0, 1, None, Some(2)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 2); + + // Pull logs from collection 1 with an end timestamp and a limit + let input = PullLogsInput::new(collection_uuid_1, 0, 1, Some(1), Some(2)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + + // Pull logs from collection 1 with a limit and a large batch size + let input = PullLogsInput::new(collection_uuid_1, 0, 100, Some(1), None); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + + // Pull logs from collection 1 with an end timestamp and a large batch size + let input = PullLogsInput::new(collection_uuid_1, 0, 100, None, Some(1)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + + // Pull logs from collection 1 with an end timestamp and a large batch size + let input = PullLogsInput::new(collection_uuid_1, 0, 100, None, Some(2)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 2); + + // Pull logs from collection 1 with an end timestamp and a limit and a large batch size + let input = PullLogsInput::new(collection_uuid_1, 0, 100, Some(1), Some(2)); + let output = operator.run(&input).await.unwrap(); + assert_eq!(output.logs().len(), 1); + } +} diff --git a/rust/worker/src/execution/orchestration/hnsw.rs b/rust/worker/src/execution/orchestration/hnsw.rs new file mode 100644 index 00000000000..699bfde4b3f --- /dev/null +++ b/rust/worker/src/execution/orchestration/hnsw.rs @@ -0,0 +1,255 @@ +use super::super::operator::{wrap, TaskMessage}; +use super::super::operators::pull_log::{PullLogsInput, PullLogsOperator, PullLogsOutput}; +use crate::distance; +use crate::distance::DistanceFunction; +use crate::errors::ChromaError; +use crate::execution::operators::brute_force_knn::{ + BruteForceKnnOperator, BruteForceKnnOperatorInput, BruteForceKnnOperatorOutput, + BruteForceKnnOperatorResult, +}; +use crate::execution::operators::pull_log::PullLogsResult; +use crate::sysdb::sysdb::SysDb; +use crate::system::System; +use crate::types::VectorQueryResult; +use crate::{ + log::log::Log, + system::{Component, Handler, Receiver}, +}; +use async_trait::async_trait; +use num_bigint::BigInt; +use std::fmt::Debug; +use std::time::{SystemTime, UNIX_EPOCH}; +use uuid::Uuid; + +/** The state of the orchestrator. +In chroma, we have a relatively fixed number of query plans that we can execute. Rather +than a flexible state machine abstraction, we just manually define the states that we +expect to encounter for a given query plan. This is a bit more rigid, but it's also simpler and easier to +understand. We can always add more abstraction later if we need it. +```plaintext + + ┌───► Brute Force ─────┐ + │ │ + Pending ─► PullLogs ─► Dedupe│ ├─► MergeResults ─► Finished + │ │ + └───► HNSW ────────────┘ + +``` +*/ +#[derive(Debug)] +enum ExecutionState { + Pending, + PullLogs, + Dedupe, + QueryKnn, + MergeResults, + Finished, +} + +#[derive(Debug)] +pub(crate) struct HnswQueryOrchestrator { + state: ExecutionState, + // Component Execution + system: System, + // Query state + query_vectors: Vec>, + k: i32, + include_embeddings: bool, + segment_id: Uuid, + // Services + log: Box, + sysdb: Box, + dispatcher: Box>, + // Result channel + result_channel: Option< + tokio::sync::oneshot::Sender>, Box>>, + >, +} + +impl HnswQueryOrchestrator { + pub(crate) fn new( + system: System, + query_vectors: Vec>, + k: i32, + include_embeddings: bool, + segment_id: Uuid, + log: Box, + sysdb: Box, + dispatcher: Box>, + ) -> Self { + HnswQueryOrchestrator { + state: ExecutionState::Pending, + system, + query_vectors, + k, + include_embeddings, + segment_id, + log, + sysdb, + dispatcher, + result_channel: None, + } + } + + /// Get the collection id for a segment id. + /// TODO: This can be cached + async fn get_collection_id_for_segment_id(&mut self, segment_id: Uuid) -> Option { + let segments = self + .sysdb + .get_segments(Some(segment_id), None, None, None, None) + .await; + match segments { + Ok(segments) => match segments.get(0) { + Some(segment) => segment.collection, + None => None, + }, + Err(e) => { + // Log an error and return + return None; + } + } + } + + async fn pull_logs(&mut self, self_address: Box>) { + self.state = ExecutionState::PullLogs; + let operator = PullLogsOperator::new(self.log.clone()); + let collection_id = match self.get_collection_id_for_segment_id(self.segment_id).await { + Some(collection_id) => collection_id, + None => { + // Log an error and reply + return + return; + } + }; + let end_timestamp = SystemTime::now().duration_since(UNIX_EPOCH); + let end_timestamp = match end_timestamp { + // TODO: change protobuf definition to use u64 instead of i64 + Ok(end_timestamp) => end_timestamp.as_nanos() as i64, + Err(e) => { + // Log an error and reply + return + return; + } + }; + let input = PullLogsInput::new(collection_id, 0, 100, None, Some(end_timestamp)); + let task = wrap(operator, input, self_address); + match self.dispatcher.send(task).await { + Ok(_) => (), + Err(e) => { + // TODO: log an error and reply to caller + } + } + } + + /// Run the orchestrator and return the result. + /// # Note + /// Use this over spawning the component directly. This method will start the component and + /// wait for it to finish before returning the result. + pub(crate) async fn run(mut self) -> Result>, Box> { + let (tx, rx) = tokio::sync::oneshot::channel(); + self.result_channel = Some(tx); + let mut handle = self.system.clone().start_component(self); + let result = rx.await; + handle.stop(); + result.unwrap() + } +} + +// ============== Component Implementation ============== + +#[async_trait] +impl Component for HnswQueryOrchestrator { + fn queue_size(&self) -> usize { + 1000 // TODO: make configurable + } + + async fn on_start(&mut self, ctx: &crate::system::ComponentContext) -> () { + self.pull_logs(ctx.sender.as_receiver()).await; + } +} + +// ============== Handlers ============== + +#[async_trait] +impl Handler for HnswQueryOrchestrator { + async fn handle( + &mut self, + message: PullLogsResult, + ctx: &crate::system::ComponentContext, + ) { + self.state = ExecutionState::Dedupe; + + // TODO: implement the remaining state transitions and operators + // TODO: don't need all this cloning and data shuffling, once we land the chunk abstraction + let mut dataset = Vec::new(); + match message { + Ok(logs) => { + for log in logs.logs().iter() { + // TODO: only adds have embeddings, unwrap is fine for now + dataset.push(log.embedding.clone().unwrap()); + } + let bf_input = BruteForceKnnOperatorInput { + data: dataset, + query: self.query_vectors[0].clone(), + k: self.k as usize, + distance_metric: DistanceFunction::Euclidean, + }; + let operator = Box::new(BruteForceKnnOperator {}); + let task = wrap(operator, bf_input, ctx.sender.as_receiver()); + match self.dispatcher.send(task).await { + Ok(_) => (), + Err(e) => { + // TODO: log an error and reply to caller + } + } + } + Err(e) => { + // Log an error + return; + } + } + } +} + +#[async_trait] +impl Handler for HnswQueryOrchestrator { + async fn handle( + &mut self, + message: BruteForceKnnOperatorResult, + ctx: &crate::system::ComponentContext, + ) { + // This is an example of the final state transition and result + let result_channel = match self.result_channel.take() { + Some(tx) => tx, + None => { + // Log an error + return; + } + }; + + match message { + Ok(output) => { + let mut result = Vec::new(); + let mut query_results = Vec::new(); + for (index, distance) in output.indices.iter().zip(output.distances.iter()) { + let query_result = VectorQueryResult { + id: index.to_string(), + seq_id: BigInt::from(0), + distance: *distance, + vector: None, + }; + query_results.push(query_result); + } + result.push(query_results); + + match result_channel.send(Ok(result)) { + Ok(_) => (), + Err(e) => { + // Log an error + } + } + } + Err(_) => { + // Log an error + } + } + } +} diff --git a/rust/worker/src/execution/orchestration/mod.rs b/rust/worker/src/execution/orchestration/mod.rs new file mode 100644 index 00000000000..902c3eaf84d --- /dev/null +++ b/rust/worker/src/execution/orchestration/mod.rs @@ -0,0 +1,3 @@ +mod hnsw; + +pub(crate) use hnsw::*; diff --git a/rust/worker/src/execution/worker_thread.rs b/rust/worker/src/execution/worker_thread.rs new file mode 100644 index 00000000000..d651a725d34 --- /dev/null +++ b/rust/worker/src/execution/worker_thread.rs @@ -0,0 +1,58 @@ +use super::{dispatcher::TaskRequestMessage, operator::TaskMessage}; +use crate::system::{Component, ComponentContext, ComponentRuntime, Handler, Receiver}; +use async_trait::async_trait; +use std::fmt::{Debug, Formatter, Result}; + +/// A worker thread is responsible for executing tasks +/// It sends requests to the dispatcher for new tasks. +/// # Implementation notes +/// - The actor loop will block until work is available +pub(super) struct WorkerThread { + dispatcher: Box>, + queue_size: usize, +} + +impl WorkerThread { + pub(super) fn new( + dispatcher: Box>, + queue_size: usize, + ) -> WorkerThread { + WorkerThread { + dispatcher, + queue_size, + } + } +} + +impl Debug for WorkerThread { + fn fmt(&self, f: &mut Formatter<'_>) -> Result { + f.debug_struct("WorkerThread").finish() + } +} + +#[async_trait] +impl Component for WorkerThread { + fn queue_size(&self) -> usize { + self.queue_size + } + + fn runtime() -> ComponentRuntime { + ComponentRuntime::Dedicated + } + + async fn on_start(&mut self, ctx: &ComponentContext) -> () { + let req = TaskRequestMessage::new(ctx.sender.as_receiver()); + let res = self.dispatcher.send(req).await; + // TODO: what to do with resp? + } +} + +#[async_trait] +impl Handler for WorkerThread { + async fn handle(&mut self, task: TaskMessage, ctx: &ComponentContext) { + task.run().await; + let req: TaskRequestMessage = TaskRequestMessage::new(ctx.sender.as_receiver()); + let res = self.dispatcher.send(req).await; + // TODO: task run should be able to error and we should send it as part of the result + } +} diff --git a/rust/worker/src/index/hnsw.rs b/rust/worker/src/index/hnsw.rs index 49eb5efb2c9..2d83afd17ea 100644 --- a/rust/worker/src/index/hnsw.rs +++ b/rust/worker/src/index/hnsw.rs @@ -307,7 +307,7 @@ extern "C" { pub mod test { use super::*; - use crate::index::types::DistanceFunction; + use crate::distance::DistanceFunction; use crate::index::utils; use rand::Rng; use rayon::prelude::*; diff --git a/rust/worker/src/index/types.rs b/rust/worker/src/index/types.rs index 7af440c947c..092e3fa0c45 100644 --- a/rust/worker/src/index/types.rs +++ b/rust/worker/src/index/types.rs @@ -1,3 +1,4 @@ +use crate::distance::DistanceFunction; use crate::errors::{ChromaError, ErrorCodes}; use crate::types::{MetadataValue, Segment}; use thiserror::Error; @@ -78,58 +79,3 @@ pub(crate) trait PersistentIndex: Index { where Self: Sized; } - -/// The distance function enum. -/// # Description -/// This enum defines the distance functions supported by indices in Chroma. -/// # Variants -/// - `Euclidean` - The Euclidean or l2 norm. -/// - `Cosine` - The cosine distance. Specifically, 1 - cosine. -/// - `InnerProduct` - The inner product. Specifically, 1 - inner product. -/// # Notes -/// See https://docs.trychroma.com/usage-guide#changing-the-distance-function -#[derive(Clone, Debug)] -pub(crate) enum DistanceFunction { - Euclidean, - Cosine, - InnerProduct, -} - -#[derive(Error, Debug)] -pub(crate) enum DistanceFunctionError { - #[error("Invalid distance function `{0}`")] - InvalidDistanceFunction(String), -} - -impl ChromaError for DistanceFunctionError { - fn code(&self) -> ErrorCodes { - match self { - DistanceFunctionError::InvalidDistanceFunction(_) => ErrorCodes::InvalidArgument, - } - } -} - -impl TryFrom<&str> for DistanceFunction { - type Error = DistanceFunctionError; - - fn try_from(value: &str) -> Result { - match value { - "l2" => Ok(DistanceFunction::Euclidean), - "cosine" => Ok(DistanceFunction::Cosine), - "ip" => Ok(DistanceFunction::InnerProduct), - _ => Err(DistanceFunctionError::InvalidDistanceFunction( - value.to_string(), - )), - } - } -} - -impl Into for DistanceFunction { - fn into(self) -> String { - match self { - DistanceFunction::Euclidean => "l2".to_string(), - DistanceFunction::Cosine => "cosine".to_string(), - DistanceFunction::InnerProduct => "ip".to_string(), - } - } -} diff --git a/rust/worker/src/ingest/ingest.rs b/rust/worker/src/ingest/ingest.rs index bacf627cb76..770b0681de8 100644 --- a/rust/worker/src/ingest/ingest.rs +++ b/rust/worker/src/ingest/ingest.rs @@ -305,12 +305,13 @@ impl PulsarIngestTopic { } } +#[async_trait] impl Component for PulsarIngestTopic { fn queue_size(&self) -> usize { 1000 } - fn on_start(&mut self, ctx: &ComponentContext) -> () { + async fn on_start(&mut self, ctx: &ComponentContext) -> () { println!("Starting PulsarIngestTopic for topic"); let stream = match self.consumer.write() { Ok(mut consumer_handle) => consumer_handle.take(), diff --git a/rust/worker/src/ingest/scheduler.rs b/rust/worker/src/ingest/scheduler.rs index 770e9bb0bbf..7a8e7d84b92 100644 --- a/rust/worker/src/ingest/scheduler.rs +++ b/rust/worker/src/ingest/scheduler.rs @@ -49,12 +49,13 @@ impl RoundRobinScheduler { } } +#[async_trait] impl Component for RoundRobinScheduler { fn queue_size(&self) -> usize { 1000 } - fn on_start(&mut self, ctx: &ComponentContext) { + async fn on_start(&mut self, ctx: &ComponentContext) { let sleep_sender = ctx.sender.clone(); let (new_tenant_tx, mut new_tenant_rx) = tokio::sync::mpsc::channel(1000); self.new_tenant_channel = Some(new_tenant_tx); diff --git a/rust/worker/src/lib.rs b/rust/worker/src/lib.rs index 1bc760b0904..1ebc7c2fe7b 100644 --- a/rust/worker/src/lib.rs +++ b/rust/worker/src/lib.rs @@ -2,7 +2,9 @@ mod assignment; mod blockstore; mod compactor; mod config; +mod distance; mod errors; +mod execution; mod index; mod ingest; mod log; @@ -14,15 +16,51 @@ mod sysdb; mod system; mod types; +use crate::sysdb::sysdb::SysDb; use config::Configurable; use memberlist::MemberlistProvider; -use crate::sysdb::sysdb::SysDb; - mod chroma_proto { tonic::include_proto!("chroma"); } +pub async fn query_service_entrypoint() { + let config = config::RootConfig::load(); + let system: system::System = system::System::new(); + let segment_manager = match segment::SegmentManager::try_from_config(&config.worker).await { + Ok(segment_manager) => segment_manager, + Err(err) => { + println!("Failed to create segment manager component: {:?}", err); + return; + } + }; + let dispatcher = match execution::dispatcher::Dispatcher::try_from_config(&config.worker).await + { + Ok(dispatcher) => dispatcher, + Err(err) => { + println!("Failed to create dispatcher component: {:?}", err); + return; + } + }; + let mut dispatcher_handle = system.start_component(dispatcher); + let mut worker_server = match server::WorkerServer::try_from_config(&config.worker).await { + Ok(worker_server) => worker_server, + Err(err) => { + println!("Failed to create worker server component: {:?}", err); + return; + } + }; + worker_server.set_segment_manager(segment_manager.clone()); + worker_server.set_system(system); + worker_server.set_dispatcher(dispatcher_handle.receiver()); + + let server_join_handle = tokio::spawn(async move { + crate::server::WorkerServer::run(worker_server).await; + }); + + let _ = tokio::join!(server_join_handle, dispatcher_handle.join()); +} + pub async fn worker_entrypoint() { let config = config::RootConfig::load(); // Create all the core components and start them @@ -102,5 +140,6 @@ pub async fn worker_entrypoint() { ingest_handle.join(), memberlist_handle.join(), scheduler_handler.join(), + server_join_handle, ); } diff --git a/rust/worker/src/log/log.rs b/rust/worker/src/log/log.rs index 83349355c11..5dae688e6d3 100644 --- a/rust/worker/src/log/log.rs +++ b/rust/worker/src/log/log.rs @@ -9,6 +9,7 @@ use crate::types::EmbeddingRecord; use crate::types::EmbeddingRecordConversionError; use async_trait::async_trait; use std::collections::HashMap; +use std::fmt::Debug; use thiserror::Error; // CollectionInfo is a struct that contains information about a collection for the @@ -30,12 +31,13 @@ pub(crate) struct CollectionRecord { } #[async_trait] -pub(crate) trait Log: Send + Sync + LogClone { +pub(crate) trait Log: Send + Sync + LogClone + Debug { async fn read( &mut self, collection_id: String, offset: i64, batch_size: i32, + end_timestamp: Option, ) -> Result>, PullLogsError>; async fn get_collections_with_new_data( @@ -62,7 +64,7 @@ impl Clone for Box { } } -#[derive(Clone)] +#[derive(Clone, Debug)] pub(crate) struct GrpcLog { client: LogServiceClient, } @@ -118,11 +120,17 @@ impl Log for GrpcLog { collection_id: String, offset: i64, batch_size: i32, + end_timestamp: Option, ) -> Result>, PullLogsError> { + let end_timestamp = match end_timestamp { + Some(end_timestamp) => end_timestamp, + None => -1, + }; let request = self.client.pull_logs(chroma_proto::PullLogsRequest { - collection_id: collection_id, + collection_id, start_from_id: offset, - batch_size: batch_size, + batch_size, + end_timestamp, }); let response = request.await; match response { @@ -184,7 +192,7 @@ impl Log for GrpcLog { pub(crate) enum PullLogsError { #[error("Failed to fetch")] FailedToPullLogs(#[from] tonic::Status), - #[error("Failed to convert proto segment")] + #[error("Failed to convert proto embedding record into EmbeddingRecord")] ConversionError(#[from] EmbeddingRecordConversionError), } @@ -222,8 +230,19 @@ pub(crate) struct LogRecord { pub(crate) record: Box, } +impl Debug for LogRecord { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + f.debug_struct("LogRecord") + .field("collection_id", &self.collection_id) + .field("log_id", &self.log_id) + .field("log_id_ts", &self.log_id_ts) + .field("record", &self.record) + .finish() + } +} + // This is used for testing only -#[derive(Clone)] +#[derive(Clone, Debug)] pub(crate) struct InMemoryLog { logs: HashMap>>, } @@ -248,11 +267,20 @@ impl Log for InMemoryLog { collection_id: String, offset: i64, batch_size: i32, + end_timestamp: Option, ) -> Result>, PullLogsError> { - let logs = self.logs.get(&collection_id).unwrap(); + let end_timestamp = match end_timestamp { + Some(end_timestamp) => end_timestamp, + None => i64::MAX, + }; + + let logs = match self.logs.get(&collection_id) { + Some(logs) => logs, + None => return Ok(Vec::new()), + }; let mut result = Vec::new(); for i in offset..(offset + batch_size as i64) { - if i < logs.len() as i64 { + if i < logs.len() as i64 && logs[i as usize].log_id_ts <= end_timestamp { result.push(logs[i as usize].record.clone()); } } diff --git a/rust/worker/src/log/mod.rs b/rust/worker/src/log/mod.rs index c7873c00ce9..cd769734c48 100644 --- a/rust/worker/src/log/mod.rs +++ b/rust/worker/src/log/mod.rs @@ -1,2 +1,17 @@ pub(crate) mod config; pub(crate) mod log; + +use crate::{ + config::{Configurable, WorkerConfig}, + errors::ChromaError, +}; + +pub(crate) async fn from_config( + config: &WorkerConfig, +) -> Result, Box> { + match &config.log { + crate::log::config::LogConfig::Grpc(_) => { + Ok(Box::new(log::GrpcLog::try_from_config(config).await?)) + } + } +} diff --git a/rust/worker/src/memberlist/memberlist_provider.rs b/rust/worker/src/memberlist/memberlist_provider.rs index ea58228ae98..c6adfbd3f38 100644 --- a/rust/worker/src/memberlist/memberlist_provider.rs +++ b/rust/worker/src/memberlist/memberlist_provider.rs @@ -176,12 +176,13 @@ impl CustomResourceMemberlistProvider { } } +#[async_trait] impl Component for CustomResourceMemberlistProvider { fn queue_size(&self) -> usize { self.queue_size } - fn on_start(&mut self, ctx: &ComponentContext) { + async fn on_start(&mut self, ctx: &ComponentContext) { self.connect_to_kube_stream(ctx); } } @@ -257,7 +258,7 @@ mod tests { let kube_ns = "chroma".to_string(); let kube_client = Client::try_default().await.unwrap(); let memberlist_provider = CustomResourceMemberlistProvider::new( - "worker-memberlist".to_string(), + "query-service-memberlist".to_string(), kube_client.clone(), kube_ns.clone(), 10, diff --git a/rust/worker/src/server.rs b/rust/worker/src/server.rs index 1ecc6ba2e70..205a51b6a97 100644 --- a/rust/worker/src/server.rs +++ b/rust/worker/src/server.rs @@ -1,28 +1,53 @@ -use std::f32::consts::E; - use crate::chroma_proto; use crate::chroma_proto::{ GetVectorsRequest, GetVectorsResponse, QueryVectorsRequest, QueryVectorsResponse, }; use crate::config::{Configurable, WorkerConfig}; use crate::errors::ChromaError; +use crate::execution::operator::TaskMessage; +use crate::execution::orchestration::HnswQueryOrchestrator; +use crate::log::log::Log; use crate::segment::SegmentManager; +use crate::sysdb::sysdb::SysDb; +use crate::system::{Receiver, System}; use crate::types::ScalarEncoding; use async_trait::async_trait; -use kube::core::request; use tonic::{transport::Server, Request, Response, Status}; use uuid::Uuid; pub struct WorkerServer { + // System + system: Option, + // Component dependencies segment_manager: Option, + dispatcher: Option>>, + // Service dependencies + log: Box, + sysdb: Box, port: u16, } #[async_trait] impl Configurable for WorkerServer { async fn try_from_config(config: &WorkerConfig) -> Result> { + let sysdb = match crate::sysdb::from_config(&config).await { + Ok(sysdb) => sysdb, + Err(err) => { + return Err(err); + } + }; + let log = match crate::log::from_config(&config).await { + Ok(log) => log, + Err(err) => { + return Err(err); + } + }; Ok(WorkerServer { segment_manager: None, + dispatcher: None, + system: None, + sysdb, + log, port: config.my_port, }) } @@ -46,6 +71,14 @@ impl WorkerServer { pub(crate) fn set_segment_manager(&mut self, segment_manager: SegmentManager) { self.segment_manager = Some(segment_manager); } + + pub(crate) fn set_dispatcher(&mut self, dispatcher: Box>) { + self.dispatcher = Some(dispatcher); + } + + pub(crate) fn set_system(&mut self, system: System) { + self.system = Some(system); + } } #[tonic::async_trait] @@ -126,6 +159,8 @@ impl chroma_proto::vector_reader_server::VectorReader for WorkerServer { }; let mut proto_results_for_all = Vec::new(); + + let mut query_vectors = Vec::new(); for proto_query_vector in request.vectors { let (query_vector, encoding) = match proto_query_vector.try_into() { Ok((vector, encoding)) => (vector, encoding), @@ -133,31 +168,58 @@ impl chroma_proto::vector_reader_server::VectorReader for WorkerServer { return Err(Status::internal(format!("Error converting vector: {}", e))); } }; + query_vectors.push(query_vector); + } + + let dispatcher = match self.dispatcher { + Some(ref dispatcher) => dispatcher, + None => { + return Err(Status::internal("No dispatcher found")); + } + }; - let results = match segment_manager - .query_vector( - &segment_uuid, - &query_vector, - request.k as usize, + let result = match self.system { + Some(ref system) => { + let orchestrator = HnswQueryOrchestrator::new( + // TODO: Should not have to clone query vectors here + system.clone(), + query_vectors.clone(), + request.k, request.include_embeddings, - ) - .await - { - Ok(results) => results, - Err(e) => { - return Err(Status::internal(format!("Error querying segment: {}", e))); - } - }; + segment_uuid, + self.log.clone(), + self.sysdb.clone(), + dispatcher.clone(), + ); + orchestrator.run().await + } + None => { + return Err(Status::internal("No system found")); + } + }; + let result = match result { + Ok(result) => result, + Err(e) => { + return Err(Status::internal(format!( + "Error running orchestrator: {}", + e + ))); + } + }; + + for result_set in result { let mut proto_results = Vec::new(); - for query_result in results { + for query_result in result_set { let proto_result = chroma_proto::VectorQueryResult { id: query_result.id, seq_id: query_result.seq_id.to_bytes_le().1, distance: query_result.distance, vector: match query_result.vector { Some(vector) => { - match (vector, ScalarEncoding::FLOAT32, query_vector.len()).try_into() { + match (vector, ScalarEncoding::FLOAT32, query_vectors[0].len()) + .try_into() + { Ok(proto_vector) => Some(proto_vector), Err(e) => { return Err(Status::internal(format!( @@ -172,11 +234,9 @@ impl chroma_proto::vector_reader_server::VectorReader for WorkerServer { }; proto_results.push(proto_result); } - - let vector_query_results = chroma_proto::VectorQueryResults { + proto_results_for_all.push(chroma_proto::VectorQueryResults { results: proto_results, - }; - proto_results_for_all.push(vector_query_results); + }); } let resp = chroma_proto::QueryVectorsResponse { diff --git a/rust/worker/src/sysdb/mod.rs b/rust/worker/src/sysdb/mod.rs index 1db5510f893..770fa5cc208 100644 --- a/rust/worker/src/sysdb/mod.rs +++ b/rust/worker/src/sysdb/mod.rs @@ -1,2 +1,17 @@ pub(crate) mod config; pub(crate) mod sysdb; + +use crate::{ + config::{Configurable, WorkerConfig}, + errors::ChromaError, +}; + +pub(crate) async fn from_config( + config: &WorkerConfig, +) -> Result, Box> { + match &config.sysdb { + crate::sysdb::config::SysDbConfig::Grpc(_) => { + Ok(Box::new(sysdb::GrpcSysDb::try_from_config(config).await?)) + } + } +} diff --git a/rust/worker/src/sysdb/sysdb.rs b/rust/worker/src/sysdb/sysdb.rs index ba8be18fdf5..990268e66ec 100644 --- a/rust/worker/src/sysdb/sysdb.rs +++ b/rust/worker/src/sysdb/sysdb.rs @@ -1,6 +1,4 @@ -use async_trait::async_trait; -use uuid::Uuid; - +use super::config::SysDbConfig; use crate::chroma_proto; use crate::config::{Configurable, WorkerConfig}; use crate::types::{CollectionConversionError, SegmentConversionError}; @@ -9,15 +7,16 @@ use crate::{ errors::{ChromaError, ErrorCodes}, types::{Collection, Segment, SegmentScope}, }; +use async_trait::async_trait; +use std::fmt::Debug; use thiserror::Error; - -use super::config::SysDbConfig; +use uuid::Uuid; const DEFAULT_DATBASE: &str = "default_database"; const DEFAULT_TENANT: &str = "default_tenant"; #[async_trait] -pub(crate) trait SysDb: Send + Sync + SysDbClone { +pub(crate) trait SysDb: Send + Sync + SysDbClone + Debug { async fn get_collections( &mut self, collection_id: Option, @@ -59,7 +58,7 @@ impl Clone for Box { } } -#[derive(Clone)] +#[derive(Clone, Debug)] // Since this uses tonic transport channel, cloning is cheap. Each client only supports // one inflight request at a time, so we need to clone the client for each requester. pub(crate) struct GrpcSysDb { diff --git a/rust/worker/src/system/executor.rs b/rust/worker/src/system/executor.rs index f5b38779183..4877273b70e 100644 --- a/rust/worker/src/system/executor.rs +++ b/rust/worker/src/system/executor.rs @@ -53,6 +53,14 @@ where } pub(super) async fn run(&mut self, mut channel: tokio::sync::mpsc::Receiver>) { + self.handler + .on_start(&ComponentContext { + system: self.inner.system.clone(), + sender: self.inner.sender.clone(), + cancellation_token: self.inner.cancellation_token.clone(), + scheduler: self.inner.scheduler.clone(), + }) + .await; loop { select! { _ = self.inner.cancellation_token.cancelled() => { diff --git a/rust/worker/src/system/scheduler.rs b/rust/worker/src/system/scheduler.rs index 0168f05ff2a..aab82e5d098 100644 --- a/rust/worker/src/system/scheduler.rs +++ b/rust/worker/src/system/scheduler.rs @@ -1,6 +1,5 @@ use parking_lot::RwLock; use std::fmt::Debug; -use std::num; use std::sync::Arc; use std::time::Duration; use tokio::select; @@ -10,12 +9,13 @@ use super::{ executor::ComponentExecutor, sender::Sender, system::System, Receiver, ReceiverImpl, Wrapper, }; +#[derive(Debug)] pub(crate) struct SchedulerTaskHandle { join_handle: Option>, cancel: tokio_util::sync::CancellationToken, } -#[derive(Clone)] +#[derive(Clone, Debug)] pub(crate) struct Scheduler { handles: Arc>>, } @@ -175,12 +175,13 @@ mod tests { } } + #[async_trait] impl Component for TestComponent { fn queue_size(&self) -> usize { self.queue_size } - fn on_start(&mut self, ctx: &ComponentContext) -> () { + async fn on_start(&mut self, ctx: &ComponentContext) -> () { let duration = Duration::from_millis(100); ctx.scheduler .schedule(ctx.sender.clone(), ScheduleMessage {}, duration, ctx); diff --git a/rust/worker/src/system/sender.rs b/rust/worker/src/system/sender.rs index df2e1bc5587..d9fb0785418 100644 --- a/rust/worker/src/system/sender.rs +++ b/rust/worker/src/system/sender.rs @@ -51,7 +51,6 @@ where } // Sender - pub(crate) struct Sender where C: Component + Send + 'static, @@ -78,6 +77,14 @@ where Err(_) => Err(ChannelError::SendError), } } + + pub(crate) fn as_receiver(&self) -> Box> + where + C: Component + Handler, + M: Debug + Send + 'static, + { + Box::new(ReceiverImpl::new(self.sender.clone())) + } } impl Clone for Sender @@ -94,7 +101,7 @@ where // Reciever Traits #[async_trait] -pub(crate) trait Receiver: Send + Sync + ReceiverClone { +pub(crate) trait Receiver: Send + Sync + Debug + ReceiverClone { async fn send(&self, message: M) -> Result<(), ChannelError>; } @@ -118,7 +125,7 @@ where } // Reciever Impls - +#[derive(Debug)] pub(super) struct ReceiverImpl where C: Component, diff --git a/rust/worker/src/system/system.rs b/rust/worker/src/system/system.rs index d2f573e98a9..0d9f4738625 100644 --- a/rust/worker/src/system/system.rs +++ b/rust/worker/src/system/system.rs @@ -1,23 +1,21 @@ -use std::fmt::Debug; -use std::sync::Arc; - +use super::scheduler::Scheduler; +use super::sender::Sender; +use super::ComponentContext; +use super::ComponentRuntime; +use super::{executor::ComponentExecutor, Component, ComponentHandle, Handler, StreamHandler}; use futures::Stream; use futures::StreamExt; +use std::fmt::Debug; +use std::sync::Arc; use tokio::runtime::Builder; use tokio::{pin, select}; -use super::ComponentRuntime; -// use super::executor::StreamComponentExecutor; -use super::scheduler::Scheduler; -use super::sender::{self, Sender, Wrapper}; -use super::{executor, ComponentContext}; -use super::{executor::ComponentExecutor, Component, ComponentHandle, Handler, StreamHandler}; - -#[derive(Clone)] +#[derive(Clone, Debug)] pub(crate) struct System { inner: Arc, } +#[derive(Debug)] struct Inner { scheduler: Scheduler, } @@ -31,19 +29,13 @@ impl System { } } - pub(crate) fn start_component(&mut self, mut component: C) -> ComponentHandle + pub(crate) fn start_component(&self, component: C) -> ComponentHandle where C: Component + Send + 'static, { let (tx, rx) = tokio::sync::mpsc::channel(component.queue_size()); let sender = Sender::new(tx); let cancel_token = tokio_util::sync::CancellationToken::new(); - let _ = component.on_start(&mut ComponentContext { - system: self.clone(), - sender: sender.clone(), - cancellation_token: cancel_token.clone(), - scheduler: self.inner.scheduler.clone(), - }); let mut executor = ComponentExecutor::new( sender.clone(), cancel_token.clone(), @@ -53,14 +45,14 @@ impl System { ); match C::runtime() { - ComponentRuntime::Global => { + ComponentRuntime::Inherit => { let join_handle = tokio::spawn(async move { executor.run(rx).await }); return ComponentHandle::new(cancel_token, Some(join_handle), sender); } ComponentRuntime::Dedicated => { println!("Spawning on dedicated thread"); // Spawn on a dedicated thread - let mut rt = Builder::new_current_thread().enable_all().build().unwrap(); + let rt = Builder::new_current_thread().enable_all().build().unwrap(); let join_handle = std::thread::spawn(move || { rt.block_on(async move { executor.run(rx).await }); }); diff --git a/rust/worker/src/system/types.rs b/rust/worker/src/system/types.rs index 554020d1276..6b41e62cf64 100644 --- a/rust/worker/src/system/types.rs +++ b/rust/worker/src/system/types.rs @@ -17,7 +17,7 @@ pub(crate) enum ComponentState { #[derive(Debug, PartialEq)] pub(crate) enum ComponentRuntime { - Global, + Inherit, Dedicated, } @@ -30,12 +30,13 @@ pub(crate) enum ComponentRuntime { /// # Methods /// - queue_size: The size of the queue to use for the component before it starts dropping messages /// - on_start: Called when the component is started +#[async_trait] pub(crate) trait Component: Send + Sized + Debug + 'static { fn queue_size(&self) -> usize; fn runtime() -> ComponentRuntime { - ComponentRuntime::Global + ComponentRuntime::Inherit } - fn on_start(&mut self, ctx: &ComponentContext) -> () {} + async fn on_start(&mut self, ctx: &ComponentContext) -> () {} } /// A handler is a component that can process messages of a given type. @@ -166,12 +167,13 @@ mod tests { } impl StreamHandler for TestComponent {} + #[async_trait] impl Component for TestComponent { fn queue_size(&self) -> usize { self.queue_size } - fn on_start(&mut self, ctx: &ComponentContext) -> () { + async fn on_start(&mut self, ctx: &ComponentContext) -> () { let test_stream = stream::iter(vec![1, 2, 3]); self.register_stream(test_stream, ctx); } diff --git a/rust/worker/src/types/collection.rs b/rust/worker/src/types/collection.rs index 049e0c4a133..ecfdeef1346 100644 --- a/rust/worker/src/types/collection.rs +++ b/rust/worker/src/types/collection.rs @@ -15,6 +15,8 @@ pub(crate) struct Collection { pub(crate) dimension: Option, pub(crate) tenant: String, pub(crate) database: String, + pub(crate) log_position: i64, + pub(crate) version: i32, } #[derive(Error, Debug)] @@ -57,6 +59,8 @@ impl TryFrom for Collection { dimension: proto_collection.dimension, tenant: proto_collection.tenant, database: proto_collection.database, + log_position: proto_collection.log_position, + version: proto_collection.version, }) } } @@ -75,6 +79,8 @@ mod test { dimension: None, tenant: "baz".to_string(), database: "qux".to_string(), + log_position: 0, + version: 0, }; let converted_collection: Collection = proto_collection.try_into().unwrap(); assert_eq!(converted_collection.id, Uuid::nil()); diff --git a/rust/worker/src/types/segment.rs b/rust/worker/src/types/segment.rs index 4b39161e2b2..d85d1293eea 100644 --- a/rust/worker/src/types/segment.rs +++ b/rust/worker/src/types/segment.rs @@ -3,6 +3,8 @@ use crate::{ chroma_proto, errors::{ChromaError, ErrorCodes}, }; +use std::collections::HashMap; +use std::vec::Vec; use thiserror::Error; use uuid::Uuid; @@ -19,6 +21,7 @@ pub(crate) struct Segment { pub(crate) topic: Option, pub(crate) collection: Option, pub(crate) metadata: Option, + pub(crate) file_path: HashMap>, } #[derive(Error, Debug)] @@ -48,6 +51,8 @@ impl TryFrom for Segment { type Error = SegmentConversionError; fn try_from(proto_segment: chroma_proto::Segment) -> Result { + let mut proto_segment = proto_segment; + let segment_uuid = match Uuid::try_parse(&proto_segment.id) { Ok(uuid) => uuid, Err(_) => return Err(SegmentConversionError::InvalidUuid), @@ -79,6 +84,12 @@ impl TryFrom for Segment { } }; + let mut file_paths = HashMap::new(); + let drain = proto_segment.file_paths.drain(); + for (key, mut value) in drain { + file_paths.insert(key, value.paths); + } + Ok(Segment { id: segment_uuid, r#type: segment_type, @@ -86,6 +97,7 @@ impl TryFrom for Segment { topic: proto_segment.topic, collection: collection_uuid, metadata: segment_metadata, + file_path: file_paths, }) } } @@ -115,6 +127,7 @@ mod tests { topic: Some("test".to_string()), collection: Some("00000000-0000-0000-0000-000000000000".to_string()), metadata: Some(metadata), + file_paths: HashMap::new(), }; let converted_segment: Segment = proto_segment.try_into().unwrap(); assert_eq!(converted_segment.id, Uuid::nil());