mndUser.c 25.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
static int32_t  mndCreateDefaultUsers(SMnode *pMnode);
static SSdbRow *mndUserActionDecode(SSdbRaw *pRaw);
static int32_t  mndUserActionInsert(SSdb *pSdb, SUserObj *pUser);
static int32_t  mndUserActionDelete(SSdb *pSdb, SUserObj *pUser);
S
Shengliang Guan 已提交
31
static int32_t  mndUserActionUpdate(SSdb *pSdb, SUserObj *pOld, SUserObj *pNew);
S
Shengliang Guan 已提交
32 33 34 35 36 37
static int32_t  mndCreateUser(SMnode *pMnode, char *acct, SCreateUserReq *pCreate, SRpcMsg *pReq);
static int32_t  mndProcessCreateUserReq(SRpcMsg *pReq);
static int32_t  mndProcessAlterUserReq(SRpcMsg *pReq);
static int32_t  mndProcessDropUserReq(SRpcMsg *pReq);
static int32_t  mndProcessGetUserAuthReq(SRpcMsg *pReq);
static int32_t  mndRetrieveUsers(SRpcMsg *pReq, SShowObj *pShow, SSDataBlock *pBlock, int32_t rows);
S
Shengliang Guan 已提交
38
static void     mndCancelGetNextUser(SMnode *pMnode, void *pIter);
S
Shengliang Guan 已提交
39 40

int32_t mndInitUser(SMnode *pMnode) {
S
Shengliang Guan 已提交
41 42 43 44 45 46 47 48 49 50
  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

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

S
Shengliang Guan 已提交
57 58
  mndAddShowRetrieveHandle(pMnode, TSDB_MGMT_TABLE_USER, mndRetrieveUsers);
  mndAddShowFreeIterHandle(pMnode, TSDB_MGMT_TABLE_USER, mndCancelGetNextUser);
S
Shengliang Guan 已提交
59 60 61 62 63 64 65
  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 已提交
66
  taosEncryptPass_c((uint8_t *)pass, strlen(pass), userObj.pass);
S
Shengliang Guan 已提交
67 68 69 70 71 72
  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) {
73
    userObj.superUser = 1;
S
Shengliang Guan 已提交
74 75 76 77 78 79
  }

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

80
  mDebug("user:%s, will be created when deploying, raw:%p", userObj.user, pRaw);
81 82

#if 0
S
Shengliang Guan 已提交
83
  return sdbWrite(pMnode->pSdb, pRaw);
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
#else
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_RETRY, TRN_TYPE_CREATE_USER, NULL);
  if (pTrans == NULL) {
    mError("user:%s, failed to create since %s", userObj.user, terrstr());
    return -1;
  }
  mDebug("trans:%d, used to create user:%s", pTrans->id, userObj.user);

  if (mndTransAppendCommitlog(pTrans, pRaw) != 0) {
    mError("trans:%d, failed to commit redo log since %s", pTrans->id, terrstr());
    mndTransDrop(pTrans);
    return -1;
  }
  sdbSetRawStatus(pRaw, 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;
#endif
S
Shengliang Guan 已提交
108 109 110 111 112 113 114 115 116 117
}

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

  return 0;
}

118
SSdbRaw *mndUserActionEncode(SUserObj *pUser) {
119 120
  terrno = TSDB_CODE_OUT_OF_MEMORY;

S
Shengliang Guan 已提交
121 122
  int32_t numOfReadDbs = taosHashGetSize(pUser->readDbs);
  int32_t numOfWriteDbs = taosHashGetSize(pUser->writeDbs);
123
  int32_t size = sizeof(SUserObj) + USER_RESERVE_SIZE + (numOfReadDbs + numOfWriteDbs) * TSDB_DB_FNAME_LEN;
S
Shengliang Guan 已提交
124

125
  SSdbRaw *pRaw = sdbAllocRaw(SDB_USER, USER_VER_NUMBER, size);
126
  if (pRaw == NULL) goto _OVER;
S
Shengliang Guan 已提交
127 128

  int32_t dataPos = 0;
129 130 131 132 133 134
  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)
135
  SDB_SET_INT32(pRaw, dataPos, pUser->authVersion, _OVER)
136 137
  SDB_SET_INT32(pRaw, dataPos, numOfReadDbs, _OVER)
  SDB_SET_INT32(pRaw, dataPos, numOfWriteDbs, _OVER)
138 139 140

  char *db = taosHashIterate(pUser->readDbs, NULL);
  while (db != NULL) {
141
    SDB_SET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER);
142 143 144 145 146
    db = taosHashIterate(pUser->readDbs, db);
  }

  db = taosHashIterate(pUser->writeDbs, NULL);
  while (db != NULL) {
147
    SDB_SET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER);
148 149 150
    db = taosHashIterate(pUser->writeDbs, db);
  }

