appwrite-utils-cli 0.0.286 → 0.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +122 -96
- package/dist/collections/attributes.d.ts +4 -0
- package/dist/collections/attributes.js +224 -0
- package/dist/collections/indexes.d.ts +4 -0
- package/dist/collections/indexes.js +27 -0
- package/dist/collections/methods.d.ts +16 -0
- package/dist/collections/methods.js +216 -0
- package/dist/databases/methods.d.ts +6 -0
- package/dist/databases/methods.js +33 -0
- package/dist/interactiveCLI.d.ts +19 -0
- package/dist/interactiveCLI.js +555 -0
- package/dist/main.js +227 -62
- package/dist/migrations/afterImportActions.js +37 -40
- package/dist/migrations/appwriteToX.d.ts +26 -25
- package/dist/migrations/appwriteToX.js +42 -6
- package/dist/migrations/attributes.js +21 -20
- package/dist/migrations/backup.d.ts +93 -87
- package/dist/migrations/collections.d.ts +6 -0
- package/dist/migrations/collections.js +149 -20
- package/dist/migrations/converters.d.ts +2 -18
- package/dist/migrations/converters.js +13 -2
- package/dist/migrations/dataLoader.d.ts +276 -161
- package/dist/migrations/dataLoader.js +535 -292
- package/dist/migrations/databases.js +8 -2
- package/dist/migrations/helper.d.ts +3 -0
- package/dist/migrations/helper.js +21 -0
- package/dist/migrations/importController.d.ts +5 -2
- package/dist/migrations/importController.js +125 -88
- package/dist/migrations/importDataActions.d.ts +9 -1
- package/dist/migrations/importDataActions.js +15 -3
- package/dist/migrations/indexes.js +3 -2
- package/dist/migrations/logging.js +20 -8
- package/dist/migrations/migrationHelper.d.ts +9 -4
- package/dist/migrations/migrationHelper.js +6 -5
- package/dist/migrations/openapi.d.ts +1 -1
- package/dist/migrations/openapi.js +33 -18
- package/dist/migrations/queue.js +3 -2
- package/dist/migrations/relationships.d.ts +2 -2
- package/dist/migrations/schemaStrings.js +53 -41
- package/dist/migrations/setupDatabase.d.ts +2 -4
- package/dist/migrations/setupDatabase.js +24 -105
- package/dist/migrations/storage.d.ts +3 -1
- package/dist/migrations/storage.js +110 -16
- package/dist/migrations/transfer.d.ts +30 -0
- package/dist/migrations/transfer.js +337 -0
- package/dist/migrations/users.d.ts +2 -1
- package/dist/migrations/users.js +78 -43
- package/dist/schemas/authUser.d.ts +2 -2
- package/dist/storage/methods.d.ts +15 -0
- package/dist/storage/methods.js +207 -0
- package/dist/storage/schemas.d.ts +687 -0
- package/dist/storage/schemas.js +175 -0
- package/dist/utils/getClientFromConfig.d.ts +4 -0
- package/dist/utils/getClientFromConfig.js +16 -0
- package/dist/utils/helperFunctions.d.ts +11 -1
- package/dist/utils/helperFunctions.js +38 -0
- package/dist/utils/retryFailedPromises.d.ts +2 -0
- package/dist/utils/retryFailedPromises.js +21 -0
- package/dist/utils/schemaStrings.d.ts +13 -0
- package/dist/utils/schemaStrings.js +403 -0
- package/dist/utils/setupFiles.js +110 -61
- package/dist/utilsController.d.ts +40 -22
- package/dist/utilsController.js +164 -84
- package/package.json +13 -15
- package/src/collections/attributes.ts +483 -0
- package/src/collections/indexes.ts +53 -0
- package/src/collections/methods.ts +331 -0
- package/src/databases/methods.ts +47 -0
- package/src/init.ts +64 -64
- package/src/interactiveCLI.ts +767 -0
- package/src/main.ts +292 -83
- package/src/migrations/afterImportActions.ts +553 -490
- package/src/migrations/appwriteToX.ts +237 -174
- package/src/migrations/attributes.ts +483 -422
- package/src/migrations/backup.ts +205 -205
- package/src/migrations/collections.ts +545 -300
- package/src/migrations/converters.ts +161 -150
- package/src/migrations/dataLoader.ts +1615 -1304
- package/src/migrations/databases.ts +44 -25
- package/src/migrations/dbHelpers.ts +92 -92
- package/src/migrations/helper.ts +40 -0
- package/src/migrations/importController.ts +448 -384
- package/src/migrations/importDataActions.ts +315 -307
- package/src/migrations/indexes.ts +40 -37
- package/src/migrations/logging.ts +29 -16
- package/src/migrations/migrationHelper.ts +207 -201
- package/src/migrations/openapi.ts +83 -70
- package/src/migrations/queue.ts +118 -119
- package/src/migrations/relationships.ts +324 -324
- package/src/migrations/schemaStrings.ts +472 -460
- package/src/migrations/setupDatabase.ts +118 -219
- package/src/migrations/storage.ts +538 -358
- package/src/migrations/transfer.ts +608 -0
- package/src/migrations/users.ts +362 -285
- package/src/migrations/validationRules.ts +63 -63
- package/src/schemas/authUser.ts +23 -23
- package/src/setup.ts +8 -8
- package/src/storage/methods.ts +371 -0
- package/src/storage/schemas.ts +205 -0
- package/src/types.ts +9 -9
- package/src/utils/getClientFromConfig.ts +17 -0
- package/src/utils/helperFunctions.ts +181 -127
- package/src/utils/index.ts +2 -2
- package/src/utils/loadConfigs.ts +59 -59
- package/src/utils/retryFailedPromises.ts +27 -0
- package/src/utils/schemaStrings.ts +473 -0
- package/src/utils/setupFiles.ts +228 -182
- package/src/utilsController.ts +325 -194
- package/tsconfig.json +37 -37
package/src/migrations/users.ts
CHANGED
@@ -1,285 +1,362 @@
|
|
1
|
-
import type { AppwriteConfig, ConfigCollection } from "appwrite-utils";
|
2
|
-
import {
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
import {
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
}
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
usersDeleted
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
}
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
item.
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
item.name || undefined
|
120
|
-
);
|
121
|
-
if (item.labels) {
|
122
|
-
await this.users.updateLabels(user.$id, item.labels);
|
123
|
-
}
|
124
|
-
if (item.prefs) {
|
125
|
-
await this.users.updatePrefs(user.$id, item.prefs);
|
126
|
-
}
|
127
|
-
return user;
|
128
|
-
} catch (e) {
|
129
|
-
if (e instanceof
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
userToReturn
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
)
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
}
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
}
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
|
284
|
-
|
285
|
-
}
|
1
|
+
import type { AppwriteConfig, ConfigCollection } from "appwrite-utils";
|
2
|
+
import {
|
3
|
+
AppwriteException,
|
4
|
+
Databases,
|
5
|
+
ID,
|
6
|
+
Query,
|
7
|
+
Users,
|
8
|
+
type Models,
|
9
|
+
} from "node-appwrite";
|
10
|
+
import {
|
11
|
+
AuthUserSchema,
|
12
|
+
type AuthUser,
|
13
|
+
type AuthUserCreate,
|
14
|
+
} from "../schemas/authUser.js";
|
15
|
+
import _ from "lodash";
|
16
|
+
import { logger } from "./logging.js";
|
17
|
+
import { splitIntoBatches } from "./migrationHelper.js";
|
18
|
+
import {
|
19
|
+
getAppwriteClient,
|
20
|
+
tryAwaitWithRetry,
|
21
|
+
} from "../utils/helperFunctions.js";
|
22
|
+
|
23
|
+
export class UsersController {
|
24
|
+
private config: AppwriteConfig;
|
25
|
+
private users: Users;
|
26
|
+
static userFields = [
|
27
|
+
"email",
|
28
|
+
"name",
|
29
|
+
"password",
|
30
|
+
"phone",
|
31
|
+
"labels",
|
32
|
+
"prefs",
|
33
|
+
"userId",
|
34
|
+
"$createdAt",
|
35
|
+
"$updatedAt",
|
36
|
+
];
|
37
|
+
|
38
|
+
constructor(config: AppwriteConfig, db: Databases) {
|
39
|
+
this.config = config;
|
40
|
+
this.users = new Users(this.config.appwriteClient!);
|
41
|
+
}
|
42
|
+
|
43
|
+
async wipeUsers() {
|
44
|
+
const allUsers = await this.getAllUsers();
|
45
|
+
console.log("Deleting all users...");
|
46
|
+
|
47
|
+
const createBatches = (finalData: any[], batchSize: number) => {
|
48
|
+
const finalBatches: any[][] = [];
|
49
|
+
for (let i = 0; i < finalData.length; i += batchSize) {
|
50
|
+
finalBatches.push(finalData.slice(i, i + batchSize));
|
51
|
+
}
|
52
|
+
return finalBatches;
|
53
|
+
};
|
54
|
+
|
55
|
+
let usersDeleted = 0;
|
56
|
+
const batchedUserPromises = createBatches(allUsers, 50); // Batch size of 10
|
57
|
+
|
58
|
+
for (const batch of batchedUserPromises) {
|
59
|
+
console.log(`Deleting ${batch.length} users...`);
|
60
|
+
await Promise.all(
|
61
|
+
batch.map((user) =>
|
62
|
+
tryAwaitWithRetry(async () => await this.users.delete(user.$id))
|
63
|
+
)
|
64
|
+
);
|
65
|
+
usersDeleted += batch.length;
|
66
|
+
if (usersDeleted % 100 === 0) {
|
67
|
+
console.log(`Deleted ${usersDeleted} users...`);
|
68
|
+
}
|
69
|
+
}
|
70
|
+
}
|
71
|
+
|
72
|
+
async getAllUsers() {
|
73
|
+
const allUsers: Models.User<Models.Preferences>[] = [];
|
74
|
+
const users = await tryAwaitWithRetry(
|
75
|
+
async () => await this.users.list([Query.limit(200)])
|
76
|
+
);
|
77
|
+
if (users.users.length === 0) {
|
78
|
+
return [];
|
79
|
+
}
|
80
|
+
if (users.users.length === 200) {
|
81
|
+
let lastDocumentId = users.users[users.users.length - 1].$id;
|
82
|
+
allUsers.push(...users.users);
|
83
|
+
while (lastDocumentId) {
|
84
|
+
const moreUsers = await tryAwaitWithRetry(
|
85
|
+
async () =>
|
86
|
+
await this.users.list([
|
87
|
+
Query.limit(200),
|
88
|
+
Query.cursorAfter(lastDocumentId),
|
89
|
+
])
|
90
|
+
);
|
91
|
+
allUsers.push(...moreUsers.users);
|
92
|
+
if (moreUsers.users.length < 200) {
|
93
|
+
break;
|
94
|
+
}
|
95
|
+
lastDocumentId = moreUsers.users[moreUsers.users.length - 1].$id;
|
96
|
+
}
|
97
|
+
} else {
|
98
|
+
allUsers.push(...users.users);
|
99
|
+
}
|
100
|
+
return allUsers;
|
101
|
+
}
|
102
|
+
|
103
|
+
async createUsersAndReturn(items: AuthUserCreate[]) {
|
104
|
+
const users = await Promise.all(
|
105
|
+
items.map((item) => this.createUserAndReturn(item))
|
106
|
+
);
|
107
|
+
return users;
|
108
|
+
}
|
109
|
+
|
110
|
+
async createUserAndReturn(item: AuthUserCreate, numAttempts?: number) {
|
111
|
+
try {
|
112
|
+
const user = await this.users.create(
|
113
|
+
item.userId || ID.unique(),
|
114
|
+
item.email || undefined,
|
115
|
+
item.phone && item.phone.length < 15 && item.phone.startsWith("+")
|
116
|
+
? item.phone
|
117
|
+
: undefined,
|
118
|
+
`changeMe${item.email?.toLowerCase()}` || `changeMePlease`,
|
119
|
+
item.name || undefined
|
120
|
+
);
|
121
|
+
if (item.labels) {
|
122
|
+
await this.users.updateLabels(user.$id, item.labels);
|
123
|
+
}
|
124
|
+
if (item.prefs) {
|
125
|
+
await this.users.updatePrefs(user.$id, item.prefs);
|
126
|
+
}
|
127
|
+
return user;
|
128
|
+
} catch (e) {
|
129
|
+
if (e instanceof AppwriteException) {
|
130
|
+
if (
|
131
|
+
e.message.toLowerCase().includes("fetch failed") ||
|
132
|
+
e.message.toLowerCase().includes("server error")
|
133
|
+
) {
|
134
|
+
const numberOfAttempts = numAttempts || 0;
|
135
|
+
if (numberOfAttempts > 5) {
|
136
|
+
throw e;
|
137
|
+
}
|
138
|
+
const user: Models.User<Models.Preferences> =
|
139
|
+
await this.createUserAndReturn(item, numberOfAttempts + 1);
|
140
|
+
return user;
|
141
|
+
}
|
142
|
+
}
|
143
|
+
if (e instanceof Error) {
|
144
|
+
logger.error("FAILED CREATING USER: ", e.message, item);
|
145
|
+
}
|
146
|
+
console.log("FAILED CREATING USER: ", e, item);
|
147
|
+
throw e;
|
148
|
+
}
|
149
|
+
}
|
150
|
+
|
151
|
+
async createAndCheckForUserAndReturn(item: AuthUserCreate) {
|
152
|
+
let userToReturn: Models.User<Models.Preferences> | undefined = undefined;
|
153
|
+
try {
|
154
|
+
// Attempt to find an existing user by email or phone.
|
155
|
+
let foundUsers: Models.User<Models.Preferences>[] = [];
|
156
|
+
if (item.email) {
|
157
|
+
const foundUsersByEmail = await this.users.list([
|
158
|
+
Query.equal("email", item.email),
|
159
|
+
]);
|
160
|
+
foundUsers = foundUsersByEmail.users;
|
161
|
+
}
|
162
|
+
if (item.phone) {
|
163
|
+
const foundUsersByPhone = await this.users.list([
|
164
|
+
Query.equal("phone", item.phone),
|
165
|
+
]);
|
166
|
+
foundUsers = foundUsers.length
|
167
|
+
? foundUsers.concat(foundUsersByPhone.users)
|
168
|
+
: foundUsersByPhone.users;
|
169
|
+
}
|
170
|
+
|
171
|
+
userToReturn = foundUsers[0] || undefined;
|
172
|
+
|
173
|
+
if (!userToReturn) {
|
174
|
+
userToReturn = await this.users.create(
|
175
|
+
item.userId || ID.unique(),
|
176
|
+
item.email || undefined,
|
177
|
+
item.phone && item.phone.length < 15 && item.phone.startsWith("+")
|
178
|
+
? item.phone
|
179
|
+
: undefined,
|
180
|
+
item.password?.toLowerCase() ||
|
181
|
+
`changeMe${item.email?.toLowerCase()}` ||
|
182
|
+
`changeMePlease`,
|
183
|
+
item.name || undefined
|
184
|
+
);
|
185
|
+
} else {
|
186
|
+
// Update user details as necessary, ensuring email uniqueness if attempting an update.
|
187
|
+
if (
|
188
|
+
item.email &&
|
189
|
+
item.email !== userToReturn.email &&
|
190
|
+
!_.isEmpty(item.email) &&
|
191
|
+
!_.isUndefined(item.email)
|
192
|
+
) {
|
193
|
+
const emailExists = await this.users.list([
|
194
|
+
Query.equal("email", item.email),
|
195
|
+
]);
|
196
|
+
if (emailExists.users.length === 0) {
|
197
|
+
userToReturn = await this.users.updateEmail(
|
198
|
+
userToReturn.$id,
|
199
|
+
item.email
|
200
|
+
);
|
201
|
+
} else {
|
202
|
+
console.log("Email update skipped: Email already exists.");
|
203
|
+
}
|
204
|
+
}
|
205
|
+
if (item.password) {
|
206
|
+
userToReturn = await this.users.updatePassword(
|
207
|
+
userToReturn.$id,
|
208
|
+
item.password.toLowerCase()
|
209
|
+
);
|
210
|
+
}
|
211
|
+
if (item.name && item.name !== userToReturn.name) {
|
212
|
+
userToReturn = await this.users.updateName(
|
213
|
+
userToReturn.$id,
|
214
|
+
item.name
|
215
|
+
);
|
216
|
+
}
|
217
|
+
if (
|
218
|
+
item.phone &&
|
219
|
+
item.phone !== userToReturn.phone &&
|
220
|
+
item.phone.length < 15 &&
|
221
|
+
item.phone.startsWith("+") &&
|
222
|
+
(_.isUndefined(userToReturn.phone) || _.isEmpty(userToReturn.phone))
|
223
|
+
) {
|
224
|
+
const userFoundWithPhone = await this.users.list([
|
225
|
+
Query.equal("phone", item.phone),
|
226
|
+
]);
|
227
|
+
if (userFoundWithPhone.total === 0) {
|
228
|
+
userToReturn = await this.users.updatePhone(
|
229
|
+
userToReturn.$id,
|
230
|
+
item.phone
|
231
|
+
);
|
232
|
+
}
|
233
|
+
}
|
234
|
+
}
|
235
|
+
if (item.$createdAt && item.$updatedAt) {
|
236
|
+
console.log(
|
237
|
+
"$createdAt and $updatedAt are not yet supported, sorry about that!"
|
238
|
+
);
|
239
|
+
}
|
240
|
+
if (item.labels && item.labels.length) {
|
241
|
+
userToReturn = await this.users.updateLabels(
|
242
|
+
userToReturn.$id,
|
243
|
+
item.labels
|
244
|
+
);
|
245
|
+
}
|
246
|
+
if (item.prefs && Object.keys(item.prefs).length) {
|
247
|
+
await this.users.updatePrefs(userToReturn.$id, item.prefs);
|
248
|
+
userToReturn.prefs = item.prefs;
|
249
|
+
}
|
250
|
+
return userToReturn;
|
251
|
+
} catch (error) {
|
252
|
+
return userToReturn;
|
253
|
+
}
|
254
|
+
}
|
255
|
+
|
256
|
+
async getUserIdByEmailOrPhone(email?: string, phone?: string) {
|
257
|
+
if (!email && !phone) {
|
258
|
+
return undefined;
|
259
|
+
}
|
260
|
+
if (email && phone) {
|
261
|
+
const foundUsersByEmail = await this.users.list([
|
262
|
+
// @ts-ignore
|
263
|
+
Query.or([Query.equal("email", email), Query.equal("phone", phone)]),
|
264
|
+
]);
|
265
|
+
if (foundUsersByEmail.users.length > 0) {
|
266
|
+
return foundUsersByEmail.users[0]?.$id;
|
267
|
+
}
|
268
|
+
} else if (email) {
|
269
|
+
const foundUsersByEmail = await this.users.list([
|
270
|
+
Query.equal("email", email),
|
271
|
+
]);
|
272
|
+
if (foundUsersByEmail.users.length > 0) {
|
273
|
+
return foundUsersByEmail.users[0]?.$id;
|
274
|
+
} else {
|
275
|
+
if (!phone) {
|
276
|
+
return undefined;
|
277
|
+
} else {
|
278
|
+
const foundUsersByPhone = await this.users.list([
|
279
|
+
Query.equal("phone", phone),
|
280
|
+
]);
|
281
|
+
if (foundUsersByPhone.users.length > 0) {
|
282
|
+
return foundUsersByPhone.users[0]?.$id;
|
283
|
+
} else {
|
284
|
+
return undefined;
|
285
|
+
}
|
286
|
+
}
|
287
|
+
}
|
288
|
+
}
|
289
|
+
if (phone) {
|
290
|
+
const foundUsersByPhone = await this.users.list([
|
291
|
+
Query.equal("phone", phone),
|
292
|
+
]);
|
293
|
+
if (foundUsersByPhone.users.length > 0) {
|
294
|
+
return foundUsersByPhone.users[0]?.$id;
|
295
|
+
} else {
|
296
|
+
return undefined;
|
297
|
+
}
|
298
|
+
}
|
299
|
+
}
|
300
|
+
|
301
|
+
transferUsersBetweenDbsLocalToRemote = async (
|
302
|
+
endpoint: string,
|
303
|
+
projectId: string,
|
304
|
+
apiKey: string
|
305
|
+
) => {
|
306
|
+
const localUsers = this.users;
|
307
|
+
const client = getAppwriteClient(endpoint, projectId, apiKey);
|
308
|
+
const remoteUsers = new Users(client);
|
309
|
+
|
310
|
+
let fromUsers = await localUsers.list([Query.limit(50)]);
|
311
|
+
|
312
|
+
if (fromUsers.users.length === 0) {
|
313
|
+
console.log(`No users found`);
|
314
|
+
return;
|
315
|
+
} else if (fromUsers.users.length < 50) {
|
316
|
+
console.log(`Transferring ${fromUsers.users.length} users to remote`);
|
317
|
+
const batchedPromises = fromUsers.users.map((user) => {
|
318
|
+
return tryAwaitWithRetry(async () => {
|
319
|
+
const toCreateObject: Partial<typeof user> = {
|
320
|
+
...user,
|
321
|
+
};
|
322
|
+
delete toCreateObject.$id;
|
323
|
+
delete toCreateObject.$createdAt;
|
324
|
+
delete toCreateObject.$updatedAt;
|
325
|
+
await remoteUsers.create(
|
326
|
+
user.$id,
|
327
|
+
user.email,
|
328
|
+
user.phone,
|
329
|
+
user.password,
|
330
|
+
user.name
|
331
|
+
);
|
332
|
+
});
|
333
|
+
});
|
334
|
+
await Promise.all(batchedPromises);
|
335
|
+
} else {
|
336
|
+
while (fromUsers.users.length === 50) {
|
337
|
+
fromUsers = await localUsers.list([
|
338
|
+
Query.limit(50),
|
339
|
+
Query.cursorAfter(fromUsers.users[fromUsers.users.length - 1].$id),
|
340
|
+
]);
|
341
|
+
const batchedPromises = fromUsers.users.map((user) => {
|
342
|
+
return tryAwaitWithRetry(async () => {
|
343
|
+
const toCreateObject: Partial<typeof user> = {
|
344
|
+
...user,
|
345
|
+
};
|
346
|
+
delete toCreateObject.$id;
|
347
|
+
delete toCreateObject.$createdAt;
|
348
|
+
delete toCreateObject.$updatedAt;
|
349
|
+
await remoteUsers.create(
|
350
|
+
user.$id,
|
351
|
+
user.email,
|
352
|
+
user.phone,
|
353
|
+
user.password,
|
354
|
+
user.name
|
355
|
+
);
|
356
|
+
});
|
357
|
+
});
|
358
|
+
await Promise.all(batchedPromises);
|
359
|
+
}
|
360
|
+
}
|
361
|
+
};
|
362
|
+
}
|