|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486 |
- 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 {
- 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<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: Awaited<ActorModel>
-
- @ForeignKey(() => UserModel)
- @Column
- userId: number
-
- @BelongsTo(() => UserModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade'
- })
- User: Awaited<UserModel>
-
- @ForeignKey(() => ApplicationModel)
- @Column
- applicationId: number
-
- @BelongsTo(() => ApplicationModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade'
- })
- Application: Awaited<ApplicationModel>
-
- @HasMany(() => VideoChannelModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoChannels: Awaited<VideoChannelModel>[]
-
- @HasMany(() => VideoPlaylistModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoPlaylists: Awaited<VideoPlaylistModel>[]
-
- @HasMany(() => VideoCommentModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'cascade',
- hooks: true
- })
- VideoComments: Awaited<VideoCommentModel>[]
-
- @HasMany(() => AccountBlocklistModel, {
- foreignKey: {
- name: 'targetAccountId',
- allowNull: false
- },
- as: 'BlockedBy',
- onDelete: 'CASCADE'
- })
- BlockedBy: Awaited<AccountBlocklistModel>[]
-
- @HasMany(() => AccountAutomaticTagPolicyModel, {
- foreignKey: {
- name: 'accountId',
- allowNull: false
- },
- onDelete: 'cascade'
- })
- AccountAutomaticTagPolicies: Awaited<AccountAutomaticTagPolicyModel>[]
-
- @HasMany(() => CommentAutomaticTagModel, {
- foreignKey: 'accountId',
- onDelete: 'CASCADE'
- })
- CommentAutomaticTags: Awaited<CommentAutomaticTagModel>[]
-
- @HasMany(() => VideoAutomaticTagModel, {
- foreignKey: 'accountId',
- onDelete: 'CASCADE'
- })
- VideoAutomaticTags: Awaited<VideoAutomaticTagModel>[]
-
- @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 getSQLAttributes (tableName: string, aliasPrefix = '') {
- return buildSQLAttributes({
- model: this,
- tableName,
- aliasPrefix
- })
- }
-
- // ---------------------------------------------------------------------------
-
- 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: 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<MAccountDefault> {
- 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<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<MAccountHost[]> {
- 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
- }
- }
|