Files
mattermost-mobile/app/database/admin/data_operator/operators.ts
Avinash Lingaloo d6a3504c08 MM_30482 [v2] DataOperator for all the isolated tables (#5182)
* MM_30475 : ADDED default schema

* MM_30475 : ADDED todo for field 'value' of default/Global entity

* MM_30476 : Created schema for SERVER DB

* MM_30476 : Server model [ IN PROGRESS ]

* MM_30476 : Including types for group, groups_in_channel and role

* MM_30476 : ADDED models for Group

- @typings absolute path has been added to the tsconfig.json

* MM_30476 : ADDED typings to current models

* MM_30476 : ADDED typings to current models

* MM_30476 : ADDED models related to TEAM section of the ERD

* MM_30476 : ADDED models for User section of the ERD

* MM_30476 : ADDED models for POST section of the ERD

* MM_30476 : ADDED models for Channel section of the ERD

* MM_30475 : Updated typings and references to MM_TABLES

* MM_30476 : Verified all field names

* MM_30476 : Verified every table associations

* MM_30476 : Verified all relation fields

* MM_30476 : Updated primary id of the main models

We will override the wdb id at component level when we create a new records.  This involves the models : channel, group, post, team and user.

* MM_30476 : Including 1:1 relationship amongs some entities

* MM_30476 : ADDED Schema Managers

* The migration array will hold all the migration steps.

*  The initial app release (e.g. v2 )will have an empty array and subsequent releases  (e.g.  v2.1 ) will have the steps listed in that array.

* On initialization, the database will perform the migration to accomodate for new columns/tables creation and while it will conserve the mobile phone's data, it will also make it conform to this new schema.

* If a migration fails, the migration process will rollback any changes.  This migration will be thoroughly tested in development before pushing it live.

* Revert "MM_30476 : ADDED Schema Managers"

This reverts commit a505bd5e11.

* MM_30478 : Converted schema_manager into a function

* MM_30478 : Updated schema manager and included patch for wdb

* MM_30478:  Updated watermelondb patch package

* MM_30478 : Update function create_schema_manager to createSqliteAdaptorOptions

* MM_30476 : Update constant name to reflect directory name

* MM_30476 : Updated msgCount from my_channel model to message_count in server schema

* MM_30482 : Added tests for schema_manager

* MM_30482 : Database Manager [ IN PROGRESS ]

* MM_30478 : Returning an sqliteAdapter instead of an object

* MM_30476 : Apply suggestions from code review

Co-authored-by: Elias Nahum <nahumhbl@gmail.com>

* MM_30476 : Updated all imports as per instruction.

* MM_30476 : Shortening object chains by destructuring

* MM_30476 : Updated schema file structure

* MM_30476 : Prettifying @typings folder

* MM_30476 : Removing useless ids

* MM_30476 : Prettify imports for decorators

* MM_30476 : ADDED documentations and lazy queries to Channel and Channel_Info

* MM_30476 : ADDED documentations for default schema

* MM_30476 : Documentation [ IN PROGRESS ]

- Following JSDoc syntax for single line comment
- Removed redundant fields in the 'membership' tables and left only the @relation records.

* MM_30476 : Documentations [ IN PROGRESS ]

* MM_30476 : Documentations [ IN PROGRESS ]

* MM_30476 : Documentations [ IN PROGRESS ]

* MM_30476 : Documentations [ IN PROGRESS]

Updated
1)  my_team and team,
2) my_channel and  channel,

to each have 1:1 relationship  with one another

* MM_30476 : Updated all Typescript definitions

* MM_30476 :Updated @relation to @immutableRelation

* MM_30476 : Updated description for previous_post_id

* MM_30478 : Updated patch package for wdb module

* MM_30478: DB Manager [IN PROGRESS ]

* MM_30478: DB Manager [IN PROGRESS]

* MM_30478: DB Manager [IN PROGRESS]

* MM_30478 : DB Manager [IN PROGRESS]

* MM_30478 : Deleting .db file on iOS

* MM_30478: Successfully deleting .db files and directory on iOS side

* MM_30478 : Update definition for default/global

* MM_30478 : Updated all models

* MM_30478 : Doing a bit of house cleaning

* MM_30478: Record of new server connection  added to default/servers db

* TS Definitely Typed Assignment issue is now FIXED

* MM_30478 : TS Definitely Typed Assignment \n  Removed all the constructors but error still in editor tabs.  But this time the app is not crashing

* MM_30478 : Attempt 1 [SUCCESSFUL]

* MM_30478 : Removing useDefineForClassFields

* MM_30478 : Retrieving the servers in a list + Improved the DB Manager and Babel config

* MM_30478 : Updated babel.config.js

* MM_30478 : Minor UI correction

