1
1
from __future__ import annotations
2
2
3
3
import ast
4
- import re
5
4
from copy import copy
6
5
from keyword import kwlist
7
6
from pathlib import Path
15
14
rewrite_changed_nodes ,
16
15
)
17
16
18
- CAMEL_CASE_SUB_PATTERN = re .compile (r"(?<!^)(?=[A-Z])" )
19
-
20
17
21
18
@click .command ()
22
19
@click .argument ("paths" , nargs = - 1 , type = click .Path (exists = True ))
23
- def rewrite_camel_case_props (paths : list [str ]) -> None :
24
- """Rewrite camelCase props to snake_case"""
25
-
20
+ def rewrite_props (paths : list [str ]) -> None :
21
+ """Rewrite snake_case props to camelCase within <PATHS>."""
26
22
for p in map (Path , paths ):
23
+ # Process each file or recursively process each Python file in directories
27
24
for f in [p ] if p .is_file () else p .rglob ("*.py" ):
28
25
result = generate_rewrite (file = f , source = f .read_text (encoding = "utf-8" ))
29
26
if result is not None :
30
27
f .write_text (result )
31
28
32
29
33
30
def generate_rewrite (file : Path , source : str ) -> str | None :
34
- tree = ast .parse (source )
31
+ """Generate the rewritten source code if changes are detected"""
32
+ tree = ast .parse (source ) # Parse the source code into an AST
35
33
36
- changed = find_nodes_to_change (tree )
34
+ changed = find_nodes_to_change (tree ) # Find nodes that need to be changed
37
35
if not changed :
38
- return None
36
+ return None # Return None if no changes are needed
39
37
40
- new = rewrite_changed_nodes (file , source , tree , changed )
38
+ new = rewrite_changed_nodes (
39
+ file , source , tree , changed
40
+ ) # Rewrite the changed nodes
41
41
return new
42
42
43
43
44
44
def find_nodes_to_change (tree : ast .AST ) -> list [ChangedNode ]:
45
+ """Find nodes in the AST that need to be changed"""
45
46
changed : list [ChangedNode ] = []
46
47
for el_info in find_element_constructor_usages (tree ):
48
+ # Check if the props need to be rewritten
47
49
if _rewrite_props (el_info .props , _construct_prop_item ):
50
+ # Add the changed node to the list
48
51
changed .append (ChangedNode (el_info .call , el_info .parents ))
49
52
return changed
50
53
51
54
52
55
def conv_attr_name (name : str ) -> str :
53
- new_name = CAMEL_CASE_SUB_PATTERN .sub ("_" , name ).lower ()
54
- return f"{ new_name } _" if new_name in kwlist else new_name
56
+ """Convert snake_case attribute name to camelCase"""
57
+ # Return early if the value is a Python keyword
58
+ if name in kwlist :
59
+ return name
60
+
61
+ # Return early if the value is not snake_case
62
+ if "_" not in name :
63
+ return name
64
+
65
+ # Split the string by underscores
66
+ components = name .split ("_" )
67
+
68
+ # Capitalize the first letter of each component except the first one
69
+ # and join them together
70
+ return components [0 ] + "" .join (x .title () for x in components [1 :])
55
71
56
72
57
73
def _construct_prop_item (key : str , value : ast .expr ) -> tuple [str , ast .expr ]:
74
+ """Construct a new prop item with the converted key and possibly modified value"""
58
75
if key == "style" and isinstance (value , (ast .Dict , ast .Call )):
76
+ # Create a copy of the value to avoid modifying the original
59
77
new_value = copy (value )
60
78
if _rewrite_props (
61
79
new_value ,
62
80
lambda k , v : (
63
81
(k , v )
64
- # avoid infinite recursion
82
+ # Avoid infinite recursion
65
83
if k == "style"
66
84
else _construct_prop_item (k , v )
67
85
),
68
86
):
87
+ # Update the value if changes were made
69
88
value = new_value
70
89
else :
90
+ # Convert the key to camelCase
71
91
key = conv_attr_name (key )
72
92
return key , value
73
93
@@ -76,12 +96,15 @@ def _rewrite_props(
76
96
props_node : ast .Dict | ast .Call ,
77
97
constructor : Callable [[str , ast .expr ], tuple [str , ast .expr ]],
78
98
) -> bool :
99
+ """Rewrite the props in the given AST node using the provided constructor"""
100
+ did_change = False
79
101
if isinstance (props_node , ast .Dict ):
80
- did_change = False
81
102
keys : list [ast .expr | None ] = []
82
103
values : list [ast .expr ] = []
104
+ # Iterate over the keys and values in the dictionary
83
105
for k , v in zip (props_node .keys , props_node .values ):
84
106
if isinstance (k , ast .Constant ) and isinstance (k .value , str ):
107
+ # Construct the new key and value
85
108
k_value , new_v = constructor (k .value , v )
86
109
if k_value != k .value or new_v is not v :
87
110
did_change = True
@@ -90,20 +113,22 @@ def _rewrite_props(
90
113
keys .append (k )
91
114
values .append (v )
92
115
if not did_change :
93
- return False
116
+ return False # Return False if no changes were made
94
117
props_node .keys = keys
95
118
props_node .values = values
96
119
else :
97
120
did_change = False
98
121
keywords : list [ast .keyword ] = []
122
+ # Iterate over the keywords in the call
99
123
for kw in props_node .keywords :
100
124
if kw .arg is not None :
125
+ # Construct the new keyword argument and value
101
126
kw_arg , kw_value = constructor (kw .arg , kw .value )
102
127
if kw_arg != kw .arg or kw_value is not kw .value :
103
128
did_change = True
104
129
kw = ast .keyword (arg = kw_arg , value = kw_value )
105
130
keywords .append (kw )
106
131
if not did_change :
107
- return False
132
+ return False # Return False if no changes were made
108
133
props_node .keywords = keywords
109
134
return True
0 commit comments