123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465 |
- import { FindOptions, Includeable, IncludeOptions, Op, Transaction, WhereOptions } from 'sequelize'
- import {
- AllowNull,
- BeforeDestroy,
- BelongsTo,
- Column,
- CreatedAt,
- DataType,
- Default,
- DefaultScope,
- ForeignKey,
- HasMany,
- Is,
- Model,
- Scopes,
- Table,
- UpdatedAt
- } from 'sequelize-typescript'
- import { ModelCache } from '@server/models/model-cache'
- import { AttributesOnly } from '@shared/typescript-utils'
- import { Account, AccountSummary } from '../../../shared/models/actors'
- import { isAccountDescriptionValid } from '../../helpers/custom-validators/accounts'
- import { CONSTRAINTS_FIELDS, SERVER_ACTOR_NAME, WEBSERVER } from '../../initializers/constants'
- import { sendDeleteActor } from '../../lib/activitypub/send/send-delete'
- import {
- MAccount,
- MAccountActor,
- MAccountAP,
- MAccountDefault,
- MAccountFormattable,
- MAccountSummaryFormattable,
- MChannelActor
- } from '../../types/models'
- import { ActorModel } from '../actor/actor'
- import { ActorFollowModel } from '../actor/actor-follow'
- import { ActorImageModel } from '../actor/actor-image'
- import { ApplicationModel } from '../application/application'
- import { ServerModel } from '../server/server'
- import { ServerBlocklistModel } from '../server/server-blocklist'
- import { UserModel } from '../user/user'
- import { getSort, throwIfNotValid } from '../utils'
- import { VideoModel } from '../video/video'
- import { VideoChannelModel } from '../video/video-channel'
- import { VideoCommentModel } from '../video/video-comment'
- import { VideoPlaylistModel } from '../video/video-playlist'
- import { AccountBlocklistModel } from './account-blocklist'
- export enum ScopeNames {
- SUMMARY = 'SUMMARY'
- }
- export type SummaryOptions = {
- actorRequired?: boolean // Default: true
- whereActor?: WhereOptions
- whereServer?: WhereOptions
- withAccountBlockerIds?: number[]
- forCount?: boolean
- }
- @DefaultScope(() => ({
- include: [
- {
- model: ActorModel, // Default scope includes avatar and server
- required: true
- }
- ]
- }))
- @Scopes(() => ({
- [ScopeNames.SUMMARY]: (options: SummaryOptions = {}) => {
- const serverInclude: IncludeOptions = {
- attributes: [ 'host' ],
- model: ServerModel.unscoped(),
- required: !!options.whereServer,
- where: options.whereServer
- }
- const actorInclude: Includeable = {
- attributes: [ 'id', 'preferredUsername', 'url', 'serverId' ],
- model: ActorModel.unscoped(),
- required: options.actorRequired ?? true,
- where: options.whereActor,
- include: [ serverInclude ]
- }
- if (options.forCount !== true) {
- actorInclude.include.push({
- model: ActorImageModel,
- as: 'Avatars',
- required: false
- })
- }
- const queryInclude: Includeable[] = [
- actorInclude
- ]
- const query: FindOptions = {
- attributes: [ 'id', 'name', 'actorId' ]
- }
- if (options.withAccountBlockerIds) {
- queryInclude.push({
- attributes: [ 'id' ],
- model: AccountBlocklistModel.unscoped(),
- as: 'BlockedBy',
- required: false,
- where: {
- accountId: {
- [Op.in]: options.withAccountBlockerIds
- }
- }
- })
- serverInclude.include = [
- {
- attributes: [ 'id' ],
- model: ServerBlocklistModel.unscoped(),
- required: false,
- where: {
- accountId: {
- [Op.in]: options.withAccountBlockerIds
- }
- }
- }
- ]
- }
- query.include = queryInclude
- return query
- }
- }))
- @Table({
- tableName: 'account',
- indexes: [
- {
- fields: [ 'actorId' ],
- unique: true
- },
- {
- fields: [ 'applicationId' ]
- },
- {
- fields: [ 'userId' ]
- }
- ]
- })
- export class AccountModel extends Model<Partial<AttributesOnly<AccountModel>>> {
- @AllowNull(false)
- @Column
- name: string
- @AllowNull(true)
- @Default(null)
- @Is('AccountDescription', value => throwIfNotValid(value, isAccountDescriptionValid, 'description', true))
- @Column(DataType.STRING(CONSTRAINTS_FIELDS.USERS.DESCRIPTION.max))
- description: string
- @CreatedAt
- createdAt: Date
- @UpdatedAt
- updatedAt: Date
- @ForeignKey(() => ActorModel)
- @Column
- actorId: number
- @BelongsTo(() => ActorModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade'
- })
- Actor: ActorModel
- @ForeignKey(() => UserModel)
- @Column
- userId: number
- @BelongsTo(() => UserModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade'
- })
- User: UserModel
- @ForeignKey(() => ApplicationModel)
- @Column
- applicationId: number
- @BelongsTo(() => ApplicationModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade'
- })
- Application: ApplicationModel
- @HasMany(() => VideoChannelModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoChannels: VideoChannelModel[]
- @HasMany(() => VideoPlaylistModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoPlaylists: VideoPlaylistModel[]
- @HasMany(() => VideoCommentModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoComments: VideoCommentModel[]
- @HasMany(() => AccountBlocklistModel, {
- foreignKey: {
- name: 'targetAccountId',
- allowNull: false
- },
- as: 'BlockedBy',
- onDelete: 'CASCADE'
- })
- BlockedBy: AccountBlocklistModel[]
- @BeforeDestroy
- static async sendDeleteIfOwned (instance: AccountModel, options) {
- if (!instance.Actor) {
- instance.Actor = await instance.$get('Actor', { transaction: options.transaction })
- }
- await ActorFollowModel.removeFollowsOf(instance.Actor.id, options.transaction)
- if (instance.isOwned()) {
- return sendDeleteActor(instance.Actor, options.transaction)
- }
- return undefined
- }
- static load (id: number, transaction?: Transaction): Promise<MAccountDefault> {
- return AccountModel.findByPk(id, { transaction })
- }
- static loadByNameWithHost (nameWithHost: string): Promise<MAccountDefault> {
- const [ accountName, host ] = nameWithHost.split('@')
- if (!host || host === WEBSERVER.HOST) return AccountModel.loadLocalByName(accountName)
- return AccountModel.loadByNameAndHost(accountName, host)
- }
- static loadLocalByName (name: string): Promise<MAccountDefault> {
- const fun = () => {
- const query = {
- where: {
- [Op.or]: [
- {
- userId: {
- [Op.ne]: null
- }
- },
- {
- applicationId: {
- [Op.ne]: null
- }
- }
- ]
- },
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- preferredUsername: name
- }
- }
- ]
- }
- return AccountModel.findOne(query)
- }
- return ModelCache.Instance.doCache({
- cacheType: 'local-account-name',
- key: name,
- fun,
- // The server actor never change, so we can easily cache it
- whitelist: () => name === SERVER_ACTOR_NAME
- })
- }
- static loadByNameAndHost (name: string, host: string): Promise<MAccountDefault> {
- const query = {
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- preferredUsername: name
- },
- include: [
- {
- model: ServerModel,
- required: true,
- where: {
- host
- }
- }
- ]
- }
- ]
- }
- return AccountModel.findOne(query)
- }
- static loadByUrl (url: string, transaction?: Transaction): Promise<MAccountDefault> {
- const query = {
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- url
- }
- }
- ],
- transaction
- }
- return AccountModel.findOne(query)
- }
- static listForApi (start: number, count: number, sort: string) {
- const query = {
- offset: start,
- limit: count,
- order: getSort(sort)
- }
- return Promise.all([
- AccountModel.count(),
- AccountModel.findAll(query)
- ]).then(([ total, data ]) => ({ total, data }))
- }
- static loadAccountIdFromVideo (videoId: number): Promise<MAccount> {
- const query = {
- include: [
- {
- attributes: [ 'id', 'accountId' ],
- model: VideoChannelModel.unscoped(),
- required: true,
- include: [
- {
- attributes: [ 'id', 'channelId' ],
- model: VideoModel.unscoped(),
- where: {
- id: videoId
- }
- }
- ]
- }
- ]
- }
- return AccountModel.findOne(query)
- }
- static listLocalsForSitemap (sort: string): Promise<MAccountActor[]> {
- const query = {
- attributes: [ ],
- offset: 0,
- order: getSort(sort),
- include: [
- {
- attributes: [ 'preferredUsername', 'serverId' ],
- model: ActorModel.unscoped(),
- where: {
- serverId: null
- }
- }
- ]
- }
- return AccountModel
- .unscoped()
- .findAll(query)
- }
- getClientUrl () {
- return WEBSERVER.URL + '/accounts/' + this.Actor.getIdentifier()
- }
- toFormattedJSON (this: MAccountFormattable): Account {
- return {
- ...this.Actor.toFormattedJSON(),
- id: this.id,
- displayName: this.getDisplayName(),
- description: this.description,
- updatedAt: this.updatedAt,
- userId: this.userId ?? undefined
- }
- }
- toFormattedSummaryJSON (this: MAccountSummaryFormattable): AccountSummary {
- const actor = this.Actor.toFormattedSummaryJSON()
- return {
- id: this.id,
- displayName: this.getDisplayName(),
- name: actor.name,
- url: actor.url,
- host: actor.host,
- avatars: actor.avatars,
- // TODO: remove, deprecated in 4.2
- avatar: actor.avatar
- }
- }
- toActivityPubObject (this: MAccountAP) {
- const obj = this.Actor.toActivityPubObject(this.name)
- return Object.assign(obj, {
- summary: this.description
- })
- }
- isOwned () {
- return this.Actor.isOwned()
- }
- isOutdated () {
- return this.Actor.isOutdated()
- }
- getDisplayName () {
- return this.name
- }
- getLocalUrl (this: MAccountActor | MChannelActor) {
- return WEBSERVER.URL + `/accounts/` + this.Actor.preferredUsername
- }
- isBlocked () {
- return this.BlockedBy && this.BlockedBy.length !== 0
- }
- }
|