# xref unique ID hash calculation # # Copyright (C) 2020 David Lamparter for NetDEF, Inc. # # This program is free software; you can redistribute it and/or modify it # under the terms of the GNU General Public License as published by the Free # Software Foundation; either version 2 of the License, or (at your option) # any later version. # # This program is distributed in the hope that it will be useful, but WITHOUT # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for # more details. # # You should have received a copy of the GNU General Public License along # with this program; see the file COPYING; if not, write to the Free Software # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA import struct from hashlib import sha256 def bititer(data, bits, startbit = True): ''' just iterate the individual bits out from a bytes object if startbit is True, an '1' bit is inserted at the very beginning goes <bits> at a time, starts at LSB. ''' bitavail, v = 0, 0 if startbit and len(data) > 0: v = data.pop(0) yield (v & ((1 << bits) - 1)) | (1 << (bits - 1)) bitavail = 9 - bits v >>= bits - 1 while len(data) > 0: while bitavail < bits: v |= data.pop(0) << bitavail bitavail += 8 yield v & ((1 << bits) - 1) bitavail -= bits v >>= bits def base32c(data): ''' Crockford base32 with extra dashes ''' chs = "0123456789ABCDEFGHJKMNPQRSTVWXYZ" o = '' if type(data) == str: data = [ord(v) for v in data] else: data = list(data) for i, bits in enumerate(bititer(data, 5)): if i == 5: o = o + '-' elif i == 10: break o = o + chs[bits] return o def uidhash(filename, hashstr, hashu32a, hashu32b): ''' xref Unique ID hash used in FRRouting ''' filename = '/'.join(filename.rsplit('/')[-2:]) hdata = filename.encode('UTF-8') + hashstr.encode('UTF-8') hdata += struct.pack('>II', hashu32a, hashu32b) i = sha256(hdata).digest() return base32c(i)