forked from Ivasoft/mattermost-mobile
* Add try again functionality to failed posts * Fix attach files on Android * feedback review * Prevent android crash when uploading files for the first time * Update the timestamp for updateAt when retrying to post * Add POST TIME TO FAIL * use function isPostFailed
180 lines
7.0 KiB
TypeScript
180 lines
7.0 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import {Q, Query, Relation} from '@nozbe/watermelondb';
|
|
import {children, field, immutableRelation, json, lazy} from '@nozbe/watermelondb/decorators';
|
|
import Model, {Associations} from '@nozbe/watermelondb/Model';
|
|
|
|
import {MM_TABLES} from '@constants/database';
|
|
import {safeParseJSON} from '@utils/helpers';
|
|
|
|
import type ChannelModel from '@typings/database/models/servers/channel';
|
|
import type DraftModel from '@typings/database/models/servers/draft';
|
|
import type FileModel from '@typings/database/models/servers/file';
|
|
import type PostModelInterface from '@typings/database/models/servers/post';
|
|
import type PostInThreadModel from '@typings/database/models/servers/posts_in_thread';
|
|
import type ReactionModel from '@typings/database/models/servers/reaction';
|
|
import type ThreadModel from '@typings/database/models/servers/thread';
|
|
import type UserModel from '@typings/database/models/servers/user';
|
|
|
|
const {CHANNEL, DRAFT, FILE, POST, POSTS_IN_THREAD, REACTION, THREAD, USER} = MM_TABLES.SERVER;
|
|
|
|
/**
|
|
* The Post model is the building block of communication in the Mattermost app.
|
|
*/
|
|
export default class PostModel extends Model implements PostModelInterface {
|
|
/** table (name) : Post */
|
|
static table = POST;
|
|
|
|
/** associations : Describes every relationship to this table. */
|
|
static associations: Associations = {
|
|
|
|
/** A CHANNEL can have multiple POST. (relationship is 1:N) */
|
|
[CHANNEL]: {type: 'belongs_to', key: 'channel_id'},
|
|
|
|
/** A POST can have multiple DRAFT. (relationship is 1:N) */
|
|
[DRAFT]: {type: 'has_many', foreignKey: 'root_id'},
|
|
|
|
/** A POST can have multiple FILE. (relationship is 1:N)*/
|
|
[FILE]: {type: 'has_many', foreignKey: 'post_id'},
|
|
|
|
/** A POST can have multiple POSTS_IN_THREAD. (relationship is 1:N)*/
|
|
[POSTS_IN_THREAD]: {type: 'has_many', foreignKey: 'root_id'},
|
|
|
|
/** A POST can have multiple REACTION. (relationship is 1:N)*/
|
|
[REACTION]: {type: 'has_many', foreignKey: 'post_id'},
|
|
|
|
/** A POST can have an associated thread. (relationship is 1:1) */
|
|
[THREAD]: {type: 'has_many', foreignKey: 'id'},
|
|
|
|
/** A USER can have multiple POST. A user can author several posts. (relationship is 1:N)*/
|
|
[USER]: {type: 'belongs_to', key: 'user_id'},
|
|
};
|
|
|
|
/** channel_id : The foreign key for the Channel to which this post belongs to. */
|
|
@field('channel_id') channelId!: string;
|
|
|
|
/** create_at : The timestamp to when this post was first created */
|
|
@field('create_at') createAt!: number;
|
|
|
|
/** delete_at : The timestamp to when this post was last archived/deleted */
|
|
@field('delete_at') deleteAt!: number;
|
|
|
|
/** update_at : The timestamp to when this post was last updated on the server */
|
|
@field('update_at') updateAt!: number;
|
|
|
|
/** edit_at : The timestamp to when this post was last edited */
|
|
@field('edit_at') editAt!: number;
|
|
|
|
/** is_pinned : A Boolean flag indicating if this Post is pinned */
|
|
@field('is_pinned') isPinned!: boolean;
|
|
|
|
/** message : Message in the post */
|
|
@field('message') message!: string;
|
|
|
|
/** metadata: All the extra data associated with this Post */
|
|
@json('metadata', safeParseJSON) metadata!: PostMetadata | null;
|
|
|
|
/** original_id : Any post will have this value empty unless it is updated */
|
|
@field('original_id') originalId!: string;
|
|
|
|
/** pending_post_id : The id given to a post before it is published on the server */
|
|
@field('pending_post_id') pendingPostId!: string;
|
|
|
|
/** previous_post_id : Id of the previous post. If this value is empty, this implies that it is not in the db and we will request it from server */
|
|
@field('previous_post_id') previousPostId!: string;
|
|
|
|
/** root_id : Used in threads. All posts under a thread will have this id in common */
|
|
@field('root_id') rootId!: string;
|
|
|
|
/** type : Type of props (e.g. system message) */
|
|
@field('type') type!: PostType;
|
|
|
|
/** user_id : The foreign key of the User who authored this post. */
|
|
@field('user_id') userId!: string;
|
|
|
|
/** props : Additional attributes for this props */
|
|
@json('props', safeParseJSON) props!: any;
|
|
|
|
// A draft can be associated with this post for as long as this post is a parent post
|
|
@lazy drafts = this.collections.get<DraftModel>(DRAFT).query(Q.on(POST, 'id', this.id));
|
|
|
|
@lazy root = this.collection.query(Q.where('id', this.rootId)) as Query<PostModel>;
|
|
|
|
/** postsInThread: The thread to which this post is associated */
|
|
@lazy postsInThread = this.collections.get<PostInThreadModel>(POSTS_IN_THREAD).query(
|
|
Q.where('root_id', this.rootId || this.id),
|
|
Q.sortBy('latest', Q.desc),
|
|
Q.take(1),
|
|
);
|
|
|
|
/** files: All the files associated with this Post */
|
|
@children(FILE) files!: Query<FileModel>;
|
|
|
|
/** reactions: All the reactions associated with this Post */
|
|
@children(REACTION) reactions!: Query<ReactionModel>;
|
|
|
|
/** author: The author of this Post */
|
|
@immutableRelation(USER, 'user_id') author!: Relation<UserModel>;
|
|
|
|
/** channel: The channel which is presenting this Post */
|
|
@immutableRelation(CHANNEL, 'channel_id') channel!: Relation<ChannelModel>;
|
|
|
|
/** thread : The thread data for the post */
|
|
@immutableRelation(THREAD, 'id') thread!: Relation<ThreadModel>;
|
|
|
|
async destroyPermanently() {
|
|
await this.reactions.destroyAllPermanently();
|
|
await this.files.destroyAllPermanently();
|
|
await this.drafts.destroyAllPermanently();
|
|
await this.collections.get(POSTS_IN_THREAD).query(
|
|
Q.where('root_id', this.id),
|
|
).destroyAllPermanently();
|
|
try {
|
|
const thread = await this.thread.fetch();
|
|
if (thread) {
|
|
await thread.destroyPermanently();
|
|
await thread.participants.destroyAllPermanently();
|
|
await thread.threadsInTeam.destroyAllPermanently();
|
|
}
|
|
} catch {
|
|
// there is no thread record for this post
|
|
}
|
|
super.destroyPermanently();
|
|
}
|
|
|
|
async hasReplies() {
|
|
if (!this.rootId) {
|
|
return (await this.postsInThread.fetch()).length > 0;
|
|
}
|
|
|
|
const root = await this.root.fetch();
|
|
if (root.length) {
|
|
return (await root[0].postsInThread.fetch()).length > 0;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
toApi = async (): Promise<Post> => ({
|
|
id: this.id,
|
|
create_at: this.createAt,
|
|
update_at: this.updateAt,
|
|
edit_at: this.editAt,
|
|
delete_at: this.deleteAt,
|
|
is_pinned: this.isPinned,
|
|
user_id: this.userId,
|
|
channel_id: this.channelId,
|
|
root_id: this.rootId,
|
|
original_id: this.originalId,
|
|
message: this.message,
|
|
type: this.type,
|
|
props: this.props,
|
|
pending_post_id: this.pendingPostId,
|
|
file_ids: (await this.files.fetchIds()),
|
|
metadata: (this.metadata ? this.metadata : {}) as PostMetadata,
|
|
hashtags: '',
|
|
reply_count: 0,
|
|
});
|
|
}
|