github.com/prebid/prebid-server@v0.275.0/adapters/taboola/taboola.go (about) 1 package taboola 2 3 import ( 4 "encoding/json" 5 "fmt" 6 "net/http" 7 "strconv" 8 "strings" 9 "text/template" 10 11 "github.com/prebid/openrtb/v19/adcom1" 12 "github.com/prebid/openrtb/v19/openrtb2" 13 14 "github.com/prebid/prebid-server/adapters" 15 "github.com/prebid/prebid-server/config" 16 "github.com/prebid/prebid-server/errortypes" 17 "github.com/prebid/prebid-server/macros" 18 "github.com/prebid/prebid-server/openrtb_ext" 19 ) 20 21 type adapter struct { 22 endpoint *template.Template 23 gvlID string 24 } 25 26 // Builder builds a new instance of Taboola adapter for the given bidder with the given config. 27 func Builder(bidderName openrtb_ext.BidderName, config config.Adapter, server config.Server) (adapters.Bidder, error) { 28 endpointTemplate, err := template.New("endpointTemplate").Parse(config.Endpoint) 29 if err != nil { 30 return nil, fmt.Errorf("unable to parse endpoint url template: %v", err) 31 } 32 33 gvlID := "" 34 if server.GvlID > 0 { 35 gvlID = strconv.Itoa(server.GvlID) 36 } 37 38 bidder := &adapter{ 39 endpoint: endpointTemplate, 40 gvlID: gvlID, 41 } 42 return bidder, nil 43 } 44 45 func (a *adapter) MakeRequests(request *openrtb2.BidRequest, requestInfo *adapters.ExtraRequestInfo) ([]*adapters.RequestData, []error) { 46 47 var requests []*adapters.RequestData 48 49 taboolaRequests, errs := createTaboolaRequests(request) 50 if len(errs) > 0 { 51 return nil, errs 52 } 53 54 for _, taboolaRequest := range taboolaRequests { 55 if len(taboolaRequest.Imp) > 0 { 56 request, err := a.buildRequest(taboolaRequest) 57 if err != nil { 58 return nil, []error{fmt.Errorf("unable to build request %v", err)} 59 } 60 requests = append(requests, request) 61 } 62 } 63 64 return requests, errs 65 } 66 67 func (a *adapter) MakeBids(request *openrtb2.BidRequest, requestData *adapters.RequestData, responseData *adapters.ResponseData) (*adapters.BidderResponse, []error) { 68 var errs []error 69 70 if responseData.StatusCode == http.StatusNoContent { 71 return nil, nil 72 } 73 74 if responseData.StatusCode == http.StatusBadRequest { 75 err := &errortypes.BadInput{ 76 Message: "Unexpected status code: 400. Bad request from publisher. Run with request.debug = 1 for more info.", 77 } 78 return nil, []error{err} 79 } 80 81 if responseData.StatusCode != http.StatusOK { 82 err := &errortypes.BadServerResponse{ 83 Message: fmt.Sprintf("Unexpected status code: %d. Run with request.debug = 1 for more info.", responseData.StatusCode), 84 } 85 return nil, []error{err} 86 } 87 88 var response openrtb2.BidResponse 89 if err := json.Unmarshal(responseData.Body, &response); err != nil { 90 return nil, []error{err} 91 } 92 93 bidResponse := adapters.NewBidderResponseWithBidsCapacity(len(request.Imp)) 94 bidResponse.Currency = response.Cur 95 for _, seatBid := range response.SeatBid { 96 for i := range seatBid.Bid { 97 bidType, err := getMediaType(seatBid.Bid[i].ImpID, request.Imp) 98 resolveMacros(&seatBid.Bid[i]) 99 if err != nil { 100 errs = append(errs, err) 101 continue 102 } 103 b := &adapters.TypedBid{ 104 Bid: &seatBid.Bid[i], 105 BidType: bidType, 106 } 107 bidResponse.Bids = append(bidResponse.Bids, b) 108 } 109 } 110 return bidResponse, errs 111 } 112 113 func (a *adapter) buildRequest(request *openrtb2.BidRequest) (*adapters.RequestData, error) { 114 requestJSON, err := json.Marshal(request) 115 if err != nil { 116 return nil, err 117 } 118 119 const ( 120 NATIVE_ENDPOINT_PREFIX = "native" 121 DISPLAY_ENDPOINT_PREFIX = "display" 122 ) 123 124 //set MediaType based on first imp 125 var mediaType string 126 if request.Imp[0].Banner != nil { 127 mediaType = DISPLAY_ENDPOINT_PREFIX 128 } else if request.Imp[0].Native != nil { 129 mediaType = NATIVE_ENDPOINT_PREFIX 130 } else { 131 return nil, fmt.Errorf("unsupported media type for imp: %v", request.Imp[0]) 132 } 133 134 url, err := a.buildEndpointURL(request.Site.ID, mediaType) 135 if err != nil { 136 return nil, err 137 } 138 139 requestData := &adapters.RequestData{ 140 Method: "POST", 141 Uri: url, 142 Body: requestJSON, 143 } 144 145 return requestData, nil 146 } 147 148 // Builds endpoint url based on adapter-specific pub settings from imp.ext 149 func (a *adapter) buildEndpointURL(publisherId string, mediaType string) (string, error) { 150 endpointParams := macros.EndpointTemplateParams{PublisherID: publisherId, MediaType: mediaType, GvlID: a.gvlID} 151 resolvedUrl, err := macros.ResolveMacros(a.endpoint, endpointParams) 152 if err != nil { 153 return "", err 154 } 155 return resolvedUrl, nil 156 } 157 158 func createTaboolaRequests(request *openrtb2.BidRequest) (taboolaRequests []*openrtb2.BidRequest, errors []error) { 159 modifiedRequest := *request 160 var nativeImp []openrtb2.Imp 161 var bannerImp []openrtb2.Imp 162 var errs []error 163 164 var taboolaExt openrtb_ext.ImpExtTaboola 165 for i := 0; i < len(modifiedRequest.Imp); i++ { 166 imp := modifiedRequest.Imp[i] 167 168 var bidderExt adapters.ExtImpBidder 169 if err := json.Unmarshal(imp.Ext, &bidderExt); err != nil { 170 errs = append(errs, err) 171 continue 172 } 173 if err := json.Unmarshal(bidderExt.Bidder, &taboolaExt); err != nil { 174 errs = append(errs, err) 175 continue 176 } 177 178 tagId := taboolaExt.TagID 179 if len(taboolaExt.TagID) < 1 { 180 tagId = taboolaExt.TagId 181 } 182 183 imp.TagID = tagId 184 modifiedRequest.Imp[i] = imp 185 186 if taboolaExt.BidFloor != 0 { 187 imp.BidFloor = taboolaExt.BidFloor 188 modifiedRequest.Imp[i] = imp 189 } 190 191 if modifiedRequest.Imp[i].Banner != nil { 192 if taboolaExt.Position != nil { 193 bannerCopy := *imp.Banner 194 bannerCopy.Pos = adcom1.PlacementPosition(*taboolaExt.Position).Ptr() 195 imp.Banner = &bannerCopy 196 modifiedRequest.Imp[i] = imp 197 } 198 bannerImp = append(bannerImp, modifiedRequest.Imp[i]) 199 } else if modifiedRequest.Imp[i].Native != nil { 200 nativeImp = append(nativeImp, modifiedRequest.Imp[i]) 201 } 202 203 } 204 205 publisher := &openrtb2.Publisher{ 206 ID: taboolaExt.PublisherId, 207 } 208 209 if modifiedRequest.Site == nil { 210 newSite := &openrtb2.Site{ 211 ID: taboolaExt.PublisherId, 212 Name: taboolaExt.PublisherId, 213 Domain: evaluateDomain(taboolaExt.PublisherDomain, request), 214 Publisher: publisher, 215 } 216 modifiedRequest.Site = newSite 217 } else { 218 modifiedSite := *modifiedRequest.Site 219 modifiedSite.Publisher = publisher 220 modifiedSite.ID = taboolaExt.PublisherId 221 modifiedSite.Name = taboolaExt.PublisherId 222 modifiedSite.Domain = evaluateDomain(taboolaExt.PublisherDomain, request) 223 modifiedRequest.Site = &modifiedSite 224 } 225 226 if taboolaExt.BCat != nil { 227 modifiedRequest.BCat = taboolaExt.BCat 228 } 229 230 if taboolaExt.BAdv != nil { 231 modifiedRequest.BAdv = taboolaExt.BAdv 232 } 233 234 if taboolaExt.PageType != "" { 235 requestExt, requestExtErr := makeRequestExt(taboolaExt.PageType) 236 if requestExtErr == nil { 237 modifiedRequest.Ext = requestExt 238 } else { 239 errs = append(errs, requestExtErr) 240 } 241 } 242 243 taboolaRequests = append(taboolaRequests, overrideBidRequestImp(&modifiedRequest, nativeImp)) 244 taboolaRequests = append(taboolaRequests, overrideBidRequestImp(&modifiedRequest, bannerImp)) 245 246 return taboolaRequests, errs 247 } 248 249 func makeRequestExt(pageType string) (json.RawMessage, error) { 250 requestExt := &RequestExt{ 251 PageType: pageType, 252 } 253 254 requestExtJson, err := json.Marshal(requestExt) 255 if err != nil { 256 return nil, fmt.Errorf("could not marshal %s, err: %s", requestExt, err) 257 } 258 return requestExtJson, nil 259 260 } 261 262 func getMediaType(impID string, imps []openrtb2.Imp) (openrtb_ext.BidType, error) { 263 for _, imp := range imps { 264 if imp.ID == impID { 265 if imp.Banner != nil { 266 return openrtb_ext.BidTypeBanner, nil 267 } else if imp.Native != nil { 268 return openrtb_ext.BidTypeNative, nil 269 } 270 } 271 } 272 273 return "", &errortypes.BadInput{ 274 Message: fmt.Sprintf("Failed to find banner/native impression \"%s\" ", impID), 275 } 276 } 277 278 func evaluateDomain(publisherDomain string, request *openrtb2.BidRequest) (result string) { 279 if publisherDomain != "" { 280 return publisherDomain 281 } 282 if request.Site != nil { 283 return request.Site.Domain 284 } 285 return "" 286 } 287 288 func overrideBidRequestImp(originBidRequest *openrtb2.BidRequest, imp []openrtb2.Imp) (bidRequest *openrtb2.BidRequest) { 289 bidRequestResult := *originBidRequest 290 bidRequestResult.Imp = imp 291 return &bidRequestResult 292 } 293 294 func resolveMacros(bid *openrtb2.Bid) { 295 if bid != nil { 296 price := strconv.FormatFloat(bid.Price, 'f', -1, 64) 297 bid.NURL = strings.Replace(bid.NURL, "${AUCTION_PRICE}", price, -1) 298 bid.AdM = strings.Replace(bid.AdM, "${AUCTION_PRICE}", price, -1) 299 } 300 }