* MM_30478 : Jest and Typescript configuration

* MM_30478 : A bit of housekeeping

* MM_30478 : Installed WDB on Android

* MM_30478 : Deletes new server record from default DB

* MM_30478 : Returns subset of server db instances

* MM_30478 : Code clean up

* MM_30478 :  Code clean up on db manager

* MM_30478 : House keeping + Patch for WDB

* MM_30478 : Android - Saving & Deleting in FilesDir [COMPLETED]

* MM_30478 : Code clean up

* MM_30478 : Code clean up

* MM_30478 : Code clean up

* MM_30478 : Test successful on Android device

* MM_30478 : Rolling back change to jest.config.js

* MM_30478 : Updated test to test_integration

* MM_30478 : Fix imports

* MM_30478 : Refactored the manual testscript

* MM_30478 : Renamed database manager test file

* MM_30478 : Code clean up

* MM_30478 : Updated manual test file with a note.

* MM_30482 : DataOperator [ IN PROGRESS ]

* MM_30482 : DataOperator - setting up the factory [ IN PROGRESS ]

* MM_30482: Code refactoring

* MM_30482 : DataOperator - setting up the factory [ IN PROGRESS ]

* MM_30482 : DataOperator - code clean up [ IN PROGRESS ]

* MM_30482 : Minor code clean up

* MM_30478 : Fixed JEST issue with TS

* MM_30478 : Fixed JEST issue with TS

* MM_30478 : Fixed JEST issue with TS

* MM_30478 : Implementing JEST test cases

* MM_30478 : Implementing JEST last  test cases

* MM_30478 : Jest fixing ts errors

* MM_30478 : Database Manager Jest testing [ IN PROGRESS ]

* MM_30482 - Fixing DataOperator [ IN PROGRESS ]

* MM_30482 : Code clean up

* MM_30482 - Creates multiple records [ IN PROGRESS ]

* MM_30482 - Creates multiple records [ IN PROGRESS ]

* MM_30482 : Update operation [ COMPLETED ]

* MM_30482 : Code clean up

* MM_30482 : Updated TS for Data Operator

* Update mobile v2 detox deps

* MM_30482 : Added factories for all isolated tables

* MM_30482 : Refactored TS

* MM_30482 : Refactored base factory

* MM_30482 : Updated JSDoc for operateBaseRecord - Delete CASE

* MM_30482 : Implementing test for Data Operator

* MM_30482 : Completed tests for all isolated tables

* MM_30482 : Renamed entity_factory into operators

* MM_30482 : Fix all imports

* MM_30482 : Update multiple records

* MM_30482 : Edge case for existing records ( update instead of create )

* MM_30482 : Edge case  - create instead of update

* MM_30482 : Code clean up

* MM_30482 : Code clean up

* MM_30482 : Code clean up

* MM_30482 : Code clean up

* Update app/database/admin/data_operator/operators.ts

Co-authored-by: Joseph Baylon <joseph.baylon@mattermost.com>

* Update app/database/admin/data_operator/operators.ts

Co-authored-by: Joseph Baylon <joseph.baylon@mattermost.com>

* Update app/database/admin/data_operator/operators.ts

Co-authored-by: Joseph Baylon <joseph.baylon@mattermost.com>

* MM_30482 : Imposing usage of correct table name for isolated entities

* MM_30482 : Code improvement as per Joseph reviews

* MM_30482 : Updated tests to validate choice of operator service wrt tableName

* MM_30482 : Updated PR as per suggestions

* MM_30482 : Updated comments to follow jsdoc conventions

Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
Co-authored-by: Avinash Lingaloo <>
Co-authored-by: Joseph Baylon <joseph.baylon@mattermost.com>
2021-02-26 08:24:53 +04:00

