@metigan/angular 1.0.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.
Files changed (48) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +581 -0
  3. package/dist/LICENSE +22 -0
  4. package/dist/README.md +581 -0
  5. package/dist/esm2022/metigan-angular.mjs +5 -0
  6. package/dist/esm2022/public-api.mjs +21 -0
  7. package/dist/esm2022/src/lib/audiences.service.mjs +157 -0
  8. package/dist/esm2022/src/lib/config.mjs +30 -0
  9. package/dist/esm2022/src/lib/contacts.service.mjs +267 -0
  10. package/dist/esm2022/src/lib/email.service.mjs +267 -0
  11. package/dist/esm2022/src/lib/errors.mjs +40 -0
  12. package/dist/esm2022/src/lib/forms.service.mjs +180 -0
  13. package/dist/esm2022/src/lib/http-client.service.mjs +111 -0
  14. package/dist/esm2022/src/lib/metigan.module.mjs +67 -0
  15. package/dist/esm2022/src/lib/metigan.service.mjs +72 -0
  16. package/dist/esm2022/src/lib/templates.service.mjs +85 -0
  17. package/dist/esm2022/src/lib/types.mjs +6 -0
  18. package/dist/fesm2022/metigan-angular.mjs +1241 -0
  19. package/dist/fesm2022/metigan-angular.mjs.map +1 -0
  20. package/dist/index.d.ts +5 -0
  21. package/dist/public-api.d.ts +15 -0
  22. package/dist/src/lib/audiences.service.d.ts +62 -0
  23. package/dist/src/lib/config.d.ts +28 -0
  24. package/dist/src/lib/contacts.service.d.ts +80 -0
  25. package/dist/src/lib/email.service.d.ts +44 -0
  26. package/dist/src/lib/errors.d.ts +24 -0
  27. package/dist/src/lib/forms.service.d.ts +67 -0
  28. package/dist/src/lib/http-client.service.d.ts +46 -0
  29. package/dist/src/lib/metigan.module.d.ts +27 -0
  30. package/dist/src/lib/metigan.service.d.ts +27 -0
  31. package/dist/src/lib/templates.service.d.ts +36 -0
  32. package/dist/src/lib/types.d.ts +329 -0
  33. package/examples/basic.component.ts +113 -0
  34. package/ng-package.json +8 -0
  35. package/package.json +68 -0
  36. package/public-api.ts +26 -0
  37. package/src/lib/audiences.service.ts +230 -0
  38. package/src/lib/config.ts +35 -0
  39. package/src/lib/contacts.service.ts +377 -0
  40. package/src/lib/email.service.ts +286 -0
  41. package/src/lib/errors.ts +45 -0
  42. package/src/lib/forms.service.ts +263 -0
  43. package/src/lib/http-client.service.ts +156 -0
  44. package/src/lib/metigan.module.ts +55 -0
  45. package/src/lib/metigan.service.ts +80 -0
  46. package/src/lib/templates.service.ts +103 -0
  47. package/src/lib/types.ts +398 -0
  48. package/tsconfig.json +38 -0
