-
-
Notifications
You must be signed in to change notification settings - Fork 6
/
generator.py
executable file
·208 lines (169 loc) · 7.33 KB
/
generator.py
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
"""
3MS Configuration Generator Script
Currently supports:
- URL-downloaded configurations
- File-downloaded configurations
- Main MCU configuration
- MMU MCU configuration
- Multi-MMU MCU configuration
- Per-stepper TMC driver selection
- Configurations with or without TMC drivers
"""
import sys
import requests
import argparse
import configparser
def format_pin(pin, mcu):
prefixes = ''
if '!' in pin: prefixes += '!'
if '~' in pin: prefixes += '~'
if '^' in pin: prefixes += '^'
pin = pin.replace('!','').replace('~','').replace('^','')
return f'{prefixes}{mcu}:{pin}'
def read_config_from_url(url):
resp = requests.get(url)
if resp.status_code != 200: return
return resp.text
def read_config_from_file(fname):
with open(fname) as file:
config = file.read()
return config
def preprocess_config(config):
updated_config = []
in_section = False
is_commented = False
for line in config.splitlines():
is_commented = False
if line.startswith('#'):
is_commented = True
line = line[1:]
if line.startswith('['):
in_section = True
if line.strip() == '' and in_section:
in_section = False
if is_commented and not in_section:
line = f'#{line}'
if in_section:
updated_config.append(line)
else:
updated_config.append('\n')
return '\n'.join(updated_config)
def get_parser(config):
parser = configparser.RawConfigParser(strict=False, comment_prefixes=(';', '#'))
parser.read_string(config)
return parser
def extract_steppers(parser, rotation_distance=25, microsteps=16):
sections = parser.sections()
stepper_sections = {}
for section in sections:
step_pin = parser.get(section, 'step_pin', fallback=None)
dir_pin = parser.get(section, 'dir_pin', fallback=None)
en_pin = parser.get(section, 'enable_pin', fallback=None)
if step_pin and dir_pin and en_pin:
stepper_sections[section] = {
'stepper_conf': {
'step_pin': step_pin,
'dir_pin': dir_pin,
'enable_pin': en_pin,
'rotation_distance': rotation_distance,
'microsteps': parser.get(section, 'microsteps', fallback=microsteps),
},
}
return stepper_sections
def extract_tmc(parser, steppers):
for section in parser.sections():
if section.split()[-1] in steppers:
if 'tmc' in section:
tmc_options = {}
for option in parser.options(section):
tmc_options[option] = parser.get(section, option)
tmc_options['interpolate'] = parser.get(section, 'interpolate', fallback='True')
tmc_options['sense_resistor'] = parser.get(section, 'sense_resistor', fallback='0.110')
steppers[section.split()[-1]][section.split()[0]] = tmc_options
return steppers
def format_all_pins(steppers, mcu_name, tmc_selections=None):
for name, data in steppers.items():
data['stepper_conf']['step_pin'] = format_pin(data['stepper_conf']['step_pin'], mcu_name)
data['stepper_conf']['dir_pin'] = format_pin(data['stepper_conf']['dir_pin'], mcu_name)
data['stepper_conf']['enable_pin'] = format_pin(data['stepper_conf']['enable_pin'], mcu_name)
if tmc_selections:
for opt, val in data[tmc_selections[name]].items():
if 'pin' in opt:
data[tmc_selections[name]][opt] = format_pin(val, mcu_name)
return steppers
def generate_mmu_hardware(steppers, tmc_selections=None):
writer = configparser.RawConfigParser(strict=False, comment_prefixes=(';', '#'), delimiters=(':',))
i = 0
for stepper, data in steppers.items():
stepper_name = f'stepper_mmu_gear_{i}' if i > 0 else 'stepper_mmu_gear'
writer.add_section(stepper_name)
for key, val in data['stepper_conf'].items():
writer.set(stepper_name, key, val)
if tmc_selections:
tmc_section = f'{tmc_selections[stepper]} {stepper_name}'
writer.add_section(tmc_section)
for key, val in data[tmc_selections[stepper]].items():
writer.set(tmc_section, key, val)
i += 1
return writer
def main(do_file=False, do_tmc=True):
if do_file:
file = input('File > ')
config = read_config_from_file(file)
else:
url = input('URL > ')
config = read_config_from_url(url)
config = preprocess_config(config)
parser = get_parser(config)
stepper_sections = extract_steppers(parser)
print('Available stepper options: ')
i = 0
stepper_keys = list(stepper_sections.keys())
for key in stepper_sections:
print(f'{i+1}) {key}')
i += 1
print('Please enter which steppers you want, seperated by spaces')
desired_steppers_ids = [int(x) for x in input('Steppers > ').split()]
desired_steppers_names = [stepper_keys[i-1] for i in desired_steppers_ids]
desired_steppers = {}
for key, value in stepper_sections.items():
if key in desired_steppers_names:
desired_steppers[key] = value
tmc_selections = {}
if do_tmc:
desired_steppers = extract_tmc(parser, desired_steppers)
for name, data in desired_steppers.items():
print(f'Stepper {name}: Which TMC driver will you use?')
i = 0
selection_opts = []
for subdata in data:
if 'tmc' in subdata:
print(f'{i+1}) {subdata}')
selection_opts.append(subdata)
i += 1
selection_id = int(input('Enter selection > '))
selection_name = selection_opts[selection_id-1]
tmc_selections[name] = selection_name
print('Please enter the MCU name.\nIf this is an external mainboard, it should be "mmu".\nIf it is your existing mainboard, it should be blank.\nIf it is an additional mainboard to support extra steppers, it should be "mmu1", "mmu2", etc.')
mcu_name = input('MCU Name > ').strip()
if mcu_name != '':
desired_steppers = format_all_pins(desired_steppers, mcu_name, tmc_selections)
# ---------- mmu.cfg ----------
writer = generate_mmu_hardware(desired_steppers, tmc_selections)
with open('mmu_hardware.cfg', 'w') as file:
writer.write(file)
writer = configparser.RawConfigParser(strict=False, comment_prefixes=(';', '#'))
if mcu_name:
mcu_section = f'mcu {mcu_name}'
writer.add_section(mcu_section)
writer.set(mcu_section, 'serial', '/dev/serial/by-id/MCU-PATH')
with open('mmu.cfg', 'w') as file:
writer.write(file)
if not mcu_name:
file.write('# mmu.cfg can be empty, as this is a main MCU configuration')
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='Generate a 3MS configuration for Happy Hare.')
parser.add_argument('--file', action='store_true', help='Read configuration from a file instead of a URL')
parser.add_argument('--no-tmc', action='store_true', help='Disable TMC driver configuration')
args = parser.parse_args()
sys.exit(main(args.file, not args.no_tmc))