UNPKG

node-querybuilder

Version:

Modeled after Codeigniter's QueryBuilder. Build and execute queries in a safe and database-agnostic way.

312 lines (290 loc) 13 kB
const chai = require('chai'); chai.Assertion.includeStack = true; const should = chai.should(); const expect = chai.expect; const settings = require('../configs').mysql; const QueryBuilder = require('../../index.js'); const check = (done, f) => { try { f(); done(); } catch(e) { done(e); } }; const connection_released = qb => { const connection = qb.connection(); expect(connection._pool._freeConnections).to.have.length(0); qb.release(); expect(connection._pool._freeConnections).to.have.length(1); }; describe('QueryBuilder() - MySQL Adapter', () => { const on_connect = err => { if (err) { console.error("Not connected!"); return; } console.log("connected!"); }; const driver = 'mysql'; const bad_user = Object.assign({},settings, {user: 'foobar'}); const bad_host = Object.assign({},settings, {host: 'nonlocalhost'}); const bad_password = Object.assign({},settings, {password: 'password'}); const bad_database = Object.assign({},settings, {database: 'bad_mock_db'}); const bad_port = Object.assign({},settings, {port: 1}); const bad_version = Object.assign({},settings, {version: 12}); it('should exist', () => { should.exist(QueryBuilder); }); it('should be a function/class', () => { QueryBuilder.should.be.a('function'); }); it('should have all the QueryBuilder properties', () => { const qb = new QueryBuilder(Object.assign({}, settings), driver); const children = [ 'where_array','where_in_array','from_array','join_array','select_array','set_array','order_by_array', 'group_by_array','having_array','limit_to','offset_val','join_clause','last_query_string','distinct_clause','aliased_tables', ]; expect(qb).to.include.keys(children); }); it('should have all the QueryBuilder methods', () => { const qb = new QueryBuilder(Object.assign({}, settings), driver); const children = [ 'reset_query','where','or_where','_where','where_in','or_where_in','where_not_in','or_where_not_in','_where_in','like', 'not_like','or_like','or_not_like','_like','from','join','select','select_min','select_max','select_avg','select_sum', '_min_max_avg_sum','distinct','group_by','having','or_having','_having','order_by','limit','offset','set', ]; children.forEach(v => { expect(qb).to.respondTo(v); }); }); it('should have all the QueryExec methods', () => { const qb = new QueryBuilder(Object.assign({}, settings), driver); const children = ['insert','insert_ignore','insert_batch','get','get_where','count','update','update_batch','delete','get_compiled_select','get_compiled_delete','get_compiled_update','get_compiled_insert','compile_select','compile_delete','compile_update','compile_insert']; children.forEach(v => { expect(qb).to.respondTo(v); }); }); it('should have all the miscellaneous methods', () => { const qb = new QueryBuilder(Object.assign({}, settings), driver); const children = ['last_query','escape','empty_table','truncate']; children.forEach(v => { expect(qb).to.respondTo(v); }); }); it('should establish a single connection given valid connection credentials', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); expect(qb, 'should have connect property').to.have.property('connect'); qb.connect(err => { check(done, () => { expect(err, 'should be connected').to.not.be.instanceof(Error); }); }); }); it('should allow us to disconnect from MySQL', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { should.exist(qb.disconnect); qb.disconnect.should.be.a('function'); qb.disconnect(err => { const connection = qb.connection(); check(done, () => { expect(err, 'should be diconnected').to.not.be.instanceof(Error); expect(connection._protocol._ended).to.be.true; }); }); }); }); it('should fail to establish a single connection given no parameters', () => { expect(() => new QueryBuilder()).to.throw(Error); }); it('should fail to establish a single connection given no connection credentials', () => { expect(() => new QueryBuilder({},driver)).to.throw(Error); }); it('should fail to establish a single connection given connection credentials with bad user', done => { let qb; try { qb = new QueryBuilder(bad_user, driver); } catch(e) { expect(e, 'should not get a connection').to.be.instanceof(Error); return; } expect(qb, 'should have connect method').to.respondTo('connect'); qb.connect(err => { check(done, () => { expect(err, 'should not be connected').to.be.instanceof(Error); }); }); }); it('should fail to establish a single connection given connection credentials with bad host', done => { let qb; try { qb = new QueryBuilder(bad_host, driver); } catch(e) { expect(e, 'should not get a connection').to.be.instanceof(Error); return; } expect(qb, 'should have connect method').to.respondTo('connect'); qb.connect(err => { check(done, () => { expect(err, 'should not be connected').to.be.instanceof(Error); }); }); }); it('should fail to establish a single connection given connection credentials with bad database', done => { let qb; try { qb = new QueryBuilder(bad_database, driver); } catch(e) { expect(e, 'should not get a connection').to.be.instanceof(Error); return; } expect(qb, 'should have connect method').to.respondTo('connect'); qb.connect(err => { check(done, () => { expect(err, 'should not be connected').to.be.instanceof(Error); }); }); }); it('should fail to establish a single connection given connection credentials with bad password', done => { let qb; try { qb = new QueryBuilder(bad_password, driver); } catch(e) { expect(e, 'should not get a connection').to.be.instanceof(Error); return; } expect(qb, 'should have connect method').to.respondTo('connect'); qb.connect(err => { check(done, () => { expect(err, 'should not be connected').to.be.instanceof(Error); }); }); }); it('should fail to establish a single connection given connection credentials with bad port', done => { let qb; try { qb = new QueryBuilder(bad_port, driver); } catch(e) { expect(e, 'should not get a connection').to.be.instanceof(Error); return; } expect(qb, 'should have connect method').to.respondTo('connect'); qb.connect(err => { check(done, () => { expect(err, 'should not be connected').to.be.instanceof(Error); }); }); }); it('should fail to establish connection if an invalid driver is specified', () => { let qb; expect(() => new QueryBuilder(settings), 'no driver specified').to.not.throw(Error); expect(() => new QueryBuilder(settings, 'foobar'), 'invalid driver specified').to.throw(Error); }); it('should fail to establish connection if an invalid driver version is specified', () => { let qb; expect(() => new QueryBuilder( Object.assign({}, settings), driver), 'valid driver version').to.not.throw(Error); expect(() => new QueryBuilder(bad_version, driver), 'invalid driver version').to.throw(Error); }); it('should allow us to retrieve our connection settings for reference', done => { const conn_settings = Object.assign({}, settings, {password: undefined}); delete conn_settings.version; const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { check(done, () => { should.exist(qb.connection_settings); qb.connection_settings.should.be.a('function'); const settings = qb.connection_settings(); expect(settings).to.be.instanceof(Object); expect(settings).to.be.eql(conn_settings); qb.disconnect(); }); }); }); it('should allow us to escape certain values', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { check(done, () => { should.exist(qb.escape); qb.escape.should.be.a('function'); expect(qb.escape(null)).to.be.eql('NULL'); expect(qb.escape('3')).to.be.eql(3); expect(qb.escape(false)).to.be.eql(0); expect(qb.escape(true)).to.be.eql(1); expect(qb.escape(null)).to.be.eql('NULL'); qb.disconnect(); }); }); }); it('should allow us to escape identifiers the MySQL way', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { check(done, () => { should.exist(qb.escape_id); qb.escape_id.should.be.a('function'); expect(qb.escape_id('foo'), 'not pre-escaped').to.be.eql('`foo`'); expect(qb.escape_id('`foo`'), 'pre-escaped').to.be.eql('```foo```'); expect(qb.escape_id('foo.bar'), 'with qualifier').to.be.eql('`foo`.`bar`'); qb.disconnect(); }); }); }); it('should allow us to execute a query', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { qb.query("select * from `cities` where `city` like 'Z%' and `state_code` = 'FL'", (err, res) => { check(done, () => { expect(err).to.not.be.instanceof(Error); expect(res).to.not.be.empty; expect(res).to.have.length(3); }); }); }); }); it('should not be able to release a non-pooled connection', done => { const qb = new QueryBuilder(Object.assign({}, settings), driver); qb.connect(err => { check(done, () => { expect(() => qb.release()).to.throw(Error); }); }); }); it('should create a connection pool object if asked', () => { const pool = new QueryBuilder(Object.assign({}, settings), driver, 'pool'); const methods = ['pool','get_connection','disconnect']; expect(pool, 'pool should be object').to.be.instanceof(Object); methods.forEach(v => { expect(pool, `pool should have method: ${v}`).to.respondTo(v); }); pool.pool.should.be.a('function'); pool.get_connection.should.be.a('function'); pool.disconnect.should.be.a('function'); }); it('should create a QueryBuilder adapter when getting a connection from the pool', done => { const qb2 = new QueryBuilder(Object.assign({}, settings), driver); const pool = new QueryBuilder(Object.assign({}, settings), driver, 'pool'); pool.get_connection(qb => { check(done, () => { expect(qb2).to.include.keys(Object.keys(qb)); }); }); }); it('should allow one to release a connection from the pool', done => { const qb2 = new QueryBuilder(Object.assign({}, settings), driver); const pool = new QueryBuilder(Object.assign({}, settings), driver, 'pool'); pool.get_connection(qb => { check(done, () => connection_released(qb)); }); }); it('should allow one use the same connection pool connection for multiple queries', done => { const pool = new QueryBuilder(Object.assign({}, settings), driver, 'pool'); pool.get_connection(qb => { qb.query('select * from `cities` where `city` = "Gainesville"', (err, res) => { if (res.length > 0) { qb.query('select * from `cities` where `state_code` = "' + res[0].state_code + '"', (err, res) => { check(done, () => connection_released(qb)); }); } else { check(done, () => connection_released(qb)); } }); }); }); });