151 152
  SDB_SET_RESERVE(pRaw, dataPos, USER_RESERVE_SIZE, _OVER)
  SDB_SET_DATALEN(pRaw, dataPos, _OVER)
153 154 155

  terrno = 0;

156
_OVER:
157 158 159 160 161
  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 已提交
162

S
Shengliang Guan 已提交
163
  mTrace("user:%s, encode to raw:%p, row:%p", pUser->user, pRaw, pUser);
S
Shengliang Guan 已提交
164
  return pRaw;
S
Shengliang Guan 已提交
165 166
}

S
Shengliang Guan 已提交
167
static SSdbRow *mndUserActionDecode(SSdbRaw *pRaw) {
168 169
  terrno = TSDB_CODE_OUT_OF_MEMORY;

S
Shengliang Guan 已提交
170
  int8_t sver = 0;
171
  if (sdbGetRawSoftVer(pRaw, &sver) != 0) goto _OVER;
S
Shengliang Guan 已提交
172

173
  if (sver != USER_VER_NUMBER) {
S
Shengliang Guan 已提交
174
    terrno = TSDB_CODE_SDB_INVALID_DATA_VER;
175
    goto _OVER;
S
Shengliang Guan 已提交
176
  }
S
Shengliang Guan 已提交
177

178
  SSdbRow *pRow = sdbAllocRow(sizeof(SUserObj));
179
  if (pRow == NULL) goto _OVER;
180

S
Shengliang Guan 已提交
181
  SUserObj *pUser = sdbGetRowObj(pRow);
182
  if (pUser == NULL) goto _OVER;
183

S
Shengliang Guan 已提交
184
  int32_t dataPos = 0;
185 186 187 188 189 190
  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)
191
  SDB_GET_INT32(pRaw, dataPos, &pUser->authVersion, _OVER)
192 193 194

  int32_t numOfReadDbs = 0;
  int32_t numOfWriteDbs = 0;
195 196
  SDB_GET_INT32(pRaw, dataPos, &numOfReadDbs, _OVER)
  SDB_GET_INT32(pRaw, dataPos, &numOfWriteDbs, _OVER)
S
Shengliang Guan 已提交
197 198 199
  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);
200
  if (pUser->readDbs == NULL || pUser->writeDbs == NULL) goto _OVER;
201 202 203

  for (int32_t i = 0; i < numOfReadDbs; ++i) {
    char db[TSDB_DB_FNAME_LEN] = {0};
204
    SDB_GET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER)
205 206 207 208 209 210
    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};
211
    SDB_GET_BINARY(pRaw, dataPos, db, TSDB_DB_FNAME_LEN, _OVER)
212 213 214 215
    int32_t len = strlen(db) + 1;
    taosHashPut(pUser->writeDbs, db, len, db, TSDB_DB_FNAME_LEN);
  }

216
  SDB_GET_RESERVE(pRaw, dataPos, USER_RESERVE_SIZE, _OVER)
S
Shengliang Guan 已提交
217
  taosInitRWLatch(&pUser->lock);
218 219 220

  terrno = 0;

221
_OVER:
222 223
  if (terrno != 0) {
    mError("user:%s, failed to decode from raw:%p since %s", pUser->user, pRaw, terrstr());
224 225
    taosHashCleanup(pUser->readDbs);
    taosHashCleanup(pUser->writeDbs);
wafwerar's avatar
wafwerar 已提交
226
    taosMemoryFreeClear(pRow);
227 228
    return NULL;
  }
S
Shengliang Guan 已提交
229

S
Shengliang Guan 已提交
230
  mTrace("user:%s, decode from raw:%p, row:%p", pUser->user, pRaw, pUser);
S
Shengliang Guan 已提交
231
  return pRow;
S
Shengliang Guan 已提交
232
}
S
Shengliang Guan 已提交
233

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

S
Shengliang Guan 已提交
237 238
  SAcctObj *pAcct = sdbAcquire(pSdb, SDB_ACCT, pUser->acct);
  if (pAcct == NULL) {
S
Shengliang Guan 已提交
239
    terrno = TSDB_CODE_MND_ACCT_NOT_EXIST;
S
Shengliang Guan 已提交
240
    mError("user:%s, failed to perform insert action since %s", pUser->user, terrstr());
S
Shengliang Guan 已提交
241
    return -1;
S
Shengliang Guan 已提交
242
  }
S
Shengliang Guan 已提交
243 244
  pUser->acctId = pAcct->acctId;
  sdbRelease(pSdb, pAcct);
S
Shengliang Guan 已提交
245

S
Shengliang Guan 已提交
246 247
  return 0;
}
S
Shengliang Guan 已提交
248

S
Shengliang Guan 已提交
249
static int32_t mndUserActionDelete(SSdb *pSdb, SUserObj *pUser) {
S
Shengliang Guan 已提交
250
  mTrace("user:%s, perform delete action, row:%p", pUser->user, pUser);
251 252
  taosHashCleanup(pUser->readDbs);
  taosHashCleanup(pUser->writeDbs);
253 254
  pUser->readDbs = NULL;
  pUser->writeDbs = NULL;
S
Shengliang Guan 已提交
255 256 257
  return 0;
}

