Last active
January 11, 2025 11:06
-
-
Save artizirk/c91e4f8c237dec07e3ad1b286f1855a7 to your computer and use it in GitHub Desktop.
Generate WireGuard IP Addresses from public key, compatible with wg-ip bash script
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env python3 | |
# need at least python3.6+ for blake2 | |
from base64 import b64decode | |
from hashlib import sha256, blake2s | |
from ipaddress import ip_address, ip_network | |
# https://github.com/chmduquesne/wg-ip | |
def gen_ip(pubkey, subnet=ip_network('fe80::/64')): | |
"""Generate wg-ip compatible addresses from WireGuard public key. | |
Uses sha256 over base64 encoded newline terminated publickey string. | |
""" | |
prefix_bytes = subnet.network_address.packed | |
mask_bytes = subnet.netmask.packed | |
suffix_bytes = sha256(pubkey.encode('ascii')+b'\n').digest() | |
address = b'' | |
for prefix, suffix, mask in zip(prefix_bytes, suffix_bytes, mask_bytes): | |
address += ((prefix & mask) | (suffix & (mask^255))).to_bytes(1, byteorder='big') | |
return ip_address(address) | |
# https://gist.github.com/reidrankin/3a39210ce437680f5cf1ac549fd1f1ff | |
# https://gist.github.com/artizirk/fca1ba915e741d60ec4fec957dfe0859 | |
def gen_lla(pubkey, subnet=ip_network('fe80::/10')): | |
"""Calculates cryptographically-bound IPv6 Link-Local Addresses from WireGuard public keys. | |
Uses blake2s hash over public key as bytes. | |
""" | |
prefix_bytes = subnet.network_address.packed | |
mask_bytes = subnet.netmask.packed | |
suffix_bytes = blake2s(pubkey).digest() | |
address = b'' | |
for prefix, suffix, mask in zip(prefix_bytes, suffix_bytes, mask_bytes): | |
address += ((prefix & mask) | (suffix & (mask^255))).to_bytes(1, byteorder='big') | |
return ip_address(address) | |
# https://gist.github.com/artizirk/c91e4f8c237dec07e3ad1b286f1855a7?permalink_comment_id=5374168#gistcomment-5374168 | |
def gen_lla_no_hash(pubkey, subnet=ip_network('fe80::/10')): | |
"""Calculates IPv6 Link-Local Addresses from WireGuard public keys. | |
Uses public key bytes as the ip address. | |
""" | |
prefix_bytes = subnet.network_address.packed | |
mask_bytes = subnet.netmask.packed | |
address = b'' | |
for prefix, suffix, mask in zip(prefix_bytes, pubkey, mask_bytes): | |
address += ((prefix & mask) | (suffix & (mask^255))).to_bytes(1, byteorder='big') | |
return ip_address(address) | |
if __name__ == '__main__': | |
print("# sha256 based address calculation") | |
pubkey = "foo" | |
subnet = ip_network("fd1a:6126:2887::/48") | |
exp_result = ip_address("fd1a:6126:2887:f9b1:d61e:21e7:96d7:8dcc") | |
real_result = gen_ip(pubkey, subnet) | |
print(exp_result, real_result, exp_result==real_result) | |
print("# blake2s based address calculation") | |
pubkey = b"\x00"*32 # 32byte public key fo all zeroes | |
exp_result = ip_address("fe8b:5ea9:9e65:3bc2:b593:db41:30d1:0a4e") | |
real_result = gen_lla(pubkey) | |
print(exp_result, real_result, exp_result==real_result) | |
pubkey = b64decode("hTvlXzX5ZoTg6BxbWHYWSZo6pkLGPdQwVoXZYqrMMgs=") | |
exp_result = ip_address("fe80:3fe9:4d26:adc4:1953:7b20:314a:3167") | |
real_result = gen_lla(pubkey) | |
print(exp_result, real_result, exp_result==real_result) | |
print("# address directly from pubkey") | |
pubkey = b"\x00"*32 # Fake pubkey | |
exp_result = ip_address("fe80::") | |
real_result = gen_lla_no_hash(pubkey) | |
print(exp_result, real_result, exp_result==real_result) | |
pubkey = b64decode("hTvlXzX5ZoTg6BxbWHYWSZo6pkLGPdQwVoXZYqrMMgs=") | |
exp_result = ip_address("febb:e55f:35f9:6684:e0e8:1c5b:5876:1649") | |
real_result = gen_lla_no_hash(pubkey) | |
print(exp_result, real_result, exp_result==real_result) | |
pubkey = b64decode("hTvlXzX5ZoTg6BxbWHYWSZo6pkLGPdQwVoXZYqrMMgs=") | |
subnet = ip_network("fe80::/64") # AFAIk somewhere in the IP spec it was said that Link Local should be /64 | |
exp_result = ip_address("fe80::e0e8:1c5b:5876:1649") | |
real_result = gen_lla_no_hash(pubkey, subnet=subnet) | |
print(exp_result, real_result, exp_result==real_result) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Nerd sniped, I've created a similar tool https://github.com/AlexanderYastrebov/wireguard-vanity-key 🤣