| Line | Hits | Source |
|---|---|---|
| 1 | /* vim: set expandtab tabstop=2 shiftwidth=2 foldmethod=marker: */ | |
| 2 | ||
| 3 | 1 | "use strict"; |
| 4 | ||
| 5 | 1 | var util = require('util'); |
| 6 | 1 | var Emitter = require('events').EventEmitter; |
| 7 | 1 | var Pool = require(__dirname + '/pool.js'); |
| 8 | 1 | var Queue = require('safequeue'); |
| 9 | ||
| 10 | 1 | var MAX_QUEUED_SIZE = 1000; |
| 11 | ||
| 12 | 1 | var READONLY = 1; |
| 13 | 1 | var WRITABLE = 2; |
| 14 | ||
| 15 | 1 | var _QueueTimeoutError = function (msg) { |
| 16 | 6 | var e = new Error(msg); |
| 17 | 6 | e.name = 'QueueTimeout'; |
| 18 | 6 | return e; |
| 19 | }; | |
| 20 | ||
| 21 | 1 | var _QueueIsFullError = function (msg) { |
| 22 | 2 | var e = new Error(msg); |
| 23 | 2 | e.name = 'QueueIsFull'; |
| 24 | 2 | return e; |
| 25 | }; | |
| 26 | ||
| 27 | /* {{{ private function _readonly() */ | |
| 28 | 1 | var _readonly = function (sql) { |
| 29 | 20 | return sql.match(/^(SELECT|SHOW|DESC|DESCRIBE|KILL)\s+/i) ? true : false; |
| 30 | }; | |
| 31 | /* }}} */ | |
| 32 | ||
| 33 | /* {{{ private function _remove() */ | |
| 34 | 1 | var _remove = function (a, o) { |
| 35 | 10 | var i = a.indexOf(o); |
| 36 | 10 | if (i > -1) { |
| 37 | 0 | a.splice(i, 1); |
| 38 | } | |
| 39 | ||
| 40 | 10 | return a; |
| 41 | }; | |
| 42 | /* }}} */ | |
| 43 | ||
| 44 | 1 | exports.create = function (options) { |
| 45 | ||
| 46 | /** | |
| 47 | * @ 心跳SQL | |
| 48 | */ | |
| 49 | 4 | var hbquery = 'SHOW VARIABLES LIKE "READ_ONLY"'; |
| 50 | ||
| 51 | /* {{{ private function hbparse() */ | |
| 52 | /** | |
| 53 | * parse heartbeat result | |
| 54 | * | |
| 55 | * @return Integer 0 : offline; 1 : readonly; 3 : writable | |
| 56 | */ | |
| 57 | 4 | var hbparse = function (res) { |
| 58 | 2 | if (!res || !res.length) { |
| 59 | 0 | return 0; |
| 60 | } | |
| 61 | ||
| 62 | 2 | var s = READONLY; |
| 63 | 2 | if (((res.shift() || {}).Value + '').match(/^(off)$/i)) { |
| 64 | 2 | s |= WRITABLE; |
| 65 | } | |
| 66 | ||
| 67 | 2 | return s; |
| 68 | }; | |
| 69 | /* }}} */ | |
| 70 | ||
| 71 | 4 | var Cluster = function () { |
| 72 | 4 | Emitter.call(this); |
| 73 | }; | |
| 74 | 4 | util.inherits(Cluster, Emitter); |
| 75 | ||
| 76 | /** | |
| 77 | * @ 连接池列表 | |
| 78 | */ | |
| 79 | 4 | var backups = {}; |
| 80 | ||
| 81 | /** | |
| 82 | * @ 读写列表 | |
| 83 | */ | |
| 84 | 4 | var rwlists = []; |
| 85 | ||
| 86 | /** | |
| 87 | * @ 读写队列 | |
| 88 | */ | |
| 89 | 4 | var rwqueue = Queue.create({'timeout' : 0, 'maxitem' : MAX_QUEUED_SIZE}); |
| 90 | 4 | rwqueue.on('timeout', function (item, timeout) { |
| 91 | 5 | (item[2])(_QueueTimeoutError('Query stays in the queue more than ' + timeout + ' ms')); |
| 92 | }); | |
| 93 | ||
| 94 | /** | |
| 95 | * @ 只读列表 | |
| 96 | */ | |
| 97 | 4 | var rolists = []; |
| 98 | ||
| 99 | /** | |
| 100 | * @ 只读队列 | |
| 101 | */ | |
| 102 | 4 | var roqueue = Queue.create({'timeout' : 0, 'maxitem' : MAX_QUEUED_SIZE}); |
| 103 | 4 | roqueue.on('timeout', function (item, timeout) { |
| 104 | 1 | (item[2])(_QueueTimeoutError('Query stays in the queue more than ' + timeout + ' ms')); |
| 105 | }); | |
| 106 | ||
| 107 | /* {{{ public prototype setHeartBeatQuery() */ | |
| 108 | 4 | Cluster.prototype.setHeartBeatQuery = function (sql, parser) { |
| 109 | 2 | Object.keys(backups).forEach(function (i) { |
| 110 | 1 | backups[i].setHeartBeatQuery(sql); |
| 111 | }); | |
| 112 | 2 | hbquery = sql; |
| 113 | 2 | if ('function' === (typeof parser)) { |
| 114 | 2 | hbparse = parser; |
| 115 | } | |
| 116 | }; | |
| 117 | /* }}} */ | |
| 118 | ||
| 119 | /* {{{ private function checkQueue() */ | |
| 120 | 4 | var checkQueue = function (queue, pool, max) { |
| 121 | 6 | var max = (~~max) || 4; |
| 122 | 6 | while (max > 0) { |
| 123 | 14 | var s = queue.shift(); |
| 124 | 14 | if (!s || !s.length) { |
| 125 | 5 | return; |
| 126 | } | |
| 127 | 9 | pool.query(s[0], s[1], s[2]); |
| 128 | 9 | max--; |
| 129 | } | |
| 130 | ||
| 131 | 1 | if (queue.size() > 0) { |
| 132 | 1 | process.nextTick(function () { |
| 133 | 1 | checkQueue(queue, pool, max); |
| 134 | }); | |
| 135 | } | |
| 136 | }; | |
| 137 | /* }}} */ | |
| 138 | ||
| 139 | /* {{{ public prototype addserver() */ | |
| 140 | 4 | Cluster.prototype.addserver = function (config) { |
| 141 | 5 | var p = Pool.create(options, config); |
| 142 | 5 | var i = p._name(); |
| 143 | 5 | backups[i] = p; |
| 144 | ||
| 145 | 5 | var _self = this; |
| 146 | 5 | p.on('error', function (e) { |
| 147 | 4 | _self.emit('error', e); |
| 148 | }); | |
| 149 | 5 | p.on('busy', function (n, c) { |
| 150 | 9 | _self.emit('busy', n, c, i); |
| 151 | }); | |
| 152 | ||
| 153 | 5 | p.setHeartBeatQuery(hbquery); |
| 154 | 5 | p.on('state', function (res) { |
| 155 | 8 | var s = hbparse(res); |
| 156 | 8 | if (!s) { |
| 157 | 5 | rolists = _remove(rolists, i); |
| 158 | 5 | rwlists = _remove(rwlists, i); |
| 159 | 5 | return; |
| 160 | } | |
| 161 | ||
| 162 | 3 | if (rolists.indexOf(i) < 0) { |
| 163 | 3 | checkQueue(roqueue, p); |
| 164 | 3 | rolists.push(i); |
| 165 | } | |
| 166 | 3 | if ((WRITABLE & s) && rwlists.indexOf(i) < 0) { |
| 167 | 2 | checkQueue(rwqueue, p); |
| 168 | 2 | rwlists.push(i); |
| 169 | } | |
| 170 | }); | |
| 171 | }; | |
| 172 | /* }}} */ | |
| 173 | ||
| 174 | /** | |
| 175 | * @ 读计数器 | |
| 176 | */ | |
| 177 | 4 | var rocount = 0; |
| 178 | ||
| 179 | /** | |
| 180 | * @ 写计数器 | |
| 181 | */ | |
| 182 | 4 | var rwcount = 0; |
| 183 | ||
| 184 | /* {{{ public function query() */ | |
| 185 | 4 | Cluster.prototype.query = function (sql, tmout, callback) { |
| 186 | ||
| 187 | 20 | if ('function' === (typeof tmout)) { |
| 188 | 4 | callback = tmout; |
| 189 | 4 | tmout = 0; |
| 190 | 20 | }; |
| 191 | ||
| 192 | 20 | if (!_readonly(sql.sql || sql)) { |
| 193 | 6 | if (rwlists.length < 1) { |
| 194 | 6 | if (rwqueue.push([sql, tmout, callback], tmout) < 0) { |
| 195 | 1 | callback(_QueueIsFullError('Too many queries queued')); |
| 196 | } | |
| 197 | } else { | |
| 198 | 0 | backups[rwlists[(++rwcount) % rwlists.length]].query(sql, tmout, callback); |
| 199 | } | |
| 200 | } else { | |
| 201 | 14 | if (rolists.length < 1) { |
| 202 | 11 | if (roqueue.push([sql, tmout, callback], tmout) < 0) { |
| 203 | 1 | callback(_QueueIsFullError('Too many queries queued')); |
| 204 | } | |
| 205 | } else { | |
| 206 | 3 | backups[rolists[(++rocount) % rolists.length]].query(sql, tmout, callback); |
| 207 | } | |
| 208 | } | |
| 209 | }; | |
| 210 | /* }}} */ | |
| 211 | ||
| 212 | 4 | return new Cluster(); |
| 213 | }; | |
| 214 |
| Line | Hits | Source |
|---|---|---|
| 1 | /* vim: set expandtab tabstop=2 shiftwidth=2 foldmethod=marker: */ | |
| 2 | ||
| 3 | 1 | "use strict"; |
| 4 | ||
| 5 | 1 | var util = require('util'); |
| 6 | 1 | var mysql = require('mysql-robin'); |
| 7 | 1 | var EventEmitter = require('events').EventEmitter; |
| 8 | 1 | var sqlString = require('mysql-robin/lib/protocol/SqlString'); |
| 9 | ||
| 10 | /** | |
| 11 | * @ Connection | |
| 12 | */ | |
| 13 | 1 | var Connection = function (options) { |
| 14 | ||
| 15 | 15 | EventEmitter.call(this); |
| 16 | ||
| 17 | /** | |
| 18 | * 0 : 未连接 | |
| 19 | * 1 : 正在连接 | |
| 20 | * 2 : 连接成功 | |
| 21 | * -1: 准备断开 | |
| 22 | */ | |
| 23 | 15 | this._flag = 0; |
| 24 | 15 | options.port = options.port || 3306; |
| 25 | 15 | this._name = util.format('%s@%s:%d', options.user, options.host, options.port); |
| 26 | 15 | this._conn = mysql.createConnection(options); |
| 27 | ||
| 28 | 15 | var _self = this; |
| 29 | 15 | this._conn.on('error', function (e) { |
| 30 | 1 | if (e && e.fatal && _self._flag > -1) { |
| 31 | 1 | _self.close(); |
| 32 | } | |
| 33 | 1 | _self.emit('error', _self._error(e)); |
| 34 | }); | |
| 35 | }; | |
| 36 | 1 | util.inherits(Connection, EventEmitter); |
| 37 | ||
| 38 | 1 | Connection.prototype._error = function (name, msg) { |
| 39 | 8 | var e; |
| 40 | 8 | if (name instanceof Error) { |
| 41 | 6 | e = name; |
| 42 | 6 | e.name = (e.name && 'Error' !== e.name) ? e.name : 'MysqlError'; |
| 43 | } else { | |
| 44 | 2 | e = new Error(msg || name); |
| 45 | 2 | e.name = name; |
| 46 | } | |
| 47 | 8 | e.message = util.format('%s (%s)', e.message, this._name); |
| 48 | 8 | return e; |
| 49 | }; | |
| 50 | ||
| 51 | 1 | Connection.prototype.close = function () { |
| 52 | ||
| 53 | 7 | if (this._flag < 0) { |
| 54 | 1 | return; |
| 55 | } | |
| 56 | ||
| 57 | 6 | this._flag = -1; |
| 58 | 6 | this._conn.end(); |
| 59 | }; | |
| 60 | ||
| 61 | 1 | Connection.prototype.query = function (sql, timeout, callback) { |
| 62 | ||
| 63 | 26 | if ((typeof sql) === 'object' && sql.params) { |
| 64 | 0 | sql = this.format(sql.sql, sql.params); |
| 65 | } | |
| 66 | ||
| 67 | 26 | var _self = this; |
| 68 | 26 | if (!timeout || timeout < 1) { |
| 69 | 6 | return this._conn.query(sql, function (e, r) { |
| 70 | 6 | callback(e ? _self._error(e) : null, r); |
| 71 | }); | |
| 72 | } | |
| 73 | ||
| 74 | 20 | var timer = setTimeout(function () { |
| 75 | 2 | callback(_self._error('QueryTimeout', 'Mysql query timeout after ' + timeout + ' ms')); |
| 76 | 2 | callback = function (e, r) { |
| 77 | 2 | _self.emit('late', e, r, sql); |
| 78 | }; | |
| 79 | }, timeout); | |
| 80 | 20 | _self._conn.query(sql, function (e, r) { |
| 81 | 20 | clearTimeout(timer); |
| 82 | 20 | timer = null; |
| 83 | 20 | callback(e ? _self._error(e) : null, r); |
| 84 | }); | |
| 85 | }; | |
| 86 | ||
| 87 | 1 | Connection.prototype.format = function (sql, params) { |
| 88 | 1 | return sql.replace(/:(\w+)/g, function (w, i) { |
| 89 | 5 | return sqlString.escape(params[i]); |
| 90 | }); | |
| 91 | }; | |
| 92 | ||
| 93 | 1 | exports.create = function (options) { |
| 94 | 15 | return new Connection(options); |
| 95 | }; | |
| 96 |
| Line | Hits | Source |
|---|---|---|
| 1 | /* vim: set expandtab tabstop=2 shiftwidth=2 foldmethod=marker: */ | |
| 2 | ||
| 3 | 2 | "use strict"; |
| 4 | ||
| 5 | 2 | var util = require('util'); |
| 6 | 2 | var Emitter = require('events').EventEmitter; |
| 7 | ||
| 8 | 2 | var Queue = require('safequeue'); |
| 9 | 2 | var Connection = require(__dirname + '/connection.js'); |
| 10 | ||
| 11 | 2 | var HEARTBEAT_TIMEOUT = 1000; |
| 12 | ||
| 13 | 2 | var _QueueTimeoutError = function (msg) { |
| 14 | 1 | var e = new Error(msg); |
| 15 | 1 | e.name = 'QueueTimeout'; |
| 16 | 1 | return e; |
| 17 | }; | |
| 18 | ||
| 19 | 2 | exports.create = function (options, config) { |
| 20 | ||
| 21 | 10 | var _options = { |
| 22 | 'maxconnections' : 4, | |
| 23 | 'maxidletime' : 60000, | |
| 24 | }; | |
| 25 | 10 | for (var i in options) { |
| 26 | 11 | _options[i] = options[i]; |
| 27 | } | |
| 28 | /** | |
| 29 | * @ 连接数组 | |
| 30 | */ | |
| 31 | 10 | var conns = []; |
| 32 | ||
| 33 | /** | |
| 34 | * @ 重连暂停 | |
| 35 | */ | |
| 36 | 10 | var pause = 100; |
| 37 | ||
| 38 | /** | |
| 39 | * @ 心跳SQL | |
| 40 | */ | |
| 41 | 10 | var hbsql = 'SHOW VARIABLES LIKE "READ_ONLY"'; |
| 42 | ||
| 43 | /** | |
| 44 | * @ 心跳计时器 | |
| 45 | */ | |
| 46 | 10 | var tbeat = null; |
| 47 | ||
| 48 | /** | |
| 49 | * @ 空闲计时器 | |
| 50 | */ | |
| 51 | 10 | var timer = {}; |
| 52 | ||
| 53 | /* {{{ private function startup() */ | |
| 54 | 10 | var startup = function (o) { |
| 55 | ||
| 56 | 13 | clearTimeout(tbeat); |
| 57 | 13 | tbeat = null; |
| 58 | ||
| 59 | 13 | var c = Connection.create(config); |
| 60 | 13 | c.on('error', function () {}); |
| 61 | ||
| 62 | 13 | var s = ''; |
| 63 | 13 | conns.unshift(c); |
| 64 | 13 | (function heartbeat() { |
| 65 | 20 | c.query(hbsql, HEARTBEAT_TIMEOUT, function (e, r) { |
| 66 | 20 | if (e) { |
| 67 | 5 | o.emit('state'); |
| 68 | 5 | conns.shift(); |
| 69 | 5 | setTimeout(function () { |
| 70 | 3 | if (c) { |
| 71 | 3 | c.close(); |
| 72 | 3 | c = null; |
| 73 | } | |
| 74 | 3 | startup(o); |
| 75 | }, pause); | |
| 76 | ||
| 77 | 5 | pause = Math.min(pause + pause, 60000); |
| 78 | 5 | o.emit('error', e); |
| 79 | } else { | |
| 80 | 15 | tbeat = setTimeout(heartbeat, 10 * HEARTBEAT_TIMEOUT); |
| 81 | 15 | pause = 100; |
| 82 | 15 | var t = JSON.stringify(r); |
| 83 | 15 | if (t !== s) { |
| 84 | 9 | s = t; |
| 85 | 9 | o.emit('state', r); |
| 86 | } | |
| 87 | } | |
| 88 | }); | |
| 89 | })(); | |
| 90 | }; | |
| 91 | /* }}} */ | |
| 92 | ||
| 93 | /* {{{ private function _remove() */ | |
| 94 | 10 | var _remove = function (c, o) { |
| 95 | 4 | var i = conns.indexOf(c); |
| 96 | 4 | if (i < 0) { |
| 97 | 0 | return; |
| 98 | } | |
| 99 | 4 | c.close(); |
| 100 | 4 | c = null; |
| 101 | 4 | conns.splice(i, 1); |
| 102 | 4 | i = o._stack.indexOf(i); |
| 103 | 4 | if (i > -1) { |
| 104 | 1 | o._stack.splice(i, 1); |
| 105 | } | |
| 106 | }; | |
| 107 | /* }}} */ | |
| 108 | ||
| 109 | /* {{{ private function execute() */ | |
| 110 | 10 | var execute = function (c, o, s) { |
| 111 | 26 | c.query(s[0], s[1], function (e, r) { |
| 112 | 26 | (s[2])(e, r); |
| 113 | 26 | if (e && e.fatal) { |
| 114 | 1 | _remove(c, o); |
| 115 | 1 | return; |
| 116 | } | |
| 117 | ||
| 118 | 25 | s = o._queue.shift(); |
| 119 | 25 | if (!s) { |
| 120 | 11 | release(o, c); |
| 121 | } else { | |
| 122 | 14 | execute(c, o, s); |
| 123 | } | |
| 124 | }); | |
| 125 | }; | |
| 126 | /* }}} */ | |
| 127 | ||
| 128 | /* {{{ private function _wakeup() */ | |
| 129 | /** | |
| 130 | * wake up a pool to execute query | |
| 131 | */ | |
| 132 | 10 | var _wakeup = function (o) { |
| 133 | 17 | var m = Math.min(_options.maxconnections, 1 + _options.maxconnections + o._stack.length - conns.length); |
| 134 | 17 | while (m && o._queue.size()) { |
| 135 | 12 | var s = o._queue.shift(); |
| 136 | 12 | var i, c; |
| 137 | 12 | do { |
| 138 | 12 | i = o._stack.pop(); |
| 139 | 12 | if (i && conns[i]) { |
| 140 | 2 | c = conns[i]; |
| 141 | 2 | clearTimeout(timer[i]); |
| 142 | 2 | timer[i] = null; |
| 143 | } | |
| 144 | } while (i && !c); | |
| 145 | ||
| 146 | 12 | if (!c) { |
| 147 | 10 | c = Connection.create(config); |
| 148 | 10 | conns.push(c); |
| 149 | 10 | ['error', 'close'].forEach(function (k) { |
| 150 | 20 | c.once(k, function (e) { |
| 151 | 2 | _remove(c, o); |
| 152 | }); | |
| 153 | }); | |
| 154 | } | |
| 155 | 12 | execute(c, o, s); |
| 156 | 12 | m--; |
| 157 | } | |
| 158 | }; | |
| 159 | /* }}} */ | |
| 160 | ||
| 161 | /* {{{ private function release() */ | |
| 162 | 10 | var release = function (o, c) { |
| 163 | 11 | var i = conns.indexOf(c); |
| 164 | // XXX: we use conns[0] to heartbeat | |
| 165 | 11 | if (i > 0) { |
| 166 | 11 | o._stack.push(i); |
| 167 | 11 | timer[i] = setTimeout(function () { |
| 168 | 1 | _remove(c, o); |
| 169 | }, _options.maxidletime); | |
| 170 | } | |
| 171 | }; | |
| 172 | /* }}} */ | |
| 173 | ||
| 174 | 10 | var Mysql = function () { |
| 175 | ||
| 176 | 10 | Emitter.call(this); |
| 177 | 10 | startup(this); |
| 178 | ||
| 179 | /** | |
| 180 | * @ 执行队列 | |
| 181 | */ | |
| 182 | 10 | var _name = this._name(); |
| 183 | 10 | this._queue = Queue.create({'timeout' : 0, 'maxitem' : 0}); |
| 184 | 10 | this._queue.on('timeout', function (item, tmout, pos) { |
| 185 | 1 | (item[2])(_QueueTimeoutError(util.format( |
| 186 | 'Query stays in the queue more than %d ms (%s)', tmout, _name))); | |
| 187 | }); | |
| 188 | ||
| 189 | 10 | var _self = this; |
| 190 | 10 | this._queue.on('fill', function () { |
| 191 | 17 | _wakeup(_self); |
| 192 | }); | |
| 193 | ||
| 194 | /** | |
| 195 | * @ 空闲连接 | |
| 196 | */ | |
| 197 | 10 | this._stack = []; |
| 198 | ||
| 199 | }; | |
| 200 | 10 | util.inherits(Mysql, Emitter); |
| 201 | ||
| 202 | /* {{{ public prototype _name() */ | |
| 203 | 10 | Mysql.prototype._name = function () { |
| 204 | 15 | return conns[0]._name; |
| 205 | }; | |
| 206 | /* }}} */ | |
| 207 | ||
| 208 | /* {{{ public prototype query() */ | |
| 209 | /** | |
| 210 | * Get one connection and run sql | |
| 211 | * | |
| 212 | * @ param {String|Object} sql | |
| 213 | * @ param {Function} cb | |
| 214 | */ | |
| 215 | 10 | Mysql.prototype.query = function (sql, tmout, cb) { |
| 216 | 27 | this._queue.push([sql, tmout, cb], tmout); |
| 217 | 27 | var n = this._queue.size(); |
| 218 | 27 | if (n > 0) { |
| 219 | 15 | this.emit('busy', n, _options.maxconnections); |
| 220 | } | |
| 221 | }; | |
| 222 | /* }}} */ | |
| 223 | ||
| 224 | /* {{{ public prototype setHeartBeatQuery() */ | |
| 225 | 10 | Mysql.prototype.setHeartBeatQuery = function (sql) { |
| 226 | 7 | hbsql = sql; |
| 227 | }; | |
| 228 | /* }}} */ | |
| 229 | ||
| 230 | 10 | return new Mysql(); |
| 231 | }; | |
| 232 |