mndUser.c 21.4 KB
Newer Older
H
refact  
Hongze Cheng 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * 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/>.
 */

S
Shengliang Guan 已提交
16
#define _DEFAULT_SOURCE
S
Shengliang Guan 已提交
17
#include "mndUser.h"
S
Shengliang Guan 已提交
18
#include "mndAuth.h"
S
Shengliang Guan 已提交
19
#include "mndDb.h"
S
Shengliang Guan 已提交
20
#include "mndShow.h"
S
Shengliang Guan 已提交
21
#include "mndTrans.h"
S
tbase64  
Shengliang Guan 已提交
22
#include "tbase64.h"
S
Shengliang Guan 已提交
23

24 25
#define USER_VER_NUMBER   1
#define USER_RESERVE_SIZE 64
S
Shengliang Guan 已提交
26

S
Shengliang Guan 已提交
27 28 29 30 31
static int32_t  mndCreateDefaultUsers(SMnode *pMnode);
static SSdbRaw *mndUserActionEncode(SUserObj *pUser);
static SSdbRow *mndUserActionDecode(SSdbRaw *pRaw);
static int32_t  mndUserActionInsert(SSdb *pSdb, SUserObj *pUser);
static int32_t  mndUserActionDelete(SSdb *pSdb, SUserObj *pUser);
S
Shengliang Guan 已提交
32
static int32_t  mndUserActionUpdate(SSdb *pSdb, SUserObj *pOld, SUserObj *pNew);
S
Shengliang Guan 已提交
33 34 35 36 37
static int32_t  mndCreateUser(SMnode *pMnode, char *acct, SCreateUserReq *pCreate, SNodeMsg *pReq);
static int32_t  mndProcessCreateUserReq(SNodeMsg *pReq);
static int32_t  mndProcessAlterUserReq(SNodeMsg *pReq);
static int32_t  mndProcessDropUserReq(SNodeMsg *pReq);
static int32_t  mndProcessGetUserAuthReq(SNodeMsg *pReq);
38
static int32_t  mndRetrieveUsers(SNodeMsg *pReq, SShowObj *pShow, SSDataBlock *pBlock, int32_t rows);
S
Shengliang Guan 已提交
39
static void     mndCancelGetNextUser(SMnode *pMnode, void *pIter);
S
Shengliang Guan 已提交
40 41 42 43 44 45 46 47 48 49 50

int32_t mndInitUser(SMnode *pMnode) {
  SSdbTable table = {.sdbType = SDB_USER,
                     .keyType = SDB_KEY_BINARY,
                     .deployFp = (SdbDeployFp)mndCreateDefaultUsers,
                     .encodeFp = (SdbEncodeFp)mndUserActionEncode,
                     .decodeFp = (SdbDecodeFp)mndUserActionDecode,
                     .insertFp = (SdbInsertFp)mndUserActionInsert,
                     .updateFp = (SdbUpdateFp)mndUserActionUpdate,
                     .deleteFp = (SdbDeleteFp)mndUserActionDelete};

S
Shengliang Guan 已提交
51 52 53
  mndSetMsgHandle(pMnode, TDMT_MND_CREATE_USER, mndProcessCreateUserReq);
  mndSetMsgHandle(pMnode, TDMT_MND_ALTER_USER, mndProcessAlterUserReq);
  mndSetMsgHandle(pMnode, TDMT_MND_DROP_USER, mndProcessDropUserReq);
S
Shengliang Guan 已提交
54
  mndSetMsgHandle(pMnode, TDMT_MND_GET_USER_AUTH, mndProcessGetUserAuthReq);
S
Shengliang Guan 已提交
55

S
Shengliang Guan 已提交
56 57
  mndAddShowRetrieveHandle(pMnode, TSDB_MGMT_TABLE_USER, mndRetrieveUsers);
  mndAddShowFreeIterHandle(pMnode, TSDB_MGMT_TABLE_USER, mndCancelGetNextUser);
S
Shengliang Guan 已提交
58 59 60 61 62 63 64
  return sdbSetTable(pMnode->pSdb, table);
}

void mndCleanupUser(SMnode *pMnode) {}

static int32_t mndCreateDefaultUser(SMnode *pMnode, char *acct, char *user, char *pass) {
  SUserObj userObj = {0};
S
Shengliang Guan 已提交
65
  taosEncryptPass_c((uint8_t *)pass, strlen(pass), userObj.pass);
S
Shengliang Guan 已提交
66 67 68 69 70 71
  tstrncpy(userObj.user, user, TSDB_USER_LEN);
  tstrncpy(userObj.acct, acct, TSDB_USER_LEN);
  userObj.createdTime = taosGetTimestampMs();
  userObj.updateTime = userObj.createdTime;

  if (strcmp(user, TSDB_DEFAULT_USER) == 0) {
72
    userObj.superUser = 1;
S
Shengliang Guan 已提交
73 74 75 76 77 78
  }

  SSdbRaw *pRaw = mndUserActionEncode(&userObj);
  if (pRaw == NULL) return -1;
  sdbSetRawStatus(pRaw, SDB_STATUS_READY);

S
Shengliang Guan 已提交
79
  mDebug("user:%s, will be created while deploy sdb, raw:%p", userObj.user, pRaw);
S
Shengliang Guan 已提交
80 81 82 83 84 85 86 87 88 89 90
  return sdbWrite(pMnode->pSdb, pRaw);
}

