用nodejs访问ActiveX对象,以操作Access数据库为例。


Posted in NodeJs onDecember 15, 2011

起因
有人提问“如果用nodejs访问sql server?”
找了找资料,发现有两类解决方法,使用第三方nodejs插件:https://github.com/orenmazor/node-tds、使用ADODB.ConnectionActiveX对象。
参考:
http://stackoverflow.com/questions/857670/how-to-connect-to-sql-server-database-from-javascript
http://stackoverflow.com/questions/4728385/connecting-to-a-remote-microsoft-sql-server-from-node-js
如果用ActiveX那么在Windows下nodejs将会无所不能,类似写asp。那它们怎么通信?得动手试试
经过
思路
用nodejs通过cscript.exe(windows脚本进程)间接访问ActiveX
cscript能解析jscript和vbscript两种脚本,无疑为方便维护选jscript开发。
参考:http://www.microsoft.com/resources/documentation/windows/xp/all/proddocs/en-us/cscript_overview.mspx?mfr=true
需解决的问题
1、跨进程通信
新版的nodejs里增加了对子进程的操作,跨进程通信不是问题。
http://nodejs.org/docs/latest/api/all.html#child_Processes

var util = require('util'), 
exec = require('child_process').exec, 
child; 
child = exec('cat *.js bad_file | wc -l', 
function (error, stdout, stderr) { 
console.log('stdout: ' + stdout); 
console.log('stderr: ' + stderr); 
if (error !== null) { 
console.log('exec error: ' + error); 
} 
});

如例我们可以拿到控制台的输出内容stdout!

2、数据库访问相关ActiveX,ADODB.Connection
参考:http://msdn.microsoft.com/en-us/library/windows/desktop/aa746471%28v=vs.85%29.aspx

var connection = new ActiveXObject("ADODB.Connection"); 
var result = 'ok'; 
try{ 
connection.Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + params.accessfile); 
connection.Execute(params.sql); 
} catch(ex){ 
result = ex.message; 
} 
return { 
result: result 
};

connection.Open(connectionString),链接字符串参数可以设置访问sql server。
参考:http://www.connectionstrings.com/sql-server-2005
3、为方便维护,特别将cscript和nodejs的脚本合并,用typeof exports判断当前运行环境。
4、字符编码cscript代码使用ascii编码
非ascii码字符进行“\uHHHH”Unicode编码。
5、命令行字符需转义,双引号、百分号在命令行有特殊意义。
参数传递使用base64编码,避免冲突
cscript环境MSXML2.DOMDocument可以做base64编解码
function base64Decode(base64){ 
var xmldom = new ActiveXObject("MSXML2.DOMDocument"); 
var adostream = new ActiveXObject("ADODB.Stream"); 
var temp = xmldom.createElement("temp"); 
temp.dataType = "bin.base64"; 
temp.text = base64; 
adostream.Charset = "utf-8"; 
adostream.Type = 1; // 1=adTypeBinary 2=adTypeText 
adostream.Open(); 
adostream.Write(temp.nodeTypedValue); 
adostream.Position = 0; 
adostream.Type = 2; // 1=adTypeBinary 2=adTypeText 
var result = adostream.ReadText(-1); // -1=adReadAll 
adostream.Close(); 
adostream = null; 
xmldom = null; 
return result; 
}

总结
调用流程
1、创建子进程,传递经过编码的参数;
2、子进程处理完毕将数据JSON格式化输出到控制台;(子进程自动结束)
3、读取控制台的数据,执行回调函数。

优势
1、使nodejs拥有访问ActiveX对象的能力;
2、实现简单,开发维护方便。

劣势
1、只能运行在Windows平台;
2、数据编解码会消耗更多cpu;
3、每次调用需要创建一个子进程重新连接。(可改进)
总结
1、具有一定实用性;
2、跨进程通信性能可继续探索。
模块代码:

