-
Notifications
You must be signed in to change notification settings - Fork 5
/
fgt-backup.py
218 lines (169 loc) · 6.41 KB
/
fgt-backup.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
import os
import csv
import sys
import requests
from datetime import datetime
# The req object is used to make https requests
req = requests.session()
# Uncomment these two lines below to disable SSL certificate warnings
# requests.packages.urllib3.disable_warnings()
# req.verify = False
# Global variables
DATE = datetime.now().strftime('%m-%d-%Y') # Today's date in the format mm-dd-yyyy
CURRENT_DIR = os.path.dirname(os.path.realpath(__file__)) # Path to the directory where this script is located
BKP_FOLDER = os.path.join(CURRENT_DIR, 'backups', DATE) # Path to the backups folder
LOGS_FOLDER = os.path.join(CURRENT_DIR, 'logs') # Path to the logs folder
online_ip, error_message = '',''
bkp_fail = []
def main():
# Read the fortigates.csv file
fortigates = read_fortigates()
# Create folders for backups and logs if they don't exist
create_folders()
# Create log file
log = create_log()
log.write(f'Backup log - {DATE}\n')
# Backup each Fortigate
for fgt in fortigates:
# Clear error_message variable before each backup
global error_message
error_message = ''
print('\n========================================')
print(f'Fortigate: {fgt["name"]}')
log.write('\n========================================\n')
log.write(f'Fortigate: {fgt["name"]}\n')
# Call the main backup function
bkp_ok = backup(fgt)
# Check if the backup was successful
if bkp_ok:
print('Backup successful!')
log.write(f'Fortigate online on IP: {online_ip}\n')
log.write('Backup successful!\n')
# If not, check if the Fortigate is offline or if there was an error
else:
if online_ip == '':
print('Fortigate offline!')
log.write('Fortigate offline!\n')
elif error_message != '':
print(f'Error message: {error_message}')
log.write(f'Error message: {error_message}\n')
print('Backup failed!')
log.write('Backup failed!\n')
bkp_fail.append(fgt['name'])
print('========================================\n')
log.write('========================================\n')
# Print the list of failed backups if there are any
if len(bkp_fail) > 0:
print('List of failed backups:')
log.write('\nList of failed backups:\n')
for fgt in bkp_fail:
print(fgt)
log.write(f'{fgt}\n')
print(f'\nCount: {len(bkp_fail)}\n')
log.write(f'\nCount: {len(bkp_fail)}\n')
log.close()
print('Backup finished!')
input('Press ENTER to exit...')
sys.exit()
def read_fortigates():
try:
# Read the fortigates.csv file and convert it to a list of dictionaries
with open(os.path.join(CURRENT_DIR, 'fortigates.csv'), 'r') as file:
fortigates = list(csv.DictReader(file, delimiter=','))
except FileNotFoundError as e:
print(f'File fortigates.csv not found: {e}')
input('Press ENTER to exit...')
sys.exit()
except Exception as e:
print(f'Error reading fortigates.csv file: {e}')
input('Press ENTER to exit...')
sys.exit()
return fortigates
def create_folders():
try:
# Create the backups and logs folders if they don't exist
# exist_ok=True prevents the function from raising an exception if the folder already exists
os.makedirs(BKP_FOLDER, exist_ok=True)
os.makedirs(LOGS_FOLDER, exist_ok=True)
except Exception as e:
print(f'Error creating folders: {e}')
input('Press ENTER to exit...')
sys.exit()
def create_log():
try:
# Create log file and remain it open to write
return open(os.path.join(LOGS_FOLDER, f'bkp-{DATE}.log'), 'a')
except Exception as e:
print(f'Error creating the log file: {e}')
input('Press ENTER to exit...')
sys.exit()
def backup(fgt):
# Mount the backup URL
url = mount_url(fgt)
if not url: return False
# Perform the backup
print(f'Fortigate online on {online_ip}, backing up...')
try:
bkp_data = req.post(url, headers={'Authorization' : 'Bearer ' + str(fgt["token"])})
except Exception as e:
global error_message
error_message = str(e)
return False
# Save and check the backup file
file_ok = save_and_check_file(fgt['name'], bkp_data)
return file_ok
def ping(ip):
# Do a simple request to the Fortigate to check if it is online
try:
req.get(f'https://{ip}', timeout=3)
return True
except:
return False
def check_online_ip(fgt):
# Access the global variable online_ip
global online_ip
# Check if the Fortigate is online on primary IP
if ping(fgt['ip_1']):
online_ip = fgt['ip_1']
return online_ip
# If not, check if the Fortigate has a secondary IP and if it is online
elif fgt['ip_2'] != '' and ping(fgt['ip_2']):
online_ip = fgt['ip_2']
return online_ip
# If the Fortigate is not available on any IP, return False
online_ip = ''
return False
def mount_url(fgt):
# URI to backup the Fortigate
URI = '/api/v2/monitor/system/config/backup?scope=global'
# Check if the Fortigate is online on both IPs
is_online = check_online_ip(fgt)
if is_online:
# If it is online, mount the URL to backup the Fortigate
return f'https://{is_online}{URI}'
else:
return ''
def save_and_check_file(name, data):
# Access the global variable error_message
global error_message
# Path to the backup file
file_path = os.path.join(BKP_FOLDER, f'{name}-bkp-{DATE}.conf')
try:
# Save the backup data to a file
with open(file_path, 'wb') as file:
for line in data:
file.write(line)
# Check if the file is a valid Fortigate configuration file
with open(file_path, 'r') as file:
first_line = file.readline()
if first_line.startswith('#config'):
return True
# If the file is not valid, delete it and change the error_message variable
os.remove(file_path)
error_message = 'Invalid backup file'
return False
except Exception as e:
error_message = str(e)
return False
if __name__ == '__main__':
main()