@pendo/openfeature-web-provider 0.1.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 +175 -0
- package/dist/PendoProvider.d.ts +102 -0
- package/dist/PendoProvider.js +205 -0
- package/dist/PendoTelemetryHook.d.ts +47 -0
- package/dist/PendoTelemetryHook.js +70 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +8 -0
- package/package.json +36 -0
package/README.md
ADDED
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
# @pendo/openfeature-web-provider
|
|
2
|
+
|
|
3
|
+
OpenFeature provider for [Pendo](https://www.pendo.io/) feature flags in web browsers.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install @pendo/openfeature-web-provider @openfeature/web-sdk
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Prerequisites
|
|
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>
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
## Usage
|
|
25
|
+
|
|
26
|
+
### Basic Setup
|
|
27
|
+
|
|
28
|
+
```typescript
|
|
29
|
+
import { OpenFeature } from '@openfeature/web-sdk';
|
|
30
|
+
import { PendoProvider } from '@pendo/openfeature-web-provider';
|
|
31
|
+
|
|
32
|
+
// Initialize the provider (waits for Pendo agent to be ready)
|
|
33
|
+
await OpenFeature.setProviderAndWait(new PendoProvider());
|
|
34
|
+
|
|
35
|
+
// Get a client
|
|
36
|
+
const client = OpenFeature.getClient();
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
### Evaluating Flags
|
|
40
|
+
|
|
41
|
+
```typescript
|
|
42
|
+
// Boolean flag
|
|
43
|
+
const showNewFeature = await client.getBooleanValue('new-checkout-flow', false);
|
|
44
|
+
|
|
45
|
+
// String flag (returns "on" or "off")
|
|
46
|
+
const variant = await client.getStringValue('checkout-variant', 'control');
|
|
47
|
+
|
|
48
|
+
// Number flag (returns 1 or 0)
|
|
49
|
+
const flagValue = await client.getNumberValue('feature-score', 0);
|
|
50
|
+
|
|
51
|
+
// Object flag (returns { enabled: true/false })
|
|
52
|
+
const config = await client.getObjectValue('feature-config', { enabled: false });
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
### Event Tracking
|
|
56
|
+
|
|
57
|
+
Track custom events to Pendo:
|
|
58
|
+
|
|
59
|
+
```typescript
|
|
60
|
+
const provider = new PendoProvider();
|
|
61
|
+
await OpenFeature.setProviderAndWait(provider);
|
|
62
|
+
|
|
63
|
+
// Track an event
|
|
64
|
+
provider.track('checkout_started', undefined, { cartValue: '99.99' });
|
|
65
|
+
|
|
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');
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
### Telemetry Hook
|
|
72
|
+
|
|
73
|
+
Automatically track all flag evaluations to Pendo using the telemetry hook:
|
|
74
|
+
|
|
75
|
+
```typescript
|
|
76
|
+
import { OpenFeature } from '@openfeature/web-sdk';
|
|
77
|
+
import { PendoProvider, PendoTelemetryHook } from '@pendo/openfeature-web-provider';
|
|
78
|
+
|
|
79
|
+
const provider = new PendoProvider();
|
|
80
|
+
const telemetryHook = new PendoTelemetryHook();
|
|
81
|
+
|
|
82
|
+
await OpenFeature.setProviderAndWait(provider);
|
|
83
|
+
OpenFeature.addHooks(telemetryHook);
|
|
84
|
+
```
|
|
85
|
+
|
|
86
|
+
#### Telemetry Hook Options
|
|
87
|
+
|
|
88
|
+
```typescript
|
|
89
|
+
const telemetryHook = new PendoTelemetryHook({
|
|
90
|
+
// Optional: Custom event name (default: "flag_evaluated")
|
|
91
|
+
eventName: 'feature_flag_evaluated',
|
|
92
|
+
|
|
93
|
+
// Optional: Filter which flags to track
|
|
94
|
+
flagFilter: (flagKey) => flagKey.startsWith('feature_'),
|
|
95
|
+
});
|
|
96
|
+
```
|
|
97
|
+
|
|
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
|
+
## Troubleshooting
|
|
148
|
+
|
|
149
|
+
### Flags always return default values
|
|
150
|
+
|
|
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
|
|
154
|
+
|
|
155
|
+
### Provider times out
|
|
156
|
+
|
|
157
|
+
Increase the `readyTimeout` option:
|
|
158
|
+
|
|
159
|
+
```typescript
|
|
160
|
+
new PendoProvider({ readyTimeout: 15000 });
|
|
161
|
+
```
|
|
162
|
+
|
|
163
|
+
### Track events not appearing
|
|
164
|
+
|
|
165
|
+
Ensure the Pendo agent's `track` function is available:
|
|
166
|
+
|
|
167
|
+
```typescript
|
|
168
|
+
if (window.pendo?.track) {
|
|
169
|
+
provider.track('my-event');
|
|
170
|
+
}
|
|
171
|
+
```
|
|
172
|
+
|
|
173
|
+
## License
|
|
174
|
+
|
|
175
|
+
MIT
|
|
@@ -0,0 +1,102 @@
|
|
|
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
|
+
}
|
|
13
|
+
export interface PendoProviderOptions {
|
|
14
|
+
/**
|
|
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.
|
|
21
|
+
* Default: 5000ms
|
|
22
|
+
*/
|
|
23
|
+
readyTimeout?: number;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* OpenFeature provider for Pendo feature flags.
|
|
27
|
+
*
|
|
28
|
+
* This provider evaluates feature flags by checking the `pendo.segmentFlags` array
|
|
29
|
+
* which is populated by the Pendo agent based on segment membership.
|
|
30
|
+
*
|
|
31
|
+
* @example
|
|
32
|
+
* ```typescript
|
|
33
|
+
* import { OpenFeature } from '@openfeature/web-sdk';
|
|
34
|
+
* import { PendoProvider } from '@pendo/openfeature-web-provider';
|
|
35
|
+
*
|
|
36
|
+
* await OpenFeature.setProviderAndWait(new PendoProvider());
|
|
37
|
+
*
|
|
38
|
+
* const client = OpenFeature.getClient();
|
|
39
|
+
* const enabled = await client.getBooleanValue('myFeature', false);
|
|
40
|
+
* ```
|
|
41
|
+
*/
|
|
42
|
+
export declare class PendoProvider implements Provider {
|
|
43
|
+
readonly metadata: ProviderMetadata;
|
|
44
|
+
readonly rulesChanged?: () => void;
|
|
45
|
+
status: ClientProviderStatus;
|
|
46
|
+
hooks?: Hook[];
|
|
47
|
+
private options;
|
|
48
|
+
private readyPromise;
|
|
49
|
+
constructor(options?: PendoProviderOptions);
|
|
50
|
+
/**
|
|
51
|
+
* Initialize the provider.
|
|
52
|
+
* Waits for Pendo to be ready before resolving.
|
|
53
|
+
*/
|
|
54
|
+
initialize(): Promise<void>;
|
|
55
|
+
/**
|
|
56
|
+
* Wait for Pendo to be ready with timeout.
|
|
57
|
+
*/
|
|
58
|
+
private waitForPendo;
|
|
59
|
+
/**
|
|
60
|
+
* Shutdown the provider.
|
|
61
|
+
*/
|
|
62
|
+
onClose(): Promise<void>;
|
|
63
|
+
/**
|
|
64
|
+
* Resolve a boolean flag value.
|
|
65
|
+
*
|
|
66
|
+
* Checks if the flagKey exists in pendo.segmentFlags array.
|
|
67
|
+
*/
|
|
68
|
+
resolveBooleanEvaluation(flagKey: string, defaultValue: boolean, _context: EvaluationContext): ResolutionDetails<boolean>;
|
|
69
|
+
/**
|
|
70
|
+
* Resolve a string flag value.
|
|
71
|
+
*
|
|
72
|
+
* For Pendo flags, this returns "on" if the flag is enabled, "off" otherwise.
|
|
73
|
+
*/
|
|
74
|
+
resolveStringEvaluation(flagKey: string, defaultValue: string, context: EvaluationContext): ResolutionDetails<string>;
|
|
75
|
+
/**
|
|
76
|
+
* Resolve a number flag value.
|
|
77
|
+
*
|
|
78
|
+
* For Pendo flags, this returns 1 if enabled, 0 if disabled.
|
|
79
|
+
*/
|
|
80
|
+
resolveNumberEvaluation(flagKey: string, defaultValue: number, context: EvaluationContext): ResolutionDetails<number>;
|
|
81
|
+
/**
|
|
82
|
+
* Resolve an object flag value.
|
|
83
|
+
*
|
|
84
|
+
* For Pendo flags, this returns { enabled: true/false }.
|
|
85
|
+
*/
|
|
86
|
+
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
|
+
/**
|
|
92
|
+
* Track a custom event in Pendo.
|
|
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
|
|
100
|
+
*/
|
|
101
|
+
track(trackingEventName: string, _context: EvaluationContext, trackingEventDetails: TrackingEventDetails): void;
|
|
102
|
+
}
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PendoProvider = void 0;
|
|
4
|
+
const web_sdk_1 = require("@openfeature/web-sdk");
|
|
5
|
+
/**
|
|
6
|
+
* OpenFeature provider for Pendo feature flags.
|
|
7
|
+
*
|
|
8
|
+
* This provider evaluates feature flags by checking the `pendo.segmentFlags` array
|
|
9
|
+
* which is populated by the Pendo agent based on segment membership.
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```typescript
|
|
13
|
+
* import { OpenFeature } from '@openfeature/web-sdk';
|
|
14
|
+
* import { PendoProvider } from '@pendo/openfeature-web-provider';
|
|
15
|
+
*
|
|
16
|
+
* await OpenFeature.setProviderAndWait(new PendoProvider());
|
|
17
|
+
*
|
|
18
|
+
* const client = OpenFeature.getClient();
|
|
19
|
+
* const enabled = await client.getBooleanValue('myFeature', false);
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
class PendoProvider {
|
|
23
|
+
constructor(options = {}) {
|
|
24
|
+
this.metadata = {
|
|
25
|
+
name: "pendo-provider",
|
|
26
|
+
};
|
|
27
|
+
this.status = web_sdk_1.ClientProviderStatus.NOT_READY;
|
|
28
|
+
this.readyPromise = null;
|
|
29
|
+
this.options = {
|
|
30
|
+
readyTimeout: 5000,
|
|
31
|
+
...options,
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Initialize the provider.
|
|
36
|
+
* Waits for Pendo to be ready before resolving.
|
|
37
|
+
*/
|
|
38
|
+
async initialize() {
|
|
39
|
+
if (this.readyPromise) {
|
|
40
|
+
return this.readyPromise;
|
|
41
|
+
}
|
|
42
|
+
this.readyPromise = this.waitForPendo();
|
|
43
|
+
await this.readyPromise;
|
|
44
|
+
this.status = web_sdk_1.ClientProviderStatus.READY;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Wait for Pendo to be ready with timeout.
|
|
48
|
+
*/
|
|
49
|
+
async waitForPendo() {
|
|
50
|
+
const timeout = this.options.readyTimeout ?? 5000;
|
|
51
|
+
const startTime = Date.now();
|
|
52
|
+
return new Promise((resolve, reject) => {
|
|
53
|
+
const check = () => {
|
|
54
|
+
// Check if pendo is available and ready
|
|
55
|
+
if (typeof window !== "undefined" && window.pendo) {
|
|
56
|
+
if (window.pendo.isReady?.() || window.pendo.segmentFlags !== undefined) {
|
|
57
|
+
resolve();
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
// Check timeout
|
|
62
|
+
if (Date.now() - startTime > timeout) {
|
|
63
|
+
// Don't reject - just resolve and work without Pendo
|
|
64
|
+
// Flags will default to false
|
|
65
|
+
console.warn("[PendoProvider] Pendo not ready within timeout. Flags will use default values.");
|
|
66
|
+
resolve();
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
// Try again
|
|
70
|
+
setTimeout(check, 100);
|
|
71
|
+
};
|
|
72
|
+
// Also listen for pendo:ready event
|
|
73
|
+
if (typeof document !== "undefined") {
|
|
74
|
+
document.addEventListener("pendo:ready", () => {
|
|
75
|
+
resolve();
|
|
76
|
+
}, { once: true });
|
|
77
|
+
}
|
|
78
|
+
check();
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Shutdown the provider.
|
|
83
|
+
*/
|
|
84
|
+
async onClose() {
|
|
85
|
+
this.status = web_sdk_1.ClientProviderStatus.NOT_READY;
|
|
86
|
+
this.readyPromise = null;
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Resolve a boolean flag value.
|
|
90
|
+
*
|
|
91
|
+
* Checks if the flagKey exists in pendo.segmentFlags array.
|
|
92
|
+
*/
|
|
93
|
+
resolveBooleanEvaluation(flagKey, defaultValue, _context) {
|
|
94
|
+
const flags = this.getSegmentFlags();
|
|
95
|
+
if (!flags) {
|
|
96
|
+
return {
|
|
97
|
+
value: defaultValue,
|
|
98
|
+
reason: "DEFAULT",
|
|
99
|
+
variant: "default",
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
const enabled = flags.includes(flagKey);
|
|
103
|
+
return {
|
|
104
|
+
value: enabled,
|
|
105
|
+
reason: enabled ? "TARGETING_MATCH" : "DEFAULT",
|
|
106
|
+
variant: enabled ? "on" : "off",
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Resolve a string flag value.
|
|
111
|
+
*
|
|
112
|
+
* For Pendo flags, this returns "on" if the flag is enabled, "off" otherwise.
|
|
113
|
+
*/
|
|
114
|
+
resolveStringEvaluation(flagKey, defaultValue, context) {
|
|
115
|
+
const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);
|
|
116
|
+
if (boolResult.reason === "DEFAULT" && !boolResult.value) {
|
|
117
|
+
return {
|
|
118
|
+
value: defaultValue,
|
|
119
|
+
reason: "DEFAULT",
|
|
120
|
+
variant: "default",
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
return {
|
|
124
|
+
value: boolResult.value ? "on" : "off",
|
|
125
|
+
reason: boolResult.reason,
|
|
126
|
+
variant: boolResult.variant,
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Resolve a number flag value.
|
|
131
|
+
*
|
|
132
|
+
* For Pendo flags, this returns 1 if enabled, 0 if disabled.
|
|
133
|
+
*/
|
|
134
|
+
resolveNumberEvaluation(flagKey, defaultValue, context) {
|
|
135
|
+
const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);
|
|
136
|
+
if (boolResult.reason === "DEFAULT" && !boolResult.value) {
|
|
137
|
+
return {
|
|
138
|
+
value: defaultValue,
|
|
139
|
+
reason: "DEFAULT",
|
|
140
|
+
variant: "default",
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
return {
|
|
144
|
+
value: boolResult.value ? 1 : 0,
|
|
145
|
+
reason: boolResult.reason,
|
|
146
|
+
variant: boolResult.variant,
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Resolve an object flag value.
|
|
151
|
+
*
|
|
152
|
+
* For Pendo flags, this returns { enabled: true/false }.
|
|
153
|
+
*/
|
|
154
|
+
resolveObjectEvaluation(flagKey, defaultValue, context) {
|
|
155
|
+
const boolResult = this.resolveBooleanEvaluation(flagKey, false, context);
|
|
156
|
+
if (boolResult.reason === "DEFAULT" && !boolResult.value) {
|
|
157
|
+
return {
|
|
158
|
+
value: defaultValue,
|
|
159
|
+
reason: "DEFAULT",
|
|
160
|
+
variant: "default",
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
return {
|
|
164
|
+
value: { enabled: boolResult.value },
|
|
165
|
+
reason: boolResult.reason,
|
|
166
|
+
variant: boolResult.variant,
|
|
167
|
+
};
|
|
168
|
+
}
|
|
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
|
+
/**
|
|
179
|
+
* Track a custom event in Pendo.
|
|
180
|
+
*
|
|
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
|
|
187
|
+
*/
|
|
188
|
+
track(trackingEventName, _context, trackingEventDetails) {
|
|
189
|
+
if (typeof window === "undefined" || !window.pendo?.track) {
|
|
190
|
+
console.warn("[PendoProvider] Pendo agent not available for tracking");
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
// Convert TrackingEventDetails to Record<string, string> for Pendo agent
|
|
194
|
+
const properties = {};
|
|
195
|
+
for (const [key, value] of Object.entries(trackingEventDetails)) {
|
|
196
|
+
if (value !== undefined && value !== null) {
|
|
197
|
+
properties[key] = String(value);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
// Fire-and-forget: delegate to Pendo agent
|
|
201
|
+
window.pendo.track(trackingEventName, properties);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
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"]}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import type { Hook, HookContext, EvaluationDetails, FlagValue } from "@openfeature/web-sdk";
|
|
2
|
+
export interface PendoTelemetryHookOptions {
|
|
3
|
+
/**
|
|
4
|
+
* Event name for flag evaluation tracking.
|
|
5
|
+
* Default: "flag_evaluated"
|
|
6
|
+
*/
|
|
7
|
+
eventName?: string;
|
|
8
|
+
/**
|
|
9
|
+
* Optional filter function to selectively track flags.
|
|
10
|
+
* Return true to track the flag, false to skip.
|
|
11
|
+
*/
|
|
12
|
+
flagFilter?: (flagKey: string) => boolean;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* OpenFeature hook that automatically tracks flag evaluations to Pendo.
|
|
16
|
+
*
|
|
17
|
+
* This hook sends an event to Pendo after each flag evaluation, allowing you
|
|
18
|
+
* to analyze feature flag usage alongside your other Pendo analytics.
|
|
19
|
+
*
|
|
20
|
+
* The hook uses the Pendo agent's track function, so the Pendo agent must be
|
|
21
|
+
* initialized on the page for tracking to work.
|
|
22
|
+
*
|
|
23
|
+
* @example
|
|
24
|
+
* ```typescript
|
|
25
|
+
* import { OpenFeature } from '@openfeature/web-sdk';
|
|
26
|
+
* import { PendoProvider, PendoTelemetryHook } from '@pendo/openfeature-web-provider';
|
|
27
|
+
*
|
|
28
|
+
* const provider = new PendoProvider();
|
|
29
|
+
* const telemetryHook = new PendoTelemetryHook();
|
|
30
|
+
*
|
|
31
|
+
* await OpenFeature.setProviderAndWait(provider);
|
|
32
|
+
* OpenFeature.addHooks(telemetryHook);
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
export declare class PendoTelemetryHook implements Hook {
|
|
36
|
+
private options;
|
|
37
|
+
constructor(options?: PendoTelemetryHookOptions);
|
|
38
|
+
/**
|
|
39
|
+
* Called after a flag is successfully evaluated.
|
|
40
|
+
* Sends the evaluation result to Pendo as a track event.
|
|
41
|
+
*/
|
|
42
|
+
after(hookContext: Readonly<HookContext<FlagValue>>, evaluationDetails: EvaluationDetails<FlagValue>): void;
|
|
43
|
+
/**
|
|
44
|
+
* Convert a flag value to a string for tracking.
|
|
45
|
+
*/
|
|
46
|
+
private stringifyValue;
|
|
47
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PendoTelemetryHook = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* OpenFeature hook that automatically tracks flag evaluations to Pendo.
|
|
6
|
+
*
|
|
7
|
+
* This hook sends an event to Pendo after each flag evaluation, allowing you
|
|
8
|
+
* to analyze feature flag usage alongside your other Pendo analytics.
|
|
9
|
+
*
|
|
10
|
+
* The hook uses the Pendo agent's track function, so the Pendo agent must be
|
|
11
|
+
* initialized on the page for tracking to work.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* import { OpenFeature } from '@openfeature/web-sdk';
|
|
16
|
+
* import { PendoProvider, PendoTelemetryHook } from '@pendo/openfeature-web-provider';
|
|
17
|
+
*
|
|
18
|
+
* const provider = new PendoProvider();
|
|
19
|
+
* const telemetryHook = new PendoTelemetryHook();
|
|
20
|
+
*
|
|
21
|
+
* await OpenFeature.setProviderAndWait(provider);
|
|
22
|
+
* OpenFeature.addHooks(telemetryHook);
|
|
23
|
+
* ```
|
|
24
|
+
*/
|
|
25
|
+
class PendoTelemetryHook {
|
|
26
|
+
constructor(options = {}) {
|
|
27
|
+
this.options = {
|
|
28
|
+
eventName: "flag_evaluated",
|
|
29
|
+
...options,
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Called after a flag is successfully evaluated.
|
|
34
|
+
* Sends the evaluation result to Pendo as a track event.
|
|
35
|
+
*/
|
|
36
|
+
after(hookContext, evaluationDetails) {
|
|
37
|
+
const { flagKey } = hookContext;
|
|
38
|
+
// Check if this flag should be tracked
|
|
39
|
+
if (this.options.flagFilter && !this.options.flagFilter(flagKey)) {
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
// Check if Pendo agent is available
|
|
43
|
+
if (typeof window === "undefined" || !window.pendo?.track) {
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
const properties = {
|
|
47
|
+
flag_key: flagKey,
|
|
48
|
+
flag_variant: evaluationDetails.variant || "unknown",
|
|
49
|
+
flag_reason: evaluationDetails.reason || "UNKNOWN",
|
|
50
|
+
flag_value: this.stringifyValue(evaluationDetails.value),
|
|
51
|
+
provider_name: hookContext.providerMetadata.name,
|
|
52
|
+
};
|
|
53
|
+
// Fire-and-forget: delegate to Pendo agent
|
|
54
|
+
window.pendo.track(this.options.eventName, properties);
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Convert a flag value to a string for tracking.
|
|
58
|
+
*/
|
|
59
|
+
stringifyValue(value) {
|
|
60
|
+
if (value === null) {
|
|
61
|
+
return "null";
|
|
62
|
+
}
|
|
63
|
+
if (typeof value === "object") {
|
|
64
|
+
return JSON.stringify(value);
|
|
65
|
+
}
|
|
66
|
+
return String(value);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
exports.PendoTelemetryHook = PendoTelemetryHook;
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUGVuZG9UZWxlbWV0cnlIb29rLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL1BlbmRvVGVsZW1ldHJ5SG9vay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFxQkE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHO0FBQ0gsTUFBYSxrQkFBa0I7SUFLN0IsWUFBWSxVQUFxQyxFQUFFO1FBQ2pELElBQUksQ0FBQyxPQUFPLEdBQUc7WUFDYixTQUFTLEVBQUUsZ0JBQWdCO1lBQzNCLEdBQUcsT0FBTztTQUNYLENBQUM7SUFDSixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsS0FBSyxDQUNILFdBQTZDLEVBQzdDLGlCQUErQztRQUUvQyxNQUFNLEVBQUUsT0FBTyxFQUFFLEdBQUcsV0FBVyxDQUFDO1FBRWhDLHVDQUF1QztRQUN2QyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNqRSxPQUFPO1FBQ1QsQ0FBQztRQUVELG9DQUFvQztRQUNwQyxJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDMUQsT0FBTztRQUNULENBQUM7UUFFRCxNQUFNLFVBQVUsR0FBMkI7WUFDekMsUUFBUSxFQUFFLE9BQU87WUFDakIsWUFBWSxFQUFFLGlCQUFpQixDQUFDLE9BQU8sSUFBSSxTQUFTO1lBQ3BELFdBQVcsRUFBRSxpQkFBaUIsQ0FBQyxNQUFNLElBQUksU0FBUztZQUNsRCxVQUFVLEVBQUUsSUFBSSxDQUFDLGNBQWMsQ0FBQyxpQkFBaUIsQ0FBQyxLQUFLLENBQUM7WUFDeEQsYUFBYSxFQUFFLFdBQVcsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJO1NBQ2pELENBQUM7UUFFRiwyQ0FBMkM7UUFDM0MsTUFBTSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxTQUFTLEVBQUUsVUFBVSxDQUFDLENBQUM7SUFDekQsQ0FBQztJQUVEOztPQUVHO0lBQ0ssY0FBYyxDQUFDLEtBQWdCO1FBQ3JDLElBQUksS0FBSyxLQUFLLElBQUksRUFBRSxDQUFDO1lBQ25CLE9BQU8sTUFBTSxDQUFDO1FBQ2hCLENBQUM7UUFDRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzlCLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMvQixDQUFDO1FBQ0QsT0FBTyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDdkIsQ0FBQztDQUNGO0FBeERELGdEQXdEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgSG9vayxcbiAgSG9va0NvbnRleHQsXG4gIEV2YWx1YXRpb25EZXRhaWxzLFxuICBGbGFnVmFsdWUsXG59IGZyb20gXCJAb3BlbmZlYXR1cmUvd2ViLXNka1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBlbmRvVGVsZW1ldHJ5SG9va09wdGlvbnMge1xuICAvKipcbiAgICogRXZlbnQgbmFtZSBmb3IgZmxhZyBldmFsdWF0aW9uIHRyYWNraW5nLlxuICAgKiBEZWZhdWx0OiBcImZsYWdfZXZhbHVhdGVkXCJcbiAgICovXG4gIGV2ZW50TmFtZT86IHN0cmluZztcblxuICAvKipcbiAgICogT3B0aW9uYWwgZmlsdGVyIGZ1bmN0aW9uIHRvIHNlbGVjdGl2ZWx5IHRyYWNrIGZsYWdzLlxuICAgKiBSZXR1cm4gdHJ1ZSB0byB0cmFjayB0aGUgZmxhZywgZmFsc2UgdG8gc2tpcC5cbiAgICovXG4gIGZsYWdGaWx0ZXI/OiAoZmxhZ0tleTogc3RyaW5nKSA9PiBib29sZWFuO1xufVxuXG4vKipcbiAqIE9wZW5GZWF0dXJlIGhvb2sgdGhhdCBhdXRvbWF0aWNhbGx5IHRyYWNrcyBmbGFnIGV2YWx1YXRpb25zIHRvIFBlbmRvLlxuICpcbiAqIFRoaXMgaG9vayBzZW5kcyBhbiBldmVudCB0byBQZW5kbyBhZnRlciBlYWNoIGZsYWcgZXZhbHVhdGlvbiwgYWxsb3dpbmcgeW91XG4gKiB0byBhbmFseXplIGZlYXR1cmUgZmxhZyB1c2FnZSBhbG9uZ3NpZGUgeW91ciBvdGhlciBQZW5kbyBhbmFseXRpY3MuXG4gKlxuICogVGhlIGhvb2sgdXNlcyB0aGUgUGVuZG8gYWdlbnQncyB0cmFjayBmdW5jdGlvbiwgc28gdGhlIFBlbmRvIGFnZW50IG11c3QgYmVcbiAqIGluaXRpYWxpemVkIG9uIHRoZSBwYWdlIGZvciB0cmFja2luZyB0byB3b3JrLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBPcGVuRmVhdHVyZSB9IGZyb20gJ0BvcGVuZmVhdHVyZS93ZWItc2RrJztcbiAqIGltcG9ydCB7IFBlbmRvUHJvdmlkZXIsIFBlbmRvVGVsZW1ldHJ5SG9vayB9IGZyb20gJ0BwZW5kby9vcGVuZmVhdHVyZS13ZWItcHJvdmlkZXInO1xuICpcbiAqIGNvbnN0IHByb3ZpZGVyID0gbmV3IFBlbmRvUHJvdmlkZXIoKTtcbiAqIGNvbnN0IHRlbGVtZXRyeUhvb2sgPSBuZXcgUGVuZG9UZWxlbWV0cnlIb29rKCk7XG4gKlxuICogYXdhaXQgT3BlbkZlYXR1cmUuc2V0UHJvdmlkZXJBbmRXYWl0KHByb3ZpZGVyKTtcbiAqIE9wZW5GZWF0dXJlLmFkZEhvb2tzKHRlbGVtZXRyeUhvb2spO1xuICogYGBgXG4gKi9cbmV4cG9ydCBjbGFzcyBQZW5kb1RlbGVtZXRyeUhvb2sgaW1wbGVtZW50cyBIb29rIHtcbiAgcHJpdmF0ZSBvcHRpb25zOiBSZXF1aXJlZDxPbWl0PFBlbmRvVGVsZW1ldHJ5SG9va09wdGlvbnMsIFwiZmxhZ0ZpbHRlclwiPj4gJiB7XG4gICAgZmxhZ0ZpbHRlcj86IChmbGFnS2V5OiBzdHJpbmcpID0+IGJvb2xlYW47XG4gIH07XG5cbiAgY29uc3RydWN0b3Iob3B0aW9uczogUGVuZG9UZWxlbWV0cnlIb29rT3B0aW9ucyA9IHt9KSB7XG4gICAgdGhpcy5vcHRpb25zID0ge1xuICAgICAgZXZlbnROYW1lOiBcImZsYWdfZXZhbHVhdGVkXCIsXG4gICAgICAuLi5vcHRpb25zLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ2FsbGVkIGFmdGVyIGEgZmxhZyBpcyBzdWNjZXNzZnVsbHkgZXZhbHVhdGVkLlxuICAgKiBTZW5kcyB0aGUgZXZhbHVhdGlvbiByZXN1bHQgdG8gUGVuZG8gYXMgYSB0cmFjayBldmVudC5cbiAgICovXG4gIGFmdGVyKFxuICAgIGhvb2tDb250ZXh0OiBSZWFkb25seTxIb29rQ29udGV4dDxGbGFnVmFsdWU+PixcbiAgICBldmFsdWF0aW9uRGV0YWlsczogRXZhbHVhdGlvbkRldGFpbHM8RmxhZ1ZhbHVlPlxuICApOiB2b2lkIHtcbiAgICBjb25zdCB7IGZsYWdLZXkgfSA9IGhvb2tDb250ZXh0O1xuXG4gICAgLy8gQ2hlY2sgaWYgdGhpcyBmbGFnIHNob3VsZCBiZSB0cmFja2VkXG4gICAgaWYgKHRoaXMub3B0aW9ucy5mbGFnRmlsdGVyICYmICF0aGlzLm9wdGlvbnMuZmxhZ0ZpbHRlcihmbGFnS2V5KSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIENoZWNrIGlmIFBlbmRvIGFnZW50IGlzIGF2YWlsYWJsZVxuICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSBcInVuZGVmaW5lZFwiIHx8ICF3aW5kb3cucGVuZG8/LnRyYWNrKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgcHJvcGVydGllczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgIGZsYWdfa2V5OiBmbGFnS2V5LFxuICAgICAgZmxhZ192YXJpYW50OiBldmFsdWF0aW9uRGV0YWlscy52YXJpYW50IHx8IFwidW5rbm93blwiLFxuICAgICAgZmxhZ19yZWFzb246IGV2YWx1YXRpb25EZXRhaWxzLnJlYXNvbiB8fCBcIlVOS05PV05cIixcbiAgICAgIGZsYWdfdmFsdWU6IHRoaXMuc3RyaW5naWZ5VmFsdWUoZXZhbHVhdGlvbkRldGFpbHMudmFsdWUpLFxuICAgICAgcHJvdmlkZXJfbmFtZTogaG9va0NvbnRleHQucHJvdmlkZXJNZXRhZGF0YS5uYW1lLFxuICAgIH07XG5cbiAgICAvLyBGaXJlLWFuZC1mb3JnZXQ6IGRlbGVnYXRlIHRvIFBlbmRvIGFnZW50XG4gICAgd2luZG93LnBlbmRvLnRyYWNrKHRoaXMub3B0aW9ucy5ldmVudE5hbWUsIHByb3BlcnRpZXMpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnQgYSBmbGFnIHZhbHVlIHRvIGEgc3RyaW5nIGZvciB0cmFja2luZy5cbiAgICovXG4gIHByaXZhdGUgc3RyaW5naWZ5VmFsdWUodmFsdWU6IEZsYWdWYWx1ZSk6IHN0cmluZyB7XG4gICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICByZXR1cm4gXCJudWxsXCI7XG4gICAgfVxuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIHJldHVybiBKU09OLnN0cmluZ2lmeSh2YWx1ZSk7XG4gICAgfVxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpO1xuICB9XG59XG4iXX0=
|
package/dist/index.d.ts
ADDED
package/dist/index.js
ADDED
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PendoTelemetryHook = exports.PendoProvider = void 0;
|
|
4
|
+
var PendoProvider_1 = require("./PendoProvider");
|
|
5
|
+
Object.defineProperty(exports, "PendoProvider", { enumerable: true, get: function () { return PendoProvider_1.PendoProvider; } });
|
|
6
|
+
var PendoTelemetryHook_1 = require("./PendoTelemetryHook");
|
|
7
|
+
Object.defineProperty(exports, "PendoTelemetryHook", { enumerable: true, get: function () { return PendoTelemetryHook_1.PendoTelemetryHook; } });
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsaURBQWdEO0FBQXZDLDhHQUFBLGFBQWEsT0FBQTtBQUV0QiwyREFBMEQ7QUFBakQsd0hBQUEsa0JBQWtCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBQZW5kb1Byb3ZpZGVyIH0gZnJvbSBcIi4vUGVuZG9Qcm92aWRlclwiO1xuZXhwb3J0IHR5cGUgeyBQZW5kb1Byb3ZpZGVyT3B0aW9ucyB9IGZyb20gXCIuL1BlbmRvUHJvdmlkZXJcIjtcbmV4cG9ydCB7IFBlbmRvVGVsZW1ldHJ5SG9vayB9IGZyb20gXCIuL1BlbmRvVGVsZW1ldHJ5SG9va1wiO1xuZXhwb3J0IHR5cGUgeyBQZW5kb1RlbGVtZXRyeUhvb2tPcHRpb25zIH0gZnJvbSBcIi4vUGVuZG9UZWxlbWV0cnlIb29rXCI7XG4iXX0=
|
package/package.json
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@pendo/openfeature-web-provider",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "OpenFeature provider for Pendo feature flags (web/browser)",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"module": "dist/index.mjs",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"files": [
|
|
9
|
+
"dist"
|
|
10
|
+
],
|
|
11
|
+
"scripts": {
|
|
12
|
+
"build": "tsc",
|
|
13
|
+
"test": "jest",
|
|
14
|
+
"lint": "eslint src"
|
|
15
|
+
},
|
|
16
|
+
"peerDependencies": {
|
|
17
|
+
"@openfeature/web-sdk": "^1.0.0"
|
|
18
|
+
},
|
|
19
|
+
"devDependencies": {
|
|
20
|
+
"@openfeature/web-sdk": "^1.0.0",
|
|
21
|
+
"@types/node": "^20.0.0",
|
|
22
|
+
"typescript": "^5.3.0"
|
|
23
|
+
},
|
|
24
|
+
"keywords": [
|
|
25
|
+
"openfeature",
|
|
26
|
+
"feature-flags",
|
|
27
|
+
"pendo",
|
|
28
|
+
"provider"
|
|
29
|
+
],
|
|
30
|
+
"license": "MIT",
|
|
31
|
+
"repository": {
|
|
32
|
+
"type": "git",
|
|
33
|
+
"url": "https://github.com/pendo-io/pendo-flags.git",
|
|
34
|
+
"directory": "packages/web-provider"
|
|
35
|
+
}
|
|
36
|
+
}
|