123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- import { Transaction } from 'sequelize'
- import {
- AfterDestroy,
- AfterUpdate,
- AllowNull,
- BelongsTo,
- Column,
- CreatedAt,
- ForeignKey,
- Model,
- Scopes,
- Table,
- UpdatedAt
- } from 'sequelize-typescript'
- import { TokensCache } from '@server/lib/auth/tokens-cache'
- import { MUserAccountId } from '@server/types/models'
- import { MOAuthTokenUser } from '@server/types/models/oauth/oauth-token'
- import { AttributesOnly } from '@shared/typescript-utils'
- import { logger } from '../../helpers/logger'
- import { AccountModel } from '../account/account'
- import { ActorModel } from '../actor/actor'
- import { UserModel } from '../user/user'
- import { OAuthClientModel } from './oauth-client'
- export type OAuthTokenInfo = {
- refreshToken: string
- refreshTokenExpiresAt: Date
- client: {
- id: number
- }
- user: MUserAccountId
- token: MOAuthTokenUser
- }
- enum ScopeNames {
- WITH_USER = 'WITH_USER'
- }
- @Scopes(() => ({
- [ScopeNames.WITH_USER]: {
- include: [
- {
- model: UserModel.unscoped(),
- required: true,
- include: [
- {
- attributes: [ 'id' ],
- model: AccountModel.unscoped(),
- required: true,
- include: [
- {
- attributes: [ 'id', 'url' ],
- model: ActorModel.unscoped(),
- required: true
- }
- ]
- }
- ]
- }
- ]
- }
- }))
- @Table({
- tableName: 'oAuthToken',
- indexes: [
- {
- fields: [ 'refreshToken' ],
- unique: true
- },
- {
- fields: [ 'accessToken' ],
- unique: true
- },
- {
- fields: [ 'userId' ]
- },
- {
- fields: [ 'oAuthClientId' ]
- }
- ]
- })
- export class OAuthTokenModel extends Model<Partial<AttributesOnly<OAuthTokenModel>>> {
- @AllowNull(false)
- @Column
- accessToken: string
- @AllowNull(false)
- @Column
- accessTokenExpiresAt: Date
- @AllowNull(false)
- @Column
- refreshToken: string
- @AllowNull(false)
- @Column
- refreshTokenExpiresAt: Date
- @Column
- authName: string
- @CreatedAt
- createdAt: Date
- @UpdatedAt
- updatedAt: Date
- @ForeignKey(() => UserModel)
- @Column
- userId: number
- @BelongsTo(() => UserModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade'
- })
- User: UserModel
- @ForeignKey(() => OAuthClientModel)
- @Column
- oAuthClientId: number
- @BelongsTo(() => OAuthClientModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade'
- })
- OAuthClients: OAuthClientModel[]
- @AfterUpdate
- @AfterDestroy
- static removeTokenCache (token: OAuthTokenModel) {
- return TokensCache.Instance.clearCacheByToken(token.accessToken)
- }
- static loadByRefreshToken (refreshToken: string) {
- const query = {
- where: { refreshToken }
- }
- return OAuthTokenModel.findOne(query)
- }
- static getByRefreshTokenAndPopulateClient (refreshToken: string) {
- const query = {
- where: {
- refreshToken
- },
- include: [ OAuthClientModel ]
- }
- return OAuthTokenModel.scope(ScopeNames.WITH_USER)
- .findOne(query)
- .then(token => {
- if (!token) return null
- return {
- refreshToken: token.refreshToken,
- refreshTokenExpiresAt: token.refreshTokenExpiresAt,
- client: {
- id: token.oAuthClientId
- },
- user: token.User,
- token
- } as OAuthTokenInfo
- })
- .catch(err => {
- logger.error('getRefreshToken error.', { err })
- throw err
- })
- }
- static getByTokenAndPopulateUser (bearerToken: string): Promise<MOAuthTokenUser> {
- const query = {
- where: {
- accessToken: bearerToken
- }
- }
- return OAuthTokenModel.scope(ScopeNames.WITH_USER)
- .findOne(query)
- .then(token => {
- if (!token) return null
- return Object.assign(token, { user: token.User })
- })
- }
- static getByRefreshTokenAndPopulateUser (refreshToken: string): Promise<MOAuthTokenUser> {
- const query = {
- where: {
- refreshToken
- }
- }
- return OAuthTokenModel.scope(ScopeNames.WITH_USER)
- .findOne(query)
- .then(token => {
- if (!token) return undefined
- return Object.assign(token, { user: token.User })
- })
- }
- static deleteUserToken (userId: number, t?: Transaction) {
- TokensCache.Instance.deleteUserToken(userId)
- const query = {
- where: {
- userId
- },
- transaction: t
- }
- return OAuthTokenModel.destroy(query)
- }
- }
|