2023-03-23 09:15:51 +00:00
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
|
|
|
|
|
*
|
|
|
|
|
* This program is free software: you can use, redistribute, and/or modify
|
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3
|
|
|
|
|
* or later ("AGPL"), as published by the Free Software Foundation.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2023-03-24 09:57:37 +00:00
|
|
|
*/
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
#include "inc/tsdbFS.h"
|
2023-03-27 08:30:22 +00:00
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
#define TSDB_FS_EDIT_MIN TSDB_FS_EDIT_COMMIT
|
|
|
|
|
#define TSDB_FS_EDIT_MAX (TSDB_FS_EDIT_MERGE + 1)
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
TSDB_FS_STATE_NONE = 0,
|
|
|
|
|
TSDB_FS_STATE_OPEN,
|
|
|
|
|
TSDB_FS_STATE_EDIT,
|
|
|
|
|
TSDB_FS_STATE_CLOSE,
|
|
|
|
|
};
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
typedef enum {
|
|
|
|
|
TSDB_FCURRENT = 1,
|
2023-05-08 08:16:45 +00:00
|
|
|
TSDB_FCURRENT_C, // for commit
|
|
|
|
|
TSDB_FCURRENT_M, // for merge
|
2023-05-08 05:20:15 +00:00
|
|
|
} EFCurrentT;
|
|
|
|
|
|
|
|
|
|
static const char *gCurrentFname[] = {
|
|
|
|
|
[TSDB_FCURRENT] = "current.json",
|
|
|
|
|
[TSDB_FCURRENT_C] = "current.json.0",
|
|
|
|
|
[TSDB_FCURRENT_M] = "current.json.1",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static int32_t create_fs(STsdb *pTsdb, STFileSystem **ppFS) {
|
2023-05-08 02:27:55 +00:00
|
|
|
ppFS[0] = taosMemoryCalloc(1, sizeof(*ppFS[0]));
|
|
|
|
|
if (ppFS[0] == NULL) {
|
2023-04-07 07:12:31 +00:00
|
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
ppFS[0]->cstate = taosArrayInit(16, sizeof(STFileSet));
|
|
|
|
|
ppFS[0]->nstate = taosArrayInit(16, sizeof(STFileSet));
|
|
|
|
|
if (ppFS[0]->cstate == NULL || ppFS[0]->nstate == NULL) {
|
|
|
|
|
taosArrayDestroy(ppFS[0]->nstate);
|
|
|
|
|
taosArrayDestroy(ppFS[0]->cstate);
|
2023-04-07 07:12:31 +00:00
|
|
|
taosMemoryFree(ppFS[0]);
|
|
|
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
}
|
2023-04-07 06:23:42 +00:00
|
|
|
|
2023-03-27 09:07:59 +00:00
|
|
|
ppFS[0]->pTsdb = pTsdb;
|
2023-05-08 02:27:55 +00:00
|
|
|
ppFS[0]->state = TSDB_FS_STATE_NONE;
|
2023-04-10 09:52:51 +00:00
|
|
|
tsem_init(&ppFS[0]->canEdit, 0, 1);
|
2023-05-08 02:27:55 +00:00
|
|
|
ppFS[0]->nextEditId = 0;
|
2023-04-07 07:12:31 +00:00
|
|
|
|
2023-03-27 09:07:59 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
static int32_t destroy_fs(STFileSystem **ppFS) {
|
|
|
|
|
if (ppFS[0] == NULL) return 0;
|
|
|
|
|
taosArrayDestroy(ppFS[0]->nstate);
|
|
|
|
|
taosArrayDestroy(ppFS[0]->cstate);
|
|
|
|
|
tsem_destroy(&ppFS[0]->canEdit);
|
|
|
|
|
taosMemoryFree(ppFS[0]);
|
|
|
|
|
ppFS[0] = NULL;
|
2023-03-27 09:07:59 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
static int32_t current_fname(STsdb *pTsdb, char *fname, EFCurrentT ftype) {
|
2023-04-10 09:52:51 +00:00
|
|
|
if (pTsdb->pVnode->pTfs) {
|
|
|
|
|
snprintf(fname, //
|
|
|
|
|
TSDB_FILENAME_LEN, //
|
|
|
|
|
"%s%s%s%s%s", //
|
|
|
|
|
tfsGetPrimaryPath(pTsdb->pVnode->pTfs), //
|
|
|
|
|
TD_DIRSEP, //
|
|
|
|
|
pTsdb->path, //
|
|
|
|
|
TD_DIRSEP, //
|
2023-05-08 05:20:15 +00:00
|
|
|
gCurrentFname[ftype]);
|
2023-04-10 09:52:51 +00:00
|
|
|
} else {
|
|
|
|
|
snprintf(fname, //
|
|
|
|
|
TSDB_FILENAME_LEN, //
|
|
|
|
|
"%s%s%s", //
|
|
|
|
|
pTsdb->path, //
|
|
|
|
|
TD_DIRSEP, //
|
2023-05-08 05:20:15 +00:00
|
|
|
gCurrentFname[ftype]);
|
2023-04-06 08:26:41 +00:00
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
static int32_t fs_from_json_str(const char *pData, STFileSystem *pFS) {
|
2023-04-10 09:52:51 +00:00
|
|
|
int32_t code = 0;
|
|
|
|
|
int32_t lino;
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
ASSERTS(0, "TODO: Not implemented yet");
|
2023-04-10 09:52:51 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
_exit:
|
|
|
|
|
return code;
|
|
|
|
|
}
|
2023-04-10 09:52:51 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
static int32_t save_json(const cJSON *json, const char *fname) {
|
2023-05-08 10:15:31 +00:00
|
|
|
int32_t code = 0;
|
2023-05-08 08:16:45 +00:00
|
|
|
|
|
|
|
|
char *data = cJSON_Print(json);
|
|
|
|
|
if (data == NULL) return TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
|
|
TdFilePtr fp = taosOpenFile(fname, TD_FILE_WRITE | TD_FILE_CREATE | TD_FILE_TRUNC);
|
|
|
|
|
if (fp == NULL) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
goto _exit;
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
if (taosWriteFile(fp, data, strlen(data) + 1) < 0) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
goto _exit;
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
if (taosFsyncFile(fp) < 0) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
goto _exit;
|
|
|
|
|
}
|
2023-04-10 09:52:51 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
taosCloseFile(&fp);
|
2023-04-10 09:52:51 +00:00
|
|
|
|
|
|
|
|
_exit:
|
2023-05-08 08:16:45 +00:00
|
|
|
taosMemoryFree(data);
|
2023-04-10 09:52:51 +00:00
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 10:15:31 +00:00
|
|
|
static int32_t load_json(const char *fname, cJSON **json) {
|
|
|
|
|
int32_t code = 0;
|
|
|
|
|
void *data = NULL;
|
|
|
|
|
|
|
|
|
|
TdFilePtr fp = taosOpenFile(fname, TD_FILE_READ);
|
|
|
|
|
if (fp == NULL) return TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
|
|
|
|
|
int64_t size;
|
|
|
|
|
if (taosFStatFile(fp, &size, NULL) < 0) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
goto _exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data = taosMemoryMalloc(size);
|
|
|
|
|
if (data == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
goto _exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (taosReadFile(fp, data, size) < 0) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
goto _exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
json[0] = cJSON_Parse(data);
|
|
|
|
|
if (json[0] == NULL) {
|
|
|
|
|
code = TSDB_CODE_FILE_CORRUPTED;
|
|
|
|
|
goto _exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
taosCloseFile(&fp);
|
|
|
|
|
if (data) taosMemoryFree(data);
|
|
|
|
|
if (code) json[0] = NULL;
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
static int32_t save_fs(int64_t eid, SArray *aTFileSet, const char *fname) {
|
2023-04-10 09:52:51 +00:00
|
|
|
int32_t code = 0;
|
2023-05-08 08:16:45 +00:00
|
|
|
int32_t lino = 0;
|
2023-04-10 09:52:51 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
cJSON *json = cJSON_CreateObject();
|
|
|
|
|
if (json == NULL) return TSDB_CODE_OUT_OF_MEMORY;
|
2023-04-10 09:52:51 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// fmtv
|
|
|
|
|
if (cJSON_AddNumberToObject(json, "fmtv", 1) == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// eid
|
|
|
|
|
if (cJSON_AddNumberToObject(json, "eid", eid) == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// fset
|
|
|
|
|
cJSON *ajson = cJSON_AddArrayToObject(json, "fset");
|
|
|
|
|
if (ajson == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
for (int32_t i = 0; i < taosArrayGetSize(aTFileSet); i++) {
|
|
|
|
|
STFileSet *pFileSet = (STFileSet *)taosArrayGet(aTFileSet, i);
|
|
|
|
|
|
|
|
|
|
cJSON *tjson = cJSON_CreateObject();
|
|
|
|
|
if (tjson == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
code = tsdbFileSetToJson(pFileSet, tjson);
|
2023-04-10 09:52:51 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
2023-05-08 08:16:45 +00:00
|
|
|
|
|
|
|
|
cJSON_AddItemToArray(ajson, tjson);
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
code = save_json(json, fname);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
2023-04-10 09:52:51 +00:00
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
if (code) {
|
2023-05-08 08:16:45 +00:00
|
|
|
tsdbError("%s failed at line %d since %s", __func__, lino, tstrerror(code));
|
2023-04-10 09:52:51 +00:00
|
|
|
}
|
2023-05-08 08:16:45 +00:00
|
|
|
cJSON_Delete(json);
|
2023-04-10 09:52:51 +00:00
|
|
|
return code;
|
2023-04-07 07:12:31 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 10:15:31 +00:00
|
|
|
static int32_t load_fs(const char *fname, SArray *aTFileSet, int64_t *eid) {
|
|
|
|
|
int32_t code = 0;
|
|
|
|
|
int32_t lino = 0;
|
|
|
|
|
|
|
|
|
|
taosArrayClear(aTFileSet);
|
|
|
|
|
|
|
|
|
|
// load json
|
|
|
|
|
cJSON *json = NULL;
|
|
|
|
|
code = load_json(fname, &json);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
|
|
|
|
|
// parse json
|
|
|
|
|
const cJSON *item;
|
|
|
|
|
|
|
|
|
|
/* fmtv */
|
|
|
|
|
item = cJSON_GetObjectItem(json, "fmtv");
|
|
|
|
|
if (cJSON_IsNumber(item)) {
|
|
|
|
|
ASSERT(item->valuedouble == 1);
|
|
|
|
|
} else {
|
|
|
|
|
code = TSDB_CODE_FILE_CORRUPTED;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* eid */
|
|
|
|
|
item = cJSON_GetObjectItem(json, "eid");
|
|
|
|
|
if (cJSON_IsNumber(item)) {
|
|
|
|
|
eid[0] = item->valuedouble;
|
|
|
|
|
} else {
|
|
|
|
|
code = TSDB_CODE_FILE_CORRUPTED;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* fset */
|
|
|
|
|
item = cJSON_GetObjectItem(json, "fset");
|
|
|
|
|
if (cJSON_IsArray(item)) {
|
|
|
|
|
const cJSON *titem;
|
|
|
|
|
cJSON_ArrayForEach(titem, item) {
|
|
|
|
|
STFileSet *pFileSet = taosArrayReserve(aTFileSet, 1);
|
|
|
|
|
if (pFileSet == NULL) {
|
|
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
code = tsdbFileSetFromJson(titem, pFileSet);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
code = TSDB_CODE_FILE_CORRUPTED;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
if (code) {
|
|
|
|
|
tsdbError("%s failed at line %d since %s, fname:%s", __func__, lino, tstrerror(code), fname);
|
|
|
|
|
}
|
|
|
|
|
if (json) cJSON_Delete(json);
|
|
|
|
|
return code;
|
2023-04-07 07:12:31 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t commit_edit(STFileSystem *pFS, tsdb_fs_edit_t etype) {
|
2023-04-07 08:52:30 +00:00
|
|
|
int32_t code;
|
|
|
|
|
char ofname[TSDB_FILENAME_LEN];
|
|
|
|
|
char nfname[TSDB_FILENAME_LEN];
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
current_fname(pFS->pTsdb, nfname, TSDB_FCURRENT);
|
|
|
|
|
current_fname(pFS->pTsdb, ofname, etype == TSDB_FS_EDIT_COMMIT ? TSDB_FCURRENT_C : TSDB_FCURRENT_M);
|
2023-04-07 08:52:30 +00:00
|
|
|
|
|
|
|
|
code = taosRenameFile(ofname, nfname);
|
|
|
|
|
if (code) {
|
|
|
|
|
code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ASSERTS(0, "TODO: Do changes to pFS");
|
|
|
|
|
|
2023-04-07 07:12:31 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t abort_edit(STFileSystem *pFS, tsdb_fs_edit_t etype) {
|
2023-04-07 08:52:30 +00:00
|
|
|
int32_t code;
|
|
|
|
|
char fname[TSDB_FILENAME_LEN];
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
current_fname(pFS->pTsdb, fname, etype == TSDB_FS_EDIT_COMMIT ? TSDB_FCURRENT_C : TSDB_FCURRENT_M);
|
2023-04-07 08:52:30 +00:00
|
|
|
|
|
|
|
|
code = taosRemoveFile(fname);
|
|
|
|
|
if (code) code = TAOS_SYSTEM_ERROR(code);
|
|
|
|
|
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t scan_file_system(STFileSystem *pFS) {
|
2023-04-07 08:52:30 +00:00
|
|
|
// ASSERTS(0, "TODO: Not implemented yet");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t scan_and_schedule_merge(STFileSystem *pFS) {
|
2023-04-07 08:52:30 +00:00
|
|
|
// ASSERTS(0, "TODO: Not implemented yet");
|
2023-04-07 07:12:31 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
static int32_t update_fs_if_needed(STFileSystem *pFS) {
|
|
|
|
|
// TODO
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int32_t open_fs(STFileSystem *pFS, int8_t rollback) {
|
2023-04-07 07:12:31 +00:00
|
|
|
int32_t code = 0;
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t lino = 0;
|
2023-04-07 07:12:31 +00:00
|
|
|
STsdb *pTsdb = pFS->pTsdb;
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
code = update_fs_if_needed(pFS);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
2023-04-07 07:12:31 +00:00
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
char fCurrent[TSDB_FILENAME_LEN];
|
|
|
|
|
char cCurrent[TSDB_FILENAME_LEN];
|
|
|
|
|
char mCurrent[TSDB_FILENAME_LEN];
|
2023-04-07 07:12:31 +00:00
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
current_fname(pTsdb, fCurrent, TSDB_FCURRENT);
|
|
|
|
|
current_fname(pTsdb, cCurrent, TSDB_FCURRENT_C);
|
2023-05-08 08:16:45 +00:00
|
|
|
current_fname(pTsdb, mCurrent, TSDB_FCURRENT_M);
|
2023-04-07 07:12:31 +00:00
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
if (taosCheckExistFile(fCurrent)) { // current.json exists
|
2023-05-08 10:15:31 +00:00
|
|
|
code = load_fs(fCurrent, pFS->cstate, &pFS->nextEditId);
|
2023-05-08 05:20:15 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
2023-04-07 07:12:31 +00:00
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
// check current.json.commit existence
|
|
|
|
|
if (taosCheckExistFile(cCurrent)) {
|
|
|
|
|
if (rollback) {
|
|
|
|
|
code = commit_edit(pFS, TSDB_FS_EDIT_COMMIT);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
} else {
|
|
|
|
|
code = abort_edit(pFS, TSDB_FS_EDIT_COMMIT);
|
2023-04-07 07:12:31 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
}
|
2023-05-08 05:20:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// check current.json.t existence
|
|
|
|
|
if (taosCheckExistFile(mCurrent)) {
|
|
|
|
|
code = abort_edit(pFS, TSDB_FS_EDIT_MERGE);
|
2023-04-07 07:12:31 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
}
|
2023-05-08 08:16:45 +00:00
|
|
|
|
|
|
|
|
code = scan_file_system(pFS);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
|
|
|
|
|
code = scan_and_schedule_merge(pFS);
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
2023-05-08 05:20:15 +00:00
|
|
|
} else {
|
2023-05-08 08:16:45 +00:00
|
|
|
code = save_fs(0, pFS->nstate, fCurrent);
|
2023-05-08 05:20:15 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
2023-04-07 07:12:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
if (code) {
|
2023-05-08 02:27:55 +00:00
|
|
|
tsdbError("vgId:%d %s failed at line %d since %s", TD_VID(pTsdb->pVnode), __func__, lino, tstrerror(code));
|
2023-04-07 07:12:31 +00:00
|
|
|
} else {
|
2023-05-08 02:27:55 +00:00
|
|
|
tsdbInfo("vgId:%d %s success", TD_VID(pTsdb->pVnode), __func__);
|
2023-04-07 07:12:31 +00:00
|
|
|
}
|
2023-03-27 09:07:59 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t close_file_system(STFileSystem *pFS) {
|
2023-04-07 07:12:31 +00:00
|
|
|
ASSERTS(0, "TODO: Not implemented yet");
|
2023-03-27 08:30:22 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
2023-03-27 09:07:59 +00:00
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t apply_edit(STFileSystem *pFS) {
|
2023-04-11 09:04:52 +00:00
|
|
|
int32_t code = 0;
|
2023-04-07 07:12:31 +00:00
|
|
|
ASSERTS(0, "TODO: Not implemented yet");
|
2023-04-11 09:04:52 +00:00
|
|
|
return code;
|
2023-04-06 08:26:41 +00:00
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t fset_cmpr_fn(const struct STFileSet *pSet1, const struct STFileSet *pSet2) {
|
2023-04-12 07:18:12 +00:00
|
|
|
if (pSet1->fid < pSet2->fid) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else if (pSet1->fid > pSet2->fid) {
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
static int32_t edit_fs(STFileSystem *pFS, const SArray *aFileOp) {
|
2023-04-11 09:04:52 +00:00
|
|
|
int32_t code = 0;
|
|
|
|
|
int32_t lino;
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
taosArrayClearEx(pFS->nstate, NULL /* TODO */);
|
2023-04-11 09:04:52 +00:00
|
|
|
|
|
|
|
|
// TODO: copy current state to new state
|
|
|
|
|
|
2023-04-11 07:34:40 +00:00
|
|
|
for (int32_t iop = 0; iop < taosArrayGetSize(aFileOp); iop++) {
|
|
|
|
|
struct SFileOp *pOp = taosArrayGet(aFileOp, iop);
|
2023-04-11 09:04:52 +00:00
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
struct STFileSet tmpSet = {.fid = pOp->fid};
|
2023-04-12 07:18:12 +00:00
|
|
|
|
|
|
|
|
int32_t idx = taosArraySearchIdx( //
|
2023-05-08 02:27:55 +00:00
|
|
|
pFS->nstate, //
|
2023-04-12 07:18:12 +00:00
|
|
|
&tmpSet, //
|
|
|
|
|
(__compar_fn_t)fset_cmpr_fn, //
|
|
|
|
|
TD_GE);
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
struct STFileSet *pSet;
|
2023-04-12 07:18:12 +00:00
|
|
|
if (idx < 0) {
|
|
|
|
|
pSet = NULL;
|
2023-05-08 02:27:55 +00:00
|
|
|
idx = taosArrayGetSize(pFS->nstate);
|
2023-04-12 07:18:12 +00:00
|
|
|
} else {
|
2023-05-08 02:27:55 +00:00
|
|
|
pSet = taosArrayGet(pFS->nstate, idx);
|
2023-04-12 07:18:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pSet == NULL || pSet->fid != pOp->fid) {
|
|
|
|
|
ASSERTS(pOp->op == TSDB_FOP_CREATE, "BUG: Invalid file operation");
|
|
|
|
|
TSDB_CHECK_CODE( //
|
|
|
|
|
code = tsdbFileSetCreate(pOp->fid, &pSet), //
|
|
|
|
|
lino, //
|
|
|
|
|
_exit);
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
if (taosArrayInsert(pFS->nstate, idx, pSet) == NULL) {
|
2023-04-12 07:18:12 +00:00
|
|
|
code = TSDB_CODE_OUT_OF_MEMORY;
|
|
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// do opration on file set
|
2023-04-11 09:04:52 +00:00
|
|
|
TSDB_CHECK_CODE( //
|
2023-04-12 07:18:12 +00:00
|
|
|
code = tsdbFileSetEdit(pSet, pOp), //
|
2023-04-11 09:04:52 +00:00
|
|
|
lino, //
|
|
|
|
|
_exit);
|
2023-04-11 07:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
2023-04-13 07:13:27 +00:00
|
|
|
// TODO: write new state to file
|
|
|
|
|
|
2023-04-11 09:04:52 +00:00
|
|
|
_exit:
|
2023-04-06 08:26:41 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t tsdbOpenFileSystem(STsdb *pTsdb, STFileSystem **ppFS, int8_t rollback) {
|
2023-03-27 09:07:59 +00:00
|
|
|
int32_t code;
|
|
|
|
|
int32_t lino;
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
code = create_fs(pTsdb, ppFS);
|
2023-03-27 09:07:59 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit);
|
|
|
|
|
|
2023-05-08 05:20:15 +00:00
|
|
|
code = open_fs(ppFS[0], rollback);
|
2023-05-08 02:27:55 +00:00
|
|
|
TSDB_CHECK_CODE(code, lino, _exit)
|
2023-03-27 09:07:59 +00:00
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
if (code) {
|
2023-05-08 02:27:55 +00:00
|
|
|
tsdbError("vgId:%d %s failed at line %d since %s", TD_VID(pTsdb->pVnode), __func__, lino, tstrerror(code));
|
2023-05-08 05:20:15 +00:00
|
|
|
destroy_fs(ppFS);
|
2023-03-27 09:07:59 +00:00
|
|
|
} else {
|
2023-05-08 02:27:55 +00:00
|
|
|
tsdbInfo("vgId:%d %s success", TD_VID(pTsdb->pVnode), __func__);
|
2023-03-27 09:07:59 +00:00
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t tsdbCloseFileSystem(STFileSystem **ppFS) {
|
2023-03-27 09:07:59 +00:00
|
|
|
if (ppFS[0] == NULL) return 0;
|
|
|
|
|
close_file_system(ppFS[0]);
|
2023-05-08 05:20:15 +00:00
|
|
|
destroy_fs(ppFS);
|
2023-03-27 09:07:59 +00:00
|
|
|
return 0;
|
|
|
|
|
}
|
2023-04-06 08:26:41 +00:00
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t tsdbFileSystemEditBegin(STFileSystem *pFS, const SArray *aFileOp, tsdb_fs_edit_t etype) {
|
2023-04-06 08:26:41 +00:00
|
|
|
int32_t code = 0;
|
2023-04-11 07:34:40 +00:00
|
|
|
int32_t lino;
|
2023-04-06 08:26:41 +00:00
|
|
|
char fname[TSDB_FILENAME_LEN];
|
|
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// current_fname(pFS->pTsdb, fname, etype == TSDB_FS_EDIT_COMMIT ? TSDB_FCURRENT_C : TSDB_FCURRENT_M);
|
2023-04-06 08:26:41 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// tsem_wait(&pFS->canEdit);
|
2023-04-06 08:26:41 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// TSDB_CHECK_CODE( //
|
|
|
|
|
// code = edit_fs(pFS, aFileOp), //
|
|
|
|
|
// lino, //
|
|
|
|
|
// _exit);
|
2023-04-11 07:34:40 +00:00
|
|
|
|
2023-05-08 08:16:45 +00:00
|
|
|
// TSDB_CHECK_CODE( //
|
|
|
|
|
// code = save_fs(pFS, fname), //
|
|
|
|
|
// lino, //
|
|
|
|
|
// _exit);
|
2023-04-06 08:26:41 +00:00
|
|
|
|
|
|
|
|
_exit:
|
|
|
|
|
if (code) {
|
|
|
|
|
tsdbError("vgId:%d %s failed at line %d since %s", //
|
|
|
|
|
TD_VID(pFS->pTsdb->pVnode), //
|
|
|
|
|
__func__, //
|
|
|
|
|
lino, //
|
|
|
|
|
tstrerror(code));
|
2023-04-07 08:52:30 +00:00
|
|
|
} else {
|
|
|
|
|
tsdbInfo("vgId:%d %s done, etype:%d", //
|
|
|
|
|
TD_VID(pFS->pTsdb->pVnode), //
|
|
|
|
|
__func__, //
|
|
|
|
|
etype);
|
2023-04-06 08:26:41 +00:00
|
|
|
}
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t tsdbFileSystemEditCommit(STFileSystem *pFS, tsdb_fs_edit_t etype) {
|
2023-04-07 08:52:30 +00:00
|
|
|
int32_t code = commit_edit(pFS, etype);
|
2023-04-10 09:52:51 +00:00
|
|
|
tsem_post(&pFS->canEdit);
|
2023-04-06 08:26:41 +00:00
|
|
|
if (code) {
|
2023-04-07 08:52:30 +00:00
|
|
|
tsdbError("vgId:%d %s failed since %s", //
|
|
|
|
|
TD_VID(pFS->pTsdb->pVnode), //
|
|
|
|
|
__func__, //
|
2023-04-06 08:26:41 +00:00
|
|
|
tstrerror(code));
|
2023-04-07 08:52:30 +00:00
|
|
|
} else {
|
|
|
|
|
tsdbInfo("vgId:%d %s done, etype:%d", //
|
|
|
|
|
TD_VID(pFS->pTsdb->pVnode), //
|
|
|
|
|
__func__, //
|
|
|
|
|
etype);
|
2023-04-06 08:26:41 +00:00
|
|
|
}
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 02:27:55 +00:00
|
|
|
int32_t tsdbFileSystemEditAbort(STFileSystem *pFS, tsdb_fs_edit_t etype) {
|
2023-04-07 08:52:30 +00:00
|
|
|
int32_t code = abort_edit(pFS, etype);
|
2023-04-06 08:26:41 +00:00
|
|
|
if (code) {
|
2023-04-07 08:52:30 +00:00
|
|
|
tsdbError("vgId:%d %s failed since %s, etype:%d", //
|
|
|
|
|
TD_VID(pFS->pTsdb->pVnode), //
|
|
|
|
|
__func__, //
|
|
|
|
|
tstrerror(code), //
|
|
|
|
|
etype);
|
2023-04-10 07:00:01 +00:00
|
|
|
} else {
|
2023-04-06 08:26:41 +00:00
|
|
|
}
|
2023-04-10 09:52:51 +00:00
|
|
|
tsem_post(&pFS->canEdit);
|
2023-04-06 08:26:41 +00:00
|
|
|
return code;
|
|
|
|
|
}
|