S
Shengliang Guan 已提交
258
static int32_t mndUserActionUpdate(SSdb *pSdb, SUserObj *pOld, SUserObj *pNew) {
S
Shengliang Guan 已提交
259
  mTrace("user:%s, perform update action, old row:%p new row:%p", pOld->user, pOld, pNew);
S
Shengliang Guan 已提交
260
  taosWLockLatch(&pOld->lock);
S
Shengliang Guan 已提交
261
  pOld->updateTime = pNew->updateTime;
D
dapan1121 已提交
262
  pOld->authVersion = pNew->authVersion;
S
Shengliang Guan 已提交
263
  memcpy(pOld->pass, pNew->pass, TSDB_PASSWORD_LEN);
wafwerar's avatar
wafwerar 已提交
264 265
  TSWAP(pOld->readDbs, pNew->readDbs);
  TSWAP(pOld->writeDbs, pNew->writeDbs);
S
Shengliang Guan 已提交
266
  taosWUnLockLatch(&pOld->lock);
267

S
Shengliang Guan 已提交
268 269 270
  return 0;
}

271
SUserObj *mndAcquireUser(SMnode *pMnode, const char *userName) {
S
Shengliang Guan 已提交
272 273 274
  SSdb     *pSdb = pMnode->pSdb;
  SUserObj *pUser = sdbAcquire(pSdb, SDB_USER, userName);
  if (pUser == NULL) {
S
Shengliang Guan 已提交
275
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
S
Shengliang Guan 已提交
276 277
  }
  return pUser;
S
Shengliang Guan 已提交
278
}
S
Shengliang Guan 已提交
279

S
Shengliang Guan 已提交
280 281 282
void mndReleaseUser(SMnode *pMnode, SUserObj *pUser) {
  SSdb *pSdb = pMnode->pSdb;
  sdbRelease(pSdb, pUser);
S
Shengliang Guan 已提交
283 284
}

S
Shengliang Guan 已提交
285
static int32_t mndCreateUser(SMnode *pMnode, char *acct, SCreateUserReq *pCreate, SRpcMsg *pReq) {
S
Shengliang Guan 已提交
286
  SUserObj userObj = {0};
S
Shengliang Guan 已提交
287 288
  taosEncryptPass_c((uint8_t *)pCreate->pass, strlen(pCreate->pass), userObj.pass);
  tstrncpy(userObj.user, pCreate->user, TSDB_USER_LEN);
S
Shengliang Guan 已提交
289 290 291
  tstrncpy(userObj.acct, acct, TSDB_USER_LEN);
  userObj.createdTime = taosGetTimestampMs();
  userObj.updateTime = userObj.createdTime;
S
Shengliang Guan 已提交
292
  userObj.superUser = pCreate->superUser;
S
Shengliang Guan 已提交
293

S
Shengliang Guan 已提交
294
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_CREATE_USER, pReq);
S
Shengliang Guan 已提交
295
  if (pTrans == NULL) {
S
Shengliang Guan 已提交
296
    mError("user:%s, failed to create since %s", pCreate->user, terrstr());
S
Shengliang Guan 已提交
297 298
    return -1;
  }
S
Shengliang Guan 已提交
299
  mDebug("trans:%d, used to create user:%s", pTrans->id, pCreate->user);
S
Shengliang Guan 已提交
300

