我正在使用带有Node.js和SQLite 3的Electron制作一个简单的应用程序。我的目标是用我的函数包装一个基本的SELECT
查询。但是,它无法从数据库获取内容。
我知道我可以像这样选择一行的firstname
列:
storage.all('select firstname from users where id = 1', [], ( err, rows ) => {
if ( err ) {
throw `Error while retrieving ${q}`;
};
rows.forEach(( row ) => {
n.webContents.send( 'userName', row.firstname );
console.log( 'query output sent: ' + row.firstname );
});
});
这里的 storage
是我的设置数据库。上面的方法效果很好,可以让我将用户名发送到前端视图。
但是如果我将其包装成如下所示的函数:
let select = ( items, db, where, sign, whereNew ) => {
// preset sql query
let q = `select ${items} from ${db}`;
if ( where && sign && whereNew ) { q += ` where ${where} ${sign} ${whereNew}`};
let arr;
console.log(q);
// perform query on the database
return storage.each(q, [], ( err, row ) => {
if ( err ) {
throw `Error while retrieving ${q}`;
};
return row[items];
});
};
然后控制台上的输出将是一个空的Database
对象(请参见下文)。
这是我调用此函数并将其记录在控制台中的方式:
let u = select( 'firstname', 'users', 'id', '=', 1 );
console.log( 'this log ' + u );
这是控制台输出:
> szwaczki-react@0.1.0 dev /Users/middleman/Documents/Programowanie/MyApps/szwaczki-react-sqlite
> concurrently "BROWSER=none npm run start" "wait-on http://localhost:3000 && electron ."
[0]
[0] > szwaczki-react@0.1.0 start /Users/middleman/Documents/Programowanie/MyApps/szwaczki-react-sqlite
[0] > react-scripts start
[0]
[0] Starting the development server...
[0]
[1] Connected to the database.sqlite
[0] Compiled with warnings.
[0]
[0] ./src/app.js
[0] Line 2: 'ReactDOM' is defined but never used no-unused-vars
[0]
[0] ./src/view/button.js
[0] Line 2: 'ReactDOM' is defined but never used no-unused-vars
[0]
[0] Search for the keywords to learn more about each warning.
[0] To ignore, add // eslint-disable-next-line to the line before.
[0]
[1] select firstname from users where id = 1
[1] this log [object Database]
[1] query output sent: Mateusz
^C[0] BROWSER=none npm run start exited with code 0
[1] wait-on http://localhost:3000 && electron . exited with code 0
我在做什么错了?
答案 0 :(得分:0)
问题是each
函数返回一个Database
对象以允许链接,而不是回调函数(as per the documentation)的return
值。
要解决此问题,并且由于SQLite的异步行为,您可以将SQLite API函数包装到Promise
中,等待它,然后返回要检索的值:
async function select (items, db, where, sign, whereNew) {
// preset sql query
let q = `select ${items} from ${db}`;
if (where && sign && whereNew) {q += ` where ${where} ${sign} ${whereNew}`};
console.log(q);
// perform query on the database
function fetch () {
return new Promise ((resolve, reject) => {
storage.each(q, [], ( err, row ) => {
if (err) {
reject (`Error while retrieving ${q}`);
} else {
resolve (row [items]);
}
});
});
}
try {
const result = await fetch ();
return result;
} catch (error) {
throw error;
}
}
顺便说一句,构建这样的SQL查询并不是一个好主意,因为它对SQL injection开放。您应该“准备”查询,例如使用Database#prepare (...)
function,这基本上可以确保除了查询之外,其他任何SQL代码都不能执行(通过转义任何特殊字符,命令等)。
答案 1 :(得分:0)
我想我已经回答了我自己的问题,但我仍然对此没有什么疑问。我遵循了异步提示,这是工作代码:
JS
NewMessageSignature
如果我想记录结果或将其发送到IPC.Renderer,则此方法很好。但是我不能将其放入变量或后端的其他任何地方,这就是这场斗争的背后思想。