PY-20775 Report missing closing brace for expression parts inside f-strings

Also fixed detection of closing brace after conversion characters and
format specifiers inside expression parts.
This commit is contained in:
Mikhail Golubev
2016-09-21 20:42:36 +03:00
parent 7f134f8908
commit 6ec6eccb64
5 changed files with 33 additions and 3 deletions

View File

@@ -120,12 +120,19 @@ public class FStringParser {
contentEndOffset = offset;
if (c1 == '}') {
rightBraceOffset = offset;
offset++;
break;
}
}
}
else if (c1 == '{') {
offset = parseFragment(offset, depth + 1);
continue;
}
else if (c1 == '}') {
rightBraceOffset = offset;
offset++;
break;
}
offset++;
}

View File

@@ -33,13 +33,21 @@ public class FStringsAnnotator extends PyAnnotator {
@Override
public void visitPyStringLiteralExpression(PyStringLiteralExpression pyString) {
for (ASTNode node : pyString.getStringNodes()) {
if (new StringNodeInfo(node).isFormatted()) {
final StringNodeInfo nodeInfo = new StringNodeInfo(node);
if (nodeInfo.isFormatted()) {
final int nodeOffset = node.getTextRange().getStartOffset();
final List<FragmentOffsets> fragments = FStringParser.parse(node.getText());
boolean hasUnclosedBrace = false;
for (FragmentOffsets fragment : fragments) {
if (fragment.getLeftBraceOffset() + 1 >= fragment.getContentEndOffset()) {
report(fragment.getContentRange().shiftRight(nodeOffset), "Empty expressions are not allowed inside f-strings");
}
if (fragment.getRightBraceOffset() == -1) {
hasUnclosedBrace = true;
}
}
if (hasUnclosedBrace) {
report(TextRange.from(nodeInfo.getContentRange().getEndOffset(), 0).shiftRight(nodeOffset), "'}' is expected");
}
}
}

View File

@@ -1,6 +1,6 @@
f'{<error descr="Empty expressions are not allowed inside f-strings"></error>}'
f'{<error descr="Empty expressions are not allowed inside f-strings"></error>'
<error descr="Missing closing quote [']">f'{<error descr="Empty expressions are not allowed inside f-strings"></error></error>
f'{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error>'
<error descr="Missing closing quote [']">f'{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error></error>
f'{<error descr="Empty expressions are not allowed inside f-strings"></error>!r}'
f'{<error descr="Empty expressions are not allowed inside f-strings"></error>:2.3}'
f'{42:2.{<error descr="Empty expressions are not allowed inside f-strings"></error>}}'

View File

@@ -0,0 +1,10 @@
f'{42}'
f'{42!r}'
f'{42!r:03}'
f'{42:03}'
f'{42!r:{y}.{z}}'
f'{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error>'
f'{42:{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error>'
f'{42!r:{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error>'
f'{{'
f'{{{<error descr="Empty expressions are not allowed inside f-strings"></error><error descr="'}' is expected"></error>'

View File

@@ -305,6 +305,11 @@ public class PythonHighlightingTest extends PyTestCase {
doTest();
}
// PY-20775
public void testFStringMissingRightBrace() {
runWithLanguageLevel(LanguageLevel.PYTHON36, () -> doTest(true, false));
}
// PY-20776
public void testFStringEmptyExpressions() {
runWithLanguageLevel(LanguageLevel.PYTHON36, () -> doTest(true, false));