Nodejs MSSQL

发布时间:2022-02-17编辑:佚名阅读(3276)

MSSQL是Nodejs用于连接Microsoft SQL Server的插件。

在nodejs中,mssql模块支持sqlserver数据库操作。今天将mssql模块的某些功能封装为一个类,方便以后调用。封装的功能有执行存储过程,执行查询语句操作等。如果本篇文章对大家有帮助,那就再好不过了!

如果需要扩展事务和其他功能,请到官网https://www.npmjs.com/package/mssql中看文档,里面有示例。

安装方法

安装指令

npm install mssql

配置Config

const config = {
    user: '...',
    password: '...',
    server: 'localhost',
    database: '...',
    options: {
        //encrypt: true, //使用windows azure,需要设置次配置。
        trustServerCertificate: true //新版要设为true,否则会报“ConnectionError: Failed to connect to localhost:1433 - self signed certificate”错误。
    }
}

user:SQL Server 的登录名

password: SQL Server的登录密码

server:SQL Server的地址

port:端口号,默认为1433

domain:设置domain后,可通过domain连接数据库

database:数据库名称

connectionTimeout:连接timeout,单位ms 默认 15000

requestTimeout:请求timeout,单位ms默认15000

parseJSON:将json数据集转化成json obj

pool.max:连接池最大连接数,默认10

pool.min:连接池最小连接数,默认0

pool.idleTimeoutMillis:设置关闭未使用连接的时间,单位ms默认30000

封装的代码如下

1.连接池的写法:

//导入mssql模块 //基于版本@4.2.1  安装指令npm install mssql@4.2.1
// pool.close()非常重要,只创建,不关闭会造成非常严重的内存泄漏。 关闭池中的所有活动连接。
let mssql=require("mssql");
//引用配置参数模块
let configFile = require("./config");
//数据库异常
mssql.on('error', err => {
   console.log("mssql异常原因:"+err.message);
});
let sql={};
//sql参数的类型
sql.direction={
    //输入参数
    Input:"input",
    //输出参数
    Output:"output",
    //返回参数
    Return:"return"
};
//配置存储过程是的输出输入
sql.sqlserver=mssql;
//默认config对象
let config=configFile.mssql_config;
/**
 * 初始化连接参数
 * @param {string} user 用户名
 * @param {string} password 密码
 * @param {string} server 服务器地址
 * @param {string} database 数据库名称
 * @param {string} port 数据库端口
 */
sql.initConfig = function(user,password,server,database,port){
    config.user = user;
    config.password =password;
    config.server =server;
    config.database= database;
    config.port = port;
}
/**
 * 执行存储过程
 * @param {string} procedure 存储过程名称
 * @param {JSON} params 存储过程参数
 * params的定义格式如:
 let params={
    //ID是存储过程的第一个参数,要去掉@符号
    ID:{
        //sqlType是该ID参数在sqlserver中的类型
        sqlType:sql.sqlserver.Int,
        //direction是表明ID参数是输入还是输出(output)参数
        direction:sql.direction.Input,
        //该ID参数的值
        inputValue:1
    },
    //Name是存储过程的第二个参数,要去掉@符号
    Name:{
        sqlType:sqlHelper.sqlserver.Int,
        direction:sqlHelper.direction.Output,
        outputValue:null
    }
};
 * @param {function} func 回调函数 共有四个参数 error:错误信息 recordsets:查询的表结果 returnValue:存储过程的返回值 affected:影响的行数
 */
sql.execute=async function(procedure,params){
    try {
        //创建数据库连接池
        var pool = new mssql.ConnectionPool(config);
        await pool.connect();//连接数据库
        let request = pool.request();
        if (params != null) {
            for (let index in params) {
                if (params[index].direction == sql.direction.Output) {
                    request.output(index, params[index].sqlType);
                }
                else {
                    request.input(index, params[index].sqlType, params[index].inputValue);
                }
            }
        }
        // result 成功返回该结构
        // {
        //     recordsets,
        //     recordset: recordsets && recordsets[0],
        //     output,
        //     rowsAffected,
        //     returnValue
        // }
        let result =await request.execute(procedure);
        for (let index in params) {
            if (params[index].direction == sql.direction.Output) {
                params[index].outputValue = request.parameters[index].value;
            }
        }
        return {state:true,data:result};
    }catch(err){
        return {state:false,data:err};
    }finally {
        await doRelease(pool);
    }
};
/**
 * 执行sql文本(带params参数)
 * @param {string} sqlText 执行的sql语句
 * @param {JSON} params sql语句中的参数
 * @param {function} func 回调函数 共有两个个参数 error:错误消息 recordsets:查询的结果
 */
