npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

pg-db

v2.0.0

Published

Transactions, named parameters, and more for node-postgres

Downloads

85

Readme

pg-db

NPM

Build Status

Overview

Helper module atop node-postgres that adds transaction management, simpler query API, event hooks, and more.

Install

$ npm install pg-db --save

Usage

// Create using default connection config of process.env.DATABASE_URL:
var db = require('pg-db')();

db.query('SELECT foo, bar, baz FROM some_table', function(err, rows){
  if( err ) return console.error('Err: %s', err);
  console.log('Rows: %j', rows);
});

Features

  • Convenient wrapper functions - callbacks for single row, multiple rows, or update row counts
  • Automatically return connections to the pool - no need to call done()
  • SQL errors automatically destory the connection - no need to call done(err)
  • Named parameters - ... WHERE foo = :foo instead of ... WHERE foo = $1
  • Transactions - automatic and transparent!
  • Event hooks - register callbacks when queries get executed - great for logging!
  • Transaction event hooks - register callbacks when a transaction completes - great for cache invalidation!

Transactions

Transactions are implemented using domains. This allows the same node-postgres client object to be used by separate parts of your application without having to manually pass it as an argument.

Any other modules that use pg-db for query execution will automatically be part of the ongoing transaction. This allows you to easily compose multiple database interactions together without needing to pass a transactional context object to every single function.

// Foo.update(foo, cb) - Updates a Foo model object
// Audit.create(message, cb) - Creates an audit record

db.tx.series([
  async.apply(Foo.update, foo),
  async.apply(Audit.create, 'Updating foo id=' + foo.id)
], cb);

Named Parameters

Named parameter support allows you to use descriptive names for parameters in SQL queries. This leads to much cleaner SQL that's easier to both read and write.

Example:

// SQL with numbered parameters:
db.queryOne('SELECT * FROM some_table WHERE foo = $1'
          , [123]
          , function(err, row) {...})

// SQL with named parameters:
db.queryOne('SELECT * FROM some_table WHERE foo = :foo'
          , {foo: 123}
          , function(err, row) {...})

A more complicated example:

// Classic style with positional parameters:
db.update('INSERT INTO user'
              + ' (id, name, email, password_hash)'
              + ' VALUES '
              + ' ($1, $2, $3, $4)'
           , [1, 'alice', '[email protected]', hash('t0ps3cret')]
           , function(err, rowCount) { /* do something */ });

// Same query with named parameters:
db.update('INSERT INTO user'
              + ' (id, name, email, password_hash)'
              + ' VALUES '
              + ' (:id, :name, :email, :passwordHash)'
           , {id: 1, name: 'alice', email: '[email protected]', passwordHash: hash('t0ps3cret')}
           , function(err, rowCount) { /* do something */ });

Another example with a model object:

var widget = {
  id: 12345,
  name: 'My Widget',
  type: 'xg17',
  owner: '[email protected]'
};

// Classic style with positional parameters:
db.update('INSERT INTO widgets'
              + ' (id, name, type, owner)'
              + ' VALUES '
              + ' ($1, $2, $3, $4)'
           , [widget.id, widget.name, widget.type, widget.owner]
           , function(err, rowCount) { /* do something */ });

// Same query with named parameters:
db.update('INSERT INTO widgets'
              + ' (id, name, type, owner)'
              + ' VALUES '
              + ' (:id, :name, :type, :owner)'
           // We can just pass in the object as is:
           , widget
           , function(err, rowCount) { /* do something */ });

API

Query API

If a transaction is in progress then all functions of the Query API will automatically use the connection client for the transactions.

If no transaction is in progress then a random connection will be retrieved from the pool of connections. After execution completes the connection will be returned to the pool.

If an error occurrs then by default the pool wil be instructed to destroy the connection. Internally, this is done by invoking done(err).

query(sql, [params], function cb(err, rows))

Execute SQL with the optional parameters and invoke the callback with the result rows.

This function is intended to be used with SQL that returns back a set of rows such as a SELECT ... statement. If no rows are returned then the callback will be invoked with an empty array.

queryOne(sql, [params], function cb(err, row))

Execute SQL with the optional parameters and invoke the callback with the first result row.

