Skip to content

denull/minusql

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

26 Commits
 
 
 
 
 
 
 
 

Repository files navigation

MinuSQL

MinuSQL (pronounced minuscule) is a lightweight, flexible SQL query builder and database abstraction layer for Node.js that supports both MySQL and PostgreSQL databases. It provides a minimalistic API for building SQL queries while maintaining type safety and security.

Features

  • Support for both MySQL and PostgreSQL
  • No dependencies
  • Fluent query builder interface
  • Automatic case conversion (snake_case ↔ camelCase)
  • Parameterized queries for security
  • Transaction support
  • Flexible result mapping
  • Built-in support for common SQL operations (SELECT, INSERT, UPDATE, DELETE)
  • JOIN operations with various join types
  • Conflict resolution for INSERT operations
  • Type-safe query building
  • EXPLAIN query support

Installation

npm install minusql

Quick Start

This library acts as a wrapper around database drivers. To use it, you first create an instance of either a MySQL or a Postgres class, passing the database client (or a pool) from the mysql or pg libraries:

MySQL

const mysql = require('mysql');
const { MySQL } = require('minusql');

const pool = mysql.createPool({
  host: 'localhost',
  user: 'root',
  password: 'password',
  database: 'mydb'
});

const db = new MySQL(pool);

// Query example
const user = await db.users.selectOne({ id: 1 });

PostgreSQL

const { Pool } = require('pg');
const { Postgres } = require('minusql');

const pool = new Pool({
  host: 'localhost',
  user: 'postgres',
  password: 'password',
  database: 'mydb'
});

const db = new Postgres(pool);

// Query example
const user = await db.users.selectOne({ id: 1 });

By default, MinuSQL automatically converts all identifiers to snake_case when building queries, and back to camelCase when handling results. You can disable this behavior by passing { convertCase: false } to the constructor.

API Documentation

To perform queries on specific tables, simply access them as properties of the db instance: for example, db.users represents the users table. To join multiple tables, call db.join() (see below).

You can also explicitly call db.from('users') to specify a table.

To perform raw queries, use db.exec(query, params).

Query Building

To perform CRUD operations on a table, use one of the following methods:

  • db.table.select(where?, options?)
  • db.table.insert(rows, options?)
  • db.table.update(update, where?)
  • db.table.delete(where?)

All of these methods return a Query object, which will be executed as soon as you await it (or call exec() on it). You can also use the Query object to inspect the built query (the text field) or call explain() to construct an EXPLAIN query from it.

There are also two convenient aliases for common select types:

  • selectAll(options?) is equivalent to select(null, options)
  • selectOne(where?, options?) is equivalent to select(where, options).one()

SELECT Queries

// Select everything
const results = await db.users.selectAll();

// Simple query
const results = await db.users.select({ id: 1, name: 'John' });

// Complex query with operators and parameter substitution
const ageMin = 18;
const permissionList = ['edit', 'delete'];
const users = await db.users.select(['and',
  ['>', Symbol('age'), {$: ageMin}],
  ['=', Symbol('status'), 'active'],
  ['or',
    { role: 'admin' },
    { permissions: ['in', {$: permissionList}] },
  ],
]);

// With specific fields
const results = await db.users
  .select(null, { fields: ['id', 'name'] });

// With ordering and limits
const results = await db.users
  .select(null, { 
    order: 'createdAt DESC', // or [[Symbol('createdAt'), 'DESC']]
    limit: 10,
    offset: 0
  });

The select(where?, options?) method takes two arguments: the first defines the query condition and the second sets additional options.

You can use a raw query string for where, but it's strongly discouraged as it won't be parameterized or escaped. Almost any condition can be expressed in a structured form instead.