static int32_t mndCreateDefaultUsers(SMnode *pMnode) {
  if (mndCreateDefaultUser(pMnode, TSDB_DEFAULT_USER, TSDB_DEFAULT_USER, TSDB_DEFAULT_PASS) != 0) {
    return -1;
  }

  return 0;
}

S
Shengliang Guan 已提交
91
static SSdbRaw *mndUserActionEncode(SUserObj *pUser) {
92 93
  terrno = TSDB_CODE_OUT_OF_MEMORY;

S
Shengliang Guan 已提交
94 95
  int32_t numOfReadDbs = taosHashGetSize(pUser->readDbs);
  int32_t numOfWriteDbs = taosHashGetSize(pUser->writeDbs);
96
  int32_t size = sizeof(SUserObj) + USER_RESERVE_SIZE + (numOfReadDbs + numOfWriteDbs) * TSDB_DB_FNAME_LEN;
S
Shengliang Guan 已提交
97

98
  SSdbRaw *pRaw = sdbAllocRaw(SDB_USER, USER_VER_NUMBER, size);
99
  if (pRaw == NULL) goto _OVER;
S
Shengliang Guan 已提交
100 101

  int32_t dataPos = 0;
102 103 104 105 106 107 108 109
  SDB_SET_BINARY(pRaw, dataPos, pUser->user, TSDB_USER_LEN, _OVER)
  SDB_SET_BINARY(pRaw, dataPos, pUser->pass, TSDB_PASSWORD_LEN, _OVER)
  SDB_SET_BINARY(pRaw, dataPos, pUser->acct, TSDB_USER_LEN, _OVER)
  SDB_SET_INT64(pRaw, dataPos, pUser->createdTime, _OVER)
  SDB_SET_INT64(pRaw, dataPos, pUser->updateTime, _OVER)
  SDB_SET_INT8(pRaw, dataPos, pUser->superUser, _OVER)
  SDB_SET_INT32(pRaw, dataPos, numOfReadDbs, _OVER)
  SDB_SET_INT32(pRaw, dataPos, numOfWriteDbs, _OVER)
110 111 112

  char *db = taosHashIterate(pUser->readDbs, NULL);
  while (db != NULL) {
113
    SDB_SET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER);
114 115 116 117 118
    db = taosHashIterate(pUser->readDbs, db);
  }

  db = taosHashIterate(pUser->writeDbs, NULL);
  while (db != NULL) {
119
    SDB_SET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER);
120 121 122
    db = taosHashIterate(pUser->writeDbs, db);
  }

123 124
  SDB_SET_RESERVE(pRaw, dataPos, USER_RESERVE_SIZE, _OVER)
  SDB_SET_DATALEN(pRaw, dataPos, _OVER)
125 126 127

  terrno = 0;

128
_OVER:
129 130 131 132 133
  if (terrno != 0) {
    mError("user:%s, failed to encode to raw:%p since %s", pUser->user, pRaw, terrstr());
    sdbFreeRaw(pRaw);
    return NULL;
  }
S
Shengliang Guan 已提交
134

S
Shengliang Guan 已提交
135
  mTrace("user:%s, encode to raw:%p, row:%p", pUser->user, pRaw, pUser);
S
Shengliang Guan 已提交
136
  return pRaw;
S
Shengliang Guan 已提交
137 138
}

S
Shengliang Guan 已提交
139
static SSdbRow *mndUserActionDecode(SSdbRaw *pRaw) {
140 141
  terrno = TSDB_CODE_OUT_OF_MEMORY;

S
Shengliang Guan 已提交
142
  int8_t sver = 0;
143
  if (sdbGetRawSoftVer(pRaw, &sver) != 0) goto _OVER;
S
Shengliang Guan 已提交
144

145
  if (sver != USER_VER_NUMBER) {
S
Shengliang Guan 已提交
146
    terrno = TSDB_CODE_SDB_INVALID_DATA_VER;
147
    goto _OVER;
S
Shengliang Guan 已提交
148
  }
S
Shengliang Guan 已提交
149

150
  SSdbRow *pRow = sdbAllocRow(sizeof(SUserObj));
151
  if (pRow == NULL) goto _OVER;
152

S
Shengliang Guan 已提交
153
  SUserObj *pUser = sdbGetRowObj(pRow);
154
  if (pUser == NULL) goto _OVER;
155

S
Shengliang Guan 已提交
156
  int32_t dataPos = 0;
157 158 159 160 161 162
  SDB_GET_BINARY(pRaw, dataPos, pUser->user, TSDB_USER_LEN, _OVER)
  SDB_GET_BINARY(pRaw, dataPos, pUser->pass, TSDB_PASSWORD_LEN, _OVER)
  SDB_GET_BINARY(pRaw, dataPos, pUser->acct, TSDB_USER_LEN, _OVER)
  SDB_GET_INT64(pRaw, dataPos, &pUser->createdTime, _OVER)
  SDB_GET_INT64(pRaw, dataPos, &pUser->updateTime, _OVER)
  SDB_GET_INT8(pRaw, dataPos, &pUser->superUser, _OVER)
163 164 165

  int32_t numOfReadDbs = 0;
  int32_t numOfWriteDbs = 0;
166 167
  SDB_GET_INT32(pRaw, dataPos, &numOfReadDbs, _OVER)
  SDB_GET_INT32(pRaw, dataPos, &numOfWriteDbs, _OVER)
S
Shengliang Guan 已提交
168 169 170
  pUser->readDbs = taosHashInit(numOfReadDbs, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_ENTRY_LOCK);
  pUser->writeDbs =
      taosHashInit(numOfWriteDbs, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_ENTRY_LOCK);
171
  if (pUser->readDbs == NULL || pUser->writeDbs == NULL) goto _OVER;
172 173 174

  for (int32_t i = 0; i < numOfReadDbs; ++i) {
    char db[TSDB_DB_FNAME_LEN] = {0};
175
    SDB_GET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER)
176 177 178 179 180 181
    int32_t len = strlen(db) + 1;
    taosHashPut(pUser->readDbs, db, len, db, TSDB_DB_FNAME_LEN);
  }

  for (int32_t i = 0; i < numOfWriteDbs; ++i) {
    char db[TSDB_DB_FNAME_LEN] = {0};
182
    SDB_GET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER)
183 184 185 186
    int32_t len = strlen(db) + 1;
    taosHashPut(pUser->writeDbs, db, len, db, TSDB_DB_FNAME_LEN);
  }

187
  SDB_GET_RESERVE(pRaw, dataPos, USER_RESERVE_SIZE, _OVER)
188 189 190

  terrno = 0;

191
_OVER:
192 193
  if (terrno != 0) {
    mError("user:%s, failed to decode from raw:%p since %s", pUser->user, pRaw, terrstr());
194 195
    taosHashCleanup(pUser->readDbs);
    taosHashCleanup(pUser->writeDbs);
wafwerar's avatar
wafwerar 已提交
196
    taosMemoryFreeClear(pRow);
197 198
    return NULL;
  }
S
Shengliang Guan 已提交
199

S
Shengliang Guan 已提交
200
  mTrace("user:%s, decode from raw:%p, row:%p", pUser->user, pRaw, pUser);
S
Shengliang Guan 已提交
201
  return pRow;
S
Shengliang Guan 已提交
202
}
S
Shengliang Guan 已提交
203

S
Shengliang Guan 已提交
204
static int32_t mndUserActionInsert(SSdb *pSdb, SUserObj *pUser) {
S
Shengliang Guan 已提交
205
  mTrace("user:%s, perform insert action, row:%p", pUser->user, pUser);
S
Shengliang Guan 已提交
206

S
Shengliang Guan 已提交
207 208
  SAcctObj *pAcct = sdbAcquire(pSdb, SDB_ACCT, pUser->acct);
  if (pAcct == NULL) {
S
Shengliang Guan 已提交
209
    terrno = TSDB_CODE_MND_ACCT_NOT_EXIST;
S
Shengliang Guan 已提交
210
    mError("user:%s, failed to perform insert action since %s", pUser->user, terrstr());
S
Shengliang Guan 已提交
211
    return -1;
S
Shengliang Guan 已提交
212
  }
S
Shengliang Guan 已提交
213 214
  pUser->acctId = pAcct->acctId;
  sdbRelease(pSdb, pAcct);
S
Shengliang Guan 已提交
215

S
Shengliang Guan 已提交
216 217
  return 0;
}
S
Shengliang Guan 已提交
218

S
Shengliang Guan 已提交
219
static int32_t mndUserActionDelete(SSdb *pSdb, SUserObj *pUser) {
S
Shengliang Guan 已提交
220
  mTrace("user:%s, perform delete action, row:%p", pUser->user, pUser);
221 222
  taosHashCleanup(pUser->readDbs);
  taosHashCleanup(pUser->writeDbs);
223 224
  pUser->readDbs = NULL;
  pUser->writeDbs = NULL;
S
Shengliang Guan 已提交
225 226 227
  return 0;
}

S
Shengliang Guan 已提交
228
static int32_t mndUserActionUpdate(SSdb *pSdb, SUserObj *pOld, SUserObj *pNew) {
S
Shengliang Guan 已提交
229
  mTrace("user:%s, perform update action, old row:%p new row:%p", pOld->user, pOld, pNew);
S
Shengliang Guan 已提交
230 231
  memcpy(pOld->pass, pNew->pass, TSDB_PASSWORD_LEN);
  pOld->updateTime = pNew->updateTime;
232

233 234
  TSWAP(pOld->readDbs, pNew->readDbs, (void *));
  TSWAP(pOld->writeDbs, pNew->writeDbs, (void *));
235

S
Shengliang Guan 已提交
236 237 238
  return 0;
}

