@vouchfor/embeds 0.0.0-experiment.2ab669a → 0.0.0-experiment.2e18d0d

Sign up to get free protection for your applications and to get access to all the features.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@vouchfor/embeds",
3
- "version": "0.0.0-experiment.2ab669a",
3
+ "version": "0.0.0-experiment.2e18d0d",
4
4
  "license": "MIT",
5
5
  "author": "Aaron Williams",
6
6
  "main": "dist/es/embeds.js",
@@ -36,11 +36,11 @@
36
36
  },
37
37
  "dependencies": {
38
38
  "@lit/task": "^1.0.0",
39
- "@vouchfor/media-player": "0.0.0-experiment.2ab669a",
39
+ "@vouchfor/media-player": "0.0.0-experiment.2e18d0d",
40
40
  "uuid": "^9.0.1"
41
41
  },
42
42
  "peerDependencies": {
43
- "lit": "^3.0.2"
43
+ "lit": "^3.1.0"
44
44
  },
45
45
  "devDependencies": {
46
46
  "@esm-bundle/chai": "^4.3.4-fix.0",
@@ -62,7 +62,7 @@
62
62
  "eslint": "^8.50.0",
63
63
  "eslint-plugin-import": "^2.28.1",
64
64
  "lint-staged": "^14.0.1",
65
- "lit": "^2.8.0",
65
+ "lit": "^3.1.0",
66
66
  "prettier": "^3.0.3",
67
67
  "react": "^18.2.0",
68
68
  "react-dom": "^18.2.0",
@@ -39,10 +39,10 @@ type Story = StoryObj<EmbedArgs>;
39
39
 
40
40
  const Embed: Story = {
41
41
  args: {
42
- env: 'dev',
42
+ env: 'local',
43
43
  apiKey: 'TVik9uTMgE-PD25UTHIS6gyl0hMBWC7AT4dkpdlLBT4VIfDWZJrQiCk6Ak7m1',
44
44
  vouchId: '6JQEIPeStt',
45
- templateId: '7d0113f7-3f9a-4bdd-97e3-07ee6eec5730',
45
+ templateId: '357fc118-e179-4171-9446-ff2b8e9d1b29',
46
46
  aspectRatio: 0,
47
47
  preload: 'none',
48
48
  autoplay: false
@@ -50,7 +50,7 @@ const Embed: Story = {
50
50
  argTypes: {
51
51
  env: {
52
52
  control: 'radio',
53
- options: ['prod', 'staging', 'dev']
53
+ options: ['local', 'dev', 'staging', 'prod']
54
54
  },
55
55
  preload: {
56
56
  control: 'radio',
@@ -1,4 +1,5 @@
1
1
  import { Task } from '@lit/task';
2
+ import { v4 as uuidv4 } from 'uuid';
2
3
 
3
4
  import type { Embed, EmbedProps } from '..';
4
5
  import type { ReactiveControllerHost } from 'lit';
@@ -31,23 +32,67 @@ class FetcherController {
31
32
  return this._fetching;
32
33
  }
33
34
 
34
- private async getVouch(env: Environment, apiKey: string, vouchId: string) {
35
+ private getVouch = async (env: Environment, apiKey: string, vouchId: string) => {
35
36
  const { embedApiUrl } = getEnvUrls(env);
36
37
 
37
- return fetch(`${embedApiUrl}/vouches/${vouchId}`, {
38
+ const cacheCheck = uuidv4();
39
+ const res = await fetch(`${embedApiUrl}/vouches/${vouchId}`, {
38
40
  method: 'GET',
39
- headers: [['X-Api-Key', apiKey]]
40
- }).then((response) => response.json());
41
- }
41
+ headers: [
42
+ ['X-Api-Key', apiKey],
43
+ ['X-Cache-Check', cacheCheck]
44
+ ]
45
+ });
46
+
47
+ const vouch = await res.json();
48
+ this.host.dispatchEvent(new CustomEvent('vouch:loaded', { detail: vouchId }));
49
+
50
+ // HACK: we're currently using API Gateway caching on the embed API without any invalidation logic,
51
+ // so to ensure that the cache stays up to date, whenever we detect a cache hit we trigger another
52
+ // API call with the `Cache-Control` header which will re-fill the cache
53
+ const resCacheCheck = res?.headers?.get('X-Cache-Check');
54
+ if (resCacheCheck && resCacheCheck !== cacheCheck) {
55
+ fetch(`${embedApiUrl}/vouches/${vouchId}`, {
56
+ method: 'GET',
57
+ headers: [
58
+ ['X-Api-Key', apiKey],
59
+ ['Cache-Control', 'max-age=0']
60
+ ]
61
+ });
62
+ }
63
+
64
+ return vouch;
65
+ };
42
66
 
43
- private async getTemplate(env: Environment, apiKey: string, templateId: string) {
67
+ private getTemplate = async (env: Environment, apiKey: string, templateId: string) => {
44
68
  const { embedApiUrl } = getEnvUrls(env);
45
69
 
46
- return fetch(`${embedApiUrl}/templates/${templateId}`, {
70
+ const cacheCheck = uuidv4();
71
+ const res = await fetch(`${embedApiUrl}/templates/${templateId}`, {
47
72
  method: 'GET',
48
- headers: [['X-Api-Key', apiKey]]
49
- }).then((response) => response.json());
50
- }
73
+ headers: [
74
+ ['X-Api-Key', apiKey],
75
+ ['X-Cache-Check', cacheCheck]
76
+ ]
77
+ });
78
+ const template = await res.json();
79
+
80
+ // HACK: we're currently using API Gateway caching on the embed API without any invalidation logic,
81
+ // so to ensure that the cache stays up to date, whenever we detect a cache hit we trigger another
82
+ // API call with the `Cache-Control` header which will re-fill the cache
83
+ const resCacheCheck = res?.headers?.get('X-Cache-Check');
84
+ if (resCacheCheck && resCacheCheck !== cacheCheck) {
85
+ fetch(`${embedApiUrl}/templates/${templateId}`, {
86
+ method: 'GET',
87
+ headers: [
88
+ ['X-Api-Key', apiKey],
89
+ ['Cache-Control', 'max-age=0']
90
+ ]
91
+ });
92
+ }
93
+
94
+ return template;
95
+ };
51
96
 
52
97
  constructor(host: EmbedHost) {
53
98
  this.host = host;
@@ -6,13 +6,13 @@ import type { ReactiveController, ReactiveControllerHost } from 'lit';
6
6
 
7
7
  import { getEnvUrls } from '~/utils/env';
8
8
 
9
- const STREAMED_THROTTLE = 2000;
9
+ // In seconds due to checking against node.currentTime
10
+ const STREAMED_THROTTLE = 10;
10
11
 
11
12
  type EmbedHost = ReactiveControllerHost & Embed;
12
13
 
13
14
  type TrackingEvent = 'VOUCH_LOADED' | 'VOUCH_RESPONSE_VIEWED' | 'VIDEO_PLAYED' | 'VIDEO_STREAMED';
14
15
  type TrackingPayload = {
15
- vouchId: string;
16
16
  answerId?: string;
17
17
  streamStart?: number;
18
18
  streamEnd?: number;
@@ -36,8 +36,9 @@ class TrackingController implements ReactiveController {
36
36
  private _hasPlayed = false;
37
37
  private _hasLoaded: BooleanMap = {};
38
38
  private _answersViewed: BooleanMap = {};
39
- private _streamedTime: TimeMap = {};
40
- private _streamedPrevTimestamp: TimeMap = {};
39
+ private _streamStartTime: TimeMap = {};
40
+ private _streamLatestTime: TimeMap = {};
41
+ private _currentlyPlayingVideo: VideoEventDetail | null = null;
41
42
 
42
43
  constructor(host: EmbedHost) {
43
44
  this.host = host;
@@ -45,11 +46,11 @@ class TrackingController implements ReactiveController {
45
46
  }
46
47
 
47
48
  private _findVouchId() {
48
- if (this.host.data) {
49
- if ('uuid' in this.host.data) {
50
- return this.host.data.uuid;
49
+ if (this.host.vouch) {
50
+ if ('uuid' in this.host.vouch) {
51
+ return this.host.vouch.uuid;
51
52
  }
52
- return this.host.data.id;
53
+ return this.host.vouch.id;
53
54
  }
54
55
  }
55
56
 
@@ -116,7 +117,7 @@ class TrackingController implements ReactiveController {
116
117
  });
117
118
 
118
119
  return {
119
- source: 'media_player',
120
+ source: this.host.trackingSource,
120
121
  time: new Date(),
121
122
  region,
122
123
  country,
@@ -128,11 +129,16 @@ class TrackingController implements ReactiveController {
128
129
  };
129
130
  };
130
131
 
131
- private _sendTrackingEvent = (event: TrackingEvent, payload: TrackingPayload) => {
132
+ private _sendTrackingEvent = (event: TrackingEvent, payload?: TrackingPayload) => {
133
+ const vouchId = this._findVouchId();
134
+
135
+ if (!vouchId || this.host.disableTracking) {
136
+ return;
137
+ }
138
+
132
139
  const { publicApiUrl } = getEnvUrls(this.host.env);
133
140
  const { client, tab, request, visitor } = this._getUids();
134
141
 
135
- // Don't send tracking if we don't have a source
136
142
  navigator.sendBeacon(
137
143
  `${publicApiUrl}/api/events`,
138
144
  JSON.stringify({
@@ -156,87 +162,74 @@ class TrackingController implements ReactiveController {
156
162
 
157
163
  // Only send loaded event once per session
158
164
  if (!this._hasLoaded[vouchId]) {
159
- this._sendTrackingEvent('VOUCH_LOADED', {
160
- vouchId
161
- });
165
+ this._sendTrackingEvent('VOUCH_LOADED');
162
166
  this._hasLoaded[vouchId] = true;
163
167
  }
164
168
  };
165
169
 
166
170
  private _handlePlay = () => {
167
- const vouchId = this._findVouchId();
168
-
169
- if (!vouchId) {
170
- return;
171
- }
172
-
173
171
  // Only send the video played event once per session
174
172
  if (!this._hasPlayed) {
175
173
  this._sendTrackingEvent('VIDEO_PLAYED', {
176
- vouchId,
177
174
  streamStart: this.host.currentTime
178
175
  });
179
176
  this._hasPlayed = true;
180
177
  }
181
178
  };
182
179
 
183
- private _handleVideoPlay = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
184
- const vouchId = this._findVouchId();
185
- if (!vouchId) {
186
- return;
187
- }
180
+ private _handleVideoPlay = ({ detail: { id, key, node } }: CustomEvent<VideoEventDetail>) => {
188
181
  // Only increment play count once per session
189
- if (!this._answersViewed[id]) {
182
+ if (!this._answersViewed[key]) {
190
183
  this._sendTrackingEvent('VOUCH_RESPONSE_VIEWED', {
191
- vouchId,
192
184
  answerId: id
193
185
  });
194
- this._answersViewed[id] = true;
186
+ this._answersViewed[key] = true;
195
187
  }
196
- this._streamedTime[id] = node.currentTime;
197
- this._streamedPrevTimestamp[id] = Date.now();
188
+
189
+ this._streamStartTime[key] = node.currentTime;
190
+ this._streamLatestTime[key] = node.currentTime;
198
191
  };
199
192
 
200
- private _handleVideoTimeUpdate = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
201
- const vouchId = this._findVouchId();
202
- if (!vouchId) {
203
- return;
193
+ private _handleVideoTimeUpdate = ({ detail: { id, key, node } }: CustomEvent<VideoEventDetail>) => {
194
+ // We only want to count any time that the video is actually playing
195
+ if (!this.host.paused) {
196
+ this._currentlyPlayingVideo = { id, key, node };
197
+ this._streamLatestTime[key] = node.currentTime;
204
198
  }
205
- const currentTimestamp = Date.now();
199
+
206
200
  if (
201
+ !node.paused &&
207
202
  !this.host.paused &&
208
203
  // Only fire the video seeked event when this video is the active one
209
204
  id === this.host.scene?.video?.id &&
210
205
  // Throttle the frequency that we send streamed events while playing
211
- currentTimestamp - this._streamedPrevTimestamp[id] > STREAMED_THROTTLE
206
+ this._streamLatestTime[key] - this._streamStartTime[key] > STREAMED_THROTTLE
212
207
  ) {
213
208
  this._sendTrackingEvent('VIDEO_STREAMED', {
214
- vouchId,
215
209
  answerId: id,
216
- streamStart: this._streamedTime[id],
217
- streamEnd: node.currentTime
210
+ streamStart: this._streamStartTime[key],
211
+ streamEnd: this._streamLatestTime[key]
218
212
  });
219
- this._streamedTime[id] = node.currentTime;
220
- this._streamedPrevTimestamp[id] = currentTimestamp;
213
+
214
+ this._streamStartTime[key] = node.currentTime;
221
215
  }
222
216
  };
223
217
 
224
- private _handleVideoPause = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
225
- const vouchId = this._findVouchId();
226
- if (!vouchId) {
227
- return;
218
+ private _handleVideoPause = ({ detail: { id, key } }: CustomEvent<VideoEventDetail>) => {
219
+ // Don't send a tracking event when seeking backwards
220
+ if (this._streamLatestTime[key] > this._streamStartTime[key]) {
221
+ // Send a video streamed event any time the video pauses then reset the streamed state
222
+ // We do this to capture the last bit of time that the video was played between the previous
223
+ // stream event and the video being paused manually or stopping because it ended
224
+ this._sendTrackingEvent('VIDEO_STREAMED', {
225
+ answerId: id,
226
+ streamStart: this._streamStartTime[key],
227
+ streamEnd: this._streamLatestTime[key]
228
+ });
228
229
  }
229
- // Send a video streamed event any time the video pauses then reset the streamed state
230
- // We do this to capture the last bit of time that the video was played between the previous
231
- // stream event and the video being paused manually or stopping because it ended
232
- this._sendTrackingEvent('VIDEO_STREAMED', {
233
- vouchId,
234
- answerId: id,
235
- streamStart: this._streamedTime[id],
236
- streamEnd: node.currentTime
237
- });
238
- delete this._streamedTime[id];
239
- delete this._streamedPrevTimestamp[id];
230
+ this._currentlyPlayingVideo = null;
231
+ delete this._streamStartTime[key];
232
+ delete this._streamLatestTime[key];
240
233
  };
241
234
 
242
235
  hostConnected() {
@@ -250,6 +243,20 @@ class TrackingController implements ReactiveController {
250
243
  }
251
244
 
252
245
  hostDisconnected() {
246
+ if (this._currentlyPlayingVideo) {
247
+ const { id, key } = this._currentlyPlayingVideo;
248
+ if (this._streamLatestTime[key] > this._streamStartTime[key]) {
249
+ // Send a video streamed event any time the video pauses then reset the streamed state
250
+ // We do this to capture the last bit of time that the video was played between the previous
251
+ // stream event and the video being paused manually or stopping because it ended
252
+ this._sendTrackingEvent('VIDEO_STREAMED', {
253
+ answerId: id,
254
+ streamStart: this._streamStartTime[key],
255
+ streamEnd: this._streamLatestTime[key]
256
+ });
257
+ }
258
+ }
259
+
253
260
  this.host.removeEventListener('vouch:loaded', this._handleVouchLoaded);
254
261
  this.host.mediaPlayer?.removeEventListener('play', this._handlePlay);
255
262
  this.host.mediaPlayer?.removeEventListener('video:play', this._handleVideoPlay);
@@ -17,6 +17,8 @@ import '@vouchfor/media-player';
17
17
  type EmbedProps = Pick<MediaPlayerProps, 'data' | 'aspectRatio' | 'preload' | 'autoplay' | 'controls'> & {
18
18
  env: Environment;
19
19
  apiKey: string;
20
+ disableTracking?: boolean;
21
+ trackingSource?: string;
20
22
  vouchId?: string;
21
23
  templateId?: string;
22
24
  };
@@ -31,6 +33,8 @@ class Embed extends LitElement {
31
33
 
32
34
  @property({ type: String }) env: EmbedProps['env'] = 'prod';
33
35
  @property({ type: String }) apiKey: EmbedProps['apiKey'] = '';
36
+ @property({ type: Boolean }) disableTracking: EmbedProps['disableTracking'] = false;
37
+ @property({ type: String }) trackingSource: EmbedProps['trackingSource'] = 'embed';
34
38
 
35
39
  @property({ type: Array }) controls: EmbedProps['controls'];
36
40
  @property({ type: String }) preload: EmbedProps['preload'] = 'auto';
@@ -55,6 +59,7 @@ class Embed extends LitElement {
55
59
  'waiting',
56
60
 
57
61
  'video:loadeddata',
62
+ 'video:seeking',
58
63
  'video:seeked',
59
64
  'video:play',
60
65
  'video:playing',
package/src/utils/env.ts CHANGED
@@ -1,15 +1,11 @@
1
- type Environment = 'dev' | 'staging' | 'prod';
1
+ type Environment = 'local' | 'dev' | 'staging' | 'prod';
2
2
 
3
3
  type GetEnvUrlsReturn = {
4
- marketingUrl: string;
5
4
  videoUrl: string;
6
5
  publicApiUrl: string;
7
6
  embedApiUrl: string;
8
- publicRecorderUrl: string;
9
7
  };
10
8
 
11
- const marketingUrl = 'https://vouchfor.com';
12
-
13
9
  const devVideoUrl = 'https://d2rxhdlm2q91uk.cloudfront.net';
14
10
  const stagingVideoUrl = 'https://d1ix11aj5kfygl.cloudfront.net';
15
11
  const prodVideoUrl = 'https://d157jlwnudd93d.cloudfront.net';
@@ -18,61 +14,51 @@ const devPublicApiUrl = 'https://bshyfw4h5a.execute-api.ap-southeast-2.amazonaws
18
14
  const stagingPublicApiUrl = 'https://gyzw7rpbq3.execute-api.ap-southeast-2.amazonaws.com/staging';
19
15
  const prodPublicApiUrl = 'https://vfcjuim1l3.execute-api.ap-southeast-2.amazonaws.com/prod';
20
16
 
17
+ const localEmbedApiUrl = 'http://localhost:6060/v2';
21
18
  const devEmbedApiUrl = 'https://embed-dev.vouchfor.com/v2';
22
19
  const stagingEmbedApiUrl = 'https://embed-staging.vouchfor.com/v2';
23
20
  const prodEmbedApiUrl = 'https://embed.vouchfor.com/v2';
24
21
 
25
- const devPublicRecorderUrl = 'https://dev.vouchfor.com';
26
- const stagingPublicRecorderUrl = 'https://staging.vouchfor.com';
27
- const prodPublicRecorderUrl = 'https://app.vouchfor.com';
28
-
29
22
  // We are handling the case where env is an unknown string so the ts error is a lie
30
23
  // eslint-disable-next-line @typescript-eslint/ban-ts-comment
31
24
  // @ts-ignore
32
25
  function getEnvUrls(env: Environment): GetEnvUrlsReturn {
33
- if (!['dev', 'staging', 'prod'].includes(env)) {
26
+ if (!['local', 'dev', 'staging', 'prod'].includes(env)) {
34
27
  throw new Error(`Unknown environment: ${env}`);
35
28
  }
36
29
 
30
+ if (env === 'local') {
31
+ return {
32
+ videoUrl: devVideoUrl,
33
+ publicApiUrl: devPublicApiUrl,
34
+ embedApiUrl: localEmbedApiUrl
35
+ };
36
+ }
37
+
37
38
  if (env === 'dev') {
38
39
  return {
39
- marketingUrl,
40
40
  videoUrl: devVideoUrl,
41
41
  publicApiUrl: devPublicApiUrl,
42
- embedApiUrl: devEmbedApiUrl,
43
- publicRecorderUrl: devPublicRecorderUrl
42
+ embedApiUrl: devEmbedApiUrl
44
43
  };
45
44
  }
46
45
 
47
46
  if (env === 'staging') {
48
47
  return {
49
- marketingUrl,
50
48
  videoUrl: stagingVideoUrl,
51
49
  publicApiUrl: stagingPublicApiUrl,
52
- embedApiUrl: stagingEmbedApiUrl,
53
- publicRecorderUrl: stagingPublicRecorderUrl
50
+ embedApiUrl: stagingEmbedApiUrl
54
51
  };
55
52
  }
56
53
 
57
54
  if (env === 'prod') {
58
55
  return {
59
- marketingUrl,
60
56
  videoUrl: prodVideoUrl,
61
57
  publicApiUrl: prodPublicApiUrl,
62
- embedApiUrl: prodEmbedApiUrl,
63
- publicRecorderUrl: prodPublicRecorderUrl
58
+ embedApiUrl: prodEmbedApiUrl
64
59
  };
65
60
  }
66
61
  }
67
62
 
68
- export {
69
- marketingUrl,
70
- devEmbedApiUrl,
71
- stagingEmbedApiUrl,
72
- prodEmbedApiUrl,
73
- devPublicRecorderUrl,
74
- stagingPublicRecorderUrl,
75
- prodPublicRecorderUrl,
76
- getEnvUrls
77
- };
63
+ export { devEmbedApiUrl, stagingEmbedApiUrl, prodEmbedApiUrl, getEnvUrls };
78
64
  export type { Environment };
@@ -1,14 +0,0 @@
1
- import type { Embed } from '..';
2
- import type { ReactiveController, ReactiveControllerHost } from 'lit';
3
- type EmbedHost = ReactiveControllerHost & Embed;
4
- declare class EventForwardController implements ReactiveController {
5
- host: EmbedHost;
6
- private _events;
7
- private _cleanup;
8
- private _forwardElementRef;
9
- constructor(host: EmbedHost, events: string[]);
10
- register(): import("lit-html/directive.js").DirectiveResult<typeof import("lit-html/directives/ref.js").RefDirective>;
11
- hostConnected(): void;
12
- hostDisconnected(): void;
13
- }
14
- export { EventForwardController };