由于我们已经涉及到了 Subject reference,我们知道 Subject 是“当前执行”用户的特定安全视图,且该 Subject 实 例绑定到一个线程来确保我们知道在线程执行期间的任何时间是谁在执行逻辑。   这意味着三个基本的东西必须始终出现,为了能够支持访问当前正在执行的 Subject:

1. 必须创建一个 Subject 实例

2. Subject 实例必须绑定当前执行的线程。

3. 在线程完成执行后(或如果该线程执行抛出异常),该 Subject 必须解除绑定来确保该线程在任何线程池环境 中保持'clean'。

Shiro 拥有为正在运行的应用程序自动地执行绑定//解除绑定逻辑的建筑组件。例如,在 Web 应用程序中,当过滤 一个请求时,Shiro 的根过滤器执行该逻辑。但由于测试环境和框架不同,我们需要自己选择自己的测试框架来执 行此绑定/解除绑定逻辑。

Test Setup

我们知道在创建一个 Subject 实例后,它必须被绑定线程。在该线程(或在这个例子中,是一个 test)完成执行后, 我们必须解除 Subject 的绑定来保持线程的'clean'.   幸运的是,现代测试框架如 JUnit 和 TestNG 已经能够在本地支持'setup'和'teardown'的概念。我们可以利用这一支持 来模拟 Shiro 在一个“完整的”应用程序中会做些什么。我们已经在下面创建了一个你能够在你自己的测试中使用 的抽象基类——随意复制和修改如果你觉得合适的话。它能够在单元测试和集成测试中使用(我在本例中使用 JUnit, 但 TestNG 也能够工作得很好):

Testing & Frameworks

在 AbstractShiroTest 类中的代码使用 Shiro 的 ThreadState 概念及一个静态的 SecurityManager。这些技术在测试和框 架代码中是很有用的,但几乎不曾在应用程序代码中使用。 大多数使用 Shiro 工作的需要确保线程的一致性的终端用户,几乎总是使用 Shiro 的自动管理机制,即 Subject.associateWith 和 Subject.execute 方法。这些方法包含在 Subject thread association 参考文献中。

Unit Testing

单元测试主要是测试你的代码,且你的代码是在有限的作用域内。当你考虑到 Shiro 时,你真正要关注的是你的代 码能够与 Shiro 的 API 正确的运行——你不会想做关于 Shiro 的实现是否工作正常(这是 Shiro 开发团队在 Shiro 的代 码库必须确保的东西)的必要测试。   测试 Shiro 的实现是否与你的实现协同工作是真实的集成测试(下面讨论)。

ExampleShiroUnitTest

由于单元测试适用于测试你的逻辑(而不是你可能调用的任何实现),这对于模拟你逻辑所依赖的任何 API 来说是 个很好的主意。这能够与 Shiro 工作得很好——你可以模拟 Subject 实例,并使它反映任何情况下你所需的反应,这 些反应是处于测试的代码做出的。         但正如上文所述,在 Shiro 测试中关键是要记住在测试执行期间任何 Subject 实例(模拟的或真实的)必须绑定到线 程。因此,我们所需要做的是绑定模拟的 Subject 以确保如预期进行。   (这个例子使用 EasyMock,但 Mockito 也同样地工作得很好):

正如你所看到的,我们没有设立一个 Shiro SecurityManager 实例或配置一个 Realm 或任何像这样的东西。我们简单 地创建一个模拟 Subject 实例,并通过调用 setSubject 方法将它绑定到线程。这将确保任何在我们测试代码中的调用 或在代码中我们正测试的 SecurityUtils.getSubject()正常工作。   请注意,setSubject 方法实现将绑定你的模拟 Subject 到线程,且它仍将存在,直到你通过一个不同的 Subject 调用 setSubject 或直到你明确地通过调用 clearSubject()将它从线程中清除。   保持 Subject 绑定到该线程多长时间(或在一个不同的测试中用来交换一个新的实例)取决于你及你的测试需求。

