forked from LBRYCommunity/lbry-sdk
refactored mnemonic.py
This commit is contained in:
parent
391b95fd12
commit
d488bfd9d4
9 changed files with 88 additions and 162 deletions
|
@ -1,159 +1,57 @@
|
||||||
# Copyright (C) 2014 Thomas Voegtlin
|
|
||||||
# Copyright (C) 2018 LBRY Inc.
|
|
||||||
|
|
||||||
import hmac
|
|
||||||
import math
|
|
||||||
import hashlib
|
import hashlib
|
||||||
import importlib
|
import asyncio
|
||||||
import unicodedata
|
import unicodedata
|
||||||
import string
|
|
||||||
from binascii import hexlify
|
from binascii import hexlify
|
||||||
from secrets import randbelow
|
from secrets import randbits
|
||||||
|
|
||||||
import pbkdf2
|
|
||||||
|
|
||||||
from lbry.crypto.hash import hmac_sha512
|
from lbry.crypto.hash import hmac_sha512
|
||||||
from .words import english
|
from . import words
|
||||||
|
|
||||||
# The hash of the mnemonic seed must begin with this
|
|
||||||
SEED_PREFIX = b'01' # Standard wallet
|
|
||||||
SEED_PREFIX_2FA = b'101' # Two-factor authentication
|
|
||||||
SEED_PREFIX_SW = b'100' # Segwit wallet
|
|
||||||
|
|
||||||
# http://www.asahi-net.or.jp/~ax2s-kmtn/ref/unicode/e_asia.html
|
|
||||||
CJK_INTERVALS = [
|
|
||||||
(0x4E00, 0x9FFF, 'CJK Unified Ideographs'),
|
|
||||||
(0x3400, 0x4DBF, 'CJK Unified Ideographs Extension A'),
|
|
||||||
(0x20000, 0x2A6DF, 'CJK Unified Ideographs Extension B'),
|
|
||||||
(0x2A700, 0x2B73F, 'CJK Unified Ideographs Extension C'),
|
|
||||||
(0x2B740, 0x2B81F, 'CJK Unified Ideographs Extension D'),
|
|
||||||
(0xF900, 0xFAFF, 'CJK Compatibility Ideographs'),
|
|
||||||
(0x2F800, 0x2FA1D, 'CJK Compatibility Ideographs Supplement'),
|
|
||||||
(0x3190, 0x319F, 'Kanbun'),
|
|
||||||
(0x2E80, 0x2EFF, 'CJK Radicals Supplement'),
|
|
||||||
(0x2F00, 0x2FDF, 'CJK Radicals'),
|
|
||||||
(0x31C0, 0x31EF, 'CJK Strokes'),
|
|
||||||
(0x2FF0, 0x2FFF, 'Ideographic Description Characters'),
|
|
||||||
(0xE0100, 0xE01EF, 'Variation Selectors Supplement'),
|
|
||||||
(0x3100, 0x312F, 'Bopomofo'),
|
|
||||||
(0x31A0, 0x31BF, 'Bopomofo Extended'),
|
|
||||||
(0xFF00, 0xFFEF, 'Halfwidth and Fullwidth Forms'),
|
|
||||||
(0x3040, 0x309F, 'Hiragana'),
|
|
||||||
(0x30A0, 0x30FF, 'Katakana'),
|
|
||||||
(0x31F0, 0x31FF, 'Katakana Phonetic Extensions'),
|
|
||||||
(0x1B000, 0x1B0FF, 'Kana Supplement'),
|
|
||||||
(0xAC00, 0xD7AF, 'Hangul Syllables'),
|
|
||||||
(0x1100, 0x11FF, 'Hangul Jamo'),
|
|
||||||
(0xA960, 0xA97F, 'Hangul Jamo Extended A'),
|
|
||||||
(0xD7B0, 0xD7FF, 'Hangul Jamo Extended B'),
|
|
||||||
(0x3130, 0x318F, 'Hangul Compatibility Jamo'),
|
|
||||||
(0xA4D0, 0xA4FF, 'Lisu'),
|
|
||||||
(0x16F00, 0x16F9F, 'Miao'),
|
|
||||||
(0xA000, 0xA48F, 'Yi Syllables'),
|
|
||||||
(0xA490, 0xA4CF, 'Yi Radicals'),
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def is_cjk(c):
|
def get_languages():
|
||||||
n = ord(c)
|
return words.languages
|
||||||
for start, end, _ in CJK_INTERVALS:
|
|
||||||
if start <= n <= end:
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def normalize_text(seed):
|
def normalize(mnemonic: str) -> str:
|
||||||
seed = unicodedata.normalize('NFKD', seed)
|
return ' '.join(unicodedata.normalize('NFKD', mnemonic).lower().split())
|
||||||
seed = seed.lower()
|
|
||||||
# remove accents
|
|
||||||
seed = ''.join([c for c in seed if not unicodedata.combining(c)])
|
def is_valid(language, mnemonic):
|
||||||
# normalize whitespaces
|
local_words = getattr(words, language)
|
||||||
seed = ' '.join(seed.split())
|
for word in normalize(mnemonic).split():
|
||||||
# remove whitespaces between CJK
|
if word not in local_words:
|
||||||
seed = ''.join([
|
return False
|
||||||
seed[i] for i in range(len(seed))
|
return bool(mnemonic)
|
||||||
if not (seed[i] in string.whitespace and is_cjk(seed[i-1]) and is_cjk(seed[i+1]))
|
|
||||||
])
|
|
||||||
|
def sync_generate(language: str) -> str:
|
||||||
|
local_words = getattr(words, language)
|
||||||
|
entropy = randbits(132)
|
||||||
|
nonce = 0
|
||||||
|
while True:
|
||||||
|
nonce += 1
|
||||||
|
i = entropy + nonce
|
||||||
|
w = []
|
||||||
|
while i:
|
||||||
|
w.append(local_words[i % 2048])
|
||||||
|
i //= 2048
|
||||||
|
seed = ' '.join(w)
|
||||||
|
if hexlify(hmac_sha512(b"Seed version", seed.encode())).startswith(b"01"):
|
||||||
|
break
|
||||||
return seed
|
return seed
|
||||||
|
|
||||||
|
|
||||||
def load_words(language_name):
|
def sync_to_seed(mnemonic: str) -> bytes:
|
||||||
if language_name == 'english':
|
return hashlib.pbkdf2_hmac('sha512', normalize(mnemonic).encode(), b'lbryum', 2048)
|
||||||
return english.words
|
|
||||||
language_module = importlib.import_module('lbry.wallet.client.words.'+language_name)
|
|
||||||
return list(map(
|
|
||||||
lambda s: unicodedata.normalize('NFKD', s),
|
|
||||||
language_module.words
|
|
||||||
))
|
|
||||||
|
|
||||||
|
|
||||||
LANGUAGE_NAMES = {
|
async def generate(language: str) -> str:
|
||||||
'en': 'english',
|
return await asyncio.get_running_loop().run_in_executor(
|
||||||
'es': 'spanish',
|
None, sync_generate, language
|
||||||
'ja': 'japanese',
|
)
|
||||||
'pt': 'portuguese',
|
|
||||||
'zh': 'chinese_simplified'
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class Mnemonic:
|
async def to_seed(mnemonic: str) -> bytes:
|
||||||
# Seed derivation no longer follows BIP39
|
return await asyncio.get_running_loop().run_in_executor(
|
||||||
# Mnemonic phrase uses a hash based checksum, instead of a words-dependent checksum
|
None, sync_to_seed, mnemonic
|
||||||
|
)
|
||||||
def __init__(self, lang='en'):
|
|
||||||
language_name = LANGUAGE_NAMES.get(lang, 'english')
|
|
||||||
self.words = load_words(language_name)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def mnemonic_to_seed(mnemonic, passphrase=''):
|
|
||||||
pbkdf2_rounds = 2048
|
|
||||||
mnemonic = normalize_text(mnemonic)
|
|
||||||
passphrase = normalize_text(passphrase)
|
|
||||||
return pbkdf2.PBKDF2(
|
|
||||||
mnemonic, passphrase, iterations=pbkdf2_rounds, macmodule=hmac, digestmodule=hashlib.sha512
|
|
||||||
).read(64)
|
|
||||||
|
|
||||||
def mnemonic_encode(self, i):
|
|
||||||
n = len(self.words)
|
|
||||||
words = []
|
|
||||||
while i:
|
|
||||||
x = i%n
|
|
||||||
i = i//n
|
|
||||||
words.append(self.words[x])
|
|
||||||
return ' '.join(words)
|
|
||||||
|
|
||||||
def mnemonic_decode(self, seed):
|
|
||||||
n = len(self.words)
|
|
||||||
words = seed.split()
|
|
||||||
i = 0
|
|
||||||
while words:
|
|
||||||
word = words.pop()
|
|
||||||
k = self.words.index(word)
|
|
||||||
i = i*n + k
|
|
||||||
return i
|
|
||||||
|
|
||||||
def make_seed(self, prefix=SEED_PREFIX, num_bits=132):
|
|
||||||
# increase num_bits in order to obtain a uniform distribution for the last word
|
|
||||||
bpw = math.log(len(self.words), 2)
|
|
||||||
# rounding
|
|
||||||
n = int(math.ceil(num_bits/bpw) * bpw)
|
|
||||||
entropy = 1
|
|
||||||
while 0 < entropy < pow(2, n - bpw):
|
|
||||||
# try again if seed would not contain enough words
|
|
||||||
entropy = randbelow(pow(2, n))
|
|
||||||
nonce = 0
|
|
||||||
while True:
|
|
||||||
nonce += 1
|
|
||||||
i = entropy + nonce
|
|
||||||
seed = self.mnemonic_encode(i)
|
|
||||||
if i != self.mnemonic_decode(seed):
|
|
||||||
raise Exception('Cannot extract same entropy from mnemonic!')
|
|
||||||
if is_new_seed(seed, prefix):
|
|
||||||
break
|
|
||||||
return seed
|
|
||||||
|
|
||||||
|
|
||||||
def is_new_seed(seed, prefix):
|
|
||||||
seed = normalize_text(seed)
|
|
||||||
seed_hash = hexlify(hmac_sha512(b"Seed version", seed.encode('utf8')))
|
|
||||||
return seed_hash.startswith(prefix)
|
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
from .english import words as en
|
||||||
|
from .french import words as fr
|
||||||
|
from .italian import words as it
|
||||||
|
from .japanese import words as ja
|
||||||
|
from .portuguese import words as pt
|
||||||
|
from .spanish import words as es
|
||||||
|
from .chinese_simplified import words as zh
|
||||||
|
languages = 'en', 'fr', 'it', 'ja', 'pt', 'es', 'zh
|
1
lbry/wallet/words/french.py
Normal file
1
lbry/wallet/words/french.py
Normal file
File diff suppressed because one or more lines are too long
1
lbry/wallet/words/italian.py
Normal file
1
lbry/wallet/words/italian.py
Normal file
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1,23 +1,42 @@
|
||||||
import unittest
|
from unittest import TestCase
|
||||||
from binascii import hexlify
|
from binascii import hexlify
|
||||||
|
|
||||||
from lbry.wallet.mnemonic import Mnemonic
|
from lbry.wallet import words
|
||||||
|
from lbry.wallet.mnemonic import (
|
||||||
|
get_languages, is_valid,
|
||||||
|
sync_generate as generate,
|
||||||
|
sync_to_seed as to_seed
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class TestMnemonic(unittest.TestCase):
|
class TestMnemonic(TestCase):
|
||||||
|
|
||||||
def test_mnemonic_to_seed(self):
|
def test_get_languages(self):
|
||||||
seed = Mnemonic.mnemonic_to_seed(mnemonic='foobar', passphrase='torba')
|
languages = get_languages()
|
||||||
|
self.assertEqual(len(languages), 6)
|
||||||
|
for lang in languages:
|
||||||
|
self.assertEqual(len(getattr(words, lang)), 2048)
|
||||||
|
|
||||||
|
def test_is_valid(self):
|
||||||
|
self.assertFalse(is_valid('en', ''))
|
||||||
|
self.assertFalse(is_valid('en', 'foo'))
|
||||||
|
self.assertFalse(is_valid('en', 'awesomeball'))
|
||||||
|
self.assertTrue(is_valid('en', 'awesome ball'))
|
||||||
|
|
||||||
|
# check normalize works (these are not the same)
|
||||||
|
self.assertTrue(is_valid('ja', 'るいじ りんご'))
|
||||||
|
self.assertTrue(is_valid('ja', 'るいじ りんご'))
|
||||||
|
|
||||||
|
def test_generate(self):
|
||||||
|
self.assertGreaterEqual(len(generate('en').split()), 11)
|
||||||
|
self.assertGreaterEqual(len(generate('ja').split()), 11)
|
||||||
|
|
||||||
|
def test_to_seed(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
hexlify(seed),
|
hexlify(to_seed(
|
||||||
b'475a419db4e991cab14f08bde2d357e52b3e7241f72c6d8a2f92782367feeee9f403dc6a37c26a3f02ab9'
|
"carbon smart garage balance margin twelve che"
|
||||||
b'dec7f5063161eb139cea00da64cd77fba2f07c49ddc'
|
"st sword toast envelope bottom stomach absent"
|
||||||
|
)),
|
||||||
|
b'919455c9f65198c3b0f8a2a656f13bd0ecc436abfabcb6a2a1f063affbccb628'
|
||||||
|
b'230200066117a30b1aa3aec2800ddbd3bf405f088dd7c98ba4f25f58d47e1baf'
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_make_seed_decode_encode(self):
|
|
||||||
iters = 10
|
|
||||||
m = Mnemonic('en')
|
|
||||||
for _ in range(iters):
|
|
||||||
seed = m.make_seed()
|
|
||||||
i = m.mnemonic_decode(seed)
|
|
||||||
self.assertEqual(m.mnemonic_encode(i), seed)
|
|
||||||
|
|
Loading…
Reference in a new issue