我有一个带有 GET 方法的 REST 控制器。它返回一个资源。owner
我想通过将字段Resource
与授权用户的登录名进行比较来验证资源是否属于授权用户。对于正常的同步请求,我会做这样的事情:
@RestController
@RequestMapping("/api")
public class AController {
private final AService aService;
public AController(AService aService) {
this.aService = aService;
}
@GetMapping("/resources/{id}")
@PostAuthorize("returnObject.ownerLogin == authentication.name")
public Resource getResource(@PathVariable Long id) {
return aService.getResource(id);
}
}
但是如果控制器方法是异步的(用 实现DeferredResult
)呢?
@RestController
@RequestMapping("/api")
public class AController {
private final AService aService;
public AController(AService aService) {
this.aService = aService;
}
@GetMapping("/resources/{id}")
@PostAuthorize("returnObject.ownerLogin == authentication.name")
public DeferredResult<Resource> getResource(@PathVariable Long id) {
DeferredResult<Resource> deferredResult = new DeferredResult<>();
aService
.getResourceAsync(id)
.thenAccept(resource -> {
deferredResult.setResult(resource);
});
return deferredResult;
}
}
界面AService
如下所示:
@Service
public class AService {
@Async
public CompletableFuture<Resource> getResourceAsync(Long id) {
// implementation...
}
public Resource getResource(Long id) {
// implementation...
}
}
Resource
类是一个简单的 DTO :
public class Resource {
private String ownerLogin;
// other fields, getters, setters
}
在第二个示例中,Spring Security 很自然地寻找实例ownerLogin
上的字段DeferredResult
。我希望它将异步解析Resource
视为SPEL 表达式returnObject
中的。@PostAuthorize
可能吗?也许有人可以建议一种替代方法?欢迎任何建议。