sql.queryWithParams=async function(sqlText,params){
    try {
        //创建数据库连接池
        var pool = new mssql.ConnectionPool(config);
        await pool.connect();
        let request = pool.request();
        request.multiple=true;
        if (params != null) {
            for(let index in params){
                request.input(index,params[index].sqlType,params[index].inputValue);
            }
        }
        let result = await request.query(sqlText);
        return {state:true,data:result};
    }catch(err){
        return {state:false,data:err};
    }finally {
        await doRelease(pool);
    }
};
/**
 * 执行sql文本
 * @param {string} sqlText 执行的sql语句
 * @param {function} func 回调函数 共有两个个参数 error:错误消息 recordsets:查询的结果
 */
sql.query=function(sqlText){
    return sql.queryWithParams(sqlText,null);
}
/**
 * 执行大批量数据的插入
 * @param {sqlserver.Table} table 需要插入的数据表
 * 数据表的定义如下:
 let table=new sql.sqlserver.Table('UserInfoTest');
 table.create=true;
 table.columns.add('name',sqlHelper.sqlserver.NVarChar(50),{nullable:true});
 table.columns.add('pwd',sqlHelper.sqlserver.VarChar(200),{nullable:true});
 table.rows.add('张1','jjasdfienf');
 table.rows.add('张2','jjasdfienf');
 table.rows.add('张3','jjasdfienf');
 * @param {function} func 回调函数 共有两个参数 error:错误信息 rowcount:插入数据的行数
 */
sql.bulkInsert=async function(tableObj){
    try {
        if(tableObj) {
            //创建数据库连接池
            var pool = new mssql.ConnectionPool(config);
            await pool.connect();
            let request = pool.request()
            let result = await request.bulk(tableObj);
            return {state:true,data:result};
        }
        else
        {
            return {state:false,data:"table parameter undefined!"};
        }
    }catch(err){
        return {state:false,data:err};
    }finally {
        await doRelease(pool);
    }
};
/**
 * 如果需要处理大批量的数据行,通常应该使用流
 * @param {string} sqlText 需要执行的sql文本
 * @param {JSON} params 输入参数
 * @param {JSON} func 表示一个回调函数的JSON对象,如下所示:
 * {
    error:function(err){
        console.log(err);
    },
    columns:function(columns){
        console.log(columns);
    },
    row:function(row){
        console.log(row);
    },
    done:function(affected){
        console.log(affected);
    }
 */
