@vouchfor/embeds 0.0.0-experiment.cb30cc7 → 0.0.0-experiment.d892f46

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.cb30cc7",
3
+ "version": "0.0.0-experiment.d892f46",
4
4
  "license": "MIT",
5
5
  "author": "Aaron Williams",
6
6
  "main": "dist/es/embeds.js",
@@ -26,7 +26,7 @@
26
26
  "lint:staged": "lint-staged",
27
27
  "prepublishOnly": "yarn build",
28
28
  "size": "size-limit",
29
- "storybook": "yarn prebuild && storybook dev -p 6006",
29
+ "storybook": "yarn prebuild && storybook dev -p 6007",
30
30
  "prebuild": "yarn build:deps && yarn generate:manifest",
31
31
  "test": "true"
32
32
  },
@@ -35,12 +35,12 @@
35
35
  "**/*.{md,json,yml}": "prettier --write"
36
36
  },
37
37
  "dependencies": {
38
- "@lit/task": "1.0.0",
39
- "@vouchfor/media-player": "0.0.0-experiment.cb30cc7",
38
+ "@lit/task": "^1.0.0",
39
+ "@vouchfor/media-player": "0.0.0-experiment.d892f46",
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: vouch?.id }));
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 !== 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 !== 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;
@@ -4,15 +4,16 @@ import type { Embed } from '..';
4
4
  import type { VideoEventDetail } from '@vouchfor/media-player';
5
5
  import type { ReactiveController, ReactiveControllerHost } from 'lit';
6
6
 
7
+ import packageJson from '../../../../package.json';
7
8
  import { getEnvUrls } from '~/utils/env';
8
9
 
9
- const STREAMED_THROTTLE = 2000;
10
+ const MINIMUM_SEND_THRESHOLD = 1;
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
+ vouchId?: string;
16
17
  answerId?: string;
17
18
  streamStart?: number;
18
19
  streamEnd?: number;
