2018-02-07 06:27:37 +00:00
|
|
|
const frappe = require('frappejs');
|
|
|
|
const mysql = require('mysql');
|
2018-02-08 10:38:53 +00:00
|
|
|
const Database = require('./database');
|
2018-02-07 06:27:37 +00:00
|
|
|
const debug = false;
|
|
|
|
|
|
|
|
|
2018-02-13 07:26:25 +00:00
|
|
|
module.exports = class mysqlDatabase extends Database{
|
2018-02-07 06:27:37 +00:00
|
|
|
constructor({ db_name, username, password, host }) {
|
2018-02-08 10:38:53 +00:00
|
|
|
super();
|
2018-02-07 06:27:37 +00:00
|
|
|
this.db_name = db_name;
|
|
|
|
this.username = username;
|
|
|
|
this.password = password;
|
|
|
|
this.host = host;
|
|
|
|
this.init_type_map();
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(db_name) {
|
|
|
|
if (db_name) {
|
|
|
|
this.db_name = db_name;
|
|
|
|
}
|
|
|
|
return new Promise(resolve => {
|
|
|
|
this.conn = new mysql.createConnection({
|
|
|
|
host : this.host,
|
|
|
|
user : this.username,
|
|
|
|
password : this.password,
|
|
|
|
database : this.db_name
|
|
|
|
});
|
|
|
|
() => {
|
|
|
|
if (debug) {
|
|
|
|
this.conn.on('trace', (trace) => console.log(trace));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async tableExists(table) {
|
2018-02-07 06:27:37 +00:00
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
const name = await this.sql(`SELECT table_name
|
|
|
|
FROM information_schema.tables
|
|
|
|
WHERE table_schema = '${this.db_name}'
|
|
|
|
AND table_name = '${table}'`);
|
|
|
|
return (name && name.length) ? true : false;
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async runCreateTableQuery(doctype, columns, values){
|
2018-02-07 06:27:37 +00:00
|
|
|
const query = `CREATE TABLE IF NOT EXISTS ${frappe.slug(doctype)} (
|
|
|
|
${columns.join(", ")})`;
|
|
|
|
|
|
|
|
return await this.run(query, values);
|
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
|
|
|
|
getColumnDefinition(df) {
|
2018-02-07 06:27:37 +00:00
|
|
|
return `${df.fieldname} ${this.type_map[df.fieldtype]} ${df.reqd && !df.default ? "not null" : ""} ${df.default ? `default '${df.default}'` : ""}`
|
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async getTableColumns(doctype) {
|
|
|
|
return (await this.sql(`SHOW COLUMNS FROM ${doctype}`)).map(d => d.Field);
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async runAlterTableQuery(doctype) {
|
|
|
|
await this.run(`ALTER TABLE ${frappe.slug(doctype)} ADD COLUMN ${this.get_column_definition(df)}`, values);
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
getOne(doctype, name, fields = '*') {
|
|
|
|
|
|
|
|
fields = this.prepareFields(fields);
|
2018-02-07 06:27:37 +00:00
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.conn.get(`select ${fields} from ${frappe.slug(doctype)}
|
|
|
|
where name = ?`, name,
|
|
|
|
(err, row) => {
|
|
|
|
resolve(row || {});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async insertOne(doctype, doc) {
|
2018-02-07 06:27:37 +00:00
|
|
|
let fields = this.get_keys(doctype);
|
|
|
|
let placeholders = fields.map(d => '?').join(', ');
|
|
|
|
|
|
|
|
if (!doc.name) {
|
2018-02-08 10:38:53 +00:00
|
|
|
doc.name = frappe.getRandomName();
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return await this.run(`insert into ${frappe.slug(doctype)}
|
|
|
|
(${fields.map(field => field.fieldname).join(", ")})
|
2018-02-08 10:38:53 +00:00
|
|
|
values (${placeholders})`, this.getFormattedValues(fields, doc));
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async updateOne(doctype, doc) {
|
|
|
|
let fields = this.getKeys(doctype);
|
2018-02-07 06:27:37 +00:00
|
|
|
let assigns = fields.map(field => `${field.fieldname} = ?`);
|
2018-02-08 10:38:53 +00:00
|
|
|
let values = this.getFormattedValues(fields, doc);
|
2018-02-07 06:27:37 +00:00
|
|
|
|
|
|
|
// additional name for where clause
|
|
|
|
values.push(doc.name);
|
|
|
|
|
|
|
|
return await this.run(`update ${frappe.slug(doctype)}
|
|
|
|
set ${assigns.join(", ")} where name=?`, values);
|
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async runDeleteOtherChildren(field, added) {
|
|
|
|
await this.run(`delete from ${frappe.slug(field.childtype)}
|
|
|
|
where
|
|
|
|
parent = ? and
|
|
|
|
name not in (${added.slice(1).map(d => '?').join(', ')})`, added);
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async deleteOne(doctype, name) {
|
|
|
|
return await this.run(`delete from ${frappe.slug(doctype)} where name=?`, name);
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
async deleteChildren(parenttype, parent) {
|
|
|
|
await this.run(`delete from ${parent} where parent=?`, parent);
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-08 10:38:53 +00:00
|
|
|
getAll({ doctype, fields, filters, start, limit, order_by = 'modified', order = 'desc' } = {}) {
|
2018-02-07 06:27:37 +00:00
|
|
|
if (!fields) {
|
2018-02-08 10:38:53 +00:00
|
|
|
fields = frappe.getMeta(doctype).getKeywordFields();
|
2018-02-07 06:27:37 +00:00
|
|
|
}
|
|
|
|
return new Promise((resolve, reject) => {
|
2018-02-08 10:38:53 +00:00
|
|
|
let conditions = this.getFilterConditions(filters);
|
2018-02-07 06:27:37 +00:00
|
|
|
|
|
|
|
this.conn.all(`select ${fields.join(", ")}
|
|
|
|
from ${frappe.slug(doctype)}
|
|
|
|
${conditions.conditions ? "where" : ""} ${conditions.conditions}
|
|
|
|
${order_by ? ("order by " + order_by) : ""} ${order_by ? (order || "asc") : ""}
|
|
|
|
${limit ? ("limit " + limit) : ""} ${start ? ("offset " + start) : ""}`, conditions.values,
|
|
|
|
(err, rows) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else {
|
|
|
|
resolve(rows);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
run(query, params) {
|
|
|
|
// TODO promisify
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.conn.query(query, params, (err) => {
|
|
|
|
if (err) {
|
|
|
|
if (debug) {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
reject(err);
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
sql(query, params) {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
this.conn.query(query, params, (err, rows) => {
|
|
|
|
resolve(rows);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async commit() {
|
|
|
|
try {
|
|
|
|
await this.run('commit');
|
|
|
|
} catch (e) {
|
|
|
|
if (e.errno !== 1) {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
init_type_map() {
|
|
|
|
this.type_map = {
|
|
|
|
'Currency': 'real'
|
2018-02-07 06:43:18 +00:00
|
|
|
, 'Int': 'INT'
|
|
|
|
, 'Float': 'decimal(18,6)'
|
2018-02-07 06:27:37 +00:00
|
|
|
, 'Percent': 'real'
|
2018-02-08 10:06:30 +00:00
|
|
|
, 'Check': 'INT(1)'
|
2018-02-07 06:27:37 +00:00
|
|
|
, 'Small Text': 'text'
|
|
|
|
, 'Long Text': 'text'
|
|
|
|
, 'Code': 'text'
|
|
|
|
, 'Text Editor': 'text'
|
|
|
|
, 'Date': 'DATE'
|
|
|
|
, 'Datetime': 'DATETIME'
|
|
|
|
, 'Time': 'TIME'
|
|
|
|
, 'Text': 'text'
|
|
|
|
, 'Data': 'VARCHAR(140)'
|
|
|
|
, 'Link': ' varchar(140)'
|
|
|
|
, 'Dynamic Link': 'text'
|
|
|
|
, 'Password': 'varchar(140)'
|
|
|
|
, 'Select': 'VARCHAR(140)'
|
|
|
|
, 'Read Only': 'varchar(140)'
|
|
|
|
, 'Attach': 'text'
|
|
|
|
, 'Attach Image': 'text'
|
|
|
|
, 'Signature': 'text'
|
|
|
|
, 'Color': 'text'
|
|
|
|
, 'Barcode': 'text'
|
|
|
|
, 'Geolocation': 'text'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|