sql.queryViaStreamWithParams= async function(sqlText,params,func){
    try {
            //创建数据库连接池
            var pool = new mssql.ConnectionPool(config);
            await pool.connect();
            let request = pool.request();
            request.stream =true;
            if(params){
                for(let index in params){
                    request.input(index,params[index].sqlType,params[index].inputValue);
                }
            }
            request.query(sqlText);
            request.on('recordset', function(columns){
                //columns是一个JSON对象,表示 返回数据表的整个结构,包括每个字段名称以及每个字段的相关属性
                //如下所示
                /*
                { id:
                { index: 0,
                    name: 'id',
                    length: undefined,
                    type: [sql.Int],
                    scale: undefined,
                    precision: undefined,
                    nullable: false,
                    caseSensitive: false,
                    identity: true,
                    readOnly: true },
                    name:
                    { index: 1,
                        name: 'name',
                        length: 100,
                        type: [sql.NVarChar],
                        scale: undefined,
                        precision: undefined,
                        nullable: true,
                        caseSensitive: false,
                        identity: false,
                        readOnly: false },
                    Pwd:
                    { index: 2,
                        name: 'Pwd',
                        length: 200,
                        type: [sql.VarChar],
                        scale: undefined,
                        precision: undefined,
                        nullable: true,
                        caseSensitive: false,
                        identity: false,
                        readOnly: false } }
                */
                func.columns(columns);
            });
            request.on('row', function(row) {
                //row是一个JSON对象,表示 每一行的数据,包括字段名和字段值
                //如 { id: 1004, name: 'jsw', Pwd: '12345678' }
                //如果行数较多,会多次进入该方法,每次只返回一行
                func.row(row);
            });
            request.on('error',async function(err) {
                //err是一个JSON对象,表示 错误信息
                //如下所示:
                /*
                { [RequestError: Incorrect syntax near the keyword 'from'.]
                    name: 'RequestError',
                        message: 'Incorrect syntax near the keyword \'from\'.',
                    code: 'EREQUEST',
                    number: 156,
                    lineNumber: 1,
                    state: 1,
                class: 15,
                    serverName: '06-PC',
                    procName: '' }
                */
                await doRelease(pool);
                func.error(err);
            });
            request.on('done',async function(affected) {
                //affected是一个数值,表示 影响的行数
                //如 0
                //该方法是最后一个执行
                await doRelease(pool);
                await func.done(affected);
            });
    }catch(err){
        console.log("代码执行异常poolFunc,error");
        await doRelease(pool);
        func.error(err);
    }
};
/**
 * 如果需要处理大批量的数据行,通常应该使用流
 * @param {string} sqlText 需要执行的sql文本
 * @param {JSON} func 表示一个回调函数的JSON对象,如下所示:
 * {
    error:function(err){
        console.log(err);
    },
    columns:function(columns){
        console.log(columns);
    },
    row:function(row){
        console.log(row);
    },
    done:function(affected){
        console.log(affected);
    }
 */
sql.queryViaStream=async function(sqlText,func){
   await sql.queryViaStreamWithParams(sqlText,null,func);
};
//释放数据库连接到数据
async function doRelease(connection) {
    try {
        if(connection)
        {
            await connection.close();//释放连接,将连接放回池中
        }
    }
    catch (err) {
        console.error(err.message);
    }
}
module.exports=sql;


 2.长连接写法

加载文件的时候,自动先连接数据库

//导入mssql模块 //基于版本@4.2.1  安装指令npm install mssql@4.2.1
let mssql=require("mssql");
//引用配置参数模块
let configFile = require("./config");
//引用通用方法模块
let common = require("./commonHelper");
//数据库异常
mssql.on('error', err => {
    console.log("mssql异常原因:"+err.message);
});
let sql={};
//sql参数的类型
sql.direction={
    //输入参数
    Input:"input",
    //输出参数
    Output:"output",
    //返回参数
    Return:"return"
};
//配置存储过程是的输出输入
sql.sqlserver=mssql;
//默认config对象
let config=configFile.mssql_config;
/**
 * 初始化连接参数
 * @param {string} user 用户名
 * @param {string} password 密码
 * @param {string} server 服务器地址
 * @param {string} database 数据库名称
 * @param {string} port 数据库端口
 */
sql.initConfig = function(user,password,server,database,port){
    config.user = user;
    config.password =password;
    config.server =server;
    config.database= database;
    config.port = port;
    pool = null;
}
let pool = null;
async function initPool()
{
    if(pool == null)
    {
        pool = await mssql.connect(config);
    }
};
(async ()=>{
    //是否初始化数据库连接
    if(configFile.init_mssql)
    {
        try {
            common.consoleLog("初始化mssql数据库连接");
            await initPool();
        }
        catch (err) {
            console.log(err);
        }
    }
})();
/**
 * 执行存储过程
 * @param {string} procedure 存储过程名称
 * @param {JSON} params 存储过程参数
 * params的定义格式如:
 let params={
    //ID是存储过程的第一个参数,要去掉@符号
    ID:{
        //sqlType是该ID参数在sqlserver中的类型
        sqlType:sql.sqlserver.Int,
        //direction是表明ID参数是输入还是输出(output)参数
        direction:sql.direction.Input,
        //该ID参数的值
        inputValue:1
    },
    //Name是存储过程的第二个参数,要去掉@符号
    Name:{
        sqlType:sqlHelper.sqlserver.Int,
        direction:sqlHelper.direction.Output,
        outputValue:null
    }
};
 * @param {function} func 回调函数 共有四个参数 error:错误信息 recordsets:查询的表结果 returnValue:存储过程的返回值 affected:影响的行数
 */
