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

Cleanmedianet: Add cleanmedianet bidder adapter #3582

Merged
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
292 changes: 292 additions & 0 deletions modules/cleanmedianetBidAdapter.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,292 @@
import * as utils from '../src/utils';
import { parse } from '../src/url';
import { registerBidder } from '../src/adapters/bidderFactory';
import { config } from '../src/config';
import { Renderer } from '../src/Renderer';
import { BANNER, VIDEO } from '../src/mediaTypes';

export const helper = {
startsWith: function(str, search) {
return str.substr(0, search.length) === search;
},
getMediaType: function(bid) {
if (bid.ext) {
if (bid.ext.media_type) {
return bid.ext.media_type.toLowerCase();
} else if (bid.ext.vast_url) {
return VIDEO;
} else {
return BANNER;
}
}
return BANNER;
}
};

export const spec = {
code: 'cleanmedianet',
aliases: [],
supportedMediaTypes: [BANNER, VIDEO],

isBidRequestValid: function(bid) {
return (
!!bid.params.supplyPartnerId &&
typeof bid.params.supplyPartnerId === 'string' &&
(typeof bid.params.bidfloor === 'undefined' ||
typeof bid.params.bidfloor === 'number') &&
(typeof bid.params['adpos'] === 'undefined' ||
typeof bid.params['adpos'] === 'number') &&
(typeof bid.params['protocols'] === 'undefined' ||
Array.isArray(bid.params['protocols'])) &&
(typeof bid.params.instl === 'undefined' ||
bid.params.instl === 0 ||
bid.params.instl === 1)
);
},

buildRequests: function(validBidRequests, bidderRequest) {
return validBidRequests.map(bidRequest => {
const {
adUnitCode,
auctionId,
mediaTypes,
params,
sizes,
transactionId
} = bidRequest;
const baseEndpoint = 'https://bidder.cleanmediaads.com';
const rtbEndpoint =
`${baseEndpoint}/r/${
params.supplyPartnerId
}/bidr?rformat=open_rtb&reqformat=rtb_json&bidder=prebid` +
(params.query ? '&' + params.query : '');
let url =
config.getConfig('pageUrl') || bidderRequest.refererInfo.referer;

const rtbBidRequest = {
id: auctionId,
site: {
domain: parse(url).hostname,
page: url,
ref: bidderRequest.refererInfo.referer
},
device: {
ua: navigator.userAgent
},
imp: [],
ext: {}
};

if (
bidderRequest.gdprConsent &&
bidderRequest.gdprConsent.consentString &&
bidderRequest.gdprConsent.gdprApplies
) {
rtbBidRequest.ext.gdpr_consent = {
consent_string: bidderRequest.gdprConsent.consentString,
consent_required: bidderRequest.gdprConsent.gdprApplies
};
}

const imp = {
id: transactionId,
instl: params.instl === 1 ? 1 : 0,
tagid: adUnitCode,
bidfloor: params.bidfloor || 0,
bidfloorcur: 'USD',
secure: helper.startsWith(
utils.getTopWindowUrl().toLowerCase(),
'http://'
)
? 0
: 1
};

const hasFavoredMediaType =
params.favoredMediaType &&
this.supportedMediaTypes.includes(params.favoredMediaType);

if (!mediaTypes || mediaTypes.banner) {
if (!hasFavoredMediaType || params.favoredMediaType === BANNER) {
const bannerImp = Object.assign({}, imp, {
banner: {
w: sizes.length ? sizes[0][0] : 300,
h: sizes.length ? sizes[0][1] : 250,
pos: params.pos || 0,
topframe: bidderRequest.refererInfo.reachedTop
}
});
rtbBidRequest.imp.push(bannerImp);
}
}

if (mediaTypes && mediaTypes.video) {
if (!hasFavoredMediaType || params.favoredMediaType === VIDEO) {
const videoImp = Object.assign({}, imp, {
video: {
w: sizes.length ? sizes[0][0] : 300,
h: sizes.length ? sizes[0][1] : 250,
protocols: params.protocols || [1, 2, 3, 4, 5, 6],
pos: params.pos || 0,
ext: {
context: mediaTypes.video.context
}
}
});
rtbBidRequest.imp.push(videoImp);
}
}

if (rtbBidRequest.imp.length === 0) {
return;
}

return {
method: 'POST',
url: rtbEndpoint,
data: rtbBidRequest,
bidRequest
};
});
},

interpretResponse: function(serverResponse, bidRequest) {
const response = serverResponse && serverResponse.body;
if (!response) {
utils.logError('empty response');
return [];
}

const bids = response.seatbid.reduce(
(acc, seatBid) => acc.concat(seatBid.bid),
[]
);
let outBids = [];

bids.forEach(bid => {
const outBid = {
adId: bidRequest.bidRequest.adUnitCode,
requestId: bidRequest.bidRequest.bidId,
cpm: bid.price,
width: bid.w,
height: bid.h,
ttl: 60 * 10,
creativeId: bid.crid,
netRevenue: true,
currency: bid.cur || response.cur,
adUnitCode: bidRequest.bidRequest.adUnitCode,
mediaType: helper.getMediaType(bid)
};

if (
utils.deepAccess(
bidRequest.bidRequest,
'mediaTypes.' + outBid.mediaType
)
) {
if (outBid.mediaType === BANNER) {
outBids.push(Object.assign({}, outBid, { ad: bid.adm }));
} else if (outBid.mediaType === VIDEO) {
const context = utils.deepAccess(
bidRequest.bidRequest,
'mediaTypes.video.context'
);
outBids.push(
Object.assign({}, outBid, {
vastUrl: bid.ext.vast_url,
vastXml: bid.adm,
renderer:
context === 'outstream'
? newRenderer(bidRequest.bidRequest, bid)
: undefined
})
);
}
}
});
return outBids;
},

getUserSyncs: function(syncOptions, serverResponses, gdprConsent) {
const syncs = [];
const gdprApplies =
gdprConsent && typeof gdprConsent.gdprApplies === 'boolean'
? gdprConsent.gdprApplies
: false;
const suffix = gdprApplies
? 'gc=' + encodeURIComponent(gdprConsent.consentString)
: 'gc=missing';
serverResponses.forEach(resp => {
if (resp.body) {
const bidResponse = resp.body;
if (bidResponse.ext && Array.isArray(bidResponse.ext['utrk'])) {
bidResponse.ext['utrk'].forEach(pixel => {
const url =
pixel.url +
(pixel.url.indexOf('?') > 0 ? '&' + suffix : '?' + suffix);
return syncs.push({ type: pixel.type, url });
});
}
if (Array.isArray(bidResponse.seatbid)) {
bidResponse.seatbid.forEach(seatBid => {
if (Array.isArray(seatBid.bid)) {
seatBid.bid.forEach(bid => {
if (bid.ext && Array.isArray(bid.ext['utrk'])) {
bid.ext['utrk'].forEach(pixel => {
const url =
pixel.url +
(pixel.url.indexOf('?') > 0
? '&' + suffix
: '?' + suffix);
return syncs.push({ type: pixel.type, url });
});
}
});
}
});
}
}
});
return syncs;
}
};

