github.com/prebid/prebid-server@v0.275.0/adapters/motorik/motorik.go (about)

     1  package motorik
     2  
     3  import (
     4  	"encoding/json"
     5  	"fmt"
     6  	"net/http"
     7  	"text/template"
     8  
     9  	"github.com/prebid/openrtb/v19/openrtb2"
    10  	"github.com/prebid/prebid-server/adapters"
    11  	"github.com/prebid/prebid-server/config"
    12  	"github.com/prebid/prebid-server/errortypes"
    13  	"github.com/prebid/prebid-server/macros"
    14  	"github.com/prebid/prebid-server/openrtb_ext"
    15  )
    16  
    17  type adapter struct {
    18  	endpoint *template.Template
    19  }
    20  
    21  func Builder(bidderName openrtb_ext.BidderName, config config.Adapter, server config.Server) (adapters.Bidder, error) {
    22  	template, err := template.New("endpointTemplate").Parse(config.Endpoint)
    23  	if err != nil {
    24  		return nil, fmt.Errorf("unable to parse endpoint url template: %v", err)
    25  	}
    26  
    27  	bidder := &adapter{
    28  		endpoint: template,
    29  	}
    30  	return bidder, nil
    31  }
    32  
    33  func getHeaders(request *openrtb2.BidRequest) http.Header {
    34  	headers := http.Header{}
    35  	headers.Add("Content-Type", "application/json;charset=utf-8")
    36  	headers.Add("Accept", "application/json")
    37  	headers.Add("X-Openrtb-Version", "2.5")
    38  
    39  	if request.Device != nil {
    40  		if len(request.Device.UA) > 0 {
    41  			headers.Add("User-Agent", request.Device.UA)
    42  		}
    43  
    44  		if len(request.Device.IPv6) > 0 {
    45  			headers.Add("X-Forwarded-For", request.Device.IPv6)
    46  		}
    47  
    48  		if len(request.Device.IP) > 0 {
    49  			headers.Add("X-Forwarded-For", request.Device.IP)
    50  		}
    51  	}
    52  
    53  	return headers
    54  }
    55  
    56  func (a *adapter) MakeRequests(openRTBRequest *openrtb2.BidRequest, reqInfo *adapters.ExtraRequestInfo) (requestsToBidder []*adapters.RequestData, errs []error) {
    57  	motorikExt, err := getImpressionExt(&openRTBRequest.Imp[0])
    58  	if err != nil {
    59  		return nil, []error{err}
    60  	}
    61  
    62  	openRTBRequest.Imp[0].Ext = nil
    63  
    64  	url, err := a.buildEndpointURL(motorikExt)
    65  	if err != nil {
    66  		return nil, []error{err}
    67  	}
    68  
    69  	reqJSON, err := json.Marshal(openRTBRequest)
    70  	if err != nil {
    71  		return nil, []error{err}
    72  	}
    73  
    74  	return []*adapters.RequestData{{
    75  		Method:  http.MethodPost,
    76  		Body:    reqJSON,
    77  		Uri:     url,
    78  		Headers: getHeaders(openRTBRequest),
    79  	}}, nil
    80  }
    81  
    82  func getImpressionExt(imp *openrtb2.Imp) (*openrtb_ext.ExtMotorik, error) {
    83  	var bidderExt adapters.ExtImpBidder
    84  	if err := json.Unmarshal(imp.Ext, &bidderExt); err != nil {
    85  		return nil, &errortypes.BadInput{
    86  			Message: "Error parsing motorikExt - " + err.Error(),
    87  		}
    88  	}
    89  	var motorikExt openrtb_ext.ExtMotorik
    90  	if err := json.Unmarshal(bidderExt.Bidder, &motorikExt); err != nil {
    91  		return nil, &errortypes.BadInput{
    92  			Message: "Error parsing bidderExt - " + err.Error(),
    93  		}
    94  	}
    95  
    96  	return &motorikExt, nil
    97  }
    98  
    99  func (a *adapter) buildEndpointURL(params *openrtb_ext.ExtMotorik) (string, error) {
   100  	endpointParams := macros.EndpointTemplateParams{AccountID: params.AccountID, SourceId: params.PlacementID}
   101  	return macros.ResolveMacros(a.endpoint, endpointParams)
   102  }
   103  
   104  func checkResponseStatusCodes(response *adapters.ResponseData) error {
   105  	if response.StatusCode == http.StatusServiceUnavailable {
   106  		return &errortypes.BadServerResponse{
   107  			Message: fmt.Sprintf("Something went wrong Status Code: [ %d ] ", response.StatusCode),
   108  		}
   109  	}
   110  
   111  	return adapters.CheckResponseStatusCodeForErrors(response)
   112  }
   113  
   114  func (a *adapter) MakeBids(openRTBRequest *openrtb2.BidRequest, requestToBidder *adapters.RequestData, bidderRawResponse *adapters.ResponseData) (bidderResponse *adapters.BidderResponse, errs []error) {
   115  	if adapters.IsResponseStatusCodeNoContent(bidderRawResponse) {
   116  		return nil, nil
   117  	}
   118  
   119  	httpStatusError := checkResponseStatusCodes(bidderRawResponse)
   120  	if httpStatusError != nil {
   121  		return nil, []error{httpStatusError}
   122  	}
   123  
   124  	responseBody := bidderRawResponse.Body
   125  	var bidResp openrtb2.BidResponse
   126  	if err := json.Unmarshal(responseBody, &bidResp); err != nil {
   127  		return nil, []error{&errortypes.BadServerResponse{
   128  			Message: "Bad Server Response",
   129  		}}
   130  	}
   131  
   132  	if len(bidResp.SeatBid) == 0 {
   133  		return nil, []error{&errortypes.BadServerResponse{
   134  			Message: "Empty SeatBid array",
   135  		}}
   136  	}
   137  
   138  	bidResponse := adapters.NewBidderResponseWithBidsCapacity(5)
   139  	var bidsArray []*adapters.TypedBid
   140  
   141  	for _, sb := range bidResp.SeatBid {
   142  		for idx, bid := range sb.Bid {
   143  			bidType, err := getMediaTypeForImp(bid.ImpID, openRTBRequest.Imp)
   144  			if err != nil {
   145  				return nil, []error{err}
   146  			}
   147  
   148  			bidsArray = append(bidsArray, &adapters.TypedBid{
   149  				Bid:     &sb.Bid[idx],
   150  				BidType: bidType,
   151  			})
   152  		}
   153  	}
   154  
   155  	bidResponse.Bids = bidsArray
   156  	return bidResponse, nil
   157  }
   158  
   159  func getMediaTypeForImp(impId string, imps []openrtb2.Imp) (openrtb_ext.BidType, error) {
   160  	for _, imp := range imps {
   161  		if imp.ID == impId {
   162  			if imp.Banner != nil {
   163  				return openrtb_ext.BidTypeBanner, nil
   164  			} else if imp.Video != nil {
   165  				return openrtb_ext.BidTypeVideo, nil
   166  			} else if imp.Native != nil {
   167  				return openrtb_ext.BidTypeNative, nil
   168  			}
   169  			break
   170  		}
   171  	}
   172  	return "", &errortypes.BadInput{
   173  		Message: fmt.Sprintf("Failed to find impression \"%s\"", impId),
   174  	}
   175  }