S
Shengliang Guan 已提交
239
SUserObj *mndAcquireUser(SMnode *pMnode, char *userName) {
S
Shengliang Guan 已提交
240 241 242
  SSdb     *pSdb = pMnode->pSdb;
  SUserObj *pUser = sdbAcquire(pSdb, SDB_USER, userName);
  if (pUser == NULL) {
S
Shengliang Guan 已提交
243
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
S
Shengliang Guan 已提交
244 245
  }
  return pUser;
S
Shengliang Guan 已提交
246
}
S
Shengliang Guan 已提交
247

S
Shengliang Guan 已提交
248 249 250
void mndReleaseUser(SMnode *pMnode, SUserObj *pUser) {
  SSdb *pSdb = pMnode->pSdb;
  sdbRelease(pSdb, pUser);
S
Shengliang Guan 已提交
251 252
}

S
Shengliang Guan 已提交
253
static int32_t mndCreateUser(SMnode *pMnode, char *acct, SCreateUserReq *pCreate, SNodeMsg *pReq) {
S
Shengliang Guan 已提交
254
  SUserObj userObj = {0};
S
Shengliang Guan 已提交
255 256
  taosEncryptPass_c((uint8_t *)pCreate->pass, strlen(pCreate->pass), userObj.pass);
  tstrncpy(userObj.user, pCreate->user, TSDB_USER_LEN);
S
Shengliang Guan 已提交
257 258 259
  tstrncpy(userObj.acct, acct, TSDB_USER_LEN);
  userObj.createdTime = taosGetTimestampMs();
  userObj.updateTime = userObj.createdTime;
S
Shengliang Guan 已提交
260
  userObj.superUser = pCreate->superUser;
S
Shengliang Guan 已提交
261

S
Shengliang Guan 已提交
262
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_CREATE_USER, &pReq->rpcMsg);
S
Shengliang Guan 已提交
263
  if (pTrans == NULL) {
S
Shengliang Guan 已提交
264
    mError("user:%s, failed to create since %s", pCreate->user, terrstr());
S
Shengliang Guan 已提交
265 266
    return -1;
  }
S
Shengliang Guan 已提交
267
  mDebug("trans:%d, used to create user:%s", pTrans->id, pCreate->user);
S
Shengliang Guan 已提交
268

S
Shengliang Guan 已提交
269
  SSdbRaw *pRedoRaw = mndUserActionEncode(&userObj);
