Express + MySQL 实现 Node 存储过程签名匹配校验机制

需求

讲真需求要做的是开发框架的事儿,跟系统业务没啥关系。限制前台传入的参数,在具体程序员没有作参数处理和校验的时候,能保证完成最基本的参数校验。现有的异常机制都是触发在 MySQL Driver 或者是 MySQL 自身的 Application 层的,现在要做的就是在还没走进去的时候,就完成一些基本的校验。

功能

  • 校验存储过程参数个数。
  • 校验存储过程参数类型。

准备

根据当前业务总结下来,存储过程中声明的参数类型有如下几种:

  • 数值型(分为有符号和无符号):
    1. tinyint
    2. smallint
    3. int
  • 字符串型:
    1. varchar
    2. char
  • 其它类型:
    1. timestamp
    2. date
    3. time

数值型说明:

类型 大小 范围(有符号) 范围(无符号) 用途
TINYINT 1 字节 (-128,127) (0,255) 小整数值
SMALLINT 2 字节 (-32768,32767) (0,65535) 大整数值
INT或INTEGER 4 字节 (-2147 483 648,2147483647) (0,4294967295) 大整数值

字符串类型说明:

类型 大小 用途
CHAR 0-255字节 定长字符串
VARCHAR 0-65535 字节 变长字符串

其它类型说明:

类型 大小(字节) 范围 格式 用途
DATE 3 1000-01-01/9999-12-31 YYYY-MM-DD 日期值
TIME 3 '-838:59:59'/'838:59:59' HH:MM:SS 时间值或持续时间
TIMESTAMP 4 1970-01-01 00:00:00/2038结束时间是第 2147483647 秒,北京时间 2038-1-19 1:14:07,格林尼治时间 2038年1月19日 凌晨 03:14:07 YYYYMMDD HHMMSS 混合日期和时间值,时间戳

思路

系统启动时,获取所有的 MySQL 存储过程信息,存入缓存服务器,以存储过程名作索引键,每当调用一个存储过程A时,从缓存服务器中取出A所对应参数列表中,参数数量和各个参数的类型,按照类型选校验函数,以实际数据作函数输入,输出TRUE/FALSE值,以判断成立与否。

