Files
mattermost-mobile/app/database/models/server/post.ts
Elias Nahum e883186fde [Gekidou] retry post (#6293)
* 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
2022-05-20 13:23:19 -04:00

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,
});
}