S
Shengliang Guan 已提交
270
  if (pRedoRaw == NULL || mndTransAppendRedolog(pTrans, pRedoRaw) != 0) {
S
Shengliang Guan 已提交
271
    mError("trans:%d, failed to append redo log since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
272
    mndTransDrop(pTrans);
S
Shengliang Guan 已提交
273
    return -1;
S
Shengliang Guan 已提交
274
  }
S
Shengliang Guan 已提交
275 276
  sdbSetRawStatus(pRedoRaw, SDB_STATUS_READY);

277 278 279
  char *param = strdup("====> test code to be deleted later <=====");
  mndTransSetCb(pTrans, TEST_TRANS_START_FUNC, TEST_TRANS_STOP_FUNC, param, strlen(param) + 1);

S
Shengliang Guan 已提交
280
  if (mndTransPrepare(pMnode, pTrans) != 0) {
S
Shengliang Guan 已提交
281
    mError("trans:%d, failed to prepare since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
282
    mndTransDrop(pTrans);
S
Shengliang Guan 已提交
283
    return -1;
S
Shengliang Guan 已提交
284 285
  }

S
Shengliang Guan 已提交
286
  mndTransDrop(pTrans);
S
Shengliang Guan 已提交
287
  return 0;
S
Shengliang Guan 已提交
288 289
}

S
Shengliang Guan 已提交
290 291
static int32_t mndProcessCreateUserReq(SNodeMsg *pReq) {
  SMnode        *pMnode = pReq->pNode;
S
Shengliang Guan 已提交
292 293 294 295 296
  int32_t        code = -1;
  SUserObj      *pUser = NULL;
  SUserObj      *pOperUser = NULL;
  SCreateUserReq createReq = {0};

S
Shengliang Guan 已提交
297 298
  if (tDeserializeSCreateUserReq(pReq->rpcMsg.pCont, pReq->rpcMsg.contLen, &createReq) != 0) {
    terrno = TSDB_CODE_INVALID_MSG;
299
    goto _OVER;
S
Shengliang Guan 已提交
300
  }
S
Shengliang Guan 已提交
301

S
Shengliang Guan 已提交
302
  mDebug("user:%s, start to create", createReq.user);
S
Shengliang Guan 已提交
303

S
Shengliang Guan 已提交
304
  if (createReq.user[0] == 0) {
S
Shengliang Guan 已提交
305
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
306
    goto _OVER;
S
Shengliang Guan 已提交
307 308
  }

S
Shengliang Guan 已提交
309
  if (createReq.pass[0] == 0) {
S
Shengliang Guan 已提交
310
    terrno = TSDB_CODE_MND_INVALID_PASS_FORMAT;
311
    goto _OVER;
S
Shengliang Guan 已提交
312 313
  }

S
Shengliang Guan 已提交
314
  pUser = mndAcquireUser(pMnode, createReq.user);
S
Shengliang Guan 已提交
315
  if (pUser != NULL) {
S
Shengliang Guan 已提交
316
    terrno = TSDB_CODE_MND_USER_ALREADY_EXIST;
317
    goto _OVER;
S
Shengliang Guan 已提交
318 319
  }

S
Shengliang Guan 已提交
320
  pOperUser = mndAcquireUser(pMnode, pReq->user);
S
Shengliang Guan 已提交
321
  if (pOperUser == NULL) {
S
Shengliang Guan 已提交
322
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
323
    goto _OVER;
S
Shengliang Guan 已提交
324 325
  }

S
Shengliang Guan 已提交
326
  if (mndCheckCreateUserAuth(pOperUser) != 0) {
327
    goto _OVER;
S
Shengliang Guan 已提交
328 329 330
  }

  code = mndCreateUser(pMnode, pOperUser->acct, &createReq, pReq);
S
Shengliang Guan 已提交
331
  if (code == 0) code = TSDB_CODE_MND_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
332

333
_OVER:
S
Shengliang Guan 已提交
334 335
  if (code != 0 && code != TSDB_CODE_MND_ACTION_IN_PROGRESS) {
    mError("user:%s, failed to create since %s", createReq.user, terrstr());
S
Shengliang Guan 已提交
336 337
  }

S
Shengliang Guan 已提交
338 339 340 341
  mndReleaseUser(pMnode, pUser);
  mndReleaseUser(pMnode, pOperUser);

  return code;
S
Shengliang Guan 已提交
342 343
}

S
Shengliang Guan 已提交
344
static int32_t mndAlterUser(SMnode *pMnode, SUserObj *pOld, SUserObj *pNew, SNodeMsg *pReq) {
S
Shengliang Guan 已提交
345
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_ALTER_USER, &pReq->rpcMsg);
S
Shengliang Guan 已提交
346
  if (pTrans == NULL) {
S
Shengliang Guan 已提交
347
    mError("user:%s, failed to alter since %s", pOld->user, terrstr());
S
Shengliang Guan 已提交
348 349
    return -1;
  }
S
Shengliang Guan 已提交
350
  mDebug("trans:%d, used to alter user:%s", pTrans->id, pOld->user);
S
Shengliang Guan 已提交
351

S
Shengliang Guan 已提交
352
  SSdbRaw *pRedoRaw = mndUserActionEncode(pNew);
S
Shengliang Guan 已提交
353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369
  if (pRedoRaw == NULL || mndTransAppendRedolog(pTrans, pRedoRaw) != 0) {
    mError("trans:%d, failed to append redo log since %s", pTrans->id, terrstr());
    mndTransDrop(pTrans);
    return -1;
  }
  sdbSetRawStatus(pRedoRaw, SDB_STATUS_READY);

  if (mndTransPrepare(pMnode, pTrans) != 0) {
    mError("trans:%d, failed to prepare since %s", pTrans->id, terrstr());
    mndTransDrop(pTrans);
    return -1;
  }

  mndTransDrop(pTrans);
  return 0;
}

S
Shengliang Guan 已提交
370
static SHashObj *mndDupDbHash(SHashObj *pOld) {
S
Shengliang Guan 已提交
371 372
  SHashObj *pNew =
      taosHashInit(taosHashGetSize(pOld), taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_ENTRY_LOCK);
S
Shengliang Guan 已提交
373 374 375 376 377 378 379 380 381 382 383
  if (pNew == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    return NULL;
  }

  char *db = taosHashIterate(pOld, NULL);
  while (db != NULL) {
    int32_t len = strlen(db) + 1;
    if (taosHashPut(pNew, db, len, db, TSDB_DB_FNAME_LEN) != 0) {
      taosHashCancelIterate(pOld, db);
      taosHashCleanup(pNew);
S
Shengliang Guan 已提交
384
      terrno = TSDB_CODE_OUT_OF_MEMORY;
S
Shengliang Guan 已提交
385 386 387 388 389 390 391 392
      return NULL;
    }
    db = taosHashIterate(pOld, db);
  }

  return pNew;
}

S
Shengliang Guan 已提交
393 394
static int32_t mndProcessAlterUserReq(SNodeMsg *pReq) {
  SMnode       *pMnode = pReq->pNode;
S
Shengliang Guan 已提交
395 396 397
  int32_t       code = -1;
  SUserObj     *pUser = NULL;
  SUserObj     *pOperUser = NULL;
S
Shengliang Guan 已提交
398
  SUserObj      newUser = {0};
S
Shengliang Guan 已提交
399 400
  SAlterUserReq alterReq = {0};

S
Shengliang Guan 已提交
401 402
  if (tDeserializeSAlterUserReq(pReq->rpcMsg.pCont, pReq->rpcMsg.contLen, &alterReq) != 0) {
    terrno = TSDB_CODE_INVALID_MSG;
403
    goto _OVER;
S
Shengliang Guan 已提交
404
  }
S
Shengliang Guan 已提交
405

S
Shengliang Guan 已提交
406
  mDebug("user:%s, start to alter", alterReq.user);
S
Shengliang Guan 已提交
407

S
Shengliang Guan 已提交
408
  if (alterReq.user[0] == 0) {
S
Shengliang Guan 已提交
409
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
410
    goto _OVER;
S
Shengliang Guan 已提交
411 412
  }

S
Shengliang Guan 已提交
413
  if (alterReq.pass[0] == 0) {
S
Shengliang Guan 已提交
414
    terrno = TSDB_CODE_MND_INVALID_PASS_FORMAT;
415
    goto _OVER;
S
Shengliang Guan 已提交
416 417
  }

S
Shengliang Guan 已提交
418
  pUser = mndAcquireUser(pMnode, alterReq.user);
S
Shengliang Guan 已提交
419 420
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
421
    goto _OVER;
S
Shengliang Guan 已提交
422 423
  }

