@grainql/analytics-web 2.8.0 → 3.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 (61) hide show
  1. package/README.md +36 -3
  2. package/dist/cjs/consent.d.ts +38 -7
  3. package/dist/cjs/consent.d.ts.map +1 -1
  4. package/dist/cjs/consent.js +82 -23
  5. package/dist/cjs/consent.js.map +1 -1
  6. package/dist/cjs/id-manager.d.ts +66 -0
  7. package/dist/cjs/id-manager.d.ts.map +1 -0
  8. package/dist/cjs/id-manager.js +212 -0
  9. package/dist/cjs/id-manager.js.map +1 -0
  10. package/dist/cjs/index.d.ts +12 -8
  11. package/dist/cjs/index.d.ts.map +1 -1
  12. package/dist/cjs/index.js.map +1 -1
  13. package/dist/cjs/page-tracking.d.ts +6 -0
  14. package/dist/cjs/page-tracking.d.ts.map +1 -1
  15. package/dist/cjs/page-tracking.js +23 -2
  16. package/dist/cjs/page-tracking.js.map +1 -1
  17. package/dist/cjs/react/hooks/useConsent.d.ts +18 -2
  18. package/dist/cjs/react/hooks/useConsent.d.ts.map +1 -1
  19. package/dist/cjs/react/hooks/useConsent.js +52 -1
  20. package/dist/cjs/react/hooks/useConsent.js.map +1 -1
  21. package/dist/consent.d.ts +38 -7
  22. package/dist/consent.d.ts.map +1 -1
  23. package/dist/consent.js +82 -23
  24. package/dist/esm/consent.d.ts +38 -7
  25. package/dist/esm/consent.d.ts.map +1 -1
  26. package/dist/esm/consent.js +82 -23
  27. package/dist/esm/consent.js.map +1 -1
  28. package/dist/esm/id-manager.d.ts +66 -0
  29. package/dist/esm/id-manager.d.ts.map +1 -0
  30. package/dist/esm/id-manager.js +208 -0
  31. package/dist/esm/id-manager.js.map +1 -0
  32. package/dist/esm/index.d.ts +12 -8
  33. package/dist/esm/index.d.ts.map +1 -1
  34. package/dist/esm/index.js.map +1 -1
  35. package/dist/esm/page-tracking.d.ts +6 -0
  36. package/dist/esm/page-tracking.d.ts.map +1 -1
  37. package/dist/esm/page-tracking.js +23 -2
  38. package/dist/esm/page-tracking.js.map +1 -1
  39. package/dist/esm/react/hooks/useConsent.d.ts +18 -2
  40. package/dist/esm/react/hooks/useConsent.d.ts.map +1 -1
  41. package/dist/esm/react/hooks/useConsent.js +49 -1
  42. package/dist/esm/react/hooks/useConsent.js.map +1 -1
  43. package/dist/id-manager.d.ts +66 -0
  44. package/dist/id-manager.d.ts.map +1 -0
  45. package/dist/id-manager.js +212 -0
  46. package/dist/index.d.ts +12 -8
  47. package/dist/index.d.ts.map +1 -1
  48. package/dist/index.global.dev.js +310 -81
  49. package/dist/index.global.dev.js.map +4 -4
  50. package/dist/index.global.js +8 -8
  51. package/dist/index.global.js.map +4 -4
  52. package/dist/index.js +72 -44
  53. package/dist/index.mjs +73 -45
  54. package/dist/page-tracking.d.ts +6 -0
  55. package/dist/page-tracking.d.ts.map +1 -1
  56. package/dist/page-tracking.js +23 -2
  57. package/dist/react/hooks/useConsent.d.ts +18 -2
  58. package/dist/react/hooks/useConsent.d.ts.map +1 -1
  59. package/dist/react/hooks/useConsent.js +52 -1
  60. package/dist/react/hooks/useConsent.mjs +49 -1
  61. package/package.json +1 -1
