Browse Source
- Added spring-security-config support - Renamed classes - Changed contracts to include the authenticated user and secured object - Added method security support Issue gh-9288pull/10933/head
19 changed files with 497 additions and 239 deletions
@ -1,61 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-2021 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 |
|
||||||
* |
|
||||||
* https://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.authorization; |
|
||||||
|
|
||||||
import org.springframework.context.ApplicationEventPublisher; |
|
||||||
import org.springframework.context.ApplicationEventPublisherAware; |
|
||||||
import org.springframework.security.authorization.event.AuthorizationFailureEvent; |
|
||||||
import org.springframework.security.authorization.event.AuthorizationSuccessEvent; |
|
||||||
|
|
||||||
/** |
|
||||||
* Default implementation of {@link AuthorizationEventPublisher} |
|
||||||
* |
|
||||||
* @author Parikshit Dutta |
|
||||||
* @since 5.5 |
|
||||||
*/ |
|
||||||
public class DefaultAuthorizationEventPublisher implements AuthorizationEventPublisher, ApplicationEventPublisherAware { |
|
||||||
|
|
||||||
private ApplicationEventPublisher applicationEventPublisher; |
|
||||||
|
|
||||||
public DefaultAuthorizationEventPublisher() { |
|
||||||
this(null); |
|
||||||
} |
|
||||||
|
|
||||||
public DefaultAuthorizationEventPublisher(ApplicationEventPublisher applicationEventPublisher) { |
|
||||||
this.applicationEventPublisher = applicationEventPublisher; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) { |
|
||||||
this.applicationEventPublisher = applicationEventPublisher; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public void publishAuthorizationSuccess(AuthorizationDecision authorizationDecision) { |
|
||||||
if (this.applicationEventPublisher != null) { |
|
||||||
this.applicationEventPublisher.publishEvent(new AuthorizationSuccessEvent(authorizationDecision)); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public void publishAuthorizationFailure(AuthorizationDecision authorizationDecision) { |
|
||||||
if (this.applicationEventPublisher != null) { |
|
||||||
this.applicationEventPublisher.publishEvent(new AuthorizationFailureEvent(authorizationDecision)); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -0,0 +1,65 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-2022 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 |
||||||
|
* |
||||||
|
* https://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.authorization; |
||||||
|
|
||||||
|
import java.util.function.Supplier; |
||||||
|
|
||||||
|
import org.springframework.context.ApplicationEventPublisher; |
||||||
|
import org.springframework.security.authorization.event.AuthorizationDeniedEvent; |
||||||
|
import org.springframework.security.authorization.event.AuthorizationGrantedEvent; |
||||||
|
import org.springframework.security.core.Authentication; |
||||||
|
import org.springframework.util.Assert; |
||||||
|
|
||||||
|
/** |
||||||
|
* An implementation of {@link AuthorizationEventPublisher} that uses Spring's event |
||||||
|
* publishing support. |
||||||
|
* |
||||||
|
* Because {@link AuthorizationGrantedEvent}s typically require additional business logic |
||||||
|
* to decide whether to publish, this implementation only publishes |
||||||
|
* {@link AuthorizationDeniedEvent}s. |
||||||
|
* |
||||||
|
* @author Parikshit Dutta |
||||||
|
* @author Josh Cummings |
||||||
|
* @since 5.7 |
||||||
|
*/ |
||||||
|
public final class SpringAuthorizationEventPublisher implements AuthorizationEventPublisher { |
||||||
|
|
||||||
|
private final ApplicationEventPublisher eventPublisher; |
||||||
|
|
||||||
|
/** |
||||||
|
* Construct this publisher using Spring's {@link ApplicationEventPublisher} |
||||||
|
* @param eventPublisher |
||||||
|
*/ |
||||||
|
public SpringAuthorizationEventPublisher(ApplicationEventPublisher eventPublisher) { |
||||||
|
Assert.notNull(eventPublisher, "eventPublisher cannot be null"); |
||||||
|
this.eventPublisher = eventPublisher; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* {@inheritDoc} |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object, |
||||||
|
AuthorizationDecision decision) { |
||||||
|
if (decision == null || decision.isGranted()) { |
||||||
|
return; |
||||||
|
} |
||||||
|
AuthorizationDeniedEvent<T> failure = new AuthorizationDeniedEvent<>(authentication, object, decision); |
||||||
|
this.eventPublisher.publishEvent(failure); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
@ -1,70 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-2021 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 |
|
||||||
* |
|
||||||
* https://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.authorization; |
|
||||||
|
|
||||||
import org.junit.jupiter.api.BeforeEach; |
|
||||||
import org.junit.jupiter.api.Test; |
|
||||||
|
|
||||||
import org.springframework.context.ApplicationEventPublisher; |
|
||||||
import org.springframework.security.authorization.event.AuthorizationFailureEvent; |
|
||||||
import org.springframework.security.authorization.event.AuthorizationSuccessEvent; |
|
||||||
|
|
||||||
import static org.mockito.ArgumentMatchers.any; |
|
||||||
import static org.mockito.ArgumentMatchers.isA; |
|
||||||
import static org.mockito.Mockito.mock; |
|
||||||
import static org.mockito.Mockito.never; |
|
||||||
import static org.mockito.Mockito.verify; |
|
||||||
|
|
||||||
/** |
|
||||||
* Tests for {@link DefaultAuthorizationEventPublisher} |
|
||||||
* |
|
||||||
* @author Parikshit Dutta |
|
||||||
*/ |
|
||||||
public class DefaultAuthorizationEventPublisherTests { |
|
||||||
|
|
||||||
ApplicationEventPublisher applicationEventPublisher; |
|
||||||
|
|
||||||
DefaultAuthorizationEventPublisher authorizationEventPublisher; |
|
||||||
|
|
||||||
@BeforeEach |
|
||||||
public void init() { |
|
||||||
this.applicationEventPublisher = mock(ApplicationEventPublisher.class); |
|
||||||
this.authorizationEventPublisher = new DefaultAuthorizationEventPublisher(); |
|
||||||
this.authorizationEventPublisher.setApplicationEventPublisher(this.applicationEventPublisher); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void testAuthenticationSuccessIsPublished() { |
|
||||||
this.authorizationEventPublisher.publishAuthorizationSuccess(mock(AuthorizationDecision.class)); |
|
||||||
verify(this.applicationEventPublisher).publishEvent(isA(AuthorizationSuccessEvent.class)); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void testAuthenticationFailureIsPublished() { |
|
||||||
this.authorizationEventPublisher.publishAuthorizationFailure(mock(AuthorizationDecision.class)); |
|
||||||
verify(this.applicationEventPublisher).publishEvent(isA(AuthorizationFailureEvent.class)); |
|
||||||
} |
|
||||||
|
|
||||||
@Test |
|
||||||
public void testNullPublisherNotInvoked() { |
|
||||||
this.authorizationEventPublisher.setApplicationEventPublisher(null); |
|
||||||
this.authorizationEventPublisher.publishAuthorizationSuccess(mock(AuthorizationDecision.class)); |
|
||||||
this.authorizationEventPublisher.publishAuthorizationFailure(mock(AuthorizationDecision.class)); |
|
||||||
verify(this.applicationEventPublisher, never()).publishEvent(any()); |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -0,0 +1,67 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-2022 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 |
||||||
|
* |
||||||
|
* https://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.authorization; |
||||||
|
|
||||||
|
import java.util.function.Supplier; |
||||||
|
|
||||||
|
import org.junit.jupiter.api.BeforeEach; |
||||||
|
import org.junit.jupiter.api.Test; |
||||||
|
|
||||||
|
import org.springframework.context.ApplicationEventPublisher; |
||||||
|
import org.springframework.security.authentication.TestAuthentication; |
||||||
|
import org.springframework.security.authorization.event.AuthorizationDeniedEvent; |
||||||
|
import org.springframework.security.core.Authentication; |
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.isA; |
||||||
|
import static org.mockito.Mockito.mock; |
||||||
|
import static org.mockito.Mockito.verify; |
||||||
|
import static org.mockito.Mockito.verifyNoInteractions; |
||||||
|
|
||||||
|
/** |
||||||
|
* Tests for {@link SpringAuthorizationEventPublisher} |
||||||
|
* |
||||||
|
* @author Parikshit Dutta |
||||||
|
*/ |
||||||
|
public class SpringAuthorizationEventPublisherTests { |
||||||
|
|
||||||
|
Supplier<Authentication> authentication = () -> TestAuthentication.authenticatedUser(); |
||||||
|
|
||||||
|
ApplicationEventPublisher applicationEventPublisher; |
||||||
|
|
||||||
|
SpringAuthorizationEventPublisher authorizationEventPublisher; |
||||||
|
|
||||||
|
@BeforeEach |
||||||
|
public void init() { |
||||||
|
this.applicationEventPublisher = mock(ApplicationEventPublisher.class); |
||||||
|
this.authorizationEventPublisher = new SpringAuthorizationEventPublisher(this.applicationEventPublisher); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testAuthenticationSuccessIsNotPublished() { |
||||||
|
AuthorizationDecision decision = new AuthorizationDecision(true); |
||||||
|
this.authorizationEventPublisher.publishAuthorizationEvent(this.authentication, mock(Object.class), decision); |
||||||
|
verifyNoInteractions(this.applicationEventPublisher); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void testAuthenticationFailureIsPublished() { |
||||||
|
AuthorizationDecision decision = new AuthorizationDecision(false); |
||||||
|
this.authorizationEventPublisher.publishAuthorizationEvent(this.authentication, mock(Object.class), decision); |
||||||
|
verify(this.applicationEventPublisher).publishEvent(isA(AuthorizationDeniedEvent.class)); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
Loading…
Reference in new issue