Skip to content

Commit ccb46a1

Browse files
authored
fix(Designer): Fixes an issue where '\"' causing json errors in expressions (#6636)
* fixes * remove logs * fix test * fix test * actually fixing it with a revert...
1 parent c4370aa commit ccb46a1

File tree

6 files changed

+112
-86
lines changed

6 files changed

+112
-86
lines changed

e2e/designer/tokens/escapeExpressionTokens.spec.ts

+23
Original file line numberDiff line numberDiff line change
@@ -42,5 +42,28 @@ test.describe(
4242
'{ "type": "Http", "inputs": { "uri": "http://test.com", "method": "GET", "body": "@{variables(\'ArrayVariable\')}@{array(split(variables (\'ArrayVariable\'), \'\\n\'))}" }, "runAfter": { "Filter_array": [ "SUCCEEDED" ] }, "runtimeConfiguration": { "contentTransfer": { "transferMode": "Chunked" } }}'
4343
);
4444
});
45+
46+
test('Expressions should maintain behavior of escaped characters in all instances the user views at', async ({ page }) => {
47+
await page.goto('/');
48+
await GoToMockWorkflow(page, 'Panel');
49+
await page.getByLabel('HTTP operation, HTTP connector').click();
50+
await page.getByTitle('Enter request content').getByRole('paragraph').click();
51+
await page.locator('[data-automation-id="msla-token-picker-entrypoint-button-expression"]').click();
52+
const viewLine = page.locator('.view-line').first();
53+
await viewLine.click();
54+
await viewLine.pressSequentially(`concat('{', '\\"ErrorDetail\\"', ':', '\\"Exchange get failed with exchange id', '-', '\\"}')`);
55+
await page.getByRole('tab', { name: 'Dynamic content' }).click();
56+
await page.getByRole('button', { name: 'Add', exact: true }).click();
57+
await page.getByRole('tab', { name: 'Code view' }).click();
58+
await expect(page.getByRole('code')).toContainText(
59+
`@{concat('{', '\\"ErrorDetail\\"', ':', '\\"Exchange get failed with exchange id', '-', '\\"}')}`
60+
);
61+
await page.getByRole('tab', { name: 'Parameters' }).click();
62+
await page.getByText('concat(...)').click();
63+
64+
await expect(page.getByRole('code')).toContainText(
65+
`concat('{', '\\"ErrorDetail\\"', ':', '\\"Exchange get failed with exchange id', '-', '\\"}')`
66+
);
67+
});
4568
}
4669
);

libs/designer-ui/src/lib/tokenpicker/tokenpickerfooter.tsx