sql.execute=async function(procedure,params){
    try {
        //创建数据库连接池
        // var pool = new mssql.ConnectionPool(config);
        // await pool.connect();//连接数据库
        await initPool();//连接数据库
        let request = pool.request();
        if (params != null) {
            for (let index in params) {
                if (params[index].direction == sql.direction.Output) {
                    request.output(index, params[index].sqlType);
                }
                else {
                    request.input(index, params[index].sqlType, params[index].inputValue);
                }
            }
        }
        // result 成功返回该结构
        // {
        //     recordsets,
        //     recordset: recordsets && recordsets[0],
        //     output,
        //     rowsAffected,
        //     returnValue
        // }
        let result =await request.execute(procedure);
        for (let index in params) {
            if (params[index].direction == sql.direction.Output) {
                params[index].outputValue = request.parameters[index].value;
            }
        }
        return {state:true,data:result};
    }catch(err){
        await doReconnection(err.message,procedure);
        return {state:false,data:err};
    }
};
/**
 * 执行sql文本(带params参数)
 * @param {string} sqlText 执行的sql语句
 * @param {JSON} params sql语句中的参数
 *  let params={
    //ID是存储过程的第一个参数,要去掉@符号
        ID:{
            //sqlType是该ID参数在sqlserver中的类型
            sqlType:sql.sqlserver.Int,
            //该ID参数的值
            inputValue:1
        }
    }
 * @param {function} func 回调函数 共有两个个参数 error:错误消息 recordsets:查询的结果
 */
sql.queryWithParams=async function(sqlText,params){
    try {
        //创建数据库连接池
        // var pool = new mssql.ConnectionPool(config);
        // await pool.connect();
        await initPool();//连接数据库
        let request = pool.request();
        request.multiple=true;
        if (params != null) {
            for(let index in params){
                request.input(index,params[index].sqlType,params[index].inputValue);
            }
        }
        let result = await request.query(sqlText);
        return {state:true,data:result};
    }catch(err){
        await doReconnection(err.message,sqlText);
        return {state:false,data:err};
    }
};
/**
 * 执行sql文本
 * @param {string} sqlText 执行的sql语句
 * @param {function} func 回调函数 共有两个个参数 error:错误消息 recordsets:查询的结果
 */
sql.query=function(sqlText){
    return sql.queryWithParams(sqlText,null);
}
/**
 * 执行大批量数据的插入
 * @param {sqlserver.Table} table 需要插入的数据表
 * 数据表的定义如下:
 let table=new sql.sqlserver.Table('UserInfoTest');
 table.create=true;
 table.columns.add('name',sqlHelper.sqlserver.NVarChar(50),{nullable:true});
 table.columns.add('pwd',sqlHelper.sqlserver.VarChar(200),{nullable:true});
 table.rows.add('张1','jjasdfienf');
 table.rows.add('张2','jjasdfienf');
 table.rows.add('张3','jjasdfienf');
 * @param {function} func 回调函数 共有两个参数 error:错误信息 rowcount:插入数据的行数
 */
sql.bulkInsert=async function(tableObj){
    try {
        if(tableObj) {
            //创建数据库连接池
            // var pool = new mssql.ConnectionPool(config);
            // await pool.connect();
            await initPool();//连接数据库
            let request = pool.request()
            let result = await request.bulk(tableObj);
            return {state:true,data:result};
        }
        else
        {
            return {state:false,data:"table parameter undefined!"};
        }
    }catch(err){
        await doReconnection(err.message,common.JSON_stringify(tableObj));
        return {state:false,data:err};
    }
};
/**
 * 如果需要处理大批量的数据行,通常应该使用流
 * @param {string} sqlText 需要执行的sql文本
 * @param {JSON} params 输入参数
 * @param {JSON} func 表示一个回调函数的JSON对象,如下所示:
 * {
    error:function(err){
        console.log(err);
    },
    columns:function(columns){
        console.log(columns);
    },
    row:function(row){
        console.log(row);
    },
    done:function(affected){
        console.log(affected);
    }
 */
