Repository: nifi
Updated Branches:
  refs/heads/master e46f4131f -> 3dc7a160c


NIFI-3478: This closes #1507. added tests for GeoEnrichIP and added support for 
expression language on 'IP Address Attribute' property.


Project: http://git-wip-us.apache.org/repos/asf/nifi/repo
Commit: http://git-wip-us.apache.org/repos/asf/nifi/commit/3dc7a160
Tree: http://git-wip-us.apache.org/repos/asf/nifi/tree/3dc7a160
Diff: http://git-wip-us.apache.org/repos/asf/nifi/diff/3dc7a160

Branch: refs/heads/master
Commit: 3dc7a160ce83a1e83edacc54941ac4d0f5fff718
Parents: e46f413
Author: JJ <[email protected]>
Authored: Tue Feb 14 11:05:51 2017 -0600
Committer: joewitt <[email protected]>
Committed: Tue Feb 14 13:33:27 2017 -0500

----------------------------------------------------------------------
 .../org/apache/nifi/processors/GeoEnrichIP.java |  11 +-
 .../apache/nifi/processors/TestGeoEnrichIP.java | 355 +++++++++++++++++++
 2 files changed, 363 insertions(+), 3 deletions(-)
----------------------------------------------------------------------


http://git-wip-us.apache.org/repos/asf/nifi/blob/3dc7a160/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/main/java/org/apache/nifi/processors/GeoEnrichIP.java
----------------------------------------------------------------------
diff --git 
a/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/main/java/org/apache/nifi/processors/GeoEnrichIP.java
 
b/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/main/java/org/apache/nifi/processors/GeoEnrichIP.java
index eccff79..c24799e 100644
--- 
a/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/main/java/org/apache/nifi/processors/GeoEnrichIP.java
+++ 
b/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/main/java/org/apache/nifi/processors/GeoEnrichIP.java
@@ -42,6 +42,7 @@ import org.apache.nifi.annotation.documentation.Tags;
 import org.apache.nifi.annotation.lifecycle.OnScheduled;
 import org.apache.nifi.annotation.lifecycle.OnStopped;
 import org.apache.nifi.components.PropertyDescriptor;
+import org.apache.nifi.expression.AttributeExpression;
 import org.apache.nifi.flowfile.FlowFile;
 import org.apache.nifi.processor.AbstractProcessor;
 import org.apache.nifi.processor.ProcessContext;
