AclClassIdUtils Default GenericConversionService

So that String, Long, and UUID conversions are automatically
supported.
This commit is contained in:
Nena Raab 2018-11-26 17:31:10 +01:00 committed by Josh Cummings
parent 0e5f1245a6
commit 918a4cd323
No known key found for this signature in database
GPG Key ID: 49EF60DD7FF83443
2 changed files with 79 additions and 43 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2017 the original author or authors. * Copyright 2002-2018 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -18,11 +18,17 @@ package org.springframework.security.acls.jdbc;
import java.io.Serializable; import java.io.Serializable;
import java.sql.ResultSet; import java.sql.ResultSet;
import java.sql.SQLException; import java.sql.SQLException;
import java.util.UUID;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
import org.springframework.core.convert.ConversionFailedException;
import org.springframework.core.convert.ConversionService; import org.springframework.core.convert.ConversionService;
import org.springframework.core.convert.TypeDescriptor;
import org.springframework.core.convert.converter.Converter;
import org.springframework.core.convert.support.GenericConversionService;
import org.springframework.security.acls.model.ObjectIdentity; import org.springframework.security.acls.model.ObjectIdentity;
import org.springframework.util.Assert;
/** /**
* Utility class for helping convert database representations of {@link ObjectIdentity#getIdentifier()} into * Utility class for helping convert database representations of {@link ObjectIdentity#getIdentifier()} into
@ -36,9 +42,14 @@ class AclClassIdUtils {
private ConversionService conversionService; private ConversionService conversionService;
public AclClassIdUtils() { public AclClassIdUtils() {
GenericConversionService genericConversionService = new GenericConversionService();
genericConversionService.addConverter(String.class, Long.class, new StringToLongConverter());
genericConversionService.addConverter(String.class, UUID.class, new StringToUUIDConverter());
this.conversionService = genericConversionService;
} }
public AclClassIdUtils(ConversionService conversionService) { public AclClassIdUtils(ConversionService conversionService) {
Assert.notNull(conversionService, "conversionService must not be null");
this.conversionService = conversionService; this.conversionService = conversionService;
} }
@ -90,17 +101,13 @@ class AclClassIdUtils {
} }
private <T> boolean canConvertFromStringTo(Class<T> targetType) { private <T> boolean canConvertFromStringTo(Class<T> targetType) {
return hasConversionService() && conversionService.canConvert(String.class, targetType); return conversionService.canConvert(String.class, targetType);
} }
private <T extends Serializable> T convertFromStringTo(String identifier, Class<T> targetType) { private <T extends Serializable> T convertFromStringTo(String identifier, Class<T> targetType) {
return conversionService.convert(identifier, targetType); return conversionService.convert(identifier, targetType);
} }
private boolean hasConversionService() {
return conversionService != null;
}
/** /**
* Converts to a {@link Long}, attempting to use the {@link ConversionService} if available. * Converts to a {@link Long}, attempting to use the {@link ConversionService} if available.
* @param identifier The identifier * @param identifier The identifier
@ -111,7 +118,7 @@ class AclClassIdUtils {
*/ */
private Long convertToLong(Serializable identifier) { private Long convertToLong(Serializable identifier) {
Long idAsLong; Long idAsLong;
if (hasConversionService()) { if (canConvertFromStringTo(Long.class)) {
idAsLong = conversionService.convert(identifier, Long.class); idAsLong = conversionService.convert(identifier, Long.class);
} else { } else {
idAsLong = Long.valueOf(identifier.toString()); idAsLong = Long.valueOf(identifier.toString());
@ -124,6 +131,31 @@ class AclClassIdUtils {
} }
public void setConversionService(ConversionService conversionService) { public void setConversionService(ConversionService conversionService) {
Assert.notNull(conversionService, "conversionService must not be null");
this.conversionService = conversionService; this.conversionService = conversionService;
} }
private static class StringToLongConverter implements Converter<String, Long> {
@Override
public Long convert(String identifierAsString) {
if (identifierAsString == null) {
throw new ConversionFailedException(TypeDescriptor.valueOf(String.class),
TypeDescriptor.valueOf(Long.class), null, null);
}
return Long.parseLong(identifierAsString);
}
}
private static class StringToUUIDConverter implements Converter<String, UUID> {
@Override
public UUID convert(String identifierAsString) {
if (identifierAsString == null) {
throw new ConversionFailedException(TypeDescriptor.valueOf(String.class),
TypeDescriptor.valueOf(UUID.class), null, null);
}
return UUID.fromString(identifierAsString);
}
}
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2017 the original author or authors. * Copyright 2002-2018 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -16,21 +16,20 @@
package org.springframework.security.acls.jdbc; package org.springframework.security.acls.jdbc;
import static org.assertj.core.api.Assertions.assertThat; import org.junit.Before;
import static org.mockito.BDDMockito.given; import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.junit.MockitoJUnitRunner;
import org.springframework.core.convert.ConversionService;
import java.io.Serializable; import java.io.Serializable;
import java.sql.ResultSet; import java.sql.ResultSet;
import java.sql.SQLException; import java.sql.SQLException;
import java.util.UUID; import java.util.UUID;
import org.junit.Before; import static org.assertj.core.api.Assertions.assertThat;
import org.junit.Test; import static org.mockito.BDDMockito.given;
import org.junit.runner.RunWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.runners.MockitoJUnitRunner;
import org.springframework.core.convert.ConversionService;
/** /**
* Tests for {@link AclClassIdUtils}. * Tests for {@link AclClassIdUtils}.
@ -46,40 +45,23 @@ public class AclClassIdUtilsTest {
private ResultSet resultSet; private ResultSet resultSet;
@Mock @Mock
private ConversionService conversionService; private ConversionService conversionService;
@InjectMocks
private AclClassIdUtils aclClassIdUtils; private AclClassIdUtils aclClassIdUtils;
@Before @Before
public void setUp() throws Exception { public void setUp() {
given(conversionService.canConvert(String.class, Long.class)).willReturn(true); aclClassIdUtils = new AclClassIdUtils();
given(conversionService.convert(DEFAULT_IDENTIFIER, Long.class)).willReturn(new Long(DEFAULT_IDENTIFIER));
given(conversionService.convert(DEFAULT_IDENTIFIER_AS_STRING, Long.class)).willReturn(new Long(DEFAULT_IDENTIFIER));
} }
@Test @Test
public void shouldReturnLongIfIdentifierIsNotStringAndNoConversionService() throws SQLException { public void shouldReturnLongIfIdentifierIsLong() throws SQLException {
// given
AclClassIdUtils aclClassIdUtilsWithoutConversionSvc = new AclClassIdUtils();
// when // when
Serializable newIdentifier = aclClassIdUtilsWithoutConversionSvc.identifierFrom(DEFAULT_IDENTIFIER, resultSet); Serializable newIdentifier = aclClassIdUtils.identifierFrom(DEFAULT_IDENTIFIER, resultSet);
// then // then
assertThat(newIdentifier).isEqualTo(DEFAULT_IDENTIFIER); assertThat(newIdentifier).isEqualTo(DEFAULT_IDENTIFIER);
} }
@Test
public void shouldReturnLongIfIdentifierIsNotString() throws SQLException {
// given
Long prevIdentifier = 999L;
// when
Serializable newIdentifier = aclClassIdUtils.identifierFrom(prevIdentifier, resultSet);
// then
assertThat(newIdentifier).isEqualTo(prevIdentifier);
}
@Test @Test
public void shouldReturnLongIfClassIdTypeIsNull() throws SQLException { public void shouldReturnLongIfClassIdTypeIsNull() throws SQLException {
// given // given
@ -117,10 +99,11 @@ public class AclClassIdUtilsTest {
} }
@Test @Test
public void shouldReturnLongIfTypeClassCannotBeConverted() throws SQLException { public void shouldReturnLongEvenIfCustomConversionServiceDoesNotSupportLongConversion() throws SQLException {
// given // given
given(resultSet.getString("class_id_type")).willReturn("java.lang.Long"); given(resultSet.getString("class_id_type")).willReturn("java.lang.Long");
given(conversionService.canConvert(String.class, Long.class)).willReturn(false); given(conversionService.canConvert(String.class, Long.class)).willReturn(false);
aclClassIdUtils.setConversionService(conversionService);
// when // when
Serializable newIdentifier = aclClassIdUtils.identifierFrom(DEFAULT_IDENTIFIER_AS_STRING, resultSet); Serializable newIdentifier = aclClassIdUtils.identifierFrom(DEFAULT_IDENTIFIER_AS_STRING, resultSet);
@ -145,10 +128,7 @@ public class AclClassIdUtilsTest {
public void shouldReturnUUIDWhenUUIDClassIdType() throws SQLException { public void shouldReturnUUIDWhenUUIDClassIdType() throws SQLException {
// given // given
UUID identifier = UUID.randomUUID(); UUID identifier = UUID.randomUUID();
String identifierAsString = identifier.toString();
given(resultSet.getString("class_id_type")).willReturn("java.util.UUID"); given(resultSet.getString("class_id_type")).willReturn("java.util.UUID");
given(conversionService.canConvert(String.class, UUID.class)).willReturn(true);
given(conversionService.convert(identifierAsString, UUID.class)).willReturn(UUID.fromString(identifierAsString));
// when // when
Serializable newIdentifier = aclClassIdUtils.identifierFrom(identifier.toString(), resultSet); Serializable newIdentifier = aclClassIdUtils.identifierFrom(identifier.toString(), resultSet);
@ -157,4 +137,28 @@ public class AclClassIdUtilsTest {
assertThat(newIdentifier).isEqualTo(identifier); assertThat(newIdentifier).isEqualTo(identifier);
} }
@Test
public void shouldReturnStringWhenStringClassIdType() throws SQLException {
// given
String identifier = "MY_STRING_IDENTIFIER";
given(resultSet.getString("class_id_type")).willReturn("java.lang.String");
// when
Serializable newIdentifier = aclClassIdUtils.identifierFrom(identifier, resultSet);
// then
assertThat(newIdentifier).isEqualTo(identifier);
}
@Test(expected = IllegalArgumentException.class)
public void shouldNotAcceptNullConversionServiceInConstruction() throws SQLException {
// when
new AclClassIdUtils(null);
}
@Test(expected = IllegalArgumentException.class)
public void shouldNotAcceptNullConversionServiceInSetter() throws SQLException {
// when
aclClassIdUtils.setConversionService(null);
}
} }