| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 |
- // Copyright 2015 The MemDB Authors.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
- // implied. See the License for the specific language governing
- // permissions and limitations under the License. See the AUTHORS file
- // for names of contributors.
- 'use strict';
- var P = require('bluebird');
- var Logger = require('memdb-logger');
- var consts = require('./consts');
- var Collection = require('./collection');
- var util = require('util');
- var utils = require('./utils');
- var Connection = function(opts){
- opts = opts || {};
- this._id = opts._id;
- this.shard = opts.shard;
- this.config = opts.config || {};
- this.collections = {};
- this.lockedKeys = utils.forceHashMap();
- this.logger = Logger.getLogger('memdb', __filename, 'shard:' + this.shard._id);
- };
- var proto = Connection.prototype;
- proto.close = function(){
- if(this.isDirty()){
- this.rollback();
- }
- for(var name in this.collections){
- this.collections[name].close();
- }
- };
- consts.collMethods.forEach(function(method){
- proto[method] = function(name){
- var collection = this.getCollection(name);
- // remove 'name' arg
- var args = [].slice.call(arguments, 1);
- this.logger.debug('[conn:%s] %s.%s(%j)', this._id, name, method, args);
- return collection[method].apply(collection, args);
- };
- });
- proto.commit = function(){
- var self = this;
- return P.each(Object.keys(this.collections), function(name){
- var collection = self.collections[name];
- return collection.commitIndex();
- })
- .then(function(){
- return self.shard.commit(self._id, Object.keys(self.lockedKeys));
- })
- .then(function(){
- self.lockedKeys = {};
- self.logger.debug('[conn:%s] commited', self._id);
- return true;
- });
- };
- // sync method
- proto.rollback = function(){
- var self = this;
- Object.keys(this.collections).forEach(function(name){
- self.collections[name].rollbackIndex();
- });
- this.shard.rollback(this._id, Object.keys(this.lockedKeys));
- this.lockedKeys = {};
- this.logger.debug('[conn:%s] rolledback', this._id);
- return true;
- };
- proto.flushBackend = function(){
- return this.shard.flushBackend(this._id);
- };
- // for internal use
- proto.$unload = function(key){
- return this.shard.$unload(key);
- };
- // for internal use
- proto.$findReadOnly = function(key, fields){
- return this.shard.find(null, key, fields);
- };
- proto.getCollection = function(name, isIndex){
- if(!isIndex && name && name.indexOf('index.') === 0){
- throw new Error('Collection name can not begin with "index."');
- }
- var self = this;
- if(!this.collections[name]){
- var collection = new Collection({
- name : name,
- shard : this.shard,
- conn : this,
- config : this.config.collections[name] || {},
- });
- collection.on('lock', function(id){
- var key = name + '$' + id;
- self.lockedKeys[key] = true;
- });
- this.collections[name] = collection;
- }
- return this.collections[name];
- };
- proto.isDirty = function(){
- return Object.keys(this.lockedKeys).length > 0;
- };
- module.exports = Connection;
|