forked from LAION-AI/Open-Assistant
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathusers.ts
105 lines (95 loc) · 3.72 KB
/
users.ts
1
2
3
4
5
6
7
8
9
10
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
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
import { Account } from "@prisma/client";
import prisma from "src/lib/prismadb";
import { AuthMethod } from "src/types/Providers";
import type { BackendUserCore } from "src/types/Users";
/**
* Returns a `BackendUserCore` that can be used for interacting with the Backend service.
*
* @param {string} id The user's web auth id.
*
*/
export const getBackendUserCore = async (id: string): Promise<BackendUserCore> => {
const user = await prisma.user.findUnique({
where: { id },
select: { id: true, name: true, accounts: true },
});
if (!user) {
throw new Error("User not found");
}
return convertToBackendUserCore(user);
};
/**
* convert a user object to a canoncial representation used for interacting with the backend
* @param user frontend user object, from prisma db
*/
export const convertToBackendUserCore = <T extends { accounts: Account[]; id: string; name: string }>(
user: T
): BackendUserCore => {
// If there are no linked accounts, just use what we have locally.
if (user.accounts.length === 0) {
return {
id: user.id,
display_name: user.name,
auth_method: "local",
};
}
// Otherwise, use the first linked account that the user created.
return {
id: user.accounts[0].providerAccountId,
display_name: user.name,
auth_method: user.accounts[0].provider as AuthMethod,
};
};
/**
* The frontend user id for discord users is saved differently from the email users
*
* this functions gets the "correct" user id for interacting with the frontend db, more specifically
* the users table, when calling `prisma.user....`
*
* Ideally, this function does not need to exist, but this might require huge migrations
*
* @param {string} id the id of the user, this field is called 'username' in the python backend's user table
* not to be confused with the user's UUID
*/
export const getFrontendUserIdForUser = async (id: string, provider: Exclude<AuthMethod, "local">) => {
const { userId } = await prisma.account.findFirst({ where: { provider: provider, providerAccountId: id } });
return userId;
};
/**
* Map backend users to their frontend ids, we might have to do a db call to
*/
export const getBatchFrontendUserIdFromBackendUser = async (users: { username: string; auth_method: AuthMethod }[]) => {
// for users signed up with email, the 'username' field from the backend is the id of the user in the frontend db
// we initialize the output for all users with the username for now:
const outputIds = users.map((user) => user.username);
// handle non-local users differently
const indicesOfNonLocalUsers = users
.map((user, idx) => ({ idx, isNonLocal: user.auth_method !== "local" }))
.filter((x) => x.isNonLocal)
.map((x) => x.idx);
if (indicesOfNonLocalUsers.length === 0) {
// no external users, save a database call
return outputIds;
}
// get the frontendUserIds for the external users
// the `username` field for external users is the id of the their account at the provider
const externalAccountIds = indicesOfNonLocalUsers.map((idx) => users[idx].username);
const externalAccounts = await prisma.account.findMany({
where: {
provider: { in: ["discord", "google"] },
providerAccountId: { in: externalAccountIds },
},
select: { userId: true, providerAccountId: true, provider: true },
});
indicesOfNonLocalUsers.forEach((userIdx) => {
// NOTE: findMany will return the values unsorted, which is why we have to 'find' here
const account = externalAccounts.find(
(a) => a.provider === users[userIdx].auth_method && a.providerAccountId === users[userIdx].username
);
// TODO check why the account is undefined
if (account) {
outputIds[userIdx] = account.userId;
}
});
return outputIds;
};