mirror of
https://github.com/meteor/meteor.git
synced 2026-05-02 03:01:46 -04:00
129 lines
3.7 KiB
JavaScript
129 lines
3.7 KiB
JavaScript
// XXX could use some tests
|
|
|
|
Session = _.extend({}, {
|
|
keys: {},
|
|
|
|
next_id: 1,
|
|
key_callbacks: {}, // key -> id -> func
|
|
key_value_callbacks: {}, // key -> value -> id -> func
|
|
|
|
// XXX remove debugging method (or improve it, but anyway, don't
|
|
// ship it in production)
|
|
dump_state: function () {
|
|
var self = this;
|
|
console.log("=== Session state ===");
|
|
for (var key in self.key_callbacks) {
|
|
var ids = _.keys(self.key_callbacks[key]);
|
|
if (!ids.length)
|
|
continue;
|
|
console.log(key + ": " + _.map(ids, function (x) {return x.substr(0,4);}).join(' '));
|
|
}
|
|
|
|
for (var key in self.key_value_callbacks) {
|
|
for (var value in self.key_value_callbacks[key]) {
|
|
var ids = _.keys(self.key_value_callbacks[key][value]);
|
|
if (!ids.length)
|
|
continue;
|
|
console.log(key + "(" + value + "): " + _.map(ids, function (x) {return x.substr(0,4);}).join(' '));
|
|
}
|
|
}
|
|
},
|
|
|
|
set: function (key, value) {
|
|
var self = this;
|
|
|
|
var old_value = self.keys[key];
|
|
if (value === old_value)
|
|
return;
|
|
self.keys[key] = value;
|
|
self._ensureKey(key);
|
|
|
|
// Old code did this:
|
|
// for (var id in self.key_callbacks[key])
|
|
// self.key_callbacks[key][id]();
|
|
//
|
|
// This works on all browsers except IE8. The problem is that the
|
|
// callback we call will (most likely) add a new callback to
|
|
// self.key_callbacks. On IE8 this results in an infinite
|
|
// loop. Suprisingly, this does work on other browsers, they don't
|
|
// include newly added keys in the iteration.
|
|
//
|
|
// Work around this on IE8 by iterating over a copy. We need to be
|
|
// careful though and not try to call callbacks that have been
|
|
// removed since we make our copy.
|
|
|
|
_.each(_.keys(self.key_callbacks[key]), function (id) {
|
|
var f = self.key_callbacks[key][id];
|
|
f && f();
|
|
});
|
|
|
|
if (old_value in self.key_value_callbacks[key])
|
|
_.each(_.keys(self.key_value_callbacks[key][old_value]), function (id) {
|
|
var f = self.key_value_callbacks[key][old_value][id];
|
|
f && f();
|
|
});
|
|
|
|
if (value in self.key_value_callbacks[key])
|
|
_.each(_.keys(self.key_value_callbacks[key][value]), function (id) {
|
|
var f = self.key_value_callbacks[key][value][id];
|
|
f && f();
|
|
});
|
|
},
|
|
|
|
get: function (key) {
|
|
var self = this;
|
|
|
|
if (Sky.deps.monitoring) {
|
|
var id = self.next_id++;
|
|
|
|
self._ensureKey(key);
|
|
self.key_callbacks[key][id] = Sky.deps.getInvalidate();
|
|
|
|
Sky.deps.cleanup(function () {
|
|
delete self.key_callbacks[key][id];
|
|
});
|
|
}
|
|
|
|
return self.keys[key];
|
|
},
|
|
|
|
equals: function (key, value) {
|
|
var self = this;
|
|
|
|
if (typeof(value) !== 'string' &&
|
|
typeof(value) !== 'number' &&
|
|
typeof(value) !== 'boolean' &&
|
|
value !== null && value !== undefined)
|
|
throw new Error("Session.equals: value can't be an object");
|
|
|
|
if (Sky.deps.monitoring) {
|
|
var id = self.next_id++;
|
|
|
|
self._ensureKey(key);
|
|
if (!(value in self.key_value_callbacks[key]))
|
|
self.key_value_callbacks[key][value] = {};
|
|
self.key_value_callbacks[key][value][id] = Sky.deps.getInvalidate();
|
|
|
|
Sky.deps.cleanup(function () {
|
|
delete self.key_value_callbacks[key][value][id];
|
|
|
|
// clean up [key][value] if it's now empty, so we don't use
|
|
// O(n) memory for n = values seen ever
|
|
for (var x in self.key_value_callbacks[key][value])
|
|
return;
|
|
delete self.key_value_callbacks[key][value];
|
|
});
|
|
}
|
|
|
|
return self.keys[key] === value;
|
|
},
|
|
|
|
_ensureKey: function (key) {
|
|
var self = this;
|
|
if (!(key in self.key_callbacks)) {
|
|
self.key_callbacks[key] = {};
|
|
self.key_value_callbacks[key] = {};
|
|
}
|
|
}
|
|
});
|