mirror of
https://github.com/nagisa77/OpenIsle.git
synced 2026-02-06 23:21:16 +08:00
125 lines
3.5 KiB
JavaScript
125 lines
3.5 KiB
JavaScript
import { toast } from '../main'
|
||
import { setToken } from './auth'
|
||
import { registerPush } from './push'
|
||
|
||
export async function googleGetIdToken() {
|
||
const config = useRuntimeConfig()
|
||
const GOOGLE_CLIENT_ID = config.public.googleClientId
|
||
|
||
return new Promise((resolve, reject) => {
|
||
if (!window.google || !GOOGLE_CLIENT_ID) {
|
||
toast.error('Google 登录不可用, 请检查网络设置与VPN')
|
||
reject()
|
||
return
|
||
}
|
||
window.google.accounts.id.initialize({
|
||
client_id: GOOGLE_CLIENT_ID,
|
||
callback: ({ credential }) => resolve(credential),
|
||
use_fedcm: true,
|
||
})
|
||
window.google.accounts.id.prompt()
|
||
})
|
||
}
|
||
|
||
export function googleAuthorize(inviteToken = '') {
|
||
const config = useRuntimeConfig()
|
||
const GOOGLE_CLIENT_ID = config.public.googleClientId
|
||
const WEBSITE_BASE_URL = config.public.websiteBaseUrl
|
||
|
||
if (!GOOGLE_CLIENT_ID) {
|
||
toast.error('Google 登录不可用, 请检查网络设置与VPN')
|
||
return
|
||
}
|
||
|
||
const redirectUri = `${WEBSITE_BASE_URL}/google-callback`
|
||
const nonce = Math.random().toString(36).slice(2)
|
||
|
||
// 明文放在 state(推荐;Google 会原样回传)
|
||
const state = new URLSearchParams({ invite_token: inviteToken }).toString()
|
||
|
||
const url =
|
||
`https://accounts.google.com/o/oauth2/v2/auth` +
|
||
`?client_id=${encodeURIComponent(GOOGLE_CLIENT_ID)}` +
|
||
`&redirect_uri=${encodeURIComponent(redirectUri)}` +
|
||
`&response_type=id_token` +
|
||
`&scope=${encodeURIComponent('openid email profile')}` +
|
||
`&nonce=${encodeURIComponent(nonce)}` +
|
||
`&state=${encodeURIComponent(state)}`
|
||
|
||
window.location.href = url
|
||
}
|
||
|
||
export async function googleAuthWithToken(
|
||
idToken,
|
||
redirect_success,
|
||
redirect_not_approved,
|
||
options = {}, // { inviteToken?: string }
|
||
) {
|
||
try {
|
||
if (!idToken) {
|
||
toast.error('缺少 id_token')
|
||
return
|
||
}
|
||
|
||
const config = useRuntimeConfig()
|
||
const API_BASE_URL = config.public.apiBaseUrl
|
||
|
||
const payload = { idToken }
|
||
if (options && options.inviteToken) {
|
||
payload.inviteToken = options.inviteToken
|
||
}
|
||
|
||
const res = await fetch(`${API_BASE_URL}/api/auth/google`, {
|
||
method: 'POST',
|
||
headers: { 'Content-Type': 'application/json', Accept: 'application/json' },
|
||
body: JSON.stringify(payload),
|
||
})
|
||
|
||
const data = await res.json().catch(() => ({}))
|
||
|
||
if (res.ok && data && data.token) {
|
||
setToken(data.token)
|
||
toast.success('登录成功')
|
||
registerPush?.()
|
||
if (typeof redirect_success === 'function') redirect_success()
|
||
return
|
||
}
|
||
|
||
if (data && data.reason_code === 'NOT_APPROVED') {
|
||
toast.info('当前为注册审核模式,请填写注册理由')
|
||
if (typeof redirect_not_approved === 'function') redirect_not_approved(data.token)
|
||
return
|
||
}
|
||
|
||
if (data && data.reason_code === 'IS_APPROVING') {
|
||
toast.info('您的注册理由正在审批中')
|
||
if (typeof redirect_success === 'function') redirect_success()
|
||
return
|
||
}
|
||
toast.error(data?.message || '登录失败')
|
||
} catch (e) {
|
||
console.error(e)
|
||
toast.error('登录失败')
|
||
}
|
||
}
|
||
|
||
export async function googleSignIn(redirect_success, redirect_not_approved) {
|
||
try {
|
||
const token = await googleGetIdToken()
|
||
await googleAuthWithToken(token, redirect_success, redirect_not_approved)
|
||
} catch {
|
||
/* ignore */
|
||
}
|
||
}
|
||
|
||
export function loginWithGoogle() {
|
||
googleSignIn(
|
||
() => {
|
||
navigateTo('/', { replace: true })
|
||
},
|
||
(token) => {
|
||
navigateTo(`/signup-reason?token=${token}`, { replace: true })
|
||
},
|
||
)
|
||
}
|