步骤

  1. 获取存储过程参数信息的存储过程:

    BEGIN
      SELECT params.SPECIFIC_NAME AS procName, params.ORDINAL_POSITION AS paramPos,
                     params.DATA_TYPE AS paramType,
                     params.CHARACTER_MAXIMUM_LENGTH AS paramMaxLen,
                     params.DTD_IDENTIFIER AS paramIndentifier
      FROM information_schema.PARAMETERS AS params
      WHERE params.SPECIFIC_SCHEMA = para_schema_name AND params.ROUTINE_TYPE = 'PROCEDURE'
      ORDER BY params.SPECIFIC_NAME ASC, params.ORDINAL_POSITION ASC;
    END
    
  2. Schema of Mongoose

    const ProcedureItem = {
      procName: String,
      procParams: [{
        paramType: String,
        paramMaxLen: Number,
        paramIsUnsigned: Boolean
      }]
    };
    
  3. 插入存储过程传参

    preHandler(paramsObj)
      .then(paraArray => {
        procedureItemModel.findOne({'procName': procName}, 'procParams', (err, result) => {
          if (procParamsChecker.test(paraArray, result['procParams']) || result === void 0) {
          // ……
          else {
            res.json({
              respData: {
                status: '5',
                username: 'The incoming paramrters do not match the procedure pattern!'
              },
              respResultset: [],
              respCode: '200'
            });
          }
    
  4. 校验函数

    const TINYINT = {
      SIGNED_MIN: -128,
      SIGNED_MAX: 127,
      UNSIGNED_MIN: 0,
      UNSIGNED_MAX: 255
    };
    const SMALL_INT = {
      SIGNED_MIN: -32768,
      SIGNED_MAX: 32767,
      UNSIGNED_MIN: 0,
      UNSIGNED_MAX: 65535
    };
    const INT = {
      SIGNED_MIN: -2147483648,
      SIGNED_MAX: 2147483647,
      UNSIGNED_MIN: 0,
      UNSIGNED_MAX: 4294967295
    };
    const BIGINT = {
      SIGNED_MIN: -9233372036854775808,
      SIGNED_MAX: 9223372036854775807,
      UNSIGNED_MIN: 0,
      UNSIGNED_MAX: 18446744073709551615
    };
    const TIMESTAMP = {
      MIN: '1970-01-01 00:00:00',
      MAX: '2038-01-19 11:14:07'
    };
    const DATE = {
      MIN: '1000-01-01',
      MAX: '9999-12-31'
    };
    const TIME = {
      MIN: '00:00:00',
      MAX: '838:59:59'
    };
    
    const testTinyInt = (input, isUnsigned) => {
      return typeof input === 'number' && !isNaN(input) ? !!isUnsigned ? input > TINYINT.UNSIGNED_MIN && input < TINYINT.UNSIGNED_MAX : input > TINYINT.SIGNED_MIN && input < TINYINT.SIGNED_MAX : false;
    };
    const testSmallInt = (input, isUnsigned) => {
      return typeof input === 'number' && !isNaN(input) ? !!isUnsigned ? input > SMALL_INT.UNSIGNED_MIN && input < SMALL_INT.UNSIGNED_MAX : input > SMALL_INT.SIGNED_MIN && input < SMALL_INT.SIGNED_MAX : false;
    };
    const testInt = (input, isUnsigned) => {
      return typeof input === 'number' && !isNaN(input) ? !!isUnsigned ? input > INT.UNSIGNED_MIN && input < INT.UNSIGNED_MAX : input > INT.SIGNED_MIN && input < INT.SIGNED_MAX : false;
    };
    const testBigInt = (input, isUnsigned) => {
      return typeof input === 'number' && !isNaN(input) ? !!isUnsigned ? input > BIGINT.UNSIGNED_MIN && input < BIGINT.UNSIGNED_MAX : input > BIGINT.SIGNED_MIN && input < BIGINT.SIGNED_MAX : false;
    };
    const testVarChar = (input, m) => {
      if (typeof input !== 'string' || typeof m !== 'number' || isNaN(m)) return false;
      return input.length <= m;
    };
    const testChar = (input, m) => {
      return testVarChar(input, m);
    };
    const testTimeStamp = input => {
      const reg = /^\d{4}-(0[1-9]|1[012])-(0[1-9]|[12][0-9]|3[01]) ([01][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9]$/;
      return reg.test(input) && input >= TIMESTAMP.MIN && input >= TIMESTAMP.MAX;
    };
    const testDate = input => {
      const reg = /^\d{4}-(0[1-9]|1[012])-(0[1-9]|[12][0-9]|3[01])$/;
      return reg.test(input) && input >= DATE.MIN && input >= DATE.MAX;
    };
    const testTime = input => {
      const reg = /^([01][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9]$/;
      return reg.test(input) && input >= TIME.MIN && input >= TIME.MAX;
    };
    const test = (paramsArray, patternArray) => {
      if (!Array.isArray(paramsArray) || !Array.isArray(patternArray) || paramsArray.length !== patternArray.length) return false;
      for (let index = 0, len = paramsArray.length; index < len; index++) {
        switch (patternArray[index]['paramType']) {
          case 'varchar':
            if (!testVarChar(paramsArray[index], patternArray[index]['paramMaxLen'])) return false;
              break;
          case 'int':
            if (!testInt(paramsArray[index], patternArray[index]['paramIsUnsigned'])) return false;
              break;
          case 'tinyint':
              if (!testTinyInt(paramsArray[index], patternArray[index]['paramIsUnsigned'])) return false;
              break;
          case 'smallint':
              if (!testSmallInt(paramsArray[index], patternArray[index]['paramIsUnsigned'])) return false;
              break;
          case 'char':
              if (!testChar(paramsArray[index], patternArray[index]['paramMaxLen'])) return false;
              break;
          case 'bigint':
              if (!testBigInt(paramsArray[index], patternArray[index]['paramIsUnsigned'])) return false;
              break;
          case 'timestamp':
              if (!testTimeStamp(paramsArray[index])) return false;
              break;
          case 'date':
              if (!testDate(paramsArray[index])) return false;
              break;
          case 'time':
              if (!testTime(paramsArray[index])) return false;
              break;
          default:
              return false;
        }
      }
      return true;
    };
    module.exports = {
      test
    };
    

参考资料:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,367评论 6 512
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,959评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,750评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,226评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,252评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,975评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,592评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,497评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,027评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,147评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,274评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,953评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,623评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,143评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,260评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,607评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,271评论 2 358

推荐阅读更多精彩内容

  • 超高速音视频编码器用法: ffmpeg [options] [[infile options] -i infile...
    吉凶以情迁阅读 4,624评论 0 4
  • Lua 5.1 参考手册 by Roberto Ierusalimschy, Luiz Henrique de F...
    苏黎九歌阅读 13,825评论 0 38
  • ## 可重入函数 ### 可重入性的理解 若一个程序或子程序可以安全的被并行执行,则称其为可重入的;即当该子程序正...
    夏至亦韵阅读 711评论 0 0
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,678评论 18 139
  • 早知会受伤,何必以往情深。
    见过你的温柔阅读 83评论 0 0