forked from prebid/prebid-server
-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
UOE-11477: Cherry-Pick Displayio bidder changes (#966)
- Loading branch information
1 parent
0cef376
commit d68a113
Showing
20 changed files
with
1,441 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,188 @@ | ||
package displayio | ||
|
||
import ( | ||
"encoding/json" | ||
"errors" | ||
"fmt" | ||
"net/http" | ||
"text/template" | ||
|
||
"github.com/prebid/openrtb/v20/openrtb2" | ||
"github.com/prebid/prebid-server/v2/adapters" | ||
"github.com/prebid/prebid-server/v2/config" | ||
"github.com/prebid/prebid-server/v2/errortypes" | ||
"github.com/prebid/prebid-server/v2/macros" | ||
"github.com/prebid/prebid-server/v2/openrtb_ext" | ||
) | ||
|
||
type adapter struct { | ||
endpoint *template.Template | ||
} | ||
|
||
type reqDioExt struct { | ||
UserSession string `json:"userSession,omitempty"` | ||
PlacementId string `json:"placementId"` | ||
InventoryId string `json:"inventoryId"` | ||
} | ||
|
||
func (adapter *adapter) MakeRequests(request *openrtb2.BidRequest, requestInfo *adapters.ExtraRequestInfo) ([]*adapters.RequestData, []error) { | ||
headers := http.Header{} | ||
headers.Add("Content-Type", "application/json;charset=utf-8") | ||
headers.Add("Accept", "application/json") | ||
headers.Add("x-openrtb-version", "2.5") | ||
|
||
result := make([]*adapters.RequestData, 0, len(request.Imp)) | ||
errs := make([]error, 0, len(request.Imp)) | ||
|
||
for _, impression := range request.Imp { | ||
var requestExt map[string]interface{} | ||
|
||
if impression.BidFloorCur == "" || impression.BidFloor == 0 { | ||
impression.BidFloorCur = "USD" | ||
} else if impression.BidFloorCur != "USD" { | ||
convertedValue, err := requestInfo.ConvertCurrency(impression.BidFloor, impression.BidFloorCur, "USD") | ||
|
||
if err != nil { | ||
errs = append(errs, err) | ||
continue | ||
} | ||
|
||
impression.BidFloor = convertedValue | ||
impression.BidFloorCur = "USD" | ||
} | ||
|
||
if len(impression.Ext) == 0 { | ||
errs = append(errs, errors.New("impression extensions required")) | ||
continue | ||
} | ||
|
||
var bidderExt adapters.ExtImpBidder | ||
err := json.Unmarshal(impression.Ext, &bidderExt) | ||
|
||
if err != nil { | ||
errs = append(errs, err) | ||
continue | ||
} | ||
|
||
var impressionExt openrtb_ext.ExtImpDisplayio | ||
err = json.Unmarshal(bidderExt.Bidder, &impressionExt) | ||
if err != nil { | ||
errs = append(errs, err) | ||
continue | ||
} | ||
|
||
dioExt := reqDioExt{PlacementId: impressionExt.PlacementId, InventoryId: impressionExt.InventoryId} | ||
|
||
requestCopy := *request | ||
|
||
err = json.Unmarshal(requestCopy.Ext, &requestExt) | ||
if err != nil { | ||
requestExt = make(map[string]interface{}) | ||
} | ||
|
||
requestExt["displayio"] = dioExt | ||
|
||
requestCopy.Ext, err = json.Marshal(requestExt) | ||
if err != nil { | ||
errs = append(errs, err) | ||
continue | ||
} | ||
|
||
requestCopy.Imp = []openrtb2.Imp{impression} | ||
body, err := json.Marshal(requestCopy) | ||
if err != nil { | ||
errs = append(errs, err) | ||
continue | ||
} | ||
|
||
url, err := adapter.buildEndpointURL(&impressionExt) | ||
if err != nil { | ||
return nil, []error{err} | ||
} | ||
|
||
result = append(result, &adapters.RequestData{ | ||
Method: "POST", | ||
Uri: url, | ||
Body: body, | ||
Headers: headers, | ||
ImpIDs: openrtb_ext.GetImpIDs(requestCopy.Imp), | ||
}) | ||
} | ||
|
||
if len(result) == 0 { | ||
return nil, errs | ||
} | ||
return result, errs | ||
} | ||
|
||
// MakeBids translates Displayio bid response to prebid-server specific format | ||
func (adapter *adapter) MakeBids(internalRequest *openrtb2.BidRequest, _ *adapters.RequestData, responseData *adapters.ResponseData) (*adapters.BidderResponse, []error) { | ||
|
||
if adapters.IsResponseStatusCodeNoContent(responseData) { | ||
return nil, nil | ||
} | ||
|
||
if err := adapters.CheckResponseStatusCodeForErrors(responseData); err != nil { | ||
return nil, []error{err} | ||
} | ||
|
||
var bidResp openrtb2.BidResponse | ||
|
||
if err := json.Unmarshal(responseData.Body, &bidResp); err != nil { | ||
msg := fmt.Sprintf("Bad server response: %d", err) | ||
return nil, []error{&errortypes.BadServerResponse{Message: msg}} | ||
} | ||
|
||
if len(bidResp.SeatBid) != 1 { | ||
msg := fmt.Sprintf("Invalid SeatBids count: %d", len(bidResp.SeatBid)) | ||
return nil, []error{&errortypes.BadServerResponse{Message: msg}} | ||
} | ||
|
||
var errs []error | ||
bidResponse := adapters.NewBidderResponse() | ||
|
||
for _, sb := range bidResp.SeatBid { | ||
for i := range sb.Bid { | ||
bidType, err := getBidMediaTypeFromMtype(&sb.Bid[i]) | ||
if err != nil { | ||
errs = append(errs, err) | ||
} else { | ||
b := &adapters.TypedBid{ | ||
Bid: &sb.Bid[i], | ||
BidType: bidType, | ||
} | ||
bidResponse.Bids = append(bidResponse.Bids, b) | ||
} | ||
} | ||
} | ||
|
||
return bidResponse, errs | ||
} | ||
|
||
func Builder(_ openrtb_ext.BidderName, config config.Adapter, _ config.Server) (adapters.Bidder, error) { | ||
endpoint, err := template.New("endpointTemplate").Parse(config.Endpoint) | ||
if err != nil { | ||
return nil, fmt.Errorf("unable to parse endpoint url template: %v", err) | ||
} | ||
|
||
bidder := &adapter{ | ||
endpoint: endpoint, | ||
} | ||
return bidder, nil | ||
} | ||
|
||
func getBidMediaTypeFromMtype(bid *openrtb2.Bid) (openrtb_ext.BidType, error) { | ||
switch bid.MType { | ||
case openrtb2.MarkupBanner: | ||
return openrtb_ext.BidTypeBanner, nil | ||
case openrtb2.MarkupVideo: | ||
return openrtb_ext.BidTypeVideo, nil | ||
default: | ||
return "", fmt.Errorf("unexpected media type for bid: %s", bid.ImpID) | ||
} | ||
} | ||
|
||
func (adapter *adapter) buildEndpointURL(params *openrtb_ext.ExtImpDisplayio) (string, error) { | ||
endpointParams := macros.EndpointTemplateParams{PublisherID: params.PublisherId} | ||
return macros.ResolveMacros(adapter.endpoint, endpointParams) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
package displayio | ||
|
||
import ( | ||
"testing" | ||
|
||
"github.com/prebid/prebid-server/v2/adapters/adapterstest" | ||
"github.com/prebid/prebid-server/v2/config" | ||
"github.com/prebid/prebid-server/v2/openrtb_ext" | ||
) | ||
|
||
func TestJsonSamples(t *testing.T) { | ||
bidder, buildErr := Builder(openrtb_ext.BidderDisplayio, | ||
config.Adapter{Endpoint: "https://adapter.endpoint/?macro={{.PublisherID}}"}, | ||
config.Server{ExternalUrl: "https://server.endpoint/"}, | ||
) | ||
|
||
if buildErr != nil { | ||
t.Fatalf("Builder returned unexpected error %v", buildErr) | ||
} | ||
|
||
adapterstest.RunJSONBidderTest(t, "displayiotest", bidder) | ||
} |
147 changes: 147 additions & 0 deletions
147
adapters/displayio/displayiotest/exemplary/multi-format.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
{ | ||
"mockBidRequest": { | ||
"id": "requestId10111011101110111011", | ||
"app": { | ||
"id": "1011" | ||
}, | ||
"imp": [ | ||
{ | ||
"id": "impId10111011101110111011", | ||
"tagid": "1011", | ||
"ext": { | ||
"bidder": { | ||
"placementId": "1011", | ||
"publisherId": "101", | ||
"inventoryId": "1011" | ||
} | ||
}, | ||
"banner": { | ||
"format": [ | ||
{ | ||
"w": 300, | ||
"h": 250 | ||
} | ||
] | ||
}, | ||
"video": { | ||
"mimes": [ | ||
"video/mp4" | ||
], | ||
"protocols": [ | ||
2, | ||
5 | ||
], | ||
"w": 640, | ||
"h": 480 | ||
}, | ||
"bidfloor": 0.5, | ||
"bidfloorcur": "USD" | ||
} | ||
] | ||
}, | ||
"httpCalls": [ | ||
{ | ||
"expectedRequest": { | ||
"uri": "https://adapter.endpoint/?macro=101", | ||
"body": { | ||
"id": "requestId10111011101110111011", | ||
"app": { | ||
"id": "1011" | ||
}, | ||
"imp": [ | ||
{ | ||
"id": "impId10111011101110111011", | ||
"tagid": "1011", | ||
"ext": { | ||
"bidder": { | ||
"placementId": "1011", | ||
"publisherId": "101", | ||
"inventoryId": "1011" | ||
} | ||
}, | ||
"banner": { | ||
"format": [ | ||
{ | ||
"w": 300, | ||
"h": 250 | ||
} | ||
] | ||
}, | ||
"video": { | ||
"mimes": [ | ||
"video/mp4" | ||
], | ||
"protocols": [ | ||
2, | ||
5 | ||
], | ||
"w": 640, | ||
"h": 480 | ||
}, | ||
"bidfloor": 0.5, | ||
"bidfloorcur": "USD" | ||
} | ||
], | ||
"ext": { | ||
"displayio": { | ||
"placementId": "1011", | ||
"inventoryId": "1011" | ||
} | ||
} | ||
}, | ||
"impIDs": [ | ||
"impId10111011101110111011" | ||
] | ||
}, | ||
"mockResponse": { | ||
"status": 200, | ||
"body": { | ||
"id": "test-request-id", | ||
"bidid": "5778926625248726496", | ||
"seatbid": [ | ||
{ | ||
"seat": "seat1", | ||
"bid": [ | ||
{ | ||
"id": "12345", | ||
"impid": "impId10111011101110111011", | ||
"price": 0.01, | ||
"adm": "<html/>", | ||
"adomain": [ | ||
"domain.test" | ||
], | ||
"w": 300, | ||
"h": 250, | ||
"mtype": 1 | ||
} | ||
] | ||
} | ||
], | ||
"cur": "USD" | ||
} | ||
} | ||
} | ||
], | ||
"expectedBidResponses": [ | ||
{ | ||
"currency": "USD", | ||
"bids": [ | ||
{ | ||
"bid": { | ||
"id": "12345", | ||
"impid": "impId10111011101110111011", | ||
"price": 0.01, | ||
"adm": "<html/>", | ||
"adomain": [ | ||
"domain.test" | ||
], | ||
"w": 300, | ||
"h": 250, | ||
"mtype": 1 | ||
}, | ||
"type": "banner" | ||
} | ||
] | ||
} | ||
] | ||
} |
Oops, something went wrong.