From d5f53a43f3502f01da2f3c272f242e9e7c31e28f Mon Sep 17 00:00:00 2001 From: Anton Babak <76536883+AntoxaAntoxic@users.noreply.github.com> Date: Fri, 31 Oct 2025 13:05:15 +0100 Subject: [PATCH 1/2] Teqblaze Alias: AppStockSSP (#4265) --- .../resources/bidder-config/teqblaze.yaml | 18 ++++++ .../org/prebid/server/it/AppStockSspTest.java | 37 ++++++++++++ .../test-appstockssp-bid-request.json | 56 +++++++++++++++++++ .../test-appstockssp-bid-response.json | 21 +++++++ .../test-auction-appstockssp-request.json | 23 ++++++++ .../test-auction-appstockssp-response.json | 38 +++++++++++++ .../server/it/test-application.properties | 2 + 7 files changed, 195 insertions(+) create mode 100644 src/test/java/org/prebid/server/it/AppStockSspTest.java create mode 100644 src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-request.json create mode 100644 src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-response.json create mode 100644 src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-request.json create mode 100644 src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-response.json diff --git a/src/main/resources/bidder-config/teqblaze.yaml b/src/main/resources/bidder-config/teqblaze.yaml index 1386c4d6469..13eb09d7f9b 100644 --- a/src/main/resources/bidder-config/teqblaze.yaml +++ b/src/main/resources/bidder-config/teqblaze.yaml @@ -19,6 +19,24 @@ adapters: url: https://usync.rocketlab.ai/pbserver?gdpr={{gdpr}}&consent={{gdpr_consent}}&us_privacy={{us_privacy}}&gpp={{gpp}}&gpp_sid={{gpp_sid}}&redirect={{redirect_url}} support-cors: false uid-macro: '[UID]' + appStockSSP: + enabled: false + # We have the following regional endpoint domains: 'lb' - for US_EAST, 'ortb-eu' - for EU, 'ortb-apac' - for APAC + # Please deploy this config in each of your datacenters with the appropriate regional subdomain + endpoint: https://#{REGION}#.al-ad.com/pserver + meta-info: + maintainer-email: sdksupport@app-stock.com + usersync: + enabled: true + cookie-family-name: appStockSSP + redirect: + url: https://csync.al-ad.com/pbserver?gdpr={{gdpr}}&gdpr_consent={{gdpr_consent}}&ccpa={{us_privacy}}&gpp={{gpp}}&gpp_sid={{gpp_sid}}&redir={{redirect_url}} + support-cors: false + uid-macro: '[UID]' + iframe: + url: "https://csync.al-ad.com/pbserverIframe?gdpr={{gdpr}}&gdpr_consent={{gdpr_consent}}&ccpa={{us_privacy}}&gpp={{gpp}}&gpp_sid={{gpp_sid}}&pbserverUrl={{redirect_url}}" + support-cors: false + uid-macro: '[UID]' meta-info: maintainer-email: github@teqblaze.com app-media-types: diff --git a/src/test/java/org/prebid/server/it/AppStockSspTest.java b/src/test/java/org/prebid/server/it/AppStockSspTest.java new file mode 100644 index 00000000000..a564320d2ce --- /dev/null +++ b/src/test/java/org/prebid/server/it/AppStockSspTest.java @@ -0,0 +1,37 @@ +package org.prebid.server.it; + +import io.restassured.response.Response; +import org.json.JSONException; +import org.junit.jupiter.api.Test; +import org.prebid.server.model.Endpoint; + +import java.io.IOException; + +import static com.github.tomakehurst.wiremock.client.WireMock.aResponse; +import static com.github.tomakehurst.wiremock.client.WireMock.equalToJson; +import static com.github.tomakehurst.wiremock.client.WireMock.post; +import static com.github.tomakehurst.wiremock.client.WireMock.urlPathEqualTo; +import static java.util.Collections.singletonList; + +public class AppStockSspTest extends IntegrationTest { + + @Test + public void openrtb2AuctionShouldRespondWithBidsFromAppStockSsp() throws IOException, JSONException { + // given + WIRE_MOCK_RULE.stubFor(post(urlPathEqualTo("/appstockssp-exchange")) + .withRequestBody(equalToJson( + jsonFrom("openrtb2/appstockssp/test-appstockssp-bid-request.json"))) + .willReturn(aResponse().withBody( + jsonFrom("openrtb2/appstockssp/test-appstockssp-bid-response.json")))); + + // when + final Response response = responseFor("openrtb2/appstockssp/test-auction-appstockssp-request.json", + Endpoint.openrtb2_auction); + + // then + assertJsonEquals( + "openrtb2/appstockssp/test-auction-appstockssp-response.json", + response, + singletonList("appStockSSP")); + } +} diff --git a/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-request.json b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-request.json new file mode 100644 index 00000000000..dcf904ba566 --- /dev/null +++ b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-request.json @@ -0,0 +1,56 @@ +{ + "id": "request_id", + "imp": [ + { + "id": "imp_id", + "banner": { + "w": 300, + "h": 250 + }, + "secure": 1, + "ext": { + "bidder": { + "type": "publisher", + "placementId": "testPlacementId" + } + } + } + ], + "site": { + "domain": "www.example.com", + "page": "http://www.example.com", + "publisher": { + "domain": "example.com" + }, + "ext": { + "amp": 0 + } + }, + "device": { + "ua": "userAgent", + "ip": "193.168.244.1" + }, + "at": 1, + "tmax": "${json-unit.any-number}", + "cur": [ + "USD" + ], + "source": { + "tid": "${json-unit.any-string}" + }, + "regs": { + "ext": { + "gdpr": 0 + } + }, + "ext": { + "prebid": { + "server": { + "externalurl": "http://localhost:8080", + "gvlid": 1, + "datacenter": "local", + "endpoint": "/openrtb2/auction" + } + } + } +} diff --git a/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-response.json b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-response.json new file mode 100644 index 00000000000..180173549d8 --- /dev/null +++ b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-appstockssp-bid-response.json @@ -0,0 +1,21 @@ +{ + "id": "request_id", + "seatbid": [ + { + "bid": [ + { + "id": "bid_id", + "impid": "imp_id", + "price": 3.33, + "crid": "creativeId", + "mtype": 1, + "ext": { + "prebid": { + "type": "banner" + } + } + } + ] + } + ] +} diff --git a/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-request.json b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-request.json new file mode 100644 index 00000000000..27c6c80e321 --- /dev/null +++ b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-request.json @@ -0,0 +1,23 @@ +{ + "id": "request_id", + "imp": [ + { + "id": "imp_id", + "banner": { + "w": 300, + "h": 250 + }, + "ext": { + "appStockSSP": { + "placementId": "testPlacementId" + } + } + } + ], + "tmax": 5000, + "regs": { + "ext": { + "gdpr": 0 + } + } +} diff --git a/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-response.json b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-response.json new file mode 100644 index 00000000000..a1e8683f099 --- /dev/null +++ b/src/test/resources/org/prebid/server/it/openrtb2/appstockssp/test-auction-appstockssp-response.json @@ -0,0 +1,38 @@ +{ + "id": "request_id", + "seatbid": [ + { + "bid": [ + { + "id": "bid_id", + "impid": "imp_id", + "exp": 300, + "price": 3.33, + "crid": "creativeId", + "mtype": 1, + "ext": { + "origbidcpm": 3.33, + "prebid": { + "type": "banner", + "meta": { + "adaptercode": "appStockSSP" + } + } + } + } + ], + "seat": "appStockSSP", + "group": 0 + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "appStockSSP": "{{ appStockSSP.response_time_ms }}" + }, + "prebid": { + "auctiontimestamp": 0 + }, + "tmaxrequest": 5000 + } +} diff --git a/src/test/resources/org/prebid/server/it/test-application.properties b/src/test/resources/org/prebid/server/it/test-application.properties index e847998f90f..c2e136cf572 100644 --- a/src/test/resources/org/prebid/server/it/test-application.properties +++ b/src/test/resources/org/prebid/server/it/test-application.properties @@ -575,6 +575,8 @@ adapters.teqblaze.aliases.pinkLion.enabled=true adapters.teqblaze.aliases.pinkLion.endpoint=http://localhost:8090/pinkLion-exchange adapters.teqblaze.aliases.rocketlab.enabled=true adapters.teqblaze.aliases.rocketlab.endpoint=http://localhost:8090/rocketlab-exchange +adapters.teqblaze.aliases.appStockSSP.enabled=true +adapters.teqblaze.aliases.appStockSSP.endpoint=http://localhost:8090/appstockssp-exchange adapters.theadx.enabled=true adapters.theadx.endpoint=http://localhost:8090/theadx-exchange adapters.tradplus.enabled=true From 3f6fa1d43b60b2c521d656ff6383708d6604e2c9 Mon Sep 17 00:00:00 2001 From: Victor Malitskyi Date: Fri, 31 Oct 2025 13:29:47 +0100 Subject: [PATCH 2/2] Unruly: Include GPID in request payload (#4259) --- .../server/bidder/unruly/UnrulyBidder.java | 14 +++++----- .../bidder/unruly/proto/UnrulyExtPrebid.java | 12 +++++++++ .../bidder/unruly/UnrulyBidderTest.java | 27 +++++++++++++++++-- 3 files changed, 45 insertions(+), 8 deletions(-) create mode 100644 src/main/java/org/prebid/server/bidder/unruly/proto/UnrulyExtPrebid.java diff --git a/src/main/java/org/prebid/server/bidder/unruly/UnrulyBidder.java b/src/main/java/org/prebid/server/bidder/unruly/UnrulyBidder.java index ee0471a937f..feac7f16ab0 100644 --- a/src/main/java/org/prebid/server/bidder/unruly/UnrulyBidder.java +++ b/src/main/java/org/prebid/server/bidder/unruly/UnrulyBidder.java @@ -15,11 +15,10 @@ import org.prebid.server.bidder.model.BidderError; import org.prebid.server.bidder.model.HttpRequest; import org.prebid.server.bidder.model.Result; +import org.prebid.server.bidder.unruly.proto.UnrulyExtPrebid; import org.prebid.server.exception.PreBidException; import org.prebid.server.json.DecodeException; import org.prebid.server.json.JacksonMapper; -import org.prebid.server.proto.openrtb.ext.ExtPrebid; -import org.prebid.server.proto.openrtb.ext.request.unruly.ExtImpUnruly; import org.prebid.server.proto.openrtb.ext.response.BidType; import org.prebid.server.proto.openrtb.ext.response.ExtBidPrebid; import org.prebid.server.proto.openrtb.ext.response.ExtBidPrebidVideo; @@ -34,7 +33,7 @@ public class UnrulyBidder implements Bidder { - private static final TypeReference> UNRULY_EXT_TYPE_REFERENCE = + private static final TypeReference UNRULY_EXT_TYPE_REFERENCE = new TypeReference<>() { }; @@ -58,14 +57,17 @@ public Result>> makeHttpRequests(BidRequest request private Imp modifyImp(Imp imp) { + final UnrulyExtPrebid unrulyExtPrebid = parseImpExt(imp); return imp.toBuilder() - .ext(mapper.mapper().valueToTree(ExtPrebid.of(null, parseImpExt(imp)))) + .ext(mapper.mapper().valueToTree(UnrulyExtPrebid.of( + unrulyExtPrebid.getBidder(), + unrulyExtPrebid.getGpid()))) .build(); } - private ExtImpUnruly parseImpExt(Imp imp) { + private UnrulyExtPrebid parseImpExt(Imp imp) { try { - return mapper.mapper().convertValue(imp.getExt(), UNRULY_EXT_TYPE_REFERENCE).getBidder(); + return mapper.mapper().convertValue(imp.getExt(), UNRULY_EXT_TYPE_REFERENCE); } catch (IllegalArgumentException e) { throw new PreBidException(e.getMessage()); } diff --git a/src/main/java/org/prebid/server/bidder/unruly/proto/UnrulyExtPrebid.java b/src/main/java/org/prebid/server/bidder/unruly/proto/UnrulyExtPrebid.java new file mode 100644 index 00000000000..f34ec91894f --- /dev/null +++ b/src/main/java/org/prebid/server/bidder/unruly/proto/UnrulyExtPrebid.java @@ -0,0 +1,12 @@ +package org.prebid.server.bidder.unruly.proto; + +import lombok.Value; +import org.prebid.server.proto.openrtb.ext.request.unruly.ExtImpUnruly; + +@Value(staticConstructor = "of") +public class UnrulyExtPrebid { + + ExtImpUnruly bidder; + + String gpid; +} diff --git a/src/test/java/org/prebid/server/bidder/unruly/UnrulyBidderTest.java b/src/test/java/org/prebid/server/bidder/unruly/UnrulyBidderTest.java index 82cee73461a..c059eb1f291 100644 --- a/src/test/java/org/prebid/server/bidder/unruly/UnrulyBidderTest.java +++ b/src/test/java/org/prebid/server/bidder/unruly/UnrulyBidderTest.java @@ -1,6 +1,7 @@ package org.prebid.server.bidder.unruly; import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.node.IntNode; import com.fasterxml.jackson.databind.node.ObjectNode; import com.fasterxml.jackson.databind.node.TextNode; @@ -19,7 +20,8 @@ import org.prebid.server.bidder.model.HttpRequest; import org.prebid.server.bidder.model.HttpResponse; import org.prebid.server.bidder.model.Result; -import org.prebid.server.proto.openrtb.ext.ExtPrebid; +import org.prebid.server.bidder.unruly.proto.UnrulyExtPrebid; +import org.prebid.server.proto.openrtb.ext.request.unruly.ExtImpUnruly; import java.util.List; import java.util.Map; @@ -65,6 +67,27 @@ public void makeHttpRequestsShouldReturnOneRequestPerImpWithExpectedHeaders() { tuple("Accept", "application/json")); } + @Test + public void makeHttpRequestsShouldPassThroughGPID() { + // given + final BidRequest bidRequest = BidRequest.builder() + .imp(singletonList(givenImp(identity()))) + .build(); + + // when + final Result>> result = target.makeHttpRequests(bidRequest); + + // then + assertThat(result.getValue()).hasSize(1) + .extracting(HttpRequest::getPayload) + .flatExtracting(BidRequest::getImp) + .extracting(Imp::getExt) + .extracting(ext -> ext.get("gpid")) + .extracting(JsonNode::asText) + .first() + .isEqualTo("gpid"); + } + @Test public void makeBidsShouldReturnErrorIfResponseBodyCouldNotBeParsed() { // given @@ -288,7 +311,7 @@ private static BidRequest givenBidRequest(Function impCustomizer) { final ObjectNode impExt = mapper.valueToTree( - ExtPrebid.of(null, mapper.createObjectNode().set("siteId", IntNode.valueOf(123)))); + UnrulyExtPrebid.of(ExtImpUnruly.of(123), "gpid")); return impCustomizer.apply( Imp.builder()