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

taosdata / TDengine / #5040

29 Apr 2026 11:44AM UTC coverage: 73.13% (-0.02%) from 73.153%
#5040

push

travis-ci

web-flow
feat(statewindow): support multi columns (#35136)

1567 of 1828 new or added lines in 18 files covered. (85.72%)

7264 existing lines in 119 files now uncovered.

277409 of 379338 relevant lines covered (73.13%)

134373925.48 hits per line

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

79.18
/source/dnode/vnode/src/meta/metaTable.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 "meta.h"
17

18
extern SDmNotifyHandle dmNotifyHdl;
19

20
int32_t metaAddTableColumn(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
21
int32_t metaDropTableColumn(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
22
int32_t metaAlterTableColumnName(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
23
int32_t metaAlterTableColumnBytes(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
24
int32_t metaUpdateTableMultiTableTagValue(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq);
25
int32_t metaUpdateTableChildTableTagValue(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq);
26
int32_t metaUpdateTableOptions2(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq);
27
int32_t metaUpdateTableColCompress2(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq);
28
int32_t metaAlterTableColumnRef(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
29
int32_t metaRemoveTableColumnRef(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pRsp);
30
int32_t metaSaveJsonVarToIdx(SMeta *pMeta, const SMetaEntry *pCtbEntry, const SSchema *pSchema);
31

32
int32_t    metaDelJsonVarFromIdx(SMeta *pMeta, const SMetaEntry *pCtbEntry, const SSchema *pSchema);
33
static int metaUpdateChangeTime(SMeta *pMeta, tb_uid_t uid, int64_t changeTimeMs);
34
static int metaDropTableByUid(SMeta *pMeta, tb_uid_t uid, int *type, tb_uid_t *pSuid, int8_t *pSysTbl);
35
void       metaDestroyTagIdxKey(STagIdxKey *pTagIdxKey);
36
// opt ins_tables query
37
static int metaDeleteBtimeIdx(SMeta *pMeta, const SMetaEntry *pME);
38
static int metaDeleteNcolIdx(SMeta *pMeta, const SMetaEntry *pME);
39

40
int32_t updataTableColCmpr(SColCmprWrapper *pWp, SSchema *pSchema, int8_t add, uint32_t compress) {
3,497,399✔
41
  int32_t nCols = pWp->nCols;
3,497,399✔
42
  int32_t ver = pWp->version;
3,497,399✔
43
  if (add) {
3,497,399✔
44
    SColCmpr *p = taosMemoryRealloc(pWp->pColCmpr, sizeof(SColCmpr) * (nCols + 1));
3,431,248✔
45
    if (p == NULL) {
3,431,248✔
46
      return terrno;
×
47
    }
48
    pWp->pColCmpr = p;
3,431,248✔
49

50
    SColCmpr *pCol = p + nCols;
3,431,248✔
51
    pCol->id = pSchema->colId;
3,431,248✔
52
    pCol->alg = compress;
3,431,248✔
53
    pWp->nCols = nCols + 1;
3,431,248✔
54
    pWp->version = ver;
3,431,248✔
55
  } else {
56
    for (int32_t i = 0; i < nCols; i++) {
15,309,399✔
57
      SColCmpr *pOCmpr = &pWp->pColCmpr[i];
15,309,399✔
58
      if (pOCmpr->id == pSchema->colId) {
15,309,399✔
59
        int32_t left = (nCols - i - 1) * sizeof(SColCmpr);
66,151✔
60
        if (left) {
66,151✔
61
          memmove(pWp->pColCmpr + i, pWp->pColCmpr + i + 1, left);
43,976✔
62
        }
63
        nCols--;
66,151✔
64
        break;
66,151✔
65
      }
66
    }
67
    pWp->nCols = nCols;
66,151✔
68
    pWp->version = ver;
66,151✔
69
  }
70
  return 0;
3,497,399✔
71
}
72

73
int32_t addTableExtSchema(SMetaEntry *pEntry, const SSchema *pColumn, int32_t newColNum, SExtSchema *pExtSchema) {
3,491,886✔
74
  // no need to add ext schema when no column needs ext schemas
75
  if (!HAS_TYPE_MOD(pColumn) && !pEntry->pExtSchemas) return 0;
3,491,886✔
76
  if (!pEntry->pExtSchemas) {
15,375✔
77
    // add a column which needs ext schema
78
    // set all extschemas to zero for all columns alrady existed
79
    pEntry->pExtSchemas = (SExtSchema *)taosMemoryCalloc(newColNum, sizeof(SExtSchema));
2,382✔
80
  } else {
81
    // already has columns with ext schema
82
    pEntry->pExtSchemas = (SExtSchema *)taosMemoryRealloc(pEntry->pExtSchemas, sizeof(SExtSchema) * newColNum);
12,993✔
83
  }
84
  if (!pEntry->pExtSchemas) return terrno;
15,375✔
85
  pEntry->pExtSchemas[newColNum - 1] = *pExtSchema;
15,375✔
86
  return 0;
15,375✔
87
}
88

89
int32_t dropTableExtSchema(SMetaEntry *pEntry, int32_t dropColId, int32_t newColNum) {
99,290✔
90
  // no ext schema, no need to drop
91
  if (!pEntry->pExtSchemas) return 0;
99,290✔
92
  if (dropColId == newColNum) {
10,918✔
93
    // drop the last column
94
    pEntry->pExtSchemas[dropColId - 1] = (SExtSchema){0};
3,952✔
95
  } else {
96
    // drop a column in the middle
97
    memmove(pEntry->pExtSchemas + dropColId, pEntry->pExtSchemas + dropColId + 1,
6,966✔
98
            (newColNum - dropColId) * sizeof(SExtSchema));
6,966✔
99
  }
100
  for (int32_t i = 0; i < newColNum; i++) {
40,892✔
101
    if (hasExtSchema(pEntry->pExtSchemas + i)) return 0;
39,304✔
102
  }
103
  taosMemoryFreeClear(pEntry->pExtSchemas);
1,588✔
104
  return 0;
1,588✔
105
}
106

107
int32_t updataTableColRef(SColRefWrapper *pWp, const SSchema *pSchema, int8_t add, SColRef *pColRef) {
105,274✔
108
  int32_t nCols = pWp->nCols;
105,274✔
109
  if (add) {
105,274✔
110
    SColRef *p = taosMemoryRealloc(pWp->pColRef, sizeof(SColRef) * (nCols + 1));
67,366✔
111
    if (p == NULL) {
67,366✔
112
      return terrno;
×
113
    }
114
    pWp->pColRef = p;
67,366✔
115

116
    SColRef *pCol = p + nCols;
67,366✔
117
    if (NULL == pColRef) {
67,366✔
118
      pCol->hasRef = false;
6,728✔
119
      pCol->id = pSchema->colId;
6,728✔
120
    } else {
121
      pCol->hasRef = pColRef->hasRef;
60,638✔
122
      pCol->id = pSchema->colId;
60,638✔
123
      if (pCol->hasRef) {
60,638✔
124
        tstrncpy(pCol->refDbName, pColRef->refDbName, TSDB_DB_NAME_LEN);
25,703✔
125
        tstrncpy(pCol->refTableName, pColRef->refTableName, TSDB_TABLE_NAME_LEN);
25,703✔
126
        tstrncpy(pCol->refColName, pColRef->refColName, TSDB_COL_NAME_LEN);
25,703✔
127
      }
128
    }
129
    pWp->nCols = nCols + 1;
67,366✔
130
    pWp->version++;
67,366✔
131
  } else {
132
    for (int32_t i = 0; i < nCols; i++) {
39,589,918✔
133
      SColRef *pOColRef = &pWp->pColRef[i];
39,589,918✔
134
      if (pOColRef->id == pSchema->colId) {
39,589,918✔
135
        int32_t left = (nCols - i - 1) * sizeof(SColRef);
37,908✔
136
        if (left) {
37,908✔
137
          memmove(pWp->pColRef + i, pWp->pColRef + i + 1, left);
34,538✔
138
        }
139
        nCols--;
37,908✔
140
        break;
37,908✔
141
      }
142
    }
143
    pWp->nCols = nCols;
37,908✔
144
    pWp->version++;
37,908✔
145
  }
146
  return 0;
105,274✔
147
}
148

149
int metaUpdateMetaRsp(tb_uid_t uid, char *tbName, SSchemaWrapper *pSchema, int64_t ownerId, STableMetaRsp *pMetaRsp) {
11,796,149✔
150
  pMetaRsp->pSchemas = taosMemoryMalloc(pSchema->nCols * sizeof(SSchema));
11,796,149✔
151
  if (NULL == pMetaRsp->pSchemas) {
11,796,149✔
152
    return terrno;
×
153
  }
154

155
  pMetaRsp->pSchemaExt = taosMemoryCalloc(1, pSchema->nCols * sizeof(SSchemaExt));
11,796,149✔
156
  if (pMetaRsp->pSchemaExt == NULL) {
11,796,149✔
157
    taosMemoryFree(pMetaRsp->pSchemas);
×
158
    return terrno;
×
159
  }
160

161
  tstrncpy(pMetaRsp->tbName, tbName, TSDB_TABLE_NAME_LEN);
11,796,149✔
162
  pMetaRsp->numOfColumns = pSchema->nCols;
11,796,149✔
163
  pMetaRsp->tableType = TSDB_NORMAL_TABLE;
11,796,149✔
164
  pMetaRsp->sversion = pSchema->version;
11,796,149✔
165
  pMetaRsp->rversion = 1;
11,796,149✔
166
  pMetaRsp->tuid = uid;
11,796,149✔
167
  pMetaRsp->virtualStb = false; // super table will never be processed here
11,796,149✔
168
  if (ownerId != 0) pMetaRsp->ownerId = ownerId;
11,796,149✔
169

170
  memcpy(pMetaRsp->pSchemas, pSchema->pSchema, pSchema->nCols * sizeof(SSchema));
11,796,149✔
171

172
  return 0;
11,796,149✔
173
}
174

175
static int32_t metaFillRspSchemaExt(const SSchemaWrapper *pSchema, const SExtSchema *pExtSchemas, SSchemaExt *pSchemaExt) {
871,994✔
176
  if (pSchema == NULL || pSchemaExt == NULL) {
871,994✔
177
    return TSDB_CODE_SUCCESS;
×
178
  }
179

180
  for (int32_t i = 0; i < pSchema->nCols; ++i) {
244,990,120✔
181
    pSchemaExt[i].colId = pSchema->pSchema[i].colId;
244,118,126✔
182
  }
183

184
  if (pExtSchemas != NULL) {
871,994✔
185
    for (int32_t i = 0; i < pSchema->nCols; ++i) {
161,712✔
186
      pSchemaExt[i].typeMod = pExtSchemas[i].typeMod;
144,621✔
187
    }
188
  }
189

190
  return TSDB_CODE_SUCCESS;
871,994✔
191
}
192

193
int32_t metaUpdateVtbMetaRsp(SMetaEntry *pEntry, char *tbName, const SSchemaWrapper *pSchema, const SColRefWrapper *pRef,
871,994✔
194
                             const SExtSchema *pExtSchemas, int64_t ownerId, STableMetaRsp *pMetaRsp, int8_t tableType) {
195
  int32_t code = TSDB_CODE_SUCCESS;
871,994✔
196
  if (!pRef) {
871,994✔
197
    return TSDB_CODE_INVALID_PARA;
×
198
  }
199
  if (pSchema) {
871,994✔
200
    pMetaRsp->pSchemas = taosMemoryMalloc(pSchema->nCols * sizeof(SSchema));
871,994✔
201
    if (NULL == pMetaRsp->pSchemas) {
871,994✔
202
      code = terrno;
×
203
      goto _return;
×
204
    }
205

206
    pMetaRsp->pSchemaExt = taosMemoryCalloc(pSchema->nCols, sizeof(SSchemaExt));
871,994✔
207
    if (pMetaRsp->pSchemaExt == NULL) {
871,994✔
208
      code = terrno;
×
209
      goto _return;
×
210
    }
211

212
    pMetaRsp->numOfColumns = pSchema->nCols;
871,994✔
213
    pMetaRsp->sversion = pSchema->version;
871,994✔
214
    memcpy(pMetaRsp->pSchemas, pSchema->pSchema, pSchema->nCols * sizeof(SSchema));
871,994✔
215

216
    code = metaFillRspSchemaExt(pSchema, pExtSchemas, pMetaRsp->pSchemaExt);
871,994✔
217
    if (code != TSDB_CODE_SUCCESS) {
871,994✔
218
      goto _return;
×
219
    }
220
  }
221

222
  if (pRef->nCols > 0) {
871,994✔
223
    pMetaRsp->pColRefs = taosMemoryMalloc(pRef->nCols * sizeof(SColRef));
871,994✔
224
    if (NULL == pMetaRsp->pColRefs) {
871,994✔
225
      code = terrno;
×
226
      goto _return;
×
227
    }
228

229
    memcpy(pMetaRsp->pColRefs, pRef->pColRef, pRef->nCols * sizeof(SColRef));
871,994✔
230
  } else {
231
    pMetaRsp->pColRefs = NULL;
×
232
  }
233

234
  tstrncpy(pMetaRsp->tbName, tbName, TSDB_TABLE_NAME_LEN);
871,994✔
235
  if (tableType == TSDB_VIRTUAL_NORMAL_TABLE) {
871,994✔
236
    pMetaRsp->tuid = pEntry->uid;
444,920✔
237
  } else if (tableType == TSDB_VIRTUAL_CHILD_TABLE) {
427,074✔
238
    pMetaRsp->tuid = pEntry->uid;
427,074✔
239
    pMetaRsp->suid = pEntry->ctbEntry.suid;
427,074✔
240
  }
241

242
  pMetaRsp->tableType = tableType;
871,994✔
243
  pMetaRsp->virtualStb = false; // super table will never be processed here
871,994✔
244
  pMetaRsp->numOfColRefs = pRef->nCols;
871,994✔
245
  pMetaRsp->rversion = pRef->version;
871,994✔
246
  if (ownerId != 0) pMetaRsp->ownerId = ownerId;
871,994✔
247

248
  // Populate tag references
249
  if (pRef->nTagRefs > 0 && pRef->pTagRef) {
871,994✔
250
    pMetaRsp->pTagRefs = taosMemoryMalloc(pRef->nTagRefs * sizeof(SColRef));
×
251
    if (NULL == pMetaRsp->pTagRefs) {
×
252
      code = terrno;
×
253
      goto _return;
×
254
    }
255
    memcpy(pMetaRsp->pTagRefs, pRef->pTagRef, pRef->nTagRefs * sizeof(SColRef));
×
256
    pMetaRsp->numOfTagRefs = pRef->nTagRefs;
×
257
  } else {
258
    pMetaRsp->pTagRefs = NULL;
871,994✔
259
    pMetaRsp->numOfTagRefs = 0;
871,994✔
260
  }
261

262
  return code;
871,994✔
263
_return:
×
264
  taosMemoryFreeClear(pMetaRsp->pSchemaExt);
×
265
  taosMemoryFreeClear(pMetaRsp->pSchemas);
×
266
  taosMemoryFreeClear(pMetaRsp->pColRefs);
×
267
  taosMemoryFreeClear(pMetaRsp->pTagRefs);
×
268
  return code;
×
269
}
270

271
int metaSaveJsonVarToIdx(SMeta *pMeta, const SMetaEntry *pCtbEntry, const SSchema *pSchema) {
288,901✔
272
  int32_t code = 0;
288,901✔
273

274
#ifdef USE_INVERTED_INDEX
275
  if (pMeta->pTagIvtIdx == NULL || pCtbEntry == NULL) {
288,901✔
276
    return TSDB_CODE_INVALID_PARA;
×
277
  }
278
  void       *data = pCtbEntry->ctbEntry.pTags;
288,901✔
279
  const char *tagName = pSchema->name;
288,901✔
280

281
  tb_uid_t    suid = pCtbEntry->ctbEntry.suid;
288,901✔
282
  tb_uid_t    tuid = pCtbEntry->uid;
288,901✔
283
  const void *pTagData = pCtbEntry->ctbEntry.pTags;
288,901✔
284
  int32_t     nTagData = 0;
288,901✔
285

286
  SArray *pTagVals = NULL;
288,901✔
287
  code = tTagToValArray((const STag *)data, &pTagVals);
288,901✔
288
  if (code) {
288,901✔
289
    return code;
×
290
  }
291

292
  SIndexMultiTerm *terms = indexMultiTermCreate();
288,901✔
293
  if (terms == NULL) {
288,901✔
294
    return terrno;
×
295
  }
296

297
  int16_t nCols = taosArrayGetSize(pTagVals);
288,901✔
298
  for (int i = 0; i < nCols; i++) {
699,934✔
299
    STagVal *pTagVal = (STagVal *)taosArrayGet(pTagVals, i);
411,033✔
300
    char     type = pTagVal->type;
411,033✔
301

302
    char   *key = pTagVal->pKey;
411,033✔
303
    int32_t nKey = strlen(key);
411,033✔
304

305
    SIndexTerm *term = NULL;
411,033✔
306
    if (type == TSDB_DATA_TYPE_NULL) {
411,033✔
307
      term = indexTermCreate(suid, ADD_VALUE, TSDB_DATA_TYPE_VARCHAR, key, nKey, NULL, 0);
3,060✔
308
    } else if (type == TSDB_DATA_TYPE_NCHAR) {
407,973✔
309
      if (pTagVal->nData > 0) {
383,929✔
310
        char *val = taosMemoryCalloc(1, pTagVal->nData + VARSTR_HEADER_SIZE);
381,889✔
311
        if (val == NULL) {
381,889✔
312
          TAOS_CHECK_GOTO(terrno, NULL, _exception);
×
313
        }
314
        int32_t len = taosUcs4ToMbs((TdUcs4 *)pTagVal->pData, pTagVal->nData, val + VARSTR_HEADER_SIZE, NULL);
381,889✔
315
        if (len < 0) {
381,889✔
316
          TAOS_CHECK_GOTO(len, NULL, _exception);
×
317
        }
318
        memcpy(val, (uint16_t *)&len, VARSTR_HEADER_SIZE);
381,889✔
319
        type = TSDB_DATA_TYPE_VARCHAR;
381,889✔
320
        term = indexTermCreate(suid, ADD_VALUE, type, key, nKey, val, len);
381,889✔
321
        taosMemoryFree(val);
381,889✔
322
      } else if (pTagVal->nData == 0) {
2,040✔
323
        term = indexTermCreate(suid, ADD_VALUE, TSDB_DATA_TYPE_VARCHAR, key, nKey, pTagVal->pData, 0);
2,040✔
324
      }
325
    } else if (type == TSDB_DATA_TYPE_DOUBLE) {
24,044✔
326
      double val = *(double *)(&pTagVal->i64);
19,964✔
327
      int    len = sizeof(val);
19,964✔
328
      term = indexTermCreate(suid, ADD_VALUE, type, key, nKey, (const char *)&val, len);
19,964✔
329
    } else if (type == TSDB_DATA_TYPE_BOOL) {
4,080✔
330
      int val = *(int *)(&pTagVal->i64);
4,080✔
331
      int len = sizeof(val);
4,080✔
332
      term = indexTermCreate(suid, ADD_VALUE, TSDB_DATA_TYPE_BOOL, key, nKey, (const char *)&val, len);
4,080✔
333
    }
334

335
    if (term != NULL) {
411,033✔
336
      int32_t ret = indexMultiTermAdd(terms, term);
411,033✔
337
      if (ret < 0) {
411,033✔
338
        metaError("vgId:%d, failed to add term to multi term, uid: %" PRId64 ", key: %s, type: %d, ret: %d",
×
339
                  TD_VID(pMeta->pVnode), tuid, key, type, ret);
340
      }
341
    } else {
342
      code = terrno;
×
343
      goto _exception;
×
344
    }
345
  }
346
  code = indexJsonPut(pMeta->pTagIvtIdx, terms, tuid);
288,901✔
347
  indexMultiTermDestroy(terms);
288,901✔
348

349
  taosArrayDestroy(pTagVals);
288,901✔
350
  return code;
288,901✔
351
_exception:
×
352
  indexMultiTermDestroy(terms);
×
353
  taosArrayDestroy(pTagVals);
×
354
#endif
355
  return code;
×
356
}
357
int metaDelJsonVarFromIdx(SMeta *pMeta, const SMetaEntry *pCtbEntry, const SSchema *pSchema) {
123,964✔
358
int32_t code = 0;
123,964✔
359
#ifdef USE_INVERTED_INDEX
360
  if (pMeta->pTagIvtIdx == NULL || pCtbEntry == NULL) {
123,964✔
361
    return TSDB_CODE_INVALID_PARA;
×
362
  }
363
  void       *data = pCtbEntry->ctbEntry.pTags;
123,964✔
364
  const char *tagName = pSchema->name;
123,964✔
365

366
  tb_uid_t    suid = pCtbEntry->ctbEntry.suid;
123,964✔
367
  tb_uid_t    tuid = pCtbEntry->uid;
123,964✔
368
  const void *pTagData = pCtbEntry->ctbEntry.pTags;
123,964✔
369
  int32_t     nTagData = 0;
123,964✔
370

371
  SArray *pTagVals = NULL;
123,964✔
372
  code = tTagToValArray((const STag *)data, &pTagVals);
123,964✔
373
  if (code) {
123,964✔
374
    return code;
×
375
  }
376

377
  SIndexMultiTerm *terms = indexMultiTermCreate();
123,964✔
378
  if (terms == NULL) {
123,964✔
379
    return terrno;
×
380
  }
381

382
  int16_t nCols = taosArrayGetSize(pTagVals);
123,964✔
383
  for (int i = 0; i < nCols; i++) {
355,429✔
384
    STagVal *pTagVal = (STagVal *)taosArrayGet(pTagVals, i);
231,465✔
385
    char     type = pTagVal->type;
231,465✔
386

387
    char   *key = pTagVal->pKey;
231,465✔
388
    int32_t nKey = strlen(key);
231,465✔
389

390
    SIndexTerm *term = NULL;
231,465✔
391
    if (type == TSDB_DATA_TYPE_NULL) {
231,465✔
392
      term = indexTermCreate(suid, DEL_VALUE, TSDB_DATA_TYPE_VARCHAR, key, nKey, NULL, 0);
×
393
    } else if (type == TSDB_DATA_TYPE_NCHAR) {
231,465✔
394
      if (pTagVal->nData > 0) {
228,915✔
395
        char *val = taosMemoryCalloc(1, pTagVal->nData + VARSTR_HEADER_SIZE);
228,915✔
396
        if (val == NULL) {
228,915✔
397
          TAOS_CHECK_GOTO(terrno, NULL, _exception);
×
398
        }
399
        int32_t len = taosUcs4ToMbs((TdUcs4 *)pTagVal->pData, pTagVal->nData, val + VARSTR_HEADER_SIZE, NULL);
228,915✔
400
        if (len < 0) {
228,915✔
401
          TAOS_CHECK_GOTO(len, NULL, _exception);
×
402
        }
403
        memcpy(val, (uint16_t *)&len, VARSTR_HEADER_SIZE);
228,915✔
404
        type = TSDB_DATA_TYPE_VARCHAR;
228,915✔
405
        term = indexTermCreate(suid, DEL_VALUE, type, key, nKey, val, len);
228,915✔
406
        taosMemoryFree(val);
228,915✔
407
      } else if (pTagVal->nData == 0) {
×
408
        term = indexTermCreate(suid, DEL_VALUE, TSDB_DATA_TYPE_VARCHAR, key, nKey, pTagVal->pData, 0);
×
409
      }
410
    } else if (type == TSDB_DATA_TYPE_DOUBLE) {
2,550✔
411
      double val = *(double *)(&pTagVal->i64);
1,530✔
412
      int    len = sizeof(val);
1,530✔
413
      term = indexTermCreate(suid, DEL_VALUE, type, key, nKey, (const char *)&val, len);
1,530✔
414
    } else if (type == TSDB_DATA_TYPE_BOOL) {
1,020✔
415
      int val = *(int *)(&pTagVal->i64);
1,020✔
416
      int len = sizeof(val);
1,020✔
417
      term = indexTermCreate(suid, DEL_VALUE, TSDB_DATA_TYPE_BOOL, key, nKey, (const char *)&val, len);
1,020✔
418
    }
419
    if (term != NULL) {
231,465✔
420
      int32_t ret = indexMultiTermAdd(terms, term);
231,465✔
421
      if (ret < 0) {
231,465✔
422
        metaError("vgId:%d, failed to add term to multi term, uid: %" PRId64 ", key: %s, type: %d, ret: %d",
×
423
                  TD_VID(pMeta->pVnode), tuid, key, type, ret);
424
      }
425
    } else {
426
      code = terrno;
×
427
      goto _exception;
×
428
    }
429
  }
430
  code = indexJsonPut(pMeta->pTagIvtIdx, terms, tuid);
123,964✔
431
  indexMultiTermDestroy(terms);
123,964✔
432
  taosArrayDestroy(pTagVals);
123,964✔
433
  return code;
123,964✔
434
_exception:
×
435
  indexMultiTermDestroy(terms);
×
436
  taosArrayDestroy(pTagVals);
×
437
#endif
438
  return code;
×
439
}
440

441
static int32_t metaDropTables(SMeta *pMeta, SArray *tbUids) {
20,732✔
442
  int32_t code = 0;
20,732✔
443
  if (taosArrayGetSize(tbUids) == 0) return TSDB_CODE_SUCCESS;
20,732✔
444

445
  int64_t    nCtbDropped = 0;
20,732✔
446
  SSHashObj *suidHash = tSimpleHashInit(16, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BIGINT));
20,732✔
447
  if (suidHash == NULL) {
20,732✔
448
    return terrno;
×
449
  }
450

451
  metaWLock(pMeta);
20,732✔
452
  for (int i = 0; i < taosArrayGetSize(tbUids); ++i) {
106,008✔
453
    tb_uid_t uid = *(tb_uid_t *)taosArrayGet(tbUids, i);
85,276✔
454
    tb_uid_t suid = 0;
85,276✔
455
    int8_t   sysTbl = 0;
85,276✔
456
    int      type;
85,276✔
457
    code = metaDropTableByUid(pMeta, uid, &type, &suid, &sysTbl);
85,276✔
458
    if (code) return code;
85,276✔
459
    if (!sysTbl && type == TSDB_CHILD_TABLE && suid != 0 && suidHash) {
85,276✔
460
      int64_t *pVal = tSimpleHashGet(suidHash, &suid, sizeof(tb_uid_t));
79,179✔
461
      if (pVal) {
79,179✔
462
        nCtbDropped = *pVal + 1;
61,654✔
463
      } else {
464
        nCtbDropped = 1;
17,525✔
465
      }
466
      code = tSimpleHashPut(suidHash, &suid, sizeof(tb_uid_t), &nCtbDropped, sizeof(int64_t));
79,179✔
467
      if (code) return code;
79,179✔
468
    }
469
    /*
470
    if (!TSDB_CACHE_NO(pMeta->pVnode->config)) {
471
      tsdbCacheDropTable(pMeta->pVnode->pTsdb, uid, suid, NULL);
472
    }
473
    */
474
    metaDebug("batch drop table:%" PRId64, uid);
85,276✔
475
  }
476
  metaULock(pMeta);
20,732✔
477

478
  // update timeseries
479
  void   *pCtbDropped = NULL;
20,732✔
480
  int32_t iter = 0;
20,732✔
481
  while ((pCtbDropped = tSimpleHashIterate(suidHash, pCtbDropped, &iter))) {
38,257✔
482
    tb_uid_t    *pSuid = tSimpleHashGetKey(pCtbDropped, NULL);
17,525✔
483
    int32_t      nCols = 0;
17,525✔
484
    int8_t       flags = 0;
17,525✔
485
    SVnodeStats *pStats = &pMeta->pVnode->config.vndStats;
17,525✔
486
    if (metaGetStbStats(pMeta->pVnode, *pSuid, NULL, &nCols, &flags) == 0) {
17,525✔
487
      if (!TABLE_IS_VIRTUAL(flags)) {
17,525✔
488
        pStats->numOfTimeSeries -= *(int64_t *)pCtbDropped * (nCols - 1);
17,525✔
489
      }
490
    }
491
  }
492
  tSimpleHashCleanup(suidHash);
20,732✔
493

494
  pMeta->changed = true;
20,732✔
495
  return 0;
20,732✔
496
}
497

498
static int32_t metaFilterTableByHash(SMeta *pMeta, SArray *uidList) {
26,127✔
499
  int32_t code = 0;
26,127✔
500
  // 1, tranverse table's
501
  // 2, validate table name using vnodeValidateTableHash
502
  // 3, push invalidated table's uid into uidList
503

504
  TBC *pCur;
26,127✔
505
  code = tdbTbcOpen(pMeta->pTbDb, &pCur, NULL);
26,127✔
506
  if (code < 0) {
26,127✔
507
    return code;
×
508
  }
509

510
  code = tdbTbcMoveToFirst(pCur);
26,127✔
511
  if (code) {
26,127✔
512
    tdbTbcClose(pCur);
×
513
    return code;
×
514
  }
515

516
  void *pData = NULL, *pKey = NULL;
26,127✔
517
  int   nData = 0, nKey = 0;
26,127✔
518

519
  while (1) {
190,424✔
520
    int32_t ret = tdbTbcNext(pCur, &pKey, &nKey, &pData, &nData);
216,551✔
521
    if (ret < 0) {
216,551✔
522
      break;
26,127✔
523
    }
524

525
    SMetaEntry me = {0};
190,424✔
526
    SDecoder   dc = {0};
190,424✔
527
    tDecoderInit(&dc, pData, nData);
190,424✔
528
    code = metaDecodeEntry(&dc, &me);
190,424✔
529
    if (code < 0) {
190,424✔
530
      tDecoderClear(&dc);
×
531
      return code;
×
532
    }
533

534
    if (me.type != TSDB_SUPER_TABLE) {
190,424✔
535
      char tbFName[TSDB_TABLE_FNAME_LEN + 1];
170,536✔
536
      snprintf(tbFName, sizeof(tbFName), "%s.%s", pMeta->pVnode->config.dbname, me.name);
170,536✔
537
      tbFName[TSDB_TABLE_FNAME_LEN] = '\0';
170,536✔
538
      if (pMeta->pVnode->mounted) tTrimMountPrefix(tbFName);
170,536✔
539
      ret = vnodeValidateTableHash(pMeta->pVnode, tbFName);
170,536✔
540
      if (ret < 0 && terrno == TSDB_CODE_VND_HASH_MISMATCH) {
170,536✔
541
        if (taosArrayPush(uidList, &me.uid) == NULL) {
85,276✔
542
          code = terrno;
×
543
          break;
×
544
        }
545
      }
546
    }
547
    tDecoderClear(&dc);
190,424✔
548
  }
549
  tdbFree(pData);
26,127✔
550
  tdbFree(pKey);
26,127✔
551
  tdbTbcClose(pCur);
26,127✔
552

553
  return 0;
26,127✔
554
}
555

556
int32_t metaTrimTables(SMeta *pMeta, int64_t version) {
26,127✔
557
  int32_t code = 0;
26,127✔
558

559
  SArray *tbUids = taosArrayInit(8, sizeof(int64_t));
26,127✔
560
  if (tbUids == NULL) {
26,127✔
561
    return terrno;
×
562
  }
563

564
  code = metaFilterTableByHash(pMeta, tbUids);
26,127✔
565
  if (code != 0) {
26,127✔
566
    goto end;
×
567
  }
568
  if (TARRAY_SIZE(tbUids) == 0) {
26,127✔
569
    goto end;
5,395✔
570
  }
571

572
  metaInfo("vgId:%d, trim %ld tables", TD_VID(pMeta->pVnode), taosArrayGetSize(tbUids));
20,732✔
573
  code = metaDropTables(pMeta, tbUids);
20,732✔
574
  if (code) goto end;
20,732✔
575

576
end:
20,732✔
577
  taosArrayDestroy(tbUids);
26,127✔
578

579
  return code;
26,127✔
580
}
581

582
int metaTtlFindExpired(SMeta *pMeta, int64_t timePointMs, SArray *tbUids, int32_t ttlDropMaxCount) {
13,614,051✔
583
  metaRLock(pMeta);
13,614,051✔
584

585
  int ret = ttlMgrFindExpired(pMeta->pTtlMgr, timePointMs, tbUids, ttlDropMaxCount);
13,625,888✔
586

587
  metaULock(pMeta);
13,608,228✔
588

589
  if (ret != 0) {
13,608,913✔
590
    metaError("ttl failed to find expired table, ret:%d", ret);
×
591
  }
592

593
  return ret;
13,610,504✔
594
}
595

596
static int metaBuildBtimeIdxKey(SBtimeIdxKey *btimeKey, const SMetaEntry *pME) {
85,276✔
597
  int64_t btime;
598
  if (pME->type == TSDB_CHILD_TABLE) {
85,276✔
599
    btime = pME->ctbEntry.btime;
79,179✔
600
  } else if (pME->type == TSDB_NORMAL_TABLE) {
6,097✔
601
    btime = pME->ntbEntry.btime;
6,097✔
602
  } else {
603
    return TSDB_CODE_FAILED;
×
604
  }
605

606
  btimeKey->btime = btime;
85,276✔
607
  btimeKey->uid = pME->uid;
85,276✔
608
  return 0;
85,276✔
609
}
610

611
static int metaBuildNColIdxKey(SNcolIdxKey *ncolKey, const SMetaEntry *pME) {
6,097✔
612
  if (pME->type == TSDB_NORMAL_TABLE) {
6,097✔
613
    ncolKey->ncol = pME->ntbEntry.schemaRow.nCols;
6,097✔
614
    ncolKey->uid = pME->uid;
6,097✔
615
  } else {
616
    return TSDB_CODE_FAILED;
×
617
  }
618
  return 0;
6,097✔
619
}
620

621
static void metaDeleteTtl(SMeta *pMeta, const SMetaEntry *pME) {
85,276✔
622
  if (pME->type != TSDB_CHILD_TABLE && pME->type != TSDB_NORMAL_TABLE) return;
85,276✔
623

624
  STtlDelTtlCtx ctx = {.uid = pME->uid, .pTxn = pMeta->txn};
85,276✔
625
  if (pME->type == TSDB_CHILD_TABLE) {
85,276✔
626
    ctx.ttlDays = pME->ctbEntry.ttlDays;
79,179✔
627
  } else {
628
    ctx.ttlDays = pME->ntbEntry.ttlDays;
6,097✔
629
  }
630

631
  int32_t ret = ttlMgrDeleteTtl(pMeta->pTtlMgr, &ctx);
85,276✔
632
  if (ret < 0) {
85,276✔
633
    metaError("vgId:%d, failed to delete ttl for table:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), pME->name,
×
634
              pME->uid, tstrerror(ret));
635
  }
636
  return;
85,276✔
637
}
638

639
static int metaDropTableByUid(SMeta *pMeta, tb_uid_t uid, int *type, tb_uid_t *pSuid, int8_t *pSysTbl) {
85,276✔
640
  void      *pData = NULL;
85,276✔
641
  int        nData = 0;
85,276✔
642
  int        rc = 0;
85,276✔
643
  SMetaEntry e = {0};
85,276✔
644
  SDecoder   dc = {0};
85,276✔
645
  int32_t    ret = 0;
85,276✔
646

647
  rc = tdbTbGet(pMeta->pUidIdx, &uid, sizeof(uid), &pData, &nData);
85,276✔
648
  if (rc < 0) {
85,276✔
649
    return rc;
×
650
  }
651
  int64_t version = ((SUidIdxVal *)pData)[0].version;
85,276✔
652

653
  rc = tdbTbGet(pMeta->pTbDb, &(STbDbKey){.version = version, .uid = uid}, sizeof(STbDbKey), &pData, &nData);
85,276✔
654
  if (rc < 0) {
85,276✔
655
    tdbFree(pData);
×
656
    return rc;
×
657
  }
658

659
  tDecoderInit(&dc, pData, nData);
85,276✔
660
  rc = metaDecodeEntry(&dc, &e);
85,276✔
661
  if (rc < 0) {
85,276✔
662
    tDecoderClear(&dc);
×
663
    return rc;
×
664
  }
665

666
  if (type) *type = e.type;
85,276✔
667

668
  if (e.type == TSDB_CHILD_TABLE) {
85,276✔
669
    if (pSuid) *pSuid = e.ctbEntry.suid;
79,179✔
670
    void *tData = NULL;
79,179✔
671
    int   tLen = 0;
79,179✔
672

673
    if (tdbTbGet(pMeta->pUidIdx, &e.ctbEntry.suid, sizeof(tb_uid_t), &tData, &tLen) == 0) {
79,179✔
674
      STbDbKey tbDbKey = {.uid = e.ctbEntry.suid, .version = ((SUidIdxVal *)tData)[0].version};
79,179✔
675
      if (tdbTbGet(pMeta->pTbDb, &tbDbKey, sizeof(tbDbKey), &tData, &tLen) == 0) {
79,179✔
676
        SDecoder   tdc = {0};
79,179✔
677
        SMetaEntry stbEntry = {0};
79,179✔
678

679
        tDecoderInit(&tdc, tData, tLen);
79,179✔
680
        ret = metaDecodeEntry(&tdc, &stbEntry);
79,179✔
681
        if (ret < 0) {
79,179✔
682
          tDecoderClear(&tdc);
×
683
          metaError("vgId:%d, failed to decode child table:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
×
684
                    e.ctbEntry.suid, tstrerror(ret));
685
          return ret;
×
686
        }
687

688
        if (pSysTbl) *pSysTbl = metaTbInFilterCache(pMeta, stbEntry.name, 1) ? 1 : 0;
79,179✔
689
        
690
        ret = metaStableTagFilterCacheUpdateUid(
79,179✔
691
          pMeta, &e, &stbEntry, STABLE_TAG_FILTER_CACHE_DROP_TABLE);
692
        if (ret < 0) {
79,179✔
693
          metaError("vgId:%d, failed to update stable tag filter cache:%s "
×
694
            "uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
695
            e.ctbEntry.suid, tstrerror(ret));
696
        }
697

698
        SSchema        *pTagColumn = NULL;
79,179✔
699
        SSchemaWrapper *pTagSchema = &stbEntry.stbEntry.schemaTag;
79,179✔
700
        if (pTagSchema->nCols == 1 && pTagSchema->pSchema[0].type == TSDB_DATA_TYPE_JSON) {
79,179✔
701
          pTagColumn = &stbEntry.stbEntry.schemaTag.pSchema[0];
×
702
          ret = metaDelJsonVarFromIdx(pMeta, &e, pTagColumn);
×
703
          if (ret < 0) {
×
704
            metaError("vgId:%d, failed to delete json var from idx:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode),
×
705
                      e.name, e.uid, tstrerror(ret));
706
          }
707
        } else {
708
          for (int i = 0; i < pTagSchema->nCols; i++) {
408,384✔
709
            pTagColumn = &stbEntry.stbEntry.schemaTag.pSchema[i];
329,205✔
710
            if (!IS_IDX_ON(pTagColumn)) continue;
329,205✔
711
            STagIdxKey *pTagIdxKey = NULL;
79,179✔
712
            int32_t     nTagIdxKey;
79,179✔
713

714
            const void *pTagData = NULL;
79,179✔
715
            int32_t     nTagData = 0;
79,179✔
716

717
            STagVal tagVal = {.cid = pTagColumn->colId};
79,179✔
718
            if (tTagGet((const STag *)e.ctbEntry.pTags, &tagVal)) {
79,179✔
719
              if (IS_VAR_DATA_TYPE(pTagColumn->type)) {
79,179✔
720
                pTagData = tagVal.pData;
×
721
                nTagData = (int32_t)tagVal.nData;
×
722
              } else {
723
                pTagData = &(tagVal.i64);
79,179✔
724
                nTagData = tDataTypes[pTagColumn->type].bytes;
79,179✔
725
              }
726
            } else {
727
              if (!IS_VAR_DATA_TYPE(pTagColumn->type)) {
×
728
                nTagData = tDataTypes[pTagColumn->type].bytes;
×
729
              }
730
            }
731

732
            if (metaCreateTagIdxKey(e.ctbEntry.suid, pTagColumn->colId, pTagData, nTagData, pTagColumn->type, uid,
79,179✔
733
                                    &pTagIdxKey, &nTagIdxKey) == 0) {
734
              ret = tdbTbDelete(pMeta->pTagIdx, pTagIdxKey, nTagIdxKey, pMeta->txn);
79,179✔
735
              if (ret < 0) {
79,179✔
736
                metaError("vgId:%d, failed to delete tag idx key:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode),
×
737
                          e.name, e.uid, tstrerror(ret));
738
              }
739
            }
740
            metaDestroyTagIdxKey(pTagIdxKey);
79,179✔
741
            pTagIdxKey = NULL;
79,179✔
742
          }
743
        }
744
        tDecoderClear(&tdc);
79,179✔
745
      }
746
      tdbFree(tData);
79,179✔
747
    }
748
  }
749

750
  ret = tdbTbDelete(pMeta->pTbDb, &(STbDbKey){.version = version, .uid = uid}, sizeof(STbDbKey), pMeta->txn);
85,276✔
751
  if (ret < 0) {
85,276✔
752
    metaError("vgId:%d, failed to delete table:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
753
              tstrerror(ret));
754
  }
755
  ret = tdbTbDelete(pMeta->pNameIdx, e.name, strlen(e.name) + 1, pMeta->txn);
85,276✔
756
  if (ret < 0) {
85,276✔
757
    metaError("vgId:%d, failed to delete name idx:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
758
              tstrerror(ret));
759
  }
760
  ret = tdbTbDelete(pMeta->pUidIdx, &uid, sizeof(uid), pMeta->txn);
85,276✔
761
  if (ret < 0) {
85,276✔
762
    metaError("vgId:%d, failed to delete uid idx:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
763
              tstrerror(ret));
764
  }
765

766
  if (e.type == TSDB_CHILD_TABLE || e.type == TSDB_NORMAL_TABLE) metaDeleteBtimeIdx(pMeta, &e);
85,276✔
767
  if (e.type == TSDB_NORMAL_TABLE) metaDeleteNcolIdx(pMeta, &e);
85,276✔
768

769
  if (e.type != TSDB_SUPER_TABLE) metaDeleteTtl(pMeta, &e);
85,276✔
770

771
  if (e.type == TSDB_CHILD_TABLE) {
85,276✔
772
    ret =
773
        tdbTbDelete(pMeta->pCtbIdx, &(SCtbIdxKey){.suid = e.ctbEntry.suid, .uid = uid}, sizeof(SCtbIdxKey), pMeta->txn);
79,179✔
774
    if (ret < 0) {
79,179✔
775
      metaError("vgId:%d, failed to delete ctb idx:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
776
                tstrerror(ret));
777
    }
778

779
    --pMeta->pVnode->config.vndStats.numOfCTables;
79,179✔
780
    metaUpdateStbStats(pMeta, e.ctbEntry.suid, -1, 0, -1);
79,179✔
781
    ret = metaUidCacheClear(pMeta, e.ctbEntry.suid);
79,179✔
782
    if (ret < 0) {
79,179✔
783
      metaError("vgId:%d, failed to clear uid cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
×
784
                e.ctbEntry.suid, tstrerror(ret));
785
    }
786
    ret = metaTbGroupCacheClear(pMeta, e.ctbEntry.suid);
79,179✔
787
    if (ret < 0) {
79,179✔
788
      metaError("vgId:%d, failed to clear group cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
×
789
                e.ctbEntry.suid, tstrerror(ret));
790
    }
791
    /*
792
    if (!TSDB_CACHE_NO(pMeta->pVnode->config)) {
793
      tsdbCacheDropTable(pMeta->pVnode->pTsdb, e.uid, e.ctbEntry.suid, NULL);
794
    }
795
    */
796
  } else if (e.type == TSDB_NORMAL_TABLE) {
6,097✔
797
    // drop schema.db (todo)
798

799
    --pMeta->pVnode->config.vndStats.numOfNTables;
6,097✔
800
    pMeta->pVnode->config.vndStats.numOfNTimeSeries -= e.ntbEntry.schemaRow.nCols - 1;
6,097✔
801

802
    /*
803
    if (!TSDB_CACHE_NO(pMeta->pVnode->config)) {
804
      tsdbCacheDropTable(pMeta->pVnode->pTsdb, e.uid, -1, &e.ntbEntry.schemaRow);
805
    }
806
    */
807
  } else if (e.type == TSDB_SUPER_TABLE) {
×
808
    ret = tdbTbDelete(pMeta->pSuidIdx, &e.uid, sizeof(tb_uid_t), pMeta->txn);
×
809
    if (ret < 0) {
×
810
      metaError("vgId:%d, failed to delete suid idx:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
811
                tstrerror(ret));
812
    }
813
    // drop schema.db (todo)
814

815
    ret = metaStatsCacheDrop(pMeta, uid);
×
816
    if (ret < 0) {
×
817
      metaError("vgId:%d, failed to drop stats cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
818
                tstrerror(ret));
819
    }
820
    ret = metaUidCacheClear(pMeta, uid);
×
821
    if (ret < 0) {
×
822
      metaError("vgId:%d, failed to clear uid cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
×
823
                tstrerror(ret));
824
    }
825
    ret = metaStableTagFilterCacheDropSTable(pMeta, uid);
×
826
    if (ret < 0) {
×
827
      metaError("vgId:%d, failed to clear stable tag filter cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
×
828
                e.uid, tstrerror(ret));
829
    }
830
    ret = metaTbGroupCacheClear(pMeta, uid);
×
831
    if (ret < 0) {
×
832
      metaError("vgId:%d, failed to clear group cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name,
×
833
                e.uid, tstrerror(ret));
834
    }
835
    --pMeta->pVnode->config.vndStats.numOfSTables;
×
836
  }
837

838
  ret = metaCacheDrop(pMeta, uid);
85,276✔
839
  if (ret < 0) {
85,276✔
840
    metaError("vgId:%d, failed to drop cache:%s uid:%" PRId64 " since %s", TD_VID(pMeta->pVnode), e.name, e.uid,
85,276✔
841
              tstrerror(ret));
842
  }
843

844
  tDecoderClear(&dc);
85,276✔
845
  tdbFree(pData);
85,276✔
846

847
  return 0;
85,276✔
848
}
849

850
static int metaDeleteBtimeIdx(SMeta *pMeta, const SMetaEntry *pME) {
85,276✔
851
  SBtimeIdxKey btimeKey = {0};
85,276✔
852
  if (metaBuildBtimeIdxKey(&btimeKey, pME) < 0) {
85,276✔
853
    return 0;
×
854
  }
855
  return tdbTbDelete(pMeta->pBtimeIdx, &btimeKey, sizeof(btimeKey), pMeta->txn);
85,276✔
856
}
857

858
int metaDeleteNcolIdx(SMeta *pMeta, const SMetaEntry *pME) {
6,097✔
859
  SNcolIdxKey ncolKey = {0};
6,097✔
860
  if (metaBuildNColIdxKey(&ncolKey, pME) < 0) {
6,097✔
861
    return 0;
×
862
  }
863
  return tdbTbDelete(pMeta->pNcolIdx, &ncolKey, sizeof(ncolKey), pMeta->txn);
6,097✔
864
}
865

866
int metaAlterTable(SMeta *pMeta, int64_t version, SVAlterTbReq *pReq, STableMetaRsp *pMetaRsp) {
13,162,256✔
867
  pMeta->changed = true;
13,162,256✔
868
  switch (pReq->action) {
13,162,256✔
869
    case TSDB_ALTER_TABLE_ADD_COLUMN:
3,839,988✔
870
    case TSDB_ALTER_TABLE_ADD_COLUMN_WITH_COMPRESS_OPTION:
871
    case TSDB_ALTER_TABLE_ADD_COLUMN_WITH_COLUMN_REF:
872
      return metaAddTableColumn(pMeta, version, pReq, pMetaRsp);
3,839,988✔
873
    case TSDB_ALTER_TABLE_DROP_COLUMN:
99,290✔
874
      return metaDropTableColumn(pMeta, version, pReq, pMetaRsp);
99,290✔
875
    case TSDB_ALTER_TABLE_UPDATE_COLUMN_BYTES:
528,364✔
876
      return metaAlterTableColumnBytes(pMeta, version, pReq, pMetaRsp);
528,364✔
877
    case TSDB_ALTER_TABLE_UPDATE_COLUMN_NAME:
48,232✔
878
      return metaAlterTableColumnName(pMeta, version, pReq, pMetaRsp);
48,232✔
879
    case TSDB_ALTER_TABLE_UPDATE_MULTI_TABLE_TAG_VAL:
8,399,168✔
880
      return metaUpdateTableMultiTableTagValue(pMeta, version, pReq);
8,399,168✔
881
    case TSDB_ALTER_TABLE_UPDATE_CHILD_TABLE_TAG_VAL:
65,214✔
882
      return metaUpdateTableChildTableTagValue(pMeta, version, pReq);
65,214✔
883
    case TSDB_ALTER_TABLE_UPDATE_OPTIONS:
27,160✔
884
      return metaUpdateTableOptions2(pMeta, version, pReq);
27,160✔
885
    case TSDB_ALTER_TABLE_UPDATE_COLUMN_COMPRESS:
6,620✔
886
      return metaUpdateTableColCompress2(pMeta, version, pReq);
6,620✔
887
    case TSDB_ALTER_TABLE_ALTER_COLUMN_REF:
84,966✔
888
      return metaAlterTableColumnRef(pMeta, version, pReq, pMetaRsp);
84,966✔
889
    case TSDB_ALTER_TABLE_REMOVE_COLUMN_REF:
63,254✔
890
      return metaRemoveTableColumnRef(pMeta, version, pReq, pMetaRsp);
63,254✔
891
    default:
×
892
      return terrno = TSDB_CODE_VND_INVALID_TABLE_ACTION;
×
893
      break;
894
  }
895
}
896

897
static int metaUpdateChangeTime(SMeta *pMeta, tb_uid_t uid, int64_t changeTimeMs) {
22,453,739✔
898
  if (!tsTtlChangeOnWrite) return 0;
22,453,739✔
899

900
  if (changeTimeMs <= 0) {
22,453,739✔
901
    metaWarn("Skip to change ttl deletetion time on write, uid: %" PRId64, uid);
×
902
    return TSDB_CODE_VERSION_NOT_COMPATIBLE;
×
903
  }
904

905
  STtlUpdCtimeCtx ctx = {.uid = uid, .changeTimeMs = changeTimeMs, .pTxn = pMeta->txn};
22,453,739✔
906

907
  return ttlMgrUpdateChangeTime(pMeta->pTtlMgr, &ctx);
22,453,739✔
908
}
909

910
int metaUpdateChangeTimeWithLock(SMeta *pMeta, tb_uid_t uid, int64_t changeTimeMs) {
1,341,139,111✔
911
  if (!tsTtlChangeOnWrite) return 0;
1,341,139,111✔
912

913
  metaWLock(pMeta);
22,475,819✔
914
  int ret = metaUpdateChangeTime(pMeta, uid, changeTimeMs);
22,453,739✔
915
  metaULock(pMeta);
22,453,739✔
916
  return ret;
22,453,739✔
917
}
918

919
int metaCreateTagIdxKey(tb_uid_t suid, int32_t cid, const void *pTagData, int32_t nTagData, int8_t type, tb_uid_t uid,
106,959,561✔
920
                        STagIdxKey **ppTagIdxKey, int32_t *nTagIdxKey) {
921
  if (IS_VAR_DATA_TYPE(type)) {
106,959,561✔
922
    *nTagIdxKey = sizeof(STagIdxKey) + nTagData + VARSTR_HEADER_SIZE + sizeof(tb_uid_t);
12,346,510✔
923
  } else {
924
    *nTagIdxKey = sizeof(STagIdxKey) + nTagData + sizeof(tb_uid_t);
94,613,051✔
925
  }
926

927
  *ppTagIdxKey = (STagIdxKey *)taosMemoryMalloc(*nTagIdxKey);
106,977,086✔
928
  if (*ppTagIdxKey == NULL) {
106,970,152✔
929
    return terrno;
×
930
  }
931

932
  taosSetInt64Aligned(&((*ppTagIdxKey)->suid), suid);
106,969,953✔
933
  (*ppTagIdxKey)->cid = cid;
106,964,248✔
934
  (*ppTagIdxKey)->isNull = (pTagData == NULL) ? 1 : 0;
106,988,439✔
935
  (*ppTagIdxKey)->type = type;
106,975,788✔
936

937
  // refactor
938
  if (IS_VAR_DATA_TYPE(type)) {
106,979,830✔
939
    memcpy((*ppTagIdxKey)->data, (uint16_t *)&nTagData, VARSTR_HEADER_SIZE);
12,386,831✔
940
    if (pTagData != NULL) memcpy((*ppTagIdxKey)->data + VARSTR_HEADER_SIZE, pTagData, nTagData);
12,370,171✔
941
    taosSetInt64Aligned((tb_uid_t *)((*ppTagIdxKey)->data + VARSTR_HEADER_SIZE + nTagData), uid);
12,370,282✔
942
  } else {
943
    if (pTagData != NULL) memcpy((*ppTagIdxKey)->data, pTagData, nTagData);
94,592,999✔
944
    taosSetInt64Aligned((tb_uid_t *)((*ppTagIdxKey)->data + nTagData), uid);
94,604,819✔
945
  }
946

947
  return 0;
106,976,604✔
948
}
949

950
void metaDestroyTagIdxKey(STagIdxKey *pTagIdxKey) {
103,599,645✔
951
  if (pTagIdxKey) taosMemoryFree(pTagIdxKey);
103,599,645✔
952
}
103,593,273✔
953

954
static void colCompressDebug(SHashObj *pColCmprObj) {
114,605,951✔
955
  void *p = taosHashIterate(pColCmprObj, NULL);
114,605,951✔
956
  while (p) {
987,599,199✔
957
    uint32_t cmprAlg = *(uint32_t *)p;
872,967,801✔
958
    col_id_t colId = *(col_id_t *)taosHashGetKey(p, NULL);
872,961,701✔
959
    p = taosHashIterate(pColCmprObj, p);
872,973,293✔
960

961
    uint8_t l1, l2, lvl;
873,082,209✔
962
    tcompressDebug(cmprAlg, &l1, &l2, &lvl);
873,090,320✔
963

964
    const char *l1str = columnEncodeStr(l1);
873,051,862✔
965
    const char *l2str = columnCompressStr(l2);
872,997,157✔
966
    const char *lvlstr = columnLevelStr(lvl);
872,989,170✔
967
    metaDebug("colId: %d, encode:%s, compress:%s,level:%s", colId, l1str, l2str, lvlstr);
872,976,956✔
968
  }
969
  return;
114,631,398✔
970
}
971

972
int32_t metaGetColCmpr(SMeta *pMeta, tb_uid_t uid, SHashObj **ppColCmprObj) {
114,651,336✔
973
  int rc = 0;
114,651,336✔
974

975
  SHashObj *pColCmprObj = taosHashInit(32, taosGetDefaultHashFunction(TSDB_DATA_TYPE_SMALLINT), false, HASH_NO_LOCK);
114,651,336✔
976
  if (pColCmprObj == NULL) {
114,648,584✔
977
    pColCmprObj = NULL;
×
978
    return TSDB_CODE_OUT_OF_MEMORY;
×
979
  }
980

981
  void      *pData = NULL;
114,648,584✔
982
  int        nData = 0;
114,652,250✔
983
  SMetaEntry e = {0};
114,651,826✔
984
  SDecoder   dc = {0};
114,654,460✔
985

986
  *ppColCmprObj = NULL;
114,657,052✔
987

988
  metaRLock(pMeta);
114,650,915✔
989
  rc = tdbTbGet(pMeta->pUidIdx, &uid, sizeof(uid), &pData, &nData);
114,661,655✔
990
  if (rc < 0) {
114,658,967✔
991
    taosHashCleanup(pColCmprObj);
37,125✔
992
    metaULock(pMeta);
37,125✔
993
    return TSDB_CODE_FAILED;
37,125✔
994
  }
995
  int64_t version = ((SUidIdxVal *)pData)[0].version;
114,621,842✔
996
  rc = tdbTbGet(pMeta->pTbDb, &(STbDbKey){.version = version, .uid = uid}, sizeof(STbDbKey), &pData, &nData);
114,618,793✔
997
  if (rc < 0) {
114,611,908✔
998
    metaULock(pMeta);
×
999
    taosHashCleanup(pColCmprObj);
×
1000
    metaError("failed to get table entry");
×
1001
    return rc;
×
1002
  }
1003

1004
  tDecoderInit(&dc, pData, nData);
114,611,908✔
1005
  rc = metaDecodeEntry(&dc, &e);
114,611,191✔
1006
  if (rc < 0) {
114,596,705✔
1007
    tDecoderClear(&dc);
×
1008
    tdbFree(pData);
×
1009
    metaULock(pMeta);
×
1010
    taosHashCleanup(pColCmprObj);
×
1011
    return rc;
×
1012
  }
1013
  if (withColCompress(e.type)) {
114,596,705✔
1014
    SColCmprWrapper *p = &e.colCmpr;
114,615,852✔
1015
    for (int32_t i = 0; i < p->nCols; i++) {
987,621,589✔
1016
      SColCmpr *pCmpr = &p->pColCmpr[i];
873,006,443✔
1017
      rc = taosHashPut(pColCmprObj, &pCmpr->id, sizeof(pCmpr->id), &pCmpr->alg, sizeof(pCmpr->alg));
872,997,783✔
1018
      if (rc < 0) {
873,005,737✔
1019
        tDecoderClear(&dc);
×
1020
        tdbFree(pData);
×
1021
        metaULock(pMeta);
×
1022
        taosHashCleanup(pColCmprObj);
×
1023
        return rc;
×
1024
      }
1025
    }
1026
  } else {
UNCOV
1027
    tDecoderClear(&dc);
×
1028
    tdbFree(pData);
×
1029
    metaULock(pMeta);
×
1030
    taosHashCleanup(pColCmprObj);
×
1031
    return 0;
×
1032
  }
1033
  tDecoderClear(&dc);
114,588,870✔
1034
  tdbFree(pData);
114,626,158✔
1035
  metaULock(pMeta);
114,616,495✔
1036

1037
  *ppColCmprObj = pColCmprObj;
114,606,036✔
1038
  colCompressDebug(pColCmprObj);
114,616,524✔
1039

1040
  return 0;
114,631,821✔
1041
}
1042
// refactor later
1043
void *metaGetIdx(SMeta *pMeta) { return pMeta->pTagIdx; }
4,565,881✔
1044
void *metaGetIvtIdx(SMeta *pMeta) { return pMeta->pTagIvtIdx; }
4,565,881✔
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