Structured condition is defined as a recursive expression:

  • If it's an array, its first element should be a SQL function name (like COALESCE) or an operator (like < or AND). All following elements are its arguments (which are parsed as nested expressions).
  • If it's an object with a $ field, it contains a parameter which will be passed along with the query. This prevents SQL injections and improves performance, and is recommended for all user-supplied data. You can also use the type field to add an explicit type cast.
  • If it's an object, its key-value pairs are converted to expressions in the form of key = value and joined using the AND operator. This is the same as ['AND', ['=', key1, value1], ['=', key2, value2], ...], just less verbose. If a value is itself an array, it's interpreted as if the key was inserted after the first element: { x: ['>', y] } is the same as ['>', Symbol('x'), y]. Keys are escaped as identifiers, and values may contain nested expressions.
  • If it's a Symbol instance, it refers to a database column and is therefore escaped as an identifier.
  • Otherwise (if it's a primitive value, like a number or string), it's escaped and inserted into the query.

There are a few special behaviors for specific SQL operators:

  • ['in', Symbol('x'), [1, 2, 3]] is converted to "x" IN (1, 2, 3)
  • ['not in', Symbol('x'), [1, 2, 3]] is converted to "x" NOT IN (1, 2, 3)
  • ['between', Symbol('x'), 1, 2] is converted to "x" BETWEEN 1 AND 2
  • ['not between', Symbol('x'), 1, 2] is converted to "x" NOT BETWEEN 1 AND 2
  • ['type', Symbol('x'), 'json'] is converted to json "x" (the type is NOT escaped)
  • ['cast', Symbol('x'), 'json'] is converted to "x"::json (the type is NOT escaped)
  • ['extract', Symbol('x'), 'month'] is converted to EXTRACT(month FROM x) (note the order change; also the last argument is NOT escaped)
  • ['case', [cond1, then1], [cond2, then2], [default]] is converted to CASE WHEN cond1 THEN then1 WHEN cond2 THEN then2 ELSE default END

Supported options are (all optional):

  • fields: a raw string or an array of columns to select
  • group: a raw string or an array of expressions to use in the GROUP BY clause
  • having: a raw string or structured condition to use in the HAVING clause
  • order: a raw string or an array of pairs [expression, 'ASC' | 'DESC'] to use in the ORDER BY clause
  • limit: a number to use in the LIMIT clause
  • offset: a number to use in the OFFSET clause

By default, the resulting query returns an array of rows. To re-map it to more suitable data structures, see "Result Mapping" below.

INSERT Queries

// Single insert
await db.users.insert({
  name: 'John', // Values will be parametrized by default (you can change this behavior by supplying "tranform" option)
  age: 30
});

// Returning inserted ID
const result = await db.users.insert({
  name: 'John',
  age:  30,
}, { returnId: true }); // (PostgreSQL only, MySQL will always add insertId to output)
// result will contain the ID of the inserted row

// Batch insert with manually parametrized values
await db.users.insert(usersToInsert.map(user => ({
  name: {$: user.name},
  age:  {$: user.age},
})), { transform: false });

// Upsert (handling conflicts)
await db.users.insert({
  id:         888352,
  name:       'John',
  age:        30,
  revision:   0,
  joinedAt:   Date.now() / 1000,
}, {
  transform: {
    joinedAt: 'timestamp', // Unixtime can be easily converted to timestamps
  },
  unique: ['id'], // Needed only for PostgreSQL (upserts on MySQL will work without it)
  conflict: {
    name:     /update/, // Update name on conflict
    age:      /max/,    // Update to largest of old and new value
    revision: ['+', Symbol('revision'), 1], // Expressions are supported here as well
    joinedAt: /fill/,   // Update only if was null
  },
});

insert accepts two parameters: rows to insert (or a single row) and options.

Supported options:

  • fields: an array of columns; if omitted, the first element's keys will be used
  • unique (PostgreSQL only): for upserts, you need to specify a list of unique fields
  • conflict: for upserts, describes the conflict resolution strategy (see below)
  • returnId (PostgreSQL only): which column to return after insertion (set to true to return column "id"). MySQL will always return id of the inserted row (along with some other information) as a insertId field in the resulting row.

Conflict resolution strategy is either false (ignore all conflicts) or an object. Its keys correspond to columns that should be updated on conflict, and values are structured expressions to set them to.

For convenience, you can pass the following predefined RegExp patterns as aliases for common strategies:

  • /update/: update to the new value on conflict
  • /fill/: only update if the old value is NULL
  • /inc/: increment old value by 1
  • /dec/: decrement old value by 1
  • /add/: add new value to the old one
  • /sub/: subtract new value from the old one
  • /max/: select the maximum out of old value and the new one
  • /min/: select the minimum out of old value and the new one

UPDATE Queries

// Basic update with a simple where condition
await db.users.update(
  { age: 31 },
  { name: 'John' }
);

// Update with a complex where condition
await db.users.update(
  { status: 'inactive', lastSeen: new Date() },
  ['and', 
    ['<', Symbol('lastLogin'), {$: oneMonthAgo}],
    ['=', Symbol('status'), 'active']
  ]
);

// Update with expressions
await db.users.update(
  { 
    loginCount: ['+', Symbol('loginCount'), 1],
    status: 'active'
  },
  { id: 42 }
);

// Update all rows (be careful!)
await db.users.update(
  { isArchived: true },
  null
);

The update(update, where?) method takes two parameters:

  1. update: An object where keys are column names and values are either direct values or expressions
  2. where: A condition to determine which rows to update (same format as in select); if null, all rows will be updated

The update values can be:

  • Simple values (strings, numbers, booleans, etc.)
  • Expressions using the same syntax as in where conditions
  • SQL functions and operators in array format

DELETE Queries

// Delete with a simple condition
await db.users.delete({ name: 'John' });

// Delete with a complex condition
await db.users.delete(['and',
  ['<', Symbol('lastLogin'), {$: sixMonthsAgo}],
  ['=', Symbol('status'), 'inactive']
]);

// Delete all rows (use with caution!)
await db.users.delete(null);

The delete(where?) method takes a single parameter:

  • where: A condition to determine which rows to delete (same format as in select); if null or omitted, all rows will be deleted

Result Mapping

MinuSQL provides various methods for transforming query results into different data structures:

// Get an array of results (default behavior)
const users = await db.users.select().toArray();

// Get an array of single column's values
const names = await db.users.select().toArray('name');


// Get just the first result or null if none found
const user = await db.users.select({ id: 1 }).one();
// Equivalent to using selectOne()
const user = await db.users.selectOne({ id: 1 });

// Get first result with transformation
const userName = await db.users.select({ id: 1 }).one('name');

// Map results to an object using a key
const usersById = await db.users.select().toObject('id');
// Result: { '1': {id: 1, name: 'John'}, '2': {id: 2, name: 'Jane'}, ... }

// Map to object with specific value
const nameById = await db.users.select().toObject('id', 'name');
// Result: { '1': 'John', '2': 'Jane', ... }

// Map to object using custom key function
const usersByFullName = await db.users.select().toObject(
  user => `${user.firstName} ${user.lastName}`
);

// Group into arrays by a key
const usersByRole = await db.users.select().toObjectArray('role', 'name');
// Result: { 'admin': ['John', 'Jane'], 'user': ['Bob', 'Alice'], ... }

// Map instance
const userMap = await db.users.select().toMap('id');
// Result: Map { 1 => {id: 1, name: 'John'}, 2 => {id: 2, name: 'Jane'}, ... }

// Map with specific value
const nameMap = await db.users.select().toMap('id', 'name');
// Result: Map { 1 => 'John', 2 => 'Jane', ... }

// Group into Map of arrays
const roleMap = await db.users.select().toMapArray('role', 'name');
// Result: Map { 'admin' => ['John', 'Jane'], 'user' => ['Bob', 'Alice'], ... }

// Extract values to a Set
const allRoles = await db.users.select().toSet('role');
// Result: Set { 'admin', 'user', 'guest', ... }

// Process each row with a function
await db.users.select().forEach(user => {
  console.log(`User ${user.name} is ${user.age} years old`);
});

// Map to class instances
class User {
  static fromRow(row) {
    const user = new User();
    user.id = row.id;
    user.name = row.name;
    return user;
  }
  
  greet() {
    return `Hello, ${this.name}!`;
  }
}

const users = await db.users.select().toArray(User);
console.log(users[0].greet()); // "Hello, John!"

The Query object provides these mapping methods:

  • one(value?): Returns the first result or null if none found, optionally transformed
  • toArray(value?): Returns results as an array, optionally transforming each row using the field parameter
  • toObject(key, value?): Maps results to an object using the specified key, optionally transforming values
  • toObjectArray(key, value?): Groups results into arrays by key
  • toMap(key, value?): Maps results to a Map
  • toMapArray(key, value?): Groups results into arrays in a Map by key
  • toSet(value): Extracts unique values from the specified field into a Set
  • forEach(fn): Executes a function for each result row

The transformation parameter (value) can be:

  • A string: extracts that property from each row
  • A function: called with (row, index, allRows) for custom transformations
  • A class: tries to instantiate objects from rows, using fromRow() static method if available
  • An object: for extracting/transforming multiple properties (recursively)
  • An array: for extracting multiple properties as an array (recursively)

key parameter supports a subset of those types:

  • A string: property to be used as key
  • A function: called with (row, index, allRows) and result is used as key
  • An array: elements will be joined with _ and used as key

Note that by default, all result keys are automatically converted from snake_case to camelCase unless convertCase: false was set.

Transactions

MinuSQL provides a simple way to work with transactions:

// Basic transaction
await db.begin(async (tx) => {
  // The tx object is a transaction-specific database instance
  await tx.users.insert({ name: 'John' });
  await tx.profiles.insert({ userId: 1, bio: 'Hello' });
  
  // If any query fails, the transaction will be automatically rolled back
  // If all succeed, it will be committed automatically
});

JOIN Operations

const results = await db.join([
  { table: 'users', as: 'u' },
  { table: 'profiles', as: 'p', on: { 'u.id': Symbol('p.userId') } },
]).selectAll();

// Same as
const results = await db.users
  .join('profiles p', { 'users.id': Symbol('p.userId') })
  .selectAll();

To join multiple tables, call db.join with the array of objects containing following fields:

  • table: name of the table or another subquery
  • as (Optional): alias to be used in AS clause
  • join (Optional): join type to be used in JOIN clause (if omitted, defaults to LEFT)
  • on: any expression in the structured format to be used in ON clause

Instead of object with join description, you can also use raw string, but it's discouraged.

Alternatively, you can call join directly on a table: db.users.join({ table: 'profiles', ... }). As a shorthand, you can also pass table name as the first argument and join condition as second.

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

License

MIT

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published