diff --git a/src/main/java/net/contargo/iris/address/api/AddressApiController.java b/src/main/java/net/contargo/iris/address/api/AddressApiController.java index 40424bec..00de0521 100644 --- a/src/main/java/net/contargo/iris/address/api/AddressApiController.java +++ b/src/main/java/net/contargo/iris/address/api/AddressApiController.java @@ -34,9 +34,12 @@ import java.util.ArrayList; import java.util.List; import java.util.Map; +import java.util.Optional; import javax.servlet.http.HttpServletRequest; +import static net.contargo.iris.address.w3w.ThreeWordMatcher.isThreeWordAddress; + import static org.slf4j.LoggerFactory.getLogger; import static org.springframework.hateoas.mvc.ControllerLinkBuilder.linkTo; @@ -144,6 +147,15 @@ public ListOfAddressListsResponse addressesByAddressDetails(@RequestParam(requir } } + if (isThreeWordAddress(street)) { + Optional addressDto = addressDtoService.getAddressesByThreeWords(street.trim()); + + if (addressDto.isPresent()) { + addressListDtos = singletonList(new AddressListDto("Result for three words " + street, + singletonList(addressDto.get()))); + } + } + Map addressDetails = NominatimUtil.parameterMap(street, postalCode, city, country, name); if (addressListDtos.isEmpty()) { diff --git a/src/main/java/net/contargo/iris/address/dto/AddressDtoService.java b/src/main/java/net/contargo/iris/address/dto/AddressDtoService.java index aaafa47e..5e905523 100644 --- a/src/main/java/net/contargo/iris/address/dto/AddressDtoService.java +++ b/src/main/java/net/contargo/iris/address/dto/AddressDtoService.java @@ -5,6 +5,7 @@ import java.util.List; import java.util.Map; +import java.util.Optional; /** @@ -116,4 +117,14 @@ public interface AddressDtoService { * @return a list of matching addresses */ List getAddressesByQuery(String query); + + + /** + * Returns an optional {@link AddressDto} matching the given three word address. + * + * @param threeWords a three word address + * + * @return an optional address + */ + Optional getAddressesByThreeWords(String threeWords); } diff --git a/src/main/java/net/contargo/iris/address/dto/AddressDtoServiceImpl.java b/src/main/java/net/contargo/iris/address/dto/AddressDtoServiceImpl.java index 63eb2309..11421947 100644 --- a/src/main/java/net/contargo/iris/address/dto/AddressDtoServiceImpl.java +++ b/src/main/java/net/contargo/iris/address/dto/AddressDtoServiceImpl.java @@ -9,6 +9,7 @@ import java.util.ArrayList; import java.util.List; import java.util.Map; +import java.util.Optional; import static java.util.Collections.singletonList; import static java.util.stream.Collectors.toList; @@ -110,4 +111,13 @@ public List getAddressesByQuery(String query) { return addressServiceWrapper.getAddressesByQuery(query).stream().map(AddressDto::new).collect(toList()); } + + + @Override + public Optional getAddressesByThreeWords(String threeWords) { + + Optional
optionalAddress = addressServiceWrapper.getAddressByThreeWords(threeWords); + + return optionalAddress.map(AddressDto::new); + } } diff --git a/src/main/java/net/contargo/iris/address/service/AddressServiceWrapper.java b/src/main/java/net/contargo/iris/address/service/AddressServiceWrapper.java index f3e8560b..c22b4420 100644 --- a/src/main/java/net/contargo/iris/address/service/AddressServiceWrapper.java +++ b/src/main/java/net/contargo/iris/address/service/AddressServiceWrapper.java @@ -7,6 +7,8 @@ import net.contargo.iris.address.staticsearch.StaticAddress; import net.contargo.iris.address.staticsearch.service.StaticAddressNotFoundException; import net.contargo.iris.address.staticsearch.service.StaticAddressService; +import net.contargo.iris.address.w3w.ThreeWordClient; +import net.contargo.iris.address.w3w.ThreeWordClientException; import net.contargo.iris.normalizer.NormalizerService; import org.apache.commons.lang.StringUtils; @@ -19,6 +21,7 @@ import java.util.Collections; import java.util.List; import java.util.Map; +import java.util.Optional; import java.util.regex.Pattern; import static net.contargo.iris.address.nominatim.service.AddressDetailKey.CITY; @@ -46,14 +49,16 @@ public class AddressServiceWrapper { private final StaticAddressService staticAddressService; private final AddressCache addressCache; private final NormalizerService normalizerService; + private final ThreeWordClient threeWordClient; public AddressServiceWrapper(AddressService addressService, StaticAddressService staticAddressService, - AddressCache cache, NormalizerService normalizerService) { + AddressCache cache, NormalizerService normalizerService, ThreeWordClient threeWordClient) { this.addressService = addressService; this.staticAddressService = staticAddressService; this.addressCache = cache; this.normalizerService = normalizerService; + this.threeWordClient = threeWordClient; } /** @@ -225,4 +230,19 @@ public List
getAddressesByQuery(String query) { return addresses; } + + + @SuppressWarnings("squid:S1166") + public Optional
getAddressByThreeWords(String threeWords) { + + try { + GeoLocation resolvedLocation = threeWordClient.resolve(threeWords); + + return Optional.of(getAddressForGeoLocation(resolvedLocation)); + } catch (ThreeWordClientException e) { + LOG.info("Cannot resolve three word address {}: {}", threeWords, e.getMessage()); + + return Optional.empty(); + } + } } diff --git a/src/main/java/net/contargo/iris/address/w3w/ForwardW3wResponse.java b/src/main/java/net/contargo/iris/address/w3w/ForwardW3wResponse.java new file mode 100644 index 00000000..0610fffd --- /dev/null +++ b/src/main/java/net/contargo/iris/address/w3w/ForwardW3wResponse.java @@ -0,0 +1,77 @@ +package net.contargo.iris.address.w3w; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonProperty; + +import net.contargo.iris.GeoLocation; + +import java.math.BigDecimal; + + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +class ForwardW3wResponse { + + private final W3wResponseGeometry geometry; + private final W3wResponseStatus status; + + @JsonCreator + ForwardW3wResponse(@JsonProperty("geometry") W3wResponseGeometry geometry, + @JsonProperty("status") W3wResponseStatus status) { + + this.geometry = geometry; + this.status = status; + } + + GeoLocation toGeolocation() { + + return new GeoLocation(geometry.lat, geometry.lon); + } + + + boolean error() { + + return status.code != null; + } + + + Integer errorCode() { + + return status.code; + } + + + String errorMessage() { + + return status.message; + } + + private static class W3wResponseGeometry { + + private final BigDecimal lat; + private final BigDecimal lon; + + @JsonCreator + private W3wResponseGeometry(@JsonProperty("lat") BigDecimal lat, + @JsonProperty("lng") BigDecimal lon) { + + this.lat = lat; + this.lon = lon; + } + } + + private static class W3wResponseStatus { + + private final Integer code; + private final String message; + + @JsonCreator + private W3wResponseStatus(@JsonProperty("code") Integer code, + @JsonProperty("message") String message) { + + this.code = code; + this.message = message; + } + } +} diff --git a/src/main/java/net/contargo/iris/address/w3w/ThreeWordClient.java b/src/main/java/net/contargo/iris/address/w3w/ThreeWordClient.java new file mode 100644 index 00000000..112780c4 --- /dev/null +++ b/src/main/java/net/contargo/iris/address/w3w/ThreeWordClient.java @@ -0,0 +1,47 @@ +package net.contargo.iris.address.w3w; + +import net.contargo.iris.GeoLocation; + +import org.springframework.http.ResponseEntity; + +import org.springframework.web.client.RestTemplate; + + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +public class ThreeWordClient { + + private static final String FORWARD_URL = + "https://api.what3words.com/v2/forward?addr={w3wAddress}&key={apiKey}&lang=de"; + + private final RestTemplate restTemplate; + private final String apiKey; + + public ThreeWordClient(RestTemplate restTemplate, String apiKey) { + + this.restTemplate = restTemplate; + this.apiKey = apiKey; + } + + public GeoLocation resolve(String threeWords) { + + ResponseEntity response = restTemplate.getForEntity(FORWARD_URL, ForwardW3wResponse.class, + threeWords, apiKey); + + checkErrorStatus(response.getBody(), threeWords); + + return response.getBody().toGeolocation(); + } + + + private static void checkErrorStatus(ForwardW3wResponse response, String threeWords) { + + if (response.error()) { + Integer code = response.errorCode(); + String message = response.errorMessage(); + + throw new ThreeWordClientException(code, message, threeWords); + } + } +} diff --git a/src/main/java/net/contargo/iris/address/w3w/ThreeWordClientException.java b/src/main/java/net/contargo/iris/address/w3w/ThreeWordClientException.java new file mode 100644 index 00000000..f76d67c5 --- /dev/null +++ b/src/main/java/net/contargo/iris/address/w3w/ThreeWordClientException.java @@ -0,0 +1,13 @@ +package net.contargo.iris.address.w3w; + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +public class ThreeWordClientException extends RuntimeException { + + ThreeWordClientException(Integer code, String message, String threeWords) { + + super("API of w3w returned error code " + code + " with message '" + message + + "' for three word address '" + threeWords + "'"); + } +} diff --git a/src/main/java/net/contargo/iris/address/w3w/ThreeWordMatcher.java b/src/main/java/net/contargo/iris/address/w3w/ThreeWordMatcher.java new file mode 100644 index 00000000..3b9d7891 --- /dev/null +++ b/src/main/java/net/contargo/iris/address/w3w/ThreeWordMatcher.java @@ -0,0 +1,17 @@ +package net.contargo.iris.address.w3w; + +import java.util.regex.Pattern; + + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +public class ThreeWordMatcher { + + private static final Pattern THREE_WORD_PATTERN = Pattern.compile("^\\p{L}+\\.\\p{L}+\\.\\p{L}+$"); + + public static boolean isThreeWordAddress(String input) { + + return input != null && THREE_WORD_PATTERN.matcher(input.trim()).matches(); + } +} diff --git a/src/main/resources/application-context.xml b/src/main/resources/application-context.xml index 24d9809d..483d5c23 100644 --- a/src/main/resources/application-context.xml +++ b/src/main/resources/application-context.xml @@ -121,8 +121,16 @@ + + + + + + + + diff --git a/src/main/resources/application.properties b/src/main/resources/application.properties index e794a683..5e613363 100644 --- a/src/main/resources/application.properties +++ b/src/main/resources/application.properties @@ -54,3 +54,6 @@ cookies.link=https://example.com routing.threads=10 routeDataRevision.mandatoryForSwissAddress=true + + +w3w.apikey= \ No newline at end of file diff --git a/src/test/java/net/contargo/iris/address/api/AddressApiControllerMvcUnitTest.java b/src/test/java/net/contargo/iris/address/api/AddressApiControllerMvcUnitTest.java index c4d6e445..da88da64 100644 --- a/src/test/java/net/contargo/iris/address/api/AddressApiControllerMvcUnitTest.java +++ b/src/test/java/net/contargo/iris/address/api/AddressApiControllerMvcUnitTest.java @@ -30,6 +30,7 @@ import java.util.HashMap; import java.util.Map; +import java.util.Optional; import static org.hamcrest.MatcherAssert.assertThat; @@ -169,6 +170,32 @@ public void addressesByAddressDetailsWithHashKey() throws Exception { } + @Test + public void addressesByAddressDetailsWithThreeWords() throws Exception { + + AddressDto addressDto = new AddressDto(new Address("Resolved address")); + when(addressDtoServiceMock.getAddressesByThreeWords("one.two.three")).thenReturn(Optional.of(addressDto)); + + ResultActions resultActions = mockMvc.perform(get("/geocodes/?street=one.two.three").accept(APPLICATION_JSON)); + resultActions.andExpect(status().isOk()); + resultActions.andExpect(content().contentType("application/json")); + resultActions.andExpect(jsonPath("$.geoCodeResponse.addresses[0].addresses[0].displayName").value( + "Resolved address")); + } + + + @Test + public void addressesByAddressDetailsWithFailure() throws Exception { + + when(addressDtoServiceMock.getAddressesByThreeWords("one.two.three")).thenReturn(Optional.empty()); + + ResultActions resultActions = mockMvc.perform(get("/geocodes/?street=one.two.three").accept(APPLICATION_JSON)); + resultActions.andExpect(status().isOk()); + resultActions.andExpect(content().contentType("application/json")); + resultActions.andExpect(jsonPath("$.geoCodeResponse.addresses").isEmpty()); + } + + @Test public void addressesByAddressDetailsWithHashKeyAndException() throws Exception { diff --git a/src/test/java/net/contargo/iris/address/dto/AddressDtoServiceImplUnitTest.java b/src/test/java/net/contargo/iris/address/dto/AddressDtoServiceImplUnitTest.java index 922f9e95..5dcadd48 100644 --- a/src/test/java/net/contargo/iris/address/dto/AddressDtoServiceImplUnitTest.java +++ b/src/test/java/net/contargo/iris/address/dto/AddressDtoServiceImplUnitTest.java @@ -15,6 +15,7 @@ import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.Optional; import static net.contargo.iris.address.nominatim.service.AddressDetailKey.CITY; import static net.contargo.iris.address.nominatim.service.AddressDetailKey.COUNTRY; @@ -211,4 +212,16 @@ public void getAddressesByQuery() { assertThat(addresses, hasSize(1)); assertThat(addresses.get(0).getDisplayName(), is("Gartenstr. 67, Karlsruhe (Südweststadt)")); } + + + @Test + public void getAddressesByThreeWords() { + + when(addressServiceWrapperMock.getAddressByThreeWords("riches.lofts.guessing")).thenReturn(Optional.of( + new Address("German Chancellery"))); + + Optional dto = sut.getAddressesByThreeWords("riches.lofts.guessing"); + + assertThat(dto.get().getDisplayName(), is("German Chancellery")); + } } diff --git a/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperCachingUnitTest.java b/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperCachingUnitTest.java index 46e07006..5e3dbe14 100644 --- a/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperCachingUnitTest.java +++ b/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperCachingUnitTest.java @@ -6,6 +6,7 @@ import net.contargo.iris.address.nominatim.service.AddressService; import net.contargo.iris.address.staticsearch.StaticAddress; import net.contargo.iris.address.staticsearch.service.StaticAddressService; +import net.contargo.iris.address.w3w.ThreeWordClient; import net.contargo.iris.normalizer.NormalizerService; import org.junit.Before; @@ -61,6 +62,8 @@ public class AddressServiceWrapperCachingUnitTest { private StaticAddressService staticAddressServiceMock; @Mock private AddressCache addressCacheMock; + @Mock + private ThreeWordClient threeWordClientMock; @Before public void setup() { @@ -69,7 +72,7 @@ public void setup() { when(normalizerServiceMock.normalize(CITY_NAME)).thenReturn(CITY_NAME_NORMALIZED); sut = new AddressServiceWrapper(addressServiceMock, staticAddressServiceMock, addressCacheMock, - normalizerServiceMock); + normalizerServiceMock, threeWordClientMock); } diff --git a/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperUnitTest.java b/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperUnitTest.java index 60c8d974..34c471f3 100644 --- a/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperUnitTest.java +++ b/src/test/java/net/contargo/iris/address/service/AddressServiceWrapperUnitTest.java @@ -7,6 +7,8 @@ import net.contargo.iris.address.staticsearch.StaticAddress; import net.contargo.iris.address.staticsearch.service.StaticAddressNotFoundException; import net.contargo.iris.address.staticsearch.service.StaticAddressService; +import net.contargo.iris.address.w3w.ThreeWordClient; +import net.contargo.iris.address.w3w.ThreeWordClientException; import net.contargo.iris.normalizer.NormalizerService; import org.junit.Before; @@ -24,6 +26,7 @@ import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.Optional; import static net.contargo.iris.address.nominatim.service.AddressDetailKey.CITY; import static net.contargo.iris.address.nominatim.service.AddressDetailKey.COUNTRY; @@ -81,6 +84,8 @@ public class AddressServiceWrapperUnitTest { private NormalizerService normalizerServiceMock; @Mock private AddressListFilter addressListFilterMock; + @Mock + private ThreeWordClient threeWordClientMock; private Map addressDetails; @@ -96,8 +101,8 @@ public void setUp() { when(normalizerServiceMock.normalize(CITYNAME_KARLSRUHE)).thenReturn(CITYNAME_KARLSRUHE_NORMALIZED); - sut = new AddressServiceWrapper(addressServiceMock, staticAddressServiceMock, cacheMock, - normalizerServiceMock); + sut = new AddressServiceWrapper(addressServiceMock, staticAddressServiceMock, cacheMock, normalizerServiceMock, + threeWordClientMock); } @@ -493,4 +498,32 @@ public void getAddressesByQueryWithHashkeyNotFound() { assertThat(addresses.get(0).getCity(), is("Karlsruhe")); assertThat(addresses.get(0).getPostcode(), is("76135")); } + + + @Test + public void getAddressByThreeWords() { + + GeoLocation geoLocation = new GeoLocation(new BigDecimal("52.520146"), new BigDecimal("13.36926")); + when(threeWordClientMock.resolve("riches.lofts.guessing")).thenReturn(geoLocation); + + Address address = new Address("German chancellery"); + when(cacheMock.getForLocation(geoLocation)).thenReturn(address); + + Optional
resolvedAddress = sut.getAddressByThreeWords("riches.lofts.guessing"); + + assertThat(resolvedAddress.get(), is(address)); + } + + + @Test + public void unresolvableThreeWordAddress() { + + doThrow(ThreeWordClientException.class).when(threeWordClientMock).resolve("riches.lofts.guessing"); + + Optional
resolvedAddress = sut.getAddressByThreeWords("riches.lofts.guessing"); + + assertThat(resolvedAddress.isPresent(), is(false)); + + verifyZeroInteractions(cacheMock); + } } diff --git a/src/test/java/net/contargo/iris/address/w3w/ThreeWordClientUnitTest.java b/src/test/java/net/contargo/iris/address/w3w/ThreeWordClientUnitTest.java new file mode 100644 index 00000000..eaec1ea9 --- /dev/null +++ b/src/test/java/net/contargo/iris/address/w3w/ThreeWordClientUnitTest.java @@ -0,0 +1,130 @@ +package net.contargo.iris.address.w3w; + +import com.fasterxml.jackson.databind.DeserializationFeature; +import com.fasterxml.jackson.databind.ObjectMapper; + +import net.contargo.iris.GeoLocation; + +import org.junit.Before; +import org.junit.Test; + +import org.junit.runner.RunWith; + +import org.mockito.Mock; + +import org.mockito.runners.MockitoJUnitRunner; + +import org.springframework.web.client.RestTemplate; + +import java.io.IOException; + +import java.math.BigDecimal; + +import static org.hamcrest.Matchers.is; + +import static org.junit.Assert.assertThat; + +import static org.mockito.Mockito.when; + +import static org.springframework.http.ResponseEntity.ok; + + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +@RunWith(MockitoJUnitRunner.class) +public class ThreeWordClientUnitTest { + + private static final String API_KEY = "apikey"; + private static final String REQUEST_URI = + "https://api.what3words.com/v2/forward?addr={w3wAddress}&key={apiKey}&lang=de"; + + private ThreeWordClient sut; + + @Mock + private RestTemplate restTemplateMock; + + @Before + public void setUp() { + + sut = new ThreeWordClient(restTemplateMock, API_KEY); + } + + + @Test + public void resolve() throws IOException { + + when(restTemplateMock.getForEntity(REQUEST_URI, ForwardW3wResponse.class, "wohin.polizist.meinung", API_KEY)) + .thenReturn(ok(response())); + + GeoLocation geoLocation = sut.resolve("wohin.polizist.meinung"); + + assertThat(geoLocation, is(new GeoLocation(new BigDecimal("52.520119"), new BigDecimal("13.369304")))); + } + + + @Test(expected = ThreeWordClientException.class) + public void resolveError() throws IOException { + + when(restTemplateMock.getForEntity(REQUEST_URI, ForwardW3wResponse.class, "wohin.polizist.meinung", API_KEY)) + .thenReturn(ok(errorResponse())); + + sut.resolve("wohin.polizist.meinung"); + } + + + private ForwardW3wResponse response() throws IOException { + + String json = "{" + + " \"thanks\": \"Thanks from all of us at index.home.raft for using a what3words API\"," + + " \"crs\": {" + + " \"type\": \"link\"," + + " \"properties\": {" + + " \"href\": \"http://spatialreference.org/ref/epsg/4326/ogcwkt/\"," + + " \"type\": \"ogcwkt\"" + + " }" + + " }," + + " \"words\": \"wohin.polizist.meinung\"," + + " \"bounds\": {" + + " \"southwest\": {" + + " \"lng\": 13.369282," + + " \"lat\": 52.520106" + + " }," + + " \"northeast\": {" + + " \"lng\": 13.369326," + + " \"lat\": 52.520133" + + " }" + + " }," + + " \"geometry\": {" + + " \"lng\": 13.369304," + + " \"lat\": 52.520119" + + " }," + + " \"language\": \"de\"," + + " \"map\": \"http://w3w.co/wohin.polizist.meinung\"," + + " \"status\": {" + + " \"reason\": \"OK\"," + + " \"status\": 200" + + " }" + + "}"; + + return new ObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false) + .readValue(json, ForwardW3wResponse.class); + } + + + private ForwardW3wResponse errorResponse() throws IOException { + + String json = "{" + + " \"thanks\": \"Thanks from all of us at index.home.raft for using a what3words API\"," + + " \"status\": {" + + " \"reason\": \"OK\"," + + " \"code\": 300," + + " \"message\": \"Invalid or non-existent 3 word address\"," + + " \"status\": 200" + + " }" + + "}"; + + return new ObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false) + .readValue(json, ForwardW3wResponse.class); + } +} diff --git a/src/test/java/net/contargo/iris/address/w3w/ThreeWordMatcherUnitTest.java b/src/test/java/net/contargo/iris/address/w3w/ThreeWordMatcherUnitTest.java new file mode 100644 index 00000000..542a177b --- /dev/null +++ b/src/test/java/net/contargo/iris/address/w3w/ThreeWordMatcherUnitTest.java @@ -0,0 +1,23 @@ +package net.contargo.iris.address.w3w; + +import org.junit.Test; + +import static org.hamcrest.Matchers.is; + +import static org.junit.Assert.assertThat; + + +/** + * @author Sandra Thieme - thieme@synyx.de + */ +public class ThreeWordMatcherUnitTest { + + @Test + public void isThreeWordAddress() { + + assertThat(ThreeWordMatcher.isThreeWordAddress("riches.lofts.guessing"), is(true)); + assertThat(ThreeWordMatcher.isThreeWordAddress(" riches.lofts.guessing "), is(true)); + assertThat(ThreeWordMatcher.isThreeWordAddress(null), is(false)); + assertThat(ThreeWordMatcher.isThreeWordAddress("Gartenstr. 67"), is(false)); + } +}