diff --git a/adapters/brave/brave.go b/adapters/brave/brave.go new file mode 100755 index 00000000000..627b6547ed9 --- /dev/null +++ b/adapters/brave/brave.go @@ -0,0 +1,157 @@ +package brave + +import ( + "encoding/json" + "fmt" + "net/http" + "text/template" + + "github.com/prebid/openrtb/v17/openrtb2" + "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 { + endpoint *template.Template +} + +// Builder builds a new instance of the Brave adapter for the given bidder with the given config. +func Builder(bidderName openrtb_ext.BidderName, config config.Adapter, server config.Server) (adapters.Bidder, error) { + uri, 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: uri, + } + return bidder, nil +} + +func (a *adapter) MakeRequests(request *openrtb2.BidRequest, requestInfo *adapters.ExtraRequestInfo) ([]*adapters.RequestData, []error) { + var braveExt *openrtb_ext.ExtImpBrave + var err error + + braveExt, err = a.getImpressionExt(&request.Imp[0]) + if err != nil { + return nil, []error{err} + } + + request.Imp[0].Ext = nil + + reqJSON, err := json.Marshal(request) + if err != nil { + return nil, []error{err} + } + + headers := http.Header{} + headers.Add("Content-Type", "application/json;charset=utf-8") + headers.Add("Accept", "application/json") + + url, err := a.buildEndpointURL(braveExt) + if err != nil { + return nil, []error{err} + } + + return []*adapters.RequestData{{ + Method: http.MethodPost, + Body: reqJSON, + Uri: url, + Headers: headers, + }}, nil +} + +func (a *adapter) getImpressionExt(imp *openrtb2.Imp) (*openrtb_ext.ExtImpBrave, error) { + var bidderExt adapters.ExtImpBidder + if err := json.Unmarshal(imp.Ext, &bidderExt); err != nil { + return nil, &errortypes.BadInput{ + Message: "ext.bidder not provided", + } + } + var braveExt openrtb_ext.ExtImpBrave + if err := json.Unmarshal(bidderExt.Bidder, &braveExt); err != nil { + return nil, &errortypes.BadInput{ + Message: "ext.bidder not provided", + } + } + return &braveExt, nil +} + +func (a *adapter) buildEndpointURL(params *openrtb_ext.ExtImpBrave) (string, error) { + endpointParams := macros.EndpointTemplateParams{PublisherID: params.PlacementID} + return macros.ResolveMacros(a.endpoint, endpointParams) +} + +func (a *adapter) MakeBids( + receivedRequest *openrtb2.BidRequest, + bidderRequest *adapters.RequestData, + bidderResponse *adapters.ResponseData, +) ( + *adapters.BidderResponse, + []error, +) { + + if bidderResponse.StatusCode == http.StatusNoContent { + return nil, []error{&errortypes.BadInput{Message: "No bid"}} + } + + if bidderResponse.StatusCode == http.StatusBadRequest { + return nil, []error{&errortypes.BadInput{ + Message: fmt.Sprintf("Unexpected status code: %d. Run with request.debug = 1 for more info", bidderResponse.StatusCode), + }} + } + + if bidderResponse.StatusCode == http.StatusServiceUnavailable { + return nil, []error{&errortypes.BadInput{ + Message: fmt.Sprintf("Service Unavailable. Status Code: [ %d ] ", bidderResponse.StatusCode), + }} + } + + if bidderResponse.StatusCode != http.StatusOK { + return nil, []error{&errortypes.BadServerResponse{ + Message: fmt.Sprintf("Something went wrong, please contact your Account Manager. Status Code: [ %d ] ", bidderResponse.StatusCode), + }} + } + + var bidResponse openrtb2.BidResponse + if err := json.Unmarshal(bidderResponse.Body, &bidResponse); err != nil { + return nil, []error{&errortypes.BadServerResponse{ + Message: "Bad Server Response", + }} + } + + if len(bidResponse.SeatBid) == 0 { + return nil, []error{&errortypes.BadServerResponse{ + Message: "Empty SeatBid array", + }} + } + + bidResponseFinal := adapters.NewBidderResponseWithBidsCapacity(len(bidResponse.SeatBid[0].Bid)) + sb := bidResponse.SeatBid[0] + + for _, bid := range sb.Bid { + bidResponseFinal.Bids = append(bidResponseFinal.Bids, &adapters.TypedBid{ + Bid: &bid, + BidType: getMediaTypeForImp(bid.ImpID, receivedRequest.Imp), + }) + } + return bidResponseFinal, nil +} + +func getMediaTypeForImp(impId string, imps []openrtb2.Imp) openrtb_ext.BidType { + mediaType := openrtb_ext.BidTypeBanner + for _, imp := range imps { + if imp.ID == impId { + if imp.Video != nil { + mediaType = openrtb_ext.BidTypeVideo + } else if imp.Native != nil { + mediaType = openrtb_ext.BidTypeNative + } + return mediaType + } + } + return mediaType +} diff --git a/adapters/brave/brave_test.go b/adapters/brave/brave_test.go new file mode 100644 index 00000000000..e8d0a474be9 --- /dev/null +++ b/adapters/brave/brave_test.go @@ -0,0 +1,30 @@ +package brave + +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 TestJsonSamples(t *testing.T) { + bidder, buildErr := Builder(openrtb_ext.BidderBrave, config.Adapter{ + Endpoint: "http://point.braveglobal.tv/?t=3&partner={{.PublisherID}}"}, + config.Server{ExternalUrl: "http://hosturl.com", GvlID: 1, DataCenter: "2"}) + + if buildErr != nil { + t.Fatalf("Builder returned unexpected error %v", buildErr) + } + + adapterstest.RunJSONBidderTest(t, "bravetest", bidder) +} + +func TestEndpointTemplateMalformed(t *testing.T) { + _, buildErr := Builder(openrtb_ext.BidderBrave, config.Adapter{ + Endpoint: "{{Malformed}}"}, + config.Server{ExternalUrl: "http://hosturl.com", GvlID: 1, DataCenter: "2"}) + + assert.Error(t, buildErr) +} diff --git a/adapters/brave/bravetest/exemplary/banner-app.json b/adapters/brave/bravetest/exemplary/banner-app.json new file mode 100644 index 00000000000..71a30978299 --- /dev/null +++ b/adapters/brave/bravetest/exemplary/banner-app.json @@ -0,0 +1,140 @@ +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "banner": { + "w":320, + "h":50 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "banner": { + "w":320, + "h":50 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w":320, + "h":50 + } + ], + "type": "banner", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w":320, + "h":50 + }, + "type": "banner" + }] + } + ] + } diff --git a/adapters/brave/bravetest/exemplary/banner-web.json b/adapters/brave/bravetest/exemplary/banner-web.json new file mode 100644 index 00000000000..0dd6dde0bd8 --- /dev/null +++ b/adapters/brave/bravetest/exemplary/banner-web.json @@ -0,0 +1,138 @@ +{ + "mockBidRequest": { + "id": "request-id", + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "banner": { + "w":320, + "h":50 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "banner": { + "w":320, + "h":50 + }, + "tagid": "tid" + } + ], + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w":320, + "h":50 + } + ], + "type": "banner", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w":320, + "h":50 + }, + "type": "banner" + }] + } + ] +} diff --git a/adapters/brave/bravetest/exemplary/native-app.json b/adapters/brave/bravetest/exemplary/native-app.json new file mode 100644 index 00000000000..05ac5aa8482 --- /dev/null +++ b/adapters/brave/bravetest/exemplary/native-app.json @@ -0,0 +1,137 @@ +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "native": { + "ver":"1.1", + "request":"{\"adunit\":2,\"assets\":[{\"id\":3,\"img\":{\"h\":120,\"hmin\":0,\"type\":3,\"w\":180,\"wmin\":0},\"required\":1},{\"id\":0,\"required\":1,\"title\":{\"len\":25}},{\"data\":{\"len\":25,\"type\":1},\"id\":4,\"required\":1},{\"data\":{\"len\":140,\"type\":2},\"id\":6,\"required\":1}],\"context\":1,\"layout\":1,\"contextsubtype\":11,\"plcmtcnt\":1,\"plcmttype\":2,\"ver\":\"1.1\",\"ext\":{\"banner\":{\"w\":320,\"h\":50}}}" + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "native": { + "ver":"1.1", + "request":"{\"adunit\":2,\"assets\":[{\"id\":3,\"img\":{\"h\":120,\"hmin\":0,\"type\":3,\"w\":180,\"wmin\":0},\"required\":1},{\"id\":0,\"required\":1,\"title\":{\"len\":25}},{\"data\":{\"len\":25,\"type\":1},\"id\":4,\"required\":1},{\"data\":{\"len\":140,\"type\":2},\"id\":6,\"required\":1}],\"context\":1,\"layout\":1,\"contextsubtype\":11,\"plcmtcnt\":1,\"plcmttype\":2,\"ver\":\"1.1\",\"ext\":{\"banner\":{\"w\":320,\"h\":50}}}" + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100" + } + ], + "type": "native", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100" + }, + "type": "native" + }] + } + ] + } diff --git a/adapters/brave/bravetest/exemplary/native-web.json b/adapters/brave/bravetest/exemplary/native-web.json new file mode 100644 index 00000000000..01126e46762 --- /dev/null +++ b/adapters/brave/bravetest/exemplary/native-web.json @@ -0,0 +1,135 @@ +{ + "mockBidRequest": { + "id": "request-id", + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "native": { + "ver":"1.1", + "request":"{\"adunit\":2,\"assets\":[{\"id\":3,\"img\":{\"h\":120,\"hmin\":0,\"type\":3,\"w\":180,\"wmin\":0},\"required\":1},{\"id\":0,\"required\":1,\"title\":{\"len\":25}},{\"data\":{\"len\":25,\"type\":1},\"id\":4,\"required\":1},{\"data\":{\"len\":140,\"type\":2},\"id\":6,\"required\":1}],\"context\":1,\"layout\":1,\"contextsubtype\":11,\"plcmtcnt\":1,\"plcmttype\":2,\"ver\":\"1.1\",\"ext\":{\"banner\":{\"w\":320,\"h\":50}}}" + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "native": { + "ver":"1.1", + "request":"{\"adunit\":2,\"assets\":[{\"id\":3,\"img\":{\"h\":120,\"hmin\":0,\"type\":3,\"w\":180,\"wmin\":0},\"required\":1},{\"id\":0,\"required\":1,\"title\":{\"len\":25}},{\"data\":{\"len\":25,\"type\":1},\"id\":4,\"required\":1},{\"data\":{\"len\":140,\"type\":2},\"id\":6,\"required\":1}],\"context\":1,\"layout\":1,\"contextsubtype\":11,\"plcmtcnt\":1,\"plcmttype\":2,\"ver\":\"1.1\",\"ext\":{\"banner\":{\"w\":320,\"h\":50}}}" + }, + "tagid": "tid" + } + ], + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100" + } + ], + "type": "native", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100" + }, + "type": "native" + }] + } + ] + } diff --git a/adapters/brave/bravetest/exemplary/video-app.json b/adapters/brave/bravetest/exemplary/video-app.json new file mode 100644 index 00000000000..7f41510921c --- /dev/null +++ b/adapters/brave/bravetest/exemplary/video-app.json @@ -0,0 +1,150 @@ +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w": 1280, + "h": 720 + } + ], + "type": "video", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w": 1280, + "h": 720 + }, + "type": "video" + }] + } + ] +} diff --git a/adapters/brave/bravetest/exemplary/video-web.json b/adapters/brave/bravetest/exemplary/video-web.json new file mode 100644 index 00000000000..5da8ff3587c --- /dev/null +++ b/adapters/brave/bravetest/exemplary/video-web.json @@ -0,0 +1,148 @@ +{ + "mockBidRequest": { + "id": "request-id", + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + { + "bid": [ + { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w": 1280, + "h": 720 + } + ], + "type": "video", + "seat": "brave" + } + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "expectedBidResponses": [ + { + "bids": [{ + "bid": { + "id": "123456789", + "impid": "impression-id", + "price": 2, + "adm": "adm code", + "adomain": [ + "testdomain.com" + ], + "crid": "100", + "w": 1280, + "h": 720 + }, + "type": "video" + }] + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/empty-seatbid-array.json b/adapters/brave/bravetest/supplemental/empty-seatbid-array.json new file mode 100644 index 00000000000..c407c8223cb --- /dev/null +++ b/adapters/brave/bravetest/supplemental/empty-seatbid-array.json @@ -0,0 +1,121 @@ +{ + "mockBidRequest": { + "id": "request-id", + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + "httpCalls": [ + { + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "site": { + "page": "test.com/page", + "domain": "test.com", + "cat": [ + "IAB9-1" + ], + "publisher": { + "id": "123456789" + } + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": { + "id": "resp-id", + "seatbid": [ + ], + "cur": "USD", + "ext": { + "responsetimemillis": { + "brave": 120 + }, + "tmaxrequest": 1000 + } + } + } + } + ], + "mockResponse": { + "status": 200, + "body": "invalid response" + }, + "expectedMakeBidsErrors": [ + { + "value": "Empty SeatBid array", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/invalid-brave-ext-object.json b/adapters/brave/bravetest/supplemental/invalid-brave-ext-object.json new file mode 100644 index 00000000000..899834e668f --- /dev/null +++ b/adapters/brave/bravetest/supplemental/invalid-brave-ext-object.json @@ -0,0 +1,29 @@ +{ + "expectedMakeRequestsErrors": [ + { + "value": "ext.bidder not provided", + "comparison": "literal" + } + ], + "mockBidRequest": { + "id": "request-id", + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": ["video/mp4"], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": "invalid data" + } + ], + "site": { + "page": "test.com" + } + }, + "httpCalls": [] +} diff --git a/adapters/brave/bravetest/supplemental/invalid-response.json b/adapters/brave/bravetest/supplemental/invalid-response.json new file mode 100644 index 00000000000..972c2b30d1d --- /dev/null +++ b/adapters/brave/bravetest/supplemental/invalid-response.json @@ -0,0 +1,97 @@ +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "banner": { + "w":320, + "h":50 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + + "httpCalls": [{ + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "banner": { + "w":320, + "h":50 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 200, + "body": "invalid response" + } + }], + "expectedMakeBidsErrors": [ + { + "value": "Bad Server Response", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/status-code-bad-request.json b/adapters/brave/bravetest/supplemental/status-code-bad-request.json new file mode 100644 index 00000000000..e80273a7d91 --- /dev/null +++ b/adapters/brave/bravetest/supplemental/status-code-bad-request.json @@ -0,0 +1,108 @@ + +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + + "httpCalls": [{ + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 400 + } + }], + "expectedBidResponses": [], + "expectedMakeBidsErrors": [ + { + "value": "Unexpected status code: 400. Run with request.debug = 1 for more info", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/status-code-no-content.json b/adapters/brave/bravetest/supplemental/status-code-no-content.json new file mode 100644 index 00000000000..eb369bd8de6 --- /dev/null +++ b/adapters/brave/bravetest/supplemental/status-code-no-content.json @@ -0,0 +1,108 @@ + +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + + "httpCalls": [{ + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 204 + } + }], + "expectedBidResponses": [], + "expectedMakeBidsErrors": [ + { + "value": "No bid", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/status-code-other-error.json b/adapters/brave/bravetest/supplemental/status-code-other-error.json new file mode 100644 index 00000000000..bd68ee85916 --- /dev/null +++ b/adapters/brave/bravetest/supplemental/status-code-other-error.json @@ -0,0 +1,108 @@ + +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + + "httpCalls": [{ + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 306 + } + }], + "expectedBidResponses": [], + "expectedMakeBidsErrors": [ + { + "value": "Something went wrong, please contact your Account Manager. Status Code: [ 306 ] ", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/bravetest/supplemental/status-code-service-unavailable.json b/adapters/brave/bravetest/supplemental/status-code-service-unavailable.json new file mode 100644 index 00000000000..f66347d8934 --- /dev/null +++ b/adapters/brave/bravetest/supplemental/status-code-service-unavailable.json @@ -0,0 +1,107 @@ +{ + "mockBidRequest": { + "id": "request-id", + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "tmax": 1000, + "user": { + "id": "some-user" + }, + "imp": [ + { + "id": "impression-id", + "tagid": "tid", + "video": { + "mimes": [ + "video/mp4" + ], + "w": 640, + "h": 480, + "minduration": 30, + "maxduration": 150 + }, + "ext": { + "bidder": { + "placementId": "f897beb0daba0253d8e59a098eef9311" + } + } + } + ] + }, + + "httpCalls": [{ + "expectedRequest": { + "headers": { + "Content-Type": [ + "application/json;charset=utf-8" + ], + "Accept": [ + "application/json" + ] + }, + "uri": "http://point.braveglobal.tv/?t=3&partner=f897beb0daba0253d8e59a098eef9311", + "body": { + "id": "request-id", + "device": { + "ua": "useragent", + "ip": "100.100.100.100", + "language": "en" + }, + "imp": [ + { + "id": "impression-id", + "video": { + "mimes": [ + "video/mp4" + ], + "minduration": 30, + "maxduration": 150, + "w": 640, + "h": 480 + }, + "tagid": "tid" + } + ], + "app": { + "publisher": { + "id": "123456789" + }, + "cat": [ + "IAB9-1" + ], + "bundle": "com.app.test", + "name": "Test App", + "id": "12345678" + }, + "user": { + "id": "some-user" + }, + "tmax": 1000 + } + }, + "mockResponse": { + "status": 503 + } + }], + "expectedBidResponses": [], + "expectedMakeBidsErrors": [ + { + "value": "Service Unavailable. Status Code: [ 503 ] ", + "comparison": "literal" + } + ] +} diff --git a/adapters/brave/params_test.go b/adapters/brave/params_test.go new file mode 100644 index 00000000000..1120a13c2f9 --- /dev/null +++ b/adapters/brave/params_test.go @@ -0,0 +1,51 @@ +package brave + +import ( + "encoding/json" + "testing" + + "github.com/prebid/prebid-server/openrtb_ext" +) + +func TestValidParams(t *testing.T) { + validator, err := openrtb_ext.NewBidderParamsValidator("../../static/bidder-params") + if err != nil { + t.Fatalf("Failed to fetch the json-schemas. %v", err) + } + + for _, validParam := range validParams { + if err := validator.Validate(openrtb_ext.BidderBrave, json.RawMessage(validParam)); err != nil { + t.Errorf("Schema rejected Brave params: %s", validParam) + } + } +} + +func TestInvalidParams(t *testing.T) { + validator, err := openrtb_ext.NewBidderParamsValidator("../../static/bidder-params") + if err != nil { + t.Fatalf("Failed to fetch the json-schemas. %v", err) + } + + for _, invalidParam := range invalidParams { + if err := validator.Validate(openrtb_ext.BidderBrave, json.RawMessage(invalidParam)); err == nil { + t.Errorf("Schema allowed unexpected params: %s", invalidParam) + } + } +} + +var validParams = []string{ + `{ "placementId": "f897beb0daba0253d8e59a098eef9311" }`, +} + +var invalidParams = []string{ + ``, + `null`, + `true`, + `5`, + `4.2`, + `[]`, + `{}`, + `{"adCode": "string", "seatCode": 5, "originalPublisherid": "string"}`, + `{ "placementid": "f897beb0daba0253d8e59a098eef9311" }`, + `{ "PlacementId": "f897beb0daba0253d8e59a098eef9311" }`, +} diff --git a/exchange/adapter_builders.go b/exchange/adapter_builders.go index f4d747333e3..05daa46f0da 100755 --- a/exchange/adapter_builders.go +++ b/exchange/adapter_builders.go @@ -51,6 +51,7 @@ import ( "github.com/prebid/prebid-server/adapters/blue" "github.com/prebid/prebid-server/adapters/bmtm" "github.com/prebid/prebid-server/adapters/boldwin" + "github.com/prebid/prebid-server/adapters/brave" "github.com/prebid/prebid-server/adapters/brightroll" "github.com/prebid/prebid-server/adapters/ccx" "github.com/prebid/prebid-server/adapters/coinzilla" @@ -211,6 +212,7 @@ func newAdapterBuilders() map[openrtb_ext.BidderName]adapters.Builder { openrtb_ext.BidderBlue: blue.Builder, openrtb_ext.BidderBmtm: bmtm.Builder, openrtb_ext.BidderBoldwin: boldwin.Builder, + openrtb_ext.BidderBrave: brave.Builder, openrtb_ext.BidderBrightroll: brightroll.Builder, openrtb_ext.BidderCcx: ccx.Builder, openrtb_ext.BidderCoinzilla: coinzilla.Builder, diff --git a/openrtb_ext/bidders.go b/openrtb_ext/bidders.go index b4baf3d7f4b..e4091b227a2 100644 --- a/openrtb_ext/bidders.go +++ b/openrtb_ext/bidders.go @@ -131,6 +131,7 @@ const ( BidderBlue BidderName = "blue" BidderBmtm BidderName = "bmtm" BidderBoldwin BidderName = "boldwin" + BidderBrave BidderName = "brave" BidderBrightroll BidderName = "brightroll" BidderCcx BidderName = "ccx" BidderCoinzilla BidderName = "coinzilla" @@ -300,6 +301,7 @@ func CoreBidderNames() []BidderName { BidderBlue, BidderBmtm, BidderBoldwin, + BidderBrave, BidderBrightroll, BidderCcx, BidderCoinzilla, diff --git a/openrtb_ext/imp_brave.go b/openrtb_ext/imp_brave.go new file mode 100644 index 00000000000..2c7a0eaf9f0 --- /dev/null +++ b/openrtb_ext/imp_brave.go @@ -0,0 +1,5 @@ +package openrtb_ext + +type ExtImpBrave struct { + PlacementID string `json:"placementId"` +} diff --git a/static/bidder-info/brave.yaml b/static/bidder-info/brave.yaml new file mode 100644 index 00000000000..1e4b7ced428 --- /dev/null +++ b/static/bidder-info/brave.yaml @@ -0,0 +1,15 @@ +endpoint: "http://point.braveglobal.tv/?t=3&partner={{.PublisherID}}" +maintainer: + email: "support@thebrave.io" +gvlVendorID: 869 +capabilities: + app: + mediaTypes: + - banner + - video + - native + site: + mediaTypes: + - banner + - video + - native diff --git a/static/bidder-params/brave.json b/static/bidder-params/brave.json new file mode 100644 index 00000000000..8e913982e96 --- /dev/null +++ b/static/bidder-params/brave.json @@ -0,0 +1,13 @@ +{ + "$schema": "http://json-schema.org/draft-04/schema#", + "title": "Brave Adapter Params", + "description": "A schema which validates params accepted by the Brave adapter", + "type": "object", + "properties": { + "placementId": { + "type": "string", + "description": "Unique Placement ID of publisher" + } + }, + "required": ["placementId"] +} \ No newline at end of file