/* * Copyright (c) 2019 TAOS Data, Inc. * * This program is free software: you can use, redistribute, and/or modify * it under the terms of the GNU Affero General Public License, version 3 * or later ("AGPL"), as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ #ifndef _TD_TQ_H_ #define _TD_TQ_H_ #include "common.h" #include "mallocator.h" #include "os.h" #include "taoserror.h" #include "taosmsg.h" #include "tlist.h" #include "tutil.h" #ifdef __cplusplus extern "C" { #endif typedef struct STqMsgHead { int32_t protoVer; int32_t msgType; int64_t cgId; int64_t clientId; } STqMsgHead; typedef struct STqOneAck { int64_t topicId; int64_t consumeOffset; } STqOneAck; typedef struct STqAcks { int32_t ackNum; // should be sorted STqOneAck acks[]; } STqAcks; typedef struct STqSetCurReq { STqMsgHead head; int64_t topicId; int64_t offset; } STqSetCurReq; typedef struct STqConsumeReq { STqMsgHead head; STqAcks acks; } STqConsumeReq; typedef struct STqMsgContent { int64_t topicId; int64_t msgLen; char msg[]; } STqMsgContent; typedef struct STqConsumeRsp { STqMsgHead head; int64_t bodySize; STqMsgContent msgs[]; } STqConsumeRsp; typedef struct STqSubscribeReq { STqMsgHead head; int32_t topicNum; int64_t topic[]; } STqSubscribeReq; typedef struct STqSubscribeRsp { STqMsgHead head; int64_t vgId; char ep[TSDB_EP_LEN]; // TSDB_EP_LEN } STqSubscribeRsp; typedef struct STqHeartbeatReq { } STqHeartbeatReq; typedef struct STqHeartbeatRsp { } STqHeartbeatRsp; typedef struct STqTopicVhandle { int64_t topicId; // executor for filter void* filterExec; // callback for mnode // trigger when vnode list associated topic change void* (*mCallback)(void*, void*); } STqTopicVhandle; #define TQ_BUFFER_SIZE 8 typedef struct STqExec { void* runtimeEnv; SSDataBlock* (*exec)(void* runtimeEnv); void* (*assign)(void* runtimeEnv, SSubmitBlk* inputData); void (*clear)(void* runtimeEnv); char* (*serialize)(struct STqExec*); struct STqExec* (*deserialize)(char*); } STqExec; typedef struct STqBufferItem { int64_t offset; // executors are identical but not concurrent // so there must be a copy in each item STqExec* executor; int32_t status; int64_t size; void* content; } STqMsgItem; typedef struct STqTopic { // char* topic; //c style, end with '\0' // int64_t cgId; // void* ahandle; int64_t nextConsumeOffset; int64_t floatingCursor; int64_t topicId; int32_t head; int32_t tail; STqMsgItem buffer[TQ_BUFFER_SIZE]; } STqTopic; typedef struct STqListHandle { STqTopic topic; struct STqListHandle* next; } STqList; typedef struct STqGroup { int64_t clientId; int64_t cgId; void* ahandle; int32_t topicNum; STqList* head; SList* topicList; // SList void* returnMsg; // SVReadMsg } STqGroup; typedef struct STqQueryMsg { STqMsgItem* item; struct STqQueryMsg* next; } STqQueryMsg; typedef struct STqLogReader { void* logHandle; int32_t (*logRead)(void* logHandle, void** data, int64_t ver); int64_t (*logGetFirstVer)(void* logHandle); int64_t (*logGetSnapshotVer)(void* logHandle); int64_t (*logGetLastVer)(void* logHandle); } STqLogReader; typedef struct STqCfg { // TODO } STqCfg; typedef struct STqMemRef { SMemAllocatorFactory* pAlloctorFactory; SMemAllocator* pAllocator; } STqMemRef; typedef struct STqSerializedHead { int16_t ver; int16_t action; int32_t checksum; int64_t ssize; char content[]; } STqSerializedHead; typedef int (*FTqSerialize)(const void* pObj, STqSerializedHead** ppHead); typedef const void* (*FTqDeserialize)(const STqSerializedHead* pHead, void** ppObj); typedef void (*FTqDelete)(void*); #define TQ_BUCKET_MASK 0xFF #define TQ_BUCKET_SIZE 256 #define TQ_PAGE_SIZE 4096 // key + offset + size #define TQ_IDX_SIZE 24 // 4096 / 24 #define TQ_MAX_IDX_ONE_PAGE 170 // 24 * 170 #define TQ_IDX_PAGE_BODY_SIZE 4080 // 4096 - 4080 #define TQ_IDX_PAGE_HEAD_SIZE 16 #define TQ_ACTION_CONST 0 #define TQ_ACTION_INUSE 1 #define TQ_ACTION_INUSE_CONT 2 #define TQ_ACTION_INTXN 3 #define TQ_SVER 0 // TODO: inplace mode is not implemented #define TQ_UPDATE_INPLACE 0 #define TQ_UPDATE_APPEND 1 #define TQ_DUP_INTXN_REWRITE 0 #define TQ_DUP_INTXN_REJECT 2 static inline bool tqUpdateAppend(int32_t tqConfigFlag) { return tqConfigFlag & TQ_UPDATE_APPEND; } static inline bool tqDupIntxnReject(int32_t tqConfigFlag) { return tqConfigFlag & TQ_DUP_INTXN_REJECT; } static const int8_t TQ_CONST_DELETE = TQ_ACTION_CONST; #define TQ_DELETE_TOKEN (void*)&TQ_CONST_DELETE typedef struct STqMetaHandle { int64_t key; int64_t offset; int64_t serializedSize; void* valueInUse; void* valueInTxn; } STqMetaHandle; typedef struct STqMetaList { STqMetaHandle handle; struct STqMetaList* next; // struct STqMetaList* inTxnPrev; // struct STqMetaList* inTxnNext; struct STqMetaList* unpersistPrev; struct STqMetaList* unpersistNext; } STqMetaList; typedef struct STqMetaStore { STqMetaList* bucket[TQ_BUCKET_SIZE]; // a table head STqMetaList* unpersistHead; // TODO:temporaral use, to be replaced by unified tfile int fileFd; // TODO:temporaral use, to be replaced by unified tfile int idxFd; char* dirPath; int32_t tqConfigFlag; FTqSerialize pSerializer; FTqDeserialize pDeserializer; FTqDelete pDeleter; } STqMetaStore; typedef struct STQ { // the collection of groups // the handle of meta kvstore char* path; STqCfg* tqConfig; STqLogReader* tqLogReader; STqMemRef tqMemRef; STqMetaStore* tqMeta; } STQ; // open in each vnode STQ* tqOpen(const char* path, STqCfg* tqConfig, STqLogReader* tqLogReader, SMemAllocatorFactory* allocFac); void tqClose(STQ*); // void* will be replace by a msg type int tqPushMsg(STQ*, void* msg, int64_t version); int tqCommit(STQ*); int tqConsume(STQ*, STqConsumeReq*); int tqSetCursor(STQ*, STqSetCurReq* pMsg); int tqBufferSetOffset(STqTopic*, int64_t offset); STqTopic* tqFindTopic(STqGroup*, int64_t topicId); STqGroup* tqGetGroup(STQ*, int64_t clientId); STqGroup* tqOpenGroup(STQ*, int64_t topicId, int64_t cgId, int64_t cId); int tqCloseGroup(STQ*, int64_t topicId, int64_t cgId, int64_t cId); int tqRegisterContext(STqGroup*, void* ahandle); int tqSendLaunchQuery(STqMsgItem*, int64_t offset); int tqSerializeGroup(const STqGroup*, STqSerializedHead**); const void* tqDeserializeGroup(const STqSerializedHead*, STqGroup**); static int tqQueryExecuting(int32_t status) { return status; } #ifdef __cplusplus } #endif #endif /*_TD_TQ_H_*/