-1
Original file line numberDiff line numberDiff line change
@@ -115,7 +115,6 @@ export function TokenPickerFooter({
115115
let currExpression: Expression | null = null;
116116
const unescapedExpressionValue = unescapeString(expression.value);
117117
try {
118-
console.log(unescapedExpressionValue);
119118
currExpression = ExpressionParser.parseExpression(unescapedExpressionValue);
120119
} catch (ex) {
121120
console.log(ex);

libs/designer/src/lib/core/utils/parameters/__test__/helper.spec.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -411,7 +411,7 @@ describe('core/utils/parameters/helper', () => {
411411
];
412412

413413
expect(parameterValueToJSONString(parameterValue, /* applyCasting */ false, /* forValidation */ true)).toBe(
414-
'{\n "newUnb3_1": @{xpath(xml(triggerBody()), \'string(/*[local-name()="DynamicsSOCSV"])\')}\n}'
414+
'{"newUnb3_1":"@xpath(xml(triggerBody()), \'string(/*[local-name()=\\"DynamicsSOCSV\\"])\')"}'
415415
);
416416
});
417417

@@ -485,7 +485,7 @@ describe('core/utils/parameters/helper', () => {
485485
];
486486

487487
expect(parameterValueToJSONString(parameterValue, /* applyCasting */ false, /* forValidation */ true)).toBe(
488-
'{\n "newUnb3_1": "@{xpath(xml(triggerBody()), \'string(/*[local-name()="DynamicsSOCSV"])\')}"\n}'
488+
'{"newUnb3_1":"@{xpath(xml(triggerBody()), \'string(/*[local-name()=\\"DynamicsSOCSV\\"])\')}"}'
489489
);
490490
});
491491

libs/designer/src/lib/core/utils/parameters/helper.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -3651,7 +3651,7 @@ export function parameterValueToJSONString(parameterValue: ValueSegment[], apply
36513651
let tokenExpression: string = expression.value;
36523652

36533653
if (isTokenValueSegment(expression)) {
3654-
const stringifiedTokenExpression = tokenExpression;
3654+
const stringifiedTokenExpression = JSON.stringify(tokenExpression).slice(1, -1);
36553655
// Note: Stringify the token expression to escape double quotes and other characters which must be escaped in JSON.
36563656
if (shouldInterpolate) {
36573657
if (applyCasting) {
@@ -3671,8 +3671,10 @@ export function parameterValueToJSONString(parameterValue: ValueSegment[], apply
36713671
const nextExpressionIsLiteral =
36723672
i < updatedParameterValue.length - 1 && updatedParameterValue[i + 1].type !== ValueSegmentType.TOKEN;
36733673
tokenExpression = `@${stringifiedTokenExpression}`;
3674-
tokenExpression = lastExpressionWasLiteral ? `"${tokenExpression}` : tokenExpression;
3675-
tokenExpression = nextExpressionIsLiteral ? `${tokenExpression}"` : `${tokenExpression}`;
3674+
// eslint-disable-next-line no-useless-escape
3675+
tokenExpression = lastExpressionWasLiteral ? `\"${tokenExpression}` : tokenExpression;
3676+
// eslint-disable-next-line no-useless-escape
3677+
tokenExpression = nextExpressionIsLiteral ? `${tokenExpression}\"` : `${tokenExpression}`;
36763678
}
36773679

36783680
parameterValueString += tokenExpression;

libs/logic-apps-shared/src/utils/src/lib/helpers/__test__/stringFunctions.spec.ts

+72-77
Original file line numberDiff line numberDiff line change
@@ -72,130 +72,125 @@ describe('canStringBeConverted', () => {
7272

7373
describe('unescapeString', () => {
7474
it('unescapes newline characters', () => {
75-
const input = 'Hello\\nWorld';
76-
const expectedOutput = 'Hello\nWorld';
77-
const result = unescapeString(input);
78-
expect(result).toBe(expectedOutput);
75+
expect(unescapeString('Hello\\nWorld')).toBe('Hello\nWorld');
7976
});
8077

8178
it('unescapes carriage return characters', () => {
82-
const input = 'Hello\\rWorld';
83-
const expectedOutput = 'Hello\rWorld';
84-
const result = unescapeString(input);
85-
expect(result).toBe(expectedOutput);
79+
expect(unescapeString('Hello\\rWorld')).toBe('Hello\rWorld');
8680
});
8781

8882
it('unescapes tab characters', () => {
89-
const input = 'Hello\\tWorld';
90-
const expectedOutput = 'Hello\tWorld';
91-
const result = unescapeString(input);
92-
expect(result).toBe(expectedOutput);
83+
expect(unescapeString('Hello\\tWorld')).toBe('Hello\tWorld');
9384
});
9485

9586
it('unescapes vertical tab characters', () => {
96-
const input = 'Hello\\vWorld';
97-
const expectedOutput = 'Hello\vWorld';
98-
const result = unescapeString(input);
99-
expect(result).toBe(expectedOutput);
87+
expect(unescapeString('Hello\\vWorld')).toBe('Hello\vWorld');
88+
});
89+
90+
it('unescapes backslashes', () => {
91+
expect(unescapeString('Hello\\\\World')).toBe('Hello\\World');
92+
});
93+
94+
it('unescapes double quotes', () => {
95+
expect(unescapeString('Hello\\"World"')).toBe('Hello"World"');
10096
});
10197

10298
it('returns the same string if there are no escape sequences', () => {
103-
const input = 'Hello World';
104-
const expectedOutput = 'Hello World';
105-
const result = unescapeString(input);
106-
expect(result).toBe(expectedOutput);
99+
expect(unescapeString('Hello World')).toBe('Hello World');
100+
});
101+
102+
it('handles multiple escape sequences in a row', () => {
103+
expect(unescapeString('Line1\\nLine2\\tTabbed')).toBe('Line1\nLine2\tTabbed');
104+
});
105+
106+
it('handles an empty string', () => {
107+
expect(unescapeString('')).toBe('');
108+
});
109+
110+
it('ignores invalid escape sequences', () => {
111+
expect(unescapeString('Hello\\xWorld')).toBe('Hello\\xWorld');
107112
});
108113
});
109114

110115
describe('escapeString', () => {
111116
it('escapes newline characters', () => {
112-
const input = 'Hello\nWorld';
113-
const expectedOutput = 'Hello\\nWorld';
114-
const result = escapeString(input);
115-
expect(result).toBe(expectedOutput);
117+
expect(escapeString('Hello\nWorld')).toBe('Hello\\nWorld');
116118
});
117119

118120
it('escapes carriage return characters', () => {
119-
const input = 'Hello\rWorld';
120-
const expectedOutput = 'Hello\\rWorld';
121-
const result = escapeString(input);
122-
expect(result).toBe(expectedOutput);
121+
expect(escapeString('Hello\rWorld')).toBe('Hello\\rWorld');
123122
});
124123

125124
it('escapes tab characters', () => {
126-
const input = 'Hello\tWorld';
127-
const expectedOutput = 'Hello\\tWorld';
128-
const result = escapeString(input);
129-
expect(result).toBe(expectedOutput);
125+
expect(escapeString('Hello\tWorld')).toBe('Hello\\tWorld');
130126
});
131127

132128
it('escapes vertical tab characters', () => {
133-
const input = 'Hello\vWorld';
134-
const expectedOutput = 'Hello\\vWorld';
135-
const result = escapeString(input);
136-
expect(result).toBe(expectedOutput);
129+
expect(escapeString('Hello\vWorld')).toBe('Hello\\vWorld');
137130
});
138131

139132
it('returns the same string if there are no special characters', () => {
140-
const input = 'Hello World';
141-
const expectedOutput = 'Hello World';
142-
const result = escapeString(input);
143-
expect(result).toBe(expectedOutput);
133+
expect(escapeString('Hello World')).toBe('Hello World');
144134
});
145135

146-
it('should correctly escape newline characters', () => {
147-
expect(escapeString('\n')).toEqual('\\n');
148-
expect(escapeString('Test\nTest')).toEqual('Test\\nTest');
136+
it('escapes newline characters', () => {
137+
expect(escapeString('\n')).toBe('\\n');
138+
expect(escapeString('Test\nTest')).toBe('Test\\nTest');
149139
});
150140

151-
it('should correctly escape backslashes and newline characters together', () => {
152-
expect(escapeString('\\\n')).toEqual('\\\\n');
153-
expect(escapeString('Test\\\nTest')).toEqual('Test\\\\nTest');
141+
it('escapes backslashes and newline characters together', () => {
142+
expect(escapeString('\\\n')).toBe('\\\\n');
143+
expect(escapeString('Test\\\nTest')).toBe('Test\\\\nTest');
154144
});
155145

156-
it('should handle an empty string', () => {
157-
expect(escapeString('')).toEqual('');
146+
it('handles an empty string', () => {
147+
expect(escapeString('')).toBe('');
158148
});
159149

160-
it('does not escape characters if requireSingleQuotesWrap is true and there are no surrounding single quotes', () => {
161-
const input = 'Test\nTest';
162-
const result = escapeString(input, true);
163-
expect(result).toBe(input); // No change, since it's not surrounded by single quotes
150+
it('does not escape characters if requireSingleQuotesWrap is true and the string is not wrapped in single quotes', () => {
151+
expect(escapeString('Test\nTest', true)).toBe('Test\nTest');
164152
});
165153

166-
it('escapes characters if requireSingleQuotesWrap is true and the string is surrounded by single quotes', () => {
167-
const input = "'Test\nTest'";
168-
const expectedOutput = "'Test\\nTest'";
169-
const result = escapeString(input, true);
170-
expect(result).toBe(expectedOutput); // Should escape \n
154+
it('escapes characters if requireSingleQuotesWrap is true and the string is wrapped in single quotes', () => {
155+
expect(escapeString("'Test\nTest'", true)).toBe("'Test\\nTest'");
171156
});
172157

173-
it('escapes characters even if the string contains multiple lines when requireSingleQuotesWrap is true and surrounded by single quotes', () => {
174-
const input = "'Test\nAnotherLine\nTest'";
175-
const expectedOutput = `'Test
158+
it('escapes multiple newlines when requireSingleQuotesWrap is true and wrapped in single quotes', () => {
159+
expect(escapeString("'Test\nAnotherLine\nTest'", true)).toBe(`'Test
176160
AnotherLine
177-
Test'`;
178-
const result = escapeString(input, true);
179-
expect(result).toBe(expectedOutput);
161+
Test'`);
162+
});
163+
164+
it('escapes characters even if requireSingleQuotesWrap is false, regardless of surrounding quotes', () => {
165+
expect(escapeString("'Test\nTest'", false)).toBe("'Test\\nTest'");
166+
});
167+
168+
it('escapes characters when requireSingleQuotesWrap is undefined, regardless of surrounding quotes', () => {
169+
expect(escapeString("'Test\nTest'")).toBe("'Test\\nTest'");
170+
});
171+
172+
it('escapes double quotes only when requireSingleQuotesWrap is true', () => {
173+
expect(escapeString(`concat('{', '"ErrorDetail"', ':', '"Exchange get failed with exchange id', '-', '"}')`, true)).toBe(
174+
`concat('{', '\\"ErrorDetail\\"', ':', '\\"Exchange get failed with exchange id', '-', '\\"}')`
175+
);
176+
expect(escapeString(`concat('{', '"ErrorDetail"', ':', '"Exchange get failed with exchange id', '-', '"}')`, false)).toBe(
177+
`concat('{', '"ErrorDetail"', ':', '"Exchange get failed with exchange id', '-', '"}')`
178+
);
179+
});
180+
181+
it('escapes double quotes and newlines when requireSingleQuotesWrap is true', () => {
182+
expect(escapeString('\'Hello\n"World"\'', true)).toBe('\'Hello\\n\\"World\\"\'');
180183
});
181184

182-
it('does not escape characters if requireSingleQuotesWrap is true and string is not surrounded by single quotes', () => {
183-
const input = 'Test\nTest';
184-
const result = escapeString(input, true);
185-
expect(result).toBe(input); // No change, since it's not surrounded by single quotes
185+
it('does not escape double quotes when requireSingleQuotesWrap is false', () => {
186+
expect(escapeString('Hello "World"', false)).toBe('Hello "World"');
186187
});
187188

188-
it('escapes characters when requireSingleQuotesWrap is false regardless of surrounding quotes', () => {
189-
const input = "'Test\nTest'";
190-
const expectedOutput = "'Test\\nTest'";
191-
const result = escapeString(input, false);
192-
expect(result).toBe(expectedOutput);
189+
it('escapes double quotes and other characters when requireSingleQuotesWrap is true and surrounded by single quotes', () => {
190+
expect(escapeString('\'Test\n"AnotherTest"\'', true)).toBe('\'Test\\n\\"AnotherTest\\"\'');
193191
});
194192

195-
it('escapes characters when requireSingleQuotesWrap is undefined regardless of surrounding quotes', () => {
196-
const input = "'Test\nTest'";
197-
const expectedOutput = "'Test\\nTest'";
198-
const result = escapeString(input);
199-
expect(result).toBe(expectedOutput);
193+
it('does not escape double quotes when requireSingleQuotesWrap is false', () => {
194+
expect(escapeString('Test "Hello"', false)).toBe('Test "Hello"');
200195
});
201196
});

libs/logic-apps-shared/src/utils/src/lib/helpers/stringFunctions.ts

+10-3
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ export const cleanResourceId = (resourceId?: string): string => {
5555
};
5656

5757
export const unescapeString = (input: string): string => {
58-
return input.replace(/\\([nrtv])/g, (_match, char) => {
58+
return input.replace(/\\([nrtv"\\])/g, (_match, char) => {
5959
switch (char) {
6060
case 'n':
6161
return '\n';
@@ -65,18 +65,23 @@ export const unescapeString = (input: string): string => {
6565
return '\t';
6666
case 'v':
6767
return '\v';
68+
case '"':
69+
return '"';
70+
case '\\':
71+
return '\\';
6872
default:
6973
return char;
7074
}
7175
});
7276
};
7377

7478
export const escapeString = (input: string, requireSingleQuotesWrap?: boolean): string => {
75-
if (requireSingleQuotesWrap && !/'.*[\n\r\t\v].*'/.test(input)) {
79+
// Only apply escaping if requireSingleQuotesWrap is true and the input is wrapped in single quotes
80+
if (requireSingleQuotesWrap && !/'.*[\n\r\t\v"].*'/.test(input)) {
7681
return input;
7782
}
7883

79-
return input?.replace(/[\n\r\t\v]/g, (char) => {
84+
return input?.replace(/[\n\r\t\v"]/g, (char) => {
8085
switch (char) {
8186
case '\n':
8287
return '\\n';
@@ -86,6 +91,8 @@ export const escapeString = (input: string, requireSingleQuotesWrap?: boolean):
8691
return '\\t';
8792
case '\v':
8893
return '\\v';
94+
case '"':
95+
return requireSingleQuotesWrap ? '\\"' : '"'; // Escape only if requireSingleQuotesWrap is true
8996
default:
9097
return char;
9198
}

0 commit comments

Comments
 (0)