301 302 303
  SSdbRaw *pCommitRaw = mndUserActionEncode(&userObj);
  if (pCommitRaw == NULL || mndTransAppendCommitlog(pTrans, pCommitRaw) != 0) {
    mError("trans:%d, failed to commit redo log since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
304
    mndTransDrop(pTrans);
S
Shengliang Guan 已提交
305
    return -1;
S
Shengliang Guan 已提交
306
  }
307
  sdbSetRawStatus(pCommitRaw, SDB_STATUS_READY);
S
Shengliang Guan 已提交
308

S
Shengliang Guan 已提交
309
  if (mndTransPrepare(pMnode, pTrans) != 0) {
S
Shengliang Guan 已提交
310
    mError("trans:%d, failed to prepare since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
311
    mndTransDrop(pTrans);
S
Shengliang Guan 已提交
312
    return -1;
S
Shengliang Guan 已提交
313 314
  }

S
Shengliang Guan 已提交
315
  mndTransDrop(pTrans);
S
Shengliang Guan 已提交
316
  return 0;
S
Shengliang Guan 已提交
317 318
}

S
Shengliang Guan 已提交
319 320
static int32_t mndProcessCreateUserReq(SRpcMsg *pReq) {
  SMnode        *pMnode = pReq->info.node;
S
Shengliang Guan 已提交
321 322 323 324 325
  int32_t        code = -1;
  SUserObj      *pUser = NULL;
  SUserObj      *pOperUser = NULL;
  SCreateUserReq createReq = {0};

S
Shengliang Guan 已提交
326
  if (tDeserializeSCreateUserReq(pReq->pCont, pReq->contLen, &createReq) != 0) {
S
Shengliang Guan 已提交
327
    terrno = TSDB_CODE_INVALID_MSG;
328
    goto _OVER;
S
Shengliang Guan 已提交
329
  }
S
Shengliang Guan 已提交
330

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

S
Shengliang Guan 已提交
333
  if (createReq.user[0] == 0) {
S
Shengliang Guan 已提交
334
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
335
    goto _OVER;
S
Shengliang Guan 已提交
336 337
  }

S
Shengliang Guan 已提交
338
  if (createReq.pass[0] == 0) {
S
Shengliang Guan 已提交
339
    terrno = TSDB_CODE_MND_INVALID_PASS_FORMAT;
340
    goto _OVER;
S
Shengliang Guan 已提交
341 342
  }

S
Shengliang Guan 已提交
343
  pUser = mndAcquireUser(pMnode, createReq.user);
S
Shengliang Guan 已提交
344
  if (pUser != NULL) {
S
Shengliang Guan 已提交
345
    terrno = TSDB_CODE_MND_USER_ALREADY_EXIST;
346
    goto _OVER;
S
Shengliang Guan 已提交
347 348
  }

S
Shengliang Guan 已提交
349
  pOperUser = mndAcquireUser(pMnode, pReq->conn.user);
S
Shengliang Guan 已提交
350
  if (pOperUser == NULL) {
S
Shengliang Guan 已提交
351
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
352
    goto _OVER;
S
Shengliang Guan 已提交
353 354
  }

S
Shengliang Guan 已提交
355
  if (mndCheckCreateUserAuth(pOperUser) != 0) {
356
    goto _OVER;
S
Shengliang Guan 已提交
357 358 359
  }

  code = mndCreateUser(pMnode, pOperUser->acct, &createReq, pReq);
S
Shengliang Guan 已提交
360
  if (code == 0) code = TSDB_CODE_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
361

362
_OVER:
S
Shengliang Guan 已提交
363
  if (code != 0 && code != TSDB_CODE_ACTION_IN_PROGRESS) {
S
Shengliang Guan 已提交
364
    mError("user:%s, failed to create since %s", createReq.user, terrstr());
S
Shengliang Guan 已提交
365 366
  }

S
Shengliang Guan 已提交
367 368 369 370
  mndReleaseUser(pMnode, pUser);
  mndReleaseUser(pMnode, pOperUser);

  return code;
S
Shengliang Guan 已提交
371 372
}

S
Shengliang Guan 已提交
373 374
static int32_t mndAlterUser(SMnode *pMnode, SUserObj *pOld, SUserObj *pNew, SRpcMsg *pReq) {
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_ALTER_USER, pReq);
S
Shengliang Guan 已提交
375
  if (pTrans == NULL) {
S
Shengliang Guan 已提交
376
    mError("user:%s, failed to alter since %s", pOld->user, terrstr());
S
Shengliang Guan 已提交
377 378
    return -1;
  }
S
Shengliang Guan 已提交
379
  mDebug("trans:%d, used to alter user:%s", pTrans->id, pOld->user);
S
Shengliang Guan 已提交
380

381 382 383
  SSdbRaw *pCommitRaw = mndUserActionEncode(pNew);
  if (pCommitRaw == NULL || mndTransAppendCommitlog(pTrans, pCommitRaw) != 0) {
    mError("trans:%d, failed to append commit log since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
384 385 386
    mndTransDrop(pTrans);
    return -1;
  }
387
  sdbSetRawStatus(pCommitRaw, SDB_STATUS_READY);
S
Shengliang Guan 已提交
388 389 390 391 392 393 394 395 396 397 398

  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 已提交
399
static SHashObj *mndDupDbHash(SHashObj *pOld) {
S
Shengliang Guan 已提交
400 401
  SHashObj *pNew =
      taosHashInit(taosHashGetSize(pOld), taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_ENTRY_LOCK);
S
Shengliang Guan 已提交
402 403 404 405 406 407 408 409 410 411 412
  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 已提交
413
      terrno = TSDB_CODE_OUT_OF_MEMORY;
S
Shengliang Guan 已提交
414 415 416 417 418 419 420 421
      return NULL;
    }
    db = taosHashIterate(pOld, db);
  }

  return pNew;
}