tearDownSubject()

在实例中的 tearDownSubject()方法使用了 Junit 4 的注释来确保该 Subject 在每个测试方法执行后被清除,不管发生 什么。这要求你设立一个新的 Subject 实例并将它设置到每个需要执行的测试中。   然而这也不是绝对必要的。例如,你可以只每个测试开始时绑定一个新的 Subject 实例(通过 setSubject),也就是 说,使用@Before-annotated 方法。但如果你将要这么做,你可以同时使用@After tearDownSubject() 方法来保持对 称及'clean'。   你可以手动地在每个方法中混合及匹配该 setup/teardown 逻辑或使用@Before 和@After 注释只要你认为合适。所有 测试完成后,AbstractShiroTest 超类在无论怎样都会将 Subject 从线程解除绑定,因为@After 注释在它的 tearDownShiro()方法中。

Integration Testing

现在我们讨论了单元测试的设置,让我们讨论一些关于集成测试的东西。集成测试是指测试跨 API 边界的实现。例 如,测试当调用 B 实现时 A 实现是否工作,且 B 实现是否做它该做的事情。   你同样也可以在 Shiro 中轻松地执行集成测试。Shiro 的 SecurityManager 实例及它所包含的东西(如 Realms 和 SessionManager 等)都是占用很少内存的非常轻量级的 POJO。这意味着你可以为每一个你执行的测试类创建并销毁 一个 SecurityManager 实例。当你的集成测试运行时,它们将使用“真实的”SecurityManager,且与你应用程序中 相像的 Subject 实例将会在运行时使用。

ExampleShiroIntegrationTest 下面的实例代码看起来与上面的单元测试实例几乎相同,但这 3 个步骤却有些不同:

1. 现在有了 step '0',它用来设立一个“真实的”SecurityManager 实例。

2. Step 1 现在通过 Subject.Builder 构造一个“真实的”Subject 实例,并将它绑定到线程。

线程的绑定与解除绑定(step 2 和 3)与单元测试实例中的作用一样。

正如你所看到的,一个具体的 SecurityManager 实现被实例化,并通过 setSecurityManager 方法使其余的测试能够对 其进行访问。然后测试方法能够使用该 SecurityManager,当使用 Subject.Builder 后通过调用 getSecurityManager()方 法。   还要注意 SecurityManager 实例在@BeforeClass 设置方法中只被设置一次——一个对于大多数测试类较为普遍的做 法。如果你想,你可以创建一个新的 SecurityManager 实例并在任何时候从任何测试方法通过 setSerurityManager 来 设置它——例如,你可能会引用两个不同的.ini 文件来构建一个根据你的测试需求而来的新 SecurityManager。   最后,与单元测试例子一样,AbstractShiroTest 超类将会清除所有 Shiro 产物(任何存在的 SecurityManager 及 Subject 实例)通过它的@AfterClass tearDownShiro()方法来确保该线程在下个测试类运行时是'clean'的。

