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

Feature/nielsen dcr #45

Merged
merged 55 commits into from
Sep 2, 2024
Merged
Show file tree
Hide file tree
Changes from 54 commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
8ff3ca1
Add tested countries
wjoosen Aug 19, 2024
7fbdeb7
Load separate static queue snippet for different country configurations
wjoosen Aug 19, 2024
b9208da
appId placeholder string
wjoosen Aug 19, 2024
8541e1b
Reference to CZ docs
wjoosen Aug 19, 2024
9516b27
Rename DTVR Content Metadata
wjoosen Aug 19, 2024
9352600
Add example content metadata object for CZ DCR
wjoosen Aug 19, 2024
04bdf84
Add ContentMetadata types split up in generic, US and CZ specific pro…
wjoosen Aug 19, 2024
1d5dc5a
Add CZ Ad metadata type
wjoosen Aug 20, 2024
38c5740
Add configuration type
wjoosen Aug 20, 2024
083b2ca
pass country configuration to static queue snippet loading function
wjoosen Aug 20, 2024
9b1d992
store configuration parameters in handler
wjoosen Aug 20, 2024
6d052ee
updateMetadata improvements
wjoosen Aug 21, 2024
ca2de34
introduce DCRContentMetadata field
wjoosen Aug 21, 2024
b6bc9a6
tweak if statement
wjoosen Aug 21, 2024
e81ae1a
only send onLoadMetadata with dtvr metadata if dtvr is enabled
wjoosen Aug 21, 2024
059f61f
only send play event if dtvr is enabled
wjoosen Aug 21, 2024
8cc288e
send loadMetadata event for DCR on first play event
wjoosen Aug 21, 2024
a12d29f
adjust types
wjoosen Aug 21, 2024
0487a5f
separate ad metadata reporting logic for dcr and dtvr
wjoosen Aug 21, 2024
a95ac85
set initial adMetadata io placeholder values
wjoosen Aug 22, 2024
05c7477
take first array item of filtered currentAds and add TODO note
wjoosen Aug 22, 2024
50f7b9f
import
wjoosen Aug 22, 2024
23cf5c7
Report playhead position
wjoosen Aug 22, 2024
b4a7ca4
only report playhead position when DCR is enabled
wjoosen Aug 22, 2024
bdae3eb
dont listen for timed metadata if DTVR is not enabled
wjoosen Aug 22, 2024
0713d95
send stop event when an ad ends
wjoosen Aug 22, 2024
4e67802
report end before post roll starts
wjoosen Aug 22, 2024
8a9fc8e
typo
wjoosen Aug 22, 2024
235f9ae
add and remove listener
wjoosen Aug 22, 2024
3a7f857
rewrite
wjoosen Aug 22, 2024
62a787b
only report setVolume for DTVR
wjoosen Aug 22, 2024
520838c
remove comments
wjoosen Aug 22, 2024
5918b31
ignore NaN duration change events
wjoosen Aug 22, 2024
2447323
detect post rolls for DAI sources too
wjoosen Aug 22, 2024
85b4c30
report stop if an ad was playing in the onEnd handler
wjoosen Aug 22, 2024
62ef2fc
report stop on more interrupt scenarios
wjoosen Aug 22, 2024
824079a
adapt build ad metadata helper to new getAdType implementation
wjoosen Aug 22, 2024
2f8dec3
Add public DCR metadata types
wjoosen Aug 22, 2024
5fce7b8
convert provided metadata to types imposed by nielsen dcr docs
wjoosen Aug 22, 2024
66b72c5
expose updateDCRContentMetadata
wjoosen Aug 22, 2024
d0edda9
add configuration parameter
wjoosen Aug 22, 2024
674e01a
add import
wjoosen Aug 22, 2024
f1cae07
edit paths
wjoosen Aug 22, 2024
0e5d9bd
edit librarylocation path
wjoosen Aug 22, 2024
0f55a07
edit serve script
wjoosen Aug 22, 2024
84afa8d
set sessionInProgress to true regardless of DCR/DTVR config
wjoosen Aug 22, 2024
e3f6673
use Ad from AdBeginEvent
wjoosen Aug 22, 2024
2448e81
update README
wjoosen Aug 22, 2024
7f6391c
typo
wjoosen Aug 22, 2024
1005a44
add necessary type exports
wjoosen Aug 22, 2024
8abc60c
finish example page
wjoosen Aug 22, 2024
e0e5fd4
Add changeset
wjoosen Aug 22, 2024
f1cbb0d
edit example metadata
wjoosen Aug 22, 2024
ecaa04b
always include nol_c2 property
wjoosen Aug 22, 2024
44bc3e5
only print logs if nol_sdkDebug === "debug"
wjoosen Aug 29, 2024
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
5 changes: 5 additions & 0 deletions .changeset/chilly-yaks-bake.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@theoplayer/nielsen-connector-web": minor
---

