@capillarytech/creatives-library 8.0.353-alpha.5 → 8.0.353-alpha.6

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 (136) hide show
  1. package/constants/unified.js +29 -0
  2. package/package.json +1 -1
  3. package/services/tests/api.test.js +35 -20
  4. package/utils/commonUtils.js +19 -1
  5. package/utils/rcsPayloadUtils.js +92 -0
  6. package/utils/templateVarUtils.js +201 -0
  7. package/utils/tests/rcsPayloadUtils.test.js +226 -0
  8. package/utils/tests/templateVarUtils.test.js +204 -0
  9. package/v2Components/CapActionButton/constants.js +7 -0
  10. package/v2Components/CapActionButton/index.js +166 -108
  11. package/v2Components/CapActionButton/index.scss +157 -6
  12. package/v2Components/CapActionButton/messages.js +19 -3
  13. package/v2Components/CapActionButton/tests/index.test.js +41 -17
  14. package/v2Components/CapTagList/index.js +10 -0
  15. package/v2Components/CommonTestAndPreview/CustomValuesEditor.js +72 -49
  16. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +8 -2
  17. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +213 -21
  18. package/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +16 -0
  19. package/v2Components/CommonTestAndPreview/DeliverySettings/index.js +85 -10
  20. package/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +30 -0
  21. package/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +79 -11
  22. package/v2Components/CommonTestAndPreview/SendTestMessage.js +10 -5
  23. package/v2Components/CommonTestAndPreview/UnifiedPreview/PreviewHeader.js +0 -17
  24. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js +157 -15
  25. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +346 -146
  26. package/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +138 -48
  27. package/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +11 -0
  28. package/v2Components/CommonTestAndPreview/constants.js +38 -4
  29. package/v2Components/CommonTestAndPreview/index.js +691 -235
  30. package/v2Components/CommonTestAndPreview/messages.js +45 -3
  31. package/v2Components/CommonTestAndPreview/previewApiUtils.js +59 -0
  32. package/v2Components/CommonTestAndPreview/sagas.js +25 -6
  33. package/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +308 -284
  34. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +231 -65
  35. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +118 -5
  36. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +341 -0
  37. package/v2Components/CommonTestAndPreview/tests/PreviewSection.test.js +8 -1
  38. package/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +34 -13
  39. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/PreviewHeader.test.js +0 -159
  40. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/RcsPreviewContent.test.js +281 -283
  41. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +199 -256
  42. package/v2Components/CommonTestAndPreview/tests/constants.test.js +1 -2
  43. package/v2Components/CommonTestAndPreview/tests/index.test.js +132 -198
  44. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +67 -0
  45. package/v2Components/CommonTestAndPreview/tests/sagas.test.js +36 -26
  46. package/v2Components/FormBuilder/index.js +11 -6
  47. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +91 -0
  48. package/v2Components/SmsFallback/constants.js +73 -0
  49. package/v2Components/SmsFallback/index.js +956 -0
  50. package/v2Components/SmsFallback/index.scss +265 -0
  51. package/v2Components/SmsFallback/messages.js +78 -0
  52. package/v2Components/SmsFallback/smsFallbackUtils.js +119 -0
  53. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +50 -0
  54. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +147 -0
  55. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +304 -0
  56. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +223 -0
  57. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +309 -0
  58. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +422 -0
  59. package/v2Components/SmsFallback/useLocalTemplateList.js +92 -0
  60. package/v2Components/TemplatePreview/_templatePreview.scss +38 -23
  61. package/v2Components/TemplatePreview/constants.js +2 -0
  62. package/v2Components/TemplatePreview/index.js +143 -31
  63. package/v2Components/TemplatePreview/tests/index.test.js +142 -0
  64. package/v2Components/TestAndPreviewSlidebox/index.js +15 -3
  65. package/v2Components/TestAndPreviewSlidebox/sagas.js +11 -4
  66. package/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +3 -1
  67. package/v2Components/VarSegmentMessageEditor/constants.js +2 -0
  68. package/v2Components/VarSegmentMessageEditor/index.js +125 -0
  69. package/v2Components/VarSegmentMessageEditor/index.scss +46 -0
  70. package/v2Containers/App/constants.js +0 -3
  71. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +43 -0
  72. package/v2Containers/CreativesContainer/SlideBoxContent.js +36 -4
  73. package/v2Containers/CreativesContainer/SlideBoxFooter.js +10 -1
  74. package/v2Containers/CreativesContainer/SlideBoxHeader.js +29 -4
  75. package/v2Containers/CreativesContainer/constants.js +9 -0
  76. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +79 -0
  77. package/v2Containers/CreativesContainer/index.js +322 -103
  78. package/v2Containers/CreativesContainer/index.scss +51 -1
  79. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +90 -0
  80. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +78 -34
  81. package/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +79 -16
  82. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +8 -0
  83. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +357 -98
  84. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +20 -15
  85. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +258 -0
  86. package/v2Containers/CreativesContainer/tests/index.test.js +71 -9
  87. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +125 -0
  88. package/v2Containers/MobilePush/Create/test/saga.test.js +2 -2
  89. package/v2Containers/Rcs/constants.js +119 -10
  90. package/v2Containers/Rcs/index.js +2445 -813
  91. package/v2Containers/Rcs/index.scss +280 -8
  92. package/v2Containers/Rcs/messages.js +34 -3
  93. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +225 -0
  94. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +98018 -70073
  95. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +0 -5
  96. package/v2Containers/Rcs/tests/index.test.js +152 -121
  97. package/v2Containers/Rcs/tests/mockData.js +38 -0
  98. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +318 -0
  99. package/v2Containers/Rcs/tests/utils.test.js +646 -30
  100. package/v2Containers/Rcs/utils.js +478 -11
  101. package/v2Containers/Sms/Create/index.js +106 -40
  102. package/v2Containers/Sms/smsFormDataHelpers.js +67 -0
  103. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +253 -0
  104. package/v2Containers/SmsTrai/Create/index.js +9 -4
  105. package/v2Containers/SmsTrai/Edit/constants.js +2 -0
  106. package/v2Containers/SmsTrai/Edit/index.js +640 -130
  107. package/v2Containers/SmsTrai/Edit/index.scss +121 -0
  108. package/v2Containers/SmsTrai/Edit/messages.js +14 -4
  109. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +4328 -2375
  110. package/v2Containers/SmsWrapper/index.js +37 -8
  111. package/v2Containers/TagList/index.js +6 -0
  112. package/v2Containers/Templates/TemplatesActionBar.js +101 -0
  113. package/v2Containers/Templates/_templates.scss +166 -9
  114. package/v2Containers/Templates/actions.js +11 -0
  115. package/v2Containers/Templates/constants.js +2 -0
  116. package/v2Containers/Templates/index.js +122 -120
  117. package/v2Containers/Templates/sagas.js +56 -12
  118. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +120 -0
  119. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1062 -1017
  120. package/v2Containers/Templates/tests/sagas.test.js +199 -16
  121. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +180 -0
  122. package/v2Containers/Templates/utils/smsTemplatesListApi.js +79 -0
  123. package/v2Containers/TemplatesV2/TemplatesV2.style.js +72 -1
  124. package/v2Containers/TemplatesV2/index.js +86 -23
  125. package/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +131 -0
  126. package/v2Containers/WeChat/MapTemplates/test/saga.test.js +9 -9
  127. package/v2Containers/WebPush/Create/index.js +8 -91
  128. package/v2Containers/WebPush/Create/index.scss +0 -7
  129. package/v2Containers/Whatsapp/index.js +3 -20
  130. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +578 -34
  131. package/v2Components/CommonTestAndPreview/UnifiedPreview/WebPushPreviewContent.js +0 -169
  132. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/WebPushPreviewContent.test.js +0 -522
  133. package/v2Containers/App/tests/constants.test.js +0 -61
  134. package/v2Containers/Templates/tests/webpush.test.js +0 -375
  135. package/v2Containers/WebPush/Create/tests/getTemplateContent.test.js +0 -338
  136. package/v2Containers/WebPush/Create/tests/testAndPreviewIntegration.test.js +0 -325
