gpt4 book ai didi

mysql - NodeJS MySQL 转储

转载 作者:搜寻专家 更新时间:2023-11-01 00:13:17 25 4
gpt4 key购买 nike

我试图编写一个基本的 cron 脚本来运行和“转储”一个 mysql 数据库。出于某种原因,当它“成功保存文件”时,它确实创建了文件,但它是空的。如果我不保存文件,而是执行 console.log,它会打印一个空字符串。关于我可能做错了什么的想法?

提前致谢。

var mysql_backup = function(){

this.backup = '';
this.mysql = require('mysql'),

this.init = function(){
this.connection = this.mysql.createConnection({
user: 'root',
password: 'root',
database: 'test'
});
}

this.query = function(sql, callback) {
this.connection.query(sql, function (error, results, fields) {
if (error) {
throw error;
}
if (results.length > 0) {
callback(results);
}
});
}

this.get_tables = function(callback){
var me = this;
me.query('SHOW TABLES',
function(tables) {
for (var table in tables){
me.query(
'SHOW CREATE TABLE ' + tables[table].Tables_in_test,
function(r){
for (var t in r) {
me.backup += "DROP TABLE " + r[t].Table + "\n\n";
me.backup += r[t]["Create Table"] + "\n\n";
}
}
)
}
me.save_backup();
});
}

this.save_backup = function(){
var fs = require('fs');
fs.writeFile("./backup_test.txt", this.backup, function(err) {
if(err) {
console.log(err);
} else {
console.log("The file was saved!");
}
});
}

};

var db = new mysql_backup;
db.init();
db.get_tables();
db.connection.destroy();

最佳答案

编写的代码甚至没有为我保存文件。似乎有一些问题。不确定这是实际代码还是在复制粘贴中丢失了一些东西。但是,根据您所拥有的:

重要的是您永远不会在代码中使用 connection.connect() 连接到数据库。

连接后要运行的代码应该在 connection.connect() 回调中。例如

