Skip to content

mnpenner/mysql3

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

mysql3

A small OOP wrapper around mariadb.

Installation

yarn add mysql3
# or
npm i mysql3
# or
pnpm add mysql3

Usage

import {createPool, sql} from 'mysql3';

async function main(pool: ConnectionPool) {
    const result = await pool.value(sql`select now()`)
    console.log(result)
}

createPool({
    host: process.env.DB_HOST,
    user: process.env.DB_USER,
    password: process.env.DB_PASSWORD,
    database: process.env.DB_NAME,
}).then(pool => main(pool).finally(pool.close)).catch(err => {
    console.error(err)
    process.exit(1)
})

See tests/test.ts for more.

API

function createPool(config: MariaDB.PoolConfig): Promise<ConnectionPool>

Connections

class ConnectionPool {
    constructor(pool: MariaDB.Pool);
    getConnection(): Promise<PoolConnection>;
    private _fwd;
    query: <TRecord = Record<string, DefaultValueType>>(query: QueryParam) => Promise<QueryResult<TRecord>>;
    exec: (query: QueryParam) => Promise<MariaDB.UpsertResult>;
    row: <TRecord extends object = Record<string, DefaultValueType>>(query: QueryParam) => Promise<TRecord | null>;
    col: <TValue = DefaultValueType>(query: SqlFrag) => Promise<TValue[]>;
    value: <TValue = DefaultValueType>(query: SqlFrag) => Promise<TValue | null>;
    exists: (query: SqlFrag) => Promise<boolean>;
    count: (query: SqlFrag) => Promise<number>;
    stream<TRecord extends object = DefaultRecordType>(query: SqlFrag): AsyncGenerator<TRecord, unknown, undefined>;
    close: () => Promise<void>;
    transaction<TReturn>(callback: (conn: PoolConnection) => Promise<TReturn>): Promise<TReturn>;
    transaction<TUnionResults = DefaultRecordType>(callback: SqlFrag[]): Promise<QueryResult<TUnionResults>[]>;
    get activeConnections(): number;
    get totalConnections(): number;
    get idleConnections(): number;
    get taskQueueSize(): number;
}
class PoolConnection {
    constructor(conn: MariaDB.PoolConnection);
    query<TRecord = DefaultRecordType>(query: QueryParam): Promise<QueryResult<TRecord>>;
    exec: ((...args: Parameters<typeof PoolConnection.prototype.query>) => Promise<MariaDB.UpsertResult>);
    row<TRecord extends object = DefaultRecordType>(query: QueryParam): Promise<TRecord | null>;
    col<TValue = DefaultValueType>(query: SqlFrag): Promise<TValue[]>;
    value<TValue = DefaultValueType>(query: SqlFrag): Promise<TValue | null>;
    exists(query: SqlFrag): Promise<boolean>;
    count(query: SqlFrag): Promise<number>;
    stream<TRecord extends object = DefaultRecordType>(query: SqlFrag): AsyncGenerator<TRecord, unknown, undefined>;
    release: () => void;
    beginTransaction: () => Promise<void>;
    commit: () => Promise<void>;
    rollback: () => Promise<void>;
    ping: () => Promise<void>;
    changeUser: (options?: MariaDB.UserConnectionConfig | undefined) => Promise<void>;
    get isValid(): boolean;
    close: () => Promise<void>;
    destroy: () => void;
    serverVersion: () => string;
}

SQL Strings

sql(strings: TemplateStringsArray, ...values: Value[]): SqlFrag

namespace sql {
    function set<T extends TableSchema<T>>(fields: InsertData<T>): SqlFrag;
    function insert<T extends TableSchema<T>>(table: TableId, data: InsertData<T>, options?: InsertOptions): SqlFrag;
    function alias(fields: Record<string, ColumnId> | Array<[column: ColumnId, alias: string]>): SqlFrag;
    function raw(sqlString: string | SqlFrag): SqlFrag;
    function point(x: number, y: number): SqlFrag;
    function id(id: Id): SqlFrag;
    function cols(...columns: Array<ColumnId>): SqlFrag;
    function values(values: Value[][]): SqlFrag;
}

enum DuplicateKey {
    IGNORE = "ignore",
    UPDATE = "update"
}

Convenience Methods

interface User {
    id: number
    name: string
}
const user: User = await conn.row<User>(sql`select id,name from users where username='mpen'`)
const databaseNames: string[] = await conn.col<string>(sql`show databases`)
const imageCount: bigint = await pool.value<bigint>(sql`select count(*) from images`)
const imageCount: number = await pool.count(sql`select * from images`)
const userExists: boolean = await pool.exists(sql`select * from users where username='mpen'`)

Transactions

const userId = ouid();
const profileId = ouid();
await pool.transaction([
    sql.insert('users', {id: userId, username: 'mpen'}),
    sql.insert('profiles', {id: profileId, userId, name: "Mark"}),
])

OR

const [userId,profileId] = await pool.transaction(async conn => {
    const user = await conn.exec(sql.insert('users', {username: 'mpen'}))
    const profile = await conn.exec(sql.insert('profiles', {userId: user.insertId, name: 'Mark'}))
    return [user.insertId,profile.insertId]
})

Streaming Results

If you need to process many rows, it's better to stream the results than to try to load the entire result set into memory via query():

const imgQuery = sql`select * from images`
const count = await pool.count(imgQuery)
let i=0;
for await(const img of pool.stream<{starred:boolean|null,date_taken_local:Date|null}>(imgQuery)) {
    console.log(`${++i}/${count}`,img.date_taken_local)
}

More Examples

console.dir(await pool.query(sql`select ${sql.as({mediaType: 'media_type', width: 'width', height: 'height', aspectRatio: 'aspect_ratio'})}
                                 from ${sql.tbl(['imagegather', 'image_files'])}
                                 limit 2`), {depth: 1})
[
  {
    mediaType: 'image/jpeg',
    width: 296,
    height: 222,
    aspectRatio: 1.33333
  },
  {
    mediaType: 'image/jpeg',
    width: 3264,
    height: 2448,
    aspectRatio: 1.33333
  },
  meta: [ [ColumnDef], [ColumnDef], [ColumnDef], [ColumnDef] ]
]
console.dir(await pool.query({
    sql: sql`select media_type,width,height from image_files limit 2`,
    rowsAsArray: true,
}), {depth: 1})
[
  [ 'image/jpeg', 296, 222 ],
  [ 'image/jpeg', 3264, 2448 ],
  meta: [ [ColumnDef], [ColumnDef], [ColumnDef] ]
]
console.log(sql`insert into foo set ${sql.set({bar:1,baz:"qu'ux"})}`)
SqlFrag { sql: "insert into foo set `bar`=1, `baz`='qu''ux'" }
console.log(sql`insert into foo(bar, baz)
                values ${sql.values([
                    [1, "qu'ux"],
                    [2, null],
                ])}`.toSqlString())
insert into foo(bar,baz) values (1,'qu''ux'), (2,NULL)

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published