Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New Adapter: Teads #3112

Merged
merged 5 commits into from
Oct 16, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
40 changes: 40 additions & 0 deletions adapters/teads/models.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
package teads

import (
"encoding/json"
"text/template"
)

type adapter struct {
endpointTemplate *template.Template
}

type defaultBidderImpExtension struct {
Bidder bidder `json:"bidder"`
}

type bidder struct {
PlacementId int `json:"placementId"`
}

type teadsImpExtension struct {
KV teadsKV `json:"kv"`
}

type teadsKV struct {
PlacementId int `json:"placementId"`
}

type teadsBidExt struct {
Prebid teadsPrebidExt `json:"prebid"`
}

type teadsPrebidExt struct {
Meta teadsPrebidMeta `json:"meta"`
}

type teadsPrebidMeta struct {
RendererName string `json:"rendererName"`
RendererVersion string `json:"rendererVersion"`
RendererData json.RawMessage `json:"rendererData"`
}
202 changes: 202 additions & 0 deletions adapters/teads/teads.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,202 @@
package teads

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

"github.com/prebid/openrtb/v19/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"
)

// Builder builds a new instance of the Teads adapter for the given bidder with the given config.
func Builder(_ openrtb_ext.BidderName, config config.Adapter, server config.Server) (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 *openrtb2.BidRequest, reqInfo *adapters.ExtraRequestInfo) ([]*adapters.RequestData, []error) {
if len(request.Imp) == 0 {
return nil, []error{&errortypes.BadInput{
Message: "No impression in the bid request",
}}
}
github-maxime-liege marked this conversation as resolved.
Show resolved Hide resolved

endpointURL, err := a.buildEndpointURL()
if endpointURL == "" {
return nil, []error{err}
}

if err := updateImpObject(request.Imp); err != nil {
return nil, []error{err}
}

reqJSON, err := json.Marshal(request)
if err != nil {
return nil, []error{&errortypes.BadInput{
Message: "Error parsing BidRequest object",
}}
}

headers := http.Header{}
headers.Add("Content-Type", "application/json;charset=utf-8")
return []*adapters.RequestData{{
Method: "POST",
Uri: endpointURL,
Body: reqJSON,
Headers: headers,
}}, []error{}
}

func updateImpObject(imps []openrtb2.Imp) error {
for i := range imps {
imp := &imps[i]

if imp.Banner != nil {
if len(imp.Banner.Format) != 0 {
bannerCopy := *imp.Banner
bannerCopy.H = &imp.Banner.Format[0].H
bannerCopy.W = &imp.Banner.Format[0].W
imp.Banner = &bannerCopy
}
}
github-maxime-liege marked this conversation as resolved.
Show resolved Hide resolved

var defaultImpExt defaultBidderImpExtension
if err := json.Unmarshal(imp.Ext, &defaultImpExt); err != nil {
return &errortypes.BadInput{
Message: "Error parsing Imp.Ext object",
}
}
if defaultImpExt.Bidder.PlacementId == 0 {
return &errortypes.BadInput{
Message: "placementId should not be 0.",
}
}
imp.TagID = strconv.Itoa(defaultImpExt.Bidder.PlacementId)
teadsImpExt := &teadsImpExtension{
KV: teadsKV{
PlacementId: defaultImpExt.Bidder.PlacementId,
},
}
if extJson, err := json.Marshal(teadsImpExt); err != nil {
return &errortypes.BadInput{
Message: "Error stringify Imp.Ext object",
}
} else {
imp.Ext = extJson
}
}
return nil
}

// Builds enpoint url based on adapter-specific pub settings from imp.ext
func (a *adapter) buildEndpointURL() (string, error) {
endpointParams := macros.EndpointTemplateParams{}
host, err := macros.ResolveMacros(a.endpointTemplate, endpointParams)

if err != nil {
return "", &errortypes.BadInput{
Message: "Unable to parse endpoint url template: " + err.Error(),
}
}

thisURI, err := url.Parse(host)
if err != nil {
return "", &errortypes.BadInput{
Message: "Malformed URL: " + err.Error(),
}
}

return thisURI.String(), nil
}

