mirror of
https://github.com/Instagram/LibCST.git
synced 2025-12-23 10:35:53 +00:00
173 lines
6.3 KiB
Python
173 lines
6.3 KiB
Python
# Copyright (c) Facebook, Inc. and its affiliates.
|
|
#
|
|
# This source code is licensed under the MIT license found in the
|
|
# LICENSE file in the root directory of this source tree.
|
|
|
|
# pyre-strict
|
|
from typing import Any
|
|
|
|
import libcst as cst
|
|
from libcst import CodeRange, parse_expression
|
|
from libcst._nodes.tests.base import CSTNodeTest
|
|
from libcst.testing.utils import data_provider
|
|
|
|
|
|
class DictTest(CSTNodeTest):
|
|
@data_provider(
|
|
[
|
|
# zero-element dict
|
|
{
|
|
"node": cst.Dict([]),
|
|
"code": "{}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 2)),
|
|
},
|
|
# one-element dict, sentinel comma value
|
|
{
|
|
"node": cst.Dict([cst.DictElement(cst.Name("k"), cst.Name("v"))]),
|
|
"code": "{k: v}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 6)),
|
|
},
|
|
{
|
|
"node": cst.Dict([cst.StarredDictElement(cst.Name("expanded"))]),
|
|
"code": "{**expanded}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 12)),
|
|
},
|
|
# two-element dict, sentinel comma value
|
|
{
|
|
"node": cst.Dict(
|
|
[
|
|
cst.DictElement(cst.Name("k1"), cst.Name("v1")),
|
|
cst.DictElement(cst.Name("k2"), cst.Name("v2")),
|
|
]
|
|
),
|
|
"code": "{k1: v1, k2: v2}",
|
|
"parser": None,
|
|
"expected_position": CodeRange((1, 0), (1, 16)),
|
|
},
|
|
# custom whitespace between brackets
|
|
{
|
|
"node": cst.Dict(
|
|
[cst.DictElement(cst.Name("k"), cst.Name("v"))],
|
|
lbrace=cst.LeftCurlyBrace(
|
|
whitespace_after=cst.SimpleWhitespace("\t")
|
|
),
|
|
rbrace=cst.RightCurlyBrace(
|
|
whitespace_before=cst.SimpleWhitespace("\t\t")
|
|
),
|
|
),
|
|
"code": "{\tk: v\t\t}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 9)),
|
|
},
|
|
# with parenthesis
|
|
{
|
|
"node": cst.Dict(
|
|
[cst.DictElement(cst.Name("k"), cst.Name("v"))],
|
|
lpar=[cst.LeftParen()],
|
|
rpar=[cst.RightParen()],
|
|
),
|
|
"code": "({k: v})",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 1), (1, 7)),
|
|
},
|
|
# starred element
|
|
{
|
|
"node": cst.Dict(
|
|
[
|
|
cst.StarredDictElement(cst.Name("one")),
|
|
cst.StarredDictElement(cst.Name("two")),
|
|
]
|
|
),
|
|
"code": "{**one, **two}",
|
|
"parser": None,
|
|
"expected_position": CodeRange((1, 0), (1, 14)),
|
|
},
|
|
# custom comma on DictElement
|
|
{
|
|
"node": cst.Dict(
|
|
[cst.DictElement(cst.Name("k"), cst.Name("v"), comma=cst.Comma())]
|
|
),
|
|
"code": "{k: v,}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 7)),
|
|
},
|
|
# custom comma on StarredDictElement
|
|
{
|
|
"node": cst.Dict(
|
|
[cst.StarredDictElement(cst.Name("expanded"), comma=cst.Comma())]
|
|
),
|
|
"code": "{**expanded,}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 13)),
|
|
},
|
|
# custom whitespace on DictElement
|
|
{
|
|
"node": cst.Dict(
|
|
[
|
|
cst.DictElement(
|
|
cst.Name("k"),
|
|
cst.Name("v"),
|
|
whitespace_before_colon=cst.SimpleWhitespace("\t"),
|
|
whitespace_after_colon=cst.SimpleWhitespace("\t\t"),
|
|
)
|
|
]
|
|
),
|
|
"code": "{k\t:\t\tv}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 8)),
|
|
},
|
|
# custom whitespace on StarredDictElement
|
|
{
|
|
"node": cst.Dict(
|
|
[
|
|
cst.DictElement(
|
|
cst.Name("k"), cst.Name("v"), comma=cst.Comma()
|
|
),
|
|
cst.StarredDictElement(
|
|
cst.Name("expanded"),
|
|
whitespace_before_value=cst.SimpleWhitespace(" "),
|
|
),
|
|
]
|
|
),
|
|
"code": "{k: v,** expanded}",
|
|
"parser": parse_expression,
|
|
"expected_position": CodeRange((1, 0), (1, 19)),
|
|
},
|
|
# missing spaces around dict is always okay
|
|
{
|
|
"node": cst.GeneratorExp(
|
|
cst.Name("a"),
|
|
cst.CompFor(
|
|
cst.Name("b"),
|
|
cst.Dict([cst.DictElement(cst.Name("k"), cst.Name("v"))]),
|
|
ifs=[
|
|
cst.CompIf(
|
|
cst.Name("c"),
|
|
whitespace_before=cst.SimpleWhitespace(""),
|
|
)
|
|
],
|
|
whitespace_after_in=cst.SimpleWhitespace(""),
|
|
),
|
|
),
|
|
"parser": parse_expression,
|
|
"code": "(a for b in{k: v}if c)",
|
|
},
|
|
]
|
|
)
|
|
def test_valid(self, **kwargs: Any) -> None:
|
|
self.validate_node(**kwargs)
|
|
|
|
@data_provider(
|
|
[
|
|
# unbalanced Dict
|
|
{
|
|
"get_node": lambda: cst.Dict([], lpar=[cst.LeftParen()]),
|
|
"expected_re": "left paren without right paren",
|
|
}
|
|
]
|
|
)
|
|
def test_invalid(self, **kwargs: Any) -> None:
|
|
self.assert_invalid(**kwargs)
|