sql.queryViaStreamWithParams= async function(sqlText,params,func){
    try {
        //创建数据库连接池
        // var pool = new mssql.ConnectionPool(config);
        // await pool.connect();
        await initPool();//连接数据库
        let request = pool.request();
        request.stream =true;
        if(params){
            for(let index in params){
                request.input(index,params[index].sqlType,params[index].inputValue);
            }
        }
        request.query(sqlText);
        request.on('recordset', function(columns){
            //columns是一个JSON对象,表示 返回数据表的整个结构,包括每个字段名称以及每个字段的相关属性
            //如下所示
            /*
            { id:
            { index: 0,
                name: 'id',
                length: undefined,
                type: [sql.Int],
                scale: undefined,
                precision: undefined,
                nullable: false,
                caseSensitive: false,
                identity: true,
                readOnly: true },
                name:
                { index: 1,
                    name: 'name',
                    length: 100,
                    type: [sql.NVarChar],
                    scale: undefined,
                    precision: undefined,
                    nullable: true,
                    caseSensitive: false,
                    identity: false,
                    readOnly: false },
                Pwd:
                { index: 2,
                    name: 'Pwd',
                    length: 200,
                    type: [sql.VarChar],
                    scale: undefined,
                    precision: undefined,
                    nullable: true,
                    caseSensitive: false,
                    identity: false,
                    readOnly: false } }
            */
            func.columns(columns);
        });
        request.on('row', function(row) {
            //row是一个JSON对象,表示 每一行的数据,包括字段名和字段值
            //如 { id: 1004, name: 'jsw', Pwd: '12345678' }
            //如果行数较多,会多次进入该方法,每次只返回一行
            func.row(row);
        });
        request.on('error',async function(err) {
            //err是一个JSON对象,表示 错误信息
            //如下所示:
            /*
            { [RequestError: Incorrect syntax near the keyword 'from'.]
                name: 'RequestError',
                    message: 'Incorrect syntax near the keyword \'from\'.',
                code: 'EREQUEST',
                number: 156,
                lineNumber: 1,
                state: 1,
            class: 15,
                serverName: '06-PC',
                procName: '' }
            */
            func.error(err);
        });
        request.on('done',async function(affected) {
            //affected是一个数值,表示 影响的行数
            //如 0
            //该方法是最后一个执行
            await func.done(affected);
        });
    }catch(err){
        console.log("代码执行异常poolFunc,error");
        await doReconnection(err.message,sqlText);
        func.error(err);
    }
};
/**
 * 如果需要处理大批量的数据行,通常应该使用流
 * @param {string} sqlText 需要执行的sql文本
 * @param {JSON} func 表示一个回调函数的JSON对象,如下所示:
 * {
    error:function(err){
        console.log(err);
    },
    columns:function(columns){
        console.log(columns);
    },
    row:function(row){
        console.log(row);
    },
    done:function(affected){
        console.log(affected);
    }
 */
sql.queryViaStream=async function(sqlText,func){
    await sql.queryViaStreamWithParams(sqlText,null,func);
};
//重新连接数据库
async function doReconnection(message,sqlText) {
    try{
        let time=new Date().Format("HH:mm:ss");
        common.writeLog("mssql_connection",time  + common.partition +sqlText+ common.partition + message);
        console.log(message);
        //释放连接,重新连接mssql
        if(message.search("Failed to connect")>=0 || message.search("Connection is closed")>=0)
        {
            pool = null;//重新初始化mssql连接
            await mssql.close();
        }
    }
    catch (err) {
        console.log(err);
    }
}
module.exports=sql;


    关键字: Nodejs MSSQL


鼓掌

1

正能量

0

0

呵呵

0


评论区