forked from prebid/prebid-server-java
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathStartioBidder.java
More file actions
149 lines (125 loc) · 5.74 KB
/
Copy pathStartioBidder.java
File metadata and controls
149 lines (125 loc) · 5.74 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
package org.prebid.server.bidder.startio;
import com.fasterxml.jackson.core.JsonPointer;
import com.fasterxml.jackson.databind.JsonNode;
import com.iab.openrtb.request.App;
import com.iab.openrtb.request.BidRequest;
import com.iab.openrtb.request.Imp;
import com.iab.openrtb.request.Site;
import com.iab.openrtb.response.Bid;
import com.iab.openrtb.response.BidResponse;
import com.iab.openrtb.response.SeatBid;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
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.json.DecodeException;
import org.prebid.server.json.JacksonMapper;
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.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
public class StartioBidder implements Bidder<BidRequest> {
private static final JsonPointer BID_TYPE_POINTER = JsonPointer.valueOf("/prebid/type");
private static final String BID_CURRENCY = "USD";
private final String endpointUrl;
private final JacksonMapper mapper;
public StartioBidder(String endpointUrl, JacksonMapper mapper) {
this.endpointUrl = HttpUtil.validateUrl(Objects.requireNonNull(endpointUrl));
this.mapper = Objects.requireNonNull(mapper);
}
@Override
public final Result<List<HttpRequest<BidRequest>>> makeHttpRequests(BidRequest bidRequest) {
if (hasNoAppOrSiteId(bidRequest)) {
return Result.withError(BidderError.badInput(
"Bidder requires either app.id or site.id to be specified."));
}
if (isSupportedCurrency(bidRequest.getCur())) {
return Result.withError(BidderError.badInput("Unsupported currency, bidder only accepts USD."));
}
final List<HttpRequest<BidRequest>> requests = new ArrayList<>();
final List<BidderError> errors = new ArrayList<>();
final List<Imp> imps = bidRequest.getImp();
for (int i = 0; i < imps.size(); i++) {
final Imp imp = imps.get(i);
if (imp.getBanner() == null && imp.getVideo() == null && imp.getXNative() == null) {
errors.add(BidderError.badInput(
"imp[%d]: Unsupported media type, bidder does not support audio.".formatted(i)));
continue;
}
final Imp modifiedImp = imp.getAudio() != null
? imp.toBuilder().audio(null).build()
: imp;
requests.add(BidderUtil.defaultRequest(
bidRequest.toBuilder().imp(Collections.singletonList(modifiedImp)).build(),
endpointUrl, mapper));
}
return Result.of(requests, errors);
}
private static boolean hasNoAppOrSiteId(BidRequest bidRequest) {
final App app = bidRequest.getApp();
final Site site = bidRequest.getSite();
return (app == null || StringUtils.isEmpty(app.getId()))
&& (site == null || StringUtils.isEmpty(site.getId()));
}
private static boolean isSupportedCurrency(List<String> currencies) {
return CollectionUtils.isNotEmpty(currencies) && !currencies.contains(BID_CURRENCY);
}
@Override
public final Result<List<BidderBid>> makeBids(BidderCall<BidRequest> httpCall, BidRequest bidRequest) {
final BidResponse response;
try {
response = mapper.decodeValue(httpCall.getResponse().getBody(), BidResponse.class);
} catch (DecodeException e) {
return Result.withError(BidderError.badServerResponse(e.getMessage()));
}
final List<BidderError> errors = new ArrayList<>();
return Result.of(extractBids(response, errors), errors);
}
private static List<BidderBid> extractBids(BidResponse bidResponse, List<BidderError> errors) {
if (bidResponse == null || CollectionUtils.isEmpty(bidResponse.getSeatbid())) {
return Collections.emptyList();
}
return bidsFromResponse(bidResponse, errors);
}
private static List<BidderBid> bidsFromResponse(BidResponse bidResponse, List<BidderError> errors) {
return bidResponse.getSeatbid().stream()
.filter(Objects::nonNull)
.map(SeatBid::getBid)
.filter(Objects::nonNull)
.flatMap(Collection::stream)
.filter(Objects::nonNull)
.map(bid -> constructBidderBid(bid, bidResponse.getCur(), errors))
.filter(Objects::nonNull)
.toList();
}
private static BidderBid constructBidderBid(Bid bid, String currency, List<BidderError> errors) {
final BidType type = getBidType(bid);
if (type != null) {
return BidderBid.of(bid, type, currency);
}
errors.add(BidderError.badServerResponse(
"Failed to parse bid media type for impression %s.".formatted(bid.getImpid())));
return null;
}
private static BidType getBidType(Bid bid) {
final JsonNode ext = bid.getExt();
final JsonNode bidType = ext != null ? ext.at(BID_TYPE_POINTER) : null;
if (bidType == null || !bidType.isTextual()) {
return null;
}
return switch (bidType.textValue()) {
case "banner" -> BidType.banner;
case "video" -> BidType.video;
case "native" -> BidType.xNative;
default -> null;
};
}
}