Add DCR support (CZ and US).
66 changes: 53 additions & 13 deletions nielsen/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,39 +12,79 @@ npm install @theoplayer/nielsen-connector-web

### Configuring the connector

Create the connector by providing the `THEOplayer` instance, the Nielsen App ID, the channelName for the asset
and optionally some Nielsen configuration.
Create the connector by providing the following mandatory parameters:

- The `THEOplayer` instance
- the Nielsen App ID
- the channelName for the asset

and optionally the following parameters:

- a `NielsenOptions` object
- a `NielsenConfiguration` object (if none is provided, the default configuration disables DCR, enables DTVR and sets the country to US)

```js
import {NielsenConnector} from "../../dist/THEOplayerNielsenConnector";
import { NielsenConnector } from '../../dist/THEOplayerNielsenConnector';

const appId = '<your app ID>';
const channelName = '<your channel name>';
// Optional

// Non-mandatory options
const options: NielsenOptions = {
containerId: 'THEOplayer',
containerId: 'THEOplayer',
optout: false
}
};

// Non-mandatory configuration (e.g. for DCR tracking with the Czech Republic SDK)
const configuration: NielsenConfiguration = {
country: NielsenCountry.CZ,
enableDTVR: false,
enableDCR: true
};
const nielsenConnector = new NielsenConnector(player, appId, channelName, options);
```

The `NielsenOptions` can have the following fields:

| Key | Value |
|-----------------|-----------------------------------------------------------------|
| ` containerId ` | HTML DOM element id of the player container. |
| --------------- | --------------------------------------------------------------- |
| `containerId` | HTML DOM element id of the player container. |
| ` nol_sdkDebug` | Enables Debug Mode which allows output to be viewed in console. |
| ` optout ` | Whether to opt-out of Nielsen Measurement. |
| `optout` | Whether to opt-out of Nielsen Measurement. |

### Passing metadata dynamically
### Passing metadata dynamically (DTVR)

The connector allows updating the current asset's metadata at any time:
The connector allows updating the current asset's metadata at any time. Note that Nielsen's [documentation](<https://engineeringportal.nielsen.com/wiki/updateMetadata_(Browser)>) prohibits updating of the values for `type`, `vidtype` or `assetid` parameters

```js
const metadata = {
['channelName']: 'newChannelName',
['customTag1']: 'customValue1',
['customTag2']: 'customValue2'
}
['customTag2']: 'customValue2'
};
nielsenConnector.updateMetadata(metadata);
```

### Passing metadata when setting a source to the player (DCR)

This can be achieved through the `updateDCRContentMetadata` method, e.g.:

```js
const metadata: NielsenDCRContentMetadataCZ = {
assetid: 'cz-500358-98731568435405',
program: 'Animated Test Content',
title: 'Big Buck Bunny',
length: '596',
airdate: '20230620 20:00:00',
isfullepisode: true,
crossId1: '915 954 39504',
c2: '651678089925925',
segB: '011',
adloadtype: AdLoadType.linear,
hasAds: HasAds.supports_ads
};

nielsenConnector.updateDCRContentMetadata(metadata);
```

Note that types are included in the package: `NielsenDCRContentMetadataUS`, `NielsenDCRContentMetadataCZ`. Please contact your THEO Technologies representative if you need support for another International DCR SDK.
2 changes: 1 addition & 1 deletion nielsen/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
"bundle": "rollup -c rollup.config.mjs",
"watch": "npm run bundle -- --watch",
"build": "npm run clean && npm run bundle",
"serve": "http-server",
"serve": "http-server ./.. -o /nielsen/test/pages/main.html",
"test": "jest"
},
"author": "THEO Technologies NV",
Expand Down
11 changes: 10 additions & 1 deletion nielsen/src/index.ts
Original file line number Diff line number Diff line change
@@ -1,2 +1,11 @@
export { NielsenConnector } from './integration/NielsenConnector';
export { NielsenOptions } from './nielsen/Types';
export {
NielsenOptions,
NielsenDCRContentMetadata,
NielsenDCRContentMetadataCZ,
NielsenDCRContentMetadataUS,
NielsenConfiguration,
NielsenCountry,
AdLoadType,
HasAds
} from './nielsen/Types';
16 changes: 13 additions & 3 deletions nielsen/src/integration/NielsenConnector.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import type { ChromelessPlayer } from 'theoplayer';
import { NielsenOptions } from '../nielsen/Types';
import { NielsenConfiguration, NielsenDCRContentMetadata, NielsenOptions } from '../nielsen/Types';
import { NielsenHandler } from './NielsenHandler';