shiro实战系列(十三)之单元测试的更多相关文章

  1. shiro实战系列(一)之入门实战

    一.什么是shiro? Apache Shiro 是一个强大而灵活的开源安全框架,它干净利落地处理身份认证,授权,企业会话管理和加密.   Apache Shiro 的首要目标是易于使用和理解.安全有 ...

  2. shiro实战系列(二)之入门实战续

    下面讲解基于实战系列一,所以相关的java文件获取pom.xml及其log4j文件同样适用于本次讲解. 一.Using Shiro Using Shiro 现在我们的 SecurityManager ...

  3. shiro实战系列(八)之安全管理器

    Apache Shiro 提供安全框架界独一无二的东西:一个完整的企业级 Session 解决方案,从最简单的命令行及智能手机 应用到最大的集群企业 Web 应用程序.   这对许多应用有着很大的影响 ...

  4. shiro实战系列(四)之配置

    Shiro之配置 Shiro 被设计成能够在任何环境下工作,从最简单的命令行应用程序到最大的的企业群集应用.由于环境的多样性,使得许多配置机制适用于它的配置. 一. 许多配置选项 Shiro的Secu ...

  5. shiro实战系列(三)之架构

    Apache Shiro 的设计目标是通过直观和易于使用来简化应用程序安全.Shiro 的核心设计体现了大多数人们是如何考虑应用程序安全的——在某些人(或某些事)与应用程序交互的背景下.   应用软件 ...

  6. shiro实战系列(十五)之Spring集成Shiro

    Shiro 的 JavaBean 兼容性使得它非常适合通过 Spring XML 或其他基于 Spring 的配置机制.Shiro 应用程序需要一个具 有单例 SecurityManager 实例的应 ...

  7. shiro实战系列(十四)之配置

    Shiro 被设计成能够在任何环境下工作,从最简单的命令行应用程序到最大的的企业群集应用.由于环境的多样性,使得许多配置机制适用于它的配置. 一. 许多配置选项 Shiro的SecurityManag ...

  8. shiro实战系列(十二)之常用专业术语

    请花 2 分钟来阅读和理解它——这很重要.真的.这里的术语和概念在文档的任何地方都被涉及到,它将在总体上 大大简化你对 Shiro 和安全的理解.   由于所使用的术语使得安全可能令人困惑.我们将通过 ...

  9. shiro实战系列(十一)之Caching

    Shiro 开发团队明白在许多应用程序中性能是至关重要的.Caching 是从第一天开始第一个建立在 Shiro 中的一流功 能,以确保安全操作保持尽可能的快.   然而,Caching 作为一个概念 ...

随机推荐

  1. JSP学习笔记(6)-使用数据库

    1.连接MySQL数据库 1.1.JDBC JDBC(Java Database Connectivity)提供了访问数据库的API,由一些Java类和接口组成,是Java运行平台核心库中的一部分.用 ...

  2. 自定义MVC框架之工具类-分页类的封装

    以前写过一个MVC框架,封装的有点low,经过一段时间的沉淀,打算重新改造下,之前这篇文章封装过一个验证码类. 这次重新改造MVC有几个很大的收获 >全部代码都是用Ubuntu+Vim编写,以前 ...

  3. Python 初识网络

    一. C/S架构:客户端(client)/服务端(server)架构 B/S架构:浏览器(browser) / 服务端(server)架构 软件cs架构: 浏览器,qq,微信等等 硬件cs架构:打印机 ...

  4. js-ES6学习笔记-修饰器

    1.修饰器对类的行为的改变,是代码编译时发生的,而不是在运行时.这意味着,修饰器能在编译阶段运行代码. 2. function testable(target) { target.isTestable ...

  5. css选取table元素的第一列

    table tr td:first-child

  6. jQuery获取json数据

    出自---小瓶子编辑 $.each()方法接受两个参数,第一个是需要遍历的对象集合(JSON对象集合),第二个是用来遍历的方法,这个方法又接受两个参数,第一个是遍历的index,第二个是当前遍历的值. ...

  7. JAVA EE期末项目-校园小商店

    校园小商店 一.项目成员及分工 我(计科二班袁文雪)和队友(计科二班蒋媛)设计了一款面对校园的网上购物商店. 我的工作:理解分析代码,编写文档. 二.项目需求分析 网上商店系统主要是实现学生网上选商品 ...

  8. If TransactionScope will close database connections

    问 Do TransactionScope work with closed database connections? using (var transaction = new Transactio ...

  9. jboss-as- 7.1.1.Final配置jndi数据源

    初次使用jboss7.1.1.final部署项目,遇到了很多困难,最终通过查看官方文档和网上资料得以解决,特此记录一下. error information 2016-05-12 12:53:20 J ...

  10. crontab 命令使用

    什么是crontab? crontab命令常见于Unix和类Unix的操作系统之中,用于设置周期性被执行的指令.该命令从标准输入设备读取指令,并将其存放于“crontab”文件中,以供之后读取和执行. ...