-
Notifications
You must be signed in to change notification settings - Fork 68
/
Copy pathnetbox-sync.py
executable file
·148 lines (106 loc) · 4.39 KB
/
netbox-sync.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
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright (c) 2020 - 2023 Ricardo Bartels. All rights reserved.
#
# netbox-sync.py
#
# This work is licensed under the terms of the MIT license.
# For a copy, see file LICENSE.txt included in this
# repository or visit: <https://opensource.org/licenses/MIT>.
self_description = """
Sync objects from various sources to NetBox
"""
from datetime import datetime
from module.common.misc import grab, get_relative_time, do_error_exit
from module.common.cli_parser import parse_command_line
from module.common.logging import setup_logging
from module.netbox.connection import NetBoxHandler
from module.netbox.inventory import NetBoxInventory
from module.sources import instantiate_sources
from module.config.parser import ConfigParser
from module.common.config import CommonConfig
from module.config.file_output import ConfigFileOutput
from module import __version__, __version_date__, __description__
def main():
start_time = datetime.now()
# parse command line
args = parse_command_line(self_description=self_description)
# write out default config file and exit if "generate_config" is defined
ConfigFileOutput(args)
# parse config files and environment variables
config_parse_handler = ConfigParser()
config_parse_handler.add_config_file_list(args.config_files)
config_parse_handler.read_config()
# read common config
common_config = CommonConfig().parse(do_log=False)
# cli option overwrites config file
log_level = grab(args, "log_level", fallback=common_config.log_level)
log_file = None
if common_config.log_to_file is True:
log_file = common_config.log_file
# setup logging
log = setup_logging(log_level, log_file)
# now we are ready to go
log.info(f"Starting {__description__} v{__version__} ({__version_date__})")
for config_file in config_parse_handler.file_list:
log.debug(f"Using config file: {config_file}")
# exit if any parser errors occurred here
config_parse_handler.log_end_exit_on_errors()
# just to print config options to log/console
CommonConfig().parse()
# initialize an empty inventory which will be used to hold and reference all objects
inventory = NetBoxInventory()
# establish NetBox connection
nb_handler = NetBoxHandler()
# if purge was selected we go ahead and remove all items which were managed by this tools
if args.purge is True:
if args.dry_run is True:
do_error_exit("Purge not available with option 'dry_run'")
nb_handler.just_delete_all_the_things()
# that's it, we are done here
exit(0)
# instantiate source handlers and get attributes
log.info("Initializing sources")
sources = instantiate_sources()
# all sources are unavailable
if len(sources) == 0:
log.error("No working sources found. Exit.")
exit(1)
# collect all dependent object classes
log.info("Querying necessary objects from NetBox. This might take a while.")
for source in sources:
nb_handler.query_current_data(source.dependent_netbox_objects)
log.info("Finished querying necessary objects from NetBox")
# resolve object relations within the initial inventory
inventory.resolve_relations()
# initialize basic data needed for syncing
nb_handler.initialize_basic_data()
# loop over sources and patch netbox data
for source in sources:
log.debug(f"Retrieving data from source '{source.name}'")
source.apply()
# add/remove tags to/from all inventory items
inventory.tag_all_the_things(nb_handler)
# update all IP addresses
inventory.query_ptr_records_for_all_ips()
if args.dry_run is True:
log.info("This is a dry run and we stop here. Running time: %s" %
get_relative_time(datetime.now() - start_time))
exit(0)
# update data in NetBox
nb_handler.update_instance()
# prune orphaned objects from NetBox
nb_handler.prune_data()
# delete tags which are not used anymore
nb_handler.delete_unused_tags()
# loop over sources and patch netbox data
for source in sources:
# closing all open connections
source.finish()
# closing NetBox connection
nb_handler.finish()
# finish
log.info("Completed NetBox Sync in %s" % get_relative_time(datetime.now() - start_time))
if __name__ == "__main__":
main()
# EOF