@@ -160,6 +160,24 @@ export const TAG_CONTENT_REGEX = /{{([^}]+)}}/g;
160
160
  export const ENTRY_TRIGGER_TAG_REGEX = /\bentryTrigger\.\w+(?:\.\w+)?(?:\(\w+\))?/g;
161
161
  export const SKIP_TAGS_REGEX_GROUPS = ["dynamic_expiry_date_after_\\d+_days.FORMAT_\\d", "unsubscribe\\(#[a-zA-Z\\d]{6}\\)", "Link_to_[a-zA-Z]", "SURVEY.*.TOKEN", "^[A-Za-z].*\\([a-zA-Z\\d]*\\)", "referral_unique_(code|url).*userid"];
162
162
 
163
+ // --- Template variable tokens (`{{var}}`, DLT `{#var#}`) ---
164
+ /** Global: all `{{…}}` placeholders in a string. */
165
+ export const DEFAULT_MUSTACHE_VAR_REGEX = /\{\{[^}]+\}\}/g;
166
+ /** Global: `{{…}}` or DLT `{#…#}` tokens (SMS combined mode). */
167
+ export const COMBINED_SMS_TEMPLATE_VAR_REGEX = /\{\{[^}]+\}\}|\{\#[^#]*\#\}/g;
168
+ /** Full-string check: one mustache token. */
169
+ export const MUSTACHE_VAR_TOKEN_FULL_STRING_REGEX = /^\{\{[^}]+\}\}$/;
170
+ /** Full-string check: one DLT hash token. */
171
+ export const DLT_HASH_VAR_TOKEN_FULL_STRING_REGEX = /^\{\#[^#]*\#\}$/;
172
+ /** Full-string with capture group: inner name for `{{ name }}`-style tokens (whitespace-tolerant). */
173
+ export const MUSTACHE_TOKEN_INNER_NAME_REGEX = /^\{\{\s*([^}]+?)\s*\}\}$/;
174
+ /** Full-string with capture group: inner name/body for `{# name #}` DLT tokens (whitespace-tolerant). */
175
+ export const DLT_HASH_TOKEN_INNER_NAME_REGEX = /^\{#\s*(.*?)\s*#\}$/;
176
+ /** Global with capture group: inner name inside `{{name}}`. */
177
+ export const MUSTACHE_VAR_NAME_CAPTURE_REGEX = /\{\{([^}]+)\}\}/g;
178
+ /** Global with capture group: inner body inside `{#body#}`. */
179
+ export const DLT_VAR_BODY_CAPTURE_REGEX = /\{\#([^#]*)\#\}/g;
180
+
163
181
  export const GET_TRANSLATION_MAPPED = {
164
182
  'en': 'en-US',
165
183
  'zh-cn': 'zh',
@@ -197,3 +215,14 @@ export const LOGOUT_FAILURE = 'cap/LOGOUT_FAILURE';
197
215
  export const JAPANESE_HELP_TEXT = 'ヘルプ :トークンの定義';
198
216
 
199
217
  export const TAG_TRANSLATION_DOC = 'https://docs.capillarytech.com/docs/tags-translation';
218
+
219
+ // --- RCS SMS fallback API contract (shared across modules: campaigns, journey, etc.) ---
220
+
221
+ /** Keys on `messageContent.*.smsFallBackContent` sent to the API: only `message` + `templateConfigs`.
222
+ * `rcsSmsFallbackVarMapped` is editor-only — merged into `message` at normalize, not sent on the wire. */
223
+ export const RCS_API_SMS_FALLBACK_KEYS = Object.freeze({
224
+ MESSAGE: 'message',
225
+ TEMPLATE_CONFIGS: 'templateConfigs',
226
+ RCS_SMS_FALLBACK_VAR_MAPPED: 'rcsSmsFallbackVarMapped',
227
+ });
228
+
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@capillarytech/creatives-library",
3
3
  "author": "meharaj",
4
- "version": "8.0.353-alpha.5",
4
+ "version": "8.0.353-alpha.6",
5
5
  "description": "Capillary creatives ui",
6
6
  "main": "./index.js",
7
7
  "module": "./index.es.js",
@@ -89,21 +89,26 @@ describe('uploadFile -- whatsapp image upload', () => {
89
89
 
90
90
  it('Uploads the file with the original filename when encodeURIComponent fails', async () => {
91
91
  // Mocking the encodeURIComponent function to throw an error
92
+ const originalEncodeURIComponent = global.encodeURIComponent;
92
93
  global.encodeURIComponent = jest.fn(() => { throw new Error('encodeURIComponent error'); });
93
94
  const blob = new Blob([''], { type: 'image/jpeg' });
94
95
  const file = new File([blob], '@%test.jpeg', { type: 'image/jpeg' });
95
- expect(
96
- uploadFile({
97
- file,
98
- assetType: 'image',
99
- fileParams: {
100
- width: 275,
101
- height: 183,
102
- error: false,
103
- },
104
- whatsappParams: {},
105
- }),
106
- ).toEqual(Promise.resolve());
96
+ try {
97
+ expect(
98
+ uploadFile({
99
+ file,
100
+ assetType: 'image',
101
+ fileParams: {
102
+ width: 275,
103
+ height: 183,
104
+ error: false,
105
+ },
106
+ whatsappParams: {},
107
+ }),
108
+ ).toEqual(Promise.resolve());
109
+ } finally {
110
+ global.encodeURIComponent = originalEncodeURIComponent;
111
+ }
107
112
  });
108
113
  });
109
114
 
@@ -1037,16 +1042,26 @@ describe('getMembersLookup', () => {
1037
1042
  expect(result).toBeInstanceOf(Promise);
1038
1043
  });
1039
1044
 
1040
- it('should call fetch with correct URL encoding and GET method', () => {
1045
+ it('should call fetch with correct URL encoding and GET method', async () => {
1041
1046
  global.fetch.mockClear();
1042
- getMembersLookup('email', 'user+test@example.com');
1047
+ await getMembersLookup('email', 'user+test@example.com');
1043
1048
  expect(global.fetch).toHaveBeenCalled();
1044
- const calls = global.fetch.mock.calls;
1045
- const anyMembersCall = calls.find((c) => c[0] && String(c[0]).includes('members'));
1046
- expect(anyMembersCall).toBeDefined();
1047
- expect(anyMembersCall[0]).toContain('identifierType=');
1048
- expect(anyMembersCall[0]).toContain('identifierValue=');
1049
- expect(anyMembersCall[1]?.method || 'GET').toBe('GET');
1049
+
1050
+ // Find the first call that uses both the members endpoint and proper encoding
1051
+ const call = global.fetch.mock.calls.find(
1052
+ ([url]) =>
1053
+ url &&
1054
+ url.includes('members') &&
1055
+ url.includes('identifierType=email') &&
1056
+ url.includes('identifierValue=user%2Btest%40example.com')
1057
+ );
1058
+ expect(call).toBeDefined();
1059
+
1060
+ // Check URL structure
1061
+ const [url, options] = call;
1062
+ expect(url).toContain('identifierType=email');
1063
+ expect(url).toContain('identifierValue=user%2Btest%40example.com');
1064
+ expect((options && options.method) || 'GET').toBe('GET');
1050
1065
  });
1051
1066
  });
1052
1067
 
@@ -539,4 +539,22 @@ export const isValidMobile = (mobile) => PHONE_REGEX.test(mobile);
539
539
  export const formatPhoneNumber = (phone) => {
540
540
  if (!phone) return '';
541
541
  return String(phone).replace(/[^\d]/g, '');
542
- };
542
+ };
543
+
544
+ /**
545
+ * TRAI sender IDs on persisted RCS SMS fallback: may live on the merged object, under
546
+ * `templateConfigs`, or (legacy) `templateConfigs.header`. Same resolution order as
547
+ * `createPayload` in `Rcs/index.js`.
548
+ */
549
+ export function extractRegisteredSenderIdsFromSmsFallbackRecord(record) {
550
+ if (!record || typeof record !== 'object') return null;
551
+ const tc = record.templateConfigs && typeof record.templateConfigs === 'object'
552
+ ? record.templateConfigs
553
+ : {};
554
+ const candidates = [record.registeredSenderIds, tc.registeredSenderIds, tc.header];
555
+ for (let i = 0; i < candidates.length; i += 1) {
556
+ const a = candidates[i];
557
+ if (Array.isArray(a) && a.length > 0) return a;
558
+ }
559
+ return null;
560
+ }
@@ -0,0 +1,92 @@
1
+ import isEmpty from 'lodash/isEmpty';
2
+ import { RCS_API_SMS_FALLBACK_KEYS } from '../constants/unified';
3
+ import {
4
+ COMBINED_SMS_TEMPLATE_VAR_REGEX,
5
+ isAnyTemplateVarToken,
6
+ splitTemplateVarString,
7
+ } from './templateVarUtils';
8
+
9
+ const {
10
+ MESSAGE,
11
+ TEMPLATE_CONFIGS,
12
+ RCS_SMS_FALLBACK_VAR_MAPPED,
13
+ } = RCS_API_SMS_FALLBACK_KEYS;
14
+
15
+ function mergeSmsFallbackSlots(templateStr, varMapped) {
16
+ const map = varMapped != null && typeof varMapped === 'object' ? varMapped : {};
17
+ if (!Object.keys(map).length) return typeof templateStr === 'string' ? templateStr : '';
18
+ const str = typeof templateStr === 'string' ? templateStr : '';
19
+ return splitTemplateVarString(str, COMBINED_SMS_TEMPLATE_VAR_REGEX)
20
+ .map((seg, i) => {
21
+ if (!isAnyTemplateVarToken(seg)) return seg;
22
+ const key = `${seg}_${i}`;
23
+ if (Object.prototype.hasOwnProperty.call(map, key)) {
24
+ const v = map[key];
25
+ return v == null ? '' : String(v);
26
+ }
27
+ return seg;
28
+ })
29
+ .join('');
30
+ }
31
+
32
+ /**
33
+ * Mutates one RCS messageContent entry: strips UI-only fields, promotes smsFallBackContent to a
34
+ * sibling of rcsContent, and folds rcsSmsFallbackVarMapped into the message string for the API.
35
+ */
36
+ export const normalizeRcsMessageContentForApi = messageContentItem => {
37
+ const {
38
+ rcsContent = {},
39
+ smsFallBackContent: rootSmsFallbackContent = {},
40
+ } = messageContentItem;
41
+ const {
42
+ smsFallBackContent: legacySmsFallbackNestedUnderCard,
43
+ ...rcsCardPayloadOnly
44
+ } = { ...rcsContent };
45
+ /* Legacy nested first, then root — so current root `smsFallBackContent` wins on key clashes. */
46
+ const mergedSmsFallbackSources = {
47
+ ...(legacySmsFallbackNestedUnderCard ?? {}),
48
+ ...rootSmsFallbackContent,
49
+ };
50
+ messageContentItem.rcsContent = rcsCardPayloadOnly;
51
+
52
+ delete messageContentItem.templateConfigs;
53
+ delete rcsCardPayloadOnly.templateConfigs;
54
+
55
+ if (Array.isArray(rcsCardPayloadOnly.cardContent)) {
56
+ rcsCardPayloadOnly.cardContent.forEach(card => {
57
+ if (Array.isArray(card.suggestions)) {
58
+ card.suggestions.forEach(suggestion => {
59
+ delete suggestion.isSaved;
60
+ });
61
+ }
62
+ });
63
+ }
64
+
65
+ const varMappedMerged = mergedSmsFallbackSources[RCS_SMS_FALLBACK_VAR_MAPPED];
66
+ const hasVarMapped =
67
+ varMappedMerged != null
68
+ && typeof varMappedMerged === 'object'
69
+ && Object.keys(varMappedMerged).length > 0;
70
+ const hasMessageKey = Object.prototype.hasOwnProperty.call(mergedSmsFallbackSources, MESSAGE);
71
+ const rawMessage = hasMessageKey ? mergedSmsFallbackSources[MESSAGE] : undefined;
72
+ const messageForApi =
73
+ hasMessageKey && hasVarMapped
74
+ ? mergeSmsFallbackSlots(rawMessage == null ? '' : String(rawMessage), varMappedMerged)
75
+ : rawMessage;
76
+
77
+ const apiSiblingSmsFallback = Object.fromEntries(
78
+ [
79
+ hasMessageKey && [MESSAGE, messageForApi],
80
+ !isEmpty(mergedSmsFallbackSources[TEMPLATE_CONFIGS]) && [
81
+ TEMPLATE_CONFIGS,
82
+ mergedSmsFallbackSources[TEMPLATE_CONFIGS],
83
+ ],
84
+ ].filter(Boolean),
85
+ );
86
+
87
+ if (Object.keys(apiSiblingSmsFallback).length) {
88
+ messageContentItem.smsFallBackContent = apiSiblingSmsFallback;
89
+ } else {
90
+ delete messageContentItem.smsFallBackContent;
91
+ }
92
+ };
@@ -0,0 +1,201 @@
1
+ /**
2
+ * Shared utilities for templates containing {{var}} and DLT `{#var#}` tokens.
3
+ * Same split process used by WhatsApp/RCS: match vars with regex, then split content at each var.
4
+ */
5
+
6
+ import {
7
+ COMBINED_SMS_TEMPLATE_VAR_REGEX,
8
+ DEFAULT_MUSTACHE_VAR_REGEX,
9
+ DLT_HASH_VAR_TOKEN_FULL_STRING_REGEX,
10
+ DLT_HASH_TOKEN_INNER_NAME_REGEX,
11
+ DLT_VAR_BODY_CAPTURE_REGEX,
12
+ MUSTACHE_TOKEN_INNER_NAME_REGEX,
13
+ MUSTACHE_VAR_NAME_CAPTURE_REGEX,
14
+ MUSTACHE_VAR_TOKEN_FULL_STRING_REGEX,
15
+ } from '../constants/unified';
16
+
17
+ export { COMBINED_SMS_TEMPLATE_VAR_REGEX, DEFAULT_MUSTACHE_VAR_REGEX } from '../constants/unified';
18
+
19
+ const isMustacheVarToken = (s) =>
20
+ typeof s === 'string' && MUSTACHE_VAR_TOKEN_FULL_STRING_REGEX.test(s);
21
+
22
+ export const isDltHashVarToken = (s) =>
23
+ typeof s === 'string' && DLT_HASH_VAR_TOKEN_FULL_STRING_REGEX.test(s);
24
+
25
+ export const isAnyTemplateVarToken = (s) => isMustacheVarToken(s) || isDltHashVarToken(s);
26
+
27
+ /**
28
+ * `RegExp.prototype.exec` only advances `lastIndex` when the `g` and/or `y` flag is set.
29
+ * A non-global regex in a `while ((m = re.exec(s)) !== null)` loop never advances and can run forever.
30
+ *
31
+ * @param {RegExp} regex
32
+ * @returns {RegExp} Same instance if already global; otherwise a new RegExp with `g` appended to flags.
33
+ */
34
+ function ensureGlobalRegexForExecLoop(regex) {
35
+ if (!regex || !(regex instanceof RegExp) || regex.global) {
36
+ return regex;
37
+ }
38
+ return new RegExp(regex.source, `${regex.flags}g`);
39
+ }
40
+
41
+ /**
42
+ * Splits `content` into alternating plain-text segments and variable tokens, using the order of
43
+ * matches in `matchedVariableTokens` (e.g. from `String.prototype.match` with a global regex).
44
+ *
45
+ * @param {string[]} matchedVariableTokens - Matched tokens in left-to-right order
46
+ * @param {string} content - Full template string
47
+ * @returns {string[]}
48
+ */
49
+ export const splitContentByOrderedVarTokens = (matchedVariableTokens, content) => {
50
+ const segmentList = [];
51
+ const tokenQueue = [...(matchedVariableTokens ?? [])];
52
+ let remainder = content ?? '';
53
+ while ((remainder?.length ?? 0) > 0) {
54
+ const nextVarToken = tokenQueue?.[0];
55
+ if (nextVarToken == null || nextVarToken === '') {
56
+ segmentList.push(remainder);
57
+ break;
58
+ }
59
+ const varStartIndex = remainder.indexOf(nextVarToken);
60
+ if (varStartIndex !== -1) {
61
+ segmentList.push(remainder.substring(0, varStartIndex));
62
+ segmentList.push(nextVarToken);
63
+ const afterVar = varStartIndex + (nextVarToken?.length ?? 0);
64
+ remainder = remainder.substring(afterVar, remainder?.length ?? 0);
65
+ tokenQueue.shift();
66
+ } else {
67
+ segmentList.push(remainder);
68
+ break;
69
+ }
70
+ }
71
+ return segmentList;
72
+ };
73
+
74
+ /**
75
+ * Splits a template string into an array of text + {{var}} segments using the given regex.
76
+ *
77
+ * @param {string} str - Template string
78
+ * @param {RegExp} [varRegex] - Regex that matches var tokens; defaults to DEFAULT_MUSTACHE_VAR_REGEX
79
+ * @returns {string[]}
80
+ */
81
+ export const splitTemplateVarString = (str = '', varRegex = DEFAULT_MUSTACHE_VAR_REGEX) => {
82
+ if (!str) return [];
83
+ const matchedVariableTokens = str.match(varRegex) || [];
84
+ return splitContentByOrderedVarTokens(matchedVariableTokens, str).filter(
85
+ (segment) => segment !== ''
86
+ );
87
+ };
88
+
89
+ /**
90
+ * Extracts unique variable names from `{{var}}` (and, when using default capture, `{#var#}`) strings.
91
+ *
92
+ * @param {string} templateStr
93
+ * @param {RegExp} [captureRegex] - regex with a single capture group for the var name; if omitted, also scans DLT `{#…#}`.
94
+ * @returns {string[]}
95
+ */
96
+ export const extractTemplateVariables = (templateStr = '', captureRegex) => {
97
+ if (!templateStr) return [];
98
+ const variables = [];
99
+ const add = (name) => {
100
+ const n = (name || '').trim();
101
+ if (n && !variables.includes(n)) variables.push(n);
102
+ };
103
+ const mustacheRe = ensureGlobalRegexForExecLoop(
104
+ captureRegex || MUSTACHE_VAR_NAME_CAPTURE_REGEX,
105
+ );
106
+ let match;
107
+ while ((match = mustacheRe.exec(templateStr)) !== null) {
108
+ add(match?.[1]);
109
+ }
110
+ if (!captureRegex) {
111
+ const dltRe = new RegExp(DLT_VAR_BODY_CAPTURE_REGEX.source, DLT_VAR_BODY_CAPTURE_REGEX.flags);
112
+ let dltMatch;
113
+ while ((dltMatch = dltRe.exec(templateStr)) !== null) {
114
+ add(dltMatch?.[1] || 'var');
115
+ }
116
+ }
117
+ return variables;
118
+ };
119
+
120
+ /**
121
+ * Looks up the inner name of a `{{name}}` or `{#name#}` token in a flat key→value map.
122
+ * Handles both exact matches and dot-path suffixes (e.g. `tag.FORMAT_1` → name `FORMAT_1`).
123
+ * Returns the resolved string value, or `undefined` if not found / blank.
124
+ */
125
+ function resolveUserVarForToken(segment, userVarMap) {
126
+ if (!userVarMap || typeof userVarMap !== 'object') return undefined;
127
+ const mMustache = segment.match(MUSTACHE_TOKEN_INNER_NAME_REGEX);
128
+ const mDlt = segment.match(DLT_HASH_TOKEN_INNER_NAME_REGEX);
129
+ const innerName = ((mMustache ? mMustache[1] : (mDlt ? mDlt[1] : '')) ?? '').trim();
130
+ if (!innerName) return undefined;
131
+ const direct = userVarMap[innerName];
132
+ if (direct != null && String(direct).trim() !== '') return String(direct);
133
+ const hit = Object.keys(userVarMap).find((k) => k === innerName || k.endsWith(`.${innerName}`));
134
+ if (hit != null && userVarMap[hit] != null && String(userVarMap[hit]).trim() !== '') return String(userVarMap[hit]);
135
+ return undefined;
136
+ }
137
+
138
+ /**
139
+ * SMS / DLT template preview: replace `{{…}}` / `{#…#}` tokens using `varMapData` keys `${token}_${index}`.
140
+ * Used by SmsTraiEdit (RCS SMS fallback) and UnifiedPreview fallback SMS bubble.
141
+ * DLT `{#…#}`: empty / unset slot values show the raw token (matches DLT preview UX); mustache `{{…}}` still
142
+ * resolves to empty when the slot is cleared.
143
+ *
144
+ * @param {string} templateStr
145
+ * @param {Object} varMapData - Slot map (`${token}_${index}` → value)
146
+ * @param {Object|null} [userVarMap] - Optional Test & Preview custom values; used as fallback when a slot
147
+ * is absent or blank. When omitted the function behaves identically to its original two-argument form.
148
+ */
149
+ export const getFallbackResolvedContent = (templateStr = '', varMapData = {}, userVarMap = null) => {
150
+ const fallbackVarSlotMap = varMapData ?? {};
151
+ const templateSegments = splitTemplateVarString(templateStr, COMBINED_SMS_TEMPLATE_VAR_REGEX);
152
+ return templateSegments
153
+ .map((segment, segmentIndex) => {
154
+ const isVariableToken = typeof segment === 'string' && isAnyTemplateVarToken(segment);
155
+ if (!isVariableToken) return segment;
156
+ const slotKey = `${segment}_${segmentIndex}`;
157
+ if (Object.prototype.hasOwnProperty.call(fallbackVarSlotMap, slotKey)) {
158
+ const slotValue = fallbackVarSlotMap[slotKey];
159
+ if (isDltHashVarToken(segment)) {
160
+ if (slotValue == null || String(slotValue).trim() === '') {
161
+ if (userVarMap) { const uv = resolveUserVarForToken(segment, userVarMap); if (uv !== undefined) return uv; }
162
+ return segment;
163
+ }
164
+ return String(slotValue);
165
+ }
166
+ if (slotValue != null && String(slotValue).trim() !== '') return String(slotValue);
167
+ if (userVarMap) { const uv = resolveUserVarForToken(segment, userVarMap); if (uv !== undefined) return uv; }
168
+ return '';
169
+ }
170
+ if (userVarMap) { const uv = resolveUserVarForToken(segment, userVarMap); if (uv !== undefined) return uv; }
171
+ if (isDltHashVarToken(segment)) return segment;
172
+ return '';
173
+ })
174
+ .join('');
175
+ };
176
+
177
+ /**
178
+ * SMS fallback **card** (library list): show filled slot text; if a slot is empty, unset, or
179
+ * whitespace-only, show the raw `{{…}}` / `{#…#}` token so “save without labels” still shows
180
+ * `{#var#}` in the card. {@link getFallbackResolvedContent} keeps DLT placeholders in preview when
181
+ * slots are empty; mustache cleared slots can still render empty.
182
+ */
183
+ export const getFallbackResolvedContentForCardDisplay = (templateStr = '', varMapData = {}) => {
184
+ const fallbackVarSlotMap = varMapData ?? {};
185
+ const templateSegments = splitTemplateVarString(templateStr, COMBINED_SMS_TEMPLATE_VAR_REGEX);
186
+ return templateSegments
187
+ .map((segment, segmentIndex) => {
188
+ const isVariableToken = typeof segment === 'string' && isAnyTemplateVarToken(segment);
189
+ if (!isVariableToken) return segment;
190
+ const slotKey = `${segment}_${segmentIndex}`;
191
+ if (Object.prototype.hasOwnProperty.call(fallbackVarSlotMap, slotKey)) {
192
+ const slotValue = fallbackVarSlotMap[slotKey];
193
+ if (slotValue == null) return segment;
194
+ const str = String(slotValue);
195
+ if (str.trim() === '') return segment;
196
+ return str;
197
+ }
198
+ return segment;
199
+ })
200
+ .join('');
201
+ };
@@ -0,0 +1,226 @@
1
+ import { normalizeRcsMessageContentForApi } from '../rcsPayloadUtils';
2
+
3
+ describe('normalizeRcsMessageContentForApi', () => {
4
+ it('defaults missing rcsContent to empty object and still normalizes root smsFallBackContent', () => {
5
+ const item = {
6
+ smsFallBackContent: { message: 'root-only' },
7
+ };
8
+ normalizeRcsMessageContentForApi(item);
9
+ expect(item.rcsContent).toEqual({});
10
+ expect(item.smsFallBackContent).toEqual({ message: 'root-only' });
11
+ });
12
+
13
+ it('moves nested smsFallBackContent out of rcsContent and merges onto root', () => {
14
+ const item = {
15
+ rcsContent: {
16
+ contentType: 'RICHCARD',
17
+ smsFallBackContent: {
18
+ message: 'nested-msg',
19
+ templateConfigs: { registeredSenderIds: ['H1'] },
20
+ },
21
+ },
22
+ smsFallBackContent: { message: 'root-msg' },
23
+ };
24
+ normalizeRcsMessageContentForApi(item);
25
+ expect(item.rcsContent).toEqual({ contentType: 'RICHCARD' });
26
+ expect(item.smsFallBackContent.message).toBe('root-msg');
27
+ expect(item.smsFallBackContent.templateConfigs).toEqual({
28
+ registeredSenderIds: ['H1'],
29
+ });
30
+ });
31
+
32
+ it('nested message merges when root has no message key', () => {
33
+ const item = {
34
+ rcsContent: {
35
+ smsFallBackContent: { message: 'from-nested' },
36
+ },
37
+ };
38
+ normalizeRcsMessageContentForApi(item);
39
+ expect(item.rcsContent).toEqual({});
40
+ expect(item.smsFallBackContent).toEqual({ message: 'from-nested' });
41
+ });
42
+
43
+ it('removes smsFallBackContent when nothing to serialize', () => {
44
+ const item = {
45
+ rcsContent: { smsFallBackContent: {} },
46
+ smsFallBackContent: {},
47
+ };
48
+ normalizeRcsMessageContentForApi(item);
49
+ expect(item.rcsContent).toEqual({});
50
+ expect(item.smsFallBackContent).toBeUndefined();
51
+ });
52
+
53
+ it('does not add templateConfigs when empty', () => {
54
+ const item = {
55
+ rcsContent: {},
56
+ smsFallBackContent: { message: 'x', templateConfigs: {} },
57
+ };
58
+ normalizeRcsMessageContentForApi(item);
59
+ expect(item.smsFallBackContent).toEqual({ message: 'x' });
60
+ });
61
+
62
+ it('serializes templateConfigs only when message is absent but templateConfigs has data', () => {
63
+ const item = {
64
+ rcsContent: {
65
+ smsFallBackContent: {
66
+ templateConfigs: { registeredSenderIds: ['Z'] },
67
+ },
68
+ },
69
+ };
70
+ normalizeRcsMessageContentForApi(item);
71
+ expect(item.smsFallBackContent).toEqual({
72
+ templateConfigs: { registeredSenderIds: ['Z'] },
73
+ });
74
+ });
75
+
76
+ it('includes message in api payload when key exists with empty string', () => {
77
+ const item = {
78
+ smsFallBackContent: { message: '', templateConfigs: { id: 't1' } },
79
+ };
80
+ normalizeRcsMessageContentForApi(item);
81
+ expect(item.smsFallBackContent).toEqual({
82
+ message: '',
83
+ templateConfigs: { id: 't1' },
84
+ });
85
+ });
86
+
87
+ it('merges when legacy nested is null-coalesced', () => {
88
+ const item = {
89
+ rcsContent: {
90
+ contentType: 'TEXT',
91
+ smsFallBackContent: null,
92
+ },
93
+ smsFallBackContent: { message: 'from-root' },
94
+ };
95
+ normalizeRcsMessageContentForApi(item);
96
+ expect(item.rcsContent).toEqual({ contentType: 'TEXT' });
97
+ expect(item.smsFallBackContent).toEqual({ message: 'from-root' });
98
+ });
99
+
100
+ it('folds rcsSmsFallbackVarMapped into message and omits var map from API shape', () => {
101
+ const msg = '{{optout}} {{fullname}} test SMS';
102
+ const varMapped = {
103
+ '{{optout}}_0': '{{city}}',
104
+ '{{fullname}}_2': '{{fullname}}',
105
+ };
106
+ const item = {
107
+ rcsContent: { contentType: 'RICHCARD' },
108
+ smsFallBackContent: {
109
+ message: msg,
110
+ rcsSmsFallbackVarMapped: varMapped,
111
+ },
112
+ };
113
+ normalizeRcsMessageContentForApi(item);
114
+ expect(item.smsFallBackContent).toEqual({
115
+ message: '{{city}} {{fullname}} test SMS',
116
+ });
117
+ expect(item.smsFallBackContent.rcsSmsFallbackVarMapped).toBeUndefined();
118
+ });
119
+
120
+ it('merges nested rcsSmsFallbackVarMapped with root message; root message wins on key clash', () => {
121
+ const nestedVar = { '{{a}}_0': '{{nested}}' };
122
+ const item = {
123
+ rcsContent: {
124
+ contentType: 'RICHCARD',
125
+ smsFallBackContent: {
126
+ message: '{{a}} nested-tail',
127
+ rcsSmsFallbackVarMapped: nestedVar,
128
+ },
129
+ },
130
+ smsFallBackContent: {
131
+ message: '{{a}} tail',
132
+ rcsSmsFallbackVarMapped: { '{{a}}_0': '{{root}}' },
133
+ },
134
+ };
135
+ normalizeRcsMessageContentForApi(item);
136
+ expect(item.smsFallBackContent.message).toBe('{{root}} tail');
137
+ expect(item.smsFallBackContent.rcsSmsFallbackVarMapped).toBeUndefined();
138
+ });
139
+
140
+ it('non-slot var map keys do not change message but var map is still stripped', () => {
141
+ const varMapped = { 0: { label: 'City', tag: '{{city}}' } };
142
+ const item = {
143
+ rcsContent: { contentType: 'RICHCARD' },
144
+ smsFallBackContent: {
145
+ message: 'Hello',
146
+ rcsSmsFallbackVarMapped: varMapped,
147
+ },
148
+ };
149
+ normalizeRcsMessageContentForApi(item);
150
+ expect(item.smsFallBackContent).toEqual({ message: 'Hello' });
151
+ });
152
+
153
+ it('omits rcsSmsFallbackVarMapped when merged value is empty object', () => {
154
+ const item = {
155
+ rcsContent: {
156
+ smsFallBackContent: { rcsSmsFallbackVarMapped: {} },
157
+ },
158
+ smsFallBackContent: { message: 'only-msg' },
159
+ };
160
+ normalizeRcsMessageContentForApi(item);
161
+ expect(item.smsFallBackContent).toEqual({ message: 'only-msg' });
162
+ expect(item.smsFallBackContent.rcsSmsFallbackVarMapped).toBeUndefined();
163
+ });
164
+
165
+ it('removes root-level templateConfigs from RCS message content item', () => {
166
+ const item = {
167
+ rcsContent: { cardType: 'STANDALONE' },
168
+ templateConfigs: {
169
+ templateId: '69b3d33934a7bd0db5bada8e',
170
+ template: '{{registered_store_name}} Its a test message',
171
+ },
172
+ };
173
+ normalizeRcsMessageContentForApi(item);
174
+ expect(item.templateConfigs).toBeUndefined();
175
+ expect(item.rcsContent).toEqual({ cardType: 'STANDALONE' });
176
+ });
177
+
178
+ it('removes templateConfigs nested inside rcsContent', () => {
179
+ const item = {
180
+ rcsContent: {
181
+ cardType: 'STANDALONE',
182
+ templateConfigs: {
183
+ templateId: '69b3d33934a7bd0db5bada8e',
184
+ template: '{{registered_store_name}} Its a test message',
185
+ },
186
+ cardContent: [{ title: 'Card' }],
187
+ },
188
+ };
189
+ normalizeRcsMessageContentForApi(item);
190
+ expect(item.templateConfigs).toBeUndefined();
191
+ expect(item.rcsContent.templateConfigs).toBeUndefined();
192
+ expect(item.rcsContent.cardContent).toEqual([{ title: 'Card' }]);
193
+ });
194
+
195
+ it('strips isSaved from each suggestion in cardContent', () => {
196
+ const item = {
197
+ rcsContent: {
198
+ cardContent: [
199
+ {
200
+ title: 'Card 1',
201
+ suggestions: [
202
+ { type: 'QUICK_REPLY', text: 'Yes', isSaved: true },
203
+ { type: 'CTA', text: 'Go', url: 'https://example.com', isSaved: false },
204
+ ],
205
+ },
206
+ ],
207
+ },
208
+ };
209
+ normalizeRcsMessageContentForApi(item);
210
+ expect(item.rcsContent.cardContent[0].suggestions).toEqual([
211
+ { type: 'QUICK_REPLY', text: 'Yes' },
212
+ { type: 'CTA', text: 'Go', url: 'https://example.com' },
213
+ ]);
214
+ });
215
+
216
+ it('handles cardContent with no suggestions gracefully', () => {
217
+ const item = {
218
+ rcsContent: {
219
+ cardContent: [{ title: 'Card 1' }, { title: 'Card 2', suggestions: [] }],
220
+ },
221
+ };
222
+ normalizeRcsMessageContentForApi(item);
223
+ expect(item.rcsContent.cardContent[0].suggestions).toBeUndefined();
224
+ expect(item.rcsContent.cardContent[1].suggestions).toEqual([]);
225
+ });
226
+ });