diff --git a/package.json b/package.json index 630b5add5..419062be2 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "remotestoragejs", "desc": "Client side JavaScript library for integrating remoteStorage", - "version": "0.10.0-beta", + "version": "0.10.0-beta2", "private": false, "license": "MIT", "repository": { diff --git a/release/0.10.0-beta2/remotestorage-nocache.amd.js b/release/0.10.0-beta2/remotestorage-nocache.amd.js new file mode 100644 index 000000000..ea0b912fc --- /dev/null +++ b/release/0.10.0-beta2/remotestorage-nocache.amd.js @@ -0,0 +1,5529 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ +define([], function() { + +/** FILE: lib/promising.js **/ +(function(global) { + function getPromise(builder) { + var promise; + + if(typeof(builder) === 'function') { + setTimeout(function() { + try { + builder(promise); + } catch(e) { + promise.reject(e); + } + }, 0); + } + + var consumers = [], success, result; + + function notifyConsumer(consumer) { + if(success) { + var nextValue; + if(consumer.fulfilled) { + try { + nextValue = [consumer.fulfilled.apply(null, result)]; + } catch(exc) { + consumer.promise.reject(exc); + return; + } + } else { + nextValue = result; + } + if(nextValue[0] && typeof(nextValue[0].then) === 'function') { + nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill.apply(null, nextValue); + } + } else { + if(consumer.rejected) { + var ret; + try { + ret = consumer.rejected.apply(null, result); + } catch(exc) { + consumer.promise.reject(exc); + return; + } + if(ret && typeof(ret.then) === 'function') { + ret.then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill(ret); + } + } else { + consumer.promise.reject.apply(null, result); + } + } + } + + function resolve(succ, res) { + if(result) { + console.error("WARNING: Can't resolve promise, already resolved!"); + return; + } + success = succ; + result = Array.prototype.slice.call(res); + setTimeout(function() { + var cl = consumers.length; + if(cl === 0 && (! success)) { + console.error("Possibly uncaught error: ", result, result[0] && result[0].stack); + } + for(var i=0;i. + */ + RemoteStorage.log = function() { + if (RemoteStorage._log) { + console.log.apply(console, arguments); + } + }; + + + RemoteStorage.prototype = { + /** + * Method: connect + * + * Connect to a remoteStorage server. + * + * Parameters: + * userAddress - The user address (user@host) to connect to. + * + * Discovers the webfinger profile of the given user address and + * initiates the OAuth dance. + * + * This method must be called *after* all required access has been claimed. + * + */ + connect: function(userAddress) { + this.setBackend('remotestorage'); + if (userAddress.indexOf('@') < 0) { + this._emit('error', new RemoteStorage.DiscoveryError("User adress doesn't contain an @.")); + return; + } + this.remote.configure(userAddress); + this._emit('connecting'); + + var discoveryTimeout = setTimeout(function() { + this._emit('error', new RemoteStorage.DiscoveryError("No storage information found at that user address.")); + }.bind(this), 5000); + + RemoteStorage.Discover(userAddress, function(href, storageApi, authURL) { + clearTimeout(discoveryTimeout); + if (!href) { + this._emit('error', new RemoteStorage.DiscoveryError("Failed to contact storage server.")); + return; + } + this._emit('authing'); + this.remote.configure(userAddress, href, storageApi); + if (! this.remote.connected) { + this.authorize(authURL); + } + }.bind(this)); + }, + + /** + * Method: disconnect + * + * "Disconnect" from remotestorage server to terminate current session. + * This method clears all stored settings and deletes the entire local + * cache. + */ + disconnect: function() { + if (this.remote) { + this.remote.configure(null, null, null, null); + } + this._setGPD({ + get: this._pendingGPD('get'), + put: this._pendingGPD('put'), + delete: this._pendingGPD('delete') + }); + var n = this._cleanups.length, i = 0; + + var oneDone = function() { + i++; + if (i >= n) { + this._init(); + RemoteStorage.log('Done cleaning up, emitting disconnected and disconnect events'); + this._emit('disconnected'); + } + }.bind(this); + + if (n > 0) { + this._cleanups.forEach(function(cleanup) { + var cleanupResult = cleanup(this); + if (typeof(cleanup) === 'object' && typeof(cleanup.then) === 'function') { + cleanupResult.then(oneDone); + } else { + oneDone(); + } + }.bind(this)); + } else { + oneDone(); + } + }, + + setBackend: function(what) { + this.backend = what; + if (this.localStorageAvailable()) { + if (what) { + localStorage['remotestorage:backend'] = what; + } else { + delete localStorage['remotestorage:backend']; + } + } + }, + + /** + * Method: onChange + * + * Add a "change" event handler to the given path. Whenever a "change" + * happens (as determined by the backend, such as e.g. + * ) and the affected path is equal to or below + * the given 'path', the given handler is called. + * + * You should usually not use this method directly, but instead use the + * "change" events provided by . + * + * Parameters: + * path - Absolute path to attach handler to. + * handler - Handler function. + */ + onChange: function(path, handler) { + if (! this._pathHandlers.change[path]) { + this._pathHandlers.change[path] = []; + } + this._pathHandlers.change[path].push(handler); + }, + + /** + * Method: enableLog + * + * Enable remoteStorage logging + */ + enableLog: function() { + RemoteStorage._log = true; + }, + + /** + * Method: disableLog + * + * Disable remoteStorage logging + */ + disableLog: function() { + RemoteStorage._log = false; + }, + + /** + * Method: log + * + * The same as . + */ + log: function() { + RemoteStorage.log.apply(RemoteStorage, arguments); + }, + + setApiKeys: function(type, keys) { + if (keys) { + this.apiKeys[type] = keys; + } else { + delete this.apiKeys[type]; + } + if (this.localStorageAvailable()) { + localStorage['remotestorage:api-keys'] = JSON.stringify(this.apiKeys); + } + }, + + /** + ** INITIALIZATION + **/ + + _init: function() { + var self = this, + readyFired = false; + + function fireReady() { + try { + if (!readyFired) { + self._emit('ready'); + readyFired = true; + } + } catch(e) { + console.error("'ready' failed: ", e, e.stack); + self._emit('error', e); + } + } + + this._loadFeatures(function(features) { + this.log('[RemoteStorage] All features loaded'); + this.local = features.local && new features.local(); + // this.remote set by WireClient._rs_init as lazy property on + // RS.prototype + + if (this.local && this.remote) { + this._setGPD(SyncedGetPutDelete, this); + this._bindChange(this.local); + } else if (this.remote) { + this._setGPD(this.remote, this.remote); + } + + if (this.remote) { + this.remote.on('connected', function(){ + fireReady(); + self._emit('connected'); + }); + this.remote.on('not-connected', function(){ + fireReady(); + self._emit('not-connected'); + }); + if (this.remote.connected) { + fireReady(); + self._emit('connected'); + } + } + + this._collectCleanupFunctions(); + + try { + this._allLoaded = true; + this._emit('features-loaded'); + } catch(exc) { + logError(exc); + this._emit('error', exc); + } + this._processPending(); + }.bind(this)); + }, + + _collectCleanupFunctions: function() { + for (var i=0; i < this.features.length; i++) { + var cleanup = this.features[i].cleanup; + if (typeof(cleanup) === 'function') { + this._cleanups.push(cleanup); + } + } + }, + + /** + ** FEATURE DETECTION + **/ + _loadFeatures: function(callback) { + var featureList = [ + 'WireClient', + 'I18n', + 'Dropbox', + 'GoogleDrive', + 'Access', + 'Caching', + 'Discover', + 'Authorize', + 'Widget', + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage', + 'Sync', + 'BaseClient', + 'Env' + ]; + var features = []; + var featuresDone = 0; + var self = this; + + function featureDone() { + featuresDone++; + if (featuresDone === featureList.length) { + setTimeout(function() { + features.caching = !!RemoteStorage.Caching; + features.sync = !!RemoteStorage.Sync; + [ + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage' + ].some(function(cachingLayer) { + if (features.some(function(feature) { return feature.name === cachingLayer; })) { + features.local = RemoteStorage[cachingLayer]; + return true; + } + }); + self.features = features; + callback(features); + }, 0); + } + } + + function featureInitialized(name) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialized."); + features.push({ + name : name, + init : RemoteStorage[name]._rs_init, + supported : true, + cleanup : RemoteStorage[name]._rs_cleanup + }); + featureDone(); + } + + function featureFailed(name, err) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialization failed ( "+err+")"); + featureDone(); + } + + function featureSupported(name, success) { + self.log("[RemoteStorage] [FEATURE "+name+"]" + success ? "":" not"+" supported"); + if (!success) { + featureDone(); + } + } + + function initFeature(name) { + var initResult; + try { + initResult = RemoteStorage[name]._rs_init(self); + } catch(e) { + featureFailed(name, e); + return; + } + if (typeof(initResult) === 'object' && typeof(initResult.then) === 'function') { + initResult.then( + function(){ featureInitialized(name); }, + function(err){ featureFailed(name, err); } + ); + } else { + featureInitialized(name); + } + } + + featureList.forEach(function(featureName) { + self.log("[RemoteStorage] [FEATURE " + featureName + "] initializing..."); + var impl = RemoteStorage[featureName]; + var supported; + + if (impl) { + supported = !impl._rs_supported || impl._rs_supported(); + + if (typeof supported === 'object') { + supported.then( + function(){ + featureSupported(featureName, true); + initFeature(featureName); + }, + function(){ + featureSupported(featureName, false); + } + ); + } + else if (typeof supported === 'boolean') { + featureSupported(featureName, supported); + if (supported) { + initFeature(featureName); + } + } + } else { + featureSupported(featureName, false); + } + }); + }, + + localStorageAvailable: function() { + try { + return !!global.localStorage; + } catch(error) { + return false; + } + }, + + /** + ** GET/PUT/DELETE INTERFACE HELPERS + **/ + + _setGPD: function(impl, context) { + function wrap(f) { + return function() { + return f.apply(context, arguments) + .then(emitUnauthorized.bind(this)); + }; + } + this.get = wrap(impl.get); + this.put = wrap(impl.put); + this.delete = wrap(impl.delete); + }, + + _pendingGPD: function(methodName) { + return function() { + var promise = promising(); + this._pending.push({ + method: methodName, + args: Array.prototype.slice.call(arguments), + promise: promise + }); + return promise; + }.bind(this); + }, + + _processPending: function() { + this._pending.forEach(function(pending) { + try { + this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject); + } catch(e) { + pending.promise.reject(e); + } + }.bind(this)); + this._pending = []; + }, + + /** + ** CHANGE EVENT HANDLING + **/ + + _bindChange: function(object) { + object.on('change', this._dispatchEvent.bind(this, 'change')); + }, + + _dispatchEvent: function(eventName, event) { + for (var path in this._pathHandlers[eventName]) { + var pl = path.length; + var self = this; + this._pathHandlers[eventName][path].forEach(function(handler) { + if (event.path.substr(0, pl) === path) { + var ev = {}; + for (var key in event) { ev[key] = event[key]; } + ev.relativePath = event.path.replace(new RegExp('^' + path), ''); + try { + handler(ev); + } catch(e) { + console.error("'change' handler failed: ", e, e.stack); + self._emit('error', e); + } + } + }); + } + } + }; + + /** + * Property: connected + * + * Boolean property indicating if remoteStorage is currently connected. + */ + Object.defineProperty(RemoteStorage.prototype, 'connected', { + get: function() { + return this.remote.connected; + } + }); + + /** + * Property: access + * + * Tracking claimed access scopes. A instance. + * + * + * Property: caching + * + * Caching settings. A instance. + * + * Not available in no-cache builds. + * + * + * Property: remote + * + * Access to the remote backend used. Usually a . + * + * + * Property: local + * + * Access to the local caching backend used. Usually either a + * or instance. + * + * Not available in no-cache builds. + */ + + global.RemoteStorage = RemoteStorage; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/eventhandling.js **/ +(function(global) { + /** + * Interface: eventhandling + */ + var methods = { + /** + * Method: addEventListener + * + * Install an event handler for the given event name + */ + addEventListener: function(eventName, handler) { + if (typeof(eventName) !== 'string') { + throw new Error('Argument eventName should be a string'); + } + if (typeof(handler) !== 'function') { + throw new Error('Argument handler should be a function'); + } + RemoteStorage.log('[Eventhandling] Adding event listener', eventName, handler); + this._validateEvent(eventName); + this._handlers[eventName].push(handler); + }, + + /** + * Method: removeEventListener + * + * Remove a previously installed event handler + */ + removeEventListener: function(eventName, handler) { + this._validateEvent(eventName); + var hl = this._handlers[eventName].length; + for (var i=0;i + **/ + methods.on = methods.addEventListener; + + /** + * Function: eventHandling + * + * Mixes event handling functionality into an object. + * + * The first parameter is always the object to be extended. + * All remaining parameter are expected to be strings, interpreted as valid event + * names. + * + * Example: + * (start code) + * var MyConstructor = function() { + * eventHandling(this, 'connected', 'disconnected'); + * + * this._emit('connected'); + * this._emit('disconnected'); + * // This would throw an exception: + * // this._emit('something-else'); + * }; + * + * var myObject = new MyConstructor(); + * myObject.on('connected', function() { console.log('connected'); }); + * myObject.on('disconnected', function() { console.log('disconnected'); }); + * // This would throw an exception as well: + * // myObject.on('something-else', function() {}); + * (end code) + */ + RemoteStorage.eventHandling = function(object) { + var eventNames = Array.prototype.slice.call(arguments, 1); + for (var key in methods) { + object[key] = methods[key]; + } + object._handlers = {}; + eventNames.forEach(function(eventName) { + object._addEvent(eventName); + }); + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/wireclient.js **/ +(function(global) { + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.WireClient + * + * WireClient Interface + * -------------------- + * + * This file exposes a get/put/delete interface on top of XMLHttpRequest. + * It requires to be configured with parameters about the remotestorage server to + * connect to. + * Each instance of WireClient is always associated with a single remotestorage + * server and access token. + * + * Usually the WireClient instance can be accessed via `remoteStorage.remote`. + * + * This is the get/put/delete interface: + * + * - #get() takes a path and optionally a ifNoneMatch option carrying a version + * string to check. It returns a promise that will be fulfilled with the HTTP + * response status, the response body, the MIME type as returned in the + * 'Content-Type' header and the current revision, as returned in the 'ETag' + * header. + * - #put() takes a path, the request body and a content type string. It also + * accepts the ifMatch and ifNoneMatch options, that map to the If-Match and + * If-None-Match headers respectively. See the remotestorage-01 specification + * for details on handling these headers. It returns a promise, fulfilled with + * the same values as the one for #get(). + * - #delete() takes a path and the ifMatch option as well. It returns a promise + * fulfilled with the same values as the one for #get(). + * + * In addition to this, the WireClient has some compatibility features to work with + * remotestorage 2012.04 compatible storages. For example it will cache revisions + * from folder listings in-memory and return them accordingly as the "revision" + * parameter in response to #get() requests. Similarly it will return 404 when it + * receives an empty folder listing, to mimic remotestorage-01 behavior. Note + * that it is not always possible to know the revision beforehand, hence it may + * be undefined at times (especially for caching-roots). + */ + + var hasLocalStorage; + var SETTINGS_KEY = "remotestorage:wireclient"; + + var API_2012 = 1, API_00 = 2, API_01 = 3, API_02 = 4, API_HEAD = 5; + + var STORAGE_APIS = { + 'draft-dejong-remotestorage-00': API_00, + 'draft-dejong-remotestorage-01': API_01, + 'draft-dejong-remotestorage-02': API_02, + 'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012 + }; + + var isArrayBufferView; + + if (typeof(ArrayBufferView) === 'function') { + isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); }; + } else { + var arrayBufferViews = [ + Int8Array, Uint8Array, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array + ]; + isArrayBufferView = function(object) { + for (var i=0;i<8;i++) { + if (object instanceof arrayBufferViews[i]) { + return true; + } + } + return false; + }; + } + + function addQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + if (str === '*') { + return '*'; + } + + return '"' + str + '"'; + } + + function stripQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + + return str.replace(/^["']|["']$/g, ''); + } + + function readBinaryData(content, mimeType, callback) { + var blob = new Blob([content], { type: mimeType }); + var reader = new FileReader(); + reader.addEventListener("loadend", function() { + callback(reader.result); // reader.result contains the contents of blob as a typed array + }); + reader.readAsArrayBuffer(blob); + } + + function cleanPath(path) { + return path.replace(/\/+/g, '/').split('/').map(encodeURIComponent).join('/'); + } + + function isFolder(path) { + return (path.substr(-1) === '/'); + } + + function isFolderDescription(body) { + return ((body['@context'] === 'http://remotestorage.io/spec/folder-description') + && (typeof(body['items']) === 'object')); + } + + function isSuccessStatus(status) { + return [201, 204, 304].indexOf(status) >= 0; + } + + function isErrorStatus(status) { + return [401, 403, 404, 412].indexOf(status) >= 0; + } + + var onErrorCb; + + /** + * Class : RemoteStorage.WireClient + **/ + RS.WireClient = function(rs) { + this.connected = false; + + /** + * Event: change + * never fired for some reason + * + * Event: connected + * fired when the wireclient connect method realizes that it is + * in posession of a token and a href + **/ + RS.eventHandling(this, 'change', 'connected', 'wire-busy', 'wire-done', 'not-connected'); + + onErrorCb = function(error){ + if (error instanceof RemoteStorage.Unauthorized) { + this.configure(undefined, undefined, undefined, null); + } + }.bind(this); + rs.on('error', onErrorCb); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + setTimeout(function() { + this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token); + }.bind(this), 0); + } + } + + this._revisionCache = {}; + + if (this.connected) { + setTimeout(this._emit.bind(this), 0, 'connected'); + } + }; + + RS.WireClient.REQUEST_TIMEOUT = 30000; + + RS.WireClient.prototype = { + /** + * Property: token + * + * Holds the bearer token of this WireClient, as obtained in the OAuth dance + * + * Example: + * (start code) + * + * remoteStorage.remote.token + * // -> 'DEADBEEF01==' + */ + + /** + * Property: href + * + * Holds the server's base URL, as obtained in the Webfinger discovery + * + * Example: + * (start code) + * + * remoteStorage.remote.href + * // -> 'https://storage.example.com/users/jblogg/' + */ + + /** + * Property: storageApi + * + * Holds the spec version the server claims to be compatible with + * + * Example: + * (start code) + * + * remoteStorage.remote.storageApi + * // -> 'draft-dejong-remotestorage-01' + */ + + _request: function(method, uri, token, headers, body, getEtag, fakeRevision) { + if ((method === 'PUT' || method === 'DELETE') && uri[uri.length - 1] === '/') { + throw "Don't " + method + " on directories!"; + } + + var promise = promising(); + var revision; + var reqType; + var self = this; + + headers['Authorization'] = 'Bearer ' + token; + + this._emit('wire-busy', { + method: method, + isFolder: isFolder(uri) + }); + + RS.WireClient.request(method, uri, { + body: body, + headers: headers + }, function(error, response) { + if (error) { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: false + }); + self.online = false; + promise.reject(error); + } else { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: true + }); + self.online = true; + if (isErrorStatus(response.status)) { + RemoteStorage.log('[WireClient] Error response status', response.status); + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = undefined; + } + promise.fulfill(response.status, undefined, undefined, revision); + } else if (isSuccessStatus(response.status) || + (response.status === 200 && method !== 'GET')) { + revision = stripQuotes(response.getResponseHeader('ETag')); + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, undefined, undefined, revision); + } else { + var mimeType = response.getResponseHeader('Content-Type'); + var body; + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = response.status === 200 ? fakeRevision : undefined; + } + + if ((!mimeType) || mimeType.match(/charset=binary/)) { + RS.WireClient.readBinaryData(response.response, mimeType, function(result) { + RemoteStorage.log('[WireClient] Successful request with unknown or binary mime-type', revision); + promise.fulfill(response.status, result, mimeType, revision); + }); + } else { + if (mimeType && mimeType.match(/^application\/json/)) { + body = JSON.parse(response.responseText); + } else { + body = response.responseText; + } + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, body, mimeType, revision); + } + } + } + }); + return promise; + }, + + configure: function(userAddress, href, storageApi, token) { + if (typeof(userAddress) !== 'undefined') { + this.userAddress = userAddress; + } + if (typeof(href) !== 'undefined') { + this.href = href; + } + if (typeof(storageApi) !== 'undefined') { + this.storageApi = storageApi; + } + if (typeof(token) !== 'undefined') { + this.token = token; + } + if (typeof(this.storageApi) !== 'undefined') { + this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD; + this.supportsRevs = this._storageApi >= API_00; + } + if (this.href && this.token) { + this.connected = true; + this.online = true; + this._emit('connected'); + } else { + this.connected = false; + } + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ + userAddress: this.userAddress, + href: this.href, + token: this.token, + storageApi: this.storageApi + }); + } + RS.WireClient.configureHooks.forEach(function(hook) { + hook.call(this); + }.bind(this)); + }, + + stopWaitingForToken: function() { + if (!this.connected) { + this._emit('not-connected'); + } + }, + + get: function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } else if (options.ifNoneMatch) { + var oldRev = this._revisionCache[path]; + } + var promise = this._request('GET', this.href + cleanPath(path), this.token, headers, + undefined, this.supportsRevs, this._revisionCache[path]); + if (isFolder(path)) { + return promise.then(function(status, body, contentType, revision) { + var itemsMap = {}; + + // New folder listing received + if (status === 200 && typeof(body) === 'object') { + // Empty folder listing of any spec + if (Object.keys(body).length === 0) { + status = 404; + } + // >= 02 spec + else if (isFolderDescription(body)) { + for (var item in body.items) { + this._revisionCache[path + item] = body.items[item].ETag; + } + itemsMap = body.items; + } + // < 02 spec + else { + Object.keys(body).forEach(function(key){ + this._revisionCache[path + key] = body[key]; + itemsMap[key] = {"ETag": body[key]}; + }.bind(this)); + } + return promising().fulfill(status, itemsMap, contentType, revision); + } else { + return promising().fulfill(status, body, contentType, revision); + } + }.bind(this)); + } else { + return promise; + } + }, + + put: function(path, body, contentType, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + if (!contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var headers = { 'Content-Type': contentType }; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } + return this._request('PUT', this.href + cleanPath(path), this.token, + headers, body, this.supportsRevs); + }, + + 'delete': function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + } + return this._request('DELETE', this.href + cleanPath(path), this.token, + headers, + undefined, this.supportsRevs); + } + }; + + // Shared cleanPath used by Dropbox + RS.WireClient.cleanPath = cleanPath; + + // Shared isArrayBufferView used by WireClient and Dropbox + RS.WireClient.isArrayBufferView = isArrayBufferView; + + RS.WireClient.readBinaryData = readBinaryData; + + // Shared request function used by WireClient, GoogleDrive and Dropbox. + RS.WireClient.request = function(method, url, options, callback) { + RemoteStorage.log('[WireClient]', method, url); + + callback = callback.bind(this); + + var timedOut = false; + + var timer = setTimeout(function() { + timedOut = true; + callback('timeout'); + }, RS.WireClient.REQUEST_TIMEOUT); + + var xhr = new XMLHttpRequest(); + xhr.open(method, url, true); + + if (options.responseType) { + xhr.responseType = options.responseType; + } + if (options.headers) { + for (var key in options.headers) { + xhr.setRequestHeader(key, options.headers[key]); + } + } + + xhr.onload = function() { + if (timedOut) { return; } + clearTimeout(timer); + callback(null, xhr); + }; + + xhr.onerror = function(error) { + if (timedOut) { return; } + clearTimeout(timer); + callback(error); + }; + + var body = options.body; + + if (typeof(body) === 'object') { + if (isArrayBufferView(body)) { + /* alright. */ + //FIXME empty block + } + else if (body instanceof ArrayBuffer) { + body = new Uint8Array(body); + } else { + body = JSON.stringify(body); + } + } + xhr.send(body); + }; + + Object.defineProperty(RemoteStorage.WireClient.prototype, 'storageType', { + get: function() { + if (this.storageApi) { + var spec = this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/); + return spec ? spec[1] : '2012.04'; + } + } + }); + + RS.WireClient.configureHooks = []; + + RS.WireClient._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + remoteStorage.remote = new RS.WireClient(remoteStorage); + this.online = true; + }; + + RS.WireClient._rs_supported = function() { + return !! global.XMLHttpRequest; + }; + + RS.WireClient._rs_cleanup = function(remoteStorage){ + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + remoteStorage.removeEventListener('error', onErrorCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/discover.js **/ +(function(global) { + + // feature detection flags + var haveXMLHttpRequest, hasLocalStorage; + // used to store settings in localStorage + var SETTINGS_KEY = 'remotestorage:discover'; + // cache loaded from localStorage + var cachedInfo = {}; + + /** + * Class: RemoteStorage.Discover + * + * This class deals with the webfinger lookup + * + * Arguments: + * userAddress - user@host + * callback - gets called with href of the storage, the type and the authURL + * Example: + * (start code) + * + * (end code) + **/ + + RemoteStorage.Discover = function(userAddress, callback) { + if (userAddress in cachedInfo) { + var info = cachedInfo[userAddress]; + callback(info.href, info.type, info.authURL); + return; + } + var hostname = userAddress.split('@')[1]; + var params = '?resource=' + encodeURIComponent('acct:' + userAddress); + var urls = [ + 'https://' + hostname + '/.well-known/webfinger' + params, + 'https://' + hostname + '/.well-known/host-meta.json' + params, + 'http://' + hostname + '/.well-known/webfinger' + params, + 'http://' + hostname + '/.well-known/host-meta.json' + params + ]; + + function tryOne() { + var xhr = new XMLHttpRequest(); + var url = urls.shift(); + if (!url) { return callback(); } + RemoteStorage.log('[Discover] Trying URL', url); + xhr.open('GET', url, true); + xhr.onabort = xhr.onerror = function() { + console.error("webfinger error", arguments, '(', url, ')'); + tryOne(); + }; + xhr.onload = function() { + if (xhr.status !== 200) { return tryOne(); } + var profile; + + try { + profile = JSON.parse(xhr.responseText); + } catch(e) { + RemoteStorage.log("[Discover] Failed to parse profile ", xhr.responseText, e); + tryOne(); + return; + } + + if (!profile.links) { + RemoteStorage.log("[Discover] Profile has no links section ", JSON.stringify(profile)); + tryOne(); + return; + } + + var link; + profile.links.forEach(function(l) { + if (l.rel === 'remotestorage') { + link = l; + } else if (l.rel === 'remoteStorage' && !link) { + link = l; + } + }); + RemoteStorage.log('[Discover] Got profile', profile, 'and link', link); + if (link) { + var authURL = link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'] + || link.properties['auth-endpoint'], + storageType = link.properties['http://remotestorage.io/spec/version'] + || link.type; + cachedInfo[userAddress] = { href: link.href, type: storageType, authURL: authURL }; + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo }); + } + callback(link.href, storageType, authURL); + } else { + tryOne(); + } + }; + xhr.send(); + } + tryOne(); + }; + + RemoteStorage.Discover._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + cachedInfo = settings.cache; + } + } + }; + + RemoteStorage.Discover._rs_supported = function() { + haveXMLHttpRequest = !! global.XMLHttpRequest; + return haveXMLHttpRequest; + }; + + RemoteStorage.Discover._rs_cleanup = function() { + if (hasLocalStorage) { + delete localStorage[SETTINGS_KEY]; + } + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/authorize.js **/ +(function(global) { + + function extractParams() { + //FF already decodes the URL fragment in document.location.hash, so use this instead: + var location = RemoteStorage.Authorize.getLocation(), + hashPos = location.href.indexOf('#'), + hash; + if (hashPos === -1) { return; } + hash = location.href.substring(hashPos+1); + // if hash is not of the form #key=val&key=val, it's probably not for us + if (hash.indexOf('=') === -1) { return; } + return hash.split('&').reduce(function(m, kvs) { + var kv = kvs.split('='); + m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]); + return m; + }, {}); + } + + RemoteStorage.Authorize = function(authURL, scope, redirectUri, clientId) { + RemoteStorage.log('[Authorize] authURL = ', authURL); + + var url = authURL; + url += authURL.indexOf('?') > 0 ? '&' : '?'; + url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, '')); + url += '&scope=' + encodeURIComponent(scope); + url += '&client_id=' + encodeURIComponent(clientId); + url += '&response_type=token'; + RemoteStorage.Authorize.setLocation(url); + }; + + RemoteStorage.prototype.authorize = function(authURL) { + this.access.setStorageType(this.remote.storageType); + var scope = this.access.scopeParameter; + + var redirectUri = String(RemoteStorage.Authorize.getLocation()); + var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0]; + + RemoteStorage.Authorize(authURL, scope, redirectUri, clientId); + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.getLocation = function () { + return global.document.location; + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.setLocation = function (location) { + if (typeof location === 'string') { + global.document.location.href = location; + } else if (typeof location === 'object') { + global.document.location = location; + } else { + throw "Invalid location " + location; + } + }; + + RemoteStorage.Authorize._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + + var onFeaturesLoaded; + RemoteStorage.Authorize._rs_init = function(remoteStorage) { + + onFeaturesLoaded = function () { + var authParamsUsed = false; + if (params) { + if (params.error) { + throw "Authorization server errored: " + params.error; + } + if (params.access_token) { + remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token); + authParamsUsed = true; + } + if (params.remotestorage) { + remoteStorage.connect(params.remotestorage); + authParamsUsed = true; + } + } + if (!authParamsUsed) { + remoteStorage.remote.stopWaitingForToken(); + } + }; + var params = extractParams(), + location; + if (params) { + location = RemoteStorage.Authorize.getLocation(); + location.hash = ''; + } + remoteStorage.on('features-loaded', onFeaturesLoaded); + }; + + RemoteStorage.Authorize._rs_cleanup = function(remoteStorage) { + remoteStorage.removeEventListener('features-loaded', onFeaturesLoaded); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/access.js **/ +(function(global) { + + var SETTINGS_KEY = "remotestorage:access"; + + /** + * Class: RemoteStorage.Access + * + * Keeps track of claimed access and scopes. + */ + RemoteStorage.Access = function() { + this.reset(); + }; + + RemoteStorage.Access.prototype = { + + /** + * Method: claim + * + * Claim access on a given scope with given mode. + * + * Parameters: + * scope - An access scope, such as "contacts" or "calendar". + * mode - Access mode to use. Either "r" or "rw". + * + * Example: + * (start code) + * remoteStorage.access.claim('contacts', 'r'); + * remoteStorage.access.claim('pictures', 'rw'); + */ + claim: function(scope, mode) { + if (typeof(scope) !== 'string' || scope.indexOf('/') !== -1 || scope.length === 0) { + throw new Error('Scope should be a non-empty string without forward slashes'); + } + if (!mode.match(/^rw?$/)) { + throw new Error('Mode should be either \'r\' or \'rw\''); + } + this._adjustRootPaths(scope); + this.scopeModeMap[scope] = mode; + }, + + get: function(scope) { + return this.scopeModeMap[scope]; + }, + + remove: function(scope) { + var savedMap = {}; + var name; + for (name in this.scopeModeMap) { + savedMap[name] = this.scopeModeMap[name]; + } + this.reset(); + delete savedMap[scope]; + for (name in savedMap) { + this.set(name, savedMap[name]); + } + }, + + /** + * Verify permission for a given scope. + */ + checkPermission: function(scope, mode) { + var actualMode = this.get(scope); + return actualMode && (mode === 'r' || actualMode === 'rw'); + }, + + /** + * Verify permission for a given path. + */ + checkPathPermission: function(path, mode) { + if (this.checkPermission('*', mode)) { + return true; + } + return !!this.checkPermission(this._getModuleName(path), mode); + }, + + reset: function() { + this.rootPaths = []; + this.scopeModeMap = {}; + }, + + /** + * Return the module name for a given path. + */ + _getModuleName: function(path) { + if (path[0] !== '/') { + throw new Error('Path should start with a slash'); + } + var moduleMatch = path.replace(/^\/public/, '').match(/^\/([^\/]*)\//); + return moduleMatch ? moduleMatch[1] : '*'; + }, + + _adjustRootPaths: function(newScope) { + if ('*' in this.scopeModeMap || newScope === '*') { + this.rootPaths = ['/']; + } else if (! (newScope in this.scopeModeMap)) { + this.rootPaths.push('/' + newScope + '/'); + this.rootPaths.push('/public/' + newScope + '/'); + } + }, + + _scopeNameForParameter: function(scope) { + if (scope.name === '*' && this.storageType) { + if (this.storageType === '2012.04') { + return ''; + } else if (this.storageType.match(/remotestorage-0[01]/)) { + return 'root'; + } + } + return scope.name; + }, + + setStorageType: function(type) { + this.storageType = type; + } + }; + + /** + * Property: scopes + * + * Holds an array of claimed scopes in the form + * > { name: "", mode: "" } + */ + Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', { + get: function() { + return Object.keys(this.scopeModeMap).map(function(key) { + return { name: key, mode: this.scopeModeMap[key] }; + }.bind(this)); + } + }); + + Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', { + get: function() { + return this.scopes.map(function(scope) { + return this._scopeNameForParameter(scope) + ':' + scope.mode; + }.bind(this)).join(' '); + } + }); + + // Documented in src/remotestorage.js + Object.defineProperty(RemoteStorage.prototype, 'access', { + get: function() { + var access = new RemoteStorage.Access(); + Object.defineProperty(this, 'access', { + value: access + }); + return access; + }, + configurable: true + }); + + RemoteStorage.Access._rs_init = function() {}; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/env.js **/ +(function(pMode) { + + var mode = pMode, + env = {}, + isBackground = false; + + + RemoteStorage.Env = function() { + return env; + }; + + RemoteStorage.Env.isBrowser = function () { + return mode === "browser"; + }; + + RemoteStorage.Env.isNode = function () { + return mode === "node"; + }; + + RemoteStorage.Env.goBackground = function () { + isBackground = true; + RemoteStorage.Env._emit("background"); + }; + + RemoteStorage.Env.goForeground = function () { + isBackground = false; + RemoteStorage.Env._emit("foreground"); + }; + + RemoteStorage.Env._rs_init = function(remoteStorage) { + RemoteStorage.eventHandling(RemoteStorage.Env, "background", "foreground"); + + if ( mode === 'browser') { + if ( typeof(document.hidden) !== "undefined" ) { + env.hiddenProperty = "hidden"; + env.visibilityChangeEvent = "visibilitychange"; + } else if ( typeof(document.mozHidden) !== "undefined" ) { + env.hiddenProperty = "mozHidden"; + env.visibilityChangeEvent = "mozvisibilitychange"; + } else if ( typeof(document.msHidden) !== "undefined" ) { + env.hiddenProperty = "msHidden"; + env.visibilityChangeEvent = "msvisibilitychange"; + } else if ( typeof(document.webkitHidden) !== "undefined" ) { + env.hiddenProperty = "webkitHidden"; + env.visibilityChangeEvent = "webkitvisibilitychange"; + } + document.addEventListener(env.visibilityChangeEvent, function () { + if ( document[env.hiddenProperty] ) { + RemoteStorage.Env.goBackground(); + } else { + RemoteStorage.Env.goForeground(); + } + }, false); + } + }; + + RemoteStorage.Env._rs_cleanup = function(remoteStorage) { + }; + +})(typeof(window) !== 'undefined' ? 'browser' : 'node'); + + +/** FILE: src/i18n.js **/ +(function() { + /** + * Class: RemoteStorage.I18n + * + * TODO add documentation + **/ + + "use strict"; + + var dictionary = { + "view_info": 'This app allows you to use your own storage. Learn more!', + "view_connect": "Connect remote storage", + "view_connecting": "Connecting %s", + "view_offline": "Offline", + "view_error_occured": "Sorry! An error occured.", + "view_confirm_reset": "Are you sure you want to reset everything? This will clear your local data and reload the page.", + "view_get_me_out": "Get me out of here!", + "view_error_plz_report": 'If this problem persists, please let us know!', + "view_unauthorized": "Unauthorized! Click here to reconnect." + }; + + RemoteStorage.I18n = { + + translate: function() { + var str = arguments[0], + params = Array.prototype.splice.call(arguments, 1); + + if (typeof dictionary[str] !== "string") { + throw "Unknown translation string: " + str; + } else { + str = dictionary[str]; + } + return (str.replace(/%s/g, function(){ return params.shift(); })); + }, + + getDictionary: function() { + return dictionary; + }, + + setDictionary: function(newDictionary) { + dictionary = newDictionary; + } + + }; +})(); + + +/** FILE: src/assets.js **/ +/** THIS FILE WAS GENERATED BY build/compile-assets.js. DO NOT CHANGE IT MANUALLY, BUT INSTEAD CHANGE THE ASSETS IN assets/. **/ +RemoteStorage.Assets = { + + connectIcon: '', + disconnectIcon: '', + dropbox: '', + googledrive: '', + remoteStorageIcon: '', + remoteStorageIconError: '', + remoteStorageIconOffline: '', + syncIcon: '', + widget: '
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ', + widgetCss: '/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}' +}; + + +/** FILE: src/widget.js **/ +(function(window) { + + var hasLocalStorage; + var LS_STATE_KEY = "remotestorage:widget:state"; + // states allowed to immediately jump into after a reload. + var VALID_ENTRY_STATES = { + initial: true, + connected: true, + offline: true + }; + + function stateSetter(widget, state) { + RemoteStorage.log('[Widget] Producing stateSetter for', state); + return function() { + RemoteStorage.log('[Widget] Setting state', state, arguments); + if (hasLocalStorage) { + localStorage[LS_STATE_KEY] = state; + } + if (widget.view) { + if (widget.rs.remote) { + widget.view.setUserAddress(widget.rs.remote.userAddress); + } + widget.view.setState(state, arguments); + } else { + widget._rememberedState = state; + } + }; + } + + function errorsHandler(widget) { + return function(error) { + if (error instanceof RemoteStorage.DiscoveryError) { + console.error('Discovery failed', error, '"' + error.message + '"'); + widget.view.setState('initial', [error.message]); + } else if (error instanceof RemoteStorage.SyncError) { + widget.view.setState('offline', []); + } else if (error instanceof RemoteStorage.Unauthorized) { + widget.view.setState('unauthorized'); + } else { + RemoteStorage.log('[Widget] Unknown error'); + widget.view.setState('error', [error]); + } + }; + } + + function flashFor(evt) { + if (evt.method === 'GET' && evt.isFolder) { + return false; + } + return true; + } + + /** + * Class: RemoteStorage.Widget + * + * The widget controller that communicates with the view and listens to + * its remoteStorage instance. + * + * While listening to the events emitted by its remoteStorage it sets + * corresponding states of the view. + * + * - connected -> connected + * - disconnected -> initial + * - connecting -> authing + * - authing -> authing + * - wire-busy -> busy + * - wire-done -> connected + * - error -> one of initial, offline, unauthorized, or error + **/ + RemoteStorage.Widget = function(remoteStorage) { + var self = this; + var requestsToFlashFor = 0; + + // setting event listeners on rs events to put + // the widget into corresponding states + this.rs = remoteStorage; + this.rs.remote.on('connected', stateSetter(this, 'connected')); + this.rs.on('disconnected', stateSetter(this, 'initial')); + this.rs.on('connecting', stateSetter(this, 'authing')); + this.rs.on('authing', stateSetter(this, 'authing')); + this.rs.on('error', errorsHandler(this)); + + if (this.rs.remote) { + this.rs.remote.on('wire-busy', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor++; + stateSetter(self, 'busy')(); + } + }); + + this.rs.remote.on('wire-done', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor--; + if (requestsToFlashFor <= 0) { + stateSetter(self, 'connected')(); + } + } + }); + } + + if (hasLocalStorage) { + var state = localStorage[LS_STATE_KEY]; + if (state && VALID_ENTRY_STATES[state]) { + this._rememberedState = state; + } + } + }; + + RemoteStorage.Widget.prototype = { + + /** + * Method: display + * + * Displays the widget via the view.display method + * + * Parameters: + * + * domID + **/ + display: function(domID) { + if (! this.view) { + this.setView(new RemoteStorage.Widget.View(this.rs)); + } + this.view.display.apply(this.view, arguments); + return this; + }, + + linkWidgetToSync: function() { + if (typeof(this.rs.sync) === 'object' && typeof(this.rs.sync.sync) === 'function') { + this.view.on('sync', this.rs.sync.sync.bind(this.rs.sync)); + } else { + RemoteStorage.log('[Widget] typeof this.rs.sync check fail', this.rs.sync); + setTimeout(this.linkWidgetToSync.bind(this), 1000); + } + }, + + /** + * Method: setView(view) + * + * Sets the view and initializes event listeners to react on + * widget (widget.view) events + **/ + setView: function(view) { + this.view = view; + this.view.on('connect', function(options) { + if (typeof(options) === 'string') { + // options is simply a useraddress + this.rs.connect(options); + } else if (options.special) { + this.rs[options.special].connect(options); + } + }.bind(this)); + this.view.on('disconnect', this.rs.disconnect.bind(this.rs)); + this.linkWidgetToSync(); + try { + this.view.on('reset', function(){ + var location = RemoteStorage.Authorize.getLocation(); + this.rs.on('disconnected', location.reload.bind(location)); + this.rs.disconnect(); + }.bind(this)); + } catch(e) { + if (e.message && e.message.match(/Unknown event/)) { + // ignored. (the 0.7 widget-view interface didn't have a 'reset' event) + } else { + throw e; + } + } + + if (this._rememberedState) { + setTimeout(stateSetter(this, this._rememberedState), 0); + delete this._rememberedState; + } + } + }; + + /** + * Method: displayWidget + * + * Same as + **/ + RemoteStorage.prototype.displayWidget = function(domID) { + return this.widget.display(domID); + }; + + RemoteStorage.Widget._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (! remoteStorage.widget) { + remoteStorage.widget = new RemoteStorage.Widget(remoteStorage); + } + }; + + RemoteStorage.Widget._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/view.js **/ +(function(window){ + + var t = RemoteStorage.I18n.translate; + + var cEl = function(){ + return document.createElement.apply(document, arguments); + }; + + function gCl(parent, className) { + return parent.getElementsByClassName(className)[0]; + } + + function gTl(parent, tagName) { + return parent.getElementsByTagName(tagName)[0]; + } + + function removeClass(el, className) { + return el.classList.remove(className); + } + + function addClass(el, className) { + return el.classList.add(className); + } + + function stopPropagation(event) { + if (typeof(event.stopPropagation) === 'function') { + event.stopPropagation(); + } else { + event.cancelBubble = true; + } + } + + function setupButton(parent, className, iconName, eventListener) { + var element = gCl(parent, className); + if (typeof iconName !== 'undefined') { + var img = gTl(element, 'img'); + (img || element).src = RemoteStorage.Assets[iconName]; + } + element.addEventListener('click', eventListener); + return element; + } + + /** + * Class: RemoteStorage.Widget.View + * + * Controls the visible widget + * + * States: + * + * initial - not connected + * authing - in auth flow + * connected - connected to remote storage, not syncing at the moment + * busy - connected, syncing at the moment + * offline - connected, but no network connectivity + * error - connected, but sync error happened + * unauthorized - connected, but request returned 401 + **/ + RemoteStorage.Widget.View = function(remoteStorage) { + this.rs = remoteStorage; + if (typeof(document) === 'undefined') { + throw "Widget not supported"; + } + RemoteStorage.eventHandling(this, + 'connect', + 'disconnect', + 'sync', + 'display', + 'reset'); + + // Re-binding the event so they can be called from the window + for (var event in this.events){ + this.events[event] = this.events[event].bind(this); + } + + this.hideBubbleOnBodyClick = function(event) { + for (var p = event.target; p !== document.body; p = p.parentElement) { + if (p.id === 'remotestorage-widget') { + return; + } + } + this.hideBubble(); + }.bind(this); + }; + + RemoteStorage.Widget.View.prototype = { + + connectGdrive: function() { + this._emit('connect', { special: 'googledrive' }); + }, + + connectDropbox: function(){ + this._emit('connect', { special: 'dropbox'}); + }, + + /** + * Method: setState + * + * Call the function that applies the state to the widget + * + * Parameters: + * + * state + * args + **/ + setState: function(state, args) { + RemoteStorage.log('[View] widget.view.setState(',state,',',args,');'); + var s = this.states[state]; + if (typeof(s) === 'undefined') { + throw new Error("Bad State assigned to view: " + state); + } + s.apply(this, args); + }, + + /** + * Method: setUserAddres + * + * Set user address of the input field + **/ + setUserAddress: function(addr) { + this.userAddress = addr || ''; + + var el; + if (this.div && (el = gTl(this.div, 'form').userAddress)) { + el.value = this.userAddress; + } + }, + + /** + * Method: toggleBubble + * + * Show the bubble when hidden and the other way around + **/ + toggleBubble: function(event) { + if (this.bubble.className.search('rs-hidden') < 0) { + this.hideBubble(event); + } else { + this.showBubble(event); + } + }, + + /** + * Method: hideBubble + * + * Hide the bubble + **/ + hideBubble: function(){ + addClass(this.bubble, 'rs-hidden'); + document.body.removeEventListener('click', this.hideBubbleOnBodyClick); + }, + + /** + * Method: showBubble + * + * Show the bubble + **/ + showBubble: function(event){ + removeClass(this.bubble, 'rs-hidden'); + if (typeof(event) !== 'undefined') { + stopPropagation(event); + } + document.body.addEventListener('click', this.hideBubbleOnBodyClick); + gTl(this.bubble,'form').userAddress.focus(); + }, + + /** + * Method: display + * + * Draw the widget inside of the dom element with the id domID + * + * Parameters: + * + * domID + * + * Returns: + * + * The widget div + **/ + display: function(domID) { + if (typeof this.div !== 'undefined') { + return this.div; + } + + var element = cEl('div'); + var style = cEl('style'); + style.innerHTML = RemoteStorage.Assets.widgetCss; + + element.id = "remotestorage-widget"; + + element.innerHTML = RemoteStorage.Assets.widget; + + element.appendChild(style); + if (domID) { + var parent = document.getElementById(domID); + if (! parent) { + throw "Failed to find target DOM element with id=\"" + domID + "\""; + } + parent.appendChild(element); + } else { + document.body.appendChild(element); + } + + // Sync button + setupButton(element, 'rs-sync', 'syncIcon', this.events.sync); + + // Disconnect button + setupButton(element, 'rs-disconnect', 'disconnectIcon', this.events.disconnect); + + // Get me out of here + setupButton(element, 'remotestorage-reset', undefined, this.events.reset); + + // Connect button + var cb = setupButton(element, 'connect', 'connectIcon', this.events.connect); + + // Input + this.form = gTl(element, 'form'); + var el = this.form.userAddress; + el.addEventListener('keyup', function(event) { + if (event.target.value) { + cb.removeAttribute('disabled'); + } else { + cb.setAttribute('disabled','disabled'); + } + }); + if (this.userAddress) { + el.value = this.userAddress; + } + + // The cube + this.cube = setupButton(element, 'rs-cube', 'remoteStorageIcon', this.toggleBubble.bind(this)); + + // Google Drive and Dropbox icons + setupButton(element, 'rs-dropbox', 'dropbox', this.connectDropbox.bind(this)); + setupButton(element, 'rs-googledrive', 'googledrive', this.connectGdrive.bind(this)); + + var bubbleDontCatch = { INPUT: true, BUTTON: true, IMG: true }; + var eventListener = function(event) { + if (! bubbleDontCatch[event.target.tagName] && ! (this.div.classList.contains('remotestorage-state-unauthorized') )) { + this.showBubble(event); + } + }.bind(this); + this.bubble = setupButton(element, 'rs-bubble', undefined, eventListener); + + this.hideBubble(); + + this.div = element; + + this.states.initial.call(this); + this.events.display.call(this); + return this.div; + }, + + _renderTranslatedInitialContent: function() { + gCl(this.div, 'rs-status-text').innerHTML = t("view_connect"); + gCl(this.div, 'remotestorage-reset').innerHTML = t("view_get_me_out"); + gCl(this.div, 'rs-error-plz-report').innerHTML = t("view_error_plz_report"); + gCl(this.div, 'remotestorage-unauthorized').innerHTML = t("view_unauthorized"); + }, + + states: { + initial: function(message) { + var cube = this.cube; + var info = message || t("view_info"); + + this._renderTranslatedInitialContent(); + + if (message) { + cube.src = RemoteStorage.Assets.remoteStorageIconError; + removeClass(this.cube, 'remotestorage-loading'); + this.showBubble(); + + // Show the red error cube for 5 seconds, then show the normal orange one again + setTimeout(function(){ + cube.src = RemoteStorage.Assets.remoteStorageIcon; + },5000); + } else { + this.hideBubble(); + } + this.div.className = "remotestorage-state-initial"; + + // Google Drive and Dropbox icons + var backends = 1; + if (this._activateBackend('dropbox')) { backends += 1; } + if (this._activateBackend('googledrive')) { backends += 1; } + gCl(this.div, 'rs-bubble-text').style.paddingRight = backends*32+8+'px'; + + // If address not empty connect button enabled + var cb = gCl(this.div, 'connect'); + if (this.form.userAddress.value) { + cb.removeAttribute('disabled'); + } + + var infoEl = gCl(this.div, 'rs-info-msg'); + infoEl.innerHTML = info; + + if (message) { + infoEl.classList.add('remotestorage-error-info'); + } else { + infoEl.classList.remove('remotestorage-error-info'); + } + }, + + authing: function() { + this.div.removeEventListener('click', this.events.connect); + this.div.className = "remotestorage-state-authing"; + gCl(this.div, 'rs-status-text').innerHTML = t("view_connecting", this.userAddress); + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone, when is that neccesary + }, + + connected: function() { + this.div.className = "remotestorage-state-connected"; + gCl(this.div, 'userAddress').innerHTML = this.userAddress; + this.cube.src = RemoteStorage.Assets.remoteStorageIcon; + removeClass(this.cube, 'remotestorage-loading'); + var icons = { + googledrive: gCl(this.div, 'rs-googledrive'), + dropbox: gCl(this.div, 'rs-dropbox') + }; + icons.googledrive.style.display = icons.dropbox.style.display = 'none'; + if (icons[this.rs.backend]) { + icons[this.rs.backend].style.display = 'inline-block'; + gCl(this.div, 'rs-bubble-text').style.paddingRight = 2*32+8+'px'; + } else { + gCl(this.div, 'rs-bubble-text').style.paddingRight = 32+8+'px'; + } + }, + + busy: function() { + this.div.className = "remotestorage-state-busy"; + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone when is that neccesary + this.hideBubble(); + }, + + offline: function() { + this.div.className = "remotestorage-state-offline"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconOffline; + gCl(this.div, 'rs-status-text').innerHTML = t("view_offline"); + }, + + error: function(err) { + var errorMsg = err; + this.div.className = "remotestorage-state-error"; + + gCl(this.div, 'rs-bubble-text').innerHTML = ''+t('view_error_occured')+''; + //FIXME I don't know what an DOMError is and my browser doesn't know too(how to handle this?) + if (err instanceof Error /*|| err instanceof DOMError*/) { + errorMsg = err.message + '\n\n' + + err.stack; + } + gCl(this.div, 'rs-error-msg').textContent = errorMsg; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + }, + + unauthorized: function() { + this.div.className = "remotestorage-state-unauthorized"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + this.div.addEventListener('click', this.events.connect); + } + }, + + events: { + /** + * Event: connect + * + * Emitted when the connect button is clicked + **/ + connect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('connect', gTl(this.div, 'form').userAddress.value); + }, + + /** + * Event: sync + * + * Emitted when the sync button is clicked + **/ + sync: function(event) { + stopPropagation(event); + event.preventDefault(); + + this._emit('sync'); + }, + + /** + * Event: disconnect + * + * Emitted when the disconnect button is clicked + **/ + disconnect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('disconnect'); + }, + + /** + * Event: reset + * + * Emitted after crash triggers disconnect + **/ + reset: function(event){ + event.preventDefault(); + var result = window.confirm(t('view_confirm_reset')); + if (result){ + this._emit('reset'); + } + }, + + /** + * Event: display + * + * Emitted when finished displaying the widget + **/ + display : function(event) { + if (event) { + event.preventDefault(); + } + this._emit('display'); + } + }, + + _activateBackend: function activateBackend(backendName) { + var className = 'rs-' + backendName; + if (this.rs.apiKeys[backendName]) { + gCl(this.div, className).style.display = 'inline-block'; + return true; + } else { + gCl(this.div, className).style.display = 'none'; + return false; + } + } + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: lib/tv4.js **/ +/** +Author: Geraint Luff and others +Year: 2013 + +This code is released into the "public domain" by its author(s). Anybody may use, alter and distribute the code without restriction. The author makes no guarantees, and takes no liability of any kind for use of this code. + +If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory. +**/ + +(function (global) { +var ValidatorContext = function (parent, collectMultiple) { + this.missing = []; + this.schemas = parent ? Object.create(parent.schemas) : {}; + this.collectMultiple = collectMultiple; + this.errors = []; + this.handleError = collectMultiple ? this.collectError : this.returnError; +}; +ValidatorContext.prototype.returnError = function (error) { + return error; +}; +ValidatorContext.prototype.collectError = function (error) { + if (error) { + this.errors.push(error); + } + return null; +} +ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) { + for (var i = startIndex; i < this.errors.length; i++) { + this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath); + } + return this; +} + +ValidatorContext.prototype.getSchema = function (url) { + if (this.schemas[url] != undefined) { + var schema = this.schemas[url]; + return schema; + } + var baseUrl = url; + var fragment = ""; + if (url.indexOf('#') != -1) { + fragment = url.substring(url.indexOf("#") + 1); + baseUrl = url.substring(0, url.indexOf("#")); + } + if (this.schemas[baseUrl] != undefined) { + var schema = this.schemas[baseUrl]; + var pointerPath = decodeURIComponent(fragment); + if (pointerPath == "") { + return schema; + } else if (pointerPath.charAt(0) != "/") { + return undefined; + } + var parts = pointerPath.split("/").slice(1); + for (var i = 0; i < parts.length; i++) { + var component = parts[i].replace("~1", "/").replace("~0", "~"); + if (schema[component] == undefined) { + schema = undefined; + break; + } + schema = schema[component]; + } + if (schema != undefined) { + return schema; + } + } + if (this.missing[baseUrl] == undefined) { + this.missing.push(baseUrl); + this.missing[baseUrl] = baseUrl; + } +}; +ValidatorContext.prototype.addSchema = function (url, schema) { + var map = {}; + map[url] = schema; + normSchema(schema, url); + searchForTrustedSchemas(map, schema, url); + for (var key in map) { + this.schemas[key] = map[key]; + } + return map; +}; + +ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) { + if (schema['$ref'] != undefined) { + schema = this.getSchema(schema['$ref']); + if (!schema) { + return null; + } + } + + var errorCount = this.errors.length; + var error = this.validateBasic(data, schema) + || this.validateNumeric(data, schema) + || this.validateString(data, schema) + || this.validateArray(data, schema) + || this.validateObject(data, schema) + || this.validateCombinations(data, schema) + || null + if (error || errorCount != this.errors.length) { + while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) { + var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null; + var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null; + if (error) { + error = error.prefixWith(dataPart, schemaPart); + } + this.prefixErrors(errorCount, dataPart, schemaPart); + } + } + + return this.handleError(error); +} + +function recursiveCompare(A, B) { + if (A === B) { + return true; + } + if (typeof A == "object" && typeof B == "object") { + if (Array.isArray(A) != Array.isArray(B)) { + return false; + } else if (Array.isArray(A)) { + if (A.length != B.length) { + return false + } + for (var i = 0; i < A.length; i++) { + if (!recursiveCompare(A[i], B[i])) { + return false; + } + } + } else { + for (var key in A) { + if (B[key] === undefined && A[key] !== undefined) { + return false; + } + } + for (var key in B) { + if (A[key] === undefined && B[key] !== undefined) { + return false; + } + } + for (var key in A) { + if (!recursiveCompare(A[key], B[key])) { + return false; + } + } + } + return true; + } + return false; +} + +ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) { + var error; + if (error = this.validateType(data, schema)) { + return error.prefixWith(null, "type"); + } + if (error = this.validateEnum(data, schema)) { + return error.prefixWith(null, "type"); + } + return null; +} + +ValidatorContext.prototype.validateType = function validateType(data, schema) { + if (schema.type == undefined) { + return null; + } + var dataType = typeof data; + if (data == null) { + dataType = "null"; + } else if (Array.isArray(data)) { + dataType = "array"; + } + var allowedTypes = schema.type; + if (typeof allowedTypes != "object") { + allowedTypes = [allowedTypes]; + } + + for (var i = 0; i < allowedTypes.length; i++) { + var type = allowedTypes[i]; + if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) { + return null; + } + } + return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType); +} + +ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) { + if (schema["enum"] == undefined) { + return null; + } + for (var i = 0; i < schema["enum"].length; i++) { + var enumVal = schema["enum"][i]; + if (recursiveCompare(data, enumVal)) { + return null; + } + } + return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data)); +} +ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) { + return this.validateMultipleOf(data, schema) + || this.validateMinMax(data, schema) + || null; +} + +ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) { + var multipleOf = schema.multipleOf || schema.divisibleBy; + if (multipleOf == undefined) { + return null; + } + if (typeof data == "number") { + if (data%multipleOf != 0) { + return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf); + } + } + return null; +} + +ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) { + if (typeof data != "number") { + return null; + } + if (schema.minimum != undefined) { + if (data < schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum"); + } + if (schema.exclusiveMinimum && data == schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum"); + } + } + if (schema.maximum != undefined) { + if (data > schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum"); + } + if (schema.exclusiveMaximum && data == schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum"); + } + } + return null; +} +ValidatorContext.prototype.validateString = function validateString(data, schema) { + return this.validateStringLength(data, schema) + || this.validateStringPattern(data, schema) + || null; +} + +ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) { + if (typeof data != "string") { + return null; + } + if (schema.minLength != undefined) { + if (data.length < schema.minLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength"); + } + } + if (schema.maxLength != undefined) { + if (data.length > schema.maxLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength"); + } + } + return null; +} + +ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) { + if (typeof data != "string" || schema.pattern == undefined) { + return null; + } + var regexp = new RegExp(schema.pattern); + if (!regexp.test(data)) { + return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern"); + } + return null; +} +ValidatorContext.prototype.validateArray = function validateArray(data, schema) { + if (!Array.isArray(data)) { + return null; + } + return this.validateArrayLength(data, schema) + || this.validateArrayUniqueItems(data, schema) + || this.validateArrayItems(data, schema) + || null; +} + +ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) { + if (schema.minItems != undefined) { + if (data.length < schema.minItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxItems != undefined) { + if (data.length > schema.maxItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) { + if (schema.uniqueItems) { + for (var i = 0; i < data.length; i++) { + for (var j = i + 1; j < data.length; j++) { + if (recursiveCompare(data[i], data[j])) { + var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems"); + if (this.handleError(error)) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) { + if (schema.items == undefined) { + return null; + } + var error; + if (Array.isArray(schema.items)) { + for (var i = 0; i < data.length; i++) { + if (i < schema.items.length) { + if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) { + return error; + } + } else if (schema.additionalItems != undefined) { + if (typeof schema.additionalItems == "boolean") { + if (!schema.additionalItems) { + error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems"); + if (this.handleError(error)) { + return error; + } + } + } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) { + return error; + } + } + } + } else { + for (var i = 0; i < data.length; i++) { + if (error = this.validateAll(data[i], schema.items, [i], ["items"])) { + return error; + } + } + } + return null; +} +ValidatorContext.prototype.validateObject = function validateObject(data, schema) { + if (typeof data != "object" || data == null || Array.isArray(data)) { + return null; + } + return this.validateObjectMinMaxProperties(data, schema) + || this.validateObjectRequiredProperties(data, schema) + || this.validateObjectProperties(data, schema) + || this.validateObjectDependencies(data, schema) + || null; +} + +ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) { + var keys = Object.keys(data); + if (schema.minProperties != undefined) { + if (keys.length < schema.minProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxProperties != undefined) { + if (keys.length > schema.maxProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) { + if (schema.required != undefined) { + for (var i = 0; i < schema.required.length; i++) { + var key = schema.required[i]; + if (data[key] === undefined) { + var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required"); + if (this.handleError(error)) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) { + var error; + for (var key in data) { + var foundMatch = false; + if (schema.properties != undefined && schema.properties[key] != undefined) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) { + return error; + } + } + if (schema.patternProperties != undefined) { + for (var patternKey in schema.patternProperties) { + var regexp = new RegExp(patternKey); + if (regexp.test(key)) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) { + return error; + } + } + } + } + if (!foundMatch && schema.additionalProperties != undefined) { + if (typeof schema.additionalProperties == "boolean") { + if (!schema.additionalProperties) { + error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties"); + if (this.handleError(error)) { + return error; + } + } + } else { + if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) { + var error; + if (schema.dependencies != undefined) { + for (var depKey in schema.dependencies) { + if (data[depKey] !== undefined) { + var dep = schema.dependencies[depKey]; + if (typeof dep == "string") { + if (data[dep] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } else if (Array.isArray(dep)) { + for (var i = 0; i < dep.length; i++) { + var requiredKey = dep[i]; + if (data[requiredKey] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } + } else { + if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) { + var error; + return this.validateAllOf(data, schema) + || this.validateAnyOf(data, schema) + || this.validateOneOf(data, schema) + || this.validateNot(data, schema) + || null; +} + +ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) { + if (schema.allOf == undefined) { + return null; + } + var error; + for (var i = 0; i < schema.allOf.length; i++) { + var subSchema = schema.allOf[i]; + if (error = this.validateAll(data, subSchema, [], ["allOf", i])) { + return error; + } + } + return null; +} + +ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) { + if (schema.anyOf == undefined) { + return null; + } + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.anyOf.length; i++) { + var subSchema = schema.anyOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["anyOf", i]); + + if (error == null && errorCount == this.errors.length) { + this.errors = this.errors.slice(0, startErrorCount); + return null; + } + if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf")); + } + } + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors); +} + +ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) { + if (schema.oneOf == undefined) { + return null; + } + var validIndex = null; + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.oneOf.length; i++) { + var subSchema = schema.oneOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["oneOf", i]); + + if (error == null && errorCount == this.errors.length) { + if (validIndex == null) { + validIndex = i; + } else { + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf"); + } + } else if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf")); + } + } + if (validIndex == null) { + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors); + } else { + this.errors = this.errors.slice(0, startErrorCount); + } + return null; +} + +ValidatorContext.prototype.validateNot = function validateNot(data, schema) { + if (schema.not == undefined) { + return null; + } + var oldErrorCount = this.errors.length; + var error = this.validateAll(data, schema.not); + var notErrors = this.errors.slice(oldErrorCount); + this.errors = this.errors.slice(0, oldErrorCount); + if (error == null && notErrors.length == 0) { + return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not") + } + return null; +} + +// parseURI() and resolveUrl() are from https://gist.github.com/1088850 +// - released as public domain by author ("Yaffle") - see comments on gist + +function parseURI(url) { + var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/); + // authority = '//' + user + ':' + pass '@' + hostname + ':' port + return (m ? { + href : m[0] || '', + protocol : m[1] || '', + authority: m[2] || '', + host : m[3] || '', + hostname : m[4] || '', + port : m[5] || '', + pathname : m[6] || '', + search : m[7] || '', + hash : m[8] || '' + } : null); +} + +function resolveUrl(base, href) {// RFC 3986 + + function removeDotSegments(input) { + var output = []; + input.replace(/^(\.\.?(\/|$))+/, '') + .replace(/\/(\.(\/|$))+/g, '/') + .replace(/\/\.\.$/, '/../') + .replace(/\/?[^\/]*/g, function (p) { + if (p === '/..') { + output.pop(); + } else { + output.push(p); + } + }); + return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : ''); + } + + href = parseURI(href || ''); + base = parseURI(base || ''); + + return !href || !base ? null : (href.protocol || base.protocol) + + (href.protocol || href.authority ? href.authority : base.authority) + + removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) + + (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) + + href.hash; +} + +function normSchema(schema, baseUri) { + if (baseUri == undefined) { + baseUri = schema.id; + } else if (typeof schema.id == "string") { + baseUri = resolveUrl(baseUri, schema.id); + schema.id = baseUri; + } + if (typeof schema == "object") { + if (Array.isArray(schema)) { + for (var i = 0; i < schema.length; i++) { + normSchema(schema[i], baseUri); + } + } else if (typeof schema['$ref'] == "string") { + schema['$ref'] = resolveUrl(baseUri, schema['$ref']); + } else { + for (var key in schema) { + if (key != "enum") { + normSchema(schema[key], baseUri); + } + } + } + } +} + +var ErrorCodes = { + INVALID_TYPE: 0, + ENUM_MISMATCH: 1, + ANY_OF_MISSING: 10, + ONE_OF_MISSING: 11, + ONE_OF_MULTIPLE: 12, + NOT_PASSED: 13, + // Numeric errors + NUMBER_MULTIPLE_OF: 100, + NUMBER_MINIMUM: 101, + NUMBER_MINIMUM_EXCLUSIVE: 102, + NUMBER_MAXIMUM: 103, + NUMBER_MAXIMUM_EXCLUSIVE: 104, + // String errors + STRING_LENGTH_SHORT: 200, + STRING_LENGTH_LONG: 201, + STRING_PATTERN: 202, + // Object errors + OBJECT_PROPERTIES_MINIMUM: 300, + OBJECT_PROPERTIES_MAXIMUM: 301, + OBJECT_REQUIRED: 302, + OBJECT_ADDITIONAL_PROPERTIES: 303, + OBJECT_DEPENDENCY_KEY: 304, + // Array errors + ARRAY_LENGTH_SHORT: 400, + ARRAY_LENGTH_LONG: 401, + ARRAY_UNIQUE: 402, + ARRAY_ADDITIONAL_ITEMS: 403 +}; + +function ValidationError(code, message, dataPath, schemaPath, subErrors) { + if (code == undefined) { + throw new Error ("No code supplied for error: "+ message); + } + this.code = code; + this.message = message; + this.dataPath = dataPath ? dataPath : ""; + this.schemaPath = schemaPath ? schemaPath : ""; + this.subErrors = subErrors ? subErrors : null; +} +ValidationError.prototype = { + prefixWith: function (dataPrefix, schemaPrefix) { + if (dataPrefix != null) { + dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1"); + this.dataPath = "/" + dataPrefix + this.dataPath; + } + if (schemaPrefix != null) { + schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1"); + this.schemaPath = "/" + schemaPrefix + this.schemaPath; + } + if (this.subErrors != null) { + for (var i = 0; i < this.subErrors.length; i++) { + this.subErrors[i].prefixWith(dataPrefix, schemaPrefix); + } + } + return this; + } +}; + +function searchForTrustedSchemas(map, schema, url) { + if (typeof schema.id == "string") { + if (schema.id.substring(0, url.length) == url) { + var remainder = schema.id.substring(url.length); + if ((url.length > 0 && url.charAt(url.length - 1) == "/") + || remainder.charAt(0) == "#" + || remainder.charAt(0) == "?") { + if (map[schema.id] == undefined) { + map[schema.id] = schema; + } + } + } + } + if (typeof schema == "object") { + for (var key in schema) { + if (key != "enum" && typeof schema[key] == "object") { + searchForTrustedSchemas(map, schema[key], url); + } + } + } + return map; +} + +var globalContext = new ValidatorContext(); + +var publicApi = { + validate: function (data, schema) { + var context = new ValidatorContext(globalContext); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + var added = context.addSchema("", schema); + var error = context.validateAll(data, schema); + this.error = error; + this.missing = context.missing; + this.valid = (error == null); + return this.valid; + }, + validateResult: function () { + var result = {}; + this.validate.apply(result, arguments); + return result; + }, + validateMultiple: function (data, schema) { + var context = new ValidatorContext(globalContext, true); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + context.addSchema("", schema); + context.validateAll(data, schema); + var result = {}; + result.errors = context.errors; + result.missing = context.missing; + result.valid = (result.errors.length == 0); + return result; + }, + addSchema: function (url, schema) { + return globalContext.addSchema(url, schema); + }, + getSchema: function (url) { + return globalContext.getSchema(url); + }, + missing: [], + error: null, + normSchema: normSchema, + resolveUrl: resolveUrl, + errorCodes: ErrorCodes +}; + +global.tv4 = publicApi; + +})(typeof(window) != 'undefined' ? window : global); + + + +/** FILE: lib/Math.uuid.js **/ +/*! + Math.uuid.js (v1.4) + http://www.broofa.com + mailto:robert@broofa.com + + Copyright (c) 2010 Robert Kieffer + Dual licensed under the MIT and GPL licenses. + + ******** + + Changes within remoteStorage.js: + 2012-10-31: + - added AMD wrapper + - moved extensions for Math object into exported object. +*/ + +/* + * Generate a random uuid. + * + * USAGE: Math.uuid(length, radix) + * length - the desired number of characters + * radix - the number of allowable values for each character. + * + * EXAMPLES: + * // No arguments - returns RFC4122, version 4 ID + * >>> Math.uuid() + * "92329D39-6F5C-4520-ABFC-AAB64544E172" + * + * // One argument - returns ID of the specified length + * >>> Math.uuid(15) // 15 character ID (default base=62) + * "VcydxgltxrVZSTV" + * + * // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62) + * >>> Math.uuid(8, 2) // 8 character ID (base=2) + * "01001010" + * >>> Math.uuid(8, 10) // 8 character ID (base=10) + * "47473046" + * >>> Math.uuid(8, 16) // 8 character ID (base=16) + * "098F4D35" + */ + // Private array of chars to use + var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); + +Math.uuid = function (len, radix) { + var chars = CHARS, uuid = [], i; + radix = radix || chars.length; + + if (len) { + // Compact form + for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix]; + } else { + // rfc4122, version 4 form + var r; + + // rfc4122 requires these characters + uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; + uuid[14] = '4'; + + // Fill in random data. At i==19 set the high bits of clock sequence as + // per rfc4122, sec. 4.1.5 + for (i = 0; i < 36; i++) { + if (!uuid[i]) { + r = 0 | Math.random()*16; + uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; + } + } + } + + return uuid.join(''); +}; + + +/** FILE: src/baseclient.js **/ +(function(global) { + + function deprecate(thing, replacement) { + console.log('WARNING: ' + thing + ' is deprecated. Use ' + + replacement + ' instead.'); + } + + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.BaseClient + * + * Provides a high-level interface to access data below a given root path. + * + * A BaseClient deals with three types of data: folders, objects and files. + * + * returns a list of all items within a folder, or undefined + * if a 404 is encountered. Items that end with a forward slash ("/") are + * child folders. + * + * / operate on JSON objects. Each object has a type. + * + * / operates on files. Each file has a MIME type. + * + * operates on either objects or files (but not folders, folders are + * created and removed implictly). + */ + RS.BaseClient = function(storage, base) { + if (base[base.length - 1] !== '/') { + throw "Not a folder: " + base; + } + + if (base === '/') { + // allow absolute and relative paths for the root scope. + this.makePath = function(path) { + return (path[0] === '/' ? '' : '/') + path; + }; + } + + /** + * Property: storage + * + * The instance this operates on. + */ + this.storage = storage; + + /** + * Property: base + * + * Base path this operates on. + * + * For the module's privateClient this would be //, for the + * corresponding publicClient /public//. + */ + this.base = base; + + var parts = this.base.split('/'); + if (parts.length > 2) { + this.moduleName = parts[1]; + } else { + this.moduleName = 'root'; + } + + /** + * Event: change + * + * Emitted when a node changes + * + * Arguments: + * event - Event object containing information about the changed node + * + * (start code) + * { + * path: path, // Path of the changed node + * origin: 'window', 'local', or 'remote' // emitted by user action within the app, local data store, or remote sync + * oldValue: oldBody, // Old body of the changed node (undefined if creation) + * newValue: newBody // New body of the changed node (undefined if deletion) + * } + * (end code) + **/ + + RS.eventHandling(this, 'change'); + this.on = this.on.bind(this); + storage.onChange(this.base, this._fireChange.bind(this)); + }; + + RS.BaseClient.prototype = { + + extend: function(object) { + for (var key in object) { + this[key] = object[key]; + } + return this; + }, + + /** + * Method: scope + * + * Returns a new operating on a subpath of the current path. + */ + scope: function(path) { + return new RS.BaseClient(this.storage, this.makePath(path)); + }, + + // folder operations + + /** + * Method: getListing + * + * Get a list of child nodes below a given path. + * + * The callback semantics of getListing are identical to those of getObject. + * + * Parameters: + * path - The path to query. It MUST end with a forward slash. + * maxAge - (optional) Maximum age of cached listing in + * milliseconds + * + * Returns: + * + * A promise for an object, representing child nodes. + * + * Keys ending in a forward slash represent *folder nodes*, while all + * other keys represent *data nodes*. + * + * For spec versions <= 01, the data node information will contain only + * the item's ETag. For later spec versions, it will also contain the + * content type and -length of the item. + * + * Example: + * (start code) + * client.getListing('').then(function(listing) { + * listing.forEach(function(item) { + * console.log(item); + * }); + * }); + * (end code) + */ + getListing: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getListing must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then( + function(status, body) { + return (status === 404) ? undefined : body; + } + ); + }, + + /** + * Method: getAll + * + * Get all objects directly below a given path. + * + * Parameters: + * path - path to the folder + * maxAge - (optional) Maximum age of cached objects in + * milliseconds + * + * Returns: + * a promise for an object in the form { path : object, ... } + * + * Example: + * (start code) + * client.getAll('').then(function(objects) { + * for (var key in objects) { + * console.log('- ' + key + ': ', objects[key]); + * } + * }); + * (end code) + */ + getAll: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getAll must be undefined or a number'); + } + + return this.storage.get(this.makePath(path), maxAge).then(function(status, body) { + if (status === 404) { return; } + if (typeof(body) === 'object') { + var promise = promising(); + var count = Object.keys(body).length, i = 0; + if (count === 0) { + // treat this like 404. it probably means a folder listing that + // has changes that haven't been pushed out yet. + return; + } + for (var key in body) { + this.storage.get(this.makePath(path + key), maxAge). + then(function(status, b) { + body[this.key] = b; + i++; + if (i === count) { promise.fulfill(body); } + }.bind({ key: key })); + } + return promise; + } + }.bind(this)); + }, + + // file operations + + /** + * Method: getFile + * + * Get the file at the given path. A file is raw data, as opposed to + * a JSON object (use for that). + * + * Except for the return value structure, getFile works exactly like + * getObject. + * + * Parameters: + * path - see getObject + * + * Returns: + * A promise for an object: + * + * mimeType - String representing the MIME Type of the document. + * data - Raw data of the document (either a string or an ArrayBuffer) + * + * Example: + * (start code) + * // Display an image: + * client.getFile('path/to/some/image').then(function(file) { + * var blob = new Blob([file.data], { type: file.mimeType }); + * var targetElement = document.findElementById('my-image-element'); + * targetElement.src = window.URL.createObjectURL(blob); + * }); + * (end code) + */ + getFile: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getFile must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getFile must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + return { + data: body, + mimeType: mimeType, + revision: revision // (this is new) + }; + }); + }, + + /** + * Method: storeFile + * + * Store raw data at a given path. + * + * Parameters: + * mimeType - MIME media type of the data being stored + * path - path relative to the module root. MAY NOT end in a forward slash. + * data - string, ArrayBuffer or ArrayBufferView of raw data to store + * + * The given mimeType will later be returned, when retrieving the data + * using . + * + * Example (UTF-8 data): + * (start code) + * client.storeFile('text/html', 'index.html', '

Hello World!

'); + * (end code) + * + * Example (Binary data): + * (start code) + * // MARKUP: + * + * // CODE: + * var input = document.getElementById('file-input'); + * var file = input.files[0]; + * var fileReader = new FileReader(); + * + * fileReader.onload = function() { + * client.storeFile(file.type, file.name, fileReader.result); + * }; + * + * fileReader.readAsArrayBuffer(file); + * (end code) + * + */ + storeFile: function(mimeType, path, body) { + if (typeof(mimeType) !== 'string') { + return promising().reject('Argument \'mimeType\' of baseClient.storeFile must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeFile must be a string'); + } + if (typeof(body) !== 'string' && typeof(body) !== 'object') { + return promising().reject('Argument \'body\' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView'); + } + + var self = this; + return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + self.makePath(path) + ") failed with status: " + status; + } + }); + }, + + // object operations + + /** + * Method: getObject + * + * Get a JSON object from given path. + * + * Parameters: + * path - relative path from the module root (without leading slash) + * + * Returns: + * A promise for the object. + * + * Example: + * (start code) + * client.getObject('/path/to/object'). + * then(function(object) { + * // object is either an object or null + * }); + * (end code) + */ + getObject: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getObject must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getObject must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + if (typeof(body) === 'object') { + return body; + } else if (typeof(body) !== 'undefined' && status === 200) { + throw "Not an object: " + this.makePath(path); + } + }); + }, + + /** + * Method: storeObject + * + * Store object at given path. Triggers synchronization. + * + * Parameters: + * + * type - unique type of this object within this module. See description below. + * path - path relative to the module root. + * object - an object to be saved to the given node. It must be serializable as JSON. + * + * Returns: + * A promise to store the object. The promise fails with a ValidationError, when validations fail. + * + * + * What about the type?: + * + * A great thing about having data on the web, is to be able to link to + * it and rearrange it to fit the current circumstances. To facilitate + * that, eventually you need to know how the data at hand is structured. + * For documents on the web, this is usually done via a MIME type. The + * MIME type of JSON objects however, is always application/json. + * To add that extra layer of "knowing what this object is", remoteStorage + * aims to use . + * A first step in that direction, is to add a *@context attribute* to all + * JSON data put into remoteStorage. + * Now that is what the *type* is for. + * + * Within remoteStorage.js, @context values are built using three components: + * http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness + * the module name - module names should be unique as well + * the type given here - naming this particular kind of object within this module + * + * In retrospect that means, that whenever you introduce a new "type" in calls to + * storeObject, you should make sure that once your code is in the wild, future + * versions of the code are compatible with the same JSON structure. + * + * How to define types?: + * + * See for examples. + */ + storeObject: function(typeAlias, path, object) { + if (typeof(typeAlias) !== 'string') { + return promising().reject('Argument \'typeAlias\' of baseClient.storeObject must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeObject must be a string'); + } + if (typeof(object) !== 'object') { + return promising().reject('Argument \'object\' of baseClient.storeObject must be an object'); + } + this._attachType(object, typeAlias); + try { + var validationResult = this.validate(object); + if (! validationResult.valid) { + return promising(function(p) { p.reject(validationResult); }); + } + } catch(exc) { + if (! (exc instanceof RS.BaseClient.Types.SchemaNotFound)) { + return promising().reject(exc); + } + } + return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status; + } + }.bind(this)); + }, + + // generic operations + + /** + * Method: remove + * + * Remove node at given path from storage. Triggers synchronization. + * + * Parameters: + * path - Path relative to the module root. + */ + remove: function(path) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.remove must be a string'); + } + return this.storage.delete(this.makePath(path)); + }, + + + cache: function(path, strategy) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.cache must be a string'; + } + if (strategy === false) { + deprecate('caching strategy ', '<"FLUSH">'); + strategy = 'FLUSH'; + } else if (strategy === undefined) { + strategy = 'ALL'; + } else if (typeof(strategy) !== 'string') { + deprecate('that caching strategy', '<"ALL">'); + strategy = 'ALL'; + } + if (strategy !== 'FLUSH' && + strategy !== 'SEEN' && + strategy !== 'ALL') { + throw 'Argument \'strategy\' of baseclient.cache must be one of ' + + '["FLUSH", "SEEN", "ALL"]'; + } + this.storage.caching.set(this.makePath(path), strategy); + return this; + }, + + flush: function(path) { + return this.storage.local.flush(path); + }, + + makePath: function(path) { + return this.base + (path || ''); + }, + + _fireChange: function(event) { + this._emit('change', event); + }, + + _cleanPath: RS.WireClient.cleanPath, + + /** + * Method: getItemURL + * + * Retrieve full URL of item + * + * Parameters: + * path - Path relative to the module root. + */ + getItemURL: function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.getItemURL must be a string'; + } + if (this.storage.connected) { + path = this._cleanPath( this.makePath(path) ); + return this.storage.remote.href + path; + } else { + return undefined; + } + }, + + uuid: function() { + return Math.uuid(); + } + + }; + + /** + * Method: RS#scope + * + * Returns a new scoped to the given path. + * + * Parameters: + * path - Root path of new BaseClient. + * + * + * Example: + * (start code) + * + * var foo = remoteStorage.scope('/foo/'); + * + * // PUTs data "baz" to path /foo/bar + * foo.storeFile('text/plain', 'bar', 'baz'); + * + * var something = foo.scope('something/'); + * + * // GETs listing from path /foo/something/bla/ + * something.getListing('bla/'); + * + * (end code) + * + */ + RS.BaseClient._rs_init = function() { + RS.prototype.scope = function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.scope must be a string'; + } + + return new RS.BaseClient(this, path); + }; + }; + + /* e.g.: + remoteStorage.defineModule('locations', function(priv, pub) { + return { + exports: { + features: priv.scope('features/').defaultType('feature'), + collections: priv.scope('collections/').defaultType('feature-collection'); + } + }; + }); + */ + + maxAgeInvalid = function(maxAge) { + return typeof(maxAge) !== 'undefined' && typeof(maxAge) !== 'number'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/baseclient/types.js **/ +(function(global) { + + RemoteStorage.BaseClient.Types = { + // -> + uris: {}, + // -> + schemas: {}, + // -> + aliases: {}, + + declare: function(moduleName, alias, uri, schema) { + var fullAlias = moduleName + '/' + alias; + + if (schema.extends) { + var extendedAlias; + var parts = schema.extends.split('/'); + if (parts.length === 1) { + extendedAlias = moduleName + '/' + parts.shift(); + } else { + extendedAlias = parts.join('/'); + } + var extendedUri = this.uris[extendedAlias]; + if (! extendedUri) { + throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'"; + } + schema.extends = this.schemas[extendedUri]; + } + + this.uris[fullAlias] = uri; + this.aliases[uri] = fullAlias; + this.schemas[uri] = schema; + }, + + resolveAlias: function(alias) { + return this.uris[alias]; + }, + + getSchema: function(uri) { + return this.schemas[uri]; + }, + + inScope: function(moduleName) { + var ml = moduleName.length; + var schemas = {}; + for (var alias in this.uris) { + if (alias.substr(0, ml + 1) === moduleName + '/') { + var uri = this.uris[alias]; + schemas[uri] = this.schemas[uri]; + } + } + return schemas; + } + }; + + var SchemaNotFound = function(uri) { + var error = new Error("Schema not found: " + uri); + error.name = "SchemaNotFound"; + return error; + }; + + SchemaNotFound.prototype = Error.prototype; + + RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound; + /** + * Class: RemoteStorage.BaseClient + **/ + RemoteStorage.BaseClient.prototype.extend({ + /** + * Method: validate(object) + * + * validates an Object against the associated schema + * the context has to have a @context property + * + * Returns: + * A validate object giving you information about errors + **/ + validate: function(object) { + var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']); + if (schema) { + return tv4.validateResult(object, schema); + } else { + throw new SchemaNotFound(object['@context']); + } + }, + + // client.declareType(alias, schema); + // /* OR */ + // client.declareType(alias, uri, schema); + declareType: function(alias, uri, schema) { + if (! schema) { + schema = uri; + uri = this._defaultTypeURI(alias); + } + RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema); + }, + + _defaultTypeURI: function(alias) { + return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias; + }, + + _attachType: function(object, alias) { + object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias); + } + }); + + Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', { + configurable: true, + get: function() { + return RemoteStorage.BaseClient.Types.inScope(this.moduleName); + } + }); + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/modules.js **/ +(function() { + + RemoteStorage.MODULES = {}; + + /* + * Method: RemoteStorage.defineModule + * + * Method for defining a new remoteStorage data module + * + * Parameters: + * moduleName - Name of the module + * builder - Builder function defining the module + * + * The module builder function should return an object containing another + * object called exports, which will be exported to any + * instance under the module's name. So when defining a locations module, + * like in the example below, it would be accessible via + * `remoteStorage.locations`, which would in turn have a `features` and a + * `collections` property. + * + * The function receives a private and a public client, which are both + * instances of . In the following example, the + * scope of privateClient is `/locations` and the scope of publicClient is + * `/public/locations`. + * + * Example: + * (start code) + * RemoteStorage.defineModule('locations', function(privateClient, publicClient) { + * return { + * exports: { + * features: privateClient.scope('features/').defaultType('feature'), + * collections: privateClient.scope('collections/').defaultType('feature-collection') + * } + * }; + * }); + * (end code) + */ + + RemoteStorage.defineModule = function(moduleName, builder) { + RemoteStorage.MODULES[moduleName] = builder; + + Object.defineProperty(RemoteStorage.prototype, moduleName, { + configurable: true, + get: function() { + var instance = this._loadModule(moduleName); + Object.defineProperty(this, moduleName, { + value: instance + }); + return instance; + } + }); + + if (moduleName.indexOf('-') !== -1) { + var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) { + return s[1].toUpperCase(); + }); + Object.defineProperty(RemoteStorage.prototype, camelizedName, { + get: function() { + return this[moduleName]; + } + }); + } + }; + + RemoteStorage.prototype._loadModule = function(moduleName) { + var builder = RemoteStorage.MODULES[moduleName]; + if (builder) { + var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'), + new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/')); + return module.exports; + } else { + throw "Unknown module: " + moduleName; + } + }; + + RemoteStorage.prototype.defineModule = function(moduleName) { + console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!"); + RemoteStorage.defineModule.apply(RemoteStorage, arguments); + }; + +})(); + + +/** FILE: src/debug/inspect.js **/ +(function() { + function loadTable(table, storage, paths) { + table.setAttribute('border', '1'); + table.style.margin = '8px'; + table.innerHTML = ''; + var thead = document.createElement('thead'); + table.appendChild(thead); + var titleRow = document.createElement('tr'); + thead.appendChild(titleRow); + ['Path', 'Content-Type', 'Revision'].forEach(function(label) { + var th = document.createElement('th'); + th.textContent = label; + thead.appendChild(th); + }); + + var tbody = document.createElement('tbody'); + table.appendChild(tbody); + + function renderRow(tr, path, contentType, revision) { + [path, contentType, revision].forEach(function(value) { + var td = document.createElement('td'); + td.textContent = value || ''; + tr.appendChild(td); + }); + } + + function loadRow(path) { + if (storage.connected === false) { return; } + function processRow(status, body, contentType, revision) { + if (status === 200) { + var tr = document.createElement('tr'); + tbody.appendChild(tr); + renderRow(tr, path, contentType, revision); + if (path[path.length - 1] === '/') { + for (var key in body) { + loadRow(path + key); + } + } + } + } + storage.get(path).then(processRow); + } + + paths.forEach(loadRow); + } + + function renderWrapper(title, table, storage, paths) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = title; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + updateButton.onclick = function() { loadTable(table, storage, paths); }; + wrapper.appendChild(updateButton); + if (storage.reset) { + var resetButton = document.createElement('button'); + resetButton.textContent = "Reset"; + resetButton.onclick = function() { + storage.reset(function(newStorage) { + storage = newStorage; + loadTable(table, storage, paths); + }); + }; + wrapper.appendChild(resetButton); + } + wrapper.appendChild(table); + loadTable(table, storage, paths); + return wrapper; + } + + function renderLocalChanges(local) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = "Outgoing changes"; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + wrapper.appendChild(updateButton); + var list = document.createElement('ul'); + list.style.fontFamily = 'courier'; + wrapper.appendChild(list); + + function updateList() { + local.changesBelow('/').then(function(changes) { + list.innerHTML = ''; + changes.forEach(function(change) { + var el = document.createElement('li'); + el.textContent = JSON.stringify(change); + list.appendChild(el); + }); + }); + } + + updateButton.onclick = updateList; + updateList(); + return wrapper; + } + + RemoteStorage.prototype.inspect = function() { + + var widget = document.createElement('div'); + widget.id = 'remotestorage-inspect'; + widget.style.position = 'absolute'; + widget.style.top = 0; + widget.style.left = 0; + widget.style.background = 'black'; + widget.style.color = 'white'; + widget.style.border = 'groove 5px #ccc'; + + var controls = document.createElement('div'); + controls.style.position = 'absolute'; + controls.style.top = 0; + controls.style.left = 0; + + var heading = document.createElement('strong'); + heading.textContent = " remotestorage.js inspector "; + + controls.appendChild(heading); + + var syncButton; + + if (this.local) { + syncButton = document.createElement('button'); + syncButton.textContent = "Synchronize"; + controls.appendChild(syncButton); + } + + var closeButton = document.createElement('button'); + closeButton.textContent = "Close"; + closeButton.onclick = function() { + document.body.removeChild(widget); + }; + controls.appendChild(closeButton); + + widget.appendChild(controls); + + var remoteTable = document.createElement('table'); + var localTable = document.createElement('table'); + widget.appendChild(renderWrapper("Remote", remoteTable, this.remote, this.caching.rootPaths)); + if (this.local) { + widget.appendChild(renderWrapper("Local", localTable, this.local, ['/'])); + widget.appendChild(renderLocalChanges(this.local)); + + syncButton.onclick = function() { + this.log('sync clicked'); + this.sync().then(function() { + this.log('SYNC FINISHED'); + loadTable(localTable, this.local, ['/']); + }.bind(this), function(err) { + console.error("SYNC FAILED", err, err.stack); + }); + }.bind(this); + } + + document.body.appendChild(widget); + }; +})(); + + +/** FILE: src/legacy.js **/ +(function() { + var util = { + getEventEmitter: function() { + var object = {}; + var args = Array.prototype.slice.call(arguments); + args.unshift(object); + RemoteStorage.eventHandling.apply(RemoteStorage, args); + object.emit = object._emit; + return object; + }, + + extend: function(target) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach(function(source) { + for (var key in source) { + target[key] = source[key]; + } + }); + return target; + }, + + asyncEach: function(array, callback) { + return this.asyncMap(array, callback). + then(function() { return array; }); + }, + + asyncMap: function(array, callback) { + var promise = promising(); + var n = array.length, i = 0; + var results = [], errors = []; + function oneDone() { + i++; + if (i === n) { + promise.fulfill(results, errors); + } + } + + array.forEach(function(item, index) { + var result; + try { + result = callback(item); + } catch(exc) { + oneDone(); + errors[index] = exc; + } + if (typeof(result) === 'object' && typeof(result.then) === 'function') { + result.then(function(res) { results[index] = res; oneDone(); }, + function(error) { errors[index] = res; oneDone(); }); + } else { + oneDone(); + results[index] = result; + } + }); + + return promise; + }, + + containingFolder: function(path) { + var folder = path.replace(/[^\/]+\/?$/, ''); + return folder === path ? null : folder; + }, + + isFolder: function(path) { + return path.substr(-1) === '/'; + }, + + baseName: function(path) { + var parts = path.split('/'); + if (util.isFolder(path)) { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + }, + + bindAll: function(object) { + for (var key in this) { + if (typeof(object[key]) === 'function') { + object[key] = object[key].bind(object); + } + } + } + }; + + Object.defineProperty(RemoteStorage.prototype, 'util', { + get: function() { + console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release."); + return util; + } + }); + +})(); + + +/** FILE: src/googledrive.js **/ +(function(global) { + /** + * Class: RemoteStorage.GoogleDrive + * + * WORK IN PROGRESS, NOT RECOMMENDED FOR PRODUCTION USE + **/ + + var RS = RemoteStorage; + + var BASE_URL = 'https://www.googleapis.com'; + var AUTH_URL = 'https://accounts.google.com/o/oauth2/auth'; + var AUTH_SCOPE = 'https://www.googleapis.com/auth/drive'; + + var GD_DIR_MIME_TYPE = 'application/vnd.google-apps.folder'; + var RS_DIR_MIME_TYPE = 'application/json; charset=UTF-8'; + + function buildQueryString(params) { + return Object.keys(params).map(function(key) { + return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]); + }).join('&'); + } + + function fileNameFromMeta(meta) { + return encodeURIComponent(meta.title) + (meta.mimeType === GD_DIR_MIME_TYPE ? '/' : ''); + } + + function metaTitleFromFileName(filename) { + if (filename.substr(-1) === '/') { + filename = filename.substr(0, filename.length - 1); + } + return decodeURIComponent(filename); + } + + function parentPath(path) { + return path.replace(/[^\/]+\/?$/, ''); + } + + function baseName(path) { + var parts = path.split('/'); + if (path.substr(-1) === '/') { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + } + + var Cache = function(maxAge) { + this.maxAge = maxAge; + this._items = {}; + }; + + Cache.prototype = { + get: function(key) { + var item = this._items[key]; + var now = new Date().getTime(); + return (item && item.t >= (now - this.maxAge)) ? item.v : undefined; + }, + + set: function(key, value) { + this._items[key] = { + v: value, + t: new Date().getTime() + }; + } + }; + + RS.GoogleDrive = function(remoteStorage, clientId) { + + RS.eventHandling(this, 'connected'); + + this.rs = remoteStorage; + this.clientId = clientId; + + this._fileIdCache = new Cache(60 * 5); // ids expire after 5 minutes (is this a good idea?) + + setTimeout(function() { + this.configure(undefined, undefined, undefined, localStorage['remotestorage:googledrive:token']); + }.bind(this), 0); + }; + + RS.GoogleDrive.prototype = { + + configure: function(_x, _y, _z, token) { // parameter list compatible with WireClient + if (token) { + localStorage['remotestorage:googledrive:token'] = token; + this.token = token; + this.connected = true; + this._emit('connected'); + } else { + this.connected = false; + delete this.token; + // not reseting backend whenever googledrive gets initialized without an token +// this.rs.setBackend(undefined); + delete localStorage['remotestorage:googledrive:token']; + } + }, + + connect: function() { + this.rs.setBackend('googledrive'); + RS.Authorize(AUTH_URL, AUTH_SCOPE, String(RS.Authorize.getLocation()), this.clientId); + }, + + get: function(path, options) { + if (path.substr(-1) === '/') { + return this._getFolder(path, options); + } else { + return this._getFile(path, options); + } + }, + + put: function(path, body, contentType, options) { + var promise = promising(); + function putDone(error, response) { + if (error) { + promise.reject(error); + } else if (response.status >= 200 && response.status < 300) { + var meta = JSON.parse(response.responseText); + promise.fulfill(200, undefined, meta.mimeType, meta.etag); + } else { + promise.reject("PUT failed with status " + response.status + " (" + response.responseText + ")"); + } + } + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + return; + } else if (id) { + this._updateFile(id, path, body, contentType, options, putDone); + } else { + this._createFile(path, body, contentType, options, putDone); + } + }); + return promise; + }, + + 'delete': function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else if (id) { + this._request('DELETE', BASE_URL + '/drive/v2/files/' + id, {}, function(deleteError, response) { + if (deleteError) { + promise.reject(deleteError); + } else if (response.status === 200 || response.status === 204) { + promise.fulfill(200); + } else { + promise.reject("Delete failed: " + response.status + " (" + response.responseText + ")"); + } + }); + } else { + // file doesn't exist. ignore. + promise.fulfill(200); + } + }); + return promise; + }, + + _updateFile: function(id, path, body, contentType, options, callback) { + callback = callback.bind(this); + var metadata = { + mimeType: contentType + }; + this._request('PUT', BASE_URL + '/upload/drive/v2/files/' + id + '?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('PUT', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }, + + _createFile: function(path, body, contentType, options, callback) { + callback = callback.bind(this); + this._getParentId(path, function(parentIdError, parentId) { + if (parentIdError) { + callback(parentIdError); + return; + } + var fileName = baseName(path); + var metadata = { + title: metaTitleFromFileName(fileName), + mimeType: contentType, + parents: [{ + kind: "drive#fileLink", + id: parentId + }] + }; + this._request('POST', BASE_URL + '/upload/drive/v2/files?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('POST', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }); + }, + + _getFile: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else { + this._getMeta(id, function(metaError, meta) { + var etagWithoutQuotes; + if (typeof(meta) === 'object' && typeof(meta.etag) === 'string') { + etagWithoutQuotes = meta.etag.substring(1, meta.etag.length-1); + } + if (metaError) { + promise.reject(metaError); + } else if (meta.downloadUrl) { + var options = {}; + if (meta.mimeType.match(/charset=binary/)) { + options.responseType = 'blob'; + } + this._request('GET', meta.downloadUrl, options, function(downloadError, response) { + if (downloadError) { + promise.reject(downloadError); + } else { + var body = response.response; + if (meta.mimeType.match(/^application\/json/)) { + try { + body = JSON.parse(body); + } catch(e) {} + } + promise.fulfill(200, body, meta.mimeType, etagWithoutQuotes); + } + }); + } else { + // empty file + promise.fulfill(200, '', meta.mimeType, etagWithoutQuotes); + } + }); + } + }); + return promise; + }, + + _getFolder: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + var query, fields, data, i, etagWithoutQuotes, itemsMap; + if (idError) { + promise.reject(idError); + } else if (! id) { + promise.fulfill(404); + } else { + query = '\'' + id + '\' in parents'; + fields = 'items(downloadUrl,etag,fileSize,id,mimeType,title)'; + this._request('GET', BASE_URL + '/drive/v2/files?' + + 'q=' + encodeURIComponent(query) + + '&fields=' + encodeURIComponent(fields) + + '&maxResults=1000', + {}, function(childrenError, response) { + if (childrenError) { + promise.reject(childrenError); + } else { + if (response.status === 200) { + try { + data = JSON.parse(response.responseText); + } catch(e) { + promise.reject('non-JSON response from GoogleDrive'); + return; + } + itemsMap = {}; + for(i=0; i= 0 || true) { + try { + body = JSON.parse(body); + mime = 'application/json; charset=UTF-8'; + } catch(e) { + RS.log("Failed parsing Json, assume it is something else then", mime, path); + } + } + promise.fulfill(status, body, mime, rev); + } + + } else { + promise.fulfill(status); + } + } + }); + return promise; + }, + /** + * Method : put(path, body, contentType, options) + * put compatible with wireclient + * also uses _revCache to check for version conflicts + * also shares via share(path) + **/ + put: function(path, body, contentType, options){ + RemoteStorage.log('dropbox.put', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = this._sharePromise(path); + + var revCache = this._revCache; + + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options && options.ifMatch && savedRev && (savedRev !== options.ifMatch) ) { + promise.fulfill(412); + return promise; + } + if (! contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || RS.WireClient.isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var url = 'https://api-content.dropbox.com/1/files_put/auto' + path + '?'; + if (options && options.ifMatch) { + url += "parent_rev="+encodeURIComponent(options.ifMatch); + } + if (body.length>150*1024*1024){ //FIXME actual content-length + //https://www.dropbox.com/developers/core/docs#chunked-upload + RemoteStorage.log('files larger than 150MB not supported yet'); + } else { + this._request('PUT', url, {body:body, headers:{'Content-Type':contentType}}, function(err, resp) { + if (err) { + promise.reject(err); + } else { + var response = JSON.parse(resp.responseText); + // if dropbox reports an file conflict they just change the name of the file + // TODO find out which stays the origianl and how to deal with this + if (response.path !== path){ + promise.fulfill(412); + this.rs.log('Dropbox created conflicting File ', response.path); + } + else { + revCache.set(path, response.rev); + promise.fulfill(resp.status); + } + } + }); + } + return promise; + }, + + /** + * Method : delete(path, options) + * similar to get and set + **/ + 'delete': function(path, options){ + RemoteStorage.log('dropbox.delete ', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = promising(); + var revCache = this._revCache; + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options.ifMatch && savedRev && (options.ifMatch !== savedRev)) { + promise.fulfill(412); + return promise; + } + + var url = 'https://api.dropbox.com/1/fileops/delete?root=auto&path='+encodeURIComponent(path); + this._request('POST', url, {}, function(err, resp){ + if (err) { + promise.reject(error); + } else { + promise.fulfill(resp.status); + revCache.delete(path); + } + }); + + return promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + delete this._itemRefs[path]; + var p = promising(); + return p.fulfill.apply(p, args); + }.bind(this)); + }, + + /** + * Method : _sharePromise(path) + * returns a promise which's then block doesn't touch the arguments given + * and calls share for the path + * + * also checks for necessity of shareing this url(already in the itemRefs or not '/public/') + **/ + _sharePromise: function(path){ + var promise = promising(); + var self = this; + if (path.match(/^\/public\/.*[^\/]$/) && typeof this._itemRefs[path] === 'undefined') { + RemoteStorage.log('shareing this one ', path); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + var p = promising(); + RemoteStorage.log('calling share now'); + self.share(path).then(function() { + RemoteStorage.log('shareing fullfilled promise',arguments); + p.fulfill.apply(p,args); + }, function(err) { + RemoteStorage.log("shareing failed" , err); + p.fulfill.apply(p,args); + }); + return p; + }); + } + return promise; + }, + + /** + * Method : share(path) + * get sher_url s from dropbox and pushes those into this._hrefCache + * returns promise + */ + share: function(path){ + var url = "https://api.dropbox.com/1/media/auto"+path; + var promise = promising(); + var itemRefs = this._itemRefs; + + // requesting shareing url + this._request('POST', url, {}, function(err, resp){ + if (err) { + RemoteStorage.log(err); + err.message = 'Shareing Dropbox Thingie("'+path+'") failed' + err.message; + promise.reject(err); + } else { + try{ + var response = JSON.parse(resp.responseText); + var url = response.url; + itemRefs[path] = url; + RemoteStorage.log("SHAREING URL :::: ",url,' for ',path); + if (hasLocalStorage) { + localStorage[SETTINGS_KEY+":shares"] = JSON.stringify(this._itemRefs); + } + promise.fulfill(url); + } catch(err) { + err.message += "share error"; + promise.reject(err); + } + } + }); + return promise; + }, + + /** + * Method : info() + * fetching user info from Dropbox returns promise + **/ + info: function() { + var url = 'https://api.dropbox.com/1/account/info'; + var promise = promising(); + // requesting user info(mainly for userAdress) + this._request('GET', url, {}, function(err, resp){ + if (err) { + promise.reject(err); + } else { + try { + var info = JSON.parse(resp.responseText); + promise.fulfill(info); + } catch(e) { + promise.reject(err); + } + } + }); + return promise; + }, + + _request: function(method, url, options, callback) { + callback = callback.bind(this); + if (! options.headers) { options.headers = {}; } + options.headers['Authorization'] = 'Bearer ' + this.token; + RS.WireClient.request.call(this, method, url, options, function(err, xhr) { + //503 means retry this later + if (xhr && xhr.status === 503) { + global.setTimeout(this._request(method, url, options, callback), 3210); + } else { + callback(err, xhr); + } + }); + }, + + /** + * method: fetchDelta + * + * this method fetches the deltas from the dropbox api, used to sync the storage + * here we retrive changes and put them into the _revCache, those values will then be used + * to determin if something has changed. + **/ + fetchDelta: function() { + var args = Array.prototype.slice.call(arguments); + var promise = promising(); + var self = this; + this._request('POST', 'https://api.dropbox.com/1/delta', { + body: this._deltaCursor ? ('cursor=' + encodeURIComponent(this._deltaCursor)) : '', + headers: { + 'Content-Type': 'application/x-www-form-urlencoded' + } + }, function(error, response) { + if (error) { + this.rs.log('fetchDeltas',error); + this.rs._emit('error', new RemoteStorage.SyncError('fetchDeltas failed'+error)); + promise.reject(error); + } else { + // break if status != 200 + if (response.status !== 200 ) { + if (response.status === 400) { + this.rs._emit('error', new RemoteStorage.Unauthorized()); + promise.fulfill.apply(promise, args); + } else { + RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText); + promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText); + } + return promise; + } + + var delta; + try { + delta = JSON.parse(response.responseText); + } catch(error) { + RS.log('fetchDeltas can not parse response',error); + return promise.reject("can not parse response of fetchDelta : "+error.message); + } + // break if no entries found + if (!delta.entries) { + RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!", delta); + return promise.reject('dropbox.fetchDeltas failed, no entries found'); + } + + // Dropbox sends the complete state + if (delta.reset) { + this._revCache = new LowerCaseCache('rev'); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + self._revCache._activatePropagation(); + var p = promising(); + return p.fulfill.apply(p,args); + }); + } + + //saving the cursor for requesting further deltas in relation to the cursor position + if (delta.cursor) { + this._deltaCursor = delta.cursor; + } + + //updating revCache + RemoteStorage.log("Delta : ",delta.entries); + delta.entries.forEach(function(entry) { + var path = entry[0]; + var rev; + if (!entry[1]){ + rev = null; + } else { + if (entry[1].is_dir) { + return; + } + rev = entry[1].rev; + } + self._revCache.set(path, rev); + }); + promise.fulfill.apply(promise, args); + } + }); + return promise; + } + }; + + //hooking and unhooking the sync + + function hookSync(rs) { + if (rs._dropboxOrigSync) { return; } // already hooked + rs._dropboxOrigSync = rs.sync.bind(rs); + rs.sync = function() { + return this.dropbox.fetchDelta.apply(this.dropbox, arguments). + then(rs._dropboxOrigSync, function(err){ + rs._emit('error', new rs.SyncError(err)); + }); + }; + } + + function unHookSync(rs) { + if (! rs._dropboxOrigSync) { return; } // not hooked + rs.sync = rs._dropboxOrigSync; + delete rs._dropboxOrigSync; + } + + // hooking and unhooking getItemURL + + function hookGetItemURL(rs) { + if (rs._origBaseClientGetItemURL) { return; } + rs._origBaseClientGetItemURL = RS.BaseClient.prototype.getItemURL; + RS.BaseClient.prototype.getItemURL = function(path){ + var ret = rs.dropbox._itemRefs[path]; + return ret ? ret : ''; + }; + } + + function unHookGetItemURL(rs){ + if (! rs._origBaseClieNtGetItemURL) { return; } + RS.BaseClient.prototype.getItemURL = rs._origBaseClietGetItemURL; + delete rs._origBaseClietGetItemURL; + } + + function hookRemote(rs){ + if (rs._origRemote) { return; } + rs._origRemote = rs.remote; + rs.remote = rs.dropbox; + } + + function unHookRemote(rs){ + if (rs._origRemote) { + rs.remote = rs._origRemote; + delete rs._origRemote; + } + } + + function hookIt(rs){ + hookRemote(rs); + if (rs.sync) { + hookSync(rs); + } + hookGetItemURL(rs); + } + + function unHookIt(rs){ + unHookRemote(rs); + unHookSync(rs); + unHookGetItemURL(rs); + } + + RS.Dropbox._rs_init = function(rs) { + hasLocalStorage = rs.localStorageAvailable(); + if ( rs.apiKeys.dropbox ) { + rs.dropbox = new RS.Dropbox(rs); + } + if (rs.backend === 'dropbox') { + hookIt(rs); + } + }; + + RS.Dropbox._rs_supported = function() { + return true; + }; + + RS.Dropbox._rs_cleanup = function(rs) { + unHookIt(rs); + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + rs.removeEventListener('error', onErrorCb); + rs.setBackend(undefined); + }; +})(this); + +return new RemoteStorage(); +}); diff --git a/release/0.10.0-beta2/remotestorage-nocache.js b/release/0.10.0-beta2/remotestorage-nocache.js new file mode 100644 index 000000000..a427d540b --- /dev/null +++ b/release/0.10.0-beta2/remotestorage-nocache.js @@ -0,0 +1,5527 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ + +/** FILE: lib/promising.js **/ +(function(global) { + function getPromise(builder) { + var promise; + + if(typeof(builder) === 'function') { + setTimeout(function() { + try { + builder(promise); + } catch(e) { + promise.reject(e); + } + }, 0); + } + + var consumers = [], success, result; + + function notifyConsumer(consumer) { + if(success) { + var nextValue; + if(consumer.fulfilled) { + try { + nextValue = [consumer.fulfilled.apply(null, result)]; + } catch(exc) { + consumer.promise.reject(exc); + return; + } + } else { + nextValue = result; + } + if(nextValue[0] && typeof(nextValue[0].then) === 'function') { + nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill.apply(null, nextValue); + } + } else { + if(consumer.rejected) { + var ret; + try { + ret = consumer.rejected.apply(null, result); + } catch(exc) { + consumer.promise.reject(exc); + return; + } + if(ret && typeof(ret.then) === 'function') { + ret.then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill(ret); + } + } else { + consumer.promise.reject.apply(null, result); + } + } + } + + function resolve(succ, res) { + if(result) { + console.error("WARNING: Can't resolve promise, already resolved!"); + return; + } + success = succ; + result = Array.prototype.slice.call(res); + setTimeout(function() { + var cl = consumers.length; + if(cl === 0 && (! success)) { + console.error("Possibly uncaught error: ", result, result[0] && result[0].stack); + } + for(var i=0;i. + */ + RemoteStorage.log = function() { + if (RemoteStorage._log) { + console.log.apply(console, arguments); + } + }; + + + RemoteStorage.prototype = { + /** + * Method: connect + * + * Connect to a remoteStorage server. + * + * Parameters: + * userAddress - The user address (user@host) to connect to. + * + * Discovers the webfinger profile of the given user address and + * initiates the OAuth dance. + * + * This method must be called *after* all required access has been claimed. + * + */ + connect: function(userAddress) { + this.setBackend('remotestorage'); + if (userAddress.indexOf('@') < 0) { + this._emit('error', new RemoteStorage.DiscoveryError("User adress doesn't contain an @.")); + return; + } + this.remote.configure(userAddress); + this._emit('connecting'); + + var discoveryTimeout = setTimeout(function() { + this._emit('error', new RemoteStorage.DiscoveryError("No storage information found at that user address.")); + }.bind(this), 5000); + + RemoteStorage.Discover(userAddress, function(href, storageApi, authURL) { + clearTimeout(discoveryTimeout); + if (!href) { + this._emit('error', new RemoteStorage.DiscoveryError("Failed to contact storage server.")); + return; + } + this._emit('authing'); + this.remote.configure(userAddress, href, storageApi); + if (! this.remote.connected) { + this.authorize(authURL); + } + }.bind(this)); + }, + + /** + * Method: disconnect + * + * "Disconnect" from remotestorage server to terminate current session. + * This method clears all stored settings and deletes the entire local + * cache. + */ + disconnect: function() { + if (this.remote) { + this.remote.configure(null, null, null, null); + } + this._setGPD({ + get: this._pendingGPD('get'), + put: this._pendingGPD('put'), + delete: this._pendingGPD('delete') + }); + var n = this._cleanups.length, i = 0; + + var oneDone = function() { + i++; + if (i >= n) { + this._init(); + RemoteStorage.log('Done cleaning up, emitting disconnected and disconnect events'); + this._emit('disconnected'); + } + }.bind(this); + + if (n > 0) { + this._cleanups.forEach(function(cleanup) { + var cleanupResult = cleanup(this); + if (typeof(cleanup) === 'object' && typeof(cleanup.then) === 'function') { + cleanupResult.then(oneDone); + } else { + oneDone(); + } + }.bind(this)); + } else { + oneDone(); + } + }, + + setBackend: function(what) { + this.backend = what; + if (this.localStorageAvailable()) { + if (what) { + localStorage['remotestorage:backend'] = what; + } else { + delete localStorage['remotestorage:backend']; + } + } + }, + + /** + * Method: onChange + * + * Add a "change" event handler to the given path. Whenever a "change" + * happens (as determined by the backend, such as e.g. + * ) and the affected path is equal to or below + * the given 'path', the given handler is called. + * + * You should usually not use this method directly, but instead use the + * "change" events provided by . + * + * Parameters: + * path - Absolute path to attach handler to. + * handler - Handler function. + */ + onChange: function(path, handler) { + if (! this._pathHandlers.change[path]) { + this._pathHandlers.change[path] = []; + } + this._pathHandlers.change[path].push(handler); + }, + + /** + * Method: enableLog + * + * Enable remoteStorage logging + */ + enableLog: function() { + RemoteStorage._log = true; + }, + + /** + * Method: disableLog + * + * Disable remoteStorage logging + */ + disableLog: function() { + RemoteStorage._log = false; + }, + + /** + * Method: log + * + * The same as . + */ + log: function() { + RemoteStorage.log.apply(RemoteStorage, arguments); + }, + + setApiKeys: function(type, keys) { + if (keys) { + this.apiKeys[type] = keys; + } else { + delete this.apiKeys[type]; + } + if (this.localStorageAvailable()) { + localStorage['remotestorage:api-keys'] = JSON.stringify(this.apiKeys); + } + }, + + /** + ** INITIALIZATION + **/ + + _init: function() { + var self = this, + readyFired = false; + + function fireReady() { + try { + if (!readyFired) { + self._emit('ready'); + readyFired = true; + } + } catch(e) { + console.error("'ready' failed: ", e, e.stack); + self._emit('error', e); + } + } + + this._loadFeatures(function(features) { + this.log('[RemoteStorage] All features loaded'); + this.local = features.local && new features.local(); + // this.remote set by WireClient._rs_init as lazy property on + // RS.prototype + + if (this.local && this.remote) { + this._setGPD(SyncedGetPutDelete, this); + this._bindChange(this.local); + } else if (this.remote) { + this._setGPD(this.remote, this.remote); + } + + if (this.remote) { + this.remote.on('connected', function(){ + fireReady(); + self._emit('connected'); + }); + this.remote.on('not-connected', function(){ + fireReady(); + self._emit('not-connected'); + }); + if (this.remote.connected) { + fireReady(); + self._emit('connected'); + } + } + + this._collectCleanupFunctions(); + + try { + this._allLoaded = true; + this._emit('features-loaded'); + } catch(exc) { + logError(exc); + this._emit('error', exc); + } + this._processPending(); + }.bind(this)); + }, + + _collectCleanupFunctions: function() { + for (var i=0; i < this.features.length; i++) { + var cleanup = this.features[i].cleanup; + if (typeof(cleanup) === 'function') { + this._cleanups.push(cleanup); + } + } + }, + + /** + ** FEATURE DETECTION + **/ + _loadFeatures: function(callback) { + var featureList = [ + 'WireClient', + 'I18n', + 'Dropbox', + 'GoogleDrive', + 'Access', + 'Caching', + 'Discover', + 'Authorize', + 'Widget', + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage', + 'Sync', + 'BaseClient', + 'Env' + ]; + var features = []; + var featuresDone = 0; + var self = this; + + function featureDone() { + featuresDone++; + if (featuresDone === featureList.length) { + setTimeout(function() { + features.caching = !!RemoteStorage.Caching; + features.sync = !!RemoteStorage.Sync; + [ + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage' + ].some(function(cachingLayer) { + if (features.some(function(feature) { return feature.name === cachingLayer; })) { + features.local = RemoteStorage[cachingLayer]; + return true; + } + }); + self.features = features; + callback(features); + }, 0); + } + } + + function featureInitialized(name) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialized."); + features.push({ + name : name, + init : RemoteStorage[name]._rs_init, + supported : true, + cleanup : RemoteStorage[name]._rs_cleanup + }); + featureDone(); + } + + function featureFailed(name, err) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialization failed ( "+err+")"); + featureDone(); + } + + function featureSupported(name, success) { + self.log("[RemoteStorage] [FEATURE "+name+"]" + success ? "":" not"+" supported"); + if (!success) { + featureDone(); + } + } + + function initFeature(name) { + var initResult; + try { + initResult = RemoteStorage[name]._rs_init(self); + } catch(e) { + featureFailed(name, e); + return; + } + if (typeof(initResult) === 'object' && typeof(initResult.then) === 'function') { + initResult.then( + function(){ featureInitialized(name); }, + function(err){ featureFailed(name, err); } + ); + } else { + featureInitialized(name); + } + } + + featureList.forEach(function(featureName) { + self.log("[RemoteStorage] [FEATURE " + featureName + "] initializing..."); + var impl = RemoteStorage[featureName]; + var supported; + + if (impl) { + supported = !impl._rs_supported || impl._rs_supported(); + + if (typeof supported === 'object') { + supported.then( + function(){ + featureSupported(featureName, true); + initFeature(featureName); + }, + function(){ + featureSupported(featureName, false); + } + ); + } + else if (typeof supported === 'boolean') { + featureSupported(featureName, supported); + if (supported) { + initFeature(featureName); + } + } + } else { + featureSupported(featureName, false); + } + }); + }, + + localStorageAvailable: function() { + try { + return !!global.localStorage; + } catch(error) { + return false; + } + }, + + /** + ** GET/PUT/DELETE INTERFACE HELPERS + **/ + + _setGPD: function(impl, context) { + function wrap(f) { + return function() { + return f.apply(context, arguments) + .then(emitUnauthorized.bind(this)); + }; + } + this.get = wrap(impl.get); + this.put = wrap(impl.put); + this.delete = wrap(impl.delete); + }, + + _pendingGPD: function(methodName) { + return function() { + var promise = promising(); + this._pending.push({ + method: methodName, + args: Array.prototype.slice.call(arguments), + promise: promise + }); + return promise; + }.bind(this); + }, + + _processPending: function() { + this._pending.forEach(function(pending) { + try { + this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject); + } catch(e) { + pending.promise.reject(e); + } + }.bind(this)); + this._pending = []; + }, + + /** + ** CHANGE EVENT HANDLING + **/ + + _bindChange: function(object) { + object.on('change', this._dispatchEvent.bind(this, 'change')); + }, + + _dispatchEvent: function(eventName, event) { + for (var path in this._pathHandlers[eventName]) { + var pl = path.length; + var self = this; + this._pathHandlers[eventName][path].forEach(function(handler) { + if (event.path.substr(0, pl) === path) { + var ev = {}; + for (var key in event) { ev[key] = event[key]; } + ev.relativePath = event.path.replace(new RegExp('^' + path), ''); + try { + handler(ev); + } catch(e) { + console.error("'change' handler failed: ", e, e.stack); + self._emit('error', e); + } + } + }); + } + } + }; + + /** + * Property: connected + * + * Boolean property indicating if remoteStorage is currently connected. + */ + Object.defineProperty(RemoteStorage.prototype, 'connected', { + get: function() { + return this.remote.connected; + } + }); + + /** + * Property: access + * + * Tracking claimed access scopes. A instance. + * + * + * Property: caching + * + * Caching settings. A instance. + * + * Not available in no-cache builds. + * + * + * Property: remote + * + * Access to the remote backend used. Usually a . + * + * + * Property: local + * + * Access to the local caching backend used. Usually either a + * or instance. + * + * Not available in no-cache builds. + */ + + global.RemoteStorage = RemoteStorage; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/eventhandling.js **/ +(function(global) { + /** + * Interface: eventhandling + */ + var methods = { + /** + * Method: addEventListener + * + * Install an event handler for the given event name + */ + addEventListener: function(eventName, handler) { + if (typeof(eventName) !== 'string') { + throw new Error('Argument eventName should be a string'); + } + if (typeof(handler) !== 'function') { + throw new Error('Argument handler should be a function'); + } + RemoteStorage.log('[Eventhandling] Adding event listener', eventName, handler); + this._validateEvent(eventName); + this._handlers[eventName].push(handler); + }, + + /** + * Method: removeEventListener + * + * Remove a previously installed event handler + */ + removeEventListener: function(eventName, handler) { + this._validateEvent(eventName); + var hl = this._handlers[eventName].length; + for (var i=0;i + **/ + methods.on = methods.addEventListener; + + /** + * Function: eventHandling + * + * Mixes event handling functionality into an object. + * + * The first parameter is always the object to be extended. + * All remaining parameter are expected to be strings, interpreted as valid event + * names. + * + * Example: + * (start code) + * var MyConstructor = function() { + * eventHandling(this, 'connected', 'disconnected'); + * + * this._emit('connected'); + * this._emit('disconnected'); + * // This would throw an exception: + * // this._emit('something-else'); + * }; + * + * var myObject = new MyConstructor(); + * myObject.on('connected', function() { console.log('connected'); }); + * myObject.on('disconnected', function() { console.log('disconnected'); }); + * // This would throw an exception as well: + * // myObject.on('something-else', function() {}); + * (end code) + */ + RemoteStorage.eventHandling = function(object) { + var eventNames = Array.prototype.slice.call(arguments, 1); + for (var key in methods) { + object[key] = methods[key]; + } + object._handlers = {}; + eventNames.forEach(function(eventName) { + object._addEvent(eventName); + }); + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/wireclient.js **/ +(function(global) { + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.WireClient + * + * WireClient Interface + * -------------------- + * + * This file exposes a get/put/delete interface on top of XMLHttpRequest. + * It requires to be configured with parameters about the remotestorage server to + * connect to. + * Each instance of WireClient is always associated with a single remotestorage + * server and access token. + * + * Usually the WireClient instance can be accessed via `remoteStorage.remote`. + * + * This is the get/put/delete interface: + * + * - #get() takes a path and optionally a ifNoneMatch option carrying a version + * string to check. It returns a promise that will be fulfilled with the HTTP + * response status, the response body, the MIME type as returned in the + * 'Content-Type' header and the current revision, as returned in the 'ETag' + * header. + * - #put() takes a path, the request body and a content type string. It also + * accepts the ifMatch and ifNoneMatch options, that map to the If-Match and + * If-None-Match headers respectively. See the remotestorage-01 specification + * for details on handling these headers. It returns a promise, fulfilled with + * the same values as the one for #get(). + * - #delete() takes a path and the ifMatch option as well. It returns a promise + * fulfilled with the same values as the one for #get(). + * + * In addition to this, the WireClient has some compatibility features to work with + * remotestorage 2012.04 compatible storages. For example it will cache revisions + * from folder listings in-memory and return them accordingly as the "revision" + * parameter in response to #get() requests. Similarly it will return 404 when it + * receives an empty folder listing, to mimic remotestorage-01 behavior. Note + * that it is not always possible to know the revision beforehand, hence it may + * be undefined at times (especially for caching-roots). + */ + + var hasLocalStorage; + var SETTINGS_KEY = "remotestorage:wireclient"; + + var API_2012 = 1, API_00 = 2, API_01 = 3, API_02 = 4, API_HEAD = 5; + + var STORAGE_APIS = { + 'draft-dejong-remotestorage-00': API_00, + 'draft-dejong-remotestorage-01': API_01, + 'draft-dejong-remotestorage-02': API_02, + 'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012 + }; + + var isArrayBufferView; + + if (typeof(ArrayBufferView) === 'function') { + isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); }; + } else { + var arrayBufferViews = [ + Int8Array, Uint8Array, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array + ]; + isArrayBufferView = function(object) { + for (var i=0;i<8;i++) { + if (object instanceof arrayBufferViews[i]) { + return true; + } + } + return false; + }; + } + + function addQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + if (str === '*') { + return '*'; + } + + return '"' + str + '"'; + } + + function stripQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + + return str.replace(/^["']|["']$/g, ''); + } + + function readBinaryData(content, mimeType, callback) { + var blob = new Blob([content], { type: mimeType }); + var reader = new FileReader(); + reader.addEventListener("loadend", function() { + callback(reader.result); // reader.result contains the contents of blob as a typed array + }); + reader.readAsArrayBuffer(blob); + } + + function cleanPath(path) { + return path.replace(/\/+/g, '/').split('/').map(encodeURIComponent).join('/'); + } + + function isFolder(path) { + return (path.substr(-1) === '/'); + } + + function isFolderDescription(body) { + return ((body['@context'] === 'http://remotestorage.io/spec/folder-description') + && (typeof(body['items']) === 'object')); + } + + function isSuccessStatus(status) { + return [201, 204, 304].indexOf(status) >= 0; + } + + function isErrorStatus(status) { + return [401, 403, 404, 412].indexOf(status) >= 0; + } + + var onErrorCb; + + /** + * Class : RemoteStorage.WireClient + **/ + RS.WireClient = function(rs) { + this.connected = false; + + /** + * Event: change + * never fired for some reason + * + * Event: connected + * fired when the wireclient connect method realizes that it is + * in posession of a token and a href + **/ + RS.eventHandling(this, 'change', 'connected', 'wire-busy', 'wire-done', 'not-connected'); + + onErrorCb = function(error){ + if (error instanceof RemoteStorage.Unauthorized) { + this.configure(undefined, undefined, undefined, null); + } + }.bind(this); + rs.on('error', onErrorCb); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + setTimeout(function() { + this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token); + }.bind(this), 0); + } + } + + this._revisionCache = {}; + + if (this.connected) { + setTimeout(this._emit.bind(this), 0, 'connected'); + } + }; + + RS.WireClient.REQUEST_TIMEOUT = 30000; + + RS.WireClient.prototype = { + /** + * Property: token + * + * Holds the bearer token of this WireClient, as obtained in the OAuth dance + * + * Example: + * (start code) + * + * remoteStorage.remote.token + * // -> 'DEADBEEF01==' + */ + + /** + * Property: href + * + * Holds the server's base URL, as obtained in the Webfinger discovery + * + * Example: + * (start code) + * + * remoteStorage.remote.href + * // -> 'https://storage.example.com/users/jblogg/' + */ + + /** + * Property: storageApi + * + * Holds the spec version the server claims to be compatible with + * + * Example: + * (start code) + * + * remoteStorage.remote.storageApi + * // -> 'draft-dejong-remotestorage-01' + */ + + _request: function(method, uri, token, headers, body, getEtag, fakeRevision) { + if ((method === 'PUT' || method === 'DELETE') && uri[uri.length - 1] === '/') { + throw "Don't " + method + " on directories!"; + } + + var promise = promising(); + var revision; + var reqType; + var self = this; + + headers['Authorization'] = 'Bearer ' + token; + + this._emit('wire-busy', { + method: method, + isFolder: isFolder(uri) + }); + + RS.WireClient.request(method, uri, { + body: body, + headers: headers + }, function(error, response) { + if (error) { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: false + }); + self.online = false; + promise.reject(error); + } else { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: true + }); + self.online = true; + if (isErrorStatus(response.status)) { + RemoteStorage.log('[WireClient] Error response status', response.status); + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = undefined; + } + promise.fulfill(response.status, undefined, undefined, revision); + } else if (isSuccessStatus(response.status) || + (response.status === 200 && method !== 'GET')) { + revision = stripQuotes(response.getResponseHeader('ETag')); + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, undefined, undefined, revision); + } else { + var mimeType = response.getResponseHeader('Content-Type'); + var body; + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = response.status === 200 ? fakeRevision : undefined; + } + + if ((!mimeType) || mimeType.match(/charset=binary/)) { + RS.WireClient.readBinaryData(response.response, mimeType, function(result) { + RemoteStorage.log('[WireClient] Successful request with unknown or binary mime-type', revision); + promise.fulfill(response.status, result, mimeType, revision); + }); + } else { + if (mimeType && mimeType.match(/^application\/json/)) { + body = JSON.parse(response.responseText); + } else { + body = response.responseText; + } + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, body, mimeType, revision); + } + } + } + }); + return promise; + }, + + configure: function(userAddress, href, storageApi, token) { + if (typeof(userAddress) !== 'undefined') { + this.userAddress = userAddress; + } + if (typeof(href) !== 'undefined') { + this.href = href; + } + if (typeof(storageApi) !== 'undefined') { + this.storageApi = storageApi; + } + if (typeof(token) !== 'undefined') { + this.token = token; + } + if (typeof(this.storageApi) !== 'undefined') { + this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD; + this.supportsRevs = this._storageApi >= API_00; + } + if (this.href && this.token) { + this.connected = true; + this.online = true; + this._emit('connected'); + } else { + this.connected = false; + } + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ + userAddress: this.userAddress, + href: this.href, + token: this.token, + storageApi: this.storageApi + }); + } + RS.WireClient.configureHooks.forEach(function(hook) { + hook.call(this); + }.bind(this)); + }, + + stopWaitingForToken: function() { + if (!this.connected) { + this._emit('not-connected'); + } + }, + + get: function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } else if (options.ifNoneMatch) { + var oldRev = this._revisionCache[path]; + } + var promise = this._request('GET', this.href + cleanPath(path), this.token, headers, + undefined, this.supportsRevs, this._revisionCache[path]); + if (isFolder(path)) { + return promise.then(function(status, body, contentType, revision) { + var itemsMap = {}; + + // New folder listing received + if (status === 200 && typeof(body) === 'object') { + // Empty folder listing of any spec + if (Object.keys(body).length === 0) { + status = 404; + } + // >= 02 spec + else if (isFolderDescription(body)) { + for (var item in body.items) { + this._revisionCache[path + item] = body.items[item].ETag; + } + itemsMap = body.items; + } + // < 02 spec + else { + Object.keys(body).forEach(function(key){ + this._revisionCache[path + key] = body[key]; + itemsMap[key] = {"ETag": body[key]}; + }.bind(this)); + } + return promising().fulfill(status, itemsMap, contentType, revision); + } else { + return promising().fulfill(status, body, contentType, revision); + } + }.bind(this)); + } else { + return promise; + } + }, + + put: function(path, body, contentType, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + if (!contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var headers = { 'Content-Type': contentType }; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } + return this._request('PUT', this.href + cleanPath(path), this.token, + headers, body, this.supportsRevs); + }, + + 'delete': function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + } + return this._request('DELETE', this.href + cleanPath(path), this.token, + headers, + undefined, this.supportsRevs); + } + }; + + // Shared cleanPath used by Dropbox + RS.WireClient.cleanPath = cleanPath; + + // Shared isArrayBufferView used by WireClient and Dropbox + RS.WireClient.isArrayBufferView = isArrayBufferView; + + RS.WireClient.readBinaryData = readBinaryData; + + // Shared request function used by WireClient, GoogleDrive and Dropbox. + RS.WireClient.request = function(method, url, options, callback) { + RemoteStorage.log('[WireClient]', method, url); + + callback = callback.bind(this); + + var timedOut = false; + + var timer = setTimeout(function() { + timedOut = true; + callback('timeout'); + }, RS.WireClient.REQUEST_TIMEOUT); + + var xhr = new XMLHttpRequest(); + xhr.open(method, url, true); + + if (options.responseType) { + xhr.responseType = options.responseType; + } + if (options.headers) { + for (var key in options.headers) { + xhr.setRequestHeader(key, options.headers[key]); + } + } + + xhr.onload = function() { + if (timedOut) { return; } + clearTimeout(timer); + callback(null, xhr); + }; + + xhr.onerror = function(error) { + if (timedOut) { return; } + clearTimeout(timer); + callback(error); + }; + + var body = options.body; + + if (typeof(body) === 'object') { + if (isArrayBufferView(body)) { + /* alright. */ + //FIXME empty block + } + else if (body instanceof ArrayBuffer) { + body = new Uint8Array(body); + } else { + body = JSON.stringify(body); + } + } + xhr.send(body); + }; + + Object.defineProperty(RemoteStorage.WireClient.prototype, 'storageType', { + get: function() { + if (this.storageApi) { + var spec = this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/); + return spec ? spec[1] : '2012.04'; + } + } + }); + + RS.WireClient.configureHooks = []; + + RS.WireClient._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + remoteStorage.remote = new RS.WireClient(remoteStorage); + this.online = true; + }; + + RS.WireClient._rs_supported = function() { + return !! global.XMLHttpRequest; + }; + + RS.WireClient._rs_cleanup = function(remoteStorage){ + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + remoteStorage.removeEventListener('error', onErrorCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/discover.js **/ +(function(global) { + + // feature detection flags + var haveXMLHttpRequest, hasLocalStorage; + // used to store settings in localStorage + var SETTINGS_KEY = 'remotestorage:discover'; + // cache loaded from localStorage + var cachedInfo = {}; + + /** + * Class: RemoteStorage.Discover + * + * This class deals with the webfinger lookup + * + * Arguments: + * userAddress - user@host + * callback - gets called with href of the storage, the type and the authURL + * Example: + * (start code) + * + * (end code) + **/ + + RemoteStorage.Discover = function(userAddress, callback) { + if (userAddress in cachedInfo) { + var info = cachedInfo[userAddress]; + callback(info.href, info.type, info.authURL); + return; + } + var hostname = userAddress.split('@')[1]; + var params = '?resource=' + encodeURIComponent('acct:' + userAddress); + var urls = [ + 'https://' + hostname + '/.well-known/webfinger' + params, + 'https://' + hostname + '/.well-known/host-meta.json' + params, + 'http://' + hostname + '/.well-known/webfinger' + params, + 'http://' + hostname + '/.well-known/host-meta.json' + params + ]; + + function tryOne() { + var xhr = new XMLHttpRequest(); + var url = urls.shift(); + if (!url) { return callback(); } + RemoteStorage.log('[Discover] Trying URL', url); + xhr.open('GET', url, true); + xhr.onabort = xhr.onerror = function() { + console.error("webfinger error", arguments, '(', url, ')'); + tryOne(); + }; + xhr.onload = function() { + if (xhr.status !== 200) { return tryOne(); } + var profile; + + try { + profile = JSON.parse(xhr.responseText); + } catch(e) { + RemoteStorage.log("[Discover] Failed to parse profile ", xhr.responseText, e); + tryOne(); + return; + } + + if (!profile.links) { + RemoteStorage.log("[Discover] Profile has no links section ", JSON.stringify(profile)); + tryOne(); + return; + } + + var link; + profile.links.forEach(function(l) { + if (l.rel === 'remotestorage') { + link = l; + } else if (l.rel === 'remoteStorage' && !link) { + link = l; + } + }); + RemoteStorage.log('[Discover] Got profile', profile, 'and link', link); + if (link) { + var authURL = link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'] + || link.properties['auth-endpoint'], + storageType = link.properties['http://remotestorage.io/spec/version'] + || link.type; + cachedInfo[userAddress] = { href: link.href, type: storageType, authURL: authURL }; + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo }); + } + callback(link.href, storageType, authURL); + } else { + tryOne(); + } + }; + xhr.send(); + } + tryOne(); + }; + + RemoteStorage.Discover._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + cachedInfo = settings.cache; + } + } + }; + + RemoteStorage.Discover._rs_supported = function() { + haveXMLHttpRequest = !! global.XMLHttpRequest; + return haveXMLHttpRequest; + }; + + RemoteStorage.Discover._rs_cleanup = function() { + if (hasLocalStorage) { + delete localStorage[SETTINGS_KEY]; + } + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/authorize.js **/ +(function(global) { + + function extractParams() { + //FF already decodes the URL fragment in document.location.hash, so use this instead: + var location = RemoteStorage.Authorize.getLocation(), + hashPos = location.href.indexOf('#'), + hash; + if (hashPos === -1) { return; } + hash = location.href.substring(hashPos+1); + // if hash is not of the form #key=val&key=val, it's probably not for us + if (hash.indexOf('=') === -1) { return; } + return hash.split('&').reduce(function(m, kvs) { + var kv = kvs.split('='); + m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]); + return m; + }, {}); + } + + RemoteStorage.Authorize = function(authURL, scope, redirectUri, clientId) { + RemoteStorage.log('[Authorize] authURL = ', authURL); + + var url = authURL; + url += authURL.indexOf('?') > 0 ? '&' : '?'; + url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, '')); + url += '&scope=' + encodeURIComponent(scope); + url += '&client_id=' + encodeURIComponent(clientId); + url += '&response_type=token'; + RemoteStorage.Authorize.setLocation(url); + }; + + RemoteStorage.prototype.authorize = function(authURL) { + this.access.setStorageType(this.remote.storageType); + var scope = this.access.scopeParameter; + + var redirectUri = String(RemoteStorage.Authorize.getLocation()); + var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0]; + + RemoteStorage.Authorize(authURL, scope, redirectUri, clientId); + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.getLocation = function () { + return global.document.location; + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.setLocation = function (location) { + if (typeof location === 'string') { + global.document.location.href = location; + } else if (typeof location === 'object') { + global.document.location = location; + } else { + throw "Invalid location " + location; + } + }; + + RemoteStorage.Authorize._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + + var onFeaturesLoaded; + RemoteStorage.Authorize._rs_init = function(remoteStorage) { + + onFeaturesLoaded = function () { + var authParamsUsed = false; + if (params) { + if (params.error) { + throw "Authorization server errored: " + params.error; + } + if (params.access_token) { + remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token); + authParamsUsed = true; + } + if (params.remotestorage) { + remoteStorage.connect(params.remotestorage); + authParamsUsed = true; + } + } + if (!authParamsUsed) { + remoteStorage.remote.stopWaitingForToken(); + } + }; + var params = extractParams(), + location; + if (params) { + location = RemoteStorage.Authorize.getLocation(); + location.hash = ''; + } + remoteStorage.on('features-loaded', onFeaturesLoaded); + }; + + RemoteStorage.Authorize._rs_cleanup = function(remoteStorage) { + remoteStorage.removeEventListener('features-loaded', onFeaturesLoaded); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/access.js **/ +(function(global) { + + var SETTINGS_KEY = "remotestorage:access"; + + /** + * Class: RemoteStorage.Access + * + * Keeps track of claimed access and scopes. + */ + RemoteStorage.Access = function() { + this.reset(); + }; + + RemoteStorage.Access.prototype = { + + /** + * Method: claim + * + * Claim access on a given scope with given mode. + * + * Parameters: + * scope - An access scope, such as "contacts" or "calendar". + * mode - Access mode to use. Either "r" or "rw". + * + * Example: + * (start code) + * remoteStorage.access.claim('contacts', 'r'); + * remoteStorage.access.claim('pictures', 'rw'); + */ + claim: function(scope, mode) { + if (typeof(scope) !== 'string' || scope.indexOf('/') !== -1 || scope.length === 0) { + throw new Error('Scope should be a non-empty string without forward slashes'); + } + if (!mode.match(/^rw?$/)) { + throw new Error('Mode should be either \'r\' or \'rw\''); + } + this._adjustRootPaths(scope); + this.scopeModeMap[scope] = mode; + }, + + get: function(scope) { + return this.scopeModeMap[scope]; + }, + + remove: function(scope) { + var savedMap = {}; + var name; + for (name in this.scopeModeMap) { + savedMap[name] = this.scopeModeMap[name]; + } + this.reset(); + delete savedMap[scope]; + for (name in savedMap) { + this.set(name, savedMap[name]); + } + }, + + /** + * Verify permission for a given scope. + */ + checkPermission: function(scope, mode) { + var actualMode = this.get(scope); + return actualMode && (mode === 'r' || actualMode === 'rw'); + }, + + /** + * Verify permission for a given path. + */ + checkPathPermission: function(path, mode) { + if (this.checkPermission('*', mode)) { + return true; + } + return !!this.checkPermission(this._getModuleName(path), mode); + }, + + reset: function() { + this.rootPaths = []; + this.scopeModeMap = {}; + }, + + /** + * Return the module name for a given path. + */ + _getModuleName: function(path) { + if (path[0] !== '/') { + throw new Error('Path should start with a slash'); + } + var moduleMatch = path.replace(/^\/public/, '').match(/^\/([^\/]*)\//); + return moduleMatch ? moduleMatch[1] : '*'; + }, + + _adjustRootPaths: function(newScope) { + if ('*' in this.scopeModeMap || newScope === '*') { + this.rootPaths = ['/']; + } else if (! (newScope in this.scopeModeMap)) { + this.rootPaths.push('/' + newScope + '/'); + this.rootPaths.push('/public/' + newScope + '/'); + } + }, + + _scopeNameForParameter: function(scope) { + if (scope.name === '*' && this.storageType) { + if (this.storageType === '2012.04') { + return ''; + } else if (this.storageType.match(/remotestorage-0[01]/)) { + return 'root'; + } + } + return scope.name; + }, + + setStorageType: function(type) { + this.storageType = type; + } + }; + + /** + * Property: scopes + * + * Holds an array of claimed scopes in the form + * > { name: "", mode: "" } + */ + Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', { + get: function() { + return Object.keys(this.scopeModeMap).map(function(key) { + return { name: key, mode: this.scopeModeMap[key] }; + }.bind(this)); + } + }); + + Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', { + get: function() { + return this.scopes.map(function(scope) { + return this._scopeNameForParameter(scope) + ':' + scope.mode; + }.bind(this)).join(' '); + } + }); + + // Documented in src/remotestorage.js + Object.defineProperty(RemoteStorage.prototype, 'access', { + get: function() { + var access = new RemoteStorage.Access(); + Object.defineProperty(this, 'access', { + value: access + }); + return access; + }, + configurable: true + }); + + RemoteStorage.Access._rs_init = function() {}; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/env.js **/ +(function(pMode) { + + var mode = pMode, + env = {}, + isBackground = false; + + + RemoteStorage.Env = function() { + return env; + }; + + RemoteStorage.Env.isBrowser = function () { + return mode === "browser"; + }; + + RemoteStorage.Env.isNode = function () { + return mode === "node"; + }; + + RemoteStorage.Env.goBackground = function () { + isBackground = true; + RemoteStorage.Env._emit("background"); + }; + + RemoteStorage.Env.goForeground = function () { + isBackground = false; + RemoteStorage.Env._emit("foreground"); + }; + + RemoteStorage.Env._rs_init = function(remoteStorage) { + RemoteStorage.eventHandling(RemoteStorage.Env, "background", "foreground"); + + if ( mode === 'browser') { + if ( typeof(document.hidden) !== "undefined" ) { + env.hiddenProperty = "hidden"; + env.visibilityChangeEvent = "visibilitychange"; + } else if ( typeof(document.mozHidden) !== "undefined" ) { + env.hiddenProperty = "mozHidden"; + env.visibilityChangeEvent = "mozvisibilitychange"; + } else if ( typeof(document.msHidden) !== "undefined" ) { + env.hiddenProperty = "msHidden"; + env.visibilityChangeEvent = "msvisibilitychange"; + } else if ( typeof(document.webkitHidden) !== "undefined" ) { + env.hiddenProperty = "webkitHidden"; + env.visibilityChangeEvent = "webkitvisibilitychange"; + } + document.addEventListener(env.visibilityChangeEvent, function () { + if ( document[env.hiddenProperty] ) { + RemoteStorage.Env.goBackground(); + } else { + RemoteStorage.Env.goForeground(); + } + }, false); + } + }; + + RemoteStorage.Env._rs_cleanup = function(remoteStorage) { + }; + +})(typeof(window) !== 'undefined' ? 'browser' : 'node'); + + +/** FILE: src/i18n.js **/ +(function() { + /** + * Class: RemoteStorage.I18n + * + * TODO add documentation + **/ + + "use strict"; + + var dictionary = { + "view_info": 'This app allows you to use your own storage. Learn more!', + "view_connect": "Connect remote storage", + "view_connecting": "Connecting %s", + "view_offline": "Offline", + "view_error_occured": "Sorry! An error occured.", + "view_confirm_reset": "Are you sure you want to reset everything? This will clear your local data and reload the page.", + "view_get_me_out": "Get me out of here!", + "view_error_plz_report": 'If this problem persists, please let us know!', + "view_unauthorized": "Unauthorized! Click here to reconnect." + }; + + RemoteStorage.I18n = { + + translate: function() { + var str = arguments[0], + params = Array.prototype.splice.call(arguments, 1); + + if (typeof dictionary[str] !== "string") { + throw "Unknown translation string: " + str; + } else { + str = dictionary[str]; + } + return (str.replace(/%s/g, function(){ return params.shift(); })); + }, + + getDictionary: function() { + return dictionary; + }, + + setDictionary: function(newDictionary) { + dictionary = newDictionary; + } + + }; +})(); + + +/** FILE: src/assets.js **/ +/** THIS FILE WAS GENERATED BY build/compile-assets.js. DO NOT CHANGE IT MANUALLY, BUT INSTEAD CHANGE THE ASSETS IN assets/. **/ +RemoteStorage.Assets = { + + connectIcon: '', + disconnectIcon: '', + dropbox: '', + googledrive: '', + remoteStorageIcon: '', + remoteStorageIconError: '', + remoteStorageIconOffline: '', + syncIcon: '', + widget: '
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ', + widgetCss: '/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}' +}; + + +/** FILE: src/widget.js **/ +(function(window) { + + var hasLocalStorage; + var LS_STATE_KEY = "remotestorage:widget:state"; + // states allowed to immediately jump into after a reload. + var VALID_ENTRY_STATES = { + initial: true, + connected: true, + offline: true + }; + + function stateSetter(widget, state) { + RemoteStorage.log('[Widget] Producing stateSetter for', state); + return function() { + RemoteStorage.log('[Widget] Setting state', state, arguments); + if (hasLocalStorage) { + localStorage[LS_STATE_KEY] = state; + } + if (widget.view) { + if (widget.rs.remote) { + widget.view.setUserAddress(widget.rs.remote.userAddress); + } + widget.view.setState(state, arguments); + } else { + widget._rememberedState = state; + } + }; + } + + function errorsHandler(widget) { + return function(error) { + if (error instanceof RemoteStorage.DiscoveryError) { + console.error('Discovery failed', error, '"' + error.message + '"'); + widget.view.setState('initial', [error.message]); + } else if (error instanceof RemoteStorage.SyncError) { + widget.view.setState('offline', []); + } else if (error instanceof RemoteStorage.Unauthorized) { + widget.view.setState('unauthorized'); + } else { + RemoteStorage.log('[Widget] Unknown error'); + widget.view.setState('error', [error]); + } + }; + } + + function flashFor(evt) { + if (evt.method === 'GET' && evt.isFolder) { + return false; + } + return true; + } + + /** + * Class: RemoteStorage.Widget + * + * The widget controller that communicates with the view and listens to + * its remoteStorage instance. + * + * While listening to the events emitted by its remoteStorage it sets + * corresponding states of the view. + * + * - connected -> connected + * - disconnected -> initial + * - connecting -> authing + * - authing -> authing + * - wire-busy -> busy + * - wire-done -> connected + * - error -> one of initial, offline, unauthorized, or error + **/ + RemoteStorage.Widget = function(remoteStorage) { + var self = this; + var requestsToFlashFor = 0; + + // setting event listeners on rs events to put + // the widget into corresponding states + this.rs = remoteStorage; + this.rs.remote.on('connected', stateSetter(this, 'connected')); + this.rs.on('disconnected', stateSetter(this, 'initial')); + this.rs.on('connecting', stateSetter(this, 'authing')); + this.rs.on('authing', stateSetter(this, 'authing')); + this.rs.on('error', errorsHandler(this)); + + if (this.rs.remote) { + this.rs.remote.on('wire-busy', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor++; + stateSetter(self, 'busy')(); + } + }); + + this.rs.remote.on('wire-done', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor--; + if (requestsToFlashFor <= 0) { + stateSetter(self, 'connected')(); + } + } + }); + } + + if (hasLocalStorage) { + var state = localStorage[LS_STATE_KEY]; + if (state && VALID_ENTRY_STATES[state]) { + this._rememberedState = state; + } + } + }; + + RemoteStorage.Widget.prototype = { + + /** + * Method: display + * + * Displays the widget via the view.display method + * + * Parameters: + * + * domID + **/ + display: function(domID) { + if (! this.view) { + this.setView(new RemoteStorage.Widget.View(this.rs)); + } + this.view.display.apply(this.view, arguments); + return this; + }, + + linkWidgetToSync: function() { + if (typeof(this.rs.sync) === 'object' && typeof(this.rs.sync.sync) === 'function') { + this.view.on('sync', this.rs.sync.sync.bind(this.rs.sync)); + } else { + RemoteStorage.log('[Widget] typeof this.rs.sync check fail', this.rs.sync); + setTimeout(this.linkWidgetToSync.bind(this), 1000); + } + }, + + /** + * Method: setView(view) + * + * Sets the view and initializes event listeners to react on + * widget (widget.view) events + **/ + setView: function(view) { + this.view = view; + this.view.on('connect', function(options) { + if (typeof(options) === 'string') { + // options is simply a useraddress + this.rs.connect(options); + } else if (options.special) { + this.rs[options.special].connect(options); + } + }.bind(this)); + this.view.on('disconnect', this.rs.disconnect.bind(this.rs)); + this.linkWidgetToSync(); + try { + this.view.on('reset', function(){ + var location = RemoteStorage.Authorize.getLocation(); + this.rs.on('disconnected', location.reload.bind(location)); + this.rs.disconnect(); + }.bind(this)); + } catch(e) { + if (e.message && e.message.match(/Unknown event/)) { + // ignored. (the 0.7 widget-view interface didn't have a 'reset' event) + } else { + throw e; + } + } + + if (this._rememberedState) { + setTimeout(stateSetter(this, this._rememberedState), 0); + delete this._rememberedState; + } + } + }; + + /** + * Method: displayWidget + * + * Same as + **/ + RemoteStorage.prototype.displayWidget = function(domID) { + return this.widget.display(domID); + }; + + RemoteStorage.Widget._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (! remoteStorage.widget) { + remoteStorage.widget = new RemoteStorage.Widget(remoteStorage); + } + }; + + RemoteStorage.Widget._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/view.js **/ +(function(window){ + + var t = RemoteStorage.I18n.translate; + + var cEl = function(){ + return document.createElement.apply(document, arguments); + }; + + function gCl(parent, className) { + return parent.getElementsByClassName(className)[0]; + } + + function gTl(parent, tagName) { + return parent.getElementsByTagName(tagName)[0]; + } + + function removeClass(el, className) { + return el.classList.remove(className); + } + + function addClass(el, className) { + return el.classList.add(className); + } + + function stopPropagation(event) { + if (typeof(event.stopPropagation) === 'function') { + event.stopPropagation(); + } else { + event.cancelBubble = true; + } + } + + function setupButton(parent, className, iconName, eventListener) { + var element = gCl(parent, className); + if (typeof iconName !== 'undefined') { + var img = gTl(element, 'img'); + (img || element).src = RemoteStorage.Assets[iconName]; + } + element.addEventListener('click', eventListener); + return element; + } + + /** + * Class: RemoteStorage.Widget.View + * + * Controls the visible widget + * + * States: + * + * initial - not connected + * authing - in auth flow + * connected - connected to remote storage, not syncing at the moment + * busy - connected, syncing at the moment + * offline - connected, but no network connectivity + * error - connected, but sync error happened + * unauthorized - connected, but request returned 401 + **/ + RemoteStorage.Widget.View = function(remoteStorage) { + this.rs = remoteStorage; + if (typeof(document) === 'undefined') { + throw "Widget not supported"; + } + RemoteStorage.eventHandling(this, + 'connect', + 'disconnect', + 'sync', + 'display', + 'reset'); + + // Re-binding the event so they can be called from the window + for (var event in this.events){ + this.events[event] = this.events[event].bind(this); + } + + this.hideBubbleOnBodyClick = function(event) { + for (var p = event.target; p !== document.body; p = p.parentElement) { + if (p.id === 'remotestorage-widget') { + return; + } + } + this.hideBubble(); + }.bind(this); + }; + + RemoteStorage.Widget.View.prototype = { + + connectGdrive: function() { + this._emit('connect', { special: 'googledrive' }); + }, + + connectDropbox: function(){ + this._emit('connect', { special: 'dropbox'}); + }, + + /** + * Method: setState + * + * Call the function that applies the state to the widget + * + * Parameters: + * + * state + * args + **/ + setState: function(state, args) { + RemoteStorage.log('[View] widget.view.setState(',state,',',args,');'); + var s = this.states[state]; + if (typeof(s) === 'undefined') { + throw new Error("Bad State assigned to view: " + state); + } + s.apply(this, args); + }, + + /** + * Method: setUserAddres + * + * Set user address of the input field + **/ + setUserAddress: function(addr) { + this.userAddress = addr || ''; + + var el; + if (this.div && (el = gTl(this.div, 'form').userAddress)) { + el.value = this.userAddress; + } + }, + + /** + * Method: toggleBubble + * + * Show the bubble when hidden and the other way around + **/ + toggleBubble: function(event) { + if (this.bubble.className.search('rs-hidden') < 0) { + this.hideBubble(event); + } else { + this.showBubble(event); + } + }, + + /** + * Method: hideBubble + * + * Hide the bubble + **/ + hideBubble: function(){ + addClass(this.bubble, 'rs-hidden'); + document.body.removeEventListener('click', this.hideBubbleOnBodyClick); + }, + + /** + * Method: showBubble + * + * Show the bubble + **/ + showBubble: function(event){ + removeClass(this.bubble, 'rs-hidden'); + if (typeof(event) !== 'undefined') { + stopPropagation(event); + } + document.body.addEventListener('click', this.hideBubbleOnBodyClick); + gTl(this.bubble,'form').userAddress.focus(); + }, + + /** + * Method: display + * + * Draw the widget inside of the dom element with the id domID + * + * Parameters: + * + * domID + * + * Returns: + * + * The widget div + **/ + display: function(domID) { + if (typeof this.div !== 'undefined') { + return this.div; + } + + var element = cEl('div'); + var style = cEl('style'); + style.innerHTML = RemoteStorage.Assets.widgetCss; + + element.id = "remotestorage-widget"; + + element.innerHTML = RemoteStorage.Assets.widget; + + element.appendChild(style); + if (domID) { + var parent = document.getElementById(domID); + if (! parent) { + throw "Failed to find target DOM element with id=\"" + domID + "\""; + } + parent.appendChild(element); + } else { + document.body.appendChild(element); + } + + // Sync button + setupButton(element, 'rs-sync', 'syncIcon', this.events.sync); + + // Disconnect button + setupButton(element, 'rs-disconnect', 'disconnectIcon', this.events.disconnect); + + // Get me out of here + setupButton(element, 'remotestorage-reset', undefined, this.events.reset); + + // Connect button + var cb = setupButton(element, 'connect', 'connectIcon', this.events.connect); + + // Input + this.form = gTl(element, 'form'); + var el = this.form.userAddress; + el.addEventListener('keyup', function(event) { + if (event.target.value) { + cb.removeAttribute('disabled'); + } else { + cb.setAttribute('disabled','disabled'); + } + }); + if (this.userAddress) { + el.value = this.userAddress; + } + + // The cube + this.cube = setupButton(element, 'rs-cube', 'remoteStorageIcon', this.toggleBubble.bind(this)); + + // Google Drive and Dropbox icons + setupButton(element, 'rs-dropbox', 'dropbox', this.connectDropbox.bind(this)); + setupButton(element, 'rs-googledrive', 'googledrive', this.connectGdrive.bind(this)); + + var bubbleDontCatch = { INPUT: true, BUTTON: true, IMG: true }; + var eventListener = function(event) { + if (! bubbleDontCatch[event.target.tagName] && ! (this.div.classList.contains('remotestorage-state-unauthorized') )) { + this.showBubble(event); + } + }.bind(this); + this.bubble = setupButton(element, 'rs-bubble', undefined, eventListener); + + this.hideBubble(); + + this.div = element; + + this.states.initial.call(this); + this.events.display.call(this); + return this.div; + }, + + _renderTranslatedInitialContent: function() { + gCl(this.div, 'rs-status-text').innerHTML = t("view_connect"); + gCl(this.div, 'remotestorage-reset').innerHTML = t("view_get_me_out"); + gCl(this.div, 'rs-error-plz-report').innerHTML = t("view_error_plz_report"); + gCl(this.div, 'remotestorage-unauthorized').innerHTML = t("view_unauthorized"); + }, + + states: { + initial: function(message) { + var cube = this.cube; + var info = message || t("view_info"); + + this._renderTranslatedInitialContent(); + + if (message) { + cube.src = RemoteStorage.Assets.remoteStorageIconError; + removeClass(this.cube, 'remotestorage-loading'); + this.showBubble(); + + // Show the red error cube for 5 seconds, then show the normal orange one again + setTimeout(function(){ + cube.src = RemoteStorage.Assets.remoteStorageIcon; + },5000); + } else { + this.hideBubble(); + } + this.div.className = "remotestorage-state-initial"; + + // Google Drive and Dropbox icons + var backends = 1; + if (this._activateBackend('dropbox')) { backends += 1; } + if (this._activateBackend('googledrive')) { backends += 1; } + gCl(this.div, 'rs-bubble-text').style.paddingRight = backends*32+8+'px'; + + // If address not empty connect button enabled + var cb = gCl(this.div, 'connect'); + if (this.form.userAddress.value) { + cb.removeAttribute('disabled'); + } + + var infoEl = gCl(this.div, 'rs-info-msg'); + infoEl.innerHTML = info; + + if (message) { + infoEl.classList.add('remotestorage-error-info'); + } else { + infoEl.classList.remove('remotestorage-error-info'); + } + }, + + authing: function() { + this.div.removeEventListener('click', this.events.connect); + this.div.className = "remotestorage-state-authing"; + gCl(this.div, 'rs-status-text').innerHTML = t("view_connecting", this.userAddress); + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone, when is that neccesary + }, + + connected: function() { + this.div.className = "remotestorage-state-connected"; + gCl(this.div, 'userAddress').innerHTML = this.userAddress; + this.cube.src = RemoteStorage.Assets.remoteStorageIcon; + removeClass(this.cube, 'remotestorage-loading'); + var icons = { + googledrive: gCl(this.div, 'rs-googledrive'), + dropbox: gCl(this.div, 'rs-dropbox') + }; + icons.googledrive.style.display = icons.dropbox.style.display = 'none'; + if (icons[this.rs.backend]) { + icons[this.rs.backend].style.display = 'inline-block'; + gCl(this.div, 'rs-bubble-text').style.paddingRight = 2*32+8+'px'; + } else { + gCl(this.div, 'rs-bubble-text').style.paddingRight = 32+8+'px'; + } + }, + + busy: function() { + this.div.className = "remotestorage-state-busy"; + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone when is that neccesary + this.hideBubble(); + }, + + offline: function() { + this.div.className = "remotestorage-state-offline"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconOffline; + gCl(this.div, 'rs-status-text').innerHTML = t("view_offline"); + }, + + error: function(err) { + var errorMsg = err; + this.div.className = "remotestorage-state-error"; + + gCl(this.div, 'rs-bubble-text').innerHTML = ''+t('view_error_occured')+''; + //FIXME I don't know what an DOMError is and my browser doesn't know too(how to handle this?) + if (err instanceof Error /*|| err instanceof DOMError*/) { + errorMsg = err.message + '\n\n' + + err.stack; + } + gCl(this.div, 'rs-error-msg').textContent = errorMsg; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + }, + + unauthorized: function() { + this.div.className = "remotestorage-state-unauthorized"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + this.div.addEventListener('click', this.events.connect); + } + }, + + events: { + /** + * Event: connect + * + * Emitted when the connect button is clicked + **/ + connect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('connect', gTl(this.div, 'form').userAddress.value); + }, + + /** + * Event: sync + * + * Emitted when the sync button is clicked + **/ + sync: function(event) { + stopPropagation(event); + event.preventDefault(); + + this._emit('sync'); + }, + + /** + * Event: disconnect + * + * Emitted when the disconnect button is clicked + **/ + disconnect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('disconnect'); + }, + + /** + * Event: reset + * + * Emitted after crash triggers disconnect + **/ + reset: function(event){ + event.preventDefault(); + var result = window.confirm(t('view_confirm_reset')); + if (result){ + this._emit('reset'); + } + }, + + /** + * Event: display + * + * Emitted when finished displaying the widget + **/ + display : function(event) { + if (event) { + event.preventDefault(); + } + this._emit('display'); + } + }, + + _activateBackend: function activateBackend(backendName) { + var className = 'rs-' + backendName; + if (this.rs.apiKeys[backendName]) { + gCl(this.div, className).style.display = 'inline-block'; + return true; + } else { + gCl(this.div, className).style.display = 'none'; + return false; + } + } + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: lib/tv4.js **/ +/** +Author: Geraint Luff and others +Year: 2013 + +This code is released into the "public domain" by its author(s). Anybody may use, alter and distribute the code without restriction. The author makes no guarantees, and takes no liability of any kind for use of this code. + +If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory. +**/ + +(function (global) { +var ValidatorContext = function (parent, collectMultiple) { + this.missing = []; + this.schemas = parent ? Object.create(parent.schemas) : {}; + this.collectMultiple = collectMultiple; + this.errors = []; + this.handleError = collectMultiple ? this.collectError : this.returnError; +}; +ValidatorContext.prototype.returnError = function (error) { + return error; +}; +ValidatorContext.prototype.collectError = function (error) { + if (error) { + this.errors.push(error); + } + return null; +} +ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) { + for (var i = startIndex; i < this.errors.length; i++) { + this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath); + } + return this; +} + +ValidatorContext.prototype.getSchema = function (url) { + if (this.schemas[url] != undefined) { + var schema = this.schemas[url]; + return schema; + } + var baseUrl = url; + var fragment = ""; + if (url.indexOf('#') != -1) { + fragment = url.substring(url.indexOf("#") + 1); + baseUrl = url.substring(0, url.indexOf("#")); + } + if (this.schemas[baseUrl] != undefined) { + var schema = this.schemas[baseUrl]; + var pointerPath = decodeURIComponent(fragment); + if (pointerPath == "") { + return schema; + } else if (pointerPath.charAt(0) != "/") { + return undefined; + } + var parts = pointerPath.split("/").slice(1); + for (var i = 0; i < parts.length; i++) { + var component = parts[i].replace("~1", "/").replace("~0", "~"); + if (schema[component] == undefined) { + schema = undefined; + break; + } + schema = schema[component]; + } + if (schema != undefined) { + return schema; + } + } + if (this.missing[baseUrl] == undefined) { + this.missing.push(baseUrl); + this.missing[baseUrl] = baseUrl; + } +}; +ValidatorContext.prototype.addSchema = function (url, schema) { + var map = {}; + map[url] = schema; + normSchema(schema, url); + searchForTrustedSchemas(map, schema, url); + for (var key in map) { + this.schemas[key] = map[key]; + } + return map; +}; + +ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) { + if (schema['$ref'] != undefined) { + schema = this.getSchema(schema['$ref']); + if (!schema) { + return null; + } + } + + var errorCount = this.errors.length; + var error = this.validateBasic(data, schema) + || this.validateNumeric(data, schema) + || this.validateString(data, schema) + || this.validateArray(data, schema) + || this.validateObject(data, schema) + || this.validateCombinations(data, schema) + || null + if (error || errorCount != this.errors.length) { + while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) { + var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null; + var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null; + if (error) { + error = error.prefixWith(dataPart, schemaPart); + } + this.prefixErrors(errorCount, dataPart, schemaPart); + } + } + + return this.handleError(error); +} + +function recursiveCompare(A, B) { + if (A === B) { + return true; + } + if (typeof A == "object" && typeof B == "object") { + if (Array.isArray(A) != Array.isArray(B)) { + return false; + } else if (Array.isArray(A)) { + if (A.length != B.length) { + return false + } + for (var i = 0; i < A.length; i++) { + if (!recursiveCompare(A[i], B[i])) { + return false; + } + } + } else { + for (var key in A) { + if (B[key] === undefined && A[key] !== undefined) { + return false; + } + } + for (var key in B) { + if (A[key] === undefined && B[key] !== undefined) { + return false; + } + } + for (var key in A) { + if (!recursiveCompare(A[key], B[key])) { + return false; + } + } + } + return true; + } + return false; +} + +ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) { + var error; + if (error = this.validateType(data, schema)) { + return error.prefixWith(null, "type"); + } + if (error = this.validateEnum(data, schema)) { + return error.prefixWith(null, "type"); + } + return null; +} + +ValidatorContext.prototype.validateType = function validateType(data, schema) { + if (schema.type == undefined) { + return null; + } + var dataType = typeof data; + if (data == null) { + dataType = "null"; + } else if (Array.isArray(data)) { + dataType = "array"; + } + var allowedTypes = schema.type; + if (typeof allowedTypes != "object") { + allowedTypes = [allowedTypes]; + } + + for (var i = 0; i < allowedTypes.length; i++) { + var type = allowedTypes[i]; + if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) { + return null; + } + } + return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType); +} + +ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) { + if (schema["enum"] == undefined) { + return null; + } + for (var i = 0; i < schema["enum"].length; i++) { + var enumVal = schema["enum"][i]; + if (recursiveCompare(data, enumVal)) { + return null; + } + } + return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data)); +} +ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) { + return this.validateMultipleOf(data, schema) + || this.validateMinMax(data, schema) + || null; +} + +ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) { + var multipleOf = schema.multipleOf || schema.divisibleBy; + if (multipleOf == undefined) { + return null; + } + if (typeof data == "number") { + if (data%multipleOf != 0) { + return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf); + } + } + return null; +} + +ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) { + if (typeof data != "number") { + return null; + } + if (schema.minimum != undefined) { + if (data < schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum"); + } + if (schema.exclusiveMinimum && data == schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum"); + } + } + if (schema.maximum != undefined) { + if (data > schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum"); + } + if (schema.exclusiveMaximum && data == schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum"); + } + } + return null; +} +ValidatorContext.prototype.validateString = function validateString(data, schema) { + return this.validateStringLength(data, schema) + || this.validateStringPattern(data, schema) + || null; +} + +ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) { + if (typeof data != "string") { + return null; + } + if (schema.minLength != undefined) { + if (data.length < schema.minLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength"); + } + } + if (schema.maxLength != undefined) { + if (data.length > schema.maxLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength"); + } + } + return null; +} + +ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) { + if (typeof data != "string" || schema.pattern == undefined) { + return null; + } + var regexp = new RegExp(schema.pattern); + if (!regexp.test(data)) { + return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern"); + } + return null; +} +ValidatorContext.prototype.validateArray = function validateArray(data, schema) { + if (!Array.isArray(data)) { + return null; + } + return this.validateArrayLength(data, schema) + || this.validateArrayUniqueItems(data, schema) + || this.validateArrayItems(data, schema) + || null; +} + +ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) { + if (schema.minItems != undefined) { + if (data.length < schema.minItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxItems != undefined) { + if (data.length > schema.maxItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) { + if (schema.uniqueItems) { + for (var i = 0; i < data.length; i++) { + for (var j = i + 1; j < data.length; j++) { + if (recursiveCompare(data[i], data[j])) { + var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems"); + if (this.handleError(error)) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) { + if (schema.items == undefined) { + return null; + } + var error; + if (Array.isArray(schema.items)) { + for (var i = 0; i < data.length; i++) { + if (i < schema.items.length) { + if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) { + return error; + } + } else if (schema.additionalItems != undefined) { + if (typeof schema.additionalItems == "boolean") { + if (!schema.additionalItems) { + error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems"); + if (this.handleError(error)) { + return error; + } + } + } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) { + return error; + } + } + } + } else { + for (var i = 0; i < data.length; i++) { + if (error = this.validateAll(data[i], schema.items, [i], ["items"])) { + return error; + } + } + } + return null; +} +ValidatorContext.prototype.validateObject = function validateObject(data, schema) { + if (typeof data != "object" || data == null || Array.isArray(data)) { + return null; + } + return this.validateObjectMinMaxProperties(data, schema) + || this.validateObjectRequiredProperties(data, schema) + || this.validateObjectProperties(data, schema) + || this.validateObjectDependencies(data, schema) + || null; +} + +ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) { + var keys = Object.keys(data); + if (schema.minProperties != undefined) { + if (keys.length < schema.minProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxProperties != undefined) { + if (keys.length > schema.maxProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) { + if (schema.required != undefined) { + for (var i = 0; i < schema.required.length; i++) { + var key = schema.required[i]; + if (data[key] === undefined) { + var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required"); + if (this.handleError(error)) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) { + var error; + for (var key in data) { + var foundMatch = false; + if (schema.properties != undefined && schema.properties[key] != undefined) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) { + return error; + } + } + if (schema.patternProperties != undefined) { + for (var patternKey in schema.patternProperties) { + var regexp = new RegExp(patternKey); + if (regexp.test(key)) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) { + return error; + } + } + } + } + if (!foundMatch && schema.additionalProperties != undefined) { + if (typeof schema.additionalProperties == "boolean") { + if (!schema.additionalProperties) { + error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties"); + if (this.handleError(error)) { + return error; + } + } + } else { + if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) { + var error; + if (schema.dependencies != undefined) { + for (var depKey in schema.dependencies) { + if (data[depKey] !== undefined) { + var dep = schema.dependencies[depKey]; + if (typeof dep == "string") { + if (data[dep] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } else if (Array.isArray(dep)) { + for (var i = 0; i < dep.length; i++) { + var requiredKey = dep[i]; + if (data[requiredKey] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } + } else { + if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) { + var error; + return this.validateAllOf(data, schema) + || this.validateAnyOf(data, schema) + || this.validateOneOf(data, schema) + || this.validateNot(data, schema) + || null; +} + +ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) { + if (schema.allOf == undefined) { + return null; + } + var error; + for (var i = 0; i < schema.allOf.length; i++) { + var subSchema = schema.allOf[i]; + if (error = this.validateAll(data, subSchema, [], ["allOf", i])) { + return error; + } + } + return null; +} + +ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) { + if (schema.anyOf == undefined) { + return null; + } + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.anyOf.length; i++) { + var subSchema = schema.anyOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["anyOf", i]); + + if (error == null && errorCount == this.errors.length) { + this.errors = this.errors.slice(0, startErrorCount); + return null; + } + if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf")); + } + } + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors); +} + +ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) { + if (schema.oneOf == undefined) { + return null; + } + var validIndex = null; + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.oneOf.length; i++) { + var subSchema = schema.oneOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["oneOf", i]); + + if (error == null && errorCount == this.errors.length) { + if (validIndex == null) { + validIndex = i; + } else { + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf"); + } + } else if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf")); + } + } + if (validIndex == null) { + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors); + } else { + this.errors = this.errors.slice(0, startErrorCount); + } + return null; +} + +ValidatorContext.prototype.validateNot = function validateNot(data, schema) { + if (schema.not == undefined) { + return null; + } + var oldErrorCount = this.errors.length; + var error = this.validateAll(data, schema.not); + var notErrors = this.errors.slice(oldErrorCount); + this.errors = this.errors.slice(0, oldErrorCount); + if (error == null && notErrors.length == 0) { + return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not") + } + return null; +} + +// parseURI() and resolveUrl() are from https://gist.github.com/1088850 +// - released as public domain by author ("Yaffle") - see comments on gist + +function parseURI(url) { + var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/); + // authority = '//' + user + ':' + pass '@' + hostname + ':' port + return (m ? { + href : m[0] || '', + protocol : m[1] || '', + authority: m[2] || '', + host : m[3] || '', + hostname : m[4] || '', + port : m[5] || '', + pathname : m[6] || '', + search : m[7] || '', + hash : m[8] || '' + } : null); +} + +function resolveUrl(base, href) {// RFC 3986 + + function removeDotSegments(input) { + var output = []; + input.replace(/^(\.\.?(\/|$))+/, '') + .replace(/\/(\.(\/|$))+/g, '/') + .replace(/\/\.\.$/, '/../') + .replace(/\/?[^\/]*/g, function (p) { + if (p === '/..') { + output.pop(); + } else { + output.push(p); + } + }); + return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : ''); + } + + href = parseURI(href || ''); + base = parseURI(base || ''); + + return !href || !base ? null : (href.protocol || base.protocol) + + (href.protocol || href.authority ? href.authority : base.authority) + + removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) + + (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) + + href.hash; +} + +function normSchema(schema, baseUri) { + if (baseUri == undefined) { + baseUri = schema.id; + } else if (typeof schema.id == "string") { + baseUri = resolveUrl(baseUri, schema.id); + schema.id = baseUri; + } + if (typeof schema == "object") { + if (Array.isArray(schema)) { + for (var i = 0; i < schema.length; i++) { + normSchema(schema[i], baseUri); + } + } else if (typeof schema['$ref'] == "string") { + schema['$ref'] = resolveUrl(baseUri, schema['$ref']); + } else { + for (var key in schema) { + if (key != "enum") { + normSchema(schema[key], baseUri); + } + } + } + } +} + +var ErrorCodes = { + INVALID_TYPE: 0, + ENUM_MISMATCH: 1, + ANY_OF_MISSING: 10, + ONE_OF_MISSING: 11, + ONE_OF_MULTIPLE: 12, + NOT_PASSED: 13, + // Numeric errors + NUMBER_MULTIPLE_OF: 100, + NUMBER_MINIMUM: 101, + NUMBER_MINIMUM_EXCLUSIVE: 102, + NUMBER_MAXIMUM: 103, + NUMBER_MAXIMUM_EXCLUSIVE: 104, + // String errors + STRING_LENGTH_SHORT: 200, + STRING_LENGTH_LONG: 201, + STRING_PATTERN: 202, + // Object errors + OBJECT_PROPERTIES_MINIMUM: 300, + OBJECT_PROPERTIES_MAXIMUM: 301, + OBJECT_REQUIRED: 302, + OBJECT_ADDITIONAL_PROPERTIES: 303, + OBJECT_DEPENDENCY_KEY: 304, + // Array errors + ARRAY_LENGTH_SHORT: 400, + ARRAY_LENGTH_LONG: 401, + ARRAY_UNIQUE: 402, + ARRAY_ADDITIONAL_ITEMS: 403 +}; + +function ValidationError(code, message, dataPath, schemaPath, subErrors) { + if (code == undefined) { + throw new Error ("No code supplied for error: "+ message); + } + this.code = code; + this.message = message; + this.dataPath = dataPath ? dataPath : ""; + this.schemaPath = schemaPath ? schemaPath : ""; + this.subErrors = subErrors ? subErrors : null; +} +ValidationError.prototype = { + prefixWith: function (dataPrefix, schemaPrefix) { + if (dataPrefix != null) { + dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1"); + this.dataPath = "/" + dataPrefix + this.dataPath; + } + if (schemaPrefix != null) { + schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1"); + this.schemaPath = "/" + schemaPrefix + this.schemaPath; + } + if (this.subErrors != null) { + for (var i = 0; i < this.subErrors.length; i++) { + this.subErrors[i].prefixWith(dataPrefix, schemaPrefix); + } + } + return this; + } +}; + +function searchForTrustedSchemas(map, schema, url) { + if (typeof schema.id == "string") { + if (schema.id.substring(0, url.length) == url) { + var remainder = schema.id.substring(url.length); + if ((url.length > 0 && url.charAt(url.length - 1) == "/") + || remainder.charAt(0) == "#" + || remainder.charAt(0) == "?") { + if (map[schema.id] == undefined) { + map[schema.id] = schema; + } + } + } + } + if (typeof schema == "object") { + for (var key in schema) { + if (key != "enum" && typeof schema[key] == "object") { + searchForTrustedSchemas(map, schema[key], url); + } + } + } + return map; +} + +var globalContext = new ValidatorContext(); + +var publicApi = { + validate: function (data, schema) { + var context = new ValidatorContext(globalContext); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + var added = context.addSchema("", schema); + var error = context.validateAll(data, schema); + this.error = error; + this.missing = context.missing; + this.valid = (error == null); + return this.valid; + }, + validateResult: function () { + var result = {}; + this.validate.apply(result, arguments); + return result; + }, + validateMultiple: function (data, schema) { + var context = new ValidatorContext(globalContext, true); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + context.addSchema("", schema); + context.validateAll(data, schema); + var result = {}; + result.errors = context.errors; + result.missing = context.missing; + result.valid = (result.errors.length == 0); + return result; + }, + addSchema: function (url, schema) { + return globalContext.addSchema(url, schema); + }, + getSchema: function (url) { + return globalContext.getSchema(url); + }, + missing: [], + error: null, + normSchema: normSchema, + resolveUrl: resolveUrl, + errorCodes: ErrorCodes +}; + +global.tv4 = publicApi; + +})(typeof(window) != 'undefined' ? window : global); + + + +/** FILE: lib/Math.uuid.js **/ +/*! + Math.uuid.js (v1.4) + http://www.broofa.com + mailto:robert@broofa.com + + Copyright (c) 2010 Robert Kieffer + Dual licensed under the MIT and GPL licenses. + + ******** + + Changes within remoteStorage.js: + 2012-10-31: + - added AMD wrapper + - moved extensions for Math object into exported object. +*/ + +/* + * Generate a random uuid. + * + * USAGE: Math.uuid(length, radix) + * length - the desired number of characters + * radix - the number of allowable values for each character. + * + * EXAMPLES: + * // No arguments - returns RFC4122, version 4 ID + * >>> Math.uuid() + * "92329D39-6F5C-4520-ABFC-AAB64544E172" + * + * // One argument - returns ID of the specified length + * >>> Math.uuid(15) // 15 character ID (default base=62) + * "VcydxgltxrVZSTV" + * + * // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62) + * >>> Math.uuid(8, 2) // 8 character ID (base=2) + * "01001010" + * >>> Math.uuid(8, 10) // 8 character ID (base=10) + * "47473046" + * >>> Math.uuid(8, 16) // 8 character ID (base=16) + * "098F4D35" + */ + // Private array of chars to use + var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); + +Math.uuid = function (len, radix) { + var chars = CHARS, uuid = [], i; + radix = radix || chars.length; + + if (len) { + // Compact form + for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix]; + } else { + // rfc4122, version 4 form + var r; + + // rfc4122 requires these characters + uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; + uuid[14] = '4'; + + // Fill in random data. At i==19 set the high bits of clock sequence as + // per rfc4122, sec. 4.1.5 + for (i = 0; i < 36; i++) { + if (!uuid[i]) { + r = 0 | Math.random()*16; + uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; + } + } + } + + return uuid.join(''); +}; + + +/** FILE: src/baseclient.js **/ +(function(global) { + + function deprecate(thing, replacement) { + console.log('WARNING: ' + thing + ' is deprecated. Use ' + + replacement + ' instead.'); + } + + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.BaseClient + * + * Provides a high-level interface to access data below a given root path. + * + * A BaseClient deals with three types of data: folders, objects and files. + * + * returns a list of all items within a folder, or undefined + * if a 404 is encountered. Items that end with a forward slash ("/") are + * child folders. + * + * / operate on JSON objects. Each object has a type. + * + * / operates on files. Each file has a MIME type. + * + * operates on either objects or files (but not folders, folders are + * created and removed implictly). + */ + RS.BaseClient = function(storage, base) { + if (base[base.length - 1] !== '/') { + throw "Not a folder: " + base; + } + + if (base === '/') { + // allow absolute and relative paths for the root scope. + this.makePath = function(path) { + return (path[0] === '/' ? '' : '/') + path; + }; + } + + /** + * Property: storage + * + * The instance this operates on. + */ + this.storage = storage; + + /** + * Property: base + * + * Base path this operates on. + * + * For the module's privateClient this would be //, for the + * corresponding publicClient /public//. + */ + this.base = base; + + var parts = this.base.split('/'); + if (parts.length > 2) { + this.moduleName = parts[1]; + } else { + this.moduleName = 'root'; + } + + /** + * Event: change + * + * Emitted when a node changes + * + * Arguments: + * event - Event object containing information about the changed node + * + * (start code) + * { + * path: path, // Path of the changed node + * origin: 'window', 'local', or 'remote' // emitted by user action within the app, local data store, or remote sync + * oldValue: oldBody, // Old body of the changed node (undefined if creation) + * newValue: newBody // New body of the changed node (undefined if deletion) + * } + * (end code) + **/ + + RS.eventHandling(this, 'change'); + this.on = this.on.bind(this); + storage.onChange(this.base, this._fireChange.bind(this)); + }; + + RS.BaseClient.prototype = { + + extend: function(object) { + for (var key in object) { + this[key] = object[key]; + } + return this; + }, + + /** + * Method: scope + * + * Returns a new operating on a subpath of the current path. + */ + scope: function(path) { + return new RS.BaseClient(this.storage, this.makePath(path)); + }, + + // folder operations + + /** + * Method: getListing + * + * Get a list of child nodes below a given path. + * + * The callback semantics of getListing are identical to those of getObject. + * + * Parameters: + * path - The path to query. It MUST end with a forward slash. + * maxAge - (optional) Maximum age of cached listing in + * milliseconds + * + * Returns: + * + * A promise for an object, representing child nodes. + * + * Keys ending in a forward slash represent *folder nodes*, while all + * other keys represent *data nodes*. + * + * For spec versions <= 01, the data node information will contain only + * the item's ETag. For later spec versions, it will also contain the + * content type and -length of the item. + * + * Example: + * (start code) + * client.getListing('').then(function(listing) { + * listing.forEach(function(item) { + * console.log(item); + * }); + * }); + * (end code) + */ + getListing: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getListing must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then( + function(status, body) { + return (status === 404) ? undefined : body; + } + ); + }, + + /** + * Method: getAll + * + * Get all objects directly below a given path. + * + * Parameters: + * path - path to the folder + * maxAge - (optional) Maximum age of cached objects in + * milliseconds + * + * Returns: + * a promise for an object in the form { path : object, ... } + * + * Example: + * (start code) + * client.getAll('').then(function(objects) { + * for (var key in objects) { + * console.log('- ' + key + ': ', objects[key]); + * } + * }); + * (end code) + */ + getAll: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getAll must be undefined or a number'); + } + + return this.storage.get(this.makePath(path), maxAge).then(function(status, body) { + if (status === 404) { return; } + if (typeof(body) === 'object') { + var promise = promising(); + var count = Object.keys(body).length, i = 0; + if (count === 0) { + // treat this like 404. it probably means a folder listing that + // has changes that haven't been pushed out yet. + return; + } + for (var key in body) { + this.storage.get(this.makePath(path + key), maxAge). + then(function(status, b) { + body[this.key] = b; + i++; + if (i === count) { promise.fulfill(body); } + }.bind({ key: key })); + } + return promise; + } + }.bind(this)); + }, + + // file operations + + /** + * Method: getFile + * + * Get the file at the given path. A file is raw data, as opposed to + * a JSON object (use for that). + * + * Except for the return value structure, getFile works exactly like + * getObject. + * + * Parameters: + * path - see getObject + * + * Returns: + * A promise for an object: + * + * mimeType - String representing the MIME Type of the document. + * data - Raw data of the document (either a string or an ArrayBuffer) + * + * Example: + * (start code) + * // Display an image: + * client.getFile('path/to/some/image').then(function(file) { + * var blob = new Blob([file.data], { type: file.mimeType }); + * var targetElement = document.findElementById('my-image-element'); + * targetElement.src = window.URL.createObjectURL(blob); + * }); + * (end code) + */ + getFile: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getFile must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getFile must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + return { + data: body, + mimeType: mimeType, + revision: revision // (this is new) + }; + }); + }, + + /** + * Method: storeFile + * + * Store raw data at a given path. + * + * Parameters: + * mimeType - MIME media type of the data being stored + * path - path relative to the module root. MAY NOT end in a forward slash. + * data - string, ArrayBuffer or ArrayBufferView of raw data to store + * + * The given mimeType will later be returned, when retrieving the data + * using . + * + * Example (UTF-8 data): + * (start code) + * client.storeFile('text/html', 'index.html', '

Hello World!

'); + * (end code) + * + * Example (Binary data): + * (start code) + * // MARKUP: + * + * // CODE: + * var input = document.getElementById('file-input'); + * var file = input.files[0]; + * var fileReader = new FileReader(); + * + * fileReader.onload = function() { + * client.storeFile(file.type, file.name, fileReader.result); + * }; + * + * fileReader.readAsArrayBuffer(file); + * (end code) + * + */ + storeFile: function(mimeType, path, body) { + if (typeof(mimeType) !== 'string') { + return promising().reject('Argument \'mimeType\' of baseClient.storeFile must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeFile must be a string'); + } + if (typeof(body) !== 'string' && typeof(body) !== 'object') { + return promising().reject('Argument \'body\' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView'); + } + + var self = this; + return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + self.makePath(path) + ") failed with status: " + status; + } + }); + }, + + // object operations + + /** + * Method: getObject + * + * Get a JSON object from given path. + * + * Parameters: + * path - relative path from the module root (without leading slash) + * + * Returns: + * A promise for the object. + * + * Example: + * (start code) + * client.getObject('/path/to/object'). + * then(function(object) { + * // object is either an object or null + * }); + * (end code) + */ + getObject: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getObject must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getObject must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + if (typeof(body) === 'object') { + return body; + } else if (typeof(body) !== 'undefined' && status === 200) { + throw "Not an object: " + this.makePath(path); + } + }); + }, + + /** + * Method: storeObject + * + * Store object at given path. Triggers synchronization. + * + * Parameters: + * + * type - unique type of this object within this module. See description below. + * path - path relative to the module root. + * object - an object to be saved to the given node. It must be serializable as JSON. + * + * Returns: + * A promise to store the object. The promise fails with a ValidationError, when validations fail. + * + * + * What about the type?: + * + * A great thing about having data on the web, is to be able to link to + * it and rearrange it to fit the current circumstances. To facilitate + * that, eventually you need to know how the data at hand is structured. + * For documents on the web, this is usually done via a MIME type. The + * MIME type of JSON objects however, is always application/json. + * To add that extra layer of "knowing what this object is", remoteStorage + * aims to use . + * A first step in that direction, is to add a *@context attribute* to all + * JSON data put into remoteStorage. + * Now that is what the *type* is for. + * + * Within remoteStorage.js, @context values are built using three components: + * http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness + * the module name - module names should be unique as well + * the type given here - naming this particular kind of object within this module + * + * In retrospect that means, that whenever you introduce a new "type" in calls to + * storeObject, you should make sure that once your code is in the wild, future + * versions of the code are compatible with the same JSON structure. + * + * How to define types?: + * + * See for examples. + */ + storeObject: function(typeAlias, path, object) { + if (typeof(typeAlias) !== 'string') { + return promising().reject('Argument \'typeAlias\' of baseClient.storeObject must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeObject must be a string'); + } + if (typeof(object) !== 'object') { + return promising().reject('Argument \'object\' of baseClient.storeObject must be an object'); + } + this._attachType(object, typeAlias); + try { + var validationResult = this.validate(object); + if (! validationResult.valid) { + return promising(function(p) { p.reject(validationResult); }); + } + } catch(exc) { + if (! (exc instanceof RS.BaseClient.Types.SchemaNotFound)) { + return promising().reject(exc); + } + } + return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status; + } + }.bind(this)); + }, + + // generic operations + + /** + * Method: remove + * + * Remove node at given path from storage. Triggers synchronization. + * + * Parameters: + * path - Path relative to the module root. + */ + remove: function(path) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.remove must be a string'); + } + return this.storage.delete(this.makePath(path)); + }, + + + cache: function(path, strategy) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.cache must be a string'; + } + if (strategy === false) { + deprecate('caching strategy ', '<"FLUSH">'); + strategy = 'FLUSH'; + } else if (strategy === undefined) { + strategy = 'ALL'; + } else if (typeof(strategy) !== 'string') { + deprecate('that caching strategy', '<"ALL">'); + strategy = 'ALL'; + } + if (strategy !== 'FLUSH' && + strategy !== 'SEEN' && + strategy !== 'ALL') { + throw 'Argument \'strategy\' of baseclient.cache must be one of ' + + '["FLUSH", "SEEN", "ALL"]'; + } + this.storage.caching.set(this.makePath(path), strategy); + return this; + }, + + flush: function(path) { + return this.storage.local.flush(path); + }, + + makePath: function(path) { + return this.base + (path || ''); + }, + + _fireChange: function(event) { + this._emit('change', event); + }, + + _cleanPath: RS.WireClient.cleanPath, + + /** + * Method: getItemURL + * + * Retrieve full URL of item + * + * Parameters: + * path - Path relative to the module root. + */ + getItemURL: function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.getItemURL must be a string'; + } + if (this.storage.connected) { + path = this._cleanPath( this.makePath(path) ); + return this.storage.remote.href + path; + } else { + return undefined; + } + }, + + uuid: function() { + return Math.uuid(); + } + + }; + + /** + * Method: RS#scope + * + * Returns a new scoped to the given path. + * + * Parameters: + * path - Root path of new BaseClient. + * + * + * Example: + * (start code) + * + * var foo = remoteStorage.scope('/foo/'); + * + * // PUTs data "baz" to path /foo/bar + * foo.storeFile('text/plain', 'bar', 'baz'); + * + * var something = foo.scope('something/'); + * + * // GETs listing from path /foo/something/bla/ + * something.getListing('bla/'); + * + * (end code) + * + */ + RS.BaseClient._rs_init = function() { + RS.prototype.scope = function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.scope must be a string'; + } + + return new RS.BaseClient(this, path); + }; + }; + + /* e.g.: + remoteStorage.defineModule('locations', function(priv, pub) { + return { + exports: { + features: priv.scope('features/').defaultType('feature'), + collections: priv.scope('collections/').defaultType('feature-collection'); + } + }; + }); + */ + + maxAgeInvalid = function(maxAge) { + return typeof(maxAge) !== 'undefined' && typeof(maxAge) !== 'number'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/baseclient/types.js **/ +(function(global) { + + RemoteStorage.BaseClient.Types = { + // -> + uris: {}, + // -> + schemas: {}, + // -> + aliases: {}, + + declare: function(moduleName, alias, uri, schema) { + var fullAlias = moduleName + '/' + alias; + + if (schema.extends) { + var extendedAlias; + var parts = schema.extends.split('/'); + if (parts.length === 1) { + extendedAlias = moduleName + '/' + parts.shift(); + } else { + extendedAlias = parts.join('/'); + } + var extendedUri = this.uris[extendedAlias]; + if (! extendedUri) { + throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'"; + } + schema.extends = this.schemas[extendedUri]; + } + + this.uris[fullAlias] = uri; + this.aliases[uri] = fullAlias; + this.schemas[uri] = schema; + }, + + resolveAlias: function(alias) { + return this.uris[alias]; + }, + + getSchema: function(uri) { + return this.schemas[uri]; + }, + + inScope: function(moduleName) { + var ml = moduleName.length; + var schemas = {}; + for (var alias in this.uris) { + if (alias.substr(0, ml + 1) === moduleName + '/') { + var uri = this.uris[alias]; + schemas[uri] = this.schemas[uri]; + } + } + return schemas; + } + }; + + var SchemaNotFound = function(uri) { + var error = new Error("Schema not found: " + uri); + error.name = "SchemaNotFound"; + return error; + }; + + SchemaNotFound.prototype = Error.prototype; + + RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound; + /** + * Class: RemoteStorage.BaseClient + **/ + RemoteStorage.BaseClient.prototype.extend({ + /** + * Method: validate(object) + * + * validates an Object against the associated schema + * the context has to have a @context property + * + * Returns: + * A validate object giving you information about errors + **/ + validate: function(object) { + var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']); + if (schema) { + return tv4.validateResult(object, schema); + } else { + throw new SchemaNotFound(object['@context']); + } + }, + + // client.declareType(alias, schema); + // /* OR */ + // client.declareType(alias, uri, schema); + declareType: function(alias, uri, schema) { + if (! schema) { + schema = uri; + uri = this._defaultTypeURI(alias); + } + RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema); + }, + + _defaultTypeURI: function(alias) { + return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias; + }, + + _attachType: function(object, alias) { + object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias); + } + }); + + Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', { + configurable: true, + get: function() { + return RemoteStorage.BaseClient.Types.inScope(this.moduleName); + } + }); + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/modules.js **/ +(function() { + + RemoteStorage.MODULES = {}; + + /* + * Method: RemoteStorage.defineModule + * + * Method for defining a new remoteStorage data module + * + * Parameters: + * moduleName - Name of the module + * builder - Builder function defining the module + * + * The module builder function should return an object containing another + * object called exports, which will be exported to any + * instance under the module's name. So when defining a locations module, + * like in the example below, it would be accessible via + * `remoteStorage.locations`, which would in turn have a `features` and a + * `collections` property. + * + * The function receives a private and a public client, which are both + * instances of . In the following example, the + * scope of privateClient is `/locations` and the scope of publicClient is + * `/public/locations`. + * + * Example: + * (start code) + * RemoteStorage.defineModule('locations', function(privateClient, publicClient) { + * return { + * exports: { + * features: privateClient.scope('features/').defaultType('feature'), + * collections: privateClient.scope('collections/').defaultType('feature-collection') + * } + * }; + * }); + * (end code) + */ + + RemoteStorage.defineModule = function(moduleName, builder) { + RemoteStorage.MODULES[moduleName] = builder; + + Object.defineProperty(RemoteStorage.prototype, moduleName, { + configurable: true, + get: function() { + var instance = this._loadModule(moduleName); + Object.defineProperty(this, moduleName, { + value: instance + }); + return instance; + } + }); + + if (moduleName.indexOf('-') !== -1) { + var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) { + return s[1].toUpperCase(); + }); + Object.defineProperty(RemoteStorage.prototype, camelizedName, { + get: function() { + return this[moduleName]; + } + }); + } + }; + + RemoteStorage.prototype._loadModule = function(moduleName) { + var builder = RemoteStorage.MODULES[moduleName]; + if (builder) { + var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'), + new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/')); + return module.exports; + } else { + throw "Unknown module: " + moduleName; + } + }; + + RemoteStorage.prototype.defineModule = function(moduleName) { + console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!"); + RemoteStorage.defineModule.apply(RemoteStorage, arguments); + }; + +})(); + + +/** FILE: src/debug/inspect.js **/ +(function() { + function loadTable(table, storage, paths) { + table.setAttribute('border', '1'); + table.style.margin = '8px'; + table.innerHTML = ''; + var thead = document.createElement('thead'); + table.appendChild(thead); + var titleRow = document.createElement('tr'); + thead.appendChild(titleRow); + ['Path', 'Content-Type', 'Revision'].forEach(function(label) { + var th = document.createElement('th'); + th.textContent = label; + thead.appendChild(th); + }); + + var tbody = document.createElement('tbody'); + table.appendChild(tbody); + + function renderRow(tr, path, contentType, revision) { + [path, contentType, revision].forEach(function(value) { + var td = document.createElement('td'); + td.textContent = value || ''; + tr.appendChild(td); + }); + } + + function loadRow(path) { + if (storage.connected === false) { return; } + function processRow(status, body, contentType, revision) { + if (status === 200) { + var tr = document.createElement('tr'); + tbody.appendChild(tr); + renderRow(tr, path, contentType, revision); + if (path[path.length - 1] === '/') { + for (var key in body) { + loadRow(path + key); + } + } + } + } + storage.get(path).then(processRow); + } + + paths.forEach(loadRow); + } + + function renderWrapper(title, table, storage, paths) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = title; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + updateButton.onclick = function() { loadTable(table, storage, paths); }; + wrapper.appendChild(updateButton); + if (storage.reset) { + var resetButton = document.createElement('button'); + resetButton.textContent = "Reset"; + resetButton.onclick = function() { + storage.reset(function(newStorage) { + storage = newStorage; + loadTable(table, storage, paths); + }); + }; + wrapper.appendChild(resetButton); + } + wrapper.appendChild(table); + loadTable(table, storage, paths); + return wrapper; + } + + function renderLocalChanges(local) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = "Outgoing changes"; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + wrapper.appendChild(updateButton); + var list = document.createElement('ul'); + list.style.fontFamily = 'courier'; + wrapper.appendChild(list); + + function updateList() { + local.changesBelow('/').then(function(changes) { + list.innerHTML = ''; + changes.forEach(function(change) { + var el = document.createElement('li'); + el.textContent = JSON.stringify(change); + list.appendChild(el); + }); + }); + } + + updateButton.onclick = updateList; + updateList(); + return wrapper; + } + + RemoteStorage.prototype.inspect = function() { + + var widget = document.createElement('div'); + widget.id = 'remotestorage-inspect'; + widget.style.position = 'absolute'; + widget.style.top = 0; + widget.style.left = 0; + widget.style.background = 'black'; + widget.style.color = 'white'; + widget.style.border = 'groove 5px #ccc'; + + var controls = document.createElement('div'); + controls.style.position = 'absolute'; + controls.style.top = 0; + controls.style.left = 0; + + var heading = document.createElement('strong'); + heading.textContent = " remotestorage.js inspector "; + + controls.appendChild(heading); + + var syncButton; + + if (this.local) { + syncButton = document.createElement('button'); + syncButton.textContent = "Synchronize"; + controls.appendChild(syncButton); + } + + var closeButton = document.createElement('button'); + closeButton.textContent = "Close"; + closeButton.onclick = function() { + document.body.removeChild(widget); + }; + controls.appendChild(closeButton); + + widget.appendChild(controls); + + var remoteTable = document.createElement('table'); + var localTable = document.createElement('table'); + widget.appendChild(renderWrapper("Remote", remoteTable, this.remote, this.caching.rootPaths)); + if (this.local) { + widget.appendChild(renderWrapper("Local", localTable, this.local, ['/'])); + widget.appendChild(renderLocalChanges(this.local)); + + syncButton.onclick = function() { + this.log('sync clicked'); + this.sync().then(function() { + this.log('SYNC FINISHED'); + loadTable(localTable, this.local, ['/']); + }.bind(this), function(err) { + console.error("SYNC FAILED", err, err.stack); + }); + }.bind(this); + } + + document.body.appendChild(widget); + }; +})(); + + +/** FILE: src/legacy.js **/ +(function() { + var util = { + getEventEmitter: function() { + var object = {}; + var args = Array.prototype.slice.call(arguments); + args.unshift(object); + RemoteStorage.eventHandling.apply(RemoteStorage, args); + object.emit = object._emit; + return object; + }, + + extend: function(target) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach(function(source) { + for (var key in source) { + target[key] = source[key]; + } + }); + return target; + }, + + asyncEach: function(array, callback) { + return this.asyncMap(array, callback). + then(function() { return array; }); + }, + + asyncMap: function(array, callback) { + var promise = promising(); + var n = array.length, i = 0; + var results = [], errors = []; + function oneDone() { + i++; + if (i === n) { + promise.fulfill(results, errors); + } + } + + array.forEach(function(item, index) { + var result; + try { + result = callback(item); + } catch(exc) { + oneDone(); + errors[index] = exc; + } + if (typeof(result) === 'object' && typeof(result.then) === 'function') { + result.then(function(res) { results[index] = res; oneDone(); }, + function(error) { errors[index] = res; oneDone(); }); + } else { + oneDone(); + results[index] = result; + } + }); + + return promise; + }, + + containingFolder: function(path) { + var folder = path.replace(/[^\/]+\/?$/, ''); + return folder === path ? null : folder; + }, + + isFolder: function(path) { + return path.substr(-1) === '/'; + }, + + baseName: function(path) { + var parts = path.split('/'); + if (util.isFolder(path)) { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + }, + + bindAll: function(object) { + for (var key in this) { + if (typeof(object[key]) === 'function') { + object[key] = object[key].bind(object); + } + } + } + }; + + Object.defineProperty(RemoteStorage.prototype, 'util', { + get: function() { + console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release."); + return util; + } + }); + +})(); + + +/** FILE: src/googledrive.js **/ +(function(global) { + /** + * Class: RemoteStorage.GoogleDrive + * + * WORK IN PROGRESS, NOT RECOMMENDED FOR PRODUCTION USE + **/ + + var RS = RemoteStorage; + + var BASE_URL = 'https://www.googleapis.com'; + var AUTH_URL = 'https://accounts.google.com/o/oauth2/auth'; + var AUTH_SCOPE = 'https://www.googleapis.com/auth/drive'; + + var GD_DIR_MIME_TYPE = 'application/vnd.google-apps.folder'; + var RS_DIR_MIME_TYPE = 'application/json; charset=UTF-8'; + + function buildQueryString(params) { + return Object.keys(params).map(function(key) { + return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]); + }).join('&'); + } + + function fileNameFromMeta(meta) { + return encodeURIComponent(meta.title) + (meta.mimeType === GD_DIR_MIME_TYPE ? '/' : ''); + } + + function metaTitleFromFileName(filename) { + if (filename.substr(-1) === '/') { + filename = filename.substr(0, filename.length - 1); + } + return decodeURIComponent(filename); + } + + function parentPath(path) { + return path.replace(/[^\/]+\/?$/, ''); + } + + function baseName(path) { + var parts = path.split('/'); + if (path.substr(-1) === '/') { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + } + + var Cache = function(maxAge) { + this.maxAge = maxAge; + this._items = {}; + }; + + Cache.prototype = { + get: function(key) { + var item = this._items[key]; + var now = new Date().getTime(); + return (item && item.t >= (now - this.maxAge)) ? item.v : undefined; + }, + + set: function(key, value) { + this._items[key] = { + v: value, + t: new Date().getTime() + }; + } + }; + + RS.GoogleDrive = function(remoteStorage, clientId) { + + RS.eventHandling(this, 'connected'); + + this.rs = remoteStorage; + this.clientId = clientId; + + this._fileIdCache = new Cache(60 * 5); // ids expire after 5 minutes (is this a good idea?) + + setTimeout(function() { + this.configure(undefined, undefined, undefined, localStorage['remotestorage:googledrive:token']); + }.bind(this), 0); + }; + + RS.GoogleDrive.prototype = { + + configure: function(_x, _y, _z, token) { // parameter list compatible with WireClient + if (token) { + localStorage['remotestorage:googledrive:token'] = token; + this.token = token; + this.connected = true; + this._emit('connected'); + } else { + this.connected = false; + delete this.token; + // not reseting backend whenever googledrive gets initialized without an token +// this.rs.setBackend(undefined); + delete localStorage['remotestorage:googledrive:token']; + } + }, + + connect: function() { + this.rs.setBackend('googledrive'); + RS.Authorize(AUTH_URL, AUTH_SCOPE, String(RS.Authorize.getLocation()), this.clientId); + }, + + get: function(path, options) { + if (path.substr(-1) === '/') { + return this._getFolder(path, options); + } else { + return this._getFile(path, options); + } + }, + + put: function(path, body, contentType, options) { + var promise = promising(); + function putDone(error, response) { + if (error) { + promise.reject(error); + } else if (response.status >= 200 && response.status < 300) { + var meta = JSON.parse(response.responseText); + promise.fulfill(200, undefined, meta.mimeType, meta.etag); + } else { + promise.reject("PUT failed with status " + response.status + " (" + response.responseText + ")"); + } + } + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + return; + } else if (id) { + this._updateFile(id, path, body, contentType, options, putDone); + } else { + this._createFile(path, body, contentType, options, putDone); + } + }); + return promise; + }, + + 'delete': function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else if (id) { + this._request('DELETE', BASE_URL + '/drive/v2/files/' + id, {}, function(deleteError, response) { + if (deleteError) { + promise.reject(deleteError); + } else if (response.status === 200 || response.status === 204) { + promise.fulfill(200); + } else { + promise.reject("Delete failed: " + response.status + " (" + response.responseText + ")"); + } + }); + } else { + // file doesn't exist. ignore. + promise.fulfill(200); + } + }); + return promise; + }, + + _updateFile: function(id, path, body, contentType, options, callback) { + callback = callback.bind(this); + var metadata = { + mimeType: contentType + }; + this._request('PUT', BASE_URL + '/upload/drive/v2/files/' + id + '?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('PUT', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }, + + _createFile: function(path, body, contentType, options, callback) { + callback = callback.bind(this); + this._getParentId(path, function(parentIdError, parentId) { + if (parentIdError) { + callback(parentIdError); + return; + } + var fileName = baseName(path); + var metadata = { + title: metaTitleFromFileName(fileName), + mimeType: contentType, + parents: [{ + kind: "drive#fileLink", + id: parentId + }] + }; + this._request('POST', BASE_URL + '/upload/drive/v2/files?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('POST', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }); + }, + + _getFile: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else { + this._getMeta(id, function(metaError, meta) { + var etagWithoutQuotes; + if (typeof(meta) === 'object' && typeof(meta.etag) === 'string') { + etagWithoutQuotes = meta.etag.substring(1, meta.etag.length-1); + } + if (metaError) { + promise.reject(metaError); + } else if (meta.downloadUrl) { + var options = {}; + if (meta.mimeType.match(/charset=binary/)) { + options.responseType = 'blob'; + } + this._request('GET', meta.downloadUrl, options, function(downloadError, response) { + if (downloadError) { + promise.reject(downloadError); + } else { + var body = response.response; + if (meta.mimeType.match(/^application\/json/)) { + try { + body = JSON.parse(body); + } catch(e) {} + } + promise.fulfill(200, body, meta.mimeType, etagWithoutQuotes); + } + }); + } else { + // empty file + promise.fulfill(200, '', meta.mimeType, etagWithoutQuotes); + } + }); + } + }); + return promise; + }, + + _getFolder: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + var query, fields, data, i, etagWithoutQuotes, itemsMap; + if (idError) { + promise.reject(idError); + } else if (! id) { + promise.fulfill(404); + } else { + query = '\'' + id + '\' in parents'; + fields = 'items(downloadUrl,etag,fileSize,id,mimeType,title)'; + this._request('GET', BASE_URL + '/drive/v2/files?' + + 'q=' + encodeURIComponent(query) + + '&fields=' + encodeURIComponent(fields) + + '&maxResults=1000', + {}, function(childrenError, response) { + if (childrenError) { + promise.reject(childrenError); + } else { + if (response.status === 200) { + try { + data = JSON.parse(response.responseText); + } catch(e) { + promise.reject('non-JSON response from GoogleDrive'); + return; + } + itemsMap = {}; + for(i=0; i= 0 || true) { + try { + body = JSON.parse(body); + mime = 'application/json; charset=UTF-8'; + } catch(e) { + RS.log("Failed parsing Json, assume it is something else then", mime, path); + } + } + promise.fulfill(status, body, mime, rev); + } + + } else { + promise.fulfill(status); + } + } + }); + return promise; + }, + /** + * Method : put(path, body, contentType, options) + * put compatible with wireclient + * also uses _revCache to check for version conflicts + * also shares via share(path) + **/ + put: function(path, body, contentType, options){ + RemoteStorage.log('dropbox.put', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = this._sharePromise(path); + + var revCache = this._revCache; + + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options && options.ifMatch && savedRev && (savedRev !== options.ifMatch) ) { + promise.fulfill(412); + return promise; + } + if (! contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || RS.WireClient.isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var url = 'https://api-content.dropbox.com/1/files_put/auto' + path + '?'; + if (options && options.ifMatch) { + url += "parent_rev="+encodeURIComponent(options.ifMatch); + } + if (body.length>150*1024*1024){ //FIXME actual content-length + //https://www.dropbox.com/developers/core/docs#chunked-upload + RemoteStorage.log('files larger than 150MB not supported yet'); + } else { + this._request('PUT', url, {body:body, headers:{'Content-Type':contentType}}, function(err, resp) { + if (err) { + promise.reject(err); + } else { + var response = JSON.parse(resp.responseText); + // if dropbox reports an file conflict they just change the name of the file + // TODO find out which stays the origianl and how to deal with this + if (response.path !== path){ + promise.fulfill(412); + this.rs.log('Dropbox created conflicting File ', response.path); + } + else { + revCache.set(path, response.rev); + promise.fulfill(resp.status); + } + } + }); + } + return promise; + }, + + /** + * Method : delete(path, options) + * similar to get and set + **/ + 'delete': function(path, options){ + RemoteStorage.log('dropbox.delete ', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = promising(); + var revCache = this._revCache; + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options.ifMatch && savedRev && (options.ifMatch !== savedRev)) { + promise.fulfill(412); + return promise; + } + + var url = 'https://api.dropbox.com/1/fileops/delete?root=auto&path='+encodeURIComponent(path); + this._request('POST', url, {}, function(err, resp){ + if (err) { + promise.reject(error); + } else { + promise.fulfill(resp.status); + revCache.delete(path); + } + }); + + return promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + delete this._itemRefs[path]; + var p = promising(); + return p.fulfill.apply(p, args); + }.bind(this)); + }, + + /** + * Method : _sharePromise(path) + * returns a promise which's then block doesn't touch the arguments given + * and calls share for the path + * + * also checks for necessity of shareing this url(already in the itemRefs or not '/public/') + **/ + _sharePromise: function(path){ + var promise = promising(); + var self = this; + if (path.match(/^\/public\/.*[^\/]$/) && typeof this._itemRefs[path] === 'undefined') { + RemoteStorage.log('shareing this one ', path); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + var p = promising(); + RemoteStorage.log('calling share now'); + self.share(path).then(function() { + RemoteStorage.log('shareing fullfilled promise',arguments); + p.fulfill.apply(p,args); + }, function(err) { + RemoteStorage.log("shareing failed" , err); + p.fulfill.apply(p,args); + }); + return p; + }); + } + return promise; + }, + + /** + * Method : share(path) + * get sher_url s from dropbox and pushes those into this._hrefCache + * returns promise + */ + share: function(path){ + var url = "https://api.dropbox.com/1/media/auto"+path; + var promise = promising(); + var itemRefs = this._itemRefs; + + // requesting shareing url + this._request('POST', url, {}, function(err, resp){ + if (err) { + RemoteStorage.log(err); + err.message = 'Shareing Dropbox Thingie("'+path+'") failed' + err.message; + promise.reject(err); + } else { + try{ + var response = JSON.parse(resp.responseText); + var url = response.url; + itemRefs[path] = url; + RemoteStorage.log("SHAREING URL :::: ",url,' for ',path); + if (hasLocalStorage) { + localStorage[SETTINGS_KEY+":shares"] = JSON.stringify(this._itemRefs); + } + promise.fulfill(url); + } catch(err) { + err.message += "share error"; + promise.reject(err); + } + } + }); + return promise; + }, + + /** + * Method : info() + * fetching user info from Dropbox returns promise + **/ + info: function() { + var url = 'https://api.dropbox.com/1/account/info'; + var promise = promising(); + // requesting user info(mainly for userAdress) + this._request('GET', url, {}, function(err, resp){ + if (err) { + promise.reject(err); + } else { + try { + var info = JSON.parse(resp.responseText); + promise.fulfill(info); + } catch(e) { + promise.reject(err); + } + } + }); + return promise; + }, + + _request: function(method, url, options, callback) { + callback = callback.bind(this); + if (! options.headers) { options.headers = {}; } + options.headers['Authorization'] = 'Bearer ' + this.token; + RS.WireClient.request.call(this, method, url, options, function(err, xhr) { + //503 means retry this later + if (xhr && xhr.status === 503) { + global.setTimeout(this._request(method, url, options, callback), 3210); + } else { + callback(err, xhr); + } + }); + }, + + /** + * method: fetchDelta + * + * this method fetches the deltas from the dropbox api, used to sync the storage + * here we retrive changes and put them into the _revCache, those values will then be used + * to determin if something has changed. + **/ + fetchDelta: function() { + var args = Array.prototype.slice.call(arguments); + var promise = promising(); + var self = this; + this._request('POST', 'https://api.dropbox.com/1/delta', { + body: this._deltaCursor ? ('cursor=' + encodeURIComponent(this._deltaCursor)) : '', + headers: { + 'Content-Type': 'application/x-www-form-urlencoded' + } + }, function(error, response) { + if (error) { + this.rs.log('fetchDeltas',error); + this.rs._emit('error', new RemoteStorage.SyncError('fetchDeltas failed'+error)); + promise.reject(error); + } else { + // break if status != 200 + if (response.status !== 200 ) { + if (response.status === 400) { + this.rs._emit('error', new RemoteStorage.Unauthorized()); + promise.fulfill.apply(promise, args); + } else { + RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText); + promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText); + } + return promise; + } + + var delta; + try { + delta = JSON.parse(response.responseText); + } catch(error) { + RS.log('fetchDeltas can not parse response',error); + return promise.reject("can not parse response of fetchDelta : "+error.message); + } + // break if no entries found + if (!delta.entries) { + RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!", delta); + return promise.reject('dropbox.fetchDeltas failed, no entries found'); + } + + // Dropbox sends the complete state + if (delta.reset) { + this._revCache = new LowerCaseCache('rev'); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + self._revCache._activatePropagation(); + var p = promising(); + return p.fulfill.apply(p,args); + }); + } + + //saving the cursor for requesting further deltas in relation to the cursor position + if (delta.cursor) { + this._deltaCursor = delta.cursor; + } + + //updating revCache + RemoteStorage.log("Delta : ",delta.entries); + delta.entries.forEach(function(entry) { + var path = entry[0]; + var rev; + if (!entry[1]){ + rev = null; + } else { + if (entry[1].is_dir) { + return; + } + rev = entry[1].rev; + } + self._revCache.set(path, rev); + }); + promise.fulfill.apply(promise, args); + } + }); + return promise; + } + }; + + //hooking and unhooking the sync + + function hookSync(rs) { + if (rs._dropboxOrigSync) { return; } // already hooked + rs._dropboxOrigSync = rs.sync.bind(rs); + rs.sync = function() { + return this.dropbox.fetchDelta.apply(this.dropbox, arguments). + then(rs._dropboxOrigSync, function(err){ + rs._emit('error', new rs.SyncError(err)); + }); + }; + } + + function unHookSync(rs) { + if (! rs._dropboxOrigSync) { return; } // not hooked + rs.sync = rs._dropboxOrigSync; + delete rs._dropboxOrigSync; + } + + // hooking and unhooking getItemURL + + function hookGetItemURL(rs) { + if (rs._origBaseClientGetItemURL) { return; } + rs._origBaseClientGetItemURL = RS.BaseClient.prototype.getItemURL; + RS.BaseClient.prototype.getItemURL = function(path){ + var ret = rs.dropbox._itemRefs[path]; + return ret ? ret : ''; + }; + } + + function unHookGetItemURL(rs){ + if (! rs._origBaseClieNtGetItemURL) { return; } + RS.BaseClient.prototype.getItemURL = rs._origBaseClietGetItemURL; + delete rs._origBaseClietGetItemURL; + } + + function hookRemote(rs){ + if (rs._origRemote) { return; } + rs._origRemote = rs.remote; + rs.remote = rs.dropbox; + } + + function unHookRemote(rs){ + if (rs._origRemote) { + rs.remote = rs._origRemote; + delete rs._origRemote; + } + } + + function hookIt(rs){ + hookRemote(rs); + if (rs.sync) { + hookSync(rs); + } + hookGetItemURL(rs); + } + + function unHookIt(rs){ + unHookRemote(rs); + unHookSync(rs); + unHookGetItemURL(rs); + } + + RS.Dropbox._rs_init = function(rs) { + hasLocalStorage = rs.localStorageAvailable(); + if ( rs.apiKeys.dropbox ) { + rs.dropbox = new RS.Dropbox(rs); + } + if (rs.backend === 'dropbox') { + hookIt(rs); + } + }; + + RS.Dropbox._rs_supported = function() { + return true; + }; + + RS.Dropbox._rs_cleanup = function(rs) { + unHookIt(rs); + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + rs.removeEventListener('error', onErrorCb); + rs.setBackend(undefined); + }; +})(this); + +remoteStorage = new RemoteStorage(); \ No newline at end of file diff --git a/release/0.10.0-beta2/remotestorage-nocache.min.js b/release/0.10.0-beta2/remotestorage-nocache.min.js new file mode 100644 index 000000000..505ee271b --- /dev/null +++ b/release/0.10.0-beta2/remotestorage-nocache.min.js @@ -0,0 +1,5 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ +(function(global){function getPromise(builder){var promise;if(typeof builder==="function"){setTimeout(function(){try{builder(promise)}catch(e){promise.reject(e)}},0)}var consumers=[],success,result;function notifyConsumer(consumer){if(success){var nextValue;if(consumer.fulfilled){try{nextValue=[consumer.fulfilled.apply(null,result)]}catch(exc){consumer.promise.reject(exc);return}}else{nextValue=result}if(nextValue[0]&&typeof nextValue[0].then==="function"){nextValue[0].then(consumer.promise.fulfill,consumer.promise.reject)}else{consumer.promise.fulfill.apply(null,nextValue)}}else{if(consumer.rejected){var ret;try{ret=consumer.rejected.apply(null,result)}catch(exc){consumer.promise.reject(exc);return}if(ret&&typeof ret.then==="function"){ret.then(consumer.promise.fulfill,consumer.promise.reject)}else{consumer.promise.fulfill(ret)}}else{consumer.promise.reject.apply(null,result)}}}function resolve(succ,res){if(result){console.error("WARNING: Can't resolve promise, already resolved!");return}success=succ;result=Array.prototype.slice.call(res);setTimeout(function(){var cl=consumers.length;if(cl===0&&!success){console.error("Possibly uncaught error: ",result,result[0]&&result[0].stack)}for(var i=0;i=n){this._init();RemoteStorage.log("Done cleaning up, emitting disconnected and disconnect events");this._emit("disconnected")}}.bind(this);if(n>0){this._cleanups.forEach(function(cleanup){var cleanupResult=cleanup(this);if(typeof cleanup==="object"&&typeof cleanup.then==="function"){cleanupResult.then(oneDone)}else{oneDone()}}.bind(this))}else{oneDone()}},setBackend:function(what){this.backend=what;if(this.localStorageAvailable()){if(what){localStorage["remotestorage:backend"]=what}else{delete localStorage["remotestorage:backend"]}}},onChange:function(path,handler){if(!this._pathHandlers.change[path]){this._pathHandlers.change[path]=[]}this._pathHandlers.change[path].push(handler)},enableLog:function(){RemoteStorage._log=true},disableLog:function(){RemoteStorage._log=false},log:function(){RemoteStorage.log.apply(RemoteStorage,arguments)},setApiKeys:function(type,keys){if(keys){this.apiKeys[type]=keys}else{delete this.apiKeys[type]}if(this.localStorageAvailable()){localStorage["remotestorage:api-keys"]=JSON.stringify(this.apiKeys)}},_init:function(){var self=this,readyFired=false;function fireReady(){try{if(!readyFired){self._emit("ready");readyFired=true}}catch(e){console.error("'ready' failed: ",e,e.stack);self._emit("error",e)}}this._loadFeatures(function(features){this.log("[RemoteStorage] All features loaded");this.local=features.local&&new features.local;if(this.local&&this.remote){this._setGPD(SyncedGetPutDelete,this);this._bindChange(this.local)}else if(this.remote){this._setGPD(this.remote,this.remote)}if(this.remote){this.remote.on("connected",function(){fireReady();self._emit("connected")});this.remote.on("not-connected",function(){fireReady();self._emit("not-connected")});if(this.remote.connected){fireReady();self._emit("connected")}}this._collectCleanupFunctions();try{this._allLoaded=true;this._emit("features-loaded")}catch(exc){logError(exc);this._emit("error",exc)}this._processPending()}.bind(this))},_collectCleanupFunctions:function(){for(var i=0;i=0}function isErrorStatus(status){return[401,403,404,412].indexOf(status)>=0}var onErrorCb;RS.WireClient=function(rs){this.connected=false;RS.eventHandling(this,"change","connected","wire-busy","wire-done","not-connected");onErrorCb=function(error){if(error instanceof RemoteStorage.Unauthorized){this.configure(undefined,undefined,undefined,null)}}.bind(this);rs.on("error",onErrorCb);if(hasLocalStorage){var settings;try{settings=JSON.parse(localStorage[SETTINGS_KEY])}catch(e){}if(settings){setTimeout(function(){this.configure(settings.userAddress,settings.href,settings.storageApi,settings.token)}.bind(this),0)}}this._revisionCache={};if(this.connected){setTimeout(this._emit.bind(this),0,"connected")}};RS.WireClient.REQUEST_TIMEOUT=3e4;RS.WireClient.prototype={_request:function(method,uri,token,headers,body,getEtag,fakeRevision){if((method==="PUT"||method==="DELETE")&&uri[uri.length-1]==="/"){throw"Don't "+method+" on directories!"}var promise=promising();var revision;var reqType;var self=this;headers["Authorization"]="Bearer "+token;this._emit("wire-busy",{method:method,isFolder:isFolder(uri)});RS.WireClient.request(method,uri,{body:body,headers:headers},function(error,response){if(error){self._emit("wire-done",{method:method,isFolder:isFolder(uri),success:false});self.online=false;promise.reject(error)}else{self._emit("wire-done",{method:method,isFolder:isFolder(uri),success:true});self.online=true;if(isErrorStatus(response.status)){RemoteStorage.log("[WireClient] Error response status",response.status);if(getEtag){revision=stripQuotes(response.getResponseHeader("ETag"))}else{revision=undefined}promise.fulfill(response.status,undefined,undefined,revision)}else if(isSuccessStatus(response.status)||response.status===200&&method!=="GET"){revision=stripQuotes(response.getResponseHeader("ETag"));RemoteStorage.log("[WireClient] Successful request",revision);promise.fulfill(response.status,undefined,undefined,revision)}else{var mimeType=response.getResponseHeader("Content-Type");var body;if(getEtag){revision=stripQuotes(response.getResponseHeader("ETag"))}else{revision=response.status===200?fakeRevision:undefined}if(!mimeType||mimeType.match(/charset=binary/)){RS.WireClient.readBinaryData(response.response,mimeType,function(result){RemoteStorage.log("[WireClient] Successful request with unknown or binary mime-type",revision);promise.fulfill(response.status,result,mimeType,revision)})}else{if(mimeType&&mimeType.match(/^application\/json/)){body=JSON.parse(response.responseText)}else{body=response.responseText}RemoteStorage.log("[WireClient] Successful request",revision);promise.fulfill(response.status,body,mimeType,revision)}}}});return promise},configure:function(userAddress,href,storageApi,token){if(typeof userAddress!=="undefined"){this.userAddress=userAddress}if(typeof href!=="undefined"){this.href=href}if(typeof storageApi!=="undefined"){this.storageApi=storageApi}if(typeof token!=="undefined"){this.token=token}if(typeof this.storageApi!=="undefined"){this._storageApi=STORAGE_APIS[this.storageApi]||API_HEAD;this.supportsRevs=this._storageApi>=API_00}if(this.href&&this.token){this.connected=true;this.online=true;this._emit("connected")}else{this.connected=false}if(hasLocalStorage){localStorage[SETTINGS_KEY]=JSON.stringify({userAddress:this.userAddress,href:this.href,token:this.token,storageApi:this.storageApi})}RS.WireClient.configureHooks.forEach(function(hook){hook.call(this)}.bind(this))},stopWaitingForToken:function(){if(!this.connected){this._emit("not-connected")}},get:function(path,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}var headers={};if(this.supportsRevs){if(options.ifNoneMatch){headers["If-None-Match"]=addQuotes(options.ifNoneMatch)}}else if(options.ifNoneMatch){var oldRev=this._revisionCache[path]}var promise=this._request("GET",this.href+cleanPath(path),this.token,headers,undefined,this.supportsRevs,this._revisionCache[path]);if(isFolder(path)){return promise.then(function(status,body,contentType,revision){var itemsMap={};if(status===200&&typeof body==="object"){if(Object.keys(body).length===0){status=404}else if(isFolderDescription(body)){for(var item in body.items){this._revisionCache[path+item]=body.items[item].ETag}itemsMap=body.items}else{Object.keys(body).forEach(function(key){this._revisionCache[path+key]=body[key];itemsMap[key]={ETag:body[key]}}.bind(this))}return promising().fulfill(status,itemsMap,contentType,revision)}else{return promising().fulfill(status,body,contentType,revision)}}.bind(this))}else{return promise}},put:function(path,body,contentType,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}if(!contentType.match(/charset=/)){contentType+="; charset="+(body instanceof ArrayBuffer||isArrayBufferView(body)?"binary":"utf-8")}var headers={"Content-Type":contentType};if(this.supportsRevs){if(options.ifMatch){headers["If-Match"]=addQuotes(options.ifMatch)}if(options.ifNoneMatch){headers["If-None-Match"]=addQuotes(options.ifNoneMatch)}}return this._request("PUT",this.href+cleanPath(path),this.token,headers,body,this.supportsRevs)},"delete":function(path,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}var headers={};if(this.supportsRevs){if(options.ifMatch){headers["If-Match"]=addQuotes(options.ifMatch)}}return this._request("DELETE",this.href+cleanPath(path),this.token,headers,undefined,this.supportsRevs)}};RS.WireClient.cleanPath=cleanPath;RS.WireClient.isArrayBufferView=isArrayBufferView;RS.WireClient.readBinaryData=readBinaryData;RS.WireClient.request=function(method,url,options,callback){RemoteStorage.log("[WireClient]",method,url);callback=callback.bind(this);var timedOut=false;var timer=setTimeout(function(){timedOut=true;callback("timeout")},RS.WireClient.REQUEST_TIMEOUT);var xhr=new XMLHttpRequest;xhr.open(method,url,true);if(options.responseType){xhr.responseType=options.responseType}if(options.headers){for(var key in options.headers){xhr.setRequestHeader(key,options.headers[key])}}xhr.onload=function(){if(timedOut){return}clearTimeout(timer);callback(null,xhr)};xhr.onerror=function(error){if(timedOut){return}clearTimeout(timer);callback(error)};var body=options.body;if(typeof body==="object"){if(isArrayBufferView(body)){}else if(body instanceof ArrayBuffer){body=new Uint8Array(body)}else{body=JSON.stringify(body)}}xhr.send(body)};Object.defineProperty(RemoteStorage.WireClient.prototype,"storageType",{get:function(){if(this.storageApi){var spec=this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/);return spec?spec[1]:"2012.04"}}});RS.WireClient.configureHooks=[];RS.WireClient._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();remoteStorage.remote=new RS.WireClient(remoteStorage);this.online=true};RS.WireClient._rs_supported=function(){return!!global.XMLHttpRequest};RS.WireClient._rs_cleanup=function(remoteStorage){if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}remoteStorage.removeEventListener("error",onErrorCb)}})(typeof window!=="undefined"?window:global);(function(global){var haveXMLHttpRequest,hasLocalStorage;var SETTINGS_KEY="remotestorage:discover";var cachedInfo={};RemoteStorage.Discover=function(userAddress,callback){if(userAddress in cachedInfo){var info=cachedInfo[userAddress];callback(info.href,info.type,info.authURL);return}var hostname=userAddress.split("@")[1];var params="?resource="+encodeURIComponent("acct:"+userAddress);var urls=["https://"+hostname+"/.well-known/webfinger"+params,"https://"+hostname+"/.well-known/host-meta.json"+params,"http://"+hostname+"/.well-known/webfinger"+params,"http://"+hostname+"/.well-known/host-meta.json"+params];function tryOne(){var xhr=new XMLHttpRequest;var url=urls.shift();if(!url){return callback()}RemoteStorage.log("[Discover] Trying URL",url);xhr.open("GET",url,true);xhr.onabort=xhr.onerror=function(){console.error("webfinger error",arguments,"(",url,")");tryOne()};xhr.onload=function(){if(xhr.status!==200){return tryOne()}var profile;try{profile=JSON.parse(xhr.responseText)}catch(e){RemoteStorage.log("[Discover] Failed to parse profile ",xhr.responseText,e);tryOne();return}if(!profile.links){RemoteStorage.log("[Discover] Profile has no links section ",JSON.stringify(profile));tryOne();return}var link;profile.links.forEach(function(l){if(l.rel==="remotestorage"){link=l}else if(l.rel==="remoteStorage"&&!link){link=l}});RemoteStorage.log("[Discover] Got profile",profile,"and link",link);if(link){var authURL=link.properties["http://tools.ietf.org/html/rfc6749#section-4.2"]||link.properties["auth-endpoint"],storageType=link.properties["http://remotestorage.io/spec/version"]||link.type;cachedInfo[userAddress]={href:link.href,type:storageType,authURL:authURL};if(hasLocalStorage){localStorage[SETTINGS_KEY]=JSON.stringify({cache:cachedInfo})}callback(link.href,storageType,authURL)}else{tryOne()}};xhr.send()}tryOne()};RemoteStorage.Discover._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();if(hasLocalStorage){var settings;try{settings=JSON.parse(localStorage[SETTINGS_KEY])}catch(e){}if(settings){cachedInfo=settings.cache}}};RemoteStorage.Discover._rs_supported=function(){haveXMLHttpRequest=!!global.XMLHttpRequest;return haveXMLHttpRequest};RemoteStorage.Discover._rs_cleanup=function(){if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}}})(typeof window!=="undefined"?window:global);(function(global){function extractParams(){var location=RemoteStorage.Authorize.getLocation(),hashPos=location.href.indexOf("#"),hash;if(hashPos===-1){return}hash=location.href.substring(hashPos+1);if(hash.indexOf("=")===-1){return}return hash.split("&").reduce(function(m,kvs){var kv=kvs.split("=");m[decodeURIComponent(kv[0])]=decodeURIComponent(kv[1]);return m},{})}RemoteStorage.Authorize=function(authURL,scope,redirectUri,clientId){RemoteStorage.log("[Authorize] authURL = ",authURL);var url=authURL;url+=authURL.indexOf("?")>0?"&":"?";url+="redirect_uri="+encodeURIComponent(redirectUri.replace(/#.*$/,""));url+="&scope="+encodeURIComponent(scope);url+="&client_id="+encodeURIComponent(clientId);url+="&response_type=token";RemoteStorage.Authorize.setLocation(url)};RemoteStorage.prototype.authorize=function(authURL){this.access.setStorageType(this.remote.storageType);var scope=this.access.scopeParameter;var redirectUri=String(RemoteStorage.Authorize.getLocation());var clientId=redirectUri.match(/^(https?:\/\/[^\/]+)/)[0];RemoteStorage.Authorize(authURL,scope,redirectUri,clientId)};RemoteStorage.Authorize.getLocation=function(){return global.document.location};RemoteStorage.Authorize.setLocation=function(location){if(typeof location==="string"){global.document.location.href=location}else if(typeof location==="object"){global.document.location=location}else{throw"Invalid location "+location}};RemoteStorage.Authorize._rs_supported=function(remoteStorage){return typeof document!=="undefined"};var onFeaturesLoaded;RemoteStorage.Authorize._rs_init=function(remoteStorage){onFeaturesLoaded=function(){var authParamsUsed=false;if(params){if(params.error){throw"Authorization server errored: "+params.error}if(params.access_token){remoteStorage.remote.configure(undefined,undefined,undefined,params.access_token);authParamsUsed=true}if(params.remotestorage){remoteStorage.connect(params.remotestorage);authParamsUsed=true}}if(!authParamsUsed){remoteStorage.remote.stopWaitingForToken()}};var params=extractParams(),location;if(params){location=RemoteStorage.Authorize.getLocation();location.hash=""}remoteStorage.on("features-loaded",onFeaturesLoaded)};RemoteStorage.Authorize._rs_cleanup=function(remoteStorage){remoteStorage.removeEventListener("features-loaded",onFeaturesLoaded)}})(typeof window!=="undefined"?window:global);(function(global){var SETTINGS_KEY="remotestorage:access";RemoteStorage.Access=function(){this.reset()};RemoteStorage.Access.prototype={claim:function(scope,mode){if(typeof scope!=="string"||scope.indexOf("/")!==-1||scope.length===0){throw new Error("Scope should be a non-empty string without forward slashes")}if(!mode.match(/^rw?$/)){throw new Error("Mode should be either 'r' or 'rw'")}this._adjustRootPaths(scope);this.scopeModeMap[scope]=mode},get:function(scope){return this.scopeModeMap[scope]},remove:function(scope){var savedMap={};var name;for(name in this.scopeModeMap){savedMap[name]=this.scopeModeMap[name]}this.reset();delete savedMap[scope];for(name in savedMap){this.set(name,savedMap[name])}},checkPermission:function(scope,mode){var actualMode=this.get(scope);return actualMode&&(mode==="r"||actualMode==="rw")},checkPathPermission:function(path,mode){if(this.checkPermission("*",mode)){return true}return!!this.checkPermission(this._getModuleName(path),mode)},reset:function(){this.rootPaths=[];this.scopeModeMap={}},_getModuleName:function(path){if(path[0]!=="/"){throw new Error("Path should start with a slash")}var moduleMatch=path.replace(/^\/public/,"").match(/^\/([^\/]*)\//);return moduleMatch?moduleMatch[1]:"*"},_adjustRootPaths:function(newScope){if("*"in this.scopeModeMap||newScope==="*"){this.rootPaths=["/"]}else if(!(newScope in this.scopeModeMap)){this.rootPaths.push("/"+newScope+"/");this.rootPaths.push("/public/"+newScope+"/")}},_scopeNameForParameter:function(scope){if(scope.name==="*"&&this.storageType){if(this.storageType==="2012.04"){return""}else if(this.storageType.match(/remotestorage-0[01]/)){return"root"}}return scope.name},setStorageType:function(type){this.storageType=type}};Object.defineProperty(RemoteStorage.Access.prototype,"scopes",{get:function(){return Object.keys(this.scopeModeMap).map(function(key){return{name:key,mode:this.scopeModeMap[key]}}.bind(this))}});Object.defineProperty(RemoteStorage.Access.prototype,"scopeParameter",{get:function(){return this.scopes.map(function(scope){return this._scopeNameForParameter(scope)+":"+scope.mode}.bind(this)).join(" ")}});Object.defineProperty(RemoteStorage.prototype,"access",{get:function(){var access=new RemoteStorage.Access;Object.defineProperty(this,"access",{value:access});return access},configurable:true});RemoteStorage.Access._rs_init=function(){}})(typeof window!=="undefined"?window:global);(function(pMode){var mode=pMode,env={},isBackground=false;RemoteStorage.Env=function(){return env};RemoteStorage.Env.isBrowser=function(){return mode==="browser"};RemoteStorage.Env.isNode=function(){return mode==="node"};RemoteStorage.Env.goBackground=function(){isBackground=true;RemoteStorage.Env._emit("background")};RemoteStorage.Env.goForeground=function(){isBackground=false;RemoteStorage.Env._emit("foreground")};RemoteStorage.Env._rs_init=function(remoteStorage){RemoteStorage.eventHandling(RemoteStorage.Env,"background","foreground");if(mode==="browser"){if(typeof document.hidden!=="undefined"){env.hiddenProperty="hidden";env.visibilityChangeEvent="visibilitychange"}else if(typeof document.mozHidden!=="undefined"){env.hiddenProperty="mozHidden";env.visibilityChangeEvent="mozvisibilitychange"}else if(typeof document.msHidden!=="undefined"){env.hiddenProperty="msHidden";env.visibilityChangeEvent="msvisibilitychange"}else if(typeof document.webkitHidden!=="undefined"){env.hiddenProperty="webkitHidden";env.visibilityChangeEvent="webkitvisibilitychange"}document.addEventListener(env.visibilityChangeEvent,function(){if(document[env.hiddenProperty]){RemoteStorage.Env.goBackground()}else{RemoteStorage.Env.goForeground()}},false)}};RemoteStorage.Env._rs_cleanup=function(remoteStorage){}})(typeof window!=="undefined"?"browser":"node");(function(){"use strict";var dictionary={view_info:'This app allows you to use your own storage. Learn more!',view_connect:"Connect remote storage",view_connecting:"Connecting %s",view_offline:"Offline",view_error_occured:"Sorry! An error occured.",view_confirm_reset:"Are you sure you want to reset everything? This will clear your local data and reload the page.",view_get_me_out:"Get me out of here!",view_error_plz_report:'If this problem persists, please let us know!',view_unauthorized:"Unauthorized! Click here to reconnect."};RemoteStorage.I18n={translate:function(){var str=arguments[0],params=Array.prototype.splice.call(arguments,1);if(typeof dictionary[str]!=="string"){throw"Unknown translation string: "+str}else{str=dictionary[str]}return str.replace(/%s/g,function(){return params.shift()})},getDictionary:function(){return dictionary},setDictionary:function(newDictionary){dictionary=newDictionary}}})();RemoteStorage.Assets={connectIcon:"",disconnectIcon:"",dropbox:"",googledrive:"",remoteStorageIcon:"",remoteStorageIconError:"",remoteStorageIconOffline:"",syncIcon:"",widget:'
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ',widgetCss:'/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}'}; +(function(window){var hasLocalStorage;var LS_STATE_KEY="remotestorage:widget:state";var VALID_ENTRY_STATES={initial:true,connected:true,offline:true};function stateSetter(widget,state){RemoteStorage.log("[Widget] Producing stateSetter for",state);return function(){RemoteStorage.log("[Widget] Setting state",state,arguments);if(hasLocalStorage){localStorage[LS_STATE_KEY]=state}if(widget.view){if(widget.rs.remote){widget.view.setUserAddress(widget.rs.remote.userAddress)}widget.view.setState(state,arguments)}else{widget._rememberedState=state}}}function errorsHandler(widget){return function(error){if(error instanceof RemoteStorage.DiscoveryError){console.error("Discovery failed",error,'"'+error.message+'"');widget.view.setState("initial",[error.message])}else if(error instanceof RemoteStorage.SyncError){widget.view.setState("offline",[])}else if(error instanceof RemoteStorage.Unauthorized){widget.view.setState("unauthorized")}else{RemoteStorage.log("[Widget] Unknown error");widget.view.setState("error",[error])}}}function flashFor(evt){if(evt.method==="GET"&&evt.isFolder){return false}return true}RemoteStorage.Widget=function(remoteStorage){var self=this;var requestsToFlashFor=0;this.rs=remoteStorage;this.rs.remote.on("connected",stateSetter(this,"connected"));this.rs.on("disconnected",stateSetter(this,"initial"));this.rs.on("connecting",stateSetter(this,"authing"));this.rs.on("authing",stateSetter(this,"authing"));this.rs.on("error",errorsHandler(this));if(this.rs.remote){this.rs.remote.on("wire-busy",function(evt){if(flashFor(evt)){requestsToFlashFor++;stateSetter(self,"busy")()}});this.rs.remote.on("wire-done",function(evt){if(flashFor(evt)){requestsToFlashFor--;if(requestsToFlashFor<=0){stateSetter(self,"connected")()}}})}if(hasLocalStorage){var state=localStorage[LS_STATE_KEY];if(state&&VALID_ENTRY_STATES[state]){this._rememberedState=state}}};RemoteStorage.Widget.prototype={display:function(domID){if(!this.view){this.setView(new RemoteStorage.Widget.View(this.rs))}this.view.display.apply(this.view,arguments);return this},linkWidgetToSync:function(){if(typeof this.rs.sync==="object"&&typeof this.rs.sync.sync==="function"){this.view.on("sync",this.rs.sync.sync.bind(this.rs.sync))}else{RemoteStorage.log("[Widget] typeof this.rs.sync check fail",this.rs.sync);setTimeout(this.linkWidgetToSync.bind(this),1e3)}},setView:function(view){this.view=view;this.view.on("connect",function(options){if(typeof options==="string"){this.rs.connect(options)}else if(options.special){this.rs[options.special].connect(options)}}.bind(this));this.view.on("disconnect",this.rs.disconnect.bind(this.rs));this.linkWidgetToSync();try{this.view.on("reset",function(){var location=RemoteStorage.Authorize.getLocation();this.rs.on("disconnected",location.reload.bind(location));this.rs.disconnect()}.bind(this))}catch(e){if(e.message&&e.message.match(/Unknown event/)){}else{throw e}}if(this._rememberedState){setTimeout(stateSetter(this,this._rememberedState),0);delete this._rememberedState}}};RemoteStorage.prototype.displayWidget=function(domID){return this.widget.display(domID)};RemoteStorage.Widget._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();if(!remoteStorage.widget){remoteStorage.widget=new RemoteStorage.Widget(remoteStorage)}};RemoteStorage.Widget._rs_supported=function(remoteStorage){return typeof document!=="undefined"}})(typeof window!=="undefined"?window:global);(function(window){var t=RemoteStorage.I18n.translate;var cEl=function(){return document.createElement.apply(document,arguments)};function gCl(parent,className){return parent.getElementsByClassName(className)[0]}function gTl(parent,tagName){return parent.getElementsByTagName(tagName)[0]}function removeClass(el,className){return el.classList.remove(className)}function addClass(el,className){return el.classList.add(className)}function stopPropagation(event){if(typeof event.stopPropagation==="function"){event.stopPropagation()}else{event.cancelBubble=true}}function setupButton(parent,className,iconName,eventListener){var element=gCl(parent,className);if(typeof iconName!=="undefined"){var img=gTl(element,"img");(img||element).src=RemoteStorage.Assets[iconName]}element.addEventListener("click",eventListener);return element}RemoteStorage.Widget.View=function(remoteStorage){this.rs=remoteStorage;if(typeof document==="undefined"){throw"Widget not supported"}RemoteStorage.eventHandling(this,"connect","disconnect","sync","display","reset");for(var event in this.events){this.events[event]=this.events[event].bind(this)}this.hideBubbleOnBodyClick=function(event){for(var p=event.target;p!==document.body;p=p.parentElement){if(p.id==="remotestorage-widget"){return}}this.hideBubble()}.bind(this)};RemoteStorage.Widget.View.prototype={connectGdrive:function(){this._emit("connect",{special:"googledrive"})},connectDropbox:function(){this._emit("connect",{special:"dropbox"})},setState:function(state,args){RemoteStorage.log("[View] widget.view.setState(",state,",",args,");");var s=this.states[state];if(typeof s==="undefined"){throw new Error("Bad State assigned to view: "+state)}s.apply(this,args)},setUserAddress:function(addr){this.userAddress=addr||"";var el;if(this.div&&(el=gTl(this.div,"form").userAddress)){el.value=this.userAddress}},toggleBubble:function(event){if(this.bubble.className.search("rs-hidden")<0){this.hideBubble(event)}else{this.showBubble(event)}},hideBubble:function(){addClass(this.bubble,"rs-hidden");document.body.removeEventListener("click",this.hideBubbleOnBodyClick)},showBubble:function(event){removeClass(this.bubble,"rs-hidden");if(typeof event!=="undefined"){stopPropagation(event)}document.body.addEventListener("click",this.hideBubbleOnBodyClick);gTl(this.bubble,"form").userAddress.focus()},display:function(domID){if(typeof this.div!=="undefined"){return this.div}var element=cEl("div");var style=cEl("style");style.innerHTML=RemoteStorage.Assets.widgetCss;element.id="remotestorage-widget";element.innerHTML=RemoteStorage.Assets.widget;element.appendChild(style);if(domID){var parent=document.getElementById(domID);if(!parent){throw'Failed to find target DOM element with id="'+domID+'"'}parent.appendChild(element)}else{document.body.appendChild(element)}setupButton(element,"rs-sync","syncIcon",this.events.sync);setupButton(element,"rs-disconnect","disconnectIcon",this.events.disconnect);setupButton(element,"remotestorage-reset",undefined,this.events.reset);var cb=setupButton(element,"connect","connectIcon",this.events.connect);this.form=gTl(element,"form");var el=this.form.userAddress;el.addEventListener("keyup",function(event){if(event.target.value){cb.removeAttribute("disabled")}else{cb.setAttribute("disabled","disabled")}});if(this.userAddress){el.value=this.userAddress}this.cube=setupButton(element,"rs-cube","remoteStorageIcon",this.toggleBubble.bind(this));setupButton(element,"rs-dropbox","dropbox",this.connectDropbox.bind(this));setupButton(element,"rs-googledrive","googledrive",this.connectGdrive.bind(this));var bubbleDontCatch={INPUT:true,BUTTON:true,IMG:true};var eventListener=function(event){if(!bubbleDontCatch[event.target.tagName]&&!this.div.classList.contains("remotestorage-state-unauthorized")){this.showBubble(event)}}.bind(this);this.bubble=setupButton(element,"rs-bubble",undefined,eventListener);this.hideBubble();this.div=element;this.states.initial.call(this);this.events.display.call(this);return this.div},_renderTranslatedInitialContent:function(){gCl(this.div,"rs-status-text").innerHTML=t("view_connect");gCl(this.div,"remotestorage-reset").innerHTML=t("view_get_me_out");gCl(this.div,"rs-error-plz-report").innerHTML=t("view_error_plz_report");gCl(this.div,"remotestorage-unauthorized").innerHTML=t("view_unauthorized")},states:{initial:function(message){var cube=this.cube;var info=message||t("view_info");this._renderTranslatedInitialContent();if(message){cube.src=RemoteStorage.Assets.remoteStorageIconError;removeClass(this.cube,"remotestorage-loading");this.showBubble();setTimeout(function(){cube.src=RemoteStorage.Assets.remoteStorageIcon},5e3)}else{this.hideBubble()}this.div.className="remotestorage-state-initial";var backends=1;if(this._activateBackend("dropbox")){backends+=1}if(this._activateBackend("googledrive")){backends+=1}gCl(this.div,"rs-bubble-text").style.paddingRight=backends*32+8+"px";var cb=gCl(this.div,"connect");if(this.form.userAddress.value){cb.removeAttribute("disabled")}var infoEl=gCl(this.div,"rs-info-msg");infoEl.innerHTML=info;if(message){infoEl.classList.add("remotestorage-error-info")}else{infoEl.classList.remove("remotestorage-error-info")}},authing:function(){this.div.removeEventListener("click",this.events.connect);this.div.className="remotestorage-state-authing";gCl(this.div,"rs-status-text").innerHTML=t("view_connecting",this.userAddress);addClass(this.cube,"remotestorage-loading")},connected:function(){this.div.className="remotestorage-state-connected";gCl(this.div,"userAddress").innerHTML=this.userAddress;this.cube.src=RemoteStorage.Assets.remoteStorageIcon;removeClass(this.cube,"remotestorage-loading");var icons={googledrive:gCl(this.div,"rs-googledrive"),dropbox:gCl(this.div,"rs-dropbox")};icons.googledrive.style.display=icons.dropbox.style.display="none";if(icons[this.rs.backend]){icons[this.rs.backend].style.display="inline-block";gCl(this.div,"rs-bubble-text").style.paddingRight=2*32+8+"px"}else{gCl(this.div,"rs-bubble-text").style.paddingRight=32+8+"px"}},busy:function(){this.div.className="remotestorage-state-busy";addClass(this.cube,"remotestorage-loading");this.hideBubble()},offline:function(){this.div.className="remotestorage-state-offline";this.cube.src=RemoteStorage.Assets.remoteStorageIconOffline;gCl(this.div,"rs-status-text").innerHTML=t("view_offline")},error:function(err){var errorMsg=err;this.div.className="remotestorage-state-error";gCl(this.div,"rs-bubble-text").innerHTML=""+t("view_error_occured")+"";if(err instanceof Error){errorMsg=err.message+"\n\n"+err.stack}gCl(this.div,"rs-error-msg").textContent=errorMsg;this.cube.src=RemoteStorage.Assets.remoteStorageIconError;this.showBubble()},unauthorized:function(){this.div.className="remotestorage-state-unauthorized";this.cube.src=RemoteStorage.Assets.remoteStorageIconError;this.showBubble();this.div.addEventListener("click",this.events.connect)}},events:{connect:function(event){stopPropagation(event);event.preventDefault();this._emit("connect",gTl(this.div,"form").userAddress.value)},sync:function(event){stopPropagation(event);event.preventDefault();this._emit("sync")},disconnect:function(event){stopPropagation(event);event.preventDefault();this._emit("disconnect")},reset:function(event){event.preventDefault();var result=window.confirm(t("view_confirm_reset"));if(result){this._emit("reset")}},display:function(event){if(event){event.preventDefault()}this._emit("display")}},_activateBackend:function activateBackend(backendName){var className="rs-"+backendName;if(this.rs.apiKeys[backendName]){gCl(this.div,className).style.display="inline-block";return true}else{gCl(this.div,className).style.display="none";return false}}}})(typeof window!=="undefined"?window:global);(function(global){var ValidatorContext=function(parent,collectMultiple){this.missing=[];this.schemas=parent?Object.create(parent.schemas):{};this.collectMultiple=collectMultiple;this.errors=[];this.handleError=collectMultiple?this.collectError:this.returnError};ValidatorContext.prototype.returnError=function(error){return error};ValidatorContext.prototype.collectError=function(error){if(error){this.errors.push(error)}return null};ValidatorContext.prototype.prefixErrors=function(startIndex,dataPath,schemaPath){for(var i=startIndex;ischema.maximum){return new ValidationError(ErrorCodes.NUMBER_MAXIMUM,"Value "+data+" is greater than maximum "+schema.maximum).prefixWith(null,"maximum")}if(schema.exclusiveMaximum&&data==schema.maximum){return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE,"Value "+data+" is equal to exclusive maximum "+schema.maximum).prefixWith(null,"exclusiveMaximum")}}return null};ValidatorContext.prototype.validateString=function validateString(data,schema){return this.validateStringLength(data,schema)||this.validateStringPattern(data,schema)||null};ValidatorContext.prototype.validateStringLength=function validateStringLength(data,schema){if(typeof data!="string"){return null}if(schema.minLength!=undefined){if(data.lengthschema.maxLength){return new ValidationError(ErrorCodes.STRING_LENGTH_LONG,"String is too long ("+data.length+" chars), maximum "+schema.maxLength).prefixWith(null,"maxLength")}}return null};ValidatorContext.prototype.validateStringPattern=function validateStringPattern(data,schema){if(typeof data!="string"||schema.pattern==undefined){return null}var regexp=new RegExp(schema.pattern);if(!regexp.test(data)){return new ValidationError(ErrorCodes.STRING_PATTERN,"String does not match pattern").prefixWith(null,"pattern")}return null};ValidatorContext.prototype.validateArray=function validateArray(data,schema){if(!Array.isArray(data)){return null}return this.validateArrayLength(data,schema)||this.validateArrayUniqueItems(data,schema)||this.validateArrayItems(data,schema)||null};ValidatorContext.prototype.validateArrayLength=function validateArrayLength(data,schema){if(schema.minItems!=undefined){if(data.lengthschema.maxItems){var error=new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG,"Array is too long ("+data.length+" chars), maximum "+schema.maxItems).prefixWith(null,"maxItems");if(this.handleError(error)){return error}}}return null};ValidatorContext.prototype.validateArrayUniqueItems=function validateArrayUniqueItems(data,schema){if(schema.uniqueItems){for(var i=0;ischema.maxProperties){var error=new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM,"Too many properties defined ("+keys.length+"), maximum "+schema.maxProperties).prefixWith(null,"maxProperties");if(this.handleError(error)){return error}}}return null};ValidatorContext.prototype.validateObjectRequiredProperties=function validateObjectRequiredProperties(data,schema){if(schema.required!=undefined){for(var i=0;i0&&url.charAt(url.length-1)=="/"||remainder.charAt(0)=="#"||remainder.charAt(0)=="?"){if(map[schema.id]==undefined){map[schema.id]=schema}}}}if(typeof schema=="object"){for(var key in schema){if(key!="enum"&&typeof schema[key]=="object"){searchForTrustedSchemas(map,schema[key],url)}}}return map}var globalContext=new ValidatorContext;var publicApi={validate:function(data,schema){var context=new ValidatorContext(globalContext);if(typeof schema=="string"){schema={$ref:schema}}var added=context.addSchema("",schema);var error=context.validateAll(data,schema);this.error=error;this.missing=context.missing;this.valid=error==null;return this.valid},validateResult:function(){var result={};this.validate.apply(result,arguments);return result},validateMultiple:function(data,schema){var context=new ValidatorContext(globalContext,true);if(typeof schema=="string"){schema={$ref:schema}}context.addSchema("",schema);context.validateAll(data,schema);var result={};result.errors=context.errors;result.missing=context.missing;result.valid=result.errors.length==0;return result},addSchema:function(url,schema){return globalContext.addSchema(url,schema)},getSchema:function(url){return globalContext.getSchema(url)},missing:[],error:null,normSchema:normSchema,resolveUrl:resolveUrl,errorCodes:ErrorCodes};global.tv4=publicApi})(typeof window!="undefined"?window:global);var CHARS="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz".split("");Math.uuid=function(len,radix){var chars=CHARS,uuid=[],i;radix=radix||chars.length;if(len){for(i=0;i2){this.moduleName=parts[1]}else{this.moduleName="root"}RS.eventHandling(this,"change");this.on=this.on.bind(this);storage.onChange(this.base,this._fireChange.bind(this))};RS.BaseClient.prototype={extend:function(object){for(var key in object){this[key]=object[key]}return this},scope:function(path){return new RS.BaseClient(this.storage,this.makePath(path))},getListing:function(path,maxAge){if(typeof path!=="string"){path="" +}else if(path.length>0&&path[path.length-1]!=="/"){throw"Not a folder: "+path}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getListing must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body){return status===404?undefined:body})},getAll:function(path,maxAge){if(typeof path!=="string"){path=""}else if(path.length>0&&path[path.length-1]!=="/"){throw"Not a folder: "+path}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getAll must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body){if(status===404){return}if(typeof body==="object"){var promise=promising();var count=Object.keys(body).length,i=0;if(count===0){return}for(var key in body){this.storage.get(this.makePath(path+key),maxAge).then(function(status,b){body[this.key]=b;i++;if(i===count){promise.fulfill(body)}}.bind({key:key}))}return promise}}.bind(this))},getFile:function(path,maxAge){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.getFile must be a string")}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getFile must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body,mimeType,revision){return{data:body,mimeType:mimeType,revision:revision}})},storeFile:function(mimeType,path,body){if(typeof mimeType!=="string"){return promising().reject("Argument 'mimeType' of baseClient.storeFile must be a string")}if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.storeFile must be a string")}if(typeof body!=="string"&&typeof body!=="object"){return promising().reject("Argument 'body' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView")}var self=this;return this.storage.put(this.makePath(path),body,mimeType).then(function(status,_body,_mimeType,revision){if(status===200||status===201){return revision}else{throw"Request (PUT "+self.makePath(path)+") failed with status: "+status}})},getObject:function(path,maxAge){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.getObject must be a string")}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getObject must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body,mimeType,revision){if(typeof body==="object"){return body}else if(typeof body!=="undefined"&&status===200){throw"Not an object: "+this.makePath(path)}})},storeObject:function(typeAlias,path,object){if(typeof typeAlias!=="string"){return promising().reject("Argument 'typeAlias' of baseClient.storeObject must be a string")}if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.storeObject must be a string")}if(typeof object!=="object"){return promising().reject("Argument 'object' of baseClient.storeObject must be an object")}this._attachType(object,typeAlias);try{var validationResult=this.validate(object);if(!validationResult.valid){return promising(function(p){p.reject(validationResult)})}}catch(exc){if(!(exc instanceof RS.BaseClient.Types.SchemaNotFound)){return promising().reject(exc)}}return this.storage.put(this.makePath(path),object,"application/json; charset=UTF-8").then(function(status,_body,_mimeType,revision){if(status===200||status===201){return revision}else{throw"Request (PUT "+this.makePath(path)+") failed with status: "+status}}.bind(this))},remove:function(path){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.remove must be a string")}return this.storage.delete(this.makePath(path))},cache:function(path,strategy){if(typeof path!=="string"){throw"Argument 'path' of baseClient.cache must be a string"}if(strategy===false){deprecate("caching strategy ",'<"FLUSH">');strategy="FLUSH"}else if(strategy===undefined){strategy="ALL"}else if(typeof strategy!=="string"){deprecate("that caching strategy",'<"ALL">');strategy="ALL"}if(strategy!=="FLUSH"&&strategy!=="SEEN"&&strategy!=="ALL"){throw"Argument 'strategy' of baseclient.cache must be one of "+'["FLUSH", "SEEN", "ALL"]'}this.storage.caching.set(this.makePath(path),strategy);return this},flush:function(path){return this.storage.local.flush(path)},makePath:function(path){return this.base+(path||"")},_fireChange:function(event){this._emit("change",event)},_cleanPath:RS.WireClient.cleanPath,getItemURL:function(path){if(typeof path!=="string"){throw"Argument 'path' of baseClient.getItemURL must be a string"}if(this.storage.connected){path=this._cleanPath(this.makePath(path));return this.storage.remote.href+path}else{return undefined}},uuid:function(){return Math.uuid()}};RS.BaseClient._rs_init=function(){RS.prototype.scope=function(path){if(typeof path!=="string"){throw"Argument 'path' of baseClient.scope must be a string"}return new RS.BaseClient(this,path)}};maxAgeInvalid=function(maxAge){return typeof maxAge!=="undefined"&&typeof maxAge!=="number"}})(typeof window!=="undefined"?window:global);(function(global){RemoteStorage.BaseClient.Types={uris:{},schemas:{},aliases:{},declare:function(moduleName,alias,uri,schema){var fullAlias=moduleName+"/"+alias;if(schema.extends){var extendedAlias;var parts=schema.extends.split("/");if(parts.length===1){extendedAlias=moduleName+"/"+parts.shift()}else{extendedAlias=parts.join("/")}var extendedUri=this.uris[extendedAlias];if(!extendedUri){throw"Type '"+fullAlias+"' tries to extend unknown schema '"+extendedAlias+"'"}schema.extends=this.schemas[extendedUri]}this.uris[fullAlias]=uri;this.aliases[uri]=fullAlias;this.schemas[uri]=schema},resolveAlias:function(alias){return this.uris[alias]},getSchema:function(uri){return this.schemas[uri]},inScope:function(moduleName){var ml=moduleName.length;var schemas={};for(var alias in this.uris){if(alias.substr(0,ml+1)===moduleName+"/"){var uri=this.uris[alias];schemas[uri]=this.schemas[uri]}}return schemas}};var SchemaNotFound=function(uri){var error=new Error("Schema not found: "+uri);error.name="SchemaNotFound";return error};SchemaNotFound.prototype=Error.prototype;RemoteStorage.BaseClient.Types.SchemaNotFound=SchemaNotFound;RemoteStorage.BaseClient.prototype.extend({validate:function(object){var schema=RemoteStorage.BaseClient.Types.getSchema(object["@context"]);if(schema){return tv4.validateResult(object,schema)}else{throw new SchemaNotFound(object["@context"])}},declareType:function(alias,uri,schema){if(!schema){schema=uri;uri=this._defaultTypeURI(alias)}RemoteStorage.BaseClient.Types.declare(this.moduleName,alias,uri,schema)},_defaultTypeURI:function(alias){return"http://remotestoragejs.com/spec/modules/"+this.moduleName+"/"+alias},_attachType:function(object,alias){object["@context"]=RemoteStorage.BaseClient.Types.resolveAlias(alias)||this._defaultTypeURI(alias)}});Object.defineProperty(RemoteStorage.BaseClient.prototype,"schemas",{configurable:true,get:function(){return RemoteStorage.BaseClient.Types.inScope(this.moduleName)}})})(typeof window!=="undefined"?window:global);(function(){RemoteStorage.MODULES={};RemoteStorage.defineModule=function(moduleName,builder){RemoteStorage.MODULES[moduleName]=builder;Object.defineProperty(RemoteStorage.prototype,moduleName,{configurable:true,get:function(){var instance=this._loadModule(moduleName);Object.defineProperty(this,moduleName,{value:instance});return instance}});if(moduleName.indexOf("-")!==-1){var camelizedName=moduleName.replace(/\-[a-z]/g,function(s){return s[1].toUpperCase()});Object.defineProperty(RemoteStorage.prototype,camelizedName,{get:function(){return this[moduleName]}})}};RemoteStorage.prototype._loadModule=function(moduleName){var builder=RemoteStorage.MODULES[moduleName];if(builder){var module=builder(new RemoteStorage.BaseClient(this,"/"+moduleName+"/"),new RemoteStorage.BaseClient(this,"/public/"+moduleName+"/"));return module.exports}else{throw"Unknown module: "+moduleName}};RemoteStorage.prototype.defineModule=function(moduleName){console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!");RemoteStorage.defineModule.apply(RemoteStorage,arguments)}})();(function(){function loadTable(table,storage,paths){table.setAttribute("border","1");table.style.margin="8px";table.innerHTML="";var thead=document.createElement("thead");table.appendChild(thead);var titleRow=document.createElement("tr");thead.appendChild(titleRow);["Path","Content-Type","Revision"].forEach(function(label){var th=document.createElement("th");th.textContent=label;thead.appendChild(th)});var tbody=document.createElement("tbody");table.appendChild(tbody);function renderRow(tr,path,contentType,revision){[path,contentType,revision].forEach(function(value){var td=document.createElement("td");td.textContent=value||"";tr.appendChild(td)})}function loadRow(path){if(storage.connected===false){return}function processRow(status,body,contentType,revision){if(status===200){var tr=document.createElement("tr");tbody.appendChild(tr);renderRow(tr,path,contentType,revision);if(path[path.length-1]==="/"){for(var key in body){loadRow(path+key)}}}}storage.get(path).then(processRow)}paths.forEach(loadRow)}function renderWrapper(title,table,storage,paths){var wrapper=document.createElement("div");var heading=document.createElement("h2");heading.textContent=title;wrapper.appendChild(heading);var updateButton=document.createElement("button");updateButton.textContent="Refresh";updateButton.onclick=function(){loadTable(table,storage,paths)};wrapper.appendChild(updateButton);if(storage.reset){var resetButton=document.createElement("button");resetButton.textContent="Reset";resetButton.onclick=function(){storage.reset(function(newStorage){storage=newStorage;loadTable(table,storage,paths)})};wrapper.appendChild(resetButton)}wrapper.appendChild(table);loadTable(table,storage,paths);return wrapper}function renderLocalChanges(local){var wrapper=document.createElement("div");var heading=document.createElement("h2");heading.textContent="Outgoing changes";wrapper.appendChild(heading);var updateButton=document.createElement("button");updateButton.textContent="Refresh";wrapper.appendChild(updateButton);var list=document.createElement("ul");list.style.fontFamily="courier";wrapper.appendChild(list);function updateList(){local.changesBelow("/").then(function(changes){list.innerHTML="";changes.forEach(function(change){var el=document.createElement("li");el.textContent=JSON.stringify(change);list.appendChild(el)})})}updateButton.onclick=updateList;updateList();return wrapper}RemoteStorage.prototype.inspect=function(){var widget=document.createElement("div");widget.id="remotestorage-inspect";widget.style.position="absolute";widget.style.top=0;widget.style.left=0;widget.style.background="black";widget.style.color="white";widget.style.border="groove 5px #ccc";var controls=document.createElement("div");controls.style.position="absolute";controls.style.top=0;controls.style.left=0;var heading=document.createElement("strong");heading.textContent=" remotestorage.js inspector ";controls.appendChild(heading);var syncButton;if(this.local){syncButton=document.createElement("button");syncButton.textContent="Synchronize";controls.appendChild(syncButton)}var closeButton=document.createElement("button");closeButton.textContent="Close";closeButton.onclick=function(){document.body.removeChild(widget)};controls.appendChild(closeButton);widget.appendChild(controls);var remoteTable=document.createElement("table");var localTable=document.createElement("table");widget.appendChild(renderWrapper("Remote",remoteTable,this.remote,this.caching.rootPaths));if(this.local){widget.appendChild(renderWrapper("Local",localTable,this.local,["/"]));widget.appendChild(renderLocalChanges(this.local));syncButton.onclick=function(){this.log("sync clicked");this.sync().then(function(){this.log("SYNC FINISHED");loadTable(localTable,this.local,["/"])}.bind(this),function(err){console.error("SYNC FAILED",err,err.stack)})}.bind(this)}document.body.appendChild(widget)}})();(function(){var util={getEventEmitter:function(){var object={};var args=Array.prototype.slice.call(arguments);args.unshift(object);RemoteStorage.eventHandling.apply(RemoteStorage,args);object.emit=object._emit;return object},extend:function(target){var sources=Array.prototype.slice.call(arguments,1);sources.forEach(function(source){for(var key in source){target[key]=source[key]}});return target},asyncEach:function(array,callback){return this.asyncMap(array,callback).then(function(){return array})},asyncMap:function(array,callback){var promise=promising();var n=array.length,i=0;var results=[],errors=[];function oneDone(){i++;if(i===n){promise.fulfill(results,errors)}}array.forEach(function(item,index){var result;try{result=callback(item)}catch(exc){oneDone();errors[index]=exc}if(typeof result==="object"&&typeof result.then==="function"){result.then(function(res){results[index]=res;oneDone()},function(error){errors[index]=res;oneDone()})}else{oneDone();results[index]=result}});return promise},containingFolder:function(path){var folder=path.replace(/[^\/]+\/?$/,"");return folder===path?null:folder},isFolder:function(path){return path.substr(-1)==="/"},baseName:function(path){var parts=path.split("/");if(util.isFolder(path)){return parts[parts.length-2]+"/"}else{return parts[parts.length-1]}},bindAll:function(object){for(var key in this){if(typeof object[key]==="function"){object[key]=object[key].bind(object)}}}};Object.defineProperty(RemoteStorage.prototype,"util",{get:function(){console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release.");return util}})})();(function(global){var RS=RemoteStorage;var BASE_URL="https://www.googleapis.com";var AUTH_URL="https://accounts.google.com/o/oauth2/auth";var AUTH_SCOPE="https://www.googleapis.com/auth/drive";var GD_DIR_MIME_TYPE="application/vnd.google-apps.folder";var RS_DIR_MIME_TYPE="application/json; charset=UTF-8";function buildQueryString(params){return Object.keys(params).map(function(key){return encodeURIComponent(key)+"="+encodeURIComponent(params[key])}).join("&")}function fileNameFromMeta(meta){return encodeURIComponent(meta.title)+(meta.mimeType===GD_DIR_MIME_TYPE?"/":"")}function metaTitleFromFileName(filename){if(filename.substr(-1)==="/"){filename=filename.substr(0,filename.length-1)}return decodeURIComponent(filename)}function parentPath(path){return path.replace(/[^\/]+\/?$/,"")}function baseName(path){var parts=path.split("/");if(path.substr(-1)==="/"){return parts[parts.length-2]+"/"}else{return parts[parts.length-1]}}var Cache=function(maxAge){this.maxAge=maxAge;this._items={}};Cache.prototype={get:function(key){var item=this._items[key];var now=(new Date).getTime();return item&&item.t>=now-this.maxAge?item.v:undefined},set:function(key,value){this._items[key]={v:value,t:(new Date).getTime()}}};RS.GoogleDrive=function(remoteStorage,clientId){RS.eventHandling(this,"connected");this.rs=remoteStorage;this.clientId=clientId;this._fileIdCache=new Cache(60*5);setTimeout(function(){this.configure(undefined,undefined,undefined,localStorage["remotestorage:googledrive:token"])}.bind(this),0)};RS.GoogleDrive.prototype={configure:function(_x,_y,_z,token){if(token){localStorage["remotestorage:googledrive:token"]=token;this.token=token;this.connected=true;this._emit("connected")}else{this.connected=false;delete this.token;delete localStorage["remotestorage:googledrive:token"]}},connect:function(){this.rs.setBackend("googledrive");RS.Authorize(AUTH_URL,AUTH_SCOPE,String(RS.Authorize.getLocation()),this.clientId)},get:function(path,options){if(path.substr(-1)==="/"){return this._getFolder(path,options)}else{return this._getFile(path,options)}},put:function(path,body,contentType,options){var promise=promising();function putDone(error,response){if(error){promise.reject(error)}else if(response.status>=200&&response.status<300){var meta=JSON.parse(response.responseText);promise.fulfill(200,undefined,meta.mimeType,meta.etag)}else{promise.reject("PUT failed with status "+response.status+" ("+response.responseText+")")}}this._getFileId(path,function(idError,id){if(idError){promise.reject(idError);return}else if(id){this._updateFile(id,path,body,contentType,options,putDone)}else{this._createFile(path,body,contentType,options,putDone)}});return promise},"delete":function(path,options){var promise=promising();this._getFileId(path,function(idError,id){if(idError){promise.reject(idError)}else if(id){this._request("DELETE",BASE_URL+"/drive/v2/files/"+id,{},function(deleteError,response){if(deleteError){promise.reject(deleteError)}else if(response.status===200||response.status===204){promise.fulfill(200)}else{promise.reject("Delete failed: "+response.status+" ("+response.responseText+")")}})}else{promise.fulfill(200)}});return promise},_updateFile:function(id,path,body,contentType,options,callback){callback=callback.bind(this);var metadata={mimeType:contentType};this._request("PUT",BASE_URL+"/upload/drive/v2/files/"+id+"?uploadType=resumable",{body:JSON.stringify(metadata),headers:{"Content-Type":"application/json; charset=UTF-8"}},function(metadataError,response){if(metadataError){callback(metadataError)}else{this._request("PUT",response.getResponseHeader("Location"),{body:contentType.match(/^application\/json/)?JSON.stringify(body):body},callback)}})},_createFile:function(path,body,contentType,options,callback){callback=callback.bind(this);this._getParentId(path,function(parentIdError,parentId){if(parentIdError){callback(parentIdError);return}var fileName=baseName(path);var metadata={title:metaTitleFromFileName(fileName),mimeType:contentType,parents:[{kind:"drive#fileLink",id:parentId}]};this._request("POST",BASE_URL+"/upload/drive/v2/files?uploadType=resumable",{body:JSON.stringify(metadata),headers:{"Content-Type":"application/json; charset=UTF-8"}},function(metadataError,response){if(metadataError){callback(metadataError)}else{this._request("POST",response.getResponseHeader("Location"),{body:contentType.match(/^application\/json/)?JSON.stringify(body):body},callback)}})})},_getFile:function(path,options){var promise=promising();this._getFileId(path,function(idError,id){if(idError){promise.reject(idError)}else{this._getMeta(id,function(metaError,meta){var etagWithoutQuotes;if(typeof meta==="object"&&typeof meta.etag==="string"){etagWithoutQuotes=meta.etag.substring(1,meta.etag.length-1)}if(metaError){promise.reject(metaError)}else if(meta.downloadUrl){var options={};if(meta.mimeType.match(/charset=binary/)){options.responseType="blob"}this._request("GET",meta.downloadUrl,options,function(downloadError,response){if(downloadError){promise.reject(downloadError)}else{var body=response.response;if(meta.mimeType.match(/^application\/json/)){try{body=JSON.parse(body)}catch(e){}}promise.fulfill(200,body,meta.mimeType,etagWithoutQuotes)}})}else{promise.fulfill(200,"",meta.mimeType,etagWithoutQuotes)}})}});return promise},_getFolder:function(path,options){var promise=promising();this._getFileId(path,function(idError,id){var query,fields,data,i,etagWithoutQuotes,itemsMap;if(idError){promise.reject(idError)}else if(!id){promise.fulfill(404)}else{query="'"+id+"' in parents";fields="items(downloadUrl,etag,fileSize,id,mimeType,title)";this._request("GET",BASE_URL+"/drive/v2/files?"+"q="+encodeURIComponent(query)+"&fields="+encodeURIComponent(fields)+"&maxResults=1000",{},function(childrenError,response){if(childrenError){promise.reject(childrenError)}else{if(response.status===200){try{data=JSON.parse(response.responseText)}catch(e){promise.reject("non-JSON response from GoogleDrive");return}itemsMap={};for(i=0;i=0||true){try{body=JSON.parse(body);mime="application/json; charset=UTF-8"}catch(e){RS.log("Failed parsing Json, assume it is something else then",mime,path)}}promise.fulfill(status,body,mime,rev)}}else{promise.fulfill(status)}}});return promise},put:function(path,body,contentType,options){RemoteStorage.log("dropbox.put",arguments);if(!this.connected){throw new Error("not connected (path: "+path+")")}path=cleanPath(path);var promise=this._sharePromise(path);var revCache=this._revCache;var savedRev=revCache.get(path);if(options&&options.ifMatch&&savedRev&&savedRev!==options.ifMatch){promise.fulfill(412);return promise}if(!contentType.match(/charset=/)){contentType+="; charset="+(body instanceof ArrayBuffer||RS.WireClient.isArrayBufferView(body)?"binary":"utf-8")}var url="https://api-content.dropbox.com/1/files_put/auto"+path+"?";if(options&&options.ifMatch){url+="parent_rev="+encodeURIComponent(options.ifMatch)}if(body.length>150*1024*1024){RemoteStorage.log("files larger than 150MB not supported yet")}else{this._request("PUT",url,{body:body,headers:{"Content-Type":contentType}},function(err,resp){if(err){promise.reject(err)}else{var response=JSON.parse(resp.responseText);if(response.path!==path){promise.fulfill(412);this.rs.log("Dropbox created conflicting File ",response.path)}else{revCache.set(path,response.rev);promise.fulfill(resp.status)}}})}return promise},"delete":function(path,options){RemoteStorage.log("dropbox.delete ",arguments);if(!this.connected){throw new Error("not connected (path: "+path+")")}path=cleanPath(path);var promise=promising();var revCache=this._revCache;var savedRev=revCache.get(path);if(options.ifMatch&&savedRev&&options.ifMatch!==savedRev){promise.fulfill(412);return promise}var url="https://api.dropbox.com/1/fileops/delete?root=auto&path="+encodeURIComponent(path);this._request("POST",url,{},function(err,resp){if(err){promise.reject(error)}else{promise.fulfill(resp.status);revCache.delete(path)}});return promise.then(function(){var args=Array.prototype.slice.call(arguments);delete this._itemRefs[path];var p=promising();return p.fulfill.apply(p,args)}.bind(this))},_sharePromise:function(path){var promise=promising();var self=this;if(path.match(/^\/public\/.*[^\/]$/)&&typeof this._itemRefs[path]==="undefined"){RemoteStorage.log("shareing this one ",path);promise.then(function(){var args=Array.prototype.slice.call(arguments);var p=promising();RemoteStorage.log("calling share now");self.share(path).then(function(){RemoteStorage.log("shareing fullfilled promise",arguments);p.fulfill.apply(p,args)},function(err){RemoteStorage.log("shareing failed",err);p.fulfill.apply(p,args)});return p})}return promise},share:function(path){var url="https://api.dropbox.com/1/media/auto"+path;var promise=promising();var itemRefs=this._itemRefs;this._request("POST",url,{},function(err,resp){if(err){RemoteStorage.log(err);err.message='Shareing Dropbox Thingie("'+path+'") failed'+err.message;promise.reject(err)}else{try{var response=JSON.parse(resp.responseText);var url=response.url;itemRefs[path]=url;RemoteStorage.log("SHAREING URL :::: ",url," for ",path);if(hasLocalStorage){localStorage[SETTINGS_KEY+":shares"]=JSON.stringify(this._itemRefs)}promise.fulfill(url)}catch(err){err.message+="share error";promise.reject(err)}}});return promise},info:function(){var url="https://api.dropbox.com/1/account/info";var promise=promising();this._request("GET",url,{},function(err,resp){if(err){promise.reject(err)}else{try{var info=JSON.parse(resp.responseText);promise.fulfill(info)}catch(e){promise.reject(err)}}});return promise},_request:function(method,url,options,callback){callback=callback.bind(this);if(!options.headers){options.headers={}}options.headers["Authorization"]="Bearer "+this.token;RS.WireClient.request.call(this,method,url,options,function(err,xhr){if(xhr&&xhr.status===503){global.setTimeout(this._request(method,url,options,callback),3210)}else{callback(err,xhr)}})},fetchDelta:function(){var args=Array.prototype.slice.call(arguments);var promise=promising();var self=this;this._request("POST","https://api.dropbox.com/1/delta",{body:this._deltaCursor?"cursor="+encodeURIComponent(this._deltaCursor):"",headers:{"Content-Type":"application/x-www-form-urlencoded"}},function(error,response){if(error){this.rs.log("fetchDeltas",error);this.rs._emit("error",new RemoteStorage.SyncError("fetchDeltas failed"+error));promise.reject(error)}else{if(response.status!==200){if(response.status===400){this.rs._emit("error",new RemoteStorage.Unauthorized);promise.fulfill.apply(promise,args) +}else{RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText);promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText)}return promise}var delta;try{delta=JSON.parse(response.responseText)}catch(error){RS.log("fetchDeltas can not parse response",error);return promise.reject("can not parse response of fetchDelta : "+error.message)}if(!delta.entries){RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!",delta);return promise.reject("dropbox.fetchDeltas failed, no entries found")}if(delta.reset){this._revCache=new LowerCaseCache("rev");promise.then(function(){var args=Array.prototype.slice.call(arguments);self._revCache._activatePropagation();var p=promising();return p.fulfill.apply(p,args)})}if(delta.cursor){this._deltaCursor=delta.cursor}RemoteStorage.log("Delta : ",delta.entries);delta.entries.forEach(function(entry){var path=entry[0];var rev;if(!entry[1]){rev=null}else{if(entry[1].is_dir){return}rev=entry[1].rev}self._revCache.set(path,rev)});promise.fulfill.apply(promise,args)}});return promise}};function hookSync(rs){if(rs._dropboxOrigSync){return}rs._dropboxOrigSync=rs.sync.bind(rs);rs.sync=function(){return this.dropbox.fetchDelta.apply(this.dropbox,arguments).then(rs._dropboxOrigSync,function(err){rs._emit("error",new rs.SyncError(err))})}}function unHookSync(rs){if(!rs._dropboxOrigSync){return}rs.sync=rs._dropboxOrigSync;delete rs._dropboxOrigSync}function hookGetItemURL(rs){if(rs._origBaseClientGetItemURL){return}rs._origBaseClientGetItemURL=RS.BaseClient.prototype.getItemURL;RS.BaseClient.prototype.getItemURL=function(path){var ret=rs.dropbox._itemRefs[path];return ret?ret:""}}function unHookGetItemURL(rs){if(!rs._origBaseClieNtGetItemURL){return}RS.BaseClient.prototype.getItemURL=rs._origBaseClietGetItemURL;delete rs._origBaseClietGetItemURL}function hookRemote(rs){if(rs._origRemote){return}rs._origRemote=rs.remote;rs.remote=rs.dropbox}function unHookRemote(rs){if(rs._origRemote){rs.remote=rs._origRemote;delete rs._origRemote}}function hookIt(rs){hookRemote(rs);if(rs.sync){hookSync(rs)}hookGetItemURL(rs)}function unHookIt(rs){unHookRemote(rs);unHookSync(rs);unHookGetItemURL(rs)}RS.Dropbox._rs_init=function(rs){hasLocalStorage=rs.localStorageAvailable();if(rs.apiKeys.dropbox){rs.dropbox=new RS.Dropbox(rs)}if(rs.backend==="dropbox"){hookIt(rs)}};RS.Dropbox._rs_supported=function(){return true};RS.Dropbox._rs_cleanup=function(rs){unHookIt(rs);if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}rs.removeEventListener("error",onErrorCb);rs.setBackend(undefined)}})(this);remoteStorage=new RemoteStorage; \ No newline at end of file diff --git a/release/0.10.0-beta2/remotestorage-node.js b/release/0.10.0-beta2/remotestorage-node.js new file mode 100644 index 000000000..31f563377 --- /dev/null +++ b/release/0.10.0-beta2/remotestorage-node.js @@ -0,0 +1,5691 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ + +/** FILE: lib/promising.js **/ +(function(global) { + function getPromise(builder) { + var promise; + + if(typeof(builder) === 'function') { + setTimeout(function() { + try { + builder(promise); + } catch(e) { + promise.reject(e); + } + }, 0); + } + + var consumers = [], success, result; + + function notifyConsumer(consumer) { + if(success) { + var nextValue; + if(consumer.fulfilled) { + try { + nextValue = [consumer.fulfilled.apply(null, result)]; + } catch(exc) { + consumer.promise.reject(exc); + return; + } + } else { + nextValue = result; + } + if(nextValue[0] && typeof(nextValue[0].then) === 'function') { + nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill.apply(null, nextValue); + } + } else { + if(consumer.rejected) { + var ret; + try { + ret = consumer.rejected.apply(null, result); + } catch(exc) { + consumer.promise.reject(exc); + return; + } + if(ret && typeof(ret.then) === 'function') { + ret.then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill(ret); + } + } else { + consumer.promise.reject.apply(null, result); + } + } + } + + function resolve(succ, res) { + if(result) { + console.error("WARNING: Can't resolve promise, already resolved!"); + return; + } + success = succ; + result = Array.prototype.slice.call(res); + setTimeout(function() { + var cl = consumers.length; + if(cl === 0 && (! success)) { + console.error("Possibly uncaught error: ", result, result[0] && result[0].stack); + } + for(var i=0;i. + */ + RemoteStorage.log = function() { + if (RemoteStorage._log) { + console.log.apply(console, arguments); + } + }; + + + RemoteStorage.prototype = { + /** + * Method: connect + * + * Connect to a remoteStorage server. + * + * Parameters: + * userAddress - The user address (user@host) to connect to. + * + * Discovers the webfinger profile of the given user address and + * initiates the OAuth dance. + * + * This method must be called *after* all required access has been claimed. + * + */ + connect: function(userAddress) { + this.setBackend('remotestorage'); + if (userAddress.indexOf('@') < 0) { + this._emit('error', new RemoteStorage.DiscoveryError("User adress doesn't contain an @.")); + return; + } + this.remote.configure(userAddress); + this._emit('connecting'); + + var discoveryTimeout = setTimeout(function() { + this._emit('error', new RemoteStorage.DiscoveryError("No storage information found at that user address.")); + }.bind(this), 5000); + + RemoteStorage.Discover(userAddress, function(href, storageApi, authURL) { + clearTimeout(discoveryTimeout); + if (!href) { + this._emit('error', new RemoteStorage.DiscoveryError("Failed to contact storage server.")); + return; + } + this._emit('authing'); + this.remote.configure(userAddress, href, storageApi); + if (! this.remote.connected) { + this.authorize(authURL); + } + }.bind(this)); + }, + + /** + * Method: disconnect + * + * "Disconnect" from remotestorage server to terminate current session. + * This method clears all stored settings and deletes the entire local + * cache. + */ + disconnect: function() { + if (this.remote) { + this.remote.configure(null, null, null, null); + } + this._setGPD({ + get: this._pendingGPD('get'), + put: this._pendingGPD('put'), + delete: this._pendingGPD('delete') + }); + var n = this._cleanups.length, i = 0; + + var oneDone = function() { + i++; + if (i >= n) { + this._init(); + RemoteStorage.log('Done cleaning up, emitting disconnected and disconnect events'); + this._emit('disconnected'); + } + }.bind(this); + + if (n > 0) { + this._cleanups.forEach(function(cleanup) { + var cleanupResult = cleanup(this); + if (typeof(cleanup) === 'object' && typeof(cleanup.then) === 'function') { + cleanupResult.then(oneDone); + } else { + oneDone(); + } + }.bind(this)); + } else { + oneDone(); + } + }, + + setBackend: function(what) { + this.backend = what; + if (this.localStorageAvailable()) { + if (what) { + localStorage['remotestorage:backend'] = what; + } else { + delete localStorage['remotestorage:backend']; + } + } + }, + + /** + * Method: onChange + * + * Add a "change" event handler to the given path. Whenever a "change" + * happens (as determined by the backend, such as e.g. + * ) and the affected path is equal to or below + * the given 'path', the given handler is called. + * + * You should usually not use this method directly, but instead use the + * "change" events provided by . + * + * Parameters: + * path - Absolute path to attach handler to. + * handler - Handler function. + */ + onChange: function(path, handler) { + if (! this._pathHandlers.change[path]) { + this._pathHandlers.change[path] = []; + } + this._pathHandlers.change[path].push(handler); + }, + + /** + * Method: enableLog + * + * Enable remoteStorage logging + */ + enableLog: function() { + RemoteStorage._log = true; + }, + + /** + * Method: disableLog + * + * Disable remoteStorage logging + */ + disableLog: function() { + RemoteStorage._log = false; + }, + + /** + * Method: log + * + * The same as . + */ + log: function() { + RemoteStorage.log.apply(RemoteStorage, arguments); + }, + + setApiKeys: function(type, keys) { + if (keys) { + this.apiKeys[type] = keys; + } else { + delete this.apiKeys[type]; + } + if (this.localStorageAvailable()) { + localStorage['remotestorage:api-keys'] = JSON.stringify(this.apiKeys); + } + }, + + /** + ** INITIALIZATION + **/ + + _init: function() { + var self = this, + readyFired = false; + + function fireReady() { + try { + if (!readyFired) { + self._emit('ready'); + readyFired = true; + } + } catch(e) { + console.error("'ready' failed: ", e, e.stack); + self._emit('error', e); + } + } + + this._loadFeatures(function(features) { + this.log('[RemoteStorage] All features loaded'); + this.local = features.local && new features.local(); + // this.remote set by WireClient._rs_init as lazy property on + // RS.prototype + + if (this.local && this.remote) { + this._setGPD(SyncedGetPutDelete, this); + this._bindChange(this.local); + } else if (this.remote) { + this._setGPD(this.remote, this.remote); + } + + if (this.remote) { + this.remote.on('connected', function(){ + fireReady(); + self._emit('connected'); + }); + this.remote.on('not-connected', function(){ + fireReady(); + self._emit('not-connected'); + }); + if (this.remote.connected) { + fireReady(); + self._emit('connected'); + } + } + + this._collectCleanupFunctions(); + + try { + this._allLoaded = true; + this._emit('features-loaded'); + } catch(exc) { + logError(exc); + this._emit('error', exc); + } + this._processPending(); + }.bind(this)); + }, + + _collectCleanupFunctions: function() { + for (var i=0; i < this.features.length; i++) { + var cleanup = this.features[i].cleanup; + if (typeof(cleanup) === 'function') { + this._cleanups.push(cleanup); + } + } + }, + + /** + ** FEATURE DETECTION + **/ + _loadFeatures: function(callback) { + var featureList = [ + 'WireClient', + 'I18n', + 'Dropbox', + 'GoogleDrive', + 'Access', + 'Caching', + 'Discover', + 'Authorize', + 'Widget', + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage', + 'Sync', + 'BaseClient', + 'Env' + ]; + var features = []; + var featuresDone = 0; + var self = this; + + function featureDone() { + featuresDone++; + if (featuresDone === featureList.length) { + setTimeout(function() { + features.caching = !!RemoteStorage.Caching; + features.sync = !!RemoteStorage.Sync; + [ + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage' + ].some(function(cachingLayer) { + if (features.some(function(feature) { return feature.name === cachingLayer; })) { + features.local = RemoteStorage[cachingLayer]; + return true; + } + }); + self.features = features; + callback(features); + }, 0); + } + } + + function featureInitialized(name) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialized."); + features.push({ + name : name, + init : RemoteStorage[name]._rs_init, + supported : true, + cleanup : RemoteStorage[name]._rs_cleanup + }); + featureDone(); + } + + function featureFailed(name, err) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialization failed ( "+err+")"); + featureDone(); + } + + function featureSupported(name, success) { + self.log("[RemoteStorage] [FEATURE "+name+"]" + success ? "":" not"+" supported"); + if (!success) { + featureDone(); + } + } + + function initFeature(name) { + var initResult; + try { + initResult = RemoteStorage[name]._rs_init(self); + } catch(e) { + featureFailed(name, e); + return; + } + if (typeof(initResult) === 'object' && typeof(initResult.then) === 'function') { + initResult.then( + function(){ featureInitialized(name); }, + function(err){ featureFailed(name, err); } + ); + } else { + featureInitialized(name); + } + } + + featureList.forEach(function(featureName) { + self.log("[RemoteStorage] [FEATURE " + featureName + "] initializing..."); + var impl = RemoteStorage[featureName]; + var supported; + + if (impl) { + supported = !impl._rs_supported || impl._rs_supported(); + + if (typeof supported === 'object') { + supported.then( + function(){ + featureSupported(featureName, true); + initFeature(featureName); + }, + function(){ + featureSupported(featureName, false); + } + ); + } + else if (typeof supported === 'boolean') { + featureSupported(featureName, supported); + if (supported) { + initFeature(featureName); + } + } + } else { + featureSupported(featureName, false); + } + }); + }, + + localStorageAvailable: function() { + try { + return !!global.localStorage; + } catch(error) { + return false; + } + }, + + /** + ** GET/PUT/DELETE INTERFACE HELPERS + **/ + + _setGPD: function(impl, context) { + function wrap(f) { + return function() { + return f.apply(context, arguments) + .then(emitUnauthorized.bind(this)); + }; + } + this.get = wrap(impl.get); + this.put = wrap(impl.put); + this.delete = wrap(impl.delete); + }, + + _pendingGPD: function(methodName) { + return function() { + var promise = promising(); + this._pending.push({ + method: methodName, + args: Array.prototype.slice.call(arguments), + promise: promise + }); + return promise; + }.bind(this); + }, + + _processPending: function() { + this._pending.forEach(function(pending) { + try { + this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject); + } catch(e) { + pending.promise.reject(e); + } + }.bind(this)); + this._pending = []; + }, + + /** + ** CHANGE EVENT HANDLING + **/ + + _bindChange: function(object) { + object.on('change', this._dispatchEvent.bind(this, 'change')); + }, + + _dispatchEvent: function(eventName, event) { + for (var path in this._pathHandlers[eventName]) { + var pl = path.length; + var self = this; + this._pathHandlers[eventName][path].forEach(function(handler) { + if (event.path.substr(0, pl) === path) { + var ev = {}; + for (var key in event) { ev[key] = event[key]; } + ev.relativePath = event.path.replace(new RegExp('^' + path), ''); + try { + handler(ev); + } catch(e) { + console.error("'change' handler failed: ", e, e.stack); + self._emit('error', e); + } + } + }); + } + } + }; + + /** + * Property: connected + * + * Boolean property indicating if remoteStorage is currently connected. + */ + Object.defineProperty(RemoteStorage.prototype, 'connected', { + get: function() { + return this.remote.connected; + } + }); + + /** + * Property: access + * + * Tracking claimed access scopes. A instance. + * + * + * Property: caching + * + * Caching settings. A instance. + * + * Not available in no-cache builds. + * + * + * Property: remote + * + * Access to the remote backend used. Usually a . + * + * + * Property: local + * + * Access to the local caching backend used. Usually either a + * or instance. + * + * Not available in no-cache builds. + */ + + global.RemoteStorage = RemoteStorage; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/eventhandling.js **/ +(function(global) { + /** + * Interface: eventhandling + */ + var methods = { + /** + * Method: addEventListener + * + * Install an event handler for the given event name + */ + addEventListener: function(eventName, handler) { + if (typeof(eventName) !== 'string') { + throw new Error('Argument eventName should be a string'); + } + if (typeof(handler) !== 'function') { + throw new Error('Argument handler should be a function'); + } + RemoteStorage.log('[Eventhandling] Adding event listener', eventName, handler); + this._validateEvent(eventName); + this._handlers[eventName].push(handler); + }, + + /** + * Method: removeEventListener + * + * Remove a previously installed event handler + */ + removeEventListener: function(eventName, handler) { + this._validateEvent(eventName); + var hl = this._handlers[eventName].length; + for (var i=0;i + **/ + methods.on = methods.addEventListener; + + /** + * Function: eventHandling + * + * Mixes event handling functionality into an object. + * + * The first parameter is always the object to be extended. + * All remaining parameter are expected to be strings, interpreted as valid event + * names. + * + * Example: + * (start code) + * var MyConstructor = function() { + * eventHandling(this, 'connected', 'disconnected'); + * + * this._emit('connected'); + * this._emit('disconnected'); + * // This would throw an exception: + * // this._emit('something-else'); + * }; + * + * var myObject = new MyConstructor(); + * myObject.on('connected', function() { console.log('connected'); }); + * myObject.on('disconnected', function() { console.log('disconnected'); }); + * // This would throw an exception as well: + * // myObject.on('something-else', function() {}); + * (end code) + */ + RemoteStorage.eventHandling = function(object) { + var eventNames = Array.prototype.slice.call(arguments, 1); + for (var key in methods) { + object[key] = methods[key]; + } + object._handlers = {}; + eventNames.forEach(function(eventName) { + object._addEvent(eventName); + }); + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/wireclient.js **/ +(function(global) { + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.WireClient + * + * WireClient Interface + * -------------------- + * + * This file exposes a get/put/delete interface on top of XMLHttpRequest. + * It requires to be configured with parameters about the remotestorage server to + * connect to. + * Each instance of WireClient is always associated with a single remotestorage + * server and access token. + * + * Usually the WireClient instance can be accessed via `remoteStorage.remote`. + * + * This is the get/put/delete interface: + * + * - #get() takes a path and optionally a ifNoneMatch option carrying a version + * string to check. It returns a promise that will be fulfilled with the HTTP + * response status, the response body, the MIME type as returned in the + * 'Content-Type' header and the current revision, as returned in the 'ETag' + * header. + * - #put() takes a path, the request body and a content type string. It also + * accepts the ifMatch and ifNoneMatch options, that map to the If-Match and + * If-None-Match headers respectively. See the remotestorage-01 specification + * for details on handling these headers. It returns a promise, fulfilled with + * the same values as the one for #get(). + * - #delete() takes a path and the ifMatch option as well. It returns a promise + * fulfilled with the same values as the one for #get(). + * + * In addition to this, the WireClient has some compatibility features to work with + * remotestorage 2012.04 compatible storages. For example it will cache revisions + * from folder listings in-memory and return them accordingly as the "revision" + * parameter in response to #get() requests. Similarly it will return 404 when it + * receives an empty folder listing, to mimic remotestorage-01 behavior. Note + * that it is not always possible to know the revision beforehand, hence it may + * be undefined at times (especially for caching-roots). + */ + + var hasLocalStorage; + var SETTINGS_KEY = "remotestorage:wireclient"; + + var API_2012 = 1, API_00 = 2, API_01 = 3, API_02 = 4, API_HEAD = 5; + + var STORAGE_APIS = { + 'draft-dejong-remotestorage-00': API_00, + 'draft-dejong-remotestorage-01': API_01, + 'draft-dejong-remotestorage-02': API_02, + 'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012 + }; + + var isArrayBufferView; + + if (typeof(ArrayBufferView) === 'function') { + isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); }; + } else { + var arrayBufferViews = [ + Int8Array, Uint8Array, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array + ]; + isArrayBufferView = function(object) { + for (var i=0;i<8;i++) { + if (object instanceof arrayBufferViews[i]) { + return true; + } + } + return false; + }; + } + + function addQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + if (str === '*') { + return '*'; + } + + return '"' + str + '"'; + } + + function stripQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + + return str.replace(/^["']|["']$/g, ''); + } + + function readBinaryData(content, mimeType, callback) { + var blob = new Blob([content], { type: mimeType }); + var reader = new FileReader(); + reader.addEventListener("loadend", function() { + callback(reader.result); // reader.result contains the contents of blob as a typed array + }); + reader.readAsArrayBuffer(blob); + } + + function cleanPath(path) { + return path.replace(/\/+/g, '/').split('/').map(encodeURIComponent).join('/'); + } + + function isFolder(path) { + return (path.substr(-1) === '/'); + } + + function isFolderDescription(body) { + return ((body['@context'] === 'http://remotestorage.io/spec/folder-description') + && (typeof(body['items']) === 'object')); + } + + function isSuccessStatus(status) { + return [201, 204, 304].indexOf(status) >= 0; + } + + function isErrorStatus(status) { + return [401, 403, 404, 412].indexOf(status) >= 0; + } + + var onErrorCb; + + /** + * Class : RemoteStorage.WireClient + **/ + RS.WireClient = function(rs) { + this.connected = false; + + /** + * Event: change + * never fired for some reason + * + * Event: connected + * fired when the wireclient connect method realizes that it is + * in posession of a token and a href + **/ + RS.eventHandling(this, 'change', 'connected', 'wire-busy', 'wire-done', 'not-connected'); + + onErrorCb = function(error){ + if (error instanceof RemoteStorage.Unauthorized) { + this.configure(undefined, undefined, undefined, null); + } + }.bind(this); + rs.on('error', onErrorCb); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + setTimeout(function() { + this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token); + }.bind(this), 0); + } + } + + this._revisionCache = {}; + + if (this.connected) { + setTimeout(this._emit.bind(this), 0, 'connected'); + } + }; + + RS.WireClient.REQUEST_TIMEOUT = 30000; + + RS.WireClient.prototype = { + /** + * Property: token + * + * Holds the bearer token of this WireClient, as obtained in the OAuth dance + * + * Example: + * (start code) + * + * remoteStorage.remote.token + * // -> 'DEADBEEF01==' + */ + + /** + * Property: href + * + * Holds the server's base URL, as obtained in the Webfinger discovery + * + * Example: + * (start code) + * + * remoteStorage.remote.href + * // -> 'https://storage.example.com/users/jblogg/' + */ + + /** + * Property: storageApi + * + * Holds the spec version the server claims to be compatible with + * + * Example: + * (start code) + * + * remoteStorage.remote.storageApi + * // -> 'draft-dejong-remotestorage-01' + */ + + _request: function(method, uri, token, headers, body, getEtag, fakeRevision) { + if ((method === 'PUT' || method === 'DELETE') && uri[uri.length - 1] === '/') { + throw "Don't " + method + " on directories!"; + } + + var promise = promising(); + var revision; + var reqType; + var self = this; + + headers['Authorization'] = 'Bearer ' + token; + + this._emit('wire-busy', { + method: method, + isFolder: isFolder(uri) + }); + + RS.WireClient.request(method, uri, { + body: body, + headers: headers + }, function(error, response) { + if (error) { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: false + }); + self.online = false; + promise.reject(error); + } else { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: true + }); + self.online = true; + if (isErrorStatus(response.status)) { + RemoteStorage.log('[WireClient] Error response status', response.status); + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = undefined; + } + promise.fulfill(response.status, undefined, undefined, revision); + } else if (isSuccessStatus(response.status) || + (response.status === 200 && method !== 'GET')) { + revision = stripQuotes(response.getResponseHeader('ETag')); + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, undefined, undefined, revision); + } else { + var mimeType = response.getResponseHeader('Content-Type'); + var body; + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = response.status === 200 ? fakeRevision : undefined; + } + + if ((!mimeType) || mimeType.match(/charset=binary/)) { + RS.WireClient.readBinaryData(response.response, mimeType, function(result) { + RemoteStorage.log('[WireClient] Successful request with unknown or binary mime-type', revision); + promise.fulfill(response.status, result, mimeType, revision); + }); + } else { + if (mimeType && mimeType.match(/^application\/json/)) { + body = JSON.parse(response.responseText); + } else { + body = response.responseText; + } + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, body, mimeType, revision); + } + } + } + }); + return promise; + }, + + configure: function(userAddress, href, storageApi, token) { + if (typeof(userAddress) !== 'undefined') { + this.userAddress = userAddress; + } + if (typeof(href) !== 'undefined') { + this.href = href; + } + if (typeof(storageApi) !== 'undefined') { + this.storageApi = storageApi; + } + if (typeof(token) !== 'undefined') { + this.token = token; + } + if (typeof(this.storageApi) !== 'undefined') { + this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD; + this.supportsRevs = this._storageApi >= API_00; + } + if (this.href && this.token) { + this.connected = true; + this.online = true; + this._emit('connected'); + } else { + this.connected = false; + } + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ + userAddress: this.userAddress, + href: this.href, + token: this.token, + storageApi: this.storageApi + }); + } + RS.WireClient.configureHooks.forEach(function(hook) { + hook.call(this); + }.bind(this)); + }, + + stopWaitingForToken: function() { + if (!this.connected) { + this._emit('not-connected'); + } + }, + + get: function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } else if (options.ifNoneMatch) { + var oldRev = this._revisionCache[path]; + } + var promise = this._request('GET', this.href + cleanPath(path), this.token, headers, + undefined, this.supportsRevs, this._revisionCache[path]); + if (isFolder(path)) { + return promise.then(function(status, body, contentType, revision) { + var itemsMap = {}; + + // New folder listing received + if (status === 200 && typeof(body) === 'object') { + // Empty folder listing of any spec + if (Object.keys(body).length === 0) { + status = 404; + } + // >= 02 spec + else if (isFolderDescription(body)) { + for (var item in body.items) { + this._revisionCache[path + item] = body.items[item].ETag; + } + itemsMap = body.items; + } + // < 02 spec + else { + Object.keys(body).forEach(function(key){ + this._revisionCache[path + key] = body[key]; + itemsMap[key] = {"ETag": body[key]}; + }.bind(this)); + } + return promising().fulfill(status, itemsMap, contentType, revision); + } else { + return promising().fulfill(status, body, contentType, revision); + } + }.bind(this)); + } else { + return promise; + } + }, + + put: function(path, body, contentType, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + if (!contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var headers = { 'Content-Type': contentType }; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } + return this._request('PUT', this.href + cleanPath(path), this.token, + headers, body, this.supportsRevs); + }, + + 'delete': function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + } + return this._request('DELETE', this.href + cleanPath(path), this.token, + headers, + undefined, this.supportsRevs); + } + }; + + // Shared cleanPath used by Dropbox + RS.WireClient.cleanPath = cleanPath; + + // Shared isArrayBufferView used by WireClient and Dropbox + RS.WireClient.isArrayBufferView = isArrayBufferView; + + RS.WireClient.readBinaryData = readBinaryData; + + // Shared request function used by WireClient, GoogleDrive and Dropbox. + RS.WireClient.request = function(method, url, options, callback) { + RemoteStorage.log('[WireClient]', method, url); + + callback = callback.bind(this); + + var timedOut = false; + + var timer = setTimeout(function() { + timedOut = true; + callback('timeout'); + }, RS.WireClient.REQUEST_TIMEOUT); + + var xhr = new XMLHttpRequest(); + xhr.open(method, url, true); + + if (options.responseType) { + xhr.responseType = options.responseType; + } + if (options.headers) { + for (var key in options.headers) { + xhr.setRequestHeader(key, options.headers[key]); + } + } + + xhr.onload = function() { + if (timedOut) { return; } + clearTimeout(timer); + callback(null, xhr); + }; + + xhr.onerror = function(error) { + if (timedOut) { return; } + clearTimeout(timer); + callback(error); + }; + + var body = options.body; + + if (typeof(body) === 'object') { + if (isArrayBufferView(body)) { + /* alright. */ + //FIXME empty block + } + else if (body instanceof ArrayBuffer) { + body = new Uint8Array(body); + } else { + body = JSON.stringify(body); + } + } + xhr.send(body); + }; + + Object.defineProperty(RemoteStorage.WireClient.prototype, 'storageType', { + get: function() { + if (this.storageApi) { + var spec = this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/); + return spec ? spec[1] : '2012.04'; + } + } + }); + + RS.WireClient.configureHooks = []; + + RS.WireClient._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + remoteStorage.remote = new RS.WireClient(remoteStorage); + this.online = true; + }; + + RS.WireClient._rs_supported = function() { + return !! global.XMLHttpRequest; + }; + + RS.WireClient._rs_cleanup = function(remoteStorage){ + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + remoteStorage.removeEventListener('error', onErrorCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/discover.js **/ +(function(global) { + + // feature detection flags + var haveXMLHttpRequest, hasLocalStorage; + // used to store settings in localStorage + var SETTINGS_KEY = 'remotestorage:discover'; + // cache loaded from localStorage + var cachedInfo = {}; + + /** + * Class: RemoteStorage.Discover + * + * This class deals with the webfinger lookup + * + * Arguments: + * userAddress - user@host + * callback - gets called with href of the storage, the type and the authURL + * Example: + * (start code) + * + * (end code) + **/ + + RemoteStorage.Discover = function(userAddress, callback) { + if (userAddress in cachedInfo) { + var info = cachedInfo[userAddress]; + callback(info.href, info.type, info.authURL); + return; + } + var hostname = userAddress.split('@')[1]; + var params = '?resource=' + encodeURIComponent('acct:' + userAddress); + var urls = [ + 'https://' + hostname + '/.well-known/webfinger' + params, + 'https://' + hostname + '/.well-known/host-meta.json' + params, + 'http://' + hostname + '/.well-known/webfinger' + params, + 'http://' + hostname + '/.well-known/host-meta.json' + params + ]; + + function tryOne() { + var xhr = new XMLHttpRequest(); + var url = urls.shift(); + if (!url) { return callback(); } + RemoteStorage.log('[Discover] Trying URL', url); + xhr.open('GET', url, true); + xhr.onabort = xhr.onerror = function() { + console.error("webfinger error", arguments, '(', url, ')'); + tryOne(); + }; + xhr.onload = function() { + if (xhr.status !== 200) { return tryOne(); } + var profile; + + try { + profile = JSON.parse(xhr.responseText); + } catch(e) { + RemoteStorage.log("[Discover] Failed to parse profile ", xhr.responseText, e); + tryOne(); + return; + } + + if (!profile.links) { + RemoteStorage.log("[Discover] Profile has no links section ", JSON.stringify(profile)); + tryOne(); + return; + } + + var link; + profile.links.forEach(function(l) { + if (l.rel === 'remotestorage') { + link = l; + } else if (l.rel === 'remoteStorage' && !link) { + link = l; + } + }); + RemoteStorage.log('[Discover] Got profile', profile, 'and link', link); + if (link) { + var authURL = link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'] + || link.properties['auth-endpoint'], + storageType = link.properties['http://remotestorage.io/spec/version'] + || link.type; + cachedInfo[userAddress] = { href: link.href, type: storageType, authURL: authURL }; + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo }); + } + callback(link.href, storageType, authURL); + } else { + tryOne(); + } + }; + xhr.send(); + } + tryOne(); + }; + + RemoteStorage.Discover._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + cachedInfo = settings.cache; + } + } + }; + + RemoteStorage.Discover._rs_supported = function() { + haveXMLHttpRequest = !! global.XMLHttpRequest; + return haveXMLHttpRequest; + }; + + RemoteStorage.Discover._rs_cleanup = function() { + if (hasLocalStorage) { + delete localStorage[SETTINGS_KEY]; + } + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/authorize.js **/ +(function(global) { + + function extractParams() { + //FF already decodes the URL fragment in document.location.hash, so use this instead: + var location = RemoteStorage.Authorize.getLocation(), + hashPos = location.href.indexOf('#'), + hash; + if (hashPos === -1) { return; } + hash = location.href.substring(hashPos+1); + // if hash is not of the form #key=val&key=val, it's probably not for us + if (hash.indexOf('=') === -1) { return; } + return hash.split('&').reduce(function(m, kvs) { + var kv = kvs.split('='); + m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]); + return m; + }, {}); + } + + RemoteStorage.Authorize = function(authURL, scope, redirectUri, clientId) { + RemoteStorage.log('[Authorize] authURL = ', authURL); + + var url = authURL; + url += authURL.indexOf('?') > 0 ? '&' : '?'; + url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, '')); + url += '&scope=' + encodeURIComponent(scope); + url += '&client_id=' + encodeURIComponent(clientId); + url += '&response_type=token'; + RemoteStorage.Authorize.setLocation(url); + }; + + RemoteStorage.prototype.authorize = function(authURL) { + this.access.setStorageType(this.remote.storageType); + var scope = this.access.scopeParameter; + + var redirectUri = String(RemoteStorage.Authorize.getLocation()); + var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0]; + + RemoteStorage.Authorize(authURL, scope, redirectUri, clientId); + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.getLocation = function () { + return global.document.location; + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.setLocation = function (location) { + if (typeof location === 'string') { + global.document.location.href = location; + } else if (typeof location === 'object') { + global.document.location = location; + } else { + throw "Invalid location " + location; + } + }; + + RemoteStorage.Authorize._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + + var onFeaturesLoaded; + RemoteStorage.Authorize._rs_init = function(remoteStorage) { + + onFeaturesLoaded = function () { + var authParamsUsed = false; + if (params) { + if (params.error) { + throw "Authorization server errored: " + params.error; + } + if (params.access_token) { + remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token); + authParamsUsed = true; + } + if (params.remotestorage) { + remoteStorage.connect(params.remotestorage); + authParamsUsed = true; + } + } + if (!authParamsUsed) { + remoteStorage.remote.stopWaitingForToken(); + } + }; + var params = extractParams(), + location; + if (params) { + location = RemoteStorage.Authorize.getLocation(); + location.hash = ''; + } + remoteStorage.on('features-loaded', onFeaturesLoaded); + }; + + RemoteStorage.Authorize._rs_cleanup = function(remoteStorage) { + remoteStorage.removeEventListener('features-loaded', onFeaturesLoaded); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/access.js **/ +(function(global) { + + var SETTINGS_KEY = "remotestorage:access"; + + /** + * Class: RemoteStorage.Access + * + * Keeps track of claimed access and scopes. + */ + RemoteStorage.Access = function() { + this.reset(); + }; + + RemoteStorage.Access.prototype = { + + /** + * Method: claim + * + * Claim access on a given scope with given mode. + * + * Parameters: + * scope - An access scope, such as "contacts" or "calendar". + * mode - Access mode to use. Either "r" or "rw". + * + * Example: + * (start code) + * remoteStorage.access.claim('contacts', 'r'); + * remoteStorage.access.claim('pictures', 'rw'); + */ + claim: function(scope, mode) { + if (typeof(scope) !== 'string' || scope.indexOf('/') !== -1 || scope.length === 0) { + throw new Error('Scope should be a non-empty string without forward slashes'); + } + if (!mode.match(/^rw?$/)) { + throw new Error('Mode should be either \'r\' or \'rw\''); + } + this._adjustRootPaths(scope); + this.scopeModeMap[scope] = mode; + }, + + get: function(scope) { + return this.scopeModeMap[scope]; + }, + + remove: function(scope) { + var savedMap = {}; + var name; + for (name in this.scopeModeMap) { + savedMap[name] = this.scopeModeMap[name]; + } + this.reset(); + delete savedMap[scope]; + for (name in savedMap) { + this.set(name, savedMap[name]); + } + }, + + /** + * Verify permission for a given scope. + */ + checkPermission: function(scope, mode) { + var actualMode = this.get(scope); + return actualMode && (mode === 'r' || actualMode === 'rw'); + }, + + /** + * Verify permission for a given path. + */ + checkPathPermission: function(path, mode) { + if (this.checkPermission('*', mode)) { + return true; + } + return !!this.checkPermission(this._getModuleName(path), mode); + }, + + reset: function() { + this.rootPaths = []; + this.scopeModeMap = {}; + }, + + /** + * Return the module name for a given path. + */ + _getModuleName: function(path) { + if (path[0] !== '/') { + throw new Error('Path should start with a slash'); + } + var moduleMatch = path.replace(/^\/public/, '').match(/^\/([^\/]*)\//); + return moduleMatch ? moduleMatch[1] : '*'; + }, + + _adjustRootPaths: function(newScope) { + if ('*' in this.scopeModeMap || newScope === '*') { + this.rootPaths = ['/']; + } else if (! (newScope in this.scopeModeMap)) { + this.rootPaths.push('/' + newScope + '/'); + this.rootPaths.push('/public/' + newScope + '/'); + } + }, + + _scopeNameForParameter: function(scope) { + if (scope.name === '*' && this.storageType) { + if (this.storageType === '2012.04') { + return ''; + } else if (this.storageType.match(/remotestorage-0[01]/)) { + return 'root'; + } + } + return scope.name; + }, + + setStorageType: function(type) { + this.storageType = type; + } + }; + + /** + * Property: scopes + * + * Holds an array of claimed scopes in the form + * > { name: "", mode: "" } + */ + Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', { + get: function() { + return Object.keys(this.scopeModeMap).map(function(key) { + return { name: key, mode: this.scopeModeMap[key] }; + }.bind(this)); + } + }); + + Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', { + get: function() { + return this.scopes.map(function(scope) { + return this._scopeNameForParameter(scope) + ':' + scope.mode; + }.bind(this)).join(' '); + } + }); + + // Documented in src/remotestorage.js + Object.defineProperty(RemoteStorage.prototype, 'access', { + get: function() { + var access = new RemoteStorage.Access(); + Object.defineProperty(this, 'access', { + value: access + }); + return access; + }, + configurable: true + }); + + RemoteStorage.Access._rs_init = function() {}; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/env.js **/ +(function(pMode) { + + var mode = pMode, + env = {}, + isBackground = false; + + + RemoteStorage.Env = function() { + return env; + }; + + RemoteStorage.Env.isBrowser = function () { + return mode === "browser"; + }; + + RemoteStorage.Env.isNode = function () { + return mode === "node"; + }; + + RemoteStorage.Env.goBackground = function () { + isBackground = true; + RemoteStorage.Env._emit("background"); + }; + + RemoteStorage.Env.goForeground = function () { + isBackground = false; + RemoteStorage.Env._emit("foreground"); + }; + + RemoteStorage.Env._rs_init = function(remoteStorage) { + RemoteStorage.eventHandling(RemoteStorage.Env, "background", "foreground"); + + if ( mode === 'browser') { + if ( typeof(document.hidden) !== "undefined" ) { + env.hiddenProperty = "hidden"; + env.visibilityChangeEvent = "visibilitychange"; + } else if ( typeof(document.mozHidden) !== "undefined" ) { + env.hiddenProperty = "mozHidden"; + env.visibilityChangeEvent = "mozvisibilitychange"; + } else if ( typeof(document.msHidden) !== "undefined" ) { + env.hiddenProperty = "msHidden"; + env.visibilityChangeEvent = "msvisibilitychange"; + } else if ( typeof(document.webkitHidden) !== "undefined" ) { + env.hiddenProperty = "webkitHidden"; + env.visibilityChangeEvent = "webkitvisibilitychange"; + } + document.addEventListener(env.visibilityChangeEvent, function () { + if ( document[env.hiddenProperty] ) { + RemoteStorage.Env.goBackground(); + } else { + RemoteStorage.Env.goForeground(); + } + }, false); + } + }; + + RemoteStorage.Env._rs_cleanup = function(remoteStorage) { + }; + +})(typeof(window) !== 'undefined' ? 'browser' : 'node'); + + +/** FILE: lib/tv4.js **/ +/** +Author: Geraint Luff and others +Year: 2013 + +This code is released into the "public domain" by its author(s). Anybody may use, alter and distribute the code without restriction. The author makes no guarantees, and takes no liability of any kind for use of this code. + +If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory. +**/ + +(function (global) { +var ValidatorContext = function (parent, collectMultiple) { + this.missing = []; + this.schemas = parent ? Object.create(parent.schemas) : {}; + this.collectMultiple = collectMultiple; + this.errors = []; + this.handleError = collectMultiple ? this.collectError : this.returnError; +}; +ValidatorContext.prototype.returnError = function (error) { + return error; +}; +ValidatorContext.prototype.collectError = function (error) { + if (error) { + this.errors.push(error); + } + return null; +} +ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) { + for (var i = startIndex; i < this.errors.length; i++) { + this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath); + } + return this; +} + +ValidatorContext.prototype.getSchema = function (url) { + if (this.schemas[url] != undefined) { + var schema = this.schemas[url]; + return schema; + } + var baseUrl = url; + var fragment = ""; + if (url.indexOf('#') != -1) { + fragment = url.substring(url.indexOf("#") + 1); + baseUrl = url.substring(0, url.indexOf("#")); + } + if (this.schemas[baseUrl] != undefined) { + var schema = this.schemas[baseUrl]; + var pointerPath = decodeURIComponent(fragment); + if (pointerPath == "") { + return schema; + } else if (pointerPath.charAt(0) != "/") { + return undefined; + } + var parts = pointerPath.split("/").slice(1); + for (var i = 0; i < parts.length; i++) { + var component = parts[i].replace("~1", "/").replace("~0", "~"); + if (schema[component] == undefined) { + schema = undefined; + break; + } + schema = schema[component]; + } + if (schema != undefined) { + return schema; + } + } + if (this.missing[baseUrl] == undefined) { + this.missing.push(baseUrl); + this.missing[baseUrl] = baseUrl; + } +}; +ValidatorContext.prototype.addSchema = function (url, schema) { + var map = {}; + map[url] = schema; + normSchema(schema, url); + searchForTrustedSchemas(map, schema, url); + for (var key in map) { + this.schemas[key] = map[key]; + } + return map; +}; + +ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) { + if (schema['$ref'] != undefined) { + schema = this.getSchema(schema['$ref']); + if (!schema) { + return null; + } + } + + var errorCount = this.errors.length; + var error = this.validateBasic(data, schema) + || this.validateNumeric(data, schema) + || this.validateString(data, schema) + || this.validateArray(data, schema) + || this.validateObject(data, schema) + || this.validateCombinations(data, schema) + || null + if (error || errorCount != this.errors.length) { + while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) { + var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null; + var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null; + if (error) { + error = error.prefixWith(dataPart, schemaPart); + } + this.prefixErrors(errorCount, dataPart, schemaPart); + } + } + + return this.handleError(error); +} + +function recursiveCompare(A, B) { + if (A === B) { + return true; + } + if (typeof A == "object" && typeof B == "object") { + if (Array.isArray(A) != Array.isArray(B)) { + return false; + } else if (Array.isArray(A)) { + if (A.length != B.length) { + return false + } + for (var i = 0; i < A.length; i++) { + if (!recursiveCompare(A[i], B[i])) { + return false; + } + } + } else { + for (var key in A) { + if (B[key] === undefined && A[key] !== undefined) { + return false; + } + } + for (var key in B) { + if (A[key] === undefined && B[key] !== undefined) { + return false; + } + } + for (var key in A) { + if (!recursiveCompare(A[key], B[key])) { + return false; + } + } + } + return true; + } + return false; +} + +ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) { + var error; + if (error = this.validateType(data, schema)) { + return error.prefixWith(null, "type"); + } + if (error = this.validateEnum(data, schema)) { + return error.prefixWith(null, "type"); + } + return null; +} + +ValidatorContext.prototype.validateType = function validateType(data, schema) { + if (schema.type == undefined) { + return null; + } + var dataType = typeof data; + if (data == null) { + dataType = "null"; + } else if (Array.isArray(data)) { + dataType = "array"; + } + var allowedTypes = schema.type; + if (typeof allowedTypes != "object") { + allowedTypes = [allowedTypes]; + } + + for (var i = 0; i < allowedTypes.length; i++) { + var type = allowedTypes[i]; + if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) { + return null; + } + } + return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType); +} + +ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) { + if (schema["enum"] == undefined) { + return null; + } + for (var i = 0; i < schema["enum"].length; i++) { + var enumVal = schema["enum"][i]; + if (recursiveCompare(data, enumVal)) { + return null; + } + } + return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data)); +} +ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) { + return this.validateMultipleOf(data, schema) + || this.validateMinMax(data, schema) + || null; +} + +ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) { + var multipleOf = schema.multipleOf || schema.divisibleBy; + if (multipleOf == undefined) { + return null; + } + if (typeof data == "number") { + if (data%multipleOf != 0) { + return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf); + } + } + return null; +} + +ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) { + if (typeof data != "number") { + return null; + } + if (schema.minimum != undefined) { + if (data < schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum"); + } + if (schema.exclusiveMinimum && data == schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum"); + } + } + if (schema.maximum != undefined) { + if (data > schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum"); + } + if (schema.exclusiveMaximum && data == schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum"); + } + } + return null; +} +ValidatorContext.prototype.validateString = function validateString(data, schema) { + return this.validateStringLength(data, schema) + || this.validateStringPattern(data, schema) + || null; +} + +ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) { + if (typeof data != "string") { + return null; + } + if (schema.minLength != undefined) { + if (data.length < schema.minLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength"); + } + } + if (schema.maxLength != undefined) { + if (data.length > schema.maxLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength"); + } + } + return null; +} + +ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) { + if (typeof data != "string" || schema.pattern == undefined) { + return null; + } + var regexp = new RegExp(schema.pattern); + if (!regexp.test(data)) { + return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern"); + } + return null; +} +ValidatorContext.prototype.validateArray = function validateArray(data, schema) { + if (!Array.isArray(data)) { + return null; + } + return this.validateArrayLength(data, schema) + || this.validateArrayUniqueItems(data, schema) + || this.validateArrayItems(data, schema) + || null; +} + +ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) { + if (schema.minItems != undefined) { + if (data.length < schema.minItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxItems != undefined) { + if (data.length > schema.maxItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) { + if (schema.uniqueItems) { + for (var i = 0; i < data.length; i++) { + for (var j = i + 1; j < data.length; j++) { + if (recursiveCompare(data[i], data[j])) { + var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems"); + if (this.handleError(error)) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) { + if (schema.items == undefined) { + return null; + } + var error; + if (Array.isArray(schema.items)) { + for (var i = 0; i < data.length; i++) { + if (i < schema.items.length) { + if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) { + return error; + } + } else if (schema.additionalItems != undefined) { + if (typeof schema.additionalItems == "boolean") { + if (!schema.additionalItems) { + error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems"); + if (this.handleError(error)) { + return error; + } + } + } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) { + return error; + } + } + } + } else { + for (var i = 0; i < data.length; i++) { + if (error = this.validateAll(data[i], schema.items, [i], ["items"])) { + return error; + } + } + } + return null; +} +ValidatorContext.prototype.validateObject = function validateObject(data, schema) { + if (typeof data != "object" || data == null || Array.isArray(data)) { + return null; + } + return this.validateObjectMinMaxProperties(data, schema) + || this.validateObjectRequiredProperties(data, schema) + || this.validateObjectProperties(data, schema) + || this.validateObjectDependencies(data, schema) + || null; +} + +ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) { + var keys = Object.keys(data); + if (schema.minProperties != undefined) { + if (keys.length < schema.minProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxProperties != undefined) { + if (keys.length > schema.maxProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) { + if (schema.required != undefined) { + for (var i = 0; i < schema.required.length; i++) { + var key = schema.required[i]; + if (data[key] === undefined) { + var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required"); + if (this.handleError(error)) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) { + var error; + for (var key in data) { + var foundMatch = false; + if (schema.properties != undefined && schema.properties[key] != undefined) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) { + return error; + } + } + if (schema.patternProperties != undefined) { + for (var patternKey in schema.patternProperties) { + var regexp = new RegExp(patternKey); + if (regexp.test(key)) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) { + return error; + } + } + } + } + if (!foundMatch && schema.additionalProperties != undefined) { + if (typeof schema.additionalProperties == "boolean") { + if (!schema.additionalProperties) { + error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties"); + if (this.handleError(error)) { + return error; + } + } + } else { + if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) { + var error; + if (schema.dependencies != undefined) { + for (var depKey in schema.dependencies) { + if (data[depKey] !== undefined) { + var dep = schema.dependencies[depKey]; + if (typeof dep == "string") { + if (data[dep] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } else if (Array.isArray(dep)) { + for (var i = 0; i < dep.length; i++) { + var requiredKey = dep[i]; + if (data[requiredKey] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } + } else { + if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) { + var error; + return this.validateAllOf(data, schema) + || this.validateAnyOf(data, schema) + || this.validateOneOf(data, schema) + || this.validateNot(data, schema) + || null; +} + +ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) { + if (schema.allOf == undefined) { + return null; + } + var error; + for (var i = 0; i < schema.allOf.length; i++) { + var subSchema = schema.allOf[i]; + if (error = this.validateAll(data, subSchema, [], ["allOf", i])) { + return error; + } + } + return null; +} + +ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) { + if (schema.anyOf == undefined) { + return null; + } + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.anyOf.length; i++) { + var subSchema = schema.anyOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["anyOf", i]); + + if (error == null && errorCount == this.errors.length) { + this.errors = this.errors.slice(0, startErrorCount); + return null; + } + if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf")); + } + } + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors); +} + +ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) { + if (schema.oneOf == undefined) { + return null; + } + var validIndex = null; + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.oneOf.length; i++) { + var subSchema = schema.oneOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["oneOf", i]); + + if (error == null && errorCount == this.errors.length) { + if (validIndex == null) { + validIndex = i; + } else { + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf"); + } + } else if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf")); + } + } + if (validIndex == null) { + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors); + } else { + this.errors = this.errors.slice(0, startErrorCount); + } + return null; +} + +ValidatorContext.prototype.validateNot = function validateNot(data, schema) { + if (schema.not == undefined) { + return null; + } + var oldErrorCount = this.errors.length; + var error = this.validateAll(data, schema.not); + var notErrors = this.errors.slice(oldErrorCount); + this.errors = this.errors.slice(0, oldErrorCount); + if (error == null && notErrors.length == 0) { + return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not") + } + return null; +} + +// parseURI() and resolveUrl() are from https://gist.github.com/1088850 +// - released as public domain by author ("Yaffle") - see comments on gist + +function parseURI(url) { + var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/); + // authority = '//' + user + ':' + pass '@' + hostname + ':' port + return (m ? { + href : m[0] || '', + protocol : m[1] || '', + authority: m[2] || '', + host : m[3] || '', + hostname : m[4] || '', + port : m[5] || '', + pathname : m[6] || '', + search : m[7] || '', + hash : m[8] || '' + } : null); +} + +function resolveUrl(base, href) {// RFC 3986 + + function removeDotSegments(input) { + var output = []; + input.replace(/^(\.\.?(\/|$))+/, '') + .replace(/\/(\.(\/|$))+/g, '/') + .replace(/\/\.\.$/, '/../') + .replace(/\/?[^\/]*/g, function (p) { + if (p === '/..') { + output.pop(); + } else { + output.push(p); + } + }); + return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : ''); + } + + href = parseURI(href || ''); + base = parseURI(base || ''); + + return !href || !base ? null : (href.protocol || base.protocol) + + (href.protocol || href.authority ? href.authority : base.authority) + + removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) + + (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) + + href.hash; +} + +function normSchema(schema, baseUri) { + if (baseUri == undefined) { + baseUri = schema.id; + } else if (typeof schema.id == "string") { + baseUri = resolveUrl(baseUri, schema.id); + schema.id = baseUri; + } + if (typeof schema == "object") { + if (Array.isArray(schema)) { + for (var i = 0; i < schema.length; i++) { + normSchema(schema[i], baseUri); + } + } else if (typeof schema['$ref'] == "string") { + schema['$ref'] = resolveUrl(baseUri, schema['$ref']); + } else { + for (var key in schema) { + if (key != "enum") { + normSchema(schema[key], baseUri); + } + } + } + } +} + +var ErrorCodes = { + INVALID_TYPE: 0, + ENUM_MISMATCH: 1, + ANY_OF_MISSING: 10, + ONE_OF_MISSING: 11, + ONE_OF_MULTIPLE: 12, + NOT_PASSED: 13, + // Numeric errors + NUMBER_MULTIPLE_OF: 100, + NUMBER_MINIMUM: 101, + NUMBER_MINIMUM_EXCLUSIVE: 102, + NUMBER_MAXIMUM: 103, + NUMBER_MAXIMUM_EXCLUSIVE: 104, + // String errors + STRING_LENGTH_SHORT: 200, + STRING_LENGTH_LONG: 201, + STRING_PATTERN: 202, + // Object errors + OBJECT_PROPERTIES_MINIMUM: 300, + OBJECT_PROPERTIES_MAXIMUM: 301, + OBJECT_REQUIRED: 302, + OBJECT_ADDITIONAL_PROPERTIES: 303, + OBJECT_DEPENDENCY_KEY: 304, + // Array errors + ARRAY_LENGTH_SHORT: 400, + ARRAY_LENGTH_LONG: 401, + ARRAY_UNIQUE: 402, + ARRAY_ADDITIONAL_ITEMS: 403 +}; + +function ValidationError(code, message, dataPath, schemaPath, subErrors) { + if (code == undefined) { + throw new Error ("No code supplied for error: "+ message); + } + this.code = code; + this.message = message; + this.dataPath = dataPath ? dataPath : ""; + this.schemaPath = schemaPath ? schemaPath : ""; + this.subErrors = subErrors ? subErrors : null; +} +ValidationError.prototype = { + prefixWith: function (dataPrefix, schemaPrefix) { + if (dataPrefix != null) { + dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1"); + this.dataPath = "/" + dataPrefix + this.dataPath; + } + if (schemaPrefix != null) { + schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1"); + this.schemaPath = "/" + schemaPrefix + this.schemaPath; + } + if (this.subErrors != null) { + for (var i = 0; i < this.subErrors.length; i++) { + this.subErrors[i].prefixWith(dataPrefix, schemaPrefix); + } + } + return this; + } +}; + +function searchForTrustedSchemas(map, schema, url) { + if (typeof schema.id == "string") { + if (schema.id.substring(0, url.length) == url) { + var remainder = schema.id.substring(url.length); + if ((url.length > 0 && url.charAt(url.length - 1) == "/") + || remainder.charAt(0) == "#" + || remainder.charAt(0) == "?") { + if (map[schema.id] == undefined) { + map[schema.id] = schema; + } + } + } + } + if (typeof schema == "object") { + for (var key in schema) { + if (key != "enum" && typeof schema[key] == "object") { + searchForTrustedSchemas(map, schema[key], url); + } + } + } + return map; +} + +var globalContext = new ValidatorContext(); + +var publicApi = { + validate: function (data, schema) { + var context = new ValidatorContext(globalContext); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + var added = context.addSchema("", schema); + var error = context.validateAll(data, schema); + this.error = error; + this.missing = context.missing; + this.valid = (error == null); + return this.valid; + }, + validateResult: function () { + var result = {}; + this.validate.apply(result, arguments); + return result; + }, + validateMultiple: function (data, schema) { + var context = new ValidatorContext(globalContext, true); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + context.addSchema("", schema); + context.validateAll(data, schema); + var result = {}; + result.errors = context.errors; + result.missing = context.missing; + result.valid = (result.errors.length == 0); + return result; + }, + addSchema: function (url, schema) { + return globalContext.addSchema(url, schema); + }, + getSchema: function (url) { + return globalContext.getSchema(url); + }, + missing: [], + error: null, + normSchema: normSchema, + resolveUrl: resolveUrl, + errorCodes: ErrorCodes +}; + +global.tv4 = publicApi; + +})(typeof(window) != 'undefined' ? window : global); + + + +/** FILE: lib/Math.uuid.js **/ +/*! + Math.uuid.js (v1.4) + http://www.broofa.com + mailto:robert@broofa.com + + Copyright (c) 2010 Robert Kieffer + Dual licensed under the MIT and GPL licenses. + + ******** + + Changes within remoteStorage.js: + 2012-10-31: + - added AMD wrapper + - moved extensions for Math object into exported object. +*/ + +/* + * Generate a random uuid. + * + * USAGE: Math.uuid(length, radix) + * length - the desired number of characters + * radix - the number of allowable values for each character. + * + * EXAMPLES: + * // No arguments - returns RFC4122, version 4 ID + * >>> Math.uuid() + * "92329D39-6F5C-4520-ABFC-AAB64544E172" + * + * // One argument - returns ID of the specified length + * >>> Math.uuid(15) // 15 character ID (default base=62) + * "VcydxgltxrVZSTV" + * + * // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62) + * >>> Math.uuid(8, 2) // 8 character ID (base=2) + * "01001010" + * >>> Math.uuid(8, 10) // 8 character ID (base=10) + * "47473046" + * >>> Math.uuid(8, 16) // 8 character ID (base=16) + * "098F4D35" + */ + // Private array of chars to use + var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); + +Math.uuid = function (len, radix) { + var chars = CHARS, uuid = [], i; + radix = radix || chars.length; + + if (len) { + // Compact form + for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix]; + } else { + // rfc4122, version 4 form + var r; + + // rfc4122 requires these characters + uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; + uuid[14] = '4'; + + // Fill in random data. At i==19 set the high bits of clock sequence as + // per rfc4122, sec. 4.1.5 + for (i = 0; i < 36; i++) { + if (!uuid[i]) { + r = 0 | Math.random()*16; + uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; + } + } + } + + return uuid.join(''); +}; + + +/** FILE: src/baseclient.js **/ +(function(global) { + + function deprecate(thing, replacement) { + console.log('WARNING: ' + thing + ' is deprecated. Use ' + + replacement + ' instead.'); + } + + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.BaseClient + * + * Provides a high-level interface to access data below a given root path. + * + * A BaseClient deals with three types of data: folders, objects and files. + * + * returns a list of all items within a folder, or undefined + * if a 404 is encountered. Items that end with a forward slash ("/") are + * child folders. + * + * / operate on JSON objects. Each object has a type. + * + * / operates on files. Each file has a MIME type. + * + * operates on either objects or files (but not folders, folders are + * created and removed implictly). + */ + RS.BaseClient = function(storage, base) { + if (base[base.length - 1] !== '/') { + throw "Not a folder: " + base; + } + + if (base === '/') { + // allow absolute and relative paths for the root scope. + this.makePath = function(path) { + return (path[0] === '/' ? '' : '/') + path; + }; + } + + /** + * Property: storage + * + * The instance this operates on. + */ + this.storage = storage; + + /** + * Property: base + * + * Base path this operates on. + * + * For the module's privateClient this would be //, for the + * corresponding publicClient /public//. + */ + this.base = base; + + var parts = this.base.split('/'); + if (parts.length > 2) { + this.moduleName = parts[1]; + } else { + this.moduleName = 'root'; + } + + /** + * Event: change + * + * Emitted when a node changes + * + * Arguments: + * event - Event object containing information about the changed node + * + * (start code) + * { + * path: path, // Path of the changed node + * origin: 'window', 'local', or 'remote' // emitted by user action within the app, local data store, or remote sync + * oldValue: oldBody, // Old body of the changed node (undefined if creation) + * newValue: newBody // New body of the changed node (undefined if deletion) + * } + * (end code) + **/ + + RS.eventHandling(this, 'change'); + this.on = this.on.bind(this); + storage.onChange(this.base, this._fireChange.bind(this)); + }; + + RS.BaseClient.prototype = { + + extend: function(object) { + for (var key in object) { + this[key] = object[key]; + } + return this; + }, + + /** + * Method: scope + * + * Returns a new operating on a subpath of the current path. + */ + scope: function(path) { + return new RS.BaseClient(this.storage, this.makePath(path)); + }, + + // folder operations + + /** + * Method: getListing + * + * Get a list of child nodes below a given path. + * + * The callback semantics of getListing are identical to those of getObject. + * + * Parameters: + * path - The path to query. It MUST end with a forward slash. + * maxAge - (optional) Maximum age of cached listing in + * milliseconds + * + * Returns: + * + * A promise for an object, representing child nodes. + * + * Keys ending in a forward slash represent *folder nodes*, while all + * other keys represent *data nodes*. + * + * For spec versions <= 01, the data node information will contain only + * the item's ETag. For later spec versions, it will also contain the + * content type and -length of the item. + * + * Example: + * (start code) + * client.getListing('').then(function(listing) { + * listing.forEach(function(item) { + * console.log(item); + * }); + * }); + * (end code) + */ + getListing: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getListing must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then( + function(status, body) { + return (status === 404) ? undefined : body; + } + ); + }, + + /** + * Method: getAll + * + * Get all objects directly below a given path. + * + * Parameters: + * path - path to the folder + * maxAge - (optional) Maximum age of cached objects in + * milliseconds + * + * Returns: + * a promise for an object in the form { path : object, ... } + * + * Example: + * (start code) + * client.getAll('').then(function(objects) { + * for (var key in objects) { + * console.log('- ' + key + ': ', objects[key]); + * } + * }); + * (end code) + */ + getAll: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getAll must be undefined or a number'); + } + + return this.storage.get(this.makePath(path), maxAge).then(function(status, body) { + if (status === 404) { return; } + if (typeof(body) === 'object') { + var promise = promising(); + var count = Object.keys(body).length, i = 0; + if (count === 0) { + // treat this like 404. it probably means a folder listing that + // has changes that haven't been pushed out yet. + return; + } + for (var key in body) { + this.storage.get(this.makePath(path + key), maxAge). + then(function(status, b) { + body[this.key] = b; + i++; + if (i === count) { promise.fulfill(body); } + }.bind({ key: key })); + } + return promise; + } + }.bind(this)); + }, + + // file operations + + /** + * Method: getFile + * + * Get the file at the given path. A file is raw data, as opposed to + * a JSON object (use for that). + * + * Except for the return value structure, getFile works exactly like + * getObject. + * + * Parameters: + * path - see getObject + * + * Returns: + * A promise for an object: + * + * mimeType - String representing the MIME Type of the document. + * data - Raw data of the document (either a string or an ArrayBuffer) + * + * Example: + * (start code) + * // Display an image: + * client.getFile('path/to/some/image').then(function(file) { + * var blob = new Blob([file.data], { type: file.mimeType }); + * var targetElement = document.findElementById('my-image-element'); + * targetElement.src = window.URL.createObjectURL(blob); + * }); + * (end code) + */ + getFile: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getFile must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getFile must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + return { + data: body, + mimeType: mimeType, + revision: revision // (this is new) + }; + }); + }, + + /** + * Method: storeFile + * + * Store raw data at a given path. + * + * Parameters: + * mimeType - MIME media type of the data being stored + * path - path relative to the module root. MAY NOT end in a forward slash. + * data - string, ArrayBuffer or ArrayBufferView of raw data to store + * + * The given mimeType will later be returned, when retrieving the data + * using . + * + * Example (UTF-8 data): + * (start code) + * client.storeFile('text/html', 'index.html', '

Hello World!

'); + * (end code) + * + * Example (Binary data): + * (start code) + * // MARKUP: + * + * // CODE: + * var input = document.getElementById('file-input'); + * var file = input.files[0]; + * var fileReader = new FileReader(); + * + * fileReader.onload = function() { + * client.storeFile(file.type, file.name, fileReader.result); + * }; + * + * fileReader.readAsArrayBuffer(file); + * (end code) + * + */ + storeFile: function(mimeType, path, body) { + if (typeof(mimeType) !== 'string') { + return promising().reject('Argument \'mimeType\' of baseClient.storeFile must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeFile must be a string'); + } + if (typeof(body) !== 'string' && typeof(body) !== 'object') { + return promising().reject('Argument \'body\' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView'); + } + + var self = this; + return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + self.makePath(path) + ") failed with status: " + status; + } + }); + }, + + // object operations + + /** + * Method: getObject + * + * Get a JSON object from given path. + * + * Parameters: + * path - relative path from the module root (without leading slash) + * + * Returns: + * A promise for the object. + * + * Example: + * (start code) + * client.getObject('/path/to/object'). + * then(function(object) { + * // object is either an object or null + * }); + * (end code) + */ + getObject: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getObject must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getObject must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + if (typeof(body) === 'object') { + return body; + } else if (typeof(body) !== 'undefined' && status === 200) { + throw "Not an object: " + this.makePath(path); + } + }); + }, + + /** + * Method: storeObject + * + * Store object at given path. Triggers synchronization. + * + * Parameters: + * + * type - unique type of this object within this module. See description below. + * path - path relative to the module root. + * object - an object to be saved to the given node. It must be serializable as JSON. + * + * Returns: + * A promise to store the object. The promise fails with a ValidationError, when validations fail. + * + * + * What about the type?: + * + * A great thing about having data on the web, is to be able to link to + * it and rearrange it to fit the current circumstances. To facilitate + * that, eventually you need to know how the data at hand is structured. + * For documents on the web, this is usually done via a MIME type. The + * MIME type of JSON objects however, is always application/json. + * To add that extra layer of "knowing what this object is", remoteStorage + * aims to use . + * A first step in that direction, is to add a *@context attribute* to all + * JSON data put into remoteStorage. + * Now that is what the *type* is for. + * + * Within remoteStorage.js, @context values are built using three components: + * http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness + * the module name - module names should be unique as well + * the type given here - naming this particular kind of object within this module + * + * In retrospect that means, that whenever you introduce a new "type" in calls to + * storeObject, you should make sure that once your code is in the wild, future + * versions of the code are compatible with the same JSON structure. + * + * How to define types?: + * + * See for examples. + */ + storeObject: function(typeAlias, path, object) { + if (typeof(typeAlias) !== 'string') { + return promising().reject('Argument \'typeAlias\' of baseClient.storeObject must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeObject must be a string'); + } + if (typeof(object) !== 'object') { + return promising().reject('Argument \'object\' of baseClient.storeObject must be an object'); + } + this._attachType(object, typeAlias); + try { + var validationResult = this.validate(object); + if (! validationResult.valid) { + return promising(function(p) { p.reject(validationResult); }); + } + } catch(exc) { + if (! (exc instanceof RS.BaseClient.Types.SchemaNotFound)) { + return promising().reject(exc); + } + } + return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status; + } + }.bind(this)); + }, + + // generic operations + + /** + * Method: remove + * + * Remove node at given path from storage. Triggers synchronization. + * + * Parameters: + * path - Path relative to the module root. + */ + remove: function(path) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.remove must be a string'); + } + return this.storage.delete(this.makePath(path)); + }, + + + cache: function(path, strategy) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.cache must be a string'; + } + if (strategy === false) { + deprecate('caching strategy ', '<"FLUSH">'); + strategy = 'FLUSH'; + } else if (strategy === undefined) { + strategy = 'ALL'; + } else if (typeof(strategy) !== 'string') { + deprecate('that caching strategy', '<"ALL">'); + strategy = 'ALL'; + } + if (strategy !== 'FLUSH' && + strategy !== 'SEEN' && + strategy !== 'ALL') { + throw 'Argument \'strategy\' of baseclient.cache must be one of ' + + '["FLUSH", "SEEN", "ALL"]'; + } + this.storage.caching.set(this.makePath(path), strategy); + return this; + }, + + flush: function(path) { + return this.storage.local.flush(path); + }, + + makePath: function(path) { + return this.base + (path || ''); + }, + + _fireChange: function(event) { + this._emit('change', event); + }, + + _cleanPath: RS.WireClient.cleanPath, + + /** + * Method: getItemURL + * + * Retrieve full URL of item + * + * Parameters: + * path - Path relative to the module root. + */ + getItemURL: function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.getItemURL must be a string'; + } + if (this.storage.connected) { + path = this._cleanPath( this.makePath(path) ); + return this.storage.remote.href + path; + } else { + return undefined; + } + }, + + uuid: function() { + return Math.uuid(); + } + + }; + + /** + * Method: RS#scope + * + * Returns a new scoped to the given path. + * + * Parameters: + * path - Root path of new BaseClient. + * + * + * Example: + * (start code) + * + * var foo = remoteStorage.scope('/foo/'); + * + * // PUTs data "baz" to path /foo/bar + * foo.storeFile('text/plain', 'bar', 'baz'); + * + * var something = foo.scope('something/'); + * + * // GETs listing from path /foo/something/bla/ + * something.getListing('bla/'); + * + * (end code) + * + */ + RS.BaseClient._rs_init = function() { + RS.prototype.scope = function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.scope must be a string'; + } + + return new RS.BaseClient(this, path); + }; + }; + + /* e.g.: + remoteStorage.defineModule('locations', function(priv, pub) { + return { + exports: { + features: priv.scope('features/').defaultType('feature'), + collections: priv.scope('collections/').defaultType('feature-collection'); + } + }; + }); + */ + + maxAgeInvalid = function(maxAge) { + return typeof(maxAge) !== 'undefined' && typeof(maxAge) !== 'number'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/baseclient/types.js **/ +(function(global) { + + RemoteStorage.BaseClient.Types = { + // -> + uris: {}, + // -> + schemas: {}, + // -> + aliases: {}, + + declare: function(moduleName, alias, uri, schema) { + var fullAlias = moduleName + '/' + alias; + + if (schema.extends) { + var extendedAlias; + var parts = schema.extends.split('/'); + if (parts.length === 1) { + extendedAlias = moduleName + '/' + parts.shift(); + } else { + extendedAlias = parts.join('/'); + } + var extendedUri = this.uris[extendedAlias]; + if (! extendedUri) { + throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'"; + } + schema.extends = this.schemas[extendedUri]; + } + + this.uris[fullAlias] = uri; + this.aliases[uri] = fullAlias; + this.schemas[uri] = schema; + }, + + resolveAlias: function(alias) { + return this.uris[alias]; + }, + + getSchema: function(uri) { + return this.schemas[uri]; + }, + + inScope: function(moduleName) { + var ml = moduleName.length; + var schemas = {}; + for (var alias in this.uris) { + if (alias.substr(0, ml + 1) === moduleName + '/') { + var uri = this.uris[alias]; + schemas[uri] = this.schemas[uri]; + } + } + return schemas; + } + }; + + var SchemaNotFound = function(uri) { + var error = new Error("Schema not found: " + uri); + error.name = "SchemaNotFound"; + return error; + }; + + SchemaNotFound.prototype = Error.prototype; + + RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound; + /** + * Class: RemoteStorage.BaseClient + **/ + RemoteStorage.BaseClient.prototype.extend({ + /** + * Method: validate(object) + * + * validates an Object against the associated schema + * the context has to have a @context property + * + * Returns: + * A validate object giving you information about errors + **/ + validate: function(object) { + var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']); + if (schema) { + return tv4.validateResult(object, schema); + } else { + throw new SchemaNotFound(object['@context']); + } + }, + + // client.declareType(alias, schema); + // /* OR */ + // client.declareType(alias, uri, schema); + declareType: function(alias, uri, schema) { + if (! schema) { + schema = uri; + uri = this._defaultTypeURI(alias); + } + RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema); + }, + + _defaultTypeURI: function(alias) { + return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias; + }, + + _attachType: function(object, alias) { + object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias); + } + }); + + Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', { + configurable: true, + get: function() { + return RemoteStorage.BaseClient.Types.inScope(this.moduleName); + } + }); + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/caching.js **/ + /** + * Class: RemoteStorage.Caching + * + * Holds/manages caching configuration. + * + * Caching strategies: + * + * For each subtree, you can set the caching strategy to 'ALL', + * 'SEEN' (default), and 'FLUSH'. + * + * - 'ALL' means that once all outgoing changes have been pushed, sync + * will start retrieving nodes to cache pro-actively. If a local + * copy exists of everything, it will check on each sync whether + * the ETag of the root folder changed, and retrieve remote changes + * if they exist. + * - 'SEEN' does this only for documents and folders that have been either + * read from or written to at least once since connecting to the current + * remote backend, plus their parent/ancestor folders up to the root + * (to make tree-based sync possible). + * - 'FLUSH' will only cache outgoing changes, and forget them as soon as + * they have been saved to remote successfully. + * + **/ + +(function(global) { + var SETTINGS_KEY = "remotestorage:caching"; + + function containingFolder(path) { + if (path === '') { + return '/'; + } + if (! path) { + throw "Path not given!"; + } + + return path.replace(/\/+/g, '/').replace(/[^\/]+\/?$/, ''); + } + + RemoteStorage.Caching = function() { + this.reset(); + }; + + RemoteStorage.Caching.prototype = { + pendingActivations: [], + + /** + * Method: set + * + * Set the caching strategy for a given path explicitly. + * + * Not needed when using /. + * + * Parameters: + * path - Path to cache + * value - Caching strategy. One of 'ALL', 'SEEN', or 'FLUSH'. + * + * Example: + * (start code) + * remoteStorage.caching.set('/bookmarks/archive') + */ + set: function(path, value) { + if (typeof(path) !== 'string') { + throw new Error('path should be a string'); + } + if (typeof(value) === 'undefined') { + throw new Error("value should be 'FLUSH', 'SEEN', or 'ALL'"); + } + + this._rootPaths[path] = value; + + if (value === 'ALL') { + if (this.activateHandler) { + this.activateHandler(path); + } else { + this.pendingActivations.push(path); + } + } + }, + + /** + * Method: enable + * + * Enable caching for a given path. + * + * Uses caching strategy 'ALL'. + * + * Parameters: + * path - Path to enable caching for + */ + enable: function(path) { + this.set(path, 'ALL'); + }, + + /** + * Method: disable + * + * Disable caching for a given path. + * + * Uses caching strategy 'FLUSH' (meaning items are only cached until + * successfully pushed to the remote). + * + * Parameters: + * path - Path to disable caching for + */ + disable: function(path) { + this.set(path, 'FLUSH'); + }, + + /** + * Method: onActivate + * + * Set a callback for when caching is activated for a path. + * + * Parameters: + * callback - Callback function + */ + onActivate: function(cb) { + var i; + RemoteStorage.log('[Caching] Setting activate handler', cb, this.pendingActivations); + this.activateHandler = cb; + for (i=0; i 0; + }, + + collectDiffTasks: function() { + var num = 0; + + return this.local.forAllNodes(function(node) { + if (num > 100) { + return; + } + + if (this.isCorrupt(node)) { + RemoteStorage.log('[Sync] WARNING: corrupt node in local cache', node); + if (typeof(node) === 'object' && node.path) { + this.addTask(node.path); + num++; + } + } else if (this.needsFetch(node) && this.access.checkPathPermission(node.path, 'r')) { + this.addTask(node.path); + num++; + } else if (this.isDocumentNode(node) && this.needsPush(node) && + this.access.checkPathPermission(node.path, 'rw')) { + this.addTask(node.path); + num++; + } + }.bind(this)).then(function() { + return num; + }, function(err) { + throw err; + }); + }, + + inConflict: function(node) { + return (node.local && node.remote && + (node.remote.body !== undefined || node.remote.itemsMap)); + }, + + needsRefresh: function(node) { + if (node.common) { + if (!node.common.timestamp) { + return true; + } + return (this.now() - node.common.timestamp > syncInterval); + } + return false; + }, + + needsFetch: function(node) { + if (this.inConflict(node)) { + return true; + } + if (node.common && node.common.itemsMap === undefined && node.common.body === undefined) { + return true; + } + if (node.remote && node.remote.itemsMap === undefined && node.remote.body === undefined) { + return true; + } + return false; + }, + + needsPush: function(node) { + if (this.inConflict(node)) { + return false; + } + if (node.local && !node.push) { + return true; + } + }, + + needsRemotePut: function(node) { + return node.local && node.local.body; + }, + + needsRemoteDelete: function(node) { + return node.local && node.local.body === false; + }, + + getParentPath: function(path) { + var parts = path.match(/^(.*\/)([^\/]+\/?)$/); + + if (parts) { + return parts[1]; + } else { + throw new Error('Not a valid path: "'+path+'"'); + } + }, + + deleteChildPathsFromTasks: function() { + for (var path in this._tasks) { + paths = this.local._getInternals().pathsFromRoot(path); + + for (var i=1; i= numToAdd) { + return true; + } + } + } + return (numAdded >= numToAdd); + }, + + collectTasks: function(alsoCheckRefresh) { + if (this.hasTasks() || this.stopped) { + promise = promising(); + promise.fulfill(); + return promise; + } + + return this.collectDiffTasks().then(function(numDiffs) { + if (numDiffs || alsoCheckRefresh === false) { + promise = promising(); + promise.fulfill(); + return promise; + } else { + return this.collectRefreshTasks(); + } + }.bind(this), function(err) { + throw err; + }); + }, + + addTask: function(path, cb) { + if (!this._tasks[path]) { + this._tasks[path] = []; + } + if (typeof(cb) === 'function') { + this._tasks[path].push(cb); + } + }, + + /** + * Method: sync + **/ + sync: function() { + var promise = promising(); + this.done = false; + + if (!this.doTasks()) { + return this.collectTasks().then(function() { + try { + this.doTasks(); + } catch(e) { + RemoteStorage.log('[Sync] doTasks error', e); + } + }.bind(this), function(e) { + RemoteStorage.log('[Sync] Sync error', e); + throw new Error('Local cache unavailable'); + }); + } else { + return promising().fulfill(); + } + }, + }; + + /** + * Method: getSyncInterval + * + * Get the value of the sync interval when application is in the foreground + * + * Returns a number of milliseconds + * + */ + RemoteStorage.prototype.getSyncInterval = function() { + return syncInterval; + }; + + /** + * Method: setSyncInterval + * + * Set the value of the sync interval when application is in the foreground + * + * Parameters: + * interval - sync interval in milliseconds + * + */ + RemoteStorage.prototype.setSyncInterval = function(interval) { + if (typeof(interval) !== 'number') { + throw interval + " is not a valid sync interval"; + } + syncInterval = parseInt(interval, 10); + if (this._syncTimer) { + this.stopSync(); + this._syncTimer = setTimeout(this.syncCycle.bind(this), interval); + } + }; + + var SyncError = function(originalError) { + var msg = 'Sync failed: '; + if (typeof(originalError) === 'object' && 'message' in originalError) { + msg += originalError.message; + } else { + msg += originalError; + } + this.originalError = originalError; + Error.apply(this, [msg]); + }; + + SyncError.prototype = Object.create(Error.prototype); + + RemoteStorage.SyncError = SyncError; + + RemoteStorage.prototype.syncCycle = function() { + if (this.sync.stopped) { + return; + } + + this.sync.on('done', function() { + RemoteStorage.log('[Sync] Sync done. Setting timer to', this.getSyncInterval()); + if (!this.sync.stopped) { + if (this._syncTimer) { + clearTimeout(this._syncTimer); + } + this._syncTimer = setTimeout(this.sync.sync.bind(this.sync), this.getSyncInterval()); + } + }.bind(this)); + + this.sync.sync(); + }; + + RemoteStorage.prototype.stopSync = function() { + if (this.sync) { + RemoteStorage.log('[Sync] Stopping sync'); + this.sync.stopped = true; + } else { + // TODO When is this ever the case and what is syncStopped for then? + RemoteStorage.log('[Sync] Will instantiate sync stopped'); + this.syncStopped = true; + } + }; + + RemoteStorage.prototype.startSync = function() { + this.sync.stopped = false; + this.syncStopped = false; + this.sync.sync(); + }; + + var syncCycleCb; + + RemoteStorage.Sync._rs_init = function(remoteStorage) { + syncCycleCb = function() { + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + + if (!remoteStorage.sync) { + // Call this now that all other modules are also ready: + remoteStorage.sync = new RemoteStorage.Sync( + remoteStorage.local, remoteStorage.remote, remoteStorage.access, + remoteStorage.caching); + + if (remoteStorage.syncStopped) { + RemoteStorage.log('[Sync] Instantiating sync stopped'); + remoteStorage.sync.stopped = true; + delete remoteStorage.syncStopped; + } + } + + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + remoteStorage.syncCycle(); + }; + + remoteStorage.on('ready', syncCycleCb); + }; + + RemoteStorage.Sync._rs_cleanup = function(remoteStorage) { + remoteStorage.stopSync(); + remoteStorage.removeEventListener('ready', syncCycleCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/cachinglayer.js **/ +(function() { + /** + * Interface: cachinglayer + * + * This module defines functions that are mixed into remoteStorage.local when + * it is instantiated (currently one of indexeddb.js, localstorage.js, or + * inmemorystorage.js). + * + * All remoteStorage.local implementations should therefore implement + * this.getNodes, this.setNodes, and this.forAllNodes. The rest is blended in + * here to create a GPD (get/put/delete) interface which the BaseClient can + * talk to. + */ + + function isFolder(path) { + return path.substr(-1) === '/'; + } + + function isDocument(path) { + return path.substr(-1) !== '/'; + } + + function deepClone(obj) { + if (obj === undefined) { + return undefined; + } else { + return JSON.parse(JSON.stringify(obj)); + } + } + + function equal(obj1, obj2) { + return JSON.stringify(obj1) === JSON.stringify(obj2); + } + + function getLatest(node) { + if (typeof(node) !== 'object' || typeof(node.path) !== 'string') { + return; + } + if (isFolder(node.path)) { + if (node.local && node.local.itemsMap) { + return node.local; + } + if (node.common && node.common.itemsMap) { + return node.common; + } + } else { + if (node.local && node.local.body && node.local.contentType) { + return node.local; + } + if (node.common && node.common.body && node.common.contentType) { + return node.common; + } + // Migration code! Once all apps use at least this version of the lib, we + // can publish clean-up code that migrates over any old-format data, and + // stop supporting it. For now, new apps will support data in both + // formats, thanks to this: + if (node.body && node.contentType) { + return { + body: node.body, + contentType: node.contentType + }; + } + } + } + + function isOutdated(node, maxAge) { + return !node || !node.timestamp || + ((new Date().getTime()) - node.timestamp > maxAge); + } + + function pathsFromRoot(path) { + var paths = [path]; + var parts = path.replace(/\/$/, '').split('/'); + + while (parts.length > 1) { + parts.pop(); + paths.push(parts.join('/')+'/'); + } + return paths; + } + + function makeNode(path, timestamp) { + var node = { path: path, common: { timestamp: timestamp } }; + + if (isFolder(path)) { + node.common.itemsMap = {}; + } + return node; + } + + function updateFolderNodeWithItemName(node, itemName, timestamp) { + if (!node.common) { + node.common = { + timestamp: timestamp, + itemsMap: {} + }; + } + if (!node.common.itemsMap) { + node.common.itemsMap = {}; + } + if (!node.local) { + node.local = deepClone(node.common); + } + if (!node.local.itemsMap) { + node.local.itemsMap = node.common.itemsMap; + } + node.local.itemsMap[itemName] = true; + + return node; + } + + var methods = { + + get: function(path, maxAge) { + var promise = promising(); + + this.getNodes([path]).then(function(objs) { + var node = getLatest(objs[path]); + if ((typeof(maxAge) === 'number') && isOutdated(node, maxAge)) { + remoteStorage.sync.queueGetRequest(path, promise); + } + + if (node) { + promise.fulfill(200, node.body || node.itemsMap, node.contentType); + } else { + promise.fulfill(404); + } + }.bind(this), function(err) { + promise.reject(err); + }.bind(this)); + + return promise; + }, + + put: function(path, body, contentType) { + var paths = pathsFromRoot(path); + var now = new Date().getTime(); + + return this._updateNodes(paths, function(nodes) { + try { + for (var i=0; i 0) { + // This folder still contains other items, don't remove any further ancestors + break; + } + } + } + return nodes; + }); + }, + + flush: function(path) { + return this._getAllDescendentPaths(path).then(function(paths) { + return this.getNodes(paths); + }.bind(this)).then(function(nodes) { + for (var path in nodes) { + var node = nodes[path]; + + if (node && node.common && node.local) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: (node.local.body === false ? undefined : node.local.body), + newValue: (node.common.body === false ? undefined : node.common.body) + }); + } + nodes[path] = undefined; + } + return this.setNodes(nodes); + }.bind(this)); + }, + + fireInitial: function() { + this.forAllNodes(function(node) { + var latest; + if (isDocument(node.path)) { + latest = getLatest(node); + if (latest) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: undefined, + oldContentType: undefined, + newValue: latest.body, + newContentType: latest.contentType + }); + } + } + }.bind(this)); + }, + + onDiff: function(diffHandler) { + this.diffHandler = diffHandler; + }, + + migrate: function(node) { + if (typeof(node) === 'object' && !node.common) { + node.common = {}; + if (typeof(node.path) === 'string') { + if (node.path.substr(-1) === '/' && typeof(node.body) === 'object') { + node.common.itemsMap = node.body; + } + } else { + //save legacy content of document node as local version + if (!node.local) { + node.local = {}; + } + node.local.body = node.body; + node.local.contentType = node.contentType; + } + } + return node; + }, + + _updateNodes: function(paths, cb) { + var self = this; + + return this.getNodes(paths).then(function(nodes) { + var existingNodes = deepClone(nodes); + var changeEvents = []; + var node; + + nodes = cb(nodes); + + for (var path in nodes) { + node = nodes[path]; + if (equal(node, existingNodes[path])) { + delete nodes[path]; + } + else if(isDocument(path)) { + changeEvents.push({ + path: path, + origin: 'window', + oldValue: node.local.previousBody, + newValue: node.local.body === false ? undefined : node.local.body, + oldContentType: node.local.previousContentType, + newContentType: node.local.contentType + }); + delete node.local.previousBody; + delete node.local.previousContentType; + } + } + + return self.setNodes(nodes).then(function() { + self._emitChangeEvents(changeEvents); + return 200; + }); + }, + function(err) { + throw(err); + }); + }, + + _emitChangeEvents: function(events) { + for (var i=0; i node) map. + * + */ + + var RS = RemoteStorage; + + var DB_VERSION = 2; + + var DEFAULT_DB_NAME = 'remotestorage'; + var DEFAULT_DB; + + RS.IndexedDB = function(database) { + this.db = database || DEFAULT_DB; + + if (!this.db) { + RemoteStorage.log("[IndexedDB] Failed to open DB"); + return undefined; + } + + RS.cachingLayer(this); + RS.eventHandling(this, 'change'); + + this.getsRunning = 0; + this.putsRunning = 0; + }; + + RS.IndexedDB.prototype = { + getNodes: function(paths) { + var promise = promising(); + var transaction = this.db.transaction(['nodes'], 'readonly'); + var nodes = transaction.objectStore('nodes'); + var retrievedNodes = {}; + var startTime = new Date().getTime(); + + this.getsRunning++; + + for (var i=0; i 64 && nChr < 91 ? + nChr - 65 + : nChr > 96 && nChr < 123 ? + nChr - 71 + : nChr > 47 && nChr < 58 ? + nChr + 4 + : nChr === 43 ? + 62 + : nChr === 47 ? + 63 + : + 0; + } + + function base64DecToArr(sBase64, nBlocksSize) { + var + sB64Enc = sBase64.replace(/[^A-Za-z0-9\+\/]/g, ""), nInLen = sB64Enc.length, + nOutLen = nBlocksSize ? Math.ceil((nInLen * 3 + 1 >> 2) / nBlocksSize) * nBlocksSize : nInLen * 3 + 1 >> 2, taBytes = new Uint8Array(nOutLen); + + for (var nMod3, nMod4, nUint24 = 0, nOutIdx = 0, nInIdx = 0; nInIdx < nInLen; nInIdx++) { + nMod4 = nInIdx & 3; + nUint24 |= b64ToUint6(sB64Enc.charCodeAt(nInIdx)) << 18 - 6 * nMod4; + if (nMod4 === 3 || nInLen - nInIdx === 1) { + for (nMod3 = 0; nMod3 < 3 && nOutIdx < nOutLen; nMod3++, nOutIdx++) { + taBytes[nOutIdx] = nUint24 >>> (16 >>> nMod3 & 24) & 255; + } + nUint24 = 0; + } + } + return taBytes; + } + + function isBinary(node) { + return node.match(/charset=binary/); + } + + function isRemoteStorageKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX || + key.substr(0, CHANGES_PREFIX.length) === CHANGES_PREFIX; + } + + function isNodeKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX; + } + + RemoteStorage.LocalStorage.prototype = { + + getNodes: function(paths) { + var promise = promising(); + var nodes = {}; + + for(i=0; i + * instance under the module's name. So when defining a locations module, + * like in the example below, it would be accessible via + * `remoteStorage.locations`, which would in turn have a `features` and a + * `collections` property. + * + * The function receives a private and a public client, which are both + * instances of . In the following example, the + * scope of privateClient is `/locations` and the scope of publicClient is + * `/public/locations`. + * + * Example: + * (start code) + * RemoteStorage.defineModule('locations', function(privateClient, publicClient) { + * return { + * exports: { + * features: privateClient.scope('features/').defaultType('feature'), + * collections: privateClient.scope('collections/').defaultType('feature-collection') + * } + * }; + * }); + * (end code) + */ + + RemoteStorage.defineModule = function(moduleName, builder) { + RemoteStorage.MODULES[moduleName] = builder; + + Object.defineProperty(RemoteStorage.prototype, moduleName, { + configurable: true, + get: function() { + var instance = this._loadModule(moduleName); + Object.defineProperty(this, moduleName, { + value: instance + }); + return instance; + } + }); + + if (moduleName.indexOf('-') !== -1) { + var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) { + return s[1].toUpperCase(); + }); + Object.defineProperty(RemoteStorage.prototype, camelizedName, { + get: function() { + return this[moduleName]; + } + }); + } + }; + + RemoteStorage.prototype._loadModule = function(moduleName) { + var builder = RemoteStorage.MODULES[moduleName]; + if (builder) { + var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'), + new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/')); + return module.exports; + } else { + throw "Unknown module: " + moduleName; + } + }; + + RemoteStorage.prototype.defineModule = function(moduleName) { + console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!"); + RemoteStorage.defineModule.apply(RemoteStorage, arguments); + }; + +})(); + + +/** FILE: src/legacy.js **/ +(function() { + var util = { + getEventEmitter: function() { + var object = {}; + var args = Array.prototype.slice.call(arguments); + args.unshift(object); + RemoteStorage.eventHandling.apply(RemoteStorage, args); + object.emit = object._emit; + return object; + }, + + extend: function(target) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach(function(source) { + for (var key in source) { + target[key] = source[key]; + } + }); + return target; + }, + + asyncEach: function(array, callback) { + return this.asyncMap(array, callback). + then(function() { return array; }); + }, + + asyncMap: function(array, callback) { + var promise = promising(); + var n = array.length, i = 0; + var results = [], errors = []; + function oneDone() { + i++; + if (i === n) { + promise.fulfill(results, errors); + } + } + + array.forEach(function(item, index) { + var result; + try { + result = callback(item); + } catch(exc) { + oneDone(); + errors[index] = exc; + } + if (typeof(result) === 'object' && typeof(result.then) === 'function') { + result.then(function(res) { results[index] = res; oneDone(); }, + function(error) { errors[index] = res; oneDone(); }); + } else { + oneDone(); + results[index] = result; + } + }); + + return promise; + }, + + containingFolder: function(path) { + var folder = path.replace(/[^\/]+\/?$/, ''); + return folder === path ? null : folder; + }, + + isFolder: function(path) { + return path.substr(-1) === '/'; + }, + + baseName: function(path) { + var parts = path.split('/'); + if (util.isFolder(path)) { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + }, + + bindAll: function(object) { + for (var key in this) { + if (typeof(object[key]) === 'function') { + object[key] = object[key].bind(object); + } + } + } + }; + + Object.defineProperty(RemoteStorage.prototype, 'util', { + get: function() { + console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release."); + return util; + } + }); + +})(); + + +/** FILE: src/nodejs_ext.js **/ +(function(global) { + global.XMLHttpRequest = require('xhr2'); + + RemoteStorage.WireClient.readBinaryData = function(content, mimeType, callback) { + callback(content); + }; +}(global)); + +if(typeof(define) == 'function' && define.amd) define([], function() { return RemoteStorage }); else module.exports = RemoteStorage; diff --git a/release/0.10.0-beta2/remotestorage.amd.js b/release/0.10.0-beta2/remotestorage.amd.js new file mode 100644 index 000000000..1e0b3ce43 --- /dev/null +++ b/release/0.10.0-beta2/remotestorage.amd.js @@ -0,0 +1,7677 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ +define([], function() { + +/** FILE: lib/promising.js **/ +(function(global) { + function getPromise(builder) { + var promise; + + if(typeof(builder) === 'function') { + setTimeout(function() { + try { + builder(promise); + } catch(e) { + promise.reject(e); + } + }, 0); + } + + var consumers = [], success, result; + + function notifyConsumer(consumer) { + if(success) { + var nextValue; + if(consumer.fulfilled) { + try { + nextValue = [consumer.fulfilled.apply(null, result)]; + } catch(exc) { + consumer.promise.reject(exc); + return; + } + } else { + nextValue = result; + } + if(nextValue[0] && typeof(nextValue[0].then) === 'function') { + nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill.apply(null, nextValue); + } + } else { + if(consumer.rejected) { + var ret; + try { + ret = consumer.rejected.apply(null, result); + } catch(exc) { + consumer.promise.reject(exc); + return; + } + if(ret && typeof(ret.then) === 'function') { + ret.then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill(ret); + } + } else { + consumer.promise.reject.apply(null, result); + } + } + } + + function resolve(succ, res) { + if(result) { + console.error("WARNING: Can't resolve promise, already resolved!"); + return; + } + success = succ; + result = Array.prototype.slice.call(res); + setTimeout(function() { + var cl = consumers.length; + if(cl === 0 && (! success)) { + console.error("Possibly uncaught error: ", result, result[0] && result[0].stack); + } + for(var i=0;i. + */ + RemoteStorage.log = function() { + if (RemoteStorage._log) { + console.log.apply(console, arguments); + } + }; + + + RemoteStorage.prototype = { + /** + * Method: connect + * + * Connect to a remoteStorage server. + * + * Parameters: + * userAddress - The user address (user@host) to connect to. + * + * Discovers the webfinger profile of the given user address and + * initiates the OAuth dance. + * + * This method must be called *after* all required access has been claimed. + * + */ + connect: function(userAddress) { + this.setBackend('remotestorage'); + if (userAddress.indexOf('@') < 0) { + this._emit('error', new RemoteStorage.DiscoveryError("User adress doesn't contain an @.")); + return; + } + this.remote.configure(userAddress); + this._emit('connecting'); + + var discoveryTimeout = setTimeout(function() { + this._emit('error', new RemoteStorage.DiscoveryError("No storage information found at that user address.")); + }.bind(this), 5000); + + RemoteStorage.Discover(userAddress, function(href, storageApi, authURL) { + clearTimeout(discoveryTimeout); + if (!href) { + this._emit('error', new RemoteStorage.DiscoveryError("Failed to contact storage server.")); + return; + } + this._emit('authing'); + this.remote.configure(userAddress, href, storageApi); + if (! this.remote.connected) { + this.authorize(authURL); + } + }.bind(this)); + }, + + /** + * Method: disconnect + * + * "Disconnect" from remotestorage server to terminate current session. + * This method clears all stored settings and deletes the entire local + * cache. + */ + disconnect: function() { + if (this.remote) { + this.remote.configure(null, null, null, null); + } + this._setGPD({ + get: this._pendingGPD('get'), + put: this._pendingGPD('put'), + delete: this._pendingGPD('delete') + }); + var n = this._cleanups.length, i = 0; + + var oneDone = function() { + i++; + if (i >= n) { + this._init(); + RemoteStorage.log('Done cleaning up, emitting disconnected and disconnect events'); + this._emit('disconnected'); + } + }.bind(this); + + if (n > 0) { + this._cleanups.forEach(function(cleanup) { + var cleanupResult = cleanup(this); + if (typeof(cleanup) === 'object' && typeof(cleanup.then) === 'function') { + cleanupResult.then(oneDone); + } else { + oneDone(); + } + }.bind(this)); + } else { + oneDone(); + } + }, + + setBackend: function(what) { + this.backend = what; + if (this.localStorageAvailable()) { + if (what) { + localStorage['remotestorage:backend'] = what; + } else { + delete localStorage['remotestorage:backend']; + } + } + }, + + /** + * Method: onChange + * + * Add a "change" event handler to the given path. Whenever a "change" + * happens (as determined by the backend, such as e.g. + * ) and the affected path is equal to or below + * the given 'path', the given handler is called. + * + * You should usually not use this method directly, but instead use the + * "change" events provided by . + * + * Parameters: + * path - Absolute path to attach handler to. + * handler - Handler function. + */ + onChange: function(path, handler) { + if (! this._pathHandlers.change[path]) { + this._pathHandlers.change[path] = []; + } + this._pathHandlers.change[path].push(handler); + }, + + /** + * Method: enableLog + * + * Enable remoteStorage logging + */ + enableLog: function() { + RemoteStorage._log = true; + }, + + /** + * Method: disableLog + * + * Disable remoteStorage logging + */ + disableLog: function() { + RemoteStorage._log = false; + }, + + /** + * Method: log + * + * The same as . + */ + log: function() { + RemoteStorage.log.apply(RemoteStorage, arguments); + }, + + setApiKeys: function(type, keys) { + if (keys) { + this.apiKeys[type] = keys; + } else { + delete this.apiKeys[type]; + } + if (this.localStorageAvailable()) { + localStorage['remotestorage:api-keys'] = JSON.stringify(this.apiKeys); + } + }, + + /** + ** INITIALIZATION + **/ + + _init: function() { + var self = this, + readyFired = false; + + function fireReady() { + try { + if (!readyFired) { + self._emit('ready'); + readyFired = true; + } + } catch(e) { + console.error("'ready' failed: ", e, e.stack); + self._emit('error', e); + } + } + + this._loadFeatures(function(features) { + this.log('[RemoteStorage] All features loaded'); + this.local = features.local && new features.local(); + // this.remote set by WireClient._rs_init as lazy property on + // RS.prototype + + if (this.local && this.remote) { + this._setGPD(SyncedGetPutDelete, this); + this._bindChange(this.local); + } else if (this.remote) { + this._setGPD(this.remote, this.remote); + } + + if (this.remote) { + this.remote.on('connected', function(){ + fireReady(); + self._emit('connected'); + }); + this.remote.on('not-connected', function(){ + fireReady(); + self._emit('not-connected'); + }); + if (this.remote.connected) { + fireReady(); + self._emit('connected'); + } + } + + this._collectCleanupFunctions(); + + try { + this._allLoaded = true; + this._emit('features-loaded'); + } catch(exc) { + logError(exc); + this._emit('error', exc); + } + this._processPending(); + }.bind(this)); + }, + + _collectCleanupFunctions: function() { + for (var i=0; i < this.features.length; i++) { + var cleanup = this.features[i].cleanup; + if (typeof(cleanup) === 'function') { + this._cleanups.push(cleanup); + } + } + }, + + /** + ** FEATURE DETECTION + **/ + _loadFeatures: function(callback) { + var featureList = [ + 'WireClient', + 'I18n', + 'Dropbox', + 'GoogleDrive', + 'Access', + 'Caching', + 'Discover', + 'Authorize', + 'Widget', + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage', + 'Sync', + 'BaseClient', + 'Env' + ]; + var features = []; + var featuresDone = 0; + var self = this; + + function featureDone() { + featuresDone++; + if (featuresDone === featureList.length) { + setTimeout(function() { + features.caching = !!RemoteStorage.Caching; + features.sync = !!RemoteStorage.Sync; + [ + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage' + ].some(function(cachingLayer) { + if (features.some(function(feature) { return feature.name === cachingLayer; })) { + features.local = RemoteStorage[cachingLayer]; + return true; + } + }); + self.features = features; + callback(features); + }, 0); + } + } + + function featureInitialized(name) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialized."); + features.push({ + name : name, + init : RemoteStorage[name]._rs_init, + supported : true, + cleanup : RemoteStorage[name]._rs_cleanup + }); + featureDone(); + } + + function featureFailed(name, err) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialization failed ( "+err+")"); + featureDone(); + } + + function featureSupported(name, success) { + self.log("[RemoteStorage] [FEATURE "+name+"]" + success ? "":" not"+" supported"); + if (!success) { + featureDone(); + } + } + + function initFeature(name) { + var initResult; + try { + initResult = RemoteStorage[name]._rs_init(self); + } catch(e) { + featureFailed(name, e); + return; + } + if (typeof(initResult) === 'object' && typeof(initResult.then) === 'function') { + initResult.then( + function(){ featureInitialized(name); }, + function(err){ featureFailed(name, err); } + ); + } else { + featureInitialized(name); + } + } + + featureList.forEach(function(featureName) { + self.log("[RemoteStorage] [FEATURE " + featureName + "] initializing..."); + var impl = RemoteStorage[featureName]; + var supported; + + if (impl) { + supported = !impl._rs_supported || impl._rs_supported(); + + if (typeof supported === 'object') { + supported.then( + function(){ + featureSupported(featureName, true); + initFeature(featureName); + }, + function(){ + featureSupported(featureName, false); + } + ); + } + else if (typeof supported === 'boolean') { + featureSupported(featureName, supported); + if (supported) { + initFeature(featureName); + } + } + } else { + featureSupported(featureName, false); + } + }); + }, + + localStorageAvailable: function() { + try { + return !!global.localStorage; + } catch(error) { + return false; + } + }, + + /** + ** GET/PUT/DELETE INTERFACE HELPERS + **/ + + _setGPD: function(impl, context) { + function wrap(f) { + return function() { + return f.apply(context, arguments) + .then(emitUnauthorized.bind(this)); + }; + } + this.get = wrap(impl.get); + this.put = wrap(impl.put); + this.delete = wrap(impl.delete); + }, + + _pendingGPD: function(methodName) { + return function() { + var promise = promising(); + this._pending.push({ + method: methodName, + args: Array.prototype.slice.call(arguments), + promise: promise + }); + return promise; + }.bind(this); + }, + + _processPending: function() { + this._pending.forEach(function(pending) { + try { + this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject); + } catch(e) { + pending.promise.reject(e); + } + }.bind(this)); + this._pending = []; + }, + + /** + ** CHANGE EVENT HANDLING + **/ + + _bindChange: function(object) { + object.on('change', this._dispatchEvent.bind(this, 'change')); + }, + + _dispatchEvent: function(eventName, event) { + for (var path in this._pathHandlers[eventName]) { + var pl = path.length; + var self = this; + this._pathHandlers[eventName][path].forEach(function(handler) { + if (event.path.substr(0, pl) === path) { + var ev = {}; + for (var key in event) { ev[key] = event[key]; } + ev.relativePath = event.path.replace(new RegExp('^' + path), ''); + try { + handler(ev); + } catch(e) { + console.error("'change' handler failed: ", e, e.stack); + self._emit('error', e); + } + } + }); + } + } + }; + + /** + * Property: connected + * + * Boolean property indicating if remoteStorage is currently connected. + */ + Object.defineProperty(RemoteStorage.prototype, 'connected', { + get: function() { + return this.remote.connected; + } + }); + + /** + * Property: access + * + * Tracking claimed access scopes. A instance. + * + * + * Property: caching + * + * Caching settings. A instance. + * + * Not available in no-cache builds. + * + * + * Property: remote + * + * Access to the remote backend used. Usually a . + * + * + * Property: local + * + * Access to the local caching backend used. Usually either a + * or instance. + * + * Not available in no-cache builds. + */ + + global.RemoteStorage = RemoteStorage; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/eventhandling.js **/ +(function(global) { + /** + * Interface: eventhandling + */ + var methods = { + /** + * Method: addEventListener + * + * Install an event handler for the given event name + */ + addEventListener: function(eventName, handler) { + if (typeof(eventName) !== 'string') { + throw new Error('Argument eventName should be a string'); + } + if (typeof(handler) !== 'function') { + throw new Error('Argument handler should be a function'); + } + RemoteStorage.log('[Eventhandling] Adding event listener', eventName, handler); + this._validateEvent(eventName); + this._handlers[eventName].push(handler); + }, + + /** + * Method: removeEventListener + * + * Remove a previously installed event handler + */ + removeEventListener: function(eventName, handler) { + this._validateEvent(eventName); + var hl = this._handlers[eventName].length; + for (var i=0;i + **/ + methods.on = methods.addEventListener; + + /** + * Function: eventHandling + * + * Mixes event handling functionality into an object. + * + * The first parameter is always the object to be extended. + * All remaining parameter are expected to be strings, interpreted as valid event + * names. + * + * Example: + * (start code) + * var MyConstructor = function() { + * eventHandling(this, 'connected', 'disconnected'); + * + * this._emit('connected'); + * this._emit('disconnected'); + * // This would throw an exception: + * // this._emit('something-else'); + * }; + * + * var myObject = new MyConstructor(); + * myObject.on('connected', function() { console.log('connected'); }); + * myObject.on('disconnected', function() { console.log('disconnected'); }); + * // This would throw an exception as well: + * // myObject.on('something-else', function() {}); + * (end code) + */ + RemoteStorage.eventHandling = function(object) { + var eventNames = Array.prototype.slice.call(arguments, 1); + for (var key in methods) { + object[key] = methods[key]; + } + object._handlers = {}; + eventNames.forEach(function(eventName) { + object._addEvent(eventName); + }); + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/wireclient.js **/ +(function(global) { + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.WireClient + * + * WireClient Interface + * -------------------- + * + * This file exposes a get/put/delete interface on top of XMLHttpRequest. + * It requires to be configured with parameters about the remotestorage server to + * connect to. + * Each instance of WireClient is always associated with a single remotestorage + * server and access token. + * + * Usually the WireClient instance can be accessed via `remoteStorage.remote`. + * + * This is the get/put/delete interface: + * + * - #get() takes a path and optionally a ifNoneMatch option carrying a version + * string to check. It returns a promise that will be fulfilled with the HTTP + * response status, the response body, the MIME type as returned in the + * 'Content-Type' header and the current revision, as returned in the 'ETag' + * header. + * - #put() takes a path, the request body and a content type string. It also + * accepts the ifMatch and ifNoneMatch options, that map to the If-Match and + * If-None-Match headers respectively. See the remotestorage-01 specification + * for details on handling these headers. It returns a promise, fulfilled with + * the same values as the one for #get(). + * - #delete() takes a path and the ifMatch option as well. It returns a promise + * fulfilled with the same values as the one for #get(). + * + * In addition to this, the WireClient has some compatibility features to work with + * remotestorage 2012.04 compatible storages. For example it will cache revisions + * from folder listings in-memory and return them accordingly as the "revision" + * parameter in response to #get() requests. Similarly it will return 404 when it + * receives an empty folder listing, to mimic remotestorage-01 behavior. Note + * that it is not always possible to know the revision beforehand, hence it may + * be undefined at times (especially for caching-roots). + */ + + var hasLocalStorage; + var SETTINGS_KEY = "remotestorage:wireclient"; + + var API_2012 = 1, API_00 = 2, API_01 = 3, API_02 = 4, API_HEAD = 5; + + var STORAGE_APIS = { + 'draft-dejong-remotestorage-00': API_00, + 'draft-dejong-remotestorage-01': API_01, + 'draft-dejong-remotestorage-02': API_02, + 'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012 + }; + + var isArrayBufferView; + + if (typeof(ArrayBufferView) === 'function') { + isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); }; + } else { + var arrayBufferViews = [ + Int8Array, Uint8Array, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array + ]; + isArrayBufferView = function(object) { + for (var i=0;i<8;i++) { + if (object instanceof arrayBufferViews[i]) { + return true; + } + } + return false; + }; + } + + function addQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + if (str === '*') { + return '*'; + } + + return '"' + str + '"'; + } + + function stripQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + + return str.replace(/^["']|["']$/g, ''); + } + + function readBinaryData(content, mimeType, callback) { + var blob = new Blob([content], { type: mimeType }); + var reader = new FileReader(); + reader.addEventListener("loadend", function() { + callback(reader.result); // reader.result contains the contents of blob as a typed array + }); + reader.readAsArrayBuffer(blob); + } + + function cleanPath(path) { + return path.replace(/\/+/g, '/').split('/').map(encodeURIComponent).join('/'); + } + + function isFolder(path) { + return (path.substr(-1) === '/'); + } + + function isFolderDescription(body) { + return ((body['@context'] === 'http://remotestorage.io/spec/folder-description') + && (typeof(body['items']) === 'object')); + } + + function isSuccessStatus(status) { + return [201, 204, 304].indexOf(status) >= 0; + } + + function isErrorStatus(status) { + return [401, 403, 404, 412].indexOf(status) >= 0; + } + + var onErrorCb; + + /** + * Class : RemoteStorage.WireClient + **/ + RS.WireClient = function(rs) { + this.connected = false; + + /** + * Event: change + * never fired for some reason + * + * Event: connected + * fired when the wireclient connect method realizes that it is + * in posession of a token and a href + **/ + RS.eventHandling(this, 'change', 'connected', 'wire-busy', 'wire-done', 'not-connected'); + + onErrorCb = function(error){ + if (error instanceof RemoteStorage.Unauthorized) { + this.configure(undefined, undefined, undefined, null); + } + }.bind(this); + rs.on('error', onErrorCb); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + setTimeout(function() { + this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token); + }.bind(this), 0); + } + } + + this._revisionCache = {}; + + if (this.connected) { + setTimeout(this._emit.bind(this), 0, 'connected'); + } + }; + + RS.WireClient.REQUEST_TIMEOUT = 30000; + + RS.WireClient.prototype = { + /** + * Property: token + * + * Holds the bearer token of this WireClient, as obtained in the OAuth dance + * + * Example: + * (start code) + * + * remoteStorage.remote.token + * // -> 'DEADBEEF01==' + */ + + /** + * Property: href + * + * Holds the server's base URL, as obtained in the Webfinger discovery + * + * Example: + * (start code) + * + * remoteStorage.remote.href + * // -> 'https://storage.example.com/users/jblogg/' + */ + + /** + * Property: storageApi + * + * Holds the spec version the server claims to be compatible with + * + * Example: + * (start code) + * + * remoteStorage.remote.storageApi + * // -> 'draft-dejong-remotestorage-01' + */ + + _request: function(method, uri, token, headers, body, getEtag, fakeRevision) { + if ((method === 'PUT' || method === 'DELETE') && uri[uri.length - 1] === '/') { + throw "Don't " + method + " on directories!"; + } + + var promise = promising(); + var revision; + var reqType; + var self = this; + + headers['Authorization'] = 'Bearer ' + token; + + this._emit('wire-busy', { + method: method, + isFolder: isFolder(uri) + }); + + RS.WireClient.request(method, uri, { + body: body, + headers: headers + }, function(error, response) { + if (error) { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: false + }); + self.online = false; + promise.reject(error); + } else { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: true + }); + self.online = true; + if (isErrorStatus(response.status)) { + RemoteStorage.log('[WireClient] Error response status', response.status); + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = undefined; + } + promise.fulfill(response.status, undefined, undefined, revision); + } else if (isSuccessStatus(response.status) || + (response.status === 200 && method !== 'GET')) { + revision = stripQuotes(response.getResponseHeader('ETag')); + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, undefined, undefined, revision); + } else { + var mimeType = response.getResponseHeader('Content-Type'); + var body; + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = response.status === 200 ? fakeRevision : undefined; + } + + if ((!mimeType) || mimeType.match(/charset=binary/)) { + RS.WireClient.readBinaryData(response.response, mimeType, function(result) { + RemoteStorage.log('[WireClient] Successful request with unknown or binary mime-type', revision); + promise.fulfill(response.status, result, mimeType, revision); + }); + } else { + if (mimeType && mimeType.match(/^application\/json/)) { + body = JSON.parse(response.responseText); + } else { + body = response.responseText; + } + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, body, mimeType, revision); + } + } + } + }); + return promise; + }, + + configure: function(userAddress, href, storageApi, token) { + if (typeof(userAddress) !== 'undefined') { + this.userAddress = userAddress; + } + if (typeof(href) !== 'undefined') { + this.href = href; + } + if (typeof(storageApi) !== 'undefined') { + this.storageApi = storageApi; + } + if (typeof(token) !== 'undefined') { + this.token = token; + } + if (typeof(this.storageApi) !== 'undefined') { + this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD; + this.supportsRevs = this._storageApi >= API_00; + } + if (this.href && this.token) { + this.connected = true; + this.online = true; + this._emit('connected'); + } else { + this.connected = false; + } + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ + userAddress: this.userAddress, + href: this.href, + token: this.token, + storageApi: this.storageApi + }); + } + RS.WireClient.configureHooks.forEach(function(hook) { + hook.call(this); + }.bind(this)); + }, + + stopWaitingForToken: function() { + if (!this.connected) { + this._emit('not-connected'); + } + }, + + get: function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } else if (options.ifNoneMatch) { + var oldRev = this._revisionCache[path]; + } + var promise = this._request('GET', this.href + cleanPath(path), this.token, headers, + undefined, this.supportsRevs, this._revisionCache[path]); + if (isFolder(path)) { + return promise.then(function(status, body, contentType, revision) { + var itemsMap = {}; + + // New folder listing received + if (status === 200 && typeof(body) === 'object') { + // Empty folder listing of any spec + if (Object.keys(body).length === 0) { + status = 404; + } + // >= 02 spec + else if (isFolderDescription(body)) { + for (var item in body.items) { + this._revisionCache[path + item] = body.items[item].ETag; + } + itemsMap = body.items; + } + // < 02 spec + else { + Object.keys(body).forEach(function(key){ + this._revisionCache[path + key] = body[key]; + itemsMap[key] = {"ETag": body[key]}; + }.bind(this)); + } + return promising().fulfill(status, itemsMap, contentType, revision); + } else { + return promising().fulfill(status, body, contentType, revision); + } + }.bind(this)); + } else { + return promise; + } + }, + + put: function(path, body, contentType, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + if (!contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var headers = { 'Content-Type': contentType }; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } + return this._request('PUT', this.href + cleanPath(path), this.token, + headers, body, this.supportsRevs); + }, + + 'delete': function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + } + return this._request('DELETE', this.href + cleanPath(path), this.token, + headers, + undefined, this.supportsRevs); + } + }; + + // Shared cleanPath used by Dropbox + RS.WireClient.cleanPath = cleanPath; + + // Shared isArrayBufferView used by WireClient and Dropbox + RS.WireClient.isArrayBufferView = isArrayBufferView; + + RS.WireClient.readBinaryData = readBinaryData; + + // Shared request function used by WireClient, GoogleDrive and Dropbox. + RS.WireClient.request = function(method, url, options, callback) { + RemoteStorage.log('[WireClient]', method, url); + + callback = callback.bind(this); + + var timedOut = false; + + var timer = setTimeout(function() { + timedOut = true; + callback('timeout'); + }, RS.WireClient.REQUEST_TIMEOUT); + + var xhr = new XMLHttpRequest(); + xhr.open(method, url, true); + + if (options.responseType) { + xhr.responseType = options.responseType; + } + if (options.headers) { + for (var key in options.headers) { + xhr.setRequestHeader(key, options.headers[key]); + } + } + + xhr.onload = function() { + if (timedOut) { return; } + clearTimeout(timer); + callback(null, xhr); + }; + + xhr.onerror = function(error) { + if (timedOut) { return; } + clearTimeout(timer); + callback(error); + }; + + var body = options.body; + + if (typeof(body) === 'object') { + if (isArrayBufferView(body)) { + /* alright. */ + //FIXME empty block + } + else if (body instanceof ArrayBuffer) { + body = new Uint8Array(body); + } else { + body = JSON.stringify(body); + } + } + xhr.send(body); + }; + + Object.defineProperty(RemoteStorage.WireClient.prototype, 'storageType', { + get: function() { + if (this.storageApi) { + var spec = this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/); + return spec ? spec[1] : '2012.04'; + } + } + }); + + RS.WireClient.configureHooks = []; + + RS.WireClient._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + remoteStorage.remote = new RS.WireClient(remoteStorage); + this.online = true; + }; + + RS.WireClient._rs_supported = function() { + return !! global.XMLHttpRequest; + }; + + RS.WireClient._rs_cleanup = function(remoteStorage){ + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + remoteStorage.removeEventListener('error', onErrorCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/discover.js **/ +(function(global) { + + // feature detection flags + var haveXMLHttpRequest, hasLocalStorage; + // used to store settings in localStorage + var SETTINGS_KEY = 'remotestorage:discover'; + // cache loaded from localStorage + var cachedInfo = {}; + + /** + * Class: RemoteStorage.Discover + * + * This class deals with the webfinger lookup + * + * Arguments: + * userAddress - user@host + * callback - gets called with href of the storage, the type and the authURL + * Example: + * (start code) + * + * (end code) + **/ + + RemoteStorage.Discover = function(userAddress, callback) { + if (userAddress in cachedInfo) { + var info = cachedInfo[userAddress]; + callback(info.href, info.type, info.authURL); + return; + } + var hostname = userAddress.split('@')[1]; + var params = '?resource=' + encodeURIComponent('acct:' + userAddress); + var urls = [ + 'https://' + hostname + '/.well-known/webfinger' + params, + 'https://' + hostname + '/.well-known/host-meta.json' + params, + 'http://' + hostname + '/.well-known/webfinger' + params, + 'http://' + hostname + '/.well-known/host-meta.json' + params + ]; + + function tryOne() { + var xhr = new XMLHttpRequest(); + var url = urls.shift(); + if (!url) { return callback(); } + RemoteStorage.log('[Discover] Trying URL', url); + xhr.open('GET', url, true); + xhr.onabort = xhr.onerror = function() { + console.error("webfinger error", arguments, '(', url, ')'); + tryOne(); + }; + xhr.onload = function() { + if (xhr.status !== 200) { return tryOne(); } + var profile; + + try { + profile = JSON.parse(xhr.responseText); + } catch(e) { + RemoteStorage.log("[Discover] Failed to parse profile ", xhr.responseText, e); + tryOne(); + return; + } + + if (!profile.links) { + RemoteStorage.log("[Discover] Profile has no links section ", JSON.stringify(profile)); + tryOne(); + return; + } + + var link; + profile.links.forEach(function(l) { + if (l.rel === 'remotestorage') { + link = l; + } else if (l.rel === 'remoteStorage' && !link) { + link = l; + } + }); + RemoteStorage.log('[Discover] Got profile', profile, 'and link', link); + if (link) { + var authURL = link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'] + || link.properties['auth-endpoint'], + storageType = link.properties['http://remotestorage.io/spec/version'] + || link.type; + cachedInfo[userAddress] = { href: link.href, type: storageType, authURL: authURL }; + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo }); + } + callback(link.href, storageType, authURL); + } else { + tryOne(); + } + }; + xhr.send(); + } + tryOne(); + }; + + RemoteStorage.Discover._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + cachedInfo = settings.cache; + } + } + }; + + RemoteStorage.Discover._rs_supported = function() { + haveXMLHttpRequest = !! global.XMLHttpRequest; + return haveXMLHttpRequest; + }; + + RemoteStorage.Discover._rs_cleanup = function() { + if (hasLocalStorage) { + delete localStorage[SETTINGS_KEY]; + } + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/authorize.js **/ +(function(global) { + + function extractParams() { + //FF already decodes the URL fragment in document.location.hash, so use this instead: + var location = RemoteStorage.Authorize.getLocation(), + hashPos = location.href.indexOf('#'), + hash; + if (hashPos === -1) { return; } + hash = location.href.substring(hashPos+1); + // if hash is not of the form #key=val&key=val, it's probably not for us + if (hash.indexOf('=') === -1) { return; } + return hash.split('&').reduce(function(m, kvs) { + var kv = kvs.split('='); + m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]); + return m; + }, {}); + } + + RemoteStorage.Authorize = function(authURL, scope, redirectUri, clientId) { + RemoteStorage.log('[Authorize] authURL = ', authURL); + + var url = authURL; + url += authURL.indexOf('?') > 0 ? '&' : '?'; + url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, '')); + url += '&scope=' + encodeURIComponent(scope); + url += '&client_id=' + encodeURIComponent(clientId); + url += '&response_type=token'; + RemoteStorage.Authorize.setLocation(url); + }; + + RemoteStorage.prototype.authorize = function(authURL) { + this.access.setStorageType(this.remote.storageType); + var scope = this.access.scopeParameter; + + var redirectUri = String(RemoteStorage.Authorize.getLocation()); + var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0]; + + RemoteStorage.Authorize(authURL, scope, redirectUri, clientId); + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.getLocation = function () { + return global.document.location; + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.setLocation = function (location) { + if (typeof location === 'string') { + global.document.location.href = location; + } else if (typeof location === 'object') { + global.document.location = location; + } else { + throw "Invalid location " + location; + } + }; + + RemoteStorage.Authorize._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + + var onFeaturesLoaded; + RemoteStorage.Authorize._rs_init = function(remoteStorage) { + + onFeaturesLoaded = function () { + var authParamsUsed = false; + if (params) { + if (params.error) { + throw "Authorization server errored: " + params.error; + } + if (params.access_token) { + remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token); + authParamsUsed = true; + } + if (params.remotestorage) { + remoteStorage.connect(params.remotestorage); + authParamsUsed = true; + } + } + if (!authParamsUsed) { + remoteStorage.remote.stopWaitingForToken(); + } + }; + var params = extractParams(), + location; + if (params) { + location = RemoteStorage.Authorize.getLocation(); + location.hash = ''; + } + remoteStorage.on('features-loaded', onFeaturesLoaded); + }; + + RemoteStorage.Authorize._rs_cleanup = function(remoteStorage) { + remoteStorage.removeEventListener('features-loaded', onFeaturesLoaded); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/access.js **/ +(function(global) { + + var SETTINGS_KEY = "remotestorage:access"; + + /** + * Class: RemoteStorage.Access + * + * Keeps track of claimed access and scopes. + */ + RemoteStorage.Access = function() { + this.reset(); + }; + + RemoteStorage.Access.prototype = { + + /** + * Method: claim + * + * Claim access on a given scope with given mode. + * + * Parameters: + * scope - An access scope, such as "contacts" or "calendar". + * mode - Access mode to use. Either "r" or "rw". + * + * Example: + * (start code) + * remoteStorage.access.claim('contacts', 'r'); + * remoteStorage.access.claim('pictures', 'rw'); + */ + claim: function(scope, mode) { + if (typeof(scope) !== 'string' || scope.indexOf('/') !== -1 || scope.length === 0) { + throw new Error('Scope should be a non-empty string without forward slashes'); + } + if (!mode.match(/^rw?$/)) { + throw new Error('Mode should be either \'r\' or \'rw\''); + } + this._adjustRootPaths(scope); + this.scopeModeMap[scope] = mode; + }, + + get: function(scope) { + return this.scopeModeMap[scope]; + }, + + remove: function(scope) { + var savedMap = {}; + var name; + for (name in this.scopeModeMap) { + savedMap[name] = this.scopeModeMap[name]; + } + this.reset(); + delete savedMap[scope]; + for (name in savedMap) { + this.set(name, savedMap[name]); + } + }, + + /** + * Verify permission for a given scope. + */ + checkPermission: function(scope, mode) { + var actualMode = this.get(scope); + return actualMode && (mode === 'r' || actualMode === 'rw'); + }, + + /** + * Verify permission for a given path. + */ + checkPathPermission: function(path, mode) { + if (this.checkPermission('*', mode)) { + return true; + } + return !!this.checkPermission(this._getModuleName(path), mode); + }, + + reset: function() { + this.rootPaths = []; + this.scopeModeMap = {}; + }, + + /** + * Return the module name for a given path. + */ + _getModuleName: function(path) { + if (path[0] !== '/') { + throw new Error('Path should start with a slash'); + } + var moduleMatch = path.replace(/^\/public/, '').match(/^\/([^\/]*)\//); + return moduleMatch ? moduleMatch[1] : '*'; + }, + + _adjustRootPaths: function(newScope) { + if ('*' in this.scopeModeMap || newScope === '*') { + this.rootPaths = ['/']; + } else if (! (newScope in this.scopeModeMap)) { + this.rootPaths.push('/' + newScope + '/'); + this.rootPaths.push('/public/' + newScope + '/'); + } + }, + + _scopeNameForParameter: function(scope) { + if (scope.name === '*' && this.storageType) { + if (this.storageType === '2012.04') { + return ''; + } else if (this.storageType.match(/remotestorage-0[01]/)) { + return 'root'; + } + } + return scope.name; + }, + + setStorageType: function(type) { + this.storageType = type; + } + }; + + /** + * Property: scopes + * + * Holds an array of claimed scopes in the form + * > { name: "", mode: "" } + */ + Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', { + get: function() { + return Object.keys(this.scopeModeMap).map(function(key) { + return { name: key, mode: this.scopeModeMap[key] }; + }.bind(this)); + } + }); + + Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', { + get: function() { + return this.scopes.map(function(scope) { + return this._scopeNameForParameter(scope) + ':' + scope.mode; + }.bind(this)).join(' '); + } + }); + + // Documented in src/remotestorage.js + Object.defineProperty(RemoteStorage.prototype, 'access', { + get: function() { + var access = new RemoteStorage.Access(); + Object.defineProperty(this, 'access', { + value: access + }); + return access; + }, + configurable: true + }); + + RemoteStorage.Access._rs_init = function() {}; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/env.js **/ +(function(pMode) { + + var mode = pMode, + env = {}, + isBackground = false; + + + RemoteStorage.Env = function() { + return env; + }; + + RemoteStorage.Env.isBrowser = function () { + return mode === "browser"; + }; + + RemoteStorage.Env.isNode = function () { + return mode === "node"; + }; + + RemoteStorage.Env.goBackground = function () { + isBackground = true; + RemoteStorage.Env._emit("background"); + }; + + RemoteStorage.Env.goForeground = function () { + isBackground = false; + RemoteStorage.Env._emit("foreground"); + }; + + RemoteStorage.Env._rs_init = function(remoteStorage) { + RemoteStorage.eventHandling(RemoteStorage.Env, "background", "foreground"); + + if ( mode === 'browser') { + if ( typeof(document.hidden) !== "undefined" ) { + env.hiddenProperty = "hidden"; + env.visibilityChangeEvent = "visibilitychange"; + } else if ( typeof(document.mozHidden) !== "undefined" ) { + env.hiddenProperty = "mozHidden"; + env.visibilityChangeEvent = "mozvisibilitychange"; + } else if ( typeof(document.msHidden) !== "undefined" ) { + env.hiddenProperty = "msHidden"; + env.visibilityChangeEvent = "msvisibilitychange"; + } else if ( typeof(document.webkitHidden) !== "undefined" ) { + env.hiddenProperty = "webkitHidden"; + env.visibilityChangeEvent = "webkitvisibilitychange"; + } + document.addEventListener(env.visibilityChangeEvent, function () { + if ( document[env.hiddenProperty] ) { + RemoteStorage.Env.goBackground(); + } else { + RemoteStorage.Env.goForeground(); + } + }, false); + } + }; + + RemoteStorage.Env._rs_cleanup = function(remoteStorage) { + }; + +})(typeof(window) !== 'undefined' ? 'browser' : 'node'); + + +/** FILE: src/i18n.js **/ +(function() { + /** + * Class: RemoteStorage.I18n + * + * TODO add documentation + **/ + + "use strict"; + + var dictionary = { + "view_info": 'This app allows you to use your own storage. Learn more!', + "view_connect": "Connect remote storage", + "view_connecting": "Connecting %s", + "view_offline": "Offline", + "view_error_occured": "Sorry! An error occured.", + "view_confirm_reset": "Are you sure you want to reset everything? This will clear your local data and reload the page.", + "view_get_me_out": "Get me out of here!", + "view_error_plz_report": 'If this problem persists, please let us know!', + "view_unauthorized": "Unauthorized! Click here to reconnect." + }; + + RemoteStorage.I18n = { + + translate: function() { + var str = arguments[0], + params = Array.prototype.splice.call(arguments, 1); + + if (typeof dictionary[str] !== "string") { + throw "Unknown translation string: " + str; + } else { + str = dictionary[str]; + } + return (str.replace(/%s/g, function(){ return params.shift(); })); + }, + + getDictionary: function() { + return dictionary; + }, + + setDictionary: function(newDictionary) { + dictionary = newDictionary; + } + + }; +})(); + + +/** FILE: src/assets.js **/ +/** THIS FILE WAS GENERATED BY build/compile-assets.js. DO NOT CHANGE IT MANUALLY, BUT INSTEAD CHANGE THE ASSETS IN assets/. **/ +RemoteStorage.Assets = { + + connectIcon: '', + disconnectIcon: '', + dropbox: '', + googledrive: '', + remoteStorageIcon: '', + remoteStorageIconError: '', + remoteStorageIconOffline: '', + syncIcon: '', + widget: '
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ', + widgetCss: '/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}' +}; + + +/** FILE: src/widget.js **/ +(function(window) { + + var hasLocalStorage; + var LS_STATE_KEY = "remotestorage:widget:state"; + // states allowed to immediately jump into after a reload. + var VALID_ENTRY_STATES = { + initial: true, + connected: true, + offline: true + }; + + function stateSetter(widget, state) { + RemoteStorage.log('[Widget] Producing stateSetter for', state); + return function() { + RemoteStorage.log('[Widget] Setting state', state, arguments); + if (hasLocalStorage) { + localStorage[LS_STATE_KEY] = state; + } + if (widget.view) { + if (widget.rs.remote) { + widget.view.setUserAddress(widget.rs.remote.userAddress); + } + widget.view.setState(state, arguments); + } else { + widget._rememberedState = state; + } + }; + } + + function errorsHandler(widget) { + return function(error) { + if (error instanceof RemoteStorage.DiscoveryError) { + console.error('Discovery failed', error, '"' + error.message + '"'); + widget.view.setState('initial', [error.message]); + } else if (error instanceof RemoteStorage.SyncError) { + widget.view.setState('offline', []); + } else if (error instanceof RemoteStorage.Unauthorized) { + widget.view.setState('unauthorized'); + } else { + RemoteStorage.log('[Widget] Unknown error'); + widget.view.setState('error', [error]); + } + }; + } + + function flashFor(evt) { + if (evt.method === 'GET' && evt.isFolder) { + return false; + } + return true; + } + + /** + * Class: RemoteStorage.Widget + * + * The widget controller that communicates with the view and listens to + * its remoteStorage instance. + * + * While listening to the events emitted by its remoteStorage it sets + * corresponding states of the view. + * + * - connected -> connected + * - disconnected -> initial + * - connecting -> authing + * - authing -> authing + * - wire-busy -> busy + * - wire-done -> connected + * - error -> one of initial, offline, unauthorized, or error + **/ + RemoteStorage.Widget = function(remoteStorage) { + var self = this; + var requestsToFlashFor = 0; + + // setting event listeners on rs events to put + // the widget into corresponding states + this.rs = remoteStorage; + this.rs.remote.on('connected', stateSetter(this, 'connected')); + this.rs.on('disconnected', stateSetter(this, 'initial')); + this.rs.on('connecting', stateSetter(this, 'authing')); + this.rs.on('authing', stateSetter(this, 'authing')); + this.rs.on('error', errorsHandler(this)); + + if (this.rs.remote) { + this.rs.remote.on('wire-busy', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor++; + stateSetter(self, 'busy')(); + } + }); + + this.rs.remote.on('wire-done', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor--; + if (requestsToFlashFor <= 0) { + stateSetter(self, 'connected')(); + } + } + }); + } + + if (hasLocalStorage) { + var state = localStorage[LS_STATE_KEY]; + if (state && VALID_ENTRY_STATES[state]) { + this._rememberedState = state; + } + } + }; + + RemoteStorage.Widget.prototype = { + + /** + * Method: display + * + * Displays the widget via the view.display method + * + * Parameters: + * + * domID + **/ + display: function(domID) { + if (! this.view) { + this.setView(new RemoteStorage.Widget.View(this.rs)); + } + this.view.display.apply(this.view, arguments); + return this; + }, + + linkWidgetToSync: function() { + if (typeof(this.rs.sync) === 'object' && typeof(this.rs.sync.sync) === 'function') { + this.view.on('sync', this.rs.sync.sync.bind(this.rs.sync)); + } else { + RemoteStorage.log('[Widget] typeof this.rs.sync check fail', this.rs.sync); + setTimeout(this.linkWidgetToSync.bind(this), 1000); + } + }, + + /** + * Method: setView(view) + * + * Sets the view and initializes event listeners to react on + * widget (widget.view) events + **/ + setView: function(view) { + this.view = view; + this.view.on('connect', function(options) { + if (typeof(options) === 'string') { + // options is simply a useraddress + this.rs.connect(options); + } else if (options.special) { + this.rs[options.special].connect(options); + } + }.bind(this)); + this.view.on('disconnect', this.rs.disconnect.bind(this.rs)); + this.linkWidgetToSync(); + try { + this.view.on('reset', function(){ + var location = RemoteStorage.Authorize.getLocation(); + this.rs.on('disconnected', location.reload.bind(location)); + this.rs.disconnect(); + }.bind(this)); + } catch(e) { + if (e.message && e.message.match(/Unknown event/)) { + // ignored. (the 0.7 widget-view interface didn't have a 'reset' event) + } else { + throw e; + } + } + + if (this._rememberedState) { + setTimeout(stateSetter(this, this._rememberedState), 0); + delete this._rememberedState; + } + } + }; + + /** + * Method: displayWidget + * + * Same as + **/ + RemoteStorage.prototype.displayWidget = function(domID) { + return this.widget.display(domID); + }; + + RemoteStorage.Widget._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (! remoteStorage.widget) { + remoteStorage.widget = new RemoteStorage.Widget(remoteStorage); + } + }; + + RemoteStorage.Widget._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/view.js **/ +(function(window){ + + var t = RemoteStorage.I18n.translate; + + var cEl = function(){ + return document.createElement.apply(document, arguments); + }; + + function gCl(parent, className) { + return parent.getElementsByClassName(className)[0]; + } + + function gTl(parent, tagName) { + return parent.getElementsByTagName(tagName)[0]; + } + + function removeClass(el, className) { + return el.classList.remove(className); + } + + function addClass(el, className) { + return el.classList.add(className); + } + + function stopPropagation(event) { + if (typeof(event.stopPropagation) === 'function') { + event.stopPropagation(); + } else { + event.cancelBubble = true; + } + } + + function setupButton(parent, className, iconName, eventListener) { + var element = gCl(parent, className); + if (typeof iconName !== 'undefined') { + var img = gTl(element, 'img'); + (img || element).src = RemoteStorage.Assets[iconName]; + } + element.addEventListener('click', eventListener); + return element; + } + + /** + * Class: RemoteStorage.Widget.View + * + * Controls the visible widget + * + * States: + * + * initial - not connected + * authing - in auth flow + * connected - connected to remote storage, not syncing at the moment + * busy - connected, syncing at the moment + * offline - connected, but no network connectivity + * error - connected, but sync error happened + * unauthorized - connected, but request returned 401 + **/ + RemoteStorage.Widget.View = function(remoteStorage) { + this.rs = remoteStorage; + if (typeof(document) === 'undefined') { + throw "Widget not supported"; + } + RemoteStorage.eventHandling(this, + 'connect', + 'disconnect', + 'sync', + 'display', + 'reset'); + + // Re-binding the event so they can be called from the window + for (var event in this.events){ + this.events[event] = this.events[event].bind(this); + } + + this.hideBubbleOnBodyClick = function(event) { + for (var p = event.target; p !== document.body; p = p.parentElement) { + if (p.id === 'remotestorage-widget') { + return; + } + } + this.hideBubble(); + }.bind(this); + }; + + RemoteStorage.Widget.View.prototype = { + + connectGdrive: function() { + this._emit('connect', { special: 'googledrive' }); + }, + + connectDropbox: function(){ + this._emit('connect', { special: 'dropbox'}); + }, + + /** + * Method: setState + * + * Call the function that applies the state to the widget + * + * Parameters: + * + * state + * args + **/ + setState: function(state, args) { + RemoteStorage.log('[View] widget.view.setState(',state,',',args,');'); + var s = this.states[state]; + if (typeof(s) === 'undefined') { + throw new Error("Bad State assigned to view: " + state); + } + s.apply(this, args); + }, + + /** + * Method: setUserAddres + * + * Set user address of the input field + **/ + setUserAddress: function(addr) { + this.userAddress = addr || ''; + + var el; + if (this.div && (el = gTl(this.div, 'form').userAddress)) { + el.value = this.userAddress; + } + }, + + /** + * Method: toggleBubble + * + * Show the bubble when hidden and the other way around + **/ + toggleBubble: function(event) { + if (this.bubble.className.search('rs-hidden') < 0) { + this.hideBubble(event); + } else { + this.showBubble(event); + } + }, + + /** + * Method: hideBubble + * + * Hide the bubble + **/ + hideBubble: function(){ + addClass(this.bubble, 'rs-hidden'); + document.body.removeEventListener('click', this.hideBubbleOnBodyClick); + }, + + /** + * Method: showBubble + * + * Show the bubble + **/ + showBubble: function(event){ + removeClass(this.bubble, 'rs-hidden'); + if (typeof(event) !== 'undefined') { + stopPropagation(event); + } + document.body.addEventListener('click', this.hideBubbleOnBodyClick); + gTl(this.bubble,'form').userAddress.focus(); + }, + + /** + * Method: display + * + * Draw the widget inside of the dom element with the id domID + * + * Parameters: + * + * domID + * + * Returns: + * + * The widget div + **/ + display: function(domID) { + if (typeof this.div !== 'undefined') { + return this.div; + } + + var element = cEl('div'); + var style = cEl('style'); + style.innerHTML = RemoteStorage.Assets.widgetCss; + + element.id = "remotestorage-widget"; + + element.innerHTML = RemoteStorage.Assets.widget; + + element.appendChild(style); + if (domID) { + var parent = document.getElementById(domID); + if (! parent) { + throw "Failed to find target DOM element with id=\"" + domID + "\""; + } + parent.appendChild(element); + } else { + document.body.appendChild(element); + } + + // Sync button + setupButton(element, 'rs-sync', 'syncIcon', this.events.sync); + + // Disconnect button + setupButton(element, 'rs-disconnect', 'disconnectIcon', this.events.disconnect); + + // Get me out of here + setupButton(element, 'remotestorage-reset', undefined, this.events.reset); + + // Connect button + var cb = setupButton(element, 'connect', 'connectIcon', this.events.connect); + + // Input + this.form = gTl(element, 'form'); + var el = this.form.userAddress; + el.addEventListener('keyup', function(event) { + if (event.target.value) { + cb.removeAttribute('disabled'); + } else { + cb.setAttribute('disabled','disabled'); + } + }); + if (this.userAddress) { + el.value = this.userAddress; + } + + // The cube + this.cube = setupButton(element, 'rs-cube', 'remoteStorageIcon', this.toggleBubble.bind(this)); + + // Google Drive and Dropbox icons + setupButton(element, 'rs-dropbox', 'dropbox', this.connectDropbox.bind(this)); + setupButton(element, 'rs-googledrive', 'googledrive', this.connectGdrive.bind(this)); + + var bubbleDontCatch = { INPUT: true, BUTTON: true, IMG: true }; + var eventListener = function(event) { + if (! bubbleDontCatch[event.target.tagName] && ! (this.div.classList.contains('remotestorage-state-unauthorized') )) { + this.showBubble(event); + } + }.bind(this); + this.bubble = setupButton(element, 'rs-bubble', undefined, eventListener); + + this.hideBubble(); + + this.div = element; + + this.states.initial.call(this); + this.events.display.call(this); + return this.div; + }, + + _renderTranslatedInitialContent: function() { + gCl(this.div, 'rs-status-text').innerHTML = t("view_connect"); + gCl(this.div, 'remotestorage-reset').innerHTML = t("view_get_me_out"); + gCl(this.div, 'rs-error-plz-report').innerHTML = t("view_error_plz_report"); + gCl(this.div, 'remotestorage-unauthorized').innerHTML = t("view_unauthorized"); + }, + + states: { + initial: function(message) { + var cube = this.cube; + var info = message || t("view_info"); + + this._renderTranslatedInitialContent(); + + if (message) { + cube.src = RemoteStorage.Assets.remoteStorageIconError; + removeClass(this.cube, 'remotestorage-loading'); + this.showBubble(); + + // Show the red error cube for 5 seconds, then show the normal orange one again + setTimeout(function(){ + cube.src = RemoteStorage.Assets.remoteStorageIcon; + },5000); + } else { + this.hideBubble(); + } + this.div.className = "remotestorage-state-initial"; + + // Google Drive and Dropbox icons + var backends = 1; + if (this._activateBackend('dropbox')) { backends += 1; } + if (this._activateBackend('googledrive')) { backends += 1; } + gCl(this.div, 'rs-bubble-text').style.paddingRight = backends*32+8+'px'; + + // If address not empty connect button enabled + var cb = gCl(this.div, 'connect'); + if (this.form.userAddress.value) { + cb.removeAttribute('disabled'); + } + + var infoEl = gCl(this.div, 'rs-info-msg'); + infoEl.innerHTML = info; + + if (message) { + infoEl.classList.add('remotestorage-error-info'); + } else { + infoEl.classList.remove('remotestorage-error-info'); + } + }, + + authing: function() { + this.div.removeEventListener('click', this.events.connect); + this.div.className = "remotestorage-state-authing"; + gCl(this.div, 'rs-status-text').innerHTML = t("view_connecting", this.userAddress); + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone, when is that neccesary + }, + + connected: function() { + this.div.className = "remotestorage-state-connected"; + gCl(this.div, 'userAddress').innerHTML = this.userAddress; + this.cube.src = RemoteStorage.Assets.remoteStorageIcon; + removeClass(this.cube, 'remotestorage-loading'); + var icons = { + googledrive: gCl(this.div, 'rs-googledrive'), + dropbox: gCl(this.div, 'rs-dropbox') + }; + icons.googledrive.style.display = icons.dropbox.style.display = 'none'; + if (icons[this.rs.backend]) { + icons[this.rs.backend].style.display = 'inline-block'; + gCl(this.div, 'rs-bubble-text').style.paddingRight = 2*32+8+'px'; + } else { + gCl(this.div, 'rs-bubble-text').style.paddingRight = 32+8+'px'; + } + }, + + busy: function() { + this.div.className = "remotestorage-state-busy"; + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone when is that neccesary + this.hideBubble(); + }, + + offline: function() { + this.div.className = "remotestorage-state-offline"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconOffline; + gCl(this.div, 'rs-status-text').innerHTML = t("view_offline"); + }, + + error: function(err) { + var errorMsg = err; + this.div.className = "remotestorage-state-error"; + + gCl(this.div, 'rs-bubble-text').innerHTML = ''+t('view_error_occured')+''; + //FIXME I don't know what an DOMError is and my browser doesn't know too(how to handle this?) + if (err instanceof Error /*|| err instanceof DOMError*/) { + errorMsg = err.message + '\n\n' + + err.stack; + } + gCl(this.div, 'rs-error-msg').textContent = errorMsg; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + }, + + unauthorized: function() { + this.div.className = "remotestorage-state-unauthorized"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + this.div.addEventListener('click', this.events.connect); + } + }, + + events: { + /** + * Event: connect + * + * Emitted when the connect button is clicked + **/ + connect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('connect', gTl(this.div, 'form').userAddress.value); + }, + + /** + * Event: sync + * + * Emitted when the sync button is clicked + **/ + sync: function(event) { + stopPropagation(event); + event.preventDefault(); + + this._emit('sync'); + }, + + /** + * Event: disconnect + * + * Emitted when the disconnect button is clicked + **/ + disconnect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('disconnect'); + }, + + /** + * Event: reset + * + * Emitted after crash triggers disconnect + **/ + reset: function(event){ + event.preventDefault(); + var result = window.confirm(t('view_confirm_reset')); + if (result){ + this._emit('reset'); + } + }, + + /** + * Event: display + * + * Emitted when finished displaying the widget + **/ + display : function(event) { + if (event) { + event.preventDefault(); + } + this._emit('display'); + } + }, + + _activateBackend: function activateBackend(backendName) { + var className = 'rs-' + backendName; + if (this.rs.apiKeys[backendName]) { + gCl(this.div, className).style.display = 'inline-block'; + return true; + } else { + gCl(this.div, className).style.display = 'none'; + return false; + } + } + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: lib/tv4.js **/ +/** +Author: Geraint Luff and others +Year: 2013 + +This code is released into the "public domain" by its author(s). Anybody may use, alter and distribute the code without restriction. The author makes no guarantees, and takes no liability of any kind for use of this code. + +If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory. +**/ + +(function (global) { +var ValidatorContext = function (parent, collectMultiple) { + this.missing = []; + this.schemas = parent ? Object.create(parent.schemas) : {}; + this.collectMultiple = collectMultiple; + this.errors = []; + this.handleError = collectMultiple ? this.collectError : this.returnError; +}; +ValidatorContext.prototype.returnError = function (error) { + return error; +}; +ValidatorContext.prototype.collectError = function (error) { + if (error) { + this.errors.push(error); + } + return null; +} +ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) { + for (var i = startIndex; i < this.errors.length; i++) { + this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath); + } + return this; +} + +ValidatorContext.prototype.getSchema = function (url) { + if (this.schemas[url] != undefined) { + var schema = this.schemas[url]; + return schema; + } + var baseUrl = url; + var fragment = ""; + if (url.indexOf('#') != -1) { + fragment = url.substring(url.indexOf("#") + 1); + baseUrl = url.substring(0, url.indexOf("#")); + } + if (this.schemas[baseUrl] != undefined) { + var schema = this.schemas[baseUrl]; + var pointerPath = decodeURIComponent(fragment); + if (pointerPath == "") { + return schema; + } else if (pointerPath.charAt(0) != "/") { + return undefined; + } + var parts = pointerPath.split("/").slice(1); + for (var i = 0; i < parts.length; i++) { + var component = parts[i].replace("~1", "/").replace("~0", "~"); + if (schema[component] == undefined) { + schema = undefined; + break; + } + schema = schema[component]; + } + if (schema != undefined) { + return schema; + } + } + if (this.missing[baseUrl] == undefined) { + this.missing.push(baseUrl); + this.missing[baseUrl] = baseUrl; + } +}; +ValidatorContext.prototype.addSchema = function (url, schema) { + var map = {}; + map[url] = schema; + normSchema(schema, url); + searchForTrustedSchemas(map, schema, url); + for (var key in map) { + this.schemas[key] = map[key]; + } + return map; +}; + +ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) { + if (schema['$ref'] != undefined) { + schema = this.getSchema(schema['$ref']); + if (!schema) { + return null; + } + } + + var errorCount = this.errors.length; + var error = this.validateBasic(data, schema) + || this.validateNumeric(data, schema) + || this.validateString(data, schema) + || this.validateArray(data, schema) + || this.validateObject(data, schema) + || this.validateCombinations(data, schema) + || null + if (error || errorCount != this.errors.length) { + while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) { + var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null; + var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null; + if (error) { + error = error.prefixWith(dataPart, schemaPart); + } + this.prefixErrors(errorCount, dataPart, schemaPart); + } + } + + return this.handleError(error); +} + +function recursiveCompare(A, B) { + if (A === B) { + return true; + } + if (typeof A == "object" && typeof B == "object") { + if (Array.isArray(A) != Array.isArray(B)) { + return false; + } else if (Array.isArray(A)) { + if (A.length != B.length) { + return false + } + for (var i = 0; i < A.length; i++) { + if (!recursiveCompare(A[i], B[i])) { + return false; + } + } + } else { + for (var key in A) { + if (B[key] === undefined && A[key] !== undefined) { + return false; + } + } + for (var key in B) { + if (A[key] === undefined && B[key] !== undefined) { + return false; + } + } + for (var key in A) { + if (!recursiveCompare(A[key], B[key])) { + return false; + } + } + } + return true; + } + return false; +} + +ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) { + var error; + if (error = this.validateType(data, schema)) { + return error.prefixWith(null, "type"); + } + if (error = this.validateEnum(data, schema)) { + return error.prefixWith(null, "type"); + } + return null; +} + +ValidatorContext.prototype.validateType = function validateType(data, schema) { + if (schema.type == undefined) { + return null; + } + var dataType = typeof data; + if (data == null) { + dataType = "null"; + } else if (Array.isArray(data)) { + dataType = "array"; + } + var allowedTypes = schema.type; + if (typeof allowedTypes != "object") { + allowedTypes = [allowedTypes]; + } + + for (var i = 0; i < allowedTypes.length; i++) { + var type = allowedTypes[i]; + if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) { + return null; + } + } + return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType); +} + +ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) { + if (schema["enum"] == undefined) { + return null; + } + for (var i = 0; i < schema["enum"].length; i++) { + var enumVal = schema["enum"][i]; + if (recursiveCompare(data, enumVal)) { + return null; + } + } + return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data)); +} +ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) { + return this.validateMultipleOf(data, schema) + || this.validateMinMax(data, schema) + || null; +} + +ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) { + var multipleOf = schema.multipleOf || schema.divisibleBy; + if (multipleOf == undefined) { + return null; + } + if (typeof data == "number") { + if (data%multipleOf != 0) { + return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf); + } + } + return null; +} + +ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) { + if (typeof data != "number") { + return null; + } + if (schema.minimum != undefined) { + if (data < schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum"); + } + if (schema.exclusiveMinimum && data == schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum"); + } + } + if (schema.maximum != undefined) { + if (data > schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum"); + } + if (schema.exclusiveMaximum && data == schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum"); + } + } + return null; +} +ValidatorContext.prototype.validateString = function validateString(data, schema) { + return this.validateStringLength(data, schema) + || this.validateStringPattern(data, schema) + || null; +} + +ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) { + if (typeof data != "string") { + return null; + } + if (schema.minLength != undefined) { + if (data.length < schema.minLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength"); + } + } + if (schema.maxLength != undefined) { + if (data.length > schema.maxLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength"); + } + } + return null; +} + +ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) { + if (typeof data != "string" || schema.pattern == undefined) { + return null; + } + var regexp = new RegExp(schema.pattern); + if (!regexp.test(data)) { + return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern"); + } + return null; +} +ValidatorContext.prototype.validateArray = function validateArray(data, schema) { + if (!Array.isArray(data)) { + return null; + } + return this.validateArrayLength(data, schema) + || this.validateArrayUniqueItems(data, schema) + || this.validateArrayItems(data, schema) + || null; +} + +ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) { + if (schema.minItems != undefined) { + if (data.length < schema.minItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxItems != undefined) { + if (data.length > schema.maxItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) { + if (schema.uniqueItems) { + for (var i = 0; i < data.length; i++) { + for (var j = i + 1; j < data.length; j++) { + if (recursiveCompare(data[i], data[j])) { + var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems"); + if (this.handleError(error)) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) { + if (schema.items == undefined) { + return null; + } + var error; + if (Array.isArray(schema.items)) { + for (var i = 0; i < data.length; i++) { + if (i < schema.items.length) { + if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) { + return error; + } + } else if (schema.additionalItems != undefined) { + if (typeof schema.additionalItems == "boolean") { + if (!schema.additionalItems) { + error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems"); + if (this.handleError(error)) { + return error; + } + } + } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) { + return error; + } + } + } + } else { + for (var i = 0; i < data.length; i++) { + if (error = this.validateAll(data[i], schema.items, [i], ["items"])) { + return error; + } + } + } + return null; +} +ValidatorContext.prototype.validateObject = function validateObject(data, schema) { + if (typeof data != "object" || data == null || Array.isArray(data)) { + return null; + } + return this.validateObjectMinMaxProperties(data, schema) + || this.validateObjectRequiredProperties(data, schema) + || this.validateObjectProperties(data, schema) + || this.validateObjectDependencies(data, schema) + || null; +} + +ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) { + var keys = Object.keys(data); + if (schema.minProperties != undefined) { + if (keys.length < schema.minProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxProperties != undefined) { + if (keys.length > schema.maxProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) { + if (schema.required != undefined) { + for (var i = 0; i < schema.required.length; i++) { + var key = schema.required[i]; + if (data[key] === undefined) { + var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required"); + if (this.handleError(error)) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) { + var error; + for (var key in data) { + var foundMatch = false; + if (schema.properties != undefined && schema.properties[key] != undefined) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) { + return error; + } + } + if (schema.patternProperties != undefined) { + for (var patternKey in schema.patternProperties) { + var regexp = new RegExp(patternKey); + if (regexp.test(key)) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) { + return error; + } + } + } + } + if (!foundMatch && schema.additionalProperties != undefined) { + if (typeof schema.additionalProperties == "boolean") { + if (!schema.additionalProperties) { + error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties"); + if (this.handleError(error)) { + return error; + } + } + } else { + if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) { + var error; + if (schema.dependencies != undefined) { + for (var depKey in schema.dependencies) { + if (data[depKey] !== undefined) { + var dep = schema.dependencies[depKey]; + if (typeof dep == "string") { + if (data[dep] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } else if (Array.isArray(dep)) { + for (var i = 0; i < dep.length; i++) { + var requiredKey = dep[i]; + if (data[requiredKey] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } + } else { + if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) { + var error; + return this.validateAllOf(data, schema) + || this.validateAnyOf(data, schema) + || this.validateOneOf(data, schema) + || this.validateNot(data, schema) + || null; +} + +ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) { + if (schema.allOf == undefined) { + return null; + } + var error; + for (var i = 0; i < schema.allOf.length; i++) { + var subSchema = schema.allOf[i]; + if (error = this.validateAll(data, subSchema, [], ["allOf", i])) { + return error; + } + } + return null; +} + +ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) { + if (schema.anyOf == undefined) { + return null; + } + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.anyOf.length; i++) { + var subSchema = schema.anyOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["anyOf", i]); + + if (error == null && errorCount == this.errors.length) { + this.errors = this.errors.slice(0, startErrorCount); + return null; + } + if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf")); + } + } + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors); +} + +ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) { + if (schema.oneOf == undefined) { + return null; + } + var validIndex = null; + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.oneOf.length; i++) { + var subSchema = schema.oneOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["oneOf", i]); + + if (error == null && errorCount == this.errors.length) { + if (validIndex == null) { + validIndex = i; + } else { + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf"); + } + } else if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf")); + } + } + if (validIndex == null) { + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors); + } else { + this.errors = this.errors.slice(0, startErrorCount); + } + return null; +} + +ValidatorContext.prototype.validateNot = function validateNot(data, schema) { + if (schema.not == undefined) { + return null; + } + var oldErrorCount = this.errors.length; + var error = this.validateAll(data, schema.not); + var notErrors = this.errors.slice(oldErrorCount); + this.errors = this.errors.slice(0, oldErrorCount); + if (error == null && notErrors.length == 0) { + return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not") + } + return null; +} + +// parseURI() and resolveUrl() are from https://gist.github.com/1088850 +// - released as public domain by author ("Yaffle") - see comments on gist + +function parseURI(url) { + var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/); + // authority = '//' + user + ':' + pass '@' + hostname + ':' port + return (m ? { + href : m[0] || '', + protocol : m[1] || '', + authority: m[2] || '', + host : m[3] || '', + hostname : m[4] || '', + port : m[5] || '', + pathname : m[6] || '', + search : m[7] || '', + hash : m[8] || '' + } : null); +} + +function resolveUrl(base, href) {// RFC 3986 + + function removeDotSegments(input) { + var output = []; + input.replace(/^(\.\.?(\/|$))+/, '') + .replace(/\/(\.(\/|$))+/g, '/') + .replace(/\/\.\.$/, '/../') + .replace(/\/?[^\/]*/g, function (p) { + if (p === '/..') { + output.pop(); + } else { + output.push(p); + } + }); + return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : ''); + } + + href = parseURI(href || ''); + base = parseURI(base || ''); + + return !href || !base ? null : (href.protocol || base.protocol) + + (href.protocol || href.authority ? href.authority : base.authority) + + removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) + + (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) + + href.hash; +} + +function normSchema(schema, baseUri) { + if (baseUri == undefined) { + baseUri = schema.id; + } else if (typeof schema.id == "string") { + baseUri = resolveUrl(baseUri, schema.id); + schema.id = baseUri; + } + if (typeof schema == "object") { + if (Array.isArray(schema)) { + for (var i = 0; i < schema.length; i++) { + normSchema(schema[i], baseUri); + } + } else if (typeof schema['$ref'] == "string") { + schema['$ref'] = resolveUrl(baseUri, schema['$ref']); + } else { + for (var key in schema) { + if (key != "enum") { + normSchema(schema[key], baseUri); + } + } + } + } +} + +var ErrorCodes = { + INVALID_TYPE: 0, + ENUM_MISMATCH: 1, + ANY_OF_MISSING: 10, + ONE_OF_MISSING: 11, + ONE_OF_MULTIPLE: 12, + NOT_PASSED: 13, + // Numeric errors + NUMBER_MULTIPLE_OF: 100, + NUMBER_MINIMUM: 101, + NUMBER_MINIMUM_EXCLUSIVE: 102, + NUMBER_MAXIMUM: 103, + NUMBER_MAXIMUM_EXCLUSIVE: 104, + // String errors + STRING_LENGTH_SHORT: 200, + STRING_LENGTH_LONG: 201, + STRING_PATTERN: 202, + // Object errors + OBJECT_PROPERTIES_MINIMUM: 300, + OBJECT_PROPERTIES_MAXIMUM: 301, + OBJECT_REQUIRED: 302, + OBJECT_ADDITIONAL_PROPERTIES: 303, + OBJECT_DEPENDENCY_KEY: 304, + // Array errors + ARRAY_LENGTH_SHORT: 400, + ARRAY_LENGTH_LONG: 401, + ARRAY_UNIQUE: 402, + ARRAY_ADDITIONAL_ITEMS: 403 +}; + +function ValidationError(code, message, dataPath, schemaPath, subErrors) { + if (code == undefined) { + throw new Error ("No code supplied for error: "+ message); + } + this.code = code; + this.message = message; + this.dataPath = dataPath ? dataPath : ""; + this.schemaPath = schemaPath ? schemaPath : ""; + this.subErrors = subErrors ? subErrors : null; +} +ValidationError.prototype = { + prefixWith: function (dataPrefix, schemaPrefix) { + if (dataPrefix != null) { + dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1"); + this.dataPath = "/" + dataPrefix + this.dataPath; + } + if (schemaPrefix != null) { + schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1"); + this.schemaPath = "/" + schemaPrefix + this.schemaPath; + } + if (this.subErrors != null) { + for (var i = 0; i < this.subErrors.length; i++) { + this.subErrors[i].prefixWith(dataPrefix, schemaPrefix); + } + } + return this; + } +}; + +function searchForTrustedSchemas(map, schema, url) { + if (typeof schema.id == "string") { + if (schema.id.substring(0, url.length) == url) { + var remainder = schema.id.substring(url.length); + if ((url.length > 0 && url.charAt(url.length - 1) == "/") + || remainder.charAt(0) == "#" + || remainder.charAt(0) == "?") { + if (map[schema.id] == undefined) { + map[schema.id] = schema; + } + } + } + } + if (typeof schema == "object") { + for (var key in schema) { + if (key != "enum" && typeof schema[key] == "object") { + searchForTrustedSchemas(map, schema[key], url); + } + } + } + return map; +} + +var globalContext = new ValidatorContext(); + +var publicApi = { + validate: function (data, schema) { + var context = new ValidatorContext(globalContext); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + var added = context.addSchema("", schema); + var error = context.validateAll(data, schema); + this.error = error; + this.missing = context.missing; + this.valid = (error == null); + return this.valid; + }, + validateResult: function () { + var result = {}; + this.validate.apply(result, arguments); + return result; + }, + validateMultiple: function (data, schema) { + var context = new ValidatorContext(globalContext, true); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + context.addSchema("", schema); + context.validateAll(data, schema); + var result = {}; + result.errors = context.errors; + result.missing = context.missing; + result.valid = (result.errors.length == 0); + return result; + }, + addSchema: function (url, schema) { + return globalContext.addSchema(url, schema); + }, + getSchema: function (url) { + return globalContext.getSchema(url); + }, + missing: [], + error: null, + normSchema: normSchema, + resolveUrl: resolveUrl, + errorCodes: ErrorCodes +}; + +global.tv4 = publicApi; + +})(typeof(window) != 'undefined' ? window : global); + + + +/** FILE: lib/Math.uuid.js **/ +/*! + Math.uuid.js (v1.4) + http://www.broofa.com + mailto:robert@broofa.com + + Copyright (c) 2010 Robert Kieffer + Dual licensed under the MIT and GPL licenses. + + ******** + + Changes within remoteStorage.js: + 2012-10-31: + - added AMD wrapper + - moved extensions for Math object into exported object. +*/ + +/* + * Generate a random uuid. + * + * USAGE: Math.uuid(length, radix) + * length - the desired number of characters + * radix - the number of allowable values for each character. + * + * EXAMPLES: + * // No arguments - returns RFC4122, version 4 ID + * >>> Math.uuid() + * "92329D39-6F5C-4520-ABFC-AAB64544E172" + * + * // One argument - returns ID of the specified length + * >>> Math.uuid(15) // 15 character ID (default base=62) + * "VcydxgltxrVZSTV" + * + * // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62) + * >>> Math.uuid(8, 2) // 8 character ID (base=2) + * "01001010" + * >>> Math.uuid(8, 10) // 8 character ID (base=10) + * "47473046" + * >>> Math.uuid(8, 16) // 8 character ID (base=16) + * "098F4D35" + */ + // Private array of chars to use + var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); + +Math.uuid = function (len, radix) { + var chars = CHARS, uuid = [], i; + radix = radix || chars.length; + + if (len) { + // Compact form + for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix]; + } else { + // rfc4122, version 4 form + var r; + + // rfc4122 requires these characters + uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; + uuid[14] = '4'; + + // Fill in random data. At i==19 set the high bits of clock sequence as + // per rfc4122, sec. 4.1.5 + for (i = 0; i < 36; i++) { + if (!uuid[i]) { + r = 0 | Math.random()*16; + uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; + } + } + } + + return uuid.join(''); +}; + + +/** FILE: src/baseclient.js **/ +(function(global) { + + function deprecate(thing, replacement) { + console.log('WARNING: ' + thing + ' is deprecated. Use ' + + replacement + ' instead.'); + } + + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.BaseClient + * + * Provides a high-level interface to access data below a given root path. + * + * A BaseClient deals with three types of data: folders, objects and files. + * + * returns a list of all items within a folder, or undefined + * if a 404 is encountered. Items that end with a forward slash ("/") are + * child folders. + * + * / operate on JSON objects. Each object has a type. + * + * / operates on files. Each file has a MIME type. + * + * operates on either objects or files (but not folders, folders are + * created and removed implictly). + */ + RS.BaseClient = function(storage, base) { + if (base[base.length - 1] !== '/') { + throw "Not a folder: " + base; + } + + if (base === '/') { + // allow absolute and relative paths for the root scope. + this.makePath = function(path) { + return (path[0] === '/' ? '' : '/') + path; + }; + } + + /** + * Property: storage + * + * The instance this operates on. + */ + this.storage = storage; + + /** + * Property: base + * + * Base path this operates on. + * + * For the module's privateClient this would be //, for the + * corresponding publicClient /public//. + */ + this.base = base; + + var parts = this.base.split('/'); + if (parts.length > 2) { + this.moduleName = parts[1]; + } else { + this.moduleName = 'root'; + } + + /** + * Event: change + * + * Emitted when a node changes + * + * Arguments: + * event - Event object containing information about the changed node + * + * (start code) + * { + * path: path, // Path of the changed node + * origin: 'window', 'local', or 'remote' // emitted by user action within the app, local data store, or remote sync + * oldValue: oldBody, // Old body of the changed node (undefined if creation) + * newValue: newBody // New body of the changed node (undefined if deletion) + * } + * (end code) + **/ + + RS.eventHandling(this, 'change'); + this.on = this.on.bind(this); + storage.onChange(this.base, this._fireChange.bind(this)); + }; + + RS.BaseClient.prototype = { + + extend: function(object) { + for (var key in object) { + this[key] = object[key]; + } + return this; + }, + + /** + * Method: scope + * + * Returns a new operating on a subpath of the current path. + */ + scope: function(path) { + return new RS.BaseClient(this.storage, this.makePath(path)); + }, + + // folder operations + + /** + * Method: getListing + * + * Get a list of child nodes below a given path. + * + * The callback semantics of getListing are identical to those of getObject. + * + * Parameters: + * path - The path to query. It MUST end with a forward slash. + * maxAge - (optional) Maximum age of cached listing in + * milliseconds + * + * Returns: + * + * A promise for an object, representing child nodes. + * + * Keys ending in a forward slash represent *folder nodes*, while all + * other keys represent *data nodes*. + * + * For spec versions <= 01, the data node information will contain only + * the item's ETag. For later spec versions, it will also contain the + * content type and -length of the item. + * + * Example: + * (start code) + * client.getListing('').then(function(listing) { + * listing.forEach(function(item) { + * console.log(item); + * }); + * }); + * (end code) + */ + getListing: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getListing must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then( + function(status, body) { + return (status === 404) ? undefined : body; + } + ); + }, + + /** + * Method: getAll + * + * Get all objects directly below a given path. + * + * Parameters: + * path - path to the folder + * maxAge - (optional) Maximum age of cached objects in + * milliseconds + * + * Returns: + * a promise for an object in the form { path : object, ... } + * + * Example: + * (start code) + * client.getAll('').then(function(objects) { + * for (var key in objects) { + * console.log('- ' + key + ': ', objects[key]); + * } + * }); + * (end code) + */ + getAll: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getAll must be undefined or a number'); + } + + return this.storage.get(this.makePath(path), maxAge).then(function(status, body) { + if (status === 404) { return; } + if (typeof(body) === 'object') { + var promise = promising(); + var count = Object.keys(body).length, i = 0; + if (count === 0) { + // treat this like 404. it probably means a folder listing that + // has changes that haven't been pushed out yet. + return; + } + for (var key in body) { + this.storage.get(this.makePath(path + key), maxAge). + then(function(status, b) { + body[this.key] = b; + i++; + if (i === count) { promise.fulfill(body); } + }.bind({ key: key })); + } + return promise; + } + }.bind(this)); + }, + + // file operations + + /** + * Method: getFile + * + * Get the file at the given path. A file is raw data, as opposed to + * a JSON object (use for that). + * + * Except for the return value structure, getFile works exactly like + * getObject. + * + * Parameters: + * path - see getObject + * + * Returns: + * A promise for an object: + * + * mimeType - String representing the MIME Type of the document. + * data - Raw data of the document (either a string or an ArrayBuffer) + * + * Example: + * (start code) + * // Display an image: + * client.getFile('path/to/some/image').then(function(file) { + * var blob = new Blob([file.data], { type: file.mimeType }); + * var targetElement = document.findElementById('my-image-element'); + * targetElement.src = window.URL.createObjectURL(blob); + * }); + * (end code) + */ + getFile: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getFile must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getFile must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + return { + data: body, + mimeType: mimeType, + revision: revision // (this is new) + }; + }); + }, + + /** + * Method: storeFile + * + * Store raw data at a given path. + * + * Parameters: + * mimeType - MIME media type of the data being stored + * path - path relative to the module root. MAY NOT end in a forward slash. + * data - string, ArrayBuffer or ArrayBufferView of raw data to store + * + * The given mimeType will later be returned, when retrieving the data + * using . + * + * Example (UTF-8 data): + * (start code) + * client.storeFile('text/html', 'index.html', '

Hello World!

'); + * (end code) + * + * Example (Binary data): + * (start code) + * // MARKUP: + * + * // CODE: + * var input = document.getElementById('file-input'); + * var file = input.files[0]; + * var fileReader = new FileReader(); + * + * fileReader.onload = function() { + * client.storeFile(file.type, file.name, fileReader.result); + * }; + * + * fileReader.readAsArrayBuffer(file); + * (end code) + * + */ + storeFile: function(mimeType, path, body) { + if (typeof(mimeType) !== 'string') { + return promising().reject('Argument \'mimeType\' of baseClient.storeFile must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeFile must be a string'); + } + if (typeof(body) !== 'string' && typeof(body) !== 'object') { + return promising().reject('Argument \'body\' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView'); + } + + var self = this; + return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + self.makePath(path) + ") failed with status: " + status; + } + }); + }, + + // object operations + + /** + * Method: getObject + * + * Get a JSON object from given path. + * + * Parameters: + * path - relative path from the module root (without leading slash) + * + * Returns: + * A promise for the object. + * + * Example: + * (start code) + * client.getObject('/path/to/object'). + * then(function(object) { + * // object is either an object or null + * }); + * (end code) + */ + getObject: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getObject must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getObject must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + if (typeof(body) === 'object') { + return body; + } else if (typeof(body) !== 'undefined' && status === 200) { + throw "Not an object: " + this.makePath(path); + } + }); + }, + + /** + * Method: storeObject + * + * Store object at given path. Triggers synchronization. + * + * Parameters: + * + * type - unique type of this object within this module. See description below. + * path - path relative to the module root. + * object - an object to be saved to the given node. It must be serializable as JSON. + * + * Returns: + * A promise to store the object. The promise fails with a ValidationError, when validations fail. + * + * + * What about the type?: + * + * A great thing about having data on the web, is to be able to link to + * it and rearrange it to fit the current circumstances. To facilitate + * that, eventually you need to know how the data at hand is structured. + * For documents on the web, this is usually done via a MIME type. The + * MIME type of JSON objects however, is always application/json. + * To add that extra layer of "knowing what this object is", remoteStorage + * aims to use . + * A first step in that direction, is to add a *@context attribute* to all + * JSON data put into remoteStorage. + * Now that is what the *type* is for. + * + * Within remoteStorage.js, @context values are built using three components: + * http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness + * the module name - module names should be unique as well + * the type given here - naming this particular kind of object within this module + * + * In retrospect that means, that whenever you introduce a new "type" in calls to + * storeObject, you should make sure that once your code is in the wild, future + * versions of the code are compatible with the same JSON structure. + * + * How to define types?: + * + * See for examples. + */ + storeObject: function(typeAlias, path, object) { + if (typeof(typeAlias) !== 'string') { + return promising().reject('Argument \'typeAlias\' of baseClient.storeObject must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeObject must be a string'); + } + if (typeof(object) !== 'object') { + return promising().reject('Argument \'object\' of baseClient.storeObject must be an object'); + } + this._attachType(object, typeAlias); + try { + var validationResult = this.validate(object); + if (! validationResult.valid) { + return promising(function(p) { p.reject(validationResult); }); + } + } catch(exc) { + if (! (exc instanceof RS.BaseClient.Types.SchemaNotFound)) { + return promising().reject(exc); + } + } + return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status; + } + }.bind(this)); + }, + + // generic operations + + /** + * Method: remove + * + * Remove node at given path from storage. Triggers synchronization. + * + * Parameters: + * path - Path relative to the module root. + */ + remove: function(path) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.remove must be a string'); + } + return this.storage.delete(this.makePath(path)); + }, + + + cache: function(path, strategy) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.cache must be a string'; + } + if (strategy === false) { + deprecate('caching strategy ', '<"FLUSH">'); + strategy = 'FLUSH'; + } else if (strategy === undefined) { + strategy = 'ALL'; + } else if (typeof(strategy) !== 'string') { + deprecate('that caching strategy', '<"ALL">'); + strategy = 'ALL'; + } + if (strategy !== 'FLUSH' && + strategy !== 'SEEN' && + strategy !== 'ALL') { + throw 'Argument \'strategy\' of baseclient.cache must be one of ' + + '["FLUSH", "SEEN", "ALL"]'; + } + this.storage.caching.set(this.makePath(path), strategy); + return this; + }, + + flush: function(path) { + return this.storage.local.flush(path); + }, + + makePath: function(path) { + return this.base + (path || ''); + }, + + _fireChange: function(event) { + this._emit('change', event); + }, + + _cleanPath: RS.WireClient.cleanPath, + + /** + * Method: getItemURL + * + * Retrieve full URL of item + * + * Parameters: + * path - Path relative to the module root. + */ + getItemURL: function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.getItemURL must be a string'; + } + if (this.storage.connected) { + path = this._cleanPath( this.makePath(path) ); + return this.storage.remote.href + path; + } else { + return undefined; + } + }, + + uuid: function() { + return Math.uuid(); + } + + }; + + /** + * Method: RS#scope + * + * Returns a new scoped to the given path. + * + * Parameters: + * path - Root path of new BaseClient. + * + * + * Example: + * (start code) + * + * var foo = remoteStorage.scope('/foo/'); + * + * // PUTs data "baz" to path /foo/bar + * foo.storeFile('text/plain', 'bar', 'baz'); + * + * var something = foo.scope('something/'); + * + * // GETs listing from path /foo/something/bla/ + * something.getListing('bla/'); + * + * (end code) + * + */ + RS.BaseClient._rs_init = function() { + RS.prototype.scope = function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.scope must be a string'; + } + + return new RS.BaseClient(this, path); + }; + }; + + /* e.g.: + remoteStorage.defineModule('locations', function(priv, pub) { + return { + exports: { + features: priv.scope('features/').defaultType('feature'), + collections: priv.scope('collections/').defaultType('feature-collection'); + } + }; + }); + */ + + maxAgeInvalid = function(maxAge) { + return typeof(maxAge) !== 'undefined' && typeof(maxAge) !== 'number'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/baseclient/types.js **/ +(function(global) { + + RemoteStorage.BaseClient.Types = { + // -> + uris: {}, + // -> + schemas: {}, + // -> + aliases: {}, + + declare: function(moduleName, alias, uri, schema) { + var fullAlias = moduleName + '/' + alias; + + if (schema.extends) { + var extendedAlias; + var parts = schema.extends.split('/'); + if (parts.length === 1) { + extendedAlias = moduleName + '/' + parts.shift(); + } else { + extendedAlias = parts.join('/'); + } + var extendedUri = this.uris[extendedAlias]; + if (! extendedUri) { + throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'"; + } + schema.extends = this.schemas[extendedUri]; + } + + this.uris[fullAlias] = uri; + this.aliases[uri] = fullAlias; + this.schemas[uri] = schema; + }, + + resolveAlias: function(alias) { + return this.uris[alias]; + }, + + getSchema: function(uri) { + return this.schemas[uri]; + }, + + inScope: function(moduleName) { + var ml = moduleName.length; + var schemas = {}; + for (var alias in this.uris) { + if (alias.substr(0, ml + 1) === moduleName + '/') { + var uri = this.uris[alias]; + schemas[uri] = this.schemas[uri]; + } + } + return schemas; + } + }; + + var SchemaNotFound = function(uri) { + var error = new Error("Schema not found: " + uri); + error.name = "SchemaNotFound"; + return error; + }; + + SchemaNotFound.prototype = Error.prototype; + + RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound; + /** + * Class: RemoteStorage.BaseClient + **/ + RemoteStorage.BaseClient.prototype.extend({ + /** + * Method: validate(object) + * + * validates an Object against the associated schema + * the context has to have a @context property + * + * Returns: + * A validate object giving you information about errors + **/ + validate: function(object) { + var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']); + if (schema) { + return tv4.validateResult(object, schema); + } else { + throw new SchemaNotFound(object['@context']); + } + }, + + // client.declareType(alias, schema); + // /* OR */ + // client.declareType(alias, uri, schema); + declareType: function(alias, uri, schema) { + if (! schema) { + schema = uri; + uri = this._defaultTypeURI(alias); + } + RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema); + }, + + _defaultTypeURI: function(alias) { + return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias; + }, + + _attachType: function(object, alias) { + object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias); + } + }); + + Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', { + configurable: true, + get: function() { + return RemoteStorage.BaseClient.Types.inScope(this.moduleName); + } + }); + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/caching.js **/ + /** + * Class: RemoteStorage.Caching + * + * Holds/manages caching configuration. + * + * Caching strategies: + * + * For each subtree, you can set the caching strategy to 'ALL', + * 'SEEN' (default), and 'FLUSH'. + * + * - 'ALL' means that once all outgoing changes have been pushed, sync + * will start retrieving nodes to cache pro-actively. If a local + * copy exists of everything, it will check on each sync whether + * the ETag of the root folder changed, and retrieve remote changes + * if they exist. + * - 'SEEN' does this only for documents and folders that have been either + * read from or written to at least once since connecting to the current + * remote backend, plus their parent/ancestor folders up to the root + * (to make tree-based sync possible). + * - 'FLUSH' will only cache outgoing changes, and forget them as soon as + * they have been saved to remote successfully. + * + **/ + +(function(global) { + var SETTINGS_KEY = "remotestorage:caching"; + + function containingFolder(path) { + if (path === '') { + return '/'; + } + if (! path) { + throw "Path not given!"; + } + + return path.replace(/\/+/g, '/').replace(/[^\/]+\/?$/, ''); + } + + RemoteStorage.Caching = function() { + this.reset(); + }; + + RemoteStorage.Caching.prototype = { + pendingActivations: [], + + /** + * Method: set + * + * Set the caching strategy for a given path explicitly. + * + * Not needed when using /. + * + * Parameters: + * path - Path to cache + * value - Caching strategy. One of 'ALL', 'SEEN', or 'FLUSH'. + * + * Example: + * (start code) + * remoteStorage.caching.set('/bookmarks/archive') + */ + set: function(path, value) { + if (typeof(path) !== 'string') { + throw new Error('path should be a string'); + } + if (typeof(value) === 'undefined') { + throw new Error("value should be 'FLUSH', 'SEEN', or 'ALL'"); + } + + this._rootPaths[path] = value; + + if (value === 'ALL') { + if (this.activateHandler) { + this.activateHandler(path); + } else { + this.pendingActivations.push(path); + } + } + }, + + /** + * Method: enable + * + * Enable caching for a given path. + * + * Uses caching strategy 'ALL'. + * + * Parameters: + * path - Path to enable caching for + */ + enable: function(path) { + this.set(path, 'ALL'); + }, + + /** + * Method: disable + * + * Disable caching for a given path. + * + * Uses caching strategy 'FLUSH' (meaning items are only cached until + * successfully pushed to the remote). + * + * Parameters: + * path - Path to disable caching for + */ + disable: function(path) { + this.set(path, 'FLUSH'); + }, + + /** + * Method: onActivate + * + * Set a callback for when caching is activated for a path. + * + * Parameters: + * callback - Callback function + */ + onActivate: function(cb) { + var i; + RemoteStorage.log('[Caching] Setting activate handler', cb, this.pendingActivations); + this.activateHandler = cb; + for (i=0; i 0; + }, + + collectDiffTasks: function() { + var num = 0; + + return this.local.forAllNodes(function(node) { + if (num > 100) { + return; + } + + if (this.isCorrupt(node)) { + RemoteStorage.log('[Sync] WARNING: corrupt node in local cache', node); + if (typeof(node) === 'object' && node.path) { + this.addTask(node.path); + num++; + } + } else if (this.needsFetch(node) && this.access.checkPathPermission(node.path, 'r')) { + this.addTask(node.path); + num++; + } else if (this.isDocumentNode(node) && this.needsPush(node) && + this.access.checkPathPermission(node.path, 'rw')) { + this.addTask(node.path); + num++; + } + }.bind(this)).then(function() { + return num; + }, function(err) { + throw err; + }); + }, + + inConflict: function(node) { + return (node.local && node.remote && + (node.remote.body !== undefined || node.remote.itemsMap)); + }, + + needsRefresh: function(node) { + if (node.common) { + if (!node.common.timestamp) { + return true; + } + return (this.now() - node.common.timestamp > syncInterval); + } + return false; + }, + + needsFetch: function(node) { + if (this.inConflict(node)) { + return true; + } + if (node.common && node.common.itemsMap === undefined && node.common.body === undefined) { + return true; + } + if (node.remote && node.remote.itemsMap === undefined && node.remote.body === undefined) { + return true; + } + return false; + }, + + needsPush: function(node) { + if (this.inConflict(node)) { + return false; + } + if (node.local && !node.push) { + return true; + } + }, + + needsRemotePut: function(node) { + return node.local && node.local.body; + }, + + needsRemoteDelete: function(node) { + return node.local && node.local.body === false; + }, + + getParentPath: function(path) { + var parts = path.match(/^(.*\/)([^\/]+\/?)$/); + + if (parts) { + return parts[1]; + } else { + throw new Error('Not a valid path: "'+path+'"'); + } + }, + + deleteChildPathsFromTasks: function() { + for (var path in this._tasks) { + paths = this.local._getInternals().pathsFromRoot(path); + + for (var i=1; i= numToAdd) { + return true; + } + } + } + return (numAdded >= numToAdd); + }, + + collectTasks: function(alsoCheckRefresh) { + if (this.hasTasks() || this.stopped) { + promise = promising(); + promise.fulfill(); + return promise; + } + + return this.collectDiffTasks().then(function(numDiffs) { + if (numDiffs || alsoCheckRefresh === false) { + promise = promising(); + promise.fulfill(); + return promise; + } else { + return this.collectRefreshTasks(); + } + }.bind(this), function(err) { + throw err; + }); + }, + + addTask: function(path, cb) { + if (!this._tasks[path]) { + this._tasks[path] = []; + } + if (typeof(cb) === 'function') { + this._tasks[path].push(cb); + } + }, + + /** + * Method: sync + **/ + sync: function() { + var promise = promising(); + this.done = false; + + if (!this.doTasks()) { + return this.collectTasks().then(function() { + try { + this.doTasks(); + } catch(e) { + RemoteStorage.log('[Sync] doTasks error', e); + } + }.bind(this), function(e) { + RemoteStorage.log('[Sync] Sync error', e); + throw new Error('Local cache unavailable'); + }); + } else { + return promising().fulfill(); + } + }, + }; + + /** + * Method: getSyncInterval + * + * Get the value of the sync interval when application is in the foreground + * + * Returns a number of milliseconds + * + */ + RemoteStorage.prototype.getSyncInterval = function() { + return syncInterval; + }; + + /** + * Method: setSyncInterval + * + * Set the value of the sync interval when application is in the foreground + * + * Parameters: + * interval - sync interval in milliseconds + * + */ + RemoteStorage.prototype.setSyncInterval = function(interval) { + if (typeof(interval) !== 'number') { + throw interval + " is not a valid sync interval"; + } + syncInterval = parseInt(interval, 10); + if (this._syncTimer) { + this.stopSync(); + this._syncTimer = setTimeout(this.syncCycle.bind(this), interval); + } + }; + + var SyncError = function(originalError) { + var msg = 'Sync failed: '; + if (typeof(originalError) === 'object' && 'message' in originalError) { + msg += originalError.message; + } else { + msg += originalError; + } + this.originalError = originalError; + Error.apply(this, [msg]); + }; + + SyncError.prototype = Object.create(Error.prototype); + + RemoteStorage.SyncError = SyncError; + + RemoteStorage.prototype.syncCycle = function() { + if (this.sync.stopped) { + return; + } + + this.sync.on('done', function() { + RemoteStorage.log('[Sync] Sync done. Setting timer to', this.getSyncInterval()); + if (!this.sync.stopped) { + if (this._syncTimer) { + clearTimeout(this._syncTimer); + } + this._syncTimer = setTimeout(this.sync.sync.bind(this.sync), this.getSyncInterval()); + } + }.bind(this)); + + this.sync.sync(); + }; + + RemoteStorage.prototype.stopSync = function() { + if (this.sync) { + RemoteStorage.log('[Sync] Stopping sync'); + this.sync.stopped = true; + } else { + // TODO When is this ever the case and what is syncStopped for then? + RemoteStorage.log('[Sync] Will instantiate sync stopped'); + this.syncStopped = true; + } + }; + + RemoteStorage.prototype.startSync = function() { + this.sync.stopped = false; + this.syncStopped = false; + this.sync.sync(); + }; + + var syncCycleCb; + + RemoteStorage.Sync._rs_init = function(remoteStorage) { + syncCycleCb = function() { + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + + if (!remoteStorage.sync) { + // Call this now that all other modules are also ready: + remoteStorage.sync = new RemoteStorage.Sync( + remoteStorage.local, remoteStorage.remote, remoteStorage.access, + remoteStorage.caching); + + if (remoteStorage.syncStopped) { + RemoteStorage.log('[Sync] Instantiating sync stopped'); + remoteStorage.sync.stopped = true; + delete remoteStorage.syncStopped; + } + } + + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + remoteStorage.syncCycle(); + }; + + remoteStorage.on('ready', syncCycleCb); + }; + + RemoteStorage.Sync._rs_cleanup = function(remoteStorage) { + remoteStorage.stopSync(); + remoteStorage.removeEventListener('ready', syncCycleCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/cachinglayer.js **/ +(function() { + /** + * Interface: cachinglayer + * + * This module defines functions that are mixed into remoteStorage.local when + * it is instantiated (currently one of indexeddb.js, localstorage.js, or + * inmemorystorage.js). + * + * All remoteStorage.local implementations should therefore implement + * this.getNodes, this.setNodes, and this.forAllNodes. The rest is blended in + * here to create a GPD (get/put/delete) interface which the BaseClient can + * talk to. + */ + + function isFolder(path) { + return path.substr(-1) === '/'; + } + + function isDocument(path) { + return path.substr(-1) !== '/'; + } + + function deepClone(obj) { + if (obj === undefined) { + return undefined; + } else { + return JSON.parse(JSON.stringify(obj)); + } + } + + function equal(obj1, obj2) { + return JSON.stringify(obj1) === JSON.stringify(obj2); + } + + function getLatest(node) { + if (typeof(node) !== 'object' || typeof(node.path) !== 'string') { + return; + } + if (isFolder(node.path)) { + if (node.local && node.local.itemsMap) { + return node.local; + } + if (node.common && node.common.itemsMap) { + return node.common; + } + } else { + if (node.local && node.local.body && node.local.contentType) { + return node.local; + } + if (node.common && node.common.body && node.common.contentType) { + return node.common; + } + // Migration code! Once all apps use at least this version of the lib, we + // can publish clean-up code that migrates over any old-format data, and + // stop supporting it. For now, new apps will support data in both + // formats, thanks to this: + if (node.body && node.contentType) { + return { + body: node.body, + contentType: node.contentType + }; + } + } + } + + function isOutdated(node, maxAge) { + return !node || !node.timestamp || + ((new Date().getTime()) - node.timestamp > maxAge); + } + + function pathsFromRoot(path) { + var paths = [path]; + var parts = path.replace(/\/$/, '').split('/'); + + while (parts.length > 1) { + parts.pop(); + paths.push(parts.join('/')+'/'); + } + return paths; + } + + function makeNode(path, timestamp) { + var node = { path: path, common: { timestamp: timestamp } }; + + if (isFolder(path)) { + node.common.itemsMap = {}; + } + return node; + } + + function updateFolderNodeWithItemName(node, itemName, timestamp) { + if (!node.common) { + node.common = { + timestamp: timestamp, + itemsMap: {} + }; + } + if (!node.common.itemsMap) { + node.common.itemsMap = {}; + } + if (!node.local) { + node.local = deepClone(node.common); + } + if (!node.local.itemsMap) { + node.local.itemsMap = node.common.itemsMap; + } + node.local.itemsMap[itemName] = true; + + return node; + } + + var methods = { + + get: function(path, maxAge) { + var promise = promising(); + + this.getNodes([path]).then(function(objs) { + var node = getLatest(objs[path]); + if ((typeof(maxAge) === 'number') && isOutdated(node, maxAge)) { + remoteStorage.sync.queueGetRequest(path, promise); + } + + if (node) { + promise.fulfill(200, node.body || node.itemsMap, node.contentType); + } else { + promise.fulfill(404); + } + }.bind(this), function(err) { + promise.reject(err); + }.bind(this)); + + return promise; + }, + + put: function(path, body, contentType) { + var paths = pathsFromRoot(path); + var now = new Date().getTime(); + + return this._updateNodes(paths, function(nodes) { + try { + for (var i=0; i 0) { + // This folder still contains other items, don't remove any further ancestors + break; + } + } + } + return nodes; + }); + }, + + flush: function(path) { + return this._getAllDescendentPaths(path).then(function(paths) { + return this.getNodes(paths); + }.bind(this)).then(function(nodes) { + for (var path in nodes) { + var node = nodes[path]; + + if (node && node.common && node.local) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: (node.local.body === false ? undefined : node.local.body), + newValue: (node.common.body === false ? undefined : node.common.body) + }); + } + nodes[path] = undefined; + } + return this.setNodes(nodes); + }.bind(this)); + }, + + fireInitial: function() { + this.forAllNodes(function(node) { + var latest; + if (isDocument(node.path)) { + latest = getLatest(node); + if (latest) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: undefined, + oldContentType: undefined, + newValue: latest.body, + newContentType: latest.contentType + }); + } + } + }.bind(this)); + }, + + onDiff: function(diffHandler) { + this.diffHandler = diffHandler; + }, + + migrate: function(node) { + if (typeof(node) === 'object' && !node.common) { + node.common = {}; + if (typeof(node.path) === 'string') { + if (node.path.substr(-1) === '/' && typeof(node.body) === 'object') { + node.common.itemsMap = node.body; + } + } else { + //save legacy content of document node as local version + if (!node.local) { + node.local = {}; + } + node.local.body = node.body; + node.local.contentType = node.contentType; + } + } + return node; + }, + + _updateNodes: function(paths, cb) { + var self = this; + + return this.getNodes(paths).then(function(nodes) { + var existingNodes = deepClone(nodes); + var changeEvents = []; + var node; + + nodes = cb(nodes); + + for (var path in nodes) { + node = nodes[path]; + if (equal(node, existingNodes[path])) { + delete nodes[path]; + } + else if(isDocument(path)) { + changeEvents.push({ + path: path, + origin: 'window', + oldValue: node.local.previousBody, + newValue: node.local.body === false ? undefined : node.local.body, + oldContentType: node.local.previousContentType, + newContentType: node.local.contentType + }); + delete node.local.previousBody; + delete node.local.previousContentType; + } + } + + return self.setNodes(nodes).then(function() { + self._emitChangeEvents(changeEvents); + return 200; + }); + }, + function(err) { + throw(err); + }); + }, + + _emitChangeEvents: function(events) { + for (var i=0; i node) map. + * + */ + + var RS = RemoteStorage; + + var DB_VERSION = 2; + + var DEFAULT_DB_NAME = 'remotestorage'; + var DEFAULT_DB; + + RS.IndexedDB = function(database) { + this.db = database || DEFAULT_DB; + + if (!this.db) { + RemoteStorage.log("[IndexedDB] Failed to open DB"); + return undefined; + } + + RS.cachingLayer(this); + RS.eventHandling(this, 'change'); + + this.getsRunning = 0; + this.putsRunning = 0; + }; + + RS.IndexedDB.prototype = { + getNodes: function(paths) { + var promise = promising(); + var transaction = this.db.transaction(['nodes'], 'readonly'); + var nodes = transaction.objectStore('nodes'); + var retrievedNodes = {}; + var startTime = new Date().getTime(); + + this.getsRunning++; + + for (var i=0; i 64 && nChr < 91 ? + nChr - 65 + : nChr > 96 && nChr < 123 ? + nChr - 71 + : nChr > 47 && nChr < 58 ? + nChr + 4 + : nChr === 43 ? + 62 + : nChr === 47 ? + 63 + : + 0; + } + + function base64DecToArr(sBase64, nBlocksSize) { + var + sB64Enc = sBase64.replace(/[^A-Za-z0-9\+\/]/g, ""), nInLen = sB64Enc.length, + nOutLen = nBlocksSize ? Math.ceil((nInLen * 3 + 1 >> 2) / nBlocksSize) * nBlocksSize : nInLen * 3 + 1 >> 2, taBytes = new Uint8Array(nOutLen); + + for (var nMod3, nMod4, nUint24 = 0, nOutIdx = 0, nInIdx = 0; nInIdx < nInLen; nInIdx++) { + nMod4 = nInIdx & 3; + nUint24 |= b64ToUint6(sB64Enc.charCodeAt(nInIdx)) << 18 - 6 * nMod4; + if (nMod4 === 3 || nInLen - nInIdx === 1) { + for (nMod3 = 0; nMod3 < 3 && nOutIdx < nOutLen; nMod3++, nOutIdx++) { + taBytes[nOutIdx] = nUint24 >>> (16 >>> nMod3 & 24) & 255; + } + nUint24 = 0; + } + } + return taBytes; + } + + function isBinary(node) { + return node.match(/charset=binary/); + } + + function isRemoteStorageKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX || + key.substr(0, CHANGES_PREFIX.length) === CHANGES_PREFIX; + } + + function isNodeKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX; + } + + RemoteStorage.LocalStorage.prototype = { + + getNodes: function(paths) { + var promise = promising(); + var nodes = {}; + + for(i=0; i + * instance under the module's name. So when defining a locations module, + * like in the example below, it would be accessible via + * `remoteStorage.locations`, which would in turn have a `features` and a + * `collections` property. + * + * The function receives a private and a public client, which are both + * instances of . In the following example, the + * scope of privateClient is `/locations` and the scope of publicClient is + * `/public/locations`. + * + * Example: + * (start code) + * RemoteStorage.defineModule('locations', function(privateClient, publicClient) { + * return { + * exports: { + * features: privateClient.scope('features/').defaultType('feature'), + * collections: privateClient.scope('collections/').defaultType('feature-collection') + * } + * }; + * }); + * (end code) + */ + + RemoteStorage.defineModule = function(moduleName, builder) { + RemoteStorage.MODULES[moduleName] = builder; + + Object.defineProperty(RemoteStorage.prototype, moduleName, { + configurable: true, + get: function() { + var instance = this._loadModule(moduleName); + Object.defineProperty(this, moduleName, { + value: instance + }); + return instance; + } + }); + + if (moduleName.indexOf('-') !== -1) { + var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) { + return s[1].toUpperCase(); + }); + Object.defineProperty(RemoteStorage.prototype, camelizedName, { + get: function() { + return this[moduleName]; + } + }); + } + }; + + RemoteStorage.prototype._loadModule = function(moduleName) { + var builder = RemoteStorage.MODULES[moduleName]; + if (builder) { + var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'), + new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/')); + return module.exports; + } else { + throw "Unknown module: " + moduleName; + } + }; + + RemoteStorage.prototype.defineModule = function(moduleName) { + console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!"); + RemoteStorage.defineModule.apply(RemoteStorage, arguments); + }; + +})(); + + +/** FILE: src/debug/inspect.js **/ +(function() { + function loadTable(table, storage, paths) { + table.setAttribute('border', '1'); + table.style.margin = '8px'; + table.innerHTML = ''; + var thead = document.createElement('thead'); + table.appendChild(thead); + var titleRow = document.createElement('tr'); + thead.appendChild(titleRow); + ['Path', 'Content-Type', 'Revision'].forEach(function(label) { + var th = document.createElement('th'); + th.textContent = label; + thead.appendChild(th); + }); + + var tbody = document.createElement('tbody'); + table.appendChild(tbody); + + function renderRow(tr, path, contentType, revision) { + [path, contentType, revision].forEach(function(value) { + var td = document.createElement('td'); + td.textContent = value || ''; + tr.appendChild(td); + }); + } + + function loadRow(path) { + if (storage.connected === false) { return; } + function processRow(status, body, contentType, revision) { + if (status === 200) { + var tr = document.createElement('tr'); + tbody.appendChild(tr); + renderRow(tr, path, contentType, revision); + if (path[path.length - 1] === '/') { + for (var key in body) { + loadRow(path + key); + } + } + } + } + storage.get(path).then(processRow); + } + + paths.forEach(loadRow); + } + + function renderWrapper(title, table, storage, paths) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = title; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + updateButton.onclick = function() { loadTable(table, storage, paths); }; + wrapper.appendChild(updateButton); + if (storage.reset) { + var resetButton = document.createElement('button'); + resetButton.textContent = "Reset"; + resetButton.onclick = function() { + storage.reset(function(newStorage) { + storage = newStorage; + loadTable(table, storage, paths); + }); + }; + wrapper.appendChild(resetButton); + } + wrapper.appendChild(table); + loadTable(table, storage, paths); + return wrapper; + } + + function renderLocalChanges(local) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = "Outgoing changes"; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + wrapper.appendChild(updateButton); + var list = document.createElement('ul'); + list.style.fontFamily = 'courier'; + wrapper.appendChild(list); + + function updateList() { + local.changesBelow('/').then(function(changes) { + list.innerHTML = ''; + changes.forEach(function(change) { + var el = document.createElement('li'); + el.textContent = JSON.stringify(change); + list.appendChild(el); + }); + }); + } + + updateButton.onclick = updateList; + updateList(); + return wrapper; + } + + RemoteStorage.prototype.inspect = function() { + + var widget = document.createElement('div'); + widget.id = 'remotestorage-inspect'; + widget.style.position = 'absolute'; + widget.style.top = 0; + widget.style.left = 0; + widget.style.background = 'black'; + widget.style.color = 'white'; + widget.style.border = 'groove 5px #ccc'; + + var controls = document.createElement('div'); + controls.style.position = 'absolute'; + controls.style.top = 0; + controls.style.left = 0; + + var heading = document.createElement('strong'); + heading.textContent = " remotestorage.js inspector "; + + controls.appendChild(heading); + + var syncButton; + + if (this.local) { + syncButton = document.createElement('button'); + syncButton.textContent = "Synchronize"; + controls.appendChild(syncButton); + } + + var closeButton = document.createElement('button'); + closeButton.textContent = "Close"; + closeButton.onclick = function() { + document.body.removeChild(widget); + }; + controls.appendChild(closeButton); + + widget.appendChild(controls); + + var remoteTable = document.createElement('table'); + var localTable = document.createElement('table'); + widget.appendChild(renderWrapper("Remote", remoteTable, this.remote, this.caching.rootPaths)); + if (this.local) { + widget.appendChild(renderWrapper("Local", localTable, this.local, ['/'])); + widget.appendChild(renderLocalChanges(this.local)); + + syncButton.onclick = function() { + this.log('sync clicked'); + this.sync().then(function() { + this.log('SYNC FINISHED'); + loadTable(localTable, this.local, ['/']); + }.bind(this), function(err) { + console.error("SYNC FAILED", err, err.stack); + }); + }.bind(this); + } + + document.body.appendChild(widget); + }; +})(); + + +/** FILE: src/legacy.js **/ +(function() { + var util = { + getEventEmitter: function() { + var object = {}; + var args = Array.prototype.slice.call(arguments); + args.unshift(object); + RemoteStorage.eventHandling.apply(RemoteStorage, args); + object.emit = object._emit; + return object; + }, + + extend: function(target) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach(function(source) { + for (var key in source) { + target[key] = source[key]; + } + }); + return target; + }, + + asyncEach: function(array, callback) { + return this.asyncMap(array, callback). + then(function() { return array; }); + }, + + asyncMap: function(array, callback) { + var promise = promising(); + var n = array.length, i = 0; + var results = [], errors = []; + function oneDone() { + i++; + if (i === n) { + promise.fulfill(results, errors); + } + } + + array.forEach(function(item, index) { + var result; + try { + result = callback(item); + } catch(exc) { + oneDone(); + errors[index] = exc; + } + if (typeof(result) === 'object' && typeof(result.then) === 'function') { + result.then(function(res) { results[index] = res; oneDone(); }, + function(error) { errors[index] = res; oneDone(); }); + } else { + oneDone(); + results[index] = result; + } + }); + + return promise; + }, + + containingFolder: function(path) { + var folder = path.replace(/[^\/]+\/?$/, ''); + return folder === path ? null : folder; + }, + + isFolder: function(path) { + return path.substr(-1) === '/'; + }, + + baseName: function(path) { + var parts = path.split('/'); + if (util.isFolder(path)) { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + }, + + bindAll: function(object) { + for (var key in this) { + if (typeof(object[key]) === 'function') { + object[key] = object[key].bind(object); + } + } + } + }; + + Object.defineProperty(RemoteStorage.prototype, 'util', { + get: function() { + console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release."); + return util; + } + }); + +})(); + + +/** FILE: src/googledrive.js **/ +(function(global) { + /** + * Class: RemoteStorage.GoogleDrive + * + * WORK IN PROGRESS, NOT RECOMMENDED FOR PRODUCTION USE + **/ + + var RS = RemoteStorage; + + var BASE_URL = 'https://www.googleapis.com'; + var AUTH_URL = 'https://accounts.google.com/o/oauth2/auth'; + var AUTH_SCOPE = 'https://www.googleapis.com/auth/drive'; + + var GD_DIR_MIME_TYPE = 'application/vnd.google-apps.folder'; + var RS_DIR_MIME_TYPE = 'application/json; charset=UTF-8'; + + function buildQueryString(params) { + return Object.keys(params).map(function(key) { + return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]); + }).join('&'); + } + + function fileNameFromMeta(meta) { + return encodeURIComponent(meta.title) + (meta.mimeType === GD_DIR_MIME_TYPE ? '/' : ''); + } + + function metaTitleFromFileName(filename) { + if (filename.substr(-1) === '/') { + filename = filename.substr(0, filename.length - 1); + } + return decodeURIComponent(filename); + } + + function parentPath(path) { + return path.replace(/[^\/]+\/?$/, ''); + } + + function baseName(path) { + var parts = path.split('/'); + if (path.substr(-1) === '/') { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + } + + var Cache = function(maxAge) { + this.maxAge = maxAge; + this._items = {}; + }; + + Cache.prototype = { + get: function(key) { + var item = this._items[key]; + var now = new Date().getTime(); + return (item && item.t >= (now - this.maxAge)) ? item.v : undefined; + }, + + set: function(key, value) { + this._items[key] = { + v: value, + t: new Date().getTime() + }; + } + }; + + RS.GoogleDrive = function(remoteStorage, clientId) { + + RS.eventHandling(this, 'connected'); + + this.rs = remoteStorage; + this.clientId = clientId; + + this._fileIdCache = new Cache(60 * 5); // ids expire after 5 minutes (is this a good idea?) + + setTimeout(function() { + this.configure(undefined, undefined, undefined, localStorage['remotestorage:googledrive:token']); + }.bind(this), 0); + }; + + RS.GoogleDrive.prototype = { + + configure: function(_x, _y, _z, token) { // parameter list compatible with WireClient + if (token) { + localStorage['remotestorage:googledrive:token'] = token; + this.token = token; + this.connected = true; + this._emit('connected'); + } else { + this.connected = false; + delete this.token; + // not reseting backend whenever googledrive gets initialized without an token +// this.rs.setBackend(undefined); + delete localStorage['remotestorage:googledrive:token']; + } + }, + + connect: function() { + this.rs.setBackend('googledrive'); + RS.Authorize(AUTH_URL, AUTH_SCOPE, String(RS.Authorize.getLocation()), this.clientId); + }, + + get: function(path, options) { + if (path.substr(-1) === '/') { + return this._getFolder(path, options); + } else { + return this._getFile(path, options); + } + }, + + put: function(path, body, contentType, options) { + var promise = promising(); + function putDone(error, response) { + if (error) { + promise.reject(error); + } else if (response.status >= 200 && response.status < 300) { + var meta = JSON.parse(response.responseText); + promise.fulfill(200, undefined, meta.mimeType, meta.etag); + } else { + promise.reject("PUT failed with status " + response.status + " (" + response.responseText + ")"); + } + } + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + return; + } else if (id) { + this._updateFile(id, path, body, contentType, options, putDone); + } else { + this._createFile(path, body, contentType, options, putDone); + } + }); + return promise; + }, + + 'delete': function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else if (id) { + this._request('DELETE', BASE_URL + '/drive/v2/files/' + id, {}, function(deleteError, response) { + if (deleteError) { + promise.reject(deleteError); + } else if (response.status === 200 || response.status === 204) { + promise.fulfill(200); + } else { + promise.reject("Delete failed: " + response.status + " (" + response.responseText + ")"); + } + }); + } else { + // file doesn't exist. ignore. + promise.fulfill(200); + } + }); + return promise; + }, + + _updateFile: function(id, path, body, contentType, options, callback) { + callback = callback.bind(this); + var metadata = { + mimeType: contentType + }; + this._request('PUT', BASE_URL + '/upload/drive/v2/files/' + id + '?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('PUT', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }, + + _createFile: function(path, body, contentType, options, callback) { + callback = callback.bind(this); + this._getParentId(path, function(parentIdError, parentId) { + if (parentIdError) { + callback(parentIdError); + return; + } + var fileName = baseName(path); + var metadata = { + title: metaTitleFromFileName(fileName), + mimeType: contentType, + parents: [{ + kind: "drive#fileLink", + id: parentId + }] + }; + this._request('POST', BASE_URL + '/upload/drive/v2/files?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('POST', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }); + }, + + _getFile: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else { + this._getMeta(id, function(metaError, meta) { + var etagWithoutQuotes; + if (typeof(meta) === 'object' && typeof(meta.etag) === 'string') { + etagWithoutQuotes = meta.etag.substring(1, meta.etag.length-1); + } + if (metaError) { + promise.reject(metaError); + } else if (meta.downloadUrl) { + var options = {}; + if (meta.mimeType.match(/charset=binary/)) { + options.responseType = 'blob'; + } + this._request('GET', meta.downloadUrl, options, function(downloadError, response) { + if (downloadError) { + promise.reject(downloadError); + } else { + var body = response.response; + if (meta.mimeType.match(/^application\/json/)) { + try { + body = JSON.parse(body); + } catch(e) {} + } + promise.fulfill(200, body, meta.mimeType, etagWithoutQuotes); + } + }); + } else { + // empty file + promise.fulfill(200, '', meta.mimeType, etagWithoutQuotes); + } + }); + } + }); + return promise; + }, + + _getFolder: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + var query, fields, data, i, etagWithoutQuotes, itemsMap; + if (idError) { + promise.reject(idError); + } else if (! id) { + promise.fulfill(404); + } else { + query = '\'' + id + '\' in parents'; + fields = 'items(downloadUrl,etag,fileSize,id,mimeType,title)'; + this._request('GET', BASE_URL + '/drive/v2/files?' + + 'q=' + encodeURIComponent(query) + + '&fields=' + encodeURIComponent(fields) + + '&maxResults=1000', + {}, function(childrenError, response) { + if (childrenError) { + promise.reject(childrenError); + } else { + if (response.status === 200) { + try { + data = JSON.parse(response.responseText); + } catch(e) { + promise.reject('non-JSON response from GoogleDrive'); + return; + } + itemsMap = {}; + for(i=0; i= 0 || true) { + try { + body = JSON.parse(body); + mime = 'application/json; charset=UTF-8'; + } catch(e) { + RS.log("Failed parsing Json, assume it is something else then", mime, path); + } + } + promise.fulfill(status, body, mime, rev); + } + + } else { + promise.fulfill(status); + } + } + }); + return promise; + }, + /** + * Method : put(path, body, contentType, options) + * put compatible with wireclient + * also uses _revCache to check for version conflicts + * also shares via share(path) + **/ + put: function(path, body, contentType, options){ + RemoteStorage.log('dropbox.put', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = this._sharePromise(path); + + var revCache = this._revCache; + + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options && options.ifMatch && savedRev && (savedRev !== options.ifMatch) ) { + promise.fulfill(412); + return promise; + } + if (! contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || RS.WireClient.isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var url = 'https://api-content.dropbox.com/1/files_put/auto' + path + '?'; + if (options && options.ifMatch) { + url += "parent_rev="+encodeURIComponent(options.ifMatch); + } + if (body.length>150*1024*1024){ //FIXME actual content-length + //https://www.dropbox.com/developers/core/docs#chunked-upload + RemoteStorage.log('files larger than 150MB not supported yet'); + } else { + this._request('PUT', url, {body:body, headers:{'Content-Type':contentType}}, function(err, resp) { + if (err) { + promise.reject(err); + } else { + var response = JSON.parse(resp.responseText); + // if dropbox reports an file conflict they just change the name of the file + // TODO find out which stays the origianl and how to deal with this + if (response.path !== path){ + promise.fulfill(412); + this.rs.log('Dropbox created conflicting File ', response.path); + } + else { + revCache.set(path, response.rev); + promise.fulfill(resp.status); + } + } + }); + } + return promise; + }, + + /** + * Method : delete(path, options) + * similar to get and set + **/ + 'delete': function(path, options){ + RemoteStorage.log('dropbox.delete ', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = promising(); + var revCache = this._revCache; + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options.ifMatch && savedRev && (options.ifMatch !== savedRev)) { + promise.fulfill(412); + return promise; + } + + var url = 'https://api.dropbox.com/1/fileops/delete?root=auto&path='+encodeURIComponent(path); + this._request('POST', url, {}, function(err, resp){ + if (err) { + promise.reject(error); + } else { + promise.fulfill(resp.status); + revCache.delete(path); + } + }); + + return promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + delete this._itemRefs[path]; + var p = promising(); + return p.fulfill.apply(p, args); + }.bind(this)); + }, + + /** + * Method : _sharePromise(path) + * returns a promise which's then block doesn't touch the arguments given + * and calls share for the path + * + * also checks for necessity of shareing this url(already in the itemRefs or not '/public/') + **/ + _sharePromise: function(path){ + var promise = promising(); + var self = this; + if (path.match(/^\/public\/.*[^\/]$/) && typeof this._itemRefs[path] === 'undefined') { + RemoteStorage.log('shareing this one ', path); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + var p = promising(); + RemoteStorage.log('calling share now'); + self.share(path).then(function() { + RemoteStorage.log('shareing fullfilled promise',arguments); + p.fulfill.apply(p,args); + }, function(err) { + RemoteStorage.log("shareing failed" , err); + p.fulfill.apply(p,args); + }); + return p; + }); + } + return promise; + }, + + /** + * Method : share(path) + * get sher_url s from dropbox and pushes those into this._hrefCache + * returns promise + */ + share: function(path){ + var url = "https://api.dropbox.com/1/media/auto"+path; + var promise = promising(); + var itemRefs = this._itemRefs; + + // requesting shareing url + this._request('POST', url, {}, function(err, resp){ + if (err) { + RemoteStorage.log(err); + err.message = 'Shareing Dropbox Thingie("'+path+'") failed' + err.message; + promise.reject(err); + } else { + try{ + var response = JSON.parse(resp.responseText); + var url = response.url; + itemRefs[path] = url; + RemoteStorage.log("SHAREING URL :::: ",url,' for ',path); + if (hasLocalStorage) { + localStorage[SETTINGS_KEY+":shares"] = JSON.stringify(this._itemRefs); + } + promise.fulfill(url); + } catch(err) { + err.message += "share error"; + promise.reject(err); + } + } + }); + return promise; + }, + + /** + * Method : info() + * fetching user info from Dropbox returns promise + **/ + info: function() { + var url = 'https://api.dropbox.com/1/account/info'; + var promise = promising(); + // requesting user info(mainly for userAdress) + this._request('GET', url, {}, function(err, resp){ + if (err) { + promise.reject(err); + } else { + try { + var info = JSON.parse(resp.responseText); + promise.fulfill(info); + } catch(e) { + promise.reject(err); + } + } + }); + return promise; + }, + + _request: function(method, url, options, callback) { + callback = callback.bind(this); + if (! options.headers) { options.headers = {}; } + options.headers['Authorization'] = 'Bearer ' + this.token; + RS.WireClient.request.call(this, method, url, options, function(err, xhr) { + //503 means retry this later + if (xhr && xhr.status === 503) { + global.setTimeout(this._request(method, url, options, callback), 3210); + } else { + callback(err, xhr); + } + }); + }, + + /** + * method: fetchDelta + * + * this method fetches the deltas from the dropbox api, used to sync the storage + * here we retrive changes and put them into the _revCache, those values will then be used + * to determin if something has changed. + **/ + fetchDelta: function() { + var args = Array.prototype.slice.call(arguments); + var promise = promising(); + var self = this; + this._request('POST', 'https://api.dropbox.com/1/delta', { + body: this._deltaCursor ? ('cursor=' + encodeURIComponent(this._deltaCursor)) : '', + headers: { + 'Content-Type': 'application/x-www-form-urlencoded' + } + }, function(error, response) { + if (error) { + this.rs.log('fetchDeltas',error); + this.rs._emit('error', new RemoteStorage.SyncError('fetchDeltas failed'+error)); + promise.reject(error); + } else { + // break if status != 200 + if (response.status !== 200 ) { + if (response.status === 400) { + this.rs._emit('error', new RemoteStorage.Unauthorized()); + promise.fulfill.apply(promise, args); + } else { + RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText); + promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText); + } + return promise; + } + + var delta; + try { + delta = JSON.parse(response.responseText); + } catch(error) { + RS.log('fetchDeltas can not parse response',error); + return promise.reject("can not parse response of fetchDelta : "+error.message); + } + // break if no entries found + if (!delta.entries) { + RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!", delta); + return promise.reject('dropbox.fetchDeltas failed, no entries found'); + } + + // Dropbox sends the complete state + if (delta.reset) { + this._revCache = new LowerCaseCache('rev'); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + self._revCache._activatePropagation(); + var p = promising(); + return p.fulfill.apply(p,args); + }); + } + + //saving the cursor for requesting further deltas in relation to the cursor position + if (delta.cursor) { + this._deltaCursor = delta.cursor; + } + + //updating revCache + RemoteStorage.log("Delta : ",delta.entries); + delta.entries.forEach(function(entry) { + var path = entry[0]; + var rev; + if (!entry[1]){ + rev = null; + } else { + if (entry[1].is_dir) { + return; + } + rev = entry[1].rev; + } + self._revCache.set(path, rev); + }); + promise.fulfill.apply(promise, args); + } + }); + return promise; + } + }; + + //hooking and unhooking the sync + + function hookSync(rs) { + if (rs._dropboxOrigSync) { return; } // already hooked + rs._dropboxOrigSync = rs.sync.bind(rs); + rs.sync = function() { + return this.dropbox.fetchDelta.apply(this.dropbox, arguments). + then(rs._dropboxOrigSync, function(err){ + rs._emit('error', new rs.SyncError(err)); + }); + }; + } + + function unHookSync(rs) { + if (! rs._dropboxOrigSync) { return; } // not hooked + rs.sync = rs._dropboxOrigSync; + delete rs._dropboxOrigSync; + } + + // hooking and unhooking getItemURL + + function hookGetItemURL(rs) { + if (rs._origBaseClientGetItemURL) { return; } + rs._origBaseClientGetItemURL = RS.BaseClient.prototype.getItemURL; + RS.BaseClient.prototype.getItemURL = function(path){ + var ret = rs.dropbox._itemRefs[path]; + return ret ? ret : ''; + }; + } + + function unHookGetItemURL(rs){ + if (! rs._origBaseClieNtGetItemURL) { return; } + RS.BaseClient.prototype.getItemURL = rs._origBaseClietGetItemURL; + delete rs._origBaseClietGetItemURL; + } + + function hookRemote(rs){ + if (rs._origRemote) { return; } + rs._origRemote = rs.remote; + rs.remote = rs.dropbox; + } + + function unHookRemote(rs){ + if (rs._origRemote) { + rs.remote = rs._origRemote; + delete rs._origRemote; + } + } + + function hookIt(rs){ + hookRemote(rs); + if (rs.sync) { + hookSync(rs); + } + hookGetItemURL(rs); + } + + function unHookIt(rs){ + unHookRemote(rs); + unHookSync(rs); + unHookGetItemURL(rs); + } + + RS.Dropbox._rs_init = function(rs) { + hasLocalStorage = rs.localStorageAvailable(); + if ( rs.apiKeys.dropbox ) { + rs.dropbox = new RS.Dropbox(rs); + } + if (rs.backend === 'dropbox') { + hookIt(rs); + } + }; + + RS.Dropbox._rs_supported = function() { + return true; + }; + + RS.Dropbox._rs_cleanup = function(rs) { + unHookIt(rs); + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + rs.removeEventListener('error', onErrorCb); + rs.setBackend(undefined); + }; +})(this); + +return new RemoteStorage(); +}); diff --git a/release/0.10.0-beta2/remotestorage.js b/release/0.10.0-beta2/remotestorage.js new file mode 100644 index 000000000..1c8c1fb8a --- /dev/null +++ b/release/0.10.0-beta2/remotestorage.js @@ -0,0 +1,7675 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ + +/** FILE: lib/promising.js **/ +(function(global) { + function getPromise(builder) { + var promise; + + if(typeof(builder) === 'function') { + setTimeout(function() { + try { + builder(promise); + } catch(e) { + promise.reject(e); + } + }, 0); + } + + var consumers = [], success, result; + + function notifyConsumer(consumer) { + if(success) { + var nextValue; + if(consumer.fulfilled) { + try { + nextValue = [consumer.fulfilled.apply(null, result)]; + } catch(exc) { + consumer.promise.reject(exc); + return; + } + } else { + nextValue = result; + } + if(nextValue[0] && typeof(nextValue[0].then) === 'function') { + nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill.apply(null, nextValue); + } + } else { + if(consumer.rejected) { + var ret; + try { + ret = consumer.rejected.apply(null, result); + } catch(exc) { + consumer.promise.reject(exc); + return; + } + if(ret && typeof(ret.then) === 'function') { + ret.then(consumer.promise.fulfill, consumer.promise.reject); + } else { + consumer.promise.fulfill(ret); + } + } else { + consumer.promise.reject.apply(null, result); + } + } + } + + function resolve(succ, res) { + if(result) { + console.error("WARNING: Can't resolve promise, already resolved!"); + return; + } + success = succ; + result = Array.prototype.slice.call(res); + setTimeout(function() { + var cl = consumers.length; + if(cl === 0 && (! success)) { + console.error("Possibly uncaught error: ", result, result[0] && result[0].stack); + } + for(var i=0;i. + */ + RemoteStorage.log = function() { + if (RemoteStorage._log) { + console.log.apply(console, arguments); + } + }; + + + RemoteStorage.prototype = { + /** + * Method: connect + * + * Connect to a remoteStorage server. + * + * Parameters: + * userAddress - The user address (user@host) to connect to. + * + * Discovers the webfinger profile of the given user address and + * initiates the OAuth dance. + * + * This method must be called *after* all required access has been claimed. + * + */ + connect: function(userAddress) { + this.setBackend('remotestorage'); + if (userAddress.indexOf('@') < 0) { + this._emit('error', new RemoteStorage.DiscoveryError("User adress doesn't contain an @.")); + return; + } + this.remote.configure(userAddress); + this._emit('connecting'); + + var discoveryTimeout = setTimeout(function() { + this._emit('error', new RemoteStorage.DiscoveryError("No storage information found at that user address.")); + }.bind(this), 5000); + + RemoteStorage.Discover(userAddress, function(href, storageApi, authURL) { + clearTimeout(discoveryTimeout); + if (!href) { + this._emit('error', new RemoteStorage.DiscoveryError("Failed to contact storage server.")); + return; + } + this._emit('authing'); + this.remote.configure(userAddress, href, storageApi); + if (! this.remote.connected) { + this.authorize(authURL); + } + }.bind(this)); + }, + + /** + * Method: disconnect + * + * "Disconnect" from remotestorage server to terminate current session. + * This method clears all stored settings and deletes the entire local + * cache. + */ + disconnect: function() { + if (this.remote) { + this.remote.configure(null, null, null, null); + } + this._setGPD({ + get: this._pendingGPD('get'), + put: this._pendingGPD('put'), + delete: this._pendingGPD('delete') + }); + var n = this._cleanups.length, i = 0; + + var oneDone = function() { + i++; + if (i >= n) { + this._init(); + RemoteStorage.log('Done cleaning up, emitting disconnected and disconnect events'); + this._emit('disconnected'); + } + }.bind(this); + + if (n > 0) { + this._cleanups.forEach(function(cleanup) { + var cleanupResult = cleanup(this); + if (typeof(cleanup) === 'object' && typeof(cleanup.then) === 'function') { + cleanupResult.then(oneDone); + } else { + oneDone(); + } + }.bind(this)); + } else { + oneDone(); + } + }, + + setBackend: function(what) { + this.backend = what; + if (this.localStorageAvailable()) { + if (what) { + localStorage['remotestorage:backend'] = what; + } else { + delete localStorage['remotestorage:backend']; + } + } + }, + + /** + * Method: onChange + * + * Add a "change" event handler to the given path. Whenever a "change" + * happens (as determined by the backend, such as e.g. + * ) and the affected path is equal to or below + * the given 'path', the given handler is called. + * + * You should usually not use this method directly, but instead use the + * "change" events provided by . + * + * Parameters: + * path - Absolute path to attach handler to. + * handler - Handler function. + */ + onChange: function(path, handler) { + if (! this._pathHandlers.change[path]) { + this._pathHandlers.change[path] = []; + } + this._pathHandlers.change[path].push(handler); + }, + + /** + * Method: enableLog + * + * Enable remoteStorage logging + */ + enableLog: function() { + RemoteStorage._log = true; + }, + + /** + * Method: disableLog + * + * Disable remoteStorage logging + */ + disableLog: function() { + RemoteStorage._log = false; + }, + + /** + * Method: log + * + * The same as . + */ + log: function() { + RemoteStorage.log.apply(RemoteStorage, arguments); + }, + + setApiKeys: function(type, keys) { + if (keys) { + this.apiKeys[type] = keys; + } else { + delete this.apiKeys[type]; + } + if (this.localStorageAvailable()) { + localStorage['remotestorage:api-keys'] = JSON.stringify(this.apiKeys); + } + }, + + /** + ** INITIALIZATION + **/ + + _init: function() { + var self = this, + readyFired = false; + + function fireReady() { + try { + if (!readyFired) { + self._emit('ready'); + readyFired = true; + } + } catch(e) { + console.error("'ready' failed: ", e, e.stack); + self._emit('error', e); + } + } + + this._loadFeatures(function(features) { + this.log('[RemoteStorage] All features loaded'); + this.local = features.local && new features.local(); + // this.remote set by WireClient._rs_init as lazy property on + // RS.prototype + + if (this.local && this.remote) { + this._setGPD(SyncedGetPutDelete, this); + this._bindChange(this.local); + } else if (this.remote) { + this._setGPD(this.remote, this.remote); + } + + if (this.remote) { + this.remote.on('connected', function(){ + fireReady(); + self._emit('connected'); + }); + this.remote.on('not-connected', function(){ + fireReady(); + self._emit('not-connected'); + }); + if (this.remote.connected) { + fireReady(); + self._emit('connected'); + } + } + + this._collectCleanupFunctions(); + + try { + this._allLoaded = true; + this._emit('features-loaded'); + } catch(exc) { + logError(exc); + this._emit('error', exc); + } + this._processPending(); + }.bind(this)); + }, + + _collectCleanupFunctions: function() { + for (var i=0; i < this.features.length; i++) { + var cleanup = this.features[i].cleanup; + if (typeof(cleanup) === 'function') { + this._cleanups.push(cleanup); + } + } + }, + + /** + ** FEATURE DETECTION + **/ + _loadFeatures: function(callback) { + var featureList = [ + 'WireClient', + 'I18n', + 'Dropbox', + 'GoogleDrive', + 'Access', + 'Caching', + 'Discover', + 'Authorize', + 'Widget', + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage', + 'Sync', + 'BaseClient', + 'Env' + ]; + var features = []; + var featuresDone = 0; + var self = this; + + function featureDone() { + featuresDone++; + if (featuresDone === featureList.length) { + setTimeout(function() { + features.caching = !!RemoteStorage.Caching; + features.sync = !!RemoteStorage.Sync; + [ + 'IndexedDB', + 'LocalStorage', + 'InMemoryStorage' + ].some(function(cachingLayer) { + if (features.some(function(feature) { return feature.name === cachingLayer; })) { + features.local = RemoteStorage[cachingLayer]; + return true; + } + }); + self.features = features; + callback(features); + }, 0); + } + } + + function featureInitialized(name) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialized."); + features.push({ + name : name, + init : RemoteStorage[name]._rs_init, + supported : true, + cleanup : RemoteStorage[name]._rs_cleanup + }); + featureDone(); + } + + function featureFailed(name, err) { + self.log("[RemoteStorage] [FEATURE "+name+"] initialization failed ( "+err+")"); + featureDone(); + } + + function featureSupported(name, success) { + self.log("[RemoteStorage] [FEATURE "+name+"]" + success ? "":" not"+" supported"); + if (!success) { + featureDone(); + } + } + + function initFeature(name) { + var initResult; + try { + initResult = RemoteStorage[name]._rs_init(self); + } catch(e) { + featureFailed(name, e); + return; + } + if (typeof(initResult) === 'object' && typeof(initResult.then) === 'function') { + initResult.then( + function(){ featureInitialized(name); }, + function(err){ featureFailed(name, err); } + ); + } else { + featureInitialized(name); + } + } + + featureList.forEach(function(featureName) { + self.log("[RemoteStorage] [FEATURE " + featureName + "] initializing..."); + var impl = RemoteStorage[featureName]; + var supported; + + if (impl) { + supported = !impl._rs_supported || impl._rs_supported(); + + if (typeof supported === 'object') { + supported.then( + function(){ + featureSupported(featureName, true); + initFeature(featureName); + }, + function(){ + featureSupported(featureName, false); + } + ); + } + else if (typeof supported === 'boolean') { + featureSupported(featureName, supported); + if (supported) { + initFeature(featureName); + } + } + } else { + featureSupported(featureName, false); + } + }); + }, + + localStorageAvailable: function() { + try { + return !!global.localStorage; + } catch(error) { + return false; + } + }, + + /** + ** GET/PUT/DELETE INTERFACE HELPERS + **/ + + _setGPD: function(impl, context) { + function wrap(f) { + return function() { + return f.apply(context, arguments) + .then(emitUnauthorized.bind(this)); + }; + } + this.get = wrap(impl.get); + this.put = wrap(impl.put); + this.delete = wrap(impl.delete); + }, + + _pendingGPD: function(methodName) { + return function() { + var promise = promising(); + this._pending.push({ + method: methodName, + args: Array.prototype.slice.call(arguments), + promise: promise + }); + return promise; + }.bind(this); + }, + + _processPending: function() { + this._pending.forEach(function(pending) { + try { + this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject); + } catch(e) { + pending.promise.reject(e); + } + }.bind(this)); + this._pending = []; + }, + + /** + ** CHANGE EVENT HANDLING + **/ + + _bindChange: function(object) { + object.on('change', this._dispatchEvent.bind(this, 'change')); + }, + + _dispatchEvent: function(eventName, event) { + for (var path in this._pathHandlers[eventName]) { + var pl = path.length; + var self = this; + this._pathHandlers[eventName][path].forEach(function(handler) { + if (event.path.substr(0, pl) === path) { + var ev = {}; + for (var key in event) { ev[key] = event[key]; } + ev.relativePath = event.path.replace(new RegExp('^' + path), ''); + try { + handler(ev); + } catch(e) { + console.error("'change' handler failed: ", e, e.stack); + self._emit('error', e); + } + } + }); + } + } + }; + + /** + * Property: connected + * + * Boolean property indicating if remoteStorage is currently connected. + */ + Object.defineProperty(RemoteStorage.prototype, 'connected', { + get: function() { + return this.remote.connected; + } + }); + + /** + * Property: access + * + * Tracking claimed access scopes. A instance. + * + * + * Property: caching + * + * Caching settings. A instance. + * + * Not available in no-cache builds. + * + * + * Property: remote + * + * Access to the remote backend used. Usually a . + * + * + * Property: local + * + * Access to the local caching backend used. Usually either a + * or instance. + * + * Not available in no-cache builds. + */ + + global.RemoteStorage = RemoteStorage; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/eventhandling.js **/ +(function(global) { + /** + * Interface: eventhandling + */ + var methods = { + /** + * Method: addEventListener + * + * Install an event handler for the given event name + */ + addEventListener: function(eventName, handler) { + if (typeof(eventName) !== 'string') { + throw new Error('Argument eventName should be a string'); + } + if (typeof(handler) !== 'function') { + throw new Error('Argument handler should be a function'); + } + RemoteStorage.log('[Eventhandling] Adding event listener', eventName, handler); + this._validateEvent(eventName); + this._handlers[eventName].push(handler); + }, + + /** + * Method: removeEventListener + * + * Remove a previously installed event handler + */ + removeEventListener: function(eventName, handler) { + this._validateEvent(eventName); + var hl = this._handlers[eventName].length; + for (var i=0;i + **/ + methods.on = methods.addEventListener; + + /** + * Function: eventHandling + * + * Mixes event handling functionality into an object. + * + * The first parameter is always the object to be extended. + * All remaining parameter are expected to be strings, interpreted as valid event + * names. + * + * Example: + * (start code) + * var MyConstructor = function() { + * eventHandling(this, 'connected', 'disconnected'); + * + * this._emit('connected'); + * this._emit('disconnected'); + * // This would throw an exception: + * // this._emit('something-else'); + * }; + * + * var myObject = new MyConstructor(); + * myObject.on('connected', function() { console.log('connected'); }); + * myObject.on('disconnected', function() { console.log('disconnected'); }); + * // This would throw an exception as well: + * // myObject.on('something-else', function() {}); + * (end code) + */ + RemoteStorage.eventHandling = function(object) { + var eventNames = Array.prototype.slice.call(arguments, 1); + for (var key in methods) { + object[key] = methods[key]; + } + object._handlers = {}; + eventNames.forEach(function(eventName) { + object._addEvent(eventName); + }); + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/wireclient.js **/ +(function(global) { + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.WireClient + * + * WireClient Interface + * -------------------- + * + * This file exposes a get/put/delete interface on top of XMLHttpRequest. + * It requires to be configured with parameters about the remotestorage server to + * connect to. + * Each instance of WireClient is always associated with a single remotestorage + * server and access token. + * + * Usually the WireClient instance can be accessed via `remoteStorage.remote`. + * + * This is the get/put/delete interface: + * + * - #get() takes a path and optionally a ifNoneMatch option carrying a version + * string to check. It returns a promise that will be fulfilled with the HTTP + * response status, the response body, the MIME type as returned in the + * 'Content-Type' header and the current revision, as returned in the 'ETag' + * header. + * - #put() takes a path, the request body and a content type string. It also + * accepts the ifMatch and ifNoneMatch options, that map to the If-Match and + * If-None-Match headers respectively. See the remotestorage-01 specification + * for details on handling these headers. It returns a promise, fulfilled with + * the same values as the one for #get(). + * - #delete() takes a path and the ifMatch option as well. It returns a promise + * fulfilled with the same values as the one for #get(). + * + * In addition to this, the WireClient has some compatibility features to work with + * remotestorage 2012.04 compatible storages. For example it will cache revisions + * from folder listings in-memory and return them accordingly as the "revision" + * parameter in response to #get() requests. Similarly it will return 404 when it + * receives an empty folder listing, to mimic remotestorage-01 behavior. Note + * that it is not always possible to know the revision beforehand, hence it may + * be undefined at times (especially for caching-roots). + */ + + var hasLocalStorage; + var SETTINGS_KEY = "remotestorage:wireclient"; + + var API_2012 = 1, API_00 = 2, API_01 = 3, API_02 = 4, API_HEAD = 5; + + var STORAGE_APIS = { + 'draft-dejong-remotestorage-00': API_00, + 'draft-dejong-remotestorage-01': API_01, + 'draft-dejong-remotestorage-02': API_02, + 'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012 + }; + + var isArrayBufferView; + + if (typeof(ArrayBufferView) === 'function') { + isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); }; + } else { + var arrayBufferViews = [ + Int8Array, Uint8Array, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array + ]; + isArrayBufferView = function(object) { + for (var i=0;i<8;i++) { + if (object instanceof arrayBufferViews[i]) { + return true; + } + } + return false; + }; + } + + function addQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + if (str === '*') { + return '*'; + } + + return '"' + str + '"'; + } + + function stripQuotes(str) { + if (typeof(str) !== 'string') { + return str; + } + + return str.replace(/^["']|["']$/g, ''); + } + + function readBinaryData(content, mimeType, callback) { + var blob = new Blob([content], { type: mimeType }); + var reader = new FileReader(); + reader.addEventListener("loadend", function() { + callback(reader.result); // reader.result contains the contents of blob as a typed array + }); + reader.readAsArrayBuffer(blob); + } + + function cleanPath(path) { + return path.replace(/\/+/g, '/').split('/').map(encodeURIComponent).join('/'); + } + + function isFolder(path) { + return (path.substr(-1) === '/'); + } + + function isFolderDescription(body) { + return ((body['@context'] === 'http://remotestorage.io/spec/folder-description') + && (typeof(body['items']) === 'object')); + } + + function isSuccessStatus(status) { + return [201, 204, 304].indexOf(status) >= 0; + } + + function isErrorStatus(status) { + return [401, 403, 404, 412].indexOf(status) >= 0; + } + + var onErrorCb; + + /** + * Class : RemoteStorage.WireClient + **/ + RS.WireClient = function(rs) { + this.connected = false; + + /** + * Event: change + * never fired for some reason + * + * Event: connected + * fired when the wireclient connect method realizes that it is + * in posession of a token and a href + **/ + RS.eventHandling(this, 'change', 'connected', 'wire-busy', 'wire-done', 'not-connected'); + + onErrorCb = function(error){ + if (error instanceof RemoteStorage.Unauthorized) { + this.configure(undefined, undefined, undefined, null); + } + }.bind(this); + rs.on('error', onErrorCb); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + setTimeout(function() { + this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token); + }.bind(this), 0); + } + } + + this._revisionCache = {}; + + if (this.connected) { + setTimeout(this._emit.bind(this), 0, 'connected'); + } + }; + + RS.WireClient.REQUEST_TIMEOUT = 30000; + + RS.WireClient.prototype = { + /** + * Property: token + * + * Holds the bearer token of this WireClient, as obtained in the OAuth dance + * + * Example: + * (start code) + * + * remoteStorage.remote.token + * // -> 'DEADBEEF01==' + */ + + /** + * Property: href + * + * Holds the server's base URL, as obtained in the Webfinger discovery + * + * Example: + * (start code) + * + * remoteStorage.remote.href + * // -> 'https://storage.example.com/users/jblogg/' + */ + + /** + * Property: storageApi + * + * Holds the spec version the server claims to be compatible with + * + * Example: + * (start code) + * + * remoteStorage.remote.storageApi + * // -> 'draft-dejong-remotestorage-01' + */ + + _request: function(method, uri, token, headers, body, getEtag, fakeRevision) { + if ((method === 'PUT' || method === 'DELETE') && uri[uri.length - 1] === '/') { + throw "Don't " + method + " on directories!"; + } + + var promise = promising(); + var revision; + var reqType; + var self = this; + + headers['Authorization'] = 'Bearer ' + token; + + this._emit('wire-busy', { + method: method, + isFolder: isFolder(uri) + }); + + RS.WireClient.request(method, uri, { + body: body, + headers: headers + }, function(error, response) { + if (error) { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: false + }); + self.online = false; + promise.reject(error); + } else { + self._emit('wire-done', { + method: method, + isFolder: isFolder(uri), + success: true + }); + self.online = true; + if (isErrorStatus(response.status)) { + RemoteStorage.log('[WireClient] Error response status', response.status); + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = undefined; + } + promise.fulfill(response.status, undefined, undefined, revision); + } else if (isSuccessStatus(response.status) || + (response.status === 200 && method !== 'GET')) { + revision = stripQuotes(response.getResponseHeader('ETag')); + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, undefined, undefined, revision); + } else { + var mimeType = response.getResponseHeader('Content-Type'); + var body; + if (getEtag) { + revision = stripQuotes(response.getResponseHeader('ETag')); + } else { + revision = response.status === 200 ? fakeRevision : undefined; + } + + if ((!mimeType) || mimeType.match(/charset=binary/)) { + RS.WireClient.readBinaryData(response.response, mimeType, function(result) { + RemoteStorage.log('[WireClient] Successful request with unknown or binary mime-type', revision); + promise.fulfill(response.status, result, mimeType, revision); + }); + } else { + if (mimeType && mimeType.match(/^application\/json/)) { + body = JSON.parse(response.responseText); + } else { + body = response.responseText; + } + RemoteStorage.log('[WireClient] Successful request', revision); + promise.fulfill(response.status, body, mimeType, revision); + } + } + } + }); + return promise; + }, + + configure: function(userAddress, href, storageApi, token) { + if (typeof(userAddress) !== 'undefined') { + this.userAddress = userAddress; + } + if (typeof(href) !== 'undefined') { + this.href = href; + } + if (typeof(storageApi) !== 'undefined') { + this.storageApi = storageApi; + } + if (typeof(token) !== 'undefined') { + this.token = token; + } + if (typeof(this.storageApi) !== 'undefined') { + this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD; + this.supportsRevs = this._storageApi >= API_00; + } + if (this.href && this.token) { + this.connected = true; + this.online = true; + this._emit('connected'); + } else { + this.connected = false; + } + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ + userAddress: this.userAddress, + href: this.href, + token: this.token, + storageApi: this.storageApi + }); + } + RS.WireClient.configureHooks.forEach(function(hook) { + hook.call(this); + }.bind(this)); + }, + + stopWaitingForToken: function() { + if (!this.connected) { + this._emit('not-connected'); + } + }, + + get: function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } else if (options.ifNoneMatch) { + var oldRev = this._revisionCache[path]; + } + var promise = this._request('GET', this.href + cleanPath(path), this.token, headers, + undefined, this.supportsRevs, this._revisionCache[path]); + if (isFolder(path)) { + return promise.then(function(status, body, contentType, revision) { + var itemsMap = {}; + + // New folder listing received + if (status === 200 && typeof(body) === 'object') { + // Empty folder listing of any spec + if (Object.keys(body).length === 0) { + status = 404; + } + // >= 02 spec + else if (isFolderDescription(body)) { + for (var item in body.items) { + this._revisionCache[path + item] = body.items[item].ETag; + } + itemsMap = body.items; + } + // < 02 spec + else { + Object.keys(body).forEach(function(key){ + this._revisionCache[path + key] = body[key]; + itemsMap[key] = {"ETag": body[key]}; + }.bind(this)); + } + return promising().fulfill(status, itemsMap, contentType, revision); + } else { + return promising().fulfill(status, body, contentType, revision); + } + }.bind(this)); + } else { + return promise; + } + }, + + put: function(path, body, contentType, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + if (!contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var headers = { 'Content-Type': contentType }; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + if (options.ifNoneMatch) { + headers['If-None-Match'] = addQuotes(options.ifNoneMatch); + } + } + return this._request('PUT', this.href + cleanPath(path), this.token, + headers, body, this.supportsRevs); + }, + + 'delete': function(path, options) { + if (!this.connected) { + throw new Error("not connected (path: " + path + ")"); + } + if (!options) { options = {}; } + var headers = {}; + if (this.supportsRevs) { + if (options.ifMatch) { + headers['If-Match'] = addQuotes(options.ifMatch); + } + } + return this._request('DELETE', this.href + cleanPath(path), this.token, + headers, + undefined, this.supportsRevs); + } + }; + + // Shared cleanPath used by Dropbox + RS.WireClient.cleanPath = cleanPath; + + // Shared isArrayBufferView used by WireClient and Dropbox + RS.WireClient.isArrayBufferView = isArrayBufferView; + + RS.WireClient.readBinaryData = readBinaryData; + + // Shared request function used by WireClient, GoogleDrive and Dropbox. + RS.WireClient.request = function(method, url, options, callback) { + RemoteStorage.log('[WireClient]', method, url); + + callback = callback.bind(this); + + var timedOut = false; + + var timer = setTimeout(function() { + timedOut = true; + callback('timeout'); + }, RS.WireClient.REQUEST_TIMEOUT); + + var xhr = new XMLHttpRequest(); + xhr.open(method, url, true); + + if (options.responseType) { + xhr.responseType = options.responseType; + } + if (options.headers) { + for (var key in options.headers) { + xhr.setRequestHeader(key, options.headers[key]); + } + } + + xhr.onload = function() { + if (timedOut) { return; } + clearTimeout(timer); + callback(null, xhr); + }; + + xhr.onerror = function(error) { + if (timedOut) { return; } + clearTimeout(timer); + callback(error); + }; + + var body = options.body; + + if (typeof(body) === 'object') { + if (isArrayBufferView(body)) { + /* alright. */ + //FIXME empty block + } + else if (body instanceof ArrayBuffer) { + body = new Uint8Array(body); + } else { + body = JSON.stringify(body); + } + } + xhr.send(body); + }; + + Object.defineProperty(RemoteStorage.WireClient.prototype, 'storageType', { + get: function() { + if (this.storageApi) { + var spec = this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/); + return spec ? spec[1] : '2012.04'; + } + } + }); + + RS.WireClient.configureHooks = []; + + RS.WireClient._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + remoteStorage.remote = new RS.WireClient(remoteStorage); + this.online = true; + }; + + RS.WireClient._rs_supported = function() { + return !! global.XMLHttpRequest; + }; + + RS.WireClient._rs_cleanup = function(remoteStorage){ + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + remoteStorage.removeEventListener('error', onErrorCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/discover.js **/ +(function(global) { + + // feature detection flags + var haveXMLHttpRequest, hasLocalStorage; + // used to store settings in localStorage + var SETTINGS_KEY = 'remotestorage:discover'; + // cache loaded from localStorage + var cachedInfo = {}; + + /** + * Class: RemoteStorage.Discover + * + * This class deals with the webfinger lookup + * + * Arguments: + * userAddress - user@host + * callback - gets called with href of the storage, the type and the authURL + * Example: + * (start code) + * + * (end code) + **/ + + RemoteStorage.Discover = function(userAddress, callback) { + if (userAddress in cachedInfo) { + var info = cachedInfo[userAddress]; + callback(info.href, info.type, info.authURL); + return; + } + var hostname = userAddress.split('@')[1]; + var params = '?resource=' + encodeURIComponent('acct:' + userAddress); + var urls = [ + 'https://' + hostname + '/.well-known/webfinger' + params, + 'https://' + hostname + '/.well-known/host-meta.json' + params, + 'http://' + hostname + '/.well-known/webfinger' + params, + 'http://' + hostname + '/.well-known/host-meta.json' + params + ]; + + function tryOne() { + var xhr = new XMLHttpRequest(); + var url = urls.shift(); + if (!url) { return callback(); } + RemoteStorage.log('[Discover] Trying URL', url); + xhr.open('GET', url, true); + xhr.onabort = xhr.onerror = function() { + console.error("webfinger error", arguments, '(', url, ')'); + tryOne(); + }; + xhr.onload = function() { + if (xhr.status !== 200) { return tryOne(); } + var profile; + + try { + profile = JSON.parse(xhr.responseText); + } catch(e) { + RemoteStorage.log("[Discover] Failed to parse profile ", xhr.responseText, e); + tryOne(); + return; + } + + if (!profile.links) { + RemoteStorage.log("[Discover] Profile has no links section ", JSON.stringify(profile)); + tryOne(); + return; + } + + var link; + profile.links.forEach(function(l) { + if (l.rel === 'remotestorage') { + link = l; + } else if (l.rel === 'remoteStorage' && !link) { + link = l; + } + }); + RemoteStorage.log('[Discover] Got profile', profile, 'and link', link); + if (link) { + var authURL = link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'] + || link.properties['auth-endpoint'], + storageType = link.properties['http://remotestorage.io/spec/version'] + || link.type; + cachedInfo[userAddress] = { href: link.href, type: storageType, authURL: authURL }; + if (hasLocalStorage) { + localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo }); + } + callback(link.href, storageType, authURL); + } else { + tryOne(); + } + }; + xhr.send(); + } + tryOne(); + }; + + RemoteStorage.Discover._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (hasLocalStorage) { + var settings; + try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {} + if (settings) { + cachedInfo = settings.cache; + } + } + }; + + RemoteStorage.Discover._rs_supported = function() { + haveXMLHttpRequest = !! global.XMLHttpRequest; + return haveXMLHttpRequest; + }; + + RemoteStorage.Discover._rs_cleanup = function() { + if (hasLocalStorage) { + delete localStorage[SETTINGS_KEY]; + } + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/authorize.js **/ +(function(global) { + + function extractParams() { + //FF already decodes the URL fragment in document.location.hash, so use this instead: + var location = RemoteStorage.Authorize.getLocation(), + hashPos = location.href.indexOf('#'), + hash; + if (hashPos === -1) { return; } + hash = location.href.substring(hashPos+1); + // if hash is not of the form #key=val&key=val, it's probably not for us + if (hash.indexOf('=') === -1) { return; } + return hash.split('&').reduce(function(m, kvs) { + var kv = kvs.split('='); + m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]); + return m; + }, {}); + } + + RemoteStorage.Authorize = function(authURL, scope, redirectUri, clientId) { + RemoteStorage.log('[Authorize] authURL = ', authURL); + + var url = authURL; + url += authURL.indexOf('?') > 0 ? '&' : '?'; + url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, '')); + url += '&scope=' + encodeURIComponent(scope); + url += '&client_id=' + encodeURIComponent(clientId); + url += '&response_type=token'; + RemoteStorage.Authorize.setLocation(url); + }; + + RemoteStorage.prototype.authorize = function(authURL) { + this.access.setStorageType(this.remote.storageType); + var scope = this.access.scopeParameter; + + var redirectUri = String(RemoteStorage.Authorize.getLocation()); + var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0]; + + RemoteStorage.Authorize(authURL, scope, redirectUri, clientId); + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.getLocation = function () { + return global.document.location; + }; + + /** + * Get current document location + * + * Override this method if access to document.location is forbidden + */ + RemoteStorage.Authorize.setLocation = function (location) { + if (typeof location === 'string') { + global.document.location.href = location; + } else if (typeof location === 'object') { + global.document.location = location; + } else { + throw "Invalid location " + location; + } + }; + + RemoteStorage.Authorize._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + + var onFeaturesLoaded; + RemoteStorage.Authorize._rs_init = function(remoteStorage) { + + onFeaturesLoaded = function () { + var authParamsUsed = false; + if (params) { + if (params.error) { + throw "Authorization server errored: " + params.error; + } + if (params.access_token) { + remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token); + authParamsUsed = true; + } + if (params.remotestorage) { + remoteStorage.connect(params.remotestorage); + authParamsUsed = true; + } + } + if (!authParamsUsed) { + remoteStorage.remote.stopWaitingForToken(); + } + }; + var params = extractParams(), + location; + if (params) { + location = RemoteStorage.Authorize.getLocation(); + location.hash = ''; + } + remoteStorage.on('features-loaded', onFeaturesLoaded); + }; + + RemoteStorage.Authorize._rs_cleanup = function(remoteStorage) { + remoteStorage.removeEventListener('features-loaded', onFeaturesLoaded); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/access.js **/ +(function(global) { + + var SETTINGS_KEY = "remotestorage:access"; + + /** + * Class: RemoteStorage.Access + * + * Keeps track of claimed access and scopes. + */ + RemoteStorage.Access = function() { + this.reset(); + }; + + RemoteStorage.Access.prototype = { + + /** + * Method: claim + * + * Claim access on a given scope with given mode. + * + * Parameters: + * scope - An access scope, such as "contacts" or "calendar". + * mode - Access mode to use. Either "r" or "rw". + * + * Example: + * (start code) + * remoteStorage.access.claim('contacts', 'r'); + * remoteStorage.access.claim('pictures', 'rw'); + */ + claim: function(scope, mode) { + if (typeof(scope) !== 'string' || scope.indexOf('/') !== -1 || scope.length === 0) { + throw new Error('Scope should be a non-empty string without forward slashes'); + } + if (!mode.match(/^rw?$/)) { + throw new Error('Mode should be either \'r\' or \'rw\''); + } + this._adjustRootPaths(scope); + this.scopeModeMap[scope] = mode; + }, + + get: function(scope) { + return this.scopeModeMap[scope]; + }, + + remove: function(scope) { + var savedMap = {}; + var name; + for (name in this.scopeModeMap) { + savedMap[name] = this.scopeModeMap[name]; + } + this.reset(); + delete savedMap[scope]; + for (name in savedMap) { + this.set(name, savedMap[name]); + } + }, + + /** + * Verify permission for a given scope. + */ + checkPermission: function(scope, mode) { + var actualMode = this.get(scope); + return actualMode && (mode === 'r' || actualMode === 'rw'); + }, + + /** + * Verify permission for a given path. + */ + checkPathPermission: function(path, mode) { + if (this.checkPermission('*', mode)) { + return true; + } + return !!this.checkPermission(this._getModuleName(path), mode); + }, + + reset: function() { + this.rootPaths = []; + this.scopeModeMap = {}; + }, + + /** + * Return the module name for a given path. + */ + _getModuleName: function(path) { + if (path[0] !== '/') { + throw new Error('Path should start with a slash'); + } + var moduleMatch = path.replace(/^\/public/, '').match(/^\/([^\/]*)\//); + return moduleMatch ? moduleMatch[1] : '*'; + }, + + _adjustRootPaths: function(newScope) { + if ('*' in this.scopeModeMap || newScope === '*') { + this.rootPaths = ['/']; + } else if (! (newScope in this.scopeModeMap)) { + this.rootPaths.push('/' + newScope + '/'); + this.rootPaths.push('/public/' + newScope + '/'); + } + }, + + _scopeNameForParameter: function(scope) { + if (scope.name === '*' && this.storageType) { + if (this.storageType === '2012.04') { + return ''; + } else if (this.storageType.match(/remotestorage-0[01]/)) { + return 'root'; + } + } + return scope.name; + }, + + setStorageType: function(type) { + this.storageType = type; + } + }; + + /** + * Property: scopes + * + * Holds an array of claimed scopes in the form + * > { name: "", mode: "" } + */ + Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', { + get: function() { + return Object.keys(this.scopeModeMap).map(function(key) { + return { name: key, mode: this.scopeModeMap[key] }; + }.bind(this)); + } + }); + + Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', { + get: function() { + return this.scopes.map(function(scope) { + return this._scopeNameForParameter(scope) + ':' + scope.mode; + }.bind(this)).join(' '); + } + }); + + // Documented in src/remotestorage.js + Object.defineProperty(RemoteStorage.prototype, 'access', { + get: function() { + var access = new RemoteStorage.Access(); + Object.defineProperty(this, 'access', { + value: access + }); + return access; + }, + configurable: true + }); + + RemoteStorage.Access._rs_init = function() {}; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/env.js **/ +(function(pMode) { + + var mode = pMode, + env = {}, + isBackground = false; + + + RemoteStorage.Env = function() { + return env; + }; + + RemoteStorage.Env.isBrowser = function () { + return mode === "browser"; + }; + + RemoteStorage.Env.isNode = function () { + return mode === "node"; + }; + + RemoteStorage.Env.goBackground = function () { + isBackground = true; + RemoteStorage.Env._emit("background"); + }; + + RemoteStorage.Env.goForeground = function () { + isBackground = false; + RemoteStorage.Env._emit("foreground"); + }; + + RemoteStorage.Env._rs_init = function(remoteStorage) { + RemoteStorage.eventHandling(RemoteStorage.Env, "background", "foreground"); + + if ( mode === 'browser') { + if ( typeof(document.hidden) !== "undefined" ) { + env.hiddenProperty = "hidden"; + env.visibilityChangeEvent = "visibilitychange"; + } else if ( typeof(document.mozHidden) !== "undefined" ) { + env.hiddenProperty = "mozHidden"; + env.visibilityChangeEvent = "mozvisibilitychange"; + } else if ( typeof(document.msHidden) !== "undefined" ) { + env.hiddenProperty = "msHidden"; + env.visibilityChangeEvent = "msvisibilitychange"; + } else if ( typeof(document.webkitHidden) !== "undefined" ) { + env.hiddenProperty = "webkitHidden"; + env.visibilityChangeEvent = "webkitvisibilitychange"; + } + document.addEventListener(env.visibilityChangeEvent, function () { + if ( document[env.hiddenProperty] ) { + RemoteStorage.Env.goBackground(); + } else { + RemoteStorage.Env.goForeground(); + } + }, false); + } + }; + + RemoteStorage.Env._rs_cleanup = function(remoteStorage) { + }; + +})(typeof(window) !== 'undefined' ? 'browser' : 'node'); + + +/** FILE: src/i18n.js **/ +(function() { + /** + * Class: RemoteStorage.I18n + * + * TODO add documentation + **/ + + "use strict"; + + var dictionary = { + "view_info": 'This app allows you to use your own storage. Learn more!', + "view_connect": "Connect remote storage", + "view_connecting": "Connecting %s", + "view_offline": "Offline", + "view_error_occured": "Sorry! An error occured.", + "view_confirm_reset": "Are you sure you want to reset everything? This will clear your local data and reload the page.", + "view_get_me_out": "Get me out of here!", + "view_error_plz_report": 'If this problem persists, please let us know!', + "view_unauthorized": "Unauthorized! Click here to reconnect." + }; + + RemoteStorage.I18n = { + + translate: function() { + var str = arguments[0], + params = Array.prototype.splice.call(arguments, 1); + + if (typeof dictionary[str] !== "string") { + throw "Unknown translation string: " + str; + } else { + str = dictionary[str]; + } + return (str.replace(/%s/g, function(){ return params.shift(); })); + }, + + getDictionary: function() { + return dictionary; + }, + + setDictionary: function(newDictionary) { + dictionary = newDictionary; + } + + }; +})(); + + +/** FILE: src/assets.js **/ +/** THIS FILE WAS GENERATED BY build/compile-assets.js. DO NOT CHANGE IT MANUALLY, BUT INSTEAD CHANGE THE ASSETS IN assets/. **/ +RemoteStorage.Assets = { + + connectIcon: '', + disconnectIcon: '', + dropbox: '', + googledrive: '', + remoteStorageIcon: '', + remoteStorageIconError: '', + remoteStorageIconOffline: '', + syncIcon: '', + widget: '
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ', + widgetCss: '/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}' +}; + + +/** FILE: src/widget.js **/ +(function(window) { + + var hasLocalStorage; + var LS_STATE_KEY = "remotestorage:widget:state"; + // states allowed to immediately jump into after a reload. + var VALID_ENTRY_STATES = { + initial: true, + connected: true, + offline: true + }; + + function stateSetter(widget, state) { + RemoteStorage.log('[Widget] Producing stateSetter for', state); + return function() { + RemoteStorage.log('[Widget] Setting state', state, arguments); + if (hasLocalStorage) { + localStorage[LS_STATE_KEY] = state; + } + if (widget.view) { + if (widget.rs.remote) { + widget.view.setUserAddress(widget.rs.remote.userAddress); + } + widget.view.setState(state, arguments); + } else { + widget._rememberedState = state; + } + }; + } + + function errorsHandler(widget) { + return function(error) { + if (error instanceof RemoteStorage.DiscoveryError) { + console.error('Discovery failed', error, '"' + error.message + '"'); + widget.view.setState('initial', [error.message]); + } else if (error instanceof RemoteStorage.SyncError) { + widget.view.setState('offline', []); + } else if (error instanceof RemoteStorage.Unauthorized) { + widget.view.setState('unauthorized'); + } else { + RemoteStorage.log('[Widget] Unknown error'); + widget.view.setState('error', [error]); + } + }; + } + + function flashFor(evt) { + if (evt.method === 'GET' && evt.isFolder) { + return false; + } + return true; + } + + /** + * Class: RemoteStorage.Widget + * + * The widget controller that communicates with the view and listens to + * its remoteStorage instance. + * + * While listening to the events emitted by its remoteStorage it sets + * corresponding states of the view. + * + * - connected -> connected + * - disconnected -> initial + * - connecting -> authing + * - authing -> authing + * - wire-busy -> busy + * - wire-done -> connected + * - error -> one of initial, offline, unauthorized, or error + **/ + RemoteStorage.Widget = function(remoteStorage) { + var self = this; + var requestsToFlashFor = 0; + + // setting event listeners on rs events to put + // the widget into corresponding states + this.rs = remoteStorage; + this.rs.remote.on('connected', stateSetter(this, 'connected')); + this.rs.on('disconnected', stateSetter(this, 'initial')); + this.rs.on('connecting', stateSetter(this, 'authing')); + this.rs.on('authing', stateSetter(this, 'authing')); + this.rs.on('error', errorsHandler(this)); + + if (this.rs.remote) { + this.rs.remote.on('wire-busy', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor++; + stateSetter(self, 'busy')(); + } + }); + + this.rs.remote.on('wire-done', function(evt) { + if (flashFor(evt)) { + requestsToFlashFor--; + if (requestsToFlashFor <= 0) { + stateSetter(self, 'connected')(); + } + } + }); + } + + if (hasLocalStorage) { + var state = localStorage[LS_STATE_KEY]; + if (state && VALID_ENTRY_STATES[state]) { + this._rememberedState = state; + } + } + }; + + RemoteStorage.Widget.prototype = { + + /** + * Method: display + * + * Displays the widget via the view.display method + * + * Parameters: + * + * domID + **/ + display: function(domID) { + if (! this.view) { + this.setView(new RemoteStorage.Widget.View(this.rs)); + } + this.view.display.apply(this.view, arguments); + return this; + }, + + linkWidgetToSync: function() { + if (typeof(this.rs.sync) === 'object' && typeof(this.rs.sync.sync) === 'function') { + this.view.on('sync', this.rs.sync.sync.bind(this.rs.sync)); + } else { + RemoteStorage.log('[Widget] typeof this.rs.sync check fail', this.rs.sync); + setTimeout(this.linkWidgetToSync.bind(this), 1000); + } + }, + + /** + * Method: setView(view) + * + * Sets the view and initializes event listeners to react on + * widget (widget.view) events + **/ + setView: function(view) { + this.view = view; + this.view.on('connect', function(options) { + if (typeof(options) === 'string') { + // options is simply a useraddress + this.rs.connect(options); + } else if (options.special) { + this.rs[options.special].connect(options); + } + }.bind(this)); + this.view.on('disconnect', this.rs.disconnect.bind(this.rs)); + this.linkWidgetToSync(); + try { + this.view.on('reset', function(){ + var location = RemoteStorage.Authorize.getLocation(); + this.rs.on('disconnected', location.reload.bind(location)); + this.rs.disconnect(); + }.bind(this)); + } catch(e) { + if (e.message && e.message.match(/Unknown event/)) { + // ignored. (the 0.7 widget-view interface didn't have a 'reset' event) + } else { + throw e; + } + } + + if (this._rememberedState) { + setTimeout(stateSetter(this, this._rememberedState), 0); + delete this._rememberedState; + } + } + }; + + /** + * Method: displayWidget + * + * Same as + **/ + RemoteStorage.prototype.displayWidget = function(domID) { + return this.widget.display(domID); + }; + + RemoteStorage.Widget._rs_init = function(remoteStorage) { + hasLocalStorage = remoteStorage.localStorageAvailable(); + if (! remoteStorage.widget) { + remoteStorage.widget = new RemoteStorage.Widget(remoteStorage); + } + }; + + RemoteStorage.Widget._rs_supported = function(remoteStorage) { + return typeof(document) !== 'undefined'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/view.js **/ +(function(window){ + + var t = RemoteStorage.I18n.translate; + + var cEl = function(){ + return document.createElement.apply(document, arguments); + }; + + function gCl(parent, className) { + return parent.getElementsByClassName(className)[0]; + } + + function gTl(parent, tagName) { + return parent.getElementsByTagName(tagName)[0]; + } + + function removeClass(el, className) { + return el.classList.remove(className); + } + + function addClass(el, className) { + return el.classList.add(className); + } + + function stopPropagation(event) { + if (typeof(event.stopPropagation) === 'function') { + event.stopPropagation(); + } else { + event.cancelBubble = true; + } + } + + function setupButton(parent, className, iconName, eventListener) { + var element = gCl(parent, className); + if (typeof iconName !== 'undefined') { + var img = gTl(element, 'img'); + (img || element).src = RemoteStorage.Assets[iconName]; + } + element.addEventListener('click', eventListener); + return element; + } + + /** + * Class: RemoteStorage.Widget.View + * + * Controls the visible widget + * + * States: + * + * initial - not connected + * authing - in auth flow + * connected - connected to remote storage, not syncing at the moment + * busy - connected, syncing at the moment + * offline - connected, but no network connectivity + * error - connected, but sync error happened + * unauthorized - connected, but request returned 401 + **/ + RemoteStorage.Widget.View = function(remoteStorage) { + this.rs = remoteStorage; + if (typeof(document) === 'undefined') { + throw "Widget not supported"; + } + RemoteStorage.eventHandling(this, + 'connect', + 'disconnect', + 'sync', + 'display', + 'reset'); + + // Re-binding the event so they can be called from the window + for (var event in this.events){ + this.events[event] = this.events[event].bind(this); + } + + this.hideBubbleOnBodyClick = function(event) { + for (var p = event.target; p !== document.body; p = p.parentElement) { + if (p.id === 'remotestorage-widget') { + return; + } + } + this.hideBubble(); + }.bind(this); + }; + + RemoteStorage.Widget.View.prototype = { + + connectGdrive: function() { + this._emit('connect', { special: 'googledrive' }); + }, + + connectDropbox: function(){ + this._emit('connect', { special: 'dropbox'}); + }, + + /** + * Method: setState + * + * Call the function that applies the state to the widget + * + * Parameters: + * + * state + * args + **/ + setState: function(state, args) { + RemoteStorage.log('[View] widget.view.setState(',state,',',args,');'); + var s = this.states[state]; + if (typeof(s) === 'undefined') { + throw new Error("Bad State assigned to view: " + state); + } + s.apply(this, args); + }, + + /** + * Method: setUserAddres + * + * Set user address of the input field + **/ + setUserAddress: function(addr) { + this.userAddress = addr || ''; + + var el; + if (this.div && (el = gTl(this.div, 'form').userAddress)) { + el.value = this.userAddress; + } + }, + + /** + * Method: toggleBubble + * + * Show the bubble when hidden and the other way around + **/ + toggleBubble: function(event) { + if (this.bubble.className.search('rs-hidden') < 0) { + this.hideBubble(event); + } else { + this.showBubble(event); + } + }, + + /** + * Method: hideBubble + * + * Hide the bubble + **/ + hideBubble: function(){ + addClass(this.bubble, 'rs-hidden'); + document.body.removeEventListener('click', this.hideBubbleOnBodyClick); + }, + + /** + * Method: showBubble + * + * Show the bubble + **/ + showBubble: function(event){ + removeClass(this.bubble, 'rs-hidden'); + if (typeof(event) !== 'undefined') { + stopPropagation(event); + } + document.body.addEventListener('click', this.hideBubbleOnBodyClick); + gTl(this.bubble,'form').userAddress.focus(); + }, + + /** + * Method: display + * + * Draw the widget inside of the dom element with the id domID + * + * Parameters: + * + * domID + * + * Returns: + * + * The widget div + **/ + display: function(domID) { + if (typeof this.div !== 'undefined') { + return this.div; + } + + var element = cEl('div'); + var style = cEl('style'); + style.innerHTML = RemoteStorage.Assets.widgetCss; + + element.id = "remotestorage-widget"; + + element.innerHTML = RemoteStorage.Assets.widget; + + element.appendChild(style); + if (domID) { + var parent = document.getElementById(domID); + if (! parent) { + throw "Failed to find target DOM element with id=\"" + domID + "\""; + } + parent.appendChild(element); + } else { + document.body.appendChild(element); + } + + // Sync button + setupButton(element, 'rs-sync', 'syncIcon', this.events.sync); + + // Disconnect button + setupButton(element, 'rs-disconnect', 'disconnectIcon', this.events.disconnect); + + // Get me out of here + setupButton(element, 'remotestorage-reset', undefined, this.events.reset); + + // Connect button + var cb = setupButton(element, 'connect', 'connectIcon', this.events.connect); + + // Input + this.form = gTl(element, 'form'); + var el = this.form.userAddress; + el.addEventListener('keyup', function(event) { + if (event.target.value) { + cb.removeAttribute('disabled'); + } else { + cb.setAttribute('disabled','disabled'); + } + }); + if (this.userAddress) { + el.value = this.userAddress; + } + + // The cube + this.cube = setupButton(element, 'rs-cube', 'remoteStorageIcon', this.toggleBubble.bind(this)); + + // Google Drive and Dropbox icons + setupButton(element, 'rs-dropbox', 'dropbox', this.connectDropbox.bind(this)); + setupButton(element, 'rs-googledrive', 'googledrive', this.connectGdrive.bind(this)); + + var bubbleDontCatch = { INPUT: true, BUTTON: true, IMG: true }; + var eventListener = function(event) { + if (! bubbleDontCatch[event.target.tagName] && ! (this.div.classList.contains('remotestorage-state-unauthorized') )) { + this.showBubble(event); + } + }.bind(this); + this.bubble = setupButton(element, 'rs-bubble', undefined, eventListener); + + this.hideBubble(); + + this.div = element; + + this.states.initial.call(this); + this.events.display.call(this); + return this.div; + }, + + _renderTranslatedInitialContent: function() { + gCl(this.div, 'rs-status-text').innerHTML = t("view_connect"); + gCl(this.div, 'remotestorage-reset').innerHTML = t("view_get_me_out"); + gCl(this.div, 'rs-error-plz-report').innerHTML = t("view_error_plz_report"); + gCl(this.div, 'remotestorage-unauthorized').innerHTML = t("view_unauthorized"); + }, + + states: { + initial: function(message) { + var cube = this.cube; + var info = message || t("view_info"); + + this._renderTranslatedInitialContent(); + + if (message) { + cube.src = RemoteStorage.Assets.remoteStorageIconError; + removeClass(this.cube, 'remotestorage-loading'); + this.showBubble(); + + // Show the red error cube for 5 seconds, then show the normal orange one again + setTimeout(function(){ + cube.src = RemoteStorage.Assets.remoteStorageIcon; + },5000); + } else { + this.hideBubble(); + } + this.div.className = "remotestorage-state-initial"; + + // Google Drive and Dropbox icons + var backends = 1; + if (this._activateBackend('dropbox')) { backends += 1; } + if (this._activateBackend('googledrive')) { backends += 1; } + gCl(this.div, 'rs-bubble-text').style.paddingRight = backends*32+8+'px'; + + // If address not empty connect button enabled + var cb = gCl(this.div, 'connect'); + if (this.form.userAddress.value) { + cb.removeAttribute('disabled'); + } + + var infoEl = gCl(this.div, 'rs-info-msg'); + infoEl.innerHTML = info; + + if (message) { + infoEl.classList.add('remotestorage-error-info'); + } else { + infoEl.classList.remove('remotestorage-error-info'); + } + }, + + authing: function() { + this.div.removeEventListener('click', this.events.connect); + this.div.className = "remotestorage-state-authing"; + gCl(this.div, 'rs-status-text').innerHTML = t("view_connecting", this.userAddress); + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone, when is that neccesary + }, + + connected: function() { + this.div.className = "remotestorage-state-connected"; + gCl(this.div, 'userAddress').innerHTML = this.userAddress; + this.cube.src = RemoteStorage.Assets.remoteStorageIcon; + removeClass(this.cube, 'remotestorage-loading'); + var icons = { + googledrive: gCl(this.div, 'rs-googledrive'), + dropbox: gCl(this.div, 'rs-dropbox') + }; + icons.googledrive.style.display = icons.dropbox.style.display = 'none'; + if (icons[this.rs.backend]) { + icons[this.rs.backend].style.display = 'inline-block'; + gCl(this.div, 'rs-bubble-text').style.paddingRight = 2*32+8+'px'; + } else { + gCl(this.div, 'rs-bubble-text').style.paddingRight = 32+8+'px'; + } + }, + + busy: function() { + this.div.className = "remotestorage-state-busy"; + addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone when is that neccesary + this.hideBubble(); + }, + + offline: function() { + this.div.className = "remotestorage-state-offline"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconOffline; + gCl(this.div, 'rs-status-text').innerHTML = t("view_offline"); + }, + + error: function(err) { + var errorMsg = err; + this.div.className = "remotestorage-state-error"; + + gCl(this.div, 'rs-bubble-text').innerHTML = ''+t('view_error_occured')+''; + //FIXME I don't know what an DOMError is and my browser doesn't know too(how to handle this?) + if (err instanceof Error /*|| err instanceof DOMError*/) { + errorMsg = err.message + '\n\n' + + err.stack; + } + gCl(this.div, 'rs-error-msg').textContent = errorMsg; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + }, + + unauthorized: function() { + this.div.className = "remotestorage-state-unauthorized"; + this.cube.src = RemoteStorage.Assets.remoteStorageIconError; + this.showBubble(); + this.div.addEventListener('click', this.events.connect); + } + }, + + events: { + /** + * Event: connect + * + * Emitted when the connect button is clicked + **/ + connect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('connect', gTl(this.div, 'form').userAddress.value); + }, + + /** + * Event: sync + * + * Emitted when the sync button is clicked + **/ + sync: function(event) { + stopPropagation(event); + event.preventDefault(); + + this._emit('sync'); + }, + + /** + * Event: disconnect + * + * Emitted when the disconnect button is clicked + **/ + disconnect: function(event) { + stopPropagation(event); + event.preventDefault(); + this._emit('disconnect'); + }, + + /** + * Event: reset + * + * Emitted after crash triggers disconnect + **/ + reset: function(event){ + event.preventDefault(); + var result = window.confirm(t('view_confirm_reset')); + if (result){ + this._emit('reset'); + } + }, + + /** + * Event: display + * + * Emitted when finished displaying the widget + **/ + display : function(event) { + if (event) { + event.preventDefault(); + } + this._emit('display'); + } + }, + + _activateBackend: function activateBackend(backendName) { + var className = 'rs-' + backendName; + if (this.rs.apiKeys[backendName]) { + gCl(this.div, className).style.display = 'inline-block'; + return true; + } else { + gCl(this.div, className).style.display = 'none'; + return false; + } + } + }; +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: lib/tv4.js **/ +/** +Author: Geraint Luff and others +Year: 2013 + +This code is released into the "public domain" by its author(s). Anybody may use, alter and distribute the code without restriction. The author makes no guarantees, and takes no liability of any kind for use of this code. + +If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory. +**/ + +(function (global) { +var ValidatorContext = function (parent, collectMultiple) { + this.missing = []; + this.schemas = parent ? Object.create(parent.schemas) : {}; + this.collectMultiple = collectMultiple; + this.errors = []; + this.handleError = collectMultiple ? this.collectError : this.returnError; +}; +ValidatorContext.prototype.returnError = function (error) { + return error; +}; +ValidatorContext.prototype.collectError = function (error) { + if (error) { + this.errors.push(error); + } + return null; +} +ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) { + for (var i = startIndex; i < this.errors.length; i++) { + this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath); + } + return this; +} + +ValidatorContext.prototype.getSchema = function (url) { + if (this.schemas[url] != undefined) { + var schema = this.schemas[url]; + return schema; + } + var baseUrl = url; + var fragment = ""; + if (url.indexOf('#') != -1) { + fragment = url.substring(url.indexOf("#") + 1); + baseUrl = url.substring(0, url.indexOf("#")); + } + if (this.schemas[baseUrl] != undefined) { + var schema = this.schemas[baseUrl]; + var pointerPath = decodeURIComponent(fragment); + if (pointerPath == "") { + return schema; + } else if (pointerPath.charAt(0) != "/") { + return undefined; + } + var parts = pointerPath.split("/").slice(1); + for (var i = 0; i < parts.length; i++) { + var component = parts[i].replace("~1", "/").replace("~0", "~"); + if (schema[component] == undefined) { + schema = undefined; + break; + } + schema = schema[component]; + } + if (schema != undefined) { + return schema; + } + } + if (this.missing[baseUrl] == undefined) { + this.missing.push(baseUrl); + this.missing[baseUrl] = baseUrl; + } +}; +ValidatorContext.prototype.addSchema = function (url, schema) { + var map = {}; + map[url] = schema; + normSchema(schema, url); + searchForTrustedSchemas(map, schema, url); + for (var key in map) { + this.schemas[key] = map[key]; + } + return map; +}; + +ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) { + if (schema['$ref'] != undefined) { + schema = this.getSchema(schema['$ref']); + if (!schema) { + return null; + } + } + + var errorCount = this.errors.length; + var error = this.validateBasic(data, schema) + || this.validateNumeric(data, schema) + || this.validateString(data, schema) + || this.validateArray(data, schema) + || this.validateObject(data, schema) + || this.validateCombinations(data, schema) + || null + if (error || errorCount != this.errors.length) { + while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) { + var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null; + var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null; + if (error) { + error = error.prefixWith(dataPart, schemaPart); + } + this.prefixErrors(errorCount, dataPart, schemaPart); + } + } + + return this.handleError(error); +} + +function recursiveCompare(A, B) { + if (A === B) { + return true; + } + if (typeof A == "object" && typeof B == "object") { + if (Array.isArray(A) != Array.isArray(B)) { + return false; + } else if (Array.isArray(A)) { + if (A.length != B.length) { + return false + } + for (var i = 0; i < A.length; i++) { + if (!recursiveCompare(A[i], B[i])) { + return false; + } + } + } else { + for (var key in A) { + if (B[key] === undefined && A[key] !== undefined) { + return false; + } + } + for (var key in B) { + if (A[key] === undefined && B[key] !== undefined) { + return false; + } + } + for (var key in A) { + if (!recursiveCompare(A[key], B[key])) { + return false; + } + } + } + return true; + } + return false; +} + +ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) { + var error; + if (error = this.validateType(data, schema)) { + return error.prefixWith(null, "type"); + } + if (error = this.validateEnum(data, schema)) { + return error.prefixWith(null, "type"); + } + return null; +} + +ValidatorContext.prototype.validateType = function validateType(data, schema) { + if (schema.type == undefined) { + return null; + } + var dataType = typeof data; + if (data == null) { + dataType = "null"; + } else if (Array.isArray(data)) { + dataType = "array"; + } + var allowedTypes = schema.type; + if (typeof allowedTypes != "object") { + allowedTypes = [allowedTypes]; + } + + for (var i = 0; i < allowedTypes.length; i++) { + var type = allowedTypes[i]; + if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) { + return null; + } + } + return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType); +} + +ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) { + if (schema["enum"] == undefined) { + return null; + } + for (var i = 0; i < schema["enum"].length; i++) { + var enumVal = schema["enum"][i]; + if (recursiveCompare(data, enumVal)) { + return null; + } + } + return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data)); +} +ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) { + return this.validateMultipleOf(data, schema) + || this.validateMinMax(data, schema) + || null; +} + +ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) { + var multipleOf = schema.multipleOf || schema.divisibleBy; + if (multipleOf == undefined) { + return null; + } + if (typeof data == "number") { + if (data%multipleOf != 0) { + return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf); + } + } + return null; +} + +ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) { + if (typeof data != "number") { + return null; + } + if (schema.minimum != undefined) { + if (data < schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum"); + } + if (schema.exclusiveMinimum && data == schema.minimum) { + return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum"); + } + } + if (schema.maximum != undefined) { + if (data > schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum"); + } + if (schema.exclusiveMaximum && data == schema.maximum) { + return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum"); + } + } + return null; +} +ValidatorContext.prototype.validateString = function validateString(data, schema) { + return this.validateStringLength(data, schema) + || this.validateStringPattern(data, schema) + || null; +} + +ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) { + if (typeof data != "string") { + return null; + } + if (schema.minLength != undefined) { + if (data.length < schema.minLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength"); + } + } + if (schema.maxLength != undefined) { + if (data.length > schema.maxLength) { + return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength"); + } + } + return null; +} + +ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) { + if (typeof data != "string" || schema.pattern == undefined) { + return null; + } + var regexp = new RegExp(schema.pattern); + if (!regexp.test(data)) { + return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern"); + } + return null; +} +ValidatorContext.prototype.validateArray = function validateArray(data, schema) { + if (!Array.isArray(data)) { + return null; + } + return this.validateArrayLength(data, schema) + || this.validateArrayUniqueItems(data, schema) + || this.validateArrayItems(data, schema) + || null; +} + +ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) { + if (schema.minItems != undefined) { + if (data.length < schema.minItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxItems != undefined) { + if (data.length > schema.maxItems) { + var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) { + if (schema.uniqueItems) { + for (var i = 0; i < data.length; i++) { + for (var j = i + 1; j < data.length; j++) { + if (recursiveCompare(data[i], data[j])) { + var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems"); + if (this.handleError(error)) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) { + if (schema.items == undefined) { + return null; + } + var error; + if (Array.isArray(schema.items)) { + for (var i = 0; i < data.length; i++) { + if (i < schema.items.length) { + if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) { + return error; + } + } else if (schema.additionalItems != undefined) { + if (typeof schema.additionalItems == "boolean") { + if (!schema.additionalItems) { + error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems"); + if (this.handleError(error)) { + return error; + } + } + } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) { + return error; + } + } + } + } else { + for (var i = 0; i < data.length; i++) { + if (error = this.validateAll(data[i], schema.items, [i], ["items"])) { + return error; + } + } + } + return null; +} +ValidatorContext.prototype.validateObject = function validateObject(data, schema) { + if (typeof data != "object" || data == null || Array.isArray(data)) { + return null; + } + return this.validateObjectMinMaxProperties(data, schema) + || this.validateObjectRequiredProperties(data, schema) + || this.validateObjectProperties(data, schema) + || this.validateObjectDependencies(data, schema) + || null; +} + +ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) { + var keys = Object.keys(data); + if (schema.minProperties != undefined) { + if (keys.length < schema.minProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties"); + if (this.handleError(error)) { + return error; + } + } + } + if (schema.maxProperties != undefined) { + if (keys.length > schema.maxProperties) { + var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties"); + if (this.handleError(error)) { + return error; + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) { + if (schema.required != undefined) { + for (var i = 0; i < schema.required.length; i++) { + var key = schema.required[i]; + if (data[key] === undefined) { + var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required"); + if (this.handleError(error)) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) { + var error; + for (var key in data) { + var foundMatch = false; + if (schema.properties != undefined && schema.properties[key] != undefined) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) { + return error; + } + } + if (schema.patternProperties != undefined) { + for (var patternKey in schema.patternProperties) { + var regexp = new RegExp(patternKey); + if (regexp.test(key)) { + foundMatch = true; + if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) { + return error; + } + } + } + } + if (!foundMatch && schema.additionalProperties != undefined) { + if (typeof schema.additionalProperties == "boolean") { + if (!schema.additionalProperties) { + error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties"); + if (this.handleError(error)) { + return error; + } + } + } else { + if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) { + return error; + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) { + var error; + if (schema.dependencies != undefined) { + for (var depKey in schema.dependencies) { + if (data[depKey] !== undefined) { + var dep = schema.dependencies[depKey]; + if (typeof dep == "string") { + if (data[dep] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } else if (Array.isArray(dep)) { + for (var i = 0; i < dep.length; i++) { + var requiredKey = dep[i]; + if (data[requiredKey] === undefined) { + error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies"); + if (this.handleError(error)) { + return error; + } + } + } + } else { + if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) { + return error; + } + } + } + } + } + return null; +} + +ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) { + var error; + return this.validateAllOf(data, schema) + || this.validateAnyOf(data, schema) + || this.validateOneOf(data, schema) + || this.validateNot(data, schema) + || null; +} + +ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) { + if (schema.allOf == undefined) { + return null; + } + var error; + for (var i = 0; i < schema.allOf.length; i++) { + var subSchema = schema.allOf[i]; + if (error = this.validateAll(data, subSchema, [], ["allOf", i])) { + return error; + } + } + return null; +} + +ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) { + if (schema.anyOf == undefined) { + return null; + } + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.anyOf.length; i++) { + var subSchema = schema.anyOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["anyOf", i]); + + if (error == null && errorCount == this.errors.length) { + this.errors = this.errors.slice(0, startErrorCount); + return null; + } + if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf")); + } + } + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors); +} + +ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) { + if (schema.oneOf == undefined) { + return null; + } + var validIndex = null; + var errors = []; + var startErrorCount = this.errors.length; + for (var i = 0; i < schema.oneOf.length; i++) { + var subSchema = schema.oneOf[i]; + + var errorCount = this.errors.length; + var error = this.validateAll(data, subSchema, [], ["oneOf", i]); + + if (error == null && errorCount == this.errors.length) { + if (validIndex == null) { + validIndex = i; + } else { + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf"); + } + } else if (error) { + errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf")); + } + } + if (validIndex == null) { + errors = errors.concat(this.errors.slice(startErrorCount)); + this.errors = this.errors.slice(0, startErrorCount); + return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors); + } else { + this.errors = this.errors.slice(0, startErrorCount); + } + return null; +} + +ValidatorContext.prototype.validateNot = function validateNot(data, schema) { + if (schema.not == undefined) { + return null; + } + var oldErrorCount = this.errors.length; + var error = this.validateAll(data, schema.not); + var notErrors = this.errors.slice(oldErrorCount); + this.errors = this.errors.slice(0, oldErrorCount); + if (error == null && notErrors.length == 0) { + return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not") + } + return null; +} + +// parseURI() and resolveUrl() are from https://gist.github.com/1088850 +// - released as public domain by author ("Yaffle") - see comments on gist + +function parseURI(url) { + var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/); + // authority = '//' + user + ':' + pass '@' + hostname + ':' port + return (m ? { + href : m[0] || '', + protocol : m[1] || '', + authority: m[2] || '', + host : m[3] || '', + hostname : m[4] || '', + port : m[5] || '', + pathname : m[6] || '', + search : m[7] || '', + hash : m[8] || '' + } : null); +} + +function resolveUrl(base, href) {// RFC 3986 + + function removeDotSegments(input) { + var output = []; + input.replace(/^(\.\.?(\/|$))+/, '') + .replace(/\/(\.(\/|$))+/g, '/') + .replace(/\/\.\.$/, '/../') + .replace(/\/?[^\/]*/g, function (p) { + if (p === '/..') { + output.pop(); + } else { + output.push(p); + } + }); + return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : ''); + } + + href = parseURI(href || ''); + base = parseURI(base || ''); + + return !href || !base ? null : (href.protocol || base.protocol) + + (href.protocol || href.authority ? href.authority : base.authority) + + removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) + + (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) + + href.hash; +} + +function normSchema(schema, baseUri) { + if (baseUri == undefined) { + baseUri = schema.id; + } else if (typeof schema.id == "string") { + baseUri = resolveUrl(baseUri, schema.id); + schema.id = baseUri; + } + if (typeof schema == "object") { + if (Array.isArray(schema)) { + for (var i = 0; i < schema.length; i++) { + normSchema(schema[i], baseUri); + } + } else if (typeof schema['$ref'] == "string") { + schema['$ref'] = resolveUrl(baseUri, schema['$ref']); + } else { + for (var key in schema) { + if (key != "enum") { + normSchema(schema[key], baseUri); + } + } + } + } +} + +var ErrorCodes = { + INVALID_TYPE: 0, + ENUM_MISMATCH: 1, + ANY_OF_MISSING: 10, + ONE_OF_MISSING: 11, + ONE_OF_MULTIPLE: 12, + NOT_PASSED: 13, + // Numeric errors + NUMBER_MULTIPLE_OF: 100, + NUMBER_MINIMUM: 101, + NUMBER_MINIMUM_EXCLUSIVE: 102, + NUMBER_MAXIMUM: 103, + NUMBER_MAXIMUM_EXCLUSIVE: 104, + // String errors + STRING_LENGTH_SHORT: 200, + STRING_LENGTH_LONG: 201, + STRING_PATTERN: 202, + // Object errors + OBJECT_PROPERTIES_MINIMUM: 300, + OBJECT_PROPERTIES_MAXIMUM: 301, + OBJECT_REQUIRED: 302, + OBJECT_ADDITIONAL_PROPERTIES: 303, + OBJECT_DEPENDENCY_KEY: 304, + // Array errors + ARRAY_LENGTH_SHORT: 400, + ARRAY_LENGTH_LONG: 401, + ARRAY_UNIQUE: 402, + ARRAY_ADDITIONAL_ITEMS: 403 +}; + +function ValidationError(code, message, dataPath, schemaPath, subErrors) { + if (code == undefined) { + throw new Error ("No code supplied for error: "+ message); + } + this.code = code; + this.message = message; + this.dataPath = dataPath ? dataPath : ""; + this.schemaPath = schemaPath ? schemaPath : ""; + this.subErrors = subErrors ? subErrors : null; +} +ValidationError.prototype = { + prefixWith: function (dataPrefix, schemaPrefix) { + if (dataPrefix != null) { + dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1"); + this.dataPath = "/" + dataPrefix + this.dataPath; + } + if (schemaPrefix != null) { + schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1"); + this.schemaPath = "/" + schemaPrefix + this.schemaPath; + } + if (this.subErrors != null) { + for (var i = 0; i < this.subErrors.length; i++) { + this.subErrors[i].prefixWith(dataPrefix, schemaPrefix); + } + } + return this; + } +}; + +function searchForTrustedSchemas(map, schema, url) { + if (typeof schema.id == "string") { + if (schema.id.substring(0, url.length) == url) { + var remainder = schema.id.substring(url.length); + if ((url.length > 0 && url.charAt(url.length - 1) == "/") + || remainder.charAt(0) == "#" + || remainder.charAt(0) == "?") { + if (map[schema.id] == undefined) { + map[schema.id] = schema; + } + } + } + } + if (typeof schema == "object") { + for (var key in schema) { + if (key != "enum" && typeof schema[key] == "object") { + searchForTrustedSchemas(map, schema[key], url); + } + } + } + return map; +} + +var globalContext = new ValidatorContext(); + +var publicApi = { + validate: function (data, schema) { + var context = new ValidatorContext(globalContext); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + var added = context.addSchema("", schema); + var error = context.validateAll(data, schema); + this.error = error; + this.missing = context.missing; + this.valid = (error == null); + return this.valid; + }, + validateResult: function () { + var result = {}; + this.validate.apply(result, arguments); + return result; + }, + validateMultiple: function (data, schema) { + var context = new ValidatorContext(globalContext, true); + if (typeof schema == "string") { + schema = {"$ref": schema}; + } + context.addSchema("", schema); + context.validateAll(data, schema); + var result = {}; + result.errors = context.errors; + result.missing = context.missing; + result.valid = (result.errors.length == 0); + return result; + }, + addSchema: function (url, schema) { + return globalContext.addSchema(url, schema); + }, + getSchema: function (url) { + return globalContext.getSchema(url); + }, + missing: [], + error: null, + normSchema: normSchema, + resolveUrl: resolveUrl, + errorCodes: ErrorCodes +}; + +global.tv4 = publicApi; + +})(typeof(window) != 'undefined' ? window : global); + + + +/** FILE: lib/Math.uuid.js **/ +/*! + Math.uuid.js (v1.4) + http://www.broofa.com + mailto:robert@broofa.com + + Copyright (c) 2010 Robert Kieffer + Dual licensed under the MIT and GPL licenses. + + ******** + + Changes within remoteStorage.js: + 2012-10-31: + - added AMD wrapper + - moved extensions for Math object into exported object. +*/ + +/* + * Generate a random uuid. + * + * USAGE: Math.uuid(length, radix) + * length - the desired number of characters + * radix - the number of allowable values for each character. + * + * EXAMPLES: + * // No arguments - returns RFC4122, version 4 ID + * >>> Math.uuid() + * "92329D39-6F5C-4520-ABFC-AAB64544E172" + * + * // One argument - returns ID of the specified length + * >>> Math.uuid(15) // 15 character ID (default base=62) + * "VcydxgltxrVZSTV" + * + * // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62) + * >>> Math.uuid(8, 2) // 8 character ID (base=2) + * "01001010" + * >>> Math.uuid(8, 10) // 8 character ID (base=10) + * "47473046" + * >>> Math.uuid(8, 16) // 8 character ID (base=16) + * "098F4D35" + */ + // Private array of chars to use + var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); + +Math.uuid = function (len, radix) { + var chars = CHARS, uuid = [], i; + radix = radix || chars.length; + + if (len) { + // Compact form + for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix]; + } else { + // rfc4122, version 4 form + var r; + + // rfc4122 requires these characters + uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; + uuid[14] = '4'; + + // Fill in random data. At i==19 set the high bits of clock sequence as + // per rfc4122, sec. 4.1.5 + for (i = 0; i < 36; i++) { + if (!uuid[i]) { + r = 0 | Math.random()*16; + uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; + } + } + } + + return uuid.join(''); +}; + + +/** FILE: src/baseclient.js **/ +(function(global) { + + function deprecate(thing, replacement) { + console.log('WARNING: ' + thing + ' is deprecated. Use ' + + replacement + ' instead.'); + } + + var RS = RemoteStorage; + + /** + * Class: RemoteStorage.BaseClient + * + * Provides a high-level interface to access data below a given root path. + * + * A BaseClient deals with three types of data: folders, objects and files. + * + * returns a list of all items within a folder, or undefined + * if a 404 is encountered. Items that end with a forward slash ("/") are + * child folders. + * + * / operate on JSON objects. Each object has a type. + * + * / operates on files. Each file has a MIME type. + * + * operates on either objects or files (but not folders, folders are + * created and removed implictly). + */ + RS.BaseClient = function(storage, base) { + if (base[base.length - 1] !== '/') { + throw "Not a folder: " + base; + } + + if (base === '/') { + // allow absolute and relative paths for the root scope. + this.makePath = function(path) { + return (path[0] === '/' ? '' : '/') + path; + }; + } + + /** + * Property: storage + * + * The instance this operates on. + */ + this.storage = storage; + + /** + * Property: base + * + * Base path this operates on. + * + * For the module's privateClient this would be //, for the + * corresponding publicClient /public//. + */ + this.base = base; + + var parts = this.base.split('/'); + if (parts.length > 2) { + this.moduleName = parts[1]; + } else { + this.moduleName = 'root'; + } + + /** + * Event: change + * + * Emitted when a node changes + * + * Arguments: + * event - Event object containing information about the changed node + * + * (start code) + * { + * path: path, // Path of the changed node + * origin: 'window', 'local', or 'remote' // emitted by user action within the app, local data store, or remote sync + * oldValue: oldBody, // Old body of the changed node (undefined if creation) + * newValue: newBody // New body of the changed node (undefined if deletion) + * } + * (end code) + **/ + + RS.eventHandling(this, 'change'); + this.on = this.on.bind(this); + storage.onChange(this.base, this._fireChange.bind(this)); + }; + + RS.BaseClient.prototype = { + + extend: function(object) { + for (var key in object) { + this[key] = object[key]; + } + return this; + }, + + /** + * Method: scope + * + * Returns a new operating on a subpath of the current path. + */ + scope: function(path) { + return new RS.BaseClient(this.storage, this.makePath(path)); + }, + + // folder operations + + /** + * Method: getListing + * + * Get a list of child nodes below a given path. + * + * The callback semantics of getListing are identical to those of getObject. + * + * Parameters: + * path - The path to query. It MUST end with a forward slash. + * maxAge - (optional) Maximum age of cached listing in + * milliseconds + * + * Returns: + * + * A promise for an object, representing child nodes. + * + * Keys ending in a forward slash represent *folder nodes*, while all + * other keys represent *data nodes*. + * + * For spec versions <= 01, the data node information will contain only + * the item's ETag. For later spec versions, it will also contain the + * content type and -length of the item. + * + * Example: + * (start code) + * client.getListing('').then(function(listing) { + * listing.forEach(function(item) { + * console.log(item); + * }); + * }); + * (end code) + */ + getListing: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getListing must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then( + function(status, body) { + return (status === 404) ? undefined : body; + } + ); + }, + + /** + * Method: getAll + * + * Get all objects directly below a given path. + * + * Parameters: + * path - path to the folder + * maxAge - (optional) Maximum age of cached objects in + * milliseconds + * + * Returns: + * a promise for an object in the form { path : object, ... } + * + * Example: + * (start code) + * client.getAll('').then(function(objects) { + * for (var key in objects) { + * console.log('- ' + key + ': ', objects[key]); + * } + * }); + * (end code) + */ + getAll: function(path, maxAge) { + if (typeof(path) !== 'string') { + path = ''; + } else if (path.length > 0 && path[path.length - 1] !== '/') { + throw "Not a folder: " + path; + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getAll must be undefined or a number'); + } + + return this.storage.get(this.makePath(path), maxAge).then(function(status, body) { + if (status === 404) { return; } + if (typeof(body) === 'object') { + var promise = promising(); + var count = Object.keys(body).length, i = 0; + if (count === 0) { + // treat this like 404. it probably means a folder listing that + // has changes that haven't been pushed out yet. + return; + } + for (var key in body) { + this.storage.get(this.makePath(path + key), maxAge). + then(function(status, b) { + body[this.key] = b; + i++; + if (i === count) { promise.fulfill(body); } + }.bind({ key: key })); + } + return promise; + } + }.bind(this)); + }, + + // file operations + + /** + * Method: getFile + * + * Get the file at the given path. A file is raw data, as opposed to + * a JSON object (use for that). + * + * Except for the return value structure, getFile works exactly like + * getObject. + * + * Parameters: + * path - see getObject + * + * Returns: + * A promise for an object: + * + * mimeType - String representing the MIME Type of the document. + * data - Raw data of the document (either a string or an ArrayBuffer) + * + * Example: + * (start code) + * // Display an image: + * client.getFile('path/to/some/image').then(function(file) { + * var blob = new Blob([file.data], { type: file.mimeType }); + * var targetElement = document.findElementById('my-image-element'); + * targetElement.src = window.URL.createObjectURL(blob); + * }); + * (end code) + */ + getFile: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getFile must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getFile must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + return { + data: body, + mimeType: mimeType, + revision: revision // (this is new) + }; + }); + }, + + /** + * Method: storeFile + * + * Store raw data at a given path. + * + * Parameters: + * mimeType - MIME media type of the data being stored + * path - path relative to the module root. MAY NOT end in a forward slash. + * data - string, ArrayBuffer or ArrayBufferView of raw data to store + * + * The given mimeType will later be returned, when retrieving the data + * using . + * + * Example (UTF-8 data): + * (start code) + * client.storeFile('text/html', 'index.html', '

Hello World!

'); + * (end code) + * + * Example (Binary data): + * (start code) + * // MARKUP: + * + * // CODE: + * var input = document.getElementById('file-input'); + * var file = input.files[0]; + * var fileReader = new FileReader(); + * + * fileReader.onload = function() { + * client.storeFile(file.type, file.name, fileReader.result); + * }; + * + * fileReader.readAsArrayBuffer(file); + * (end code) + * + */ + storeFile: function(mimeType, path, body) { + if (typeof(mimeType) !== 'string') { + return promising().reject('Argument \'mimeType\' of baseClient.storeFile must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeFile must be a string'); + } + if (typeof(body) !== 'string' && typeof(body) !== 'object') { + return promising().reject('Argument \'body\' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView'); + } + + var self = this; + return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + self.makePath(path) + ") failed with status: " + status; + } + }); + }, + + // object operations + + /** + * Method: getObject + * + * Get a JSON object from given path. + * + * Parameters: + * path - relative path from the module root (without leading slash) + * + * Returns: + * A promise for the object. + * + * Example: + * (start code) + * client.getObject('/path/to/object'). + * then(function(object) { + * // object is either an object or null + * }); + * (end code) + */ + getObject: function(path, maxAge) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.getObject must be a string'); + } + if (maxAgeInvalid(maxAge)) { + return promising().reject('Argument \'maxAge\' of baseClient.getObject must be undefined or a number'); + } + return this.storage.get(this.makePath(path), maxAge).then(function(status, body, mimeType, revision) { + if (typeof(body) === 'object') { + return body; + } else if (typeof(body) !== 'undefined' && status === 200) { + throw "Not an object: " + this.makePath(path); + } + }); + }, + + /** + * Method: storeObject + * + * Store object at given path. Triggers synchronization. + * + * Parameters: + * + * type - unique type of this object within this module. See description below. + * path - path relative to the module root. + * object - an object to be saved to the given node. It must be serializable as JSON. + * + * Returns: + * A promise to store the object. The promise fails with a ValidationError, when validations fail. + * + * + * What about the type?: + * + * A great thing about having data on the web, is to be able to link to + * it and rearrange it to fit the current circumstances. To facilitate + * that, eventually you need to know how the data at hand is structured. + * For documents on the web, this is usually done via a MIME type. The + * MIME type of JSON objects however, is always application/json. + * To add that extra layer of "knowing what this object is", remoteStorage + * aims to use . + * A first step in that direction, is to add a *@context attribute* to all + * JSON data put into remoteStorage. + * Now that is what the *type* is for. + * + * Within remoteStorage.js, @context values are built using three components: + * http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness + * the module name - module names should be unique as well + * the type given here - naming this particular kind of object within this module + * + * In retrospect that means, that whenever you introduce a new "type" in calls to + * storeObject, you should make sure that once your code is in the wild, future + * versions of the code are compatible with the same JSON structure. + * + * How to define types?: + * + * See for examples. + */ + storeObject: function(typeAlias, path, object) { + if (typeof(typeAlias) !== 'string') { + return promising().reject('Argument \'typeAlias\' of baseClient.storeObject must be a string'); + } + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.storeObject must be a string'); + } + if (typeof(object) !== 'object') { + return promising().reject('Argument \'object\' of baseClient.storeObject must be an object'); + } + this._attachType(object, typeAlias); + try { + var validationResult = this.validate(object); + if (! validationResult.valid) { + return promising(function(p) { p.reject(validationResult); }); + } + } catch(exc) { + if (! (exc instanceof RS.BaseClient.Types.SchemaNotFound)) { + return promising().reject(exc); + } + } + return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) { + if (status === 200 || status === 201) { + return revision; + } else { + throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status; + } + }.bind(this)); + }, + + // generic operations + + /** + * Method: remove + * + * Remove node at given path from storage. Triggers synchronization. + * + * Parameters: + * path - Path relative to the module root. + */ + remove: function(path) { + if (typeof(path) !== 'string') { + return promising().reject('Argument \'path\' of baseClient.remove must be a string'); + } + return this.storage.delete(this.makePath(path)); + }, + + + cache: function(path, strategy) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.cache must be a string'; + } + if (strategy === false) { + deprecate('caching strategy ', '<"FLUSH">'); + strategy = 'FLUSH'; + } else if (strategy === undefined) { + strategy = 'ALL'; + } else if (typeof(strategy) !== 'string') { + deprecate('that caching strategy', '<"ALL">'); + strategy = 'ALL'; + } + if (strategy !== 'FLUSH' && + strategy !== 'SEEN' && + strategy !== 'ALL') { + throw 'Argument \'strategy\' of baseclient.cache must be one of ' + + '["FLUSH", "SEEN", "ALL"]'; + } + this.storage.caching.set(this.makePath(path), strategy); + return this; + }, + + flush: function(path) { + return this.storage.local.flush(path); + }, + + makePath: function(path) { + return this.base + (path || ''); + }, + + _fireChange: function(event) { + this._emit('change', event); + }, + + _cleanPath: RS.WireClient.cleanPath, + + /** + * Method: getItemURL + * + * Retrieve full URL of item + * + * Parameters: + * path - Path relative to the module root. + */ + getItemURL: function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.getItemURL must be a string'; + } + if (this.storage.connected) { + path = this._cleanPath( this.makePath(path) ); + return this.storage.remote.href + path; + } else { + return undefined; + } + }, + + uuid: function() { + return Math.uuid(); + } + + }; + + /** + * Method: RS#scope + * + * Returns a new scoped to the given path. + * + * Parameters: + * path - Root path of new BaseClient. + * + * + * Example: + * (start code) + * + * var foo = remoteStorage.scope('/foo/'); + * + * // PUTs data "baz" to path /foo/bar + * foo.storeFile('text/plain', 'bar', 'baz'); + * + * var something = foo.scope('something/'); + * + * // GETs listing from path /foo/something/bla/ + * something.getListing('bla/'); + * + * (end code) + * + */ + RS.BaseClient._rs_init = function() { + RS.prototype.scope = function(path) { + if (typeof(path) !== 'string') { + throw 'Argument \'path\' of baseClient.scope must be a string'; + } + + return new RS.BaseClient(this, path); + }; + }; + + /* e.g.: + remoteStorage.defineModule('locations', function(priv, pub) { + return { + exports: { + features: priv.scope('features/').defaultType('feature'), + collections: priv.scope('collections/').defaultType('feature-collection'); + } + }; + }); + */ + + maxAgeInvalid = function(maxAge) { + return typeof(maxAge) !== 'undefined' && typeof(maxAge) !== 'number'; + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/baseclient/types.js **/ +(function(global) { + + RemoteStorage.BaseClient.Types = { + // -> + uris: {}, + // -> + schemas: {}, + // -> + aliases: {}, + + declare: function(moduleName, alias, uri, schema) { + var fullAlias = moduleName + '/' + alias; + + if (schema.extends) { + var extendedAlias; + var parts = schema.extends.split('/'); + if (parts.length === 1) { + extendedAlias = moduleName + '/' + parts.shift(); + } else { + extendedAlias = parts.join('/'); + } + var extendedUri = this.uris[extendedAlias]; + if (! extendedUri) { + throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'"; + } + schema.extends = this.schemas[extendedUri]; + } + + this.uris[fullAlias] = uri; + this.aliases[uri] = fullAlias; + this.schemas[uri] = schema; + }, + + resolveAlias: function(alias) { + return this.uris[alias]; + }, + + getSchema: function(uri) { + return this.schemas[uri]; + }, + + inScope: function(moduleName) { + var ml = moduleName.length; + var schemas = {}; + for (var alias in this.uris) { + if (alias.substr(0, ml + 1) === moduleName + '/') { + var uri = this.uris[alias]; + schemas[uri] = this.schemas[uri]; + } + } + return schemas; + } + }; + + var SchemaNotFound = function(uri) { + var error = new Error("Schema not found: " + uri); + error.name = "SchemaNotFound"; + return error; + }; + + SchemaNotFound.prototype = Error.prototype; + + RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound; + /** + * Class: RemoteStorage.BaseClient + **/ + RemoteStorage.BaseClient.prototype.extend({ + /** + * Method: validate(object) + * + * validates an Object against the associated schema + * the context has to have a @context property + * + * Returns: + * A validate object giving you information about errors + **/ + validate: function(object) { + var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']); + if (schema) { + return tv4.validateResult(object, schema); + } else { + throw new SchemaNotFound(object['@context']); + } + }, + + // client.declareType(alias, schema); + // /* OR */ + // client.declareType(alias, uri, schema); + declareType: function(alias, uri, schema) { + if (! schema) { + schema = uri; + uri = this._defaultTypeURI(alias); + } + RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema); + }, + + _defaultTypeURI: function(alias) { + return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias; + }, + + _attachType: function(object, alias) { + object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias); + } + }); + + Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', { + configurable: true, + get: function() { + return RemoteStorage.BaseClient.Types.inScope(this.moduleName); + } + }); + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/caching.js **/ + /** + * Class: RemoteStorage.Caching + * + * Holds/manages caching configuration. + * + * Caching strategies: + * + * For each subtree, you can set the caching strategy to 'ALL', + * 'SEEN' (default), and 'FLUSH'. + * + * - 'ALL' means that once all outgoing changes have been pushed, sync + * will start retrieving nodes to cache pro-actively. If a local + * copy exists of everything, it will check on each sync whether + * the ETag of the root folder changed, and retrieve remote changes + * if they exist. + * - 'SEEN' does this only for documents and folders that have been either + * read from or written to at least once since connecting to the current + * remote backend, plus their parent/ancestor folders up to the root + * (to make tree-based sync possible). + * - 'FLUSH' will only cache outgoing changes, and forget them as soon as + * they have been saved to remote successfully. + * + **/ + +(function(global) { + var SETTINGS_KEY = "remotestorage:caching"; + + function containingFolder(path) { + if (path === '') { + return '/'; + } + if (! path) { + throw "Path not given!"; + } + + return path.replace(/\/+/g, '/').replace(/[^\/]+\/?$/, ''); + } + + RemoteStorage.Caching = function() { + this.reset(); + }; + + RemoteStorage.Caching.prototype = { + pendingActivations: [], + + /** + * Method: set + * + * Set the caching strategy for a given path explicitly. + * + * Not needed when using /. + * + * Parameters: + * path - Path to cache + * value - Caching strategy. One of 'ALL', 'SEEN', or 'FLUSH'. + * + * Example: + * (start code) + * remoteStorage.caching.set('/bookmarks/archive') + */ + set: function(path, value) { + if (typeof(path) !== 'string') { + throw new Error('path should be a string'); + } + if (typeof(value) === 'undefined') { + throw new Error("value should be 'FLUSH', 'SEEN', or 'ALL'"); + } + + this._rootPaths[path] = value; + + if (value === 'ALL') { + if (this.activateHandler) { + this.activateHandler(path); + } else { + this.pendingActivations.push(path); + } + } + }, + + /** + * Method: enable + * + * Enable caching for a given path. + * + * Uses caching strategy 'ALL'. + * + * Parameters: + * path - Path to enable caching for + */ + enable: function(path) { + this.set(path, 'ALL'); + }, + + /** + * Method: disable + * + * Disable caching for a given path. + * + * Uses caching strategy 'FLUSH' (meaning items are only cached until + * successfully pushed to the remote). + * + * Parameters: + * path - Path to disable caching for + */ + disable: function(path) { + this.set(path, 'FLUSH'); + }, + + /** + * Method: onActivate + * + * Set a callback for when caching is activated for a path. + * + * Parameters: + * callback - Callback function + */ + onActivate: function(cb) { + var i; + RemoteStorage.log('[Caching] Setting activate handler', cb, this.pendingActivations); + this.activateHandler = cb; + for (i=0; i 0; + }, + + collectDiffTasks: function() { + var num = 0; + + return this.local.forAllNodes(function(node) { + if (num > 100) { + return; + } + + if (this.isCorrupt(node)) { + RemoteStorage.log('[Sync] WARNING: corrupt node in local cache', node); + if (typeof(node) === 'object' && node.path) { + this.addTask(node.path); + num++; + } + } else if (this.needsFetch(node) && this.access.checkPathPermission(node.path, 'r')) { + this.addTask(node.path); + num++; + } else if (this.isDocumentNode(node) && this.needsPush(node) && + this.access.checkPathPermission(node.path, 'rw')) { + this.addTask(node.path); + num++; + } + }.bind(this)).then(function() { + return num; + }, function(err) { + throw err; + }); + }, + + inConflict: function(node) { + return (node.local && node.remote && + (node.remote.body !== undefined || node.remote.itemsMap)); + }, + + needsRefresh: function(node) { + if (node.common) { + if (!node.common.timestamp) { + return true; + } + return (this.now() - node.common.timestamp > syncInterval); + } + return false; + }, + + needsFetch: function(node) { + if (this.inConflict(node)) { + return true; + } + if (node.common && node.common.itemsMap === undefined && node.common.body === undefined) { + return true; + } + if (node.remote && node.remote.itemsMap === undefined && node.remote.body === undefined) { + return true; + } + return false; + }, + + needsPush: function(node) { + if (this.inConflict(node)) { + return false; + } + if (node.local && !node.push) { + return true; + } + }, + + needsRemotePut: function(node) { + return node.local && node.local.body; + }, + + needsRemoteDelete: function(node) { + return node.local && node.local.body === false; + }, + + getParentPath: function(path) { + var parts = path.match(/^(.*\/)([^\/]+\/?)$/); + + if (parts) { + return parts[1]; + } else { + throw new Error('Not a valid path: "'+path+'"'); + } + }, + + deleteChildPathsFromTasks: function() { + for (var path in this._tasks) { + paths = this.local._getInternals().pathsFromRoot(path); + + for (var i=1; i= numToAdd) { + return true; + } + } + } + return (numAdded >= numToAdd); + }, + + collectTasks: function(alsoCheckRefresh) { + if (this.hasTasks() || this.stopped) { + promise = promising(); + promise.fulfill(); + return promise; + } + + return this.collectDiffTasks().then(function(numDiffs) { + if (numDiffs || alsoCheckRefresh === false) { + promise = promising(); + promise.fulfill(); + return promise; + } else { + return this.collectRefreshTasks(); + } + }.bind(this), function(err) { + throw err; + }); + }, + + addTask: function(path, cb) { + if (!this._tasks[path]) { + this._tasks[path] = []; + } + if (typeof(cb) === 'function') { + this._tasks[path].push(cb); + } + }, + + /** + * Method: sync + **/ + sync: function() { + var promise = promising(); + this.done = false; + + if (!this.doTasks()) { + return this.collectTasks().then(function() { + try { + this.doTasks(); + } catch(e) { + RemoteStorage.log('[Sync] doTasks error', e); + } + }.bind(this), function(e) { + RemoteStorage.log('[Sync] Sync error', e); + throw new Error('Local cache unavailable'); + }); + } else { + return promising().fulfill(); + } + }, + }; + + /** + * Method: getSyncInterval + * + * Get the value of the sync interval when application is in the foreground + * + * Returns a number of milliseconds + * + */ + RemoteStorage.prototype.getSyncInterval = function() { + return syncInterval; + }; + + /** + * Method: setSyncInterval + * + * Set the value of the sync interval when application is in the foreground + * + * Parameters: + * interval - sync interval in milliseconds + * + */ + RemoteStorage.prototype.setSyncInterval = function(interval) { + if (typeof(interval) !== 'number') { + throw interval + " is not a valid sync interval"; + } + syncInterval = parseInt(interval, 10); + if (this._syncTimer) { + this.stopSync(); + this._syncTimer = setTimeout(this.syncCycle.bind(this), interval); + } + }; + + var SyncError = function(originalError) { + var msg = 'Sync failed: '; + if (typeof(originalError) === 'object' && 'message' in originalError) { + msg += originalError.message; + } else { + msg += originalError; + } + this.originalError = originalError; + Error.apply(this, [msg]); + }; + + SyncError.prototype = Object.create(Error.prototype); + + RemoteStorage.SyncError = SyncError; + + RemoteStorage.prototype.syncCycle = function() { + if (this.sync.stopped) { + return; + } + + this.sync.on('done', function() { + RemoteStorage.log('[Sync] Sync done. Setting timer to', this.getSyncInterval()); + if (!this.sync.stopped) { + if (this._syncTimer) { + clearTimeout(this._syncTimer); + } + this._syncTimer = setTimeout(this.sync.sync.bind(this.sync), this.getSyncInterval()); + } + }.bind(this)); + + this.sync.sync(); + }; + + RemoteStorage.prototype.stopSync = function() { + if (this.sync) { + RemoteStorage.log('[Sync] Stopping sync'); + this.sync.stopped = true; + } else { + // TODO When is this ever the case and what is syncStopped for then? + RemoteStorage.log('[Sync] Will instantiate sync stopped'); + this.syncStopped = true; + } + }; + + RemoteStorage.prototype.startSync = function() { + this.sync.stopped = false; + this.syncStopped = false; + this.sync.sync(); + }; + + var syncCycleCb; + + RemoteStorage.Sync._rs_init = function(remoteStorage) { + syncCycleCb = function() { + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + + if (!remoteStorage.sync) { + // Call this now that all other modules are also ready: + remoteStorage.sync = new RemoteStorage.Sync( + remoteStorage.local, remoteStorage.remote, remoteStorage.access, + remoteStorage.caching); + + if (remoteStorage.syncStopped) { + RemoteStorage.log('[Sync] Instantiating sync stopped'); + remoteStorage.sync.stopped = true; + delete remoteStorage.syncStopped; + } + } + + RemoteStorage.log('[Sync] syncCycleCb calling syncCycle'); + remoteStorage.syncCycle(); + }; + + remoteStorage.on('ready', syncCycleCb); + }; + + RemoteStorage.Sync._rs_cleanup = function(remoteStorage) { + remoteStorage.stopSync(); + remoteStorage.removeEventListener('ready', syncCycleCb); + }; + +})(typeof(window) !== 'undefined' ? window : global); + + +/** FILE: src/cachinglayer.js **/ +(function() { + /** + * Interface: cachinglayer + * + * This module defines functions that are mixed into remoteStorage.local when + * it is instantiated (currently one of indexeddb.js, localstorage.js, or + * inmemorystorage.js). + * + * All remoteStorage.local implementations should therefore implement + * this.getNodes, this.setNodes, and this.forAllNodes. The rest is blended in + * here to create a GPD (get/put/delete) interface which the BaseClient can + * talk to. + */ + + function isFolder(path) { + return path.substr(-1) === '/'; + } + + function isDocument(path) { + return path.substr(-1) !== '/'; + } + + function deepClone(obj) { + if (obj === undefined) { + return undefined; + } else { + return JSON.parse(JSON.stringify(obj)); + } + } + + function equal(obj1, obj2) { + return JSON.stringify(obj1) === JSON.stringify(obj2); + } + + function getLatest(node) { + if (typeof(node) !== 'object' || typeof(node.path) !== 'string') { + return; + } + if (isFolder(node.path)) { + if (node.local && node.local.itemsMap) { + return node.local; + } + if (node.common && node.common.itemsMap) { + return node.common; + } + } else { + if (node.local && node.local.body && node.local.contentType) { + return node.local; + } + if (node.common && node.common.body && node.common.contentType) { + return node.common; + } + // Migration code! Once all apps use at least this version of the lib, we + // can publish clean-up code that migrates over any old-format data, and + // stop supporting it. For now, new apps will support data in both + // formats, thanks to this: + if (node.body && node.contentType) { + return { + body: node.body, + contentType: node.contentType + }; + } + } + } + + function isOutdated(node, maxAge) { + return !node || !node.timestamp || + ((new Date().getTime()) - node.timestamp > maxAge); + } + + function pathsFromRoot(path) { + var paths = [path]; + var parts = path.replace(/\/$/, '').split('/'); + + while (parts.length > 1) { + parts.pop(); + paths.push(parts.join('/')+'/'); + } + return paths; + } + + function makeNode(path, timestamp) { + var node = { path: path, common: { timestamp: timestamp } }; + + if (isFolder(path)) { + node.common.itemsMap = {}; + } + return node; + } + + function updateFolderNodeWithItemName(node, itemName, timestamp) { + if (!node.common) { + node.common = { + timestamp: timestamp, + itemsMap: {} + }; + } + if (!node.common.itemsMap) { + node.common.itemsMap = {}; + } + if (!node.local) { + node.local = deepClone(node.common); + } + if (!node.local.itemsMap) { + node.local.itemsMap = node.common.itemsMap; + } + node.local.itemsMap[itemName] = true; + + return node; + } + + var methods = { + + get: function(path, maxAge) { + var promise = promising(); + + this.getNodes([path]).then(function(objs) { + var node = getLatest(objs[path]); + if ((typeof(maxAge) === 'number') && isOutdated(node, maxAge)) { + remoteStorage.sync.queueGetRequest(path, promise); + } + + if (node) { + promise.fulfill(200, node.body || node.itemsMap, node.contentType); + } else { + promise.fulfill(404); + } + }.bind(this), function(err) { + promise.reject(err); + }.bind(this)); + + return promise; + }, + + put: function(path, body, contentType) { + var paths = pathsFromRoot(path); + var now = new Date().getTime(); + + return this._updateNodes(paths, function(nodes) { + try { + for (var i=0; i 0) { + // This folder still contains other items, don't remove any further ancestors + break; + } + } + } + return nodes; + }); + }, + + flush: function(path) { + return this._getAllDescendentPaths(path).then(function(paths) { + return this.getNodes(paths); + }.bind(this)).then(function(nodes) { + for (var path in nodes) { + var node = nodes[path]; + + if (node && node.common && node.local) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: (node.local.body === false ? undefined : node.local.body), + newValue: (node.common.body === false ? undefined : node.common.body) + }); + } + nodes[path] = undefined; + } + return this.setNodes(nodes); + }.bind(this)); + }, + + fireInitial: function() { + this.forAllNodes(function(node) { + var latest; + if (isDocument(node.path)) { + latest = getLatest(node); + if (latest) { + this._emit('change', { + path: node.path, + origin: 'local', + oldValue: undefined, + oldContentType: undefined, + newValue: latest.body, + newContentType: latest.contentType + }); + } + } + }.bind(this)); + }, + + onDiff: function(diffHandler) { + this.diffHandler = diffHandler; + }, + + migrate: function(node) { + if (typeof(node) === 'object' && !node.common) { + node.common = {}; + if (typeof(node.path) === 'string') { + if (node.path.substr(-1) === '/' && typeof(node.body) === 'object') { + node.common.itemsMap = node.body; + } + } else { + //save legacy content of document node as local version + if (!node.local) { + node.local = {}; + } + node.local.body = node.body; + node.local.contentType = node.contentType; + } + } + return node; + }, + + _updateNodes: function(paths, cb) { + var self = this; + + return this.getNodes(paths).then(function(nodes) { + var existingNodes = deepClone(nodes); + var changeEvents = []; + var node; + + nodes = cb(nodes); + + for (var path in nodes) { + node = nodes[path]; + if (equal(node, existingNodes[path])) { + delete nodes[path]; + } + else if(isDocument(path)) { + changeEvents.push({ + path: path, + origin: 'window', + oldValue: node.local.previousBody, + newValue: node.local.body === false ? undefined : node.local.body, + oldContentType: node.local.previousContentType, + newContentType: node.local.contentType + }); + delete node.local.previousBody; + delete node.local.previousContentType; + } + } + + return self.setNodes(nodes).then(function() { + self._emitChangeEvents(changeEvents); + return 200; + }); + }, + function(err) { + throw(err); + }); + }, + + _emitChangeEvents: function(events) { + for (var i=0; i node) map. + * + */ + + var RS = RemoteStorage; + + var DB_VERSION = 2; + + var DEFAULT_DB_NAME = 'remotestorage'; + var DEFAULT_DB; + + RS.IndexedDB = function(database) { + this.db = database || DEFAULT_DB; + + if (!this.db) { + RemoteStorage.log("[IndexedDB] Failed to open DB"); + return undefined; + } + + RS.cachingLayer(this); + RS.eventHandling(this, 'change'); + + this.getsRunning = 0; + this.putsRunning = 0; + }; + + RS.IndexedDB.prototype = { + getNodes: function(paths) { + var promise = promising(); + var transaction = this.db.transaction(['nodes'], 'readonly'); + var nodes = transaction.objectStore('nodes'); + var retrievedNodes = {}; + var startTime = new Date().getTime(); + + this.getsRunning++; + + for (var i=0; i 64 && nChr < 91 ? + nChr - 65 + : nChr > 96 && nChr < 123 ? + nChr - 71 + : nChr > 47 && nChr < 58 ? + nChr + 4 + : nChr === 43 ? + 62 + : nChr === 47 ? + 63 + : + 0; + } + + function base64DecToArr(sBase64, nBlocksSize) { + var + sB64Enc = sBase64.replace(/[^A-Za-z0-9\+\/]/g, ""), nInLen = sB64Enc.length, + nOutLen = nBlocksSize ? Math.ceil((nInLen * 3 + 1 >> 2) / nBlocksSize) * nBlocksSize : nInLen * 3 + 1 >> 2, taBytes = new Uint8Array(nOutLen); + + for (var nMod3, nMod4, nUint24 = 0, nOutIdx = 0, nInIdx = 0; nInIdx < nInLen; nInIdx++) { + nMod4 = nInIdx & 3; + nUint24 |= b64ToUint6(sB64Enc.charCodeAt(nInIdx)) << 18 - 6 * nMod4; + if (nMod4 === 3 || nInLen - nInIdx === 1) { + for (nMod3 = 0; nMod3 < 3 && nOutIdx < nOutLen; nMod3++, nOutIdx++) { + taBytes[nOutIdx] = nUint24 >>> (16 >>> nMod3 & 24) & 255; + } + nUint24 = 0; + } + } + return taBytes; + } + + function isBinary(node) { + return node.match(/charset=binary/); + } + + function isRemoteStorageKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX || + key.substr(0, CHANGES_PREFIX.length) === CHANGES_PREFIX; + } + + function isNodeKey(key) { + return key.substr(0, NODES_PREFIX.length) === NODES_PREFIX; + } + + RemoteStorage.LocalStorage.prototype = { + + getNodes: function(paths) { + var promise = promising(); + var nodes = {}; + + for(i=0; i + * instance under the module's name. So when defining a locations module, + * like in the example below, it would be accessible via + * `remoteStorage.locations`, which would in turn have a `features` and a + * `collections` property. + * + * The function receives a private and a public client, which are both + * instances of . In the following example, the + * scope of privateClient is `/locations` and the scope of publicClient is + * `/public/locations`. + * + * Example: + * (start code) + * RemoteStorage.defineModule('locations', function(privateClient, publicClient) { + * return { + * exports: { + * features: privateClient.scope('features/').defaultType('feature'), + * collections: privateClient.scope('collections/').defaultType('feature-collection') + * } + * }; + * }); + * (end code) + */ + + RemoteStorage.defineModule = function(moduleName, builder) { + RemoteStorage.MODULES[moduleName] = builder; + + Object.defineProperty(RemoteStorage.prototype, moduleName, { + configurable: true, + get: function() { + var instance = this._loadModule(moduleName); + Object.defineProperty(this, moduleName, { + value: instance + }); + return instance; + } + }); + + if (moduleName.indexOf('-') !== -1) { + var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) { + return s[1].toUpperCase(); + }); + Object.defineProperty(RemoteStorage.prototype, camelizedName, { + get: function() { + return this[moduleName]; + } + }); + } + }; + + RemoteStorage.prototype._loadModule = function(moduleName) { + var builder = RemoteStorage.MODULES[moduleName]; + if (builder) { + var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'), + new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/')); + return module.exports; + } else { + throw "Unknown module: " + moduleName; + } + }; + + RemoteStorage.prototype.defineModule = function(moduleName) { + console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!"); + RemoteStorage.defineModule.apply(RemoteStorage, arguments); + }; + +})(); + + +/** FILE: src/debug/inspect.js **/ +(function() { + function loadTable(table, storage, paths) { + table.setAttribute('border', '1'); + table.style.margin = '8px'; + table.innerHTML = ''; + var thead = document.createElement('thead'); + table.appendChild(thead); + var titleRow = document.createElement('tr'); + thead.appendChild(titleRow); + ['Path', 'Content-Type', 'Revision'].forEach(function(label) { + var th = document.createElement('th'); + th.textContent = label; + thead.appendChild(th); + }); + + var tbody = document.createElement('tbody'); + table.appendChild(tbody); + + function renderRow(tr, path, contentType, revision) { + [path, contentType, revision].forEach(function(value) { + var td = document.createElement('td'); + td.textContent = value || ''; + tr.appendChild(td); + }); + } + + function loadRow(path) { + if (storage.connected === false) { return; } + function processRow(status, body, contentType, revision) { + if (status === 200) { + var tr = document.createElement('tr'); + tbody.appendChild(tr); + renderRow(tr, path, contentType, revision); + if (path[path.length - 1] === '/') { + for (var key in body) { + loadRow(path + key); + } + } + } + } + storage.get(path).then(processRow); + } + + paths.forEach(loadRow); + } + + function renderWrapper(title, table, storage, paths) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = title; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + updateButton.onclick = function() { loadTable(table, storage, paths); }; + wrapper.appendChild(updateButton); + if (storage.reset) { + var resetButton = document.createElement('button'); + resetButton.textContent = "Reset"; + resetButton.onclick = function() { + storage.reset(function(newStorage) { + storage = newStorage; + loadTable(table, storage, paths); + }); + }; + wrapper.appendChild(resetButton); + } + wrapper.appendChild(table); + loadTable(table, storage, paths); + return wrapper; + } + + function renderLocalChanges(local) { + var wrapper = document.createElement('div'); + //wrapper.style.display = 'inline-block'; + var heading = document.createElement('h2'); + heading.textContent = "Outgoing changes"; + wrapper.appendChild(heading); + var updateButton = document.createElement('button'); + updateButton.textContent = "Refresh"; + wrapper.appendChild(updateButton); + var list = document.createElement('ul'); + list.style.fontFamily = 'courier'; + wrapper.appendChild(list); + + function updateList() { + local.changesBelow('/').then(function(changes) { + list.innerHTML = ''; + changes.forEach(function(change) { + var el = document.createElement('li'); + el.textContent = JSON.stringify(change); + list.appendChild(el); + }); + }); + } + + updateButton.onclick = updateList; + updateList(); + return wrapper; + } + + RemoteStorage.prototype.inspect = function() { + + var widget = document.createElement('div'); + widget.id = 'remotestorage-inspect'; + widget.style.position = 'absolute'; + widget.style.top = 0; + widget.style.left = 0; + widget.style.background = 'black'; + widget.style.color = 'white'; + widget.style.border = 'groove 5px #ccc'; + + var controls = document.createElement('div'); + controls.style.position = 'absolute'; + controls.style.top = 0; + controls.style.left = 0; + + var heading = document.createElement('strong'); + heading.textContent = " remotestorage.js inspector "; + + controls.appendChild(heading); + + var syncButton; + + if (this.local) { + syncButton = document.createElement('button'); + syncButton.textContent = "Synchronize"; + controls.appendChild(syncButton); + } + + var closeButton = document.createElement('button'); + closeButton.textContent = "Close"; + closeButton.onclick = function() { + document.body.removeChild(widget); + }; + controls.appendChild(closeButton); + + widget.appendChild(controls); + + var remoteTable = document.createElement('table'); + var localTable = document.createElement('table'); + widget.appendChild(renderWrapper("Remote", remoteTable, this.remote, this.caching.rootPaths)); + if (this.local) { + widget.appendChild(renderWrapper("Local", localTable, this.local, ['/'])); + widget.appendChild(renderLocalChanges(this.local)); + + syncButton.onclick = function() { + this.log('sync clicked'); + this.sync().then(function() { + this.log('SYNC FINISHED'); + loadTable(localTable, this.local, ['/']); + }.bind(this), function(err) { + console.error("SYNC FAILED", err, err.stack); + }); + }.bind(this); + } + + document.body.appendChild(widget); + }; +})(); + + +/** FILE: src/legacy.js **/ +(function() { + var util = { + getEventEmitter: function() { + var object = {}; + var args = Array.prototype.slice.call(arguments); + args.unshift(object); + RemoteStorage.eventHandling.apply(RemoteStorage, args); + object.emit = object._emit; + return object; + }, + + extend: function(target) { + var sources = Array.prototype.slice.call(arguments, 1); + sources.forEach(function(source) { + for (var key in source) { + target[key] = source[key]; + } + }); + return target; + }, + + asyncEach: function(array, callback) { + return this.asyncMap(array, callback). + then(function() { return array; }); + }, + + asyncMap: function(array, callback) { + var promise = promising(); + var n = array.length, i = 0; + var results = [], errors = []; + function oneDone() { + i++; + if (i === n) { + promise.fulfill(results, errors); + } + } + + array.forEach(function(item, index) { + var result; + try { + result = callback(item); + } catch(exc) { + oneDone(); + errors[index] = exc; + } + if (typeof(result) === 'object' && typeof(result.then) === 'function') { + result.then(function(res) { results[index] = res; oneDone(); }, + function(error) { errors[index] = res; oneDone(); }); + } else { + oneDone(); + results[index] = result; + } + }); + + return promise; + }, + + containingFolder: function(path) { + var folder = path.replace(/[^\/]+\/?$/, ''); + return folder === path ? null : folder; + }, + + isFolder: function(path) { + return path.substr(-1) === '/'; + }, + + baseName: function(path) { + var parts = path.split('/'); + if (util.isFolder(path)) { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + }, + + bindAll: function(object) { + for (var key in this) { + if (typeof(object[key]) === 'function') { + object[key] = object[key].bind(object); + } + } + } + }; + + Object.defineProperty(RemoteStorage.prototype, 'util', { + get: function() { + console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release."); + return util; + } + }); + +})(); + + +/** FILE: src/googledrive.js **/ +(function(global) { + /** + * Class: RemoteStorage.GoogleDrive + * + * WORK IN PROGRESS, NOT RECOMMENDED FOR PRODUCTION USE + **/ + + var RS = RemoteStorage; + + var BASE_URL = 'https://www.googleapis.com'; + var AUTH_URL = 'https://accounts.google.com/o/oauth2/auth'; + var AUTH_SCOPE = 'https://www.googleapis.com/auth/drive'; + + var GD_DIR_MIME_TYPE = 'application/vnd.google-apps.folder'; + var RS_DIR_MIME_TYPE = 'application/json; charset=UTF-8'; + + function buildQueryString(params) { + return Object.keys(params).map(function(key) { + return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]); + }).join('&'); + } + + function fileNameFromMeta(meta) { + return encodeURIComponent(meta.title) + (meta.mimeType === GD_DIR_MIME_TYPE ? '/' : ''); + } + + function metaTitleFromFileName(filename) { + if (filename.substr(-1) === '/') { + filename = filename.substr(0, filename.length - 1); + } + return decodeURIComponent(filename); + } + + function parentPath(path) { + return path.replace(/[^\/]+\/?$/, ''); + } + + function baseName(path) { + var parts = path.split('/'); + if (path.substr(-1) === '/') { + return parts[parts.length-2]+'/'; + } else { + return parts[parts.length-1]; + } + } + + var Cache = function(maxAge) { + this.maxAge = maxAge; + this._items = {}; + }; + + Cache.prototype = { + get: function(key) { + var item = this._items[key]; + var now = new Date().getTime(); + return (item && item.t >= (now - this.maxAge)) ? item.v : undefined; + }, + + set: function(key, value) { + this._items[key] = { + v: value, + t: new Date().getTime() + }; + } + }; + + RS.GoogleDrive = function(remoteStorage, clientId) { + + RS.eventHandling(this, 'connected'); + + this.rs = remoteStorage; + this.clientId = clientId; + + this._fileIdCache = new Cache(60 * 5); // ids expire after 5 minutes (is this a good idea?) + + setTimeout(function() { + this.configure(undefined, undefined, undefined, localStorage['remotestorage:googledrive:token']); + }.bind(this), 0); + }; + + RS.GoogleDrive.prototype = { + + configure: function(_x, _y, _z, token) { // parameter list compatible with WireClient + if (token) { + localStorage['remotestorage:googledrive:token'] = token; + this.token = token; + this.connected = true; + this._emit('connected'); + } else { + this.connected = false; + delete this.token; + // not reseting backend whenever googledrive gets initialized without an token +// this.rs.setBackend(undefined); + delete localStorage['remotestorage:googledrive:token']; + } + }, + + connect: function() { + this.rs.setBackend('googledrive'); + RS.Authorize(AUTH_URL, AUTH_SCOPE, String(RS.Authorize.getLocation()), this.clientId); + }, + + get: function(path, options) { + if (path.substr(-1) === '/') { + return this._getFolder(path, options); + } else { + return this._getFile(path, options); + } + }, + + put: function(path, body, contentType, options) { + var promise = promising(); + function putDone(error, response) { + if (error) { + promise.reject(error); + } else if (response.status >= 200 && response.status < 300) { + var meta = JSON.parse(response.responseText); + promise.fulfill(200, undefined, meta.mimeType, meta.etag); + } else { + promise.reject("PUT failed with status " + response.status + " (" + response.responseText + ")"); + } + } + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + return; + } else if (id) { + this._updateFile(id, path, body, contentType, options, putDone); + } else { + this._createFile(path, body, contentType, options, putDone); + } + }); + return promise; + }, + + 'delete': function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else if (id) { + this._request('DELETE', BASE_URL + '/drive/v2/files/' + id, {}, function(deleteError, response) { + if (deleteError) { + promise.reject(deleteError); + } else if (response.status === 200 || response.status === 204) { + promise.fulfill(200); + } else { + promise.reject("Delete failed: " + response.status + " (" + response.responseText + ")"); + } + }); + } else { + // file doesn't exist. ignore. + promise.fulfill(200); + } + }); + return promise; + }, + + _updateFile: function(id, path, body, contentType, options, callback) { + callback = callback.bind(this); + var metadata = { + mimeType: contentType + }; + this._request('PUT', BASE_URL + '/upload/drive/v2/files/' + id + '?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('PUT', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }, + + _createFile: function(path, body, contentType, options, callback) { + callback = callback.bind(this); + this._getParentId(path, function(parentIdError, parentId) { + if (parentIdError) { + callback(parentIdError); + return; + } + var fileName = baseName(path); + var metadata = { + title: metaTitleFromFileName(fileName), + mimeType: contentType, + parents: [{ + kind: "drive#fileLink", + id: parentId + }] + }; + this._request('POST', BASE_URL + '/upload/drive/v2/files?uploadType=resumable', { + body: JSON.stringify(metadata), + headers: { + 'Content-Type': 'application/json; charset=UTF-8' + } + }, function(metadataError, response) { + if (metadataError) { + callback(metadataError); + } else { + this._request('POST', response.getResponseHeader('Location'), { + body: contentType.match(/^application\/json/) ? JSON.stringify(body) : body + }, callback); + } + }); + }); + }, + + _getFile: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + if (idError) { + promise.reject(idError); + } else { + this._getMeta(id, function(metaError, meta) { + var etagWithoutQuotes; + if (typeof(meta) === 'object' && typeof(meta.etag) === 'string') { + etagWithoutQuotes = meta.etag.substring(1, meta.etag.length-1); + } + if (metaError) { + promise.reject(metaError); + } else if (meta.downloadUrl) { + var options = {}; + if (meta.mimeType.match(/charset=binary/)) { + options.responseType = 'blob'; + } + this._request('GET', meta.downloadUrl, options, function(downloadError, response) { + if (downloadError) { + promise.reject(downloadError); + } else { + var body = response.response; + if (meta.mimeType.match(/^application\/json/)) { + try { + body = JSON.parse(body); + } catch(e) {} + } + promise.fulfill(200, body, meta.mimeType, etagWithoutQuotes); + } + }); + } else { + // empty file + promise.fulfill(200, '', meta.mimeType, etagWithoutQuotes); + } + }); + } + }); + return promise; + }, + + _getFolder: function(path, options) { + var promise = promising(); + this._getFileId(path, function(idError, id) { + var query, fields, data, i, etagWithoutQuotes, itemsMap; + if (idError) { + promise.reject(idError); + } else if (! id) { + promise.fulfill(404); + } else { + query = '\'' + id + '\' in parents'; + fields = 'items(downloadUrl,etag,fileSize,id,mimeType,title)'; + this._request('GET', BASE_URL + '/drive/v2/files?' + + 'q=' + encodeURIComponent(query) + + '&fields=' + encodeURIComponent(fields) + + '&maxResults=1000', + {}, function(childrenError, response) { + if (childrenError) { + promise.reject(childrenError); + } else { + if (response.status === 200) { + try { + data = JSON.parse(response.responseText); + } catch(e) { + promise.reject('non-JSON response from GoogleDrive'); + return; + } + itemsMap = {}; + for(i=0; i= 0 || true) { + try { + body = JSON.parse(body); + mime = 'application/json; charset=UTF-8'; + } catch(e) { + RS.log("Failed parsing Json, assume it is something else then", mime, path); + } + } + promise.fulfill(status, body, mime, rev); + } + + } else { + promise.fulfill(status); + } + } + }); + return promise; + }, + /** + * Method : put(path, body, contentType, options) + * put compatible with wireclient + * also uses _revCache to check for version conflicts + * also shares via share(path) + **/ + put: function(path, body, contentType, options){ + RemoteStorage.log('dropbox.put', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = this._sharePromise(path); + + var revCache = this._revCache; + + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options && options.ifMatch && savedRev && (savedRev !== options.ifMatch) ) { + promise.fulfill(412); + return promise; + } + if (! contentType.match(/charset=/)) { + contentType += '; charset=' + ((body instanceof ArrayBuffer || RS.WireClient.isArrayBufferView(body)) ? 'binary' : 'utf-8'); + } + var url = 'https://api-content.dropbox.com/1/files_put/auto' + path + '?'; + if (options && options.ifMatch) { + url += "parent_rev="+encodeURIComponent(options.ifMatch); + } + if (body.length>150*1024*1024){ //FIXME actual content-length + //https://www.dropbox.com/developers/core/docs#chunked-upload + RemoteStorage.log('files larger than 150MB not supported yet'); + } else { + this._request('PUT', url, {body:body, headers:{'Content-Type':contentType}}, function(err, resp) { + if (err) { + promise.reject(err); + } else { + var response = JSON.parse(resp.responseText); + // if dropbox reports an file conflict they just change the name of the file + // TODO find out which stays the origianl and how to deal with this + if (response.path !== path){ + promise.fulfill(412); + this.rs.log('Dropbox created conflicting File ', response.path); + } + else { + revCache.set(path, response.rev); + promise.fulfill(resp.status); + } + } + }); + } + return promise; + }, + + /** + * Method : delete(path, options) + * similar to get and set + **/ + 'delete': function(path, options){ + RemoteStorage.log('dropbox.delete ', arguments); + if (! this.connected) { throw new Error("not connected (path: " + path + ")"); } + path = cleanPath(path); + + var promise = promising(); + var revCache = this._revCache; + //check if file has changed and return 412 + var savedRev = revCache.get(path); + if (options.ifMatch && savedRev && (options.ifMatch !== savedRev)) { + promise.fulfill(412); + return promise; + } + + var url = 'https://api.dropbox.com/1/fileops/delete?root=auto&path='+encodeURIComponent(path); + this._request('POST', url, {}, function(err, resp){ + if (err) { + promise.reject(error); + } else { + promise.fulfill(resp.status); + revCache.delete(path); + } + }); + + return promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + delete this._itemRefs[path]; + var p = promising(); + return p.fulfill.apply(p, args); + }.bind(this)); + }, + + /** + * Method : _sharePromise(path) + * returns a promise which's then block doesn't touch the arguments given + * and calls share for the path + * + * also checks for necessity of shareing this url(already in the itemRefs or not '/public/') + **/ + _sharePromise: function(path){ + var promise = promising(); + var self = this; + if (path.match(/^\/public\/.*[^\/]$/) && typeof this._itemRefs[path] === 'undefined') { + RemoteStorage.log('shareing this one ', path); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + var p = promising(); + RemoteStorage.log('calling share now'); + self.share(path).then(function() { + RemoteStorage.log('shareing fullfilled promise',arguments); + p.fulfill.apply(p,args); + }, function(err) { + RemoteStorage.log("shareing failed" , err); + p.fulfill.apply(p,args); + }); + return p; + }); + } + return promise; + }, + + /** + * Method : share(path) + * get sher_url s from dropbox and pushes those into this._hrefCache + * returns promise + */ + share: function(path){ + var url = "https://api.dropbox.com/1/media/auto"+path; + var promise = promising(); + var itemRefs = this._itemRefs; + + // requesting shareing url + this._request('POST', url, {}, function(err, resp){ + if (err) { + RemoteStorage.log(err); + err.message = 'Shareing Dropbox Thingie("'+path+'") failed' + err.message; + promise.reject(err); + } else { + try{ + var response = JSON.parse(resp.responseText); + var url = response.url; + itemRefs[path] = url; + RemoteStorage.log("SHAREING URL :::: ",url,' for ',path); + if (hasLocalStorage) { + localStorage[SETTINGS_KEY+":shares"] = JSON.stringify(this._itemRefs); + } + promise.fulfill(url); + } catch(err) { + err.message += "share error"; + promise.reject(err); + } + } + }); + return promise; + }, + + /** + * Method : info() + * fetching user info from Dropbox returns promise + **/ + info: function() { + var url = 'https://api.dropbox.com/1/account/info'; + var promise = promising(); + // requesting user info(mainly for userAdress) + this._request('GET', url, {}, function(err, resp){ + if (err) { + promise.reject(err); + } else { + try { + var info = JSON.parse(resp.responseText); + promise.fulfill(info); + } catch(e) { + promise.reject(err); + } + } + }); + return promise; + }, + + _request: function(method, url, options, callback) { + callback = callback.bind(this); + if (! options.headers) { options.headers = {}; } + options.headers['Authorization'] = 'Bearer ' + this.token; + RS.WireClient.request.call(this, method, url, options, function(err, xhr) { + //503 means retry this later + if (xhr && xhr.status === 503) { + global.setTimeout(this._request(method, url, options, callback), 3210); + } else { + callback(err, xhr); + } + }); + }, + + /** + * method: fetchDelta + * + * this method fetches the deltas from the dropbox api, used to sync the storage + * here we retrive changes and put them into the _revCache, those values will then be used + * to determin if something has changed. + **/ + fetchDelta: function() { + var args = Array.prototype.slice.call(arguments); + var promise = promising(); + var self = this; + this._request('POST', 'https://api.dropbox.com/1/delta', { + body: this._deltaCursor ? ('cursor=' + encodeURIComponent(this._deltaCursor)) : '', + headers: { + 'Content-Type': 'application/x-www-form-urlencoded' + } + }, function(error, response) { + if (error) { + this.rs.log('fetchDeltas',error); + this.rs._emit('error', new RemoteStorage.SyncError('fetchDeltas failed'+error)); + promise.reject(error); + } else { + // break if status != 200 + if (response.status !== 200 ) { + if (response.status === 400) { + this.rs._emit('error', new RemoteStorage.Unauthorized()); + promise.fulfill.apply(promise, args); + } else { + RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText); + promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText); + } + return promise; + } + + var delta; + try { + delta = JSON.parse(response.responseText); + } catch(error) { + RS.log('fetchDeltas can not parse response',error); + return promise.reject("can not parse response of fetchDelta : "+error.message); + } + // break if no entries found + if (!delta.entries) { + RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!", delta); + return promise.reject('dropbox.fetchDeltas failed, no entries found'); + } + + // Dropbox sends the complete state + if (delta.reset) { + this._revCache = new LowerCaseCache('rev'); + promise.then(function(){ + var args = Array.prototype.slice.call(arguments); + self._revCache._activatePropagation(); + var p = promising(); + return p.fulfill.apply(p,args); + }); + } + + //saving the cursor for requesting further deltas in relation to the cursor position + if (delta.cursor) { + this._deltaCursor = delta.cursor; + } + + //updating revCache + RemoteStorage.log("Delta : ",delta.entries); + delta.entries.forEach(function(entry) { + var path = entry[0]; + var rev; + if (!entry[1]){ + rev = null; + } else { + if (entry[1].is_dir) { + return; + } + rev = entry[1].rev; + } + self._revCache.set(path, rev); + }); + promise.fulfill.apply(promise, args); + } + }); + return promise; + } + }; + + //hooking and unhooking the sync + + function hookSync(rs) { + if (rs._dropboxOrigSync) { return; } // already hooked + rs._dropboxOrigSync = rs.sync.bind(rs); + rs.sync = function() { + return this.dropbox.fetchDelta.apply(this.dropbox, arguments). + then(rs._dropboxOrigSync, function(err){ + rs._emit('error', new rs.SyncError(err)); + }); + }; + } + + function unHookSync(rs) { + if (! rs._dropboxOrigSync) { return; } // not hooked + rs.sync = rs._dropboxOrigSync; + delete rs._dropboxOrigSync; + } + + // hooking and unhooking getItemURL + + function hookGetItemURL(rs) { + if (rs._origBaseClientGetItemURL) { return; } + rs._origBaseClientGetItemURL = RS.BaseClient.prototype.getItemURL; + RS.BaseClient.prototype.getItemURL = function(path){ + var ret = rs.dropbox._itemRefs[path]; + return ret ? ret : ''; + }; + } + + function unHookGetItemURL(rs){ + if (! rs._origBaseClieNtGetItemURL) { return; } + RS.BaseClient.prototype.getItemURL = rs._origBaseClietGetItemURL; + delete rs._origBaseClietGetItemURL; + } + + function hookRemote(rs){ + if (rs._origRemote) { return; } + rs._origRemote = rs.remote; + rs.remote = rs.dropbox; + } + + function unHookRemote(rs){ + if (rs._origRemote) { + rs.remote = rs._origRemote; + delete rs._origRemote; + } + } + + function hookIt(rs){ + hookRemote(rs); + if (rs.sync) { + hookSync(rs); + } + hookGetItemURL(rs); + } + + function unHookIt(rs){ + unHookRemote(rs); + unHookSync(rs); + unHookGetItemURL(rs); + } + + RS.Dropbox._rs_init = function(rs) { + hasLocalStorage = rs.localStorageAvailable(); + if ( rs.apiKeys.dropbox ) { + rs.dropbox = new RS.Dropbox(rs); + } + if (rs.backend === 'dropbox') { + hookIt(rs); + } + }; + + RS.Dropbox._rs_supported = function() { + return true; + }; + + RS.Dropbox._rs_cleanup = function(rs) { + unHookIt(rs); + if (hasLocalStorage){ + delete localStorage[SETTINGS_KEY]; + } + rs.removeEventListener('error', onErrorCb); + rs.setBackend(undefined); + }; +})(this); + +remoteStorage = new RemoteStorage(); \ No newline at end of file diff --git a/release/0.10.0-beta2/remotestorage.min.js b/release/0.10.0-beta2/remotestorage.min.js new file mode 100644 index 000000000..eb84f11ff --- /dev/null +++ b/release/0.10.0-beta2/remotestorage.min.js @@ -0,0 +1,6 @@ +/** remotestorage.js 0.10.0-beta2, http://remotestorage.io, MIT-licensed **/ +(function(global){function getPromise(builder){var promise;if(typeof builder==="function"){setTimeout(function(){try{builder(promise)}catch(e){promise.reject(e)}},0)}var consumers=[],success,result;function notifyConsumer(consumer){if(success){var nextValue;if(consumer.fulfilled){try{nextValue=[consumer.fulfilled.apply(null,result)]}catch(exc){consumer.promise.reject(exc);return}}else{nextValue=result}if(nextValue[0]&&typeof nextValue[0].then==="function"){nextValue[0].then(consumer.promise.fulfill,consumer.promise.reject)}else{consumer.promise.fulfill.apply(null,nextValue)}}else{if(consumer.rejected){var ret;try{ret=consumer.rejected.apply(null,result)}catch(exc){consumer.promise.reject(exc);return}if(ret&&typeof ret.then==="function"){ret.then(consumer.promise.fulfill,consumer.promise.reject)}else{consumer.promise.fulfill(ret)}}else{consumer.promise.reject.apply(null,result)}}}function resolve(succ,res){if(result){console.error("WARNING: Can't resolve promise, already resolved!");return}success=succ;result=Array.prototype.slice.call(res);setTimeout(function(){var cl=consumers.length;if(cl===0&&!success){console.error("Possibly uncaught error: ",result,result[0]&&result[0].stack)}for(var i=0;i=n){this._init();RemoteStorage.log("Done cleaning up, emitting disconnected and disconnect events");this._emit("disconnected")}}.bind(this);if(n>0){this._cleanups.forEach(function(cleanup){var cleanupResult=cleanup(this);if(typeof cleanup==="object"&&typeof cleanup.then==="function"){cleanupResult.then(oneDone)}else{oneDone()}}.bind(this))}else{oneDone()}},setBackend:function(what){this.backend=what;if(this.localStorageAvailable()){if(what){localStorage["remotestorage:backend"]=what}else{delete localStorage["remotestorage:backend"]}}},onChange:function(path,handler){if(!this._pathHandlers.change[path]){this._pathHandlers.change[path]=[]}this._pathHandlers.change[path].push(handler)},enableLog:function(){RemoteStorage._log=true},disableLog:function(){RemoteStorage._log=false},log:function(){RemoteStorage.log.apply(RemoteStorage,arguments)},setApiKeys:function(type,keys){if(keys){this.apiKeys[type]=keys}else{delete this.apiKeys[type]}if(this.localStorageAvailable()){localStorage["remotestorage:api-keys"]=JSON.stringify(this.apiKeys)}},_init:function(){var self=this,readyFired=false;function fireReady(){try{if(!readyFired){self._emit("ready");readyFired=true}}catch(e){console.error("'ready' failed: ",e,e.stack);self._emit("error",e)}}this._loadFeatures(function(features){this.log("[RemoteStorage] All features loaded");this.local=features.local&&new features.local;if(this.local&&this.remote){this._setGPD(SyncedGetPutDelete,this);this._bindChange(this.local)}else if(this.remote){this._setGPD(this.remote,this.remote)}if(this.remote){this.remote.on("connected",function(){fireReady();self._emit("connected")});this.remote.on("not-connected",function(){fireReady();self._emit("not-connected")});if(this.remote.connected){fireReady();self._emit("connected")}}this._collectCleanupFunctions();try{this._allLoaded=true;this._emit("features-loaded")}catch(exc){logError(exc);this._emit("error",exc)}this._processPending()}.bind(this))},_collectCleanupFunctions:function(){for(var i=0;i=0}function isErrorStatus(status){return[401,403,404,412].indexOf(status)>=0}var onErrorCb;RS.WireClient=function(rs){this.connected=false;RS.eventHandling(this,"change","connected","wire-busy","wire-done","not-connected");onErrorCb=function(error){if(error instanceof RemoteStorage.Unauthorized){this.configure(undefined,undefined,undefined,null)}}.bind(this);rs.on("error",onErrorCb);if(hasLocalStorage){var settings;try{settings=JSON.parse(localStorage[SETTINGS_KEY])}catch(e){}if(settings){setTimeout(function(){this.configure(settings.userAddress,settings.href,settings.storageApi,settings.token)}.bind(this),0)}}this._revisionCache={};if(this.connected){setTimeout(this._emit.bind(this),0,"connected")}};RS.WireClient.REQUEST_TIMEOUT=3e4;RS.WireClient.prototype={_request:function(method,uri,token,headers,body,getEtag,fakeRevision){if((method==="PUT"||method==="DELETE")&&uri[uri.length-1]==="/"){throw"Don't "+method+" on directories!"}var promise=promising();var revision;var reqType;var self=this;headers["Authorization"]="Bearer "+token;this._emit("wire-busy",{method:method,isFolder:isFolder(uri)});RS.WireClient.request(method,uri,{body:body,headers:headers},function(error,response){if(error){self._emit("wire-done",{method:method,isFolder:isFolder(uri),success:false});self.online=false;promise.reject(error)}else{self._emit("wire-done",{method:method,isFolder:isFolder(uri),success:true});self.online=true;if(isErrorStatus(response.status)){RemoteStorage.log("[WireClient] Error response status",response.status);if(getEtag){revision=stripQuotes(response.getResponseHeader("ETag"))}else{revision=undefined}promise.fulfill(response.status,undefined,undefined,revision)}else if(isSuccessStatus(response.status)||response.status===200&&method!=="GET"){revision=stripQuotes(response.getResponseHeader("ETag"));RemoteStorage.log("[WireClient] Successful request",revision);promise.fulfill(response.status,undefined,undefined,revision)}else{var mimeType=response.getResponseHeader("Content-Type");var body;if(getEtag){revision=stripQuotes(response.getResponseHeader("ETag"))}else{revision=response.status===200?fakeRevision:undefined}if(!mimeType||mimeType.match(/charset=binary/)){RS.WireClient.readBinaryData(response.response,mimeType,function(result){RemoteStorage.log("[WireClient] Successful request with unknown or binary mime-type",revision);promise.fulfill(response.status,result,mimeType,revision)})}else{if(mimeType&&mimeType.match(/^application\/json/)){body=JSON.parse(response.responseText)}else{body=response.responseText}RemoteStorage.log("[WireClient] Successful request",revision);promise.fulfill(response.status,body,mimeType,revision)}}}});return promise},configure:function(userAddress,href,storageApi,token){if(typeof userAddress!=="undefined"){this.userAddress=userAddress}if(typeof href!=="undefined"){this.href=href}if(typeof storageApi!=="undefined"){this.storageApi=storageApi}if(typeof token!=="undefined"){this.token=token}if(typeof this.storageApi!=="undefined"){this._storageApi=STORAGE_APIS[this.storageApi]||API_HEAD;this.supportsRevs=this._storageApi>=API_00}if(this.href&&this.token){this.connected=true;this.online=true;this._emit("connected")}else{this.connected=false}if(hasLocalStorage){localStorage[SETTINGS_KEY]=JSON.stringify({userAddress:this.userAddress,href:this.href,token:this.token,storageApi:this.storageApi})}RS.WireClient.configureHooks.forEach(function(hook){hook.call(this)}.bind(this))},stopWaitingForToken:function(){if(!this.connected){this._emit("not-connected")}},get:function(path,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}var headers={};if(this.supportsRevs){if(options.ifNoneMatch){headers["If-None-Match"]=addQuotes(options.ifNoneMatch)}}else if(options.ifNoneMatch){var oldRev=this._revisionCache[path]}var promise=this._request("GET",this.href+cleanPath(path),this.token,headers,undefined,this.supportsRevs,this._revisionCache[path]);if(isFolder(path)){return promise.then(function(status,body,contentType,revision){var itemsMap={};if(status===200&&typeof body==="object"){if(Object.keys(body).length===0){status=404}else if(isFolderDescription(body)){for(var item in body.items){this._revisionCache[path+item]=body.items[item].ETag}itemsMap=body.items}else{Object.keys(body).forEach(function(key){this._revisionCache[path+key]=body[key];itemsMap[key]={ETag:body[key]}}.bind(this))}return promising().fulfill(status,itemsMap,contentType,revision)}else{return promising().fulfill(status,body,contentType,revision)}}.bind(this))}else{return promise}},put:function(path,body,contentType,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}if(!contentType.match(/charset=/)){contentType+="; charset="+(body instanceof ArrayBuffer||isArrayBufferView(body)?"binary":"utf-8")}var headers={"Content-Type":contentType};if(this.supportsRevs){if(options.ifMatch){headers["If-Match"]=addQuotes(options.ifMatch)}if(options.ifNoneMatch){headers["If-None-Match"]=addQuotes(options.ifNoneMatch)}}return this._request("PUT",this.href+cleanPath(path),this.token,headers,body,this.supportsRevs)},"delete":function(path,options){if(!this.connected){throw new Error("not connected (path: "+path+")")}if(!options){options={}}var headers={};if(this.supportsRevs){if(options.ifMatch){headers["If-Match"]=addQuotes(options.ifMatch)}}return this._request("DELETE",this.href+cleanPath(path),this.token,headers,undefined,this.supportsRevs)}};RS.WireClient.cleanPath=cleanPath;RS.WireClient.isArrayBufferView=isArrayBufferView;RS.WireClient.readBinaryData=readBinaryData;RS.WireClient.request=function(method,url,options,callback){RemoteStorage.log("[WireClient]",method,url);callback=callback.bind(this);var timedOut=false;var timer=setTimeout(function(){timedOut=true;callback("timeout")},RS.WireClient.REQUEST_TIMEOUT);var xhr=new XMLHttpRequest;xhr.open(method,url,true);if(options.responseType){xhr.responseType=options.responseType}if(options.headers){for(var key in options.headers){xhr.setRequestHeader(key,options.headers[key])}}xhr.onload=function(){if(timedOut){return}clearTimeout(timer);callback(null,xhr)};xhr.onerror=function(error){if(timedOut){return}clearTimeout(timer);callback(error)};var body=options.body;if(typeof body==="object"){if(isArrayBufferView(body)){}else if(body instanceof ArrayBuffer){body=new Uint8Array(body)}else{body=JSON.stringify(body)}}xhr.send(body)};Object.defineProperty(RemoteStorage.WireClient.prototype,"storageType",{get:function(){if(this.storageApi){var spec=this.storageApi.match(/draft-dejong-(remotestorage-\d\d)/);return spec?spec[1]:"2012.04"}}});RS.WireClient.configureHooks=[];RS.WireClient._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();remoteStorage.remote=new RS.WireClient(remoteStorage);this.online=true};RS.WireClient._rs_supported=function(){return!!global.XMLHttpRequest};RS.WireClient._rs_cleanup=function(remoteStorage){if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}remoteStorage.removeEventListener("error",onErrorCb)}})(typeof window!=="undefined"?window:global);(function(global){var haveXMLHttpRequest,hasLocalStorage;var SETTINGS_KEY="remotestorage:discover";var cachedInfo={};RemoteStorage.Discover=function(userAddress,callback){if(userAddress in cachedInfo){var info=cachedInfo[userAddress];callback(info.href,info.type,info.authURL);return}var hostname=userAddress.split("@")[1];var params="?resource="+encodeURIComponent("acct:"+userAddress);var urls=["https://"+hostname+"/.well-known/webfinger"+params,"https://"+hostname+"/.well-known/host-meta.json"+params,"http://"+hostname+"/.well-known/webfinger"+params,"http://"+hostname+"/.well-known/host-meta.json"+params];function tryOne(){var xhr=new XMLHttpRequest;var url=urls.shift();if(!url){return callback()}RemoteStorage.log("[Discover] Trying URL",url);xhr.open("GET",url,true);xhr.onabort=xhr.onerror=function(){console.error("webfinger error",arguments,"(",url,")");tryOne()};xhr.onload=function(){if(xhr.status!==200){return tryOne()}var profile;try{profile=JSON.parse(xhr.responseText)}catch(e){RemoteStorage.log("[Discover] Failed to parse profile ",xhr.responseText,e);tryOne();return}if(!profile.links){RemoteStorage.log("[Discover] Profile has no links section ",JSON.stringify(profile));tryOne();return}var link;profile.links.forEach(function(l){if(l.rel==="remotestorage"){link=l}else if(l.rel==="remoteStorage"&&!link){link=l}});RemoteStorage.log("[Discover] Got profile",profile,"and link",link);if(link){var authURL=link.properties["http://tools.ietf.org/html/rfc6749#section-4.2"]||link.properties["auth-endpoint"],storageType=link.properties["http://remotestorage.io/spec/version"]||link.type;cachedInfo[userAddress]={href:link.href,type:storageType,authURL:authURL};if(hasLocalStorage){localStorage[SETTINGS_KEY]=JSON.stringify({cache:cachedInfo})}callback(link.href,storageType,authURL)}else{tryOne()}};xhr.send()}tryOne()};RemoteStorage.Discover._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();if(hasLocalStorage){var settings;try{settings=JSON.parse(localStorage[SETTINGS_KEY])}catch(e){}if(settings){cachedInfo=settings.cache}}};RemoteStorage.Discover._rs_supported=function(){haveXMLHttpRequest=!!global.XMLHttpRequest;return haveXMLHttpRequest};RemoteStorage.Discover._rs_cleanup=function(){if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}}})(typeof window!=="undefined"?window:global);(function(global){function extractParams(){var location=RemoteStorage.Authorize.getLocation(),hashPos=location.href.indexOf("#"),hash;if(hashPos===-1){return}hash=location.href.substring(hashPos+1);if(hash.indexOf("=")===-1){return}return hash.split("&").reduce(function(m,kvs){var kv=kvs.split("=");m[decodeURIComponent(kv[0])]=decodeURIComponent(kv[1]);return m},{})}RemoteStorage.Authorize=function(authURL,scope,redirectUri,clientId){RemoteStorage.log("[Authorize] authURL = ",authURL);var url=authURL;url+=authURL.indexOf("?")>0?"&":"?";url+="redirect_uri="+encodeURIComponent(redirectUri.replace(/#.*$/,""));url+="&scope="+encodeURIComponent(scope);url+="&client_id="+encodeURIComponent(clientId);url+="&response_type=token";RemoteStorage.Authorize.setLocation(url)};RemoteStorage.prototype.authorize=function(authURL){this.access.setStorageType(this.remote.storageType);var scope=this.access.scopeParameter;var redirectUri=String(RemoteStorage.Authorize.getLocation());var clientId=redirectUri.match(/^(https?:\/\/[^\/]+)/)[0];RemoteStorage.Authorize(authURL,scope,redirectUri,clientId)};RemoteStorage.Authorize.getLocation=function(){return global.document.location};RemoteStorage.Authorize.setLocation=function(location){if(typeof location==="string"){global.document.location.href=location}else if(typeof location==="object"){global.document.location=location}else{throw"Invalid location "+location}};RemoteStorage.Authorize._rs_supported=function(remoteStorage){return typeof document!=="undefined"};var onFeaturesLoaded;RemoteStorage.Authorize._rs_init=function(remoteStorage){onFeaturesLoaded=function(){var authParamsUsed=false;if(params){if(params.error){throw"Authorization server errored: "+params.error}if(params.access_token){remoteStorage.remote.configure(undefined,undefined,undefined,params.access_token);authParamsUsed=true}if(params.remotestorage){remoteStorage.connect(params.remotestorage);authParamsUsed=true}}if(!authParamsUsed){remoteStorage.remote.stopWaitingForToken()}};var params=extractParams(),location;if(params){location=RemoteStorage.Authorize.getLocation();location.hash=""}remoteStorage.on("features-loaded",onFeaturesLoaded)};RemoteStorage.Authorize._rs_cleanup=function(remoteStorage){remoteStorage.removeEventListener("features-loaded",onFeaturesLoaded)}})(typeof window!=="undefined"?window:global);(function(global){var SETTINGS_KEY="remotestorage:access";RemoteStorage.Access=function(){this.reset()};RemoteStorage.Access.prototype={claim:function(scope,mode){if(typeof scope!=="string"||scope.indexOf("/")!==-1||scope.length===0){throw new Error("Scope should be a non-empty string without forward slashes")}if(!mode.match(/^rw?$/)){throw new Error("Mode should be either 'r' or 'rw'")}this._adjustRootPaths(scope);this.scopeModeMap[scope]=mode},get:function(scope){return this.scopeModeMap[scope]},remove:function(scope){var savedMap={};var name;for(name in this.scopeModeMap){savedMap[name]=this.scopeModeMap[name]}this.reset();delete savedMap[scope];for(name in savedMap){this.set(name,savedMap[name])}},checkPermission:function(scope,mode){var actualMode=this.get(scope);return actualMode&&(mode==="r"||actualMode==="rw")},checkPathPermission:function(path,mode){if(this.checkPermission("*",mode)){return true}return!!this.checkPermission(this._getModuleName(path),mode)},reset:function(){this.rootPaths=[];this.scopeModeMap={}},_getModuleName:function(path){if(path[0]!=="/"){throw new Error("Path should start with a slash")}var moduleMatch=path.replace(/^\/public/,"").match(/^\/([^\/]*)\//);return moduleMatch?moduleMatch[1]:"*"},_adjustRootPaths:function(newScope){if("*"in this.scopeModeMap||newScope==="*"){this.rootPaths=["/"]}else if(!(newScope in this.scopeModeMap)){this.rootPaths.push("/"+newScope+"/");this.rootPaths.push("/public/"+newScope+"/")}},_scopeNameForParameter:function(scope){if(scope.name==="*"&&this.storageType){if(this.storageType==="2012.04"){return""}else if(this.storageType.match(/remotestorage-0[01]/)){return"root"}}return scope.name},setStorageType:function(type){this.storageType=type}};Object.defineProperty(RemoteStorage.Access.prototype,"scopes",{get:function(){return Object.keys(this.scopeModeMap).map(function(key){return{name:key,mode:this.scopeModeMap[key]}}.bind(this))}});Object.defineProperty(RemoteStorage.Access.prototype,"scopeParameter",{get:function(){return this.scopes.map(function(scope){return this._scopeNameForParameter(scope)+":"+scope.mode}.bind(this)).join(" ")}});Object.defineProperty(RemoteStorage.prototype,"access",{get:function(){var access=new RemoteStorage.Access;Object.defineProperty(this,"access",{value:access});return access},configurable:true});RemoteStorage.Access._rs_init=function(){}})(typeof window!=="undefined"?window:global);(function(pMode){var mode=pMode,env={},isBackground=false;RemoteStorage.Env=function(){return env};RemoteStorage.Env.isBrowser=function(){return mode==="browser"};RemoteStorage.Env.isNode=function(){return mode==="node"};RemoteStorage.Env.goBackground=function(){isBackground=true;RemoteStorage.Env._emit("background")};RemoteStorage.Env.goForeground=function(){isBackground=false;RemoteStorage.Env._emit("foreground")};RemoteStorage.Env._rs_init=function(remoteStorage){RemoteStorage.eventHandling(RemoteStorage.Env,"background","foreground");if(mode==="browser"){if(typeof document.hidden!=="undefined"){env.hiddenProperty="hidden";env.visibilityChangeEvent="visibilitychange"}else if(typeof document.mozHidden!=="undefined"){env.hiddenProperty="mozHidden";env.visibilityChangeEvent="mozvisibilitychange"}else if(typeof document.msHidden!=="undefined"){env.hiddenProperty="msHidden";env.visibilityChangeEvent="msvisibilitychange"}else if(typeof document.webkitHidden!=="undefined"){env.hiddenProperty="webkitHidden";env.visibilityChangeEvent="webkitvisibilitychange"}document.addEventListener(env.visibilityChangeEvent,function(){if(document[env.hiddenProperty]){RemoteStorage.Env.goBackground()}else{RemoteStorage.Env.goForeground()}},false)}};RemoteStorage.Env._rs_cleanup=function(remoteStorage){}})(typeof window!=="undefined"?"browser":"node");(function(){"use strict";var dictionary={view_info:'This app allows you to use your own storage. Learn more!',view_connect:"Connect remote storage",view_connecting:"Connecting %s",view_offline:"Offline",view_error_occured:"Sorry! An error occured.",view_confirm_reset:"Are you sure you want to reset everything? This will clear your local data and reload the page.",view_get_me_out:"Get me out of here!",view_error_plz_report:'If this problem persists, please let us know!',view_unauthorized:"Unauthorized! Click here to reconnect."};RemoteStorage.I18n={translate:function(){var str=arguments[0],params=Array.prototype.splice.call(arguments,1);if(typeof dictionary[str]!=="string"){throw"Unknown translation string: "+str}else{str=dictionary[str]}return str.replace(/%s/g,function(){return params.shift()})},getDictionary:function(){return dictionary},setDictionary:function(newDictionary){dictionary=newDictionary}}})();RemoteStorage.Assets={connectIcon:"",disconnectIcon:"",dropbox:"",googledrive:"",remoteStorageIcon:"",remoteStorageIconError:"",remoteStorageIconOffline:"",syncIcon:"",widget:'
{{view_connect}}
{{ERROR_MSG}}

{{view_error_plz_report}}

{{USER_ADDRESS}}

{{view_unauthorized}}

{{view_info}}
Connect to Dropbox Connect to Google Drive ',widgetCss:'/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}.rs-bubble .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-backends{position:relative;top:5px;right:0;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget form input[type=email]:focus{background:#223;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .rs-content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble.rs-hidden{display:none;}.remotestorage-state-busy .rs-bubble{display:block;}.remotestorage-state-busy .remotestorage-connected{display:block;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}'}; +(function(window){var hasLocalStorage;var LS_STATE_KEY="remotestorage:widget:state";var VALID_ENTRY_STATES={initial:true,connected:true,offline:true};function stateSetter(widget,state){RemoteStorage.log("[Widget] Producing stateSetter for",state);return function(){RemoteStorage.log("[Widget] Setting state",state,arguments);if(hasLocalStorage){localStorage[LS_STATE_KEY]=state}if(widget.view){if(widget.rs.remote){widget.view.setUserAddress(widget.rs.remote.userAddress)}widget.view.setState(state,arguments)}else{widget._rememberedState=state}}}function errorsHandler(widget){return function(error){if(error instanceof RemoteStorage.DiscoveryError){console.error("Discovery failed",error,'"'+error.message+'"');widget.view.setState("initial",[error.message])}else if(error instanceof RemoteStorage.SyncError){widget.view.setState("offline",[])}else if(error instanceof RemoteStorage.Unauthorized){widget.view.setState("unauthorized")}else{RemoteStorage.log("[Widget] Unknown error");widget.view.setState("error",[error])}}}function flashFor(evt){if(evt.method==="GET"&&evt.isFolder){return false}return true}RemoteStorage.Widget=function(remoteStorage){var self=this;var requestsToFlashFor=0;this.rs=remoteStorage;this.rs.remote.on("connected",stateSetter(this,"connected"));this.rs.on("disconnected",stateSetter(this,"initial"));this.rs.on("connecting",stateSetter(this,"authing"));this.rs.on("authing",stateSetter(this,"authing"));this.rs.on("error",errorsHandler(this));if(this.rs.remote){this.rs.remote.on("wire-busy",function(evt){if(flashFor(evt)){requestsToFlashFor++;stateSetter(self,"busy")()}});this.rs.remote.on("wire-done",function(evt){if(flashFor(evt)){requestsToFlashFor--;if(requestsToFlashFor<=0){stateSetter(self,"connected")()}}})}if(hasLocalStorage){var state=localStorage[LS_STATE_KEY];if(state&&VALID_ENTRY_STATES[state]){this._rememberedState=state}}};RemoteStorage.Widget.prototype={display:function(domID){if(!this.view){this.setView(new RemoteStorage.Widget.View(this.rs))}this.view.display.apply(this.view,arguments);return this},linkWidgetToSync:function(){if(typeof this.rs.sync==="object"&&typeof this.rs.sync.sync==="function"){this.view.on("sync",this.rs.sync.sync.bind(this.rs.sync))}else{RemoteStorage.log("[Widget] typeof this.rs.sync check fail",this.rs.sync);setTimeout(this.linkWidgetToSync.bind(this),1e3)}},setView:function(view){this.view=view;this.view.on("connect",function(options){if(typeof options==="string"){this.rs.connect(options)}else if(options.special){this.rs[options.special].connect(options)}}.bind(this));this.view.on("disconnect",this.rs.disconnect.bind(this.rs));this.linkWidgetToSync();try{this.view.on("reset",function(){var location=RemoteStorage.Authorize.getLocation();this.rs.on("disconnected",location.reload.bind(location));this.rs.disconnect()}.bind(this))}catch(e){if(e.message&&e.message.match(/Unknown event/)){}else{throw e}}if(this._rememberedState){setTimeout(stateSetter(this,this._rememberedState),0);delete this._rememberedState}}};RemoteStorage.prototype.displayWidget=function(domID){return this.widget.display(domID)};RemoteStorage.Widget._rs_init=function(remoteStorage){hasLocalStorage=remoteStorage.localStorageAvailable();if(!remoteStorage.widget){remoteStorage.widget=new RemoteStorage.Widget(remoteStorage)}};RemoteStorage.Widget._rs_supported=function(remoteStorage){return typeof document!=="undefined"}})(typeof window!=="undefined"?window:global);(function(window){var t=RemoteStorage.I18n.translate;var cEl=function(){return document.createElement.apply(document,arguments)};function gCl(parent,className){return parent.getElementsByClassName(className)[0]}function gTl(parent,tagName){return parent.getElementsByTagName(tagName)[0]}function removeClass(el,className){return el.classList.remove(className)}function addClass(el,className){return el.classList.add(className)}function stopPropagation(event){if(typeof event.stopPropagation==="function"){event.stopPropagation()}else{event.cancelBubble=true}}function setupButton(parent,className,iconName,eventListener){var element=gCl(parent,className);if(typeof iconName!=="undefined"){var img=gTl(element,"img");(img||element).src=RemoteStorage.Assets[iconName]}element.addEventListener("click",eventListener);return element}RemoteStorage.Widget.View=function(remoteStorage){this.rs=remoteStorage;if(typeof document==="undefined"){throw"Widget not supported"}RemoteStorage.eventHandling(this,"connect","disconnect","sync","display","reset");for(var event in this.events){this.events[event]=this.events[event].bind(this)}this.hideBubbleOnBodyClick=function(event){for(var p=event.target;p!==document.body;p=p.parentElement){if(p.id==="remotestorage-widget"){return}}this.hideBubble()}.bind(this)};RemoteStorage.Widget.View.prototype={connectGdrive:function(){this._emit("connect",{special:"googledrive"})},connectDropbox:function(){this._emit("connect",{special:"dropbox"})},setState:function(state,args){RemoteStorage.log("[View] widget.view.setState(",state,",",args,");");var s=this.states[state];if(typeof s==="undefined"){throw new Error("Bad State assigned to view: "+state)}s.apply(this,args)},setUserAddress:function(addr){this.userAddress=addr||"";var el;if(this.div&&(el=gTl(this.div,"form").userAddress)){el.value=this.userAddress}},toggleBubble:function(event){if(this.bubble.className.search("rs-hidden")<0){this.hideBubble(event)}else{this.showBubble(event)}},hideBubble:function(){addClass(this.bubble,"rs-hidden");document.body.removeEventListener("click",this.hideBubbleOnBodyClick)},showBubble:function(event){removeClass(this.bubble,"rs-hidden");if(typeof event!=="undefined"){stopPropagation(event)}document.body.addEventListener("click",this.hideBubbleOnBodyClick);gTl(this.bubble,"form").userAddress.focus()},display:function(domID){if(typeof this.div!=="undefined"){return this.div}var element=cEl("div");var style=cEl("style");style.innerHTML=RemoteStorage.Assets.widgetCss;element.id="remotestorage-widget";element.innerHTML=RemoteStorage.Assets.widget;element.appendChild(style);if(domID){var parent=document.getElementById(domID);if(!parent){throw'Failed to find target DOM element with id="'+domID+'"'}parent.appendChild(element)}else{document.body.appendChild(element)}setupButton(element,"rs-sync","syncIcon",this.events.sync);setupButton(element,"rs-disconnect","disconnectIcon",this.events.disconnect);setupButton(element,"remotestorage-reset",undefined,this.events.reset);var cb=setupButton(element,"connect","connectIcon",this.events.connect);this.form=gTl(element,"form");var el=this.form.userAddress;el.addEventListener("keyup",function(event){if(event.target.value){cb.removeAttribute("disabled")}else{cb.setAttribute("disabled","disabled")}});if(this.userAddress){el.value=this.userAddress}this.cube=setupButton(element,"rs-cube","remoteStorageIcon",this.toggleBubble.bind(this));setupButton(element,"rs-dropbox","dropbox",this.connectDropbox.bind(this));setupButton(element,"rs-googledrive","googledrive",this.connectGdrive.bind(this));var bubbleDontCatch={INPUT:true,BUTTON:true,IMG:true};var eventListener=function(event){if(!bubbleDontCatch[event.target.tagName]&&!this.div.classList.contains("remotestorage-state-unauthorized")){this.showBubble(event)}}.bind(this);this.bubble=setupButton(element,"rs-bubble",undefined,eventListener);this.hideBubble();this.div=element;this.states.initial.call(this);this.events.display.call(this);return this.div},_renderTranslatedInitialContent:function(){gCl(this.div,"rs-status-text").innerHTML=t("view_connect");gCl(this.div,"remotestorage-reset").innerHTML=t("view_get_me_out");gCl(this.div,"rs-error-plz-report").innerHTML=t("view_error_plz_report");gCl(this.div,"remotestorage-unauthorized").innerHTML=t("view_unauthorized")},states:{initial:function(message){var cube=this.cube;var info=message||t("view_info");this._renderTranslatedInitialContent();if(message){cube.src=RemoteStorage.Assets.remoteStorageIconError;removeClass(this.cube,"remotestorage-loading");this.showBubble();setTimeout(function(){cube.src=RemoteStorage.Assets.remoteStorageIcon},5e3)}else{this.hideBubble()}this.div.className="remotestorage-state-initial";var backends=1;if(this._activateBackend("dropbox")){backends+=1}if(this._activateBackend("googledrive")){backends+=1}gCl(this.div,"rs-bubble-text").style.paddingRight=backends*32+8+"px";var cb=gCl(this.div,"connect");if(this.form.userAddress.value){cb.removeAttribute("disabled")}var infoEl=gCl(this.div,"rs-info-msg");infoEl.innerHTML=info;if(message){infoEl.classList.add("remotestorage-error-info")}else{infoEl.classList.remove("remotestorage-error-info")}},authing:function(){this.div.removeEventListener("click",this.events.connect);this.div.className="remotestorage-state-authing";gCl(this.div,"rs-status-text").innerHTML=t("view_connecting",this.userAddress);addClass(this.cube,"remotestorage-loading")},connected:function(){this.div.className="remotestorage-state-connected";gCl(this.div,"userAddress").innerHTML=this.userAddress;this.cube.src=RemoteStorage.Assets.remoteStorageIcon;removeClass(this.cube,"remotestorage-loading");var icons={googledrive:gCl(this.div,"rs-googledrive"),dropbox:gCl(this.div,"rs-dropbox")};icons.googledrive.style.display=icons.dropbox.style.display="none";if(icons[this.rs.backend]){icons[this.rs.backend].style.display="inline-block";gCl(this.div,"rs-bubble-text").style.paddingRight=2*32+8+"px"}else{gCl(this.div,"rs-bubble-text").style.paddingRight=32+8+"px"}},busy:function(){this.div.className="remotestorage-state-busy";addClass(this.cube,"remotestorage-loading");this.hideBubble()},offline:function(){this.div.className="remotestorage-state-offline";this.cube.src=RemoteStorage.Assets.remoteStorageIconOffline;gCl(this.div,"rs-status-text").innerHTML=t("view_offline")},error:function(err){var errorMsg=err;this.div.className="remotestorage-state-error";gCl(this.div,"rs-bubble-text").innerHTML=""+t("view_error_occured")+"";if(err instanceof Error){errorMsg=err.message+"\n\n"+err.stack}gCl(this.div,"rs-error-msg").textContent=errorMsg;this.cube.src=RemoteStorage.Assets.remoteStorageIconError;this.showBubble()},unauthorized:function(){this.div.className="remotestorage-state-unauthorized";this.cube.src=RemoteStorage.Assets.remoteStorageIconError;this.showBubble();this.div.addEventListener("click",this.events.connect)}},events:{connect:function(event){stopPropagation(event);event.preventDefault();this._emit("connect",gTl(this.div,"form").userAddress.value)},sync:function(event){stopPropagation(event);event.preventDefault();this._emit("sync")},disconnect:function(event){stopPropagation(event);event.preventDefault();this._emit("disconnect")},reset:function(event){event.preventDefault();var result=window.confirm(t("view_confirm_reset"));if(result){this._emit("reset")}},display:function(event){if(event){event.preventDefault()}this._emit("display")}},_activateBackend:function activateBackend(backendName){var className="rs-"+backendName;if(this.rs.apiKeys[backendName]){gCl(this.div,className).style.display="inline-block";return true}else{gCl(this.div,className).style.display="none";return false}}}})(typeof window!=="undefined"?window:global);(function(global){var ValidatorContext=function(parent,collectMultiple){this.missing=[];this.schemas=parent?Object.create(parent.schemas):{};this.collectMultiple=collectMultiple;this.errors=[];this.handleError=collectMultiple?this.collectError:this.returnError};ValidatorContext.prototype.returnError=function(error){return error};ValidatorContext.prototype.collectError=function(error){if(error){this.errors.push(error)}return null};ValidatorContext.prototype.prefixErrors=function(startIndex,dataPath,schemaPath){for(var i=startIndex;ischema.maximum){return new ValidationError(ErrorCodes.NUMBER_MAXIMUM,"Value "+data+" is greater than maximum "+schema.maximum).prefixWith(null,"maximum")}if(schema.exclusiveMaximum&&data==schema.maximum){return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE,"Value "+data+" is equal to exclusive maximum "+schema.maximum).prefixWith(null,"exclusiveMaximum")}}return null};ValidatorContext.prototype.validateString=function validateString(data,schema){return this.validateStringLength(data,schema)||this.validateStringPattern(data,schema)||null};ValidatorContext.prototype.validateStringLength=function validateStringLength(data,schema){if(typeof data!="string"){return null}if(schema.minLength!=undefined){if(data.lengthschema.maxLength){return new ValidationError(ErrorCodes.STRING_LENGTH_LONG,"String is too long ("+data.length+" chars), maximum "+schema.maxLength).prefixWith(null,"maxLength")}}return null};ValidatorContext.prototype.validateStringPattern=function validateStringPattern(data,schema){if(typeof data!="string"||schema.pattern==undefined){return null}var regexp=new RegExp(schema.pattern);if(!regexp.test(data)){return new ValidationError(ErrorCodes.STRING_PATTERN,"String does not match pattern").prefixWith(null,"pattern")}return null};ValidatorContext.prototype.validateArray=function validateArray(data,schema){if(!Array.isArray(data)){return null}return this.validateArrayLength(data,schema)||this.validateArrayUniqueItems(data,schema)||this.validateArrayItems(data,schema)||null};ValidatorContext.prototype.validateArrayLength=function validateArrayLength(data,schema){if(schema.minItems!=undefined){if(data.lengthschema.maxItems){var error=new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG,"Array is too long ("+data.length+" chars), maximum "+schema.maxItems).prefixWith(null,"maxItems");if(this.handleError(error)){return error}}}return null};ValidatorContext.prototype.validateArrayUniqueItems=function validateArrayUniqueItems(data,schema){if(schema.uniqueItems){for(var i=0;ischema.maxProperties){var error=new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM,"Too many properties defined ("+keys.length+"), maximum "+schema.maxProperties).prefixWith(null,"maxProperties");if(this.handleError(error)){return error}}}return null};ValidatorContext.prototype.validateObjectRequiredProperties=function validateObjectRequiredProperties(data,schema){if(schema.required!=undefined){for(var i=0;i0&&url.charAt(url.length-1)=="/"||remainder.charAt(0)=="#"||remainder.charAt(0)=="?"){if(map[schema.id]==undefined){map[schema.id]=schema}}}}if(typeof schema=="object"){for(var key in schema){if(key!="enum"&&typeof schema[key]=="object"){searchForTrustedSchemas(map,schema[key],url)}}}return map}var globalContext=new ValidatorContext;var publicApi={validate:function(data,schema){var context=new ValidatorContext(globalContext);if(typeof schema=="string"){schema={$ref:schema}}var added=context.addSchema("",schema);var error=context.validateAll(data,schema);this.error=error;this.missing=context.missing;this.valid=error==null;return this.valid},validateResult:function(){var result={};this.validate.apply(result,arguments);return result},validateMultiple:function(data,schema){var context=new ValidatorContext(globalContext,true);if(typeof schema=="string"){schema={$ref:schema}}context.addSchema("",schema);context.validateAll(data,schema);var result={};result.errors=context.errors;result.missing=context.missing;result.valid=result.errors.length==0;return result},addSchema:function(url,schema){return globalContext.addSchema(url,schema)},getSchema:function(url){return globalContext.getSchema(url)},missing:[],error:null,normSchema:normSchema,resolveUrl:resolveUrl,errorCodes:ErrorCodes};global.tv4=publicApi})(typeof window!="undefined"?window:global);var CHARS="0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz".split("");Math.uuid=function(len,radix){var chars=CHARS,uuid=[],i;radix=radix||chars.length;if(len){for(i=0;i2){this.moduleName=parts[1]}else{this.moduleName="root"}RS.eventHandling(this,"change");this.on=this.on.bind(this);storage.onChange(this.base,this._fireChange.bind(this))};RS.BaseClient.prototype={extend:function(object){for(var key in object){this[key]=object[key]}return this},scope:function(path){return new RS.BaseClient(this.storage,this.makePath(path))},getListing:function(path,maxAge){if(typeof path!=="string"){path="" +}else if(path.length>0&&path[path.length-1]!=="/"){throw"Not a folder: "+path}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getListing must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body){return status===404?undefined:body})},getAll:function(path,maxAge){if(typeof path!=="string"){path=""}else if(path.length>0&&path[path.length-1]!=="/"){throw"Not a folder: "+path}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getAll must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body){if(status===404){return}if(typeof body==="object"){var promise=promising();var count=Object.keys(body).length,i=0;if(count===0){return}for(var key in body){this.storage.get(this.makePath(path+key),maxAge).then(function(status,b){body[this.key]=b;i++;if(i===count){promise.fulfill(body)}}.bind({key:key}))}return promise}}.bind(this))},getFile:function(path,maxAge){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.getFile must be a string")}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getFile must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body,mimeType,revision){return{data:body,mimeType:mimeType,revision:revision}})},storeFile:function(mimeType,path,body){if(typeof mimeType!=="string"){return promising().reject("Argument 'mimeType' of baseClient.storeFile must be a string")}if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.storeFile must be a string")}if(typeof body!=="string"&&typeof body!=="object"){return promising().reject("Argument 'body' of baseClient.storeFile must be a string, ArrayBuffer, or ArrayBufferView")}var self=this;return this.storage.put(this.makePath(path),body,mimeType).then(function(status,_body,_mimeType,revision){if(status===200||status===201){return revision}else{throw"Request (PUT "+self.makePath(path)+") failed with status: "+status}})},getObject:function(path,maxAge){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.getObject must be a string")}if(maxAgeInvalid(maxAge)){return promising().reject("Argument 'maxAge' of baseClient.getObject must be undefined or a number")}return this.storage.get(this.makePath(path),maxAge).then(function(status,body,mimeType,revision){if(typeof body==="object"){return body}else if(typeof body!=="undefined"&&status===200){throw"Not an object: "+this.makePath(path)}})},storeObject:function(typeAlias,path,object){if(typeof typeAlias!=="string"){return promising().reject("Argument 'typeAlias' of baseClient.storeObject must be a string")}if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.storeObject must be a string")}if(typeof object!=="object"){return promising().reject("Argument 'object' of baseClient.storeObject must be an object")}this._attachType(object,typeAlias);try{var validationResult=this.validate(object);if(!validationResult.valid){return promising(function(p){p.reject(validationResult)})}}catch(exc){if(!(exc instanceof RS.BaseClient.Types.SchemaNotFound)){return promising().reject(exc)}}return this.storage.put(this.makePath(path),object,"application/json; charset=UTF-8").then(function(status,_body,_mimeType,revision){if(status===200||status===201){return revision}else{throw"Request (PUT "+this.makePath(path)+") failed with status: "+status}}.bind(this))},remove:function(path){if(typeof path!=="string"){return promising().reject("Argument 'path' of baseClient.remove must be a string")}return this.storage.delete(this.makePath(path))},cache:function(path,strategy){if(typeof path!=="string"){throw"Argument 'path' of baseClient.cache must be a string"}if(strategy===false){deprecate("caching strategy ",'<"FLUSH">');strategy="FLUSH"}else if(strategy===undefined){strategy="ALL"}else if(typeof strategy!=="string"){deprecate("that caching strategy",'<"ALL">');strategy="ALL"}if(strategy!=="FLUSH"&&strategy!=="SEEN"&&strategy!=="ALL"){throw"Argument 'strategy' of baseclient.cache must be one of "+'["FLUSH", "SEEN", "ALL"]'}this.storage.caching.set(this.makePath(path),strategy);return this},flush:function(path){return this.storage.local.flush(path)},makePath:function(path){return this.base+(path||"")},_fireChange:function(event){this._emit("change",event)},_cleanPath:RS.WireClient.cleanPath,getItemURL:function(path){if(typeof path!=="string"){throw"Argument 'path' of baseClient.getItemURL must be a string"}if(this.storage.connected){path=this._cleanPath(this.makePath(path));return this.storage.remote.href+path}else{return undefined}},uuid:function(){return Math.uuid()}};RS.BaseClient._rs_init=function(){RS.prototype.scope=function(path){if(typeof path!=="string"){throw"Argument 'path' of baseClient.scope must be a string"}return new RS.BaseClient(this,path)}};maxAgeInvalid=function(maxAge){return typeof maxAge!=="undefined"&&typeof maxAge!=="number"}})(typeof window!=="undefined"?window:global);(function(global){RemoteStorage.BaseClient.Types={uris:{},schemas:{},aliases:{},declare:function(moduleName,alias,uri,schema){var fullAlias=moduleName+"/"+alias;if(schema.extends){var extendedAlias;var parts=schema.extends.split("/");if(parts.length===1){extendedAlias=moduleName+"/"+parts.shift()}else{extendedAlias=parts.join("/")}var extendedUri=this.uris[extendedAlias];if(!extendedUri){throw"Type '"+fullAlias+"' tries to extend unknown schema '"+extendedAlias+"'"}schema.extends=this.schemas[extendedUri]}this.uris[fullAlias]=uri;this.aliases[uri]=fullAlias;this.schemas[uri]=schema},resolveAlias:function(alias){return this.uris[alias]},getSchema:function(uri){return this.schemas[uri]},inScope:function(moduleName){var ml=moduleName.length;var schemas={};for(var alias in this.uris){if(alias.substr(0,ml+1)===moduleName+"/"){var uri=this.uris[alias];schemas[uri]=this.schemas[uri]}}return schemas}};var SchemaNotFound=function(uri){var error=new Error("Schema not found: "+uri);error.name="SchemaNotFound";return error};SchemaNotFound.prototype=Error.prototype;RemoteStorage.BaseClient.Types.SchemaNotFound=SchemaNotFound;RemoteStorage.BaseClient.prototype.extend({validate:function(object){var schema=RemoteStorage.BaseClient.Types.getSchema(object["@context"]);if(schema){return tv4.validateResult(object,schema)}else{throw new SchemaNotFound(object["@context"])}},declareType:function(alias,uri,schema){if(!schema){schema=uri;uri=this._defaultTypeURI(alias)}RemoteStorage.BaseClient.Types.declare(this.moduleName,alias,uri,schema)},_defaultTypeURI:function(alias){return"http://remotestoragejs.com/spec/modules/"+this.moduleName+"/"+alias},_attachType:function(object,alias){object["@context"]=RemoteStorage.BaseClient.Types.resolveAlias(alias)||this._defaultTypeURI(alias)}});Object.defineProperty(RemoteStorage.BaseClient.prototype,"schemas",{configurable:true,get:function(){return RemoteStorage.BaseClient.Types.inScope(this.moduleName)}})})(typeof window!=="undefined"?window:global);(function(global){var SETTINGS_KEY="remotestorage:caching";function containingFolder(path){if(path===""){return"/"}if(!path){throw"Path not given!"}return path.replace(/\/+/g,"/").replace(/[^\/]+\/?$/,"")}RemoteStorage.Caching=function(){this.reset()};RemoteStorage.Caching.prototype={pendingActivations:[],set:function(path,value){if(typeof path!=="string"){throw new Error("path should be a string")}if(typeof value==="undefined"){throw new Error("value should be 'FLUSH', 'SEEN', or 'ALL'")}this._rootPaths[path]=value;if(value==="ALL"){if(this.activateHandler){this.activateHandler(path)}else{this.pendingActivations.push(path)}}},enable:function(path){this.set(path,"ALL")},disable:function(path){this.set(path,"FLUSH")},onActivate:function(cb){var i;RemoteStorage.log("[Caching] Setting activate handler",cb,this.pendingActivations);this.activateHandler=cb;for(i=0;i0},collectDiffTasks:function(){var num=0;return this.local.forAllNodes(function(node){if(num>100){return}if(this.isCorrupt(node)){RemoteStorage.log("[Sync] WARNING: corrupt node in local cache",node);if(typeof node==="object"&&node.path){this.addTask(node.path);num++}}else if(this.needsFetch(node)&&this.access.checkPathPermission(node.path,"r")){this.addTask(node.path);num++}else if(this.isDocumentNode(node)&&this.needsPush(node)&&this.access.checkPathPermission(node.path,"rw")){this.addTask(node.path);num++}}.bind(this)).then(function(){return num},function(err){throw err})},inConflict:function(node){return node.local&&node.remote&&(node.remote.body!==undefined||node.remote.itemsMap)},needsRefresh:function(node){if(node.common){if(!node.common.timestamp){return true}return this.now()-node.common.timestamp>syncInterval}return false},needsFetch:function(node){if(this.inConflict(node)){return true}if(node.common&&node.common.itemsMap===undefined&&node.common.body===undefined){return true}if(node.remote&&node.remote.itemsMap===undefined&&node.remote.body===undefined){return true}return false},needsPush:function(node){if(this.inConflict(node)){return false}if(node.local&&!node.push){return true}},needsRemotePut:function(node){return node.local&&node.local.body},needsRemoteDelete:function(node){return node.local&&node.local.body===false},getParentPath:function(path){var parts=path.match(/^(.*\/)([^\/]+\/?)$/);if(parts){return parts[1]}else{throw new Error('Not a valid path: "'+path+'"')}},deleteChildPathsFromTasks:function(){for(var path in this._tasks){paths=this.local._getInternals().pathsFromRoot(path);for(var i=1;i=numToAdd){return true}}}return numAdded>=numToAdd},collectTasks:function(alsoCheckRefresh){if(this.hasTasks()||this.stopped){promise=promising();promise.fulfill();return promise}return this.collectDiffTasks().then(function(numDiffs){if(numDiffs||alsoCheckRefresh===false){promise=promising();promise.fulfill();return promise}else{return this.collectRefreshTasks()}}.bind(this),function(err){throw err})},addTask:function(path,cb){if(!this._tasks[path]){this._tasks[path]=[]}if(typeof cb==="function"){this._tasks[path].push(cb)}},sync:function(){var promise=promising();this.done=false;if(!this.doTasks()){return this.collectTasks().then(function(){try{this.doTasks()}catch(e){RemoteStorage.log("[Sync] doTasks error",e)}}.bind(this),function(e){RemoteStorage.log("[Sync] Sync error",e);throw new Error("Local cache unavailable")})}else{return promising().fulfill()}}};RemoteStorage.prototype.getSyncInterval=function(){return syncInterval};RemoteStorage.prototype.setSyncInterval=function(interval){if(typeof interval!=="number"){throw interval+" is not a valid sync interval"}syncInterval=parseInt(interval,10);if(this._syncTimer){this.stopSync();this._syncTimer=setTimeout(this.syncCycle.bind(this),interval)}};var SyncError=function(originalError){var msg="Sync failed: ";if(typeof originalError==="object"&&"message"in originalError){msg+=originalError.message}else{msg+=originalError}this.originalError=originalError;Error.apply(this,[msg])};SyncError.prototype=Object.create(Error.prototype);RemoteStorage.SyncError=SyncError;RemoteStorage.prototype.syncCycle=function(){if(this.sync.stopped){return}this.sync.on("done",function(){RemoteStorage.log("[Sync] Sync done. Setting timer to",this.getSyncInterval());if(!this.sync.stopped){if(this._syncTimer){clearTimeout(this._syncTimer)}this._syncTimer=setTimeout(this.sync.sync.bind(this.sync),this.getSyncInterval())}}.bind(this));this.sync.sync()};RemoteStorage.prototype.stopSync=function(){if(this.sync){RemoteStorage.log("[Sync] Stopping sync");this.sync.stopped=true}else{RemoteStorage.log("[Sync] Will instantiate sync stopped");this.syncStopped=true}};RemoteStorage.prototype.startSync=function(){this.sync.stopped=false;this.syncStopped=false;this.sync.sync()};var syncCycleCb;RemoteStorage.Sync._rs_init=function(remoteStorage){syncCycleCb=function(){RemoteStorage.log("[Sync] syncCycleCb calling syncCycle");if(!remoteStorage.sync){remoteStorage.sync=new RemoteStorage.Sync(remoteStorage.local,remoteStorage.remote,remoteStorage.access,remoteStorage.caching);if(remoteStorage.syncStopped){RemoteStorage.log("[Sync] Instantiating sync stopped");remoteStorage.sync.stopped=true;delete remoteStorage.syncStopped}}RemoteStorage.log("[Sync] syncCycleCb calling syncCycle");remoteStorage.syncCycle()};remoteStorage.on("ready",syncCycleCb)};RemoteStorage.Sync._rs_cleanup=function(remoteStorage){remoteStorage.stopSync();remoteStorage.removeEventListener("ready",syncCycleCb)}})(typeof window!=="undefined"?window:global);(function(){function isFolder(path){return path.substr(-1)==="/"}function isDocument(path){return path.substr(-1)!=="/"}function deepClone(obj){if(obj===undefined){return undefined}else{return JSON.parse(JSON.stringify(obj))}}function equal(obj1,obj2){return JSON.stringify(obj1)===JSON.stringify(obj2)}function getLatest(node){if(typeof node!=="object"||typeof node.path!=="string"){return}if(isFolder(node.path)){if(node.local&&node.local.itemsMap){return node.local}if(node.common&&node.common.itemsMap){return node.common}}else{if(node.local&&node.local.body&&node.local.contentType){return node.local}if(node.common&&node.common.body&&node.common.contentType){return node.common}if(node.body&&node.contentType){return{body:node.body,contentType:node.contentType}}}}function isOutdated(node,maxAge){return!node||!node.timestamp||(new Date).getTime()-node.timestamp>maxAge}function pathsFromRoot(path){var paths=[path];var parts=path.replace(/\/$/,"").split("/");while(parts.length>1){parts.pop();paths.push(parts.join("/")+"/")}return paths}function makeNode(path,timestamp){var node={path:path,common:{timestamp:timestamp}};if(isFolder(path)){node.common.itemsMap={}}return node}function updateFolderNodeWithItemName(node,itemName,timestamp){if(!node.common){node.common={timestamp:timestamp,itemsMap:{}}}if(!node.common.itemsMap){node.common.itemsMap={}}if(!node.local){node.local=deepClone(node.common)}if(!node.local.itemsMap){node.local.itemsMap=node.common.itemsMap}node.local.itemsMap[itemName]=true;return node}var methods={get:function(path,maxAge){var promise=promising();this.getNodes([path]).then(function(objs){var node=getLatest(objs[path]);if(typeof maxAge==="number"&&isOutdated(node,maxAge)){remoteStorage.sync.queueGetRequest(path,promise)}if(node){promise.fulfill(200,node.body||node.itemsMap,node.contentType)}else{promise.fulfill(404)}}.bind(this),function(err){promise.reject(err)}.bind(this));return promise},put:function(path,body,contentType){var paths=pathsFromRoot(path);var now=(new Date).getTime();return this._updateNodes(paths,function(nodes){try{for(var i=0;i0){break}}}return nodes})},flush:function(path){return this._getAllDescendentPaths(path).then(function(paths){return this.getNodes(paths)}.bind(this)).then(function(nodes){for(var path in nodes){var node=nodes[path];if(node&&node.common&&node.local){this._emit("change",{path:node.path,origin:"local",oldValue:node.local.body===false?undefined:node.local.body,newValue:node.common.body===false?undefined:node.common.body})}nodes[path]=undefined}return this.setNodes(nodes)}.bind(this))},fireInitial:function(){this.forAllNodes(function(node){var latest;if(isDocument(node.path)){latest=getLatest(node);if(latest){this._emit("change",{path:node.path,origin:"local",oldValue:undefined,oldContentType:undefined,newValue:latest.body,newContentType:latest.contentType})}}}.bind(this))},onDiff:function(diffHandler){this.diffHandler=diffHandler},migrate:function(node){if(typeof node==="object"&&!node.common){node.common={};if(typeof node.path==="string"){if(node.path.substr(-1)==="/"&&typeof node.body==="object"){node.common.itemsMap=node.body}}else{if(!node.local){node.local={}}node.local.body=node.body;node.local.contentType=node.contentType}}return node},_updateNodes:function(paths,cb){var self=this;return this.getNodes(paths).then(function(nodes){var existingNodes=deepClone(nodes);var changeEvents=[];var node;nodes=cb(nodes);for(var path in nodes){node=nodes[path];if(equal(node,existingNodes[path])){delete nodes[path]}else if(isDocument(path)){changeEvents.push({path:path,origin:"window",oldValue:node.local.previousBody,newValue:node.local.body===false?undefined:node.local.body,oldContentType:node.local.previousContentType,newContentType:node.local.contentType});delete node.local.previousBody;delete node.local.previousContentType}}return self.setNodes(nodes).then(function(){self._emitChangeEvents(changeEvents);return 200})},function(err){throw err})},_emitChangeEvents:function(events){for(var i=0;i64&&nChr<91?nChr-65:nChr>96&&nChr<123?nChr-71:nChr>47&&nChr<58?nChr+4:nChr===43?62:nChr===47?63:0}function base64DecToArr(sBase64,nBlocksSize){var sB64Enc=sBase64.replace(/[^A-Za-z0-9\+\/]/g,""),nInLen=sB64Enc.length,nOutLen=nBlocksSize?Math.ceil((nInLen*3+1>>2)/nBlocksSize)*nBlocksSize:nInLen*3+1>>2,taBytes=new Uint8Array(nOutLen);for(var nMod3,nMod4,nUint24=0,nOutIdx=0,nInIdx=0;nInIdx>>(16>>>nMod3&24)&255}nUint24=0}}return taBytes}function isBinary(node){return node.match(/charset=binary/)}function isRemoteStorageKey(key){return key.substr(0,NODES_PREFIX.length)===NODES_PREFIX||key.substr(0,CHANGES_PREFIX.length)===CHANGES_PREFIX}function isNodeKey(key){return key.substr(0,NODES_PREFIX.length)===NODES_PREFIX}RemoteStorage.LocalStorage.prototype={getNodes:function(paths){var promise=promising();var nodes={};for(i=0;i=now-this.maxAge?item.v:undefined},set:function(key,value){this._items[key]={v:value,t:(new Date).getTime()}}};RS.GoogleDrive=function(remoteStorage,clientId){RS.eventHandling(this,"connected");this.rs=remoteStorage;this.clientId=clientId;this._fileIdCache=new Cache(60*5);setTimeout(function(){this.configure(undefined,undefined,undefined,localStorage["remotestorage:googledrive:token"])}.bind(this),0)};RS.GoogleDrive.prototype={configure:function(_x,_y,_z,token){if(token){localStorage["remotestorage:googledrive:token"]=token;this.token=token;this.connected=true;this._emit("connected")}else{this.connected=false;delete this.token;delete localStorage["remotestorage:googledrive:token"]}},connect:function(){this.rs.setBackend("googledrive");RS.Authorize(AUTH_URL,AUTH_SCOPE,String(RS.Authorize.getLocation()),this.clientId)},get:function(path,options){if(path.substr(-1)==="/"){return this._getFolder(path,options)}else{return this._getFile(path,options)}},put:function(path,body,contentType,options){var promise=promising();function putDone(error,response){if(error){promise.reject(error)}else if(response.status>=200&&response.status<300){var meta=JSON.parse(response.responseText);promise.fulfill(200,undefined,meta.mimeType,meta.etag)}else{promise.reject("PUT failed with status "+response.status+" ("+response.responseText+")")}}this._getFileId(path,function(idError,id){if(idError){promise.reject(idError);return}else if(id){this._updateFile(id,path,body,contentType,options,putDone)}else{this._createFile(path,body,contentType,options,putDone)}});return promise},"delete":function(path,options){var promise=promising();this._getFileId(path,function(idError,id){if(idError){promise.reject(idError)}else if(id){this._request("DELETE",BASE_URL+"/drive/v2/files/"+id,{},function(deleteError,response){if(deleteError){promise.reject(deleteError)}else if(response.status===200||response.status===204){promise.fulfill(200)}else{promise.reject("Delete failed: "+response.status+" ("+response.responseText+")")}})}else{promise.fulfill(200)}});return promise},_updateFile:function(id,path,body,contentType,options,callback){callback=callback.bind(this);var metadata={mimeType:contentType};this._request("PUT",BASE_URL+"/upload/drive/v2/files/"+id+"?uploadType=resumable",{body:JSON.stringify(metadata),headers:{"Content-Type":"application/json; charset=UTF-8"}},function(metadataError,response){if(metadataError){callback(metadataError)}else{this._request("PUT",response.getResponseHeader("Location"),{body:contentType.match(/^application\/json/)?JSON.stringify(body):body},callback)}})},_createFile:function(path,body,contentType,options,callback){callback=callback.bind(this);this._getParentId(path,function(parentIdError,parentId){if(parentIdError){callback(parentIdError);return}var fileName=baseName(path);var metadata={title:metaTitleFromFileName(fileName),mimeType:contentType,parents:[{kind:"drive#fileLink",id:parentId}]};this._request("POST",BASE_URL+"/upload/drive/v2/files?uploadType=resumable",{body:JSON.stringify(metadata),headers:{"Content-Type":"application/json; charset=UTF-8"}},function(metadataError,response){if(metadataError){callback(metadataError)}else{this._request("POST",response.getResponseHeader("Location"),{body:contentType.match(/^application\/json/)?JSON.stringify(body):body},callback)}})})},_getFile:function(path,options){var promise=promising();this._getFileId(path,function(idError,id){if(idError){promise.reject(idError)}else{this._getMeta(id,function(metaError,meta){var etagWithoutQuotes;if(typeof meta==="object"&&typeof meta.etag==="string"){etagWithoutQuotes=meta.etag.substring(1,meta.etag.length-1)}if(metaError){promise.reject(metaError)}else if(meta.downloadUrl){var options={};if(meta.mimeType.match(/charset=binary/)){options.responseType="blob"}this._request("GET",meta.downloadUrl,options,function(downloadError,response){if(downloadError){promise.reject(downloadError)}else{var body=response.response;if(meta.mimeType.match(/^application\/json/)){try{body=JSON.parse(body)}catch(e){}}promise.fulfill(200,body,meta.mimeType,etagWithoutQuotes)}})}else{promise.fulfill(200,"",meta.mimeType,etagWithoutQuotes)}})}});return promise},_getFolder:function(path,options){var promise=promising();this._getFileId(path,function(idError,id){var query,fields,data,i,etagWithoutQuotes,itemsMap;if(idError){promise.reject(idError)}else if(!id){promise.fulfill(404)}else{query="'"+id+"' in parents";fields="items(downloadUrl,etag,fileSize,id,mimeType,title)";this._request("GET",BASE_URL+"/drive/v2/files?"+"q="+encodeURIComponent(query)+"&fields="+encodeURIComponent(fields)+"&maxResults=1000",{},function(childrenError,response){if(childrenError){promise.reject(childrenError)}else{if(response.status===200){try{data=JSON.parse(response.responseText)}catch(e){promise.reject("non-JSON response from GoogleDrive");return}itemsMap={};for(i=0;i=0||true){try{body=JSON.parse(body);mime="application/json; charset=UTF-8"}catch(e){RS.log("Failed parsing Json, assume it is something else then",mime,path)}}promise.fulfill(status,body,mime,rev)}}else{promise.fulfill(status)}}});return promise},put:function(path,body,contentType,options){RemoteStorage.log("dropbox.put",arguments);if(!this.connected){throw new Error("not connected (path: "+path+")")}path=cleanPath(path);var promise=this._sharePromise(path);var revCache=this._revCache;var savedRev=revCache.get(path);if(options&&options.ifMatch&&savedRev&&savedRev!==options.ifMatch){promise.fulfill(412);return promise}if(!contentType.match(/charset=/)){contentType+="; charset="+(body instanceof ArrayBuffer||RS.WireClient.isArrayBufferView(body)?"binary":"utf-8") +}var url="https://api-content.dropbox.com/1/files_put/auto"+path+"?";if(options&&options.ifMatch){url+="parent_rev="+encodeURIComponent(options.ifMatch)}if(body.length>150*1024*1024){RemoteStorage.log("files larger than 150MB not supported yet")}else{this._request("PUT",url,{body:body,headers:{"Content-Type":contentType}},function(err,resp){if(err){promise.reject(err)}else{var response=JSON.parse(resp.responseText);if(response.path!==path){promise.fulfill(412);this.rs.log("Dropbox created conflicting File ",response.path)}else{revCache.set(path,response.rev);promise.fulfill(resp.status)}}})}return promise},"delete":function(path,options){RemoteStorage.log("dropbox.delete ",arguments);if(!this.connected){throw new Error("not connected (path: "+path+")")}path=cleanPath(path);var promise=promising();var revCache=this._revCache;var savedRev=revCache.get(path);if(options.ifMatch&&savedRev&&options.ifMatch!==savedRev){promise.fulfill(412);return promise}var url="https://api.dropbox.com/1/fileops/delete?root=auto&path="+encodeURIComponent(path);this._request("POST",url,{},function(err,resp){if(err){promise.reject(error)}else{promise.fulfill(resp.status);revCache.delete(path)}});return promise.then(function(){var args=Array.prototype.slice.call(arguments);delete this._itemRefs[path];var p=promising();return p.fulfill.apply(p,args)}.bind(this))},_sharePromise:function(path){var promise=promising();var self=this;if(path.match(/^\/public\/.*[^\/]$/)&&typeof this._itemRefs[path]==="undefined"){RemoteStorage.log("shareing this one ",path);promise.then(function(){var args=Array.prototype.slice.call(arguments);var p=promising();RemoteStorage.log("calling share now");self.share(path).then(function(){RemoteStorage.log("shareing fullfilled promise",arguments);p.fulfill.apply(p,args)},function(err){RemoteStorage.log("shareing failed",err);p.fulfill.apply(p,args)});return p})}return promise},share:function(path){var url="https://api.dropbox.com/1/media/auto"+path;var promise=promising();var itemRefs=this._itemRefs;this._request("POST",url,{},function(err,resp){if(err){RemoteStorage.log(err);err.message='Shareing Dropbox Thingie("'+path+'") failed'+err.message;promise.reject(err)}else{try{var response=JSON.parse(resp.responseText);var url=response.url;itemRefs[path]=url;RemoteStorage.log("SHAREING URL :::: ",url," for ",path);if(hasLocalStorage){localStorage[SETTINGS_KEY+":shares"]=JSON.stringify(this._itemRefs)}promise.fulfill(url)}catch(err){err.message+="share error";promise.reject(err)}}});return promise},info:function(){var url="https://api.dropbox.com/1/account/info";var promise=promising();this._request("GET",url,{},function(err,resp){if(err){promise.reject(err)}else{try{var info=JSON.parse(resp.responseText);promise.fulfill(info)}catch(e){promise.reject(err)}}});return promise},_request:function(method,url,options,callback){callback=callback.bind(this);if(!options.headers){options.headers={}}options.headers["Authorization"]="Bearer "+this.token;RS.WireClient.request.call(this,method,url,options,function(err,xhr){if(xhr&&xhr.status===503){global.setTimeout(this._request(method,url,options,callback),3210)}else{callback(err,xhr)}})},fetchDelta:function(){var args=Array.prototype.slice.call(arguments);var promise=promising();var self=this;this._request("POST","https://api.dropbox.com/1/delta",{body:this._deltaCursor?"cursor="+encodeURIComponent(this._deltaCursor):"",headers:{"Content-Type":"application/x-www-form-urlencoded"}},function(error,response){if(error){this.rs.log("fetchDeltas",error);this.rs._emit("error",new RemoteStorage.SyncError("fetchDeltas failed"+error));promise.reject(error)}else{if(response.status!==200){if(response.status===400){this.rs._emit("error",new RemoteStorage.Unauthorized);promise.fulfill.apply(promise,args)}else{RemoteStorage.log("!!!!dropbox.fetchDelta returned "+response.status+response.responseText);promise.reject("dropbox.fetchDelta returned "+response.status+response.responseText)}return promise}var delta;try{delta=JSON.parse(response.responseText)}catch(error){RS.log("fetchDeltas can not parse response",error);return promise.reject("can not parse response of fetchDelta : "+error.message)}if(!delta.entries){RemoteStorage.log("!!!!!DropBox.fetchDeltas() NO ENTRIES FOUND!!",delta);return promise.reject("dropbox.fetchDeltas failed, no entries found")}if(delta.reset){this._revCache=new LowerCaseCache("rev");promise.then(function(){var args=Array.prototype.slice.call(arguments);self._revCache._activatePropagation();var p=promising();return p.fulfill.apply(p,args)})}if(delta.cursor){this._deltaCursor=delta.cursor}RemoteStorage.log("Delta : ",delta.entries);delta.entries.forEach(function(entry){var path=entry[0];var rev;if(!entry[1]){rev=null}else{if(entry[1].is_dir){return}rev=entry[1].rev}self._revCache.set(path,rev)});promise.fulfill.apply(promise,args)}});return promise}};function hookSync(rs){if(rs._dropboxOrigSync){return}rs._dropboxOrigSync=rs.sync.bind(rs);rs.sync=function(){return this.dropbox.fetchDelta.apply(this.dropbox,arguments).then(rs._dropboxOrigSync,function(err){rs._emit("error",new rs.SyncError(err))})}}function unHookSync(rs){if(!rs._dropboxOrigSync){return}rs.sync=rs._dropboxOrigSync;delete rs._dropboxOrigSync}function hookGetItemURL(rs){if(rs._origBaseClientGetItemURL){return}rs._origBaseClientGetItemURL=RS.BaseClient.prototype.getItemURL;RS.BaseClient.prototype.getItemURL=function(path){var ret=rs.dropbox._itemRefs[path];return ret?ret:""}}function unHookGetItemURL(rs){if(!rs._origBaseClieNtGetItemURL){return}RS.BaseClient.prototype.getItemURL=rs._origBaseClietGetItemURL;delete rs._origBaseClietGetItemURL}function hookRemote(rs){if(rs._origRemote){return}rs._origRemote=rs.remote;rs.remote=rs.dropbox}function unHookRemote(rs){if(rs._origRemote){rs.remote=rs._origRemote;delete rs._origRemote}}function hookIt(rs){hookRemote(rs);if(rs.sync){hookSync(rs)}hookGetItemURL(rs)}function unHookIt(rs){unHookRemote(rs);unHookSync(rs);unHookGetItemURL(rs)}RS.Dropbox._rs_init=function(rs){hasLocalStorage=rs.localStorageAvailable();if(rs.apiKeys.dropbox){rs.dropbox=new RS.Dropbox(rs)}if(rs.backend==="dropbox"){hookIt(rs)}};RS.Dropbox._rs_supported=function(){return true};RS.Dropbox._rs_cleanup=function(rs){unHookIt(rs);if(hasLocalStorage){delete localStorage[SETTINGS_KEY]}rs.removeEventListener("error",onErrorCb);rs.setBackend(undefined)}})(this);remoteStorage=new RemoteStorage; \ No newline at end of file diff --git a/src/version.js b/src/version.js index 3207ef8c1..159a9fd0e 100644 --- a/src/version.js +++ b/src/version.js @@ -3,7 +3,7 @@ RemoteStorage.version = RemoteStorage.prototype.version = { product: 0, major: 10, minor: 0, - postfix: 'beta' + postfix: 'beta2' }; RemoteStorage.version.toString = function() {