@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 +61 -82
- package/dist/PendoProvider.d.ts +33 -33
- package/dist/PendoProvider.js +74 -38
- package/dist/PendoTelemetryHook.d.ts +2 -1
- package/dist/PendoTelemetryHook.js +5 -4
- package/dist/types.d.ts +22 -0
- package/dist/types.js +6 -0
- package/package.json +6 -2
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
|
|
14
|
-
|
|
15
|
-
```
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
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
|
|
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 =
|
|
43
|
+
const showNewFeature = client.getBooleanValue('new-checkout-flow', false);
|
|
44
44
|
|
|
45
45
|
// String flag (returns "on" or "off")
|
|
46
|
-
const variant =
|
|
46
|
+
const variant = client.getStringValue('checkout-variant', 'control');
|
|
47
47
|
|
|
48
48
|
// Number flag (returns 1 or 0)
|
|
49
|
-
const flagValue =
|
|
49
|
+
const flagValue = client.getNumberValue('feature-score', 0);
|
|
50
50
|
|
|
51
51
|
// Object flag (returns { enabled: true/false })
|
|
52
|
-
const config =
|
|
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
|
|
61
|
-
await OpenFeature.setProviderAndWait(provider);
|
|
60
|
+
const client = OpenFeature.getClient();
|
|
62
61
|
|
|
63
|
-
// Track an event
|
|
64
|
-
|
|
62
|
+
// Track an event (delegates to pendo.track)
|
|
63
|
+
client.track('checkout_started', { cartValue: '99.99' });
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
### Configuration Options
|
|
65
67
|
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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.
|
|
152
|
-
2.
|
|
153
|
-
3.
|
|
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
|
-
###
|
|
144
|
+
### Flags not updating
|
|
164
145
|
|
|
165
|
-
|
|
146
|
+
The provider automatically detects flag changes. If flags aren't updating:
|
|
166
147
|
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
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
|
|
package/dist/PendoProvider.d.ts
CHANGED
|
@@ -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
|
-
|
|
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
|
-
*
|
|
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
|
|
29
|
-
*
|
|
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 =
|
|
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
|
|
41
|
+
readonly events: OpenFeatureEventEmitter;
|
|
45
42
|
status: ClientProviderStatus;
|
|
46
43
|
hooks?: Hook[];
|
|
47
44
|
private options;
|
|
48
|
-
private
|
|
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
|
|
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
|
-
*
|
|
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
|
}
|
package/dist/PendoProvider.js
CHANGED
|
@@ -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
|
|
9
|
-
*
|
|
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 =
|
|
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.
|
|
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
|
|
50
|
+
* Waits for Pendo to be ready and sets up flag change detection.
|
|
37
51
|
*/
|
|
38
52
|
async initialize() {
|
|
39
|
-
if
|
|
40
|
-
|
|
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
|
-
|
|
43
|
-
|
|
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
|
|
65
|
+
const timeout = this.options.readyTimeout;
|
|
51
66
|
const startTime = Date.now();
|
|
52
|
-
return new Promise((resolve
|
|
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
|
|
85
|
+
// Also listen for pendo_ready event
|
|
73
86
|
if (typeof document !== "undefined") {
|
|
74
|
-
document.addEventListener("
|
|
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.
|
|
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 (
|
|
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
|
-
*
|
|
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
|
|
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
|
|
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":";;;AASA,kDAA4D;AA2B5D;;;;;;;;;;;;;;;;GAgBG;AACH,MAAa,aAAa;IAYxB,YAAY,UAAgC,EAAE;QAXrC,aAAQ,GAAqB;YACpC,IAAI,EAAE,gBAAgB;SACvB,CAAC;QAGF,WAAM,GAAyB,8BAAoB,CAAC,SAAS,CAAC;QAItD,iBAAY,GAAyB,IAAI,CAAC;QAGhD,IAAI,CAAC,OAAO,GAAG;YACb,YAAY,EAAE,IAAI;YAClB,GAAG,OAAO;SACX,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,UAAU;QACd,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC,YAAY,CAAC;QAC3B,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACxC,MAAM,IAAI,CAAC,YAAY,CAAC;QACxB,IAAI,CAAC,MAAM,GAAG,8BAAoB,CAAC,KAAK,CAAC;IAC3C,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,YAAY;QACxB,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC;QAClD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE7B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,KAAK,GAAG,GAAG,EAAE;gBACjB,wCAAwC;gBACxC,IAAI,OAAO,MAAM,KAAK,WAAW,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;oBAClD,IAAI,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,EAAE,IAAI,MAAM,CAAC,KAAK,CAAC,YAAY,KAAK,SAAS,EAAE,CAAC;wBACxE,OAAO,EAAE,CAAC;wBACV,OAAO;oBACT,CAAC;gBACH,CAAC;gBAED,gBAAgB;gBAChB,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,GAAG,OAAO,EAAE,CAAC;oBACrC,qDAAqD;oBACrD,8BAA8B;oBAC9B,OAAO,CAAC,IAAI,CACV,gFAAgF,CACjF,CAAC;oBACF,OAAO,EAAE,CAAC;oBACV,OAAO;gBACT,CAAC;gBAED,YAAY;gBACZ,UAAU,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;YACzB,CAAC,CAAC;YAEF,oCAAoC;YACpC,IAAI,OAAO,QAAQ,KAAK,WAAW,EAAE,CAAC;gBACpC,QAAQ,CAAC,gBAAgB,CACvB,aAAa,EACb,GAAG,EAAE;oBACH,OAAO,EAAE,CAAC;gBACZ,CAAC,EACD,EAAE,IAAI,EAAE,IAAI,EAAE,CACf,CAAC;YACJ,CAAC;YAED,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO;QACX,IAAI,CAAC,MAAM,GAAG,8BAAoB,CAAC,SAAS,CAAC;QAC7C,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;IAC3B,CAAC;IAED;;;;OAIG;IACH,wBAAwB,CACtB,OAAe,EACf,YAAqB,EACrB,QAA2B;QAE3B,MAAM,KAAK,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAErC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO;gBACL,KAAK,EAAE,YAAY;gBACnB,MAAM,EAAE,SAAS;gBACjB,OAAO,EAAE,SAAS;aACnB,CAAC;QACJ,CAAC;QAED,MAAM,OAAO,GAAG,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAExC,OAAO;YACL,KAAK,EAAE,OAAO;YACd,MAAM,EAAE,OAAO,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,SAAS;YAC/C,OAAO,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK;SAChC,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CACrB,OAAe,EACf,YAAoB,EACpB,OAA0B;QAE1B,MAAM,UAAU,GAAG,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;QAE1E,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YACzD,OAAO;gBACL,KAAK,EAAE,YAAY;gBACnB,MAAM,EAAE,SAAS;gBACjB,OAAO,EAAE,SAAS;aACnB,CAAC;QACJ,CAAC;QAED,OAAO;YACL,KAAK,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK;YACtC,MAAM,EAAE,UAAU,CAAC,MAAM;YACzB,OAAO,EAAE,UAAU,CAAC,OAAO;SAC5B,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CACrB,OAAe,EACf,YAAoB,EACpB,OAA0B;QAE1B,MAAM,UAAU,GAAG,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;QAE1E,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YACzD,OAAO;gBACL,KAAK,EAAE,YAAY;gBACnB,MAAM,EAAE,SAAS;gBACjB,OAAO,EAAE,SAAS;aACnB,CAAC;QACJ,CAAC;QAED,OAAO;YACL,KAAK,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAC/B,MAAM,EAAE,UAAU,CAAC,MAAM;YACzB,OAAO,EAAE,UAAU,CAAC,OAAO;SAC5B,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CACrB,OAAe,EACf,YAAe,EACf,OAA0B;QAE1B,MAAM,UAAU,GAAG,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;QAE1E,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YACzD,OAAO;gBACL,KAAK,EAAE,YAAY;gBACnB,MAAM,EAAE,SAAS;gBACjB,OAAO,EAAE,SAAS;aACnB,CAAC;QACJ,CAAC;QAED,OAAO;YACL,KAAK,EAAE,EAAE,OAAO,EAAE,UAAU,CAAC,KAAK,EAAkB;YACpD,MAAM,EAAE,UAAU,CAAC,MAAM;YACzB,OAAO,EAAE,UAAU,CAAC,OAAO;SAC5B,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,eAAe;QACrB,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;YAClC,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,OAAO,MAAM,CAAC,KAAK,EAAE,YAAY,CAAC;IACpC,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK,CACH,iBAAyB,EACzB,QAA2B,EAC3B,oBAA0C;QAE1C,IAAI,OAAO,MAAM,KAAK,WAAW,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC;YAC1D,OAAO,CAAC,IAAI,CAAC,wDAAwD,CAAC,CAAC;YACvE,OAAO;QACT,CAAC;QAED,yEAAyE;QACzE,MAAM,UAAU,GAA2B,EAAE,CAAC;QAC9C,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,oBAAoB,CAAC,EAAE,CAAC;YAChE,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;gBAC1C,UAAU,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;YAClC,CAAC;QACH,CAAC;QAED,2CAA2C;QAC3C,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,iBAAiB,EAAE,UAAU,CAAC,CAAC;IACpD,CAAC;CACF;AAhPD,sCAgPC","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
|
|
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
|
|
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
|
|
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
|
|
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,
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUGVuZG9UZWxlbWV0cnlIb29rLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL1BlbmRvVGVsZW1ldHJ5SG9vay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFNQSxtQkFBaUI7QUFnQmpCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9CRztBQUNILE1BQWEsa0JBQWtCO0lBSzdCLFlBQVksVUFBcUMsRUFBRTtRQUNqRCxJQUFJLENBQUMsT0FBTyxHQUFHO1lBQ2IsU0FBUyxFQUFFLGdCQUFnQjtZQUMzQixHQUFHLE9BQU87U0FDWCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7T0FHRztJQUNILEtBQUssQ0FDSCxXQUE2QyxFQUM3QyxpQkFBK0M7UUFFL0MsTUFBTSxFQUFFLE9BQU8sRUFBRSxHQUFHLFdBQVcsQ0FBQztRQUVoQyx1Q0FBdUM7UUFDdkMsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDakUsT0FBTztRQUNULENBQUM7UUFFRCxzQ0FBc0M7UUFDdEMsSUFBSSxPQUFPLE1BQU0sS0FBSyxXQUFXLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQzFELE9BQU87UUFDVCxDQUFDO1FBRUQsTUFBTSxVQUFVLEdBQTJCO1lBQ3pDLFFBQVEsRUFBRSxPQUFPO1lBQ2pCLFlBQVksRUFBRSxpQkFBaUIsQ0FBQyxPQUFPLElBQUksU0FBUztZQUNwRCxXQUFXLEVBQUUsaUJBQWlCLENBQUMsTUFBTSxJQUFJLFNBQVM7WUFDbEQsVUFBVSxFQUFFLElBQUksQ0FBQyxjQUFjLENBQUMsaUJBQWlCLENBQUMsS0FBSyxDQUFDO1lBQ3hELGFBQWEsRUFBRSxXQUFXLENBQUMsZ0JBQWdCLENBQUMsSUFBSTtTQUNqRCxDQUFDO1FBRUYsNkNBQTZDO1FBQzdDLE1BQU0sQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLFVBQVUsQ0FBQyxDQUFDO0lBQ3pELENBQUM7SUFFRDs7T0FFRztJQUNLLGNBQWMsQ0FBQyxLQUFnQjtRQUNyQyxJQUFJLEtBQUssS0FBSyxJQUFJLEVBQUUsQ0FBQztZQUNuQixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQ0QsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUM5QixPQUFPLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDL0IsQ0FBQztRQUNELE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3ZCLENBQUM7Q0FDRjtBQXhERCxnREF3REMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7XG4gIEhvb2ssXG4gIEhvb2tDb250ZXh0LFxuICBFdmFsdWF0aW9uRGV0YWlscyxcbiAgRmxhZ1ZhbHVlLFxufSBmcm9tIFwiQG9wZW5mZWF0dXJlL3dlYi1zZGtcIjtcbmltcG9ydCBcIi4vdHlwZXNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBQZW5kb1RlbGVtZXRyeUhvb2tPcHRpb25zIHtcbiAgLyoqXG4gICAqIEV2ZW50IG5hbWUgZm9yIGZsYWcgZXZhbHVhdGlvbiB0cmFja2luZy5cbiAgICogRGVmYXVsdDogXCJmbGFnX2V2YWx1YXRlZFwiXG4gICAqL1xuICBldmVudE5hbWU/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGZpbHRlciBmdW5jdGlvbiB0byBzZWxlY3RpdmVseSB0cmFjayBmbGFncy5cbiAgICogUmV0dXJuIHRydWUgdG8gdHJhY2sgdGhlIGZsYWcsIGZhbHNlIHRvIHNraXAuXG4gICAqL1xuICBmbGFnRmlsdGVyPzogKGZsYWdLZXk6IHN0cmluZykgPT4gYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBPcGVuRmVhdHVyZSBob29rIHRoYXQgYXV0b21hdGljYWxseSB0cmFja3MgZmxhZyBldmFsdWF0aW9ucyB0byBQZW5kby5cbiAqXG4gKiBUaGlzIGhvb2sgc2VuZHMgYW4gZXZlbnQgdG8gUGVuZG8gYWZ0ZXIgZWFjaCBmbGFnIGV2YWx1YXRpb24sIGFsbG93aW5nIHlvdVxuICogdG8gYW5hbHl6ZSBmZWF0dXJlIGZsYWcgdXNhZ2UgYWxvbmdzaWRlIHlvdXIgb3RoZXIgUGVuZG8gYW5hbHl0aWNzLlxuICpcbiAqIFRoZSBob29rIHVzZXMgdGhlIFBlbmRvIFdlYiBTREsncyB0cmFjayBmdW5jdGlvbiwgc28gdGhlIFBlbmRvIFdlYiBTREsgbXVzdCBiZVxuICogaW5pdGlhbGl6ZWQgb24gdGhlIHBhZ2UgZm9yIHRyYWNraW5nIHRvIHdvcmsuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IE9wZW5GZWF0dXJlIH0gZnJvbSAnQG9wZW5mZWF0dXJlL3dlYi1zZGsnO1xuICogaW1wb3J0IHsgUGVuZG9Qcm92aWRlciwgUGVuZG9UZWxlbWV0cnlIb29rIH0gZnJvbSAnQHBlbmRvL29wZW5mZWF0dXJlLXdlYi1wcm92aWRlcic7XG4gKlxuICogY29uc3QgcHJvdmlkZXIgPSBuZXcgUGVuZG9Qcm92aWRlcigpO1xuICogY29uc3QgdGVsZW1ldHJ5SG9vayA9IG5ldyBQZW5kb1RlbGVtZXRyeUhvb2soKTtcbiAqXG4gKiBhd2FpdCBPcGVuRmVhdHVyZS5zZXRQcm92aWRlckFuZFdhaXQocHJvdmlkZXIpO1xuICogT3BlbkZlYXR1cmUuYWRkSG9va3ModGVsZW1ldHJ5SG9vayk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNsYXNzIFBlbmRvVGVsZW1ldHJ5SG9vayBpbXBsZW1lbnRzIEhvb2sge1xuICBwcml2YXRlIG9wdGlvbnM6IFJlcXVpcmVkPE9taXQ8UGVuZG9UZWxlbWV0cnlIb29rT3B0aW9ucywgXCJmbGFnRmlsdGVyXCI+PiAmIHtcbiAgICBmbGFnRmlsdGVyPzogKGZsYWdLZXk6IHN0cmluZykgPT4gYm9vbGVhbjtcbiAgfTtcblxuICBjb25zdHJ1Y3RvcihvcHRpb25zOiBQZW5kb1RlbGVtZXRyeUhvb2tPcHRpb25zID0ge30pIHtcbiAgICB0aGlzLm9wdGlvbnMgPSB7XG4gICAgICBldmVudE5hbWU6IFwiZmxhZ19ldmFsdWF0ZWRcIixcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxsZWQgYWZ0ZXIgYSBmbGFnIGlzIHN1Y2Nlc3NmdWxseSBldmFsdWF0ZWQuXG4gICAqIFNlbmRzIHRoZSBldmFsdWF0aW9uIHJlc3VsdCB0byBQZW5kbyBhcyBhIHRyYWNrIGV2ZW50LlxuICAgKi9cbiAgYWZ0ZXIoXG4gICAgaG9va0NvbnRleHQ6IFJlYWRvbmx5PEhvb2tDb250ZXh0PEZsYWdWYWx1ZT4+LFxuICAgIGV2YWx1YXRpb25EZXRhaWxzOiBFdmFsdWF0aW9uRGV0YWlsczxGbGFnVmFsdWU+XG4gICk6IHZvaWQge1xuICAgIGNvbnN0IHsgZmxhZ0tleSB9ID0gaG9va0NvbnRleHQ7XG5cbiAgICAvLyBDaGVjayBpZiB0aGlzIGZsYWcgc2hvdWxkIGJlIHRyYWNrZWRcbiAgICBpZiAodGhpcy5vcHRpb25zLmZsYWdGaWx0ZXIgJiYgIXRoaXMub3B0aW9ucy5mbGFnRmlsdGVyKGZsYWdLZXkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gQ2hlY2sgaWYgUGVuZG8gV2ViIFNESyBpcyBhdmFpbGFibGVcbiAgICBpZiAodHlwZW9mIHdpbmRvdyA9PT0gXCJ1bmRlZmluZWRcIiB8fCAhd2luZG93LnBlbmRvPy50cmFjaykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgICBmbGFnX2tleTogZmxhZ0tleSxcbiAgICAgIGZsYWdfdmFyaWFudDogZXZhbHVhdGlvbkRldGFpbHMudmFyaWFudCB8fCBcInVua25vd25cIixcbiAgICAgIGZsYWdfcmVhc29uOiBldmFsdWF0aW9uRGV0YWlscy5yZWFzb24gfHwgXCJVTktOT1dOXCIsXG4gICAgICBmbGFnX3ZhbHVlOiB0aGlzLnN0cmluZ2lmeVZhbHVlKGV2YWx1YXRpb25EZXRhaWxzLnZhbHVlKSxcbiAgICAgIHByb3ZpZGVyX25hbWU6IGhvb2tDb250ZXh0LnByb3ZpZGVyTWV0YWRhdGEubmFtZSxcbiAgICB9O1xuXG4gICAgLy8gRmlyZS1hbmQtZm9yZ2V0OiBkZWxlZ2F0ZSB0byBQZW5kbyBXZWIgU0RLXG4gICAgd2luZG93LnBlbmRvLnRyYWNrKHRoaXMub3B0aW9ucy5ldmVudE5hbWUsIHByb3BlcnRpZXMpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnQgYSBmbGFnIHZhbHVlIHRvIGEgc3RyaW5nIGZvciB0cmFja2luZy5cbiAgICovXG4gIHByaXZhdGUgc3RyaW5naWZ5VmFsdWUodmFsdWU6IEZsYWdWYWx1ZSk6IHN0cmluZyB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICByZXR1cm4gXCJudWxsXCI7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIHJldHVybiBKU09OLnN0cmluZ2lmeSh2YWx1ZSk7XG4gICAgfVxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpO1xuICB9XG59XG4iXX0=
|
package/dist/types.d.ts
ADDED
|
@@ -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.
|
|
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"
|