|
@ -7,24 +7,184 @@
|
|
|
'use strict';
|
|
|
|
|
|
let ImDb = require('../../repository/mysql/db/im.db');
|
|
|
var ObjectUtil = require("../../util/object.util.js");
|
|
|
let ObjectUtil = require("../../util/object.util.js");
|
|
|
let DbUtil = require("../../util/db.util.js");
|
|
|
|
|
|
let async = require("async");
|
|
|
let log = require("../../util/log.js");
|
|
|
let fs = require('fs');
|
|
|
var mongoose = require("mongoose");
|
|
|
let mongoose = require("mongoose");
|
|
|
let vprintf = require('sprintf-js').vsprintf;
|
|
|
|
|
|
const migrationFile = "../../resources/schema/ichat_1.2.8_data_migration.sql";
|
|
|
const MIGRATION_SCRIPT_File_NAME = "../../resources/schema/ichat_1.2.8_data_migration.sql";
|
|
|
|
|
|
class Migration {
|
|
|
constructor() {
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 迁移P2P及其消息
|
|
|
* 迁移系统消息
|
|
|
*/
|
|
|
static migrateSystem() {
|
|
|
let data = "-- System sessions: \n";
|
|
|
async.waterfall([
|
|
|
function (callback) {
|
|
|
// 所有的系统会话
|
|
|
let sql = "select distinct to_uid participant_id from im_new.msg_system order by to_uid";
|
|
|
ImDb.execQuery({
|
|
|
sql: sql,
|
|
|
args: [],
|
|
|
handler: function (err, res) {
|
|
|
if (err) {
|
|
|
return callback(err, res);
|
|
|
}
|
|
|
|
|
|
let buffer = "insert into sessions(id, name, type, create_date) values ";
|
|
|
res.forEach(function (session) {
|
|
|
buffer += vprintf("\n('%s', '%s', 0, now()),", [
|
|
|
DbUtil.stringArrayHash(['system', session.participant_id]),
|
|
|
"系统消息"
|
|
|
]);
|
|
|
});
|
|
|
|
|
|
data += buffer.substr(0, buffer.length - 1) + ";\n";
|
|
|
|
|
|
callback(null);
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
function (callback) {
|
|
|
// 所有的系统消息
|
|
|
let sql = "SELECT msg_id 'id', 'session_id', 'system' `sender_id`, '系统通知' `sender_name`, 1 `content_type`, DATA 'content', timestamp, `type` 'business_type', to_uid FROM im_new.msg_system ORDER BY to_uid";
|
|
|
ImDb.execQuery({
|
|
|
sql: sql,
|
|
|
args: [],
|
|
|
handler: function (err, res) {
|
|
|
if (err) {
|
|
|
return callback(err, res);
|
|
|
}
|
|
|
|
|
|
let buffer = "insert into system_messages(id, session_id, sender_id, sender_name, content_type, content, timestamp, business_type) values";
|
|
|
res.forEach(function (message) {
|
|
|
buffer += vprintf("\n('%s', '%s', '%s', '%s', 1, '%s', %s, '%s'),", [
|
|
|
mongoose.Types.ObjectId().toString(),
|
|
|
DbUtil.stringArrayHash(['system', message.to_uid]),
|
|
|
'system',
|
|
|
'系统',
|
|
|
message.content.replace(/'/g, "''"),
|
|
|
ObjectUtil.timestampToLong(message.timestamp),
|
|
|
message.business_type]);
|
|
|
});
|
|
|
|
|
|
data += buffer.substr(0, buffer.length - 1) + ";\n\n";
|
|
|
|
|
|
callback(null, null);
|
|
|
}
|
|
|
});
|
|
|
}],
|
|
|
function (err, res) {
|
|
|
if (err) {
|
|
|
log.error("Error occurs while migrate system sessions: ", err);
|
|
|
} else {
|
|
|
fs.writeFileSync(MIGRATION_SCRIPT_File_NAME, data + "\n");
|
|
|
|
|
|
log.info("Migrate System sessions succeed.");
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 迁移MUC及其消息。将原先的P2P中与患者有关的会话全部过来出来,再将其转为会话。议题从家庭医生库中提取出来。
|
|
|
*/
|
|
|
static migrateMUC() {
|
|
|
// 选择出所有的咨询组
|
|
|
let sql = "select g.to_gid from im_new.msg_group g where length(g.to_gid) > 20 group by g.to_gid order by g.to_gid";
|
|
|
|
|
|
// 选择出所有的咨询组成员
|
|
|
sql = "select g.to_gid, g.from_uid from im_new.msg_group g where length(g.to_gid) > 20 group by g.to_gid order by g.to_gid";
|
|
|
|
|
|
// 选择出所有的咨询组消息
|
|
|
sql = "select g.to_gid, g.from_uid, g.msg_id, g.`type`, g.content, g.timestamp, g.at_uid from im_new.msg_group g where length(g.to_gid) > 20 order by g.to_gid";
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 迁移P2P及其消息,新的IM中P2P仅给医生使用,因此过滤出原先发送与接收人均是医生的会话。
|
|
|
*/
|
|
|
static migrateP2P() {
|
|
|
let data = "-- P2P sessions: \n";
|
|
|
async.waterfall([
|
|
|
function (callback) {
|
|
|
// 所有的P2P 会话,成员与消息
|
|
|
let sql = "SELECT a.from_uid sender_id, d.name sender_name, a.to_uid to_id, e.name to_name, a.type content_type, a.content, a.timestamp " +
|
|
|
" FROM im_new.msg_p2p a " +
|
|
|
" LEFT JOIN wlyy.wlyy_doctor d ON a.from_uid = d.code " +
|
|
|
" LEFT JOIN wlyy.wlyy_doctor e on a.to_uid = e.code " +
|
|
|
" WHERE a.from_uid IN (SELECT code FROM wlyy.wlyy_doctor d) AND a.to_uid IN (SELECT code FROM wlyy.wlyy_doctor d)";
|
|
|
|
|
|
ImDb.execQuery({
|
|
|
sql: sql,
|
|
|
args: [],
|
|
|
handler: function (err, res) {
|
|
|
if(err) return callback(err, null);
|
|
|
|
|
|
let sessionsBuffer = "insert into sessions(id, name, type, create_date) values ";
|
|
|
let participantBuffer = "insert into participants(session_id, participant_id, participant_role, last_fetch_time) values ";
|
|
|
let messageBuffer = "insert into p2p_messages(id, session_id, sender_id, sender_name, content_type, content, timestamp) values ";
|
|
|
res.forEach(function (message) {
|
|
|
let sessionId = DbUtil.stringArrayHash([message.sender_id, message.to_id]);
|
|
|
|
|
|
// 会话
|
|
|
sessionsBuffer += vprintf("\n('%s', '%s', %s, %s),", [
|
|
|
sessionId,
|
|
|
'P2P',
|
|
|
1,
|
|
|
new Date('2016-11-22 12:00:00').getTime()
|
|
|
]);
|
|
|
|
|
|
// 成员
|
|
|
participantBuffer += vprintf("\n('%s', '%s', %s, %s),", [
|
|
|
sessionId,
|
|
|
message.sender_id,
|
|
|
0,
|
|
|
new Date().getTime()
|
|
|
]);
|
|
|
|
|
|
participantBuffer += vprintf("\n('%s', '%s', %s, %s),", [
|
|
|
sessionId,
|
|
|
message.to_id,
|
|
|
0,
|
|
|
new Date().getTime()
|
|
|
]);
|
|
|
|
|
|
// 消息
|
|
|
messageBuffer += vprintf("\n('%s', '%s', '%s', '%s', '%s', '%s', %s),", [
|
|
|
mongoose.Types.ObjectId().toString(),
|
|
|
sessionId,
|
|
|
message.sender_id,
|
|
|
message.sender_name,
|
|
|
message.content_type,
|
|
|
message.content.replace(/'/g, "''"),
|
|
|
ObjectUtil.timestampToLong(message.timestamp)
|
|
|
]);
|
|
|
});
|
|
|
|
|
|
data += sessionsBuffer.substr(0, sessionsBuffer.length - 1) + " ON DUPLICATE KEY UPDATE id = id;\n\n";
|
|
|
data += participantBuffer.substr(0, participantBuffer.length - 1) + " ON DUPLICATE KEY UPDATE session_id = session_id;\n\n";
|
|
|
data += messageBuffer.substr(0, messageBuffer.length - 1) + " ON DUPLICATE KEY UPDATE id = id;\n\n";
|
|
|
callback(null);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
],
|
|
|
function (err, res) {
|
|
|
if (err) {
|
|
|
log.error("Error occurs while migrate p2p sessions: ", err);
|
|
|
} else {
|
|
|
fs.writeFileSync(MIGRATION_SCRIPT_File_NAME, data + "\n");
|
|
|
|
|
|
log.info("Migrate P2P sessions succeed.");
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
@ -34,6 +194,7 @@ class Migration {
|
|
|
* 3 原消息取出来后,需要重新生成UUID
|
|
|
*/
|
|
|
static migrateGroups() {
|
|
|
let data = "-- Group sessions: \n";
|
|
|
async.waterfall([
|
|
|
function (callback) {
|
|
|
// 选择出所有的行政组,作为会话
|
|
@ -49,9 +210,9 @@ class Migration {
|
|
|
return callback(err, res);
|
|
|
}
|
|
|
|
|
|
let sqls = "insert into sessions(id, name, type, create_date) values ";
|
|
|
let buffer = "insert into sessions(id, name, type, create_date) values ";
|
|
|
sessions.forEach(function (session) {
|
|
|
sqls += vprintf("\n('%s', '%s', %s, %s)", [
|
|
|
buffer += vprintf("\n('%s', '%s', %s, %s),", [
|
|
|
session.id,
|
|
|
session.name,
|
|
|
session.type,
|
|
@ -59,7 +220,7 @@ class Migration {
|
|
|
]);
|
|
|
});
|
|
|
|
|
|
fs.writeFileSync(migrationFile, sqls + ";\n\n");
|
|
|
data += buffer.substr(0, buffer.length - 1) + ";\n\n";
|
|
|
callback(null);
|
|
|
}
|
|
|
});
|
|
@ -78,9 +239,9 @@ class Migration {
|
|
|
return callback(err, res);
|
|
|
}
|
|
|
|
|
|
let sqls = "insert into participants(session_id, participant_id, participant_role, last_fetch_time) values ";
|
|
|
let buffer = "insert into participants(session_id, participant_id, participant_role, last_fetch_time) values ";
|
|
|
participants.forEach(function (participant) {
|
|
|
sqls += vprintf("\n('%s', '%s', %s, %s)", [
|
|
|
buffer += vprintf("\n('%s', '%s', %s, %s),", [
|
|
|
participant.session_id,
|
|
|
participant.participant_id,
|
|
|
participant.participant_role,
|
|
@ -88,7 +249,7 @@ class Migration {
|
|
|
]);
|
|
|
});
|
|
|
|
|
|
fs.appendFileSync(migrationFile, sqls + ";\n\n");
|
|
|
data += buffer.substr(0, buffer.length - 1) + ";\n\n";
|
|
|
callback(null);
|
|
|
}
|
|
|
});
|
|
@ -108,71 +269,60 @@ class Migration {
|
|
|
return callback(err, res);
|
|
|
}
|
|
|
|
|
|
let sqls = "insert into group_messages(id, session_id, sender_id, sender_name, content_type, content, timestamp, at) values ";
|
|
|
let buffer = "insert into group_messages(id, session_id, sender_id, sender_name, content_type, content, timestamp, at) values ";
|
|
|
messages.forEach(function (message) {
|
|
|
sqls += vprintf("\n('%s', '%s', '%s', '%s', '%s', '%s', %s, '%s')", [
|
|
|
buffer += vprintf("\n('%s', '%s', '%s', '%s', '%s', '%s', %s, '%s'),", [
|
|
|
mongoose.Types.ObjectId().toString(),
|
|
|
message.session_id,
|
|
|
message.sender_id,
|
|
|
message.sender_name,
|
|
|
message.content_type,
|
|
|
message.content,
|
|
|
message.content.replace(/'/g, "''"),
|
|
|
ObjectUtil.timestampToLong(message.timestamp),
|
|
|
message.at
|
|
|
]);
|
|
|
});
|
|
|
|
|
|
fs.appendFileSync(migrationFile, sqls + ";\n\n", null, function (err) {
|
|
|
if (err) {
|
|
|
return callback(err, null);
|
|
|
}
|
|
|
});
|
|
|
data += buffer.substr(0, buffer.length - 1) + ";\n\n";
|
|
|
callback(null, null);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
callback(null, null, "OK");
|
|
|
}
|
|
|
],
|
|
|
function (err, res) {
|
|
|
if (err) {
|
|
|
log.error("Error occures while migrate group sessions: ", err);
|
|
|
log.error("Error occurs while migrate group sessions: ", err);
|
|
|
} else {
|
|
|
fs.appendFileSync(MIGRATION_SCRIPT_File_NAME, data, null);
|
|
|
|
|
|
log.info("Migrate group sessions succeed.");
|
|
|
|
|
|
process.exit(err != null ? 1 : 0);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 迁移MUC及其消息
|
|
|
*/
|
|
|
static migrateMUC() {
|
|
|
// 选择出所有的咨询组
|
|
|
let sql = "select g.to_gid from im_new.msg_group g where length(g.to_gid) > 20 group by g.to_gid order by g.to_gid";
|
|
|
|
|
|
// 选择出所有的咨询组成员
|
|
|
sql = "select g.to_gid, g.from_uid from im_new.msg_group g where length(g.to_gid) > 20 group by g.to_gid order by g.to_gid";
|
|
|
|
|
|
// 选择出所有的咨询组消息
|
|
|
sql = "select g.to_gid, g.from_uid, g.msg_id, g.`type`, g.content, g.timestamp, g.at_uid from im_new.msg_group g where length(g.to_gid) > 20 order by g.to_gid";
|
|
|
}
|
|
|
}
|
|
|
|
|
|
async.waterfall([
|
|
|
function (callback) {
|
|
|
Migration.P2P();
|
|
|
//Migration.migrateSystem();
|
|
|
callback(null);
|
|
|
},
|
|
|
function (callback) {
|
|
|
Migration.migrateMUC();
|
|
|
//Migration.migrateMUC();
|
|
|
callback(null);
|
|
|
},
|
|
|
function (callback) {
|
|
|
Migration.migrateGroups();
|
|
|
Migration.migrateP2P();
|
|
|
callback(null);
|
|
|
},
|
|
|
function (callback) {
|
|
|
//Migration.migrateGroups();
|
|
|
callback(null);
|
|
|
}
|
|
|
],
|
|
|
function (err, res) {
|
|
|
process.exit(err != null ? 1 : 0);
|
|
|
|
|
|
});
|
|
|
|
|
|
|