• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

taosdata / TDengine / #5058

17 May 2026 01:15AM UTC coverage: 73.387% (-0.02%) from 73.406%
#5058

push

travis-ci

web-flow
feat (TDgpt): Dynamic Model Synchronization Enhancements (#35344)

* refactor: do some internal refactor.

* fix: fix multiprocess sync issue.

* feat: add dynamic anomaly detection and forecasting services

* fix: log error message for undeploying model in exception handling

* Potential fix for pull request finding

Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>

* Potential fix for pull request finding

Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>

* Potential fix for pull request finding

Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>

* Potential fix for pull request finding

Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>

* fix: handle undeploy when model exists only on disk

Agent-Logs-Url: https://github.com/taosdata/TDengine/sessions/286aafa0-c3ce-4c27-b803-2707571e9dc1

Co-authored-by: hjxilinx <8252296+hjxilinx@users.noreply.github.com>

* fix: guard dynamic registry concurrent access

Agent-Logs-Url: https://github.com/taosdata/TDengine/sessions/5e4db858-6458-40f4-ac28-d1b1b7f97c18

Co-authored-by: hjxilinx <8252296+hjxilinx@users.noreply.github.com>

* fix: tighten service list locking scope

Agent-Logs-Url: https://github.com/taosdata/TDengine/sessions/5e4db858-6458-40f4-ac28-d1b1b7f97c18

Co-authored-by: hjxilinx <8252296+hjxilinx@users.noreply.github.com>

* fix: restore prophet support and update tests per review feedback

Agent-Logs-Url: https://github.com/taosdata/TDengine/sessions/92298ae1-7da6-4d07-b20e-101c7cd0b26b

Co-authored-by: hjxilinx <8252296+hjxilinx@users.noreply.github.com>

* fix: improve test name and move copy inside lock scope

Agent-Logs-Url: https://github.com/taosdata/TDengine/sessions/92298ae1-7da6-4d07-b20e-101c7cd0b26b

Co-authored-by: hjxilinx <8252296+hjxilinx@users.noreply.github.com>

* Potential fix for pull request finding

Co-au... (continued)

281656 of 383795 relevant lines covered (73.39%)

135114337.11 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

76.65
/source/libs/executor/src/projectoperator.c
1
/*
2
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
3
 *
4
 * This program is free software: you can use, redistribute, and/or modify
5
 * it under the terms of the GNU Affero General Public License, version 3
6
 * or later ("AGPL"), as published by the Free Software Foundation.
7
 *
8
 * This program is distributed in the hope that it will be useful, but WITHOUT
9
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10
 * FITNESS FOR A PARTICULAR PURPOSE.
11
 *
12
 * You should have received a copy of the GNU Affero General Public License
13
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
14
 */
15

16
#include "executorInt.h"
17
#include "filter.h"
18
#include "functionMgt.h"
19
#include "operator.h"
20
#include "querytask.h"
21
#include "taoserror.h"
22
#include "tdatablock.h"
23

24
typedef struct SProjectOperatorInfo {
25
  SOptrBasicInfo binfo;
26
  SAggSupporter  aggSup;
27
  SArray*        pPseudoColInfo;
28
  SLimitInfo     limitInfo;
29
  bool           mergeDataBlocks;
30
  SSDataBlock*   pFinalRes;
31
  bool           inputIgnoreGroup;
32
  bool           outputIgnoreGroup;
33
} SProjectOperatorInfo;
34

35
typedef struct SIndefOperatorInfo {
36
  SOptrBasicInfo binfo;
37
  SAggSupporter  aggSup;
38
  SArray*        pPseudoColInfo;
39
  SExprSupp      scalarSup;
40
  uint64_t       groupId;
41
  SSDataBlock*   pNextGroupRes;
42
} SIndefOperatorInfo;
43

44
static int32_t      doGenerateSourceData(SOperatorInfo* pOperator);
45
static int32_t      doProjectOperation(SOperatorInfo* pOperator, SSDataBlock** pResBlock);
46
static int32_t      doApplyIndefinitFunction(SOperatorInfo* pOperator, SSDataBlock** pResBlock);
47
int32_t projectApplyOperator(SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock, int32_t outputSlotId, int32_t* numOfRows, bool createNewColModel, const void* pExtraParams);
48

49
static bool hasLagLeadFunc(const SExprSupp* pSup) {
9,606,892✔
50
  if (pSup == NULL || pSup->pCtx == NULL) {
9,606,892✔
51
    return false;
×
52
  }
53

54
  for (int32_t i = 0; i < pSup->numOfExprs; ++i) {
31,277,163✔
55
    EFunctionType type = fmGetFuncTypeFromId(pSup->pCtx[i].functionId);
21,703,359✔
56
    if (type == FUNCTION_TYPE_LAG || type == FUNCTION_TYPE_LEAD) {
21,703,359✔
57
      return true;
33,505✔
58
    }
59
  }
60

61
  return false;
9,573,804✔
62
}
63

64
static void destroyProjectOperatorInfo(void* param) {
160,106,472✔
65
  if (NULL == param) {
160,106,472✔
66
    return;
×
67
  }
68

69
  SProjectOperatorInfo* pInfo = (SProjectOperatorInfo*)param;
160,106,472✔
70
  cleanupBasicInfo(&pInfo->binfo);
160,106,472✔
71
  cleanupAggSup(&pInfo->aggSup);
160,110,940✔
72
  taosArrayDestroy(pInfo->pPseudoColInfo);
160,083,164✔
73

74
  blockDataDestroy(pInfo->pFinalRes);
160,094,825✔
75
  taosMemoryFreeClear(param);
160,091,945✔
76
}
77

78
static void destroyIndefinitOperatorInfo(void* param) {
3,480,244✔
79
  SIndefOperatorInfo* pInfo = (SIndefOperatorInfo*)param;
3,480,244✔
80
  if (pInfo == NULL) {
3,480,244✔
81
    return;
×
82
  }
83

84
  cleanupBasicInfo(&pInfo->binfo);
3,480,244✔
85
  taosArrayDestroy(pInfo->pPseudoColInfo);
3,481,078✔
86
  cleanupAggSup(&pInfo->aggSup);
3,480,079✔
87
  cleanupExprSupp(&pInfo->scalarSup);
3,480,079✔
88

89
  taosMemoryFreeClear(param);
3,480,244✔
90
}
91

92
static void cleanupProcessByRowIter(SqlFunctionCtx* pCtx) {
×
93
  SFuncInputRowIter* pIter = &pCtx->rowIter;
×
94

95
  if (pIter->pPrevRowBlock != NULL) {
×
96
    blockDataDestroy(pIter->pPrevRowBlock);
×
97
  }
98
  taosMemoryFreeClear(pIter->pPrevData);
×
99
  taosMemoryFreeClear(pIter->pPrevPk);
×
100
  memset(pIter, 0, sizeof(*pIter));
×
101
}
×
102

103
static int32_t resetProcessByRowCtx(SqlFunctionCtx* pCtx) {
×
104
  SResultRowEntryInfo* pResInfo = GET_RES_INFO(pCtx);
×
105
  char*                pOutput = pCtx->pOutput;
×
106

107
  if (pResInfo->initialized && pCtx->fpSet.cleanup != NULL) {
×
108
    pCtx->fpSet.cleanup(pCtx);
×
109
  }
110

111
  cleanupProcessByRowIter(pCtx);
×
112
  pResInfo->initialized = false;
×
113
  pResInfo->numOfRes = 0;
×
114
  pCtx->bInputFinished = false;
×
115

116
  pCtx->pOutput = NULL;
×
117
  int32_t code = pCtx->fpSet.init(pCtx, pResInfo);
×
118
  pCtx->pOutput = pOutput;
×
119
  return code;
×
120
}
121

122
static bool allProcessByRowCtxSameFuncId(SArray* pProcessByRowFunctionCtx) {
×
123
  if (pProcessByRowFunctionCtx == NULL || taosArrayGetSize(pProcessByRowFunctionCtx) <= 1) {
×
124
    return true;
×
125
  }
126

127
  SqlFunctionCtx** ppFirstCtx = taosArrayGet(pProcessByRowFunctionCtx, 0);
×
128
  if (ppFirstCtx == NULL || *ppFirstCtx == NULL) {
×
129
    return false;
×
130
  }
131

132
  int32_t funcId = (*ppFirstCtx)->functionId;
×
133
  for (int32_t i = 1; i < taosArrayGetSize(pProcessByRowFunctionCtx); ++i) {
×
134
    SqlFunctionCtx** ppCtx = taosArrayGet(pProcessByRowFunctionCtx, i);
×
135
    if (ppCtx == NULL || *ppCtx == NULL || (*ppCtx)->functionId != funcId) {
×
136
      return false;
×
137
    }
138
  }
139

140
  return true;
×
141
}
142

143
static int32_t processByRowInExternalWindows(SArray* pGroupedCtxArray, SSDataBlock* pSrcBlock,
×
144
                                             SStreamRuntimeFuncInfo* pStreamInfo) {
145
  int32_t code = TSDB_CODE_SUCCESS;
×
146
  int32_t lino = 0;
×
147

148
  int32_t  ctxNum = taosArrayGetSize(pGroupedCtxArray);
×
149
  int32_t  idxNum = taosArrayGetSize(pStreamInfo->pStreamBlkWinIdx);
×
150
  int32_t  totalRows = 0;
×
151
  SArray*  pInputWinIdx = NULL;
×
152
  int32_t* pStartRows = NULL;
×
153
  int64_t* pNumRows = NULL;
×
154
  int32_t* pOffsets = NULL;
×
155

156
  pInputWinIdx = taosArrayInit(idxNum, sizeof(int64_t));
×
157
  TSDB_CHECK_NULL(pInputWinIdx, code, lino, _exit, terrno);
×
158
  TSDB_CHECK_NULL(taosArrayAddBatch(pInputWinIdx, TARRAY_DATA(pStreamInfo->pStreamBlkWinIdx), idxNum), code, lino,
×
159
                  _exit, terrno);
160

161
  pStartRows = taosMemoryCalloc(ctxNum, sizeof(int32_t));
×
162
  TSDB_CHECK_NULL(pStartRows, code, lino, _exit, terrno);
×
163
  pNumRows = taosMemoryCalloc(ctxNum, sizeof(int64_t));
×
164
  TSDB_CHECK_NULL(pNumRows, code, lino, _exit, terrno);
×
165
  pOffsets = taosMemoryCalloc(ctxNum, sizeof(int32_t));
×
166
  TSDB_CHECK_NULL(pOffsets, code, lino, _exit, terrno);
×
167

168
  for (int32_t i = 0; i < ctxNum; ++i) {
×
169
    SqlFunctionCtx** ppCtx = taosArrayGet(pGroupedCtxArray, i);
×
170
    TSDB_CHECK_NULL(ppCtx, code, lino, _exit, terrno);
×
171
    TSDB_CHECK_NULL(*ppCtx, code, lino, _exit, terrno);
×
172
    pStartRows[i] = (*ppCtx)->input.startRowIndex;
×
173
    pNumRows[i] = (*ppCtx)->input.numOfRows;
×
174
    pOffsets[i] = (*ppCtx)->offset;
×
175
  }
176

177
  taosArrayClear(pStreamInfo->pStreamBlkWinIdx);
×
178

179
  for (int32_t i = 0; i < idxNum; ++i) {
×
180
    int64_t* pCurr = taosArrayGet(pInputWinIdx, i);
×
181
    int32_t* pCurrPair = (int32_t*)pCurr;
×
182
    int32_t  winIdx = pCurrPair[0];
×
183
    int32_t  rowStart = pCurrPair[1];
×
184
    int32_t  rowEnd = pSrcBlock->info.rows;
×
185

186
    if (i + 1 < idxNum) {
×
187
      int64_t* pNext = taosArrayGet(pInputWinIdx, i + 1);
×
188
      rowEnd = ((int32_t*)pNext)[1];
×
189
    }
190

191
    if (rowEnd <= rowStart) {
×
192
      continue;
×
193
    }
194

195
    for (int32_t j = 0; j < ctxNum; ++j) {
×
196
      SqlFunctionCtx** ppCtx = taosArrayGet(pGroupedCtxArray, j);
×
197
      SqlFunctionCtx*  pCtx = *ppCtx;
×
198

199
      pCtx->input.startRowIndex = rowStart;
×
200
      pCtx->input.numOfRows = rowEnd - rowStart;
×
201
      pCtx->offset = pOffsets[j] + totalRows;
×
202
      TAOS_CHECK_EXIT(resetProcessByRowCtx(pCtx));
×
203
    }
204

205
    SqlFunctionCtx** ppFirstCtx = taosArrayGet(pGroupedCtxArray, 0);
×
206
    TAOS_CHECK_EXIT((*ppFirstCtx)->fpSet.processFuncByRow(pGroupedCtxArray));
×
207

208
    int32_t winRows = (*ppFirstCtx)->resultInfo->numOfRes;
×
209
    if (winRows > 0) {
×
210
      int64_t  val = 0;
×
211
      int32_t* pOutPair = (int32_t*)&val;
×
212
      pOutPair[0] = winIdx;
×
213
      pOutPair[1] = totalRows;
×
214
      TSDB_CHECK_NULL(taosArrayPush(pStreamInfo->pStreamBlkWinIdx, &val), code, lino, _exit, terrno);
×
215
      totalRows += winRows;
×
216
    }
217
  }
218

219
  for (int32_t i = 0; i < ctxNum; ++i) {
×
220
    SqlFunctionCtx** ppCtx = taosArrayGet(pGroupedCtxArray, i);
×
221
    SqlFunctionCtx*  pCtx = *ppCtx;
×
222
    pCtx->input.startRowIndex = pStartRows[i];
×
223
    pCtx->input.numOfRows = pNumRows[i];
×
224
    pCtx->offset = pOffsets[i];
×
225
    pCtx->resultInfo->numOfRes = totalRows;
×
226
  }
227

228
_exit:
×
229
  if (pInputWinIdx != NULL) {
×
230
    taosArrayDestroy(pInputWinIdx);
×
231
  }
232
  taosMemoryFreeClear(pStartRows);
×
233
  taosMemoryFreeClear(pNumRows);
×
234
  taosMemoryFreeClear(pOffsets);
×
235
  return code;
×
236
}
237

238
static int32_t assignPlaceHolderInExternalWindows(SColumnInfoData* pResColData, int64_t offset, int64_t rows,
×
239
                                                  int16_t funcId, SStreamRuntimeFuncInfo* pInfo, SNode* pParamNode) {
240
  int32_t code = TSDB_CODE_SUCCESS;
×
241
  int32_t lino = 0;
×
242

243
  int32_t originIdx = pInfo->curIdx;
×
244
  int32_t idxNum = taosArrayGetSize(pInfo->pStreamBlkWinIdx);
×
245
  SArray* pInputWinIdx = taosArrayInit(idxNum, sizeof(int64_t));
×
246
  TSDB_CHECK_NULL(pInputWinIdx, code, lino, _exit, terrno);
×
247
  TSDB_CHECK_NULL(taosArrayAddBatch(pInputWinIdx, TARRAY_DATA(pInfo->pStreamBlkWinIdx), idxNum), code, lino, _exit,
×
248
                  terrno);
249

250
  for (int32_t i = 0; i < idxNum; ++i) {
×
251
    int64_t* pCurr = taosArrayGet(pInputWinIdx, i);
×
252
    int32_t* pCurrPair = (int32_t*)pCurr;
×
253
    int32_t  winIdx = pCurrPair[0];
×
254
    int32_t  rowStart = pCurrPair[1];
×
255
    int32_t  rowEnd = rows;
×
256

257
    if (i + 1 < idxNum) {
×
258
      int64_t* pNext = taosArrayGet(pInputWinIdx, i + 1);
×
259
      rowEnd = ((int32_t*)pNext)[1];
×
260
    }
261

262
    if (rowEnd <= rowStart) {
×
263
      continue;
×
264
    }
265

266
    pInfo->curIdx = winIdx;
×
267
    TAOS_CHECK_EXIT(scalarAssignPlaceHolderRes(pResColData, offset + rowStart, rowEnd - rowStart, funcId, pInfo,
×
268
                                               pParamNode));
269
  }
270

271
_exit:
×
272
  pInfo->curIdx = originIdx;
×
273
  taosArrayDestroy(pInputWinIdx);
×
274
  return code;
×
275
}
276

277
void streamOperatorReleaseState(SOperatorInfo* pOperator) {
×
278
  SOperatorInfo* downstream = pOperator->pDownstream[0];
×
279
  if (downstream->fpSet.releaseStreamStateFn) {
×
280
    downstream->fpSet.releaseStreamStateFn(downstream);
×
281
  }
282
}
×
283

284
void streamOperatorReloadState(SOperatorInfo* pOperator) {
×
285
  SOperatorInfo* downstream = pOperator->pDownstream[0];
×
286
  if (downstream->fpSet.reloadStreamStateFn) {
×
287
    downstream->fpSet.reloadStreamStateFn(downstream);
×
288
  }
289
}
×
290

291
static int32_t resetProjectOperState(SOperatorInfo* pOper) {
10,054,965✔
292
  SProjectOperatorInfo* pProject = pOper->info;
10,054,965✔
293
  SExecTaskInfo*           pTaskInfo = pOper->pTaskInfo;
10,054,965✔
294
  pOper->status = OP_NOT_OPENED;
10,054,965✔
295

296
  resetBasicOperatorState(&pProject->binfo);
10,054,965✔
297
  SProjectPhysiNode* pPhynode = (SProjectPhysiNode*)pOper->pPhyNode;
10,054,965✔
298

299
  pProject->limitInfo = (SLimitInfo){0};
10,054,965✔
300
  initLimitInfo(pPhynode->node.pLimit, pPhynode->node.pSlimit, &pProject->limitInfo);
10,054,965✔
301

302
  blockDataCleanup(pProject->pFinalRes);
10,054,965✔
303

304
  int32_t code = resetAggSup(&pOper->exprSupp, &pProject->aggSup, pTaskInfo, pPhynode->pProjections, NULL,
20,106,154✔
305
    sizeof(int64_t) * 2 + POINTER_BYTES, pTaskInfo->id.str, NULL,
10,054,703✔
306
    &pTaskInfo->storageAPI.functionStore);
307
  if (code == 0){
10,052,745✔
308
    code = setFunctionResultOutput(pOper, &pProject->binfo, &pProject->aggSup, MAIN_SCAN, pOper->exprSupp.numOfExprs);
10,053,007✔
309
  }
310
  return 0;
10,052,774✔
311
}
312

313
int32_t createProjectOperatorInfo(SOperatorInfo* downstream, SProjectPhysiNode* pProjPhyNode, SExecTaskInfo* pTaskInfo,
160,077,978✔
314
                                  SOperatorInfo** pOptrInfo) {
315
  QRY_PARAM_CHECK(pOptrInfo);
160,077,978✔
316

317
  int32_t code = TSDB_CODE_SUCCESS;
160,067,949✔
318
  SProjectOperatorInfo* pInfo = taosMemoryCalloc(1, sizeof(SProjectOperatorInfo));
160,067,949✔
319
  SOperatorInfo*        pOperator = taosMemoryCalloc(1, sizeof(SOperatorInfo));
159,931,199✔
320
  if (pInfo == NULL || pOperator == NULL) {
160,012,844✔
321
    code = terrno;
160✔
322
    goto _error;
×
323
  }
324

325
  pOperator->pPhyNode = pProjPhyNode;
160,013,607✔
326
  pOperator->exprSupp.hasWindowOrGroup = false;
160,035,699✔
327
  pOperator->pTaskInfo = pTaskInfo;
160,042,230✔
328
  initOperatorCostInfo(pOperator);
160,061,125✔
329

330
  int32_t    lino = 0;
160,087,265✔
331

332
  SSDataBlock* pResBlock = createDataBlockFromDescNode(pProjPhyNode->node.pOutputDataBlockDesc);
160,087,265✔
333
  TSDB_CHECK_NULL(pResBlock, code, lino, _error, terrno);
160,123,394✔
334

335
  initLimitInfo(pProjPhyNode->node.pLimit, pProjPhyNode->node.pSlimit, &pInfo->limitInfo);
160,123,394✔
336

337
  pInfo->binfo.pRes = pResBlock;
160,085,508✔
338
  pInfo->pFinalRes = NULL;
160,100,789✔
339

340
  code = createOneDataBlock(pResBlock, false, &pInfo->pFinalRes);
160,094,441✔
341
  TSDB_CHECK_CODE(code, lino, _error);
160,070,243✔
342

343
  pInfo->binfo.inputTsOrder = pProjPhyNode->node.inputTsOrder;
160,070,243✔
344
  pInfo->binfo.outputTsOrder = pProjPhyNode->node.outputTsOrder;
160,058,070✔
345
  pInfo->inputIgnoreGroup = pProjPhyNode->inputIgnoreGroup;
160,036,008✔
346
  pInfo->outputIgnoreGroup = pProjPhyNode->ignoreGroupId;
160,037,965✔
347

348
  if (pTaskInfo->execModel == OPTR_EXEC_MODEL_QUEUE) {
160,040,470✔
349
    pInfo->mergeDataBlocks = false;
381,948✔
350
  } else {
351
    if (!pProjPhyNode->ignoreGroupId) {
159,605,519✔
352
      pInfo->mergeDataBlocks = false;
2,501,011✔
353
    } else {
354
      pInfo->mergeDataBlocks = pProjPhyNode->mergeDataBlock;
157,153,636✔
355
    }
356
  }
357

358
  int32_t numOfRows = 4096;
160,041,797✔
359
  size_t  keyBufSize = sizeof(int64_t) + sizeof(int64_t) + POINTER_BYTES;
160,041,797✔
360

361
  // Make sure the size of SSDataBlock will never exceed the size of 2MB.
362
  int32_t TWOMB = 2 * 1024 * 1024;
160,041,797✔
363
  if (numOfRows * pResBlock->info.rowSize > TWOMB) {
160,041,797✔
364
    numOfRows = TWOMB / pResBlock->info.rowSize;
5,201,222✔
365
  }
366

367
  initResultSizeInfo(&pOperator->resultInfo, numOfRows);
160,026,915✔
368
  
369
  int32_t    numOfCols = 0;
160,071,136✔
370
  SExprInfo* pExprInfo = NULL;
160,045,336✔
371
  code = createExprInfo(pProjPhyNode->pProjections, NULL, &pExprInfo, &numOfCols);
160,070,199✔
372
  TSDB_CHECK_CODE(code, lino, _error);
160,043,591✔
373
  code = initAggSup(&pOperator->exprSupp, &pInfo->aggSup, pExprInfo, numOfCols, keyBufSize, pTaskInfo->id.str,
159,736,155✔
374
                    NULL, &pTaskInfo->storageAPI.functionStore);
375
  TSDB_CHECK_CODE(code, lino, _error);
159,732,294✔
376

377
  initBasicInfo(&pInfo->binfo, pResBlock);
159,732,294✔
378
  code = setFunctionResultOutput(pOperator, &pInfo->binfo, &pInfo->aggSup, MAIN_SCAN, numOfCols);
159,723,702✔
379
  TSDB_CHECK_CODE(code, lino, _error);
159,703,532✔
380

381
  code = filterInitFromNode((SNode*)pProjPhyNode->node.pConditions, &pOperator->exprSupp.pFilterInfo, 0,
159,751,414✔
382
                            pTaskInfo->pStreamRuntimeInfo);
159,703,532✔
383
  TSDB_CHECK_CODE(code, lino, _error);
159,721,146✔
384

385
  code = setRowTsColumnOutputInfo(pOperator->exprSupp.pCtx, numOfCols, &pInfo->pPseudoColInfo);
159,721,146✔
386
  TSDB_CHECK_CODE(code, lino, _error);
159,700,104✔
387

388
  setOperatorInfo(pOperator, "ProjectOperator", QUERY_NODE_PHYSICAL_PLAN_PROJECT, false, OP_NOT_OPENED, pInfo,
159,700,104✔
389
                  pTaskInfo);
390
  pOperator->fpSet = createOperatorFpSet(optrDummyOpenFn, doProjectOperation, NULL, destroyProjectOperatorInfo,
159,735,961✔
391
                                         optrDefaultBufFn, NULL, optrDefaultGetNextExtFn, NULL);
392
  setOperatorStreamStateFn(pOperator, streamOperatorReleaseState, streamOperatorReloadState);
159,700,885✔
393
  setOperatorResetStateFn(pOperator, resetProjectOperState);
159,687,711✔
394

395
  if (NULL != downstream) {
159,714,917✔
396
    code = appendDownstream(pOperator, &downstream, 1);
157,703,690✔
397
    if (code != TSDB_CODE_SUCCESS) {
157,648,173✔
398
      goto _error;
×
399
    }
400
  }
401

402
  *pOptrInfo = pOperator;
159,659,400✔
403
  return TSDB_CODE_SUCCESS;
159,676,446✔
404

405
_error:
307,436✔
406
  if (pInfo != NULL) destroyProjectOperatorInfo(pInfo);
307,436✔
407
  destroyOperatorAndDownstreams(pOperator, &downstream, 1);
307,436✔
408
  pTaskInfo->code = code;
307,436✔
409
  return code;
307,436✔
410
}
411

412
static int32_t discardGroupDataBlock(SSDataBlock* pBlock, SLimitInfo* pLimitInfo) {
327,777,970✔
413
  if (pLimitInfo->remainGroupOffset > 0) {
327,777,970✔
414
    // it is the first group
415
    if (pLimitInfo->currentGroupId == 0 || pLimitInfo->currentGroupId == pBlock->info.id.groupId) {
1,098,241✔
416
      pLimitInfo->currentGroupId = pBlock->info.id.groupId;
286,083✔
417
      return PROJECT_RETRIEVE_CONTINUE;
286,083✔
418
    } else if (pLimitInfo->currentGroupId != pBlock->info.id.groupId) {
812,158✔
419
      // now it is the data from a new group
420
      pLimitInfo->remainGroupOffset -= 1;
812,158✔
421
      pLimitInfo->currentGroupId = pBlock->info.id.groupId;
812,158✔
422

423
      // ignore data block in current group
424
      if (pLimitInfo->remainGroupOffset > 0) {
812,158✔
425
        return PROJECT_RETRIEVE_CONTINUE;
762,630✔
426
      }
427

428
      pLimitInfo->currentGroupId = 0;
49,528✔
429
    }
430
  }
431

432
  return PROJECT_RETRIEVE_DONE;
326,721,869✔
433
}
434

435
static int32_t setInfoForNewGroup(SSDataBlock* pBlock, SLimitInfo* pLimitInfo, SOperatorInfo* pOperator) {
326,720,504✔
436
  // remainGroupOffset == 0
437
  // here check for a new group data, we need to handle the data of the previous group.
438
  if (!(pLimitInfo->remainGroupOffset == 0 || pLimitInfo->remainGroupOffset == -1)) {
326,720,504✔
439
    qError("project failed at: %s:%d", __func__, __LINE__);
×
440
    return TSDB_CODE_INVALID_PARA;
×
441
  }
442

443
  bool newGroup = false;
326,758,833✔
444
  if (0 == pBlock->info.id.groupId) {
326,758,833✔
445
    pLimitInfo->numOfOutputGroups = 1;
298,409,380✔
446
  } else if (pLimitInfo->currentGroupId != pBlock->info.id.groupId) {
28,308,684✔
447
    pLimitInfo->currentGroupId = pBlock->info.id.groupId;
23,912,547✔
448
    pLimitInfo->numOfOutputGroups += 1;
23,912,547✔
449
    newGroup = true;
23,912,547✔
450
  } else {
451
    return PROJECT_RETRIEVE_CONTINUE;
4,396,137✔
452
  }
453

454
  if ((pLimitInfo->slimit.limit >= 0) && (pLimitInfo->slimit.limit < pLimitInfo->numOfOutputGroups)) {
322,313,896✔
455
    setOperatorCompleted(pOperator);
155,233✔
456
    return PROJECT_RETRIEVE_DONE;
155,233✔
457
  }
458

459
  // reset the value for a new group data
460
  // existing rows that belongs to previous group.
461
  if (newGroup) {
322,145,442✔
462
    resetLimitInfoForNextGroup(pLimitInfo);
23,756,710✔
463
  }
464

465
  return PROJECT_RETRIEVE_CONTINUE;
322,094,800✔
466
}
467

468
// todo refactor
469
static int32_t doIngroupLimitOffset(SLimitInfo* pLimitInfo, uint64_t groupId, SSDataBlock* pBlock,
327,373,163✔
470
                                    SOperatorInfo* pOperator) {
471
  // set current group id
472
  pLimitInfo->currentGroupId = groupId;
327,373,163✔
473
  bool limitReached = applyLimitOffset(pLimitInfo, pBlock, pOperator->pTaskInfo);
327,427,287✔
474
  if (pBlock->info.rows == 0 && 0 != pLimitInfo->limit.limit) {
327,438,906✔
475
    return PROJECT_RETRIEVE_CONTINUE;
5,397,672✔
476
  } else {
477
    if (limitReached && (pLimitInfo->slimit.limit >= 0 && pLimitInfo->slimit.limit <= pLimitInfo->numOfOutputGroups)) {
322,041,090✔
478
      setOperatorCompleted(pOperator);
89,438✔
479
    } else if (limitReached && groupId == 0) {
321,953,323✔
480
      setOperatorCompleted(pOperator);
8,429,123✔
481
    }
482
  }
483

484
  return PROJECT_RETRIEVE_DONE;
321,952,898✔
485
}
486

487
int32_t doProjectOperation(SOperatorInfo* pOperator, SSDataBlock** pResBlock) {
584,313,219✔
488
  QRY_PARAM_CHECK(pResBlock);
584,313,219✔
489

490
  SProjectOperatorInfo* pProjectInfo = pOperator->info;
584,377,064✔
491
  SOptrBasicInfo*       pInfo = &pProjectInfo->binfo;
584,347,865✔
492
  SExprSupp*            pSup = &pOperator->exprSupp;
584,370,949✔
493
  SSDataBlock*          pRes = pInfo->pRes;
584,320,280✔
494
  SSDataBlock*          pFinalRes = pProjectInfo->pFinalRes;
584,371,135✔
495
  int32_t               code = 0;
584,344,890✔
496
  int32_t               lino = 0;
584,344,890✔
497
  int32_t               order = pInfo->inputTsOrder;
584,344,890✔
498
  int32_t               scanFlag = 0;
584,339,616✔
499

500
  blockDataCleanup(pFinalRes);
584,339,616✔
501
  SExecTaskInfo* pTaskInfo = pOperator->pTaskInfo;
584,300,251✔
502
  bool           withExternalWindow = pTaskInfo->pStreamRuntimeInfo != NULL &&
598,486,501✔
503
                                      pTaskInfo->pStreamRuntimeInfo->funcInfo.withExternalWindow;
14,283,175✔
504

505
  if (pOperator->status == OP_EXEC_DONE) {
584,214,241✔
506
    return code;
90,363,798✔
507
  }
508

509
  SOperatorInfo* downstream = pOperator->numOfDownstream > 0 ? pOperator->pDownstream[0] : NULL;
493,897,391✔
510
  SLimitInfo*    pLimitInfo = &pProjectInfo->limitInfo;
493,970,221✔
511

512
  if (downstream == NULL) {
493,989,733✔
513
    code = doGenerateSourceData(pOperator);
2,028,006✔
514
    QUERY_CHECK_CODE(code, lino, _end);
2,028,006✔
515

516
    if (pProjectInfo->outputIgnoreGroup) {
2,028,006✔
517
      pRes->info.id.groupId = 0;
2,027,571✔
518
    }
519

520
    *pResBlock = (pRes->info.rows > 0)? pRes:NULL;
2,028,006✔
521
    return code;
2,027,571✔
522
  }
523

524
  while (1) {
129,171,568✔
525
    while (1) {
6,446,385✔
526
      blockDataCleanup(pRes);
627,579,680✔
527

528
      // The downstream exec may change the value of the newgroup, so use a local variable instead.
529
      SSDataBlock* pBlock = getNextBlockFromDownstream(pOperator, 0);
627,531,511✔
530
      if (pBlock == NULL) {
624,799,213✔
531
        qDebug("set op close, exec %d, status %d rows %" PRId64 , pTaskInfo->execModel, pOperator->status, pFinalRes->info.rows);
297,097,986✔
532
        setOperatorCompleted(pOperator);
297,103,703✔
533
        break;
297,091,451✔
534
      }
535
//      if (pTaskInfo->execModel == OPTR_EXEC_MODEL_QUEUE) {
536
//        qDebug("set status recv");
537
//        pOperator->status = OP_EXEC_RECV;
538
//      }
539

540
      if (pProjectInfo->inputIgnoreGroup) {
327,701,227✔
541
        pBlock->info.id.groupId = 0;
5,651,720✔
542
      }
543

544
      int32_t status = discardGroupDataBlock(pBlock, pLimitInfo);
327,716,998✔
545
      if (status == PROJECT_RETRIEVE_CONTINUE) {
327,782,289✔
546
        continue;
1,048,713✔
547
      }
548

549
      (void) setInfoForNewGroup(pBlock, pLimitInfo, pOperator);
326,733,576✔
550
      if (pOperator->status == OP_EXEC_DONE) {
326,642,910✔
551
        break;
155,233✔
552
      }
553

554
      if (pProjectInfo->mergeDataBlocks) {
326,572,843✔
555
        pFinalRes->info.scanFlag = scanFlag = pBlock->info.scanFlag;
141,796,857✔
556
      } else {
557
        pRes->info.scanFlag = scanFlag = pBlock->info.scanFlag;
184,797,143✔
558
      }
559

560
      code = setInputDataBlock(pSup, pBlock, order, scanFlag, false);
326,541,371✔
561
      QUERY_CHECK_CODE(code, lino, _end);
326,585,999✔
562

563
      code = blockDataEnsureCapacity(pInfo->pRes, pInfo->pRes->info.rows + pBlock->info.rows);
326,585,999✔
564
      QUERY_CHECK_CODE(code, lino, _end);
326,614,255✔
565

566
      code = projectApplyFunctions(pSup->pExprInfo, pInfo->pRes, pBlock, pSup->pCtx, pSup->numOfExprs,
653,124,002✔
567
                                   pProjectInfo->pPseudoColInfo, GET_STM_RTINFO(pOperator->pTaskInfo),
326,625,788✔
568
                                   pOperator->pTaskInfo);
569
      QUERY_CHECK_CODE(code, lino, _end);
326,536,581✔
570

571
      status = doIngroupLimitOffset(pLimitInfo, pBlock->info.id.groupId, pInfo->pRes, pOperator);
325,328,138✔
572
      if (status == PROJECT_RETRIEVE_CONTINUE) {
325,330,864✔
573
        continue;
5,397,672✔
574
      }
575

576
      break;
319,933,192✔
577
    }
578

579
    if (pProjectInfo->mergeDataBlocks) {
617,179,876✔
580
      if (pRes->info.rows > 0) {
246,144,399✔
581
        pFinalRes->info.id.groupId = 0;  // clear groupId
138,157,063✔
582
        pFinalRes->info.version = pRes->info.version;
138,157,063✔
583
        // keep baseGId from current upstream block; already set above for this merge round
584

585
        // continue merge data, ignore the group id
586
        code = blockDataMerge(pFinalRes, pRes);
138,157,063✔
587
        QUERY_CHECK_CODE(code, lino, _end);
138,157,063✔
588

589
        if (!withExternalWindow && pFinalRes->info.rows + pRes->info.rows <= pOperator->resultInfo.threshold &&
138,157,063✔
590
            (pOperator->status != OP_EXEC_DONE)) {
134,311,169✔
591
          continue;
129,130,896✔
592
        }
593
      }
594

595
      // do apply filter
596
      if (pOperator->exprSupp.pFilterInfo != NULL) {
117,014,044✔
597
        filterSetExecContext(pOperator->exprSupp.pFilterInfo, pOperator->pTaskInfo, isTaskKilled);
2,348✔
598
      }
599
      code = doFilter(pFinalRes, pOperator->exprSupp.pFilterInfo, NULL, NULL);
117,013,770✔
600
      QUERY_CHECK_CODE(code, lino, _end);
117,013,496✔
601

602
      // when apply the limit/offset for each group, pRes->info.rows may be 0, due to limit constraint.
603
      if (pFinalRes->info.rows > 0 || (pOperator->status == OP_EXEC_DONE)) {
117,013,496✔
604
        qDebug("project return %" PRId64 " rows, status %d", pFinalRes->info.rows, pOperator->status);
116,972,824✔
605
        break;
116,972,557✔
606
      }
607
    } else {
608
      // do apply filter
609
      if (pRes->info.rows > 0) {
371,043,260✔
610
        code = doFilter(pRes, pOperator->exprSupp.pFilterInfo, NULL, NULL);
181,728,582✔
611
        QUERY_CHECK_CODE(code, lino, _end);
181,655,756✔
612

613
        if (pRes->info.rows == 0) {
181,655,756✔
614
          continue;
×
615
        }
616
      }
617

618
      // no results generated
619
      break;
371,040,842✔
620
    }
621
  }
622

623
  SSDataBlock* p = pProjectInfo->mergeDataBlocks ? pFinalRes : pRes;
488,013,399✔
624
  p->info.dataLoad = 1;
488,048,183✔
625

626
  if (pProjectInfo->outputIgnoreGroup) {
488,004,079✔
627
    p->info.id.groupId = 0;
479,865,724✔
628
  }
629

630
  if (pTaskInfo->execModel == OPTR_EXEC_MODEL_STREAM) {
488,077,888✔
631
    printDataBlock(p, getStreamOpName(pOperator->operatorType), GET_TASKID(pTaskInfo), pTaskInfo->id.queryId);
8,180,663✔
632
  }
633

634
  *pResBlock = (p->info.rows > 0)? p:NULL;
488,045,492✔
635

636
_end:
489,204,376✔
637
  if (code != TSDB_CODE_SUCCESS) {
489,204,376✔
638
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
1,208,443✔
639
    pTaskInfo->code = code;
1,208,443✔
640
    T_LONG_JMP(pTaskInfo->env, code);
1,208,443✔
641
  }
642
  return code;
487,995,933✔
643
}
644

645
static int32_t resetIndefinitOutputOperState(SOperatorInfo* pOper) {
497✔
646
  SIndefOperatorInfo* pInfo = pOper->info;
497✔
647
  SExecTaskInfo*           pTaskInfo = pOper->pTaskInfo;
497✔
648
  SIndefRowsFuncPhysiNode* pPhynode = (SIndefRowsFuncPhysiNode*)pOper->pPhyNode;
497✔
649
  pOper->status = OP_NOT_OPENED;
497✔
650

651
  resetBasicOperatorState(&pInfo->binfo);
497✔
652

653
  pInfo->groupId = 0;
497✔
654
  pInfo->pNextGroupRes = NULL;
497✔
655
  int32_t code = resetAggSup(&pOper->exprSupp, &pInfo->aggSup, pTaskInfo, pPhynode->pFuncs, NULL,
994✔
656
    sizeof(int64_t) * 2 + POINTER_BYTES, pTaskInfo->id.str, NULL,
497✔
657
    &pTaskInfo->storageAPI.functionStore);
658
  if (code == 0){
497✔
659
    code = setFunctionResultOutput(pOper, &pInfo->binfo, &pInfo->aggSup, MAIN_SCAN, pOper->exprSupp.numOfExprs);
497✔
660
  }
661

662
  if (code == 0) {
497✔
663
    code = resetExprSupp(&pInfo->scalarSup, pTaskInfo, pPhynode->pExprs, NULL,
497✔
664
                         &pTaskInfo->storageAPI.functionStore);
665
  }
666
  return 0;
497✔
667
}
668

669
int32_t createIndefinitOutputOperatorInfo(SOperatorInfo* downstream, SPhysiNode* pNode,
3,480,233✔
670
                                                 SExecTaskInfo* pTaskInfo, SOperatorInfo** pOptrInfo) {
671
  QRY_PARAM_CHECK(pOptrInfo);
3,480,233✔
672
  int32_t code = 0;
3,480,233✔
673
  int32_t lino = 0;
3,480,233✔
674
  int32_t numOfRows = 4096;
3,480,233✔
675
  size_t  keyBufSize = sizeof(int64_t) + sizeof(int64_t) + POINTER_BYTES;
3,480,233✔
676

677
  SIndefOperatorInfo* pInfo = taosMemoryCalloc(1, sizeof(SIndefOperatorInfo));
3,480,233✔
678
  SOperatorInfo*      pOperator = taosMemoryCalloc(1, sizeof(SOperatorInfo));
3,478,982✔
679
  if (pInfo == NULL || pOperator == NULL) {
3,480,650✔
680
    code = terrno;
417✔
681
    goto _error;
×
682
  }
683

684
  pOperator->pPhyNode = pNode;
3,480,233✔
685
  pOperator->pTaskInfo = pTaskInfo;
3,480,233✔
686
  initOperatorCostInfo(pOperator);
3,479,805✔
687

688
  SExprSupp* pSup = &pOperator->exprSupp;
3,481,078✔
689
  pSup->hasWindowOrGroup = false;
3,481,078✔
690

691
  SIndefRowsFuncPhysiNode* pPhyNode = (SIndefRowsFuncPhysiNode*)pNode;
3,479,816✔
692

693
  if (pPhyNode->pExprs != NULL) {
3,479,816✔
694
    int32_t    num = 0;
25,615✔
695
    SExprInfo* pSExpr = NULL;
25,615✔
696
    code = createExprInfo(pPhyNode->pExprs, NULL, &pSExpr, &num);
25,615✔
697
    QUERY_CHECK_CODE(code, lino, _error);
25,615✔
698

699
    code = initExprSupp(&pInfo->scalarSup, pSExpr, num, &pTaskInfo->storageAPI.functionStore);
25,615✔
700
    if (code != TSDB_CODE_SUCCESS) {
25,615✔
701
      goto _error;
×
702
    }
703
  }
704

705
  SSDataBlock* pResBlock = createDataBlockFromDescNode(pPhyNode->node.pOutputDataBlockDesc);
3,479,399✔
706
  TSDB_CHECK_NULL(pResBlock, code, lino, _error, terrno);
3,481,078✔
707

708
  // Make sure the size of SSDataBlock will never exceed the size of 2MB.
709
  int32_t TWOMB = 2 * 1024 * 1024;
3,481,078✔
710
  if (numOfRows * pResBlock->info.rowSize > TWOMB) {
3,481,078✔
711
    numOfRows = TWOMB / pResBlock->info.rowSize;
×
712
  }
713

714
  initBasicInfo(&pInfo->binfo, pResBlock);
3,480,244✔
715
  initResultSizeInfo(&pOperator->resultInfo, numOfRows);
3,480,244✔
716
  code = blockDataEnsureCapacity(pResBlock, numOfRows);
3,479,399✔
717
  TSDB_CHECK_CODE(code, lino, _error);
3,481,078✔
718

719
  int32_t    numOfExpr = 0;
3,481,078✔
720
  SExprInfo* pExprInfo = NULL;
3,481,078✔
721
  code = createExprInfo(pPhyNode->pFuncs, NULL, &pExprInfo, &numOfExpr);
3,481,078✔
722
  TSDB_CHECK_CODE(code, lino, _error);
3,478,818✔
723

724
  code = initAggSup(pSup, &pInfo->aggSup, pExprInfo, numOfExpr, keyBufSize, pTaskInfo->id.str,
3,478,818✔
725
                            NULL, &pTaskInfo->storageAPI.functionStore);
726
  TSDB_CHECK_CODE(code, lino, _error);
3,479,246✔
727

728
  code = setFunctionResultOutput(pOperator, &pInfo->binfo, &pInfo->aggSup, MAIN_SCAN, numOfExpr);
3,479,246✔
729
  TSDB_CHECK_CODE(code, lino, _error);
3,477,174✔
730

731
  code = filterInitFromNode((SNode*)pPhyNode->node.pConditions, &pOperator->exprSupp.pFilterInfo, 0,
3,479,816✔
732
                            pTaskInfo->pStreamRuntimeInfo);
3,477,174✔
733
  TSDB_CHECK_CODE(code, lino, _error);
3,478,401✔
734

735
  pInfo->binfo.pRes = pResBlock;
3,478,401✔
736
  pInfo->binfo.inputTsOrder = pNode->inputTsOrder;
3,478,818✔
737
  pInfo->binfo.outputTsOrder = pNode->outputTsOrder;
3,478,390✔
738
  code = setRowTsColumnOutputInfo(pSup->pCtx, numOfExpr, &pInfo->pPseudoColInfo);
3,476,757✔
739
  TSDB_CHECK_CODE(code, lino, _error);
3,480,661✔
740

741
  setOperatorInfo(pOperator, "IndefinitOperator", QUERY_NODE_PHYSICAL_PLAN_INDEF_ROWS_FUNC, false, OP_NOT_OPENED, pInfo,
3,480,661✔
742
                  pTaskInfo);
743
  pOperator->fpSet = createOperatorFpSet(optrDummyOpenFn, doApplyIndefinitFunction, NULL, destroyIndefinitOperatorInfo,
3,480,661✔
744
                                         optrDefaultBufFn, NULL, optrDefaultGetNextExtFn, NULL);
745
                                         
746
  setOperatorResetStateFn(pOperator, resetIndefinitOutputOperState);
3,478,137✔
747
  code = appendDownstream(pOperator, &downstream, 1);
3,479,816✔
748
  if (code != TSDB_CODE_SUCCESS) {
3,479,827✔
749
    goto _error;
×
750
  }
751

752
  *pOptrInfo = pOperator;
3,479,827✔
753
  return TSDB_CODE_SUCCESS;
3,479,827✔
754

755
_error:
×
756
  if (pInfo != NULL) destroyIndefinitOperatorInfo(pInfo);
×
757
  destroyOperatorAndDownstreams(pOperator, &downstream, 1);
×
758
  pTaskInfo->code = code;
×
759
  return code;
×
760
}
761

762
static void doHandleDataBlock(SOperatorInfo* pOperator, SSDataBlock* pBlock, SOperatorInfo* downstream,
53,575,308✔
763
                              SExecTaskInfo* pTaskInfo) {
764
  SIndefOperatorInfo* pIndefInfo = pOperator->info;
53,575,308✔
765
  SOptrBasicInfo*     pInfo = &pIndefInfo->binfo;
53,575,308✔
766
  SExprSupp*          pSup = &pOperator->exprSupp;
53,575,308✔
767

768
  int32_t order = pInfo->inputTsOrder;
53,575,308✔
769
  int32_t scanFlag = pBlock->info.scanFlag;
53,574,726✔
770
  int32_t code = TSDB_CODE_SUCCESS;
53,574,726✔
771

772
  // there is an scalar expression that needs to be calculated before apply the group aggregation.
773
  SExprSupp* pScalarSup = &pIndefInfo->scalarSup;
53,574,726✔
774
  if (pScalarSup->pExprInfo != NULL) {
53,574,726✔
775
    code = projectApplyFunctions(pScalarSup->pExprInfo, pBlock, pBlock, pScalarSup->pCtx, pScalarSup->numOfExprs,
155,956✔
776
                                 pIndefInfo->pPseudoColInfo, GET_STM_RTINFO(pOperator->pTaskInfo),
77,978✔
777
                                 pOperator->pTaskInfo);
778
    if (code != TSDB_CODE_SUCCESS) {
77,978✔
779
      T_LONG_JMP(pTaskInfo->env, code);
×
780
    }
781
  }
782

783
  code = setInputDataBlock(pSup, pBlock, order, scanFlag, false);
53,574,726✔
784
  if (code) {
53,575,308✔
785
    T_LONG_JMP(pTaskInfo->env, code);
×
786
  }
787

788
  code = blockDataEnsureCapacity(pInfo->pRes, pInfo->pRes->info.rows + pBlock->info.rows);
53,575,308✔
789
  if (code != TSDB_CODE_SUCCESS) {
53,574,726✔
790
    T_LONG_JMP(pTaskInfo->env, code);
×
791
  }
792

793
  code = projectApplyFunctions(pSup->pExprInfo, pInfo->pRes, pBlock, pSup->pCtx, pSup->numOfExprs,
107,150,034✔
794
                               pIndefInfo->pPseudoColInfo, GET_STM_RTINFO(pOperator->pTaskInfo),
53,574,726✔
795
                               pOperator->pTaskInfo);
796
  if (code != TSDB_CODE_SUCCESS) {
53,575,308✔
797
    T_LONG_JMP(pTaskInfo->env, code);
20,064✔
798
  }
799
}
53,555,244✔
800

801
int32_t doApplyIndefinitFunction(SOperatorInfo* pOperator, SSDataBlock** pResBlock) {
11,914,573✔
802
  QRY_PARAM_CHECK(pResBlock);
11,914,573✔
803
  SIndefOperatorInfo* pIndefInfo = pOperator->info;
11,916,252✔
804
  SOptrBasicInfo*     pInfo = &pIndefInfo->binfo;
11,915,824✔
805
  SExprSupp*          pSup = &pOperator->exprSupp;
11,915,835✔
806
  int32_t             code = TSDB_CODE_SUCCESS;
11,915,001✔
807
  int32_t             lino = 0;
11,915,001✔
808
  SSDataBlock*        pRes = pInfo->pRes;
11,915,001✔
809

810
  blockDataCleanup(pRes);
11,912,740✔
811

812
  SExecTaskInfo* pTaskInfo = pOperator->pTaskInfo;
11,915,418✔
813
  if (pOperator->status == OP_EXEC_DONE) {
11,915,418✔
814
    return code;
2,307,681✔
815
  }
816

817
  SOperatorInfo* downstream = pOperator->pDownstream[0];
9,605,904✔
818
  bool           noSplitOutput = hasLagLeadFunc(pSup);
9,606,475✔
819

820
  while (1) {
3,151,009✔
821
    // here we need to handle the existsed group results
822
    if (pIndefInfo->pNextGroupRes != NULL) {  // todo extract method
12,758,735✔
823
      for (int32_t k = 0; k < pSup->numOfExprs; ++k) {
23,313,711✔
824
        SqlFunctionCtx* pCtx = &pSup->pCtx[k];
17,291,962✔
825

826
        SResultRowEntryInfo* pResInfo = GET_RES_INFO(pCtx);
17,291,962✔
827
        if (pResInfo->initialized && pCtx->fpSet.cleanup != NULL) {
17,291,962✔
828
          pCtx->fpSet.cleanup(pCtx);
2,431✔
829
        }
830
        pResInfo->initialized = false;
17,291,962✔
831
        pCtx->pOutput = NULL;
17,291,962✔
832
      }
833

834
      doHandleDataBlock(pOperator, pIndefInfo->pNextGroupRes, downstream, pTaskInfo);
6,021,749✔
835
      pIndefInfo->pNextGroupRes = NULL;
6,021,749✔
836
    }
837

838
    if (noSplitOutput || pInfo->pRes->info.rows < pOperator->resultInfo.threshold) {
12,758,318✔
839
      while (1) {
44,247,236✔
840
        // The downstream exec may change the value of the newgroup, so use a local variable instead.
841
        SSDataBlock* pBlock = getNextBlockFromDownstream(pOperator, 0);
57,004,709✔
842
        if (pBlock == NULL) {
57,006,816✔
843
          setOperatorCompleted(pOperator);
3,408,368✔
844
          break;
3,408,368✔
845
        }
846
        pInfo->pRes->info.scanFlag = pBlock->info.scanFlag;
53,598,448✔
847

848
        if (pIndefInfo->groupId == 0 && pBlock->info.id.groupId != 0) {
53,598,448✔
849
          pIndefInfo->groupId = pBlock->info.id.groupId;  // this is the initial group result
445,065✔
850
        } else {
851
          if (pIndefInfo->groupId != pBlock->info.id.groupId) {  // reset output buffer and computing status
53,153,383✔
852
            pIndefInfo->groupId = pBlock->info.id.groupId;
6,044,889✔
853
            pIndefInfo->pNextGroupRes = pBlock;
6,044,889✔
854
            break;
6,044,889✔
855
          }
856
        }
857

858
        doHandleDataBlock(pOperator, pBlock, downstream, pTaskInfo);
47,553,559✔
859
        // External-window outputs carry per-window row ranges in stream runtime state.
860
        // Return as soon as this operator has a result block so the downstream state
861
        // still matches the block we are about to hand back to the runner.
862
        if (pTaskInfo->pStreamRuntimeInfo != NULL && pTaskInfo->pStreamRuntimeInfo->funcInfo.withExternalWindow &&
47,533,495✔
863
            pInfo->pRes->info.rows > 0) {
×
864
          break;
×
865
        }
866
        if (!noSplitOutput && pInfo->pRes->info.rows >= pOperator->resultInfo.threshold) {
47,533,495✔
867
          break;
3,286,259✔
868
        }
869
      }
870
    }
871

872
    code = doFilter(pInfo->pRes, pOperator->exprSupp.pFilterInfo, NULL, NULL);
12,741,030✔
873
    QUERY_CHECK_CODE(code, lino, _end);
12,739,516✔
874

875
    size_t rows = pInfo->pRes->info.rows;
12,739,516✔
876
    if (rows > 0 || pOperator->status == OP_EXEC_DONE) {
12,738,682✔
877
      break;
878
    } else {
879
      blockDataCleanup(pInfo->pRes);
3,151,009✔
880
    }
881
  }
882

883
  *pResBlock = (pInfo->pRes->info.rows> 0) ? pInfo->pRes : NULL;
9,587,673✔
884

885
_end:
9,587,673✔
886
  if (code != TSDB_CODE_SUCCESS) {
9,587,673✔
887
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
×
888
    pTaskInfo->code = code;
×
889
    T_LONG_JMP(pTaskInfo->env, code);
×
890
  }
891
  return code;
9,587,673✔
892
}
893

894
int32_t initCtxOutputBuffer(SqlFunctionCtx* pCtx, int32_t size) {
173,287,588✔
895
  int32_t code = TSDB_CODE_SUCCESS;
173,287,588✔
896
  for (int32_t j = 0; j < size; ++j) {
814,403,060✔
897
    struct SResultRowEntryInfo* pResInfo = GET_RES_INFO(&pCtx[j]);
641,186,832✔
898
    if (isRowEntryInitialized(pResInfo) || fmIsPseudoColumnFunc(pCtx[j].functionId) || pCtx[j].functionId == -1 ||
698,099,909✔
899
        fmIsScalarFunc(pCtx[j].functionId)) {
56,976,628✔
900
      continue;
636,928,154✔
901
    }
902

903
    code = pCtx[j].fpSet.init(&pCtx[j], pCtx[j].resultInfo);
4,219,311✔
904
    if (code) {
4,187,318✔
905
      return code;
×
906
    }
907
  }
908

909
  return 0;
173,216,228✔
910
}
911

912
/*
913
 * The start of each column SResultRowEntryInfo is denote by RowCellInfoOffset.
914
 * Note that in case of top/bottom query, the whole multiple rows of result is treated as only one row of results.
915
 * +------------+-----------------result column 1------------+------------------result column 2-----------+
916
 * | SResultRow | SResultRowEntryInfo | intermediate buffer1 | SResultRowEntryInfo | intermediate buffer 2|
917
 * +------------+--------------------------------------------+--------------------------------------------+
918
 *           offset[0]                                  offset[1]                                   offset[2]
919
 */
920
// TODO refactor: some function move away
921
int32_t setFunctionResultOutput(struct SOperatorInfo* pOperator, SOptrBasicInfo* pInfo, SAggSupporter* pSup, int32_t stage,
173,291,030✔
922
                             int32_t numOfExprs) {
923
  SExecTaskInfo*  pTaskInfo = pOperator->pTaskInfo;
173,291,030✔
924
  SqlFunctionCtx* pCtx = pOperator->exprSupp.pCtx;
173,320,776✔
925
  int32_t*        rowEntryInfoOffset = pOperator->exprSupp.rowEntryInfoOffset;
173,290,075✔
926

927
  SResultRowInfo* pResultRowInfo = &pInfo->resultRowInfo;
173,297,212✔
928
  initResultRowInfo(pResultRowInfo);
173,273,663✔
929

930
  int64_t     tid = 0;
173,218,837✔
931
  int64_t     groupId = 0;
173,220,462✔
932
  SResultRow* pRow = doSetResultOutBufByKey(pSup->pResultBuf, pResultRowInfo, (char*)&tid, sizeof(tid), true, groupId,
173,220,462✔
933
                                            pTaskInfo, false, pSup, true);
934
  if (pRow == NULL || pTaskInfo->code != 0) {
173,232,844✔
935
    return pTaskInfo->code;
66,028✔
936
  }
937

938
  for (int32_t i = 0; i < numOfExprs; ++i) {
814,375,043✔
939
    struct SResultRowEntryInfo* pEntry = getResultEntryInfo(pRow, i, rowEntryInfoOffset);
641,125,925✔
940
    cleanupResultRowEntry(pEntry);
641,184,014✔
941

942
    pCtx[i].resultInfo = pEntry;
641,149,605✔
943
    pCtx[i].scanFlag = stage;
641,140,988✔
944
  }
945

946
  return initCtxOutputBuffer(pCtx, numOfExprs);
173,249,118✔
947
}
948

949
int32_t setRowTsColumnOutputInfo(SqlFunctionCtx* pCtx, int32_t numOfCols, SArray** pResList) {
163,364,165✔
950
  QRY_PARAM_CHECK(pResList);
163,364,165✔
951
  SArray* pList = taosArrayInit(4, sizeof(int32_t));
163,313,167✔
952
  if (pList == NULL) {
163,371,104✔
953
    return terrno;
×
954
  }
955

956
  for (int32_t i = 0; i < numOfCols; ++i) {
766,847,833✔
957
    if (fmIsPseudoColumnFunc(pCtx[i].functionId) && !fmIsPlaceHolderFunc(pCtx[i].functionId)) {
603,601,510✔
958
      void* px = taosArrayPush(pList, &i);
167,353✔
959
      if (px == NULL) {
167,353✔
960
        return terrno;
×
961
      }
962
    }
963
  }
964

965
  *pResList = pList;
163,284,136✔
966
  return 0;
163,288,888✔
967
}
968

969
int32_t doGenerateSourceData(SOperatorInfo* pOperator) {
2,028,006✔
970
  SProjectOperatorInfo* pProjectInfo = pOperator->info;
2,028,006✔
971

972
  SExprSupp*   pSup = &pOperator->exprSupp;
2,028,006✔
973
  SSDataBlock* pRes = pProjectInfo->binfo.pRes;
2,028,006✔
974
  SExprInfo*   pExpr = pSup->pExprInfo;
2,028,006✔
975
  SExecTaskInfo* pTaskInfo = pOperator->pTaskInfo;
2,028,006✔
976

977
  int32_t code = blockDataEnsureCapacity(pRes, pOperator->resultInfo.capacity);
2,028,006✔
978
  if (code) {
2,028,006✔
979
    return code;
×
980
  }
981

982
  for (int32_t k = 0; k < pSup->numOfExprs; ++k) {
4,060,811✔
983
    int32_t outputSlotId = pExpr[k].base.resSchema.slotId;
2,032,805✔
984

985
    if (pExpr[k].pExpr->nodeType == QUERY_NODE_VALUE) {
2,032,805✔
986
      SColumnInfoData* pColInfoData = taosArrayGet(pRes->pDataBlock, outputSlotId);
2,025,205✔
987
      if (pColInfoData == NULL) {
2,024,770✔
988
        return terrno;
×
989
      }
990

991
      int32_t type = pExpr[k].base.pParam[0].param.nType;
2,024,770✔
992
      if (TSDB_DATA_TYPE_NULL == type) {
2,024,770✔
993
        colDataSetNNULL(pColInfoData, 0, 1);
994
      } else {
995
        code = colDataSetVal(pColInfoData, 0, taosVariantGet(&pExpr[k].base.pParam[0].param, type), false);
2,021,446✔
996
        if (code) {
2,021,446✔
997
          return code;
×
998
        }
999
      }
1000
    } else if (pExpr[k].pExpr->nodeType == QUERY_NODE_FUNCTION) {
7,600✔
1001
      SqlFunctionCtx* pfCtx = &pSup->pCtx[k];
6,175✔
1002

1003
      // UDF scalar functions will be calculated here, for example, select foo(n) from (select 1 n).
1004
      // UDF aggregate functions will be handled in agg operator.
1005
      if (fmIsScalarFunc(pfCtx->functionId)) {
6,175✔
1006
        SArray* pBlockList = taosArrayInit(4, POINTER_BYTES);
6,175✔
1007
        if (pBlockList == NULL) {
6,175✔
1008
          return terrno;
×
1009
        }
1010

1011
        void* px = taosArrayPush(pBlockList, &pRes);
6,175✔
1012
        if (px == NULL) {
6,175✔
1013
          return terrno;
×
1014
        }
1015

1016
        SColumnInfoData* pResColData = taosArrayGet(pRes->pDataBlock, outputSlotId);
6,175✔
1017
        if (pResColData == NULL) {
6,175✔
1018
          return terrno;
×
1019
        }
1020

1021
        SColumnInfoData  idata = {.info = pResColData->info, .hasNull = true};
6,175✔
1022

1023
        SScalarParam dest = {.columnData = &idata};
6,175✔
1024
        gTaskScalarExtra.pStreamInfo  = GET_STM_RTINFO(pOperator->pTaskInfo);
6,175✔
1025
        gTaskScalarExtra.pStreamRange = NULL;
6,175✔
1026
        gTaskScalarExtra.pTaskInfo    = pOperator->pTaskInfo;
6,175✔
1027
        gTaskScalarExtra.isTaskKilled = isTaskKilled;
6,175✔
1028
        code = scalarCalculate((SNode*)pExpr[k].pExpr->_function.pFunctNode, pBlockList, &dest, &gTaskScalarExtra);
6,175✔
1029
        if (code != TSDB_CODE_SUCCESS) {
6,175✔
1030
          taosArrayDestroy(pBlockList);
×
1031
          return code;
×
1032
        }
1033

1034
        int32_t startOffset = pRes->info.rows;
6,175✔
1035
        if (pRes->info.capacity <= 0) {
6,175✔
1036
          qError("project failed at: %s:%d", __func__, __LINE__);
×
1037
          return TSDB_CODE_QRY_EXECUTOR_INTERNAL_ERROR;
×
1038
        }
1039
        code = colDataAssign(pResColData, &idata, dest.numOfRows, &pRes->info);
6,175✔
1040
        if (code) {
6,175✔
1041
          return code;
×
1042
        }
1043

1044
        colDataDestroy(&idata);
6,175✔
1045
        taosArrayDestroy(pBlockList);
6,175✔
1046
      } else {
1047
        return TSDB_CODE_OPS_NOT_SUPPORT;
×
1048
      }
1049
    } else if (pExpr[k].pExpr->nodeType == QUERY_NODE_OPERATOR) {
1,425✔
1050
      TAOS_CHECK_RETURN(projectApplyOperator(&pExpr[k], pRes, NULL, outputSlotId, NULL, false, &gTaskScalarExtra));
1,425✔
1051
    } else {
1052
      return TSDB_CODE_OPS_NOT_SUPPORT;
×
1053
    }
1054
  }
1055

1056
  pRes->info.rows = 1;
2,028,006✔
1057
  code = doFilter(pRes, pOperator->exprSupp.pFilterInfo, NULL, NULL);
2,028,006✔
1058
  if (code) {
2,027,571✔
1059
    pTaskInfo->code = code;
×
1060
    return code;
×
1061
  }
1062

1063
  (void) doIngroupLimitOffset(&pProjectInfo->limitInfo, 0, pRes, pOperator);
2,027,571✔
1064

1065
  setOperatorCompleted(pOperator);
2,028,006✔
1066

1067
  return code;
2,028,006✔
1068
}
1069

1070
static void setPseudoOutputColInfo(SSDataBlock* pResult, SqlFunctionCtx* pCtx, SArray* pPseudoList) {
553,103,231✔
1071
  size_t num = (pPseudoList != NULL) ? taosArrayGetSize(pPseudoList) : 0;
553,103,231✔
1072
  for (int32_t i = 0; i < num; ++i) {
555,786,551✔
1073
    pCtx[i].pOutput = taosArrayGet(pResult->pDataBlock, i);
2,551,318✔
1074
    if (pCtx[i].pOutput == NULL) {
2,551,318✔
1075
      qError("failed to get the output buf, ptr is null");
×
1076
    }
1077
  }
1078
}
553,235,233✔
1079

1080
int32_t projectApplyColumn(SSDataBlock* pResult, SSDataBlock* pSrcBlock, int32_t outputSlotId, SqlFunctionCtx* pfCtx, int32_t* numOfRows, bool createNewColModel) {
1,238,635,332✔
1081
  int32_t code = 0, lino = 0;
1,238,635,332✔
1082
  SInputColumnInfoData* pInputData = &pfCtx->input;
1,238,635,332✔
1083
  SColumnInfoData* pColInfoData = taosArrayGet(pResult->pDataBlock, outputSlotId);
1,238,676,464✔
1084
  TSDB_CHECK_NULL(pColInfoData, code, lino, _exit, terrno);
1,238,684,185✔
1085

1086
  if (pResult->info.rows > 0 && !createNewColModel) {
1,238,684,185✔
1087
    if (pInputData->pData[0] == NULL) {
11,113,532✔
1088
      int32_t slotId = pfCtx->param[0].pCol->slotId;
11,113,532✔
1089

1090
      SColumnInfoData* pInput = taosArrayGet(pSrcBlock->pDataBlock, slotId);
11,113,532✔
1091
      TSDB_CHECK_NULL(pInput, code, lino, _exit, terrno);
11,113,975✔
1092

1093
      TAOS_CHECK_EXIT(colDataMergeCol(pColInfoData, pResult->info.rows, (int32_t*)&pResult->info.capacity, pInput,
11,113,975✔
1094
                            pSrcBlock->info.rows));
1095
      *numOfRows = pSrcBlock->info.rows;
11,113,089✔
1096
      return code;
11,113,975✔
1097
    }
1098
    
1099
    TAOS_CHECK_EXIT(colDataMergeCol(pColInfoData, pResult->info.rows, (int32_t*)&pResult->info.capacity,
×
1100
                          pInputData->pData[0], pInputData->numOfRows));
1101
    *numOfRows = pInputData->numOfRows;
×
1102
    return code;
×
1103
  } 
1104
  
1105
  if (pInputData->pData[0] == NULL) {
1,227,589,140✔
1106
    int32_t slotId = pfCtx->param[0].pCol->slotId;
8,936,727✔
1107

1108
    SColumnInfoData* pInput = taosArrayGet(pSrcBlock->pDataBlock, slotId);
8,936,727✔
1109
    TSDB_CHECK_NULL(pInput, code, lino, _exit, terrno);
8,937,170✔
1110

1111
    TAOS_CHECK_EXIT(colDataAssign(pColInfoData, pInput, pSrcBlock->info.rows, &pResult->info));
8,937,170✔
1112
    *numOfRows = pSrcBlock->info.rows;
8,936,727✔
1113

1114
    return code;
8,936,727✔
1115
  }
1116
  
1117
  TAOS_CHECK_EXIT(colDataAssign(pColInfoData, pInputData->pData[0], pInputData->numOfRows, &pResult->info));
1,218,548,033✔
1118
  *numOfRows = pInputData->numOfRows;
1,218,624,801✔
1119

1120
_exit:
1,218,635,259✔
1121

1122
  if (code) {
1,218,635,259✔
1123
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
×
1124
  }
1125
  
1126
  return code;
1,218,598,841✔
1127
}
1128

1129

1130
int32_t projectApplyValue(SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock, int32_t outputSlotId, int32_t* numOfRows, bool createNewColModel) {
19,583,992✔
1131
  int32_t code = 0, lino = 0;
19,583,992✔
1132
  SColumnInfoData* pColInfoData = taosArrayGet(pResult->pDataBlock, outputSlotId);
19,583,992✔
1133
  TSDB_CHECK_NULL(pColInfoData, code, lino, _exit, terrno);
19,580,414✔
1134

1135
  int32_t offset = createNewColModel ? 0 : pResult->info.rows;
19,580,414✔
1136
  int32_t type = pExpr->base.pParam[0].param.nType;
19,582,251✔
1137
  if (TSDB_DATA_TYPE_NULL == type) {
19,575,173✔
1138
    colDataSetNNULL(pColInfoData, offset, pSrcBlock->info.rows);
720,870✔
1139
  } else {
1140
    char* p = taosVariantGet(&pExpr->base.pParam[0].param, type);
18,854,303✔
1141
    for (int32_t i = 0; i < pSrcBlock->info.rows; ++i) {
2,147,483,647✔
1142
      TAOS_CHECK_EXIT(colDataSetVal(pColInfoData, i + offset, p, false));
2,147,483,647✔
1143
    }
1144
  }
1145

1146
  *numOfRows = pSrcBlock->info.rows;
50,690,196✔
1147

1148
_exit:
19,583,635✔
1149

1150
  if (code) {
19,583,635✔
1151
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
×
1152
  }
1153
  
1154
  return code;
19,577,764✔
1155
}
1156

1157

1158

1159
int32_t projectApplyOperator(SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock, int32_t outputSlotId, int32_t* numOfRows, bool createNewColModel, const void* pExtraParams) {
91,974,189✔
1160
  int32_t code = 0, lino = 0;
91,974,189✔
1161
  SArray* pBlockList = NULL;
91,974,189✔
1162
  if (NULL != pSrcBlock) {
91,974,189✔
1163
    pBlockList = taosArrayInit(4, POINTER_BYTES);
91,975,120✔
1164
    TSDB_CHECK_NULL(pBlockList, code, lino, _exit, terrno);
91,976,867✔
1165

1166
    void* px = taosArrayPush(pBlockList, &pSrcBlock);
91,974,523✔
1167
    TSDB_CHECK_NULL(px, code, lino, _exit, terrno);
91,974,523✔
1168
  }
1169
  
1170
  SColumnInfoData* pResColData = taosArrayGet(pResult->pDataBlock, outputSlotId);
91,973,592✔
1171
  TSDB_CHECK_NULL(pResColData, code, lino, _exit, terrno);
91,973,598✔
1172

1173
  SColumnInfoData idata = {.info = pResColData->info, .hasNull = true};
91,973,598✔
1174
  SScalarParam dest = {.columnData = &idata};
91,973,634✔
1175
  gTaskScalarExtra.pStreamInfo = (void*)pExtraParams;
91,974,900✔
1176
  gTaskScalarExtra.pStreamRange = NULL;
91,974,900✔
1177
  TAOS_CHECK_EXIT(scalarCalculate(pExpr->pExpr->_optrRoot.pRootNode, pBlockList, &dest, &gTaskScalarExtra));
91,972,735✔
1178

1179
  if (pResult->info.rows > 0 && !createNewColModel) {
90,649,278✔
1180
    code = colDataMergeCol(pResColData, pResult->info.rows, (int32_t*)&pResult->info.capacity, &idata, dest.numOfRows);
7,088✔
1181
  } else {
1182
    code = colDataAssign(pResColData, &idata, dest.numOfRows, &pResult->info);
90,640,269✔
1183
  }
1184

1185
  colDataDestroy(&idata);
90,659,037✔
1186
  TAOS_CHECK_EXIT(code);
90,657,249✔
1187

1188
  if (numOfRows) {
90,657,249✔
1189
    *numOfRows = dest.numOfRows;
90,656,420✔
1190
  }
1191
  
1192
_exit:
91,978,908✔
1193

1194
  if (code < 0) {
91,975,597✔
1195
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
1,321,063✔
1196
  }
1197

1198
  taosArrayDestroy(pBlockList);
91,975,597✔
1199
  
1200
  return code;
91,976,524✔
1201
}
1202

1203

1204
int32_t projectApplyFunction(SqlFunctionCtx* pCtx, SqlFunctionCtx* pfCtx, SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock, 
607,189,247✔
1205
                                    int32_t outputSlotId, int32_t* numOfRows, bool createNewColModel, const void* pExtraParams, 
1206
                                    SArray* pPseudoList, SArray** processByRowFunctionCtx, bool doSelectFunc) {
1207
  int32_t code = 0, lino = 0;
607,189,247✔
1208
  SArray* pBlockList = NULL;
607,189,247✔
1209
  SColumnInfoData* pResColData = taosArrayGet(pResult->pDataBlock, outputSlotId);
607,189,247✔
1210
  TSDB_CHECK_NULL(pResColData, code, lino, _exit, terrno);
607,242,173✔
1211

1212
  if (fmIsPlaceHolderFunc(pfCtx->functionId) && pExtraParams && pfCtx->pExpr->base.pParamList && 1 == pfCtx->pExpr->base.pParamList->length) {
607,242,173✔
1213
    SNode* pParamNode = nodesListGetNode(pfCtx->pExpr->base.pParamList, 0);
3,304,408✔
1214
    SStreamRuntimeFuncInfo* pStreamInfo = (SStreamRuntimeFuncInfo*)pExtraParams;
3,304,408✔
1215
    if (pStreamInfo != NULL && pStreamInfo->withExternalWindow && pStreamInfo->pStreamBlkWinIdx != NULL &&
3,304,408✔
1216
        taosArrayGetSize(pStreamInfo->pStreamBlkWinIdx) > 1) {
×
1217
      TAOS_CHECK_EXIT(assignPlaceHolderInExternalWindows(pResColData, pResult->info.rows, pSrcBlock->info.rows,
×
1218
                                                         pfCtx->functionId, pStreamInfo, pParamNode));
1219
    } else {
1220
      TAOS_CHECK_EXIT(scalarAssignPlaceHolderRes(pResColData, pResult->info.rows, pSrcBlock->info.rows,
3,304,408✔
1221
                                                 pfCtx->functionId, pExtraParams, pParamNode));
1222
    }
1223
    *numOfRows = pSrcBlock->info.rows;
3,304,408✔
1224

1225
    return code;
3,304,408✔
1226
  }
1227

1228
  if (fmIsScalarFunc(pfCtx->functionId) || fmIsPlaceHolderFunc(pfCtx->functionId)) {
603,873,283✔
1229
    pBlockList = taosArrayInit(4, POINTER_BYTES);
445,845,192✔
1230
    TSDB_CHECK_NULL(pBlockList, code, lino, _exit, terrno);
445,898,282✔
1231

1232
    void* px = taosArrayPush(pBlockList, &pSrcBlock);
445,861,299✔
1233
    TSDB_CHECK_NULL(px, code, lino, _exit, terrno);
445,861,299✔
1234

1235
    SColumnInfoData idata = {.info = pResColData->info, .hasNull = true};
445,861,299✔
1236
    SScalarParam dest = {.columnData = &idata};
445,881,446✔
1237
    gTaskScalarExtra.pStreamInfo = (void*)pExtraParams;
445,913,626✔
1238
    gTaskScalarExtra.pStreamRange = NULL;
445,913,626✔
1239
    TAOS_CHECK_EXIT(scalarCalculate((SNode*)pExpr->pExpr->_function.pFunctNode, pBlockList, &dest, &gTaskScalarExtra));
445,929,808✔
1240

1241
    if (pResult->info.rows > 0 && !createNewColModel) {
445,689,530✔
1242
      code = colDataMergeCol(pResColData, pResult->info.rows, (int32_t*)&pResult->info.capacity, &idata, dest.numOfRows);
19,980,082✔
1243
    } else {
1244
      SColumnInfo oriInfo = pResColData->info;
425,811,299✔
1245
      code = colDataAssign(pResColData, &idata, dest.numOfRows, &pResult->info);
425,895,065✔
1246
      // restore the original column info to satisfy the output column schema
1247
      pResColData->info = oriInfo;
425,932,062✔
1248
    }
1249

1250
    colDataDestroy(&idata);
445,889,131✔
1251
    taosArrayDestroy(pBlockList);
445,895,747✔
1252
    TAOS_CHECK_EXIT(code);
445,891,405✔
1253

1254
    *numOfRows = dest.numOfRows;
445,891,405✔
1255

1256
    return code;
445,894,747✔
1257
  }
1258

1259
  if (fmIsIndefiniteRowsFunc(pfCtx->functionId)) {
157,975,589✔
1260
    SResultRowEntryInfo* pResInfo = GET_RES_INFO(pfCtx);
65,018,800✔
1261
    TAOS_CHECK_EXIT(pfCtx->fpSet.init(pfCtx, pResInfo));
65,018,800✔
1262

1263

1264
    pfCtx->pOutput = (char*)pResColData;
65,018,800✔
1265
    TSDB_CHECK_NULL(pfCtx->pOutput, code, lino, _exit, terrno);
65,018,800✔
1266

1267
    pfCtx->offset = createNewColModel ? 0 : pResult->info.rows;  // set the start offset
65,018,800✔
1268

1269
    // set the timestamp(_rowts) output buffer
1270
    if (taosArrayGetSize(pPseudoList) > 0) {
65,019,382✔
1271
      int32_t* outputColIndex = taosArrayGet(pPseudoList, 0);
1,062,038✔
1272
      TSDB_CHECK_NULL(outputColIndex, code, lino, _exit, terrno);
1,062,038✔
1273

1274
      pfCtx->pTsOutput = (SColumnInfoData*)pCtx[*outputColIndex].pOutput;
1,062,038✔
1275
    }
1276

1277
    // link pDstBlock to set selectivity value
1278
    if (pfCtx->subsidiaries.num > 0) {
65,018,965✔
1279
      pfCtx->pDstBlock = pResult;
49,399,134✔
1280
    }
1281

1282
    code = pfCtx->fpSet.process(pfCtx);
65,019,382✔
1283
    if (code != TSDB_CODE_SUCCESS) {
65,019,382✔
1284
      if (pfCtx->fpSet.cleanup != NULL) {
14,124✔
1285
        pfCtx->fpSet.cleanup(pfCtx);
×
1286
      }
1287
      TAOS_CHECK_EXIT(code);
14,124✔
1288
    }
1289

1290
    *numOfRows = pResInfo->numOfRes;
65,005,258✔
1291
    
1292
    if (fmIsProcessByRowFunc(pfCtx->functionId)) {
65,005,258✔
1293
      if (NULL == *processByRowFunctionCtx) {
61,358,865✔
1294
        *processByRowFunctionCtx = taosArrayInit(1, sizeof(SqlFunctionCtx*));
61,266,813✔
1295
        TSDB_CHECK_NULL(*processByRowFunctionCtx, code, lino, _exit, terrno);
61,266,813✔
1296
      }
1297

1298
      void* px = taosArrayPush(*processByRowFunctionCtx, &pfCtx);
61,358,865✔
1299
      TSDB_CHECK_NULL(px, code, lino, _exit, terrno);
61,358,865✔
1300
    }
1301

1302
    return code;
65,005,258✔
1303
  } 
1304

1305
  if (fmIsAggFunc(pfCtx->functionId)) {
92,956,789✔
1306
    // selective value output should be set during corresponding function execution
1307
    if (!doSelectFunc && fmIsSelectValueFunc(pfCtx->functionId)) {
90,405,471✔
1308
      return code;
49,575,145✔
1309
    }
1310
    
1311
    // _group_key function for "partition by tbname" + csum(col_name) query
1312
    int32_t slotId = pfCtx->param[0].pCol->slotId;
40,830,326✔
1313

1314
    // todo handle the json tag
1315
    SColumnInfoData* pInput = taosArrayGet(pSrcBlock->pDataBlock, slotId);
40,830,326✔
1316
    TSDB_CHECK_NULL(pInput, code, lino, _exit, terrno);
40,830,326✔
1317

1318
    for (int32_t f = 0; f < pSrcBlock->info.rows; ++f) {
2,147,483,647✔
1319
      bool isNull = colDataIsNull_s(pInput, f);
2,147,483,647✔
1320
      if (isNull) {
2,147,483,647✔
1321
        colDataSetNULL(pResColData, pResult->info.rows + f);
280,321,780✔
1322
      } else {
1323
        char* data = colDataGetData(pInput, f);
2,147,483,647✔
1324
        TAOS_CHECK_EXIT(colDataSetVal(pResColData, pResult->info.rows + f, data, isNull));
2,147,483,647✔
1325
      }
1326
    }
1327

1328
    *numOfRows = pSrcBlock->info.rows;
40,830,326✔
1329

1330
    return code;
40,830,326✔
1331
  } 
1332
  
1333
  if (fmIsGroupIdFunc(pfCtx->functionId)) {
2,551,318✔
1334
    for (int32_t f = 0; f < pSrcBlock->info.rows; ++f) {
×
1335
      TAOS_CHECK_EXIT(colDataSetVal(pResColData, pResult->info.rows + f, (const char*)&pSrcBlock->info.id.groupId, false));
×
1336
    }
1337

1338
    *numOfRows = pSrcBlock->info.rows;
×
1339
    return code;
×
1340
  }
1341
  
1342
_exit:
2,551,318✔
1343

1344
  if (code) {
2,602,010✔
1345
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
50,692✔
1346
  }
1347

1348
  taosArrayDestroy(pBlockList);
2,602,010✔
1349
  
1350
  return code;
2,602,010✔
1351
}
1352

1353

1354
int32_t projectApplyFunctionsWithSelect(SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock,
564,305,291✔
1355
                                        SqlFunctionCtx* pCtx, int32_t numOfOutput, SArray* pPseudoList,
1356
                                        const void* pExtraParams, bool doSelectFunc, bool hasIndefRowsFunc,
1357
                                        SExecTaskInfo* pTaskInfo) {
1358
  int32_t lino = 0;
564,305,291✔
1359
  int32_t code = TSDB_CODE_SUCCESS;
564,305,291✔
1360

1361
  SExecTaskInfo* savedTaskInfo = gTaskScalarExtra.pTaskInfo;
564,305,291✔
1362
  __typeof__(gTaskScalarExtra.isTaskKilled) savedIsTaskKilled = gTaskScalarExtra.isTaskKilled;
564,363,795✔
1363

1364
  if (pTaskInfo != NULL) {
564,401,899✔
1365
    gTaskScalarExtra.pTaskInfo    = pTaskInfo;
564,422,667✔
1366
    gTaskScalarExtra.isTaskKilled = isTaskKilled;
564,390,169✔
1367
  }
1368

1369
  if (hasIndefRowsFunc) {
564,390,291✔
1370
    setPseudoOutputColInfo(pResult, pCtx, pPseudoList);
553,265,414✔
1371
  }
1372
  pResult->info.dataLoad = 1;
564,310,141✔
1373

1374
  SArray* processByRowFunctionCtx = NULL;
564,364,749✔
1375
  SArray* pProcessedFuncIds = NULL;
564,228,616✔
1376
  SArray* pGroupedCtxArray = NULL;
564,228,616✔
1377
  if (pSrcBlock == NULL) {
564,228,616✔
1378
    for (int32_t k = 0; k < numOfOutput; ++k) {
×
1379
      int32_t outputSlotId = pExpr[k].base.resSchema.slotId;
×
1380

1381
      if (pExpr[k].pExpr->nodeType != QUERY_NODE_VALUE) {
×
1382
        qError("project failed at: %s:%d", __func__, __LINE__);
×
1383
        TAOS_CHECK_EXIT(TSDB_CODE_INVALID_PARA);
×
1384
      }
1385
      SColumnInfoData* pColInfoData = taosArrayGet(pResult->pDataBlock, outputSlotId);
×
1386
      TSDB_CHECK_NULL(pColInfoData, code, lino, _exit, terrno);
×
1387

1388
      int32_t type = pExpr[k].base.pParam[0].param.nType;
×
1389
      if (TSDB_DATA_TYPE_NULL == type) {
×
1390
        colDataSetNNULL(pColInfoData, 0, 1);
1391
      } else {
1392
        TAOS_CHECK_EXIT(colDataSetVal(pColInfoData, 0, taosVariantGet(&pExpr[k].base.pParam[0].param, type), false));
×
1393
      }
1394
    }
1395

1396
    pResult->info.rows = 1;
×
1397
    goto _exit;
×
1398
  }
1399

1400
  if (pResult != pSrcBlock) {
564,228,616✔
1401
    pResult->info.id.groupId = pSrcBlock->info.id.groupId;
407,280,362✔
1402
    if (pSrcBlock->info.parTbName[0]) {
407,366,839✔
1403
      tstrncpy(pResult->info.parTbName, pSrcBlock->info.parTbName, TSDB_TABLE_NAME_LEN);
×
1404
    }
1405
    qTrace("%s, parName:%s,groupId:%" PRIu64, __FUNCTION__, pSrcBlock->info.parTbName, pResult->info.id.groupId);
407,330,062✔
1406
  }
1407

1408
  // if the source equals to the destination, it is to create a new column as the result of scalar
1409
  // function or some operators.
1410
  bool createNewColModel = (pResult == pSrcBlock);
564,234,894✔
1411
  if (createNewColModel) {
564,234,894✔
1412
    TAOS_CHECK_EXIT(blockDataEnsureCapacity(pResult, pResult->info.rows));
157,012,556✔
1413
  }
1414

1415
  int32_t numOfRows = 0;
564,236,754✔
1416

1417
  for (int32_t k = 0; k < numOfOutput; ++k) {
2,147,483,647✔
1418
    int32_t               outputSlotId = pExpr[k].base.resSchema.slotId;
1,957,189,009✔
1419
    SqlFunctionCtx*       pfCtx = &pCtx[k];
1,957,345,075✔
1420
    switch (pExpr[k].pExpr->nodeType) {
1,957,300,894✔
1421
      case QUERY_NODE_COLUMN: {
1,238,638,181✔
1422
        TAOS_CHECK_EXIT(projectApplyColumn(pResult, pSrcBlock, outputSlotId, pfCtx, &numOfRows, createNewColModel));
1,238,638,181✔
1423
        break;
1,238,643,079✔
1424
      } 
1425
      case QUERY_NODE_VALUE: {
19,571,449✔
1426
        TAOS_CHECK_EXIT(projectApplyValue(&pExpr[k], pResult, pSrcBlock, outputSlotId, &numOfRows, createNewColModel));
19,571,449✔
1427
        break;
19,575,980✔
1428
      } 
1429
      case QUERY_NODE_OPERATOR: {
91,970,400✔
1430
        TAOS_CHECK_EXIT(projectApplyOperator(&pExpr[k], pResult, pSrcBlock, outputSlotId, &numOfRows, createNewColModel, pExtraParams));
91,970,400✔
1431
        break;
90,645,361✔
1432
      } 
1433
      case QUERY_NODE_FUNCTION: {
607,211,592✔
1434
        TAOS_CHECK_EXIT(projectApplyFunction(pCtx, pfCtx, &pExpr[k], pResult, pSrcBlock, outputSlotId, &numOfRows, createNewColModel, pExtraParams, pPseudoList, &processByRowFunctionCtx, doSelectFunc));
607,211,592✔
1435
        break;
607,011,523✔
1436
      }
1437
      default: {
×
1438
        qError("invalid project expr nodeType:%d", pExpr[k].pExpr->nodeType);
×
1439
        TAOS_CHECK_EXIT(TSDB_CODE_OPS_NOT_SUPPORT);
×
1440
      }
1441
    }
1442
  }
1443

1444
  if (processByRowFunctionCtx && taosArrayGetSize(processByRowFunctionCtx) > 0) {
562,893,850✔
1445
    int32_t processByRowSize = taosArrayGetSize(processByRowFunctionCtx);
61,189,433✔
1446
    SStreamRuntimeFuncInfo* pStreamInfo = (SStreamRuntimeFuncInfo*)pExtraParams;
61,266,813✔
1447
    bool splitByExternalWindow = pSrcBlock != NULL && pStreamInfo != NULL && pStreamInfo->withExternalWindow &&
61,266,813✔
1448
                                 pStreamInfo->pStreamBlkWinIdx != NULL &&
×
1449
                                 taosArrayGetSize(pStreamInfo->pStreamBlkWinIdx) > 1 &&
122,533,626✔
1450
                                 allProcessByRowCtxSameFuncId(processByRowFunctionCtx);
×
1451
    pProcessedFuncIds = taosArrayInit(4, sizeof(int32_t));
61,266,813✔
1452
    TSDB_CHECK_NULL(pProcessedFuncIds, code, lino, _exit, terrno);
61,266,813✔
1453

1454
    for (int32_t i = 0; i < processByRowSize; ++i) {
122,615,928✔
1455
      SqlFunctionCtx** ppCurrCtx = taosArrayGet(processByRowFunctionCtx, i);
61,358,376✔
1456
      TSDB_CHECK_NULL(ppCurrCtx, code, lino, _exit, terrno);
61,358,376✔
1457
      TSDB_CHECK_NULL(*ppCurrCtx, code, lino, _exit, terrno);
61,358,376✔
1458

1459
      bool    processed = false;
61,358,376✔
1460
      int32_t processedNum = taosArrayGetSize(pProcessedFuncIds);
61,358,376✔
1461
      for (int32_t j = 0; j < processedNum; ++j) {
61,381,435✔
1462
        int32_t* pFuncId = taosArrayGet(pProcessedFuncIds, j);
91,563✔
1463
        TSDB_CHECK_NULL(pFuncId, code, lino, _exit, terrno);
91,563✔
1464
        if (*pFuncId == (*ppCurrCtx)->functionId) {
91,563✔
1465
          processed = true;
68,504✔
1466
          break;
68,504✔
1467
        }
1468
      }
1469

1470
      if (processed) {
61,358,376✔
1471
        continue;
68,504✔
1472
      }
1473

1474
      pGroupedCtxArray = taosArrayInit(2, sizeof(SqlFunctionCtx*));
61,289,872✔
1475
      TSDB_CHECK_NULL(pGroupedCtxArray, code, lino, _exit, terrno);
61,289,872✔
1476

1477
      for (int32_t j = i; j < processByRowSize; ++j) {
122,672,239✔
1478
        SqlFunctionCtx** ppTmpCtx = taosArrayGet(processByRowFunctionCtx, j);
61,382,367✔
1479
        TSDB_CHECK_NULL(ppTmpCtx, code, lino, _exit, terrno);
61,382,367✔
1480
        TSDB_CHECK_NULL(*ppTmpCtx, code, lino, _exit, terrno);
61,382,367✔
1481

1482
        if ((*ppTmpCtx)->functionId == (*ppCurrCtx)->functionId) {
61,382,367✔
1483
          void* px = taosArrayPush(pGroupedCtxArray, ppTmpCtx);
61,358,865✔
1484
          TSDB_CHECK_NULL(px, code, lino, _exit, terrno);
61,358,865✔
1485
        }
1486
      }
1487

1488
      if (splitByExternalWindow) {
61,289,872✔
1489
        TAOS_CHECK_EXIT(processByRowInExternalWindows(pGroupedCtxArray, pSrcBlock, pStreamInfo));
×
1490
      } else {
1491
        TAOS_CHECK_EXIT((*ppCurrCtx)->fpSet.processFuncByRow(pGroupedCtxArray));
61,289,872✔
1492
      }
1493
      taosArrayDestroy(pGroupedCtxArray);
61,280,611✔
1494
      pGroupedCtxArray = NULL;
61,280,611✔
1495

1496
      void* px = taosArrayPush(pProcessedFuncIds, &(*ppCurrCtx)->functionId);
61,280,611✔
1497
      TSDB_CHECK_NULL(px, code, lino, _exit, terrno);
61,280,611✔
1498

1499
      numOfRows = (*ppCurrCtx)->resultInfo->numOfRes;
61,280,611✔
1500
    }
1501

1502
    taosArrayDestroy(pProcessedFuncIds);
61,257,552✔
1503
    pProcessedFuncIds = NULL;
61,257,552✔
1504
  }
1505

1506
  if (!createNewColModel) {
562,961,969✔
1507
    pResult->info.rows += numOfRows;
406,110,636✔
1508
  }
1509

1510
_exit:
564,373,756✔
1511
  gTaskScalarExtra.pTaskInfo    = savedTaskInfo;
564,365,859✔
1512
  gTaskScalarExtra.isTaskKilled = savedIsTaskKilled;
564,392,592✔
1513

1514
  if (pGroupedCtxArray) {
564,349,784✔
1515
    taosArrayDestroy(pGroupedCtxArray);
9,261✔
1516
  }
1517
  if (pProcessedFuncIds) {
564,349,784✔
1518
    taosArrayDestroy(pProcessedFuncIds);
9,261✔
1519
  }
1520
  if (processByRowFunctionCtx) {
564,349,784✔
1521
    taosArrayDestroy(processByRowFunctionCtx);
61,266,813✔
1522
  }
1523
  if (code) {
564,349,784✔
1524
    qError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
1,381,016✔
1525
  }
1526
  return code;
564,349,784✔
1527
}
1528

1529
int32_t projectApplyFunctions(SExprInfo* pExpr, SSDataBlock* pResult, SSDataBlock* pSrcBlock, SqlFunctionCtx* pCtx,
553,132,145✔
1530
                              int32_t numOfOutput, SArray* pPseudoList, const void* pExtraParams,
1531
                              SExecTaskInfo* pTaskInfo) {
1532
  return projectApplyFunctionsWithSelect(pExpr, pResult, pSrcBlock, pCtx, numOfOutput, pPseudoList, pExtraParams,
553,132,145✔
1533
                                         false, true, pTaskInfo);
1534
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc