見出し画像

Pythonでできること1:パスワード解析

1.概要

 Python(プログラミング)でできることの紹介+Python基礎操作の理解を深めるための記事です。第一回目はパスワード解析のブルートフォースアタック(総当たり攻撃)を紹介します。
※パスワードクラッキングの推奨をしているわけではございません。

2.ブルートフォースアタックとは?

 簡単に言うとパスワードを想定される全パターンで試す処理です。例として数字は0~9の10種類のため4桁の数字であれば0000~9999の計10,000パターンをパスワード入力して正解を確認する処理となります。

3.Pythonでやってみよう:2桁のパスワード

 例えばパスワードが2桁の小文字のアルファベット(a~z)だということが分かっていたとします。その場合のブルートフォースは下記のとおりです。
※numpyを紹介していないためリスト形式でも作成しています。

[In]
import string
import pandas as pd
import numpy as np

words = list(string.ascii_lowercase) #out-> ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z']
passwords = []
#a~zの26個の後ろに26個のa~zを足して2桁の文字列を作成する。=>26×26=676パターン
for word_1 in words:
   for word_2 in words:
       passwords.append(word_1+word_2)

#passwords = np.array(passwords).reshape(26,26) #2626列に整形
passwords = [passwords[26*i:26*(i+1)] for i in range(26)] #リストVersion

df = pd.DataFrame(passwords, columns=words, index=words)
display(df) #print()みたいなやつ

 a~zの後ろにa~zを足し2桁の文字列を作成すると26×26=676パターンです。これをパスワードにぶつければパスワードを当てることができます。

2桁アルファベット - コピー

【参考】
 複数ループに"product"を使用することでシンプルに記載できます。

[IN]
import string
import pandas as pd
import numpy as np
from itertools import product

words = list(string.ascii_lowercase) #out-> ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z']
passwords = []
#a~zの26個の後ろに26個のa~zを足して2桁の文字列を作成する。=>26×26=676パターン
for i in product(words, repeat=2):
       passwords.append(i[0] + i[1])

passwords = np.array(passwords).reshape(26,26) #2626列に整形
df = pd.DataFrame(passwords, columns=words, index=words)
display(df) 

4.Pythonでやってみよう:6桁のパスワード

私の大好きな半沢直樹ではセントラル証券において同様のパスワード解析をしておりました(答えはzansin)。具体的なやり方は多分いろいろあると思いますが上の応用として下記でできます。

[In]
import string

words = list(string.ascii_lowercase) #out-> ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z']
print(f'パスワード総数:{26**6}')
#約31億パターンあるためリストに入れるとメモリに乗らないのでリスト化はしない。
for word_1 in words:
   for word_2 in words:
       for word_3 in words:
           for word_4 in words:
               for word_5 in words:
                   for word_6 in words:
                       word_1+word_2+word_3+word_4+word_5+word_6                                             

最後に得られた値をパスワードにぶつければセントラル証券のパスワードもクラッキングできます。
ちなみに上記は作成したパスワードで何か処理はしておりませんが、for文を回すだけなら2min程度で終了します。

5.学ぶべき教訓

総当たりだと パスワードのパターン数=選択肢^文字数 となります。つまり下記によりパスワード解析が困難になります。

●文字だけでなく数字や記号を入れる。
●パスワードの桁数を増やす
●そもそもパスワードの桁数を表示させない。(何桁かをしぼらせない)


6.参考

6-1.関数化

 汎用性を上げるため、python標準ライブラリのproductを使用して関数化しました。桁数と文字列(イテラブル)を渡せば指定した桁数の文字を出力できます。

[IN]
import string
from itertools import product

def generate_passwords(words, repeat):
    for i in product(words, repeat=repeat):
        yield ''.join(i)

words = string.digits #'0123456789'

passwords_num = list(generate_passwords(words, 2))
print(passwords_num) #2桁数値のパスワードを作成する->100個

print(list(generate_passwords(string.ascii_lowercase, 2))) #2桁の英字を作成する->676個

