2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 16:18:33 +00:00
books/backends/sqlite.js

193 lines
6.0 KiB
JavaScript
Raw Normal View History

2018-01-16 11:39:17 +05:30
const frappe = require('frappejs');
2018-01-12 17:55:07 +05:30
const sqlite3 = require('sqlite3').verbose();
const Database = require('./database');
2018-02-07 18:53:52 +05:30
const debug = false;
2018-01-12 17:55:07 +05:30
module.exports = class sqliteDatabase extends Database {
constructor({ dbPath }) {
super();
this.dbPath = dbPath;
2018-01-31 18:34:46 +05:30
}
connect(dbPath) {
if (dbPath) {
this.dbPath = dbPath;
2018-01-31 18:34:46 +05:30
}
return new Promise(resolve => {
this.conn = new sqlite3.Database(this.dbPath, () => {
2018-01-31 18:34:46 +05:30
if (debug) {
this.conn.on('trace', (trace) => console.log(trace));
}
resolve();
});
});
}
async tableExists(table) {
const name = await this.sql(`SELECT name FROM sqlite_master WHERE type='table' AND name='${table}'`);
return (name && name.length) ? true : false;
2018-01-31 18:34:46 +05:30
}
async runCreateTableQuery(doctype, columns, values) {
2018-01-31 18:34:46 +05:30
const query = `CREATE TABLE IF NOT EXISTS ${frappe.slug(doctype)} (
2018-01-12 17:55:07 +05:30
${columns.join(", ")})`;
2018-01-31 18:34:46 +05:30
return await this.run(query, values);
}
getColumnDefinition(df) {
2018-02-09 18:25:55 +05:30
return `${df.fieldname} ${this.type_map[df.fieldtype]} ${df.required && !df.default ? "not null" : ""} ${df.default ? `default ${df.default}` : ""}`
2018-01-31 18:34:46 +05:30
}
async getTableColumns(doctype) {
return (await this.sql(`PRAGMA table_info(${doctype})`)).map(d => d.name);
2018-01-31 18:34:46 +05:30
}
async runAlterTableQuery(doctype, field, values) {
await this.run(`ALTER TABLE ${frappe.slug(doctype)} ADD COLUMN ${this.getColumnDefinition(field)}`, values);
2018-01-31 18:34:46 +05:30
}
getOne(doctype, name, fields = '*') {
fields = this.prepareFields(fields);
2018-01-31 18:34:46 +05:30
return new Promise((resolve, reject) => {
this.conn.get(`select ${fields} from ${frappe.slug(doctype)}
2018-01-12 17:55:07 +05:30
where name = ?`, name,
2018-01-31 18:34:46 +05:30
(err, row) => {
resolve(row || {});
});
});
}
async insertOne(doctype, doc) {
let fields = this.getKeys(doctype);
2018-01-31 18:34:46 +05:30
let placeholders = fields.map(d => '?').join(', ');
2018-02-01 16:37:36 +05:30
if (!doc.name) {
doc.name = frappe.getRandomName();
2018-02-01 16:37:36 +05:30
}
2018-01-31 18:34:46 +05:30
return await this.run(`insert into ${frappe.slug(doctype)}
2018-02-01 16:37:36 +05:30
(${fields.map(field => field.fieldname).join(", ")})
values (${placeholders})`, this.getFormattedValues(fields, doc));
2018-01-31 18:34:46 +05:30
}
2018-01-12 17:55:07 +05:30
async updateOne(doctype, doc) {
let fields = this.getKeys(doctype);
2018-01-31 18:34:46 +05:30
let assigns = fields.map(field => `${field.fieldname} = ?`);
let values = this.getFormattedValues(fields, doc);
2018-01-31 18:26:21 +05:30
2018-01-31 18:34:46 +05:30
// additional name for where clause
values.push(doc.name);
2018-01-31 18:26:21 +05:30
2018-01-31 18:34:46 +05:30
return await this.run(`update ${frappe.slug(doctype)}
2018-01-12 17:55:07 +05:30
set ${assigns.join(", ")} where name=?`, values);
2018-01-31 18:34:46 +05:30
}
async runDeleteOtherChildren(field, added) {
// delete other children
// `delete from doctype where parent = ? and name not in (?, ?, ?)}`
await this.run(`delete from ${frappe.slug(field.childtype)}
where
parent = ? and
name not in (${added.slice(1).map(d => '?').join(', ')})`, added);
2018-01-31 18:34:46 +05:30
}
async deleteOne(doctype, name) {
return await this.run(`delete from ${frappe.slug(doctype)} where name=?`, name);
2018-02-01 16:37:36 +05:30
}
async deleteChildren(parenttype, parent) {
2018-02-12 17:31:31 +05:30
await this.run(`delete from ${parenttype} where parent=?`, parent);
}
async deleteSingleValues(name) {
await frappe.db.run('delete from single_value where parent=?', name)
2018-02-01 16:37:36 +05:30
}
getAll({ doctype, fields, filters, start, limit, order_by = 'modified', order = 'desc' } = {}) {
2018-01-31 18:34:46 +05:30
if (!fields) {
fields = frappe.getMeta(doctype).getKeywordFields();
2018-01-31 18:34:46 +05:30
}
return new Promise((resolve, reject) => {
let conditions = this.getFilterConditions(filters);
2018-02-12 17:31:31 +05:30
let query = `select ${fields.join(", ")}
2018-01-12 17:55:07 +05:30
from ${frappe.slug(doctype)}
${conditions.conditions ? "where" : ""} ${conditions.conditions}
${order_by ? ("order by " + order_by) : ""} ${order_by ? (order || "asc") : ""}
2018-02-12 17:31:31 +05:30
${limit ? ("limit " + limit) : ""} ${start ? ("offset " + start) : ""}`;
this.conn.all(query, conditions.values,
2018-01-31 18:34:46 +05:30
(err, rows) => {
if (err) {
reject(err);
} else {
resolve(rows);
}
});
});
}
run(query, params) {
return new Promise((resolve, reject) => {
this.conn.run(query, params, (err) => {
if (err) {
2018-02-01 16:37:36 +05:30
if (debug) {
2018-02-07 22:14:59 +05:30
console.log(err);
2018-02-01 16:37:36 +05:30
}
2018-01-31 18:34:46 +05:30
reject(err);
} else {
resolve();
}
});
});
}
sql(query, params) {
return new Promise((resolve) => {
this.conn.all(query, params, (err, rows) => {
resolve(rows);
});
});
}
async commit() {
try {
await this.run('commit');
} catch (e) {
if (e.errno !== 1) {
throw e;
}
}
}
initTypeMap() {
2018-01-31 18:34:46 +05:30
this.type_map = {
'Currency': 'real'
, 'Int': 'integer'
, 'Float': 'real'
, 'Percent': 'real'
, 'Check': 'integer'
, 'Small Text': 'text'
, 'Long Text': 'text'
, 'Code': 'text'
, 'Text Editor': 'text'
, 'Date': 'text'
, 'Datetime': 'text'
, 'Time': 'text'
, 'Text': 'text'
, 'Data': 'text'
, 'Link': 'text'
, 'Dynamic Link': 'text'
, 'Password': 'text'
, 'Select': 'text'
, 'Read Only': 'text'
, 'Attach': 'text'
, 'Attach Image': 'text'
, 'Signature': 'text'
, 'Color': 'text'
, 'Barcode': 'text'
, 'Geolocation': 'text'
}
}
2018-01-12 17:55:07 +05:30
}