Skip to content

Commit

Permalink
New Adapter: Onetag (#1695)
Browse files Browse the repository at this point in the history
  • Loading branch information
onetag-dev authored Feb 11, 2021
1 parent 56dfc60 commit e93a781
Show file tree
Hide file tree
Showing 25 changed files with 1,249 additions and 0 deletions.
152 changes: 152 additions & 0 deletions adapters/onetag/onetag.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
package onetag

import (
"encoding/json"
"fmt"
"net/http"
"text/template"

"github.com/mxmCherry/openrtb"
"github.com/prebid/prebid-server/adapters"
"github.com/prebid/prebid-server/config"
"github.com/prebid/prebid-server/errortypes"
"github.com/prebid/prebid-server/macros"
"github.com/prebid/prebid-server/openrtb_ext"
)

type adapter struct {
endpointTemplate template.Template
}

func Builder(bidderName openrtb_ext.BidderName, config config.Adapter) (adapters.Bidder, error) {
template, err := template.New("endpointTemplate").Parse(config.Endpoint)
if err != nil {
return nil, fmt.Errorf("unable to parse endpoint url template: %v", err)
}

bidder := &adapter{
endpointTemplate: *template,
}

return bidder, nil
}

func (a *adapter) MakeRequests(request *openrtb.BidRequest, requestInfo *adapters.ExtraRequestInfo) ([]*adapters.RequestData, []error) {
pubID := ""
for idx, imp := range request.Imp {
onetagExt, err := getImpressionExt(imp)
if err != nil {
return nil, []error{err}
}
if onetagExt.PubId != "" {
if pubID == "" {
pubID = onetagExt.PubId
} else if pubID != onetagExt.PubId {
return nil, []error{&errortypes.BadInput{
Message: "There must be only one publisher ID",
}}
}
} else {
return nil, []error{&errortypes.BadInput{
Message: "The publisher ID must not be empty",
}}
}
request.Imp[idx].Ext = onetagExt.Ext
}

url, err := a.buildEndpointURL(pubID)
if err != nil {
return nil, []error{err}
}

requestJSON, err := json.Marshal(request)
if err != nil {
return nil, []error{err}
}

requestData := &adapters.RequestData{
Method: "POST",
Uri: url,
Body: requestJSON,
}

return []*adapters.RequestData{requestData}, nil
}

func getImpressionExt(imp openrtb.Imp) (*openrtb_ext.ExtImpOnetag, error) {
var bidderExt adapters.ExtImpBidder
if err := json.Unmarshal(imp.Ext, &bidderExt); err != nil {
return nil, &errortypes.BadInput{
Message: "Bidder extension not provided or can't be unmarshalled",
}
}

var onetagExt openrtb_ext.ExtImpOnetag
if err := json.Unmarshal(bidderExt.Bidder, &onetagExt); err != nil {
return nil, &errortypes.BadInput{
Message: "Error while unmarshaling bidder extension",
}
}

return &onetagExt, nil
}

func (a *adapter) buildEndpointURL(pubID string) (string, error) {
endpointParams := macros.EndpointTemplateParams{PublisherID: pubID}
return macros.ResolveMacros(a.endpointTemplate, endpointParams)
}

func (a *adapter) MakeBids(request *openrtb.BidRequest, requestData *adapters.RequestData, responseData *adapters.ResponseData) (*adapters.BidderResponse, []error) {
if responseData.StatusCode == http.StatusNoContent {
return nil, nil
}

if responseData.StatusCode != http.StatusOK {
err := &errortypes.BadServerResponse{
Message: fmt.Sprintf("Unexpected status code: %d. Run with request.debug = 1 for more info.", responseData.StatusCode),
}
return nil, []error{err}
}

var response openrtb.BidResponse
if err := json.Unmarshal(responseData.Body, &response); err != nil {
return nil, []error{err}
}

bidResponse := adapters.NewBidderResponseWithBidsCapacity(len(request.Imp))
bidResponse.Currency = response.Cur
for _, seatBid := range response.SeatBid {
for _, bid := range seatBid.Bid {
bidMediaType, err := getMediaTypeForBid(request.Imp, bid)
if err != nil {
return nil, []error{err}
}
b := &adapters.TypedBid{
Bid: &bid,
BidType: bidMediaType,
}
bidResponse.Bids = append(bidResponse.Bids, b)
}
}
return bidResponse, nil
}

func getMediaTypeForBid(impressions []openrtb.Imp, bid openrtb.Bid) (openrtb_ext.BidType, error) {
for _, impression := range impressions {
if impression.ID == bid.ImpID {
if impression.Banner != nil {
return openrtb_ext.BidTypeBanner, nil
}
if impression.Video != nil {
return openrtb_ext.BidTypeVideo, nil
}
if impression.Native != nil {
return openrtb_ext.BidTypeNative, nil
}
}
}

return "", &errortypes.BadServerResponse{
Message: fmt.Sprintf("The impression with ID %s is not present into the request", bid.ImpID),
}
}
26 changes: 26 additions & 0 deletions adapters/onetag/onetag_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
package onetag

import (
"testing"

"github.com/prebid/prebid-server/adapters/adapterstest"
"github.com/prebid/prebid-server/config"
"github.com/prebid/prebid-server/openrtb_ext"
"github.com/stretchr/testify/assert"
)

func TestEndpointTemplateMalformed(t *testing.T) {
_, buildErr := Builder(openrtb_ext.BidderOneTag, config.Adapter{
Endpoint: "{{Malformed}}"})

assert.Error(t, buildErr)
}

func TestJsonSamples(t *testing.T) {
bidder, buildErr := Builder(openrtb_ext.BidderOneTag, config.Adapter{
Endpoint: "https://example.com/prebid-server/{{.PublisherID}}"})

assert.NoError(t, buildErr, "Builder returned unexpected error %v", buildErr)

adapterstest.RunJSONBidderTest(t, "onetagtest", bidder)
}
78 changes: 78 additions & 0 deletions adapters/onetag/onetagtest/exemplary/no-bid.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
{
"mockBidRequest": {
"id": "test-request-id",
"device": {
"ip": "123.123.123.123",
"ifa": "zxcjbzxmc-zxcbmz-zxbcz-zxczx"
},
"app": {
"id": "1",
"bundle": "com.wls.testwlsapplication"
},
"imp": [
{
"id": "test-imp-id",
"instl": 1,
"video": {
"placement": 3,
"mimes": [
"video/mp4"
],
"protocols": [
2,
5
],
"w": 1024,
"h": 576
},
"ext": {
"bidder": {
"pubId": "386276e072"
}
}
}
]
},
"httpCalls": [
{
"expectedRequest": {
"uri": "https://example.com/prebid-server/386276e072",
"headers": {},
"body": {
"id": "test-request-id",
"device": {
"ip": "123.123.123.123",
"ifa": "zxcjbzxmc-zxcbmz-zxbcz-zxczx"
},
"app": {
"id": "1",
"bundle": "com.wls.testwlsapplication"
},
"imp": [
{
"id": "test-imp-id",
"instl": 1,
"video": {
"placement": 3,
"mimes": [
"video/mp4"
],
"protocols": [
2,
5
],
"w": 1024,
"h": 576
}
}
]
}
},
"mockResponse": {
"status": 204,
"body": {}
}
}
],
"expectedBidResponses": []
}
Loading

0 comments on commit e93a781

Please sign in to comment.