質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.62%

C#でAES暗号の復号する際のストリームについて

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 4,363

porepore

score 15

お世話になります。
現在C#でファイルをAES暗号で暗号化しファイル出力、
復号の際は暗号化ファイルを読み込み復号しファイル出力するソースコードを書いています。
復号の際には暗号化ファイルを読み込み、標準出力したいと思っていますがうまくいかないため、
どのようにすれば良いか教えていただきたいです。
自分としてはコメントの①のFileStream outsをほかのストリームにすれば良いのではないかと考えています。
お忙しいと思いますが宜しくお願いします。

static void Main()
    {
    string sfile = @"C:\test\enc.txt"; //暗号化前ファイル
        string dfile = @"C:\test\dec.txt"; //暗号化ファイル
        string efile = @"C:\test\out.txt"; //復号ファイル
        string key   = @"hoge";            //パスワード
        AesEnc(sfile, dfile, key);
                AesDec(dfile, efile, key);
    }
//暗号化
static void AesEnc(string sFile, string dFile, string key)
    {
        AesCryptoServiceProvider aes = new AesCryptoServiceProvider();

        aes.BlockSize = 128;
        aes.KeySize = 128;
        aes.Mode = CipherMode.CBC;
        aes.Padding = PaddingMode.PKCS7;

        aes.GenerateIV();
        byte[] byteIV = aes.IV;

        byte[] bytesPassword = Encoding.UTF8.GetBytes(key);
        byte[] bytesKey = new byte[16];        

        for (int i = 0; i < 16; i++)
        {
            if (i < bytesPassword.Length)
            {
                bytesKey[i] = bytesPassword[i];
            }
            else
            {
                bytesKey[i] = 0;                }
        }

        aes.Key = bytesKey;

        ICryptoTransform encrypt = aes.CreateEncryptor();
        FileStream outfs = new FileStream(dFile, FileMode.Create, FileAccess.Write);
        outfs.Write(byteIV, 0, 16);

        CryptoStream cs = new CryptoStream(outfs, encrypt, CryptoStreamMode.Write);
        FileStream fs = new FileStream(sFile, FileMode.Open, FileAccess.Read);

        byte[] buf = new byte[1024];
        int len;

        while ((len = fs.Read(buf, 0, buf.Length)) > 0)
        {
            cs.Write(buf, 0, len);

        }
        fs.Close();
        cs.Close();
        encrypt.Dispose();
        outfs.Close();
    }

//復号
static void AesDec(string sFile, string dFile, string key)
    {
        AesCryptoServiceProvider aes = new AesCryptoServiceProvider();

        aes.BlockSize = 128;
        aes.KeySize = 128;
        aes.Mode = CipherMode.CBC;
        aes.Padding = PaddingMode.PKCS7;

        byte[] bytesPassword = Encoding.UTF8.GetBytes(key);
        byte[] bytesKey = new byte[16];    
        byte[] bytesIV = new byte[16];

        for (int i = 0; i < 16; i++)
        {
            if (i < bytesPassword.Length)
            {
                bytesKey[i] = bytesPassword[i];
            }
            else
            {
                bytesKey[i] = 0;
            }
        }

        aes.Key = bytesKey;

        FileStream fs = new FileStream(sFile, FileMode.Open, FileAccess.Read);

        fs.Read(bytesIV, 0, 16);
        aes.IV = bytesIV;

        ICryptoTransform encrypt = aes.CreateDecryptor();
                //①
        FileStream outfs = new FileStream(dFile, FileMode.Create, FileAccess.Write);
        CryptoStream cs = new CryptoStream(outfs, encrypt, CryptoStreamMode.Write);

        byte[] buf = new byte[1024];

        int len;

        while ((len = fs.Read(buf, 0, buf.Length)) > 0)
        {
            cs.Write(buf, 0, len);
            Console.WriteLine(Encoding.GetEncoding("shift_jis").GetString(buf));
        }

        fs.Close();
        cs.Close();
        encrypt.Dispose();
        outfs.Close();
    }
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • porepore

    2016/08/26 22:30

    Console.WriteLine(outfs.ToString());としても表示されないことから
    ほかのストリームを使用すればできるのではないかと思っています。
    (全く見当違いかもしれません・・・)

    キャンセル

  • flied_onion

    2016/08/26 22:46

    追加ありがとうございます。
    うーん、StringWriterなどにする手はありますが、気になるのは、バッファの境界に漢字の真ん中が来た場合ですかね。

    キャンセル

  • porepore

    2016/08/26 22:49

    実際の運用で使う場合はファイル容量が1kb以下になると想定しています。
    バッファも適当に決めているのでもう少し増やしても問題ないかと思います。

    キャンセル

回答 1

checkベストアンサー

0

一番簡単なのはMemoryStreamを使う事ですが、下の実装だとすべて読み終わってから出力することになります。
小さいサイズないいですが、巨大なファイルには向きません。

// AesDecの途中から
            ICryptoTransform encrypt = aes.CreateDecryptor();
            // FileStream outfs = new FileStream(dFile, FileMode.Create, FileAccess.Write);
            using (var ms = new MemoryStream()) {

                CryptoStream cs = new CryptoStream(ms, encrypt, CryptoStreamMode.Write);

                byte[] buf = new byte[1024];

                int len;

                while ((len = fs.Read(buf, 0, buf.Length)) > 0) {
                    cs.Write(buf, 0, len);
                    // Console.WriteLine(Encoding.GetEncoding("shift_jis").GetString(buf));
                }

                fs.Close();
                cs.Close();
                encrypt.Dispose();

                Console.WriteLine(Encoding.GetEncoding("shift_jis").GetString(ms.ToArray()));

            }
            // outfs.Close();
        }

ループの途中で出力する手もありますが、出力していいタイミングかの判定が難しいかもしれませんね。
とりあえず、MemoryStreamの例を載せておきます。

なお、bufは暗号化されたデータなので(bufをcsに書き込むことで、csに設定されたstreamが復号されたバイトデータを得る)、bufを使ってもデータは得られません。
あと、途中までの読み取りで、WriteLineしてしまうと元々のデータに無いタイミングで改行してしまうので、Whileの中で逐次出力したい場合はWriteにしましょう。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/08/26 23:37

    ありがとうございます。
    この方法でうまくいきました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.62%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る