-
Notifications
You must be signed in to change notification settings - Fork 2.1k
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
Insticator Bid Adapter: add new bid adapter #7277
Changes from 1 commit
52d73d8
ef52a60
05e88db
886b75a
c036429
0be3938
33160a1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,272 @@ | ||
import { config } from '../src/config.js'; | ||
import { BANNER } from '../src/mediaTypes.js'; | ||
import { registerBidder } from '../src/adapters/bidderFactory.js'; | ||
import { | ||
deepAccess, | ||
generateUUID, | ||
logError, | ||
} from '../src/utils.js'; | ||
import { getStorageManager } from '../src/storageManager.js'; | ||
|
||
export const storage = getStorageManager(); | ||
const BIDDER_CODE = 'insticator'; | ||
const ENDPOINT = 'https://ex.ingage.tech/v1/openrtb'; // production endpoint | ||
const USER_ID_KEY = 'hb_insticator_uid'; | ||
const USER_ID_COOKIE_EXP = 2592000000; // 30 days | ||
const BID_TTL = 300; // 5 minutes | ||
|
||
config.setDefaults({ | ||
insticator: { | ||
endpointUrl: ENDPOINT, | ||
bidTTL: BID_TTL, | ||
}, | ||
}); | ||
|
||
function getUserId() { | ||
let uid; | ||
|
||
if (storage.localStorageIsEnabled()) { | ||
uid = localStorage.getItem(USER_ID_KEY); | ||
} else { | ||
uid = storage.getCookie(USER_ID_KEY); | ||
} | ||
|
||
if (uid && uid.length !== 36) { | ||
uid = undefined; | ||
} | ||
|
||
return uid; | ||
} | ||
|
||
function setUserId(userId) { | ||
if (storage.localStorageIsEnabled()) { | ||
localStorage.setItem(USER_ID_KEY, userId); | ||
} | ||
|
||
if (storage.cookiesAreEnabled()) { | ||
const expires = new Date(Date.now() + USER_ID_COOKIE_EXP).toISOString(); | ||
storage.setCookie(USER_ID_KEY, userId, expires); | ||
} | ||
} | ||
|
||
function buildImpression(bidRequest) { | ||
const format = []; | ||
const sizes = | ||
deepAccess(bidRequest, 'mediaTypes.banner.sizes') || bidRequest.sizes; | ||
|
||
for (const size of sizes) { | ||
format.push({ | ||
w: size[0], | ||
h: size[1], | ||
}); | ||
} | ||
|
||
return { | ||
id: bidRequest.bidId, | ||
tagid: bidRequest.adUnitCode, | ||
banner: { | ||
format, | ||
}, | ||
ext: { | ||
insticator: { | ||
adUnitId: bidRequest.params.adUnitId, | ||
}, | ||
}, | ||
}; | ||
} | ||
|
||
function buildDevice() { | ||
const device = { | ||
w: window.innerWidth, | ||
h: window.innerHeight, | ||
js: true, | ||
ext: { | ||
localStorage: storage.localStorageIsEnabled(), | ||
cookies: storage.cookiesAreEnabled(), | ||
}, | ||
}; | ||
|
||
const deviceConfig = config.getConfig('device'); | ||
|
||
if (typeof deviceConfig === 'object') { | ||
Object.assign(device, deviceConfig); | ||
} | ||
|
||
return device; | ||
} | ||
|
||
function buildRegs(bidderRequest) { | ||
if (bidderRequest.gdprConsent) { | ||
return { | ||
ext: { | ||
gdpr: bidderRequest.gdprConsent.gdprApplies ? 1 : 0, | ||
gdprConsentString: bidderRequest.gdprConsent.consentString, | ||
}, | ||
}; | ||
} | ||
|
||
return {}; | ||
} | ||
|
||
function buildUser() { | ||
const userId = getUserId() || generateUUID(); | ||
|
||
setUserId(userId); | ||
|
||
return { | ||
id: userId, | ||
}; | ||
} | ||
|
||
function buildRequest(validBidRequests, bidderRequest) { | ||
const req = { | ||
id: bidderRequest.bidderRequestId, | ||
tmax: bidderRequest.timeout, | ||
source: { | ||
fd: 1, | ||
tid: bidderRequest.auctionId, | ||
}, | ||
site: { | ||
domain: location.hostname, | ||
page: location.href, | ||
ref: bidderRequest.refererInfo.referer, | ||
}, | ||
device: buildDevice(), | ||
regs: buildRegs(bidderRequest), | ||
user: buildUser(), | ||
imp: validBidRequests.map((bidRequest) => buildImpression(bidRequest)), | ||
}; | ||
|
||
const params = config.getConfig('insticator.params'); | ||
|
||
if (params) { | ||
req.ext = { | ||
insticator: params, | ||
}; | ||
} | ||
|
||
return req; | ||
} | ||
|
||
function buildBid(bid, bidderRequest) { | ||
const originalBid = bidderRequest.bids.find((b) => b.bidId === bid.impid); | ||
|
||
return { | ||
requestId: bid.impid, | ||
creativeId: bid.crid, | ||
cpm: bid.price, | ||
currency: 'USD', | ||
netRevenue: true, | ||
ttl: bid.exp || config.getConfig('insticator.bidTTL') || BID_TTL, | ||
width: bid.w, | ||
height: bid.h, | ||
mediaType: 'banner', | ||
ad: bid.adm, | ||
adUnitCode: originalBid.adUnitCode, | ||
}; | ||
} | ||
|
||
function buildBidSet(seatbid, bidderRequest) { | ||
return seatbid.bid.map((bid) => buildBid(bid, bidderRequest)); | ||
} | ||
|
||
function validateSize(size) { | ||
return ( | ||
size instanceof Array && | ||
size.length === 2 && | ||
typeof size[0] === 'number' && | ||
typeof size[1] === 'number' | ||
); | ||
} | ||
|
||
function validateSizes(sizes) { | ||
return ( | ||
sizes instanceof Array && | ||
sizes.length > 0 && | ||
sizes.map(validateSize).reduce((a, b) => a && b, true) | ||
); | ||
} | ||
|
||
export const spec = { | ||
code: BIDDER_CODE, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You can include the |
||
supportedMediaTypes: [BANNER], | ||
|
||
isBidRequestValid: function (bid) { | ||
if (!bid.params.adUnitId) { | ||
logError('insticator: missing adUnitId bid parameter'); | ||
return false; | ||
} | ||
|
||
if (!(BANNER in bid.mediaTypes)) { | ||
logError('insticator: expected banner in mediaTypes'); | ||
return false; | ||
} | ||
|
||
if ( | ||
!validateSizes(bid.sizes) && | ||
!validateSizes(bid.mediaTypes.banner.sizes) | ||
) { | ||
logError('insticator: banner sizes not specified or invalid'); | ||
return false; | ||
} | ||
|
||
return true; | ||
}, | ||
|
||
buildRequests: function (validBidRequests, bidderRequest) { | ||
const requests = []; | ||
|
||
if (validBidRequests.length > 0) { | ||
requests.push({ | ||
method: 'POST', | ||
url: config.getConfig('insticator.endpointUrl') || ENDPOINT, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Can you please add a check for that? |
||
options: { | ||
contentType: 'application/json', | ||
withCredentials: true, | ||
}, | ||
data: JSON.stringify(buildRequest(validBidRequests, bidderRequest)), | ||
bidderRequest, | ||
}); | ||
} | ||
|
||
return requests; | ||
}, | ||
|
||
interpretResponse: function (serverResponse, request) { | ||
const bidderRequest = request.bidderRequest; | ||
const body = serverResponse.body; | ||
|
||
if (!body || body.id !== bidderRequest.bidderRequestId) { | ||
logError('insticator: response id does not match bidderRequestId'); | ||
return []; | ||
} | ||
|
||
if (!body.seatbid) { | ||
return []; | ||
} | ||
|
||
const bidsets = body.seatbid.map((seatbid) => | ||
buildBidSet(seatbid, bidderRequest) | ||
); | ||
|
||
return bidsets.reduce((a, b) => a.concat(b), []); | ||
}, | ||
|
||
getUserSyncs: function (options, responses) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do you wanna check here for Gdpr and Usp consent? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We check the GDPR and USP consent on the backend side and pass them via ext.sync. |
||
const syncs = []; | ||
|
||
for (const response of responses) { | ||
if ( | ||
response.body && | ||
response.body.ext && | ||
response.body.ext.sync instanceof Array | ||
) { | ||
syncs.push(...response.body.ext.sync); | ||
} | ||
} | ||
|
||
return syncs; | ||
}, | ||
}; | ||
|
||
registerBidder(spec); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
Overview | ||
======== | ||
|
||
``` | ||
Module Name: Insticator Adapter | ||
Module Type: Bidder Adapter | ||
Maintainer: contact@insticator.com | ||
``` | ||
|
||
Description | ||
=========== | ||
|
||
This module connects publishers to Insticator exchange of demand sources through Prebid.js. | ||
|
||
### Supported Media Types | ||
|
||
| Type | Support | ||
| --- | --- | ||
| Banner | Fully supported for all approved sizes. | ||
|
||
# Bid Parameters | ||
|
||
Each of the Insticator-specific parameters provided under the `adUnits[].bids[].params` | ||
object are detailed here. | ||
|
||
### Banner | ||
|
||
| Key | Scope | Type | Description | ||
| --- | --- | --- | --- | ||
| adUnitId | Required | String | The ad unit ID provided by Insticator. | ||
|
||
|
||
# Test Parameters | ||
``` | ||
var adUnits = [ | ||
{ | ||
code: 'test-div', | ||
sizes: [[300, 250]], | ||
bids: [ | ||
{ | ||
bidder: 'insticator', | ||
params: { | ||
adUnitId: 'test' | ||
} | ||
} | ||
] | ||
} | ||
] | ||
``` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If you have a GVL ID, which I think you do looking at your docs PR, you can pass that while initialising Storage Manager. That'll ensure your adapter doesn't gets blocked (when user has given consent) if GDPR Enforcement module is used. You can refer
appnexus
to see a sample usage.