S
Shengliang Guan 已提交
422 423
static int32_t mndProcessAlterUserReq(SRpcMsg *pReq) {
  SMnode       *pMnode = pReq->info.node;
S
Shengliang Guan 已提交
424 425
  SSdb         *pSdb = pMnode->pSdb;
  void         *pIter = NULL;
S
Shengliang Guan 已提交
426 427 428
  int32_t       code = -1;
  SUserObj     *pUser = NULL;
  SUserObj     *pOperUser = NULL;
S
Shengliang Guan 已提交
429
  SUserObj      newUser = {0};
S
Shengliang Guan 已提交
430 431
  SAlterUserReq alterReq = {0};

S
Shengliang Guan 已提交
432
  if (tDeserializeSAlterUserReq(pReq->pCont, pReq->contLen, &alterReq) != 0) {
S
Shengliang Guan 已提交
433
    terrno = TSDB_CODE_INVALID_MSG;
434
    goto _OVER;
S
Shengliang Guan 已提交
435
  }
S
Shengliang Guan 已提交
436

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

S
Shengliang Guan 已提交
439
  if (alterReq.user[0] == 0) {
S
Shengliang Guan 已提交
440
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
441 442 443 444 445
    goto _OVER;
  }

  if (TSDB_ALTER_USER_PASSWD == alterReq.alterType && alterReq.pass[0] == 0) {
    terrno = TSDB_CODE_MND_INVALID_PASS_FORMAT;
446
    goto _OVER;
S
Shengliang Guan 已提交
447 448
  }

S
Shengliang Guan 已提交
449
  pUser = mndAcquireUser(pMnode, alterReq.user);
S
Shengliang Guan 已提交
450 451
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
452
    goto _OVER;
S
Shengliang Guan 已提交
453 454
  }

S
Shengliang Guan 已提交
455
  pOperUser = mndAcquireUser(pMnode, pReq->conn.user);
S
Shengliang Guan 已提交
456 457
  if (pOperUser == NULL) {
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
458
    goto _OVER;
S
Shengliang Guan 已提交
459 460
  }

S
Shengliang Guan 已提交
461 462 463 464
  if (mndCheckAlterUserAuth(pOperUser, pUser, &alterReq) != 0) {
    goto _OVER;
  }

S
Shengliang Guan 已提交
465
  memcpy(&newUser, pUser, sizeof(SUserObj));
S
Shengliang Guan 已提交
466 467
  newUser.authVersion++;
  newUser.updateTime = taosGetTimestampMs();
S
Shengliang Guan 已提交
468 469

  taosRLockLatch(&pUser->lock);
S
Shengliang Guan 已提交
470 471
  newUser.readDbs = mndDupDbHash(pUser->readDbs);
  newUser.writeDbs = mndDupDbHash(pUser->writeDbs);
S
Shengliang Guan 已提交
472 473
  taosRUnLockLatch(&pUser->lock);

S
Shengliang Guan 已提交
474
  if (newUser.readDbs == NULL || newUser.writeDbs == NULL) {
475
    goto _OVER;
S
Shengliang Guan 已提交
476 477 478 479 480
  }

  if (alterReq.alterType == TSDB_ALTER_USER_PASSWD) {
    char pass[TSDB_PASSWORD_LEN + 1] = {0};
    taosEncryptPass_c((uint8_t *)alterReq.pass, strlen(alterReq.pass), pass);
481
    memcpy(newUser.pass, pass, TSDB_PASSWORD_LEN);
S
Shengliang Guan 已提交
482 483 484
  }

  if (alterReq.alterType == TSDB_ALTER_USER_SUPERUSER) {
S
Shengliang Guan 已提交
485
    newUser.superUser = alterReq.superUser;
S
Shengliang Guan 已提交
486 487 488
  }

  if (alterReq.alterType == TSDB_ALTER_USER_ADD_READ_DB || alterReq.alterType == TSDB_ALTER_USER_ADD_ALL_DB) {
489
    if (strcmp(alterReq.dbname, "1.*") != 0) {
S
Shengliang Guan 已提交
490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508
      int32_t len = strlen(alterReq.dbname) + 1;
      SDbObj *pDb = mndAcquireDb(pMnode, alterReq.dbname);
      if (pDb == NULL) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
      if (taosHashPut(newUser.readDbs, alterReq.dbname, len, alterReq.dbname, TSDB_DB_FNAME_LEN) != 0) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
    } else {
      while (1) {
        SDbObj *pDb = NULL;
        pIter = sdbFetch(pSdb, SDB_DB, pIter, (void **)&pDb);
        if (pIter == NULL) break;
        int32_t len = strlen(pDb->name) + 1;
        taosHashPut(newUser.readDbs, pDb->name, len, pDb->name, TSDB_DB_FNAME_LEN);
        sdbRelease(pSdb, pDb);
      }
S
Shengliang Guan 已提交
509
    }
S
Shengliang Guan 已提交
510 511 512
  }

  if (alterReq.alterType == TSDB_ALTER_USER_ADD_WRITE_DB || alterReq.alterType == TSDB_ALTER_USER_ADD_ALL_DB) {
513
    if (strcmp(alterReq.dbname, "1.*") != 0) {
S
Shengliang Guan 已提交
514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532
      int32_t len = strlen(alterReq.dbname) + 1;
      SDbObj *pDb = mndAcquireDb(pMnode, alterReq.dbname);
      if (pDb == NULL) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
      if (taosHashPut(newUser.writeDbs, alterReq.dbname, len, alterReq.dbname, TSDB_DB_FNAME_LEN) != 0) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
    } else {
      while (1) {
        SDbObj *pDb = NULL;
        pIter = sdbFetch(pSdb, SDB_DB, pIter, (void **)&pDb);
        if (pIter == NULL) break;
        int32_t len = strlen(pDb->name) + 1;
        taosHashPut(newUser.writeDbs, pDb->name, len, pDb->name, TSDB_DB_FNAME_LEN);
        sdbRelease(pSdb, pDb);
      }
S
Shengliang Guan 已提交
533 534 535
    }
  }

