github.com/prebid/prebid-server@v0.275.0/adapters/krushmedia/krushmedia.go (about) 1 package krushmedia 2 3 import ( 4 "encoding/json" 5 "fmt" 6 "net/http" 7 "strconv" 8 "text/template" 9 10 "github.com/prebid/openrtb/v19/openrtb2" 11 "github.com/prebid/prebid-server/adapters" 12 "github.com/prebid/prebid-server/config" 13 "github.com/prebid/prebid-server/errortypes" 14 "github.com/prebid/prebid-server/macros" 15 "github.com/prebid/prebid-server/openrtb_ext" 16 ) 17 18 type KrushmediaAdapter struct { 19 endpoint *template.Template 20 } 21 22 // Builder builds a new instance of the KrushmediaA adapter for the given bidder with the given config. 23 func Builder(bidderName openrtb_ext.BidderName, config config.Adapter, server config.Server) (adapters.Bidder, error) { 24 template, err := template.New("endpointTemplate").Parse(config.Endpoint) 25 if err != nil { 26 return nil, fmt.Errorf("unable to parse endpoint url template: %v", err) 27 } 28 29 bidder := &KrushmediaAdapter{ 30 endpoint: template, 31 } 32 return bidder, nil 33 } 34 35 func getHeaders(request *openrtb2.BidRequest) *http.Header { 36 headers := http.Header{} 37 headers.Add("Content-Type", "application/json;charset=utf-8") 38 headers.Add("Accept", "application/json") 39 headers.Add("X-Openrtb-Version", "2.5") 40 41 if request.Device != nil { 42 if len(request.Device.UA) > 0 { 43 headers.Add("User-Agent", request.Device.UA) 44 } 45 46 if len(request.Device.IP) > 0 { 47 headers.Add("X-Forwarded-For", request.Device.IP) 48 } 49 50 if len(request.Device.Language) > 0 { 51 headers.Add("Accept-Language", request.Device.Language) 52 } 53 54 if request.Device.DNT != nil { 55 headers.Add("Dnt", strconv.Itoa(int(*request.Device.DNT))) 56 } 57 } 58 59 return &headers 60 } 61 62 func (a *KrushmediaAdapter) MakeRequests( 63 openRTBRequest *openrtb2.BidRequest, 64 reqInfo *adapters.ExtraRequestInfo, 65 ) ( 66 requestsToBidder []*adapters.RequestData, 67 errs []error, 68 ) { 69 70 request := *openRTBRequest 71 72 var errors []error 73 var krushmediaExt *openrtb_ext.ExtKrushmedia 74 var err error 75 76 if len(request.Imp) > 0 { 77 krushmediaExt, err = a.getImpressionExt(&(request.Imp[0])) 78 if err != nil { 79 errors = append(errors, err) 80 } 81 request.Imp[0].Ext = nil 82 } else { 83 errors = append(errors, &errortypes.BadInput{ 84 Message: "Missing Imp Object", 85 }) 86 } 87 88 if len(errors) > 0 { 89 return nil, errors 90 } 91 92 url, err := a.buildEndpointURL(krushmediaExt) 93 if err != nil { 94 return nil, []error{err} 95 } 96 97 reqJSON, err := json.Marshal(request) 98 if err != nil { 99 return nil, []error{err} 100 } 101 102 return []*adapters.RequestData{{ 103 Method: http.MethodPost, 104 Body: reqJSON, 105 Uri: url, 106 Headers: *getHeaders(&request), 107 }}, nil 108 } 109 110 func (a *KrushmediaAdapter) getImpressionExt(imp *openrtb2.Imp) (*openrtb_ext.ExtKrushmedia, error) { 111 var bidderExt adapters.ExtImpBidder 112 if err := json.Unmarshal(imp.Ext, &bidderExt); err != nil { 113 return nil, &errortypes.BadInput{ 114 Message: "Bidder extension not provided or can't be unmarshalled", 115 } 116 } 117 var krushmediaExt openrtb_ext.ExtKrushmedia 118 if err := json.Unmarshal(bidderExt.Bidder, &krushmediaExt); err != nil { 119 return nil, &errortypes.BadInput{ 120 Message: "Error while unmarshaling bidder extension", 121 } 122 } 123 return &krushmediaExt, nil 124 } 125 126 func (a *KrushmediaAdapter) buildEndpointURL(params *openrtb_ext.ExtKrushmedia) (string, error) { 127 endpointParams := macros.EndpointTemplateParams{AccountID: params.AccountID} 128 return macros.ResolveMacros(a.endpoint, endpointParams) 129 } 130 131 func (a *KrushmediaAdapter) MakeBids( 132 openRTBRequest *openrtb2.BidRequest, 133 requestToBidder *adapters.RequestData, 134 bidderRawResponse *adapters.ResponseData, 135 ) ( 136 bidderResponse *adapters.BidderResponse, 137 errs []error, 138 ) { 139 if bidderRawResponse.StatusCode == http.StatusNoContent { 140 return nil, nil 141 } 142 143 if bidderRawResponse.StatusCode == http.StatusBadRequest { 144 return nil, []error{&errortypes.BadInput{ 145 Message: fmt.Sprintf("Unexpected status code: [ %d ]", bidderRawResponse.StatusCode), 146 }} 147 } 148 149 if bidderRawResponse.StatusCode == http.StatusServiceUnavailable { 150 return nil, nil 151 } 152 153 if bidderRawResponse.StatusCode != http.StatusOK { 154 return nil, []error{&errortypes.BadServerResponse{ 155 Message: fmt.Sprintf("Something went wrong, please contact your Account Manager. Status Code: [ %d ] ", bidderRawResponse.StatusCode), 156 }} 157 } 158 159 responseBody := bidderRawResponse.Body 160 var bidResp openrtb2.BidResponse 161 if err := json.Unmarshal(responseBody, &bidResp); err != nil { 162 return nil, []error{&errortypes.BadServerResponse{ 163 Message: "Bad Server Response", 164 }} 165 } 166 167 bidResponse := adapters.NewBidderResponseWithBidsCapacity(len(bidResp.SeatBid[0].Bid)) 168 sb := bidResp.SeatBid[0] 169 170 for _, bid := range sb.Bid { 171 bidResponse.Bids = append(bidResponse.Bids, &adapters.TypedBid{ 172 Bid: &bid, 173 BidType: getMediaTypeForImp(bid.ImpID, openRTBRequest.Imp), 174 }) 175 } 176 return bidResponse, nil 177 } 178 179 func getMediaTypeForImp(impId string, imps []openrtb2.Imp) openrtb_ext.BidType { 180 mediaType := openrtb_ext.BidTypeBanner 181 for _, imp := range imps { 182 if imp.ID == impId { 183 if imp.Video != nil { 184 mediaType = openrtb_ext.BidTypeVideo 185 } else if imp.Native != nil { 186 mediaType = openrtb_ext.BidTypeNative 187 } 188 return mediaType 189 } 190 } 191 return mediaType 192 }