import { Account, AccountSummary } from '@peertube/peertube-models' import { ModelCache } from '@server/models/shared/model-cache.js' import { FindOptions, IncludeOptions, Includeable, Op, Transaction, WhereOptions } from 'sequelize' import { AfterDestroy, AllowNull, BeforeDestroy, BelongsTo, Column, CreatedAt, DataType, Default, DefaultScope, ForeignKey, HasMany, Is, Scopes, Table, UpdatedAt } from 'sequelize-typescript' import { isAccountDescriptionValid } from '../../helpers/custom-validators/accounts.js' import { CONSTRAINTS_FIELDS, SERVER_ACTOR_NAME, WEBSERVER } from '../../initializers/constants.js' import { sendDeleteActor } from '../../lib/activitypub/send/send-delete.js' import { MAccount, MAccountAP, MAccountDefault, MAccountFormattable, MAccountHost, MAccountSummaryFormattable, MChannelHost } from '../../types/models/index.js' import { ActorFollowModel } from '../actor/actor-follow.js' import { ActorImageModel } from '../actor/actor-image.js' import { ActorModel } from '../actor/actor.js' import { ApplicationModel } from '../application/application.js' import { AccountAutomaticTagPolicyModel } from '../automatic-tag/account-automatic-tag-policy.js' import { CommentAutomaticTagModel } from '../automatic-tag/comment-automatic-tag.js' import { VideoAutomaticTagModel } from '../automatic-tag/video-automatic-tag.js' import { ServerBlocklistModel } from '../server/server-blocklist.js' import { ServerModel } from '../server/server.js' import { SequelizeModel, buildSQLAttributes, getSort, throwIfNotValid } from '../shared/index.js' import { UserModel } from '../user/user.js' import { VideoChannelModel } from '../video/video-channel.js' import { VideoCommentModel } from '../video/video-comment.js' import { VideoPlaylistModel } from '../video/video-playlist.js' import { VideoModel } from '../video/video.js' import { AccountBlocklistModel } from './account-blocklist.js' 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 SequelizeModel { @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: Awaited @ForeignKey(() => UserModel) @Column userId: number @BelongsTo(() => UserModel, { foreignKey: { allowNull: true }, onDelete: 'cascade' }) User: Awaited @ForeignKey(() => ApplicationModel) @Column applicationId: number @BelongsTo(() => ApplicationModel, { foreignKey: { allowNull: true }, onDelete: 'cascade' }) Application: Awaited @HasMany(() => VideoChannelModel, { foreignKey: { allowNull: false }, onDelete: 'cascade', hooks: true }) VideoChannels: Awaited[] @HasMany(() => VideoPlaylistModel, { foreignKey: { allowNull: false }, onDelete: 'cascade', hooks: true }) VideoPlaylists: Awaited[] @HasMany(() => VideoCommentModel, { foreignKey: { allowNull: true }, onDelete: 'cascade', hooks: true }) VideoComments: Awaited[] @HasMany(() => AccountBlocklistModel, { foreignKey: { name: 'targetAccountId', allowNull: false }, as: 'BlockedBy', onDelete: 'CASCADE' }) BlockedBy: Awaited[] @HasMany(() => AccountAutomaticTagPolicyModel, { foreignKey: { name: 'accountId', allowNull: false }, onDelete: 'cascade' }) AccountAutomaticTagPolicies: Awaited[] @HasMany(() => CommentAutomaticTagModel, { foreignKey: 'accountId', onDelete: 'CASCADE' }) CommentAutomaticTags: Awaited[] @HasMany(() => VideoAutomaticTagModel, { foreignKey: 'accountId', onDelete: 'CASCADE' }) VideoAutomaticTags: Awaited[] @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 } @AfterDestroy static async deleteActorIfRemote (instance: AccountModel, options) { if (!instance.Actor) { instance.Actor = await instance.$get('Actor', { transaction: options.transaction }) } // Remote actor, delete it if (instance.Actor.serverId) { await instance.Actor.destroy({ transaction: options.transaction }) } } // --------------------------------------------------------------------------- static getSQLAttributes (tableName: string, aliasPrefix = '') { return buildSQLAttributes({ model: this, tableName, aliasPrefix }) } // --------------------------------------------------------------------------- static load (id: number, transaction?: Transaction): Promise { return AccountModel.findByPk(id, { transaction }) } static loadByNameWithHost (nameWithHost: string): Promise { const [ accountName, host ] = nameWithHost.split('@') if (!host || host === WEBSERVER.HOST) return AccountModel.loadLocalByName(accountName) return AccountModel.loadByNameAndHost(accountName, host) } static loadLocalByName (name: string): Promise { const fun = () => { const query = { where: { [Op.or]: [ { userId: { [Op.ne]: null } }, { applicationId: { [Op.ne]: null } } ] }, include: [ { model: ActorModel, required: true, where: ActorModel.wherePreferredUsername(name) } ] } return AccountModel.findOne(query) } return ModelCache.Instance.doCache({ cacheType: 'server-account', 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 { const query = { include: [ { model: ActorModel, required: true, where: ActorModel.wherePreferredUsername(name), include: [ { model: ServerModel, required: true, where: { host } } ] } ] } return AccountModel.findOne(query) } static loadByUrl (url: string, transaction?: Transaction): Promise { 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 { 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 { const query = { attributes: [ ], offset: 0, order: getSort(sort), include: [ { attributes: [ 'preferredUsername', 'serverId' ], model: ActorModel.unscoped(), where: { serverId: null } } ] } return AccountModel .unscoped() .findAll(query) } toFormattedJSON (this: MAccountFormattable): Account { return { ...this.Actor.toFormattedJSON(false), 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 } } async toActivityPubObject (this: MAccountAP) { const obj = await 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 } // Avoid error when running this method on MAccount... | MChannel... getClientUrl (this: MAccountHost | MChannelHost) { return WEBSERVER.URL + '/a/' + this.Actor.getIdentifier() + '/video-channels' } isBlocked () { return this.BlockedBy && this.BlockedBy.length !== 0 } }