sharp-db
Version:
Classes for running SQL and building select queries for MySQL in Node
796 lines (764 loc) • 56.5 kB
HTML
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width,initial-scale=1">
<title>Home - Documentation</title>
<script src="scripts/prettify/prettify.js"></script>
<script src="scripts/prettify/lang-css.js"></script>
<!--[if lt IE 9]>
<script src="//html5shiv.googlecode.com/svn/trunk/html5.js"></script>
<![endif]-->
<link type="text/css" rel="stylesheet" href="https://code.ionicframework.com/ionicons/2.0.1/css/ionicons.min.css">
<link type="text/css" rel="stylesheet" href="styles/prettify-tomorrow.css">
<link type="text/css" rel="stylesheet" href="styles/jsdoc-default.css">
</head>
<body>
<input type="checkbox" id="nav-trigger" class="nav-trigger" />
<label for="nav-trigger" class="navicon-button x">
<div class="navicon"></div>
</label>
<label for="nav-trigger" class="overlay"></label>
<nav>
<li class="nav-link nav-home-link"><a href="index.html">Home</a></li><li class="nav-heading">Classes</li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="DataBroker.html">DataBroker</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="DataBroker.html#cleanup">cleanup</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="DataBroker.html#created_and_modified">created_and_modified</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="DataBroker.html#createdAndModified">createdAndModified</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="DataBroker.html#delete">delete</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="DataBroker.html#insert">insert</a></span></li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="Db.html">Db</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#.destroyAll">destroyAll</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#.endAll">endAll</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#.factory">factory</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#.withInstance">withInstance</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#beginTransaction">beginTransaction</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#bindArgs">bindArgs</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#commit">commit</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#connect">connect</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#connectOnce">connectOnce</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#delete">delete</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#deleteFrom">deleteFrom</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#destroy">destroy</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#emit">emit</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#emitError">emitError</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#end">end</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#escape">escape</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#escapeLike">escapeLike</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#escapeQuoteless">escapeQuoteless</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#exportAsSql">exportAsSql</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#insert">insert</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#insertExtended">insertExtended</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#insertInto">insertInto</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#insertIntoOnDuplicateKeyUpdate">insertIntoOnDuplicateKeyUpdate</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#multiQuery">multiQuery</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#query">query</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#quote">quote</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#rollback">rollback</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#select">select</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectByKey">selectByKey</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectExists">selectExists</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectFirst">selectFirst</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectFrom">selectFrom</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectGrouped">selectGrouped</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectHash">selectHash</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectId">selectId</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectIndexed">selectIndexed</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectList">selectList</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectOrCreate">selectOrCreate</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectOrCreateId">selectOrCreateId</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectUuid">selectUuid</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#selectValue">selectValue</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#startTransaction">startTransaction</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#tpl">tpl</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#update">update</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Db.html#updateTable">updateTable</a></span></li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="DbEvent.html">DbEvent</a></span></li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="Parser.html">Parser</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_extractSubqueries">_extractSubqueries</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleConditions">_handleConditions</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleCrossJoin">_handleCrossJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleFrom">_handleFrom</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleFullJoin">_handleFullJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleFullOuterJoin">_handleFullOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleGroupBy">_handleGroupBy</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleHaving">_handleHaving</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleInnerJoin">_handleInnerJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleLeftJoin">_handleLeftJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleLeftOuterJoin">_handleLeftOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleLimit">_handleLimit</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleOffset">_handleOffset</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleOrderBy">_handleOrderBy</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleRightJoin">_handleRightJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleRightOuterJoin">_handleRightOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleSelect">_handleSelect</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_handleWhere">_handleWhere</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_injectSubqueries">_injectSubqueries</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_split">_split</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#_stripComments">_stripComments</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Parser.html#parse">parse</a></span></li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="Select.html">Select</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#._extractBindingName">_extractBindingName</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#.init">init</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#.parse">parse</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_conditions">_conditions</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_isEntirelyDigits">_isEntirelyDigits</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_isEntirelyDigitsNoZeros">_isEntirelyDigitsNoZeros</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_isPlaceholder">_isPlaceholder</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_spliceChildData">_spliceChildData</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#_spliceSiblingData">_spliceSiblingData</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#bind">bind</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#column">column</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#columns">columns</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#crossJoin">crossJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#escape">escape</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#escapeLike">escapeLike</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#escapeQuoteless">escapeQuoteless</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetch">fetch</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchFirst">fetchFirst</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchGrouped">fetchGrouped</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchHash">fetchHash</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchIndexed">fetchIndexed</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchList">fetchList</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fetchValue">fetchValue</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#foundRows">foundRows</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#from">from</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fullJoin">fullJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#fullOuterJoin">fullOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#getClone">getClone</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#getFoundRowsQuery">getFoundRowsQuery</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#getFoundRowsSql">getFoundRowsSql</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#groupBy">groupBy</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#having">having</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#innerJoin">innerJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#join">join</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#leftJoin">leftJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#leftOuterJoin">leftOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#limit">limit</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#normalized">normalized</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#offset">offset</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#option">option</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#orderBy">orderBy</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#orHaving">orHaving</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#orWhere">orWhere</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#page">page</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#parse">parse</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#reset">reset</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#rightJoin">rightJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#rightOuterJoin">rightOuterJoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#sortField">sortField</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#table">table</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#tables">tables</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#toBoundSql">toBoundSql</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#toString">toString</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#unbind">unbind</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#unjoin">unjoin</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#where">where</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#whereBetween">whereBetween</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#withChildData">withChildData</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Select.html#withSiblingData">withSiblingData</a></span></li><li class="nav-heading"><span class="nav-item-type type-class">C</span><span class="nav-item-name"><a href="Ssh.html">Ssh</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Ssh.html#end">end</a></span></li><li class="nav-item"><span class="nav-item-type type-function">F</span><span class="nav-item-name"><a href="Ssh.html#tunnelTo">tunnelTo</a></span></li>
</nav>
<div id="main">
<section class="readme">
<article><h1>sharp-db</h1>
<p><a href="https://travis-ci.org/kensnyder/sharp-db"><img src="https://travis-ci.com/kensnyder/sharp-db.svg?branch=master&v=1.7.1" alt="Build Status"></a>
<a href="https://codecov.io/gh/kensnyder/sharp-db"><img src="https://codecov.io/gh/kensnyder/sharp-db/branch/master/graph/badge.svg?v=1.7.1" alt="Code Coverage"></a>
<a href="https://opensource.org/licenses/ISC"><img src="https://img.shields.io/github/license/kensnyder/sharp-db.svg?v=1.7.1" alt="ISC License"></a></p>
<p>Classes for running SQL and building select queries for MySQL in Node</p>
<h2>Installation</h2>
<pre class="prettyprint source lang-bash"><code>npm install sharp-db
</code></pre>
<h2>Table of Contents</h2>
<ul>
<li><a href="#db">Db</a>
<ul>
<li><a href="#connection">Connection</a></li>
<li><a href="#instantiation">Instantiation</a></li>
<li><a href="#ssh-tunneling">SSH Tunneling</a></li>
<li><a href="#basic-use">Basic Use</a></li>
<li><a href="#bindings">Bindings</a></li>
<li><a href="#methods">Methods</a></li>
<li><a href="#useful-query-options">Useful Query Options</a></li>
<li><a href="#solutions-to-common-problems">Solutions to Common Problems</a></li>
</ul>
</li>
<li><a href="#select">Select</a>
<ul>
<li><a href="#selectparse">Select.parse()</a></li>
<li><a href="#building-the-query">Building the Query</a></li>
<li><a href="#fetching-data">Fetching Data</a></li>
<li><a href="#counting-results">Counting Results</a></li>
<li><a href="#specifying-the-db-instance-to-use">Set the <code>Db</code> instance</a></li>
<li><a href="#dependent-data">Dependent Data</a></li>
<li><a href="#other-methods">Other Methods</a></li>
<li><a href="#selectparse-limitations">Select.parse() Limitations</a></li>
</ul>
</li>
<li><a href="#databroker">DataBroker</a>
<ul>
<li><a href="#use-in-integration-tests">Use in Integration Tests</a></li>
<li><a href="#insertions">Insertions</a></li>
<li><a href="#deletions">Deletions</a></li>
</ul>
</li>
<li><a href="#sqlbuilder">SqlBuilder</a>
<ul>
<li><a href="#function-list">Function List</a></li>
</ul>
</li>
<li><a href="#querylogger">QueryLogger</a>
<ul>
<li><a href="#logging-queries">Logging queries</a></li>
</ul>
</li>
<li><a href="./CONTRIBUTING.md">How to Contribute</a></li>
<li><a href="./LICENSE.md">ISC License</a></li>
</ul>
<h2>Db</h2>
<h3>Connection</h3>
<p>Connection can be configured with environmental variables or in the constructor.</p>
<table>
<thead>
<tr>
<th>Option</th>
<th>ENV name</th>
<th>Default</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>host</code></td>
<td>DB_HOST</td>
<td>127.0.0.1</td>
</tr>
<tr>
<td><code>port</code></td>
<td>DB_PORT</td>
<td>3306</td>
</tr>
<tr>
<td><code>user</code></td>
<td>DB_USER</td>
<td>root</td>
</tr>
<tr>
<td><code>password</code></td>
<td>DB_PASSWORD</td>
<td><em>empty string</em></td>
</tr>
<tr>
<td><code>database</code></td>
<td>DB_DATABASE</td>
<td>undefined</td>
</tr>
<tr>
<td><code>charset</code></td>
<td>DB_CHARSET</td>
<td>utf8mb4</td>
</tr>
</tbody>
</table>
<p>See node's mysqljs for <a href="https://github.com/mysqljs/mysql#connection-options">other options</a>.</p>
<h3>Instantiation</h3>
<p>Connect to MySQL server</p>
<pre class="prettyprint source lang-js"><code>const { Db } = require('sharp-db');
// read options from ENV
const db1 = Db.factory();
// specify options in constructor
const db2 = new Db({
host: '127.0.0.1',
user: 'root',
password: '',
port: 3306,
});
// instance that was last created
const db2Again = Db.factory();
// Don't forget to close the connection when done
db1.end();
db2.end();
</code></pre>
<h4>Auto factory and end</h4>
<p>You can use <code>await Db.withInstance(db => /* do stuff with db */)</code>
to get an instance, do something, and then close the connection.</p>
<pre class="prettyprint source lang-js"><code>const { Db } = require('sharp-db');
// read options from ENV, instantiate and call db.end() automatically
const { error, domain } = await Db.withInstance(async db => {
const sql = 'SELECT email FROM users WHERE id = 5';
const { results: email } = await db.selectValue(sql);
return {
domain: email.split('@').pop(),
};
});
</code></pre>
<p><code>Db.withInstance()</code> will return one of the following:</p>
<ol>
<li>
<p><code>{ error }</code> where error is an object from mysql2. <a href="https://github.com/mysqljs/mysql#error-handling">Full docs</a>. Summary:</p>
<ul>
<li><code>error.sqlMessage</code> The textual description of the error</li>
<li><code>error.code</code> The string error code such as <code>PROTOCOL_CONNECTION_LOST</code></li>
<li><code>error.errno</code> The associated number code</li>
<li><code>error.fatal</code> True if the error caused the connection to close</li>
<li><code>error.sql</code> The full SQL of the failed query</li>
<li><code>error.sqlState</code> The five-character SQLSTATE code</li>
</ul>
</li>
<li>
<p>Whatever value is returned from the handler. We suggest always returning an object, expecting the caller to check <code>.error</code>.</p>
</li>
</ol>
<p><strong>WARNING:</strong> Your handler function must return a promise that resolves AFTER
your query has returned a result. Failing to do so will result in <code>db.end()</code>
being called before your query is run. You may see an Error similar to the
following:</p>
<p><code>[ERROR] -1 (N/A): Can't add new command when connection is in closed state</code></p>
<p>For example:</p>
<pre class="prettyprint source lang-js"><code>// WILL FAIL:
const { error } = Db.withInstance(db => {
db.insertInto('users', user);
});
// WILL SUCCEED:
const { error, results } = Db.withInstance(db => {
return db.insertInto('users', user);
});
// ALSO OK:
const { error, newUserId } = Db.withInstance(async db => {
const { insertId } = await db.insertInto('users', user);
return {
newUserId: insertId,
};
});
</code></pre>
<h3>SSH Tunneling</h3>
<p>Connect to MySQL through an SSH tunnel</p>
<pre class="prettyprint source lang-js"><code>const { Db } = require('sharp-db');
const db = Db.factory({
// MySQL connection as first argument
host: '127.0.0.1',
port: 3306,
user: 'root',
password: '',
}, {
// SSH connection as second argument
host: 'example.com',
port: 22,
user: 'ubuntu',
privateKey: '~/.ssh/example.com.pem',
});
</code></pre>
<p>SSH Tunnel Options</p>
<table>
<thead>
<tr>
<th>Option</th>
<th>ENV name</th>
<th>Default</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>host</code></td>
<td>DB_SSH_HOST</td>
<td>"localhost"</td>
</tr>
<tr>
<td><code>port</code></td>
<td>DB_SSH_PORT</td>
<td>22</td>
</tr>
<tr>
<td><code>user</code></td>
<td>DB_SSH_USER</td>
<td><em>none</em></td>
</tr>
<tr>
<td><code>password</code></td>
<td>DB_SSH_PASSWORD</td>
<td><em>none</em></td>
</tr>
<tr>
<td><code>privateKey</code></td>
<td>DB_SSH_PRIVATE_KEY</td>
<td><em>none</em></td>
</tr>
<tr>
<td><code>localPort</code></td>
<td>DB_SSH_LOCAL_PORT</td>
<td>12346</td>
</tr>
</tbody>
</table>
<p>See all options in <a href="https://github.com/mscdex/ssh2#client-methods">ssh2's npm package</a>.</p>
<h3>Basic use</h3>
<p>All code examples below assume the <code>Db</code> instance has been stored in <code>db</code>.</p>
<h4>Plain select queries</h4>
<pre class="prettyprint source lang-js"><code>const { Db } = require('sharp-db');
const db = Db.factory();
const { query, results, fields } = await db.select('SELECT * FROM users');
// query is the final query executed after value binding
// results is an Array of objects representing the query results
// fields is an Array of objects representing the columns that were returned
// Don't forget to close the connection when done
db.destroy();
</code></pre>
<p>Relevant properties of each <code>fields</code> item:</p>
<table>
<thead>
<tr>
<th>Item</th>
<th>Description</th>
<th>Example</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>characterSet</code></td>
<td><a href="https://github.com/mysqljs/mysql/blob/master/lib/protocol/constants/charsets.js">Character set constant</a></td>
<td>45</td>
</tr>
<tr>
<td><code>encoding</code></td>
<td>Character set name</td>
<td>utf8</td>
</tr>
<tr>
<td><code>name</code></td>
<td>Name of column</td>
<td>my_column</td>
</tr>
<tr>
<td><code>columnLength</code></td>
<td>Number of <em>bytes</em> of field</td>
<td>400</td>
</tr>
<tr>
<td><code>columnType</code></td>
<td><a href="https://github.com/mysqljs/mysql/blob/master/lib/protocol/constants/types.js">Data type constant</a></td>
<td>253</td>
</tr>
<tr>
<td><code>flags</code></td>
<td><a href="https://github.com/mysqljs/mysql/blob/master/lib/protocol/constants/field_flags.js">Field flag constant</a></td>
<td>33</td>
</tr>
</tbody>
</table>
<h3>Bindings</h3>
<p>Question-mark and colon-prefixed bindings are supported.</p>
<h4>Binding with Question Marks</h4>
<pre class="prettyprint source lang-js"><code>const sql = 'SELECT * FROM users WHERE is_active = ? AND department_id = ?';
const { results: users } = await db.select(sql, true, 5);
</code></pre>
<h4>Named Bindings</h4>
<pre class="prettyprint source lang-js"><code>const sql = 'SELECT * FROM users WHERE is_active = :isActive AND department_id = :departmentId';
const { results: users } = await db.select(sql, {
isActive: true,
departmentId: 5,
});
</code></pre>
<h4>Binding data types</h4>
<pre class="prettyprint source lang-js"><code>const { results: users } = await db.select(sql, {
isActive: true, // Boolean
departmentId: 5, // Number
createdAt: '2020-02-14', // Strings
statusCode: [1, 2, 3], // Arrays e.g. IN(1, 2, 3)
deletedAt: null, // null e.g. NULL
});
</code></pre>
<h3>Methods</h3>
<h4>selectFirst(sql, ...bindValues)</h4>
<p>Get only the first row.</p>
<pre class="prettyprint source lang-js"><code>const { results: row } = await db.selectFirst(sql);
</code></pre>
<p>Example results: <code>{ id: 1, name: "John" }</code></p>
<h4>selectValue(sql, ...bindValues)</h4>
<p>Get only the first column of the first row.</p>
<pre class="prettyprint source lang-js"><code>const { results: value } = await db.selectValue(sql);
</code></pre>
<p>Example results: <code>"John"</code></p>
<h4>selectHash(sql, ...bindValues)</h4>
<p>Get an Object with column-value pairs.</p>
<pre class="prettyprint source lang-js"><code>const { results: hash } = await db.selectHash(sql);
</code></pre>
<p>Example results: <code>{ "1": "John", "2": "Jane" }</code></p>
<h4>selectList(sql, ...bindValues)</h4>
<p>Get an Array of values for the first column of the first row.</p>
<pre class="prettyprint source lang-js"><code>const { results: list } = await db.selectList(sql);
</code></pre>
<p>Example results: <code>["John", "Jane"]</code></p>
<h4>selectExists(sql, ...bindValues)</h4>
<p>Return true if query returns any rows.</p>
<pre class="prettyprint source lang-js"><code>const { results: doesExist } = await db.selectExists(sql);
</code></pre>
<p>Example results: <code>true</code></p>
<h4>selectIndexed(indexColumn, sql, ...bindValues)</h4>
<p>Return an Object where every result row is indexed by the given field.</p>
<pre class="prettyprint source lang-js"><code>const { results: usersById } = await db.selectIndexed('id', sql);
</code></pre>
<p>Example results:</p>
<pre class="prettyprint source lang-js"><code>results = {
"1": { id: 1, name: "John" },
"2": { id: 2, name: "Jane" },
}
</code></pre>
<h4>selectGrouped(groupColumn, sql, ...bindValues)</h4>
<p>Return an Object where every result row is indexed by the given field.</p>
<pre class="prettyprint source lang-js"><code>const { results: usersGroupedByOrg } = await db.selectGrouped('org', sql);
</code></pre>
<p>Example results:</p>
<pre class="prettyprint source lang-js"><code>results = {
"Marketing": [
{ id: 1, name: "John", org: "Marketing" },
{ id: 2, name: "Jane", org: "Marketing" },
],
"Finance": [
{ id: 3, name: "Jose", org: "Finance" },
],
}
</code></pre>
<h4>selectOrCreate(table, criteria[, newValues])</h4>
<p>Select a record or create a new record. Good when normalizing data that is frequently referenced.</p>
<p>For example, say I have a table <code>hits</code> with a column <code>url_id</code> and a table <code>urls</code> with columns <code>id</code> and <code>url</code>.</p>
<p>I want to add a new hit record with a given URL. You might write this:</p>
<pre class="prettyprint source lang-js"><code>const newHit = {
date: '2021-10-15 17:43:24',
url: 'https://example.com',
}
const { results } = await db.selectOrCreate('urls', { url: newHit.url });
await db.insert('hits', {
date: newHit.date,
url_id: results.id,
});
</code></pre>
<h3>Useful Query Options</h3>
<p>SQL can actually be an Object with options.</p>
<pre class="prettyprint source lang-js"><code>const options = {
sql: `
SELECT users.*, avatars.*
FROM users
INNER JOIN avatars ON avatars.user_id = users.id
WHERE users.is_active = ?
`,
// kill query if not completed within 30 seconds
timeout: 30000,
// return records with keys `users` and `avatars` with their own fields nested underneath
nestTables: true,
// you can also bind values here using question marks
values: [true],
};
const { results } = await db.select(options);
</code></pre>
<h4>nestTables Example</h4>
<p>Given a query of:</p>
<pre class="prettyprint source lang-sql"><code>SELECT users.*, avatars.*
FROM users
INNER JOIN avatars ON avatars.user_id = users.id
WHERE users.is_active = ?
</code></pre>
<p>nesting tables will return a data structure such as:</p>
<pre class="prettyprint source lang-js"><code>results = [
{
users: {
id: 1,
name: 'John Doe',
is_active: true,
},
avatars: {
id: 101,
user_id: 1,
url: 'http://example.com/john.png'
}
},
{
users: {
id: 2,
name: 'Jane Doe',
is_active: true,
},
avatars: {
id: 102,
user_id: 2,
url: 'http://example.com/jane.png'
}
}
]
</code></pre>
<h4>selectFrom(table, fields, values)</h4>
<p>Build and run a simple select statement.</p>
<pre class="prettyprint source lang-js"><code>const { results } = await db.selectFrom('users', ['fname','lname'], {
'id >': 5,
is_active: true,
department_id: [1,2],
});
</code></pre>
<h4>insert(sql, ...bindVars)</h4>
<p>Run an insert statement; return the id of the new record if applicable.</p>
<pre class="prettyprint source lang-js"><code>const { insertId } = await db.insert("INSERT INTO users SET name='John', email='john@example.com'");
</code></pre>
<h4>insertInto(table, values)</h4>
<p>Build and run an insert statement; return the id of the new record if applicable.</p>
<pre class="prettyprint source lang-js"><code>const { insertId } = await db.insertInto('users', {
name: 'John',
email: 'john@example.com',
});
</code></pre>
<h4>insertExtended(table, rows)</h4>
<p>Build and run an extended insert statement; return the id of the last record if applicable.</p>
<pre class="prettyprint source lang-js"><code>const { insertId } = await db.insertExtended('users', [
{ name: 'John', email: 'john@example.com' },
{ name: 'Jane', email: 'jane@example.com' },
]);
</code></pre>
<h4>insertIntoOnDuplicateKeyUpdate(table, insert, update)</h4>
<p>Build and run an insert statement; return the id of the new record if applicable.</p>
<pre class="prettyprint source lang-js"><code>const { insertId, affectedRows } = await db.insertIntoOnDuplicateKeyUpdate(
'users',
{
sso_ref: 'A123456',
name: 'Jane Doe',
created_at: '2020-02-02',
},
{
name: 'Jane Doe Carter',
modified_at: '2020-02-02',
}
);
</code></pre>
<h4>update(sql, ...bindValues)</h4>
<p>Run an update statement; return the number of affected rows.</p>
<pre class="prettyprint source lang-js"><code>const { affectedRows } = await db.update(
"UPDATE users SET name = ? WHERE id = ?",
'Jane Doe Carter',
5
);
</code></pre>
<h4>updateTable(table, set, where)</h4>
<p>Build and run an update statement; return the number of affected rows.</p>
<pre class="prettyprint source lang-js"><code>const { affectedRows } = await db.updateTable(
'users',
{ name: 'Jane Doe Carter' },
{ id: 5 }
);
</code></pre>
<h4>delete(sql, ...bindValues)</h4>
<p>Run a delete statement; return the number of affected rows.</p>
<pre class="prettyprint source lang-js"><code>const { affectedRows } = await db.delete(
"DELETE FROM users WHERE id = ? LIMIT 1",
5
);
</code></pre>
<h4>deleteFrom(table, where, limit)</h4>
<p>Build and run a delete statement; return the number of affected rows.</p>
<pre class="prettyprint source lang-js"><code>const { affectedRows } = await db.deleteFrom('users', { id: 5 }, 1);
</code></pre>
<h4>query(sql, ...bindValues)</h4>
<p>Run any type of statement.</p>
<pre class="prettyprint source lang-js"><code>const { query, results, fields } = await db.query(
'SELECT * FROM users'
);
</code></pre>
<h4>multiQuery(sql, ...bindValues)</h4>
<p>Run multiple statements delimited by semicolon.</p>
<pre class="prettyprint source lang-js"><code>const { query, results, fields } = await db.query(
'SELECT * FROM users; SELECT * FROM tags'
);
</code></pre>
<h3>Solutions to Common Problems</h3>
<h4>Connection is in closed state</h4>
<p><code>Error: Can't add new command when connection is in closed state</code></p>
<p>Make sure you use <code>await</code> your results before closing your connection.</p>
<h4>ECONNRESET or error event</h4>
<p><code>Error: read ECONNRESET</code> or <code>Emitted 'error' event on Client instance</code></p>
<p>Your SSH connection may have timed out. To keep connection alive, you
can send keepalive packets.</p>
<pre class="prettyprint source lang-js"><code>const sshConfig = {
// ...
// How often (in milliseconds) to send SSH-level keepalive packets to the server (in a similar way as OpenSSH's ServerAliveInterval config option). Set to 0 to disable. Default: 0
keepaliveInterval: 30,
// How many consecutive, unanswered SSH-level keepalive packets that can be sent to the server before disconnection (similar to OpenSSH's ServerAliveCountMax config option). Default: 3
keepaliveCountMax: 120,
}
const db = new Db(mysqlConfig, sshConfig);
</code></pre>
<h2>Select</h2>
<p>A Select object represents a SQL SELECT query and allows dynamically adding
clauses including JOIN, WHERE, ORDER BY, LIMIT, OFFSET.</p>
<h3>Select.parse()</h3>
<p>The easiest way to define a base query is to use <code>Select.parse(sql)</code> and then
add criteria as needed.</p>
<pre class="prettyprint source lang-js"><code>const { Select } = require('sharp-db');
const query = Select.parse(`
SELECT u.id, u.fname, u.lname, u.email, p.phone
FROM users
LEFT JOIN phone_numbers p ON p.user_id = u.id
AND p.type = 'main'
WHERE u.is_active = 1
`);
if (email) {
query.where('u.email', email);
}
if (areaCode) {
query.where('p.phone', 'LIKE ?%', areaCode);
}
query.sort(sortField);
query.limit(limitTo);
</code></pre>
<p>You can also define binding in the base query itself.</p>
<pre class="prettyprint source lang-js"><code>const query = Select.parse(`
SELECT u.id, u.fname, u.lname, u.email, a.city, a.zip
FROM users
LEFT JOIN addresses a ON a.user_id = u.id
WHERE a.state = :state
`);
query.bind('state', state);
</code></pre>
<p>And you can bind multiple values at once.</p>
<pre class="prettyprint source lang-js"><code>const query = Select.parse(`
SELECT u.id, u.fname, u.lname, u.email, a.city, a.zip
FROM users
LEFT JOIN addresses a ON a.user_id = u.id
WHERE a.state = :state
AND a.city IN (:city)
`);
query.bind({ state, city });
</code></pre>
<h3>Building the Query</h3>
<p>The following are the most common methods for building queries.</p>
<ul>
<li><code>query.columns(columnNames)</code> - Add column names to fetch</li>
<li><code>query.column(columnName)</code> - Add a column name to fetch</li>
<li><code>query.table(tableName)</code> - Specify the table in the FROM clause</li>
<li><code>query.from(tableName)</code> - Same as above</li>
<li><code>query.innerJoin(expression)</code> - Add an INNER JOIN expression</li>
<li><code>query.leftJoin(expression)</code> - Add a LEFT JOIN expression</li>
<li><code>query.fullJoin(expression)</code> - Add a FULL JOIN expression</li>
<li><code>query.rightJoin(expression)</code> - Add a RIGHT JOIN expression</li>
<li><code>query.crossJoin(expression)</code> - Add a CROSS JOIN expression</li>
<li><code>query.leftOuterJoin(expression)</code> - Add a LEFT OUTER JOIN expression</li>
<li><code>query.fullOuterJoin(expression)</code> - Add a FULL OUTER JOIN expression</li>
<li><code>query.rightOuterJoin(expression)</code> - Add a RIGHT OUTER JOIN expression</li>
<li><code>query.groupBy(column)</code> - Group by a column or expression</li>
<li><code>query.where(column, operator, value)</code> - Require column satisfy operator</li>
<li><code>query.where(column, value)</code> - Require column equal a value</li>
<li><code>query.where(expression)</code> - Add an arbitrary WHERE expression</li>
<li><code>query.where(columnValuePairs)</code> - Add multiple conditions</li>
<li><code>query.whereBetween(column, twoValueArray)</code> - Require value BETWEEN, < or ></li>
<li><code>query.orWhere(conditions)</code> - Specify multiple <code>where()</code>s joined by <code>OR</code></li>
<li><code>query.having(column, operator, value)</code> - Having column satisfy operator</li>
<li><code>query.having(column, value)</code> - Having column equal value</li>
<li><code>query.having(column, value)</code> - Having column equal value</li>
<li><code>query.having(expression)</code> - Having an arbitrary expression</li>
<li><code>query.orHaving(expressions)</code> - Multiple <code>having()</code>s joined by OR</li>
<li><code>query.orderBy(column)</code> - Add ORDER BY clause</li>
<li><code>query.sortField(column, mapNames)</code> - Add ORDER BY clause with mapNames</li>
<li><code>query.limit(num)</code> - Limit by the given number</li>
<li><code>query.offset(num)</code> - Specify an offset</li>
<li><code>query.page(num)</code> - Automatically calculate offset based on limit and page</li>
</ul>
<h3>Fetching Data</h3>
<p>The methods to fetch data mirror those of Db.</p>
<ul>
<li><code>query.fetch()</code> - equivalent to <code>db.select()</code></li>
<li><code>query.fetchFirst()</code> - equivalent to <code>db.selectFirst()</code></li>
<li><code>query.fetchHashed()</code> - equivalent to <code>db.selectHashed()</code></li>
<li><code>query.fetchList()</code> - equivalent to <code>db.selectList()</code></li>
<li><code>query.fetchValue()</code> - equivalent to <code>db.selectValue()</code></li>
<li><code>query.fetchIndexed(byField)</code> - equivalent to <code>db.selectIndexed(byField)</code></li>
<li><code>query.fetchGrouped(byField)</code> - equivalent to <code>db.selectGrouped(byField)</code></li>
</ul>
<h3>Counting Results</h3>
<p>One powerful feature of Select is that it can construct a count query to fetch
the number of results that would have been returned if there were no LIMIT.</p>
<pre class="prettyprint source lang-js"><code>const query = Select.parse('SELECT id, name FROM users LIMIT 5');
const { results: users } = await query.fetch();
const { results: count } = await query.foundRows();
// will run the following query:
// SELECT COUNT(*) AS foundRows FROM users
</code></pre>
<h3>Specifying the <code>Db</code> Instance to Use</h3>
<p>There are three ways to specify the <code>Db</code> instance to fetch data with:</p>
<ol>
<li><code>query = Select.parse(sql, db)</code></li>
<li><code>query = new Select(db)</code></li>
<li><code>query.db = db</code></li>
</ol>
<p>If no instance is specified, <code>Db.factory()</code> is used.</p>
<h3>Dependent Data</h3>
<p>A Select object can splice in sibling or child data for each row.</p>
<h4>withSiblingData(propertyName, siblingSql)</h4>
<p>Example:</p>
<pre class="prettyprint source lang-js"><code>const query = Select.parse('SELECT id, name FROM users');
query.withSiblingData(
'homeAddress',
Select.parse(`
SELECT * FROM addresses
WHERE addresses.user_id IN(:id)
AND addresses.type = 'home'
AND addresses.deleted_at IS NULL
`),
);
query.withSiblingData(
'workAddress',
Select.parse(`
SELECT * FROM addresses
WHERE addresses.user_id IN(:id)
AND addresses.type = 'work'
AND addresses.deleted_at IS NULL
`),
);
const { results } = await query.fetch();
</code></pre>
<p>...and <code>results</code> for example may equal:</p>
<pre class="prettyprint source lang-js"><code>results = [
{
id: 1,
name: 'John',
homeAddress: {
id: 11,
type: 'home',
is_active: 1,
user_id: 1,
street: '123 Any St.',
city: 'Any Town',
state: 'CA'
},
workAddress: {
id: 12,
type: 'work',
is_active: 1,
user_id: 1,
street: '123 Commerce Dr.',
city: 'Any Town',
state: 'CA',
},
},
{
id: 2,
name: 'Jane',
// rows without sibling data will be null
homeAddress: null,
workAddress: {
id: 12,
type: 'work',
is_active: 1,
user_id: 2,
street: '123 Tower Blvd.',
city: 'Any Town',
state: 'CA',
},
}
]
</code></pre>
<h4>withChildData(propertyName, childSql)</h4>
<p>Example:</p>
<pre class="prettyprint source lang-js"><code>const query = Select.parse('SELECT id, headline, published_by FROM posts');
query.withChildData(
'theComments',
Select.parse('SELECT * FROM comments WHERE comments.post_id IN(:id)')
);
query.withChildData(
'theTags',
Select.parse(`
SELECT posts_tags.post_id, tags.* FROM tags
INNER JOIN posts_tags ON posts_tags.tag_id = tags.id
WHERE posts_tags.post_id IN(:id)
`)
);
query.withSiblingData(
'thePublisher',
Select.parse('SELECT id, name FROM users WHERE user_id IN(:published_by)')
);
const { results } = await query.fetch();
</code></pre>
<p>...and <code>results</code> for example may equal:</p>
<pre class="prettyprint source lang-js"><code>results = [
{
id: 1,
headline: 'Turmoil in China',
published_by: 1001,
theComments: [
{
id: 11,
post_id: 1,
user_id: 101,
text: 'Sad to hear',
},
{
id: 12,
post_id: 1,
user_id: 102,
text: 'Hope it works out',
},
],
theTags: [
{
id: 101,
post_id: 1,
name: 'China',
},
{
id: