- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有一个使用 Resteasy 并在 Wildfly 上运行的 Web 服务。
身份验证的工作原理是让用户在每个请求中传递授权 header 。理想情况下,我需要检查用户是否拥有他们尝试访问的资源,但我想知道最好或最简洁的方法是什么。
例如,一个用户拥有多个“礼物 list ”。我有一个端点 www.example.com/api/giftlist/7,它应该检索 ID 为 7 的礼品列表,但前提是授权 header 来自该列表的所有者。在代码中,它看起来像这样:
/**
* Retrieve a list by it's ID and return in JSON format.
* @param id the ID of the list to return
* @return the list.
*/
@GET
@Path("/{id}")
@ApiOperation(value = "Get a gift list by ID.", response = GiftList.class)
@Produces(MediaType.APPLICATION_JSON)
public GiftList getGiftListById(@HeaderParam("Authorization") String authorization, @PathParam("id") Long id){
User user = null;
AccessToken token = dao.find(AccessToken.class, authorization);
if(token !=null){
user = token.getUser();
}
GiftList giftList = dao.find(GiftList.class, id);
if(giftList == null){
throw new WebApplicationException(Response.Status.NOT_FOUND);
}
if(!giftList.isOwnedBy(user)) {
throw new WebApplicationException(Response.Status.FORBIDDEN);
}
return giftList;
}
请注意,几乎所有此方法都与识别用户是否存在以及他们是否拥有他们尝试访问的资源有关。然后需要为 PUT、POST 和 DELETE 重复此逻辑,这一切都变得相当困惑。
我尝试使用拦截器,这样做我能够检查用户的角色,但我无法访问他们尝试访问的资源的类型或 ID。请注意,GiftLists 不是此应用程序中的唯一资源。我正在寻找一种更简洁的方法来避免对大多数资源的每个操作都执行此操作。也许使用拦截器,但我不确定如何访问 @PathParam 值并获取正确的类型,然后从数据库中检索它并检查所有权。
这一定是一个常见问题,所以我确定有一些常用的约定或模式?我对身份验证和使用拦截器进行了无休止的谷歌搜索,但似乎没有一个能真正帮助解决这种“所有权”问题。
减少一些但不是所有困惑的一种方法是使用拦截器来接收请求并在拦截器中找到用户,但我不确定如何传递该用户对象从那里进入方法本身,以节省再次查找它。
另一个想法是我可以将用户或授权 header 传递给 DAO 并让 DAO 在用户未被授权访问特定资源时抛出异常,但安全逻辑是否存在于 DAO 区域中 -这似乎也不对(对我来说)——例如 dao.findGiftList(authorization, ID)
——但是处理这个授权是 DAO 的责任吗?
最佳答案
要在您的 RESTful 网络服务中启用身份验证和授权,您可以使用 JAAS
要在 Wildfly 上执行此操作,您可以:
在您的 Wildfly 配置中配置一个安全域,例如下面使用数据库登录模块的配置(如果您将其作为独立服务器运行,则在 standalone.xml 配置中)
<security-domain name="test" cache-type="default">
<authentication>
<login-module code="Database" flag="required">
<module-option name="dsJndiName" value="java:/TestDS"/>
<module-option name="principalsQuery" value="select password from User where login = ? and (disabled is null or disabled = 0) and activated = 1"/>
<module-option name="rolesQuery" value="select name,'Roles' from Role r, User_Role ur, User u where u.login=? and u.id = ur.userId and r.id = ur.roleId"/>
<module-option name="hashAlgorithm" value="SHA-256"/>
<module-option name="hashEncoding" value="base64"/>
<module-option name="unauthenticatedIdentity" value="guest"/>
</login-module>
</authentication>
</security-domain>
在 jboss-ejb3(如果您使用 RESTful EJB web 服务)或 webapp/WEB-INF 目录中的 jboss-web.xml 文件中引用它。
jboss-web.xml
<?xml version="1.0" encoding="UTF-8"?>
<!-- Configure usage of the security domain "other" -->
<jboss-web>
<security-domain>test</security-domain>
</jboss-web>
jboss-ejb3.xml
<jboss:ejb-jar xmlns:jboss="http://www.jboss.com/xml/ns/javaee"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:s="urn:security"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.jboss.com/xml/ns/javaee http://www.jboss.org/j2ee/schema/jboss-ejb3-2_0.xsd
http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/ejb-jar_3_1.xsd"
version="3.1"
impl-version="2.0">
<assembly-descriptor>
<s:security>
<ejb-name>*</ejb-name>
<s:security-domain>test</s:security-domain>
</s:security>
</assembly-descriptor>
</jboss:ejb-jar>
像下面的示例一样配置安全约束和登录配置,在 webapp 中启用 BASIC 身份验证:(有关详细信息,请参阅 this link)
<security-constraint>
<web-resource-collection>
<web-resource-name>REST services</web-resource-name>
<url-pattern>/rs/user/*</url-pattern>
</web-resource-collection>
<auth-constraint>
<role-name>*</role-name>
</auth-constraint>
</security-constraint>
<login-config>
<auth-method>BASIC</auth-method>
<realm-name>TestRealm</realm-name>
</login-config>
...
<security-role>
<role-name>*</role-name>
</security-role>
然后使用@PermitAll 或@RolesAllowed 注释您的 REST 方法,以允许公共(public)访问或仅允许某些角色的授权访问,如本例所示:
@Path("/giftlists")
public class GiftLists{
@Resource
private SessionContext sessionContext;
@GET
@Consumes(MediaType.APPLICATION_JSON)
@RolesAllowed(USER) // Allow only authenticated users to access this
@Path("/{giftListId}")
public void getGiftListById(@NotNull @PathParam("giftListId") Long giftListId) {
User user = userDAO.findUserByLogin(sessionContext.getCallerPrincipal().getName());
GiftList giftList = giftListDAO.findGiftListByIdAndUser(giftListId, user); // user is provided to your DAO method / query, so giftList is returned only when User owns it. (No FORBIDDEN error)
if(giftList == null){
throw new WebApplicationException(Response.Status.NOT_FOUND);
}
}
}
所以对于这个例子,你有两个改进:
关于java - Resteasy 授权设计——检查一个用户是否拥有一个资源,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34315838/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!