[OUT]
['00', '01', '02', '03', '04', '05', '06', '07', '08', '09', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23', '24', '25', '26', '27', '28', '29', '30', '31', '32', '33', '34', '35', '36', '37', '38', '39', '40', '41', '42', '43', '44', '45', '46', '47', '48', '49', '50', '51', '52', '53', '54', '55', '56', '57', '58', '59', '60', '61', '62', '63', '64', '65', '66', '67', '68', '69', '70', '71', '72', '73', '74', '75', '76', '77', '78', '79', '80', '81', '82', '83', '84', '85', '86', '87', '88', '89', '90', '91', '92', '93', '94', '95', '96', '97', '98', '99']

['aa', 'ab', 'ac', 'ad', 'ae', 'af', 'ag', 'ah', 'ai', 'aj', 'ak', 'al', 'am', 'an', 'ao', 'ap', 'aq', 'ar', 'as', 'at', 'au', 'av', 'aw', 'ax', 'ay', 'az', 'ba', 'bb', 'bc', 'bd', 'be', 'bf', 'bg', 'bh', 'bi', 'bj', 'bk', 'bl', 'bm', 'bn', 'bo', 'bp', 'bq', 'br', 'bs', 'bt', 'bu', 'bv', 'bw', 'bx', 'by', 'bz', 'ca', 'cb', 'cc', 'cd', 'ce', 'cf', 'cg', 'ch', 'ci', 'cj', 'ck', 'cl', 'cm', 'cn', 'co', 'cp', 'cq', 'cr', 'cs', 'ct', 'cu', 'cv', 'cw', 'cx', 'cy', 'cz', 'da', 'db', 'dc', 'dd', 'de', 'df', 'dg', 'dh', 'di', 'dj', 'dk', 'dl', 'dm', 'dn', 'do', 'dp', 'dq', 'dr', 'ds', 'dt', 'du', 'dv', 'dw', 'dx', 'dy', 'dz', 'ea', 'eb', 'ec', 'ed', 'ee', 'ef', 'eg', 'eh', 'ei', 'ej', 'ek', 'el', 'em', 'en', 'eo', 'ep', 'eq', 'er', 'es', 'et', 'eu', 'ev', 'ew', 'ex', 'ey', 'ez', 'fa', 'fb', 'fc', 'fd', 'fe', 'ff', 'fg', 'fh', 'fi', 'fj', 'fk', 'fl', 'fm', 'fn', 'fo', 'fp', 'fq', 'fr', 'fs', 'ft', 'fu', 'fv', 'fw', 'fx', 'fy', 'fz', 'ga', 'gb', 'gc', 'gd', 'ge', 'gf', 'gg', 'gh', 'gi', 'gj', 'gk', 'gl', 'gm', 'gn', 'go', 'gp', 'gq', 'gr', 'gs', 'gt', 'gu', 'gv', 'gw', 'gx', 'gy', 'gz', 'ha', 'hb', 'hc', 'hd', 'he', 'hf', 'hg', 'hh', 'hi', 'hj', 'hk', 'hl', 'hm', 'hn', 'ho', 'hp', 'hq', 'hr', 'hs', 'ht', 'hu', 'hv', 'hw', 'hx', 'hy', 'hz', 'ia', 'ib', 'ic', 'id', 'ie', 'if', 'ig', 'ih', 'ii', 'ij', 'ik', 'il', 'im', 'in', 'io', 'ip', 'iq', 'ir', 'is', 'it', 'iu', 'iv', 'iw', 'ix', 'iy', 'iz', 'ja', 'jb', 'jc', 'jd', 'je', 'jf', 'jg', 'jh', 'ji', 'jj', 'jk', 'jl', 'jm', 'jn', 'jo', 'jp', 'jq', 'jr', 'js', 'jt', 'ju', 'jv', 'jw', 'jx', 'jy', 'jz', 'ka', 'kb', 'kc', 'kd', 'ke', 'kf', 'kg', 'kh', 'ki', 'kj', 'kk', 'kl', 'km', 'kn', 'ko', 'kp', 'kq', 'kr', 'ks', 'kt', 'ku', 'kv', 'kw', 'kx', 'ky', 'kz', 'la', 'lb', 'lc', 'ld', 'le', 'lf', 'lg', 'lh', 'li', 'lj', 'lk', 'll', 'lm', 'ln', 'lo', 'lp', 'lq', 'lr', 'ls', 'lt', 'lu', 'lv', 'lw', 'lx', 'ly', 'lz', 'ma', 'mb', 'mc', 'md', 'me', 'mf', 'mg', 'mh', 'mi', 'mj', 'mk', 'ml', 'mm', 'mn', 'mo', 'mp', 'mq', 'mr', 'ms', 'mt', 'mu', 'mv', 'mw', 'mx', 'my', 'mz', 'na', 'nb', 'nc', 'nd', 'ne', 'nf', 'ng', 'nh', 'ni', 'nj', 'nk', 'nl', 'nm', 'nn', 'no', 'np', 'nq', 'nr', 'ns', 'nt', 'nu', 'nv', 'nw', 'nx', 'ny', 'nz', 'oa', 'ob', 'oc', 'od', 'oe', 'of', 'og', 'oh', 'oi', 'oj', 'ok', 'ol', 'om', 'on', 'oo', 'op', 'oq', 'or', 'os', 'ot', 'ou', 'ov', 'ow', 'ox', 'oy', 'oz', 'pa', 'pb', 'pc', 'pd', 'pe', 'pf', 'pg', 'ph', 'pi', 'pj', 'pk', 'pl', 'pm', 'pn', 'po', 'pp', 'pq', 'pr', 'ps', 'pt', 'pu', 'pv', 'pw', 'px', 'py', 'pz', 'qa', 'qb', 'qc', 'qd', 'qe', 'qf', 'qg', 'qh', 'qi', 'qj', 'qk', 'ql', 'qm', 'qn', 'qo', 'qp', 'qq', 'qr', 'qs', 'qt', 'qu', 'qv', 'qw', 'qx', 'qy', 'qz', 'ra', 'rb', 'rc', 'rd', 're', 'rf', 'rg', 'rh', 'ri', 'rj', 'rk', 'rl', 'rm', 'rn', 'ro', 'rp', 'rq', 'rr', 'rs', 'rt', 'ru', 'rv', 'rw', 'rx', 'ry', 'rz', 'sa', 'sb', 'sc', 'sd', 'se', 'sf', 'sg', 'sh', 'si', 'sj', 'sk', 'sl', 'sm', 'sn', 'so', 'sp', 'sq', 'sr', 'ss', 'st', 'su', 'sv', 'sw', 'sx', 'sy', 'sz', 'ta', 'tb', 'tc', 'td', 'te', 'tf', 'tg', 'th', 'ti', 'tj', 'tk', 'tl', 'tm', 'tn', 'to', 'tp', 'tq', 'tr', 'ts', 'tt', 'tu', 'tv', 'tw', 'tx', 'ty', 'tz', 'ua', 'ub', 'uc', 'ud', 'ue', 'uf', 'ug', 'uh', 'ui', 'uj', 'uk', 'ul', 'um', 'un', 'uo', 'up', 'uq', 'ur', 'us', 'ut', 'uu', 'uv', 'uw', 'ux', 'uy', 'uz', 'va', 'vb', 'vc', 'vd', 've', 'vf', 'vg', 'vh', 'vi', 'vj', 'vk', 'vl', 'vm', 'vn', 'vo', 'vp', 'vq', 'vr', 'vs', 'vt', 'vu', 'vv', 'vw', 'vx', 'vy', 'vz', 'wa', 'wb', 'wc', 'wd', 'we', 'wf', 'wg', 'wh', 'wi', 'wj', 'wk', 'wl', 'wm', 'wn', 'wo', 'wp', 'wq', 'wr', 'ws', 'wt', 'wu', 'wv', 'ww', 'wx', 'wy', 'wz', 'xa', 'xb', 'xc', 'xd', 'xe', 'xf', 'xg', 'xh', 'xi', 'xj', 'xk', 'xl', 'xm', 'xn', 'xo', 'xp', 'xq', 'xr', 'xs', 'xt', 'xu', 'xv', 'xw', 'xx', 'xy', 'xz', 'ya', 'yb', 'yc', 'yd', 'ye', 'yf', 'yg', 'yh', 'yi', 'yj', 'yk', 'yl', 'ym', 'yn', 'yo', 'yp', 'yq', 'yr', 'ys', 'yt', 'yu', 'yv', 'yw', 'yx', 'yy', 'yz', 'za', 'zb', 'zc', 'zd', 'ze', 'zf', 'zg', 'zh', 'zi', 'zj', 'zk', 'zl', 'zm', 'zn', 'zo', 'zp', 'zq', 'zr', 'zs', 'zt', 'zu', 'zv', 'zw', 'zx', 'zy', 'zz']

6-2.参考記事

あとがき

表示や使いやすさのためにpandas, numpyを使っておりますが、実はfor文とリストが使えればこのくらいのことは簡単にできます。基礎を実践的に学ぶ練習になると思います。

それにしても1個目がこのネタでいいのかな・・・・・

いいなと思ったら応援しよう!