@@ -0,0 +1,157 @@
1
+ /**
2
+ * Metigan Audiences Service
3
+ * Service for audience/list management
4
+ */
5
+ import { Injectable } from '@angular/core';
6
+ import { throwError } from 'rxjs';
7
+ import { HttpParams } from '@angular/common/http';
8
+ import { API_URL, DEFAULT_TIMEOUT, DEFAULT_RETRY_COUNT, DEFAULT_RETRY_DELAY } from './config';
9
+ import { ValidationError, MetiganError } from './errors';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "./http-client.service";
12
+ export class MetiganAudiencesService {
13
+ http;
14
+ apiKey = '';
15
+ apiUrl = API_URL;
16
+ timeout = DEFAULT_TIMEOUT;
17
+ retryCount = DEFAULT_RETRY_COUNT;
18
+ retryDelay = DEFAULT_RETRY_DELAY;
19
+ constructor(http) {
20
+ this.http = http;
21
+ }
22
+ /**
23
+ * Initialize the service with API key and options
24
+ */
25
+ initialize(apiKey, options) {
26
+ if (!apiKey) {
27
+ throw new MetiganError('API key is required');
28
+ }
29
+ this.apiKey = apiKey;
30
+ this.apiUrl = options?.apiUrl || API_URL;
31
+ this.timeout = options?.timeout || DEFAULT_TIMEOUT;
32
+ this.retryCount = options?.retryCount || DEFAULT_RETRY_COUNT;
33
+ this.retryDelay = options?.retryDelay || DEFAULT_RETRY_DELAY;
34
+ }
35
+ /**
36
+ * Get default headers
37
+ */
38
+ getHeaders() {
39
+ return {
40
+ 'Content-Type': 'application/json',
41
+ 'x-api-key': this.apiKey,
42
+ 'User-Agent': 'AngularSDK/1.0'
43
+ };
44
+ }
45
+ /**
46
+ * Create a new audience
47
+ */
48
+ create(options) {
49
+ if (!this.apiKey) {
50
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
51
+ }
52
+ if (!options.name) {
53
+ return throwError(() => new ValidationError('Audience name is required'));
54
+ }
55
+ return this.http.post(`${this.apiUrl}/api/audiences`, options, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
56
+ }
57
+ /**
58
+ * Get audience by ID
59
+ */
60
+ get(audienceId) {
61
+ if (!this.apiKey) {
62
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
63
+ }
64
+ if (!audienceId) {
65
+ return throwError(() => new ValidationError('Audience ID is required'));
66
+ }
67
+ return this.http.get(`${this.apiUrl}/api/audiences/${audienceId}`, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
68
+ }
69
+ /**
70
+ * Update an audience
71
+ */
72
+ update(audienceId, options) {
73
+ if (!this.apiKey) {
74
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
75
+ }
76
+ if (!audienceId) {
77
+ return throwError(() => new ValidationError('Audience ID is required'));
78
+ }
79
+ return this.http.put(`${this.apiUrl}/api/audiences/${audienceId}`, options, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
80
+ }
81
+ /**
82
+ * Delete an audience
83
+ */
84
+ delete(audienceId) {
85
+ if (!this.apiKey) {
86
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
87
+ }
88
+ if (!audienceId) {
89
+ return throwError(() => new ValidationError('Audience ID is required'));
90
+ }
91
+ return this.http.delete(`${this.apiUrl}/api/audiences/${audienceId}`, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
92
+ }
93
+ /**
94
+ * List all audiences
95
+ */
96
+ list(options) {
97
+ if (!this.apiKey) {
98
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
99
+ }
100
+ let params = new HttpParams();
101
+ if (options?.page) {
102
+ params = params.set('page', options.page.toString());
103
+ }
104
+ if (options?.limit) {
105
+ params = params.set('limit', options.limit.toString());
106
+ }
107
+ return this.http.get(`${this.apiUrl}/api/audiences`, {
108
+ headers: this.getHeaders(),
109
+ params: params
110
+ }, this.retryCount, this.retryDelay);
111
+ }
112
+ /**
113
+ * Get audience statistics
114
+ */
115
+ getStats(audienceId) {
116
+ if (!this.apiKey) {
117
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
118
+ }
119
+ if (!audienceId) {
120
+ return throwError(() => new ValidationError('Audience ID is required'));
121
+ }
122
+ return this.http.get(`${this.apiUrl}/api/audiences/${audienceId}/stats`, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
123
+ }
124
+ /**
125
+ * Clean audience (remove bounced/unsubscribed contacts)
126
+ */
127
+ clean(audienceId) {
128
+ if (!this.apiKey) {
129
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
130
+ }
131
+ if (!audienceId) {
132
+ return throwError(() => new ValidationError('Audience ID is required'));
133
+ }
134
+ return this.http.post(`${this.apiUrl}/api/audiences/${audienceId}/clean`, {}, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
135
+ }
136
+ /**
137
+ * Merge audiences (source into target)
138
+ */
139
+ merge(sourceAudienceId, targetAudienceId) {
140
+ if (!this.apiKey) {
141
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
142
+ }
143
+ if (!sourceAudienceId || !targetAudienceId) {
144
+ return throwError(() => new ValidationError('Source and target audience IDs are required'));
145
+ }
146
+ return this.http.post(`${this.apiUrl}/api/audiences/${sourceAudienceId}/merge`, { targetAudienceId }, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
147
+ }
148
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganAudiencesService, deps: [{ token: i1.MetiganHttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
149
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganAudiencesService, providedIn: 'root' });
150
+ }
151
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganAudiencesService, decorators: [{
152
+ type: Injectable,
153
+ args: [{
154
+ providedIn: 'root'
155
+ }]
156
+ }], ctorParameters: () => [{ type: i1.MetiganHttpClient }] });
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audiences.service.js","sourceRoot":"","sources":["../../../../src/lib/audiences.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Metigan Audiences Service\r\n * Service for audience/list management\r\n */\r\n\r\nimport { Injectable } from '@angular/core';\r\nimport { Observable, throwError } from 'rxjs';\r\nimport { HttpParams } from '@angular/common/http';\r\nimport { MetiganHttpClient } from './http-client.service';\r\nimport { API_URL, DEFAULT_TIMEOUT, DEFAULT_RETRY_COUNT, DEFAULT_RETRY_DELAY } from './config';\r\nimport { ValidationError, MetiganError } from './errors';\r\nimport {\r\n  Audience,\r\n  CreateAudienceOptions,\r\n  UpdateAudienceOptions,\r\n  AudienceListResponse,\r\n  AudienceStats,\r\n  PaginationOptions\r\n} from './types';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class MetiganAudiencesService {\r\n  private apiKey: string = '';\r\n  private apiUrl: string = API_URL;\r\n  private timeout: number = DEFAULT_TIMEOUT;\r\n  private retryCount: number = DEFAULT_RETRY_COUNT;\r\n  private retryDelay: number = DEFAULT_RETRY_DELAY;\r\n\r\n  constructor(private http: MetiganHttpClient) {}\r\n\r\n  /**\r\n   * Initialize the service with API key and options\r\n   */\r\n  initialize(apiKey: string, options?: { apiUrl?: string; timeout?: number; retryCount?: number; retryDelay?: number }): void {\r\n    if (!apiKey) {\r\n      throw new MetiganError('API key is required');\r\n    }\r\n\r\n    this.apiKey = apiKey;\r\n    this.apiUrl = options?.apiUrl || API_URL;\r\n    this.timeout = options?.timeout || DEFAULT_TIMEOUT;\r\n    this.retryCount = options?.retryCount || DEFAULT_RETRY_COUNT;\r\n    this.retryDelay = options?.retryDelay || DEFAULT_RETRY_DELAY;\r\n  }\r\n\r\n  /**\r\n   * Get default headers\r\n   */\r\n  private getHeaders() {\r\n    return {\r\n      'Content-Type': 'application/json',\r\n      'x-api-key': this.apiKey,\r\n      'User-Agent': 'AngularSDK/1.0'\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Create a new audience\r\n   */\r\n  create(options: CreateAudienceOptions): Observable<Audience> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!options.name) {\r\n      return throwError(() => new ValidationError('Audience name is required'));\r\n    }\r\n\r\n    return this.http.post<Audience>(\r\n      `${this.apiUrl}/api/audiences`,\r\n      options,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get audience by ID\r\n   */\r\n  get(audienceId: string): Observable<Audience> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.get<Audience>(\r\n      `${this.apiUrl}/api/audiences/${audienceId}`,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Update an audience\r\n   */\r\n  update(audienceId: string, options: UpdateAudienceOptions): Observable<Audience> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.put<Audience>(\r\n      `${this.apiUrl}/api/audiences/${audienceId}`,\r\n      options,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete an audience\r\n   */\r\n  delete(audienceId: string): Observable<void> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.delete<void>(\r\n      `${this.apiUrl}/api/audiences/${audienceId}`,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * List all audiences\r\n   */\r\n  list(options?: PaginationOptions): Observable<AudienceListResponse> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    let params = new HttpParams();\r\n    if (options?.page) {\r\n      params = params.set('page', options.page.toString());\r\n    }\r\n    if (options?.limit) {\r\n      params = params.set('limit', options.limit.toString());\r\n    }\r\n\r\n    return this.http.get<AudienceListResponse>(\r\n      `${this.apiUrl}/api/audiences`,\r\n      {\r\n        headers: this.getHeaders(),\r\n        params: params\r\n      },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get audience statistics\r\n   */\r\n  getStats(audienceId: string): Observable<AudienceStats> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.get<AudienceStats>(\r\n      `${this.apiUrl}/api/audiences/${audienceId}/stats`,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Clean audience (remove bounced/unsubscribed contacts)\r\n   */\r\n  clean(audienceId: string): Observable<{ removed: number }> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.post<{ removed: number }>(\r\n      `${this.apiUrl}/api/audiences/${audienceId}/clean`,\r\n      {},\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Merge audiences (source into target)\r\n   */\r\n  merge(sourceAudienceId: string, targetAudienceId: string): Observable<Audience> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!sourceAudienceId || !targetAudienceId) {\r\n      return throwError(() => new ValidationError('Source and target audience IDs are required'));\r\n    }\r\n\r\n    return this.http.post<Audience>(\r\n      `${this.apiUrl}/api/audiences/${sourceAudienceId}/merge`,\r\n      { targetAudienceId },\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n}\r\n\r\n"]}
@@ -0,0 +1,30 @@
1
+ /**
2
+ * Metigan Library Configuration
3
+ * Central configuration file for all modules
4
+ */
5
+ /**
6
+ * Default API URL for Metigan services
7
+ * All API calls are routed through this single endpoint
8
+ *
9
+ * Can be overridden with METIGAN_API_URL environment variable for testing
10
+ */
11
+ export const API_URL = (typeof process !== 'undefined' && process.env?.['METIGAN_API_URL'])
12
+ || 'https://api.metigan.com';
13
+ /**
14
+ * SDK Version
15
+ */
16
+ export const SDK_VERSION = '1.0.0';
17
+ /**
18
+ * Default timeout for API requests (in milliseconds)
19
+ */
20
+ export const DEFAULT_TIMEOUT = 30000;
21
+ /**
22
+ * Default retry configuration
23
+ */
24
+ export const DEFAULT_RETRY_COUNT = 3;
25
+ export const DEFAULT_RETRY_DELAY = 1000;
26
+ /**
27
+ * Maximum file size for attachments (7MB)
28
+ */
29
+ export const MAX_FILE_SIZE = 7 * 1024 * 1024;
30
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUg7Ozs7O0dBS0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsQ0FBQyxPQUFPLE9BQU8sS0FBSyxXQUFXLElBQUksT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDLGlCQUFpQixDQUFDLENBQUM7T0FDdEYseUJBQXlCLENBQUM7QUFFL0I7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDO0FBRW5DOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sZUFBZSxHQUFHLEtBQUssQ0FBQztBQUVyQzs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUFHLENBQUMsQ0FBQztBQUNyQyxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLENBQUM7QUFFeEM7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUcsQ0FBQyxHQUFHLElBQUksR0FBRyxJQUFJLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogTWV0aWdhbiBMaWJyYXJ5IENvbmZpZ3VyYXRpb25cclxuICogQ2VudHJhbCBjb25maWd1cmF0aW9uIGZpbGUgZm9yIGFsbCBtb2R1bGVzXHJcbiAqL1xyXG5cclxuLyoqXHJcbiAqIERlZmF1bHQgQVBJIFVSTCBmb3IgTWV0aWdhbiBzZXJ2aWNlc1xyXG4gKiBBbGwgQVBJIGNhbGxzIGFyZSByb3V0ZWQgdGhyb3VnaCB0aGlzIHNpbmdsZSBlbmRwb2ludFxyXG4gKiBcclxuICogQ2FuIGJlIG92ZXJyaWRkZW4gd2l0aCBNRVRJR0FOX0FQSV9VUkwgZW52aXJvbm1lbnQgdmFyaWFibGUgZm9yIHRlc3RpbmdcclxuICovXHJcbmV4cG9ydCBjb25zdCBBUElfVVJMID0gKHR5cGVvZiBwcm9jZXNzICE9PSAndW5kZWZpbmVkJyAmJiBwcm9jZXNzLmVudj8uWydNRVRJR0FOX0FQSV9VUkwnXSkgXHJcbiAgfHwgJ2h0dHBzOi8vYXBpLm1ldGlnYW4uY29tJztcclxuXHJcbi8qKlxyXG4gKiBTREsgVmVyc2lvblxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IFNES19WRVJTSU9OID0gJzEuMC4wJztcclxuXHJcbi8qKlxyXG4gKiBEZWZhdWx0IHRpbWVvdXQgZm9yIEFQSSByZXF1ZXN0cyAoaW4gbWlsbGlzZWNvbmRzKVxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IERFRkFVTFRfVElNRU9VVCA9IDMwMDAwO1xyXG5cclxuLyoqXHJcbiAqIERlZmF1bHQgcmV0cnkgY29uZmlndXJhdGlvblxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IERFRkFVTFRfUkVUUllfQ09VTlQgPSAzO1xyXG5leHBvcnQgY29uc3QgREVGQVVMVF9SRVRSWV9ERUxBWSA9IDEwMDA7XHJcblxyXG4vKipcclxuICogTWF4aW11bSBmaWxlIHNpemUgZm9yIGF0dGFjaG1lbnRzICg3TUIpXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgTUFYX0ZJTEVfU0laRSA9IDcgKiAxMDI0ICogMTAyNDtcclxuXHJcbiJdfQ==
@@ -0,0 +1,267 @@
1
+ /**
2
+ * Metigan Contacts Service
3
+ * Service for contact/subscriber management
4
+ */
5
+ import { Injectable } from '@angular/core';
6
+ import { throwError } from 'rxjs';
7
+ import { HttpParams } from '@angular/common/http';
8
+ import { API_URL, DEFAULT_TIMEOUT, DEFAULT_RETRY_COUNT, DEFAULT_RETRY_DELAY } from './config';
9
+ import { ValidationError, MetiganError } from './errors';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "./http-client.service";
12
+ export class MetiganContactsService {
13
+ http;
14
+ apiKey = '';
15
+ apiUrl = API_URL;
16
+ timeout = DEFAULT_TIMEOUT;
17
+ retryCount = DEFAULT_RETRY_COUNT;
18
+ retryDelay = DEFAULT_RETRY_DELAY;
19
+ constructor(http) {
20
+ this.http = http;
21
+ }
22
+ /**
23
+ * Initialize the service with API key and options
24
+ */
25
+ initialize(apiKey, options) {
26
+ if (!apiKey) {
27
+ throw new MetiganError('API key is required');
28
+ }
29
+ this.apiKey = apiKey;
30
+ this.apiUrl = options?.apiUrl || API_URL;
31
+ this.timeout = options?.timeout || DEFAULT_TIMEOUT;
32
+ this.retryCount = options?.retryCount || DEFAULT_RETRY_COUNT;
33
+ this.retryDelay = options?.retryDelay || DEFAULT_RETRY_DELAY;
34
+ }
35
+ /**
36
+ * Get default headers
37
+ */
38
+ getHeaders() {
39
+ return {
40
+ 'Content-Type': 'application/json',
41
+ 'x-api-key': this.apiKey,
42
+ 'User-Agent': 'AngularSDK/1.0'
43
+ };
44
+ }
45
+ /**
46
+ * Validate email format
47
+ */
48
+ validateEmail(email) {
49
+ if (!email || typeof email !== 'string')
50
+ return false;
51
+ const parts = email.split('@');
52
+ if (parts.length !== 2)
53
+ return false;
54
+ if (parts[0].length === 0)
55
+ return false;
56
+ const domainParts = parts[1].split('.');
57
+ if (domainParts.length < 2)
58
+ return false;
59
+ if (domainParts.some(part => part.length === 0))
60
+ return false;
61
+ return true;
62
+ }
63
+ /**
64
+ * Create a new contact
65
+ */
66
+ create(options) {
67
+ if (!this.apiKey) {
68
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
69
+ }
70
+ if (!options.email || !this.validateEmail(options.email)) {
71
+ return throwError(() => new ValidationError('Valid email is required'));
72
+ }
73
+ if (!options.audienceId) {
74
+ return throwError(() => new ValidationError('Audience ID is required'));
75
+ }
76
+ return this.http.post(`${this.apiUrl}/api/contacts`, options, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
77
+ }
78
+ /**
79
+ * Get contact by ID
80
+ */
81
+ get(contactId) {
82
+ if (!this.apiKey) {
83
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
84
+ }
85
+ if (!contactId) {
86
+ return throwError(() => new ValidationError('Contact ID is required'));
87
+ }
88
+ return this.http.get(`${this.apiUrl}/api/contacts/${contactId}`, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
89
+ }
90
+ /**
91
+ * Get contact by email
92
+ */
93
+ getByEmail(email, audienceId) {
94
+ if (!this.apiKey) {
95
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
96
+ }
97
+ if (!email || !this.validateEmail(email)) {
98
+ return throwError(() => new ValidationError('Valid email is required'));
99
+ }
100
+ if (!audienceId) {
101
+ return throwError(() => new ValidationError('Audience ID is required'));
102
+ }
103
+ let params = new HttpParams();
104
+ params = params.set('email', email);
105
+ params = params.set('audienceId', audienceId);
106
+ return this.http.get(`${this.apiUrl}/api/contacts`, {
107
+ headers: this.getHeaders(),
108
+ params: params
109
+ }, this.retryCount, this.retryDelay);
110
+ }
111
+ /**
112
+ * Update a contact
113
+ */
114
+ update(contactId, options) {
115
+ if (!this.apiKey) {
116
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
117
+ }
118
+ if (!contactId) {
119
+ return throwError(() => new ValidationError('Contact ID is required'));
120
+ }
121
+ return this.http.put(`${this.apiUrl}/api/contacts/${contactId}`, options, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
122
+ }
123
+ /**
124
+ * Delete a contact
125
+ */
126
+ delete(contactId) {
127
+ if (!this.apiKey) {
128
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
129
+ }
130
+ if (!contactId) {
131
+ return throwError(() => new ValidationError('Contact ID is required'));
132
+ }
133
+ return this.http.delete(`${this.apiUrl}/api/contacts/${contactId}`, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
134
+ }
135
+ /**
136
+ * List contacts with filters
137
+ */
138
+ list(filters) {
139
+ if (!this.apiKey) {
140
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
141
+ }
142
+ let params = new HttpParams();
143
+ if (filters?.audienceId) {
144
+ params = params.set('audienceId', filters.audienceId);
145
+ }
146
+ if (filters?.status) {
147
+ params = params.set('status', filters.status);
148
+ }
149
+ if (filters?.tag) {
150
+ params = params.set('tag', filters.tag);
151
+ }
152
+ if (filters?.search) {
153
+ params = params.set('search', filters.search);
154
+ }
155
+ if (filters?.page) {
156
+ params = params.set('page', filters.page.toString());
157
+ }
158
+ if (filters?.limit) {
159
+ params = params.set('limit', filters.limit.toString());
160
+ }
161
+ return this.http.get(`${this.apiUrl}/api/contacts`, {
162
+ headers: this.getHeaders(),
163
+ params: params
164
+ }, this.retryCount, this.retryDelay);
165
+ }
166
+ /**
167
+ * Search contacts
168
+ */
169
+ search(query, audienceId) {
170
+ if (!this.apiKey) {
171
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
172
+ }
173
+ if (!query) {
174
+ return throwError(() => new ValidationError('Search query is required'));
175
+ }
176
+ let params = new HttpParams();
177
+ params = params.set('search', query);
178
+ if (audienceId) {
179
+ params = params.set('audienceId', audienceId);
180
+ }
181
+ return this.http.get(`${this.apiUrl}/api/contacts/search`, {
182
+ headers: this.getHeaders(),
183
+ params: params
184
+ }, this.retryCount, this.retryDelay);
185
+ }
186
+ /**
187
+ * Bulk import contacts
188
+ */
189
+ bulkImport(contacts, audienceId) {
190
+ if (!this.apiKey) {
191
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
192
+ }
193
+ if (!contacts || contacts.length === 0) {
194
+ return throwError(() => new ValidationError('Contacts array is required'));
195
+ }
196
+ if (!audienceId) {
197
+ return throwError(() => new ValidationError('Audience ID is required'));
198
+ }
199
+ return this.http.post(`${this.apiUrl}/api/contacts/bulk`, { contacts, audienceId }, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
200
+ }
201
+ /**
202
+ * Subscribe a contact
203
+ */
204
+ subscribe(contactId) {
205
+ if (!this.apiKey) {
206
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
207
+ }
208
+ if (!contactId) {
209
+ return throwError(() => new ValidationError('Contact ID is required'));
210
+ }
211
+ return this.http.post(`${this.apiUrl}/api/contacts/${contactId}/subscribe`, {}, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
212
+ }
213
+ /**
214
+ * Unsubscribe a contact
215
+ */
216
+ unsubscribe(contactId) {
217
+ if (!this.apiKey) {
218
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
219
+ }
220
+ if (!contactId) {
221
+ return throwError(() => new ValidationError('Contact ID is required'));
222
+ }
223
+ return this.http.post(`${this.apiUrl}/api/contacts/${contactId}/unsubscribe`, {}, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
224
+ }
225
+ /**
226
+ * Add tags to a contact
227
+ */
228
+ addTags(contactId, tags) {
229
+ if (!this.apiKey) {
230
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
231
+ }
232
+ if (!contactId) {
233
+ return throwError(() => new ValidationError('Contact ID is required'));
234
+ }
235
+ if (!tags || tags.length === 0) {
236
+ return throwError(() => new ValidationError('Tags array is required'));
237
+ }
238
+ return this.http.post(`${this.apiUrl}/api/contacts/${contactId}/tags`, { tags }, { headers: this.getHeaders() }, this.retryCount, this.retryDelay);
239
+ }
240
+ /**
241
+ * Remove tags from a contact
242
+ */
243
+ removeTags(contactId, tags) {
244
+ if (!this.apiKey) {
245
+ return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));
246
+ }
247
+ if (!contactId) {
248
+ return throwError(() => new ValidationError('Contact ID is required'));
249
+ }
250
+ if (!tags || tags.length === 0) {
251
+ return throwError(() => new ValidationError('Tags array is required'));
252
+ }
253
+ return this.http.delete(`${this.apiUrl}/api/contacts/${contactId}/tags`, {
254
+ headers: this.getHeaders(),
255
+ params: { tags: tags.join(',') }
256
+ }, this.retryCount, this.retryDelay);
257
+ }
258
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganContactsService, deps: [{ token: i1.MetiganHttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
259
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganContactsService, providedIn: 'root' });
260
+ }
261
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MetiganContactsService, decorators: [{
262
+ type: Injectable,
263
+ args: [{
264
+ providedIn: 'root'
265
+ }]
266
+ }], ctorParameters: () => [{ type: i1.MetiganHttpClient }] });
267
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contacts.service.js","sourceRoot":"","sources":["../../../../src/lib/contacts.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Metigan Contacts Service\r\n * Service for contact/subscriber management\r\n */\r\n\r\nimport { Injectable } from '@angular/core';\r\nimport { Observable, throwError } from 'rxjs';\r\nimport { HttpParams } from '@angular/common/http';\r\nimport { MetiganHttpClient } from './http-client.service';\r\nimport { API_URL, DEFAULT_TIMEOUT, DEFAULT_RETRY_COUNT, DEFAULT_RETRY_DELAY } from './config';\r\nimport { ValidationError, MetiganError } from './errors';\r\nimport {\r\n  Contact,\r\n  CreateContactOptions,\r\n  UpdateContactOptions,\r\n  ContactListFilters,\r\n  ContactListResponse,\r\n  BulkContactResult\r\n} from './types';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class MetiganContactsService {\r\n  private apiKey: string = '';\r\n  private apiUrl: string = API_URL;\r\n  private timeout: number = DEFAULT_TIMEOUT;\r\n  private retryCount: number = DEFAULT_RETRY_COUNT;\r\n  private retryDelay: number = DEFAULT_RETRY_DELAY;\r\n\r\n  constructor(private http: MetiganHttpClient) {}\r\n\r\n  /**\r\n   * Initialize the service with API key and options\r\n   */\r\n  initialize(apiKey: string, options?: { apiUrl?: string; timeout?: number; retryCount?: number; retryDelay?: number }): void {\r\n    if (!apiKey) {\r\n      throw new MetiganError('API key is required');\r\n    }\r\n\r\n    this.apiKey = apiKey;\r\n    this.apiUrl = options?.apiUrl || API_URL;\r\n    this.timeout = options?.timeout || DEFAULT_TIMEOUT;\r\n    this.retryCount = options?.retryCount || DEFAULT_RETRY_COUNT;\r\n    this.retryDelay = options?.retryDelay || DEFAULT_RETRY_DELAY;\r\n  }\r\n\r\n  /**\r\n   * Get default headers\r\n   */\r\n  private getHeaders() {\r\n    return {\r\n      'Content-Type': 'application/json',\r\n      'x-api-key': this.apiKey,\r\n      'User-Agent': 'AngularSDK/1.0'\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Validate email format\r\n   */\r\n  private validateEmail(email: string): boolean {\r\n    if (!email || typeof email !== 'string') return false;\r\n    const parts = email.split('@');\r\n    if (parts.length !== 2) return false;\r\n    if (parts[0].length === 0) return false;\r\n    const domainParts = parts[1].split('.');\r\n    if (domainParts.length < 2) return false;\r\n    if (domainParts.some(part => part.length === 0)) return false;\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Create a new contact\r\n   */\r\n  create(options: CreateContactOptions): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!options.email || !this.validateEmail(options.email)) {\r\n      return throwError(() => new ValidationError('Valid email is required'));\r\n    }\r\n\r\n    if (!options.audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.post<Contact>(\r\n      `${this.apiUrl}/api/contacts`,\r\n      options,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get contact by ID\r\n   */\r\n  get(contactId: string): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    return this.http.get<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}`,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get contact by email\r\n   */\r\n  getByEmail(email: string, audienceId: string): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!email || !this.validateEmail(email)) {\r\n      return throwError(() => new ValidationError('Valid email is required'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    let params = new HttpParams();\r\n    params = params.set('email', email);\r\n    params = params.set('audienceId', audienceId);\r\n\r\n    return this.http.get<Contact>(\r\n      `${this.apiUrl}/api/contacts`,\r\n      {\r\n        headers: this.getHeaders(),\r\n        params: params\r\n      },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Update a contact\r\n   */\r\n  update(contactId: string, options: UpdateContactOptions): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    return this.http.put<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}`,\r\n      options,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete a contact\r\n   */\r\n  delete(contactId: string): Observable<void> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    return this.http.delete<void>(\r\n      `${this.apiUrl}/api/contacts/${contactId}`,\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * List contacts with filters\r\n   */\r\n  list(filters?: ContactListFilters): Observable<ContactListResponse> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    let params = new HttpParams();\r\n    if (filters?.audienceId) {\r\n      params = params.set('audienceId', filters.audienceId);\r\n    }\r\n    if (filters?.status) {\r\n      params = params.set('status', filters.status);\r\n    }\r\n    if (filters?.tag) {\r\n      params = params.set('tag', filters.tag);\r\n    }\r\n    if (filters?.search) {\r\n      params = params.set('search', filters.search);\r\n    }\r\n    if (filters?.page) {\r\n      params = params.set('page', filters.page.toString());\r\n    }\r\n    if (filters?.limit) {\r\n      params = params.set('limit', filters.limit.toString());\r\n    }\r\n\r\n    return this.http.get<ContactListResponse>(\r\n      `${this.apiUrl}/api/contacts`,\r\n      {\r\n        headers: this.getHeaders(),\r\n        params: params\r\n      },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Search contacts\r\n   */\r\n  search(query: string, audienceId?: string): Observable<Contact[]> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!query) {\r\n      return throwError(() => new ValidationError('Search query is required'));\r\n    }\r\n\r\n    let params = new HttpParams();\r\n    params = params.set('search', query);\r\n    if (audienceId) {\r\n      params = params.set('audienceId', audienceId);\r\n    }\r\n\r\n    return this.http.get<Contact[]>(\r\n      `${this.apiUrl}/api/contacts/search`,\r\n      {\r\n        headers: this.getHeaders(),\r\n        params: params\r\n      },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Bulk import contacts\r\n   */\r\n  bulkImport(contacts: CreateContactOptions[], audienceId: string): Observable<BulkContactResult> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contacts || contacts.length === 0) {\r\n      return throwError(() => new ValidationError('Contacts array is required'));\r\n    }\r\n\r\n    if (!audienceId) {\r\n      return throwError(() => new ValidationError('Audience ID is required'));\r\n    }\r\n\r\n    return this.http.post<BulkContactResult>(\r\n      `${this.apiUrl}/api/contacts/bulk`,\r\n      { contacts, audienceId },\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Subscribe a contact\r\n   */\r\n  subscribe(contactId: string): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    return this.http.post<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}/subscribe`,\r\n      {},\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Unsubscribe a contact\r\n   */\r\n  unsubscribe(contactId: string): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    return this.http.post<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}/unsubscribe`,\r\n      {},\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Add tags to a contact\r\n   */\r\n  addTags(contactId: string, tags: string[]): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    if (!tags || tags.length === 0) {\r\n      return throwError(() => new ValidationError('Tags array is required'));\r\n    }\r\n\r\n    return this.http.post<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}/tags`,\r\n      { tags },\r\n      { headers: this.getHeaders() },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Remove tags from a contact\r\n   */\r\n  removeTags(contactId: string, tags: string[]): Observable<Contact> {\r\n    if (!this.apiKey) {\r\n      return throwError(() => new MetiganError('Service not initialized. Call initialize() first.'));\r\n    }\r\n\r\n    if (!contactId) {\r\n      return throwError(() => new ValidationError('Contact ID is required'));\r\n    }\r\n\r\n    if (!tags || tags.length === 0) {\r\n      return throwError(() => new ValidationError('Tags array is required'));\r\n    }\r\n\r\n    return this.http.delete<Contact>(\r\n      `${this.apiUrl}/api/contacts/${contactId}/tags`,\r\n      {\r\n        headers: this.getHeaders(),\r\n        params: { tags: tags.join(',') }\r\n      },\r\n      this.retryCount,\r\n      this.retryDelay\r\n    );\r\n  }\r\n}\r\n\r\n"]}