@@ -1 +1 @@
1
- {"version":3,"file":"useConsent.js","sourceRoot":"","sources":["../../../../src/react/hooks/useConsent.ts"],"names":[],"mappings":";AAAA;;GAEG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAMH,gCAyDC;AA7DD,6CAA+B;AAC/B,2DAAwD;AAGxD,SAAgB,UAAU;IACxB,MAAM,MAAM,GAAG,IAAA,qCAAiB,GAAE,CAAC;IACnC,MAAM,CAAC,YAAY,EAAE,eAAe,CAAC,GAAG,KAAK,CAAC,QAAQ,CAAsB,IAAI,CAAC,CAAC;IAElF,KAAK,CAAC,SAAS,CAAC,GAAG,EAAE;QACnB,IAAI,CAAC,MAAM;YAAE,OAAO;QAEpB,4BAA4B;QAC5B,MAAM,YAAY,GAAG,MAAM,CAAC,eAAe,EAAE,CAAC;QAC9C,eAAe,CAAC,YAAY,CAAC,CAAC;QAE9B,6BAA6B;QAC7B,MAAM,QAAQ,GAAG,CAAC,KAAmB,EAAE,EAAE;YACvC,eAAe,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC,CAAC;QAEF,MAAM,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;QAEjC,OAAO,GAAG,EAAE;YACV,MAAM,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACpC,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IAEb,MAAM,YAAY,GAAG,KAAK,CAAC,WAAW,CACpC,CAAC,UAAqB,EAAE,EAAE;QACxB,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;QAClC,CAAC;IACH,CAAC,EACD,CAAC,MAAM,CAAC,CACT,CAAC;IAEF,MAAM,aAAa,GAAG,KAAK,CAAC,WAAW,CACrC,CAAC,UAAqB,EAAE,EAAE;QACxB,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QACnC,CAAC;IACH,CAAC,EACD,CAAC,MAAM,CAAC,CACT,CAAC;IAEF,MAAM,UAAU,GAAG,KAAK,CAAC,WAAW,CAClC,CAAC,QAAiB,EAAE,EAAE;QACpB,IAAI,CAAC,MAAM;YAAE,OAAO,KAAK,CAAC;QAC1B,OAAO,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;IACrC,CAAC,EACD,CAAC,MAAM,CAAC,CACT,CAAC;IAEF,OAAO;QACL,YAAY;QACZ,YAAY;QACZ,aAAa;QACb,UAAU;QACV,SAAS,EAAE,YAAY,EAAE,OAAO,IAAI,KAAK;QACzC,UAAU,EAAE,YAAY,EAAE,UAAU,IAAI,EAAE;KAC3C,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"useConsent.js","sourceRoot":"","sources":["../../../../src/react/hooks/useConsent.ts"],"names":[],"mappings":"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"}
package/dist/consent.d.ts CHANGED
@@ -1,8 +1,13 @@
1
1
  /**
2
- * Consent management for Grain Analytics
3
- * Handles GDPR-compliant consent tracking and state management
2
+ * Consent management for Grain Analytics v2.0
3
+ * Privacy-first, cookie-less by default
4
+ *
5
+ * Consent Modes:
6
+ * - cookieless: Default mode, daily rotating IDs, no consent needed
7
+ * - gdpr-strict: Requires explicit consent, falls back to cookieless
8
+ * - gdpr-opt-out: Permanent IDs by default, cookieless on opt-out
4
9
  */
5
- export type ConsentMode = 'opt-in' | 'opt-out' | 'disabled';
10
+ export type ConsentMode = 'cookieless' | 'gdpr-strict' | 'gdpr-opt-out';
6
11
  export interface ConsentState {
7
12
  granted: boolean;
8
13
  categories: string[];
@@ -13,6 +18,7 @@ export declare const DEFAULT_CONSENT_CATEGORIES: string[];
13
18
  export declare const CONSENT_VERSION = "1.0.0";
14
19
  /**
15
20
  * Consent manager for handling user consent state
21
+ * v2.0: Cookie-less by default, privacy-first approach
16
22
  */
17
23
  export declare class ConsentManager {
18
24
  private consentState;
@@ -23,9 +29,8 @@ export declare class ConsentManager {
23
29
  /**
24
30
  * Load consent state from localStorage
25
31
  *
26
- * GDPR Compliance: In opt-in mode, we can use localStorage for consent preferences
27
- * since storing consent choices is a legitimate interest and necessary for compliance.
28
- * The consent preference itself is not tracking data.
32
+ * GDPR Compliance: localStorage only used for storing consent preferences
33
+ * (not for tracking), which is a legitimate interest for compliance.
29
34
  */
30
35
  private loadConsentState;
31
36
  /**
@@ -45,11 +50,28 @@ export declare class ConsentManager {
45
50
  */
46
51
  getConsentState(): ConsentState | null;
47
52
  /**
48
- * Check if user has granted consent
53
+ * Check if user has granted consent for permanent IDs
49
54
  */
50
55
  hasConsent(category?: string): boolean;
56
+ /**
57
+ * Check if permanent IDs are allowed
58
+ */
59
+ shouldUsePermanentId(): boolean;
60
+ /**
61
+ * Check if we should strip query parameters from URLs
62
+ * Query params stripped unless:
63
+ * - Mode is gdpr-opt-out, OR
64
+ * - Mode is gdpr-strict AND consent given
65
+ */
66
+ shouldStripQueryParams(): boolean;
67
+ /**
68
+ * Check if we can track events (always true in v2.0)
69
+ * Even cookieless mode allows basic analytics with daily IDs
70
+ */
71
+ canTrack(): boolean;
51
72
  /**
52
73
  * Check if we should wait for consent before tracking
74
+ * Only relevant for GDPR Strict mode
53
75
  */
54
76
  shouldWaitForConsent(): boolean;
55
77
  /**
@@ -68,5 +90,14 @@ export declare class ConsentManager {
68
90
  * Clear all consent data
69
91
  */
70
92
  clearConsent(): void;
93
+ /**
94
+ * Get current consent mode
95
+ */
96
+ getConsentMode(): ConsentMode;
97
+ /**
98
+ * Get ID mode based on consent state
99
+ * Returns 'cookieless' or 'permanent'
100
+ */
101
+ getIdMode(): 'cookieless' | 'permanent';
71
102
  }
72
103
  //# sourceMappingURL=consent.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"consent.d.ts","sourceRoot":"","sources":["../src/consent.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,MAAM,MAAM,WAAW,GAAG,QAAQ,GAAG,SAAS,GAAG,UAAU,CAAC;AAE5D,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,IAAI,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,eAAO,MAAM,0BAA0B,UAA2C,CAAC;AACnF,eAAO,MAAM,eAAe,UAAU,CAAC;AAEvC;;GAEG;AACH,qBAAa,cAAc;IACzB,OAAO,CAAC,YAAY,CAA6B;IACjD,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,SAAS,CAA4C;gBAEjD,QAAQ,EAAE,MAAM,EAAE,WAAW,GAAE,WAAuB;IAMlE;;;;;;OAMG;IACH,OAAO,CAAC,gBAAgB;IA2BxB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAUxB;;OAEG;IACH,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IAczC;;OAEG;IACH,aAAa,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IA8B1C;;OAEG;IACH,eAAe,IAAI,YAAY,GAAG,IAAI;IAItC;;OAEG;IACH,UAAU,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO;IAwBtC;;OAEG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;OAEG;IACH,WAAW,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAI1D;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAO7D;;OAEG;IACH,OAAO,CAAC,eAAe;IAYvB;;OAEG;IACH,YAAY,IAAI,IAAI;CAUrB"}
1
+ {"version":3,"file":"consent.d.ts","sourceRoot":"","sources":["../src/consent.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,MAAM,MAAM,WAAW,GAAG,YAAY,GAAG,aAAa,GAAG,cAAc,CAAC;AAExE,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,IAAI,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,eAAO,MAAM,0BAA0B,UAA2C,CAAC;AACnF,eAAO,MAAM,eAAe,UAAU,CAAC;AAEvC;;;GAGG;AACH,qBAAa,cAAc;IACzB,OAAO,CAAC,YAAY,CAA6B;IACjD,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,SAAS,CAA4C;gBAEjD,QAAQ,EAAE,MAAM,EAAE,WAAW,GAAE,WAA0B;IAMrE;;;;;OAKG;IACH,OAAO,CAAC,gBAAgB;IA2BxB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAUxB;;OAEG;IACH,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IAczC;;OAEG;IACH,aAAa,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IA8B1C;;OAEG;IACH,eAAe,IAAI,YAAY,GAAG,IAAI;IAItC;;OAEG;IACH,UAAU,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO;IAiCtC;;OAEG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;;;;OAKG;IACH,sBAAsB,IAAI,OAAO;IAgBjC;;;OAGG;IACH,QAAQ,IAAI,OAAO;IAInB;;;OAGG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;OAEG;IACH,WAAW,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAI1D;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAO7D;;OAEG;IACH,OAAO,CAAC,eAAe;IAYvB;;OAEG;IACH,YAAY,IAAI,IAAI;IAWpB;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;;OAGG;IACH,SAAS,IAAI,YAAY,GAAG,WAAW;CAGxC"}
package/dist/consent.js CHANGED
@@ -1,7 +1,12 @@
1
1
  "use strict";
2
2
  /**
3
- * Consent management for Grain Analytics
4
- * Handles GDPR-compliant consent tracking and state management
3
+ * Consent management for Grain Analytics v2.0
4
+ * Privacy-first, cookie-less by default
5
+ *
6
+ * Consent Modes:
7
+ * - cookieless: Default mode, daily rotating IDs, no consent needed
8
+ * - gdpr-strict: Requires explicit consent, falls back to cookieless
9
+ * - gdpr-opt-out: Permanent IDs by default, cookieless on opt-out
5
10
  */
6
11
  Object.defineProperty(exports, "__esModule", { value: true });
7
12
  exports.ConsentManager = exports.CONSENT_VERSION = exports.DEFAULT_CONSENT_CATEGORIES = void 0;
@@ -9,9 +14,10 @@ exports.DEFAULT_CONSENT_CATEGORIES = ['necessary', 'analytics', 'functional'];
9
14
  exports.CONSENT_VERSION = '1.0.0';
10
15
  /**
11
16
  * Consent manager for handling user consent state
17
+ * v2.0: Cookie-less by default, privacy-first approach
12
18
  */
13
19
  class ConsentManager {
14
- constructor(tenantId, consentMode = 'opt-out') {
20
+ constructor(tenantId, consentMode = 'cookieless') {
15
21
  this.consentState = null;
16
22
  this.listeners = [];
17
23
  this.consentMode = consentMode;
@@ -21,9 +27,8 @@ class ConsentManager {
21
27
  /**
22
28
  * Load consent state from localStorage
23
29
  *
24
- * GDPR Compliance: In opt-in mode, we can use localStorage for consent preferences
25
- * since storing consent choices is a legitimate interest and necessary for compliance.
26
- * The consent preference itself is not tracking data.
30
+ * GDPR Compliance: localStorage only used for storing consent preferences
31
+ * (not for tracking), which is a legitimate interest for compliance.
27
32
  */
28
33
  loadConsentState() {
29
34
  if (typeof window === 'undefined')
@@ -37,8 +42,8 @@ class ConsentManager {
37
42
  timestamp: new Date(parsed.timestamp),
38
43
  };
39
44
  }
40
- else if (this.consentMode === 'opt-out' || this.consentMode === 'disabled') {
41
- // Auto-grant consent for opt-out and disabled modes
45
+ else if (this.consentMode === 'gdpr-opt-out') {
46
+ // Auto-grant consent for opt-out mode (user hasn't opted out yet)
42
47
  this.consentState = {
43
48
  granted: true,
44
49
  categories: exports.DEFAULT_CONSENT_CATEGORIES,
@@ -47,10 +52,10 @@ class ConsentManager {
47
52
  };
48
53
  this.saveConsentState();
49
54
  }
50
- // Note: In opt-in mode without stored consent, consentState remains null (no consent)
55
+ // Note: cookieless and gdpr-strict modes without stored consent no permanent tracking
51
56
  }
52
57
  catch (error) {
53
- // Silent failure - consent will be requested when needed
58
+ // Silent failure - will use cookieless mode by default
54
59
  }
55
60
  }
56
61
  /**
@@ -120,32 +125,73 @@ class ConsentManager {
120
125
  return this.consentState ? { ...this.consentState } : null;
121
126
  }
122
127
  /**
123
- * Check if user has granted consent
128
+ * Check if user has granted consent for permanent IDs
124
129
  */
125
130
  hasConsent(category) {
126
- // Disabled mode always returns true (no consent required)
127
- if (this.consentMode === 'disabled') {
128
- return true;
131
+ // Cookie-less mode: no consent needed (no permanent tracking)
132
+ if (this.consentMode === 'cookieless') {
133
+ return false; // No permanent IDs
129
134
  }
130
- // No consent state in opt-in mode means no consent
131
- if (this.consentMode === 'opt-in' && !this.consentState) {
132
- return false;
135
+ // GDPR Strict: requires explicit consent
136
+ if (this.consentMode === 'gdpr-strict') {
137
+ if (!this.consentState?.granted) {
138
+ return false;
139
+ }
133
140
  }
134
- // Check consent state
135
- if (!this.consentState?.granted) {
136
- return false;
141
+ // GDPR Opt-out: consent by default unless explicitly revoked
142
+ if (this.consentMode === 'gdpr-opt-out') {
143
+ // If no state, assume consent (opt-out model)
144
+ if (!this.consentState) {
145
+ return true;
146
+ }
147
+ // Check if consent was revoked
148
+ if (!this.consentState.granted) {
149
+ return false;
150
+ }
137
151
  }
138
152
  // Check specific category if provided
139
- if (category) {
153
+ if (category && this.consentState) {
140
154
  return this.consentState.categories.includes(category);
141
155
  }
142
- return true;
156
+ return this.consentState?.granted ?? (this.consentMode === 'gdpr-opt-out');
157
+ }
158
+ /**
159
+ * Check if permanent IDs are allowed
160
+ */
161
+ shouldUsePermanentId() {
162
+ return this.hasConsent();
163
+ }
164
+ /**
165
+ * Check if we should strip query parameters from URLs
166
+ * Query params stripped unless:
167
+ * - Mode is gdpr-opt-out, OR
168
+ * - Mode is gdpr-strict AND consent given
169
+ */
170
+ shouldStripQueryParams() {
171
+ if (this.consentMode === 'cookieless') {
172
+ return true; // Always strip in cookieless mode
173
+ }
174
+ if (this.consentMode === 'gdpr-strict') {
175
+ return !this.hasConsent(); // Strip unless consented
176
+ }
177
+ if (this.consentMode === 'gdpr-opt-out') {
178
+ return false; // Don't strip in opt-out mode
179
+ }
180
+ return true; // Default: strip
181
+ }
182
+ /**
183
+ * Check if we can track events (always true in v2.0)
184
+ * Even cookieless mode allows basic analytics with daily IDs
185
+ */
186
+ canTrack() {
187
+ return true; // All modes allow some form of tracking
143
188
  }
144
189
  /**
145
190
  * Check if we should wait for consent before tracking
191
+ * Only relevant for GDPR Strict mode
146
192
  */
147
193
  shouldWaitForConsent() {
148
- return this.consentMode === 'opt-in' && !this.consentState?.granted;
194
+ return this.consentMode === 'gdpr-strict' && !this.consentState?.granted;
149
195
  }
150
196
  /**
151
197
  * Add consent change listener
@@ -191,6 +237,19 @@ class ConsentManager {
191
237
  // Silent failure - consent state may not be fully cleared
192
238
  }
193
239
  }
240
+ /**
241
+ * Get current consent mode
242
+ */
243
+ getConsentMode() {
244
+ return this.consentMode;
245
+ }
246
+ /**
247
+ * Get ID mode based on consent state
248
+ * Returns 'cookieless' or 'permanent'
249
+ */
250
+ getIdMode() {
251
+ return this.shouldUsePermanentId() ? 'permanent' : 'cookieless';
252
+ }
194
253
  }
195
254
  exports.ConsentManager = ConsentManager;
196
255
  //# sourceMappingURL=consent.js.map
@@ -1,8 +1,13 @@
1
1
  /**
2
- * Consent management for Grain Analytics
3
- * Handles GDPR-compliant consent tracking and state management
2
+ * Consent management for Grain Analytics v2.0
3
+ * Privacy-first, cookie-less by default
4
+ *
5
+ * Consent Modes:
6
+ * - cookieless: Default mode, daily rotating IDs, no consent needed
7
+ * - gdpr-strict: Requires explicit consent, falls back to cookieless
8
+ * - gdpr-opt-out: Permanent IDs by default, cookieless on opt-out
4
9
  */
5
- export type ConsentMode = 'opt-in' | 'opt-out' | 'disabled';
10
+ export type ConsentMode = 'cookieless' | 'gdpr-strict' | 'gdpr-opt-out';
6
11
  export interface ConsentState {
7
12
  granted: boolean;
8
13
  categories: string[];
@@ -13,6 +18,7 @@ export declare const DEFAULT_CONSENT_CATEGORIES: string[];
13
18
  export declare const CONSENT_VERSION = "1.0.0";
14
19
  /**
15
20
  * Consent manager for handling user consent state
21
+ * v2.0: Cookie-less by default, privacy-first approach
16
22
  */
17
23
  export declare class ConsentManager {
18
24
  private consentState;
@@ -23,9 +29,8 @@ export declare class ConsentManager {
23
29
  /**
24
30
  * Load consent state from localStorage
25
31
  *
26
- * GDPR Compliance: In opt-in mode, we can use localStorage for consent preferences
27
- * since storing consent choices is a legitimate interest and necessary for compliance.
28
- * The consent preference itself is not tracking data.
32
+ * GDPR Compliance: localStorage only used for storing consent preferences
33
+ * (not for tracking), which is a legitimate interest for compliance.
29
34
  */
30
35
  private loadConsentState;
31
36
  /**
@@ -45,11 +50,28 @@ export declare class ConsentManager {
45
50
  */
46
51
  getConsentState(): ConsentState | null;
47
52
  /**
48
- * Check if user has granted consent
53
+ * Check if user has granted consent for permanent IDs
49
54
  */
50
55
  hasConsent(category?: string): boolean;
56
+ /**
57
+ * Check if permanent IDs are allowed
58
+ */
59
+ shouldUsePermanentId(): boolean;
60
+ /**
61
+ * Check if we should strip query parameters from URLs
62
+ * Query params stripped unless:
63
+ * - Mode is gdpr-opt-out, OR
64
+ * - Mode is gdpr-strict AND consent given
65
+ */
66
+ shouldStripQueryParams(): boolean;
67
+ /**
68
+ * Check if we can track events (always true in v2.0)
69
+ * Even cookieless mode allows basic analytics with daily IDs
70
+ */
71
+ canTrack(): boolean;
51
72
  /**
52
73
  * Check if we should wait for consent before tracking
74
+ * Only relevant for GDPR Strict mode
53
75
  */
54
76
  shouldWaitForConsent(): boolean;
55
77
  /**
@@ -68,5 +90,14 @@ export declare class ConsentManager {
68
90
  * Clear all consent data
69
91
  */
70
92
  clearConsent(): void;
93
+ /**
94
+ * Get current consent mode
95
+ */
96
+ getConsentMode(): ConsentMode;
97
+ /**
98
+ * Get ID mode based on consent state
99
+ * Returns 'cookieless' or 'permanent'
100
+ */
101
+ getIdMode(): 'cookieless' | 'permanent';
71
102
  }
72
103
  //# sourceMappingURL=consent.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"consent.d.ts","sourceRoot":"","sources":["../../src/consent.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,MAAM,MAAM,WAAW,GAAG,QAAQ,GAAG,SAAS,GAAG,UAAU,CAAC;AAE5D,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,IAAI,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,eAAO,MAAM,0BAA0B,UAA2C,CAAC;AACnF,eAAO,MAAM,eAAe,UAAU,CAAC;AAEvC;;GAEG;AACH,qBAAa,cAAc;IACzB,OAAO,CAAC,YAAY,CAA6B;IACjD,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,SAAS,CAA4C;gBAEjD,QAAQ,EAAE,MAAM,EAAE,WAAW,GAAE,WAAuB;IAMlE;;;;;;OAMG;IACH,OAAO,CAAC,gBAAgB;IA2BxB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAUxB;;OAEG;IACH,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IAczC;;OAEG;IACH,aAAa,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IA8B1C;;OAEG;IACH,eAAe,IAAI,YAAY,GAAG,IAAI;IAItC;;OAEG;IACH,UAAU,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO;IAwBtC;;OAEG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;OAEG;IACH,WAAW,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAI1D;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAO7D;;OAEG;IACH,OAAO,CAAC,eAAe;IAYvB;;OAEG;IACH,YAAY,IAAI,IAAI;CAUrB"}
1
+ {"version":3,"file":"consent.d.ts","sourceRoot":"","sources":["../../src/consent.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,MAAM,MAAM,WAAW,GAAG,YAAY,GAAG,aAAa,GAAG,cAAc,CAAC;AAExE,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,IAAI,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,eAAO,MAAM,0BAA0B,UAA2C,CAAC;AACnF,eAAO,MAAM,eAAe,UAAU,CAAC;AAEvC;;;GAGG;AACH,qBAAa,cAAc;IACzB,OAAO,CAAC,YAAY,CAA6B;IACjD,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,SAAS,CAA4C;gBAEjD,QAAQ,EAAE,MAAM,EAAE,WAAW,GAAE,WAA0B;IAMrE;;;;;OAKG;IACH,OAAO,CAAC,gBAAgB;IA2BxB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAUxB;;OAEG;IACH,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IAczC;;OAEG;IACH,aAAa,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IA8B1C;;OAEG;IACH,eAAe,IAAI,YAAY,GAAG,IAAI;IAItC;;OAEG;IACH,UAAU,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO;IAiCtC;;OAEG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;;;;OAKG;IACH,sBAAsB,IAAI,OAAO;IAgBjC;;;OAGG;IACH,QAAQ,IAAI,OAAO;IAInB;;;OAGG;IACH,oBAAoB,IAAI,OAAO;IAI/B;;OAEG;IACH,WAAW,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAI1D;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,GAAG,IAAI;IAO7D;;OAEG;IACH,OAAO,CAAC,eAAe;IAYvB;;OAEG;IACH,YAAY,IAAI,IAAI;IAWpB;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;;OAGG;IACH,SAAS,IAAI,YAAY,GAAG,WAAW;CAGxC"}
@@ -1,14 +1,20 @@
1
1
  /**
2
- * Consent management for Grain Analytics
3
- * Handles GDPR-compliant consent tracking and state management
2
+ * Consent management for Grain Analytics v2.0
3
+ * Privacy-first, cookie-less by default
4
+ *
5
+ * Consent Modes:
6
+ * - cookieless: Default mode, daily rotating IDs, no consent needed
7
+ * - gdpr-strict: Requires explicit consent, falls back to cookieless
8
+ * - gdpr-opt-out: Permanent IDs by default, cookieless on opt-out
4
9
  */
5
10
  export const DEFAULT_CONSENT_CATEGORIES = ['necessary', 'analytics', 'functional'];
6
11
  export const CONSENT_VERSION = '1.0.0';
7
12
  /**
8
13
  * Consent manager for handling user consent state
14
+ * v2.0: Cookie-less by default, privacy-first approach
9
15
  */
10
16
  export class ConsentManager {
11
- constructor(tenantId, consentMode = 'opt-out') {
17
+ constructor(tenantId, consentMode = 'cookieless') {
12
18
  this.consentState = null;
13
19
  this.listeners = [];
14
20
  this.consentMode = consentMode;
@@ -18,9 +24,8 @@ export class ConsentManager {
18
24
  /**
19
25
  * Load consent state from localStorage
20
26
  *
21
- * GDPR Compliance: In opt-in mode, we can use localStorage for consent preferences
22
- * since storing consent choices is a legitimate interest and necessary for compliance.
23
- * The consent preference itself is not tracking data.
27
+ * GDPR Compliance: localStorage only used for storing consent preferences
28
+ * (not for tracking), which is a legitimate interest for compliance.
24
29
  */
25
30
  loadConsentState() {
26
31
  if (typeof window === 'undefined')
@@ -34,8 +39,8 @@ export class ConsentManager {
34
39
  timestamp: new Date(parsed.timestamp),
35
40
  };
36
41
  }
37
- else if (this.consentMode === 'opt-out' || this.consentMode === 'disabled') {
38
- // Auto-grant consent for opt-out and disabled modes
42
+ else if (this.consentMode === 'gdpr-opt-out') {
43
+ // Auto-grant consent for opt-out mode (user hasn't opted out yet)
39
44
  this.consentState = {
40
45
  granted: true,
41
46
  categories: DEFAULT_CONSENT_CATEGORIES,
@@ -44,10 +49,10 @@ export class ConsentManager {
44
49
  };
45
50
  this.saveConsentState();
46
51
  }
47
- // Note: In opt-in mode without stored consent, consentState remains null (no consent)
52
+ // Note: cookieless and gdpr-strict modes without stored consent no permanent tracking
48
53
  }
49
54
  catch (error) {
50
- // Silent failure - consent will be requested when needed
55
+ // Silent failure - will use cookieless mode by default
51
56
  }
52
57
  }
53
58
  /**
@@ -117,32 +122,73 @@ export class ConsentManager {
117
122
  return this.consentState ? { ...this.consentState } : null;
118
123
  }
119
124
  /**
120
- * Check if user has granted consent
125
+ * Check if user has granted consent for permanent IDs
121
126
  */
122
127
  hasConsent(category) {
123
- // Disabled mode always returns true (no consent required)
124
- if (this.consentMode === 'disabled') {
125
- return true;
128
+ // Cookie-less mode: no consent needed (no permanent tracking)
129
+ if (this.consentMode === 'cookieless') {
130
+ return false; // No permanent IDs
126
131
  }
127
- // No consent state in opt-in mode means no consent
128
- if (this.consentMode === 'opt-in' && !this.consentState) {
129
- return false;
132
+ // GDPR Strict: requires explicit consent
133
+ if (this.consentMode === 'gdpr-strict') {
134
+ if (!this.consentState?.granted) {
135
+ return false;
136
+ }
130
137
  }
131
- // Check consent state
132
- if (!this.consentState?.granted) {
133
- return false;
138
+ // GDPR Opt-out: consent by default unless explicitly revoked
139
+ if (this.consentMode === 'gdpr-opt-out') {
140
+ // If no state, assume consent (opt-out model)
141
+ if (!this.consentState) {
142
+ return true;
143
+ }
144
+ // Check if consent was revoked
145
+ if (!this.consentState.granted) {
146
+ return false;
147
+ }
134
148
  }
135
149
  // Check specific category if provided
136
- if (category) {
150
+ if (category && this.consentState) {
137
151
  return this.consentState.categories.includes(category);
138
152
  }
139
- return true;
153
+ return this.consentState?.granted ?? (this.consentMode === 'gdpr-opt-out');
154
+ }
155
+ /**
156
+ * Check if permanent IDs are allowed
157
+ */
158
+ shouldUsePermanentId() {
159
+ return this.hasConsent();
160
+ }
161
+ /**
162
+ * Check if we should strip query parameters from URLs
163
+ * Query params stripped unless:
164
+ * - Mode is gdpr-opt-out, OR
165
+ * - Mode is gdpr-strict AND consent given
166
+ */
167
+ shouldStripQueryParams() {
168
+ if (this.consentMode === 'cookieless') {
169
+ return true; // Always strip in cookieless mode
170
+ }
171
+ if (this.consentMode === 'gdpr-strict') {
172
+ return !this.hasConsent(); // Strip unless consented
173
+ }
174
+ if (this.consentMode === 'gdpr-opt-out') {
175
+ return false; // Don't strip in opt-out mode
176
+ }
177
+ return true; // Default: strip
178
+ }
179
+ /**
180
+ * Check if we can track events (always true in v2.0)
181
+ * Even cookieless mode allows basic analytics with daily IDs
182
+ */
183
+ canTrack() {
184
+ return true; // All modes allow some form of tracking
140
185
  }
141
186
  /**
142
187
  * Check if we should wait for consent before tracking
188
+ * Only relevant for GDPR Strict mode
143
189
  */
144
190
  shouldWaitForConsent() {
145
- return this.consentMode === 'opt-in' && !this.consentState?.granted;
191
+ return this.consentMode === 'gdpr-strict' && !this.consentState?.granted;
146
192
  }
147
193
  /**
148
194
  * Add consent change listener
@@ -188,5 +234,18 @@ export class ConsentManager {
188
234
  // Silent failure - consent state may not be fully cleared
189
235
  }
190
236
  }
237
+ /**
238
+ * Get current consent mode
239
+ */
240
+ getConsentMode() {
241
+ return this.consentMode;
242
+ }
243
+ /**
244
+ * Get ID mode based on consent state
245
+ * Returns 'cookieless' or 'permanent'
246
+ */
247
+ getIdMode() {
248
+ return this.shouldUsePermanentId() ? 'permanent' : 'cookieless';
249
+ }
191
250
  }
192
251
  //# sourceMappingURL=consent.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"consent.js","sourceRoot":"","sources":["../../src/consent.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"consent.js","sourceRoot":"","sources":["../../src/consent.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,66 @@
1
+ /**
2
+ * ID Manager for Grain Analytics
3
+ * Generates and manages user IDs based on consent mode
4
+ *
5
+ * Privacy-first implementation:
6
+ * - Cookie-less mode: Daily rotating IDs (no persistence)
7
+ * - GDPR Strict: Permanent IDs only with consent
8
+ * - GDPR Opt-out: Permanent IDs by default
9
+ */
10
+ export type IdMode = 'cookieless' | 'permanent';
11
+ export interface IdConfig {
12
+ mode: IdMode;
13
+ tenantId: string;
14
+ useLocalStorage?: boolean;
15
+ }
16
+ /**
17
+ * ID Manager class
18
+ * Handles both daily rotating IDs and permanent IDs
19
+ */
20
+ export declare class IdManager {
21
+ private config;
22
+ private cachedDailyId;
23
+ private dailyIdDate;
24
+ private permanentId;
25
+ constructor(config: IdConfig);
26
+ /**
27
+ * Generate a daily rotating ID
28
+ * Rotates at midnight in user's local timezone
29
+ * Provides same-day continuity without persistent tracking
30
+ */
31
+ generateDailyRotatingId(): string;
32
+ /**
33
+ * Generate or retrieve permanent user ID
34
+ * Only used when consent is given
35
+ */
36
+ generatePermanentId(): string;
37
+ /**
38
+ * Get the current user ID based on mode
39
+ */
40
+ getCurrentUserId(): string;
41
+ /**
42
+ * Switch ID mode (e.g., when consent is granted/revoked)
43
+ */
44
+ setMode(mode: IdMode): void;
45
+ /**
46
+ * Load permanent ID from localStorage
47
+ */
48
+ private loadPermanentId;
49
+ /**
50
+ * Save permanent ID to localStorage
51
+ */
52
+ private savePermanentId;
53
+ /**
54
+ * Clear permanent ID from localStorage
55
+ */
56
+ private clearPermanentId;
57
+ /**
58
+ * Get info about current ID for debugging
59
+ */
60
+ getIdInfo(): {
61
+ mode: IdMode;
62
+ id: string;
63
+ isDailyRotating: boolean;
64
+ };
65
+ }
66
+ //# sourceMappingURL=id-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"id-manager.d.ts","sourceRoot":"","sources":["../../src/id-manager.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,MAAM,MAAM,MAAM,GAAG,YAAY,GAAG,WAAW,CAAC;AAEhD,MAAM,WAAW,QAAQ;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,eAAe,CAAC,EAAE,OAAO,CAAC;CAC3B;AA2DD;;;GAGG;AACH,qBAAa,SAAS;IACpB,OAAO,CAAC,MAAM,CAAW;IACzB,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,WAAW,CAAuB;IAC1C,OAAO,CAAC,WAAW,CAAuB;gBAE9B,MAAM,EAAE,QAAQ;IAS5B;;;;OAIG;IACH,uBAAuB,IAAI,MAAM;IAoBjC;;;OAGG;IACH,mBAAmB,IAAI,MAAM;IAyB7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;IAQ1B;;OAEG;IACH,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAkB3B;;OAEG;IACH,OAAO,CAAC,eAAe;IAiBvB;;OAEG;IACH,OAAO,CAAC,eAAe;IAWvB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAWxB;;OAEG;IACH,SAAS,IAAI;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,EAAE,EAAE,MAAM,CAAC;QAAC,eAAe,EAAE,OAAO,CAAA;KAAE;CAQpE"}