@@ -81,6 +82,7 @@ public class GeoEnrichIP extends AbstractProcessor {
 
     public static final PropertyDescriptor GEO_DATABASE_FILE = new 
PropertyDescriptor.Builder()
             .name("Geo Database File")
+            .displayName("Geo Database File")
             .description("Path to Maxmind Geo Enrichment Database File")
             .required(true)
             .addValidator(StandardValidators.FILE_EXISTS_VALIDATOR)
@@ -88,9 +90,12 @@ public class GeoEnrichIP extends AbstractProcessor {
 
     public static final PropertyDescriptor IP_ADDRESS_ATTRIBUTE = new 
PropertyDescriptor.Builder()
             .name("IP Address Attribute")
+            .displayName("IP Address Attribute")
             .required(true)
             .description("The name of an attribute whose value is a dotted 
decimal IP address for which enrichment should occur")
+            .expressionLanguageSupported(true)
             .addValidator(StandardValidators.NON_EMPTY_VALIDATOR)
+            
.addValidator(StandardValidators.createAttributeExpressionLanguageValidator(AttributeExpression.ResultType.STRING))
             .build();
 
     public static final Relationship REL_FOUND = new Relationship.Builder()
@@ -105,7 +110,7 @@ public class GeoEnrichIP extends AbstractProcessor {
 
     private Set<Relationship> relationships;
     private List<PropertyDescriptor> propertyDescriptors;
-    private final AtomicReference<DatabaseReader> databaseReaderRef = new 
AtomicReference<>(null);
+    final AtomicReference<DatabaseReader> databaseReaderRef = new 
AtomicReference<>(null);
 
     @Override
     public Set<Relationship> getRelationships() {
@@ -118,7 +123,7 @@ public class GeoEnrichIP extends AbstractProcessor {
     }
 
     @OnScheduled
-    public final void onScheduled(final ProcessContext context) throws 
IOException {
+    public void onScheduled(final ProcessContext context) throws IOException {
         final String dbFileString = 
context.getProperty(GEO_DATABASE_FILE).getValue();
         final File dbFile = new File(dbFileString);
         final StopWatch stopWatch = new StopWatch(true);
@@ -157,7 +162,7 @@ public class GeoEnrichIP extends AbstractProcessor {
         }
 
         final DatabaseReader dbReader = databaseReaderRef.get();
-        final String ipAttributeName = 
context.getProperty(IP_ADDRESS_ATTRIBUTE).getValue();
+        final String ipAttributeName = 
context.getProperty(IP_ADDRESS_ATTRIBUTE).evaluateAttributeExpressions(flowFile).getValue();
         final String ipAttributeValue = flowFile.getAttribute(ipAttributeName);
         if (StringUtils.isEmpty(ipAttributeName)) { //TODO need to add 
additional validation - should look like an IPv4 or IPv6 addr for instance
             session.transfer(flowFile, REL_NOT_FOUND);

http://git-wip-us.apache.org/repos/asf/nifi/blob/3dc7a160/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/test/java/org/apache/nifi/processors/TestGeoEnrichIP.java
----------------------------------------------------------------------
diff --git 
a/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/test/java/org/apache/nifi/processors/TestGeoEnrichIP.java
 
b/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/test/java/org/apache/nifi/processors/TestGeoEnrichIP.java
new file mode 100644
index 0000000..a404b56
--- /dev/null
+++ 
b/nifi-nar-bundles/nifi-enrich-bundle/nifi-enrich-processors/src/test/java/org/apache/nifi/processors/TestGeoEnrichIP.java
@@ -0,0 +1,355 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements.  See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You 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.apache.nifi.processors;
+
+import com.fasterxml.jackson.databind.DeserializationFeature;
+import com.fasterxml.jackson.databind.InjectableValues;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.maxmind.geoip2.exception.GeoIp2Exception;
+import com.maxmind.geoip2.model.CityResponse;
+import org.apache.nifi.annotation.lifecycle.OnScheduled;
+import org.apache.nifi.flowfile.FlowFile;
+import org.apache.nifi.processor.ProcessContext;
+import org.apache.nifi.processors.maxmind.DatabaseReader;
+import org.apache.nifi.util.MockFlowFile;
+import org.apache.nifi.util.TestRunner;
+import org.apache.nifi.util.TestRunners;
+import org.junit.Before;
+import org.junit.Test;
+
+import java.io.IOException;
+import java.net.InetAddress;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertNotNull;
+import static org.junit.Assert.assertNull;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.verify;
+import static org.mockito.Mockito.verifyNoMoreInteractions;
+import static org.mockito.Mockito.when;
+
+@SuppressWarnings("WeakerAccess")
+public class TestGeoEnrichIP {
+    DatabaseReader databaseReader;
+    GeoEnrichIP geoEnrichIP;
+    TestRunner testRunner;
+
+    @Before
+    public void setUp() throws Exception {
+        databaseReader = mock(DatabaseReader.class);
+        geoEnrichIP = new TestableGeoEnrichIP();
+        testRunner = TestRunners.newTestRunner(geoEnrichIP);
+    }
+
+    @Test
+    public void verifyNonExistentIpFlowsToNotFoundRelationship() throws 
Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        testRunner.enqueue(new byte[0], Collections.emptyMap());
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(1, notFound.size());
+        assertEquals(0, found.size());
+
+        verify(databaseReader).city(InetAddress.getByName(null));
+    }
+
+    @Test
+    public void successfulMaxMindResponseShouldFlowToFoundRelationship() 
throws Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        final CityResponse cityResponse = getFullCityResponse();
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenReturn(cityResponse);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(0, notFound.size());
+        assertEquals(1, found.size());
+
+        FlowFile finishedFound = found.get(0);
+        assertNotNull(finishedFound.getAttribute("ip.geo.lookup.micros"));
+        assertEquals("Minneapolis", finishedFound.getAttribute("ip.geo.city"));
+        assertEquals("44.98", finishedFound.getAttribute("ip.geo.latitude"));
+        assertEquals("93.2636", 
finishedFound.getAttribute("ip.geo.longitude"));
+        assertEquals("Minnesota", 
finishedFound.getAttribute("ip.geo.subdivision.0"));
+        assertEquals("MN", 
finishedFound.getAttribute("ip.geo.subdivision.isocode.0"));
+        assertNull(finishedFound.getAttribute("ip.geo.subdivision.1"));
+        assertEquals("TT", 
finishedFound.getAttribute("ip.geo.subdivision.isocode.1"));
+        assertEquals("United States of America", 
finishedFound.getAttribute("ip.geo.country"));
+        assertEquals("US", 
finishedFound.getAttribute("ip.geo.country.isocode"));
+        assertEquals("55401", finishedFound.getAttribute("ip.geo.postalcode"));
+    }
+
+    @Test
+    public void 
successfulMaxMindResponseShouldFlowToFoundRelationshipWhenLatAndLongAreNotSet() 
throws Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        final CityResponse cityResponse = getNullLatAndLongCityResponse();
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenReturn(cityResponse);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(0, notFound.size());
+        assertEquals(1, found.size());
+
+        FlowFile finishedFound = found.get(0);
+        assertNotNull(finishedFound.getAttribute("ip.geo.lookup.micros"));
+        assertEquals("Minneapolis", finishedFound.getAttribute("ip.geo.city"));
+        assertNull(finishedFound.getAttribute("ip.geo.latitude"));
+        assertNull(finishedFound.getAttribute("ip.geo.longitude"));
+        assertEquals("Minnesota", 
finishedFound.getAttribute("ip.geo.subdivision.0"));
+        assertEquals("MN", 
finishedFound.getAttribute("ip.geo.subdivision.isocode.0"));
+        assertNull(finishedFound.getAttribute("ip.geo.subdivision.1"));
+        assertEquals("TT", 
finishedFound.getAttribute("ip.geo.subdivision.isocode.1"));
+        assertEquals("United States of America", 
finishedFound.getAttribute("ip.geo.country"));
+        assertEquals("US", 
finishedFound.getAttribute("ip.geo.country.isocode"));
+        assertEquals("55401", finishedFound.getAttribute("ip.geo.postalcode"));
+    }
+
+    @Test
+    public void 
evaluatingExpressionLanguageShouldAndFindingIpFieldWithSuccessfulLookUpShouldFlowToFoundRelationship()
 throws Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, 
"${ip.fields:substringBefore(',')}");
+
+        final CityResponse cityResponse = getNullLatAndLongCityResponse();
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenReturn(cityResponse);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip.fields", "ip0,ip1,ip2");
+        attributes.put("ip0", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(0, notFound.size());
+        assertEquals(1, found.size());
+
+        FlowFile finishedFound = found.get(0);
+        assertNotNull(finishedFound.getAttribute("ip0.geo.lookup.micros"));
+        assertEquals("Minneapolis", 
finishedFound.getAttribute("ip0.geo.city"));
+        assertNull(finishedFound.getAttribute("ip0.geo.latitude"));
+        assertNull(finishedFound.getAttribute("ip0.geo.longitude"));
+        assertEquals("Minnesota", 
finishedFound.getAttribute("ip0.geo.subdivision.0"));
+        assertEquals("MN", 
finishedFound.getAttribute("ip0.geo.subdivision.isocode.0"));
+        assertNull(finishedFound.getAttribute("ip0.geo.subdivision.1"));
+        assertEquals("TT", 
finishedFound.getAttribute("ip0.geo.subdivision.isocode.1"));
+        assertEquals("United States of America", 
finishedFound.getAttribute("ip0.geo.country"));
+        assertEquals("US", 
finishedFound.getAttribute("ip0.geo.country.isocode"));
+        assertEquals("55401", 
finishedFound.getAttribute("ip0.geo.postalcode"));
+    }
+
+    @Test
+    public void shouldFlowToNotFoundWhenNullResponseFromMaxMind() throws 
Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenReturn(null);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(1, notFound.size());
+        assertEquals(0, found.size());
+    }
+
+    @SuppressWarnings("unchecked")
+    @Test
+    public void shouldFlowToNotFoundWhenIOExceptionThrownFromMaxMind() throws 
Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenThrow(IOException.class);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(1, notFound.size());
+        assertEquals(0, found.size());
+    }
+
+    @SuppressWarnings("unchecked")
+    @Test
+    public void shouldFlowToNotFoundWhenGeoIp2ExceptionThrownFromMaxMind() 
throws Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        
when(databaseReader.city(InetAddress.getByName("1.2.3.4"))).thenThrow(GeoIp2Exception.class);
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "1.2.3.4");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(1, notFound.size());
+        assertEquals(0, found.size());
+    }
+
+    @SuppressWarnings("unchecked")
+    @Test
+    public void 
whenInetAddressThrowsUnknownHostFlowFileShouldBeSentToNotFound() throws 
Exception {
+        testRunner.setProperty(GeoEnrichIP.GEO_DATABASE_FILE, "./");
+        testRunner.setProperty(GeoEnrichIP.IP_ADDRESS_ATTRIBUTE, "ip");
+
+        final Map<String, String> attributes = new HashMap<>();
+        attributes.put("ip", "somenonexistentdomain.comm");
+
+        testRunner.enqueue(new byte[0], attributes);
+
+        testRunner.run();
+
+        List<MockFlowFile> notFound = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_NOT_FOUND);
+        List<MockFlowFile> found = 
testRunner.getFlowFilesForRelationship(GeoEnrichIP.REL_FOUND);
+
+        assertEquals(1, notFound.size());
+        assertEquals(0, found.size());
+
+        verify(databaseReader).close();
+        verifyNoMoreInteractions(databaseReader);
+    }
+
+    private CityResponse getFullCityResponse() throws Exception {
+        // Taken from MaxMind unit tests.
+        final String maxMindCityResponse = "{" + "\"city\":{" + 
"\"confidence\":76,"
+            + "\"geoname_id\":9876," + "\"names\":{" + "\"en\":\"Minneapolis\""
+            + "}" + "}," + "\"continent\":{" + "\"code\":\"NA\","
+            + "\"geoname_id\":42," + "\"names\":{" + "\"en\":\"North America\""
+            + "}" + "}," + "\"country\":{" + "\"confidence\":99,"
+            + "\"iso_code\":\"US\"," + "\"geoname_id\":1," + "\"names\":{"
+            + "\"en\":\"United States of America\"" + "}" + "},"
+            + "\"location\":{" + "\"accuracy_radius\":1500,"
+            + "\"latitude\":44.98," + "\"longitude\":93.2636,"
+            + "\"metro_code\":765," + "\"time_zone\":\"America/Chicago\""
+            + "}," + "\"postal\":{\"confidence\": 33, \"code\":\"55401\"},"
+            + "\"registered_country\":{" + "\"geoname_id\":2,"
+            + "\"iso_code\":\"CA\"," + "\"names\":{" + "\"en\":\"Canada\""
+            + "}" + "}," + "\"represented_country\":{" + "\"geoname_id\":3,"
+            + "\"iso_code\":\"GB\"," + "\"names\":{"
+            + "\"en\":\"United Kingdom\"" + "}," + "\"type\":\"C<military>\""
+            + "}," + "\"subdivisions\":[{" + "\"confidence\":88,"
+            + "\"geoname_id\":574635," + "\"iso_code\":\"MN\"," + "\"names\":{"
+            + "\"en\":\"Minnesota\"" + "}" + "}," + "{\"iso_code\":\"TT\"}],"
+            + "\"traits\":{" + "\"autonomous_system_number\":1234,"
+            + "\"autonomous_system_organization\":\"AS Organization\","
+            + "\"domain\":\"example.com\"," + "\"ip_address\":\"1.2.3.4\","
+            + "\"is_anonymous_proxy\":true,"
+            + "\"is_satellite_provider\":true," + "\"isp\":\"Comcast\","
+            + "\"organization\":\"Blorg\"," + "\"user_type\":\"college\""
+            + "}," + "\"maxmind\":{\"queries_remaining\":11}" + "}";
+
+        InjectableValues inject = new 
InjectableValues.Std().addValue("locales", Collections.singletonList("en"));
+        ObjectMapper mapper = new ObjectMapper();
+        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, 
false);
+
+        return new 
ObjectMapper().reader(CityResponse.class).with(inject).readValue(maxMindCityResponse);
+    }
+
+    private CityResponse getNullLatAndLongCityResponse() throws Exception {
+        // Taken from MaxMind unit tests and modified.
+        final String maxMindCityResponse = "{" + "\"city\":{" + 
"\"confidence\":76,"
+            + "\"geoname_id\":9876," + "\"names\":{" + "\"en\":\"Minneapolis\""
+            + "}" + "}," + "\"continent\":{" + "\"code\":\"NA\","
+            + "\"geoname_id\":42," + "\"names\":{" + "\"en\":\"North America\""
+            + "}" + "}," + "\"country\":{" + "\"confidence\":99,"
+            + "\"iso_code\":\"US\"," + "\"geoname_id\":1," + "\"names\":{"
+            + "\"en\":\"United States of America\"" + "}" + "},"
+            + "\"location\":{" + "\"accuracy_radius\":1500,"
+            + "\"metro_code\":765," + "\"time_zone\":\"America/Chicago\""
+            + "}," + "\"postal\":{\"confidence\": 33, \"code\":\"55401\"},"
+            + "\"registered_country\":{" + "\"geoname_id\":2,"
+            + "\"iso_code\":\"CA\"," + "\"names\":{" + "\"en\":\"Canada\""
+            + "}" + "}," + "\"represented_country\":{" + "\"geoname_id\":3,"
+            + "\"iso_code\":\"GB\"," + "\"names\":{"
+            + "\"en\":\"United Kingdom\"" + "}," + "\"type\":\"C<military>\""
+            + "}," + "\"subdivisions\":[{" + "\"confidence\":88,"
+            + "\"geoname_id\":574635," + "\"iso_code\":\"MN\"," + "\"names\":{"
+            + "\"en\":\"Minnesota\"" + "}" + "}," + "{\"iso_code\":\"TT\"}],"
+            + "\"traits\":{" + "\"autonomous_system_number\":1234,"
+            + "\"autonomous_system_organization\":\"AS Organization\","
+            + "\"domain\":\"example.com\"," + "\"ip_address\":\"1.2.3.4\","
+            + "\"is_anonymous_proxy\":true,"
+            + "\"is_satellite_provider\":true," + "\"isp\":\"Comcast\","
+            + "\"organization\":\"Blorg\"," + "\"user_type\":\"college\""
+            + "}," + "\"maxmind\":{\"queries_remaining\":11}" + "}";
+
+        InjectableValues inject = new 
InjectableValues.Std().addValue("locales", Collections.singletonList("en"));
+        ObjectMapper mapper = new ObjectMapper();
+        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, 
false);
+
+        return new 
ObjectMapper().reader(CityResponse.class).with(inject).readValue(maxMindCityResponse);
+    }
+
+    class TestableGeoEnrichIP extends GeoEnrichIP {
+        @OnScheduled
+        @Override
+        public void onScheduled(ProcessContext context) throws IOException {
+            databaseReaderRef.set(databaseReader);
+        }
+    }
+}

Reply via email to