S
Shengliang Guan 已提交
424
  pOperUser = mndAcquireUser(pMnode, pReq->user);
S
Shengliang Guan 已提交
425 426
  if (pOperUser == NULL) {
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
427
    goto _OVER;
S
Shengliang Guan 已提交
428 429 430
  }

  memcpy(&newUser, pUser, sizeof(SUserObj));
S
Shengliang Guan 已提交
431 432 433
  newUser.readDbs = mndDupDbHash(pUser->readDbs);
  newUser.writeDbs = mndDupDbHash(pUser->writeDbs);
  if (newUser.readDbs == NULL || newUser.writeDbs == NULL) {
434
    goto _OVER;
S
Shengliang Guan 已提交
435 436 437 438 439 440 441 442 443 444
  }

  int32_t len = strlen(alterReq.dbname) + 1;
  SDbObj *pDb = mndAcquireDb(pMnode, alterReq.dbname);
  mndReleaseDb(pMnode, pDb);

  if (alterReq.alterType == TSDB_ALTER_USER_PASSWD) {
    char pass[TSDB_PASSWORD_LEN + 1] = {0};
    taosEncryptPass_c((uint8_t *)alterReq.pass, strlen(alterReq.pass), pass);
    memcpy(pUser->pass, pass, TSDB_PASSWORD_LEN);
S
Shengliang Guan 已提交
445
  } else if (alterReq.alterType == TSDB_ALTER_USER_SUPERUSER) {
S
Shengliang Guan 已提交
446
    newUser.superUser = alterReq.superUser;
S
Shengliang Guan 已提交
447
  } else if (alterReq.alterType == TSDB_ALTER_USER_ADD_READ_DB) {
S
Shengliang Guan 已提交
448 449
    if (pDb == NULL) {
      terrno = TSDB_CODE_MND_DB_NOT_EXIST;
450
      goto _OVER;
S
Shengliang Guan 已提交
451 452 453
    }
    if (taosHashPut(newUser.readDbs, alterReq.dbname, len, alterReq.dbname, TSDB_DB_FNAME_LEN) != 0) {
      terrno = TSDB_CODE_OUT_OF_MEMORY;
454
      goto _OVER;
S
Shengliang Guan 已提交
455
    }
S
Shengliang Guan 已提交
456
  } else if (alterReq.alterType == TSDB_ALTER_USER_REMOVE_READ_DB) {
S
Shengliang Guan 已提交
457 458
    if (taosHashRemove(newUser.readDbs, alterReq.dbname, len) != 0) {
      terrno = TSDB_CODE_MND_DB_NOT_EXIST;
459
      goto _OVER;
S
Shengliang Guan 已提交
460
    }
S
Shengliang Guan 已提交
461
  } else if (alterReq.alterType == TSDB_ALTER_USER_CLEAR_READ_DB) {
S
Shengliang Guan 已提交
462 463 464 465
    taosHashClear(newUser.readDbs);
  } else if (alterReq.alterType == TSDB_ALTER_USER_ADD_WRITE_DB) {
    if (pDb == NULL) {
      terrno = TSDB_CODE_MND_DB_NOT_EXIST;
466
      goto _OVER;
S
Shengliang Guan 已提交
467 468 469
    }
    if (taosHashPut(newUser.writeDbs, alterReq.dbname, len, alterReq.dbname, TSDB_DB_FNAME_LEN) != 0) {
      terrno = TSDB_CODE_OUT_OF_MEMORY;
470
      goto _OVER;
S
Shengliang Guan 已提交
471 472 473 474
    }
  } else if (alterReq.alterType == TSDB_ALTER_USER_REMOVE_WRITE_DB) {
    if (taosHashRemove(newUser.writeDbs, alterReq.dbname, len) != 0) {
      terrno = TSDB_CODE_MND_DB_NOT_EXIST;
475
      goto _OVER;
S
Shengliang Guan 已提交
476
    }
S
Shengliang Guan 已提交
477
  } else if (alterReq.alterType == TSDB_ALTER_USER_CLEAR_WRITE_DB) {
S
Shengliang Guan 已提交
478 479 480
    taosHashClear(newUser.writeDbs);
  } else {
    terrno = TSDB_CODE_MND_INVALID_ALTER_OPER;
481
    goto _OVER;
S
Shengliang Guan 已提交
482 483
  }

S
Shengliang Guan 已提交
484
  newUser.updateTime = taosGetTimestampMs();