func (a *adapter) MakeBids(internalRequest *openrtb2.BidRequest, _ *adapters.RequestData, response *adapters.ResponseData) (*adapters.BidderResponse, []error) {
if adapters.IsResponseStatusCodeNoContent(response) {
return nil, nil
}

err := adapters.CheckResponseStatusCodeForErrors(response)
if err != nil {
return nil, []error{err}
}

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

bidderResponse := adapters.NewBidderResponseWithBidsCapacity(len(bidResp.SeatBid))

for _, sb := range bidResp.SeatBid {
for i := 0; i < len(sb.Bid); i++ {
bid := sb.Bid[i]

bidExtTeads, err := getTeadsRendererFromBidExt(bid.Ext)
if err != nil {
return nil, err
}
bidType, err := getMediaTypeForImp(bid.ImpID, internalRequest.Imp)
if err != nil {
return nil, err
}
bidderResponse.Bids = append(bidderResponse.Bids, &adapters.TypedBid{
Bid: &bid,
BidMeta: &openrtb_ext.ExtBidPrebidMeta{
RendererName: bidExtTeads.Prebid.Meta.RendererName,
RendererVersion: bidExtTeads.Prebid.Meta.RendererVersion,
github-maxime-liege marked this conversation as resolved.
Show resolved Hide resolved
},
BidType: bidType,
})
}
}
if bidResp.Cur != "" {
bidderResponse.Currency = bidResp.Cur
}
return bidderResponse, nil
}

func getTeadsRendererFromBidExt(ext json.RawMessage) (*teadsBidExt, []error) {
var bidExtTeads teadsBidExt
if err := json.Unmarshal(ext, &bidExtTeads); err != nil {
return nil, []error{err}
}
if bidExtTeads.Prebid.Meta.RendererName == "" {
return nil, []error{&errortypes.BadInput{
Message: "RendererName should not be empty if present",
}}
}
if bidExtTeads.Prebid.Meta.RendererVersion == "" {
return nil, []error{&errortypes.BadInput{
Message: "RendererVersion should not be empty if present",
}}
}
return &bidExtTeads, nil
}

func getMediaTypeForImp(impID string, imps []openrtb2.Imp) (openrtb_ext.BidType, []error) {
for _, imp := range imps {
if imp.ID == impID {
if imp.Video != nil {
onkarvhanumante marked this conversation as resolved.
Show resolved Hide resolved
return openrtb_ext.BidTypeVideo, nil
}
return openrtb_ext.BidTypeBanner, nil
}
}
return openrtb_ext.BidType(""), []error{&errortypes.BadInput{
Message: "Imp ids were not equals",
}}
}
27 changes: 27 additions & 0 deletions adapters/teads/teads_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package teads

import (
"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"
"testing"
)

func TestJsonSamples(t *testing.T) {
bidder, buildErr := Builder(openrtb_ext.BidderTeads, config.Adapter{
Endpoint: "https://a.teads.tv/prebid-server/bid-request"}, config.Server{ExternalUrl: "https://a.teads.tv/prebid-server/bid-request", GvlID: 1, DataCenter: "2"})

if buildErr != nil {
t.Fatalf("Builder returned unexpected error %v", buildErr)
}

adapterstest.RunJSONBidderTest(t, "teadstest", bidder)
}

func TestEndpointTemplateMalformed(t *testing.T) {
_, buildErr := Builder(openrtb_ext.BidderTeads, config.Adapter{
Endpoint: "{{Malformed}}"}, config.Server{ExternalUrl: "https://a.teads.tv/prebid-server/bid-request", GvlID: 1, DataCenter: "2"})

assert.Error(t, buildErr)
}
Loading