export class NielsenConnector {
Expand All @@ -13,14 +13,24 @@ export class NielsenConnector {
* @param instanceName User-defined string value for describing the player/site.
* @param options Additional options.
*/
constructor(player: ChromelessPlayer, appId: string, instanceName: string, options?: NielsenOptions) {
this.nielsenHandler = new NielsenHandler(player, appId, instanceName, options);
constructor(
player: ChromelessPlayer,
appId: string,
instanceName: string,
options?: NielsenOptions,
configuration?: NielsenConfiguration
) {
this.nielsenHandler = new NielsenHandler(player, appId, instanceName, options, configuration);
}

updateMetadata(metadata: { [key: string]: string }): void {
this.nielsenHandler.updateMetadata(metadata);
}

updateDCRContentMetadata(metadata: NielsenDCRContentMetadata): void {
this.nielsenHandler.updateDCRContentMetadata(metadata);
}

destroy() {
this.nielsenHandler.destroy();
}
Expand Down
133 changes: 116 additions & 17 deletions nielsen/src/integration/NielsenHandler.ts
Original file line number Diff line number Diff line change
@@ -1,21 +1,40 @@
import type {
Ad,
AdBreakEvent,
AdEvent,
AddTrackEvent,
ChromelessPlayer,
DurationChangeEvent,
TextTrack,
TextTrackEnterCueEvent,
TimeUpdateEvent,
VolumeChangeEvent
} from 'theoplayer';
import { loadNielsenLibrary } from '../nielsen/NOLBUNDLE';
import { AdMetadata, ContentMetadata, NielsenOptions } from '../nielsen/Types';
import { getAdType } from '../utils/Util';
import {
AdMetadata,
DCRContentMetadata,
DTVRContentMetadata,
NielsenConfiguration,
NielsenCountry,
NielsenDCRContentMetadata,
NielsenOptions
} from '../nielsen/Types';
import { buildDCRAdMetadata, buildDCRContentMetadata, getAdType } from '../utils/Util';

const EMSG_PRIV_SUFFIX = 'PRIV{';
const EMSG_PAYLOAD_SUFFIX = 'payload=';

export class NielsenHandler {
private player: ChromelessPlayer;

private dcrEnabled: boolean;
private dtvrEnabled: boolean;
private country: NielsenCountry = NielsenCountry.US;

private metadata: DCRContentMetadata | undefined;
private lastReportedPlayheadPosition: number | undefined;

private nSdkInstance: any;

private sessionInProgress: boolean = false;
Expand All @@ -24,46 +43,79 @@ export class NielsenHandler {

private decoder = new TextDecoder('utf-8');

constructor(player: ChromelessPlayer, appId: string, instanceName: string, options?: NielsenOptions) {
constructor(
player: ChromelessPlayer,
appId: string,
instanceName: string,
options?: NielsenOptions,
configuration?: NielsenConfiguration
) {
this.player = player;
this.nSdkInstance = loadNielsenLibrary(appId, instanceName, options);
this.dcrEnabled = configuration?.enableDCR ?? false;
this.dtvrEnabled = configuration?.enableDTVR ?? true;
this.country = configuration?.country ?? NielsenCountry.US;
this.nSdkInstance = loadNielsenLibrary(appId, instanceName, options, this.country);

this.addEventListeners();
}

updateMetadata(metadata: { [key: string]: string }): void {
this.nSdkInstance.ggPM('updateMetadata', metadata);
switch (this.country) {
case NielsenCountry.US: {
const { type, vidtype, assetid, ...updateableParameters } = metadata;
console.log(`[NIELSEN] updateMetadata: ${{ type, vidtype, assetid }} will not be updated`);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we want to keep this log? Or was this for testing?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I thought it made sense to warn developers about the fact that not any passed metadata can be updated mid playback. I have put it behind a check for nol_sdkDebug === "debug", so it only shows if logging is enabled for the Nielsen calls also.

this.nSdkInstance.ggPM('updateMetadata', updateableParameters);
break;
}
case NielsenCountry.CZ:
default:
}
}

updateDCRContentMetadata(metadata: NielsenDCRContentMetadata): void {
if (!this.dcrEnabled) return;
this.metadata = buildDCRContentMetadata(metadata, this.country);
}

private addEventListeners(): void {
this.player.addEventListener('play', this.onPlay);
this.player.addEventListener('pause', this.onInterrupt);
this.player.addEventListener('waiting', this.onInterrupt);
this.player.addEventListener('ended', this.onEnd);
this.player.addEventListener('sourcechange', this.onSourceChange);
this.player.addEventListener('volumechange', this.onVolumeChange);
this.player.addEventListener('loadedmetadata', this.onLoadMetadata);
this.player.addEventListener('durationchange', this.onDurationChange);
this.player.addEventListener('timeupdate', this.onTimeUpdate);

this.player.textTracks.addEventListener('addtrack', this.onAddTrack);

if (this.player.ads) {
this.player.ads.addEventListener('adbegin', this.onAdBegin);
this.player.ads.addEventListener('adend', this.onAdEnd);
this.player.ads.addEventListener('adbreakbegin', this.onAdBreakBegin);
}

window.addEventListener('beforeunload', this.onEnd);
}

private removeEventListeners(): void {
this.player.removeEventListener('play', this.onPlay);
this.player.removeEventListener('pause', this.onInterrupt);
this.player.removeEventListener('waiting', this.onInterrupt);
this.player.removeEventListener('ended', this.onEnd);
this.player.removeEventListener('sourcechange', this.onSourceChange);
this.player.removeEventListener('volumechange', this.onVolumeChange);
this.player.removeEventListener('loadedmetadata', this.onLoadMetadata);
this.player.removeEventListener('durationchange', this.onDurationChange);
this.player.removeEventListener('timeupdate', this.onTimeUpdate);

this.player.textTracks.removeEventListener('addtrack', this.onAddTrack);

if (this.player.ads) {
this.player.ads.removeEventListener('adbegin', this.onAdBegin);
this.player.ads.removeEventListener('adend', this.onAdEnd);
this.player.ads.removeEventListener('adbreakbegin', this.onAdBreakBegin);
}

window.removeEventListener('beforeunload', this.onEnd);
Expand All @@ -73,7 +125,13 @@ export class NielsenHandler {
this.maybeSendPlayEvent();
};

private onInterrupt = () => {
if (!this.dcrEnabled) return;
this.nSdkInstance.ggPM('stop', this.getPlayHeadPosition());
};

private onEnd = () => {
if (this.dcrEnabled && this.player.ads?.playing) this.nSdkInstance.ggPM('stop', this.getPlayHeadPosition());
this.endSession();
};

Expand All @@ -83,24 +141,36 @@ export class NielsenHandler {
};

private onVolumeChange = (event: VolumeChangeEvent) => {
if (!this.dtvrEnabled) return;
const volumeLevel = this.player.muted ? 0 : event.volume * 100;
this.nSdkInstance.ggPM('setVolume', volumeLevel);
};

private onDurationChange = () => {
private onDurationChange = ({ duration }: DurationChangeEvent) => {
if (isNaN(duration)) return;
this.duration = this.player.duration;
this.maybeSendPlayEvent();
};

private onTimeUpdate = ({ currentTime }: TimeUpdateEvent) => {
if (!this.dcrEnabled) return;
const currentTimeFloor = Math.floor(currentTime);
if (currentTimeFloor === this.lastReportedPlayheadPosition) return;
this.lastReportedPlayheadPosition = currentTimeFloor;
this.nSdkInstance.ggPM('setPlayheadPosition', currentTimeFloor);
};

private onLoadMetadata = () => {
const data: ContentMetadata = {
if (!this.dtvrEnabled) return;
const data: DTVRContentMetadata = {
type: 'content',
adModel: '1' // Always '1' for DTVR
};
this.nSdkInstance.ggPM('loadMetadata', data);
};

private onAddTrack = (event: AddTrackEvent) => {
if (!this.dtvrEnabled) return;
if (event.track.kind === 'metadata') {
const track = event.track as TextTrack;
if (track.type === 'id3' || track.type === 'emsg') {
Expand Down Expand Up @@ -164,25 +234,54 @@ export class NielsenHandler {
}
};

private onAdBegin = () => {
const currentAd = this.player.ads!.currentAds.filter((ad: Ad) => ad.type === 'linear');
const type = getAdType(this.player.ads!.currentAdBreak!);
const adMetadata: AdMetadata = {
type,
assetid: currentAd[0].id!
};
this.nSdkInstance.ggPM('loadMetadata', adMetadata);
private onAdBegin = ({ ad }: AdEvent<'adbegin'>) => {
if (ad.type !== 'linear') return;
const { adBreak } = ad;
const { timeOffset } = adBreak;
const offset = this.player.ads?.dai?.contentTimeForStreamTime(timeOffset) ?? timeOffset;
const duration = this.player.ads?.dai?.contentTimeForStreamTime(this.duration) ?? this.duration;
const type = getAdType(offset, duration);
if (this.dtvrEnabled) {
const dtvrAdMetadata: AdMetadata = {
type,
assetid: ad.id!
};
this.nSdkInstance.ggPM('loadMetadata', dtvrAdMetadata);
}
if (this.dcrEnabled) {
const dcrAdMetadata = buildDCRAdMetadata(ad, this.country, this.duration);
this.nSdkInstance.ggPM('loadMetadata', dcrAdMetadata);
}
};

private onAdEnd = () => {
if (!this.dcrEnabled) return;
this.nSdkInstance.ggPM('stop', this.getPlayHeadPosition());
};

private onAdBreakBegin = ({ adBreak }: AdBreakEvent<'adbreakbegin'>) => {
if (!this.dcrEnabled) return;
const { timeOffset } = adBreak;
const offset = this.player.ads?.dai?.contentTimeForStreamTime(timeOffset) ?? timeOffset;
const duration = this.player.ads?.dai?.contentTimeForStreamTime(this.duration) ?? this.duration;
const isPostroll = getAdType(offset, duration) === 'postroll';
if (!isPostroll) return;
this.endSession();
};

private maybeSendPlayEvent(): void {
if (!this.sessionInProgress && !Number.isNaN(this.duration)) {
this.sessionInProgress = true;
if (this.sessionInProgress || Number.isNaN(this.duration)) return;
this.sessionInProgress = true;
if (this.dtvrEnabled) {
const metadataObject = {
channelName: this.player.src,
length: this.duration
};
this.nSdkInstance.ggPM('play', metadataObject);
}
if (this.dcrEnabled) {
this.nSdkInstance.ggPM('loadMetadata', this.metadata);
}
}

private endSession(): void {
Expand Down
Loading