connection.connect(function (err, empty) {
if (err)
throw new Error ('Panic');

// if no error, we are off to the races...
}

但是,即使您快速重构代码以将最后几行包含在获取连接回调中,您仍然会遇到问题,因为您在进行各种 SQL 调用之前就破坏了连接,因此您会想要将代码移动到某种最终回调中。

即使在您这样做之后,您仍然会有一个空文件,因为您是从“SHOW TABLES”回调中调用 save_backup,而不是在您通过内部回调实际填充它之后获得 CREATE TABLE 语句并填充备份属性。

这是对代码的最小重写,它将实现您的预​​期。需要注意的重要一点是“计数器”,它管理何时写入文件和关闭连接。如果是我的话,我会做出其他改变,包括:

  • 使用“自己”而不是“我”
  • 使用数字 for 循环而不是 for (... in ...) 语法
  • 让我自己的回调符合(err,stuff)的 Node 约定
  • 一个更实质性的变化是我将重写它以使用 promise ,因为这样做可以让您免于因深度嵌套回调固有的混淆而感到悲伤。我个人喜欢 Q 库,但这里有多个选项。

希望这对您有所帮助。

var mysql_backup = function(){
this.backup = '';
this.mysql = require('mysql');

this.init = function(){
this.connection = this.mysql.createConnection({
user : 'root',
password : 'root',
database : 'test'
});

};

this.query = function(sql, callback) {
this.connection.query(sql, function (error, results, fields) {
if (error) {
throw error;
}
if (results.length > 0) {
callback(results);
}
});
};

this.get_tables = function(callback){
var counter = 0;
var me = this;
this.query('SHOW TABLES',
function(tables) {
for (table in tables){
counter++;
me.query(
'SHOW CREATE TABLE ' + tables[table].Tables_in_mvc,
function(r){
for (t in r) {
me.backup += "DROP TABLE " + r[t].Table + "\n\n";
me.backup += r[t]["Create Table"] + "\n\n";
}
counter--;
if (counter === 0){
me.save_backup();
me.connection.destroy();

}
}
)
}
});
};

this.save_backup = function(){
var fs = require('fs');
fs.writeFile("./backup_test.txt", this.backup, function(err) {
if(err) {
console.log(err);
} else {
console.log("The file was saved!");
}
});
}

};

var db = new mysql_backup;
db.init();
db.connection.connect(function (err){
if (err) console.log(err);
db.get_tables(function(x){;});

});

更新:如果你很好奇,这里有一个使用 promises 的大量评论的实现。请注意,没有解释 Q promise 库函数的注释,它比原始版本稍微短一些,还提供更全面的错误处理。

var MysqlBackup = function(connectionInfo, filename){

var Q = require('q');
var self = this;
this.backup = '';
// my personal preference is to simply require() inline if I am only
// going to use something a single time. I am certain some will find
// this a terrible practice
this.connection = require('mysql').createConnection(connectionInfo);

function getTables(){
// return a promise from invoking the node-style 'query' method
// of self.connection with parameter 'SHOW TABLES'.
return Q.ninvoke(self.connection,'query', 'SHOW TABLES');
};

function doTableEntries(theResults){

// note that because promises only pass a single parameter around,
// if the 'denodeify-ed' callback has more than two parameters (the
// first being the err param), the parameters will be stuffed into
// an array. In this case, the content of the 'fields' param of the
// mysql callback is in theResults[1]

var tables = theResults[0];
// create an array of promises resulting from another Q.ninvoke()
// query call, chained to .then(). Note that then() expects a function,
// so recordEntry() in fact builds and returns a new one-off function
// for actually recording the entry (see recordEntry() impl. below)

var tableDefinitionGetters = [];
for (var i = 0; i < tables.length ; i++){
// I noticed in your original code that your Tables_in_[] did not
// match your connection details ('mvc' vs 'test'), but the below
// should work and is a more generalized solution
var tableName = tables[i]['Tables_in_'+connectionInfo.database];

tableDefinitionGetters.push(Q.ninvoke(self.connection, 'query', 'SHOW CREATE TABLE ' + tableName)
.then(recordEntry(tableName)) );
}

// now that you have an array of promises, you can use Q.allSettled
// to return a promise which will be settled (resolved or rejected)
// when all of the promises in the array are settled. Q.all is similar,
// but its promise will be rejected (immediately) if any promise in the
// array is rejected. I tend to use allSettled() in most cases.

return Q.allSettled(tableDefinitionGetters);
};

function recordEntry (tableName){
return function(createTableQryResult){
self.backup += "DROP TABLE " + tableName + "\n\n";
self.backup += createTableQryResult[0][0]["Create Table"] + "\n\n";
};
};

function saveFile(){
// Q.denodeify return a promise-enabled version of a node-style function
// the below is probably excessively terse with its immediate invocation
return (Q.denodeify(require('fs').writeFile))(filename, self.backup);
}

// with the above all done, now you can actually make the magic happen,
// starting with the promise-return Q.ninvoke to connect to the DB
// note that the successive .then()s will be executed iff (if and only
// if) the preceding item resolves successfully, .catch() will get
// executed in the event of any upstream error, and finally() will
// get executed no matter what.

Q.ninvoke(this.connection, 'connect')
.then(getTables)
.then(doTableEntries)
.then(saveFile)
.then( function() {console.log('Success'); } )
.catch( function(err) {console.log('Something went awry', err); } )
.finally( function() {self.connection.destroy(); } );
};

var myConnection = {
host : '127.0.0.1',
user : 'root',
password : 'root',
database : 'test'
};

// I have left this as constructor-based calling approach, but the
// constructor just does it all so I just ignore the return value

new MysqlBackup(myConnection,'./backup_test.txt');

关于mysql - NodeJS MySQL 转储,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22109487/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com