prebid.js 5.17.0 → 6.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (145) hide show
  1. package/.babelrc.js +3 -6
  2. package/README.md +3 -1
  3. package/browsers.json +1 -8
  4. package/integrationExamples/gpt/akamaidap_segments_example.html +132 -0
  5. package/modules/.submodules.json +1 -0
  6. package/modules/adfBidAdapter.js +21 -16
  7. package/modules/adgenerationBidAdapter.js +28 -4
  8. package/modules/adkernelBidAdapter.js +2 -1
  9. package/modules/admixerBidAdapter.js +11 -0
  10. package/modules/adtelligentBidAdapter.js +2 -1
  11. package/modules/airgridRtdProvider.js +1 -1
  12. package/modules/akamaiDapRtdProvider.js +474 -0
  13. package/modules/akamaiDapRtdProvider.md +47 -0
  14. package/modules/aolBidAdapter.js +2 -1
  15. package/modules/appnexusBidAdapter.js +5 -3
  16. package/modules/atsAnalyticsAdapter.js +67 -46
  17. package/modules/atsAnalyticsAdapter.md +1 -0
  18. package/modules/betweenBidAdapter.js +20 -3
  19. package/modules/bliinkBidAdapter.js +58 -32
  20. package/modules/bliinkBidAdapter.md +29 -6
  21. package/modules/browsiRtdProvider.js +106 -18
  22. package/modules/cleanioRtdProvider.js +192 -0
  23. package/modules/cleanioRtdProvider.md +59 -0
  24. package/modules/codefuelBidAdapter.js +183 -0
  25. package/modules/codefuelBidAdapter.md +111 -0
  26. package/modules/connectIdSystem.js +104 -0
  27. package/modules/connectIdSystem.md +33 -0
  28. package/modules/cwireBidAdapter.js +272 -0
  29. package/modules/cwireBidAdapter.md +43 -0
  30. package/modules/deepintentBidAdapter.js +106 -9
  31. package/modules/deepintentBidAdapter.md +36 -1
  32. package/modules/deltaprojectsBidAdapter.js +252 -0
  33. package/modules/deltaprojectsBidAdapter.md +32 -0
  34. package/modules/dgkeywordRtdProvider.js +0 -1
  35. package/modules/engageyaBidAdapter.js +157 -0
  36. package/modules/gridBidAdapter.js +1 -0
  37. package/modules/gumgumBidAdapter.js +8 -0
  38. package/modules/inskinBidAdapter.js +7 -3
  39. package/modules/ixBidAdapter.js +8 -1
  40. package/modules/jixieBidAdapter.js +8 -2
  41. package/modules/justpremiumBidAdapter.js +6 -1
  42. package/modules/limelightDigitalBidAdapter.js +22 -2
  43. package/modules/livewrappedAnalyticsAdapter.js +53 -3
  44. package/modules/mediakeysBidAdapter.js +2 -1
  45. package/modules/multibid/index.js +3 -3
  46. package/modules/nativoBidAdapter.js +6 -2
  47. package/modules/nextMillenniumBidAdapter.js +12 -3
  48. package/modules/oguryBidAdapter.js +36 -7
  49. package/modules/openxBidAdapter.js +34 -22
  50. package/modules/operaadsBidAdapter.js +21 -1
  51. package/modules/otmBidAdapter.js +146 -0
  52. package/modules/otmBidAdapter.md +27 -26
  53. package/modules/outbrainBidAdapter.js +5 -0
  54. package/modules/pixfutureBidAdapter.js +24 -4
  55. package/modules/pixfutureBidAdapter.md +127 -0
  56. package/modules/playwireBidAdapter.md +61 -0
  57. package/modules/prebidServerBidAdapter/index.js +1 -1
  58. package/modules/proxistoreBidAdapter.js +4 -6
  59. package/modules/publinkIdSystem.js +11 -6
  60. package/modules/pubmaticBidAdapter.js +9 -0
  61. package/modules/pubmaticBidAdapter.md +1 -1
  62. package/modules/rtdModule/index.js +2 -2
  63. package/modules/sonobiBidAdapter.js +7 -0
  64. package/modules/sortableBidAdapter.js +1 -0
  65. package/modules/talkadsBidAdapter.js +129 -0
  66. package/modules/talkadsBidAdapter.md +60 -0
  67. package/modules/teadsBidAdapter.js +3 -0
  68. package/modules/tripleliftBidAdapter.js +22 -5
  69. package/modules/trustxBidAdapter.js +8 -6
  70. package/modules/undertoneBidAdapter.js +9 -5
  71. package/modules/undertoneBidAdapter.md +5 -1
  72. package/modules/unicornBidAdapter.js +3 -3
  73. package/modules/userId/eids.js +18 -0
  74. package/modules/userId/eids.md +7 -0
  75. package/modules/userId/userId.md +12 -0
  76. package/modules/ventesBidAdapter.js +370 -0
  77. package/modules/ventesBidAdapter.md +94 -0
  78. package/modules/videobyteBidAdapter.js +13 -6
  79. package/modules/videobyteBidAdapter.md +49 -0
  80. package/modules/visxBidAdapter.js +15 -22
  81. package/modules/yahoosspBidAdapter.js +637 -0
  82. package/modules/yahoosspBidAdapter.md +795 -0
  83. package/modules/yieldlabBidAdapter.js +48 -3
  84. package/modules/yieldlabBidAdapter.md +16 -1
  85. package/modules/yieldmoSyntheticInventoryModule.js +46 -0
  86. package/modules/yieldmoSyntheticInventoryModule.md +68 -0
  87. package/package.json +1 -1
  88. package/src/adapterManager.js +5 -0
  89. package/src/adapters/bidderFactory.js +4 -3
  90. package/src/auction.js +11 -11
  91. package/src/constants.json +1 -0
  92. package/src/secureCreatives.js +6 -7
  93. package/src/targeting.js +11 -9
  94. package/test/spec/modules/adfBidAdapter_spec.js +83 -29
  95. package/test/spec/modules/adgenerationBidAdapter_spec.js +121 -50
  96. package/test/spec/modules/adtelligentBidAdapter_spec.js +1 -0
  97. package/test/spec/modules/akamaiDapRtdProvider_spec.js +246 -0
  98. package/test/spec/modules/appnexusBidAdapter_spec.js +2 -1
  99. package/test/spec/modules/atsAnalyticsAdapter_spec.js +42 -9
  100. package/test/spec/modules/betweenBidAdapter_spec.js +41 -0
  101. package/test/spec/modules/bliinkBidAdapter_spec.js +87 -36
  102. package/test/spec/modules/browsiRtdProvider_spec.js +62 -7
  103. package/test/spec/modules/cleanioRtdProvider_spec.js +188 -0
  104. package/test/spec/modules/codefuelBidAdapter_spec.js +316 -0
  105. package/test/spec/modules/connectIdSystem_spec.js +189 -0
  106. package/test/spec/modules/cwireBidAdapter_spec.js +246 -0
  107. package/test/spec/modules/deepintentBidAdapter_spec.js +153 -3
  108. package/test/spec/modules/deltaprojectsBidAdapter_spec.js +399 -0
  109. package/test/spec/modules/engageyaBidAdapter_spec.js +286 -0
  110. package/test/spec/modules/gumgumBidAdapter_spec.js +5 -1
  111. package/test/spec/modules/ixBidAdapter_spec.js +13 -3
  112. package/test/spec/modules/jixieBidAdapter_spec.js +13 -11
  113. package/test/spec/modules/justpremiumBidAdapter_spec.js +9 -2
  114. package/test/spec/modules/limelightDigitalBidAdapter_spec.js +155 -1
  115. package/test/spec/modules/livewrappedAnalyticsAdapter_spec.js +67 -12
  116. package/test/spec/modules/multibid_spec.js +31 -31
  117. package/test/spec/modules/nextMillenniumBidAdapter_spec.js +13 -1
  118. package/test/spec/modules/oguryBidAdapter_spec.js +125 -37
  119. package/test/spec/modules/openxBidAdapter_spec.js +85 -13
  120. package/test/spec/modules/operaadsBidAdapter_spec.js +38 -6
  121. package/test/spec/modules/otmBidAdapter_spec.js +67 -0
  122. package/test/spec/modules/outbrainBidAdapter_spec.js +18 -0
  123. package/test/spec/modules/publinkIdSystem_spec.js +6 -6
  124. package/test/spec/modules/pubmaticBidAdapter_spec.js +39 -1
  125. package/test/spec/modules/sonobiBidAdapter_spec.js +34 -1
  126. package/test/spec/modules/sortableBidAdapter_spec.js +11 -0
  127. package/test/spec/modules/talkadsBidAdapter_spec.js +231 -0
  128. package/test/spec/modules/teadsBidAdapter_spec.js +132 -0
  129. package/test/spec/modules/tripleliftBidAdapter_spec.js +128 -0
  130. package/test/spec/modules/trustxBidAdapter_spec.js +3 -3
  131. package/test/spec/modules/undertoneBidAdapter_spec.js +52 -0
  132. package/test/spec/modules/unicornBidAdapter_spec.js +4 -4
  133. package/test/spec/modules/ventesBidAdapter_spec.js +845 -0
  134. package/test/spec/modules/videobyteBidAdapter_spec.js +2 -2
  135. package/test/spec/modules/visxBidAdapter_spec.js +48 -4
  136. package/test/spec/modules/yahoosspBidAdapter_spec.js +1332 -0
  137. package/test/spec/modules/yieldlabBidAdapter_spec.js +65 -1
  138. package/test/spec/modules/yieldmoSyntheticInventoryModule_spec.js +89 -0
  139. package/test/spec/unit/core/adapterManager_spec.js +32 -0
  140. package/test/spec/unit/core/bidderFactory_spec.js +61 -1
  141. package/test/spec/unit/pbjs_api_spec.js +37 -2
  142. package/test/spec/unit/secureCreatives_spec.js +54 -25
  143. package/wdio.conf.js +1 -1
  144. package/modules/turktelekomBidAdapter.md +0 -49
  145. package/yarn.lock +0 -13122
