2021-08-25 03:29:52 +00:00
## 6. Proxy
2021-10-26 06:40:32 +00:00
As the user access layer of Milvus, Proxy mainly plays a role that does some checks and preprocessing for requests from
2021-10-30 10:55:26 +00:00
clients and then forwards these requests to other components, such as Root Coordinator, Data Coordinator, Query
2021-10-29 05:18:41 +00:00
Coordinator, Index Coordinator. The below figure shows how Proxy interacts with other components.
2021-08-25 03:29:52 +00:00
< img src = "./graphs/proxy.png" width = 700 >
Proxy divides requests from clients into three classes: `DdRequest` , `DmRequest` , `DqRequest` .
2021-10-29 05:18:41 +00:00
DdRequest is the shorthand of `Data Definition Request` . It means an operation on the meta information of
2021-08-25 03:29:52 +00:00
collections, including two parts. One part is of the writing operations on collections, such as defining a schema,
2021-12-01 07:32:10 +00:00
creating or dropping a partition, creating or dropping the index, etc. Another part is of the reading operations on
2021-08-25 03:29:52 +00:00
collections, such as listing all collections or partitions, checking if a collection or a partition exists.
DmRequest means `Data Manipulation Request` . These requests perform a writing operation on collections, including
inserting records into a collection, deleting some specific records of a collection.
DqRequest means `Data Query Request` . These requests perform a reading operation on collections, such as searching on a
2021-12-01 07:32:10 +00:00
collection, querying for specific records of a collection, etc.
2021-08-25 03:29:52 +00:00
For every request, Proxy will first check if it's valid to be executed by Milvus and if the request is invalid then
Proxy will return the error to clients and won't continue to forward this request to other components. The check
2021-12-17 14:08:50 +00:00
operation of Proxy includes two parts, one part is static check and another is dynamic check. The static check includes
2021-12-20 14:59:05 +00:00
parameters check, constraints check, etc. The dynamic check will check some related dependencies of the request, take
2021-12-22 11:15:55 +00:00
search requests as an example, Proxy should check if the related collection exists in Milvus.
2021-08-25 03:29:52 +00:00
2021-10-25 12:11:42 +00:00
Also, Proxy will do some preprocessing for every request. Proxy will do little things for some requests in the
2021-12-08 04:09:04 +00:00
preprocessing stage and a lot more for other requests. Every object in Milvus will be assigned with an `ID` , such as
2021-10-25 12:11:42 +00:00
`CollectionID` , `PartitionID` , `IndexID` , `SegmentID` , etc. Components in Milvus communicate with each other by the
2021-10-30 10:55:26 +00:00
object IDs, however, users only know the object name. So as a user access layer of Milvus, Proxy should translate
2021-12-15 12:59:10 +00:00
the object name into object ID. Also taking search request as an example, Proxy should translate the `CollectionName` into
2021-12-17 10:56:17 +00:00
`CollectionID` and then the Query Node will recognize the request. Proxy holds a cache that translates object name into
2021-08-25 03:29:52 +00:00
object id and dynamically updates the cache.
2021-10-14 10:06:43 +00:00
#### 6.0 Service Discovery based on etcd
As you know, Proxy depends on some other components. So how Proxy knows the other nodes' information such as host and port,
2021-10-16 14:26:52 +00:00
it is also called how Milvus implements the service discovery mechanism? As a cloud-native vector database, Milvus uses
etcd to provide service registration and service discovery. Every node in Milvus registers its node information(including host,
port, ID etc) into etcd after startup. Nodes should specify the prefix and key of etcd when registration. Nodes with the
same type have the same prefix in etcd, while nodes with different types have different prefixes. Every key in etcd can be assigned with a lease, and you can specify the `time-to-live(ttl)` of the
lease. Milvus uses this mechanism to check if a node is online. When a node is healthy, it will continuously renew the
2021-10-14 10:06:43 +00:00
lease in etcd. Otherwise, if some exceptions occurred in a node, or a node stopped to renew the lease, etcd would delete
2021-10-16 14:26:52 +00:00
the related node information. By using this mechanism, nodes in Milvus know if there are any other nodes going to be
2021-10-14 10:06:43 +00:00
online or offline and synchronize the latest node information.
2021-08-25 03:29:52 +00:00
2021-10-29 09:38:48 +00:00
#### 6.1 Interaction with Root Coordinator
2021-08-25 03:29:52 +00:00
Proxy will forward the DdRequest to Root Coordinator. These requests include:
2021-10-16 14:26:52 +00:00
- CreateCollection
- DropCollection
- HasCollection
- DescribeCollection
- ShowCollections
- CreatePartition
- DropPartition
- HasPartition
- ShowPartitions
- CreateIndex
- DropIndex
- DescribeIndex
- GetIndexBuildProgress
- GetIndexState
2021-08-25 03:29:52 +00:00
2021-10-15 09:50:43 +00:00
Proxy handles the DdRequest sequentially. When and only when the earlier entered requests are done, the next request
2021-10-16 14:26:52 +00:00
would be handled. Proxy forwards these requests to Root Coordinator, waits until getting results from Root Coordinator, and then
2021-10-15 09:50:43 +00:00
returns to clients with results or errors.
2021-08-25 03:29:52 +00:00
2021-12-20 15:00:59 +00:00
Milvus does not support transactions, but it should guarantee the deterministic execution of every operation. A timestamp
2021-10-16 14:26:52 +00:00
is tagged on each request. When a request enters Milvus, Proxy tags a timestamp that was assigned by Root Coordinator.
The component that assigns timestamp in Root Coordinator is called `Timestamp Oracle (TSO)` . TSO ensures that each
2021-10-15 09:50:43 +00:00
timestamp is globally increasing.
2021-08-25 03:29:52 +00:00
Milvus 2.0 implements the unified Lambda architecture, which integrates the processing of the incremental and historical
data. Compared with the Kappa architecture, Milvus 2.0 introduces log backfill, which stores log snapshots and indexes
in the object storage to improve failure recovery efficiency and query performance. To break unbounded (stream) data
down into bounded windows, Milvus embraces a new watermark mechanism, which slices the stream data into multiple message
packs according to write time or event time, and maintains a timeline for users to query by time.
2021-12-20 15:02:53 +00:00
To support this watermark mechanism, Proxy should report the timestamp statistics of the physical channel to Root
2021-08-25 03:29:52 +00:00
Coordinator periodically. When Proxy knows all operations of a specific were done before a `ts` , then Proxy will report
2021-11-04 07:30:45 +00:00
the `ts` and inform Root Coordinator that updates the timestamp statistics.
2021-08-25 03:29:52 +00:00
2021-10-26 02:36:34 +00:00
Proxy holds a cache about meta information of collections. The meta information includes `CollectionID` , `Schema` ,
`PartitionID` , etc. Components in Milvus communicate with each other using `CollectionID` and `PartitionID` , so the
2021-12-22 11:17:56 +00:00
object name in a request will be translated to object ID in Proxy. When the meta is not hit in the cache, Proxy will update
2021-08-25 03:29:52 +00:00
the cache from Root Coordinator. At the same time, in order to keep the consistency of cache, when there are any changes
2021-10-26 02:36:34 +00:00
of meta information in Root Coordinator, it will inform all Proxies to clear the related meta cache, and any newer
requests will get the latest meta information.
2021-08-25 03:29:52 +00:00
2021-12-17 14:22:55 +00:00
For inserts to a collection that is auto_id configured in the collection schema, Proxy assigns a primary key for
2022-01-05 05:21:24 +00:00
every row of insert requests. For now, the only supported data type of auto-generated primary field is `int64` . Proxy
2021-12-29 04:05:53 +00:00
applies for a batch of primary keys from Root Coordinator and caches them for local assignments. When the primary keys in the cache
2021-12-23 07:19:41 +00:00
are not enough, Proxy will continue to apply for another batch of primary keys.
2021-10-16 13:59:21 +00:00
Proxy forwards ReleaseCollection and ReleasePartition to Query Coordinator, Query Coordinator then informs Root
Coordinator the events. After that, Root Coordinator will inform all Proxies to close the search-related and
2021-08-25 03:29:52 +00:00
search-result-related message streams.
#### 6.2 Interaction with MsgStream
2021-10-19 09:06:42 +00:00
In Milvus 2.0, the log broker serves as the system 'backbone': All data insert and update operations must go through the
2021-08-25 03:29:52 +00:00
log broker, and worker nodes execute CRUD operations by subscribing to and consuming logs. This design reduces system
complexity by moving core functions such as data persistence and flashback down to the storage layer, and log pub-sub
make the system even more flexible and better positioned for future scaling.
So, we should configure a group of Virtual Channels for every collection. Now every virtual channel has a unique related
2021-12-22 11:24:14 +00:00
physical channel. Take pulsar as an example, the physical channel mentioned here means the topic of pulsar.
2021-08-25 03:29:52 +00:00
For DmRequest, data will be written to DmChannels, while for DqRequest, requests will be written to DqRequestChannel,
and the corresponding results of query requests will be written to DqResultChannel.
As the number of tables increases, the number of DmChannels increases on demand, and the number of physical channels
also increases on demand. In the future, the number of physical channels in the system can also be limited to a fixed
number, such as 1024. In this case, the same physical channel will be mapped to virtual channels of different
2021-12-22 11:26:07 +00:00
collections. Shown in the figure below.
2021-08-25 03:29:52 +00:00
![collection_dm_channels ](./graphs/collection_dm_channels.png )
2021-10-27 10:56:52 +00:00
When a collection is created, Root Coordinator need to decide the number of its DmChannels and the physical
channels mapped by each virtual channel, and persist these kinds of information as the meta information of the collection;
In addition, when the system finds that the collection receives DmRequest frequently, we can allocate more virtual channels
2021-08-25 03:29:52 +00:00
to the collection to increase the parallelism and thus increase the system throughput. This function is a key point of
future work.
For DqRequest, request and result data are written to the stream. The request data will be written to DqRequestChannel,
2021-12-20 08:51:54 +00:00
and the result data will be written to DqResultChannel. Proxy will write the request of the collection into the
2023-10-24 01:30:10 +00:00
DqRequestChannel, and the DqRequestChannel will be jointly subscribed by a group of query nodes. When all query nodes
2021-08-25 03:29:52 +00:00
receive the DqRequest, they will write the query results into the DqResultChannel corresponding to the collection. As
2021-12-20 08:51:54 +00:00
the consumer of the DqResultChannel, Proxy is responsible for collecting the query results and aggregating them,
2021-08-25 03:29:52 +00:00
The result is then returned to the client.
2021-12-20 08:51:54 +00:00
The allocation logic of DqRequestChannel and DqResultChannel of a collection is allocated by the Query Coordinator. Proxy needs to ask the Query Coordinator for the names of DqRequestChannel and DqResultChannel of a collection.
2021-08-25 03:29:52 +00:00
DqRequestChannel and DqResultChannel do not need to be persisted and can be freely allocated by Query Coordinator. In
the actual implementation, the DqRequestChannel of each collection can be exclusive, and the DqResultChannel can be
2021-12-22 07:05:17 +00:00
exclusive or shared by all collections on Proxy. When Proxy applies for the DqRequestChannel and DqResultChannel
2021-12-20 08:51:54 +00:00
information of the collection from the Query Coordinator, it can attach Proxy's own ID: ProxyID.
2021-08-25 03:29:52 +00:00
2021-12-23 15:23:16 +00:00
With DqRequestChannel of the collection, Proxy will create a msgstream object to generate data into
2021-12-22 07:05:17 +00:00
DqRequestChannel. With the DqResultChannel of the collection, Proxy will create a msgstream object, and Proxy will
2021-08-25 03:29:52 +00:00
consume the data in the DqResultChannel. When these msgstream objects are closed, messages cannot be written to or
consumed from them.
![proxy_channels ](./graphs/proxy_channels.png )
#### 6.3 Interaction with Data Coordinator
In Milvus, segment is the basic read-write and search unit of data. After consuming the data in DmChannel, the data
nodes will store the data in the object storage in the unit of segment (in the actual implementation, the segment will
be divided into multiple small files for writing). In Milvus, segment is uniquely identified by segment ID, and the
allocation logic of segment ID is the responsibility of the Data Coordinator. The SegmentID to which each row of data is
2021-12-14 10:41:30 +00:00
written needs to be determined before writing to DmChannel. For a write operation, Proxy will hash each row of data
2021-10-28 12:28:59 +00:00
again according to the hash value of its primary key, and then determine into which DmChannel each row of data enters. After
2021-12-14 10:41:30 +00:00
collecting the number of pieces to be written by each DmChannel, it applies to the data coordinator for which SegmentIDs the
2021-12-15 11:13:41 +00:00
newly written data of these dmchannels belong to. In the specific implementation, Proxy needs to preallocate some
2021-10-28 12:28:59 +00:00
quotas to the Data Coordinator to avoid frequent direct GRPC communication with the Data Coordinator.
2021-08-25 03:29:52 +00:00
One consideration for uniformly assigning SegmentIDs by Data Coordinator is that Data Coordinator is responsible for
coordinating the total number of each segment not to be too large, and the location is near a water level, so that the
size of the segment is limited to a certain range.
2021-12-15 11:13:41 +00:00
Other interactions between Proxy and Data Coordinator are mainly reflected in Proxy querying Data Coordinator for
2021-08-25 03:29:52 +00:00
the status and statistical information of the segment of the collection. LoadCollection is an example. The
synchronization semantics of the current LoadCollection needs to know the number of rows currently persisted, so the
Proxy needs to ask the Data Coordinator for the total number of rows currently persisted.
#### 6.4 Interaction with Query Coordinator
2021-12-15 11:13:41 +00:00
For LoadCollection, LoadPartition, ReleaseCollection, ReleasePartition requests, Proxy directly forwards these
requests to Query Coordinator for execution after checking and preprocessing these requests. When Proxy receives
2021-11-05 09:38:02 +00:00
feedback from Query Coordinator, it returns the feedback results to the clients.
2021-08-25 03:29:52 +00:00
The semantics of the Load operation is to load Collection or Partition from persistent storage into the memory of Query
Nodes, or import streaming data into QueryNode so that it can be queried. If the load operation is not performed, the
query operation on the Collection or Partition cannot be performed. For the Load operation, Query Coordinator is
responsible for allocating DmChannels to different queryNodes and subscribing to them, and is responsible for receiving
those stream data. QueryCoordinator also allocates and loads the segments that have been persisted in storage in
Query Nodes.
The semantics of the Release operation is the reverse operation of the Load operation, and the function is to unload the
data of the Collection or Partition from the memory. For Release operations, Query Coordinator is responsible for
notifying query nodes to unload the corresponding Collection or Partition in memory, and then sending the
ReleaseDqlMessageStream command to Root Coordinator, and Root Coordinator is responsible for broadcasting the
2021-12-29 04:07:37 +00:00
ReleaseDqlMessageStream command to all Proxies, so that all related streams used to send search requests and receive
2021-08-25 03:29:52 +00:00
search result in Proxy will be closed.
2021-11-05 09:38:02 +00:00
The other interaction between Proxy and Query Coordinator is that Proxy needs to query from Query Coordinator for statistics
2021-08-25 03:29:52 +00:00
about Collection, Partition, and Segment. Taking ShowCollections as an example, if the ShowCollections parameter
specifies that the query is for Collections that have been loaded into memory, the ShowCollection request will be
forwarded to QueryCoordinator, and QueryCoordinator will return a list of all the recorded Collections loaded into
memory. Taking LoadCollection as another example, its synchronization semantics is that the number of rows loaded in
2021-12-17 10:55:57 +00:00
the memory must be no less than the number of rows that have been persisted. This requires Proxy to ask the Query
2021-08-25 03:29:52 +00:00
Coordinator for the sum of the number of rows currently loaded into the query nodes in the Collection.
#### 6.5 Decouple Functionality and Communication
![decouple ](./graphs/decouple.jpeg )
As shown in the figure above, there are interactions between various types of components in the Milvus2.0 system. The
implementation of these interactions will vary according to the deployment of Milvus. Milvus Standalone allows the
various components of Milvus as a whole independent process to be deployed on a single node. Milvus Cluster distributes
all components on multiple nodes. In Milvus Standalone, the interaction between components can be parameter transfer
between functions or communication between Grpc. The log system can be either Pulsar or RocksDb. In Milvus Cluster, the
communication between components is mostly undertaken by grpc, and the message flow is mostly by Pulsar.
2021-11-12 11:13:11 +00:00
Therefore, in the original design, Milvus 2.0 decoupled the core function of the component and the communication between
2021-12-17 10:55:57 +00:00
components. Taking Proxy as an example, the core function of Proxy component is determined and has nothing to do
2021-08-25 03:29:52 +00:00
with the deployment form. In the project's internal/proxy directory, it contains the functions of the core components of
2021-12-23 07:21:33 +00:00
Proxy; and internal/distributed/proxy contain the core functions of Proxy in the deployment of cluster distributed
2021-08-25 03:29:52 +00:00
which contains the re-encapsulation and communication implementation of Proxy. The following article will mainly
2021-12-17 10:55:57 +00:00
introduce the functions of Proxy core layer.
2021-08-25 03:29:52 +00:00
#### 6.6 Core Components of Proxy
Proxy is mainly composed of four modules: taskScheduler, channelsMgr, channelsTimeTicker, globalMetaCache. taskScheduler
is responsible for task scheduling; channelsMgr is responsible for the management of DmChannels, DqRequestChannel,
DqResultChannel and corresponding MsgStream objects of each Collection; channelsTimeTicker is responsible for collecting
the timestamp information of all physical Channels regularly; globalMetaCache is responsible for caching the metadata of
Collection and Partition.
##### 6.6.1 taskScheduler
There are three main functions in taskScheduler:
2021-10-16 14:26:52 +00:00
- Schedule task
- Maintain the snapshot of timestamp statistics
2021-12-23 07:23:24 +00:00
- Receive the search results from all streams and then distribute them to related task
2021-08-25 03:29:52 +00:00
taskScheduler maintains three queues: ddQueue, dmQueue and dqQueue correspond to DdRequest, DmRequest, and DqRequest
respectively. The interface of taskQueue is defined as follows:
```go
2021-10-29 09:38:48 +00:00
type taskQueue interface {
utChan() < -chan int
utEmpty() bool
utFull() bool
addUnissuedTask(t task) error
FrontUnissuedTask() task
PopUnissuedTask() task
AddActiveTask(t task)
PopActiveTask(tID UniqueID) task
getTaskByReqID(reqID UniqueID) task
TaskDoneTest(ts Timestamp) bool
Enqueue(t task) error
setMaxTaskNum(num int64)
getMaxTaskNum() int64
2021-08-25 03:29:52 +00:00
}
```
Proxy encapsulates each request with a corresponding task object. Each task object implements the task interface. The
definition of the task interface is as follows:
```go
type task interface {
2021-10-29 09:38:48 +00:00
TraceCtx() context.Context
ID() UniqueID // return ReqID
SetID(uid UniqueID) // set ReqID
Name() string
Type() commonpb.MsgType
BeginTs() Timestamp
EndTs() Timestamp
SetTs(ts Timestamp)
OnEnqueue() error
PreExecute(ctx context.Context) error
Execute(ctx context.Context) error
PostExecute(ctx context.Context) error
WaitToFinish() error
Notify(err error)
2021-08-25 03:29:52 +00:00
}
```
Each specific task object must implement the interface defined by the task.
2021-12-24 05:39:30 +00:00
The key members of taskQueue are unissuedTasks of type List and activateTasks of type maps. Among them, unissuedTasks
2021-08-25 03:29:52 +00:00
contain all tasks that have not been scheduled, and activateTasks contain all tasks that are being scheduled.
When the external caller of taskScheduler stuffs the task into the corresponding taskQueue, it will call the OnEnqueue
interface of the task. When OnEnqueue is called, the SetID of the task will be called to assign the taskID to the task.
The taskID is globally unique and is used to identify the task. OnEnqueue will also call task.SetTs to set the timestamp
for the task. It can be seen that the timestamp of entering the queue must be greater than the timestamp that already
exists in the queue, and it will also be greater than the timestamp of the task that exists in activateTask. At the end
of the task's OnEnqueue, call the taskQueue's addUnissuedTask to add the task to the unissuedTasks. When OnEnqueue is
2021-11-06 04:24:59 +00:00
executed, the external caller of taskScheduler calls WaitToFinish of the task to synchronously block and wait for the
2021-12-24 05:41:19 +00:00
execution of the task to be done.
2021-08-25 03:29:52 +00:00
When taskScheduler's background scheduling coroutine decides to schedule a task, it will call the taskQueue's
PopUnissuedTask to remove a task from unissuedTasks, and then call the taskQueue's AddActivateTask to put the task in
the activateTasks Map. Then perform operations on the task. In the execution process of the task, its PreExecute,
Execute, and PostExecute interfaces will be called in sequence. If an exception occurs in a certain step, the error will
be returned in advance and the subsequent steps will be skipped. Whether it is a successful execution or an error, the
Notify method will be called, this method will wake up the coroutine that is blocking and waiting for the Task. When the
task is executed, the PopActivateTask of the taskQueue is called to take the task out of activateTasks.
The following figure is a schematic diagram of taskScheduler's scheduling of DdQueue.
For the task of taskScheduler in DdQueue, it must be scheduled serially, the task that enters the queue first is
executed, and at most only one task is executing. After the task at the head of the queue is executed, other tasks in
the queue can be scheduled.
![task_scheduler_1 ](./graphs/task_scheduler_1.png )
2021-11-09 11:52:33 +00:00
The following figure is a schematic diagram of taskScheduler's scheduling of DmQueue.
2021-08-25 03:29:52 +00:00
The tasks in DmQueue can be scheduled in parallel. In a scheduling process, taskScheduler will execute several tasks
from each task concurrently.
![task_scheduler_2 ](./graphs/task_scheduler_2.png )
2021-11-09 11:52:33 +00:00
The following figure is a schematic diagram of taskScheduler's scheduling of DqQueue.
2021-08-25 03:29:52 +00:00
2021-09-13 09:12:19 +00:00
![task_scheduler_1 ](./graphs/task_scheduler_2.png )
2021-08-25 03:29:52 +00:00
The tasks in DqQueue can be scheduled in parallel. In a scheduling process, taskScheduler will execute several tasks
concurrently.
In order to facilitate the channelsTimeTicker component to obtain the synchronization point information corresponding to all
2021-11-09 11:52:33 +00:00
DmChannels, the taskScheduler needs to maintain a copy of the time statistics of the physical channels of all currently
2021-08-25 03:29:52 +00:00
unexecuted and executing tasks in the DmQueue. The member pChanSatisticsInfos is a map containing the mapping from pChan
to pChanStatInfo pointers. Among them, pChan is an alias of string, and pChanStatInfo is a custom structure, defined as
follows:
```go
type Timestamp = uint64
type pChan = string
type pChanStatInfo struct {
maxTs Timestamp
minTs Timestamp
tsSet map[Timestamp] struct{}
}
```
pChan represents the name of the physical channel corresponding to the DmChannel. The pChanStatInfo structure contains 3
members minTs, maxTs and tsSet. minTs and maxTs respectively represent the minimum and maximum timestamps of all tasks
related to the pChan in the current DmQueue. tsSet represents the set of timestamps of all tasks in DmQueue. When the
task enters the DmQueue, it will call the addPChanStats method of the queue, add the task's own timestamp to the
collection of pChanStatInfo's tsSet, and use the task's own maxTs to recalculate the pChanStatInfo's maxTs. The
calculation method is if its own maxTs is greater than pChanStatInfo MaxTs in pChanStatInfo, then update maxTs in
pChanStatInfo. Since the newly added timestamp is definitely greater than the existing timestamp, there is no need to
update minTs. When the PopActivateTask interface of the Queue is called to take the task out of the DmTaskQueue, the
popPChanStats interface of the Queue is called to delete the task timestamp from the timestamp set tsSet of
pChanStatInfo, and recalculate the minimum timestamp minTs.
DmQueue maintains the timestamp statistics of pChans and provides the method getPChanStatsInfo to the caller.
pChanStatistics is defined as follows:
```go
type pChanStatistics struct {
minTs Timestamp
maxTs Timestamp
}
func (queue *DmTaskQueue) getPChanStatsInfo() (map[pChan]*pChanStatistics, error)
```
The return value of this method includes a pChan to pChanStatistics pointer mapping. pChanStatistics includes two
members minTs and maxTs, which respectively represent the minimum and maximum timestamps of all tasks that have not been
completed on pChan. The channelsTimeTicker collects timestamp information mainly depends on this method. The awakened
coroutine reduces the timestamp result and sends the final timestamp result back to the RootCoord.
2021-12-24 08:40:19 +00:00
taskScheduler is also responsible for collecting the results of search requests. For each search request, when Proxy
2021-08-25 03:29:52 +00:00
writes the request to the DqRequestChannel, it will attach the ReqID, and the query nodes will also bring the ReqID back
when writing the search result back to the DqResultChannel. taskScheduler will start a background coroutine to consume
2021-12-28 12:28:02 +00:00
search results from DqResultChannel, and then distribute messages according to the ReqID in it. When several results of
2021-08-25 03:29:52 +00:00
the same ReqID are collected and the termination condition is reached, these results will be passed to the blocking task
2021-10-16 14:26:52 +00:00
coroutine which is waiting. The waken task will reduce the search results and then send the final search result to
2021-08-25 03:29:52 +00:00
clients.
##### 6.6.2 channelsMgr
channelsMgr is responsible for the management of DmChannels, DqRequestChannel, DqResultChannel and corresponding
MsgStream objects of each Collection. The interface is defined as follows:
```go
type channelsMgr interface {
getChannels(collectionID UniqueID) ([]pChan, error)
getVChannels(collectionID UniqueID) ([]vChan, error)
createDQLStream(collectionID UniqueID) error
getDQLStream(collectionID UniqueID) (msgstream.MsgStream, error)
removeDQLStream(collectionID UniqueID) error
removeAllDQLStream() error
createDMLMsgStream(collectionID UniqueID) error
getDMLStream(collectionID UniqueID) (msgstream.MsgStream, error)
removeDMLStream(collectionID UniqueID) error
removeAllDMLStream() error
}
```
2021-11-19 10:01:32 +00:00
- getChannels and getVChannels
2021-08-25 03:29:52 +00:00
getVChannels returns a list that represents all virtual DmChannels of collection, getChannels returns a list that
2021-12-24 05:43:09 +00:00
represents all physical DmChannels of collection. The two lists correspond one-to-one according to a position.
2021-08-25 03:29:52 +00:00
2021-11-19 10:01:32 +00:00
- createDMLStream and getDMLStream
2021-08-25 03:29:52 +00:00
2021-12-14 08:03:25 +00:00
createDMLStream creates the dml message stream of a collection;
2021-08-25 03:29:52 +00:00
2021-12-14 08:03:25 +00:00
getDMLStream returns the dml message stream of a collection;
2021-08-25 03:29:52 +00:00
2021-12-29 04:09:20 +00:00
Proxy uses these dml message streams to write dml data, such as insert requests.
2021-08-25 03:29:52 +00:00
2021-11-19 10:01:32 +00:00
- createDQLStream and getDQLStream
2021-08-25 03:29:52 +00:00
2021-12-14 08:03:25 +00:00
createDQLStream creates the dql message stream of a collection;
2021-08-25 03:29:52 +00:00
2021-12-14 08:03:25 +00:00
getDQLStream returns the dql message stream of a collection;
2021-08-25 03:29:52 +00:00
2021-12-27 09:56:19 +00:00
Proxy uses these dql message streams to send search requests.
2021-08-25 03:29:52 +00:00
The Remove related operation is to delete the corresponding message stream object, but the stream is not immediately
2021-12-27 09:58:07 +00:00
closed because maybe there are some data that needs to be written into stream currently.
2021-08-25 03:29:52 +00:00
##### 6.6.3 channelsTimeTicker
2021-12-10 02:33:08 +00:00
channelsTimeTicker is responsible for regularly collecting the synchronization timestamp information of all physical channels.
2021-08-25 03:29:52 +00:00
```go
type channelsTimeTicker interface {
start() error
close() error
addPChan(pchan pChan) error
removePChan(pchan pChan) error
getLastTick(pchan pChan) (Timestamp, error)
getMinTsStatistics() (map[pChan]Timestamp, error)
}
```
2021-11-19 10:01:32 +00:00
- addPChan and removePChan
2021-08-25 03:29:52 +00:00
addPChan adds a physical channel to channelsTimeTicker, channelsTimeTicker only sends the information of existing
pChans to Root Coordinator;
- getMinTsStatistics
getMinTsStatistics returns the timestamp statistics of physical channels, there is a background coroutine in Proxy
that call getMinTsStatistics periodically and then send the timestamp statistics to Root Coordinator;
- getLastTick
2021-12-28 12:18:01 +00:00
getLastTick returns the minimum timestamp which has already been synchronized of a physical channel;
2021-08-25 03:29:52 +00:00
channelsTimeTicker will maintain the map minTsStatistics that can be synchronized and the map currents that will be
synchronized. They are all mappings from pChan to Timestamp. The channelsTimeTicker itself has a background coroutine,
which periodically calls getPChanStatsInfo of DmQueue to obtain the minimum and maximum timestamp information pChanStats
of all unfinished tasks. channelsTimeTicker will also request a timestamp from Root Coordinator as now.
channelsTimeTicker updates minTsStatistics and currents according to the relationship between now, minTsStatistics,
currents, and pChanStats. The specific algorithm is as follows:
```go
now = get a timestamp from RootCoordinator
For each pChan in currents, do
current = currents[pChan]
if pChan not exists in pChanStats, then
minTsStatistics[pChan] = current
currents[pChan] = now
else:
minTs = pChanStats[pChan].minTs
maxTs = pChanStats[pChan].maxTs
if minTs > current, then
minTsStatistics[pChan] = min(minTs, now)
next:= min(now + sendTimeTickMsgInterval, maxTs)
currents[pChan] = next
```
##### 6.6.4 globalMetaCache
globalMetaCache is responsible for caching the meta-information of Collection and Partition. These meta-information
include CollectionID, PartitionID, CollectionSchema, etc.
The interface of Cache is defined as follows:
```go
type Cache interface {
GetCollectionID(ctx context.Context, collectionName string) (typeutil.UniqueID, error)
GetPartitionID(ctx context.Context, collectionName string, partitionName string) (typeutil.UniqueID, error)
GetPartitions(ctx context.Context, collectionName string) (map[string]typeutil.UniqueID, error)
GetCollectionSchema(ctx context.Context, collectionName string) (*schemapb.CollectionSchema, error)
RemoveCollection(ctx context.Context, collectionName string)
RemovePartition(ctx context.Context, collectionName string, partitionName string)
}
```
- getCollectionID
GetCollectionID returns the collection ID of collection;
- GetPartitions
GetPartitions returns a mapping which maps all partition names to partition IDs of collection;
- GetPartition
2021-12-27 09:48:32 +00:00
GetPartition returns the partition ID of a specific partition and collection;
2021-08-25 03:29:52 +00:00
- GetCollectionSchema
GetCollectionSchema returns the schema of collection;
- RemoveCollection
RemoveCollection removes the meta information of collection;
- RemovePartition
2021-11-06 08:21:53 +00:00
RemovePartition removes the meta information of partition;