S
Shengliang Guan 已提交
536
  if (alterReq.alterType == TSDB_ALTER_USER_REMOVE_READ_DB || alterReq.alterType == TSDB_ALTER_USER_REMOVE_ALL_DB) {
S
Shengliang 已提交
537
    if (strcmp(alterReq.dbname, "1.*") != 0) {
S
Shengliang Guan 已提交
538 539 540 541 542 543 544 545 546 547 548
      int32_t len = strlen(alterReq.dbname) + 1;
      SDbObj *pDb = mndAcquireDb(pMnode, alterReq.dbname);
      if (pDb == NULL) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
      taosHashRemove(newUser.readDbs, alterReq.dbname, len);
    } else {
      taosHashClear(newUser.readDbs);
    }
  }
S
Shengliang Guan 已提交
549

S
Shengliang Guan 已提交
550
  if (alterReq.alterType == TSDB_ALTER_USER_REMOVE_WRITE_DB || alterReq.alterType == TSDB_ALTER_USER_REMOVE_ALL_DB) {
S
Shengliang 已提交
551
    if (strcmp(alterReq.dbname, "1.*") != 0) {
S
Shengliang Guan 已提交
552 553 554 555 556 557 558 559 560 561
      int32_t len = strlen(alterReq.dbname) + 1;
      SDbObj *pDb = mndAcquireDb(pMnode, alterReq.dbname);
      if (pDb == NULL) {
        mndReleaseDb(pMnode, pDb);
        goto _OVER;
      }
      taosHashRemove(newUser.writeDbs, alterReq.dbname, len);
    } else {
      taosHashClear(newUser.writeDbs);
    }
S
Shengliang Guan 已提交
562 563
  }

S
Shengliang Guan 已提交
564
  code = mndAlterUser(pMnode, pUser, &newUser, pReq);
S
Shengliang Guan 已提交
565
  if (code == 0) code = TSDB_CODE_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
566

567
_OVER:
S
Shengliang Guan 已提交
568
  if (code != 0 && code != TSDB_CODE_ACTION_IN_PROGRESS) {
S
Shengliang Guan 已提交
569
    mError("user:%s, failed to alter since %s", alterReq.user, terrstr());
S
Shengliang Guan 已提交
570 571
  }

S
Shengliang Guan 已提交
572 573
  mndReleaseUser(pMnode, pOperUser);
  mndReleaseUser(pMnode, pUser);
S
Shengliang Guan 已提交
574 575
  taosHashCleanup(newUser.writeDbs);
  taosHashCleanup(newUser.readDbs);
S
Shengliang Guan 已提交
576 577

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

S
Shengliang Guan 已提交
580 581
static int32_t mndDropUser(SMnode *pMnode, SRpcMsg *pReq, SUserObj *pUser) {
  STrans *pTrans = mndTransCreate(pMnode, TRN_POLICY_ROLLBACK, TRN_TYPE_DROP_USER, pReq);
S
Shengliang Guan 已提交
582 583 584 585 586 587
  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);

588 589 590
  SSdbRaw *pCommitRaw = mndUserActionEncode(pUser);
  if (pCommitRaw == NULL || mndTransAppendCommitlog(pTrans, pCommitRaw) != 0) {
    mError("trans:%d, failed to append commit log since %s", pTrans->id, terrstr());
S
Shengliang Guan 已提交
591 592 593
    mndTransDrop(pTrans);
    return -1;
  }
594
  sdbSetRawStatus(pCommitRaw, SDB_STATUS_DROPPED);
S
Shengliang Guan 已提交
595 596 597 598 599 600 601 602 603 604 605

  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 已提交
