Skip to content

Commit f785d99

Browse files
sangarbeosulzhenko
authored andcommitted
New Adapter: Seedtag (#3916)
1 parent 3d55952 commit f785d99

File tree

12 files changed

+679
-0
lines changed

12 files changed

+679
-0
lines changed
Lines changed: 152 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
package org.prebid.server.bidder.seedtag;
2+
3+
import com.fasterxml.jackson.core.type.TypeReference;
4+
import com.iab.openrtb.request.BidRequest;
5+
import com.iab.openrtb.request.Imp;
6+
import com.iab.openrtb.response.Bid;
7+
import com.iab.openrtb.response.BidResponse;
8+
import com.iab.openrtb.response.SeatBid;
9+
import org.apache.commons.collections4.CollectionUtils;
10+
import org.prebid.server.bidder.Bidder;
11+
import org.prebid.server.bidder.model.BidderBid;
12+
import org.prebid.server.bidder.model.BidderCall;
13+
import org.prebid.server.bidder.model.BidderError;
14+
import org.prebid.server.bidder.model.HttpRequest;
15+
import org.prebid.server.bidder.model.Price;
16+
import org.prebid.server.bidder.model.Result;
17+
import org.prebid.server.currency.CurrencyConversionService;
18+
import org.prebid.server.exception.PreBidException;
19+
import org.prebid.server.json.DecodeException;
20+
import org.prebid.server.json.JacksonMapper;
21+
import org.prebid.server.proto.openrtb.ext.ExtPrebid;
22+
import org.prebid.server.proto.openrtb.ext.request.seedtag.ExtImpSeedtag;
23+
import org.prebid.server.proto.openrtb.ext.response.BidType;
24+
import org.prebid.server.util.BidderUtil;
25+
import org.prebid.server.util.HttpUtil;
26+
27+
import java.math.BigDecimal;
28+
import java.util.ArrayList;
29+
import java.util.Collection;
30+
import java.util.Collections;
31+
import java.util.List;
32+
import java.util.Objects;
33+
34+
public class SeedtagBidder implements Bidder<BidRequest> {
35+
36+
private static final TypeReference<ExtPrebid<?, ExtImpSeedtag>> SEEDTAG_EXT_TYPE_REFERENCE =
37+
new TypeReference<>() {
38+
};
39+
private static final String BIDDER_CURRENCY = "USD";
40+
41+
private final String endpointUrl;
42+
private final JacksonMapper mapper;
43+
private final CurrencyConversionService currencyConversionService;
44+
45+
public SeedtagBidder(String endpointUrl,
46+
CurrencyConversionService currencyConversionService,
47+
JacksonMapper mapper) {
48+
49+
this.endpointUrl = HttpUtil.validateUrl(Objects.requireNonNull(endpointUrl));
50+
this.currencyConversionService = Objects.requireNonNull(currencyConversionService);
51+
this.mapper = Objects.requireNonNull(mapper);
52+
}
53+
54+
@Override
55+
public Result<List<HttpRequest<BidRequest>>> makeHttpRequests(BidRequest request) {
56+
final List<Imp> modifiedImps = new ArrayList<>();
57+
final List<BidderError> errors = new ArrayList<>();
58+
59+
for (Imp imp : request.getImp()) {
60+
try {
61+
final Price bidFloorPrice = resolveBidFloor(imp, request);
62+
63+
modifiedImps.add(modifyImp(imp, bidFloorPrice));
64+
} catch (PreBidException e) {
65+
errors.add(BidderError.badInput(e.getMessage()));
66+
}
67+
}
68+
69+
if (modifiedImps.size() < 1) {
70+
return Result.withErrors(errors);
71+
}
72+
73+
final BidRequest modifiedBidRequest = request.toBuilder()
74+
.imp(modifiedImps)
75+
.build();
76+
return Result.of(
77+
Collections.singletonList(BidderUtil.defaultRequest(modifiedBidRequest, endpointUrl, mapper)),
78+
errors);
79+
}
80+
81+
private static Imp modifyImp(Imp imp, Price bidFloorPrice) {
82+
return imp.toBuilder()
83+
.bidfloorcur(bidFloorPrice.getCurrency())
84+
.bidfloor(bidFloorPrice.getValue())
85+
.build();
86+
}
87+
88+
private Price resolveBidFloor(Imp imp, BidRequest bidRequest) {
89+
final Price initialBidFloorPrice = Price.of(imp.getBidfloorcur(), imp.getBidfloor());
90+
return BidderUtil.shouldConvertBidFloor(initialBidFloorPrice, BIDDER_CURRENCY)
91+
? convertBidFloor(initialBidFloorPrice, imp.getId(), bidRequest)
92+
: initialBidFloorPrice;
93+
}
94+
95+
private Price convertBidFloor(Price bidFloorPrice, String impId, BidRequest bidRequest) {
96+
final BigDecimal convertedPrice = currencyConversionService.convertCurrency(
97+
bidFloorPrice.getValue(),
98+
bidRequest,
99+
bidFloorPrice.getCurrency(),
100+
BIDDER_CURRENCY);
101+
102+
return Price.of(BIDDER_CURRENCY, convertedPrice);
103+
}
104+
105+
@Override
106+
public final Result<List<BidderBid>> makeBids(BidderCall<BidRequest> httpCall, BidRequest bidRequest) {
107+
try {
108+
final BidResponse bidResponse = mapper.decodeValue(httpCall.getResponse().getBody(), BidResponse.class);
109+
final List<BidderError> errors = new ArrayList<>();
110+
final List<BidderBid> bidderBids = extractBids(bidResponse, errors);
111+
return Result.of(bidderBids, errors);
112+
} catch (DecodeException | PreBidException e) {
113+
return Result.withError(BidderError.badServerResponse(e.getMessage()));
114+
}
115+
}
116+
117+
private static List<BidderBid> extractBids(BidResponse bidResponse, List<BidderError> errors) {
118+
if (bidResponse == null || CollectionUtils.isEmpty(bidResponse.getSeatbid())) {
119+
return Collections.emptyList();
120+
}
121+
122+
return bidResponse.getSeatbid().stream()
123+
.filter(Objects::nonNull)
124+
.map(SeatBid::getBid)
125+
.filter(Objects::nonNull)
126+
.flatMap(Collection::stream)
127+
.filter(Objects::nonNull)
128+
.map(bid -> makeBidderBid(bid, errors))
129+
.filter(Objects::nonNull)
130+
.toList();
131+
}
132+
133+
private static BidderBid makeBidderBid(Bid bid, List<BidderError> errors) {
134+
final BidType bidType;
135+
try {
136+
bidType = getBidType(bid);
137+
} catch (PreBidException e) {
138+
errors.add(BidderError.badServerResponse(e.getMessage()));
139+
return null;
140+
}
141+
142+
return BidderBid.of(bid, bidType, BIDDER_CURRENCY);
143+
}
144+
145+
private static BidType getBidType(Bid bid) {
146+
return switch (bid.getMtype()) {
147+
case 1 -> BidType.banner;
148+
case 2 -> BidType.video;
149+
default -> throw new PreBidException("Invalid bid.mtype for bid.id: '%s'".formatted(bid.getId()));
150+
};
151+
}
152+
}
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
package org.prebid.server.proto.openrtb.ext.request.seedtag;
2+
3+
import com.fasterxml.jackson.annotation.JsonProperty;
4+
import lombok.Value;
5+
6+
@Value(staticConstructor = "of")
7+
public class ExtImpSeedtag {
8+
9+
@JsonProperty("adUnitId")
10+
String adUnitId;
11+
12+
}
Lines changed: 46 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
package org.prebid.server.spring.config.bidder;
2+
3+
import org.prebid.server.bidder.BidderDeps;
4+
import org.prebid.server.bidder.seedtag.SeedtagBidder;
5+
import org.prebid.server.currency.CurrencyConversionService;
6+
import org.prebid.server.json.JacksonMapper;
7+
import org.prebid.server.spring.config.bidder.model.BidderConfigurationProperties;
8+
import org.prebid.server.spring.config.bidder.util.BidderDepsAssembler;
9+
import org.prebid.server.spring.config.bidder.util.UsersyncerCreator;
10+
import org.prebid.server.spring.env.YamlPropertySourceFactory;
11+
import org.springframework.beans.factory.annotation.Value;
12+
import org.springframework.boot.context.properties.ConfigurationProperties;
13+
import org.springframework.context.annotation.Bean;
14+
import org.springframework.context.annotation.Configuration;
15+
import org.springframework.context.annotation.PropertySource;
16+
17+
import jakarta.validation.constraints.NotBlank;
18+
19+
@Configuration
20+
@PropertySource(value = "classpath:/bidder-config/seedtag.yaml", factory = YamlPropertySourceFactory.class)
21+
public class SeedtagConfiguration {
22+
23+
private static final String BIDDER_NAME = "seedtag";
24+
25+
@Bean("seedtagConfigurationProperties")
26+
@ConfigurationProperties("adapters.seedtag")
27+
BidderConfigurationProperties configurationProperties() {
28+
return new BidderConfigurationProperties();
29+
}
30+
31+
@Bean
32+
BidderDeps seedtagBidderDeps(BidderConfigurationProperties seedtagConfigurationProperties,
33+
@NotBlank @Value("${external-url}") String externalUrl,
34+
CurrencyConversionService currencyConversionService,
35+
JacksonMapper mapper) {
36+
37+
return BidderDepsAssembler.forBidder(BIDDER_NAME)
38+
.withConfig(seedtagConfigurationProperties)
39+
.usersyncerCreator(UsersyncerCreator.create(externalUrl))
40+
.bidderCreator(config -> new SeedtagBidder(
41+
config.getEndpoint(),
42+
currencyConversionService,
43+
mapper))
44+
.assemble();
45+
}
46+
}
Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
adapters:
2+
seedtag:
3+
endpoint: "https://s.seedtag.com/c/openrtb?partner=prebidserver"
4+
endpoint-compression: gzip
5+
meta-info:
6+
maintainer-email: [email protected]
7+
site-media-types:
8+
- banner
9+
- video
10+
supported-vendors:
11+
vendor-id: 157
12+
usersync:
13+
cookie-family-name: seedtag
14+
iframe:
15+
url: https://s.seedtag.com/cs/cookiesync/prebid?gdpr={{gdpr}}&gdpr_consent={{gdpr_consent}}&usp_consent={{us_privacy}}&redirect={{redirect_url}}
16+
support-cors: false
17+
uid-macro: '$UID'
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
{
2+
"$schema": "http://json-schema.org/draft-04/schema#",
3+
"title": "Seedtag Adapter Params",
4+
"description": "A schema which validates params accepted by the Seedtag adapter",
5+
"type": "object",
6+
"properties": {
7+
"adUnitId": {
8+
"type": "string",
9+
"description": "Ad Unit ID",
10+
"minLength": 1
11+
}
12+
},
13+
"required": [
14+
"adUnitId"
15+
]
16+
}

0 commit comments

Comments
 (0)