@@ -20,7 +20,7 @@ export const analyticsUrl = 'https://analytics.rlcdn.com';
20
20
  let handlerRequest = [];
21
21
  let handlerResponse = [];
22
22
 
23
- let atsAnalyticsAdapterVersion = 2;
23
+ let atsAnalyticsAdapterVersion = 3;
24
24
 
25
25
  let browsersList = [
26
26
  /* Googlebot */
@@ -222,7 +222,8 @@ function bidRequestedHandler(args) {
222
222
  auction_start: new Date(args.auctionStart).toJSON(),
223
223
  domain: window.location.hostname,
224
224
  pid: atsAnalyticsAdapter.context.pid,
225
- adapter_version: atsAnalyticsAdapterVersion
225
+ adapter_version: atsAnalyticsAdapterVersion,
226
+ bid_won: false
226
227
  };
227
228
  });
228
229
  return requests;
@@ -251,13 +252,14 @@ export function parseBrowser() {
251
252
  }
252
253
  }
253
254
 
254
- function sendDataToAnalytic () {
255
+ function sendDataToAnalytic (events) {
255
256
  // send data to ats analytic endpoint
256
257
  try {
257
- let dataToSend = {'Data': atsAnalyticsAdapter.context.events};
258
+ let dataToSend = {'Data': events};
258
259
  let strJSON = JSON.stringify(dataToSend);
259
260
  logInfo('ATS Analytics - tried to send analytics data!');
260
261
  ajax(analyticsUrl, function () {
262
+ logInfo('ATS Analytics - events sent successfully!');
261
263
  }, strJSON, {method: 'POST', contentType: 'application/json'});
262
264
  } catch (err) {
263
265
  logError('ATS Analytics - request encounter an error: ', err);
@@ -265,7 +267,7 @@ function sendDataToAnalytic () {
265
267
  }
266
268
 
267
269
  // preflight request, to check did publisher have permission to send data to analytics endpoint
268
- function preflightRequest (envelopeSourceCookieValue) {
270
+ function preflightRequest (envelopeSourceCookieValue, events) {
269
271
  logInfo('ATS Analytics - preflight request!');
270
272
  ajax(preflightUrl + atsAnalyticsAdapter.context.pid,
271
273
  {
@@ -276,7 +278,8 @@ function preflightRequest (envelopeSourceCookieValue) {
276
278
  atsAnalyticsAdapter.setSamplingCookie(samplingRate);
277
279
  let samplingRateNumber = Number(samplingRate);
278
280
  if (data && samplingRate && atsAnalyticsAdapter.shouldFireRequest(samplingRateNumber) && envelopeSourceCookieValue != null) {
279
- sendDataToAnalytic();
281
+ logInfo('ATS Analytics - events to send: ', events);
282
+ sendDataToAnalytic(events);
280
283
  }
281
284
  },
282
285
  error: function () {
@@ -286,29 +289,6 @@ function preflightRequest (envelopeSourceCookieValue) {
286
289
  }, undefined, {method: 'GET', crossOrigin: true});
287
290
  }
288
291
 
289
- function callHandler(evtype, args) {
290
- if (evtype === CONSTANTS.EVENTS.BID_REQUESTED) {
291
- handlerRequest = handlerRequest.concat(bidRequestedHandler(args));
292
- } else if (evtype === CONSTANTS.EVENTS.BID_RESPONSE) {
293
- handlerResponse.push(bidResponseHandler(args));
294
- }
295
- if (evtype === CONSTANTS.EVENTS.AUCTION_END) {
296
- if (handlerRequest.length) {
297
- let events = [];
298
- if (handlerResponse.length) {
299
- events = handlerRequest.filter(request => handlerResponse.filter(function(response) {
300
- if (request.bid_id === response.bid_id) {
301
- Object.assign(request, response);
302
- }
303
- }));
304
- } else {
305
- events = handlerRequest;
306
- }
307
- atsAnalyticsAdapter.context.events = events;
308
- }
309
- }
310
- }
311
-
312
292
  let atsAnalyticsAdapter = Object.assign(adapter(
313
293
  {
314
294
  analyticsType
@@ -316,22 +296,7 @@ let atsAnalyticsAdapter = Object.assign(adapter(
316
296
  {
317
297
  track({eventType, args}) {
318
298
  if (typeof args !== 'undefined') {
319
- callHandler(eventType, args);
320
- }
321
- if (eventType === CONSTANTS.EVENTS.AUCTION_END) {
322
- let envelopeSourceCookieValue = storage.getCookie('_lr_env_src_ats');
323
- try {
324
- let samplingRateCookie = storage.getCookie('_lr_sampling_rate');
325
- if (!samplingRateCookie) {
326
- preflightRequest(envelopeSourceCookieValue);
327
- } else {
328
- if (atsAnalyticsAdapter.shouldFireRequest(parseInt(samplingRateCookie)) && envelopeSourceCookieValue != null) {
329
- sendDataToAnalytic();
330
- }
331
- }
332
- } catch (err) {
333
- logError('ATS Analytics - preflight request encounter an error: ', err);
334
- }
299
+ atsAnalyticsAdapter.callHandler(eventType, args);
335
300
  }
336
301
  }
337
302
  });
@@ -369,13 +334,69 @@ atsAnalyticsAdapter.enableAnalytics = function (config) {
369
334
  }
370
335
  atsAnalyticsAdapter.context = {
371
336
  events: [],
372
- pid: config.options.pid
337
+ pid: config.options.pid,
338
+ bidWonTimeout: config.options.bidWonTimeout
373
339
  };
374
340
  let initOptions = config.options;
375
341
  logInfo('ATS Analytics - adapter enabled! ');
376
342
  atsAnalyticsAdapter.originEnableAnalytics(initOptions); // call the base class function
377
343
  };
378
344
 
345
+ atsAnalyticsAdapter.callHandler = function (evtype, args) {
346
+ if (evtype === CONSTANTS.EVENTS.BID_REQUESTED) {
347
+ handlerRequest = handlerRequest.concat(bidRequestedHandler(args));
348
+ } else if (evtype === CONSTANTS.EVENTS.BID_RESPONSE) {
349
+ handlerResponse.push(bidResponseHandler(args));
350
+ }
351
+ if (evtype === CONSTANTS.EVENTS.AUCTION_END) {
352
+ let bidWonTimeout = atsAnalyticsAdapter.context.bidWonTimeout ? atsAnalyticsAdapter.context.bidWonTimeout : 2000;
353
+ let events = [];
354
+ setTimeout(() => {
355
+ let winningBids = $$PREBID_GLOBAL$$.getAllWinningBids();
356
+ logInfo('ATS Analytics - winning bids: ', winningBids)
357
+ // prepare format data for sending to analytics endpoint
358
+ if (handlerRequest.length) {
359
+ let wonEvent = {};
360
+ if (handlerResponse.length) {
361
+ events = handlerRequest.filter(request => handlerResponse.filter(function (response) {
362
+ if (request.bid_id === response.bid_id) {
363
+ Object.assign(request, response);
364
+ }
365
+ }));
366
+ if (winningBids.length) {
367
+ events = events.filter(event => winningBids.filter(function (won) {
368
+ wonEvent.bid_id = won.requestId;
369
+ wonEvent.bid_won = true;
370
+ if (event.bid_id === wonEvent.bid_id) {
371
+ Object.assign(event, wonEvent);
372
+ }
373
+ }))
374
+ }
375
+ } else {
376
+ events = handlerRequest;
377
+ }
378
+ // check should we send data to analytics or not, check first cookie value _lr_sampling_rate
379
+ try {
380
+ let envelopeSourceCookieValue = storage.getCookie('_lr_env_src_ats');
381
+ let samplingRateCookie = storage.getCookie('_lr_sampling_rate');
382
+ if (!samplingRateCookie) {
383
+ preflightRequest(envelopeSourceCookieValue, events);
384
+ } else {
385
+ if (atsAnalyticsAdapter.shouldFireRequest(parseInt(samplingRateCookie)) && envelopeSourceCookieValue != null) {
386
+ logInfo('ATS Analytics - events to send: ', events);
387
+ sendDataToAnalytic(events);
388
+ }
389
+ }
390
+ // empty events array to not send duplicate events
391
+ events = [];
392
+ } catch (err) {
393
+ logError('ATS Analytics - preflight request encounter an error: ', err);
394
+ }
395
+ }
396
+ }, bidWonTimeout);
397
+ }
398
+ }
399
+
379
400
  adaptermanager.registerAnalyticsAdapter({
380
401
  adapter: atsAnalyticsAdapter,
381
402
  code: 'atsAnalytics',
@@ -17,6 +17,7 @@ Analytics adapter for Authenticated Traffic Solution(ATS), provided by LiveRamp.
17
17
  provider: 'atsAnalytics',
18
18
  options: {
19
19
  pid: '999', // publisher ID
20
+ bidWonTimeout: 2000 // on auction end for how long to wait for bid_won events, by default it's 2000 miliseconds, if it's not set it will be 2000 miliseconds.
20
21
  }
21
22
  }
22
23
  ```
@@ -3,12 +3,14 @@ import { getAdUnitSizes, parseSizesInput } from '../src/utils.js';
3
3
  import { getRefererInfo } from '../src/refererDetection.js';
4
4
 
5
5
  const BIDDER_CODE = 'between';
6
- const ENDPOINT = 'https://ads.betweendigital.com/adjson?t=prebid';
6
+ let ENDPOINT = 'https://ads.betweendigital.com/adjson?t=prebid';
7
+ const CODE_TYPES = ['inpage', 'preroll', 'midroll', 'postroll'];
7
8
 
9
+ const includes = require('core-js-pure/features/array/includes.js');
8
10
  export const spec = {
9
11
  code: BIDDER_CODE,
10
12
  aliases: ['btw'],
11
- supportedMediaTypes: ['banner'],
13
+ supportedMediaTypes: ['banner', 'video'],
12
14
  /**
13
15
  * Determines whether or not the given bid request is valid.
14
16
  *
@@ -30,6 +32,8 @@ export const spec = {
30
32
  const refInfo = getRefererInfo();
31
33
 
32
34
  validBidRequests.forEach((i) => {
35
+ const video = i.mediaTypes && i.mediaTypes.video;
36
+
33
37
  let params = {
34
38
  eids: getUsersIds(i),
35
39
  sizes: parseSizesInput(getAdUnitSizes(i)),
@@ -38,12 +42,21 @@ export const spec = {
38
42
  tz: getTz(),
39
43
  fl: getFl(),
40
44
  rr: getRr(),
41
- s: i.params.s,
45
+ s: i.params && i.params.s,
42
46
  bidid: i.bidId,
43
47
  transactionid: i.transactionId,
44
48
  auctionid: i.auctionId
45
49
  };
46
50
 
51
+ if (video) {
52
+ params.mediaType = 2;
53
+ params.maxd = video.maxd;
54
+ params.mind = video.mind;
55
+ params.pos = 'atf';
56
+ ENDPOINT += '&jst=pvc';
57
+ params.codeType = includes(CODE_TYPES, video.codeType) ? video.codeType : 'inpage';
58
+ }
59
+
47
60
  if (i.params.itu !== undefined) {
48
61
  params.itu = i.params.itu;
49
62
  }
@@ -94,12 +107,15 @@ export const spec = {
94
107
  */
95
108
  interpretResponse: function(serverResponse, bidRequest) {
96
109
  const bidResponses = [];
110
+
97
111
  for (var i = 0; i < serverResponse.body.length; i++) {
98
112
  let bidResponse = {
99
113
  requestId: serverResponse.body[i].bidid,
100
114
  cpm: serverResponse.body[i].cpm || 0,
101
115
  width: serverResponse.body[i].w,
102
116
  height: serverResponse.body[i].h,
117
+ vastXml: serverResponse.body[i].vastXml,
118
+ mediaType: serverResponse.body[i].mediaType,
103
119
  ttl: serverResponse.body[i].ttl,
104
120
  creativeId: serverResponse.body[i].creativeid,
105
121
  currency: serverResponse.body[i].currency || 'RUB',
@@ -109,6 +125,7 @@ export const spec = {
109
125
  advertiserDomains: serverResponse.body[i].adomain ? serverResponse.body[i].adomain : []
110
126
  }
111
127
  };
128
+
112
129
  bidResponses.push(bidResponse);
113
130
  }
114
131
  return bidResponses;
@@ -10,10 +10,9 @@ export const META_KEYWORDS = 'keywords'
10
10
  export const META_DESCRIPTION = 'description'
11
11
 
12
12
  const VIDEO = 'video'
13
- const NATIVE = 'native'
14
13
  const BANNER = 'banner'
15
14
 
16
- const supportedMediaTypes = [BANNER, VIDEO, NATIVE]
15
+ const supportedMediaTypes = [BANNER, VIDEO]
17
16
  const aliasBidderCode = ['bk']
18
17
 
19
18
  export function getMetaList(name) {
@@ -90,7 +89,11 @@ export const parseXML = (content) => {
90
89
  if (typeof content !== 'string' || content.length === 0) return null
91
90
 
92
91
  const parser = new DOMParser()
93
- const xml = parser.parseFromString(content, 'text/xml')
92
+ let xml;
93
+
94
+ try {
95
+ xml = parser.parseFromString(content, 'text/xml')
96
+ } catch (e) {}
94
97
 
95
98
  if (xml &&
96
99
  xml.getElementsByTagName('VAST')[0] &&
@@ -104,19 +107,19 @@ export const parseXML = (content) => {
104
107
  /**
105
108
  * @param bidRequest
106
109
  * @param bliinkCreative
107
- * @return {{cpm, netRevenue: boolean, ad: string, requestId, width: number, currency: string, mediaType: string, vastXml, ttl: number, height: number}|null}
110
+ * @return {{cpm, netRevenue: boolean, requestId, width: (*|number), currency, ttl: number, creativeId, height: (*|number)} & {mediaType: string, vastXml}}
108
111
  */
109
112
  export const buildBid = (bidRequest, bliinkCreative) => {
110
113
  if (!bidRequest && !bliinkCreative) return null
111
114
 
112
115
  const body = {
113
116
  requestId: bidRequest.bidId,
117
+ currency: bliinkCreative.currency,
114
118
  cpm: bliinkCreative.price,
115
119
  creativeId: bliinkCreative.creativeId,
116
- currency: 'EUR',
120
+ width: (bidRequest.sizes && bidRequest.sizes[0][0]) || 1,
121
+ height: (bidRequest.sizes && bidRequest.sizes[0][1]) || 1,
117
122
  netRevenue: false,
118
- width: 1,
119
- height: 1,
120
123
  ttl: 3600,
121
124
  }
122
125
 
@@ -131,14 +134,20 @@ export const buildBid = (bidRequest, bliinkCreative) => {
131
134
 
132
135
  delete bidRequest['bids']
133
136
 
134
- return Object.assign(body, {
135
- currency: bliinkCreative.currency,
136
- width: 1,
137
- height: 1,
138
- mediaType: VIDEO,
139
- ad: '<html lang="en"></html>',
140
- vastXml: bliinkCreative.content,
141
- })
137
+ switch (bliinkCreative.media_type) {
138
+ case VIDEO:
139
+ return Object.assign(body, {
140
+ mediaType: VIDEO,
141
+ vastXml: bliinkCreative.content,
142
+ })
143
+ case BANNER:
144
+ return Object.assign(body, {
145
+ mediaType: BANNER,
146
+ ad: (bliinkCreative && bliinkCreative.content && bliinkCreative.content.creative && bliinkCreative.content.creative.adm) || '',
147
+ })
148
+ default:
149
+ break;
150
+ }
142
151
  }
143
152
 
144
153
  /**
@@ -209,7 +218,7 @@ export const buildRequests = (_, bidderRequest) => {
209
218
  * @return
210
219
  */
211
220
  const interpretResponse = (serverResponse, request) => {
212
- if ((serverResponse && serverResponse.mode === 'no-ad') && (!request.params)) {
221
+ if ((serverResponse && serverResponse.mode === 'no-ad')) {
213
222
  return []
214
223
  }
215
224
 
@@ -218,23 +227,40 @@ const interpretResponse = (serverResponse, request) => {
218
227
 
219
228
  const xml = parseXML(body)
220
229
 
221
- if (xml) {
222
- const price = xml.getElementsByTagName('Price') && xml.getElementsByTagName('Price')[0]
223
- const currency = xml.getElementsByTagName('Currency') && xml.getElementsByTagName('Currency')[0]
224
- const creativeId = xml.getElementsByTagName('CreativeId') && xml.getElementsByTagName('CreativeId')[0]
225
-
226
- const creative = {
227
- content: body,
228
- price: (price && price.textContent) || 0,
229
- currency: (currency && currency.textContent) || 'EUR',
230
- creativeId: creativeId || 0,
231
- media_type: 'video',
232
- }
233
-
234
- return buildBid(serverBody.bids[0], creative);
230
+ let creative;
231
+
232
+ switch (serverBody.bids[0].params.placement) {
233
+ case xml && VIDEO:
234
+ const price = xml.getElementsByTagName('Price') && xml.getElementsByTagName('Price')[0]
235
+ const currency = xml.getElementsByTagName('Currency') && xml.getElementsByTagName('Currency')[0]
236
+ const creativeId = xml.getElementsByTagName('CreativeId') && xml.getElementsByTagName('CreativeId')[0]
237
+
238
+ creative = {
239
+ content: body,
240
+ price: (price && price.textContent) || 0,
241
+ currency: (currency && currency.textContent) || 'EUR',
242
+ creativeId: creativeId || 0,
243
+ media_type: 'video',
244
+ }
245
+
246
+ return buildBid(serverBody.bids[0], creative)
247
+ case BANNER:
248
+ if (body) {
249
+ creative = {
250
+ content: body,
251
+ price: body.price,
252
+ currency: body.currency,
253
+ creativeId: 0,
254
+ media_type: 'banner',
255
+ }
256
+
257
+ return buildBid(serverBody.bids[0], creative)
258
+ }
259
+
260
+ break
261
+ default:
262
+ break
235
263
  }
236
-
237
- return []
238
264
  }
239
265
 
240
266
  /**
@@ -31,7 +31,7 @@ const adUnits = [
31
31
  bidder: 'bliink',
32
32
  params: {
33
33
  placement: 'banner',
34
- tagId: '14f30eca-85d2-11e8-9eed-0242ac120007'
34
+ tagId: '41'
35
35
  }
36
36
  }
37
37
  ]
@@ -50,11 +50,34 @@ const adUnits = [
50
50
  mediaTypes: {
51
51
  video: {
52
52
  context: 'instream',
53
- playerSize: [640, 480],
54
- mimes: ['video/mp4'],
55
- protocols: [1, 2, 3, 4, 5, 6, 7, 8],
56
- playbackmethod: [2],
57
- skip: 1
53
+ playerSize: [[640,480]],
54
+ }
55
+ },
56
+ bids: [
57
+ {
58
+ bidder: 'bliink',
59
+ params: {
60
+ tagId: '41',
61
+ placement: 'video',
62
+ }
63
+ }
64
+ ]
65
+ }
66
+ ]
67
+ ```
68
+
69
+ ## Sample outstream Video Ad Unit
70
+
71
+ ```js
72
+ const adUnits = [
73
+ {
74
+ code: '/19968336/prebid_cache_video_adunit',
75
+ sizes: [[640,480]],
76
+ mediaType: 'video',
77
+ mediaTypes: {
78
+ video: {
79
+ context: 'outstream',
80
+ playerSize: [[640,480]],
58
81
  }
59
82
  },
60
83
  bids: [
@@ -15,21 +15,27 @@
15
15
  * @property {?string} keyName
16
16
  */
17
17
 
18
- import { deepClone, logError, isGptPubadsDefined } from '../src/utils.js';
18
+ import { deepClone, logError, isGptPubadsDefined, isNumber, isFn, deepSetValue } from '../src/utils.js';
19
19
  import {submodule} from '../src/hook.js';
20
20
  import {ajaxBuilder} from '../src/ajax.js';
21
21
  import {loadExternalScript} from '../src/adloader.js';
22
22
  import {getStorageManager} from '../src/storageManager.js';
23
23
  import find from 'core-js-pure/features/array/find.js';
24
+ import {getGlobal} from '../src/prebidGlobal.js';
25
+ import includes from 'core-js-pure/features/array/includes.js';
24
26
 
25
27
  const storage = getStorageManager();
26
28
 
27
29
  /** @type {ModuleParams} */
28
30
  let _moduleParams = {};
29
31
  /** @type {null|Object} */
30
- let _predictionsData = null;
32
+ let _browsiData = null;
31
33
  /** @type {string} */
32
34
  const DEF_KEYNAME = 'browsiViewability';
35
+ /** @type {null | function} */
36
+ let _dataReadyCallback = null;
37
+ /** @type {null|Object} */
38
+ let _ic = {};
33
39
 
34
40
  /**
35
41
  * add browsi script to page
@@ -78,29 +84,49 @@ export function collectData() {
78
84
  getPredictionsFromServer(`//${_moduleParams.url}/prebid?${toUrlParams(predictorData)}`);
79
85
  }
80
86
 
87
+ /**
88
+ * wait for data from server
89
+ * call callback when data is ready
90
+ * @param {function} callback
91
+ */
92
+ function waitForData(callback) {
93
+ if (_browsiData) {
94
+ _dataReadyCallback = null;
95
+ callback(_browsiData);
96
+ } else {
97
+ _dataReadyCallback = callback;
98
+ }
99
+ }
100
+
81
101
  export function setData(data) {
82
- _predictionsData = data;
102
+ _browsiData = data;
103
+ if (isFn(_dataReadyCallback)) {
104
+ _dataReadyCallback(_browsiData);
105
+ _dataReadyCallback = null;
106
+ }
83
107
  }
84
108
 
85
- function sendDataToModule(adUnitsCodes) {
109
+ function getRTD(auc) {
86
110
  try {
87
- const _predictions = (_predictionsData && _predictionsData.p) || {};
88
- return adUnitsCodes.reduce((rp, adUnitCode) => {
89
- if (!adUnitCode) {
111
+ const _bp = (_browsiData && _browsiData.p) || {};
112
+ return auc.reduce((rp, uc) => {
113
+ _ic[uc] = _ic[uc] || 0;
114
+ const _c = _ic[uc];
115
+ if (!uc) {
90
116
  return rp
91
117
  }
92
- const adSlot = getSlotByCode(adUnitCode);
93
- const identifier = adSlot ? getMacroId(_predictionsData['pmd'], adSlot) : adUnitCode;
94
- const predictionData = _predictions[identifier];
95
- rp[adUnitCode] = getKVObject(-1, _predictionsData['kn']);
96
- if (!predictionData) {
118
+ const adSlot = getSlotByCode(uc);
119
+ const identifier = adSlot ? getMacroId(_browsiData['pmd'], adSlot) : uc;
120
+ const _pd = _bp[identifier];
121
+ rp[uc] = getKVObject(-1);
122
+ if (!_pd) {
97
123
  return rp
98
124
  }
99
- if (predictionData.p) {
100
- if (!isIdMatchingAdUnit(adSlot, predictionData.w)) {
125
+ if (_pd.ps) {
126
+ if (!isIdMatchingAdUnit(adSlot, _pd.w)) {
101
127
  return rp;
102
128
  }
103
- rp[adUnitCode] = getKVObject(predictionData.p, _predictionsData.kn);
129
+ rp[uc] = getKVObject(getCurrentData(_pd.ps, _c));
104
130
  }
105
131
  return rp;
106
132
  }, {});
@@ -109,6 +135,31 @@ function sendDataToModule(adUnitsCodes) {
109
135
  }
110
136
  }
111
137
 
138
+ /**
139
+ * get prediction
140
+ * return -1 if prediction not found
141
+ * @param {object} predictionObject
142
+ * @param {number} _c
143
+ * @return {number}
144
+ */
145
+ export function getCurrentData(predictionObject, _c) {
146
+ if (!predictionObject || !isNumber(_c)) {
147
+ return -1;
148
+ }
149
+ if (isNumber(predictionObject[_c])) {
150
+ return predictionObject[_c];
151
+ }
152
+ if (Object.keys(predictionObject).length > 1) {
153
+ while (_c > 0) {
154
+ _c--;
155
+ if (isNumber(predictionObject[_c])) {
156
+ return predictionObject[_c];
157
+ }
158
+ }
159
+ }
160
+ return -1;
161
+ }
162
+
112
163
  /**
113
164
  * get all slots on page
114
165
  * @return {Object[]} slot GoogleTag slots
@@ -122,12 +173,16 @@ function getAllSlots() {
122
173
  * @param {string?} keyName
123
174
  * @return {Object} key:value
124
175
  */
125
- function getKVObject(p, keyName) {
176
+ function getKVObject(p) {
126
177
  const prValue = p < 0 ? 'NA' : (Math.floor(p * 10) / 10).toFixed(2);
127
178
  let prObject = {};
128
- prObject[((_moduleParams['keyName'] || keyName || DEF_KEYNAME).toString())] = prValue.toString();
179
+ prObject[getKey()] = prValue.toString();
129
180
  return prObject;
130
181
  }
182
+
183
+ function getKey() {
184
+ return ((_moduleParams['keyName'] || (_browsiData && _browsiData['kn']) || DEF_KEYNAME).toString())
185
+ }
131
186
  /**
132
187
  * check if placement id matches one of given ad units
133
188
  * @param {Object} slot google slot
@@ -238,6 +293,28 @@ function toUrlParams(data) {
238
293
  .join('&');
239
294
  }
240
295
 
296
+ function setBidRequestsData(bidObj, callback) {
297
+ let adUnitCodes = bidObj.adUnitCodes;
298
+ let adUnits = bidObj.adUnits || getGlobal().adUnits || [];
299
+ if (adUnitCodes) {
300
+ adUnits = adUnits.filter(au => includes(adUnitCodes, au.code));
301
+ } else {
302
+ adUnitCodes = adUnits.map(au => au.code);
303
+ }
304
+ waitForData(() => {
305
+ const data = getRTD(adUnitCodes);
306
+ if (data) {
307
+ adUnits.forEach(adUnit => {
308
+ const adUnitCode = adUnit.code;
309
+ if (data[adUnitCode]) {
310
+ deepSetValue(adUnit, 'ortb2Imp.ext.data.browsi', {[getKey()]: data[adUnitCode][getKey()]});
311
+ }
312
+ });
313
+ }
314
+ callback();
315
+ })
316
+ }
317
+
241
318
  /** @type {RtdSubmodule} */
242
319
  export const browsiSubmodule = {
243
320
  /**
@@ -250,10 +327,21 @@ export const browsiSubmodule = {
250
327
  * @function
251
328
  * @param {string[]} adUnitsCodes
252
329
  */
253
- getTargetingData: sendDataToModule,
330
+ getTargetingData: getTargetingData,
254
331
  init: init,
332
+ getBidRequestData: setBidRequestsData
255
333
  };
256
334
 
335
+ function getTargetingData(uc) {
336
+ const targetingData = getRTD(uc);
337
+ uc.forEach(auc => {
338
+ if (isNumber(_ic[auc])) {
339
+ _ic[auc] = _ic[auc] + 1;
340
+ }
341
+ });
342
+ return targetingData;
343
+ }
344
+
257
345
  function init(moduleConfig) {
258
346
  _moduleParams = moduleConfig.params;
259
347
  if (_moduleParams && _moduleParams.siteKey && _moduleParams.pubKey && _moduleParams.url) {