This project is about exporting the Access Control Policies from a Cisco Firepower Management Center as a CSV file.
Please note the copyrights and licenses of the individual products and projects used. Accordingly, the rights are held by the original developers. If you have any questions or problems, please contact me, maybe we can find a solution together :)
As the regular documentation of the switched firewall rules in a Cisco Firepower Management Center takes a lot of time, this process, which takes place regularly at various customers, is to be automated. For example, a process in which an inventory list of the rules can be compared with the currently activated rules can take several hours.
By automating the process, this hour-long effort can be simplified to around 5 minutes. To do this, a Python script is executed on the customer's jump host, which contacts the FMC and dials in using the administrator login data. A CSV file is generated by a database query, which contains some basic information relevant to us. However, as Python must be installed for this, which may not be very advisable for security reasons, this process can also be optimized. Using the Python program pyInstaller, a stand-alone Microsoft executable file can be created from a script that requires Python to be executed and installed on the host, which also works without requiring a Python installation on the host.
Although a single executable file is much easier to handle, it can also make sense to use a "normal" program that refers to subdirectories with binary files. This is because the libraries used by the executable can be viewed by the integrated virus scanner, especially for KRITIS customers. This means that the virus protection system can still report and intervene in the event of a compromised library. As an additional security measure, the executable file can also be provided with a certificate from the internal CA.
The script on which the executable is based runs under the BSD 3-Clause "New" or "Revised" license. This means that the script can also be used in commercial areas. More information about the script from the original developer:
https://developer.cisco.com/codeexchange/github/repo/raghukul-cisco/csvExportFirepower/
The script itself:
import re
import getpass
import sys
import warnings
import requests
import time
import json
from fireREST import FMC
from netaddr import *
import ipaddress
import datetime
#Global Variable
globalRules = {}
def acpCSV(fmc, selection):
print ("Inside ACP")
for policy in selection:
ac_policy = policy
policyFile = open(ac_policy + ".csv", "w")
rules = fmc.policy.accesspolicy.accessrule.get(policy)
refRules = {}
policyFile.write("#, name,sourceZones, destZones, sourceNetworks, destNetworks, sourcePorts,destPorts, URL, APP, VLAN, Action, Enable, ipsPolicy, variableSet, filePolicy, logBegin, logEnd, sendEventsToFMC, syslogConfig, TimeRangeObject, Comments\n")
pos = 0
for ele in rules:
#print("##########################")
pos = pos + 1
#Atomic mandatory fields in policies
try:
ips = ele['ipsPolicy']['name']
except KeyError:
ips = "NULL"
try:
file = ele['filePolicy']['name']
except KeyError:
file = "NULL"
try:
variable = ele['variableSet']['name']
except KeyError:
variable = "NULL"
try:
fmcEvents = ele['sendEventsToFMC']
except KeyError:
fmcEvents = ele['sendEventsToFMC']
try:
begin = ele['logBegin']
except KeyError:
begin = "NULL"
try:
end = ele['logEnd']
except KeyError:
end = "NULL"
try:
syslog = ele['enableSyslog']
except KeyError:
syslog = "NULL"
enable = ele['enabled']
#TimeBased Objects
if ("timeRangeObjects" in ele.keys()):
for local in ele['timeRangeObjects']:
trObj = local['name']
else:
trObj = "NULL"
#Check for time Range ACL if present
#Zones Extraction
sZones = []
dZones = []
try:
temp = ele['sourceZones']
for key,value in temp.items():
for local in value:
sZones.append(local['name'])
except KeyError:
sZones.append("Any")
try:
temp = ele['destinationZones']
for key,value in temp.items():
for local in value:
dZones.append(local['name'])
except KeyError:
dZones.append("Any")
#Network Extraction
sNetwork = []
dNetwork = []
try:
temp = ele['sourceNetworks']
for key, value in temp.items():
if ('literals' in key):
for local in value:
sNetwork.append(local['value'])
if ('objects' in key):
for local in value:
sNetwork.append(local['name'])
except KeyError:
sNetwork.append("Any")
try:
temp = ele['destinationNetworks']
for key, value in temp.items():
if ('literals' in key):
for local in value:
dNetwork.append(local['value'])
if ('objects' in key):
for local in value:
dNetwork.append(local['name'])
except KeyError:
dNetwork.append("Any")
print (ele['name'], sNetwork, dNetwork)
#Port Extraction
sPort = []
dPort = []
try:
temp = ele['sourcePorts']
for key, value in temp.items():
for local in value:
if ('PortLiteral' in local['type']):
sPort.append(local['type'])
else:
sPort.append(local['name'])
except KeyError:
sPort.append("Any")
try:
temp = ele['destinationPorts']
for key, value in temp.items():
for local in value:
if ('PortLiteral' in local['type']):
dPort.append(local['port'])
else:
dPort.append(local['name'])
except KeyError:
dPort.append("Any")
#Applications and URL
app = []
url = []
try:
temp = (ele['urls'])
for key,value in temp.items():
if ("urlCategoriesWithReputation" in key):
for local in value:
url.append(local['category']['name'])
if ("literals" in key):
for local in value:
url.append(local['url'])
if ("objects" in key):
for local in value:
url.append(local['name'])
except KeyError:
url.append("Any")
try:
temp = ele['applications']
for key, value in temp.items():
for local in value:
if ("tags" in local.keys()):
item = local['tags']
for ite in item:
app.append(ite['name'])
if ("Application" in local['type']):
app.append(local['name'])
except KeyError:
app.append("Any")
#VLAN Tags
vlan = []
try:
temp = ele['vlanTags']
for key, value in temp.items():
for local in value:
if ('VlanTagLiteral' in local['type']):
vlan.append(local['startTag'])
else:
vlan.append(local['name'])
except KeyError:
vlan.append("Any")
#Comment History
comment = []
try:
temp = ele['commentHistoryList']
comment.append(temp)
except KeyError:
comment.append("Not Applicable")
# Write CSV Header
refRules[pos] = [pos, ele['name'], sZones, dZones, sNetwork, dNetwork, sPort, dPort, url, app, vlan, ele['action'], enable, ips, variable,file,begin, end, fmcEvents, syslog,trObj,comment]
print("Writing rule #{0} to CSV...".format(pos))
policyFile.write("{0}, {1}, {2}, {3}, {4}, {5}, {6}, {7}, {8}, {9}, {10}, {11}, {12}, {13}, {14}, {15}, {16}, {17}, {18}, {19}, {20}, {21}\n".format(pos,ele['name'],';'.join(sZones), ';'.join(dZones),';'.join(sNetwork), ';'.join(dNetwork),';'.join(sPort), ';'.join(dPort), url, app, vlan, ele['action'],enable, ips, variable, file, begin, end, fmcEvents, syslog, trObj, comment))
globalRules[policy] = refRules
print("File is at: ./{0}.csv".format(ac_policy))
policyFile.close()
def getInput():
hostname = input("Enter the IP Address of the FMC: ")
username = input("Enter the username for the FMC: ")
password = getpass.getpass("Enter the password associated with the username entered: ")
fmc = FMC(hostname=hostname, username=username, password=password, domain='Global')
acPolicies = fmc.policy.accesspolicy.get()
pol = {}
print ("ACP available in global domain: ")
for policy in acPolicies:
pol[policy['name']] = policy['id']
print ("\tName: ", policy['name'])
acp = input("Enter the ACP Name (case sensitive) if you want specific ACP to export(multiple values should be comma seperated). By default all the ACP would be exported, press return for default behaviour: ")
selection = []
if (acp):
temp = acp.split(",")
for local in temp:
local = local.replace(" ", "")
selection.append(pol[local])
else:
for local in acPolicies:
selection.append(local['id'])
acpCSV(fmc, selection)
getInput()
To convert the script into an executable, you need Python 3 on your computer to prepare the program. The following dependencies must then be installed on the PC (also possible without administrator rights) using the following commands:
pip install fireREST
pip install netaddr
pip install datetime
pip install ipaddress
pip install pyinstaller
Then navigate to the folder in which the script.py is stored. A single executable can then be created with the following command:
pyinstaller --onefile .\policyCSV.py
Or alternatively, a program that does not carry all dependencies in one file and instead works with subdirectories:
pyinstaller .\policyCSV.py
This program is now basically ready and can be transferred to the target system. Neither Python nor the dependencies installed above via pip need to be installed on the target system, as these have already been integrated into the executable by the pyInstaller.