606 607
static int32_t mndProcessDropUserReq(SRpcMsg *pReq) {
  SMnode      *pMnode = pReq->info.node;
S
Shengliang Guan 已提交
608 609 610 611 612
  int32_t      code = -1;
  SUserObj    *pUser = NULL;
  SUserObj    *pOperUser = NULL;
  SDropUserReq dropReq = {0};

S
Shengliang Guan 已提交
613
  if (tDeserializeSDropUserReq(pReq->pCont, pReq->contLen, &dropReq) != 0) {
S
Shengliang Guan 已提交
614
    terrno = TSDB_CODE_INVALID_MSG;
615
    goto _OVER;
S
Shengliang Guan 已提交
616
  }
S
Shengliang Guan 已提交
617

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

S
Shengliang Guan 已提交
620
  if (dropReq.user[0] == 0) {
S
Shengliang Guan 已提交
621
    terrno = TSDB_CODE_MND_INVALID_USER_FORMAT;
622
    goto _OVER;
S
Shengliang Guan 已提交
623 624
  }

S
Shengliang Guan 已提交
625
  pUser = mndAcquireUser(pMnode, dropReq.user);
S
Shengliang Guan 已提交
626 627
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
628
    goto _OVER;
S
Shengliang Guan 已提交
629 630
  }

S
Shengliang Guan 已提交
631
  pOperUser = mndAcquireUser(pMnode, pReq->conn.user);
S
Shengliang Guan 已提交
632 633
  if (pOperUser == NULL) {
    terrno = TSDB_CODE_MND_NO_USER_FROM_CONN;
634
    goto _OVER;
S
Shengliang Guan 已提交
635 636
  }

S
Shengliang Guan 已提交
637
  if (mndCheckDropUserAuth(pOperUser) != 0) {
638
    goto _OVER;
S
Shengliang Guan 已提交
639 640
  }

S
Shengliang Guan 已提交
641
  code = mndDropUser(pMnode, pReq, pUser);
S
Shengliang Guan 已提交
642
  if (code == 0) code = TSDB_CODE_ACTION_IN_PROGRESS;
S
Shengliang Guan 已提交
643

644
_OVER:
S
Shengliang Guan 已提交
645
  if (code != 0 && code != TSDB_CODE_ACTION_IN_PROGRESS) {
S
Shengliang Guan 已提交
646
    mError("user:%s, failed to drop since %s", dropReq.user, terrstr());
S
Shengliang Guan 已提交
647 648
  }

S
Shengliang Guan 已提交
649 650 651 652
  mndReleaseUser(pMnode, pOperUser);
  mndReleaseUser(pMnode, pUser);

  return code;
S
Shengliang Guan 已提交
653 654
}

655
static int32_t mndSetUserAuthRsp(SMnode *pMnode, SUserObj *pUser, SGetUserAuthRsp *pRsp) {
D
dapan 已提交
656 657 658
  memcpy(pRsp->user, pUser->user, TSDB_USER_LEN);
  pRsp->superAuth = pUser->superUser;
  pRsp->version = pUser->authVersion;
659
  taosRLockLatch(&pUser->lock);
D
dapan 已提交
660 661
  pRsp->readDbs = mndDupDbHash(pUser->readDbs);
  pRsp->writeDbs = mndDupDbHash(pUser->writeDbs);
662
  taosRUnLockLatch(&pUser->lock);
D
dapan 已提交
663 664 665 666 667
  pRsp->createdDbs = taosHashInit(4, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BINARY), true, HASH_NO_LOCK);
  if (NULL == pRsp->createdDbs) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    return -1;
  }
668

D
dapan 已提交
669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686
  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(pRsp->createdDbs, pDb->name, len, pDb->name, len);
    }

    sdbRelease(pSdb, pDb);
  }

  return 0;
}

S
Shengliang Guan 已提交
687 688
static int32_t mndProcessGetUserAuthReq(SRpcMsg *pReq) {
  SMnode         *pMnode = pReq->info.node;
S
Shengliang Guan 已提交
689 690 691 692 693
  int32_t         code = -1;
  SUserObj       *pUser = NULL;
  SGetUserAuthReq authReq = {0};
  SGetUserAuthRsp authRsp = {0};

S
Shengliang Guan 已提交
694
  if (tDeserializeSGetUserAuthReq(pReq->pCont, pReq->contLen, &authReq) != 0) {
S
Shengliang Guan 已提交
695
    terrno = TSDB_CODE_INVALID_MSG;
696
    goto _OVER;
S
Shengliang Guan 已提交
697
  }
S
Shengliang Guan 已提交
698 699 700 701 702 703

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

  pUser = mndAcquireUser(pMnode, authReq.user);
  if (pUser == NULL) {
    terrno = TSDB_CODE_MND_USER_NOT_EXIST;
704
    goto _OVER;
S
Shengliang Guan 已提交
705 706
  }

D
dapan 已提交
707 708 709
  code = mndSetUserAuthRsp(pMnode, pUser, &authRsp);
  if (code) {
    goto _OVER;
S
Shengliang Guan 已提交
710 711
  }

S
Shengliang Guan 已提交
712
  int32_t contLen = tSerializeSGetUserAuthRsp(NULL, 0, &authRsp);
S
Shengliang Guan 已提交
713 714 715
  void   *pRsp = rpcMallocCont(contLen);
  if (pRsp == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
716
    goto _OVER;
S
Shengliang Guan 已提交
717 718
  }

S
Shengliang Guan 已提交
719
  tSerializeSGetUserAuthRsp(pRsp, contLen, &authRsp);
S
Shengliang Guan 已提交
720

S
Shengliang Guan 已提交
721 722
  pReq->info.rsp = pRsp;
  pReq->info.rspLen = contLen;
S
Shengliang Guan 已提交
723 724
  code = 0;

725
_OVER:
726

S
Shengliang Guan 已提交
727
  mndReleaseUser(pMnode, pUser);
S
Shengliang Guan 已提交
728
  tFreeSGetUserAuthRsp(&authRsp);
S
Shengliang Guan 已提交
729 730 731 732

  return code;
}

