前言
关于ASVS是OWASP开源项目下的关于应用安全的评估标准,可服务于后期RFC评审等等标准对接,经查询3.1版本暂无中文版,于是粗略翻译供查看,附原文链接:
https://www.owasp.org/index.php/Category:OWASP_Application_Security_Verification_Standard_Project
什么是ASVS
OWASP应用程序安全验证标准(ASVS)项目为测试web应用程序技术安全控制提供了基础,并为开发人员提供了安全开发人员的需求列表
V1:架构、设计和威胁建模需求
- 所有的应用程序组件都被识别出来,并知道需要这些组件。
- 安全控制从来不只在客户端执行,而是在各自的远程端点上执行。
- 应用程序和所有连接的远程服务的高级体系结构已经被定义,安全性已经在该体系结构中得到了解决。
- 在应用程序上下文中被认为敏感的数据被清楚地标识出来。
- 所有应用程序组件都是根据它们提供的业务功能和/或安全功能来定义的。
- 已经为应用程序和相关的远程服务生成了一个威胁模型,用于识别潜在的威胁和对策。
- 所有的安全控制都有一个集中的实现。
- 组件通过定义的安全控制(如网络分割、防火墙规则或基于云的安全组)彼此隔离。
- 存在执行应用程序更新的机制。
- 安全性在软件开发生命周期的所有部分中都得到了解决。
- 所有应用程序组件、库、模块、框架、平台和操作系统都没有已知的漏洞
- 对于如何管理加密密钥(如果有的话),有一个明确的策略,并强制执行加密密钥的生命周期。理想情况下,遵循一个关键的管理标准,比如NIST SP 800-57。
V2:身份验证需求
- 验证所有的页面和资源都受到服务器端身份验证的保护,除了那些特别要公开的页面和资源。
- 验证应用程序没有自动填写凭据——可以作为隐藏字段、URL参数、Ajax请求,也可以作为表单,因为这意味着纯文本、可逆或去隐密码存储。随机时间限制是可以接受的作为立场,如保护更改密码表格或忘记密码表格。
- 验证所有身份验证控制是否安全,以确保攻击者不能进入
- 验证密码输入字段,允许或者鼓励使用密码,并且不防止输入长密码或高度复杂的密码。
- 验证所有的鉴定函数(如忘记密码、更改密码、更改电子邮件、管理2FA令牌等)都具有安全控制,作为主要的身份验证机制(如登录表单)。
- 验证修改密码功能包括旧密码、新密码和密码确认。
- 验证所有身份验证决策可以被记录,而不存储敏感的会话标识符或密码。这应该包括对安全性调查所需的相关元数据的请求。
- 验证帐户密码是使用盐进行散列处理的一种方式,并且有足够的工作因素来击败暴力破解和密码散列恢复攻击。
- 验证所有应用程序数据是通过加密的通道(例如TLS)传输的。
- 验证忘记的密码功能和其他路径恢复没有显示当前密码,并且新密码没有以明文发送给用户。应该使用一次性密码重置链接。
- 验证不能通过登录、密码重置或忘记帐户功能进行信息枚举。
- 验证应用程序框架或应用程序使用的任何组件没有使用默认密码(如“admin/password”)。
- 验证反自动化已就绪,以防止违反证书测试、暴力破解和帐户锁定攻击。
- 验证所有访问应用程序外部服务的身份验证凭证都已加密并存储在受保护位置。
- 验证忘记密码和其他恢复路径使用TOTP或其他软标记、移动推送或其他脱机恢复机制。SMS的使用已经被NIST弃用,不应该使用。
- 验证帐户锁定被划分为软锁和硬锁状态,这些不是互斥的。如果一个帐户由于暴力攻击而暂时被软锁定,这不应该重置硬锁定状态。
- 验证如果需要保密问题,这些问题不会违反隐私法,并且足够强大,可以保护帐户免受恶意恢复。
- 验证高值应用程序可以配置为不允许使用大量以前配置的密码。
- 验证敏感操作(例如更改密码、更改电子邮件地址、添加新的订单等等)需要重新验证(例如密码或2FA令牌)。这是对CSRF措施的补充,而不是相反。
- 验证已采取措施阻止常用密码和弱密码短语的使用。
- 验证所有身份验证挑战(无论成功还是失败)都应该在相同的平均响应时间内响应。
- 验证源代码或在线源代码仓库中不包含机密、API密钥和密码。
- 验证用户可以注册并使用TOTP验证、双因素、生物特征识别(Touch ID或类似的)或等效的多因素验证机制,以提供对单因素凭证披露的保护
- 验证对管理接口的访问是严格控制的,不允许不受信任的机构访问。
- 验证应用程序与基于浏览器和第三方的密码管理器兼容,除非被基于政策的风险禁止。
V3:会话管理验证需求。
- 验证当用户注销时会话无效。
- 验证在一定的不活动周期后会话超时。
- 验证在管理可配置的最大时间周期之后的会话超时,而不考虑活动(绝对超时)。
- 验证所有需要身份验证的页面都具有简单和可见的注销功能。
- 测试会话ID从不在URL、错误消息或日志中公开。这包括验证应用程序不支持会话cookie的URL重写。
- 验证所有成功的身份验证和重新身份验证都生成一个新的会话和会话ID。
- 验证应用程序框架生成的会话ID是被应用程序识别为活跃的。
- 根据诸如随机性、惟一性、对统计和图形分析的抵抗力和信息泄漏等标准测试会话ID。
- 验证会话ID存储在cookie中使用“path”属性确定范围;并且启用了“HttpOnly”和“Secure”cookie标志。
- 验证应用程序跟踪所有活跃会话,并且允许用户有选择地或全局地终止会话。
- 验证高价值的应用程序,在成功更改密码过程之后,用户将被提示终止所有其他活动会话。
- TBA
V4:访问控制验证需求
- 验证最小权限原则的存在——用户应该只能够访问具有特定授权的功能、数据文件、URL、控制器、服务和其他资源。这意味着防止欺骗和权限提升。
- 验证对敏感记录的访问是受保护的,这样每个用户只能访问经过授权的对象或数据(例如,防止用户篡改参数以查看或更改另一个用户的帐户)。
- 验证禁止了目录遍历,除非故意要求。此外,应用程序不应该允许发现或者披露文件或者目录元数据,比如Thumbs.db,DS_Store,.git或者.svn文件夹。
- 验证准入控制失败的安全性。
- 验证表示层隐含的相同访问控制规则在服务器端强制执行。
- 验证访问控制所使用的所有用户、数据属性和策略信息不能被终端用户操作,除非经过特别授权。
- 验证存在用于保护对每种受保护资源访问的集中机制(包括调用外部授权服务的库)。‘
- 验证所有访问控制决策被记录,并且所有失败决策也被记录。
- 验证应用程序或框架使用强随机反CSRF令牌或者具有其他事务保护机制。
- 验证系统可以防止聚合或连续访问受保护的功能、资源或数据。例如,考虑使用资源管理器来限制每小时的编辑量,或者防止单个用户剪贴整个数据库。
- 验证应用程序对低价值系统具有额外的授权(例如:提高或者自适应的身份认证),以及对高价值应用程序执行反欺诈控制的职责分离(根据应用程序的风险和过去的欺诈行为)。
12.验证应用程序正确地执行上下文敏感的授权,来防止通过参数篡改来进行未经授权的操作。
V5:输入验证和输出编码验证需求
- 验证服务端输入验证失败导致请求拒绝并被记录。
- 验证在服务端执行输入验证的过程。
- 验证应用程序使用了集中式输入验证控制机制。
- 验证所有数据库查询都受到使用参数化查询保护或正确使用ORM来避免SQL注入。
- 验证应用程序不受LDAP注入的影响,或者安全控制防止LDAP注入。
- 验证应用程序不受OS命令注入的影响,或者安全控制防止OS命令注入。
- 验证应用程序在使用文件路径为内容时,不受远程文件包含或本地文件包含的影响。
- 验证应用程序不受Xpath注入或者XML注入攻击的影响。
- 验证放置在HTML或其他web客户端代码中的所有字符串变量都是准确地根据上下文进行人工编码,或者使用自动根据上下文编码的模板,以确保应用程序不受反射、存储或DOM跨站点脚本(XSS)攻击的影响。
- 验证应用程序不包含大量参数分配(即自动变量绑定)漏洞。
- 验证应用程序有针对HTTP参数污染攻击的防御,特别是如果应用程序框架没有区分请求参数的来源(GET、POST、cookies、header、环境等)。
- 验证所有输入数据都是经过验证的,不仅是HTML表单字段,还包括所有输入源,如REST调用、查询参数、HTTP头、cookie、批处理文件、RSS提要等;使用积极的验证(白名单),然后使用较小的验证表单,如灰色列表(消除已知的坏字符串),或者拒绝恶意的输入(黑名单)。
- 验证结构化数据是强类型的,并根据已定义的模式进行验证,包括允许的字符、长度和样式(例如信用卡号或手机号,或验证两个相关字段是否合理,例如验证郊区和邮政编码是否匹配)。
- 验证非结构化数据被审查执行通用安全措施,例如允许的字符串和长度,在给定的上下文存在潜在危害应该避免的字符串。
- 验证所有来自WYSIWYG编辑器或类似的不受信任的HTML输入都通过HTML审查库或框架特性得到了正确的审查。
- 验证数据从一个DOM环境传输到另一个DOM环境,传输使用安全的JavaScript方法,例如使用innerText或.val来确保应用程序不受DOM跨站点脚本攻击的影响。
- 验证在浏览器或基于JavaScript后端解析JSON时,JSON.parse被用来解析JSON文档,不要使用eval()解析JSON。
- 验证应用程序是否存在服务端请求伪造漏洞。
- 验证应用程序正确地限制XML解析器只使用最严格的配置,并确保禁用诸如解析外部实体之类的危险特性。
- 验证在反序列化无法避免时禁止反序列化不可信数据或者对反序列化进行极度防护。
V7:密码验证需求
- 验证所有加密模块失败的安全性,并且以不支持填充Oracle的方式处理错误。
- 验证所有随机数、随机文件名、随机GUIDs和随机字符串都是在攻击者无法猜测的情况下,使用加密模块的已批准的随机数生成器生成的
- 验证应用程序使用的加密算法是否已根据FIPS 140-2或等效标准进行验证。
- 验证加密模块根据已发布的安全策略在其已批准的模式下运行。
- 验证是否有明确的策略来管理加密密钥(例如,生成、分发、撤销、过期)。验证这个密钥的生命周期被合理执行。
- 加密服务的所有使用者没有直接访问密钥原材料。隔离加密进程,包括主秘密,并考虑使用虚拟化或物理硬件密钥库(HSM)。
- 验证个人身份信息和其他敏感数据是加密存储的。
- 验证在内存中维护的敏感密码或密钥材料在不再需要时就被用零覆盖,以减轻内存转储攻击。
- 验证所有密钥和密码都是可替换的,并在安装时生成或替换。
- 验证随机数是在适当的熵中创建的,即使应用程序处于重载状态,应用程序在这种情况下会优雅地降级。
V8:错误处理和日志验证需求
- 验证应用程序不输出包含敏感数据的错误消息或堆栈跟踪,这些数据可能帮助攻击者,包括会话ID、软件/框架版本和个人信息。
- 验证安全控件中的错误处理逻辑在默认情况下拒绝访问。
- 验证安全日志控件提供记录与安全性相关的成功登录特别是失败事件的能力。
- 验证每个日志事件包含必要的信息,以便在事件发生时对时间轴进行详细的调查。
- 验证包含不可信数据的所有事件将不会作为代码在预期的日志软件查看中执行。
- 验证安全日志受保护不被未经授权的访问和修改。
- 验证应用程序不记录根据本地隐私法律或规章定义的敏感数据、由风险评估定义的组织敏感数据,或可能帮助攻击者的敏感身份验证数据,包括用户的会话标识符、密码、散列或API令牌。
- 验证所有不可打印的符号和字段分隔符在日志条目中正确编码,以防止日志注入。
- 验证来自受信任和不受信任源的日志字段在日志条目中是可区分的。
- 验证审计日志或类似的日志允许密钥事务的不可否认性。
- 验证安全日志有某种形式的完整性检查或控制,以防止未经授权的修改。
- 验证日志存储在与应用程序运行的不同的分区上,并进行适当的日志轮换。
- 确认时间源被同步到正确的时间和时区。
V9:数据保护验证要求
- 验证所有包含敏感信息的表单都禁用了客户端缓存,包括自动完成特性。
- 验证应用程序处理的敏感数据列表是否被识别,并且有一个明确的策略,即如何在相关的数据保护指令下控制、加密和强制访问这些数据。
- 验证所有敏感数据都被通过HTTP消息体或报头发送到服务器(例如,URL参数从不用于发送敏感数据)。
- 验证应用程序设置了足够的反缓存头信息,这样,应用程序或用户输入的任何敏感和个人信息都不应该被主流的现代浏览器缓存到磁盘上(例如:访问:cache来检查磁盘缓存)。
- 验证服务器上存储的包含敏感数据的所有缓存或者临时副本,在授权用户访问敏感数据后包含不被未经授权访问或者权限无效。
- 验证是否有一种方法可以在需求保留策略结束时从应用程序中删除每种类型的敏感数据。
- 验证应用程序最小化请求中的参数数量,例如隐藏字段、Ajax变量、cookie和header值。
- 验证应用程序是否有能力检测和告警异常数量的数据采集请求,例如屏幕抓取。
- 验证存储在客户端存储器中的数据(如HTML5本地存储、会话存储、IndexedDB、常规cookie或Flash cookie)不包含敏感数据或个人身份信息。
- 如果数据时在相关数据保护指令或者需要记录访问日志中收集的,则验证是否记录了对敏感数据的访问。
- 验证在内存中维护的敏感信息在不再需要时就被用零覆盖,以减轻内存转储攻击。
- 验证在客户端或会话终止后,已验证的数据已从客户端存储中(如浏览器DOM)清除。
V10:通信验证需求
- 验证可构建一条路径从可信的CA证书到美国传输层安全(TLS)服务器证书,并且每个服务器证书都是有效的。
- 验证TLS用于经过身份验证的或涉及敏感数据或功能的所有连接(包括外部和后端连接),并且不会退回到不安全或未加密的协议。确保最强的选择是首选算法。
- 验证后端TLS连接失败被记录。
- 使用配置的信任锚点和撤销信息来验证所有客户端证书的证书路径。
- 验证所有涉及敏感信息或功能的外部系统连接都经过了身份验证。
- 验证应用程序使用的单一标准的TLS被配置为运行在批准的操作模式中。
- 验证TLS证书公钥固定(HPKP)是通过生产和备份公钥实现的。有关更多信息,请参阅下面的参考资料。
- 验证HTTP严格安全传输请求头包含在所有请求和所有子域,例如严格安全传输:max-age=15724800;includeSubdomains。
- 验证产品网站URL已提交给由web浏览器供应商维护的严格安全传输域的预加载列表。请参阅下面的参考资料。
- 验证完美的前向保密配置以减少被动攻击者记录流量。
- 验证是否启用并配置了适当的证书撤销,如在线证书状态协议(OCSP)封套。
- 验证仅使用强算法、密码和协议通过所有证书层次结构,包括所选机构的根证书和中间证书。
- 验证TLS设置是否符合当前的主要实践,特别是当常见配置、密码和算法变得不安全时。
V13:恶意代码验证要求
- 验证所有的恶意活动都被充分地沙箱、容器化或隔离,以延迟和阻止攻击者攻击其他应用程序。
- 验证应用程序源代码和尽可能多的第三方库不包含后门、复活节彩蛋以及在身份验证、访问控制、输入验证和高价值业务员逻辑中的逻辑缺陷。
V15:业务逻辑验证需求
- 验证应用程序将只按顺序的步骤处理业务逻辑流,所有步骤都在实际的人工时间内进行处理,而不是乱序执行、跳过步骤、从另一个用户处理步骤或提交的事务过于迅速。
- 验证应用程序是否具有业务限制,并根据每个用户正确地实施,使用可配置的警报和对自动或异常攻击的自动反应。
V16:文件和资源验证需求
- 验证URL重定向和转发只允许白名单目标地址,或者在重定向到可能不受信任的内容时显示警告。
- 验证提交给应用程序的不受信任的文件数据没有直接用于文件I/O命令,特别是为了防止路径遍历、本地文件包含、文件MIME类型、反射文件下载和OS命令注入漏洞。
- 验证从不可信源获取的文件被验证为可预期类型,并通过反病毒扫描器进行扫描,以防止已知的恶意内容的上传。
- 验证不受信任的数据不用于包含、类加载或反射功能,以防止远程/本地代码执行漏洞。
- 验证不可信的数据不会被用于跨域资源共享(CORS)以保护不受任意远程内容的影响。
- 验证从不受信任的源获取的文件存储在webroot之外,具有有限的权限,最好具有强验证。
- 验证网站或应用服务器默认配置为拒绝访问网站或应用服务器之外的远程资源或系统。
- 验证应用程序代码没有执行从不可信源获得的上传数据。
- 验证不支持、不安全或已弃用的客户端技术没有被使用,例如NSAPI插件、Flash、Shockwave、Active-X、Silverlight、NACL或客户端Java applets。
- 验证跨域资源共享(CORS)Access-Control-Origin请求头并不仅仅显示请求的源头或支持“null”源。
V17:手机验证的需求
V18:API和Web服务验证需求
- 验证客户端和服务端之间使用了相同的编码风格。
- 验证Web服务应用程序中对管理功能的访问仅限于Web服务管理员。
- 在接受输入之前,验证XML或JSON模式是否启用并进行验证。
- 验证所有输入都被限制在适当的大小界限。
- 验证基于SOAP的web服务是否符合Web Services-Interoperability(WS-I)基本概要。这实质上意味着TLS加密。
- 通过使用一种或多种以下方法来保护REST服务不受跨站请求伪造的影响:双提交cookie模式,CSRF随机数,ORIGIN请求头检查和referrer请求头检查。
- 验证REST服务明确地检查传入的Content-Type是否为预期类型,例如application/xml或者application/json。
- 验证使用JSON Web签名或者SOAP请求的WS-Security的Payload信息是否已签名,以确保客户端与服务端之间传输的可靠性。
- 验证可供选择的和不太安全的访问路径不存在。
V19:配置验证需求
- 验证所有组件都是最新的,具有适当的安全配置和版本。这应该包括删除不必要的配置和文件夹,如示例应用程序、平台文档和默认或示例用户。
- 验证组件之间的通信(例如应用服务器和数据库服务器之间的通信)是加密的,特别是当组件位于不同的容器或不同的系统中时。
- 验证组件之间的通信(例如应用服务器和数据库服务器之间的通信)使用具有最小必要权限的帐户进行验证。
- 验证应用程序部署足够地沙箱、容器或隔离化,以延迟和阻止攻击者攻击其他应用程序。
- 验证应用程序构建和部署过程是在安全且可重复的方法中执行的,例如CI / CD自动化和自动配置管理。
- 验证授权管理员有能力验证所有与安全相关的配置的完整性,以检测篡改。
- 验证所有的应用程序组件都已签名。
- 验证第三方组件来自可信任的仓库。
- 验证系统级语言的构建过程是否启用了所有安全标志,如ASLR、DEP和安全检查。
- 验证所有应用程序资产都由应用程序承载,比如JavaScript库、CSS样式表和web字体都由应用程序承载,而不是依赖于CDN或外部供应商。
- 验证所有应用程序组件、服务和服务器都使用它们自己的低特权服务帐户,这在应用程序之间不共享,管理员也不使用。
V20:IoT验证需求
- 验证类似USB或串行接口的应用层调试已禁用。
- 验证加密密钥对每个设备都是唯一的。
- 如果适用,请验证嵌入式/物联网操作系统是否启用了诸如ASLR和DEP之类的内存保护控件。
- 验证是否禁用了诸如JTAG或SWD之类的片上调试接口,或者是否启用并适当配置了可用的保护机制。
- 验证设备上不会展示物理调试头。
- 验证敏感数据没有在设备上未加密存储。
- 验证设备能够防止敏感信息泄露。
- 验证固件应用程序使用传输安全性保护数据传输。
- 验证固件应用程序有效验证服务端连接的数字签名。
- 验证无线通信是相互验证的。
- 验证无线通信在加密通道进行传输。
- 验证固件应用程序将数字签名固定到可信任的服务器上。
- 验证物理防篡改和防篡改检测特性的存在,包括epxy。
- 验证芯片上的识别标记已删除。
- 验证芯片制造商提供的任何可用的知识产权保护技术已启用。
- 验证安全控件启用来阻止对固件进行逆向工程(例如删除冗长的调试字符串)。
- 验证设备在加载之前验证引导映像签名。
- 验证固件更新过程不容易受到time-of-check和time-of-use攻击的影响。
- 验证设备在安装之前使用代码签名并验证固件升级文件。
- 验证设备不能被降级为有效固件的旧版本。
- 验证在嵌入式设备上使用加密安全的伪随机数生成器(例如使用芯片提供的随机数生成器)。
- 验证设备在检测到篡改或收到无效消息时擦拭固件和敏感数据。
- 验证只使用支持禁用调试接口的微控制器(例如JTAG、SWD)。
- 验证只使用对去盖和侧道攻击提供有效保护的微控制器。
- 验证敏感痕迹没有暴露在印刷电路板的外层。
- 验证芯片间的通信是加密的。
- 验证设备在执行前使用代码签名并对代码进行验证。
- 验证在内存中维护的敏感信息当不在需要时就用零覆盖。
- 验证固件应用程序使用内核容器在应用程序之间进行隔离。