Vmeda.Online/src/providers/sync.ts

203 lines
7.0 KiB
TypeScript
Raw Normal View History

// (C) Copyright 2015 Martin Dougiamas
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import { Injectable } from '@angular/core';
import { CoreEventsProvider } from './events';
import { CoreSitesProvider } from './sites';
/*
* Service that provides some features regarding synchronization.
*/
@Injectable()
export class CoreSyncProvider {
// Variables for the database.
protected SYNC_TABLE = 'sync';
protected tableSchema = {
name: this.SYNC_TABLE,
columns: [
{
name: 'component',
type: 'TEXT',
notNull: true
},
{
name: 'id',
type: 'TEXT',
notNull: true
},
{
name: 'time',
type: 'INTEGER'
},
{
name: 'warnings',
type: 'TEXT'
}
],
primaryKeys: ['component', 'id']
};
// Store blocked sync objects.
2018-01-29 09:05:20 +00:00
protected blockedItems: { [siteId: string]: { [blockId: string]: { [operation: string]: boolean } } } = {};
constructor(eventsProvider: CoreEventsProvider, private sitesProvider: CoreSitesProvider) {
this.sitesProvider.createTableFromSchema(this.tableSchema);
// Unblock all blocks on logout.
eventsProvider.on(CoreEventsProvider.LOGOUT, (data) => {
this.clearAllBlocks(data.siteId);
});
}
/**
* Block a component and ID so it cannot be synchronized.
*
* @param {string} component Component name.
* @param {number} id Unique ID per component.
* @param {string} [operation] Operation name. If not defined, a default text is used.
* @param {string} [siteId] Site ID. If not defined, current site.
*/
2018-01-29 09:05:20 +00:00
blockOperation(component: string, id: number, operation?: string, siteId?: string): void {
siteId = siteId || this.sitesProvider.getCurrentSiteId();
const uniqueId = this.getUniqueSyncBlockId(component, id);
if (!this.blockedItems[siteId]) {
this.blockedItems[siteId] = {};
}
if (!this.blockedItems[siteId][uniqueId]) {
this.blockedItems[siteId][uniqueId] = {};
}
operation = operation || '-';
this.blockedItems[siteId][uniqueId][operation] = true;
}
/**
* Clear all blocks for a site or all sites.
*
* @param {string} [siteId] If set, clear the blocked objects only for this site. Otherwise clear them for all sites.
*/
2018-01-29 09:05:20 +00:00
clearAllBlocks(siteId?: string): void {
if (siteId) {
delete this.blockedItems[siteId];
} else {
this.blockedItems = {};
}
}
/**
* Clear all blocks for a certain component.
*
* @param {string} component Component name.
* @param {number} id Unique ID per component.
* @param {string} [siteId] Site ID. If not defined, current site.
*/
2018-01-29 09:05:20 +00:00
clearBlocks(component: string, id: number, siteId?: string): void {
siteId = siteId || this.sitesProvider.getCurrentSiteId();
const uniqueId = this.getUniqueSyncBlockId(component, id);
if (this.blockedItems[siteId]) {
delete this.blockedItems[siteId][uniqueId];
}
}
/**
* Returns a sync record.
* @param {string} component Component name.
* @param {string | number} id Unique ID per component.
* @param {string} [siteId] Site ID. If not defined, current site.
* @return {Promise<any>} Record if found or reject.
*/
getSyncRecord(component: string, id: string | number, siteId?: string): Promise<any> {
return this.sitesProvider.getSiteDb(siteId).then((db) => {
return db.getRecord(this.SYNC_TABLE, { component: component, id: id });
});
}
/**
* Inserts or Updates info of a sync record.
* @param {string} component Component name.
* @param {string | number} id Unique ID per component.
* @param {any} data Data that updates the record.
* @param {string} [siteId] Site ID. If not defined, current site.
* @return {Promise<any>} Promise resolved with done.
*/
insertOrUpdateSyncRecord(component: string, id: string | number, data: any, siteId?: string): Promise<any> {
return this.sitesProvider.getSiteDb(siteId).then((db) => {
data.component = component;
data.id = id;
return db.insertOrUpdateRecord(this.SYNC_TABLE, data, { component: component, id: id });
});
}
/**
* Convenience function to create unique identifiers for a component and id.
*
* @param {string} component Component name.
* @param {number} id Unique ID per component.
* @return {string} Unique sync id.
*/
2018-01-29 09:05:20 +00:00
protected getUniqueSyncBlockId(component: string, id: number): string {
return component + '#' + id;
}
/**
* Check if a component is blocked.
* One block can have different operations. Here we check how many operations are being blocking the object.
*
* @param {string} component Component name.
* @param {number} id Unique ID per component.
* @param {string} [siteId] Site ID. If not defined, current site.
* @return {boolean} Whether it's blocked.
*/
2018-01-29 09:05:20 +00:00
isBlocked(component: string, id: number, siteId?: string): boolean {
siteId = siteId || this.sitesProvider.getCurrentSiteId();
if (!this.blockedItems[siteId]) {
return false;
}
const uniqueId = this.getUniqueSyncBlockId(component, id);
if (!this.blockedItems[siteId][uniqueId]) {
return false;
}
return Object.keys(this.blockedItems[siteId][uniqueId]).length > 0;
}
/**
* Unblock an operation on a component and ID.
*
* @param {string} component Component name.
* @param {number} id Unique ID per component.
* @param {string} [operation] Operation name. If not defined, a default text is used.
* @param {string} [siteId] Site ID. If not defined, current site.
*/
2018-01-29 09:05:20 +00:00
unblockOperation(component: string, id: number, operation?: string, siteId?: string): void {
operation = operation || '-';
siteId = siteId || this.sitesProvider.getCurrentSiteId();
const uniqueId = this.getUniqueSyncBlockId(component, id);
if (this.blockedItems[siteId] && this.blockedItems[siteId][uniqueId]) {
delete this.blockedItems[siteId][uniqueId][operation];
}
}
}