-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathpreconfig.py
executable file
·225 lines (199 loc) · 7.3 KB
/
preconfig.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
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
import argparse
import csv
import datetime
import getpass
import os
from jinja2 import Environment, FileSystemLoader
from pyedgeconnect import Orchestrator
# Parse runtime arguments
parser = argparse.ArgumentParser()
parser.add_argument(
"-c",
"--csv",
help="Specify source csv file for preconfigs",
type=str,
required=True,
)
parser.add_argument(
"-u",
"--upload",
help="Upload created valid preconfigs to Orchestrator",
action=argparse.BooleanOptionalAction,
)
parser.add_argument(
"-aa",
"--autoapply",
help="Mark preconfigs for auto-approve",
action=argparse.BooleanOptionalAction,
)
parser.add_argument(
"-j",
"--jinja",
help="specify source jinja2 template",
type=str,
default="ec_preconfig_template.jinja2",
)
parser.add_argument(
"-o",
"--orch",
help="specify Orchestrator URL",
type=str,
)
args = parser.parse_args()
# Set Orchestrator FQDN/IP via arguments, environment variable,
# or user input
if vars(args)["orch"] is not None:
orch_url = vars(args)["orch"]
elif os.getenv("ORCH_URL") is not None:
orch_url = os.getenv("ORCH_URL")
else:
orch_url = input("Orchstrator IP or FQDN: ")
# Set Orchestrator API Key via environment variable or user input
if os.getenv("ORCH_API_KEY") is not None:
orch_api_key = os.getenv("ORCH_API_KEY")
else:
orch_api_key_input = getpass.getpass(
"Orchstrator API Key (enter to skip): "
)
if len(orch_api_key_input) == 0:
orch_api_key = None
# Set user and password if present in environment variable
orch_user = os.getenv("ORCH_USER")
orch_pw = os.getenv("ORCH_PASSWORD")
else:
orch_api_key = orch_api_key_input
# Instantiate Orchestrator with ``log_console`` enabled for
# printing log messages to terminal
orch = Orchestrator(
orch_url,
api_key=orch_api_key,
log_console=True,
verify_ssl=False,
)
# If not using API key, login to Orchestrator with username/password
if orch_api_key is None:
# If username/password not in environment variables, prompt user
if orch_user is None:
orch_user = input("Enter Orchestrator username: ")
orch_pw = getpass.getpass("Enter Orchestrator password: ")
# Check if multi-factor authentication required
mfa_prompt = input("Are you using MFA for this user (y/n)?: ")
if mfa_prompt == "y":
orch.send_mfa(orch_user, orch_pw, temp_code=False)
token = input("Enter MFA token: ")
else:
token = ""
# Login to Orchestrator
confirm_auth = orch.login(orch_user, orch_pw, mfacode=token)
# Check that user/pass authentication works before proceeding
if confirm_auth:
pass
else:
print("Authentication to Orchestrator Failed")
exit()
# If API key specified, check that key is valid before proceeding
else:
confirm_auth = orch.get_orchestrator_hello()
if confirm_auth != "There was an internal server error.":
pass
else:
print("Authentication to Orchestrator Failed")
exit()
# Specify CSV file for generating preconfigs
# This is a mandatory runtime argument
if vars(args)["csv"] is not None:
csv_filename = vars(args)["csv"]
else:
print("Source CSV file not specified, exiting")
exit()
# Setting if configs should be uploaded to Orchestrator, argument
# defaults to False if not specified
upload_to_orch = vars(args)["upload"]
# Setting if discovered appliance with matching serial number or tag
# will be automatically approved and deployed with corresponding
# preconfig. Argument defaults to False if not specified
auto_apply = vars(args)["autoapply"]
# Specify alternate Jinja2 template file for generating preconfig
# in the templates directory. Otherwise use default template.
ec_template_file = vars(args)["jinja"]
# Retrieve Jinja2 template for generating EdgeConnect Preconfig YAML
# Setting ``trim_blocks`` and ``lstrip_blocks`` reduces excessive
# whitepsace from the jinja template conditionals etc.
env = Environment(
loader=FileSystemLoader("templates"),
trim_blocks=True,
lstrip_blocks=True,
)
ec_template = env.get_template(ec_template_file)
# Local directory for configuration outputs
output_directory = "preconfig_outputs/"
if not os.path.exists(output_directory):
os.makedirs(output_directory)
# Open CSV file with configuration data
with open(csv_filename, encoding="utf-8-sig") as csvfile:
csv_dict = csv.DictReader(csvfile)
# Set initial row number for row identification of data
# First row is headers
row_number = 2
# Generate Edge Connect YAML preconfig for each row in data
for row in csv_dict:
# Render CSV values through the Jinja template
yaml_preconfig = ec_template.render(data=row)
# Set value for serial number if provided
appliance_serial = row.get("serial_number")
if appliance_serial is None:
appliance_serial = ""
else:
pass
# Validate preconfig via Orchestrator
validate = orch.validate_preconfig(
preconfig_name=row["hostname"],
yaml_preconfig=yaml_preconfig,
auto_apply=auto_apply,
)
# Write local YAML file to see resulting YAML file locally
# whether validate passes or fails
yaml_filename = f'{row["hostname"]}_preconfig.yml'
with open(output_directory + yaml_filename, "w") as preconfig_file:
write_data = preconfig_file.write(yaml_preconfig)
# If the validate function passes on Orchestrator, move on
# to check if uploading to Orchestrator option selected
if validate.status_code == 200:
# If upload option was chosen, upload preconfig to
# Orchestrator with selected auto-apply settings
if upload_to_orch is True:
# In this example the appliance hostname from the CSV
# data (row["hostname"]) is used both for the name of
# the preconfig to appear in Orchestrator, as well as
# the tag on the preconfig that could be used to match
# against a discovered appliance
# Additionally a comment is added with the current
# date
comment_timestamp = datetime.date.today().strftime("%d %B %Y")
orch.create_preconfig(
preconfig_name=row["hostname"],
yaml_preconfig=yaml_preconfig,
auto_apply=auto_apply,
tag=row["hostname"],
serial_number=appliance_serial,
comment=f"Created/Uploaded @ {comment_timestamp}",
)
print(f'Posted EC Preconfig {row["hostname"]}')
else:
pass
else:
print(
f'Preconfig for {row["hostname"]}'
f" failed validation | error: {validate.text}"
)
# Write local YAML file of failed config for reference
yaml_filename = f'{row["hostname"]}_preconfig-FAILED.yml'
with open(output_directory + yaml_filename, "w") as preconfig_file:
write_data = preconfig_file.write(yaml_preconfig)
# Increment row number when iterating to next row in CSV
row_number += 1
# if not using API key, logout from Orchestrator
if orch_api_key is None:
orch.logout()
else:
pass