This function is intended to be used with SQL that returns back a single row. If no rows are returned then the callback is invoked with a null value for row. If more than one row is returned then the callback is invoked with an Error.

NOTE: This function will return an Error if multiple rows are returned. This is intentional as it probably means your SQL is wrong.

update(sql, [params], function cb(err, rowCount))

Execute SQL with the optional parameters and invoke the callback with the number of rows that were modified.

This function is intended to be used with SQL that performs DML (e.g. INSERT, UPDATE, DELETE).

execute(sql, [params], function cb(err, result))

Execute SQL with the optional parameters and invoke the callback with the raw result object returned by node-postgres.

This function is used internally by query, queryOne, and update. It's useful when you'd like to use both the rowCount and rows. Otherwise it's probably more convenient to use one of the other functions.

Transaction API - Control Flow

db.tx(function task(cb), function cb(err, result))

Executes a task in a transaction and invokes the callback with the result of the task.

db.tx.series(tasks, function cb(err, results))

Convenience wrapper for executing a series of tasks within a transaction.

Internally this executes the tasks by calling async.series.

db.tx.parallel(tasks, function cb(err, results))

Convenience wrapper for executing tasks in parallel within a transaction.

Internally this executes the tasks by calling async.parallel.

db.tx.auto(tasks, function cb(err, results))

Convenience wrapper for executing multiple tasks that depended on each other within a transaction.

Internally this executes the tasks by calling async.auto.

db.tx.waterfall(tasks, function cb(err, results))

Convenience wrapper for executing tasks in a waterfall, passing each result to the next task, within a transaction.

Internally this executes the tasks by calling async.waterfall.

Transaction API - Query and DML

Each of these functions checks whether a transaction is currently in progress and then invokes the equivalent non-tx function of the same name.

db.tx.query(sql, [params], function cb(err, rows))

Ensure we're running within a transaction and execute the command.

If no transaction is in progress then the callback is invoked with an error. Otherwise this behaves exactly like the non-tx version.

db.tx.queryOne(sql, [params], function cb(err, row))

Ensure we're running within a transaction and execute the command.

If no transaction is in progress then the callback is invoked with an error. Otherwise this behaves exactly like the non-tx version.

db.tx.update(sql, [params], function cb(err, rowCount))

Ensure we're running within a transaction and execute the command.

If no transaction is in progress then the callback is invoked with an error. Otherwise this behaves exactly like the non-tx version.

db.tx.execute(sql, [params], function cb(err, result))

Ensure we're running within a transaction and execute the command.

If no transaction is in progress then the callback is invoked with an error. Otherwise this behaves exactly like the non-tx version.

Transaction API - Success or Failure Hooks

The transaction API allows for registering callbacks to execute on completion of the current transaction. If no transaction is in progress then an error will be thrown.

If a callback has an arity of 0, i.e. function() {...}, then it is assumed to be a synchronous function. Otherwise it is assumed to accept a single paramater for the callback function that should be invoked, i.e. function(cb) {...}.

Any errors thrown or asynchronously returned back from callbacks are ignored. Multiple callbacks are executed in the order they are registered.

db.tx.onSuccess(function([cb]) callback)

Register a callback function to execute if the transaction is successful (i.e. after successful COMMIT)).

db.tx.onFailure(function([cb]) callback)

Register a callback function to execute if the transaction is unsuccessful (e.g. a ROLLBACK is issued).

Events API

db.on(event, function(data...))

Register a callback to execute when a given event occurs.

The follow event types are supported:

  • execute - triggered whenever a query is executed.
  • executeComplete - triggered after a query is executed.
  • begin - triggered before a transaction is started.
  • beginComplete - triggered after a transaction is started.
  • commit - triggered when a transaction is about to be committed.
  • commitComplete - triggered after a transaction is committed.
  • rollback - triggered when a transaction is about to be rolled back.
  • rollbackComplete - triggered after a transaction is rolled back.

Building and Testing

To buld the module run:

$ make

To run the tests first create a test/env file. You can use test/env.example as a template. Edit the DATABASE_URL property to point to a Postgres database.

Then, to run the tests run:

$ make test

License

See the LICENSE file for details.