var Access = { 
create: function(params){ 
var fso = new ActiveXObject("Scripting.FileSystemObject"); 
var result = 'ok'; 
if (!fso.FileExists(params.accessfile)){ 
var adoxcatalog = new ActiveXObject("ADOX.Catalog"); 
try { 
adoxcatalog.Create("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + params.accessfile); 
} catch(ex) { 
result = ex.message; 
return; 
} 
adoxcatalog = null; 
} else { 
result = 'exists'; 
} 
return { 
result: result 
}; 
}, 
existsTable: function(params){ 
var connection = new ActiveXObject("ADODB.Connection"); 
var result = 'ok', exists = false; 
try{ 
connection.Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + params.accessfile); 
var recordset = connection.OpenSchema(20/*adSchemaTables*/); 
recordset.MoveFirst(); 
while (!recordset.EOF){ 
if (recordset("TABLE_TYPE") == "TABLE" && recordset("TABLE_NAME") == params.tablename){ 
exists = true; 
break; 
} 
recordset.MoveNext(); 
} 
recordset.Close(); 
recordset = null; 
} catch(ex){ 
result = ex.message; 
} 
return { 
"result": result, 
"exists": exists 
}; 
}, 
execute: function(params){ 
var connection = new ActiveXObject("ADODB.Connection"); 
var result = 'ok'; 
try{ 
connection.Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + params.accessfile); 
connection.Execute(params.sql); 
} catch(ex){ 
result = ex.message; 
} 
return { 
result: result 
}; 
}, 
query: function(params){ 
var connection = new ActiveXObject("ADODB.Connection"); 
var result = 'ok', records = []; 
try{ 
connection.Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + params.accessfile); 
var recordset = new ActiveXObject("ADODB.Recordset"); 
recordset.Open(params.sql, connection); 
var fields = []; 
var enumer = new Enumerator(recordset.Fields); 
for (; !enumer.atEnd(); enumer.moveNext()){ 
fields.push(enumer.item().name); 
} 
recordset.MoveFirst(); 
while (!recordset.EOF) { 
var item = {}; 
for (var i = 0; i < fields.length; i++){ 
var fieldname = fields[i]; 
item[fieldname] = recordset(fieldname).value; 
} 
records.push(item); 
recordset.MoveNext(); 
} 
recordset.Close(); 
recordset = null; 
} catch(ex){ 
result = ex.message; 
} 
return { 
result: result, 
records: records 
}; 
} 
}; 
if (/^u/.test(typeof exports)){ // cscript 
void function(){ 
//from http://tangram.baidu.com/api.html#baidu.json 
var JSON = { 
stringify: (function () { 
/** 
* 字符串处理时需要转义的字符表 
* @private 
*/ 
var escapeMap = { 
"\b": '\\b', 
"\t": '\\t', 
"\n": '\\n', 
"\f": '\\f', 
"\r": '\\r', 
'"' : '\\"', 
"\\": '\\\\' 
}; 
/** 
* 字符串序列化 
* @private 
*/ 
function encodeString(source) { 
if (/["\\\x00-\x1f]/.test(source)) { 
source = source.replace( 
/["\\\x00-\x1f]/g, 
function (match) { 
var c = escapeMap[match]; 
if (c) { 
return c; 
} 
c = match.charCodeAt(); 
return "\\u00" 
+ Math.floor(c / 16).toString(16) 
+ (c % 16).toString(16); 
}); 
} 
return '"' + source + '"'; 
} 
/** 
* 数组序列化 
* @private 
*/ 
function encodeArray(source) { 
var result = ["["], 
l = source.length, 
preComma, i, item; 
for (i = 0; i < l; i++) { 
item = source[i]; 
switch (typeof item) { 
case "undefined": 
case "function": 
case "unknown": 
break; 
default: 
if(preComma) { 
result.push(','); 
} 
result.push(JSON.stringify(item)); 
preComma = 1; 
} 
} 
result.push("]"); 
return result.join(""); 
} 
/** 
* 处理日期序列化时的补零 
* @private 
*/ 
function pad(source) { 
return source < 10 ? '0' + source : source; 
} 
/** 
* 日期序列化 
* @private 
*/ 
function encodeDate(source){ 
return '"' + source.getFullYear() + "-" 
+ pad(source.getMonth() + 1) + "-" 
+ pad(source.getDate()) + "T" 
+ pad(source.getHours()) + ":" 
+ pad(source.getMinutes()) + ":" 
+ pad(source.getSeconds()) + '"'; 
} 
return function (value) { 
switch (typeof value) { 
case 'undefined': 
return 'undefined'; 
case 'number': 
return isFinite(value) ? String(value) : "null"; 
case 'string': 
return encodeString(value).replace(/[^\x00-\xff]/g, function(all) { 
return "\\u" + (0x10000 + all.charCodeAt(0)).toString(16).substring(1); 
}); 
case 'boolean': 
return String(value); 
default: 
if (value === null) { 
return 'null'; 
} 
if (value instanceof Array) { 
return encodeArray(value); 
} 
if (value instanceof Date) { 
return encodeDate(value); 
} 
var result = ['{'], 
encode = JSON.stringify, 
preComma, 
item; 
for (var key in value) { 
if (Object.prototype.hasOwnProperty.call(value, key)) { 
item = value[key]; 
switch (typeof item) { 
case 'undefined': 
case 'unknown': 
case 'function': 
break; 
default: 
if (preComma) { 
result.push(','); 
} 
preComma = 1; 
result.push(encode(key) + ':' + encode(item)); 
} 
} 
} 
result.push('}'); 
return result.join(''); 
} 
}; 
})(), 
parse: function (data) { 
return (new Function("return (" + data + ")"))(); 
} 
} 
//http://blog.csdn.net/cuixiping/article/details/409468 
function base64Decode(base64){ 
var xmldom = new ActiveXObject("MSXML2.DOMDocument"); 
var adostream = new ActiveXObject("ADODB.Stream"); 
var temp = xmldom.createElement("temp"); 
temp.dataType = "bin.base64"; 
temp.text = base64; 
adostream.Charset = "utf-8"; 
adostream.Type = 1; // 1=adTypeBinary 2=adTypeText 
adostream.Open(); 
adostream.Write(temp.nodeTypedValue); 
adostream.Position = 0; 
adostream.Type = 2; // 1=adTypeBinary 2=adTypeText 
var result = adostream.ReadText(-1); // -1=adReadAll 
adostream.Close(); 
adostream = null; 
xmldom = null; 
return result; 
} 
WScript.StdOut.Write('<json>'); 
var method = Access[WScript.Arguments(0)]; 
var result = null; 
if (method){ 
result = method(JSON.parse(base64Decode(WScript.Arguments(1)))); 
} 
WScript.StdOut.Write(JSON.stringify(result)); 
WScript.StdOut.Write('</json>'); 
}(); 
} else { // nodejs 
void function(){ 
function json4stdout(stdout){ 
if (!stdout) return; 
var result = null; 
String(stdout).replace(/<json>([\s\S]+)<\/json>/, function(){ 
result = JSON.parse(arguments[1]); 
}); 
return result; 
} 
var util = require('util'), exec = require('child_process').exec; 
for (var name in Access){ 
exports[name] = (function(funcname){ 
return function(params, callback){ 
console.log([funcname, params]); 
exec( 
util.format( 
'cscript.exe /e:jscript "%s" %s "%s"', __filename, 
funcname, 
(new Buffer(JSON.stringify(params))).toString('base64') 
), 
function (error, stdout, stderr) { 
if (error != null) { 
console.log('exec error: ' + error); 
return; 
} 
console.log('stdout: ' + stdout); 
callback && callback(json4stdout(stdout)); 
} 
); 
} 
})(name); 
} 
}(); 
}

调用代码:
var access = require('./access.js'); 
var util = require('util'); 
var accessfile = 'demo.mdb'; 
access.create({ accessfile: accessfile }, function(data){ 
console.log(data); 
}); 
access.existsTable({ accessfile: accessfile, tablename: 'demo' }, function(data){ 
if (data.result == 'ok' && !data.exists){ 
access.execute({ 
accessfile: 'demo.mdb', 
sql: "CREATE TABLE demo(id Counter Primary key, data Text(100))" 
}); 
} 
}); 
access.execute({ 
accessfile: 'demo.mdb', 
sql: util.format("INSERT INTO demo(data) VALUES('zswang 路过!%s')", +new Date) 
}, function(data){ 
console.log(data); 
}); 
access.query({ 
accessfile: 'demo.mdb', 
sql: "SELECT * FROM demo" 
}, function(data){ 
console.log(data); 
});

最新代码:http://code.google.com/p/nodejs-demo/source/browse/#svn%2Ftrunk%2Fdatabase
NodeJs 相关文章推荐
nodejs实现的一个简单聊天室功能分享
Dec 06 NodeJs
NodeJS使用formidable实现文件上传
Oct 27 NodeJs
Ajax异步文件上传与NodeJS express服务端处理
Apr 01 NodeJs
nodejs6下使用koa2框架实例
May 18 NodeJs
NodeJS链接MySql数据库的操作方法
Jun 27 NodeJs
详解使用PM2管理nodejs进程
Oct 24 NodeJs
NodeJS如何实现同步的方法示例
Aug 24 NodeJs
NodeJs 文件系统操作模块fs使用方法详解
Nov 26 NodeJs
nodejs 使用http进行post或get请求的实例(携带cookie)
Jan 03 NodeJs
nodejs读取图片返回给浏览器显示
Jul 25 NodeJs
nodeJs的安装与npm全局环境变量的配置详解
Jan 06 NodeJs
nodejs使用Sequelize框架操作数据库的实现
Oct 21 NodeJs
golang、python、php、c++、c、java、Nodejs性能对比
Mar 12 #NodeJs
PHPStorm 2020.1 调试 Nodejs的多种方法详解
Sep 17 #NodeJs
nodejs实用示例 缩址还原
Dec 28 #NodeJs
nodejs 后缀名判断限制代码
Mar 31 #NodeJs
NodeJS框架Express的模板视图机制分析
Jul 19 #NodeJs
NodeJS 模块开发及发布详解分享
Mar 07 #NodeJs
nodejs入门详解(多篇文章结合)
Mar 07 #NodeJs
You might like
解析strtr函数的效率问题
2013/06/26 PHP
php求今天、昨天、明天时间戳的简单实现方法
2016/07/28 PHP
浅析document.createDocumentFragment()与js效率
2013/07/08 Javascript
jquery()函数的三种语法介绍
2013/10/09 Javascript
jQuery中bind,live,delegate与one方法的用法及区别解析
2013/12/30 Javascript
javascript中直接引用Microsoft的COM生成Word
2014/01/20 Javascript
获取select元素被选中的文本内容的js代码
2014/01/29 Javascript
js代码实现随机颜色的小方块
2015/07/30 Javascript
使用canvas实现仿新浪微博头像截取上传功能
2015/09/02 Javascript
javascript禁止超链接跳转的方法
2016/02/02 Javascript
node.js实现博客小爬虫的实例代码
2016/10/08 Javascript
原生js编写2048小游戏
2017/03/17 Javascript
Angular.JS中指令ng-if的注意事项小结
2017/06/21 Javascript
详解webpack分离css单独打包
2017/06/21 Javascript
vue子父组件通信的实现代码
2017/07/09 Javascript
vue获取DOM元素并设置属性的两种实现方法
2017/09/30 Javascript
详解VUE单页应用骨架屏方案
2019/01/17 Javascript
微信小程序 导入图标实现过程详解
2019/10/11 Javascript
python用于url解码和中文解析的小脚本(python url decoder)
2013/08/11 Python
python中的函数用法入门教程
2014/09/02 Python
Python HTMLParser模块解析html获取url实例
2015/04/08 Python
深入理解Django的自定义过滤器
2017/10/17 Python
Python实现的字典值比较功能示例
2018/01/08 Python
通过python实现弹窗广告拦截过程详解
2019/07/10 Python
python七种方法判断字符串是否包含子串
2020/08/18 Python
Nordgreen台湾官网:极简北欧设计手表
2019/08/21 全球购物
资生堂英国官网:Shiseido英国
2020/12/30 全球购物
介绍一下.NET构架下remoting和webservice
2014/05/08 面试题
党支部书记先进事迹
2014/01/17 职场文书
高级工程师英文求职信
2014/03/19 职场文书
企业党的群众路线教育实践活动领导班子对照检查材料
2014/09/25 职场文书
乡镇干部个人对照检查材料思想汇报(原创篇)
2014/09/28 职场文书
2014大学校园光棍节活动策划书
2014/09/29 职场文书
列车乘务员工作不细心检讨书
2014/10/07 职场文书
Centos环境下Postgresql 安装配置及环境变量配置技巧
2021/05/18 PostgreSQL
Vue3中的Refs和Ref详情
2021/11/11 Vue.js