2 ###############################################################################
4 # libloc - A library to determine the location of someone on the Internet #
6 # Copyright (C) 2020 IPFire Development Team <info@ipfire.org> #
8 # This library is free software; you can redistribute it and/or #
9 # modify it under the terms of the GNU Lesser General Public #
10 # License as published by the Free Software Foundation; either #
11 # version 2.1 of the License, or (at your option) any later version. #
13 # This library is distributed in the hope that it will be useful, #
14 # but WITHOUT ANY WARRANTY; without even the implied warranty of #
15 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU #
16 # Lesser General Public License for more details. #
18 ###############################################################################
29 log
= logging
.getLogger("location.export")
33 _location
.NETWORK_FLAG_ANONYMOUS_PROXY
: "A1",
34 _location
.NETWORK_FLAG_SATELLITE_PROVIDER
: "A2",
35 _location
.NETWORK_FLAG_ANYCAST
: "A3",
38 class OutputWriter(object):
42 def __init__(self
, f
, prefix
=None, flatten
=True):
43 self
.f
, self
.prefix
, self
.flatten
= f
, prefix
, flatten
45 # The previously written network
46 self
._last
_network
= None
48 # Immediately write the header
52 def open(cls
, filename
, **kwargs
):
54 Convenience function to open a file
56 f
= open(filename
, cls
.mode
)
58 return cls(f
, **kwargs
)
61 return "<%s f=%s>" % (self
.__class
__.__name
__, self
.f
)
63 def _flatten(self
, network
):
65 Checks if the given network needs to be written to file,
66 or if it is a subnet of the previously written network.
68 if self
._last
_network
and network
.is_subnet_of(self
._last
_network
):
71 # Remember this network for the next call
72 self
._last
_network
= network
75 def _write_header(self
):
77 The header of the file
81 def _write_footer(self
):
83 The footer of the file
87 def _write_network(self
, network
):
88 self
.f
.write("%s\n" % network
)
90 def write(self
, network
):
91 if self
.flatten
and self
._flatten
(network
):
92 log
.debug("Skipping writing network %s" % network
)
95 # Write the network to file
96 self
._write
_network
(network
)
100 Called when all data has been written
108 class IpsetOutputWriter(OutputWriter
):
114 def _write_header(self
):
115 self
.f
.write("create %s hash:net family inet hashsize 1024 maxelem 65536\n" % self
.prefix
)
117 def _write_network(self
, network
):
118 self
.f
.write("add %s %s\n" % (self
.prefix
, network
))
121 class NftablesOutputWriter(OutputWriter
):
127 def _write_header(self
):
128 self
.f
.write("define %s = {\n" % self
.prefix
)
130 def _write_footer(self
):
133 def _write_network(self
, network
):
134 self
.f
.write(" %s,\n" % network
)
137 class XTGeoIPOutputWriter(OutputWriter
):
139 Formats the output in that way, that it can be loaded by
140 the xt_geoip kernel module from xtables-addons.
145 def _write_network(self
, network
):
146 n
= ipaddress
.ip_network("%s" % network
)
148 for address
in (n
.network_address
, n
.broadcast_address
):
149 bytes
= socket
.inet_pton(
150 socket
.AF_INET6
if address
.version
== 6 else socket
.AF_INET
,
158 "ipset" : IpsetOutputWriter
,
159 "list" : OutputWriter
,
160 "nftables" : NftablesOutputWriter
,
161 "xt_geoip" : XTGeoIPOutputWriter
,
164 class Exporter(object):
165 def __init__(self
, db
, writer
):
166 self
.db
, self
.writer
= db
, writer
168 def export(self
, directory
, families
, countries
, asns
):
169 for family
in families
:
170 log
.debug("Exporting family %s" % family
)
174 # Create writers for countries
175 for country_code
in countries
:
176 filename
= self
._make
_filename
(
177 directory
, prefix
=country_code
, suffix
=self
.writer
.suffix
, family
=family
,
180 writers
[country_code
] = self
.writer
.open(filename
, prefix
="CC_%s" % country_code
)
182 # Create writers for ASNs
184 filename
= self
._make
_filename
(
185 directory
, "AS%s" % asn
, suffix
=self
.writer
.suffix
, family
=family
,
188 writers
[asn
] = self
.writer
.open(filename
, prefix
="AS%s" % asn
)
190 # Get all networks that match the family
191 networks
= self
.db
.search_networks(family
=family
)
193 # Walk through all networks
194 for network
in networks
:
195 # Write matching countries
197 writers
[network
.country_code
].write(network
)
201 # Write matching ASNs
203 writers
[network
.asn
].write(network
)
209 if network
.has_flag(flag
):
210 # Fetch the "fake" country code
211 country
= flags
[flag
]
214 writers
[country
].write(network
)
218 # Write everything to the filesystem
219 for writer
in writers
.values():
222 def _make_filename(self
, directory
, prefix
, suffix
, family
):
223 filename
= "%s.%s%s" % (
224 prefix
, suffix
, "6" if family
== socket
.AF_INET6
else "4"
227 return os
.path
.join(directory
, filename
)