StreamRelay.NET.exe のHash/HMAC/MAC機能を呼び出しているプラグインの実装方法について説明する
StreamRelay.NET.exe/StreamRelay.NET.x86.exeには、いくつかの機能をプラグインとして実装できるようにインターフェイスを公開している。
今回は、Hash/HMAC/MACを計算するためのインターフェイスを使って、プラグインを作成する方法を記述する。
既存のプラグイン
ストリームのデータを横目で見ながら何かの計算値を出力するような機能に使っている。
hashとか、HMACとか、MACとか、CRCとか、ただのバイト数カウントするヤツとか・・・
プラグイン的には、このあたり。
- Plugins\Plugin.Classless.Hasher.dll
- Plugins\Plugin.BouncyCastle.Crypto.dll
- Plugins\Plugin.CSharpCode.SharpZipLib
実装したプラグインの呼び出され方
-[Local/Remote]Proxy オプションのnetstreamスキームの引数として渡された引数に応じて、ストリームを包んだり、暗号鍵の攪拌(IPluginPasswordStirringEntity)に呼ばれたりする
という感じ。
HASH と MAC
- 秘密鍵が必要なものがMAC
- 秘密鍵が必要でないものがHASH
という感じなので、
- hash
- hash, crc, など
- mac
- hmac, mac, など
な感じ
実装するインターフェイス
以下のインターフェイスを継承したクラスを実装すればよい
- IPlugin.dll 中の jp.dip.rocketeer.Plugins.IPluginHashMacEntity インターフェイス
- IPlugin.dll 中の jp.dip.rocketeer.Plugins.IPluginHashMac インターフェイス
このクラス2つだけ実装すればよい。
jp.dip.rocketeer.Plugins.IPluginHashMacEntityインターフェイスは、jp.dip.rocketeer.Plugins.IPluginPasswordStirringEntityインターフェイスを継承しているので、パスワード攪拌関数も兼ねていることになる。
jp.dip.rocketeer.Plugins.IPluginHashMac インターフェイス
StreamRelay.NET.exeや、他のプラグインから、引数/オプションの解析処理中に呼び出して、引数の指定内容によって、IPluginHashMacEntityやIPluginHashMacEntity[]を返す。
すると、IPluginHashMacEntity/IPluginHashMacEntity[]から、Streamを包んでStreamを返すものを呼び出したり、場合によっては暗号鍵の攪拌関数として呼ばれたりする。
という感じ。
具体的には、-(Local|Remote)Proxy の netstreamスキーム指定時の、ハッシュストリームなどの解析で
IPluginHashMacEntity CreateHashEntity(String iHashArgorithmName, Boolean IsLocal, Boolean IsInput, ref String errMsg)
または
IPluginHashMacEntity CreateMacEntity(String iMacArgorithmName, byte[] key, Boolean IsLocal, Boolean IsInput,ref String errMsg)
を呼び出すので、自分の引数であれば、IPluginHashMacEntityオブジェクトを返してほしい。
具体的には「[Input|Output]HashAlgorithm=<<name>>」または「[Input|Output]KeyMacAlgorithm(Hex)=<<name>>;<<Key|HexKey>>」をnetstreamスキーム(-(Local|Remote)Proxy)で指定されたような時に、呼び出される。
もしくは、(-Local|-Remote)Proxy の netstreamスキーム指定時の、「[Input|Output]AllHashAlgorithm(Hex)(=<<Key>>)」という指定によって
IPluginHashMacEntity[] CreateAllHashEntity(Boolean IsLocal, Boolean IsInput)
または
IPluginHashMacEntity[] CreateAllMacEntity(byte[] key, Boolean IsLocal, Boolean IsInput)
が呼び出された時には、全てのIPluginHashMacEntityオブジェクトを配列で返してほしい。
そして、Stream(HashMacStreamは後述)がCloseしたタイミングで byte[] GetHash() が呼ばれるので、計算結果を返してほしい。
jp.dip.rocketeer.Plugins.IPluginHashMac インターフェイスで定義しているメソッド/プロパティ
実装する必要があるのは以下
jp.dip.rocketeer.Plugins.IPluginHashMacインターフェイスで定義されているのは、以下のとおり
- Dictionary ListDescription(Boolean IsHash)
- IsHash=trueならハッシュの引数文字列などを返す、IsHash=falseならハッシュの引数文字列などを返す
- IPluginHashMacEntity CreateHashEntity(String iHashArgorithmName, Boolean IsLocal, Boolean IsInput, ref String errMsg)
- 名称から Hash オブジェクトの実体を作成する
- IPluginHashMacEntity[] CreateAllHashEntity(Boolean IsLocal, Boolean IsInput)
- 全ての Hash オブジェクトの実体を作成する
- IPluginHashMacEntity CreateMacEntity(String iMacArgorithmName, byte[] key, Boolean IsLocal, Boolean IsInput,ref String errMsg)
- 名称から Mac オブジェクトの実体を作成する
- IPluginHashMacEntity[] CreateAllMacEntity(byte[] key, Boolean IsLocal, Boolean IsInput)
- 全ての Mac オブジェクトの実体を作成する
jp.dip.rocketeer.Plugins.IPluginBaseインターフェイスで定義されている分については、インターフェイス IPluginBase についてを参照
jp.dip.rocketeer.Plugins.IPluginHashMacEntity インターフェイスで定義しているメソッド/プロパティ
実装する必要があるのは以下
jp.dip.rocketeer.Plugins.IPluginHashMacEntityインターフェイスで定義されているのは、以下のとおり
- Boolean IsInput { get; set; }
- InputStream or OutputStream
- Boolean IsHash { get; set; }
- Hash or Mac
- Boolean IsLocal { get; set; }
- 接続するストリームがローカル(サーバ)側かリモート(クライアント)側か
- byte[] GetHash()
- バッファリングしている Hash/Mac の値を返す
- HashMacStream CreateStream(Stream baseStream)
- 引数のストリームにHash/Mac ストリームを被せるようにして返す(後述)
jp.dip.rocketeer.Plugins.IPluginPasswordStirringEntityインターフェイスで定義されている分については、パスワード攪拌関数プラグインの実装方法を参照
jp.dip.rocketeer.Plugins.HashMacStream クラス
jp.dip.rocketeer.Plugins.HashMacStream クラスは、jp.dip.rocketeer.Plugins.HashMacStream#Close() 時に、jp.dip.rocketeer.Plugins.IPluginHashMacEntity#GetHash() を呼び出して計算値(ハッシュ値など)を出力するようにしている。
よって、jp.dip.rocketeer.Plugins.HashMacStream クラスは、jp.dip.rocketeer.Plugins.IPluginHashMacEntity#CreateStream(Stream)で返してほしい。
具体的には、jp.dip.rocketeer.Plugins.IPluginHashMacEntity#CreateStream(Stream)内部で、jp.dip.rocketeer.Plugins.HashMacStreamのコンストラクタを呼べばいいと思う。
つまり、
jp.dip.rocketeer.Plugins.HashMacStreamのコンストラクタは、
HashMacStream(IPluginHashMacEntity iObj, Stream readStream, Stream writeStream)
なので、
jp.dip.rocketeer.Plugins.IPluginHashMacEntity#CreateStream(Stream baseStream)内部で、HashMacStreamを作って返してくれればいい。
なんとなくのなんちゃってコードだとこんな感じ
public class exampleHash : IPluginHashMacEntity{
// 例えば、入力方向のハッシュ値を計算する
base.IsInput = true;
// 例えばMD5とする
public MD5 md5;
// HashMacStreamkのCloseで呼ばれる
public override byte[] GetHash(){
// cryptoStreamに渡した計算オブジェクトから最終結果を取得して、返す
return this.md5.hash;
}
CryptoStream cryptoStream;
HashMacStream CreateStream(Stream baseStream){
// ストリーム(接続)ごとに計算するので、新規作成(ディープコピー)しておく
IPluginHashMacEntity ko = this.Copy();
// baseStreamストリームと、IPluginHashMacEntityオブジェクトを使って、
// 例えば、ハッシュを計算するストリームCryptoStreamを生成する
cryptoStream = new CryptoStream(baseStream, ko.md5);
// ハッシュを計算するストリームを包んだHashMacStream(System.IO.Stream)を返す
// 入力側は計算するのでcryptoStreamだけど、
// 出力側は計算しないのでbaseStreamそのままを引数に与える
return new HashMacStream(ko, cryptoStream, baseStream);
}}
みたいな感じで HashMacStreamを生成して、返してほしい。