@pendo/openfeature-web-provider 0.1.0 → 0.2.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.
package/README.md CHANGED
@@ -10,15 +10,15 @@ npm install @pendo/openfeature-web-provider @openfeature/web-sdk
10
10
 
11
11
  ## Prerequisites
12
12
 
13
- The Pendo agent must be installed on your page. The provider reads flags from `window.pendo.segmentFlags` which is populated by the agent.
14
-
15
- ```html
16
- <script>
17
- (function(apiKey){
18
- // Pendo agent snippet
19
- // ...
20
- })('YOUR_PENDO_API_KEY');
21
- </script>
13
+ The Pendo Web SDK must be installed and initialized on your page with `requestSegmentFlags: true` enabled.
14
+
15
+ ```javascript
16
+ // Initialize Pendo with segment flags enabled
17
+ pendo.initialize({
18
+ visitor: { id: 'user-123' },
19
+ account: { id: 'account-456' },
20
+ requestSegmentFlags: true // Required for feature flags
21
+ });
22
22
  ```
23
23
 
24
24
  ## Usage
@@ -29,7 +29,7 @@ The Pendo agent must be installed on your page. The provider reads flags from `w
29
29
  import { OpenFeature } from '@openfeature/web-sdk';
30
30
  import { PendoProvider } from '@pendo/openfeature-web-provider';
31
31
 
32
- // Initialize the provider (waits for Pendo agent to be ready)
32
+ // Initialize the provider (waits for Pendo to be ready)
33
33
  await OpenFeature.setProviderAndWait(new PendoProvider());
34
34
 
35
35
  // Get a client
@@ -40,16 +40,16 @@ const client = OpenFeature.getClient();
40
40
 
41
41
  ```typescript
42
42
  // Boolean flag
43
- const showNewFeature = await client.getBooleanValue('new-checkout-flow', false);
43
+ const showNewFeature = client.getBooleanValue('new-checkout-flow', false);
44
44
 
45
45
  // String flag (returns "on" or "off")
46
- const variant = await client.getStringValue('checkout-variant', 'control');
46
+ const variant = client.getStringValue('checkout-variant', 'control');
47
47
 
48
48
  // Number flag (returns 1 or 0)
49
- const flagValue = await client.getNumberValue('feature-score', 0);
49
+ const flagValue = client.getNumberValue('feature-score', 0);
50
50
 
51
51
  // Object flag (returns { enabled: true/false })
52
- const config = await client.getObjectValue('feature-config', { enabled: false });
52
+ const config = client.getObjectValue('feature-config', { enabled: false });
53
53
  ```
54
54
 
55
55
  ### Event Tracking
@@ -57,18 +57,47 @@ const config = await client.getObjectValue('feature-config', { enabled: false })
57
57
  Track custom events to Pendo:
58
58
 
59
59
  ```typescript
60
- const provider = new PendoProvider();
61
- await OpenFeature.setProviderAndWait(provider);
60
+ const client = OpenFeature.getClient();
62
61
 
63
- // Track an event
64
- provider.track('checkout_started', undefined, { cartValue: '99.99' });
62
+ // Track an event (delegates to pendo.track)
63
+ client.track('checkout_started', { cartValue: '99.99' });
64
+ ```
65
+
66
+ ### Configuration Options
65
67
 
66
- // The context parameter is ignored in the web provider
67
- // since the Pendo agent already knows the current visitor
68
- provider.track('feature_used');
68
+ ```typescript
69
+ const provider = new PendoProvider({
70
+ // Timeout waiting for the Pendo Web SDK to be ready (default: 5000ms)
71
+ readyTimeout: 10000,
72
+ });
69
73
  ```
70
74
 
71
- ### Telemetry Hook
75
+ ## How It Works
76
+
77
+ 1. The provider waits for the Pendo Web SDK to be ready
78
+ 2. Flag evaluation checks if the flag key exists in `pendo.segmentFlags`
79
+ 3. When Pendo updates flags (on metadata/identity changes), the provider emits `ConfigurationChanged`
80
+ 4. React/Angular OpenFeature SDKs automatically re-render when flags change
81
+
82
+ ## Automatic Flag Updates
83
+
84
+ The provider subscribes to Pendo's `Events.segmentFlagsUpdated` event to detect when flags are updated. This happens automatically when:
85
+
86
+ - Pendo initializes and loads initial flags
87
+ - Visitor metadata changes
88
+ - Visitor identity changes (via `pendo.identify()`)
89
+
90
+ When flags update, the provider emits a `ConfigurationChanged` event, which triggers re-renders in React/Angular SDKs.
91
+
92
+ ## Resolution Details
93
+
94
+ | Scenario | Reason | Variant |
95
+ |----------|--------|---------|
96
+ | Flag key in segmentFlags | `TARGETING_MATCH` | `on` |
97
+ | Flag key not in segmentFlags | `DEFAULT` | `off` |
98
+ | Pendo not ready / no flags | `DEFAULT` | `default` |
99
+
100
+ ## Telemetry Hook
72
101
 
73
102
  Automatically track all flag evaluations to Pendo using the telemetry hook:
74
103
 
@@ -83,7 +112,7 @@ await OpenFeature.setProviderAndWait(provider);
83
112
  OpenFeature.addHooks(telemetryHook);
84
113
  ```
85
114
 
86
- #### Telemetry Hook Options
115
+ ### Telemetry Hook Options
87
116
 
88
117
  ```typescript
