亲手打造:构建第一个Apache Shiro应用

构建第一个Apache Shiro应用

如果您是Apache Shiro的新手,这个简短的教程将向您展示如何设置基于Apache Shiro的初始的且非常简单的安全应用。我们将一路讨论Shiro的核心概念,以帮助您熟悉Shiro的设计和API。

如果您不想按照本教程进行实际的文件编辑,则可从如下位置获取几乎相同的示例应用并参考它:

1)Apache Shiro的Git仓库中:

 https://github.com/apache/shiro/tree/master/samples/quickstart

2)源码发行包:在Apache Shiro的源代码发行包的samples/quickstart目录中。源代码发行包可从"这里"(http://shiro.apache.org/download.html)页面获得。

1、设置

在这个简单的例子中,我们将创建一个非常简单的命令行应用程序,它将运行并快速退出,这样只是让您体会Shiro的API。

提示:

对任何应用——Apache Shiro的设计从一开始就支持任何应用程序,无论从最小的命令行应用程序还是到最大的集群Web应用程序。 尽管我们为本教程创建了一个简单的应用程序,但请注意,无论您的应用程序的创建方式或部署方式如何,都会应用相同的使用模式。

本教程需要Java 1.5或更高版本。我们也将使用Apache Maven作为我们的构建工具,但当然这不是使用Apache Shiro所必需的。您可能会获取Shiro的所有.jars包,并将它们以您喜欢的任何方式整合到您的应用程序中,例如使用Apache Ant和Ivy。

对于本教程,请确保您使用的是Maven 2.2.1或更高版本。您应该能够在命令提示符下键入mvn --version并查看与以下类似的内容。

1.1检查Maven安装情况

在不同的平台上,打开命令行终端,输入如下命令:

mvn –-version 或者mvn –v回车,可以看到类似如下信息:


现在,在文件系统上创建一个新目录,例如shiro-tutorial,并将以下Maven pom.xml文件保存在该目录中。

1.2构建pom.xml

我这里在window平台创建目录如下:E:\myTest\shiro-tutorial。

在shiro-tutorial目录下创建pom.xml文件,此maven的xml文件内容如下:

pom.xml

1.3 构建类

我们将运行一个简单的命令行应用程序,因此我们需要创建一个Java类,其拥有公开静态方法void main(String  args)方法。

在包含pom.xml文件的同一目录中(上文创建的目录),创建一个* src/main/java子目录。在src/main/java中创建一个包含以下内容的Tutorial.java文件:

现在不要担心import语句问题,我们很快就会理解它们。但现在,我们已经有了一个典型的命令行'shell'程序。这些程序要做的就是打印出"My First Apache Shiro Application"并退出。

4- 测试运行

为测试我们的教程应用程序,请在命令行模式下,进入到教程项目的根目录(例如shiro-tutorial)下,在命令提示符中执行以下命令:mvn compile exec:java

你会看到我们的小教程"应用程序"运行并退出。您应该看到类似以下内容(注意粗体文本,表示我们的输出):


注意,在执行上述命令时,mvn会执行一下必要的初始化工作,如下载相关组建包等,下次再运行时,就不会再下载了J

我们已验证应用程序成功运行- 现在让我们启用Apache Shiro。继续本教程时,您可以在每次添加更多代码后运行mvn compile exec:java,以查看更改后运行结果

2、启用Shio

在应用程序中启用Shiro首先要了解的是,Shiro中的几乎所有内容都与称为SecurityManager的中央/核心组件有关。对于那些熟悉Java安全性的人来说,这是Shiro的SecurityManager概念 - 它与java.lang.SecurityManager不同。

虽然我们在体系结构章节中详细介绍Shiro的设计,但现在知道Shiro SecurityManager是应用程序的Shiro环境核心,并且每个应用程序必须存在一个SecurityManager,这就足够好了。因此,我们必须在我们的教程应用程序中做的第一件事是设置SecurityManager实例。

2.1配置

虽然我们可以直接实例化SecurityManager类,但Shiro的SecurityManager实现具有足够的配置选项和内部组件,这使得在Java源代码中很难做到这一点,因此使用灵活的基于文本格式文件来配置SecurityManager会容易得多。

为此,Shiro通过基于文本的INI配置提供默认的"公分母"解决方案。现在人们对使用庞大的XML文件感到厌倦,INI易于阅读、使用简单,并且只需很少的依赖关系。稍后您将会看到,通过对对象图导航的简单了解,INI可以有效地用于配置像SecurityManager这样的简单对象图。

注意:SecurityManager有很多配置项。

Shiro的SecurityManager实现和所有支持组件都与JavaBean兼容。 这使得Shiro可以使用几乎任意配置格式进行配置,如XML(Spring,JBoss,Guice等),YAML,JSON,Groovy Builder标记等等。 INI只是Shiro的'公分母'格式,允许在任何环境下进行配置,以防其他选项无法使用。

2.2shiro.ini

我们使用INI文件来为这个简单的应用程序配置Shiro安全管理器SecurityManager。首先,从pom.xml所在的同一目录下创建一个src/main/resources目录。然后在该新目录中使用以下内容创建一个shiro.ini文件(配置清单src/main/resources/shiro.ini):

# =============================================================================

# Tutorial INI configuration

#

# Usernames/passwords are based on the classic Mel Brooks' film "Spaceballs" :)

# =============================================================================

# -----------------------------------------------------------------------------

# Users and their (optional) assigned roles

# username = password, role1, role2, ..., roleN

# -----------------------------------------------------------------------------

[users]

root = secret, admin

guest = guest, guest

presidentskroob = 12345, president

darkhelmet = ludicrousspeed, darklord, schwartz

lonestarr = vespa, goodguy, schwartz

# -----------------------------------------------------------------------------

# Roles with assigned permissions

# roleName = perm1, perm2, ..., permN

# -----------------------------------------------------------------------------

[roles]

admin = *

schwartz = lightsaber:*

goodguy = winnebago:drive:eagle5


正如你所看到的,这个配置基本上建立了一小组静态用户帐户,对于我们第一个应用程序已经足够好了。在后面的章节中,您将看到我们如何使用更复杂的用户数据源,如关系数据库,LDAP和ActiveDirectory等等。

2.3引用配置

现在我们已经定义了一个INI文件,可以在我们的教程应用程序类中创建SecurityManager实例。更改main方法如下:


我们在初始代码中添加了3行代码后,Shiro在示例应用程序中启用!这是多容易是吧?

随意运行mvn compile exec:java,并看到所有东西仍然能够成功运行(由于Shiro默认的调试日志记录或更低日志等级,所以你不会看到任何Shiro日志消息 - 如果它启动并且没有错误地运行,那么你就了解到一切都是正常的)。

这是上述补充内容的描述:

1.我们使用Shiro的IniSecurityManagerFactory实现来获取位于classpath根目录下的shiro.ini文件。这种实现反映了Shiro对工厂方法设计模式的支持。  classpath:前缀是一个资源指示符,告诉shiro从哪里加载ini文件(其他前缀,如url:和file:也受支持)。

2.调用factory.getInstance()方法,该方法解析INI文件并返回反映配置的SecurityManager实例。

3.在这个简单的例子中,我们将SecurityManager设置为一个可以通过JVM访问的静态(内存)单例。但是请注意,如果您将在单个JVM中拥有多个支持Shiro的应用程序,那么这是不可取的。对于这个简单的例子,这是可以的,但更复杂的应用程序环境通常会将SecurityManager放置在特定于应用程序的内存中(例如在Web应用程序的ServletContext或Spring,Guice或JBoss DI容器实例中)

3、使用Shiro

我们的SecurityManager已经准备就绪,现在可以开始做我们真正关心的事情 - 执行安全操作。

在确保我们的应用程序安全时,我们自问最相关的问题可能是"谁是当前用户?"或"当前用户是否允许执行X?"?在我们编写代码或设计用户接口时,常常会提出这些问题:应用程序通常基于用户故事而构建,并且您希望基于每个用户来展现(和保护)功能。因此,我们在应用程序中考虑安全性的最自然方式是基于当前用户。  Shiro的API从根本上代表了"目前用户"主张与其主体概念。

几乎在所有环境中,您都可以通过以下调用获取当前正在执行的用户:

Subject currentUser = SecurityUtils.getSubject();

使用SecurityUtils.getSubject(),我们可以获得当前正在执行的Subject。主体是一个安全术语,基本上意味着"当前正在执行的用户的特定安全视图"。它不被称为"用户",因为"用户"一词通常与人类相关联。在安全领域,"主体"这个术语可能意味着一个人,也可能是一个第三方进程、cron作业、守护进程帐户或任何类似的东西。它只是意味着"当前与软件交互的东西"。不过,对于大多数意图和目的,您可以将主体视为Shiro的"用户"概念。

独立应用程序中的getSubject()调用,其可能会根据特定于应用程序的位置中的用户数据以及服务器环境(例如Web应用程序)返回Subject,并根据与当前线程或传入请求关联的用户数据获取Subject  。

现在你有一个主体,你可以用它做什么?

如果您想在应用程序的当前会话期间向用户提供可用的内容,则可以获得他们的会话:

Session session = currentUser.getSession();

session.setAttribute( "someKey", "aValue" );

此Session是一个Shiro特定的实例,它提供了大多数你习惯的常规HttpSession的实例,但有一些额外的好处和一个很大的区别:它不需要HTTP环境!

如果在Web应用程序内部署,默认情况下会话将基于HttpSession。但是,在非Web环境中,像这个简单的教程应用程序,Shiro默认会自动使用其企业会话管理。这意味着无论部署环境如何,您都可以在任何层中的应用程序中使用相同的API!这将打开一个全新的应用程序世界,因为任何需要会话的应用程序都不需要强制使用HttpSession或EJB Stateful Session Beans。而且,任何客户端技术现在都可以共享会话数据。

所以现在你可以获得一个Subject主体和Session会话。但那些真正有用的,如检查是否允许他们做事、检查角色和权限等是关于什么的?

其实,我们只能对已知的用户进行这些检查。上面的Subject实例代表当前用户,但是谁是当前用户?那么,他们是匿名的——也就是说,直到他们至少登录一次。所以,让我们来这样做:

就这么简单!

但是,如果他们的登录尝试失败呢?你可以捕捉各种具体的例外情况,告诉你到底发生了什么,并允许你相应地处理和做出响应:

您可以检查许多不同类型的例外情况,或者根据Shiro可能无法解释的自定义条件抛出自己的例外情况。有关更多信息,请参阅AuthenticationException

JavaDoc(http://shiro.apache.org/static/current/apidocs/org/apache/shiro/authc/AuthenticationException.html)。

【顺便一提:安全最佳做法是为用户提供通用登录失败消息,因为您不希望帮助攻击者试图进入系统。

那么,到现在为止,我们有一个登录用户。我们还能做什么?

让我们看看他们是谁:

//print their identifying principal (in this case, a username):

log.info( "User [" + currentUser.getPrincipal() + "] logged in successfully." );

也可以测试它们是否具有特定的角色:

if (

currentUser.hasRole( "schwartz") ) {

log.info("May the

Schwartz be with you!");

} else{

log.info( "Hello, mere

mortal.");

}

还可以看到他们是否有权对某种类型的实体采取行动:

if (

currentUser.isPermitted( "lightsaber:weild") ) {

log.info("You may use a

lightsaber ring. Use it wisely.");

} else{

log.info("Sorry,

lightsaber rings are for schwartz masters only.");

}

另外,我们可以执行非常强大的实例级权限检查 - 查看用户是否有权访问特定类型实例的功能:

if (

currentUser.isPermitted( "winnebago:drive:eagle5") ) {

log.info("You are

permitted to 'drive' the 'winnebago' with license plate (id) 'eagle5'. " + "Here are the

keys - have fun!");

 }else{

log.info("Sorry, you

aren't allowed to drive the 'eagle5' winnebago!");

}

是不是感觉小菜一碟,是吧?确实。

最后,当用户完成使用应用程序时,他们可以注销:

currentUser.logout(); //removesall identifying information and invalidates their session too.

4、最终Tutorial类

在添加上面的代码示例之后,这里是我们最终的Tutorial类文件。 可随意编辑并使用它,并根据需要更改安全检查(和INI配置)。完整教程示例代码如下:




5、示例小结

希望这篇入门教程能帮助您了解如何在基本应用程序中设置Shiro以及Shiro的主要设计概念Subject和SecurityManager。

但这是一个相当简单的应用程序。您可能会问自己,"如果我不想使用INI用户帐户,而想连接到更复杂的用户数据源,该怎么办?"

要回答这个问题,需要对Shiro的体系结构和配置支持机制有更深入的了解。后续我们将介绍Shiro的架构。

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

推荐阅读更多精彩内容