S
Shengliang Guan 已提交
485

S
Shengliang Guan 已提交
486
  if (mndCheckAlterUserAuth(pOperUser, pUser, pDb, &alterReq) != 0) {
487
    goto _OVER;
S
Shengliang Guan 已提交
488 489
  }

S
Shengliang Guan 已提交
490
  code = mndAlterUser(pMnode, pUser, &newUser, pReq);
S
Shengliang Guan 已提交
491
  if (code == 0) code = TSDB_CODE_MND_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
492

493
_OVER:
S
Shengliang Guan 已提交
494 495
  if (code != 0 && code != TSDB_CODE_MND_ACTION_IN_PROGRESS) {
    mError("user:%s, failed to alter since %s", alterReq.user, terrstr());
S
Shengliang Guan 已提交
496 497
  }

S
Shengliang Guan 已提交
498 499
  mndReleaseUser(pMnode, pOperUser);
  mndReleaseUser(pMnode, pUser);
S
Shengliang Guan 已提交
500 501
  taosHashCleanup(newUser.writeDbs);
  taosHashCleanup(newUser.readDbs);
S
Shengliang Guan 已提交
502 503

  return code;
S
Shengliang Guan 已提交
504 505
}

S
Shengliang Guan 已提交
506
static int32_t mndDropUser(SMnode *pMnode, SNodeMsg *pReq, SUserObj *pUser) {
S
Shengliang Guan 已提交
507
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_DROP_USER, &pReq->rpcMsg);
S
Shengliang Guan 已提交
508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531
  if (pTrans == NULL) {
    mError("user:%s, failed to drop since %s", pUser->user, terrstr());
    return -1;
  }
  mDebug("trans:%d, used to drop user:%s", pTrans->id, pUser->user);

  SSdbRaw *pRedoRaw = mndUserActionEncode(pUser);
  if (pRedoRaw == NULL || mndTransAppendRedolog(pTrans, pRedoRaw) != 0) {
    mError("trans:%d, failed to append redo log since %s", pTrans->id, terrstr());
    mndTransDrop(pTrans);
    return -1;
  }
  sdbSetRawStatus(pRedoRaw, SDB_STATUS_DROPPED);

  if (mndTransPrepare(pMnode, pTrans) != 0) {
    mError("trans:%d, failed to prepare since %s", pTrans->id, terrstr());
    mndTransDrop(pTrans);
    return -1;
  }

  mndTransDrop(pTrans);
  return 0;
}

S
Shengliang Guan 已提交
532 533
static int32_t mndProcessDropUserReq(SNodeMsg *pReq) {
  SMnode      *pMnode = pReq->pNode;
S
Shengliang Guan 已提交
534 535 536 537 538
  int32_t      code = -1;
  SUserObj    *pUser = NULL;
  SUserObj    *pOperUser = NULL;
  SDropUserReq dropReq = {0};

S
Shengliang Guan 已提交
539 540
  if (tDeserializeSDropUserReq(pReq->rpcMsg.pCont, pReq->rpcMsg.contLen, &dropReq) != 0) {
    terrno = TSDB_CODE_INVALID_MSG;
541
    goto _OVER;
S
Shengliang Guan 已提交
542
  }
S
Shengliang Guan 已提交
543

S
Shengliang Guan 已提交
544
  mDebug("user:%s, start to drop", dropReq.user);
S
Shengliang Guan 已提交
545

S
Shengliang Guan 已提交
546
  if (dropReq.user[0] == 0) {
S
Shengliang Guan 已提交
547
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
548
    goto _OVER;
S
Shengliang Guan 已提交
549 550
  }

S
Shengliang Guan 已提交
551
  pUser = mndAcquireUser(pMnode, dropReq.user);
S
Shengliang Guan 已提交
552 553
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
554
    goto _OVER;
S
Shengliang Guan 已提交
555 556
  }

S
Shengliang Guan 已提交
557
  pOperUser = mndAcquireUser(pMnode, pReq->user);
S
Shengliang Guan 已提交
558 559
  if (pOperUser == NULL) {
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
560
    goto _OVER;
S
Shengliang Guan 已提交
561 562
  }

S
Shengliang Guan 已提交
563
  if (mndCheckDropUserAuth(pOperUser) != 0) {
564
    goto _OVER;
S
Shengliang Guan 已提交
565 566
  }

S
Shengliang Guan 已提交
567 568
  code = mndDropUser(pMnode, pReq, pUser);
  if (code == 0) code = TSDB_CODE_MND_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
569

570
_OVER:
S
Shengliang Guan 已提交
571 572
  if (code != 0 && code != TSDB_CODE_MND_ACTION_IN_PROGRESS) {
    mError("user:%s, failed to drop since %s", dropReq.user, terrstr());
S
Shengliang Guan 已提交
573 574
  }

S
Shengliang Guan 已提交
575 576 577 578
  mndReleaseUser(pMnode, pOperUser);
  mndReleaseUser(pMnode, pUser);

  return code;
S
Shengliang Guan 已提交
579 580
}

