Skip to content

Commit b9c7feb

Browse files
committed
1.0.1: Fix added rows in diff calculation
* multipleQueries test created
1 parent 8a9d0f6 commit b9c7feb

File tree

6 files changed

+164
-11
lines changed

6 files changed

+164
-11
lines changed

README.md

-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,5 @@
11
# mysql-live-select [![Build Status](https://travis-ci.org/numtel/mysql-live-select.svg?branch=master)](https://travis-ci.org/numtel/mysql-live-select)
22

3-
### :boom: :exclamation: Notice!
4-
5-
This readme is for the updated 1.0 release but has not been included in `numtel:mysql` yet and is not quite ready for use. I have found that the updated diff calculation provides superfluous data for `added` rows. Please wait to update from `0.0.24`. **See [the readme for the current stable version on this page](https://github.com/numtel/mysql-live-select/blob/89691160b7e1fbfde1ae7055980668ceb4182f8a/README.md).**
6-
73
NPM Package to provide events when a MySQL select statement result set changes.
84

95
Built using the [`zongji` Binlog Tailer](https://github.com/nevill/zongji) and [`node-mysql`](https://github.com/felixge/node-mysql) projects.

lib/QueryCache.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,7 @@ QueryCache.prototype.invalidate = function() {
111111
self.initialized = true;
112112
self.lastUpdate = Date.now();
113113

114-
if(self.needUpdate) update();
114+
if(self.needUpdate === true) update();
115115
});
116116
}
117117

lib/differ.js