211 lines
7.7 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import {MM_TABLES} from '@constants/database';
import {Q} from '@nozbe/watermelondb';
import Model from '@nozbe/watermelondb/Model';
import App from '@typings/database/app';
import CustomEmoji from '@typings/database/custom_emoji';
import {
DataFactory,
RawApp,
RawCustomEmoji,
RawGlobal,
RawRole,
RawServers,
RawSystem,
RawTermsOfService,
} from '@typings/database/database';
import Global from '@typings/database/global';
import Role from '@typings/database/role';
import Servers from '@typings/database/servers';
import System from '@typings/database/system';
import TermsOfService from '@typings/database/terms_of_service';
import {OperationType} from './index';
const {APP, GLOBAL, SERVERS} = MM_TABLES.DEFAULT;
const {CUSTOM_EMOJI, ROLE, SYSTEM, TERMS_OF_SERVICE} = MM_TABLES.SERVER;
/**
* operateAppRecord: Prepares record of entity 'App' from the DEFAULT database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateAppRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawApp;
const generator = (app: App) => {
app._raw.id = record?.id ?? app.id;
app.buildNumber = record?.buildNumber ?? '';
app.createdAt = record?.createdAt ?? 0;
app.versionNumber = record?.versionNumber ?? '';
};
return operateBaseRecord({db, optType, tableName: APP, value, generator});
};
/**
* operateGlobalRecord: Prepares record of entity 'Global' from the DEFAULT database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateGlobalRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawGlobal;
const generator = (global: Global) => {
global._raw.id = record?.id ?? global.id;
global.name = record?.name ?? '';
global.value = record?.value ?? 0;
};
return operateBaseRecord({db, optType, tableName: GLOBAL, value, generator});
};
/**
* operateServersRecord: Prepares record of entity 'Servers' from the DEFAULT database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateServersRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawServers;
const generator = (servers: Servers) => {
servers._raw.id = record?.id ?? servers.id;
servers.dbPath = record?.dbPath ?? '';
servers.displayName = record?.displayName ?? 0;
servers.mentionCount = record?.mentionCount ?? 0;
servers.unreadCount = record?.unreadCount ?? 0;
servers.url = record?.url ?? 0;
};
return operateBaseRecord({db, optType, tableName: SERVERS, value, generator});
};
/**
* operateCustomEmojiRecord: Prepares record of entity 'CustomEmoji' from the SERVER database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateCustomEmojiRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawCustomEmoji;
const generator = (emoji: CustomEmoji) => {
emoji._raw.id = record?.id ?? emoji.id;
emoji.name = record?.name ?? '';
};
return operateBaseRecord({db, optType, tableName: CUSTOM_EMOJI, value, generator});
};
/**
* operateRoleRecord: Prepares record of entity 'Role' from the SERVER database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateRoleRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawRole;
const generator = (role: Role) => {
role._raw.id = record?.id ?? role.id;
role.name = record?.name ?? '';
role.permissions = record?.permissions ?? [];
};
return operateBaseRecord({db, optType, tableName: ROLE, value, generator});
};
/**
* operateSystemRecord: Prepares record of entity 'System' from the SERVER database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateSystemRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawSystem;
const generator = (system: System) => {
system._raw.id = record?.id ?? system.id;
system.name = record?.name ?? '';
system.value = record?.value ?? '';
};
return operateBaseRecord({db, optType, tableName: SYSTEM, value, generator});
};
/**
* operateTermsOfServiceRecord: Prepares record of entity 'TermsOfService' from the SERVER database for update or create actions.
* @param {DataFactory} operator
* @param {Database} operator.db
* @param {OperationType} operator.optType
* @param {RecordValue} operator.value
* @returns {Promise<void>}
*/
export const operateTermsOfServiceRecord = async ({db, optType, value}: DataFactory) => {
const record = value as RawTermsOfService;
const generator = (tos: TermsOfService) => {
tos._raw.id = record?.id ?? tos.id;
tos.acceptedAt = record?.acceptedAt ?? 0;
};
return operateBaseRecord({db, optType, tableName: TERMS_OF_SERVICE, value, generator});
};
/**
* operateBaseRecord: The 'id' of a record is key to this function. Please note that - at the moment - if WatermelonDB
* encounters an existing record during a CREATE operation, it silently fails the operation.
*
* In our case, we check to see if we have an existing 'id' and if so, we'll update the record with the data.
* For an UPDATE operation, we fetch the existing record using the 'id' value and then we do the update operation;
* if no record is found for that 'id', we'll create it a new record.
*
* @param {DataFactory} operatorBase
* @param {Database} operatorBase.db
* @param {OperationType} operatorBase.optType
* @param {string} operatorBase.tableName
* @param {any} operatorBase.value
* @param {((model: Model) => void)} operatorBase.generator
* @returns {Promise<any>}
*/
const operateBaseRecord = async ({db, optType, tableName, value, generator}: DataFactory) => {
// We query first to see if we have a record on that entity with the current value.id
const appRecord = await db.collections.get(tableName!).query(Q.where('id', value.id)).fetch() as Model[];
const isPresent = appRecord.length > 0;
if ((isPresent && optType === OperationType.CREATE) || (isPresent && optType === OperationType.UPDATE)) {
// Two possible scenarios:
// 1. We are dealing with either duplicates here and if so, we'll update instead of create
// 2. This is just a normal update operation
const record = appRecord[0];
return record.prepareUpdate(() => generator!(record));
}
if ((!isPresent && optType === OperationType.UPDATE) || (optType === OperationType.CREATE)) {
// Two possible scenarios
// 1. We don't have a record yet to update; so we create it
// 2. This is just a normal create operation
return db.collections.get(tableName!).prepareCreate(generator);
}
return null;
};