Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
183 changes: 183 additions & 0 deletions src/main/java/org/prebid/server/bidder/floxis/FloxisBidder.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,183 @@
package org.prebid.server.bidder.floxis;

import com.fasterxml.jackson.core.type.TypeReference;
import com.iab.openrtb.request.BidRequest;
import com.iab.openrtb.request.Imp;
import com.iab.openrtb.response.Bid;
import com.iab.openrtb.response.BidResponse;
import com.iab.openrtb.response.SeatBid;
import io.vertx.core.http.HttpMethod;
import org.apache.commons.collections4.CollectionUtils;
import org.prebid.server.bidder.Bidder;
import org.prebid.server.bidder.model.BidderBid;
import org.prebid.server.bidder.model.BidderCall;
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.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.floxis.ExtImpFloxis;
import org.prebid.server.proto.openrtb.ext.response.BidType;
import org.prebid.server.util.BidderUtil;
import org.prebid.server.util.HttpUtil;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Objects;

public class FloxisBidder implements Bidder<BidRequest> {

private static final TypeReference<ExtPrebid<?, ExtImpFloxis>> FLOXIS_EXT_TYPE_REFERENCE =
new TypeReference<>() {
};

private static final String HOST_MACRO = "{{Host}}";
private static final String SEAT_MACRO = "{{SeatId}}";

// Fixed allowlist mapping the bidder's region param to a Floxis RTB host. Routing is
// never derived from request-supplied hostnames; an unknown or empty region falls back
// to us-e.
private static final Map<String, String> REGION_HOSTS = Map.of(
"us-e", "rtb-us-e.floxis.tech",
"eu", "rtb-eu.floxis.tech",
"apac", "rtb-apac.floxis.tech");

private static final String DEFAULT_REGION = "us-e";

private final String endpointUrl;
private final JacksonMapper mapper;

public FloxisBidder(String endpointUrl, JacksonMapper mapper) {
this.endpointUrl = HttpUtil.validateUrl(Objects.requireNonNull(endpointUrl));
this.mapper = Objects.requireNonNull(mapper);
}

@Override
public Result<List<HttpRequest<BidRequest>>> makeHttpRequests(BidRequest request) {
if (CollectionUtils.isEmpty(request.getImp())) {
return Result.withError(BidderError.badInput("no impressions in the bid request"));
}

final ExtImpFloxis extImp;
try {
extImp = parseImpExt(request.getImp().getFirst());
} catch (PreBidException e) {
return Result.withError(BidderError.badInput(e.getMessage()));
}

// The request body is forwarded unchanged; no caller-owned struct is mutated.
return Result.withValue(HttpRequest.<BidRequest>builder()
.method(HttpMethod.POST)
.uri(resolveUrl(endpointUrl, extImp))
.headers(HttpUtil.headers())
.impIds(BidderUtil.impIds(request))
.payload(request)
.body(mapper.encodeToBytes(request))
.build());
}

private ExtImpFloxis parseImpExt(Imp imp) {
try {
return mapper.mapper().convertValue(imp.getExt(), FLOXIS_EXT_TYPE_REFERENCE).getBidder();
} catch (IllegalArgumentException e) {
throw new PreBidException("invalid imp.ext.bidder for imp %s: %s".formatted(imp.getId(), e.getMessage()));
}
}

private static String resolveHost(String region) {
final String host = region == null ? null : REGION_HOSTS.get(region);
return host != null ? host : REGION_HOSTS.get(DEFAULT_REGION);
}

private static String resolveUrl(String endpoint, ExtImpFloxis extImp) {
return endpoint
.replace(HOST_MACRO, resolveHost(extImp.getRegion()))
.replace(SEAT_MACRO, HttpUtil.encodeUrl(extImp.getSeat()));
}

@Override
public Result<List<BidderBid>> makeBids(BidderCall<BidRequest> httpCall, BidRequest bidRequest) {
final BidResponse bidResponse;
try {
bidResponse = mapper.decodeValue(httpCall.getResponse().getBody(), BidResponse.class);
} catch (DecodeException e) {
return Result.withError(BidderError.badServerResponse(e.getMessage()));
}

if (bidResponse == null || CollectionUtils.isEmpty(bidResponse.getSeatbid())) {
return Result.empty();
}

final List<BidderError> errors = new ArrayList<>();
final List<BidderBid> bids = new ArrayList<>();
for (SeatBid seatBid : bidResponse.getSeatbid()) {
if (seatBid == null || CollectionUtils.isEmpty(seatBid.getBid())) {
continue;
}
for (Bid bid : seatBid.getBid()) {
try {
bids.add(BidderBid.of(bid, getMediaTypeForBid(bidRequest.getImp(), bid), bidResponse.getCur()));
} catch (PreBidException e) {
errors.add(BidderError.badServerResponse(e.getMessage()));
}
}
}

return Result.of(bids, errors);
}

// Resolves the bid's media type. When bid.mtype (OpenRTB 2.6) is set it is treated as
// authoritative. When unset, a single-format imp's media type is used; multi-format imps
// without mtype cannot be disambiguated and surface an error.
private static BidType getMediaTypeForBid(List<Imp> imps, Bid bid) {
final Integer mtype = bid.getMtype();
if (mtype != null && mtype != 0) {
return switch (mtype) {
case 1 -> BidType.banner;
case 2 -> BidType.video;
case 3 -> BidType.audio;
case 4 -> BidType.xNative;
default -> throw new PreBidException(
"unsupported bid.mtype %d for impression %s".formatted(mtype, bid.getImpid()));
};
}

for (Imp imp : imps) {
if (!Objects.equals(imp.getId(), bid.getImpid())) {
continue;
}
int formats = 0;
BidType resolved = null;
if (imp.getBanner() != null) {
formats++;
resolved = BidType.banner;
}
if (imp.getVideo() != null) {
formats++;
resolved = BidType.video;
}
if (imp.getAudio() != null) {
formats++;
resolved = BidType.audio;
}
if (imp.getXNative() != null) {
formats++;
resolved = BidType.xNative;
}
if (formats == 1) {
return resolved;
} else if (formats > 1) {
throw new PreBidException(
"bid for multi-format imp %s requires bid.mtype to disambiguate".formatted(bid.getImpid()));
} else {
throw new PreBidException(
"unable to resolve media type for impression %s".formatted(bid.getImpid()));
}
}

throw new PreBidException("unable to find impression %s for bid".formatted(bid.getImpid()));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package org.prebid.server.proto.openrtb.ext.request.floxis;

import com.fasterxml.jackson.annotation.JsonProperty;
import lombok.Value;

@Value(staticConstructor = "of")
public class ExtImpFloxis {

String seat;

@JsonProperty("region")
String region;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package org.prebid.server.spring.config.bidder;

import org.prebid.server.bidder.BidderDeps;
import org.prebid.server.bidder.floxis.FloxisBidder;
import org.prebid.server.json.JacksonMapper;
import org.prebid.server.spring.config.bidder.model.BidderConfigurationProperties;
import org.prebid.server.spring.config.bidder.util.BidderDepsAssembler;
import org.prebid.server.spring.config.bidder.util.UsersyncerCreator;
import org.prebid.server.spring.env.YamlPropertySourceFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;

import jakarta.validation.constraints.NotBlank;

@Configuration
@PropertySource(value = "classpath:/bidder-config/floxis.yaml",
factory = YamlPropertySourceFactory.class)
public class FloxisConfiguration {

private static final String BIDDER_NAME = "floxis";

@Bean("floxisConfigurationProperties")
@ConfigurationProperties("adapters.floxis")
BidderConfigurationProperties configurationProperties() {
return new BidderConfigurationProperties();
}

@Bean
BidderDeps floxisBidderDeps(BidderConfigurationProperties floxisConfigurationProperties,
@NotBlank @Value("${external-url}") String externalUrl,
JacksonMapper mapper) {

return BidderDepsAssembler.forBidder(BIDDER_NAME)
.withConfig(floxisConfigurationProperties)
.usersyncerCreator(UsersyncerCreator.create(externalUrl))
.bidderCreator(config -> new FloxisBidder(config.getEndpoint(), mapper))
.assemble();
}

}
24 changes: 24 additions & 0 deletions src/main/resources/bidder-config/floxis.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
adapters:
floxis:
endpoint: https://{{Host}}/pbs?seat={{SeatId}}
modifying-vast-xml-allowed: false
meta-info:
maintainer-email: prebid@floxis.tech
app-media-types:
- banner
- video
- native
- audio
site-media-types:
- banner
- video
- native
- audio
supported-vendors:
vendor-id: 0
usersync:
cookie-family-name: floxis
redirect:
url: "https://px-us-e.floxis.tech/sync?gdpr={{gdpr}}&gdpr_consent={{gdpr_consent}}&gpp={{gpp}}&gpp_sid={{gpp_sid}}&us_privacy={{us_privacy}}&dest={{redirect_url}}"
support-cors: false
uid-macro: "${USER_ID}"
20 changes: 20 additions & 0 deletions src/main/resources/static/bidder-params/floxis.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
{
"$schema": "http://json-schema.org/draft-04/schema#",
"title": "Floxis Adapter Params",
"description": "A schema which validates params accepted by the Floxis adapter",
"type": "object",
"additionalProperties": false,
"properties": {
"seat": {
"type": "string",
"minLength": 1,
"description": "The Floxis seat ID this publisher buys through"
},
"region": {
"type": "string",
"enum": ["us-e", "eu", "apac"],
"description": "The Floxis RTB region; defaults to us-e when omitted"
}
},
"required": ["seat"]
}
Loading