89
118
  const telemetryHook = new PendoTelemetryHook({
@@ -95,62 +124,14 @@ const telemetryHook = new PendoTelemetryHook({
95
124
  });
96
125
  ```
97
126
 
98
- #### Event Payload
99
-
100
- Each flag evaluation sends a track event with these properties:
101
-
102
- | Property | Description |
103
- |----------|-------------|
104
- | `flag_key` | The flag that was evaluated |
105
- | `flag_variant` | "on", "off", or variant name |
106
- | `flag_reason` | "TARGETING_MATCH", "DEFAULT", or "ERROR" |
107
- | `flag_value` | Stringified value |
108
- | `provider_name` | "pendo-provider" |
109
-
110
- ### Configuration Options
111
-
112
- ```typescript
113
- const provider = new PendoProvider({
114
- // API key for Pendo initialization (optional if already initialized)
115
- apiKey: 'YOUR_API_KEY',
116
-
117
- // Timeout waiting for Pendo agent to be ready (default: 5000ms)
118
- readyTimeout: 10000,
119
- });
120
- ```
121
-
122
- ## How It Works
123
-
124
- 1. The provider waits for the Pendo agent to initialize
125
- 2. Flag evaluation checks if the flag key exists in `window.pendo.segmentFlags`
126
- 3. If the flag key is present, the flag is enabled; otherwise, it returns the default value
127
-
128
- ## Context
129
-
130
- The web provider does not require explicit context for flag evaluation since the Pendo agent maintains visitor/account identity automatically. However, you can pass context for consistency with the OpenFeature API:
131
-
132
- ```typescript
133
- const enabled = await client.getBooleanValue('my-flag', false, {
134
- targetingKey: 'user-123', // Ignored by web provider
135
- accountId: 'account-456', // Ignored by web provider
136
- });
137
- ```
138
-
139
- ## Resolution Details
140
-
141
- | Scenario | Reason | Variant |
142
- |----------|--------|---------|
143
- | Flag key in segmentFlags | `TARGETING_MATCH` | `on` |
144
- | Flag key not in segmentFlags | `DEFAULT` | `off` |
145
- | Pendo not ready | `DEFAULT` | `default` |
146
-
147
127
  ## Troubleshooting
148
128
 
149
129
  ### Flags always return default values
150
130
 
151
- 1. Check that the Pendo agent is properly installed and initialized
152
- 2. Verify the visitor is in a segment that has the flag enabled
153
- 3. Check browser console for `[PendoProvider]` warnings
131
+ 1. Ensure `requestSegmentFlags: true` is set in your Pendo initialization
132
+ 2. Check that Pendo is properly initialized before the provider
133
+ 3. Verify the visitor is in a segment with the flag enabled
134
+ 4. Check browser console for `[PendoProvider]` warnings
154
135
 
155
136
  ### Provider times out
156
137
 
@@ -160,15 +141,13 @@ Increase the `readyTimeout` option:
160
141
  new PendoProvider({ readyTimeout: 15000 });
161
142
  ```
162
143
 
163
- ### Track events not appearing
144
+ ### Flags not updating
164
145
 
165
- Ensure the Pendo agent's `track` function is available:
146
+ The provider automatically detects flag changes. If flags aren't updating:
166
147
 
167
- ```typescript
168
- if (window.pendo?.track) {
169
- provider.track('my-event');
170
- }
171
- ```
148
+ 1. Verify Pendo is receiving metadata/identity updates
149
+ 2. Check that `requestSegmentFlags: true` is enabled
150
+ 3. Ensure your segments are configured correctly in Pendo
172
151
 
173
152
  ## License
174
153
 
@@ -1,23 +1,9 @@
1
1
  import type { Provider, ProviderMetadata, ResolutionDetails, EvaluationContext, JsonValue, Hook, TrackingEventDetails } from "@openfeature/web-sdk";
2
- import { ClientProviderStatus } from "@openfeature/web-sdk";
3
- declare global {
4
- interface Window {
5
- pendo?: {
6
- segmentFlags?: string[];
7
- isReady?: () => boolean;
8
- initialize?: (options: unknown) => void;
9
- track?: (event: string, properties?: Record<string, string>) => void;
10
- };
11
- }
12
- }
2
+ import { ClientProviderStatus, OpenFeatureEventEmitter } from "@openfeature/web-sdk";
3
+ import "./types";
13
4
  export interface PendoProviderOptions {
14
5
  /**
15
- * Optional: Pendo API key for initialization.
16
- * If not provided, assumes Pendo is already initialized on the page.
17
- */
18
- apiKey?: string;
19
- /**
20
- * Optional: Timeout in milliseconds to wait for Pendo to be ready.
6
+ * Timeout in milliseconds to wait for Pendo to be ready.
21
7
  * Default: 5000ms
22
8
  */
23
9
  readyTimeout?: number;
@@ -25,37 +11,55 @@ export interface PendoProviderOptions {
25
11
  /**
26
12
  * OpenFeature provider for Pendo feature flags.
27
13
  *
28
- * This provider evaluates feature flags by checking the `pendo.segmentFlags` array
29
- * which is populated by the Pendo agent based on segment membership.
14
+ * This provider integrates with the Pendo Web SDK's segment-flags plugin
15
+ * to evaluate feature flags based on segment membership.
16
+ *
17
+ * Prerequisites:
18
+ * - Pendo Web SDK must be installed and initialized on the page
19
+ * - `requestSegmentFlags: true` must be set in the Pendo configuration
30
20
  *
31
21
  * @example
32
22
  * ```typescript
33
23
  * import { OpenFeature } from '@openfeature/web-sdk';
34
24
  * import { PendoProvider } from '@pendo/openfeature-web-provider';
35
25
  *
26
+ * // Pendo must be initialized with requestSegmentFlags enabled:
27
+ * // pendo.initialize({
28
+ * // visitor: { id: 'user-123' },
29
+ * // account: { id: 'account-456' },
30
+ * // requestSegmentFlags: true
31
+ * // });
32
+ *
36
33
  * await OpenFeature.setProviderAndWait(new PendoProvider());
37
34
  *
38
35
  * const client = OpenFeature.getClient();
39
- * const enabled = await client.getBooleanValue('myFeature', false);
36
+ * const enabled = client.getBooleanValue('myFeature', false);
40
37
  * ```
41
38
  */
42
39
  export declare class PendoProvider implements Provider {
43
40
  readonly metadata: ProviderMetadata;
44
- readonly rulesChanged?: () => void;
41
+ readonly events: OpenFeatureEventEmitter;
45
42
  status: ClientProviderStatus;
46
43
  hooks?: Hook[];
47
44
  private options;
48
- private readyPromise;
45
+ private flagChangeDetectionSetup;
46
+ private flagChangeHandler;
49
47
  constructor(options?: PendoProviderOptions);
50
48
  /**
51
49
  * Initialize the provider.
52
- * Waits for Pendo to be ready before resolving.
50
+ * Waits for Pendo to be ready and sets up flag change detection.
53
51
  */
54
52
  initialize(): Promise<void>;
55
53
  /**
56
54
  * Wait for Pendo to be ready with timeout.
57
55
  */
58
56
  private waitForPendo;
57
+ /**
58
+ * Set up detection for when segment flags change.
59
+ * Listens for the segmentFlagsUpdated event from Pendo.
60
+ * Idempotent - only runs once.
61
+ */
62
+ private setupFlagChangeDetection;
59
63
  /**
60
64
  * Shutdown the provider.
61
65
  */
@@ -84,19 +88,15 @@ export declare class PendoProvider implements Provider {
84
88
  * For Pendo flags, this returns { enabled: true/false }.
85
89
  */
86
90
  resolveObjectEvaluation<T extends JsonValue>(flagKey: string, defaultValue: T, context: EvaluationContext): ResolutionDetails<T>;
87
- /**
88
- * Get the segment flags array from Pendo.
89
- */
90
- private getSegmentFlags;
91
91
  /**
92
92
  * Track a custom event in Pendo.
93
93
  *
94
- * This method delegates to the Pendo agent's track function.
95
- * The Pendo agent must be initialized on the page for this to work.
96
- *
97
- * @param trackingEventName - The name of the event to track
98
- * @param _context - Evaluation context (unused in web provider, Pendo agent handles visitor context)
99
- * @param trackingEventDetails - Optional additional event properties
94
+ * Delegates to the Pendo Web SDK's track function.
100
95
  */
101
96
  track(trackingEventName: string, _context: EvaluationContext, trackingEventDetails: TrackingEventDetails): void;
97
+ /**
98
+ * Get the segment flags array from Pendo.
99
+ * Returns null if Pendo is not available or flags haven't been loaded.
100
+ */
101
+ private getSegmentFlags;
102
102
  }
@@ -2,21 +2,33 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.PendoProvider = void 0;
4
4
  const web_sdk_1 = require("@openfeature/web-sdk");
5
+ require("./types");
5
6
  /**
6
7
  * OpenFeature provider for Pendo feature flags.
7
8
  *
8
- * This provider evaluates feature flags by checking the `pendo.segmentFlags` array
9
- * which is populated by the Pendo agent based on segment membership.
9
+ * This provider integrates with the Pendo Web SDK's segment-flags plugin
10
+ * to evaluate feature flags based on segment membership.
11
+ *
12
+ * Prerequisites:
13
+ * - Pendo Web SDK must be installed and initialized on the page
14
+ * - `requestSegmentFlags: true` must be set in the Pendo configuration
10
15
  *
11
16
  * @example
12
17
  * ```typescript
13
18
  * import { OpenFeature } from '@openfeature/web-sdk';
14
19
  * import { PendoProvider } from '@pendo/openfeature-web-provider';
15
20
  *
21
+ * // Pendo must be initialized with requestSegmentFlags enabled:
22
+ * // pendo.initialize({
23
+ * // visitor: { id: 'user-123' },
24
+ * // account: { id: 'account-456' },
25
+ * // requestSegmentFlags: true
26
+ * // });
27
+ *
16
28
  * await OpenFeature.setProviderAndWait(new PendoProvider());
17
29
  *
18
30
  * const client = OpenFeature.getClient();
19
- * const enabled = await client.getBooleanValue('myFeature', false);
31
+ * const enabled = client.getBooleanValue('myFeature', false);
20
32
  * ```
21
33
  */
22
34
  class PendoProvider {
@@ -24,8 +36,10 @@ class PendoProvider {
24
36
  this.metadata = {
25
37
  name: "pendo-provider",
26
38
  };
39
+ this.events = new web_sdk_1.OpenFeatureEventEmitter();
27
40
  this.status = web_sdk_1.ClientProviderStatus.NOT_READY;
28
- this.readyPromise = null;
41
+ this.flagChangeDetectionSetup = false;
42
+ this.flagChangeHandler = null;
29
43
  this.options = {
30
44
  readyTimeout: 5000,
31
45
  ...options,
@@ -33,23 +47,24 @@ class PendoProvider {
33
47
  }
34
48
  /**
35
49
  * Initialize the provider.
36
- * Waits for Pendo to be ready before resolving.
50
+ * Waits for Pendo to be ready and sets up flag change detection.
37
51
  */
38
52
  async initialize() {
39
- if (this.readyPromise) {
40
- return this.readyPromise;
53
+ // Listen for pendo_ready to set up detection even if we timeout initially
54
+ if (typeof document !== "undefined") {
55
+ document.addEventListener("pendo_ready", () => this.setupFlagChangeDetection(), { once: true });
41
56
  }
42
- this.readyPromise = this.waitForPendo();
43
- await this.readyPromise;
57
+ await this.waitForPendo();
58
+ this.setupFlagChangeDetection();
44
59
  this.status = web_sdk_1.ClientProviderStatus.READY;
45
60
  }
46
61
  /**
47
62
  * Wait for Pendo to be ready with timeout.
48
63
  */
49
64
  async waitForPendo() {
50
- const timeout = this.options.readyTimeout ?? 5000;
65
+ const timeout = this.options.readyTimeout;
51
66
  const startTime = Date.now();
52
- return new Promise((resolve, reject) => {
67
+ return new Promise((resolve) => {
53
68
  const check = () => {
54
69
  // Check if pendo is available and ready
55
70
  if (typeof window !== "undefined" && window.pendo) {
@@ -60,8 +75,6 @@ class PendoProvider {
60
75
  }
61
76
  // Check timeout
62
77
  if (Date.now() - startTime > timeout) {
63
- // Don't reject - just resolve and work without Pendo
64
- // Flags will default to false
65
78
  console.warn("[PendoProvider] Pendo not ready within timeout. Flags will use default values.");
66
79
  resolve();
67
80
  return;
@@ -69,21 +82,46 @@ class PendoProvider {
69
82
  // Try again
70
83
  setTimeout(check, 100);
71
84
  };
72
- // Also listen for pendo:ready event
85
+ // Also listen for pendo_ready event
73
86
  if (typeof document !== "undefined") {
74
- document.addEventListener("pendo:ready", () => {
75
- resolve();
76
- }, { once: true });
87
+ document.addEventListener("pendo_ready", () => resolve(), { once: true });
77
88
  }
78
89
  check();
79
90
  });
80
91
  }
92
+ /**
93
+ * Set up detection for when segment flags change.
94
+ * Listens for the segmentFlagsUpdated event from Pendo.
95
+ * Idempotent - only runs once.
96
+ */
97
+ setupFlagChangeDetection() {
98
+ if (this.flagChangeDetectionSetup) {
99
+ return;
100
+ }
101
+ if (typeof window === "undefined" || !window.pendo?.Events?.segmentFlagsUpdated) {
102
+ return;
103
+ }
104
+ this.flagChangeDetectionSetup = true;
105
+ // Create handler and store reference for cleanup
106
+ this.flagChangeHandler = () => {
107
+ this.events.emit(web_sdk_1.ProviderEvents.ConfigurationChanged);
108
+ };
109
+ // Subscribe to Pendo's segmentFlagsUpdated event
110
+ window.pendo.Events.segmentFlagsUpdated.on(this.flagChangeHandler);
111
+ }
81
112
  /**
82
113
  * Shutdown the provider.
83
114
  */
84
115
  async onClose() {
116
+ // Unsubscribe from Pendo events
117
+ if (typeof window !== "undefined" &&
118
+ window.pendo?.Events?.segmentFlagsUpdated &&
119
+ this.flagChangeHandler) {
120
+ window.pendo.Events.segmentFlagsUpdated.off(this.flagChangeHandler);
121
+ }
85
122
  this.status = web_sdk_1.ClientProviderStatus.NOT_READY;
86
- this.readyPromise = null;
123
+ this.flagChangeHandler = null;
124
+ this.flagChangeDetectionSetup = false;
87
125
  }
88
126
  /**
89
127
  * Resolve a boolean flag value.
@@ -92,7 +130,7 @@ class PendoProvider {
92
130
  */
93
131
  resolveBooleanEvaluation(flagKey, defaultValue, _context) {
94
132
  const flags = this.getSegmentFlags();
95
- if (!flags) {
133
+ if (flags === null) {
96
134
  return {
97
135
  value: defaultValue,
98
136
  reason: "DEFAULT",
@@ -166,40 +204,38 @@ class PendoProvider {
166
204
  variant: boolResult.variant,
167
205
  };
168
206
  }
169
- /**
170
- * Get the segment flags array from Pendo.
171
- */
172
- getSegmentFlags() {
173
- if (typeof window === "undefined") {
174
- return undefined;
175
- }
176
- return window.pendo?.segmentFlags;
177
- }
178
207
  /**
179
208
  * Track a custom event in Pendo.
180
209
  *
181
- * This method delegates to the Pendo agent's track function.
182
- * The Pendo agent must be initialized on the page for this to work.
183
- *
184
- * @param trackingEventName - The name of the event to track
185
- * @param _context - Evaluation context (unused in web provider, Pendo agent handles visitor context)
186
- * @param trackingEventDetails - Optional additional event properties
210
+ * Delegates to the Pendo Web SDK's track function.
187
211
  */
188
212
  track(trackingEventName, _context, trackingEventDetails) {
189
213
  if (typeof window === "undefined" || !window.pendo?.track) {
190
- console.warn("[PendoProvider] Pendo agent not available for tracking");
214
+ console.warn("[PendoProvider] Pendo Web SDK not available for tracking");
191
215
  return;
192
216
  }
193
- // Convert TrackingEventDetails to Record<string, string> for Pendo agent
217
+ // Convert TrackingEventDetails to Record<string, string> for Pendo Web SDK
194
218
  const properties = {};
195
219
  for (const [key, value] of Object.entries(trackingEventDetails)) {
196
220
  if (value !== undefined && value !== null) {
197
221
  properties[key] = String(value);
198
222
  }
199
223
  }
200
- // Fire-and-forget: delegate to Pendo agent
201
224
  window.pendo.track(trackingEventName, properties);
202
225
  }
226
+ /**
227
+ * Get the segment flags array from Pendo.
228
+ * Returns null if Pendo is not available or flags haven't been loaded.
229
+ */
230
+ getSegmentFlags() {
231
+ if (typeof window === "undefined" || !window.pendo) {
232
+ return null;
233
+ }
234
+ const flags = window.pendo.segmentFlags;
235
+ // Return null if flags haven't been loaded yet (undefined)
236
+ // Return the array (even if empty) if flags have been loaded
237
+ return flags === undefined ? null : flags;
238
+ }
203
239
  }
204
240
  exports.PendoProvider = PendoProvider;
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PendoProvider.js","sourceRoot":"","sources":["../src/PendoProvider.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Provider,\n  ProviderMetadata,\n  ResolutionDetails,\n  EvaluationContext,\n  JsonValue,\n  Hook,\n  TrackingEventDetails,\n} from \"@openfeature/web-sdk\";\nimport { ClientProviderStatus } from \"@openfeature/web-sdk\";\n\ndeclare global {\n  interface Window {\n    pendo?: {\n      segmentFlags?: string[];\n      isReady?: () => boolean;\n      initialize?: (options: unknown) => void;\n      track?: (event: string, properties?: Record<string, string>) => void;\n    };\n  }\n}\n\nexport interface PendoProviderOptions {\n  /**\n   * Optional: Pendo API key for initialization.\n   * If not provided, assumes Pendo is already initialized on the page.\n   */\n  apiKey?: string;\n\n  /**\n   * Optional: Timeout in milliseconds to wait for Pendo to be ready.\n   * Default: 5000ms\n   */\n  readyTimeout?: number;\n}\n\n/**\n * OpenFeature provider for Pendo feature flags.\n *\n * This provider evaluates feature flags by checking the `pendo.segmentFlags` array\n * which is populated by the Pendo agent based on segment membership.\n *\n * @example\n * ```typescript\n * import { OpenFeature } from '@openfeature/web-sdk';\n * import { PendoProvider } from '@pendo/openfeature-web-provider';\n *\n * await OpenFeature.setProviderAndWait(new PendoProvider());\n *\n * const client = OpenFeature.getClient();\n * const enabled = await client.getBooleanValue('myFeature', false);\n * ```\n */\nexport class PendoProvider implements Provider {\n  readonly metadata: ProviderMetadata = {\n    name: \"pendo-provider\",\n  };\n\n  readonly rulesChanged?: () => void;\n  status: ClientProviderStatus = ClientProviderStatus.NOT_READY;\n  hooks?: Hook[];\n\n  private options: PendoProviderOptions;\n  private readyPromise: Promise<void> | null = null;\n\n  constructor(options: PendoProviderOptions = {}) {\n    this.options = {\n      readyTimeout: 5000,\n      ...options,\n    };\n  }\n\n  /**\n   * Initialize the provider.\n   * Waits for Pendo to be ready before resolving.\n   */\n  async initialize(): Promise<void> {\n    if (this.readyPromise) {\n      return this.readyPromise;\n    }\n\n    this.readyPromise = this.waitForPendo();\n    await this.readyPromise;\n    this.status = ClientProviderStatus.READY;\n  }\n\n  /**\n   * Wait for Pendo to be ready with timeout.\n   */\n  private async waitForPendo(): Promise<void> {\n    const timeout = this.options.readyTimeout ?? 5000;\n    const startTime = Date.now();\n\n    return new Promise((resolve, reject) => {\n      const check = () => {\n        // Check if pendo is available and ready\n        if (typeof window !== \"undefined\" && window.pendo) {\n          if (window.pendo.isReady?.() || window.pendo.segmentFlags !== undefined) {\n            resolve();\n            return;\n          }\n        }\n\n        // Check timeout\n        if (Date.now() - startTime > timeout) {\n          // Don't reject - just resolve and work without Pendo\n          // Flags will default to false\n          console.warn(\n            \"[PendoProvider] Pendo not ready within timeout. Flags will use default values.\"\n          );\n          resolve();\n          return;\n        }\n\n        // Try again\n        setTimeout(check, 100);\n      };\n\n      // Also listen for pendo:ready event\n      if (typeof document !== \"undefined\") {\n        document.addEventListener(\n          \"pendo:ready\",\n          () => {\n            resolve();\n          },\n          { once: true }\n        );\n      }\n\n      check();\n    });\n  }\n\n  /**\n   * Shutdown the provider.\n   */\n  async onClose(): Promise<void> {\n    this.status = ClientProviderStatus.NOT_READY;\n    this.readyPromise = null;\n  }\n\n  /**\n   * Resolve a boolean flag value.\n   *\n   * Checks if the flagKey exists in pendo.segmentFlags array.\n   */\n  resolveBooleanEvaluation(\n    flagKey: string,\n    defaultValue: boolean,\n    _context: EvaluationContext\n  ): ResolutionDetails<boolean> {\n    const flags = this.getSegmentFlags();\n\n    if (!flags) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    const enabled = flags.includes(flagKey);\n\n    return {\n      value: enabled,\n      reason: enabled ? \"TARGETING_MATCH\" : \"DEFAULT\",\n      variant: enabled ? \"on\" : \"off\",\n    };\n  }\n\n  /**\n   * Resolve a string flag value.\n   *\n   * For Pendo flags, this returns \"on\" if the flag is enabled, \"off\" otherwise.\n   */\n  resolveStringEvaluation(\n    flagKey: string,\n    defaultValue: string,\n    context: EvaluationContext\n  ): ResolutionDetails<string> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: boolResult.value ? \"on\" : \"off\",\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Resolve a number flag value.\n   *\n   * For Pendo flags, this returns 1 if enabled, 0 if disabled.\n   */\n  resolveNumberEvaluation(\n    flagKey: string,\n    defaultValue: number,\n    context: EvaluationContext\n  ): ResolutionDetails<number> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: boolResult.value ? 1 : 0,\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Resolve an object flag value.\n   *\n   * For Pendo flags, this returns { enabled: true/false }.\n   */\n  resolveObjectEvaluation<T extends JsonValue>(\n    flagKey: string,\n    defaultValue: T,\n    context: EvaluationContext\n  ): ResolutionDetails<T> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: { enabled: boolResult.value } as unknown as T,\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Get the segment flags array from Pendo.\n   */\n  private getSegmentFlags(): string[] | undefined {\n    if (typeof window === \"undefined\") {\n      return undefined;\n    }\n\n    return window.pendo?.segmentFlags;\n  }\n\n  /**\n   * Track a custom event in Pendo.\n   *\n   * This method delegates to the Pendo agent's track function.\n   * The Pendo agent must be initialized on the page for this to work.\n   *\n   * @param trackingEventName - The name of the event to track\n   * @param _context - Evaluation context (unused in web provider, Pendo agent handles visitor context)\n   * @param trackingEventDetails - Optional additional event properties\n   */\n  track(\n    trackingEventName: string,\n    _context: EvaluationContext,\n    trackingEventDetails: TrackingEventDetails\n  ): void {\n    if (typeof window === \"undefined\" || !window.pendo?.track) {\n      console.warn(\"[PendoProvider] Pendo agent not available for tracking\");\n      return;\n    }\n\n    // Convert TrackingEventDetails to Record<string, string> for Pendo agent\n    const properties: Record<string, string> = {};\n    for (const [key, value] of Object.entries(trackingEventDetails)) {\n      if (value !== undefined && value !== null) {\n        properties[key] = String(value);\n      }\n    }\n\n    // Fire-and-forget: delegate to Pendo agent\n    window.pendo.track(trackingEventName, properties);\n  }\n}\n"]}
241
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PendoProvider.js","sourceRoot":"","sources":["../src/PendoProvider.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Provider,\n  ProviderMetadata,\n  ResolutionDetails,\n  EvaluationContext,\n  JsonValue,\n  Hook,\n  TrackingEventDetails,\n} from \"@openfeature/web-sdk\";\nimport {\n  ClientProviderStatus,\n  OpenFeatureEventEmitter,\n  ProviderEvents,\n} from \"@openfeature/web-sdk\";\nimport \"./types\";\n\nexport interface PendoProviderOptions {\n  /**\n   * Timeout in milliseconds to wait for Pendo to be ready.\n   * Default: 5000ms\n   */\n  readyTimeout?: number;\n}\n\n/**\n * OpenFeature provider for Pendo feature flags.\n *\n * This provider integrates with the Pendo Web SDK's segment-flags plugin\n * to evaluate feature flags based on segment membership.\n *\n * Prerequisites:\n * - Pendo Web SDK must be installed and initialized on the page\n * - `requestSegmentFlags: true` must be set in the Pendo configuration\n *\n * @example\n * ```typescript\n * import { OpenFeature } from '@openfeature/web-sdk';\n * import { PendoProvider } from '@pendo/openfeature-web-provider';\n *\n * // Pendo must be initialized with requestSegmentFlags enabled:\n * // pendo.initialize({\n * //   visitor: { id: 'user-123' },\n * //   account: { id: 'account-456' },\n * //   requestSegmentFlags: true\n * // });\n *\n * await OpenFeature.setProviderAndWait(new PendoProvider());\n *\n * const client = OpenFeature.getClient();\n * const enabled = client.getBooleanValue('myFeature', false);\n * ```\n */\nexport class PendoProvider implements Provider {\n  readonly metadata: ProviderMetadata = {\n    name: \"pendo-provider\",\n  };\n\n  readonly events = new OpenFeatureEventEmitter();\n  status: ClientProviderStatus = ClientProviderStatus.NOT_READY;\n  hooks?: Hook[];\n\n  private options: Required<PendoProviderOptions>;\n  private flagChangeDetectionSetup = false;\n  private flagChangeHandler: (() => void) | null = null;\n\n  constructor(options: PendoProviderOptions = {}) {\n    this.options = {\n      readyTimeout: 5000,\n      ...options,\n    };\n  }\n\n  /**\n   * Initialize the provider.\n   * Waits for Pendo to be ready and sets up flag change detection.\n   */\n  async initialize(): Promise<void> {\n    // Listen for pendo_ready to set up detection even if we timeout initially\n    if (typeof document !== \"undefined\") {\n      document.addEventListener(\n        \"pendo_ready\",\n        () => this.setupFlagChangeDetection(),\n        { once: true }\n      );\n    }\n\n    await this.waitForPendo();\n    this.setupFlagChangeDetection();\n    this.status = ClientProviderStatus.READY;\n  }\n\n  /**\n   * Wait for Pendo to be ready with timeout.\n   */\n  private async waitForPendo(): Promise<void> {\n    const timeout = this.options.readyTimeout;\n    const startTime = Date.now();\n\n    return new Promise((resolve) => {\n      const check = () => {\n        // Check if pendo is available and ready\n        if (typeof window !== \"undefined\" && window.pendo) {\n          if (window.pendo.isReady?.() || window.pendo.segmentFlags !== undefined) {\n            resolve();\n            return;\n          }\n        }\n\n        // Check timeout\n        if (Date.now() - startTime > timeout) {\n          console.warn(\n            \"[PendoProvider] Pendo not ready within timeout. Flags will use default values.\"\n          );\n          resolve();\n          return;\n        }\n\n        // Try again\n        setTimeout(check, 100);\n      };\n\n      // Also listen for pendo_ready event\n      if (typeof document !== \"undefined\") {\n        document.addEventListener(\n          \"pendo_ready\",\n          () => resolve(),\n          { once: true }\n        );\n      }\n\n      check();\n    });\n  }\n\n  /**\n   * Set up detection for when segment flags change.\n   * Listens for the segmentFlagsUpdated event from Pendo.\n   * Idempotent - only runs once.\n   */\n  private setupFlagChangeDetection(): void {\n    if (this.flagChangeDetectionSetup) {\n      return;\n    }\n\n    if (typeof window === \"undefined\" || !window.pendo?.Events?.segmentFlagsUpdated) {\n      return;\n    }\n\n    this.flagChangeDetectionSetup = true;\n\n    // Create handler and store reference for cleanup\n    this.flagChangeHandler = () => {\n      this.events.emit(ProviderEvents.ConfigurationChanged);\n    };\n\n    // Subscribe to Pendo's segmentFlagsUpdated event\n    window.pendo.Events.segmentFlagsUpdated.on(this.flagChangeHandler);\n  }\n\n  /**\n   * Shutdown the provider.\n   */\n  async onClose(): Promise<void> {\n    // Unsubscribe from Pendo events\n    if (\n      typeof window !== \"undefined\" &&\n      window.pendo?.Events?.segmentFlagsUpdated &&\n      this.flagChangeHandler\n    ) {\n      window.pendo.Events.segmentFlagsUpdated.off(this.flagChangeHandler);\n    }\n\n    this.status = ClientProviderStatus.NOT_READY;\n    this.flagChangeHandler = null;\n    this.flagChangeDetectionSetup = false;\n  }\n\n  /**\n   * Resolve a boolean flag value.\n   *\n   * Checks if the flagKey exists in pendo.segmentFlags array.\n   */\n  resolveBooleanEvaluation(\n    flagKey: string,\n    defaultValue: boolean,\n    _context: EvaluationContext\n  ): ResolutionDetails<boolean> {\n    const flags = this.getSegmentFlags();\n\n    if (flags === null) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    const enabled = flags.includes(flagKey);\n\n    return {\n      value: enabled,\n      reason: enabled ? \"TARGETING_MATCH\" : \"DEFAULT\",\n      variant: enabled ? \"on\" : \"off\",\n    };\n  }\n\n  /**\n   * Resolve a string flag value.\n   *\n   * For Pendo flags, this returns \"on\" if the flag is enabled, \"off\" otherwise.\n   */\n  resolveStringEvaluation(\n    flagKey: string,\n    defaultValue: string,\n    context: EvaluationContext\n  ): ResolutionDetails<string> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: boolResult.value ? \"on\" : \"off\",\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Resolve a number flag value.\n   *\n   * For Pendo flags, this returns 1 if enabled, 0 if disabled.\n   */\n  resolveNumberEvaluation(\n    flagKey: string,\n    defaultValue: number,\n    context: EvaluationContext\n  ): ResolutionDetails<number> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: boolResult.value ? 1 : 0,\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Resolve an object flag value.\n   *\n   * For Pendo flags, this returns { enabled: true/false }.\n   */\n  resolveObjectEvaluation<T extends JsonValue>(\n    flagKey: string,\n    defaultValue: T,\n    context: EvaluationContext\n  ): ResolutionDetails<T> {\n    const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);\n\n    if (boolResult.reason === \"DEFAULT\" && !boolResult.value) {\n      return {\n        value: defaultValue,\n        reason: \"DEFAULT\",\n        variant: \"default\",\n      };\n    }\n\n    return {\n      value: { enabled: boolResult.value } as unknown as T,\n      reason: boolResult.reason,\n      variant: boolResult.variant,\n    };\n  }\n\n  /**\n   * Track a custom event in Pendo.\n   *\n   * Delegates to the Pendo Web SDK's track function.\n   */\n  track(\n    trackingEventName: string,\n    _context: EvaluationContext,\n    trackingEventDetails: TrackingEventDetails\n  ): void {\n    if (typeof window === \"undefined\" || !window.pendo?.track) {\n      console.warn(\"[PendoProvider] Pendo Web SDK not available for tracking\");\n      return;\n    }\n\n    // Convert TrackingEventDetails to Record<string, string> for Pendo Web SDK\n    const properties: Record<string, string> = {};\n    for (const [key, value] of Object.entries(trackingEventDetails)) {\n      if (value !== undefined && value !== null) {\n        properties[key] = String(value);\n      }\n    }\n\n    window.pendo.track(trackingEventName, properties);\n  }\n\n  /**\n   * Get the segment flags array from Pendo.\n   * Returns null if Pendo is not available or flags haven't been loaded.\n   */\n  private getSegmentFlags(): string[] | null {\n    if (typeof window === \"undefined\" || !window.pendo) {\n      return null;\n    }\n\n    const flags = window.pendo.segmentFlags;\n\n    // Return null if flags haven't been loaded yet (undefined)\n    // Return the array (even if empty) if flags have been loaded\n    return flags === undefined ? null : flags;\n  }\n}\n"]}
@@ -1,4 +1,5 @@
1
1
  import type { Hook, HookContext, EvaluationDetails, FlagValue } from "@openfeature/web-sdk";
2
+ import "./types";
2
3
  export interface PendoTelemetryHookOptions {
3
4
  /**
4
5
  * Event name for flag evaluation tracking.
@@ -17,7 +18,7 @@ export interface PendoTelemetryHookOptions {
17
18
  * This hook sends an event to Pendo after each flag evaluation, allowing you
18
19
  * to analyze feature flag usage alongside your other Pendo analytics.
19
20
  *
20
- * The hook uses the Pendo agent's track function, so the Pendo agent must be
21
+ * The hook uses the Pendo Web SDK's track function, so the Pendo Web SDK must be
21
22
  * initialized on the page for tracking to work.
22
23
  *
23
24
  * @example
@@ -1,13 +1,14 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.PendoTelemetryHook = void 0;
4
+ require("./types");
4
5
  /**
5
6
  * OpenFeature hook that automatically tracks flag evaluations to Pendo.
6
7
  *
7
8
  * This hook sends an event to Pendo after each flag evaluation, allowing you
8
9
  * to analyze feature flag usage alongside your other Pendo analytics.
9
10
  *
10
- * The hook uses the Pendo agent's track function, so the Pendo agent must be
11
+ * The hook uses the Pendo Web SDK's track function, so the Pendo Web SDK must be
11
12
  * initialized on the page for tracking to work.
12
13
  *
13
14
  * @example
@@ -39,7 +40,7 @@ class PendoTelemetryHook {
39
40
  if (this.options.flagFilter && !this.options.flagFilter(flagKey)) {
40
41
  return;
41
42
  }
42
- // Check if Pendo agent is available
43
+ // Check if Pendo Web SDK is available
43
44
  if (typeof window === "undefined" || !window.pendo?.track) {
44
45
  return;
45
46
  }
@@ -50,7 +51,7 @@ class PendoTelemetryHook {
50
51
  flag_value: this.stringifyValue(evaluationDetails.value),
51
52
  provider_name: hookContext.providerMetadata.name,
52
53
  };
53
- // Fire-and-forget: delegate to Pendo agent
54
+ // Fire-and-forget: delegate to Pendo Web SDK
54
55
  window.pendo.track(this.options.eventName, properties);
55
56
  }
56
57
  /**
@@ -67,4 +68,4 @@ class PendoTelemetryHook {
67
68
  }
68
69
  }
69
70
  exports.PendoTelemetryHook = PendoTelemetryHook;
70
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUGVuZG9UZWxlbWV0cnlIb29rLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL1BlbmRvVGVsZW1ldHJ5SG9vay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFxQkE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHO0FBQ0gsTUFBYSxrQkFBa0I7SUFLN0IsWUFBWSxVQUFxQyxFQUFFO1FBQ2pELElBQUksQ0FBQyxPQUFPLEdBQUc7WUFDYixTQUFTLEVBQUUsZ0JBQWdCO1lBQzNCLEdBQUcsT0FBTztTQUNYLENBQUM7SUFDSixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsS0FBSyxDQUNILFdBQTZDLEVBQzdDLGlCQUErQztRQUUvQyxNQUFNLEVBQUUsT0FBTyxFQUFFLEdBQUcsV0FBVyxDQUFDO1FBRWhDLHVDQUF1QztRQUN2QyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNqRSxPQUFPO1FBQ1QsQ0FBQztRQUVELG9DQUFvQztRQUNwQyxJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDMUQsT0FBTztRQUNULENBQUM7UUFFRCxNQUFNLFVBQVUsR0FBMkI7WUFDekMsUUFBUSxFQUFFLE9BQU87WUFDakIsWUFBWSxFQUFFLGlCQUFpQixDQUFDLE9BQU8sSUFBSSxTQUFTO1lBQ3BELFdBQVcsRUFBRSxpQkFBaUIsQ0FBQyxNQUFNLElBQUksU0FBUztZQUNsRCxVQUFVLEVBQUUsSUFBSSxDQUFDLGNBQWMsQ0FBQyxpQkFBaUIsQ0FBQyxLQUFLLENBQUM7WUFDeEQsYUFBYSxFQUFFLFdBQVcsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJO1NBQ2pELENBQUM7UUFFRiwyQ0FBMkM7UUFDM0MsTUFBTSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxTQUFTLEVBQUUsVUFBVSxDQUFDLENBQUM7SUFDekQsQ0FBQztJQUVEOztPQUVHO0lBQ0ssY0FBYyxDQUFDLEtBQWdCO1FBQ3JDLElBQUksS0FBSyxLQUFLLElBQUksRUFBRSxDQUFDO1lBQ25CLE9BQU8sTUFBTSxDQUFDO1FBQ2hCLENBQUM7UUFDRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzlCLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMvQixDQUFDO1FBQ0QsT0FBTyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDdkIsQ0FBQztDQUNGO0FBeERELGdEQXdEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgSG9vayxcbiAgSG9va0NvbnRleHQsXG4gIEV2YWx1YXRpb25EZXRhaWxzLFxuICBGbGFnVmFsdWUsXG59IGZyb20gXCJAb3BlbmZlYXR1cmUvd2ViLXNka1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBlbmRvVGVsZW1ldHJ5SG9va09wdGlvbnMge1xuICAvKipcbiAgICogRXZlbnQgbmFtZSBmb3IgZmxhZyBldmFsdWF0aW9uIHRyYWNraW5nLlxuICAgKiBEZWZhdWx0OiBcImZsYWdfZXZhbHVhdGVkXCJcbiAgICovXG4gIGV2ZW50TmFtZT86IHN0cmluZztcblxuICAvKipcbiAgICogT3B0aW9uYWwgZmlsdGVyIGZ1bmN0aW9uIHRvIHNlbGVjdGl2ZWx5IHRyYWNrIGZsYWdzLlxuICAgKiBSZXR1cm4gdHJ1ZSB0byB0cmFjayB0aGUgZmxhZywgZmFsc2UgdG8gc2tpcC5cbiAgICovXG4gIGZsYWdGaWx0ZXI/OiAoZmxhZ0tleTogc3RyaW5nKSA9PiBib29sZWFuO1xufVxuXG4vKipcbiAqIE9wZW5GZWF0dXJlIGhvb2sgdGhhdCBhdXRvbWF0aWNhbGx5IHRyYWNrcyBmbGFnIGV2YWx1YXRpb25zIHRvIFBlbmRvLlxuICpcbiAqIFRoaXMgaG9vayBzZW5kcyBhbiBldmVudCB0byBQZW5kbyBhZnRlciBlYWNoIGZsYWcgZXZhbHVhdGlvbiwgYWxsb3dpbmcgeW91XG4gKiB0byBhbmFseXplIGZlYXR1cmUgZmxhZyB1c2FnZSBhbG9uZ3NpZGUgeW91ciBvdGhlciBQZW5kbyBhbmFseXRpY3MuXG4gKlxuICogVGhlIGhvb2sgdXNlcyB0aGUgUGVuZG8gYWdlbnQncyB0cmFjayBmdW5jdGlvbiwgc28gdGhlIFBlbmRvIGFnZW50IG11c3QgYmVcbiAqIGluaXRpYWxpemVkIG9uIHRoZSBwYWdlIGZvciB0cmFja2luZyB0byB3b3JrLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBPcGVuRmVhdHVyZSB9IGZyb20gJ0BvcGVuZmVhdHVyZS93ZWItc2RrJztcbiAqIGltcG9ydCB7IFBlbmRvUHJvdmlkZXIsIFBlbmRvVGVsZW1ldHJ5SG9vayB9IGZyb20gJ0BwZW5kby9vcGVuZmVhdHVyZS13ZWItcHJvdmlkZXInO1xuICpcbiAqIGNvbnN0IHByb3ZpZGVyID0gbmV3IFBlbmRvUHJvdmlkZXIoKTtcbiAqIGNvbnN0IHRlbGVtZXRyeUhvb2sgPSBuZXcgUGVuZG9UZWxlbWV0cnlIb29rKCk7XG4gKlxuICogYXdhaXQgT3BlbkZlYXR1cmUuc2V0UHJvdmlkZXJBbmRXYWl0KHByb3ZpZGVyKTtcbiAqIE9wZW5GZWF0dXJlLmFkZEhvb2tzKHRlbGVtZXRyeUhvb2spO1xuICogYGBgXG4gKi9cbmV4cG9ydCBjbGFzcyBQZW5kb1RlbGVtZXRyeUhvb2sgaW1wbGVtZW50cyBIb29rIHtcbiAgcHJpdmF0ZSBvcHRpb25zOiBSZXF1aXJlZDxPbWl0PFBlbmRvVGVsZW1ldHJ5SG9va09wdGlvbnMsIFwiZmxhZ0ZpbHRlclwiPj4gJiB7XG4gICAgZmxhZ0ZpbHRlcj86IChmbGFnS2V5OiBzdHJpbmcpID0+IGJvb2xlYW47XG4gIH07XG5cbiAgY29uc3RydWN0b3Iob3B0aW9uczogUGVuZG9UZWxlbWV0cnlIb29rT3B0aW9ucyA9IHt9KSB7XG4gICAgdGhpcy5vcHRpb25zID0ge1xuICAgICAgZXZlbnROYW1lOiBcImZsYWdfZXZhbHVhdGVkXCIsXG4gICAgICAuLi5vcHRpb25zLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ2FsbGVkIGFmdGVyIGEgZmxhZyBpcyBzdWNjZXNzZnVsbHkgZXZhbHVhdGVkLlxuICAgKiBTZW5kcyB0aGUgZXZhbHVhdGlvbiByZXN1bHQgdG8gUGVuZG8gYXMgYSB0cmFjayBldmVudC5cbiAgICovXG4gIGFmdGVyKFxuICAgIGhvb2tDb250ZXh0OiBSZWFkb25seTxIb29rQ29udGV4dDxGbGFnVmFsdWU+PixcbiAgICBldmFsdWF0aW9uRGV0YWlsczogRXZhbHVhdGlvbkRldGFpbHM8RmxhZ1ZhbHVlPlxuICApOiB2b2lkIHtcbiAgICBjb25zdCB7IGZsYWdLZXkgfSA9IGhvb2tDb250ZXh0O1xuXG4gICAgLy8gQ2hlY2sgaWYgdGhpcyBmbGFnIHNob3VsZCBiZSB0cmFja2VkXG4gICAgaWYgKHRoaXMub3B0aW9ucy5mbGFnRmlsdGVyICYmICF0aGlzLm9wdGlvbnMuZmxhZ0ZpbHRlcihmbGFnS2V5KSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIENoZWNrIGlmIFBlbmRvIGFnZW50IGlzIGF2YWlsYWJsZVxuICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSBcInVuZGVmaW5lZFwiIHx8ICF3aW5kb3cucGVuZG8/LnRyYWNrKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgcHJvcGVydGllczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgIGZsYWdfa2V5OiBmbGFnS2V5LFxuICAgICAgZmxhZ192YXJpYW50OiBldmFsdWF0aW9uRGV0YWlscy52YXJpYW50IHx8IFwidW5rbm93blwiLFxuICAgICAgZmxhZ19yZWFzb246IGV2YWx1YXRpb25EZXRhaWxzLnJlYXNvbiB8fCBcIlVOS05PV05cIixcbiAgICAgIGZsYWdfdmFsdWU6IHRoaXMuc3RyaW5naWZ5VmFsdWUoZXZhbHVhdGlvbkRldGFpbHMudmFsdWUpLFxuICAgICAgcHJvdmlkZXJfbmFtZTogaG9va0NvbnRleHQucHJvdmlkZXJNZXRhZGF0YS5uYW1lLFxuICAgIH07XG5cbiAgICAvLyBGaXJlLWFuZC1mb3JnZXQ6IGRlbGVnYXRlIHRvIFBlbmRvIGFnZW50XG4gICAgd2luZG93LnBlbmRvLnRyYWNrKHRoaXMub3B0aW9ucy5ldmVudE5hbWUsIHByb3BlcnRpZXMpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnQgYSBmbGFnIHZhbHVlIHRvIGEgc3RyaW5nIGZvciB0cmFja2luZy5cbiAgICovXG4gIHByaXZhdGUgc3RyaW5naWZ5VmFsdWUodmFsdWU6IEZsYWdWYWx1ZSk6IHN0cmluZyB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICByZXR1cm4gXCJudWxsXCI7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIHJldHVybiBKU09OLnN0cmluZ2lmeSh2YWx1ZSk7XG4gICAgfVxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpO1xuICB9XG59XG4iXX0=
71
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUGVuZG9UZWxlbWV0cnlIb29rLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL1BlbmRvVGVsZW1ldHJ5SG9vay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFNQSxtQkFBaUI7QUFnQmpCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9CRztBQUNILE1BQWEsa0JBQWtCO0lBSzdCLFlBQVksVUFBcUMsRUFBRTtRQUNqRCxJQUFJLENBQUMsT0FBTyxHQUFHO1lBQ2IsU0FBUyxFQUFFLGdCQUFnQjtZQUMzQixHQUFHLE9BQU87U0FDWCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7T0FHRztJQUNILEtBQUssQ0FDSCxXQUE2QyxFQUM3QyxpQkFBK0M7UUFFL0MsTUFBTSxFQUFFLE9BQU8sRUFBRSxHQUFHLFdBQVcsQ0FBQztRQUVoQyx1Q0FBdUM7UUFDdkMsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDakUsT0FBTztRQUNULENBQUM7UUFFRCxzQ0FBc0M7UUFDdEMsSUFBSSxPQUFPLE1BQU0sS0FBSyxXQUFXLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQzFELE9BQU87UUFDVCxDQUFDO1FBRUQsTUFBTSxVQUFVLEdBQTJCO1lBQ3pDLFFBQVEsRUFBRSxPQUFPO1lBQ2pCLFlBQVksRUFBRSxpQkFBaUIsQ0FBQyxPQUFPLElBQUksU0FBUztZQUNwRCxXQUFXLEVBQUUsaUJBQWlCLENBQUMsTUFBTSxJQUFJLFNBQVM7WUFDbEQsVUFBVSxFQUFFLElBQUksQ0FBQyxjQUFjLENBQUMsaUJBQWlCLENBQUMsS0FBSyxDQUFDO1lBQ3hELGFBQWEsRUFBRSxXQUFXLENBQUMsZ0JBQWdCLENBQUMsSUFBSTtTQUNqRCxDQUFDO1FBRUYsNkNBQTZDO1FBQzdDLE1BQU0sQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLFVBQVUsQ0FBQyxDQUFDO0lBQ3pELENBQUM7SUFFRDs7T0FFRztJQUNLLGNBQWMsQ0FBQyxLQUFnQjtRQUNyQyxJQUFJLEtBQUssS0FBSyxJQUFJLEVBQUUsQ0FBQztZQUNuQixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQ0QsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUM5QixPQUFPLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDL0IsQ0FBQztRQUNELE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3ZCLENBQUM7Q0FDRjtBQXhERCxnREF3REMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7XG4gIEhvb2ssXG4gIEhvb2tDb250ZXh0LFxuICBFdmFsdWF0aW9uRGV0YWlscyxcbiAgRmxhZ1ZhbHVlLFxufSBmcm9tIFwiQG9wZW5mZWF0dXJlL3dlYi1zZGtcIjtcbmltcG9ydCBcIi4vdHlwZXNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBQZW5kb1RlbGVtZXRyeUhvb2tPcHRpb25zIHtcbiAgLyoqXG4gICAqIEV2ZW50IG5hbWUgZm9yIGZsYWcgZXZhbHVhdGlvbiB0cmFja2luZy5cbiAgICogRGVmYXVsdDogXCJmbGFnX2V2YWx1YXRlZFwiXG4gICAqL1xuICBldmVudE5hbWU/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGZpbHRlciBmdW5jdGlvbiB0byBzZWxlY3RpdmVseSB0cmFjayBmbGFncy5cbiAgICogUmV0dXJuIHRydWUgdG8gdHJhY2sgdGhlIGZsYWcsIGZhbHNlIHRvIHNraXAuXG4gICAqL1xuICBmbGFnRmlsdGVyPzogKGZsYWdLZXk6IHN0cmluZykgPT4gYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBPcGVuRmVhdHVyZSBob29rIHRoYXQgYXV0b21hdGljYWxseSB0cmFja3MgZmxhZyBldmFsdWF0aW9ucyB0byBQZW5kby5cbiAqXG4gKiBUaGlzIGhvb2sgc2VuZHMgYW4gZXZlbnQgdG8gUGVuZG8gYWZ0ZXIgZWFjaCBmbGFnIGV2YWx1YXRpb24sIGFsbG93aW5nIHlvdVxuICogdG8gYW5hbHl6ZSBmZWF0dXJlIGZsYWcgdXNhZ2UgYWxvbmdzaWRlIHlvdXIgb3RoZXIgUGVuZG8gYW5hbHl0aWNzLlxuICpcbiAqIFRoZSBob29rIHVzZXMgdGhlIFBlbmRvIFdlYiBTREsncyB0cmFjayBmdW5jdGlvbiwgc28gdGhlIFBlbmRvIFdlYiBTREsgbXVzdCBiZVxuICogaW5pdGlhbGl6ZWQgb24gdGhlIHBhZ2UgZm9yIHRyYWNraW5nIHRvIHdvcmsuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IE9wZW5GZWF0dXJlIH0gZnJvbSAnQG9wZW5mZWF0dXJlL3dlYi1zZGsnO1xuICogaW1wb3J0IHsgUGVuZG9Qcm92aWRlciwgUGVuZG9UZWxlbWV0cnlIb29rIH0gZnJvbSAnQHBlbmRvL29wZW5mZWF0dXJlLXdlYi1wcm92aWRlcic7XG4gKlxuICogY29uc3QgcHJvdmlkZXIgPSBuZXcgUGVuZG9Qcm92aWRlcigpO1xuICogY29uc3QgdGVsZW1ldHJ5SG9vayA9IG5ldyBQZW5kb1RlbGVtZXRyeUhvb2soKTtcbiAqXG4gKiBhd2FpdCBPcGVuRmVhdHVyZS5zZXRQcm92aWRlckFuZFdhaXQocHJvdmlkZXIpO1xuICogT3BlbkZlYXR1cmUuYWRkSG9va3ModGVsZW1ldHJ5SG9vayk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNsYXNzIFBlbmRvVGVsZW1ldHJ5SG9vayBpbXBsZW1lbnRzIEhvb2sge1xuICBwcml2YXRlIG9wdGlvbnM6IFJlcXVpcmVkPE9taXQ8UGVuZG9UZWxlbWV0cnlIb29rT3B0aW9ucywgXCJmbGFnRmlsdGVyXCI+PiAmIHtcbiAgICBmbGFnRmlsdGVyPzogKGZsYWdLZXk6IHN0cmluZykgPT4gYm9vbGVhbjtcbiAgfTtcblxuICBjb25zdHJ1Y3RvcihvcHRpb25zOiBQZW5kb1RlbGVtZXRyeUhvb2tPcHRpb25zID0ge30pIHtcbiAgICB0aGlzLm9wdGlvbnMgPSB7XG4gICAgICBldmVudE5hbWU6IFwiZmxhZ19ldmFsdWF0ZWRcIixcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxsZWQgYWZ0ZXIgYSBmbGFnIGlzIHN1Y2Nlc3NmdWxseSBldmFsdWF0ZWQuXG4gICAqIFNlbmRzIHRoZSBldmFsdWF0aW9uIHJlc3VsdCB0byBQZW5kbyBhcyBhIHRyYWNrIGV2ZW50LlxuICAgKi9cbiAgYWZ0ZXIoXG4gICAgaG9va0NvbnRleHQ6IFJlYWRvbmx5PEhvb2tDb250ZXh0PEZsYWdWYWx1ZT4+LFxuICAgIGV2YWx1YXRpb25EZXRhaWxzOiBFdmFsdWF0aW9uRGV0YWlsczxGbGFnVmFsdWU+XG4gICk6IHZvaWQge1xuICAgIGNvbnN0IHsgZmxhZ0tleSB9ID0gaG9va0NvbnRleHQ7XG5cbiAgICAvLyBDaGVjayBpZiB0aGlzIGZsYWcgc2hvdWxkIGJlIHRyYWNrZWRcbiAgICBpZiAodGhpcy5vcHRpb25zLmZsYWdGaWx0ZXIgJiYgIXRoaXMub3B0aW9ucy5mbGFnRmlsdGVyKGZsYWdLZXkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gQ2hlY2sgaWYgUGVuZG8gV2ViIFNESyBpcyBhdmFpbGFibGVcbiAgICBpZiAodHlwZW9mIHdpbmRvdyA9PT0gXCJ1bmRlZmluZWRcIiB8fCAhd2luZG93LnBlbmRvPy50cmFjaykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgICBmbGFnX2tleTogZmxhZ0tleSxcbiAgICAgIGZsYWdfdmFyaWFudDogZXZhbHVhdGlvbkRldGFpbHMudmFyaWFudCB8fCBcInVua25vd25cIixcbiAgICAgIGZsYWdfcmVhc29uOiBldmFsdWF0aW9uRGV0YWlscy5yZWFzb24gfHwgXCJVTktOT1dOXCIsXG4gICAgICBmbGFnX3ZhbHVlOiB0aGlzLnN0cmluZ2lmeVZhbHVlKGV2YWx1YXRpb25EZXRhaWxzLnZhbHVlKSxcbiAgICAgIHByb3ZpZGVyX25hbWU6IGhvb2tDb250ZXh0LnByb3ZpZGVyTWV0YWRhdGEubmFtZSxcbiAgICB9O1xuXG4gICAgLy8gRmlyZS1hbmQtZm9yZ2V0OiBkZWxlZ2F0ZSB0byBQZW5kbyBXZWIgU0RLXG4gICAgd2luZG93LnBlbmRvLnRyYWNrKHRoaXMub3B0aW9ucy5ldmVudE5hbWUsIHByb3BlcnRpZXMpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnQgYSBmbGFnIHZhbHVlIHRvIGEgc3RyaW5nIGZvciB0cmFja2luZy5cbiAgICovXG4gIHByaXZhdGUgc3RyaW5naWZ5VmFsdWUodmFsdWU6IEZsYWdWYWx1ZSk6IHN0cmluZyB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICByZXR1cm4gXCJudWxsXCI7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIHJldHVybiBKU09OLnN0cmluZ2lmeSh2YWx1ZSk7XG4gICAgfVxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,22 @@
1
+ /**
2
+ * Type declarations for Pendo Web SDK integration.
3
+ */
4
+ interface PendoEventEmitter {
5
+ on: (handler: (data?: unknown) => void) => void;
6
+ off: (handler: (data?: unknown) => void) => void;
7
+ trigger: (data?: unknown) => void;
8
+ }
9
+ declare global {
10
+ interface Window {
11
+ pendo?: {
12
+ segmentFlags?: string[] | null;
13
+ isReady?: () => boolean;
14
+ initialize?: (options: unknown) => void;
15
+ track?: (event: string, properties?: Record<string, string>) => void;
16
+ Events?: {
17
+ segmentFlagsUpdated?: PendoEventEmitter;
18
+ };
19
+ };
20
+ }
21
+ }
22
+ export {};
package/dist/types.js ADDED
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ /**
3
+ * Type declarations for Pendo Web SDK integration.
4
+ */
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUeXBlIGRlY2xhcmF0aW9ucyBmb3IgUGVuZG8gV2ViIFNESyBpbnRlZ3JhdGlvbi5cbiAqL1xuXG5pbnRlcmZhY2UgUGVuZG9FdmVudEVtaXR0ZXIge1xuICBvbjogKGhhbmRsZXI6IChkYXRhPzogdW5rbm93bikgPT4gdm9pZCkgPT4gdm9pZDtcbiAgb2ZmOiAoaGFuZGxlcjogKGRhdGE/OiB1bmtub3duKSA9PiB2b2lkKSA9PiB2b2lkO1xuICB0cmlnZ2VyOiAoZGF0YT86IHVua25vd24pID0+IHZvaWQ7XG59XG5cbmRlY2xhcmUgZ2xvYmFsIHtcbiAgaW50ZXJmYWNlIFdpbmRvdyB7XG4gICAgcGVuZG8/OiB7XG4gICAgICBzZWdtZW50RmxhZ3M/OiBzdHJpbmdbXSB8IG51bGw7XG4gICAgICBpc1JlYWR5PzogKCkgPT4gYm9vbGVhbjtcbiAgICAgIGluaXRpYWxpemU/OiAob3B0aW9uczogdW5rbm93bikgPT4gdm9pZDtcbiAgICAgIHRyYWNrPzogKGV2ZW50OiBzdHJpbmcsIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+KSA9PiB2b2lkO1xuICAgICAgRXZlbnRzPzoge1xuICAgICAgICBzZWdtZW50RmxhZ3NVcGRhdGVkPzogUGVuZG9FdmVudEVtaXR0ZXI7XG4gICAgICB9O1xuICAgIH07XG4gIH1cbn1cblxuLy8gVGhpcyBleHBvcnQgbWFrZXMgdGhpcyBmaWxlIGEgbW9kdWxlXG5leHBvcnQge307XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pendo/openfeature-web-provider",
3
- "version": "0.1.0",
3
+ "version": "0.2.0",
4
4
  "description": "OpenFeature provider for Pendo feature flags (web/browser)",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.mjs",
@@ -11,7 +11,11 @@
11
11
  "scripts": {
12
12
  "build": "tsc",
13
13
  "test": "jest",
14
- "lint": "eslint src"
14
+ "lint": "eslint src",
15
+ "prepublishOnly": "npm run build && npm test"
16
+ },
17
+ "publishConfig": {
18
+ "access": "public"
15
19
  },
16
20
  "peerDependencies": {
17
21
  "@openfeature/web-sdk": "^1.0.0"