From e9417e3eb5caae984d9ef9c34e721e5a02ca787f Mon Sep 17 00:00:00 2001 From: EvgeniiMunin <35193823+EvgeniiMunin@users.noreply.github.com> Date: Fri, 17 Jan 2025 05:14:56 -0800 Subject: [PATCH 1/3] Greenbids fix geolookup: fetch from official MaxMind URL + mock dbReader UT (#3626) --- .../data/config/DatabaseReaderFactory.java | 114 +++++++++++++++--- .../GreenbidsRealTimeDataConfiguration.java | 9 +- .../GreenbidsRealTimeDataProperties.java | 6 + .../core/GreenbidsInferenceDataService.java | 37 ++++-- .../GreenbidsInferenceDataServiceTest.java | 63 ++++++++-- .../time/data/core/OnnxModelRunnerTest.java | 4 +- .../data/util/TestBidRequestProvider.java | 12 ++ ...meDataProcessedAuctionRequestHookTest.java | 38 ++++-- .../greenbids/GreenbidsAnalyticsReporter.java | 2 + .../GreenbidsAnalyticsReporterTest.java | 2 + 10 files changed, 233 insertions(+), 54 deletions(-) diff --git a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/DatabaseReaderFactory.java b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/DatabaseReaderFactory.java index a40c98ebb25..322cdf48ff4 100644 --- a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/DatabaseReaderFactory.java +++ b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/DatabaseReaderFactory.java @@ -1,52 +1,128 @@ package org.prebid.server.hooks.modules.greenbids.real.time.data.config; +import io.netty.handler.codec.http.HttpResponseStatus; +import io.vertx.core.file.FileSystem; +import io.vertx.core.http.HttpClientOptions; +import io.vertx.core.http.HttpClientRequest; +import io.vertx.core.http.HttpClientResponse; +import com.maxmind.db.Reader; +import io.vertx.core.Future; import io.vertx.core.Promise; import io.vertx.core.Vertx; import com.maxmind.geoip2.DatabaseReader; +import io.vertx.core.file.OpenOptions; +import io.vertx.core.http.HttpMethod; +import io.vertx.core.http.RequestOptions; +import org.apache.commons.compress.archivers.tar.TarArchiveEntry; +import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; import org.prebid.server.exception.PreBidException; +import org.prebid.server.log.Logger; +import org.prebid.server.log.LoggerFactory; import org.prebid.server.vertx.Initializable; -import java.io.FileOutputStream; import java.io.IOException; -import java.io.InputStream; -import java.net.URL; import java.nio.file.Files; import java.nio.file.Path; import java.util.concurrent.atomic.AtomicReference; +import java.util.zip.GZIPInputStream; public class DatabaseReaderFactory implements Initializable { - private final String geoLiteCountryUrl; + private static final Logger logger = LoggerFactory.getLogger(DatabaseReaderFactory.class); + + private final GreenbidsRealTimeDataProperties properties; private final Vertx vertx; private final AtomicReference databaseReaderRef = new AtomicReference<>(); - public DatabaseReaderFactory(String geoLitCountryUrl, Vertx vertx) { - this.geoLiteCountryUrl = geoLitCountryUrl; + private final FileSystem fileSystem; + + public DatabaseReaderFactory(GreenbidsRealTimeDataProperties properties, Vertx vertx) { + this.properties = properties; this.vertx = vertx; + this.fileSystem = vertx.fileSystem(); } @Override public void initialize(Promise initializePromise) { + downloadAndExtract() + .onSuccess(databaseReaderRef::set) + .mapEmpty() + .onComplete(initializePromise); + } + + private Future downloadAndExtract() { + final String downloadUrl = properties.getGeoLiteCountryPath(); + final String tmpPath = properties.getTmpPath(); + return downloadFile(downloadUrl, tmpPath) + .compose(ignored -> vertx.executeBlocking(() -> extractMMDB(tmpPath))) + .onComplete(ar -> removeFile(tmpPath)); + } + + private Future downloadFile(String downloadUrl, String tmpPath) { + return fileSystem.open(tmpPath, new OpenOptions()) + .compose(tmpFile -> sendHttpRequest(downloadUrl) + .onFailure(ignore -> tmpFile.close()) + .compose(response -> response.pipeTo(tmpFile))); + } + + private Future sendHttpRequest(String url) { + final RequestOptions options = new RequestOptions() + .setFollowRedirects(true) + .setMethod(HttpMethod.GET) + .setTimeout(properties.getTimeoutMs()) + .setAbsoluteURI(url); + + final HttpClientOptions httpClientOptions = new HttpClientOptions() + .setConnectTimeout(properties.getTimeoutMs().intValue()) + .setMaxRedirects(properties.getMaxRedirects()); - vertx.executeBlocking(() -> { - try { - final URL url = new URL(geoLiteCountryUrl); - final Path databasePath = Files.createTempFile("GeoLite2-Country", ".mmdb"); + return vertx.createHttpClient(httpClientOptions).request(options) + .compose(HttpClientRequest::send) + .map(this::validateResponse); + } + + private HttpClientResponse validateResponse(HttpClientResponse response) { + final int statusCode = response.statusCode(); + if (statusCode != HttpResponseStatus.OK.code()) { + throw new PreBidException("Got unexpected response from server with status code %s and message %s" + .formatted(statusCode, response.statusMessage())); + } + return response; + } - try (InputStream inputStream = url.openStream(); - FileOutputStream outputStream = new FileOutputStream(databasePath.toFile())) { - inputStream.transferTo(outputStream); + private DatabaseReader extractMMDB(String tarGzPath) { + try (GZIPInputStream gis = new GZIPInputStream(Files.newInputStream(Path.of(tarGzPath))); + TarArchiveInputStream tarInput = new TarArchiveInputStream(gis)) { + + TarArchiveEntry currentEntry; + boolean hasDatabaseFile = false; + while ((currentEntry = tarInput.getNextTarEntry()) != null) { + if (currentEntry.getName().contains("GeoLite2-Country.mmdb")) { + hasDatabaseFile = true; + break; } + } + + if (!hasDatabaseFile) { + throw new PreBidException("GeoLite2-Country.mmdb not found in the archive"); + } + + return new DatabaseReader.Builder(tarInput) + .fileMode(Reader.FileMode.MEMORY).build(); + } catch (IOException e) { + throw new PreBidException("Failed to extract MMDB file", e); + } + } - databaseReaderRef.set(new DatabaseReader.Builder(databasePath.toFile()).build()); - } catch (IOException e) { - throw new PreBidException("Failed to initialize DatabaseReader from URL", e); + private void removeFile(String filePath) { + fileSystem.exists(filePath).onSuccess(exists -> { + if (exists) { + fileSystem.delete(filePath) + .onFailure(err -> logger.error("Failed to remove file {}", filePath, err)); } - return null; - }).mapEmpty() - .onComplete(initializePromise); + }); } public DatabaseReader getDatabaseReader() { diff --git a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataConfiguration.java b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataConfiguration.java index 959352d1908..a6b47f7d6ad 100644 --- a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataConfiguration.java +++ b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataConfiguration.java @@ -5,6 +5,7 @@ import com.google.cloud.storage.Storage; import com.google.cloud.storage.StorageOptions; import io.vertx.core.Vertx; +import org.prebid.server.geolocation.CountryCodeMapper; import org.prebid.server.hooks.modules.greenbids.real.time.data.model.filter.ThrottlingThresholds; import org.prebid.server.hooks.modules.greenbids.real.time.data.core.ThrottlingThresholdsFactory; import org.prebid.server.hooks.modules.greenbids.real.time.data.core.GreenbidsInferenceDataService; @@ -32,13 +33,15 @@ public class GreenbidsRealTimeDataConfiguration { @Bean DatabaseReaderFactory databaseReaderFactory(GreenbidsRealTimeDataProperties properties, Vertx vertx) { - return new DatabaseReaderFactory(properties.getGeoLiteCountryPath(), vertx); + return new DatabaseReaderFactory(properties, vertx); } @Bean - GreenbidsInferenceDataService greenbidsInferenceDataService(DatabaseReaderFactory databaseReaderFactory) { + GreenbidsInferenceDataService greenbidsInferenceDataService(DatabaseReaderFactory databaseReaderFactory, + CountryCodeMapper countryCodeMapper) { + return new GreenbidsInferenceDataService( - databaseReaderFactory, ObjectMapperProvider.mapper()); + databaseReaderFactory, ObjectMapperProvider.mapper(), countryCodeMapper); } @Bean diff --git a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataProperties.java b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataProperties.java index 86736a6011f..4752a2e8840 100644 --- a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataProperties.java +++ b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/config/GreenbidsRealTimeDataProperties.java @@ -11,6 +11,8 @@ public class GreenbidsRealTimeDataProperties { String geoLiteCountryPath; + String tmpPath; + String gcsBucketName; Integer cacheExpirationMinutes; @@ -18,4 +20,8 @@ public class GreenbidsRealTimeDataProperties { String onnxModelCacheKeyPrefix; String thresholdsCacheKeyPrefix; + + Long timeoutMs; + + Integer maxRedirects; } diff --git a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataService.java b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataService.java index 3bd3e37b859..c16a8de917f 100644 --- a/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataService.java +++ b/extra/modules/greenbids-real-time-data/src/main/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataService.java @@ -6,6 +6,7 @@ import com.fasterxml.jackson.databind.node.ObjectNode; import com.iab.openrtb.request.BidRequest; import com.iab.openrtb.request.Device; +import com.iab.openrtb.request.Geo; import com.iab.openrtb.request.Imp; import com.maxmind.geoip2.DatabaseReader; import com.maxmind.geoip2.exception.GeoIp2Exception; @@ -13,6 +14,7 @@ import com.maxmind.geoip2.record.Country; import org.apache.commons.lang3.StringUtils; import org.prebid.server.exception.PreBidException; +import org.prebid.server.geolocation.CountryCodeMapper; import org.prebid.server.hooks.modules.greenbids.real.time.data.config.DatabaseReaderFactory; import org.prebid.server.hooks.modules.greenbids.real.time.data.model.data.ThrottlingMessage; import org.prebid.server.proto.openrtb.ext.request.ExtImpPrebid; @@ -25,6 +27,7 @@ import java.util.Collection; import java.util.Iterator; import java.util.List; +import java.util.Locale; import java.util.Objects; import java.util.Optional; import java.util.stream.Collectors; @@ -35,9 +38,14 @@ public class GreenbidsInferenceDataService { private final ObjectMapper mapper; - public GreenbidsInferenceDataService(DatabaseReaderFactory dbReaderFactory, ObjectMapper mapper) { + private final CountryCodeMapper countryCodeMapper; + + public GreenbidsInferenceDataService(DatabaseReaderFactory dbReaderFactory, + ObjectMapper mapper, + CountryCodeMapper countryCodeMapper) { this.databaseReaderFactory = Objects.requireNonNull(dbReaderFactory); this.mapper = Objects.requireNonNull(mapper); + this.countryCodeMapper = Objects.requireNonNull(countryCodeMapper); } public List extractThrottlingMessagesFromBidRequest(BidRequest bidRequest) { @@ -86,23 +94,38 @@ private List extractMessagesForImp( final String ip = Optional.ofNullable(bidRequest.getDevice()) .map(Device::getIp) .orElse(null); - final String countryFromIp = getCountry(ip); + final String country = Optional.ofNullable(bidRequest.getDevice()) + .map(Device::getGeo) + .map(Geo::getCountry) + .map(countryCodeMapper::mapToAlpha2) + .map(GreenbidsInferenceDataService::getCountryNameFromAlpha2) + .filter(c -> !c.isEmpty()) + .orElseGet(() -> getCountry(ip)); + return createThrottlingMessages( bidderNode, impId, greenbidsUserAgent, - countryFromIp, + country, hostname, hourBucket, minuteQuadrant); } - private String getCountry(String ip) { - if (ip == null) { - return null; - } + private static String getCountryNameFromAlpha2(String isoCode) { + return StringUtils.isBlank(isoCode) + ? StringUtils.EMPTY + : new Locale(StringUtils.EMPTY, isoCode).getDisplayCountry(); + } + private String getCountry(String ip) { final DatabaseReader databaseReader = databaseReaderFactory.getDatabaseReader(); + return ip != null && databaseReader != null + ? getCountryFromIpUsingDatabase(databaseReader, ip) + : null; + } + + private String getCountryFromIpUsingDatabase(DatabaseReader databaseReader, String ip) { try { final InetAddress inetAddress = InetAddress.getByName(ip); final CountryResponse response = databaseReader.country(inetAddress); diff --git a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataServiceTest.java b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataServiceTest.java index 1ac1bcc5cb1..fafc759ca11 100644 --- a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataServiceTest.java +++ b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/GreenbidsInferenceDataServiceTest.java @@ -15,6 +15,7 @@ import org.mockito.Mock; import org.mockito.junit.jupiter.MockitoExtension; import org.prebid.server.exception.PreBidException; +import org.prebid.server.geolocation.CountryCodeMapper; import org.prebid.server.hooks.modules.greenbids.real.time.data.config.DatabaseReaderFactory; import org.prebid.server.hooks.modules.greenbids.real.time.data.model.data.ThrottlingMessage; import org.prebid.server.hooks.modules.greenbids.real.time.data.util.TestBidRequestProvider; @@ -50,16 +51,20 @@ public class GreenbidsInferenceDataServiceTest { @Mock private Country country; + @Mock + private CountryCodeMapper countryCodeMapper; + private GreenbidsInferenceDataService target; @BeforeEach public void setUp() { when(databaseReaderFactory.getDatabaseReader()).thenReturn(databaseReader); - target = new GreenbidsInferenceDataService(databaseReaderFactory, TestBidRequestProvider.MAPPER); + target = new GreenbidsInferenceDataService( + databaseReaderFactory, TestBidRequestProvider.MAPPER, countryCodeMapper); } @Test - public void extractThrottlingMessagesFromBidRequestShouldReturnValidThrottlingMessages() + public void extractThrottlingMessagesFromBidRequestShouldReturnValidThrottlingMessagesWhenGeoIsNull() throws IOException, GeoIp2Exception { // given final Banner banner = givenBanner(); @@ -79,20 +84,57 @@ public void extractThrottlingMessagesFromBidRequestShouldReturnValidThrottlingMe when(databaseReader.country(any(InetAddress.class))).thenReturn(countryResponse); when(countryResponse.getCountry()).thenReturn(country); - when(country.getName()).thenReturn("US"); + when(country.getName()).thenReturn("United States"); // when final List throttlingMessages = target.extractThrottlingMessagesFromBidRequest(bidRequest); // then assertThat(throttlingMessages).isNotEmpty(); - assertThat(throttlingMessages.getFirst().getBidder()).isEqualTo("rubicon"); - assertThat(throttlingMessages.get(1).getBidder()).isEqualTo("appnexus"); - assertThat(throttlingMessages.getLast().getBidder()).isEqualTo("pubmatic"); + assertThat(throttlingMessages) + .extracting(ThrottlingMessage::getBidder) + .containsExactly("rubicon", "appnexus", "pubmatic"); throttlingMessages.forEach(message -> { assertThat(message.getAdUnitCode()).isEqualTo("adunitcodevalue"); - assertThat(message.getCountry()).isEqualTo("US"); + assertThat(message.getCountry()).isEqualTo("United States"); + assertThat(message.getHostname()).isEqualTo("www.leparisien.fr"); + assertThat(message.getDevice()).isEqualTo("PC"); + assertThat(message.getHourBucket()).isEqualTo(String.valueOf(expectedHourBucket)); + assertThat(message.getMinuteQuadrant()).isEqualTo(String.valueOf(expectedMinuteQuadrant)); + }); + } + + @Test + public void extractThrottlingMessagesFromBidRequestShouldReturnValidThrottlingMessagesWhenGeoDefined() { + // given + final Banner banner = givenBanner(); + final Imp imp = Imp.builder() + .id("adunitcodevalue") + .ext(givenImpExt()) + .banner(banner) + .build(); + final Device device = givenDevice(identity(), "FRA"); + final BidRequest bidRequest = givenBidRequest(request -> request, List.of(imp), device, null); + + final ZonedDateTime timestamp = ZonedDateTime.now(ZoneId.of("UTC")); + final Integer expectedHourBucket = timestamp.getHour(); + final Integer expectedMinuteQuadrant = (timestamp.getMinute() / 15) + 1; + + when(countryCodeMapper.mapToAlpha2("FRA")).thenReturn("FR"); + + // when + final List throttlingMessages = target.extractThrottlingMessagesFromBidRequest(bidRequest); + + // then + assertThat(throttlingMessages).isNotEmpty(); + assertThat(throttlingMessages) + .extracting(ThrottlingMessage::getBidder) + .containsExactly("rubicon", "appnexus", "pubmatic"); + + throttlingMessages.forEach(message -> { + assertThat(message.getAdUnitCode()).isEqualTo("adunitcodevalue"); + assertThat(message.getCountry()).isEqualTo("France"); assertThat(message.getHostname()).isEqualTo("www.leparisien.fr"); assertThat(message.getDevice()).isEqualTo("PC"); assertThat(message.getHourBucket()).isEqualTo(String.valueOf(expectedHourBucket)); @@ -121,10 +163,9 @@ public void extractThrottlingMessagesFromBidRequestShouldHandleMissingIp() { // then assertThat(throttlingMessages).isNotEmpty(); - - assertThat(throttlingMessages.getFirst().getBidder()).isEqualTo("rubicon"); - assertThat(throttlingMessages.get(1).getBidder()).isEqualTo("appnexus"); - assertThat(throttlingMessages.getLast().getBidder()).isEqualTo("pubmatic"); + assertThat(throttlingMessages) + .extracting(ThrottlingMessage::getBidder) + .containsExactly("rubicon", "appnexus", "pubmatic"); throttlingMessages.forEach(message -> { assertThat(message.getAdUnitCode()).isEqualTo("adunitcodevalue"); diff --git a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/OnnxModelRunnerTest.java b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/OnnxModelRunnerTest.java index 4a18b0a0fc0..51c4b34ff91 100644 --- a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/OnnxModelRunnerTest.java +++ b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/core/OnnxModelRunnerTest.java @@ -29,7 +29,7 @@ public void setUp() throws OrtException, IOException { public void runModelShouldReturnProbabilitiesWhenValidThrottlingInferenceRow() throws OrtException { // given final String[][] throttlingInferenceRow = {{ - "Chrome 59", "rubicon", "adunitcodevalue", "US", "www.leparisien.fr", "PC", "10", "1"}}; + "Chrome 59", "rubicon", "adunitcodevalue", "US", "www.leparisien.fr", "PC", "10", "1"}}; // when final OrtSession.Result actualResult = target.runModel(throttlingInferenceRow); @@ -58,7 +58,7 @@ public void runModelShouldReturnProbabilitiesWhenValidThrottlingInferenceRow() t public void runModelShouldThrowOrtExceptionWhenNonValidThrottlingInferenceRow() { // given final String[][] throttlingInferenceRowWithMissingColumn = {{ - "Chrome 59", "adunitcodevalue", "US", "www.leparisien.fr", "PC", "10", "1"}}; + "Chrome 59", "adunitcodevalue", "US", "www.leparisien.fr", "PC", "10", "1"}}; // when & then assertThatThrownBy(() -> target.runModel(throttlingInferenceRowWithMissingColumn)) diff --git a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/util/TestBidRequestProvider.java b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/util/TestBidRequestProvider.java index 11ca069e447..c66ba26d953 100644 --- a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/util/TestBidRequestProvider.java +++ b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/util/TestBidRequestProvider.java @@ -7,6 +7,7 @@ import com.iab.openrtb.request.BidRequest; import com.iab.openrtb.request.Device; import com.iab.openrtb.request.Format; +import com.iab.openrtb.request.Geo; import com.iab.openrtb.request.Imp; import com.iab.openrtb.request.Site; import org.prebid.server.json.ObjectMapperProvider; @@ -68,12 +69,23 @@ public static ObjectNode givenImpExt() { return extNode; } + public static Device givenDevice(UnaryOperator deviceCustomizer, String countryAlpha3) { + final String userAgent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36" + + " (KHTML, like Gecko) Chrome/59.0.3071.115 Safari/537.36"; + final Geo geo = givenGeoWithCountry(countryAlpha3); + return deviceCustomizer.apply(Device.builder().ua(userAgent).ip("151.101.194.216").geo(geo)).build(); + } + public static Device givenDevice(UnaryOperator deviceCustomizer) { final String userAgent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36" + " (KHTML, like Gecko) Chrome/59.0.3071.115 Safari/537.36"; return deviceCustomizer.apply(Device.builder().ua(userAgent).ip("151.101.194.216")).build(); } + public static Geo givenGeoWithCountry(String countryAlpha3) { + return Geo.builder().country(countryAlpha3).build(); + } + public static Device givenDeviceWithoutUserAgent(UnaryOperator deviceCustomizer) { return deviceCustomizer.apply(Device.builder().ip("151.101.194.216")).build(); } diff --git a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/v1/GreenbidsRealTimeDataProcessedAuctionRequestHookTest.java b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/v1/GreenbidsRealTimeDataProcessedAuctionRequestHookTest.java index 157b70b9474..284741a1b68 100644 --- a/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/v1/GreenbidsRealTimeDataProcessedAuctionRequestHookTest.java +++ b/extra/modules/greenbids-real-time-data/src/test/java/org/prebid/server/hooks/modules/greenbids/real/time/data/v1/GreenbidsRealTimeDataProcessedAuctionRequestHookTest.java @@ -12,10 +12,12 @@ import com.iab.openrtb.request.Device; import com.iab.openrtb.request.Imp; import com.maxmind.geoip2.DatabaseReader; +import com.maxmind.geoip2.exception.GeoIp2Exception; +import com.maxmind.geoip2.model.CountryResponse; +import com.maxmind.geoip2.record.Country; import io.vertx.core.Future; import io.vertx.core.Vertx; import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; import org.mockito.Mock; @@ -23,6 +25,7 @@ import org.prebid.server.analytics.reporter.greenbids.model.ExplorationResult; import org.prebid.server.analytics.reporter.greenbids.model.Ortb2ImpExtResult; import org.prebid.server.auction.model.AuctionContext; +import org.prebid.server.geolocation.CountryCodeMapper; import org.prebid.server.hooks.execution.v1.analytics.ActivityImpl; import org.prebid.server.hooks.execution.v1.analytics.AppliedToImpl; import org.prebid.server.hooks.execution.v1.analytics.ResultImpl; @@ -52,12 +55,9 @@ import org.prebid.server.proto.openrtb.ext.request.ExtRequest; import org.prebid.server.proto.openrtb.ext.request.ExtRequestPrebid; -import java.io.FileOutputStream; import java.io.IOException; -import java.io.InputStream; -import java.net.URL; +import java.net.InetAddress; import java.nio.file.Files; -import java.nio.file.Path; import java.nio.file.Paths; import java.util.Collections; import java.util.List; @@ -66,6 +66,7 @@ import static java.util.function.UnaryOperator.identity; import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.ArgumentMatchers.any; import static org.mockito.Mock.Strictness.LENIENT; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -88,15 +89,30 @@ public class GreenbidsRealTimeDataProcessedAuctionRequestHookTest { @Mock(strictness = LENIENT) private DatabaseReaderFactory databaseReaderFactory; + @Mock(strictness = LENIENT) + private DatabaseReader databaseReader; + + @Mock(strictness = LENIENT) + private CountryResponse countryResponse; + + @Mock(strictness = LENIENT) + private Country country; + @Mock - private DatabaseReader dbReader; + private CountryCodeMapper countryCodeMapper; private GreenbidsRealTimeDataProcessedAuctionRequestHook target; @BeforeEach - public void setUp() throws IOException { + public void setUp() throws IOException, GeoIp2Exception { final Storage storage = StorageOptions.newBuilder() .setProjectId("test_project").build().getService(); + + when(country.getName()).thenReturn("United States"); + when(countryResponse.getCountry()).thenReturn(country); + when(databaseReader.country(any(InetAddress.class))).thenReturn(countryResponse); + when(databaseReaderFactory.getDatabaseReader()).thenReturn(databaseReader); + final FilterService filterService = new FilterService(); final OnnxModelRunnerFactory onnxModelRunnerFactory = new OnnxModelRunnerFactory(); final ThrottlingThresholdsFactory throttlingThresholdsFactory = new ThrottlingThresholdsFactory(); @@ -118,10 +134,11 @@ public void setUp() throws IOException { final OnnxModelRunnerWithThresholds onnxModelRunnerWithThresholds = new OnnxModelRunnerWithThresholds( modelCache, thresholdCache); - when(databaseReaderFactory.getDatabaseReader()).thenReturn(dbReader); + final GreenbidsInferenceDataService greenbidsInferenceDataService = new GreenbidsInferenceDataService( databaseReaderFactory, - TestBidRequestProvider.MAPPER); + TestBidRequestProvider.MAPPER, + countryCodeMapper); final GreenbidsInvocationService greenbidsInvocationService = new GreenbidsInvocationService(); target = new GreenbidsRealTimeDataProcessedAuctionRequestHook( TestBidRequestProvider.MAPPER, @@ -162,7 +179,6 @@ public void callShouldExitEarlyWhenPartnerNotActivatedInBidRequest() { assertThat(result.analyticsTags()).isNull(); } - @Disabled("Broken until dbReader is mocked") @Test public void callShouldNotFilterBiddersAndReturnAnalyticsTagWhenExploration() throws OrtException, IOException { // given @@ -217,7 +233,6 @@ public void callShouldNotFilterBiddersAndReturnAnalyticsTagWhenExploration() thr assertThat(fingerprint).isNotNull(); } - @Disabled("Broken until dbReader is mocked") @Test public void callShouldFilterBiddersBasedOnModelWhenAnyFeatureNotAvailable() throws OrtException, IOException { // given @@ -278,7 +293,6 @@ public void callShouldFilterBiddersBasedOnModelWhenAnyFeatureNotAvailable() thro assertThat(resultBidRequest).usingRecursiveComparison().isEqualTo(expectedBidRequest); } - @Disabled("Broken until dbReader is mocked") @Test public void callShouldFilterBiddersBasedOnModelResults() throws OrtException, IOException { // given diff --git a/src/main/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporter.java b/src/main/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporter.java index 65de3f02ebc..daf51c9d74f 100644 --- a/src/main/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporter.java +++ b/src/main/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporter.java @@ -35,6 +35,7 @@ import org.prebid.server.auction.model.AuctionContext; import org.prebid.server.auction.model.BidRejectionTracker; import org.prebid.server.exception.PreBidException; +import org.prebid.server.hooks.execution.model.ExecutionStatus; import org.prebid.server.hooks.execution.model.GroupExecutionOutcome; import org.prebid.server.hooks.execution.model.HookExecutionContext; import org.prebid.server.hooks.execution.model.HookExecutionOutcome; @@ -205,6 +206,7 @@ private Map extractAnalyticsResultFromAnalyticsTag(Au .map(GroupExecutionOutcome::getHooks) .flatMap(Collection::stream) .filter(hook -> "greenbids-real-time-data".equals(hook.getHookId().getModuleCode())) + .filter(hook -> hook.getStatus() == ExecutionStatus.success) .map(HookExecutionOutcome::getAnalyticsTags) .map(Tags::activities) .flatMap(Collection::stream) diff --git a/src/test/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporterTest.java b/src/test/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporterTest.java index cccdbb4e149..3e5e8678c2c 100644 --- a/src/test/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporterTest.java +++ b/src/test/java/org/prebid/server/analytics/reporter/greenbids/GreenbidsAnalyticsReporterTest.java @@ -39,6 +39,7 @@ import org.prebid.server.auction.model.AuctionContext; import org.prebid.server.auction.model.BidRejectionReason; import org.prebid.server.auction.model.BidRejectionTracker; +import org.prebid.server.hooks.execution.model.ExecutionStatus; import org.prebid.server.hooks.execution.model.GroupExecutionOutcome; import org.prebid.server.hooks.execution.model.HookExecutionContext; import org.prebid.server.hooks.execution.model.HookExecutionOutcome; @@ -656,6 +657,7 @@ private static HookExecutionContext givenHookExecutionContextWithAnalyticsTag() final HookExecutionOutcome hookExecutionOutcome = HookExecutionOutcome.builder() .hookId(HookId.of("greenbids-real-time-data", null)) .analyticsTags(tags) + .status(ExecutionStatus.success) .build(); final GroupExecutionOutcome groupExecutionOutcome = GroupExecutionOutcome.of( From f0c65af9130862df88a4252142690aae100a3bf5 Mon Sep 17 00:00:00 2001 From: bretg Date: Fri, 17 Jan 2025 08:31:03 -0500 Subject: [PATCH 2/3] IX: added comment about endpoint (#3665) --- src/main/resources/bidder-config/ix.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/src/main/resources/bidder-config/ix.yaml b/src/main/resources/bidder-config/ix.yaml index 1a5fa58aa44..725c2494701 100644 --- a/src/main/resources/bidder-config/ix.yaml +++ b/src/main/resources/bidder-config/ix.yaml @@ -1,5 +1,6 @@ adapters: ix: + # Please contact the maintainer to obtain endpoint details endpoint: https:// ortb-version: "2.6" meta-info: From 6e460543c923f8799089f8cbd20a4ed58f608023 Mon Sep 17 00:00:00 2001 From: tradplus <58809719+tradplus@users.noreply.github.com> Date: Fri, 17 Jan 2025 21:32:51 +0800 Subject: [PATCH 3/3] TradPlus: Setting zoneId is optional (#3680) --- src/main/resources/static/bidder-params/tradplus.json | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/main/resources/static/bidder-params/tradplus.json b/src/main/resources/static/bidder-params/tradplus.json index deae1392d1d..345662fe456 100644 --- a/src/main/resources/static/bidder-params/tradplus.json +++ b/src/main/resources/static/bidder-params/tradplus.json @@ -15,7 +15,6 @@ } }, "required": [ - "accountId", - "zoneId" + "accountId" ] }