+15-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,9 @@ exports.generate = function(oldHashes, newData) {
99

1010
// Need copy of curHashes so duplicates can be checked off
1111
var curHashes2 = curHashes.slice();
12-
var addedRows = newData.map(function(row) {
12+
var addedRows = newData.filter(function(row) {
13+
return oldHashes.indexOf(row._hash) === -1;
14+
}).map(function(row) {
1315
// Prepare row meta-data
1416
row._index = curHashes2.indexOf(row._hash) + 1;
1517

@@ -74,6 +76,11 @@ exports.generate = function(oldHashes, newData) {
7476
}
7577

7678
exports.apply = function(data, diff) {
79+
data = _.clone(data, true).map(function(row, index) {
80+
row._index = index + 1;
81+
return row;
82+
});
83+
7784
var newResults = data.slice();
7885

7986
diff.removed !== null && diff.removed.forEach(
@@ -98,5 +105,11 @@ exports.apply = function(data, diff) {
98105
diff.added !== null && diff.added.forEach(
99106
function(added) { newResults[added._index - 1] = added; });
100107

101-
return newResults.filter(function(row) { return row !== undefined; });
108+
var result = newResults.filter(function(row) { return row !== undefined; });
109+
110+
return result.map(function(row) {
111+
row = _.clone(row);
112+
delete row._index;
113+
return row;
114+
});
102115
}

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "mysql-live-select",
3-
"version": "1.0.0",
3+
"version": "1.0.1",
44
"description": "Live updating MySQL SELECT statements",
55
"main": "lib/LiveMysql.js",
66
"repository": "https://github.com/numtel/mysql-live-select",

test/fixtures/multipleQueries.js

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
exports.characters = {
2+
columns: { id: 'INT UNSIGNED', name: 'VARCHAR(45)', visits: 'INT UNSIGNED' },
3+
initial: [
4+
{ id: 1, name: 'Celie', visits: 10 },
5+
{ id: 2, name: 'Nettie', visits: 15 },
6+
{ id: 3, name: 'Harpo', visits: 20 },
7+
{ id: 4, name: 'Shug', visits: 25 },
8+
],
9+
select: 'SELECT * FROM $table$ ORDER BY visits DESC',
10+
// Optional 'condition' function (not specified on this case)
11+
// Execute a query after initialization and each update thereafter
12+
queries: [
13+
'UPDATE $table$ SET visits=visits+5 WHERE id=4',
14+
'UPDATE $table$ SET visits=visits+15 WHERE id=1',
15+
'INSERT INTO $table$ (id, name, visits) VALUES (5, \'Squeak\', 4)',
16+
'DELETE FROM $table$ WHERE id=3'
17+
],
18+
// Required, an array of diff objects: one for initial data and one for each
19+
// query
20+
expectedDiffs: [
21+
{ removed: null,
22+
moved: null,
23+
copied: null,
24+
added:
25+
[ { id: 4, name: 'Shug', visits: 25, _index: 1 },
26+
{ id: 3, name: 'Harpo', visits: 20, _index: 2 },
27+
{ id: 2, name: 'Nettie', visits: 15, _index: 3 },
28+
{ id: 1, name: 'Celie', visits: 10, _index: 4 } ] },
29+
{ removed: [ { _index: 1 } ],
30+
moved: null,
31+
copied: null,
32+
added: [ { id: 4, name: 'Shug', visits: 30, _index: 1 } ] },
33+
{ removed: [ { _index: 2 } ],
34+
moved:
35+
[ { old_index: 2, new_index: 3 },
36+
{ old_index: 3, new_index: 4 } ],
37+
copied: null,
38+
added: [ { id: 1, name: 'Celie', visits: 25, _index: 2 } ] },
39+
{ removed: null,
40+
moved: null,
41+
copied: null,
42+
added: [ { id: 5, name: 'Squeak', visits: 4, _index: 5 } ] },
43+
{ removed: [ { _index: 5 } ],
44+
moved:
45+
[ { old_index: 4, new_index: 3 },
46+
{ old_index: 5, new_index: 4 } ],
47+
copied: null,
48+
added: null }
49+
],
50+
// Optionally, specify an array of expected data for each event,
51+
// in order to test the differ's correctness
52+
expectedDatas: [
53+
[ { id: 4, name: 'Shug', visits: 25 },
54+
{ id: 3, name: 'Harpo', visits: 20 },
55+
{ id: 2, name: 'Nettie', visits: 15 },
56+
{ id: 1, name: 'Celie', visits: 10 } ],
57+
[ { id: 4, name: 'Shug', visits: 30 },
58+
{ id: 3, name: 'Harpo', visits: 20 },
59+
{ id: 2, name: 'Nettie', visits: 15 },
60+
{ id: 1, name: 'Celie', visits: 10 } ],
61+
[ { id: 4, name: 'Shug', visits: 30 },
62+
{ id: 1, name: 'Celie', visits: 25 },
63+
{ id: 3, name: 'Harpo', visits: 20 },
64+
{ id: 2, name: 'Nettie', visits: 15 } ],
65+
[ { id: 4, name: 'Shug', visits: 30 },
66+
{ id: 1, name: 'Celie', visits: 25 },
67+
{ id: 3, name: 'Harpo', visits: 20 },
68+
{ id: 2, name: 'Nettie', visits: 15 },
69+
{ id: 5, name: 'Squeak', visits: 4 } ],
70+
[ { id: 4, name: 'Shug', visits: 30 },
71+
{ id: 1, name: 'Celie', visits: 25 },
72+
{ id: 2, name: 'Nettie', visits: 15 },
73+
{ id: 5, name: 'Squeak', visits: 4 } ]
74+
]
75+
}

test/index.js

+72-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
/* mysql-live-select, MIT License [email protected]
22
test/index.js - Test Suite */
3+
var _ = require('lodash');
34
var LiveMysql = require('../');
45
var settings = require('./settings/mysql');
56
var querySequence = require('./helpers/querySequence');
67
var Connector = require('./helpers/connector');
8+
var multipleQueriesData = require('./fixtures/multipleQueries');
79
var server = new Connector(settings);
810

911
module.exports = {
@@ -148,15 +150,82 @@ module.exports = {
148150
});
149151
});
150152
},
151-
pauseAndResume: function(test){
153+
// Cases specified in test/fixtures/multipleQueries.js
154+
multipleQueries: function(test) {
155+
var tablePrefix = 'multiple_queries_';
156+
// Milliseconds between each query execution
157+
var queryWaitTime = 100;
158+
server.on('ready', function(conn, esc, escId, queries) {
159+
Object.keys(multipleQueriesData).forEach(function(queryName) {
160+
var table = tablePrefix + queryName;
161+
var tableEsc = escId(table);
162+
var details = multipleQueriesData[queryName];
163+
164+
var columnDefStr = _.map(details.columns, function(typeStr, name) {
165+
return escId(name) + ' ' + typeStr;
166+
}).join(', ');
167+
var columnList = Object.keys(details.columns);
168+
var initDataStr = details.initial.map(function(rowData) {
169+
return '(' + columnList.map(function(column) {
170+
return esc(rowData[column]);
171+
}).join(', ') + ')';
172+
}).join(', ');
173+
var replaceTable = function(query) {
174+
return query.replace(/\$table\$/g, tableEsc);
175+
};
176+
querySequence(conn.db, [
177+
'DROP TABLE IF EXISTS ' + tableEsc,
178+
'CREATE TABLE ' + tableEsc + ' (' + columnDefStr + ')',
179+
'INSERT INTO ' + tableEsc + ' (' + columnList.map(escId).join(', ') +
180+
') VALUES ' + initDataStr,
181+
], function(results) {
182+
var actualDiffs = [];
183+
var actualDatas = [];
184+
var curQuery = 0;
185+
var oldData = [];
186+
conn.select(replaceTable(details.select), [ {
187+
table: table,
188+
database: server.database,
189+
condition: details.condition
190+
} ]).on('update', function(diff, rows) {
191+
actualDiffs.push(diff);
192+
actualDatas.push(LiveMysql.applyDiff(oldData, diff));
193+
194+
oldData = rows;
195+
196+
if(curQuery < details.queries.length) {
197+
setTimeout(function() {
198+
querySequence(conn.db,
199+
[ replaceTable(details.queries[curQuery++]) ],
200+
function(results){ /* do nothing with results */ });
201+
}, queryWaitTime);
202+
}
203+
204+
if(actualDiffs.length === details.expectedDiffs.length) {
205+
test.deepEqual(actualDiffs, details.expectedDiffs,
206+
'Diff Mismatch on ' + queryName);
207+
208+
if(details.expectedDatas) {
209+
test.deepEqual(actualDatas, details.expectedDatas,
210+
'Data Mismatch on ' + queryName);
211+
}
212+
test.done();
213+
}
214+
});
215+
216+
});
217+
});
218+
});
219+
},
220+
pauseAndResume: function(test) {
152221
var waitTime = 500;
153222
var table = 'pause_resume';
154-
server.on('ready', function(conn, esc, escId, queries){
223+
server.on('ready', function(conn, esc, escId, queries) {
155224
querySequence(conn.db, [
156225
'DROP TABLE IF EXISTS ' + escId(table),
157226
'CREATE TABLE ' + escId(table) + ' (col INT UNSIGNED)',
158227
'INSERT INTO ' + escId(table) + ' (col) VALUES (10)',
159-
], function(results){
228+
], function(results) {
160229
var pauseTime = Date.now();
161230
conn.select('SELECT * FROM ' + escId(table), [ {
162231
table: table,

0 commit comments

Comments
 (0)