learn-tech/专栏/Java核心技术面试精讲/32如何写出安全的Java代码?-极客时间.md
2024-10-16 06:37:41 +08:00

134 lines
11 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

因收到Google相关通知网站将会择期关闭。相关通知内容
32 如何写出安全的Java代码-极客时间
在上一讲中,我们已经初步接触了 Java 安全,今天我们将一起探讨更多 Java 开发中可能影响到安全的场合。很多安全问题,在特定的上下文,存在着不同的定义,尽管本质是相似或一致的,这是由于 Java 平台自身的特性所带来特有的问题。今天这一讲我将侧重于 Java 开发者的角度谈代码安全,而不是讲广义的安全风险。
今天我要问你的问题是,如何写出安全的 Java 代码?
典型回答
这个问题可能有点宽泛我们可以用特定类型的安全风险为例如拒绝服务DoS攻击分析 Java 开发者需要重点考虑的点。
DoS 是一种常见的网络攻击,有人也称其为“洪水攻击”。最常见的表现是,利用大量机器发送请求,将目标网站的带宽或者其他资源耗尽,导致其无法响应正常用户的请求。
我认为,从 Java 语言的角度,更加需要重视的是程序级别的攻击,也就是利用 Java、JVM 或应用程序的瑕疵,进行低成本的 DoS 攻击,这也是想要写出安全的 Java 代码所必须考虑的。例如:
如果使用的是早期的 JDK 和 Applet 等技术,攻击者构建合法但恶劣的程序就相对容易,例如,将其线程优先级设置为最高,做一些看起来无害但空耗资源的事情。幸运的是类似技术已经逐步退出历史舞台,在 JDK 9 以后,相关模块就已经被移除。
上一讲中提到的哈希碰撞攻击,就是个典型的例子,对方可以轻易消耗系统有限的 CPU 和线程资源。从这个角度思考,类似加密、解密、图形处理等计算密集型任务,都要防范被恶意滥用,以免攻击者通过直接调用或者间接触发方式,消耗系统资源。
利用 Java 构建类似上传文件或者其他接受输入的服务需要对消耗系统内存或存储的上限有所控制因为我们不能将系统安全依赖于用户的合理使用。其中特别注意的是涉及解压缩功能时就需要防范Zip bomb等特定攻击。
另外Java 程序中需要明确释放的资源有很多种,比如文件描述符、数据库连接,甚至是再入锁,任何情况下都应该保证资源释放成功,否则即使平时能够正常运行,也可能被攻击者利用而耗尽某类资源,这也算是可能的 DoS 攻击来源。
所以可以看出,实现安全的 Java 代码,需要从功能设计到实现细节,都充分考虑可能的安全影响。
考点分析
关于今天的问题,以典型的 DoS 攻击作为切入点,将问题聚焦在 Java 开发中,我介绍了 Java 应用设计、实现的注意事项,后面还会介绍更加全面的实践。
其实安全问题实际就是软件的缺陷,软件安全并不存在一劳永逸的秘籍,既离不开设计、架构中的风险分析,也离不开编码、测试等阶段的安全实践手段。对于面试官来说,考察安全问题,除了对特定安全领域知识的考察,更多是要看面试者的 Java 编程基本功和知识的积累。
所以,我会在后面会循序渐进探讨 Java 安全编程,这里面没有什么黑科技,只有规范的开发标准,很多安全问题其实是态度问题,取决于你是否真的认真对待它。
我将以一些典型的代码片段为出发点,分析一些非常容易被忽略的安全风险,并介绍安全问题频发的热点场景,如 Java 序列化和反序列化。
从软件生命周期的角度,探讨设计、开发、测试、部署等不同阶段,有哪些常见的安全策略或工具。
知识扩展
首先,我们一起来看一段不起眼的条件判断代码,这里可能有什么问题吗?
// a, b, c都是int类型的数值
if (a + b < c) {
//
}
你可能会纳闷这是再常见不过的一个条件判断了能有什么安全隐患
这里的隐患是数值类型需要防范溢出否则这不仅仅可能会带来逻辑错误在特定情况下可能导致严重的安全漏洞
从语言特性来说Java JVM 提供了很多基础性的改进相比于传统的 CC++ 等语言对于数组越界等处理要完善的多原生的避免了缓冲区溢出等攻击方式提高了软件的安全性但这并不代表完全杜绝了问题Java 程序可能调用本地代码也就是 JNI 技术错误的数值可能导致 C/C++ 层面的数据越界等问题这是很危险的
所以上面的条件判断需要判断其数值范围例如写成类似下面结构
if (a < c b)
再来看一个例子请看下面的一段异常处理代码
try {
// 业务代码
} catch (Exception e) {
throw new RuntimeException(hostname + port + doesnt response);
}
这段代码将敏感信息包含在异常消息中试想如果是一个 Web 应用异常也没有良好的包装起来很有可能就把内部信息暴露给终端客户古人曾经告诫我们言多必失是很有道理的虽然其本意不是指软件安全但尽量少暴露信息也是保证安全的基本原则之一即使我们并不认为某个信息有安全风险我的建议也是如果没有必要不要暴露出来
这种暴露还可能通过其他方式发生比如某著名的编程技术网站就被曝光过所有用户名和密码这些信息都是明文存储传输过程也未必进行加密类似这种情况暴露只是个时间早晚的问题
对于安全标准特别高的系统甚至可能要求敏感信息被使用后要立即明确在内存中销毁以免被探测或者避免在发生 core dump 意外暴露
第三Java 提供了序列化等创新的特性广泛使用在远程调用等方面但也带来了复杂的安全问题直到今天序列化仍然是个安全问题频发的场景
针对序列化通常建议
敏感信息不要被序列化在编码中建议使用 transient 关键字将其保护起来
反序列化中建议在 readObject 中实现与对象构件过程相同的安全检查和数据检查
另外 JDK 9 Java 引入了过滤器机制以保证反序列化过程中数据都要经过基本验证才可以使用其原理是通过黑名单和白名单限定安全或者不安全的类型并且你可以进行定制然后通过环境变量灵活进行配置 更加具体的使用你可以参考 ObjectInputFilter
通过前面的介绍你可能注意到很多安全问题都是源于非常基本的编程细节类似 Immutable封装等设计都存在着安全性的考虑从实践的角度让每个人都了解和掌握这些原则有必要但并不太现实有没有什么工程实践手段可以帮助我们排查安全隐患呢
开发和测试阶段
在实际开发中各种功能点五花八门未必能考虑的全面我建议没有必要所有都需要自己去从头实现尽量使用广泛验证过的工具类库不管是来自于 JDK 自身还是 Apache 等第三方组织都在社区的反馈下持续地完善代码安全
开发过程中应用代码规约标准是避免安全问题的有效手段我特别推荐来自孤尽的阿里巴巴 Java 开发手册以及其配套工具充分总结了业界在 Java 等领域的实践经验将规约实践系统性地引入国内的软件开发可以有效提高代码质量
当然凡事都是有代价的规约会增加一定的开发成本可能对迭代的节奏产生一定影响所以对于不同阶段不同需求的团队可以根据自己的情况对规约进行适应性的调整
落实到实际开发流程中 OpenJDK 团队为例我们应用了几个不同角度的实践
在早期设计阶段就由安全专家组对新特性进行风险评估
开发过程中尤其是 code review 阶段应用 OpenJDK 自身定制的代码规范
利用多种静态分析工具如FindBugsParfait等帮助早期发现潜在安全风险并对相应问题采取零容忍态度强制要求解决
甚至 OpenJDK 会默认将任何编译等警告都当作错误对待并体现在 CI 流程中
在代码 check-in 等关键环节利用 hook 机制去调用规则检查工具以保证不合规代码不能进入 OpenJDK 代码库
关于静态分析工具的选择我们选取的原则是足够好没有什么工具能够发现所有问题所以在保证功能的前提下影响更大的是分析效率换句话说是代码分析的噪音高低不管分析有多么的完备如果太多误报就会导致有用信息被噪音覆盖也不利于后续其他程序化的处理反倒不利于排查问题
以上这些是为了保证 JDK 作为基础平台的苛刻质量要求在实际产品中你需要斟酌具体什么程度的要求是合理的
部署阶段
JDK 自身的也是个软件难免会存在实现瑕疵我们平时看到 JDK 更新的安全漏洞补丁其实就是在修补这些漏洞我最近还注意到某大厂后台被曝出了使用的 JDK 版本存在序列化相关的漏洞类似这种情况大多数都是因为使用的 JDK 是较低版本算是可以通过部署解决的问题
如果是安全敏感型产品建议关注 JDK 在加解密方面的路线图同样的标准也应用于其他语言和平台很多早期认为非常安全的算法已经被攻破及时地升级基础软件是安全的必要条件
攻击和防守是不对称的只要有一个严重漏洞对于攻击者就足够了所以不能对黑盒形式的部署心存侥幸这并不能保证系统的安全攻击者可以利用对软件设计的猜测结合一系列手段探测出漏洞
今天我以 DoS 等典型攻击方式为例分析了其在 Java 平台上的特定表现并从更多安全编码的细节帮你体会安全问题的普遍性最后我介绍了软件开发周期中的安全实践希望能对你的工作有所帮助
一课一练
关于今天我们讨论的题目你做到心中有数了吗你在开发中遇到过 Java 特定的安全问题吗是怎么解决的呢
请你在留言区写写你对这个问题的思考我会选出经过认真思考的留言送给你一份学习奖励礼券欢迎你与我一起讨论
别忘了今晚 8 点半我会做客极客 Live和你一起聊聊 Java 面试那些事儿极客时间App 内点击极客 Live即可加入直播今晚我们不见不散
你的朋友是不是也在准备面试呢你可以请朋友读把今天的题目分享给好友或许你能帮到他