176 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			176 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
// @ts-check
 | 
						|
 | 
						|
const {db} = require("../passthrough")
 | 
						|
const U = require("./orm-defs")
 | 
						|
 | 
						|
/**
 | 
						|
 * @template {keyof U.Models} Table
 | 
						|
 * @template {keyof U.Models[Table]} Col
 | 
						|
 * @param {Table} table
 | 
						|
 * @param {Col[] | Col} cols
 | 
						|
 * @param {Partial<U.Models[Table]>} where
 | 
						|
 * @param {string} [e]
 | 
						|
 */
 | 
						|
function select(table, cols, where = {}, e = "") {
 | 
						|
	if (!Array.isArray(cols)) cols = [cols]
 | 
						|
	const parameters = []
 | 
						|
	const wheres = Object.entries(where).map(([col, value]) => {
 | 
						|
		parameters.push(value)
 | 
						|
		return `"${col}" = ?`
 | 
						|
	})
 | 
						|
	const whereString = wheres.length ? " WHERE " + wheres.join(" AND ") : ""
 | 
						|
	/** @type {U.Prepared<Pick<U.Models[Table], Col>>} */
 | 
						|
	const prepared = db.prepare(`SELECT ${cols.map(k => `"${String(k)}"`).join(", ")} FROM ${table} ${whereString} ${e}`)
 | 
						|
	prepared.get = prepared.get.bind(prepared, ...parameters)
 | 
						|
	prepared.all = prepared.all.bind(prepared, ...parameters)
 | 
						|
	return prepared
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * @template {keyof U.Models} Table
 | 
						|
 * @template {keyof U.Merge<U.Models[Table]>} Col
 | 
						|
 */
 | 
						|
class From {
 | 
						|
	/**
 | 
						|
	 * @param {Table} table
 | 
						|
	 */
 | 
						|
	constructor(table) {
 | 
						|
		/** @private @type {Table[]} */
 | 
						|
		this.tables = [table]
 | 
						|
		/** @private */
 | 
						|
		this.directions = []
 | 
						|
		/** @private */
 | 
						|
		this.sql = ""
 | 
						|
		/** @private */
 | 
						|
		this.cols = []
 | 
						|
		/** @private */
 | 
						|
		this.using = []
 | 
						|
		/** @private */
 | 
						|
		this.isPluck = false
 | 
						|
		/** @private */
 | 
						|
		this.parameters = []
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @template {keyof U.Models} Table2
 | 
						|
	 * @param {Table2} table
 | 
						|
	 * @param {Col & (keyof U.Models[Table2])} col
 | 
						|
	 * @param {"inner" | "left"} [direction]
 | 
						|
	 */
 | 
						|
	join(table, col, direction = "inner") {
 | 
						|
		/** @type {From<Table | Table2, keyof U.Merge<U.Models[Table | Table2]>>} */
 | 
						|
		// @ts-ignore
 | 
						|
		const r = this
 | 
						|
		r.tables.push(table)
 | 
						|
		r.directions.push(direction.toUpperCase())
 | 
						|
		r.using.push(col)
 | 
						|
		return r
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @template {Col} Select
 | 
						|
	 * @param {Col[] | Select[]} cols
 | 
						|
	 */
 | 
						|
	select(...cols) {
 | 
						|
		/** @type {From<Table, Select>} */
 | 
						|
		const r = this
 | 
						|
		r.cols = cols
 | 
						|
		return r
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @template {Col} Select
 | 
						|
	 * @param {Select} col
 | 
						|
	 */
 | 
						|
	pluck(col) {
 | 
						|
		/** @type {Pluck<Table, Select>} */
 | 
						|
		// @ts-ignore
 | 
						|
		const r = this
 | 
						|
		r.cols = [col]
 | 
						|
		r.isPluck = true
 | 
						|
		return r
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @param {string} sql
 | 
						|
	 */
 | 
						|
	and(sql) {
 | 
						|
		this.sql += " " + sql
 | 
						|
		return this
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * @param {Partial<U.Models[Table]>} conditions
 | 
						|
	 */
 | 
						|
	where(conditions) {
 | 
						|
		const wheres = Object.entries(conditions).map(([col, value]) => {
 | 
						|
			this.parameters.push(value)
 | 
						|
			return `"${col}" = ?`
 | 
						|
		})
 | 
						|
		this.sql += " WHERE " + wheres.join(" AND ")
 | 
						|
		return this
 | 
						|
	}
 | 
						|
 | 
						|
	prepare() {
 | 
						|
		let sql = `SELECT ${this.cols.map(k => `"${k}"`).join(", ")} FROM ${this.tables[0]} `
 | 
						|
		for (let i = 1; i < this.tables.length; i++) {
 | 
						|
			const table = this.tables[i]
 | 
						|
			const col = this.using[i-1]
 | 
						|
			const direction = this.directions[i-1]
 | 
						|
			sql += `${direction} JOIN ${table} USING (${col}) `
 | 
						|
		}
 | 
						|
		sql += this.sql
 | 
						|
		/** @type {U.Prepared<Pick<U.Merge<U.Models[Table]>, Col>>} */
 | 
						|
		let prepared = db.prepare(sql)
 | 
						|
		if (this.isPluck) prepared = prepared.pluck()
 | 
						|
		return prepared
 | 
						|
	}
 | 
						|
 | 
						|
	get(..._) {
 | 
						|
		const prepared = this.prepare()
 | 
						|
		return prepared.get(...this.parameters, ..._)
 | 
						|
	}
 | 
						|
 | 
						|
	all(..._) {
 | 
						|
		const prepared = this.prepare()
 | 
						|
		return prepared.all(...this.parameters, ..._)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* c8 ignore start - this code is only used for types and does not actually execute */
 | 
						|
/**
 | 
						|
 * @template {keyof U.Models} Table
 | 
						|
 * @template {keyof U.Merge<U.Models[Table]>} Col
 | 
						|
 */
 | 
						|
class Pluck extends From {
 | 
						|
	// @ts-ignore
 | 
						|
	prepare() {
 | 
						|
		/** @type {U.Prepared<U.Merge<U.Models[Table]>[Col]>} */
 | 
						|
		// @ts-ignore
 | 
						|
		const prepared = super.prepare()
 | 
						|
		return prepared
 | 
						|
	}
 | 
						|
 | 
						|
	get(..._) {
 | 
						|
		const prepared = this.prepare()
 | 
						|
		return prepared.get(..._)
 | 
						|
	}
 | 
						|
 | 
						|
	all(..._) {
 | 
						|
		const prepared = this.prepare()
 | 
						|
		return prepared.all(..._)
 | 
						|
	}
 | 
						|
}
 | 
						|
/* c8 ignore stop */
 | 
						|
 | 
						|
/**
 | 
						|
 * @template {keyof U.Models} Table
 | 
						|
 * @param {Table} table
 | 
						|
 */
 | 
						|
function from(table) {
 | 
						|
	return new From(table)
 | 
						|
}
 | 
						|
 | 
						|
module.exports.from = from
 | 
						|
module.exports.select = select
 |