Skip to content

Commit b67e701

Browse files
authored
Address issue with trailing backslashes in config path not squiggled (#3573)
1 parent 77f3747 commit b67e701

File tree

3 files changed

+54
-8
lines changed

3 files changed

+54
-8
lines changed

Extension/src/LanguageServer/configurations.ts

+2-5
Original file line numberDiff line numberDiff line change
@@ -762,13 +762,10 @@ export class CppProperties {
762762
}
763763

764764
if (handleSquiggles) {
765-
// Replace all \<escape character> with \\<character>.
765+
// Replace all \<escape character> with \\<character>, except for \"
766766
// Otherwise, the JSON.parse result will have the \<escape character> missing.
767-
readResults = readResults.replace(/\\/g, '\\\\');
768-
readResults = readResults.replace(/\\\\"/g, '\\"'); // Need to revert the change to \".
769-
readResults = readResults.replace(/\\\\"/g, '\\"'); // Need to do it again for \\".
767+
readResults = util.escapeForSquiggles(readResults);
770768
}
771-
772769
// Try to use the same configuration as before the change.
773770
let newJson: ConfigurationJson = JSON.parse(readResults);
774771
if (!newJson || !newJson.configurations || newJson.configurations.length === 0) {

Extension/src/common.ts

+32-1
Original file line numberDiff line numberDiff line change
@@ -800,4 +800,35 @@ export function extractCompilerPathAndArgs(inputCompilerPath: string): CompilerP
800800
}
801801
}
802802
return { compilerPath, additionalArgs };
803-
}
803+
}
804+
805+
export function escapeForSquiggles(s: string): string {
806+
// Replace all \<escape character> with \\<character>, except for \"
807+
// Otherwise, the JSON.parse result will have the \<escape character> missing.
808+
let newResults: string = "";
809+
let lastWasBackslash: Boolean = false;
810+
let lastBackslashWasEscaped: Boolean = false;
811+
for (let i: number = 0; i < s.length; i++) {
812+
if (s[i] === '\\') {
813+
if (lastWasBackslash) {
814+
newResults += "\\";
815+
lastBackslashWasEscaped = !lastBackslashWasEscaped;
816+
} else {
817+
lastBackslashWasEscaped = false;
818+
}
819+
newResults += "\\";
820+
lastWasBackslash = true;
821+
} else {
822+
if (lastWasBackslash && (lastBackslashWasEscaped || (s[i] !== '"'))) {
823+
newResults += "\\";
824+
}
825+
lastWasBackslash = false;
826+
lastBackslashWasEscaped = false;
827+
newResults += s[i];
828+
}
829+
}
830+
if (lastWasBackslash) {
831+
newResults += "\\";
832+
}
833+
return newResults;
834+
}

Extension/test/unitTests/common.test.ts

+20-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
* ------------------------------------------------------------------------------------------ */
55

66
import * as assert from "assert";
7-
import { resolveVariables } from "../../src/common";
7+
import { resolveVariables, escapeForSquiggles } from "../../src/common";
88

99
suite("Common Utility validation", () => {
1010
suite("resolveVariables", () => {
@@ -198,6 +198,25 @@ suite("Common Utility validation", () => {
198198
.shouldLookupSymbol("Root");
199199
});
200200

201+
test("escapeForSquiggles:", () => {
202+
let testEscapeForSquigglesScenario: any = (input: string, expectedOutput: string) => {
203+
let result: string = escapeForSquiggles(input);
204+
if (result !== expectedOutput) {
205+
throw new Error(`escapeForSquiggles failure: for \"${input}\", \"${result}\" !== \"${expectedOutput}\"`);
206+
}
207+
};
208+
209+
testEscapeForSquigglesScenario("\\", "\\\\"); // single backslash
210+
testEscapeForSquigglesScenario("\\\"", "\\\""); // escaped quote
211+
testEscapeForSquigglesScenario("\\\t", "\\\\\t"); // escaped non-quote
212+
testEscapeForSquigglesScenario("\\\\\"", "\\\\\\\\\""); // escaped backslash, unescaped quote
213+
testEscapeForSquigglesScenario("\\\\\t", "\\\\\\\\\t"); // escaped backslash, unescaped non-quote
214+
testEscapeForSquigglesScenario("\\t", "\\\\t"); // escaped non-quote
215+
testEscapeForSquigglesScenario("\\\\\\t", "\\\\\\\\\\\\t"); // escaped backslash, unescaped non-quote
216+
testEscapeForSquigglesScenario("\"\"", "\"\""); // empty quoted string
217+
testEscapeForSquigglesScenario("\"\\\\\"", "\"\\\\\\\\\""); // quoted string containing escaped backslash
218+
});
219+
201220
interface ResolveTestFlowEnvironment {
202221
withEnvironment(additionalEnvironment: {[key: string]: string | string[]}): ResolveTestFlowAssert;
203222
shouldLookupSymbol: (key: string) => void;
@@ -229,6 +248,5 @@ suite("Common Utility validation", () => {
229248
}
230249
};
231250
}
232-
233251
});
234252
});

0 commit comments

Comments
 (0)