- Introduction
- Requirements
- Partner Module Overview
- Utility Libraries
- Linting
- Debugging
- Testing
- Code Submission Guidelines
Welcome to the Index Exchange Partner Certification Process!
Below you will find everything you need to complete the certification process and be a part of the Header Tag Wrapper!
README.md
- This is the main documentation file which should contain everything you need to complete the certification process. If anything is unclear, please refer to this document first.open-x-htb.js
- This is your partner module file, by default it contains a template divided into multiple sections which need to be completed.open-x-htb-validator.js
- This is the validator file for the configuration object that will be passed into your module.open-x-htb-exports.js
- A file that contains all of the modules exports (i.e. any functions that need to be exposed to the outside world).
- Complete the open-x-htb.js file
- open-x-htb.js is where all of your adapter code will live.
- In order to complete the partner module correctly, please refer to the Partner Module Overview and the Utility Libraries sections.
- Please refer to the Partner Requirements and Guidelines when creating your module. Ensure requirements are met to streamline the review process.
- Complete the open-x-htb-validator.js file
- This file is where your partner-specific configurations will need to be validated.
- Things like type and null checks will be done here.
- Complete the open-x-htb-exports.js file
- This file will contain any functions that need to be exported or exposed to the outside world. Things like render functions, custom callbacks, etc. Any legacy render functions will also need to be exposed here. Anything added to the
shellInterface.OpenXHtb
will be accessible throughwindow.headertag.OpenXHtb
- This file will contain any functions that need to be exported or exposed to the outside world. Things like render functions, custom callbacks, etc. Any legacy render functions will also need to be exposed here. Anything added to the
- Submitting for Review
- Once the module has been verified submit a pull request from the
development-v2
branch to themaster-v2
branch for the Index Exchange team to review. If everything is approved, your adapter will be officially certified!
- Once the module has been verified submit a pull request from the
In order for your module to be successfully certified, please refer to the following list of requirements and guidelines. Items under required must be satisfied in order to pass the certification process. Items under guidelines are recommended for an optimal setup and should be followed if possible.
- The only targeting keys that can be set are predetermined by Index. The partner module should not be setting targeting on other keys.
- Must support the following browsers: IE 9+, Edge, Chrome, Safari, and Firefox
- Must run system tests in all supported browsers
- Must not use polyfills
- Please use our helper libraries when possible. Refer to our Utility Libraries documentation below. All of the utility functions in this library are designed to be backwards compatible with supported browsers to ease cross-browser compatibility.
- Must provide cache busting parameter. Cache busting is required to prevent network caches.
- Partner endpoint domain must be consistent. Load balancing should be performed by the endpoint.
- Your endpoint should support HTTPS request. When wrapper loads in secure pages, all requests need to be HTTPS. If you're unable to provide a secure endpoint, we will not be able to make requests to your ad servers.
- Your module should support a single request architecture (SRA) which has a capability to send multiple bid requests in a single HTTP request.
- Partner should use AJAX method to make bid requests. Please use our Network utility library for making network requests, particularly the
Network.ajax
method. This ensures the requests will work with all browsers.
- Returned bid must be a net bid.
- Pass on bid must be explicit in the response.
- All returned bids shall be in the currency trafficked by the publisher.
- Size must be returned in the response. The sizes must also match the requested size. This size parameter will be validated.
- Encrypted prices must be flagged in the response.
- Pixels/beacons must only be dropped only when the partner wins the auction and their ad renders.
- Dropping pixels during auction slows down the execution of auctions and is not allowed by Index Exchange.
- DFP line items, creatives, and render functions will be set up by the Index Exchange team.
The Header Tag Wrapper passes each partner a configuration object that has been configured based on a publisher's website. This object contains all the configuration information required for the wrapper, including partner-specific slot mappings, timeouts, and any other partner-specific configuration. The partner-specific slot mappings dictate how ad slots on the page will map to partner-specific configuration. There are 2 concepts to be familiar with when understanding how slots on the page are mapped back to partner-specific configuration. Header Tag Slots, which refers to htSlots and partner-specific configuration, which refers to xSlots in the codebase.
- htSlots - This is an abstraction of the googletag.slot object.
- These will need to be mapped to xSlots.
- xSlots - These are also an abstraction for partner-specific configuration that will be mapped to htSlots.
- These represent a single partner-specific configuration.
- An xSlot is how a partner can map their ad server specific identifiers (placementIDs, siteIDs, zoneIDs, etc) to the
htSlot
object. - It can represent a single or multiple sizes.
- Multiple xSlots can be mapped to the same htSlot.
Example Partner Configuration Mapping
{
"partners": {
"OpenXHtb": {
"enabled": true,
"configs": {
"xSlots": {
"xSlot1": {
"placementID": "123",
"sizes": [ [300, 250], [300, 600] ]
},
"xSlot2": {
"placementID": "345",
"sizes": [ [300, 250] ]
}
},
"mapping": {
"htSlotID-1": [ "xSlot1" ],
"htSlotID-2": [ "xSlot2" ]
}
}
}
}
}
Based on the mapping defined in the partner config, Parcels will be generated for every xSlot/htSlot pair. Parcels are objects that carry different kinds of information throughout the wrapper. In the context of the adapter, parcels are the input into your adapter. Parcels carry information regarding which slots on the publisher's page need demand. More specifically parcels contain information like xSlot reference, htSlot references, demand (after its been applied by the adapter in parseResponse), etc. Each parcel represents a single combination of an htSlot and an xSlot.
Each parcel is an object in the following form:
{
"partnerId": "OPNX",
"htSlot": {
"__type__": "HeaderTagSlot"
},
"ref": "googletag.Slot", // reference to the slot on the page, in this example it is a googletag slot
"xSlotRef": { // contains the reference to the xSlot object from the partner configuration.
"placementID": "123",
"sizes": [ [300, 250], [300, 600] ]
},
"xSlotName": "1",
"requestId": "_fwyvecpA" // generated by the wrapper is used to map the creative back to a piece of demand.
// This will show up as targeting under the id key, and is used by the render function.
}
These parcels will be fed into both of the functions that your adapter needs to implement:
-
First
generateRequestObj
will need to craft a bid request for theparcels
that need demand. -
Second
parseResponse
will take the same parcels and apply the demand (set specific properties in the parcel objects) that is returned from the bid requests to the same parcels and send them back to the wrapper.
- The Header Tag Wrapper script tag is loaded on the page.
- Wrapper specific configuration validation is performed.
- All the partner modules are instantiated.
- Partner-specific configuration validation is performed - checking that all the required fields are provided and that they are in the correct format.
- An external request for demand is made to the wrapper. This can be via a googletag display or refresh call, or by other methods depending on the wrapper product in use.
The wrapper requests demand from the partner modules for the required slots (provided in the form of parcels).
- The wrapper calls
generateRequestObj(returnParcels)
for every partner module. - The adapter then crafts and returns a request object based on the parcels (containing slot information) specified.
- The wrapper then sends out a bid request using the request object.
- Depending on how the adapter is set up and whether JSONP is supported, a response callback (
adResponseCallback
) is called. - The adapter parses the response (
parseResponse
) and attaches the demand to the same returnParcels. It also registers the ad creative with the wrapper's render service. - The returnParcels are then sent back to the wrapper.
- The wrapper calls
- The wrapper applies targeting using the demand from the returnParcels.
- If the partner wins the auction in the ad server, their creative code will be returned and executed.
- The creative code contains a call to the wrapper's render function.
- The partner ad is rendered.
In this section you will be filling out the open-x-htb.js, open-x-htb-exports.js, and the open-x-htb-validator.js files to create your module.
Before you get started on writing the actual code for your module, you need to figure out what your partner configuration (refer to Configuration) object will look like. This is crucial because it will determine the input (parcels) to your module's core functions.
Once you have a basic idea of what this will look like, and how you will uniquely identify each slot on your server (via xSlot placementId or other inventory codes) you will need to validate this configuration. This validation will be performed by the wrapper using the open-x-htb-validator.js
file.
The open-x-htb-validator.js
file contains a single export, a partnerValidator
function, that takes in the configuration object that will be fed to your module's constructor (refer to Configuration for an example layout) and validates it via type checks. The type checks are performed using an external library called schema-inspector
, for which the documentation can be found here https://github.com/Atinux/schema-inspector.
Once you have filled this file out, you can continue actually writing your module!
This section involves setting up the general partner configuration such as name, default pricing strategy as well as the general format of incoming/outgoing bids for the adapter. Please read the following descriptions and update the __profile
variable if necessary.
- partnerId - This is simply the name of our module, generally if your module is a bidder the name will end with Htb. The format of the name should be PartnerName{Type}.
- namespace - Should be the same as partnerId, it is the namespace that is used internally to store all of variables/functions related to your module, i.e. adResponseCallbacks.
- statsId - A unique identifier used for analytics that will be provided for you.
- version - If this is the first iteration of your module, please leave this field at 2.0.0.
- targetingType - The targeting type of your bidder, the default is slot for slot level targeting but could also be page.
- enabledAnalytics - The analytics that the wrapper will track for the module. requestTime is the only currently supported analytic, which records different times around when bid requests occur.
- features - Extra features that a partner can support
- demandExpiry - Setting an expiry time on the demand that partner returns.
- rateLimiting - Used for limiting the amount of requests to a given partner for a given slot on pages that support rate limiting in DFP.
- targetingKeys - Different targeting keys that will be used to record demand for a given parcel.
- id - This key will be used to trace back the creative that has won in DFP for rendering.
- om - This key signals the open market bid in CPM.
- pm - This key signals the private market bid in CPM.
- pmid - This key signals the private market deal id.
OpenX use pm as private market bid targeting, e.g. ix_ox_pm=300x250_1. p.s.: targeting value is composed of **ad unit size**: 300x250 and **deal_id**: 1
- bidUnitInCents - This tells the wrapper the unit of the bid price returned by your endpoint. The unit is in terms of cents, thus it should be set to 100 if your endpoint returns dollars, 1 if your endpoint returns cents, 0.1 if your endpoint returns tenth of a cent, etc. Note that this value must be an integer power of 10.
The last three properties are critical for the wrapper to understand how to interact with the endpoint:
- callbackType:
- Partner.CallbackTypes.ID -
Use this option if your endpoint accepts an arbitrary identifier in requests which will be returned in the matching response. You will need to set this callbackId in the
generateRequestObj
function and retrieve it in the adResponseCallback function. This is the preferred method for matching requests and responses. Pass the following reference into your request object when generating the request object ingenerateRequestObj
.SpaceCamp.NAMESPACE + '.' + __profile.namespace + '.adResponseCallback'
. You will need to fill out theadResponseCallback
function to store the ad response in the__baseClass._adResponseStore[callbackId]
object, wherecallbackId
is the same callbackId that was passed back to the wrapper as part of the object that is returned ingenerateRequestObj
. The wrapper will use thecallbackId
to reference to pull your stored ad response from thatadResponseStore
and feed it back toparseResponse
. You will also need to expose this function inside your exports file, see below exports section for more information. - Partner.CallbackTypes.CALLBACK_NAME -
Use this option if your endpoint has no parameter which can be used as a callback ID. The wrapper will generate a new callback function for each request, and use the function name to tie requests to responses. Similarly to the above ID method, you can reference the generated ad response callback as such:
'window.' + SpaceCamp.NAMESPACE + '.' + __profile.namespace + '.adResponseCallbacks.' + callbackId
, wherecallbackId
is the same unique callbackId that is passed back to the wrapper as part of the object that is returned ingenerateRequestObj
. Unlike the ID method above, you do not need to fill out theadResponseCallback
function. The response will simply be passed into theparseResponse
function. - Partner.CallbackTypes.NONE -
Use this option if your endpoint supports AJAX only and will return a pure JSON response rather than JSONP with a callback function. In this mode your endpoint will not receive any demand requests if the user is using a browser which does not fully support AJAX, such as Internet Explorer 9 or earlier. In this mode,
you also do NOT need to provide a callback function in
generateRequestObj
.
- Partner.CallbackTypes.ID -
Use this option if your endpoint accepts an arbitrary identifier in requests which will be returned in the matching response. You will need to set this callbackId in the
- architecture:
- Partner.Architectures.MRA - Use this option (Multi-Request Architecture) if your endpoint requires a separate network request for each ad slot. In this mode your endpoint will receive one network request for every xSlot mapping active in the current wrapper demand request.
- Partner.Architectures.FSRA - Use this option (Fully Single-Request Architecture) if your endpoint can handle any number of slots in a single request in any combination, including multiple requests for the same slot. In this mode your endpoint will receive a single network request per wrapper demand request.
- Partner.Architectures.SRA - Use this option (Single-Request Architecture) if your endpoint can handle any number of slots in a single request, but cannot repeat the same xSlot more than once in a given request. e.g., you use a placementId and can receive a request for multiple placementIds at once, but not for the same placementId twice. The wrapper will arrange the mapped xSlots into the minimum possible number of network requests to your endpoint.
- requestType:
- Partner.RequestTypes.ANY - Use any request type when making requests. The wrapper will attempt to make an AJAX request for your bid requests, if XHR is not supported, the wrapper will attempt to make a JSONP request if possible.
- Partner.RequestTypes.AJAX - Use only AJAX for bid requests. Note, if the browser does not support ajax the bid requests will not go out.
- Partner.RequestTypes.JSONP - Use only JSONP for bid requests.
This step is for crafting a bid request URL given a specific set of parcels.
For this step, you must fill out the generateRequestObj(returnParcels)
function. This function takes in an array of returnParcels.
These are the parcel objects that contain the different slots for which demand needs to be requested.
The wrapper will ensure that the array contains an appropriate set of parcels to pass into a single network request for your endpoint based on the value set in __profile.architecture. Note, in the particular case your architecture is MRA, this array will have length 1.
Using this array of parcels, the adapter must craft a request object that will be used to send out the bid request for these slots. This object must contain the request URL, an object containing query parameters, and a callbackId.
The final returned object should looks something like this:
{
url: 'http://bidserver.com/api/bids' // base request URL for a GET request
data: { // query string object that will be attached to the base URL
slots: [
{
placementId: 54321,
sizes: [[300, 250]]
},{
placementId: 12345,
sizes: [[300, 600]]
},{
placementId: 654321,
sizes: [[728, 90]]
}
],
site: 'http://google.com'
},
callbackId: '_23sd2ij4i1' //unique id used for pairing requests and responses
}
If your endpoint uses POST please add the following networkParamOverrides
object to your return object:
{
url: 'http://bidserver.com/api/bids' // base request URL for a POST request
data: { // will be the payload in the POST request
slots: [
{
placementId: 54321,
sizes: [[300, 250]]
},{
placementId: 12345,
sizes: [[300, 600]]
},{
placementId: 654321,
sizes: [[728, 90]]
}
],
site: 'http://google.com'
},
callbackId: '_23sd2ij4i1' //unique id used for pairing requests and responses,
/* Signals a POST request and the content type */
networkParamOverrides: {
method: 'POST',
contentType: 'text/plain'
}
}
More information can be found in the comment section of the function itself.
Once the request from Step 2 finishes the adResponseCallback
will be called to store the returned response in a adResponseStore
object.
If __profile.callbackType
is set to CALLBACK_NAME
or NONE
, the wrapper will handle the callback for you and you can remove this function. If it is set to ID, you must retrieve the callback ID from the network response and store that response in the _adResponseStore
object keyed by the callback ID.
See the function in the template for details.
In this step the adapter must parse the returned demand from the bid response and attach it the returnParcels objects.
The returnParcels array will be one of the same arrays that was passed to generateRequestObj
earlier.
This step involves first matching the returned bids to the internal returnParcels objects. This can be done via some identifier that was setup for an xSlot (for example, a placementId) in the partner configuration and the same id being present in the bid response object.
This function first iterates over all of the returned bids, parsing them and attaching the demand to the returnParcel objects (which will be implicitly passed back to the wrapper). This step also involves registering the creative (if returned) with the render service, which is responsible for storing and rendering that creative if the corresponding demand wins the DFP auction.
In order to complete this step correctly, please fill out the section which includes the matching criteria. This step is necessary to map returnParcel objects to the returned bids correctly. In order to do this, we need to use some common criteria that is present both in the xSlot configuration and in the returned bids (usually placementIds or inventory codes).
Also please fill out each of the variables for each bid, these will be attached to the parcel objects to store the demand:
- bidPrice - The bid price for the given slot
- bidWidth - The width of the given slot
- bidHeight - The height of the given slot
- bidCreative - The creative/adm for the given slot that will be rendered if is the winner. Needs to be decoded and ready for a doc.write.
- bidDealId - The dealId if applicable for this slot.
- bidIsPass - True/false value for if the module returned a pass for this slot.
- pixelUrl - If you require firing a tracking pixel after your creative wins and your tracking pixel is not part of the original adm that is rendered, please provide the decoded URL of the tracking pixel here. This
pixelUrl
variable will then be passed to the__renderPixel
function, which will then be called by either your module's specificrender
function or a generic rendering function depending on the line item setup.
After filling out these objects, the resulting returnParcel objects should look something like this:
{
"partnerId": "OPNX",
"htSlot": {
"__type__": "HeaderTagSlot"
},
"ref": "googletag.Slot", // reference to the slot on the page, in this example it is a googletag slot
"xSlotRef": { // contains the reference to the xSlot object from the partner configuration.
"placementID": "123",
"sizes": [ [300, 250], [300, 600] ]
},
"xSlotName": "1",
"requestId": "_fwyvecpA",
// notice these new fields with demand
"targetingType": "slot"
"targeting" : {
"ix_ox_id": ["_230l09jd2"],
"ix_ox_cpm": ["300x250_2.50"]
},
"price": 2.50,
"size": [300,250],
"adm": "<script> test creative </script>"
}
This step is only required if your adapter needs to fire a tracking pixel after your creative renders. The function __renderPixel
will be called right after we render your winning creative.
It will be called with the parameter pixelUrl
that needs to be filled out in __parseResponse
.
In this step, you will be required to fill out the exports file for your module. This file will contain all of the functions that will need to be exposed to outside page if they need to be accessed outside of the wrapper. In the usual case, all you will need to change in this file is your partner module's name in the included snippet:
shellInterface.OpenXHtb = { //shell interface is the window variable that is accessible through the window object, currently this will always be window.headertag
render: SpaceCamp.services.RenderService.renderDfpAd.bind(null, 'OpenXHtb')
};
This snippet, exposes your module's render function to the outside world via the window.headertag
namespace.
If your module requires using a custom adResponse callback via Partner.CallbackTypes.ID callback type, that callback will need to be exposed here. Which would look something like this:
if (__directInterface.Layers.PartnersLayer.Partners.OpenXHtb) {
shellInterface.OpenXHtb = shellInterface.OpenXHtb || {};
shellInterface.OpenXHtb.adResponseCallback = __directInterface.Layers.PartnersLayer.Partners.OpenXHtb.adResponseCallback;
}
If your module requires using a custom adResponse callback via Partner.CallbackTypes.NAME callback type, that callback swill need to be exposed here. Which would look something like this:
if (__directInterface.Layers.PartnersLayer.Partners.OpenXHtb) {
shellInterface.OpenXHtb = shellInterface.OpenXHtb || {};
shellInterface.OpenXHtb.adResponseCallbacks = __directInterface.Layers.PartnersLayer.Partners.OpenXHtb.adResponseCallbacks;
}
There are a lot of helper objects available to you in you partner module.
isObject(entity)
- Return true if entity is an object.isArray(obj)
- Return true if obj is an array.isNumber(entity)
- Return true if entity is a number.isString(entity)
- Return true if entity is a string.isBoolean(entity)
- Return true if entity is a boolean.isFunction(entity)
- Return true if entity is a function.isRegex(entity)
- Return true if entity is a regex.isEmpty(entity)
- if entity is a string, return true if string is empty.
- if entity is an object, return true if the object has no properties.
- if entity is an array, return true if the array is empty.
arrayDelete(arr, value)
- Delete given value from an object or array.randomSplice(arr)
- Returns a randomly spliced item from an array.deepCopy(entity)
- Return a deep copy of the entity.mergeObjects(entity1, entity2, ...)
- Takes the first entity and overwrites it with the next entity, returning the final object.mergeArrays(arr1, arr2, ...)
- Merge all of the specified arrays into one and return it.tryCatchWrapper(fn, args, errorMessage, context)
- Wrap the given arguments into a try catch block. Returning a function.isArraySubset(arr1, arr2, matcher)
- Return true ifarr1
is a subset ofarr2
.
now()
- Return the number of milliseconds since 1970/01/01.generateUniqueId(len, charSet)
- Creates a unique identifier of the given length using the optionally specified charset:ALPHANUM
: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789',ALPHA
: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz',ALPHA_UPPER
: 'ABCDEFGHIJKLMNOPQRSTUVWXYZ',ALPHA_LOWER
: 'abcdefghijklmnopqrstuvwxyz',NUM
: '0123456789'
getTimezoneOffset()
- Returns the timezone offset.documentWrite(doc, data)
- doc.write the data using the specified documentdoc
.
arrayToString(arr)
- Returns the string representation of an array in the form of '300x250'.stringToArray(str)
- Return the array rep representation of a string in the form of [300, 250].
getProtocol(httpValue, httpsValue)
- Returndocument.location.protocol
orhttpValue
ifdocument.location.protocol
is http andhttpsValue
ifdocument.location.protocol
is https.isLocalStorageSupported()
- Checks if local storage is supported.getViewportWidth()
- Return viewport width.getViewportHeight()
- Return viewport height.isTopFrame()
- Checks to see if the code is being run in the top frame or iframe.getScreenWidth()
- Returns screen.width.getScreenHeight()
- Returns screen.height.getReferrer()
- Return document.referrer.getPageUrl()
- Return the page's URL.getHostname()
- Return the page's hostname.getNearestEntity(entityName)
- Returns the entity withentityName
in the nearestwindow
scope.createHiddenIFrame(srcUrl, scope)
- Generate a hidden iframe and then append it to the body. Use thesrcUrl
andscope
if provided.
getDeviceType()
- Returns the device type.
apply(rawBid)
- Transform rawBid into the configured format. This includes, rounding/flooring according to the configuration that was used to instantiate the library. The bidTransformConfig is an object of the format:floor
- Minimum acceptable bid price.inputCentsMultiplier
- Multiply input bids by this to get cents. This is identical to bidUnitInCents in__profile
.outputCentsDivisor
- Divide output bids in cents by this.outputPrecision
- Decimal places in output.roundingType
- Should always be 1.buckets
- Buckets specifying rounding steps.
Note that bid transformer instances suitable for DFP targeting and price reporting are already provided via __baseClass._bidTransformers
. It is recommended to use the provided instances as they are sufficient for almost all use cases.
All code must pass the linting before it is submitted for review. Follow the instructions here to run the linter.
To walk through your bidder code and debug, follow the instructions here.
To implement the system tests, follow the instructions here.
Follow the steps here to submit your code for review to Index Exchange.