From 580990a5eb4a79892c48e3a3fce3386fe80e6cc2 Mon Sep 17 00:00:00 2001 From: Determinant Date: Sat, 2 Feb 2019 22:15:54 -0500 Subject: finish cache design --- build/background.js | 987 ++++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 925 insertions(+), 62 deletions(-) (limited to 'build/background.js') diff --git a/build/background.js b/build/background.js index ec5d6af..7804de6 100644 --- a/build/background.js +++ b/build/background.js @@ -1,4 +1,728 @@ (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i 1 + +// lruList is a yallist where the head is the youngest +// item, and the tail is the oldest. the list contains the Hit +// objects as the entries. +// Each Hit object has a reference to its Yallist.Node. This +// never changes. +// +// cache is a Map (or PseudoMap) that matches the keys to +// the Yallist.Node object. +class LRUCache { + constructor (options) { + if (typeof options === 'number') + options = { max: options } + + if (!options) + options = {} + + if (options.max && (typeof options.max !== 'number' || options.max < 0)) + throw new TypeError('max must be a non-negative number') + // Kind of weird to have a default max of Infinity, but oh well. + const max = this[MAX] = options.max || Infinity + + const lc = options.length || naiveLength + this[LENGTH_CALCULATOR] = (typeof lc !== 'function') ? naiveLength : lc + this[ALLOW_STALE] = options.stale || false + if (options.maxAge && typeof options.maxAge !== 'number') + throw new TypeError('maxAge must be a number') + this[MAX_AGE] = options.maxAge || 0 + this[DISPOSE] = options.dispose + this[NO_DISPOSE_ON_SET] = options.noDisposeOnSet || false + this[UPDATE_AGE_ON_GET] = options.updateAgeOnGet || false + this.reset() + } + + // resize the cache when the max changes. + set max (mL) { + if (typeof mL !== 'number' || mL < 0) + throw new TypeError('max must be a non-negative number') + + this[MAX] = mL || Infinity + trim(this) + } + get max () { + return this[MAX] + } + + set allowStale (allowStale) { + this[ALLOW_STALE] = !!allowStale + } + get allowStale () { + return this[ALLOW_STALE] + } + + set maxAge (mA) { + if (typeof mA !== 'number') + throw new TypeError('maxAge must be a non-negative number') + + this[MAX_AGE] = mA + trim(this) + } + get maxAge () { + return this[MAX_AGE] + } + + // resize the cache when the lengthCalculator changes. + set lengthCalculator (lC) { + if (typeof lC !== 'function') + lC = naiveLength + + if (lC !== this[LENGTH_CALCULATOR]) { + this[LENGTH_CALCULATOR] = lC + this[LENGTH] = 0 + this[LRU_LIST].forEach(hit => { + hit.length = this[LENGTH_CALCULATOR](hit.value, hit.key) + this[LENGTH] += hit.length + }) + } + trim(this) + } + get lengthCalculator () { return this[LENGTH_CALCULATOR] } + + get length () { return this[LENGTH] } + get itemCount () { return this[LRU_LIST].length } + + rforEach (fn, thisp) { + thisp = thisp || this + for (let walker = this[LRU_LIST].tail; walker !== null;) { + const prev = walker.prev + forEachStep(this, fn, walker, thisp) + walker = prev + } + } + + forEach (fn, thisp) { + thisp = thisp || this + for (let walker = this[LRU_LIST].head; walker !== null;) { + const next = walker.next + forEachStep(this, fn, walker, thisp) + walker = next + } + } + + keys () { + return this[LRU_LIST].toArray().map(k => k.key) + } + + values () { + return this[LRU_LIST].toArray().map(k => k.value) + } + + reset () { + if (this[DISPOSE] && + this[LRU_LIST] && + this[LRU_LIST].length) { + this[LRU_LIST].forEach(hit => this[DISPOSE](hit.key, hit.value)) + } + + this[CACHE] = new Map() // hash of items by key + this[LRU_LIST] = new Yallist() // list of items in order of use recency + this[LENGTH] = 0 // length of items in the list + } + + dump () { + return this[LRU_LIST].map(hit => + isStale(this, hit) ? false : { + k: hit.key, + v: hit.value, + e: hit.now + (hit.maxAge || 0) + }).toArray().filter(h => h) + } + + dumpLru () { + return this[LRU_LIST] + } + + set (key, value, maxAge) { + maxAge = maxAge || this[MAX_AGE] + + if (maxAge && typeof maxAge !== 'number') + throw new TypeError('maxAge must be a number') + + const now = maxAge ? Date.now() : 0 + const len = this[LENGTH_CALCULATOR](value, key) + + if (this[CACHE].has(key)) { + if (len > this[MAX]) { + del(this, this[CACHE].get(key)) + return false + } + + const node = this[CACHE].get(key) + const item = node.value + + // dispose of the old one before overwriting + // split out into 2 ifs for better coverage tracking + if (this[DISPOSE]) { + if (!this[NO_DISPOSE_ON_SET]) + this[DISPOSE](key, item.value) + } + + item.now = now + item.maxAge = maxAge + item.value = value + this[LENGTH] += len - item.length + item.length = len + this.get(key) + trim(this) + return true + } + + const hit = new Entry(key, value, len, now, maxAge) + + // oversized objects fall out of cache automatically. + if (hit.length > this[MAX]) { + if (this[DISPOSE]) + this[DISPOSE](key, value) + + return false + } + + this[LENGTH] += hit.length + this[LRU_LIST].unshift(hit) + this[CACHE].set(key, this[LRU_LIST].head) + trim(this) + return true + } + + has (key) { + if (!this[CACHE].has(key)) return false + const hit = this[CACHE].get(key).value + return !isStale(this, hit) + } + + get (key) { + return get(this, key, true) + } + + peek (key) { + return get(this, key, false) + } + + pop () { + const node = this[LRU_LIST].tail + if (!node) + return null + + del(this, node) + return node.value + } + + del (key) { + del(this, this[CACHE].get(key)) + } + + load (arr) { + // reset the cache + this.reset() + + const now = Date.now() + // A previous serialized cache has the most recent items first + for (let l = arr.length - 1; l >= 0; l--) { + const hit = arr[l] + const expiresAt = hit.e || 0 + if (expiresAt === 0) + // the item was created without expiration in a non aged cache + this.set(hit.k, hit.v) + else { + const maxAge = expiresAt - now + // dont add already expired items + if (maxAge > 0) { + this.set(hit.k, hit.v, maxAge) + } + } + } + } + + prune () { + this[CACHE].forEach((value, key) => get(this, key, false)) + } +} + +const get = (self, key, doUse) => { + const node = self[CACHE].get(key) + if (node) { + const hit = node.value + if (isStale(self, hit)) { + del(self, node) + if (!self[ALLOW_STALE]) + return undefined + } else { + if (doUse) { + if (self[UPDATE_AGE_ON_GET]) + node.value.now = Date.now() + self[LRU_LIST].unshiftNode(node) + } + } + return hit.value + } +} + +const isStale = (self, hit) => { + if (!hit || (!hit.maxAge && !self[MAX_AGE])) + return false + + const diff = Date.now() - hit.now + return hit.maxAge ? diff > hit.maxAge + : self[MAX_AGE] && (diff > self[MAX_AGE]) +} + +const trim = self => { + if (self[LENGTH] > self[MAX]) { + for (let walker = self[LRU_LIST].tail; + self[LENGTH] > self[MAX] && walker !== null;) { + // We know that we're about to delete this one, and also + // what the next least recently used key will be, so just + // go ahead and set it now. + const prev = walker.prev + del(self, walker) + walker = prev + } + } +} + +const del = (self, node) => { + if (node) { + const hit = node.value + if (self[DISPOSE]) + self[DISPOSE](hit.key, hit.value) + + self[LENGTH] -= hit.length + self[CACHE].delete(hit.key) + self[LRU_LIST].removeNode(node) + } +} + +class Entry { + constructor (key, value, length, now, maxAge) { + this.key = key + this.value = value + this.length = length + this.now = now + this.maxAge = maxAge || 0 + } +} + +const forEachStep = (self, fn, node, thisp) => { + let hit = node.value + if (isStale(self, hit)) { + del(self, node) + if (!self[ALLOW_STALE]) + hit = undefined + } + if (hit) + fn.call(thisp, hit.value, hit.key, self) +} + +module.exports = LRUCache + +},{"yallist":3}],2:[function(require,module,exports){ +'use strict' +module.exports = function (Yallist) { + Yallist.prototype[Symbol.iterator] = function* () { + for (let walker = this.head; walker; walker = walker.next) { + yield walker.value + } + } +} + +},{}],3:[function(require,module,exports){ +'use strict' +module.exports = Yallist + +Yallist.Node = Node +Yallist.create = Yallist + +function Yallist (list) { + var self = this + if (!(self instanceof Yallist)) { + self = new Yallist() + } + + self.tail = null + self.head = null + self.length = 0 + + if (list && typeof list.forEach === 'function') { + list.forEach(function (item) { + self.push(item) + }) + } else if (arguments.length > 0) { + for (var i = 0, l = arguments.length; i < l; i++) { + self.push(arguments[i]) + } + } + + return self +} + +Yallist.prototype.removeNode = function (node) { + if (node.list !== this) { + throw new Error('removing node which does not belong to this list') + } + + var next = node.next + var prev = node.prev + + if (next) { + next.prev = prev + } + + if (prev) { + prev.next = next + } + + if (node === this.head) { + this.head = next + } + if (node === this.tail) { + this.tail = prev + } + + node.list.length-- + node.next = null + node.prev = null + node.list = null +} + +Yallist.prototype.unshiftNode = function (node) { + if (node === this.head) { + return + } + + if (node.list) { + node.list.removeNode(node) + } + + var head = this.head + node.list = this + node.next = head + if (head) { + head.prev = node + } + + this.head = node + if (!this.tail) { + this.tail = node + } + this.length++ +} + +Yallist.prototype.pushNode = function (node) { + if (node === this.tail) { + return + } + + if (node.list) { + node.list.removeNode(node) + } + + var tail = this.tail + node.list = this + node.prev = tail + if (tail) { + tail.next = node + } + + this.tail = node + if (!this.head) { + this.head = node + } + this.length++ +} + +Yallist.prototype.push = function () { + for (var i = 0, l = arguments.length; i < l; i++) { + push(this, arguments[i]) + } + return this.length +} + +Yallist.prototype.unshift = function () { + for (var i = 0, l = arguments.length; i < l; i++) { + unshift(this, arguments[i]) + } + return this.length +} + +Yallist.prototype.pop = function () { + if (!this.tail) { + return undefined + } + + var res = this.tail.value + this.tail = this.tail.prev + if (this.tail) { + this.tail.next = null + } else { + this.head = null + } + this.length-- + return res +} + +Yallist.prototype.shift = function () { + if (!this.head) { + return undefined + } + + var res = this.head.value + this.head = this.head.next + if (this.head) { + this.head.prev = null + } else { + this.tail = null + } + this.length-- + return res +} + +Yallist.prototype.forEach = function (fn, thisp) { + thisp = thisp || this + for (var walker = this.head, i = 0; walker !== null; i++) { + fn.call(thisp, walker.value, i, this) + walker = walker.next + } +} + +Yallist.prototype.forEachReverse = function (fn, thisp) { + thisp = thisp || this + for (var walker = this.tail, i = this.length - 1; walker !== null; i--) { + fn.call(thisp, walker.value, i, this) + walker = walker.prev + } +} + +Yallist.prototype.get = function (n) { + for (var i = 0, walker = this.head; walker !== null && i < n; i++) { + // abort out of the list early if we hit a cycle + walker = walker.next + } + if (i === n && walker !== null) { + return walker.value + } +} + +Yallist.prototype.getReverse = function (n) { + for (var i = 0, walker = this.tail; walker !== null && i < n; i++) { + // abort out of the list early if we hit a cycle + walker = walker.prev + } + if (i === n && walker !== null) { + return walker.value + } +} + +Yallist.prototype.map = function (fn, thisp) { + thisp = thisp || this + var res = new Yallist() + for (var walker = this.head; walker !== null;) { + res.push(fn.call(thisp, walker.value, this)) + walker = walker.next + } + return res +} + +Yallist.prototype.mapReverse = function (fn, thisp) { + thisp = thisp || this + var res = new Yallist() + for (var walker = this.tail; walker !== null;) { + res.push(fn.call(thisp, walker.value, this)) + walker = walker.prev + } + return res +} + +Yallist.prototype.reduce = function (fn, initial) { + var acc + var walker = this.head + if (arguments.length > 1) { + acc = initial + } else if (this.head) { + walker = this.head.next + acc = this.head.value + } else { + throw new TypeError('Reduce of empty list with no initial value') + } + + for (var i = 0; walker !== null; i++) { + acc = fn(acc, walker.value, i) + walker = walker.next + } + + return acc +} + +Yallist.prototype.reduceReverse = function (fn, initial) { + var acc + var walker = this.tail + if (arguments.length > 1) { + acc = initial + } else if (this.tail) { + walker = this.tail.prev + acc = this.tail.value + } else { + throw new TypeError('Reduce of empty list with no initial value') + } + + for (var i = this.length - 1; walker !== null; i--) { + acc = fn(acc, walker.value, i) + walker = walker.prev + } + + return acc +} + +Yallist.prototype.toArray = function () { + var arr = new Array(this.length) + for (var i = 0, walker = this.head; walker !== null; i++) { + arr[i] = walker.value + walker = walker.next + } + return arr +} + +Yallist.prototype.toArrayReverse = function () { + var arr = new Array(this.length) + for (var i = 0, walker = this.tail; walker !== null; i++) { + arr[i] = walker.value + walker = walker.prev + } + return arr +} + +Yallist.prototype.slice = function (from, to) { + to = to || this.length + if (to < 0) { + to += this.length + } + from = from || 0 + if (from < 0) { + from += this.length + } + var ret = new Yallist() + if (to < from || to < 0) { + return ret + } + if (from < 0) { + from = 0 + } + if (to > this.length) { + to = this.length + } + for (var i = 0, walker = this.head; walker !== null && i < from; i++) { + walker = walker.next + } + for (; walker !== null && i < to; i++, walker = walker.next) { + ret.push(walker.value) + } + return ret +} + +Yallist.prototype.sliceReverse = function (from, to) { + to = to || this.length + if (to < 0) { + to += this.length + } + from = from || 0 + if (from < 0) { + from += this.length + } + var ret = new Yallist() + if (to < from || to < 0) { + return ret + } + if (from < 0) { + from = 0 + } + if (to > this.length) { + to = this.length + } + for (var i = this.length, walker = this.tail; walker !== null && i > to; i--) { + walker = walker.prev + } + for (; walker !== null && i > from; i--, walker = walker.prev) { + ret.push(walker.value) + } + return ret +} + +Yallist.prototype.reverse = function () { + var head = this.head + var tail = this.tail + for (var walker = head; walker !== null; walker = walker.prev) { + var p = walker.prev + walker.prev = walker.next + walker.next = p + } + this.head = tail + this.tail = head + return this +} + +function push (self, item) { + self.tail = new Node(item, self.tail, null, self) + if (!self.head) { + self.head = self.tail + } + self.length++ +} + +function unshift (self, item) { + self.head = new Node(item, null, self.head, self) + if (!self.tail) { + self.tail = self.head + } + self.length++ +} + +function Node (value, prev, next, list) { + if (!(this instanceof Node)) { + return new Node(value, prev, next, list) + } + + this.list = list + this.value = value + + if (prev) { + prev.next = this + this.prev = prev + } else { + this.prev = null + } + + if (next) { + next.prev = this + this.next = next + } else { + this.next = null + } +} + +try { + // add if support for Symbol.iterator is present + require('./iterator.js')(Yallist) +} catch (er) {} + +},{"./iterator.js":2}],4:[function(require,module,exports){ "use strict"; var gapi = _interopRequireWildcard(require("./gapi")); @@ -31,7 +755,7 @@ chrome.runtime.onConnect.addListener(function (port) { }); } else if (msg.type == 4) { calData[msg.data.id].getEvents(new Date(msg.data.start), new Date(msg.data.end)).catch(function (e) { - console.log("cannot load calendar ".concat(msg.data.id)); + console.log("cannot load calendar ".concat(msg.data.id), e); return []; }).then(function (data) { console.log(data); @@ -66,7 +790,7 @@ chrome.browserAction.onClicked.addListener(function () { }); }); -},{"./gapi":2}],2:[function(require,module,exports){ +},{"./gapi":5}],5:[function(require,module,exports){ "use strict"; Object.defineProperty(exports, "__esModule", { @@ -77,6 +801,10 @@ exports.getCalendars = getCalendars; exports.getColors = getColors; exports.GCalendar = void 0; +var _lruCache = _interopRequireDefault(require("lru-cache")); + +function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } + function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } @@ -99,7 +827,6 @@ function _iterableToArrayLimit(arr, i) { var _arr = []; var _n = true; var _d = function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } -/* global chrome */ var gapi_base = 'https://www.googleapis.com/calendar/v3'; var GApiError = { invalidSyncToken: 1, @@ -107,11 +834,17 @@ var GApiError = { }; function to_params(dict) { - return Object.entries(dict).map(function (_ref) { + return Object.entries(dict).filter(function (_ref) { var _ref2 = _slicedToArray(_ref, 2), k = _ref2[0], v = _ref2[1]; + return v; + }).map(function (_ref3) { + var _ref4 = _slicedToArray(_ref3, 2), + k = _ref4[0], + v = _ref4[1]; + return "".concat(encodeURIComponent(k), "=").concat(encodeURIComponent(v)); }).join('&'); } @@ -161,8 +894,11 @@ function getEvent(calId, eventId, token) { }); } -function getEvents(calId, token, syncToken) { - var resultsPerRequest = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 100; +function _getEvents(calId, token) { + var syncToken = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : null; + var timeMin = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : null; + var timeMax = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : null; + var resultsPerRequest = arguments.length > 5 && arguments[5] !== undefined ? arguments[5] : 100; var results = []; var singleFetch = function singleFetch(pageToken, syncToken) { @@ -170,6 +906,8 @@ function getEvents(calId, token, syncToken) { access_token: token, pageToken: pageToken, syncToken: syncToken, + timeMin: timeMin, + timeMax: timeMax, maxResults: resultsPerRequest })), { method: 'GET', @@ -197,51 +935,116 @@ var GCalendar = /*#__PURE__*/ function () { function GCalendar(calId, name) { + var _this = this; + + var options = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : { + maxCachedItems: 100, + nDaysPerSlot: 10, + largeQuery: 10 + }; + _classCallCheck(this, GCalendar); this.calId = calId; this.name = name; this.token = getAuthToken(); this.syncToken = ''; - this.cache = {}; + this.cache = new _lruCache.default({ + max: options.maxCachedItems, + dispose: function dispose(k, v) { + return _this.onRemoveSlot(k, v); + } + }); + this.eventMeta = {}; + this.options = options; + this.divider = 8.64e7 * this.options.nDaysPerSlot; } _createClass(GCalendar, [{ + key: "dateToCacheKey", + value: function dateToCacheKey(date) { + return Math.floor(date / this.divider); + } + }, { + key: "dateRangeToCacheKeys", + value: function dateRangeToCacheKeys(range) { + return { + start: this.dateToCacheKey(range.start), + end: this.dateToCacheKey(new Date(range.end.getTime() - 1)) + }; + } + }, { key: "getSlot", value: function getSlot(k) { - if (!this.cache[k]) this.cache[k] = {}; - return this.cache[k]; + if (!this.cache.has(k)) { + var res = {}; + this.cache.set(k, res); + return res; + } else return this.cache.get(k); + } + }, { + key: "onRemoveSlot", + value: function onRemoveSlot(k, v) { + for (var id in v) { + console.assert(this.eventMeta[id]); + var keys = this.eventMeta[id].keys; + keys.delete(k); + if (keys.size === 0) delete this.eventMeta[id]; + } + } + }, { + key: "slotStartDate", + value: function slotStartDate(k) { + return new Date(k * this.divider); + } + }, { + key: "slotEndDate", + value: function slotEndDate(k) { + return new Date((k + 1) * this.divider); } }, { key: "addEvent", value: function addEvent(e) { - var ks = GCalendar.dateToCacheKey(e.start); - var ke = GCalendar.dateToCacheKey(new Date(e.end.getTime() - 1)); + var evict = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; + //console.log('adding event', e); + if (this.eventMeta.hasOwnProperty(e.id)) this.removeEvent(e); + var r = this.dateRangeToCacheKeys(e); + var ks = r.start; + var ke = r.end; + var t = this.cache.length; + var keys = new Set(); + + for (var i = ks; i <= ke; i++) { + keys.add(i); + if (!this.cache.has(i)) t++; + } + + this.eventMeta[e.id] = { + keys: keys, + summary: e.summary + }; + if (!evict && t > this.options.maxCachedItems) return; if (ks === ke) this.getSlot(ks)[e.id] = { start: e.start, end: e.end, - id: e.id, - summary: e.summary + id: e.id };else { this.getSlot(ks)[e.id] = { start: e.start, - end: GCalendar.slotEndDate(ks), - id: e.id, - summary: e.summary + end: this.slotEndDate(ks), + id: e.id }; this.getSlot(ke)[e.id] = { - start: GCalendar.slotStartDate(ke), + start: this.slotStartDate(ke), end: e.end, - id: e.id, - summary: e.summary + id: e.id }; for (var k = ks + 1; k < ke; k++) { this.getSlot(k)[e.id] = { - start: GCalendar.slotStartDate(k), - end: GCalendar.slotEndDate(k), - id: e.id, - summary: e.summary + start: this.slotStartDate(k), + end: this.slotEndDate(k), + id: e.id }; } } @@ -249,17 +1052,20 @@ function () { }, { key: "removeEvent", value: function removeEvent(e) { - var ks = GCalendar.dateToCacheKey(e.start); - var ke = GCalendar.dateToCacheKey(new Date(e.end.getTime() - 1)); + var _this2 = this; - for (var k = ks; k <= ke; k++) { - delete this.getSlot(k)[e.id]; - } + var keys = this.eventMeta[e.id].keys; + console.assert(keys); + keys.forEach(function (k) { + return delete _this2.getSlot(k)[e.id]; + }); + delete this.eventMeta[e.id]; } }, { key: "getSlotEvents", value: function getSlotEvents(k, start, end) { - var s = this.getSlot(k); + var s = this.getSlot(k); //console.log(s); + var results = []; for (var id in s) { @@ -268,7 +1074,7 @@ function () { id: id, start: s[id].start < start ? start : s[id].start, end: s[id].end > end ? end : s[id].end, - summary: s[id].summary + summary: this.eventMeta[id].summary }); } } @@ -277,10 +1083,11 @@ function () { } }, { key: "getCachedEvents", - value: function getCachedEvents(start, end) { - var ks = GCalendar.dateToCacheKey(start); - var ke = GCalendar.dateToCacheKey(new Date(end.getTime() - 1)); - var results = this.getSlotEvents(ks, start, end); + value: function getCachedEvents(_r) { + var r = this.dateRangeToCacheKeys(_r); + var ks = r.start; + var ke = r.end; + var results = this.getSlotEvents(ks, _r.start, _r.end); for (var k = ks + 1; k < ke; k++) { var s = this.getSlot(k); @@ -290,59 +1097,115 @@ function () { } } - if (ke > ks) results.push.apply(results, _toConsumableArray(this.getSlotEvents(ke, start, end))); + if (ke > ks) results.push.apply(results, _toConsumableArray(this.getSlotEvents(ke, _r.start, _r.end))); return results; } }, { key: "sync", value: function sync() { - var _this = this; + var _this3 = this; return this.token.then(function (token) { - return getEvents(_this.calId, token, _this.syncToken).then(function (r) { - _this.syncToken = r.nextSyncToken; + return _getEvents(_this3.calId, token, _this3.syncToken).then(function (r) { + _this3.syncToken = r.nextSyncToken; var pm_results = r.results.map(function (e) { - return e.start ? Promise.resolve(e) : getEvent(_this.calId, e.id, token); + return e.start ? Promise.resolve(e) : getEvent(_this3.calId, e.id, token); }); return Promise.all(pm_results).then(function (results) { return results.forEach(function (e) { e.start = new Date(e.start.dateTime); e.end = new Date(e.end.dateTime); - if (e.status === 'confirmed') _this.addEvent(e);else if (e.status === 'cancelled') _this.removeEvent(e); + if (e.status === 'confirmed') _this3.addEvent(e);else if (e.status === 'cancelled') _this3.removeEvent(e); }); }); }); }).catch(function (e) { if (e === GApiError.invalidSyncToken) { - _this.syncToken = ''; + _this3.syncToken = ''; - _this.sync(); + _this3.sync(); } else throw e; }); } }, { key: "getEvents", value: function getEvents(start, end) { - var _this2 = this; + var _this4 = this; - return this.sync().then(function () { - return _this2.getCachedEvents(start, end); + var r = this.dateRangeToCacheKeys({ + start: start, + end: end }); - } - }], [{ - key: "dateToCacheKey", - value: function dateToCacheKey(date) { - return Math.floor(date / 8.64e7); - } - }, { - key: "slotStartDate", - value: function slotStartDate(k) { - return new Date(k * 8.64e7); - } - }, { - key: "slotEndDate", - value: function slotEndDate(k) { - return new Date((k + 1) * 8.64e7); + var query = {}; + + for (var k = r.start; k <= r.end; k++) { + if (!this.cache.has(k)) { + if (!query.hasOwnProperty('start')) query.start = k; + query.end = k; + } + } + + console.log("start: ".concat(start, " end: ").concat(end)); + + if (query.hasOwnProperty('start')) { + console.assert(query.start <= query.end); + + if (query.end - query.start + 1 > this.options.largeQuery) { + console.log("encounter large query, use direct fetch"); + return this.token.then(function (token) { + return _getEvents(_this4.calId, token, null, start.toISOString(), end.toISOString()).then(function (r) { + var results = []; + r.results.forEach(function (e) { + console.assert(e.start); + e.start = new Date(e.start.dateTime); + e.end = new Date(e.end.dateTime); + results.push(e); + }); + return results.filter(function (e) { + return !(e.start >= end || e.end <= start); + }).map(function (e) { + return { + id: e.id, + start: e.start < start ? start : e.start, + end: e.end > end ? end : e.end, + summary: e.summary + }; + }); + }); + }); + } + + console.log("fetching short event list"); + return this.token.then(function (token) { + return _getEvents(_this4.calId, token, null, _this4.slotStartDate(query.start).toISOString(), _this4.slotEndDate(query.end).toISOString()).then(function (r) { + if (_this4.syncToken === '') _this4.syncToken = r.nextSyncToken; + return r.results.forEach(function (e) { + if (e.status === 'confirmed') { + console.assert(e.start); + e.start = new Date(e.start.dateTime); + e.end = new Date(e.end.dateTime); + + _this4.addEvent(e, true); + } + }); + }); + }).then(function () { + return _this4.sync(); + }).then(function () { + return _this4.getCachedEvents({ + start: start, + end: end + }); + }); + } else { + console.log("cache hit"); + return this.sync().then(function () { + return _this4.getCachedEvents({ + start: start, + end: end + }); + }); + } } }]); @@ -351,5 +1214,5 @@ function () { exports.GCalendar = GCalendar; -},{}]},{},[1]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/background.js","src/gapi.js"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()","import * as gapi from './gapi';\n\nlet patterns = [];\nlet calendars = {};\nlet calData = {};\n\nchrome.runtime.onConnect.addListener(function(port) {\n    console.assert(port.name == 'main');\n    port.onMessage.addListener(function(msg) {\n        console.log(msg);\n        if (msg.type == 0) {\n            patterns = msg.data;\n        }\n        else if (msg.type == 1) {\n            port.postMessage({ id: msg.id, type: 1, data: patterns });\n        }\n        else if (msg.type == 2) {\n            calendars = msg.data;\n        }\n        else if (msg.type == 3) {\n            port.postMessage({ id: msg.id, type: 3, data: calendars });\n        }\n        else if (msg.type == 4) {\n            calData[msg.data.id].getEvents(new Date(msg.data.start), new Date(msg.data.end))\n                .catch(e => {\n                    console.log(`cannot load calendar ${msg.data.id}`);\n                    return [];\n                })\n                .then(data => {\n                console.log(data);\n                let resp = { id: msg.id, type: 4, data: data.map(e => {\n                    return {\n                        id: e.id,\n                        start: e.start.getTime(),\n                        end: e.end.getTime()\n                    }\n                })};\n                console.log(resp);\n                port.postMessage(resp);\n            });\n        }\n        else if (msg.type == 5) {\n            calendars = msg.data;\n            for (let id in calendars) {\n                if (!calData.hasOwnProperty(id))\n                    calData[id] = new gapi.GCalendar(id, calendars[id].summary);\n            }\n        }\n        else {\n            console.error(\"unknown msg type\");\n        }\n    });\n});\n\nchrome.browserAction.onClicked.addListener(function() {\n    chrome.tabs.create({url: 'index.html'});\n});\n\n","/* global chrome */\nconst gapi_base = 'https://www.googleapis.com/calendar/v3';\n\nconst GApiError = {\n    invalidSyncToken: 1,\n    otherError: 2,\n};\n\nfunction to_params(dict) {\n    return Object.entries(dict).map(([k, v]) => `${encodeURIComponent(k)}=${encodeURIComponent(v)}`).join('&');\n}\n\nexport function getAuthToken() {\n    return new Promise(resolver =>\n        chrome.identity.getAuthToken(\n            {interactive: true}, token => resolver(token)));\n}\n\nexport function getCalendars(token) {\n    return fetch(`${gapi_base}/users/me/calendarList?${to_params({access_token: token})}`,\n            { method: 'GET', async: true })\n        .then(response => response.json())\n        .then(data => data.items);\n}\n\nexport function getColors(token) {\n    return fetch(`${gapi_base}/colors?${to_params({access_token: token})}`,\n        { method: 'GET', async: true })\n        .then(response => response.json());\n}\n\nfunction getEvent(calId, eventId, token) {\n    return fetch(`${gapi_base}/calendars/${calId}/events/${eventId}?${to_params({access_token: token})}`,\n        { method: 'GET', async: true })\n        .then(response => response.json());\n}\n\nfunction getEvents(calId, token, syncToken, resultsPerRequest=100) {\n    let results = [];\n    const singleFetch = (pageToken, syncToken) => fetch(`${gapi_base}/calendars/${calId}/events?${to_params({\n            access_token: token,\n            pageToken,\n            syncToken,\n            maxResults: resultsPerRequest\n        })}`, { method: 'GET', async: true })\n            .then(response => {\n                if (response.status === 200)\n                    return response.json();\n                else if (response.status === 410)\n                    throw GApiError.invalidSyncToken;\n                else throw GApiError.otherErrors;\n            })\n            .then(data => {\n                results.push(...data.items);\n                if (data.nextPageToken) {\n                    return singleFetch(data.nextPageToken, '');\n                } else {\n                    return ({\n                        nextSyncToken: data.nextSyncToken,\n                        results\n                    });\n                }\n            })\n\n    return singleFetch('', syncToken);\n}\n\nexport class GCalendar {\n    constructor(calId, name) {\n        this.calId = calId;\n        this.name = name;\n        this.token = getAuthToken();\n        this.syncToken = '';\n        this.cache = {};\n    }\n\n    static dateToCacheKey(date) {\n        return Math.floor(date / 8.64e7);\n    }\n\n    getSlot(k) {\n        if (!this.cache[k])\n            this.cache[k] = {};\n        return this.cache[k];\n    }\n\n    static slotStartDate(k) { return new Date(k * 8.64e7); }\n    static slotEndDate(k) { return new Date((k + 1) * 8.64e7); }\n\n    addEvent(e) {\n        let ks = GCalendar.dateToCacheKey(e.start);\n        let ke = GCalendar.dateToCacheKey(new Date(e.end.getTime() - 1));\n        if (ks === ke)\n            this.getSlot(ks)[e.id] = {\n                start: e.start,\n                end: e.end,\n                id: e.id,\n                summary: e.summary};\n        else\n        {\n            this.getSlot(ks)[e.id] = {\n                start: e.start,\n                end: GCalendar.slotEndDate(ks),\n                id: e.id,\n                summary: e.summary};\n            this.getSlot(ke)[e.id] = {\n                start: GCalendar.slotStartDate(ke),\n                end: e.end,\n                id: e.id,\n                summary: e.summary};\n            for (let k = ks + 1; k < ke; k++)\n                this.getSlot(k)[e.id] = {\n                    start: GCalendar.slotStartDate(k),\n                    end: GCalendar.slotEndDate(k),\n                    id: e.id,\n                    summary: e.summary};\n        }\n    }\n\n    removeEvent(e) {\n        let ks = GCalendar.dateToCacheKey(e.start);\n        let ke = GCalendar.dateToCacheKey(new Date(e.end.getTime() - 1));\n        for (let k = ks; k <= ke; k++)\n            delete this.getSlot(k)[e.id];\n    }\n\n    getSlotEvents(k, start, end) {\n        let s = this.getSlot(k);\n        let results = [];\n        for (let id in s) {\n            if (!(s[id].start >= end || s[id].end <= start))\n            {\n                results.push({\n                    id,\n                    start: s[id].start < start ? start: s[id].start,\n                    end: s[id].end > end ? end: s[id].end,\n                    summary: s[id].summary\n                });\n            }\n        }\n        return results;\n    }\n\n    getCachedEvents(start, end) {\n        let ks = GCalendar.dateToCacheKey(start);\n        let ke = GCalendar.dateToCacheKey(new Date(end.getTime() - 1));\n        let results = this.getSlotEvents(ks, start, end);\n        for (let k = ks + 1; k < ke; k++)\n        {\n            let s = this.getSlot(k);\n            for (let id in s)\n                results.push(s[id]);\n        }\n        if (ke > ks)\n            results.push(...this.getSlotEvents(ke, start, end));\n        return results;\n    }\n\n    sync() {\n        return this.token.then(token => getEvents(this.calId, token, this.syncToken).then(r => {\n            this.syncToken = r.nextSyncToken;\n            let pm_results = r.results.map(e => e.start ? Promise.resolve(e) : getEvent(this.calId, e.id, token));\n            return Promise.all(pm_results).then(results => results.forEach(e => {\n                e.start = new Date(e.start.dateTime);\n                e.end = new Date(e.end.dateTime);\n                if (e.status === 'confirmed')\n                    this.addEvent(e);\n                else if (e.status === 'cancelled')\n                    this.removeEvent(e);\n            }));\n        })).catch(e => {\n            if (e === GApiError.invalidSyncToken) {\n                this.syncToken = '';\n                this.sync();\n            } else throw e;\n        });\n    }\n\n    getEvents(start, end) {\n        return this.sync().then(() => this.getCachedEvents(start, end));\n    }\n}\n"]} +},{"lru-cache":1}]},{},[4]) +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCJub2RlX21vZHVsZXMvbHJ1LWNhY2hlL2luZGV4LmpzIiwibm9kZV9tb2R1bGVzL3lhbGxpc3QvaXRlcmF0b3IuanMiLCJub2RlX21vZHVsZXMveWFsbGlzdC95YWxsaXN0LmpzIiwic3JjL2JhY2tncm91bmQuanMiLCJzcmMvZ2FwaS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDOVVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNSQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FB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