S
Shengliang Guan 已提交
581 582
static int32_t mndProcessGetUserAuthReq(SNodeMsg *pReq) {
  SMnode         *pMnode = pReq->pNode;
S
Shengliang Guan 已提交
583 584 585 586 587
  int32_t         code = -1;
  SUserObj       *pUser = NULL;
  SGetUserAuthReq authReq = {0};
  SGetUserAuthRsp authRsp = {0};

S
Shengliang Guan 已提交
588 589
  if (tDeserializeSGetUserAuthReq(pReq->rpcMsg.pCont, pReq->rpcMsg.contLen, &authReq) != 0) {
    terrno = TSDB_CODE_INVALID_MSG;
590
    goto _OVER;
S
Shengliang Guan 已提交
591
  }
S
Shengliang Guan 已提交
592 593 594 595 596 597

  mTrace("user:%s, start to get auth", authReq.user);

  pUser = mndAcquireUser(pMnode, authReq.user);
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
598
    goto _OVER;
S
Shengliang Guan 已提交
599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621
  }

  memcpy(authRsp.user, pUser->user, TSDB_USER_LEN);
  authRsp.superAuth = pUser->superUser;
  authRsp.readDbs = mndDupDbHash(pUser->readDbs);
  authRsp.writeDbs = mndDupDbHash(pUser->writeDbs);

  SSdb *pSdb = pMnode->pSdb;
  void *pIter = NULL;
  while (1) {
    SDbObj *pDb = NULL;
    pIter = sdbFetch(pSdb, SDB_DB, pIter, (void **)&pDb);
    if (pIter == NULL) break;

    if (strcmp(pDb->createUser, pUser->user) == 0) {
      int32_t len = strlen(pDb->name) + 1;
      taosHashPut(authRsp.readDbs, pDb->name, len, pDb->name, len);
      taosHashPut(authRsp.writeDbs, pDb->name, len, pDb->name, len);
    }

    sdbRelease(pSdb, pDb);
  }

S
Shengliang Guan 已提交
622
  int32_t contLen = tSerializeSGetUserAuthRsp(NULL, 0, &authRsp);
S
Shengliang Guan 已提交
623 624 625
  void   *pRsp = rpcMallocCont(contLen);
  if (pRsp == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
626
    goto _OVER;
S
Shengliang Guan 已提交
627 628
  }

S
Shengliang Guan 已提交
629
  tSerializeSGetUserAuthRsp(pRsp, contLen, &authRsp);
S
Shengliang Guan 已提交
630

S
Shengliang Guan 已提交
631 632
  pReq->pRsp = pRsp;
  pReq->rspLen = contLen;
S
Shengliang Guan 已提交
633 634
  code = 0;

635
_OVER:
S
Shengliang Guan 已提交
636
  mndReleaseUser(pMnode, pUser);
S
Shengliang Guan 已提交
637
  tFreeSGetUserAuthRsp(&authRsp);
S
Shengliang Guan 已提交
638 639 640 641

  return code;
}

642
static int32_t mndRetrieveUsers(SNodeMsg *pReq, SShowObj *pShow, SSDataBlock *pBlock, int32_t rows) {
S
Shengliang Guan 已提交
643
  SMnode   *pMnode = pReq->pNode;
S
Shengliang Guan 已提交
644 645 646 647 648 649 650 651 652 653 654
  SSdb     *pSdb = pMnode->pSdb;
  int32_t   numOfRows = 0;
  SUserObj *pUser = NULL;
  int32_t   cols = 0;
  char     *pWrite;

  while (numOfRows < rows) {
    pShow->pIter = sdbFetch(pSdb, SDB_USER, pShow->pIter, (void **)&pUser);
    if (pShow->pIter == NULL) break;

    cols = 0;
655
    SColumnInfoData *pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
656 657 658 659

    char name[TSDB_USER_LEN + VARSTR_HEADER_SIZE] = {0};
    STR_WITH_MAXSIZE_TO_VARSTR(name, pUser->user, pShow->bytes[cols]);

660
    colDataAppend(pColInfo, numOfRows, (const char *)name, false);
661

wafwerar's avatar
wafwerar 已提交
662 663
    cols++;
    pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
664

665 666
    const char *src = pUser->superUser ? "super" : "normal";
    char        b[10 + VARSTR_HEADER_SIZE] = {0};
667
    STR_WITH_SIZE_TO_VARSTR(b, src, strlen(src));
668
    colDataAppend(pColInfo, numOfRows, (const char *)b, false);
669

wafwerar's avatar
wafwerar 已提交
670 671
    cols++;
    pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
672
    colDataAppend(pColInfo, numOfRows, (const char *)&pUser->createdTime, false);
S
Shengliang Guan 已提交
673 674 675 676 677

    numOfRows++;
    sdbRelease(pSdb, pUser);
  }

678
  pShow->numOfRows += numOfRows;
S
Shengliang Guan 已提交
679 680 681 682 683 684
  return numOfRows;
}

static void mndCancelGetNextUser(SMnode *pMnode, void *pIter) {
  SSdb *pSdb = pMnode->pSdb;
  sdbCancelFetch(pSdb, pIter);
S
Shengliang Guan 已提交
685
}