S
Shengliang Guan 已提交
733 734
static int32_t mndRetrieveUsers(SRpcMsg *pReq, SShowObj *pShow, SSDataBlock *pBlock, int32_t rows) {
  SMnode   *pMnode = pReq->info.node;
S
Shengliang Guan 已提交
735 736 737 738 739 740 741 742 743 744 745
  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;
746
    SColumnInfoData *pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
747 748

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

751
    colDataAppend(pColInfo, numOfRows, (const char *)name, false);
752

wafwerar's avatar
wafwerar 已提交
753 754
    cols++;
    pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
755

756 757
    const char *src = pUser->superUser ? "super" : "normal";
    char        b[10 + VARSTR_HEADER_SIZE] = {0};
758
    STR_WITH_SIZE_TO_VARSTR(b, src, strlen(src));
759
    colDataAppend(pColInfo, numOfRows, (const char *)b, false);
760

wafwerar's avatar
wafwerar 已提交
761 762
    cols++;
    pColInfo = taosArrayGet(pBlock->pDataBlock, cols);
763
    colDataAppend(pColInfo, numOfRows, (const char *)&pUser->createdTime, false);
S
Shengliang Guan 已提交
764 765 766 767 768

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

769
  pShow->numOfRows += numOfRows;
S
Shengliang Guan 已提交
770 771 772 773 774 775
  return numOfRows;
}

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

778 779
int32_t mndValidateUserAuthInfo(SMnode *pMnode, SUserAuthVersion *pUsers, int32_t numOfUses, void **ppRsp,
                                int32_t *pRspLen) {
D
dapan 已提交
780 781 782 783 784 785 786 787 788 789 790 791 792 793 794
  SUserAuthBatchRsp batchRsp = {0};
  batchRsp.pArray = taosArrayInit(numOfUses, sizeof(SGetUserAuthRsp));
  if (batchRsp.pArray == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    return -1;
  }

  int32_t code = 0;
  for (int32_t i = 0; i < numOfUses; ++i) {
    SUserObj *pUser = mndAcquireUser(pMnode, pUsers[i].user);
    if (pUser == NULL) {
      mError("user:%s, failed to auth user since %s", pUsers[i].user, terrstr());
      continue;
    }

D
dapan1121 已提交
795
    pUsers[i].version = ntohl(pUsers[i].version);
D
dapan 已提交
796 797 798 799
    if (pUser->authVersion <= pUsers[i].version) {
      mndReleaseUser(pMnode, pUser);
      continue;
    }
800

D
dapan 已提交
801 802 803 804 805 806 807 808 809 810 811 812 813 814 815
    SGetUserAuthRsp rsp = {0};
    code = mndSetUserAuthRsp(pMnode, pUser, &rsp);
    if (code) {
      mndReleaseUser(pMnode, pUser);
      tFreeSGetUserAuthRsp(&rsp);
      goto _OVER;
    }

    taosArrayPush(batchRsp.pArray, &rsp);
    mndReleaseUser(pMnode, pUser);
  }

  if (taosArrayGetSize(batchRsp.pArray) <= 0) {
    *ppRsp = NULL;
    *pRspLen = 0;
816

D
dapan 已提交
817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839
    tFreeSUserAuthBatchRsp(&batchRsp);
    return 0;
  }

  int32_t rspLen = tSerializeSUserAuthBatchRsp(NULL, 0, &batchRsp);
  void   *pRsp = taosMemoryMalloc(rspLen);
  if (pRsp == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    tFreeSUserAuthBatchRsp(&batchRsp);
    return -1;
  }
  tSerializeSUserAuthBatchRsp(pRsp, rspLen, &batchRsp);

  *ppRsp = pRsp;
  *pRspLen = rspLen;

  tFreeSUserAuthBatchRsp(&batchRsp);
  return 0;

_OVER:

  *ppRsp = NULL;
  *pRspLen = 0;
840

D
dapan 已提交
841 842 843
  tFreeSUserAuthBatchRsp(&batchRsp);
  return code;
}