| /*jshint strict: false */ |
| /*global chrome */ |
| var merge = require('./merge'); |
| exports.extend = require('pouchdb-extend'); |
| exports.ajax = require('./deps/ajax'); |
| exports.createBlob = require('./deps/blob'); |
| exports.uuid = require('./deps/uuid'); |
| exports.getArguments = require('argsarray'); |
| var buffer = require('./deps/buffer'); |
| var errors = require('./deps/errors'); |
| var EventEmitter = require('events').EventEmitter; |
| var collections = require('./deps/collections'); |
| exports.Map = collections.Map; |
| exports.Set = collections.Set; |
| |
| if (typeof global.Promise === 'function') { |
| exports.Promise = global.Promise; |
| } else { |
| exports.Promise = require('bluebird'); |
| } |
| var Promise = exports.Promise; |
| |
| function toObject(array) { |
| var obj = {}; |
| array.forEach(function (item) { obj[item] = true; }); |
| return obj; |
| } |
| // List of top level reserved words for doc |
| var reservedWords = toObject([ |
| '_id', |
| '_rev', |
| '_attachments', |
| '_deleted', |
| '_revisions', |
| '_revs_info', |
| '_conflicts', |
| '_deleted_conflicts', |
| '_local_seq', |
| '_rev_tree', |
| //replication documents |
| '_replication_id', |
| '_replication_state', |
| '_replication_state_time', |
| '_replication_state_reason', |
| '_replication_stats' |
| ]); |
| |
| // List of reserved words that should end up the document |
| var dataWords = toObject([ |
| '_attachments', |
| //replication documents |
| '_replication_id', |
| '_replication_state', |
| '_replication_state_time', |
| '_replication_state_reason', |
| '_replication_stats' |
| ]); |
| |
| exports.clone = function (obj) { |
| return exports.extend(true, {}, obj); |
| }; |
| exports.inherits = require('inherits'); |
| // Determine id an ID is valid |
| // - invalid IDs begin with an underescore that does not begin '_design' or |
| // '_local' |
| // - any other string value is a valid id |
| // Returns the specific error object for each case |
| exports.invalidIdError = function (id) { |
| var err; |
| if (!id) { |
| err = new TypeError(errors.MISSING_ID.message); |
| err.status = 412; |
| } else if (typeof id !== 'string') { |
| err = new TypeError(errors.INVALID_ID.message); |
| err.status = 400; |
| } else if (/^_/.test(id) && !(/^_(design|local)/).test(id)) { |
| err = new TypeError(errors.RESERVED_ID.message); |
| err.status = 400; |
| } |
| if (err) { |
| throw err; |
| } |
| }; |
| |
| function isChromeApp() { |
| return (typeof chrome !== "undefined" && |
| typeof chrome.storage !== "undefined" && |
| typeof chrome.storage.local !== "undefined"); |
| } |
| |
| // Pretty dumb name for a function, just wraps callback calls so we dont |
| // to if (callback) callback() everywhere |
| exports.call = exports.getArguments(function (args) { |
| if (!args.length) { |
| return; |
| } |
| var fun = args.shift(); |
| if (typeof fun === 'function') { |
| fun.apply(this, args); |
| } |
| }); |
| |
| exports.isLocalId = function (id) { |
| return (/^_local/).test(id); |
| }; |
| |
| // check if a specific revision of a doc has been deleted |
| // - metadata: the metadata object from the doc store |
| // - rev: (optional) the revision to check. defaults to winning revision |
| exports.isDeleted = function (metadata, rev) { |
| if (!rev) { |
| rev = merge.winningRev(metadata); |
| } |
| var dashIndex = rev.indexOf('-'); |
| if (dashIndex !== -1) { |
| rev = rev.substring(dashIndex + 1); |
| } |
| var deleted = false; |
| merge.traverseRevTree(metadata.rev_tree, |
| function (isLeaf, pos, id, acc, opts) { |
| if (id === rev) { |
| deleted = !!opts.deleted; |
| } |
| }); |
| |
| return deleted; |
| }; |
| |
| exports.filterChange = function (opts) { |
| return function (change) { |
| var req = {}; |
| var hasFilter = opts.filter && typeof opts.filter === 'function'; |
| |
| req.query = opts.query_params; |
| if (opts.filter && hasFilter && !opts.filter.call(this, change.doc, req)) { |
| return false; |
| } |
| if (opts.doc_ids && opts.doc_ids.indexOf(change.id) === -1) { |
| return false; |
| } |
| if (!opts.include_docs) { |
| delete change.doc; |
| } else { |
| for (var att in change.doc._attachments) { |
| if (change.doc._attachments.hasOwnProperty(att)) { |
| change.doc._attachments[att].stub = true; |
| } |
| } |
| } |
| return true; |
| }; |
| }; |
| |
| // Preprocess documents, parse their revisions, assign an id and a |
| // revision for new writes that are missing them, etc |
| exports.parseDoc = function (doc, newEdits) { |
| var nRevNum; |
| var newRevId; |
| var revInfo; |
| var error; |
| var opts = {status: 'available'}; |
| if (doc._deleted) { |
| opts.deleted = true; |
| } |
| |
| if (newEdits) { |
| if (!doc._id) { |
| doc._id = exports.uuid(); |
| } |
| newRevId = exports.uuid(32, 16).toLowerCase(); |
| if (doc._rev) { |
| revInfo = /^(\d+)-(.+)$/.exec(doc._rev); |
| if (!revInfo) { |
| var err = new TypeError("invalid value for property '_rev'"); |
| err.status = 400; |
| } |
| doc._rev_tree = [{ |
| pos: parseInt(revInfo[1], 10), |
| ids: [revInfo[2], {status: 'missing'}, [[newRevId, opts, []]]] |
| }]; |
| nRevNum = parseInt(revInfo[1], 10) + 1; |
| } else { |
| doc._rev_tree = [{ |
| pos: 1, |
| ids : [newRevId, opts, []] |
| }]; |
| nRevNum = 1; |
| } |
| } else { |
| if (doc._revisions) { |
| doc._rev_tree = [{ |
| pos: doc._revisions.start - doc._revisions.ids.length + 1, |
| ids: doc._revisions.ids.reduce(function (acc, x) { |
| if (acc === null) { |
| return [x, opts, []]; |
| } else { |
| return [x, {status: 'missing'}, [acc]]; |
| } |
| }, null) |
| }]; |
| nRevNum = doc._revisions.start; |
| newRevId = doc._revisions.ids[0]; |
| } |
| if (!doc._rev_tree) { |
| revInfo = /^(\d+)-(.+)$/.exec(doc._rev); |
| if (!revInfo) { |
| error = new TypeError(errors.BAD_ARG.message); |
| error.status = errors.BAD_ARG.status; |
| throw error; |
| } |
| nRevNum = parseInt(revInfo[1], 10); |
| newRevId = revInfo[2]; |
| doc._rev_tree = [{ |
| pos: parseInt(revInfo[1], 10), |
| ids: [revInfo[2], opts, []] |
| }]; |
| } |
| } |
| |
| exports.invalidIdError(doc._id); |
| |
| doc._rev = [nRevNum, newRevId].join('-'); |
| |
| var result = {metadata : {}, data : {}}; |
| for (var key in doc) { |
| if (doc.hasOwnProperty(key)) { |
| var specialKey = key[0] === '_'; |
| if (specialKey && !reservedWords[key]) { |
| error = new Error(errors.DOC_VALIDATION.message + ': ' + key); |
| error.status = errors.DOC_VALIDATION.status; |
| throw error; |
| } else if (specialKey && !dataWords[key]) { |
| result.metadata[key.slice(1)] = doc[key]; |
| } else { |
| result.data[key] = doc[key]; |
| } |
| } |
| } |
| return result; |
| }; |
| |
| exports.isCordova = function () { |
| return (typeof cordova !== "undefined" || |
| typeof PhoneGap !== "undefined" || |
| typeof phonegap !== "undefined"); |
| }; |
| |
| exports.hasLocalStorage = function () { |
| if (isChromeApp()) { |
| return false; |
| } |
| try { |
| return global.localStorage; |
| } catch (e) { |
| return false; |
| } |
| }; |
| exports.Changes = Changes; |
| exports.inherits(Changes, EventEmitter); |
| function Changes() { |
| if (!(this instanceof Changes)) { |
| return new Changes(); |
| } |
| var self = this; |
| EventEmitter.call(this); |
| this.isChrome = isChromeApp(); |
| this.listeners = {}; |
| this.hasLocal = false; |
| if (!this.isChrome) { |
| this.hasLocal = exports.hasLocalStorage(); |
| } |
| if (this.isChrome) { |
| chrome.storage.onChanged.addListener(function (e) { |
| // make sure it's event addressed to us |
| if (e.db_name != null) { |
| //object only has oldValue, newValue members |
| self.emit(e.dbName.newValue); |
| } |
| }); |
| } else if (this.hasLocal) { |
| if (global.addEventListener) { |
| global.addEventListener("storage", function (e) { |
| self.emit(e.key); |
| }); |
| } else { |
| global.attachEvent("storage", function (e) { |
| self.emit(e.key); |
| }); |
| } |
| } |
| |
| } |
| Changes.prototype.addListener = function (dbName, id, db, opts) { |
| if (this.listeners[id]) { |
| return; |
| } |
| function eventFunction() { |
| db.changes({ |
| include_docs: opts.include_docs, |
| conflicts: opts.conflicts, |
| continuous: false, |
| descending: false, |
| filter: opts.filter, |
| view: opts.view, |
| since: opts.since, |
| query_params: opts.query_params, |
| onChange: function (c) { |
| if (c.seq > opts.since && !opts.cancelled) { |
| opts.since = c.seq; |
| exports.call(opts.onChange, c); |
| } |
| } |
| }); |
| } |
| this.listeners[id] = eventFunction; |
| this.on(dbName, eventFunction); |
| }; |
| |
| Changes.prototype.removeListener = function (dbName, id) { |
| if (!(id in this.listeners)) { |
| return; |
| } |
| EventEmitter.prototype.removeListener.call(this, dbName, |
| this.listeners[id]); |
| }; |
| |
| |
| Changes.prototype.notifyLocalWindows = function (dbName) { |
| //do a useless change on a storage thing |
| //in order to get other windows's listeners to activate |
| if (this.isChrome) { |
| chrome.storage.local.set({dbName: dbName}); |
| } else if (this.hasLocal) { |
| localStorage[dbName] = (localStorage[dbName] === "a") ? "b" : "a"; |
| } |
| }; |
| |
| Changes.prototype.notify = function (dbName) { |
| this.emit(dbName); |
| this.notifyLocalWindows(dbName); |
| }; |
| |
| if (!process.browser || !('atob' in global)) { |
| exports.atob = function (str) { |
| var base64 = new buffer(str, 'base64'); |
| // Node.js will just skip the characters it can't encode instead of |
| // throwing and exception |
| if (base64.toString('base64') !== str) { |
| throw ("Cannot base64 encode full string"); |
| } |
| return base64.toString('binary'); |
| }; |
| } else { |
| exports.atob = function (str) { |
| return atob(str); |
| }; |
| } |
| |
| if (!process.browser || !('btoa' in global)) { |
| exports.btoa = function (str) { |
| return new buffer(str, 'binary').toString('base64'); |
| }; |
| } else { |
| exports.btoa = function (str) { |
| return btoa(str); |
| }; |
| } |
| |
| // From http://stackoverflow.com/questions/14967647/ (continues on next line) |
| // encode-decode-image-with-base64-breaks-image (2013-04-21) |
| exports.fixBinary = function (bin) { |
| if (!process.browser) { |
| // don't need to do this in Node |
| return bin; |
| } |
| |
| var length = bin.length; |
| var buf = new ArrayBuffer(length); |
| var arr = new Uint8Array(buf); |
| for (var i = 0; i < length; i++) { |
| arr[i] = bin.charCodeAt(i); |
| } |
| return buf; |
| }; |
| |
| exports.once = function (fun) { |
| var called = false; |
| return exports.getArguments(function (args) { |
| if (called) { |
| throw new Error('once called more than once'); |
| } else { |
| called = true; |
| fun.apply(this, args); |
| } |
| }); |
| }; |
| |
| exports.toPromise = function (func) { |
| //create the function we will be returning |
| return exports.getArguments(function (args) { |
| var self = this; |
| var tempCB = |
| (typeof args[args.length - 1] === 'function') ? args.pop() : false; |
| // if the last argument is a function, assume its a callback |
| var usedCB; |
| if (tempCB) { |
| // if it was a callback, create a new callback which calls it, |
| // but do so async so we don't trap any errors |
| usedCB = function (err, resp) { |
| process.nextTick(function () { |
| tempCB(err, resp); |
| }); |
| }; |
| } |
| var promise = new Promise(function (fulfill, reject) { |
| var resp; |
| try { |
| var callback = exports.once(function (err, mesg) { |
| if (err) { |
| reject(err); |
| } else { |
| fulfill(mesg); |
| } |
| }); |
| // create a callback for this invocation |
| // apply the function in the orig context |
| args.push(callback); |
| resp = func.apply(self, args); |
| if (resp && typeof resp.then === 'function') { |
| fulfill(resp); |
| } |
| } catch (e) { |
| reject(e); |
| } |
| }); |
| // if there is a callback, call it back |
| if (usedCB) { |
| promise.then(function (result) { |
| usedCB(null, result); |
| }, usedCB); |
| } |
| promise.cancel = function () { |
| return this; |
| }; |
| return promise; |
| }); |
| }; |
| |
| exports.adapterFun = function (name, callback) { |
| return exports.toPromise(exports.getArguments(function (args) { |
| if (this._closed) { |
| return Promise.reject(new Error('database is closed')); |
| } |
| var self = this; |
| if (!this.taskqueue.isReady) { |
| return new exports.Promise(function (fulfill, reject) { |
| self.taskqueue.addTask(function (failed) { |
| if (failed) { |
| reject(failed); |
| } else { |
| fulfill(self[name].apply(self, args)); |
| } |
| }); |
| }); |
| } |
| return callback.apply(this, args); |
| })); |
| }; |
| //Can't find original post, but this is close |
| //http://stackoverflow.com/questions/6965107/ (continues on next line) |
| //converting-between-strings-and-arraybuffers |
| exports.arrayBufferToBinaryString = function (buffer) { |
| var binary = ""; |
| var bytes = new Uint8Array(buffer); |
| var length = bytes.byteLength; |
| for (var i = 0; i < length; i++) { |
| binary += String.fromCharCode(bytes[i]); |
| } |
| return binary; |
| }; |
| |
| exports.cancellableFun = function (fun, self, opts) { |
| |
| opts = opts ? exports.clone(true, {}, opts) : {}; |
| |
| var emitter = new EventEmitter(); |
| var oldComplete = opts.complete || function () { }; |
| var complete = opts.complete = exports.once(function (err, resp) { |
| if (err) { |
| oldComplete(err); |
| } else { |
| emitter.emit('end', resp); |
| oldComplete(null, resp); |
| } |
| emitter.removeAllListeners(); |
| }); |
| var oldOnChange = opts.onChange || function () {}; |
| var lastChange = 0; |
| self.on('destroyed', function () { |
| emitter.removeAllListeners(); |
| }); |
| opts.onChange = function (change) { |
| oldOnChange(change); |
| if (change.seq <= lastChange) { |
| return; |
| } |
| lastChange = change.seq; |
| emitter.emit('change', change); |
| if (change.deleted) { |
| emitter.emit('delete', change); |
| } else if (change.changes.length === 1 && |
| change.changes[0].rev.slice(0, 1) === '1-') { |
| emitter.emit('create', change); |
| } else { |
| emitter.emit('update', change); |
| } |
| }; |
| var promise = new Promise(function (fulfill, reject) { |
| opts.complete = function (err, res) { |
| if (err) { |
| reject(err); |
| } else { |
| fulfill(res); |
| } |
| }; |
| }); |
| |
| promise.then(function (result) { |
| complete(null, result); |
| }, complete); |
| |
| // this needs to be overwridden by caller, dont fire complete until |
| // the task is ready |
| promise.cancel = function () { |
| promise.isCancelled = true; |
| if (self.taskqueue.isReady) { |
| opts.complete(null, {status: 'cancelled'}); |
| } |
| }; |
| |
| if (!self.taskqueue.isReady) { |
| self.taskqueue.addTask(function () { |
| if (promise.isCancelled) { |
| opts.complete(null, {status: 'cancelled'}); |
| } else { |
| fun(self, opts, promise); |
| } |
| }); |
| } else { |
| fun(self, opts, promise); |
| } |
| promise.on = emitter.on.bind(emitter); |
| promise.once = emitter.once.bind(emitter); |
| promise.addListener = emitter.addListener.bind(emitter); |
| promise.removeListener = emitter.removeListener.bind(emitter); |
| promise.removeAllListeners = emitter.removeAllListeners.bind(emitter); |
| promise.setMaxListeners = emitter.setMaxListeners.bind(emitter); |
| promise.listeners = emitter.listeners.bind(emitter); |
| promise.emit = emitter.emit.bind(emitter); |
| return promise; |
| }; |
| |
| exports.MD5 = exports.toPromise(require('./deps/md5')); |
| |
| // designed to give info to browser users, who are disturbed |
| // when they see 404s in the console |
| exports.explain404 = function (str) { |
| if (process.browser && 'console' in global && 'info' in console) { |
| console.info('The above 404 is totally normal. ' + |
| str + '\n\u2665 the PouchDB team'); |
| } |
| }; |