mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-07-02 03:02:19 +08:00
94 lines
2.7 KiB
TypeScript
94 lines
2.7 KiB
TypeScript
import { AuthConfig, BaseAuth } from '@nocobase/auth';
|
|
import { Issuer } from 'openid-client';
|
|
import { cookieName } from '../constants';
|
|
|
|
export class OIDCAuth extends BaseAuth {
|
|
constructor(config: AuthConfig) {
|
|
const { ctx } = config;
|
|
super({
|
|
...config,
|
|
userCollection: ctx.db.getCollection('users'),
|
|
});
|
|
}
|
|
|
|
getRedirectUri() {
|
|
const ctx = this.ctx;
|
|
const { http, port } = this.getOptions();
|
|
const protocol = http ? 'http' : 'https';
|
|
const host = port ? `${ctx.hostname}${port ? `:${port}` : ''}` : ctx.host;
|
|
return `${protocol}://${host}/api/oidc:redirect`;
|
|
}
|
|
|
|
getOptions() {
|
|
return this.options?.oidc || {};
|
|
}
|
|
|
|
mapField(userInfo: { [source: string]: any }) {
|
|
const { fieldMap } = this.getOptions();
|
|
if (!fieldMap) {
|
|
return userInfo;
|
|
}
|
|
fieldMap.forEach((item: { source: string; target: string }) => {
|
|
const { source, target } = item;
|
|
if (userInfo[source]) {
|
|
userInfo[target] = userInfo[source];
|
|
}
|
|
});
|
|
return userInfo;
|
|
}
|
|
|
|
async createOIDCClient() {
|
|
const { issuer, clientId, clientSecret, idTokenSignedResponseAlg } = this.getOptions();
|
|
const oidc = await Issuer.discover(issuer);
|
|
return new oidc.Client({
|
|
client_id: clientId,
|
|
client_secret: clientSecret,
|
|
id_token_signed_response_alg: idTokenSignedResponseAlg || 'RS256',
|
|
});
|
|
}
|
|
|
|
async validate() {
|
|
const ctx = this.ctx;
|
|
const {
|
|
params: { values },
|
|
} = ctx.action;
|
|
const token = ctx.cookies.get(cookieName);
|
|
const search = new URLSearchParams(values.state);
|
|
if (search.get('token') !== token) {
|
|
ctx.app.logger.warn('odic-auth: state mismatch');
|
|
return null;
|
|
}
|
|
const client = await this.createOIDCClient();
|
|
const tokens = await client.callback(this.getRedirectUri(), {
|
|
code: values.code,
|
|
iss: values.iss,
|
|
});
|
|
const userInfo: { [key: string]: any } = await client.userinfo(tokens.access_token);
|
|
const mappedUserInfo = this.mapField(userInfo);
|
|
const { nickname, name, sub, email, phone } = mappedUserInfo;
|
|
const username = nickname || name || sub;
|
|
// Compatible processing
|
|
// When email is provided, use email to find user
|
|
// If found, associate the user with the current authenticator
|
|
if (email) {
|
|
const user = await this.userRepository.findOne({
|
|
filter: { email },
|
|
});
|
|
if (user) {
|
|
await this.authenticator.addUser(user, {
|
|
through: {
|
|
uuid: sub,
|
|
},
|
|
});
|
|
return user;
|
|
}
|
|
}
|
|
|
|
return await this.authenticator.findOrCreateUser(sub, {
|
|
nickname: username,
|
|
email: email ?? null,
|
|
phone: phone ?? null,
|
|
});
|
|
}
|
|
}
|