Handle curly braces inside strings in StringUtils.matchToClosingParenthesis
This is required to extract fully more complex YouTube nsig functions.
This commit is contained in:
parent
d12003651b
commit
52ded6e3d7
2 changed files with 56 additions and 4 deletions
|
@ -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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
Reference in a new issue