Handle curly braces inside strings in StringUtils.matchToClosingParenthesis

This is required to extract fully more complex YouTube nsig functions.
This commit is contained in:
ThetaDev 2022-08-12 13:30:33 +02:00 committed by AudricV
parent d12003651b
commit 52ded6e3d7
No known key found for this signature in database
GPG key ID: DA92EC7905614198
2 changed files with 56 additions and 4 deletions

View file

@ -23,14 +23,13 @@ public final class StringUtils {
} }
startIndex += start.length(); startIndex += start.length();
int endIndex = startIndex; int endIndex = findNextParenthesis(string, startIndex, true);
while (string.charAt(endIndex) != '{') {
++endIndex;
}
++endIndex; ++endIndex;
int openParenthesis = 1; int openParenthesis = 1;
while (openParenthesis > 0) { while (openParenthesis > 0) {
endIndex = findNextParenthesis(string, endIndex, false);
switch (string.charAt(endIndex)) { switch (string.charAt(endIndex)) {
case '{': case '{':
++openParenthesis; ++openParenthesis;
@ -46,4 +45,47 @@ public final class StringUtils {
return string.substring(startIndex, endIndex); return string.substring(startIndex, endIndex);
} }
private static int findNextParenthesis(@Nonnull final String string,
final int offset,
final boolean onlyOpen) {
boolean lastEscaped = false;
char quote = ' ';
for (int i = offset; i < string.length(); i++) {
boolean thisEscaped = false;
final char c = string.charAt(i);
switch (c) {
case '{':
if (quote == ' ') {
return i;
}
break;
case '}':
if (!onlyOpen && quote == ' ') {
return i;
}
break;
case '\\':
if (!lastEscaped) {
thisEscaped = true;
}
break;
case '\'':
case '"':
if (!lastEscaped) {
if (quote == ' ') {
quote = c;
} else if (quote == c) {
quote = ' ';
}
}
}
lastEscaped = thisEscaped;
}
return -1;
}
} }

View file

@ -58,4 +58,14 @@ public class StringUtilsTest {
assertEquals(expected, substring); assertEquals(expected, substring);
} }
@Test
void find_closing_with_quotes() {
final String expected = "{return \",}\\\"/\"}";
final String string = "function(d){return \",}\\\"/\"}";
final String substring = matchToClosingParenthesis(string, "function(d)");
assertEquals(expected, substring);
}
} }