原型污染漏洞源于JavaScript原型继承机制被恶意篡改,攻击者通过不安全的对象合并、深拷贝或动态赋值将__proto__等敏感键注入Object.prototype,影响所有对象行为;典型场景包括lodash.merge处理用户输入、手动递归赋值未过滤敏感键、JSON.parse后直接合并及动态路径赋值。
识别原型污染漏洞,关键在于发现代码中是否允许用户输入直接影响对象原型链。它不依赖复杂利用链,往往一个合并操作、一次反序列化或一段动态赋值就可能触发。
以下几类操作是典型高危入口,需逐行审查:
_.merge、_.set、_.defaultsDeep(lodash ≤4.17.20)等函数处理用户提交的 JSON、查询参数或表单数据for (let key in source) target[key] = source[key],且未过滤 __proto__、constructor、prototype
JSON.parse() 结果不做清洗,直接合并进配置对象或路由参数容器obj[path] = value,其中 path 来自 URL 或请求体(例如 "a.__proto__.b")在开发或测试环境运行以下检查,能直观确认系统是否已受污染:
JSON.parse('{"__proto__":{"polluted":true}}'),再检查 ({ }).polluted === true 是否成立JSON.parse('{"constructor":{"prototype":{"xss":1}}}'),随后验证 ({}).xss === 1
true,说明原型已被写入,当前环境存在可利用漏洞防护不是加一层“安全中间件”,而是贯穿输入解析、对象操作、运行时环境三个环节:
__proto__、constructor、prototype 的键路径,正则推荐 /b(__proto__|constructor|prototype)b/i
Object.create(null) 替代 {},避免继承污染源头Object.assign({}, input)(浅拷贝);必须深拷贝时,选用 structuredClone()(现代环境)或自研函数跳过敏感键--disable-proto=delete 类 Node.js 启动参数防御不能止于代码修复,真实攻击常绕过静态检查:
JSON.parse、req.body、fs.readFileSync 入口添加字段扫描,命中敏感键立即告警{"__proto__":{"isAdmin":true}}),验证是否被静默接受eslint-plugin-security,标记 _.merge、JSON.parse 后无清洗的调用位置