Revision control
Copy as Markdown
Other Tools
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
function shouldSyncCalendar(calendar) {
if (calendar.type == "caldav") {
return true;
}
if (calendar.type == "ics") {
return calendar.uri.schemeIs("http") || calendar.uri.schemeIs("https");
}
return false;
}
/**
* CalendarRecord represents the state of an add-on in an application.
*
* Each add-on has its own record for each application ID it is installed
* on.
*
* The ID of add-on records is a randomly-generated GUID. It is random instead
* of deterministic so the URIs of the records cannot be guessed and so
* compromised server credentials won't result in disclosure of the specific
* add-ons present in a Sync account.
*
* The record contains the following fields:
*
*/
export function CalendarRecord(collection, id) {
CryptoWrapper.call(this, collection, id);
}
CalendarRecord.prototype = {
_logName: "Record.Calendar",
};
Object.setPrototypeOf(CalendarRecord.prototype, CryptoWrapper.prototype);
Utils.deferGetSet(CalendarRecord, "cleartext", [
"name",
"type",
"url",
"username",
]);
CalendarRecord.from = function (data) {
const record = new CalendarRecord(undefined, data.id);
for (const [key, value] of Object.entries(data)) {
record.cleartext[key] = value;
}
return record;
};
export function CalendarsEngine(service) {
SyncEngine.call(this, "Calendars", service);
}
CalendarsEngine.prototype = {
_storeObj: CalendarStore,
_trackerObj: CalendarTracker,
_recordObj: CalendarRecord,
version: 2,
syncPriority: 6,
/**
* Returns a changeset for this sync. Engine implementations can override this
* method to bypass the tracker for certain or all changed items.
*/
async getChangedIDs() {
return this._tracker.getChangedIDs();
},
};
Object.setPrototypeOf(CalendarsEngine.prototype, SyncEngine.prototype);
function CalendarStore(name, engine) {
CachedStore.call(this, name, engine);
}
CalendarStore.prototype = {
/**
* Create an item in the store from a record.
*
* This is called by the default implementation of applyIncoming(). If using
* applyIncomingBatch(), this won't be called unless your store calls it.
*
* @param {object} record - The store record to create an item from.
*/
async create(record) {
await super.create(record);
if (!["caldav", "ics"].includes(record.type)) {
this._log.trace(
`Skipping creation of unknown item type ("${record.type}"): ${record.id}`
);
return;
}
const calendar = cal.manager.createCalendar(
record.type,
Services.io.newURI(record.url)
);
calendar.name = record.name;
if (record.username) {
calendar.setProperty("username", record.username);
}
// Set this *after* the properties so it can pick up the session ID or username.
calendar.id = record.id;
cal.manager.registerCalendar(calendar);
if (!calendar.getProperty("disabled")) {
calendar.refresh();
}
},
/**
* Remove an item in the store from a record.
*
* This is called by the default implementation of applyIncoming(). If using
* applyIncomingBatch(), this won't be called unless your store calls it.
*
* @param {object} record - The store record to delete an item from.
*/
async remove(record) {
await super.remove(record);
const calendar = cal.manager.getCalendarById(record.id);
if (!calendar) {
this._log.trace("Asked to remove record that doesn't exist, ignoring");
return;
}
cal.manager.removeCalendar(calendar);
},
/**
* Update an item from a record.
*
* This is called by the default implementation of applyIncoming(). If using
* applyIncomingBatch(), this won't be called unless your store calls it.
*
* @param {object} record - The record to use to update an item from.
*/
async update(record) {
await super.update(record);
const calendar = cal.manager.getCalendarById(record.id);
if (!calendar) {
this._log.trace("Skipping update for unknown item: " + record.id);
return;
}
if (calendar.type != record.type) {
throw new Components.Exception(
`Refusing to change calendar type from ${calendar.type} to ${record.type}`,
Cr.NS_ERROR_FAILURE
);
}
if (calendar.uri.spec != record.url) {
throw new Error(
`Refusing to change calendar URL from "${calendar.uri.spec}" to "${record.url}"`
);
}
calendar.name = record.name;
if (record.username) {
calendar.setProperty("username", record.username);
} else {
calendar.deleteProperty("username");
}
},
/**
* Obtain the set of all known record IDs.
*
* @returns {object} an object with ID strings as keys and values of true.
* The values are ignored.
*/
async getAllIDs() {
const ids = await super.getAllIDs();
for (const c of cal.manager.getCalendars()) {
if (shouldSyncCalendar(c)) {
ids[c.id] = true;
}
}
return ids;
},
/**
* Create a record from the specified ID.
*
* If the ID is known, the record should be populated with metadata from
* the store. If the ID is not known, the record should be created with the
* delete field set to true.
*
* @param {string} id - Record ID
* @param {object} collection - Collection to add record to. This is typically
* passed into the constructor for the newly-created record.
* @returns {object} record type for this engine
*/
async createRecord(id, collection) {
const record = new CalendarRecord(collection, id);
const data = await super.getCreateRecordData(id);
const calendar = cal.manager.getCalendarById(id);
// If we don't know about this ID, mark the record as deleted.
if (!calendar && !data) {
record.deleted = true;
return record;
}
if (data) {
for (const [key, value] of Object.entries(data)) {
record.cleartext[key] = value;
}
}
if (calendar) {
record.name = calendar.name;
record.type = calendar.type;
record.url = calendar.uri.spec;
record.username = calendar.getProperty("username") || undefined;
super.update(record);
}
return record;
},
};
Object.setPrototypeOf(CalendarStore.prototype, CachedStore.prototype);
function CalendarTracker(name, engine) {
Tracker.call(this, name, engine);
}
CalendarTracker.prototype = {
QueryInterface: cal.generateQI([
"calICalendarManagerObserver",
"nsIObserver",
]),
_changedIDs: new Set(),
ignoreAll: false,
_watchedPrefs: ["name", "username"],
async getChangedIDs() {
const changes = {};
for (const id of this._changedIDs) {
changes[id] = 0;
}
return changes;
},
clearChangedIDs() {
this._changedIDs.clear();
},
onStart() {
Services.prefs.addObserver("calendar.registry.", this);
cal.manager.addObserver(this);
},
onStop() {
Services.prefs.removeObserver("calendar.registry.", this);
cal.manager.removeObserver(this);
},
observe(subject, topic, data) {
if (this.ignoreAll) {
return;
}
const id = data.split(".")[2];
const prefName = data.substring(id.length + 19);
if (!this._watchedPrefs.includes(prefName)) {
return;
}
const calendar = cal.manager.getCalendarById(id);
if (calendar && shouldSyncCalendar(calendar) && !this._changedIDs.has(id)) {
this._changedIDs.add(id);
this.score += SCORE_INCREMENT_XLARGE;
}
},
onCalendarRegistered(calendar) {
if (this.ignoreAll) {
return;
}
if (shouldSyncCalendar(calendar)) {
this._changedIDs.add(calendar.id);
this.score += SCORE_INCREMENT_XLARGE;
}
},
onCalendarUnregistering() {},
onCalendarDeleting(calendar) {
if (this.ignoreAll) {
return;
}
if (shouldSyncCalendar(calendar)) {
this.engine._store.markDeleted(calendar.id);
this._changedIDs.add(calendar.id);
this.score += SCORE_INCREMENT_XLARGE;
}
},
};
Object.setPrototypeOf(CalendarTracker.prototype, Tracker.prototype);