WIP: kunsi-junos #55
1 changed files with 149 additions and 0 deletions
149
libs/juniper.py
Normal file
149
libs/juniper.py
Normal file
|
@ -0,0 +1,149 @@
|
|||
import random
|
||||
|
||||
# copied from https://github.com/peering-manager/peering-manager/blob/main/devices/crypto/juniper.py
|
||||
|
||||
|
||||
# This code is the result of the attempt at converting a Perl module, the expected
|
||||
# result might not actually be what we really want it to be ¯\_(ツ)_/¯
|
||||
#
|
||||
# https://metacpan.org/pod/Crypt::Juniper
|
||||
|
||||
|
||||
MAGIC = "$9$"
|
||||
|
||||
FAMILY = [
|
||||
"QzF3n6/9CAtpu0O",
|
||||
"B1IREhcSyrleKvMW8LXx",
|
||||
"7N-dVbwsY2g4oaJZGUDj",
|
||||
"iHkq.mPf5T",
|
||||
]
|
||||
EXTRA = {}
|
||||
for counter, value in enumerate(FAMILY):
|
||||
for character in value:
|
||||
EXTRA[character] = 3 - counter
|
||||
|
||||
NUM_ALPHA = [x for x in "".join(FAMILY)]
|
||||
ALPHA_NUM = {NUM_ALPHA[x]: x for x in range(0, len(NUM_ALPHA))}
|
||||
|
||||
ENCODING = [
|
||||
[1, 4, 32],
|
||||
[1, 16, 32],
|
||||
[1, 8, 32],
|
||||
[1, 64],
|
||||
[1, 32],
|
||||
[1, 4, 16, 128],
|
||||
[1, 32, 64],
|
||||
]
|
||||
|
||||
|
||||
def __nibble(cref, length):
|
||||
nib = cref[0:length]
|
||||
rest = cref[length:]
|
||||
|
||||
if len(nib) != length:
|
||||
raise Exception(f"Ran out of characters: hit '{nib}', expecting {length} chars")
|
||||
|
||||
return nib, rest
|
||||
|
||||
|
||||
def __gap(c1, c2):
|
||||
return (ALPHA_NUM[str(c2)] - ALPHA_NUM[str(c1)]) % (len(NUM_ALPHA)) - 1
|
||||
|
||||
|
||||
def __gap_decode(gaps, dec):
|
||||
num = 0
|
||||
|
||||
if len(gaps) != len(dec):
|
||||
raise Exception("Nibble and decode size not the same.")
|
||||
|
||||
for x in range(0, len(gaps)):
|
||||
num += gaps[x] * dec[x]
|
||||
|
||||
return chr(num % 256)
|
||||
|
||||
|
||||
def __reverse(current):
|
||||
reversed = list(current)
|
||||
reversed.reverse()
|
||||
return reversed
|
||||
|
||||
|
||||
def __gap_encode(pc, prev, encode):
|
||||
__ord = ord(pc)
|
||||
|
||||
crypt = ""
|
||||
gaps = []
|
||||
for mod in __reverse(encode):
|
||||
gaps.insert(0, int(__ord / mod))
|
||||
__ord %= mod
|
||||
|
||||
for gap in gaps:
|
||||
gap += ALPHA_NUM[prev] + 1
|
||||
prev = NUM_ALPHA[gap % len(NUM_ALPHA)]
|
||||
crypt += prev
|
||||
|
||||
return crypt
|
||||
|
||||
|
||||
def __randc(counter=0):
|
||||
return_value = ""
|
||||
for _ in range(counter):
|
||||
return_value += NUM_ALPHA[random.randrange(len(NUM_ALPHA))]
|
||||
return return_value
|
||||
|
||||
|
||||
def is_encrypted(value):
|
||||
return value.startswith(MAGIC)
|
||||
|
||||
|
||||
def decrypt(value):
|
||||
if not value:
|
||||
return ""
|
||||
|
||||
if not is_encrypted(value):
|
||||
return value
|
||||
|
||||
chars = value.split("$9$", 1)[1]
|
||||
first, chars = __nibble(chars, 1)
|
||||
toss, chars = __nibble(chars, EXTRA[first])
|
||||
previous = first
|
||||
decrypted = ""
|
||||
|
||||
while chars:
|
||||
decode = ENCODING[len(decrypted) % len(ENCODING)]
|
||||
nibble, chars = __nibble(chars, len(decode))
|
||||
gaps = []
|
||||
for i in nibble:
|
||||
g = __gap(previous, i)
|
||||
previous = i
|
||||
gaps += [g]
|
||||
decrypted += __gap_decode(gaps, decode)
|
||||
|
||||
return decrypted
|
||||
|
||||
|
||||
def encrypt(value, salt=None):
|
||||
if not value:
|
||||
return ""
|
||||
|
||||
if not isinstance(value, str):
|
||||
value = str(value)
|
||||
|
||||
if is_encrypted(value):
|
||||
return value
|
||||
|
||||
if not salt:
|
||||
salt = __randc(1)
|
||||
rand = __randc(EXTRA[salt])
|
||||
|
||||
position = 0
|
||||
previous = salt
|
||||
crypted = MAGIC + salt + rand
|
||||
|
||||
for x in value:
|
||||
encode = ENCODING[position % len(ENCODING)]
|
||||
crypted += __gap_encode(x, previous, encode)
|
||||
previous = crypted[-1]
|
||||
position += 1
|
||||
|
||||
return crypted
|
Loading…
Reference in a new issue