ランポート署名(ランポートしょめい、英: Lamport signature)とは、デジタル署名を構築するための手法である。ランポート署名は、任意の一方向性関数を用いて構成でき、一般には暗号学的ハッシュ関数を用いて実装される。
将来的に量子コンピュータの登場でRSAのような多くの暗号技術が脅威にさらされる中で、長いハッシュ値を持つハッシュ関数を用いたランポート署名は安全性を担保できると考えられている。
ランポート署名において、署名鍵は1つのメッセージの署名にしか利用できないという制約がある。だが、ハッシュ木を利用することで、1つの鍵で、複数のメッセージの署名をすることが可能である。
ランポート署名は、1979 年にLeslie Lamportによって発明され、発明者の名前にちなんで名付けられた。[1]
署名されるメッセージは固定の長さ( ビット)であるとする。もし、任意の長さのメッセージに署名したい場合には、まずハッシュ長がビットの暗号学的に安全なハッシュ関数でハッシュ値を計算し、そのハッシュ値に対して署名の手続きをすることで署名することができる。
を一方向性関数とする。
署名者は、 の定義域から独立一様ランダムに 個の値 および をランダムに選ぶ。次に、各 に を作用させることで
および を計算する。
署名鍵(秘密鍵)は であり、検証鍵(公開鍵)は である。
ビットのメッセージを とする。
署名は
である。
メッセージ と署名 を持つ検証者は、全ての に対して
が成り立つことをチェックし、成り立てば署名を受理する。
署名を偽造するためには、偽造者は公開されている から、 を満たす を計算しなければならない。したがって、 が一方向性関数である限り、偽造は不可能である。
一方向性関数として、出力が256ビットであるハッシュ関数hash( )を用いるとしよう。
署名に用いる秘密鍵は、ランダムな256×2個の256ビット列 である。
検証に用いる公開鍵は、256×2個の256ビットハッシュ値 である。それぞれ128 Kibit である。
メッセージ(あるいはそのハッシュ値)は256ビットであり、各ビットに対して乱数のペア が署名鍵として選ばれている。そこで,メッセージの ビット目が0ならば を署名に入れ、1ならば を署名に入れる。例えばメッセージが であった場合、
- () より、
- () より、
- () より、
- () より、
とおき、署名を とする。署名は 256×256 bits = 64 Kibitである。
検証者が256ビットのメッセージと、256個のからなる署名を得たとする。検証鍵は、256個のハッシュ値のペア から成る。そこで、署名内の各 のハッシュ値を計算し、メッセージの ビット目が0ならば とハッシュ値を等しいかを、1ならば とハッシュ値と等しいかを確認する。
例えば上の例の場合、
- () が成り立つか?
- () が成り立つか?
- () が成り立つか?
- () が成り立つか?
を確認する。全て成り立つならば、署名を正しいものとする。
ランポート署名の安全性は、一方向関数として用いるハッシュ関数の安全性と、ハッシュ関数の入力長に依存している。
ハッシュ関数が ビットのメッセージダイジェスト(ハッシュ値)を出力する場合、
理想的な原像計算困難性を持つハッシュ関数において、
一つのハッシュ値の原像を求めるために必要な計算量は である。(古典的計算機モデルの場合。)また、量子計算機の場合、グローバーのアルゴリズムを用いることで、原像を求めるために必要な計算量は に減らすことができることが知られている。
したがって、 ないしは が十分大きくなるように、長いハッシュ値を出力するハッシュ関数を利用する必要がある。
一方、十分大きな を選択したとしても、ハッシュ関数に入力される が短い場合、ランポート署名は容易に署名偽造が可能である。例えば がわずか16ビットの場合、ハッシュ長 に関係なく、 回のハッシュ計算で入力を全数探索でき、原像を求められる。よって、バランスの取れたシステム設計においては、ハッシュ関数への入力と出力は、同程度の長さとする。
グローバーのアルゴリズムの存在により、耐量子計算機安全性のためには、公開鍵の各要素 、秘密鍵の各要素 および署名の各要素 は、システムのセキュリティパラメータの 2 倍以上でなければならない。つまり、
- 80 ビットセキュリティのシステムでは、160 ビット以上の長さを持つ要素を利用する。
- 128 ビットセキュリティのシステムでは、256 ビット以上の長さを持つ要素を利用するである。
ただし、上記の攻撃コストの見積もりは、理想的なハッシュ関数を前提とし、1つのハッシュ値をターゲットとして原像を求めようとする攻撃に限定しているため、注意が必要である。従来の計算機モデルにおいて 個の原像を同時に求める場合、1つの原像あたりのコストが から に減少することが知られている
[2]。
複数の原像を同時に計算する攻撃を考慮して最適な要素サイズを選択することは、未解決の問題である。 512 ビット要素や SHA-512 など、より大きな要素サイズと強力なハッシュ関数を利用すれば、これらの未知の問題に対して、より大きなセキュリティマージンが保証される。
秘密鍵のすべての乱数は、十分な長さ(通常、秘密鍵内の1つの乱数と同じ長さ)のシードと暗号論的擬似乱数生成器から生成できる。したがって、シードのみを秘密鍵として保存しておくことができる。乱数は必要な時にシードから再生成すればよい。
同様に、1つのシードと疑似乱数生成器を用いて、多数の鍵ペア(公開鍵と秘密鍵)を作成することもできる。その場合、ランダムアクセス可能な擬似乱数生成器を使用する必要がある。
耐量子計算機安全性を考えるならば、BBS のような古典的な擬似乱数生成器ではなく、耐量子安全なものを使うべきである。
ランポート署名は、ハッシュリストと組み合わせることで、公開鍵をただ一つのハッシュ値とすることができる。すなわち、 個のハッシュ値 から、さらにハッシュ値 を計算し, のみを公開鍵として公開する。 を用いて署名の検証を行うためには、署名には、メッセージの各ビットに対応する乱数だけでなく、各ビットに対応していない乱数のハッシュ値を含める必要がある。その結果、署名のサイズは約2倍になる。
ハッシュリストの代わりに暗号学的アキュムレータを使用することで、署名にハッシュ値を追加しないで済む方法も知られている[3]。
Winternitz署名の圧縮手法を用いると、秘密鍵、公開鍵、署名のサイズを小さくすることができ、その代わり、署名を生成・検証する計算量が大きく増加する。
この圧縮手法では、メッセージの1ビットごとに乱数およびハッシュ値を2つずつ用意する代わりに、メッセージを固定長のチャンクに分割し,チャンクごとに乱数および(多重)ハッシュ値を1つずつ用意する。したがって、チャンクを ビットとすると、鍵のサイズはオリジナルのランポート署名の鍵サイズに比べて 程度に小さくなり、署名のサイズも 程度に小さくなる。署名の生成・検証には、チャンクごとに多重ハッシュを計算する必要があり、コストは約 倍になる
[4]
[5]。
さらに、上述のようにハッシュリストを使用して、公開鍵をハッシュ値一つのみに短縮することもできる。(署名サイズは2倍になる。)
ランポート署名では、一つの(個の値からなる)鍵は、一つのメッセージの署名にしか利用できない。もし、多くのメッセージに署名をしたいならば、多くの公開鍵を公開しておかなければならない。しかし、これら多くの公開鍵にハッシュ木を適用することで、ただ一つのトップハッシュのみを公開しておくことができる。このとき、署名には検証に必要はハッシュ木の一部の情報を含める必要があり、署名長は少し長くなるが、トップハッシュを公開鍵として多くのメッセージの検証をすることができる[4]。