@savvagent/angular 1.0.1
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/CHANGELOG.md +26 -0
- package/LICENSE +21 -0
- package/README.md +484 -0
- package/coverage/base.css +224 -0
- package/coverage/block-navigation.js +87 -0
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +131 -0
- package/coverage/lcov-report/base.css +224 -0
- package/coverage/lcov-report/block-navigation.js +87 -0
- package/coverage/lcov-report/favicon.png +0 -0
- package/coverage/lcov-report/index.html +131 -0
- package/coverage/lcov-report/module.ts.html +289 -0
- package/coverage/lcov-report/prettify.css +1 -0
- package/coverage/lcov-report/prettify.js +2 -0
- package/coverage/lcov-report/service.ts.html +1846 -0
- package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/coverage/lcov-report/sorter.js +210 -0
- package/coverage/lcov.info +242 -0
- package/coverage/module.ts.html +289 -0
- package/coverage/prettify.css +1 -0
- package/coverage/prettify.js +2 -0
- package/coverage/service.ts.html +1846 -0
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +210 -0
- package/dist/README.md +484 -0
- package/dist/esm2022/index.mjs +15 -0
- package/dist/esm2022/module.mjs +75 -0
- package/dist/esm2022/savvagent-angular.mjs +5 -0
- package/dist/esm2022/service.mjs +473 -0
- package/dist/fesm2022/savvagent-angular.mjs +563 -0
- package/dist/fesm2022/savvagent-angular.mjs.map +1 -0
- package/dist/index.d.ts +13 -0
- package/dist/module.d.ts +57 -0
- package/dist/service.d.ts +319 -0
- package/jest.config.js +40 -0
- package/ng-package.json +8 -0
- package/package.json +73 -0
- package/setup-jest.ts +2 -0
- package/src/index.spec.ts +144 -0
- package/src/index.ts +38 -0
- package/src/module.spec.ts +283 -0
- package/src/module.ts +68 -0
- package/src/service.spec.ts +945 -0
- package/src/service.ts +587 -0
- package/test-utils/angular-core-mock.ts +28 -0
- package/test-utils/angular-testing-mock.ts +87 -0
- package/tsconfig.json +33 -0
- package/tsconfig.spec.json +11 -0
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { NgModule } from '@angular/core';
|
|
2
|
+
import { SavvagentService, SAVVAGENT_CONFIG } from './service';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Angular module for Savvagent feature flags.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* ```typescript
|
|
9
|
+
* // app.module.ts
|
|
10
|
+
* import { SavvagentModule } from '@savvagent/angular';
|
|
11
|
+
*
|
|
12
|
+
* @NgModule({
|
|
13
|
+
* imports: [
|
|
14
|
+
* SavvagentModule.forRoot({
|
|
15
|
+
* config: {
|
|
16
|
+
* apiKey: 'sdk_your_api_key',
|
|
17
|
+
* baseUrl: 'https://api.savvagent.com'
|
|
18
|
+
* },
|
|
19
|
+
* defaultContext: {
|
|
20
|
+
* applicationId: 'my-app',
|
|
21
|
+
* environment: 'production',
|
|
22
|
+
* userId: 'user-123'
|
|
23
|
+
* }
|
|
24
|
+
* })
|
|
25
|
+
* ]
|
|
26
|
+
* })
|
|
27
|
+
* export class AppModule {}
|
|
28
|
+
* ```
|
|
29
|
+
*
|
|
30
|
+
* @example
|
|
31
|
+
* ```typescript
|
|
32
|
+
* // For standalone components (Angular 14+)
|
|
33
|
+
* import { SavvagentModule } from '@savvagent/angular';
|
|
34
|
+
*
|
|
35
|
+
* bootstrapApplication(AppComponent, {
|
|
36
|
+
* providers: [
|
|
37
|
+
* importProvidersFrom(
|
|
38
|
+
* SavvagentModule.forRoot({
|
|
39
|
+
* config: { apiKey: 'sdk_...' }
|
|
40
|
+
* })
|
|
41
|
+
* )
|
|
42
|
+
* ]
|
|
43
|
+
* });
|
|
44
|
+
* ```
|
|
45
|
+
*/
|
|
46
|
+
export class SavvagentModule {
|
|
47
|
+
/**
|
|
48
|
+
* Configure the Savvagent module with API key and default context.
|
|
49
|
+
*
|
|
50
|
+
* @param savvagentConfig - Configuration including API key and optional default context
|
|
51
|
+
* @returns Module with providers
|
|
52
|
+
*/
|
|
53
|
+
static forRoot(savvagentConfig) {
|
|
54
|
+
return {
|
|
55
|
+
ngModule: SavvagentModule,
|
|
56
|
+
providers: [
|
|
57
|
+
{
|
|
58
|
+
provide: SAVVAGENT_CONFIG,
|
|
59
|
+
useValue: savvagentConfig
|
|
60
|
+
},
|
|
61
|
+
SavvagentService
|
|
62
|
+
]
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
|
66
|
+
static ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.14", ngImport: i0, type: SavvagentModule });
|
|
67
|
+
static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentModule, providers: [SavvagentService] });
|
|
68
|
+
}
|
|
69
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentModule, decorators: [{
|
|
70
|
+
type: NgModule,
|
|
71
|
+
args: [{
|
|
72
|
+
providers: [SavvagentService]
|
|
73
|
+
}]
|
|
74
|
+
}] });
|
|
75
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL21vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUF1QixNQUFNLGVBQWUsQ0FBQztBQUM5RCxPQUFPLEVBQUUsZ0JBQWdCLEVBQW1CLGdCQUFnQixFQUFFLE1BQU0sV0FBVyxDQUFDOztBQUVoRjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F5Q0c7QUFJSCxNQUFNLE9BQU8sZUFBZTtJQUMxQjs7Ozs7T0FLRztJQUNILE1BQU0sQ0FBQyxPQUFPLENBQUMsZUFBZ0M7UUFDN0MsT0FBTztZQUNMLFFBQVEsRUFBRSxlQUFlO1lBQ3pCLFNBQVMsRUFBRTtnQkFDVDtvQkFDRSxPQUFPLEVBQUUsZ0JBQWdCO29CQUN6QixRQUFRLEVBQUUsZUFBZTtpQkFDMUI7Z0JBQ0QsZ0JBQWdCO2FBQ2pCO1NBQ0YsQ0FBQztJQUNKLENBQUM7d0dBbEJVLGVBQWU7eUdBQWYsZUFBZTt5R0FBZixlQUFlLGFBRmYsQ0FBQyxnQkFBZ0IsQ0FBQzs7NEZBRWxCLGVBQWU7a0JBSDNCLFFBQVE7bUJBQUM7b0JBQ1IsU0FBUyxFQUFFLENBQUMsZ0JBQWdCLENBQUM7aUJBQzlCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUsIE1vZHVsZVdpdGhQcm92aWRlcnMgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFNhdnZhZ2VudFNlcnZpY2UsIFNhdnZhZ2VudENvbmZpZywgU0FWVkFHRU5UX0NPTkZJRyB9IGZyb20gJy4vc2VydmljZSc7XG5cbi8qKlxuICogQW5ndWxhciBtb2R1bGUgZm9yIFNhdnZhZ2VudCBmZWF0dXJlIGZsYWdzLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBhcHAubW9kdWxlLnRzXG4gKiBpbXBvcnQgeyBTYXZ2YWdlbnRNb2R1bGUgfSBmcm9tICdAc2F2dmFnZW50L2FuZ3VsYXInO1xuICpcbiAqIEBOZ01vZHVsZSh7XG4gKiAgIGltcG9ydHM6IFtcbiAqICAgICBTYXZ2YWdlbnRNb2R1bGUuZm9yUm9vdCh7XG4gKiAgICAgICBjb25maWc6IHtcbiAqICAgICAgICAgYXBpS2V5OiAnc2RrX3lvdXJfYXBpX2tleScsXG4gKiAgICAgICAgIGJhc2VVcmw6ICdodHRwczovL2FwaS5zYXZ2YWdlbnQuY29tJ1xuICogICAgICAgfSxcbiAqICAgICAgIGRlZmF1bHRDb250ZXh0OiB7XG4gKiAgICAgICAgIGFwcGxpY2F0aW9uSWQ6ICdteS1hcHAnLFxuICogICAgICAgICBlbnZpcm9ubWVudDogJ3Byb2R1Y3Rpb24nLFxuICogICAgICAgICB1c2VySWQ6ICd1c2VyLTEyMydcbiAqICAgICAgIH1cbiAqICAgICB9KVxuICogICBdXG4gKiB9KVxuICogZXhwb3J0IGNsYXNzIEFwcE1vZHVsZSB7fVxuICogYGBgXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIEZvciBzdGFuZGFsb25lIGNvbXBvbmVudHMgKEFuZ3VsYXIgMTQrKVxuICogaW1wb3J0IHsgU2F2dmFnZW50TW9kdWxlIH0gZnJvbSAnQHNhdnZhZ2VudC9hbmd1bGFyJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgaW1wb3J0UHJvdmlkZXJzRnJvbShcbiAqICAgICAgIFNhdnZhZ2VudE1vZHVsZS5mb3JSb290KHtcbiAqICAgICAgICAgY29uZmlnOiB7IGFwaUtleTogJ3Nka18uLi4nIH1cbiAqICAgICAgIH0pXG4gKiAgICAgKVxuICogICBdXG4gKiB9KTtcbiAqIGBgYFxuICovXG5ATmdNb2R1bGUoe1xuICBwcm92aWRlcnM6IFtTYXZ2YWdlbnRTZXJ2aWNlXVxufSlcbmV4cG9ydCBjbGFzcyBTYXZ2YWdlbnRNb2R1bGUge1xuICAvKipcbiAgICogQ29uZmlndXJlIHRoZSBTYXZ2YWdlbnQgbW9kdWxlIHdpdGggQVBJIGtleSBhbmQgZGVmYXVsdCBjb250ZXh0LlxuICAgKlxuICAgKiBAcGFyYW0gc2F2dmFnZW50Q29uZmlnIC0gQ29uZmlndXJhdGlvbiBpbmNsdWRpbmcgQVBJIGtleSBhbmQgb3B0aW9uYWwgZGVmYXVsdCBjb250ZXh0XG4gICAqIEByZXR1cm5zIE1vZHVsZSB3aXRoIHByb3ZpZGVyc1xuICAgKi9cbiAgc3RhdGljIGZvclJvb3Qoc2F2dmFnZW50Q29uZmlnOiBTYXZ2YWdlbnRDb25maWcpOiBNb2R1bGVXaXRoUHJvdmlkZXJzPFNhdnZhZ2VudE1vZHVsZT4ge1xuICAgIHJldHVybiB7XG4gICAgICBuZ01vZHVsZTogU2F2dmFnZW50TW9kdWxlLFxuICAgICAgcHJvdmlkZXJzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBTQVZWQUdFTlRfQ09ORklHLFxuICAgICAgICAgIHVzZVZhbHVlOiBzYXZ2YWdlbnRDb25maWdcbiAgICAgICAgfSxcbiAgICAgICAgU2F2dmFnZW50U2VydmljZVxuICAgICAgXVxuICAgIH07XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './index';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2F2dmFnZW50LWFuZ3VsYXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2F2dmFnZW50LWFuZ3VsYXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0=
|
|
@@ -0,0 +1,473 @@
|
|
|
1
|
+
import { Injectable, Inject, InjectionToken, Optional } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject, from, of, Subject } from 'rxjs';
|
|
3
|
+
import { map, takeUntil, catchError, distinctUntilChanged } from 'rxjs/operators';
|
|
4
|
+
import { FlagClient } from '@savvagent/sdk';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Injection token for Savvagent configuration
|
|
8
|
+
*/
|
|
9
|
+
export const SAVVAGENT_CONFIG = new InjectionToken('SAVVAGENT_CONFIG');
|
|
10
|
+
/**
|
|
11
|
+
* Angular service for Savvagent feature flags.
|
|
12
|
+
* Provides reactive flag evaluation using RxJS Observables.
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```typescript
|
|
16
|
+
* // In your component
|
|
17
|
+
* @Component({...})
|
|
18
|
+
* export class MyComponent {
|
|
19
|
+
* newFeature$ = this.savvagent.flag$('new-feature');
|
|
20
|
+
*
|
|
21
|
+
* constructor(private savvagent: SavvagentService) {}
|
|
22
|
+
* }
|
|
23
|
+
*
|
|
24
|
+
* // In your template
|
|
25
|
+
* <div *ngIf="(newFeature$ | async)?.value">
|
|
26
|
+
* New feature content!
|
|
27
|
+
* </div>
|
|
28
|
+
* ```
|
|
29
|
+
*/
|
|
30
|
+
export class SavvagentService {
|
|
31
|
+
client = null;
|
|
32
|
+
destroy$ = new Subject();
|
|
33
|
+
isReady$ = new BehaviorSubject(false);
|
|
34
|
+
defaultContext = {};
|
|
35
|
+
flagSubjects = new Map();
|
|
36
|
+
constructor(config) {
|
|
37
|
+
if (config) {
|
|
38
|
+
this.initialize(config);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Initialize the Savvagent client with configuration.
|
|
43
|
+
* Call this if not using the SAVVAGENT_CONFIG injection token.
|
|
44
|
+
*
|
|
45
|
+
* @param savvagentConfig - Configuration including API key and default context
|
|
46
|
+
*
|
|
47
|
+
* @example
|
|
48
|
+
* ```typescript
|
|
49
|
+
* @Component({...})
|
|
50
|
+
* export class AppComponent implements OnInit {
|
|
51
|
+
* constructor(private savvagent: SavvagentService) {}
|
|
52
|
+
*
|
|
53
|
+
* ngOnInit() {
|
|
54
|
+
* this.savvagent.initialize({
|
|
55
|
+
* config: { apiKey: 'sdk_...' },
|
|
56
|
+
* defaultContext: {
|
|
57
|
+
* applicationId: 'my-app',
|
|
58
|
+
* environment: 'development',
|
|
59
|
+
* userId: 'user-123'
|
|
60
|
+
* }
|
|
61
|
+
* });
|
|
62
|
+
* }
|
|
63
|
+
* }
|
|
64
|
+
* ```
|
|
65
|
+
*/
|
|
66
|
+
initialize(savvagentConfig) {
|
|
67
|
+
if (this.client) {
|
|
68
|
+
console.warn('[Savvagent] Client already initialized. Call close() first to reinitialize.');
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
try {
|
|
72
|
+
this.client = new FlagClient(savvagentConfig.config);
|
|
73
|
+
// Convert DefaultFlagContext to FlagContext format (camelCase to snake_case)
|
|
74
|
+
if (savvagentConfig.defaultContext) {
|
|
75
|
+
this.defaultContext = {
|
|
76
|
+
application_id: savvagentConfig.defaultContext.applicationId,
|
|
77
|
+
environment: savvagentConfig.defaultContext.environment,
|
|
78
|
+
organization_id: savvagentConfig.defaultContext.organizationId,
|
|
79
|
+
user_id: savvagentConfig.defaultContext.userId,
|
|
80
|
+
anonymous_id: savvagentConfig.defaultContext.anonymousId,
|
|
81
|
+
session_id: savvagentConfig.defaultContext.sessionId,
|
|
82
|
+
language: savvagentConfig.defaultContext.language,
|
|
83
|
+
attributes: savvagentConfig.defaultContext.attributes,
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
this.isReady$.next(true);
|
|
87
|
+
// Subscribe to override changes to re-evaluate all active flags
|
|
88
|
+
this.client.onOverrideChange(() => {
|
|
89
|
+
this.reEvaluateAllFlags();
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
catch (error) {
|
|
93
|
+
console.error('[Savvagent] Failed to initialize client:', error);
|
|
94
|
+
savvagentConfig.config.onError?.(error);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Observable that emits true when the client is ready.
|
|
99
|
+
*/
|
|
100
|
+
get ready$() {
|
|
101
|
+
return this.isReady$.asObservable();
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Check if the client is ready.
|
|
105
|
+
*/
|
|
106
|
+
get isReady() {
|
|
107
|
+
return this.isReady$.value;
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Get the underlying FlagClient instance for advanced use cases.
|
|
111
|
+
*/
|
|
112
|
+
get flagClient() {
|
|
113
|
+
return this.client;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Merge default context with per-call context.
|
|
117
|
+
*/
|
|
118
|
+
mergeContext(context) {
|
|
119
|
+
return {
|
|
120
|
+
...this.defaultContext,
|
|
121
|
+
...context,
|
|
122
|
+
attributes: {
|
|
123
|
+
...this.defaultContext.attributes,
|
|
124
|
+
...context?.attributes,
|
|
125
|
+
},
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Get a reactive Observable for a feature flag.
|
|
130
|
+
* Automatically updates when the flag value changes.
|
|
131
|
+
*
|
|
132
|
+
* @param flagKey - The feature flag key to evaluate
|
|
133
|
+
* @param options - Configuration options
|
|
134
|
+
* @returns Observable of flag evaluation state
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* ```typescript
|
|
138
|
+
* // In your component
|
|
139
|
+
* newFeature$ = this.savvagent.flag$('new-feature', {
|
|
140
|
+
* defaultValue: false,
|
|
141
|
+
* realtime: true,
|
|
142
|
+
* context: { attributes: { plan: 'pro' } }
|
|
143
|
+
* });
|
|
144
|
+
*
|
|
145
|
+
* // In template
|
|
146
|
+
* <ng-container *ngIf="newFeature$ | async as flag">
|
|
147
|
+
* <app-loading *ngIf="flag.loading"></app-loading>
|
|
148
|
+
* <app-new-feature *ngIf="flag.value"></app-new-feature>
|
|
149
|
+
* <app-old-feature *ngIf="!flag.value && !flag.loading"></app-old-feature>
|
|
150
|
+
* </ng-container>
|
|
151
|
+
* ```
|
|
152
|
+
*/
|
|
153
|
+
flag$(flagKey, options = {}) {
|
|
154
|
+
const { context, defaultValue = false, realtime = true } = options;
|
|
155
|
+
const mergedContext = this.mergeContext(context);
|
|
156
|
+
const cacheKey = this.getCacheKey(flagKey, mergedContext);
|
|
157
|
+
// Check if we already have a subject for this flag+context
|
|
158
|
+
if (!this.flagSubjects.has(cacheKey)) {
|
|
159
|
+
const subject = new BehaviorSubject({
|
|
160
|
+
value: defaultValue,
|
|
161
|
+
loading: true,
|
|
162
|
+
error: null,
|
|
163
|
+
result: null,
|
|
164
|
+
});
|
|
165
|
+
this.flagSubjects.set(cacheKey, subject);
|
|
166
|
+
// Initial evaluation
|
|
167
|
+
this.evaluateAndEmit(flagKey, mergedContext, defaultValue, subject);
|
|
168
|
+
// Set up real-time subscription if enabled
|
|
169
|
+
if (realtime && this.client) {
|
|
170
|
+
const unsubscribe = this.client.subscribe(flagKey, () => {
|
|
171
|
+
this.evaluateAndEmit(flagKey, mergedContext, defaultValue, subject);
|
|
172
|
+
});
|
|
173
|
+
// Clean up subscription when subject is complete
|
|
174
|
+
subject.pipe(takeUntil(this.destroy$)).subscribe({
|
|
175
|
+
complete: () => unsubscribe(),
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
return this.flagSubjects.get(cacheKey).asObservable().pipe(takeUntil(this.destroy$), distinctUntilChanged((a, b) => a.value === b.value &&
|
|
180
|
+
a.loading === b.loading &&
|
|
181
|
+
a.error === b.error));
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Generate a cache key for a flag+context combination.
|
|
185
|
+
*/
|
|
186
|
+
getCacheKey(flagKey, context) {
|
|
187
|
+
return `${flagKey}:${JSON.stringify(context)}`;
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* Evaluate a flag and emit the result to a subject.
|
|
191
|
+
*/
|
|
192
|
+
async evaluateAndEmit(flagKey, context, defaultValue, subject) {
|
|
193
|
+
if (!this.client) {
|
|
194
|
+
subject.next({
|
|
195
|
+
value: defaultValue,
|
|
196
|
+
loading: false,
|
|
197
|
+
error: new Error('Savvagent client not initialized'),
|
|
198
|
+
result: null,
|
|
199
|
+
});
|
|
200
|
+
return;
|
|
201
|
+
}
|
|
202
|
+
try {
|
|
203
|
+
const result = await this.client.evaluate(flagKey, context);
|
|
204
|
+
subject.next({
|
|
205
|
+
value: result.value,
|
|
206
|
+
loading: false,
|
|
207
|
+
error: null,
|
|
208
|
+
result,
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
catch (error) {
|
|
212
|
+
subject.next({
|
|
213
|
+
value: defaultValue,
|
|
214
|
+
loading: false,
|
|
215
|
+
error: error,
|
|
216
|
+
result: null,
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Re-evaluate all active flag subscriptions.
|
|
222
|
+
* Called when overrides change.
|
|
223
|
+
*/
|
|
224
|
+
reEvaluateAllFlags() {
|
|
225
|
+
this.flagSubjects.forEach((subject, cacheKey) => {
|
|
226
|
+
const [flagKey, contextJson] = cacheKey.split(':', 2);
|
|
227
|
+
const context = JSON.parse(contextJson || '{}');
|
|
228
|
+
const currentValue = subject.value;
|
|
229
|
+
this.evaluateAndEmit(flagKey, context, currentValue.value, subject);
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
/**
|
|
233
|
+
* Get a flag value as a simple Observable<boolean>.
|
|
234
|
+
* Useful when you only need the value without loading/error states.
|
|
235
|
+
*
|
|
236
|
+
* @param flagKey - The feature flag key to evaluate
|
|
237
|
+
* @param options - Configuration options
|
|
238
|
+
* @returns Observable of boolean flag value
|
|
239
|
+
*
|
|
240
|
+
* @example
|
|
241
|
+
* ```typescript
|
|
242
|
+
* isFeatureEnabled$ = this.savvagent.flagValue$('my-feature');
|
|
243
|
+
*
|
|
244
|
+
* // In template
|
|
245
|
+
* <button *ngIf="isFeatureEnabled$ | async">New Button</button>
|
|
246
|
+
* ```
|
|
247
|
+
*/
|
|
248
|
+
flagValue$(flagKey, options = {}) {
|
|
249
|
+
return this.flag$(flagKey, options).pipe(map((result) => result.value), distinctUntilChanged());
|
|
250
|
+
}
|
|
251
|
+
/**
|
|
252
|
+
* Evaluate a feature flag once (non-reactive).
|
|
253
|
+
* For reactive updates, use flag$() instead.
|
|
254
|
+
*
|
|
255
|
+
* @param flagKey - The feature flag key to evaluate
|
|
256
|
+
* @param context - Optional context for targeting
|
|
257
|
+
* @returns Promise with detailed evaluation result
|
|
258
|
+
*
|
|
259
|
+
* @example
|
|
260
|
+
* ```typescript
|
|
261
|
+
* async checkFeature() {
|
|
262
|
+
* const result = await this.savvagent.evaluate('new-feature');
|
|
263
|
+
* if (result.value) {
|
|
264
|
+
* // Feature is enabled
|
|
265
|
+
* }
|
|
266
|
+
* }
|
|
267
|
+
* ```
|
|
268
|
+
*/
|
|
269
|
+
async evaluate(flagKey, context) {
|
|
270
|
+
if (!this.client) {
|
|
271
|
+
throw new Error('Savvagent client not initialized');
|
|
272
|
+
}
|
|
273
|
+
return this.client.evaluate(flagKey, this.mergeContext(context));
|
|
274
|
+
}
|
|
275
|
+
/**
|
|
276
|
+
* Check if a feature flag is enabled (non-reactive).
|
|
277
|
+
*
|
|
278
|
+
* @param flagKey - The feature flag key to evaluate
|
|
279
|
+
* @param context - Optional context for targeting
|
|
280
|
+
* @returns Promise<boolean>
|
|
281
|
+
*/
|
|
282
|
+
async isEnabled(flagKey, context) {
|
|
283
|
+
if (!this.client) {
|
|
284
|
+
return false;
|
|
285
|
+
}
|
|
286
|
+
return this.client.isEnabled(flagKey, this.mergeContext(context));
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* Execute code conditionally based on flag value.
|
|
290
|
+
*
|
|
291
|
+
* @param flagKey - The flag key to check
|
|
292
|
+
* @param callback - Function to execute if flag is enabled
|
|
293
|
+
* @param context - Optional context for targeting
|
|
294
|
+
*
|
|
295
|
+
* @example
|
|
296
|
+
* ```typescript
|
|
297
|
+
* await this.savvagent.withFlag('analytics-enabled', async () => {
|
|
298
|
+
* await this.analytics.track('page_view');
|
|
299
|
+
* });
|
|
300
|
+
* ```
|
|
301
|
+
*/
|
|
302
|
+
async withFlag(flagKey, callback, context) {
|
|
303
|
+
if (!this.client) {
|
|
304
|
+
return null;
|
|
305
|
+
}
|
|
306
|
+
return this.client.withFlag(flagKey, callback, this.mergeContext(context));
|
|
307
|
+
}
|
|
308
|
+
/**
|
|
309
|
+
* Track an error with flag context.
|
|
310
|
+
*
|
|
311
|
+
* @param flagKey - The flag key associated with the error
|
|
312
|
+
* @param error - The error that occurred
|
|
313
|
+
* @param context - Optional context
|
|
314
|
+
*/
|
|
315
|
+
trackError(flagKey, error, context) {
|
|
316
|
+
this.client?.trackError(flagKey, error, this.mergeContext(context));
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Set the user ID for logged-in users.
|
|
320
|
+
*
|
|
321
|
+
* @param userId - The user ID (or null to clear)
|
|
322
|
+
*/
|
|
323
|
+
setUserId(userId) {
|
|
324
|
+
this.client?.setUserId(userId);
|
|
325
|
+
}
|
|
326
|
+
/**
|
|
327
|
+
* Get the current user ID.
|
|
328
|
+
*/
|
|
329
|
+
getUserId() {
|
|
330
|
+
return this.client?.getUserId() || null;
|
|
331
|
+
}
|
|
332
|
+
/**
|
|
333
|
+
* Get the current anonymous ID.
|
|
334
|
+
*/
|
|
335
|
+
getAnonymousId() {
|
|
336
|
+
return this.client?.getAnonymousId() || null;
|
|
337
|
+
}
|
|
338
|
+
/**
|
|
339
|
+
* Set a custom anonymous ID.
|
|
340
|
+
*/
|
|
341
|
+
setAnonymousId(id) {
|
|
342
|
+
this.client?.setAnonymousId(id);
|
|
343
|
+
}
|
|
344
|
+
// =====================
|
|
345
|
+
// Local Override Methods
|
|
346
|
+
// =====================
|
|
347
|
+
/**
|
|
348
|
+
* Set a local override for a flag.
|
|
349
|
+
* Overrides take precedence over server values.
|
|
350
|
+
*
|
|
351
|
+
* @param flagKey - The flag key to override
|
|
352
|
+
* @param value - The override value
|
|
353
|
+
*/
|
|
354
|
+
setOverride(flagKey, value) {
|
|
355
|
+
this.client?.setOverride(flagKey, value);
|
|
356
|
+
}
|
|
357
|
+
/**
|
|
358
|
+
* Clear a local override for a flag.
|
|
359
|
+
*/
|
|
360
|
+
clearOverride(flagKey) {
|
|
361
|
+
this.client?.clearOverride(flagKey);
|
|
362
|
+
}
|
|
363
|
+
/**
|
|
364
|
+
* Clear all local overrides.
|
|
365
|
+
*/
|
|
366
|
+
clearAllOverrides() {
|
|
367
|
+
this.client?.clearAllOverrides();
|
|
368
|
+
}
|
|
369
|
+
/**
|
|
370
|
+
* Check if a flag has a local override.
|
|
371
|
+
*/
|
|
372
|
+
hasOverride(flagKey) {
|
|
373
|
+
return this.client?.hasOverride(flagKey) || false;
|
|
374
|
+
}
|
|
375
|
+
/**
|
|
376
|
+
* Get the override value for a flag.
|
|
377
|
+
*/
|
|
378
|
+
getOverride(flagKey) {
|
|
379
|
+
return this.client?.getOverride(flagKey);
|
|
380
|
+
}
|
|
381
|
+
/**
|
|
382
|
+
* Get all current overrides.
|
|
383
|
+
*/
|
|
384
|
+
getOverrides() {
|
|
385
|
+
return this.client?.getOverrides() || {};
|
|
386
|
+
}
|
|
387
|
+
/**
|
|
388
|
+
* Set multiple overrides at once.
|
|
389
|
+
*/
|
|
390
|
+
setOverrides(overrides) {
|
|
391
|
+
this.client?.setOverrides(overrides);
|
|
392
|
+
}
|
|
393
|
+
// =====================
|
|
394
|
+
// Flag Discovery Methods
|
|
395
|
+
// =====================
|
|
396
|
+
/**
|
|
397
|
+
* Get all flags for the application.
|
|
398
|
+
*
|
|
399
|
+
* @param environment - Environment to evaluate (default: 'development')
|
|
400
|
+
* @returns Observable of flag definitions
|
|
401
|
+
*/
|
|
402
|
+
getAllFlags$(environment = 'development') {
|
|
403
|
+
if (!this.client) {
|
|
404
|
+
return of([]);
|
|
405
|
+
}
|
|
406
|
+
return from(this.client.getAllFlags(environment)).pipe(catchError((error) => {
|
|
407
|
+
console.error('[Savvagent] Failed to fetch all flags:', error);
|
|
408
|
+
return of([]);
|
|
409
|
+
}));
|
|
410
|
+
}
|
|
411
|
+
/**
|
|
412
|
+
* Get all flags for the application (Promise-based).
|
|
413
|
+
*/
|
|
414
|
+
async getAllFlags(environment = 'development') {
|
|
415
|
+
if (!this.client) {
|
|
416
|
+
return [];
|
|
417
|
+
}
|
|
418
|
+
return this.client.getAllFlags(environment);
|
|
419
|
+
}
|
|
420
|
+
/**
|
|
421
|
+
* Get enterprise-scoped flags only.
|
|
422
|
+
*/
|
|
423
|
+
async getEnterpriseFlags(environment = 'development') {
|
|
424
|
+
if (!this.client) {
|
|
425
|
+
return [];
|
|
426
|
+
}
|
|
427
|
+
return this.client.getEnterpriseFlags(environment);
|
|
428
|
+
}
|
|
429
|
+
// =====================
|
|
430
|
+
// Cache & Connection
|
|
431
|
+
// =====================
|
|
432
|
+
/**
|
|
433
|
+
* Clear the flag cache.
|
|
434
|
+
*/
|
|
435
|
+
clearCache() {
|
|
436
|
+
this.client?.clearCache();
|
|
437
|
+
}
|
|
438
|
+
/**
|
|
439
|
+
* Check if real-time connection is active.
|
|
440
|
+
*/
|
|
441
|
+
isRealtimeConnected() {
|
|
442
|
+
return this.client?.isRealtimeConnected() || false;
|
|
443
|
+
}
|
|
444
|
+
/**
|
|
445
|
+
* Close the client and cleanup resources.
|
|
446
|
+
*/
|
|
447
|
+
close() {
|
|
448
|
+
this.client?.close();
|
|
449
|
+
this.client = null;
|
|
450
|
+
this.isReady$.next(false);
|
|
451
|
+
this.flagSubjects.forEach((subject) => subject.complete());
|
|
452
|
+
this.flagSubjects.clear();
|
|
453
|
+
}
|
|
454
|
+
ngOnDestroy() {
|
|
455
|
+
this.destroy$.next();
|
|
456
|
+
this.destroy$.complete();
|
|
457
|
+
this.close();
|
|
458
|
+
}
|
|
459
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentService, deps: [{ token: SAVVAGENT_CONFIG, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
460
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentService, providedIn: 'root' });
|
|
461
|
+
}
|
|
462
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SavvagentService, decorators: [{
|
|
463
|
+
type: Injectable,
|
|
464
|
+
args: [{
|
|
465
|
+
providedIn: 'root'
|
|
466
|
+
}]
|
|
467
|
+
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
|
468
|
+
type: Optional
|
|
469
|
+
}, {
|
|
470
|
+
type: Inject,
|
|
471
|
+
args: [SAVVAGENT_CONFIG]
|
|
472
|
+
}] }] });
|
|
473
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../../src/service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, OnDestroy, Inject, InjectionToken, Optional } from '@angular/core';\nimport { BehaviorSubject, Observable, from, of, Subject } from 'rxjs';\nimport { map, takeUntil, catchError, distinctUntilChanged } from 'rxjs/operators';\nimport { FlagClient, FlagClientConfig, FlagContext, FlagEvaluationResult, FlagDefinition } from '@savvagent/sdk';\n\n/**\n * Default context values that apply to all flag evaluations\n * Per SDK Developer Guide: https://docs.savvagent.com/sdk-developer-guide\n */\nexport interface DefaultFlagContext {\n  /** Application ID for application-scoped flags */\n  applicationId?: string;\n  /** Environment (development, staging, production) */\n  environment?: string;\n  /** Organization ID for multi-tenant apps */\n  organizationId?: string;\n  /** Default user ID (required for percentage rollouts) */\n  userId?: string;\n  /** Default anonymous ID (alternative to userId for anonymous users) */\n  anonymousId?: string;\n  /** Session ID as fallback identifier */\n  sessionId?: string;\n  /** User's language code (e.g., \"en\", \"es\") */\n  language?: string;\n  /** Default attributes for targeting */\n  attributes?: Record<string, any>;\n}\n\n/**\n * Configuration for the Savvagent Angular service\n */\nexport interface SavvagentConfig {\n  /** SDK API key configuration */\n  config: FlagClientConfig;\n  /** Default context values applied to all flag evaluations */\n  defaultContext?: DefaultFlagContext;\n}\n\n/**\n * Injection token for Savvagent configuration\n */\nexport const SAVVAGENT_CONFIG = new InjectionToken<SavvagentConfig>('SAVVAGENT_CONFIG');\n\n/**\n * Result from flag evaluation as an Observable\n */\nexport interface FlagObservableResult {\n  /** Current flag value */\n  value: boolean;\n  /** Whether the flag is currently being evaluated */\n  loading: boolean;\n  /** Error if evaluation failed */\n  error: Error | null;\n  /** Detailed evaluation result */\n  result: FlagEvaluationResult | null;\n}\n\n/**\n * Options for flag evaluation\n */\nexport interface FlagOptions {\n  /** Context for flag evaluation (user_id, attributes, etc.) */\n  context?: FlagContext;\n  /** Default value to use while loading or on error */\n  defaultValue?: boolean;\n  /** Enable real-time updates for this flag */\n  realtime?: boolean;\n}\n\n/**\n * Angular service for Savvagent feature flags.\n * Provides reactive flag evaluation using RxJS Observables.\n *\n * @example\n * ```typescript\n * // In your component\n * @Component({...})\n * export class MyComponent {\n *   newFeature$ = this.savvagent.flag$('new-feature');\n *\n *   constructor(private savvagent: SavvagentService) {}\n * }\n *\n * // In your template\n * <div *ngIf=\"(newFeature$ | async)?.value\">\n *   New feature content!\n * </div>\n * ```\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class SavvagentService implements OnDestroy {\n  private client: FlagClient | null = null;\n  private destroy$ = new Subject<void>();\n  private isReady$ = new BehaviorSubject<boolean>(false);\n  private defaultContext: FlagContext = {};\n  private flagSubjects = new Map<string, BehaviorSubject<FlagObservableResult>>();\n\n  constructor(\n    @Optional() @Inject(SAVVAGENT_CONFIG) config?: SavvagentConfig\n  ) {\n    if (config) {\n      this.initialize(config);\n    }\n  }\n\n  /**\n   * Initialize the Savvagent client with configuration.\n   * Call this if not using the SAVVAGENT_CONFIG injection token.\n   *\n   * @param savvagentConfig - Configuration including API key and default context\n   *\n   * @example\n   * ```typescript\n   * @Component({...})\n   * export class AppComponent implements OnInit {\n   *   constructor(private savvagent: SavvagentService) {}\n   *\n   *   ngOnInit() {\n   *     this.savvagent.initialize({\n   *       config: { apiKey: 'sdk_...' },\n   *       defaultContext: {\n   *         applicationId: 'my-app',\n   *         environment: 'development',\n   *         userId: 'user-123'\n   *       }\n   *     });\n   *   }\n   * }\n   * ```\n   */\n  initialize(savvagentConfig: SavvagentConfig): void {\n    if (this.client) {\n      console.warn('[Savvagent] Client already initialized. Call close() first to reinitialize.');\n      return;\n    }\n\n    try {\n      this.client = new FlagClient(savvagentConfig.config);\n\n      // Convert DefaultFlagContext to FlagContext format (camelCase to snake_case)\n      if (savvagentConfig.defaultContext) {\n        this.defaultContext = {\n          application_id: savvagentConfig.defaultContext.applicationId,\n          environment: savvagentConfig.defaultContext.environment,\n          organization_id: savvagentConfig.defaultContext.organizationId,\n          user_id: savvagentConfig.defaultContext.userId,\n          anonymous_id: savvagentConfig.defaultContext.anonymousId,\n          session_id: savvagentConfig.defaultContext.sessionId,\n          language: savvagentConfig.defaultContext.language,\n          attributes: savvagentConfig.defaultContext.attributes,\n        };\n      }\n\n      this.isReady$.next(true);\n\n      // Subscribe to override changes to re-evaluate all active flags\n      this.client.onOverrideChange(() => {\n        this.reEvaluateAllFlags();\n      });\n    } catch (error) {\n      console.error('[Savvagent] Failed to initialize client:', error);\n      savvagentConfig.config.onError?.(error as Error);\n    }\n  }\n\n  /**\n   * Observable that emits true when the client is ready.\n   */\n  get ready$(): Observable<boolean> {\n    return this.isReady$.asObservable();\n  }\n\n  /**\n   * Check if the client is ready.\n   */\n  get isReady(): boolean {\n    return this.isReady$.value;\n  }\n\n  /**\n   * Get the underlying FlagClient instance for advanced use cases.\n   */\n  get flagClient(): FlagClient | null {\n    return this.client;\n  }\n\n  /**\n   * Merge default context with per-call context.\n   */\n  private mergeContext(context?: FlagContext): FlagContext {\n    return {\n      ...this.defaultContext,\n      ...context,\n      attributes: {\n        ...this.defaultContext.attributes,\n        ...context?.attributes,\n      },\n    };\n  }\n\n  /**\n   * Get a reactive Observable for a feature flag.\n   * Automatically updates when the flag value changes.\n   *\n   * @param flagKey - The feature flag key to evaluate\n   * @param options - Configuration options\n   * @returns Observable of flag evaluation state\n   *\n   * @example\n   * ```typescript\n   * // In your component\n   * newFeature$ = this.savvagent.flag$('new-feature', {\n   *   defaultValue: false,\n   *   realtime: true,\n   *   context: { attributes: { plan: 'pro' } }\n   * });\n   *\n   * // In template\n   * <ng-container *ngIf=\"newFeature$ | async as flag\">\n   *   <app-loading *ngIf=\"flag.loading\"></app-loading>\n   *   <app-new-feature *ngIf=\"flag.value\"></app-new-feature>\n   *   <app-old-feature *ngIf=\"!flag.value && !flag.loading\"></app-old-feature>\n   * </ng-container>\n   * ```\n   */\n  flag$(flagKey: string, options: FlagOptions = {}): Observable<FlagObservableResult> {\n    const { context, defaultValue = false, realtime = true } = options;\n    const mergedContext = this.mergeContext(context);\n    const cacheKey = this.getCacheKey(flagKey, mergedContext);\n\n    // Check if we already have a subject for this flag+context\n    if (!this.flagSubjects.has(cacheKey)) {\n      const subject = new BehaviorSubject<FlagObservableResult>({\n        value: defaultValue,\n        loading: true,\n        error: null,\n        result: null,\n      });\n      this.flagSubjects.set(cacheKey, subject);\n\n      // Initial evaluation\n      this.evaluateAndEmit(flagKey, mergedContext, defaultValue, subject);\n\n      // Set up real-time subscription if enabled\n      if (realtime && this.client) {\n        const unsubscribe = this.client.subscribe(flagKey, () => {\n          this.evaluateAndEmit(flagKey, mergedContext, defaultValue, subject);\n        });\n\n        // Clean up subscription when subject is complete\n        subject.pipe(takeUntil(this.destroy$)).subscribe({\n          complete: () => unsubscribe(),\n        });\n      }\n    }\n\n    return this.flagSubjects.get(cacheKey)!.asObservable().pipe(\n      takeUntil(this.destroy$),\n      distinctUntilChanged((a, b) =>\n        a.value === b.value &&\n        a.loading === b.loading &&\n        a.error === b.error\n      )\n    );\n  }\n\n  /**\n   * Generate a cache key for a flag+context combination.\n   */\n  private getCacheKey(flagKey: string, context: FlagContext): string {\n    return `${flagKey}:${JSON.stringify(context)}`;\n  }\n\n  /**\n   * Evaluate a flag and emit the result to a subject.\n   */\n  private async evaluateAndEmit(\n    flagKey: string,\n    context: FlagContext,\n    defaultValue: boolean,\n    subject: BehaviorSubject<FlagObservableResult>\n  ): Promise<void> {\n    if (!this.client) {\n      subject.next({\n        value: defaultValue,\n        loading: false,\n        error: new Error('Savvagent client not initialized'),\n        result: null,\n      });\n      return;\n    }\n\n    try {\n      const result = await this.client.evaluate(flagKey, context);\n      subject.next({\n        value: result.value,\n        loading: false,\n        error: null,\n        result,\n      });\n    } catch (error) {\n      subject.next({\n        value: defaultValue,\n        loading: false,\n        error: error as Error,\n        result: null,\n      });\n    }\n  }\n\n  /**\n   * Re-evaluate all active flag subscriptions.\n   * Called when overrides change.\n   */\n  private reEvaluateAllFlags(): void {\n    this.flagSubjects.forEach((subject, cacheKey) => {\n      const [flagKey, contextJson] = cacheKey.split(':', 2);\n      const context = JSON.parse(contextJson || '{}');\n      const currentValue = subject.value;\n      this.evaluateAndEmit(flagKey, context, currentValue.value, subject);\n    });\n  }\n\n  /**\n   * Get a flag value as a simple Observable<boolean>.\n   * Useful when you only need the value without loading/error states.\n   *\n   * @param flagKey - The feature flag key to evaluate\n   * @param options - Configuration options\n   * @returns Observable of boolean flag value\n   *\n   * @example\n   * ```typescript\n   * isFeatureEnabled$ = this.savvagent.flagValue$('my-feature');\n   *\n   * // In template\n   * <button *ngIf=\"isFeatureEnabled$ | async\">New Button</button>\n   * ```\n   */\n  flagValue$(flagKey: string, options: FlagOptions = {}): Observable<boolean> {\n    return this.flag$(flagKey, options).pipe(\n      map((result) => result.value),\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n   * Evaluate a feature flag once (non-reactive).\n   * For reactive updates, use flag$() instead.\n   *\n   * @param flagKey - The feature flag key to evaluate\n   * @param context - Optional context for targeting\n   * @returns Promise with detailed evaluation result\n   *\n   * @example\n   * ```typescript\n   * async checkFeature() {\n   *   const result = await this.savvagent.evaluate('new-feature');\n   *   if (result.value) {\n   *     // Feature is enabled\n   *   }\n   * }\n   * ```\n   */\n  async evaluate(flagKey: string, context?: FlagContext): Promise<FlagEvaluationResult> {\n    if (!this.client) {\n      throw new Error('Savvagent client not initialized');\n    }\n    return this.client.evaluate(flagKey, this.mergeContext(context));\n  }\n\n  /**\n   * Check if a feature flag is enabled (non-reactive).\n   *\n   * @param flagKey - The feature flag key to evaluate\n   * @param context - Optional context for targeting\n   * @returns Promise<boolean>\n   */\n  async isEnabled(flagKey: string, context?: FlagContext): Promise<boolean> {\n    if (!this.client) {\n      return false;\n    }\n    return this.client.isEnabled(flagKey, this.mergeContext(context));\n  }\n\n  /**\n   * Execute code conditionally based on flag value.\n   *\n   * @param flagKey - The flag key to check\n   * @param callback - Function to execute if flag is enabled\n   * @param context - Optional context for targeting\n   *\n   * @example\n   * ```typescript\n   * await this.savvagent.withFlag('analytics-enabled', async () => {\n   *   await this.analytics.track('page_view');\n   * });\n   * ```\n   */\n  async withFlag<T>(\n    flagKey: string,\n    callback: () => T | Promise<T>,\n    context?: FlagContext\n  ): Promise<T | null> {\n    if (!this.client) {\n      return null;\n    }\n    return this.client.withFlag(flagKey, callback, this.mergeContext(context));\n  }\n\n  /**\n   * Track an error with flag context.\n   *\n   * @param flagKey - The flag key associated with the error\n   * @param error - The error that occurred\n   * @param context - Optional context\n   */\n  trackError(flagKey: string, error: Error, context?: FlagContext): void {\n    this.client?.trackError(flagKey, error, this.mergeContext(context));\n  }\n\n  /**\n   * Set the user ID for logged-in users.\n   *\n   * @param userId - The user ID (or null to clear)\n   */\n  setUserId(userId: string | null): void {\n    this.client?.setUserId(userId);\n  }\n\n  /**\n   * Get the current user ID.\n   */\n  getUserId(): string | null {\n    return this.client?.getUserId() || null;\n  }\n\n  /**\n   * Get the current anonymous ID.\n   */\n  getAnonymousId(): string | null {\n    return this.client?.getAnonymousId() || null;\n  }\n\n  /**\n   * Set a custom anonymous ID.\n   */\n  setAnonymousId(id: string): void {\n    this.client?.setAnonymousId(id);\n  }\n\n  // =====================\n  // Local Override Methods\n  // =====================\n\n  /**\n   * Set a local override for a flag.\n   * Overrides take precedence over server values.\n   *\n   * @param flagKey - The flag key to override\n   * @param value - The override value\n   */\n  setOverride(flagKey: string, value: boolean): void {\n    this.client?.setOverride(flagKey, value);\n  }\n\n  /**\n   * Clear a local override for a flag.\n   */\n  clearOverride(flagKey: string): void {\n    this.client?.clearOverride(flagKey);\n  }\n\n  /**\n   * Clear all local overrides.\n   */\n  clearAllOverrides(): void {\n    this.client?.clearAllOverrides();\n  }\n\n  /**\n   * Check if a flag has a local override.\n   */\n  hasOverride(flagKey: string): boolean {\n    return this.client?.hasOverride(flagKey) || false;\n  }\n\n  /**\n   * Get the override value for a flag.\n   */\n  getOverride(flagKey: string): boolean | undefined {\n    return this.client?.getOverride(flagKey);\n  }\n\n  /**\n   * Get all current overrides.\n   */\n  getOverrides(): Record<string, boolean> {\n    return this.client?.getOverrides() || {};\n  }\n\n  /**\n   * Set multiple overrides at once.\n   */\n  setOverrides(overrides: Record<string, boolean>): void {\n    this.client?.setOverrides(overrides);\n  }\n\n  // =====================\n  // Flag Discovery Methods\n  // =====================\n\n  /**\n   * Get all flags for the application.\n   *\n   * @param environment - Environment to evaluate (default: 'development')\n   * @returns Observable of flag definitions\n   */\n  getAllFlags$(environment: string = 'development'): Observable<FlagDefinition[]> {\n    if (!this.client) {\n      return of([]);\n    }\n    return from(this.client.getAllFlags(environment)).pipe(\n      catchError((error) => {\n        console.error('[Savvagent] Failed to fetch all flags:', error);\n        return of([]);\n      })\n    );\n  }\n\n  /**\n   * Get all flags for the application (Promise-based).\n   */\n  async getAllFlags(environment: string = 'development'): Promise<FlagDefinition[]> {\n    if (!this.client) {\n      return [];\n    }\n    return this.client.getAllFlags(environment);\n  }\n\n  /**\n   * Get enterprise-scoped flags only.\n   */\n  async getEnterpriseFlags(environment: string = 'development'): Promise<FlagDefinition[]> {\n    if (!this.client) {\n      return [];\n    }\n    return this.client.getEnterpriseFlags(environment);\n  }\n\n  // =====================\n  // Cache & Connection\n  // =====================\n\n  /**\n   * Clear the flag cache.\n   */\n  clearCache(): void {\n    this.client?.clearCache();\n  }\n\n  /**\n   * Check if real-time connection is active.\n   */\n  isRealtimeConnected(): boolean {\n    return this.client?.isRealtimeConnected() || false;\n  }\n\n  /**\n   * Close the client and cleanup resources.\n   */\n  close(): void {\n    this.client?.close();\n    this.client = null;\n    this.isReady$.next(false);\n    this.flagSubjects.forEach((subject) => subject.complete());\n    this.flagSubjects.clear();\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.close();\n  }\n}\n"]}
|