about summary refs log tree commit diff
path: root/src/etc/generate-deriving-span-tests.py
blob: 2e810d7df97ba2b85b88b89ba441536185441710 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#!/usr/bin/env python

"""
This script creates a pile of UI tests check that all the
derives have spans that point to the fields, rather than the
#[derive(...)] line.

sample usage: src/etc/generate-deriving-span-tests.py
"""

import os
import stat

TEST_DIR = os.path.abspath(
    os.path.join(os.path.dirname(__file__), "../test/ui/derives/")
)

TEMPLATE = """\
// This file was auto-generated using 'src/etc/generate-deriving-span-tests.py'

{error_deriving}
struct Error;
{code}
fn main() {{}}
"""

ENUM_STRING = """
#[derive({traits})]
enum Enum {{
   A(
     Error {errors}
     )
}}
"""
ENUM_STRUCT_VARIANT_STRING = """
#[derive({traits})]
enum Enum {{
   A {{
     x: Error {errors}
   }}
}}
"""
STRUCT_STRING = """
#[derive({traits})]
struct Struct {{
    x: Error {errors}
}}
"""
STRUCT_TUPLE_STRING = """
#[derive({traits})]
struct Struct(
    Error {errors}
);
"""

ENUM_TUPLE, ENUM_STRUCT, STRUCT_FIELDS, STRUCT_TUPLE = range(4)


def create_test_case(type, trait, super_traits, error_count):
    string = [
        ENUM_STRING,
        ENUM_STRUCT_VARIANT_STRING,
        STRUCT_STRING,
        STRUCT_TUPLE_STRING,
    ][type]
    all_traits = ",".join([trait] + super_traits)
    super_traits = ",".join(super_traits)
    error_deriving = "#[derive(%s)]" % super_traits if super_traits else ""

    errors = "\n".join("//~%s ERROR" % ("^" * n) for n in range(error_count))
    code = string.format(traits=all_traits, errors=errors)
    return TEMPLATE.format(error_deriving=error_deriving, code=code)


def write_file(name, string):
    test_file = os.path.join(TEST_DIR, "derives-span-%s.rs" % name)

    # set write permission if file exists, so it can be changed
    if os.path.exists(test_file):
        os.chmod(test_file, stat.S_IWUSR)

    with open(test_file, "w") as f:
        f.write(string)

    # mark file read-only
    os.chmod(test_file, stat.S_IRUSR | stat.S_IRGRP | stat.S_IROTH)


ENUM = 1
STRUCT = 2
ALL = STRUCT | ENUM

traits = {
    "Default": (STRUCT, [], 1),
    "FromPrimitive": (0, [], 0),  # only works for C-like enums
    "Decodable": (0, [], 0),  # FIXME: quoting gives horrible spans
    "Encodable": (0, [], 0),  # FIXME: quoting gives horrible spans
}

for trait, supers, errs in [
    ("Clone", [], 1),
    ("PartialEq", [], 2),
    ("PartialOrd", ["PartialEq"], 1),
    ("Eq", ["PartialEq"], 1),
    ("Ord", ["Eq", "PartialOrd", "PartialEq"], 1),
    ("Debug", [], 1),
    ("Hash", [], 1),
]:
    traits[trait] = (ALL, supers, errs)

for trait, (types, super_traits, error_count) in traits.items():

    def mk(ty, t=trait, st=super_traits, ec=error_count):
        return create_test_case(ty, t, st, ec)

    if types & ENUM:
        write_file(trait + "-enum", mk(ENUM_TUPLE))
        write_file(trait + "-enum-struct-variant", mk(ENUM_STRUCT))
    if types & STRUCT:
        write_file(trait + "-struct", mk(STRUCT_FIELDS))
        write_file(trait + "-tuple-struct", mk(STRUCT_TUPLE))