function newRenderer(bidRequest, bid, rendererOptions = {}) {
const renderer = Renderer.install({
url:
(bidRequest.params && bidRequest.params.rendererUrl) ||
(bid.ext && bid.ext.renderer_url) ||
'//s.wlplayer.com/video/latest/renderer.js',
config: rendererOptions,
loaded: false
});
try {
renderer.setRender(renderOutstream);
} catch (err) {
utils.logWarn('Prebid Error calling setRender on renderer', err);
}
return renderer;
}

function renderOutstream(bid) {
bid.renderer.push(() => {
const unitId = bid.adUnitCode + '/' + bid.adId;
window['GamoshiPlayer'].renderAd({
id: unitId,
debug: window.location.href.indexOf('pbjsDebug') >= 0,
placement: document.getElementById(bid.adUnitCode),
width: bid.width,
height: bid.height,
events: {
ALL_ADS_COMPLETED: () =>
window.setTimeout(() => {
window['GamoshiPlayer'].removeAd(unitId);
}, 300)
},
vastUrl: bid.vastUrl,
vastXml: bid.vastXml
});
});
}

registerBidder(spec);
66 changes: 66 additions & 0 deletions modules/cleanmedianetBidAdapter.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
# Overview

```
Module Name: Clean Media Net Adapter
Module Type: Bidder Adapter
Maintainer: dev@cleanmedia.net
```

# Description

Connects to Clean Media Net's Programmatic advertising platform as a service.

Clean Media bid adapter supports Banner & Video (Instream and Outstream).
The *only* required parameter (in the `params` section) is the `supplyPartnerId` parameter.

# Test Parameters
```
var adUnits = [
// Banner adUnit
{
code: 'banner-div',
sizes: [[300, 250]],
bids: [{
bidder: 'cleanmedianet',
params: {
// ID of the supply partner you created in the Clean Media Net dashboard
supplyPartnerId: '1253',
// OPTIONAL: custom bid floor
bidfloor: 0.01,
// OPTIONAL: if you know the ad position on the page, specify it here
// (this corresponds to "Ad Position" in OpenRTB 2.3, section 5.4)
//adpos: 1,
// OPTIONAL: whether this is an interstitial placement (0 or 1)
// (see "instl" property in "Imp" object in the OpenRTB 2.3, section 3.2.2)
//instl: 0
}
}]
},
// Video outstream adUnit
{
code: 'video-outstream',
sizes: [[300, 250]],
mediaTypes: {
video: {
context: 'outstream',
playerSize: [300, 250]
}
},
bids: [ {
bidder: 'cleanmedianet',
params: {
// ID of the supply partner you created in the dashboard
supplyPartnerId: '1254',
// OPTIONAL: custom bid floor
bidfloor: 0.01,
// OPTIONAL: if you know the ad position on the page, specify it here
// (this corresponds to "Ad Position" in OpenRTB 2.3, section 5.4)
//adpos: 1,
// OPTIONAL: whether this is an interstitial placement (0 or 1)
// (see "instl" property in "Imp" object in the OpenRTB 2.3, section 3.2.2)
//instl: 0
}
}]
}
];
```
Loading