/* * Copyright 2002-2016 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.springframework.security.data.repository.query; import org.springframework.data.repository.query.spi.EvaluationContextExtensionSupport; import org.springframework.security.access.expression.SecurityExpressionRoot; import org.springframework.security.core.Authentication; import org.springframework.security.core.context.SecurityContext; import org.springframework.security.core.context.SecurityContextHolder; /** * <p> * By defining this object as a Bean, Spring Security is exposed as SpEL expressions for * creating Spring Data queries. * * <p> * With Java based configuration, we can define the bean using the following: * * <p> * For example, if you return a UserDetails that extends the following User object: * * <pre> * @Entity * public class User { * @GeneratedValue(strategy = GenerationType.AUTO) * @Id * private Long id; * * ... * } * </pre> * * <p> * And you have a Message object that looks like the following: * * <pre> * @Entity * public class Message { * @Id * @GeneratedValue(strategy = GenerationType.AUTO) * private Long id; * * @OneToOne * private User to; * * ... * } * </pre> * * You can use the following {@code Query} annotation to search for only messages that are * to the current user: * * <pre> * @Repository * public interface SecurityMessageRepository extends MessageRepository { * * @Query("select m from Message m where m.to.id = ?#{ principal?.id }") * List<Message> findAll(); * } * </pre> * * This works because the principal in this instance is a User which has an id field on * it. * * @since 4.0 * @author Rob Winch */ public class SecurityEvaluationContextExtension extends EvaluationContextExtensionSupport { private Authentication authentication; /** * Creates a new instance that uses the current {@link Authentication} found on the * {@link org.springframework.security.core.context.SecurityContextHolder}. */ public SecurityEvaluationContextExtension() { } /** * Creates a new instance that always uses the same {@link Authentication} object. * * @param authentication the {@link Authentication} to use */ public SecurityEvaluationContextExtension(Authentication authentication) { this.authentication = authentication; } public String getExtensionId() { return "security"; } @Override public Object getRootObject() { Authentication authentication = getAuthentication(); return new SecurityExpressionRoot(authentication) { }; } private Authentication getAuthentication() { if (this.authentication != null) { return this.authentication; } SecurityContext context = SecurityContextHolder.getContext(); return context.getAuthentication(); } }