@@ -36,21 +37,23 @@ class TrackingController implements ReactiveController {
36
37
  private _hasPlayed = false;
37
38
  private _hasLoaded: BooleanMap = {};
38
39
  private _answersViewed: BooleanMap = {};
39
- private _streamedTime: TimeMap = {};
40
- private _streamedPrevTimestamp: TimeMap = {};
40
+ private _streamStartTime: TimeMap = {};
41
+ private _streamLatestTime: TimeMap = {};
42
+ private _currentlyPlayingVideo: VideoEventDetail | null = null;
41
43
 
42
44
  constructor(host: EmbedHost) {
43
45
  this.host = host;
44
46
  host.addController(this);
45
47
  }
46
48
 
47
- private _findVouchId() {
48
- if (this.host.data) {
49
- if ('uuid' in this.host.data) {
50
- return this.host.data.uuid;
51
- }
52
- return this.host.data.id;
49
+ private _findVouchId(payload?: TrackingPayload) {
50
+ if (payload && 'vouchId' in payload) {
51
+ return payload.vouchId;
52
+ }
53
+ if (this.host.vouch) {
54
+ return this.host.vouch.id;
53
55
  }
56
+ return null;
54
57
  }
55
58
 
56
59
  private _createVisitor = (visitorId: string) => {
@@ -116,7 +119,7 @@ class TrackingController implements ReactiveController {
116
119
  });
117
120
 
118
121
  return {
119
- source: 'media_player',
122
+ source: this.host.trackingSource,
120
123
  time: new Date(),
121
124
  region,
122
125
  country,
@@ -128,27 +131,54 @@ class TrackingController implements ReactiveController {
128
131
  };
129
132
  };
130
133
 
131
- private _sendTrackingEvent = (event: TrackingEvent, payload: TrackingPayload) => {
134
+ private _sendTrackingEvent = (event: TrackingEvent, payload?: TrackingPayload) => {
135
+ const vouchId = this._findVouchId(payload);
136
+
137
+ if (!vouchId || this.host.disableTracking) {
138
+ return;
139
+ }
140
+
132
141
  const { publicApiUrl } = getEnvUrls(this.host.env);
133
142
  const { client, tab, request, visitor } = this._getUids();
134
143
 
135
- // Don't send tracking if we don't have a source
136
144
  navigator.sendBeacon(
137
145
  `${publicApiUrl}/api/events`,
138
146
  JSON.stringify({
139
147
  event,
140
- payload,
148
+ payload: {
149
+ ...payload,
150
+ vouchId
151
+ },
141
152
  context: {
142
153
  'x-uid-client': client,
143
154
  'x-uid-tab': tab,
144
155
  'x-uid-request': request,
145
156
  'x-uid-visitor': visitor,
146
- 'x-reporting-metadata': this._getReportingMetadata()
157
+ 'x-reporting-metadata': this._getReportingMetadata(),
158
+ 'x-embeds-version': packageJson.version
147
159
  }
148
160
  })
149
161
  );
150
162
  };
151
163
 
164
+ private _streamEnded = () => {
165
+ if (this._currentlyPlayingVideo) {
166
+ const { id, key } = this._currentlyPlayingVideo;
167
+ // Don't send a tracking event when seeking backwards
168
+ if (this._streamLatestTime[key] > this._streamStartTime[key] + MINIMUM_SEND_THRESHOLD) {
169
+ // Send a video streamed event any time the stream ends to capture the time between starting
170
+ // the video and the video stopping for any reason (pausing, deleting the embed node or closing the browser)
171
+ this._sendTrackingEvent('VIDEO_STREAMED', {
172
+ answerId: id,
173
+ streamStart: this._streamStartTime[key],
174
+ streamEnd: this._streamLatestTime[key]
175
+ });
176
+ }
177
+ delete this._streamStartTime[key];
178
+ delete this._streamLatestTime[key];
179
+ }
180
+ };
181
+
152
182
  private _handleVouchLoaded = ({ detail: vouchId }: CustomEvent<string>) => {
153
183
  if (!vouchId) {
154
184
  return;
@@ -156,91 +186,70 @@ class TrackingController implements ReactiveController {
156
186
 
157
187
  // Only send loaded event once per session
158
188
  if (!this._hasLoaded[vouchId]) {
159
- this._sendTrackingEvent('VOUCH_LOADED', {
160
- vouchId
161
- });
189
+ this._sendTrackingEvent('VOUCH_LOADED', { vouchId });
162
190
  this._hasLoaded[vouchId] = true;
163
191
  }
164
192
  };
165
193
 
166
194
  private _handlePlay = () => {
167
- const vouchId = this._findVouchId();
168
-
169
- if (!vouchId) {
170
- return;
171
- }
172
-
173
195
  // Only send the video played event once per session
174
196
  if (!this._hasPlayed) {
175
197
  this._sendTrackingEvent('VIDEO_PLAYED', {
176
- vouchId,
177
198
  streamStart: this.host.currentTime
178
199
  });
179
200
  this._hasPlayed = true;
180
201
  }
181
202
  };
182
203
 
183
- private _handleVideoPlay = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
184
- const vouchId = this._findVouchId();
185
- if (!vouchId) {
186
- return;
187
- }
204
+ private _handleVideoPlay = ({ detail: { id, key, node } }: CustomEvent<VideoEventDetail>) => {
188
205
  // Only increment play count once per session
189
- if (!this._answersViewed[id]) {
206
+ if (!this._answersViewed[key]) {
190
207
  this._sendTrackingEvent('VOUCH_RESPONSE_VIEWED', {
191
- vouchId,
192
208
  answerId: id
193
209
  });
194
- this._answersViewed[id] = true;
210
+ this._answersViewed[key] = true;
195
211
  }
196
- this._streamedTime[id] = node.currentTime;
197
- this._streamedPrevTimestamp[id] = Date.now();
198
- };
199
212
 
200
- private _handleVideoTimeUpdate = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
201
- const vouchId = this._findVouchId();
202
- if (!vouchId) {
203
- return;
213
+ if (!this._streamStartTime[key]) {
214
+ this._streamStartTime[key] = node.currentTime;
215
+ this._streamLatestTime[key] = node.currentTime;
204
216
  }
205
- const currentTimestamp = Date.now();
217
+ };
218
+
219
+ private _handleVideoTimeUpdate = ({ detail: { id, key, node } }: CustomEvent<VideoEventDetail>) => {
206
220
  if (
221
+ // We only want to count any time that the video is actually playing
207
222
  !this.host.paused &&
208
- // Only fire the video seeked event when this video is the active one
209
- id === this.host.scene?.video?.id &&
210
- // Throttle the frequency that we send streamed events while playing
211
- currentTimestamp - this._streamedPrevTimestamp[id] > STREAMED_THROTTLE
223
+ // Only update the latest time if this event fires for the currently active video
224
+ id === this.host.scene?.video?.id
212
225
  ) {
226
+ this._currentlyPlayingVideo = { id, key, node };
227
+ this._streamLatestTime[key] = node.currentTime;
228
+ }
229
+ };
230
+
231
+ private _handleVideoPause = ({ detail: { id, key } }: CustomEvent<VideoEventDetail>) => {
232
+ if (this._streamLatestTime[key] > this._streamStartTime[key] + MINIMUM_SEND_THRESHOLD) {
213
233
  this._sendTrackingEvent('VIDEO_STREAMED', {
214
- vouchId,
215
234
  answerId: id,
216
- streamStart: this._streamedTime[id],
217
- streamEnd: node.currentTime
235
+ streamStart: this._streamStartTime[key],
236
+ streamEnd: this._streamLatestTime[key]
218
237
  });
219
- this._streamedTime[id] = node.currentTime;
220
- this._streamedPrevTimestamp[id] = currentTimestamp;
221
238
  }
239
+ delete this._streamStartTime[key];
240
+ delete this._streamLatestTime[key];
222
241
  };
223
242
 
224
- private _handleVideoPause = ({ detail: { id, node } }: CustomEvent<VideoEventDetail>) => {
225
- const vouchId = this._findVouchId();
226
- if (!vouchId) {
227
- return;
243
+ private _handleVisibilityChange = () => {
244
+ if (document.visibilityState === 'hidden') {
245
+ this.host.pause();
246
+ this._streamEnded();
228
247
  }
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];
240
248
  };
241
249
 
242
250
  hostConnected() {
243
251
  requestAnimationFrame(() => {
252
+ document.addEventListener('visibilitychange', this._handleVisibilityChange);
244
253
  this.host.addEventListener('vouch:loaded', this._handleVouchLoaded);
245
254
  this.host.mediaPlayer?.addEventListener('play', this._handlePlay);
246
255
  this.host.mediaPlayer?.addEventListener('video:play', this._handleVideoPlay);
@@ -250,6 +259,8 @@ class TrackingController implements ReactiveController {
250
259
  }
251
260
 
252
261
  hostDisconnected() {
262
+ this._streamEnded();
263
+ document.removeEventListener('visibilitychange', this._handleVisibilityChange);
253
264
  this.host.removeEventListener('vouch:loaded', this._handleVouchLoaded);
254
265
  this.host.mediaPlayer?.removeEventListener('play', this._handlePlay);
255
266
  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
 
21
- const devEmbedApiUrl = 'http://localhost:6060/v1';
22
- const stagingEmbedApiUrl = 'https://embed-staging.vouchfor.com/v1';
23
- const prodEmbedApiUrl = 'https://embed.vouchfor.com/v1';
24
-
25
- const devPublicRecorderUrl = 'https://dev.vouchfor.com';
26
- const stagingPublicRecorderUrl = 'https://staging.vouchfor.com';
27
- const prodPublicRecorderUrl = 'https://app.vouchfor.com';
17
+ const localEmbedApiUrl = 'http://localhost:6060/v2';
18
+ const devEmbedApiUrl = 'https://embed-dev.vouchfor.com/v2';
19
+ const stagingEmbedApiUrl = 'https://embed-staging.vouchfor.com/v2';
20
+ const prodEmbedApiUrl = 'https://embed.vouchfor.com/v2';
28
21
 
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 };
@@ -1,18 +0,0 @@
1
- type Environment = 'dev' | 'staging' | 'prod';
2
- type GetEnvUrlsReturn = {
3
- marketingUrl: string;
4
- videoUrl: string;
5
- publicApiUrl: string;
6
- embedApiUrl: string;
7
- publicRecorderUrl: string;
8
- };
9
- declare const marketingUrl = "https://vouchfor.com";
10
- declare const devEmbedApiUrl = "http://localhost:6060/v1";
11
- declare const stagingEmbedApiUrl = "https://embed-staging.vouchfor.com/v1";
12
- declare const prodEmbedApiUrl = "https://embed.vouchfor.com/v1";
13
- declare const devPublicRecorderUrl = "https://dev.vouchfor.com";
14
- declare const stagingPublicRecorderUrl = "https://staging.vouchfor.com";
15
- declare const prodPublicRecorderUrl = "https://app.vouchfor.com";
16
- declare function getEnvUrls(env: Environment): GetEnvUrlsReturn;
17
- export { marketingUrl, devEmbedApiUrl, stagingEmbedApiUrl, prodEmbedApiUrl, devPublicRecorderUrl, stagingPublicRecorderUrl, prodPublicRecorderUrl, getEnvUrls };
18
- export type { Environment };
File without changes
File without changes