Help us understand the problem. What is going on with this article?

UnityでStreamおよび暗号化入門

本記事の対象

暗号化初心者
手を動かして学びたい
将来的にAssetBundle暗号化したい

本記事について

以下のステップでStreamと暗号化を実践的に学習する
1. ファイルをStreamで読んでByteで出力
2. 複数Byteずつ読んで出力
3. 複数Byteずつ読んで255でXORして出力
4. 読み込んだByteを適当な文字列でXORして出力
5. XORで復号化しながら読み込んだ文字列を出力
6. XORで暗号化しながら書き込む

StreamとXORの内容についての説明はしない。

関数の実行が楽なのでUnity Test Runnerを使用する。

前提

Unity Test Runner使えること (関数呼ぶだけなので別の方法でも可)

環境

Unity 2019.3.8

スクリーンショット 2020-08-01 21.51.34.png

ファイルをStreamで読んでByteで出力

helloworldという文字列をStreamで読んで出力する

まずは読み込むファイルを用意
プロジェクト/Test/sample.txt を作成

Test/sample.txt
helloworld

そしてこれをFileStreamで読む。

StreamTest.cs
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using NUnit.Framework;
using UnityEngine;


public class StreamTest
{
    string samplePath = "Test/sample.txt"; // helloworld

    [Test]
    public void Test1ファイルをStreamで読んでByteで出力()
    {
        var sb = new StringBuilder();
        using (var fs = new FileStream(samplePath, FileMode.Open))
        {

            for (int i = 0; i < fs.Length; i++)
            {
                // 1byteずつ読む
                Debug.Log($"Position {fs.Position} Length {fs.Length}");
                int b = fs.ReadByte(); // byteなので 0 ~ 255
                sb.Append(b.ToString() + " ");
            }
        }
        Debug.Log(sb.ToString());
    }
...

Stream.ReadByteで1Byteずつ取得することが可能。
また、fs.Positionとfs.Lengthも出力してみる。

結果は以下

Position 0 Length 10
Position 1 Length 10
...
Position 9 Length 10
104 101 108 108 111 119 111 114 108 100 

この結果から
Lengthは「helloworld」の文字数と一致 -> 1文字1byte
Positionは0スタート
ということがわかる。
また、byteの数値は文字コードのASCIIの10進数と一致した。
ASCII wikipedia

h -> 104
e -> 101

複数Byteまとめて読む

Test1ではReadByte()を使ったが、次からはまとめて読む。

Stream.Read

    [Test]
    public void Test2複数Byteずつ読んで出力()
    {
        var sb = new StringBuilder();

        using (var fs = new FileStream(samplePath, FileMode.Open))
        {

            for (int i = 0; i < fs.Length; i++)
            {
                byte[] buffer = new byte[8];
                // 8byteずつ読む
                int readCount = fs.Read(buffer, 0, buffer.Length);
                for (int j = 0; j < readCount; j++)
                {
                    // byteを10進数のintに変換
                    sb.Append(Convert.ToInt32(buffer[j]).ToString() + " ");
                }
            }
        }

        Debug.Log(sb.ToString());
    }

ポイント

// ReadByteの時は読み込んだ1byteが0~255の数値として帰ってくる
int b = fs.ReadByte(); // byteなので 0 ~ 255
// しかし、Readを使うと、引数のbufferに読んだ結果が格納され返り値は読んだbyte数になる。
int readCount = fs.Read(buffer, 0, buffer.Length);

結果はTest1と同様に
104 101 108 108 111 119 111 114 108 100

複数Byteずつ読んで255でXORして出力

xorについて簡単にすると、以下のような感じ

0001 0111
xor
0000 1111
=>
0001 1000

要するに比較対象が同じ=>0, 違う=>1となる。
255は1111 1111なのでxorすると以下のように0と1が逆になる。

0001 0111
xor
1111 1111
=>
1110 1000

csharpでは^を使うとxorすることができる。

それを踏まえて以下のコードを動かす。

    [Test]
    public void Test3複数Byteずつ読んで255XORして出力()
    {
        var sb = new StringBuilder();

        // xorで暗号化
        // わかりやすくするために 1111 1111をキーにする
        byte key = 255;
        using (var fs = new FileStream(samplePath, FileMode.Open))
        {

            for (int i = 0; i < fs.Length; i++)
            {
                byte[] buffer = new byte[8];
                int readCount = fs.Read(buffer, 0, buffer.Length);
                for (int j = 0; j < readCount; j++)
                {
                    buffer[j] ^= key; // ここでxor
                    sb.Append(Convert.ToInt32(buffer[j]).ToString() + " ");
                }
            }
        }

        Debug.Log(sb.ToString());
    }

元々のhelloworldが以下であり、
104 101 108 108 111 119 111 114 108 100

上記のコードの結果は以下となった。
151 154 147 147 144 136 144 141 147 155

0110 1000 // 104 <- 元。helloworldのh
1111 1111 // 255 <- key
1001 0111 // 151 <- 結果

なのでxorできている。

読み込んだByteを適当な文字列でXORして出力

適当な文字列からxorするための1byteを取得するようにする。

文字列は1234abcdとする。
文字列をByte化する方法は以下

        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);

GetBytes

あとはこの配列のインデックスをずらせばOK

以下のコードで実行する。

    [Test]
    public void Test4読み込んだByteを適当な文字列でXORして出力()
    {
        var sb = new StringBuilder();
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);
        foreach (var b in keyBytes)
        {
            sb.Append(Convert.ToInt32(b).ToString() + " ");
        }
        Debug.Log(sb.ToString());
        sb.Length = 0;

        using (var fs = new FileStream(samplePath, FileMode.Open))
        {
            for (int i = 0; i < fs.Length; i++)
            {
                byte[] buffer = new byte[8];
                // Positionを保持 (0スタート)
                var startPosition = fs.Position;
                // 8byteずつ読む
                int readCount = fs.Read(buffer, 0, buffer.Length); // byteなので 0 ~ 255
                for (int j = 0; j < readCount; j++)
                {
                    // byteの位置からkeyのindexを算出
                    var keyIndex = (startPosition + j) % keyBytes.Length;
                    buffer[j] ^= keyBytes[keyIndex]; // xor
                    // byteを10進数のintに変換
                    sb.Append(Convert.ToInt32(buffer[j]).ToString() + " ");
                }
            }
        }

        Debug.Log(sb.ToString());
    }

文字列 1234abcd
49 50 51 52 97 98 99 100

結果
89 87 95 88 14 21 12 22 93 86

確認

// 1byte目
0110 1000 // 104 h
0011 0001 // 49 1
0101 1001 // 89 結果 
なので1OK

// 8byte目
0111 0010 // 114 r
0110 0100 // 100 d
0001 0110 // 22 結果 
なのでok

// 9byte目
0110 1100 // 108 l
0011 0001 // 49 1 (2週目 1234abcd  1 <-)
0101 1101 // 93 結果
でOK

読み書きする際にXORするStreamを自作する。

事前にキーとなる文字列を渡しておいて、それを使ってXORするStreamを自作。

具体的にはReadとWriteを自作する。

    public class XORCryptStream : Stream
    {
        Stream baseStream;
        byte[] key;

        public XORCryptStream(Stream baseStream, byte[] key)
        {
            this.baseStream = baseStream;
            this.key = key;
        }

        public override bool CanRead => baseStream.CanRead;

        public override bool CanSeek => baseStream.CanSeek;

        public override bool CanWrite => baseStream.CanWrite;

        public override long Length => baseStream.Length;

        public override long Position { get { return baseStream.Position; } set { baseStream.Position = value; } }

        public override void Flush() => baseStream.Flush();

        public override long Seek(long offset, SeekOrigin origin) => baseStream.Seek(offset, origin);

        public override void SetLength(long value) => baseStream.SetLength(value);

        /// <summary>
        /// 復号化しつつ読む
        /// </summary>
        /// <param name="buffer">結果を格納するbuffer</param>
        /// <param name="offset">bufferのoffset番目から格納</param>
        /// <param name="count">何byte読み込むか</param>
        /// <returns>読み取ったbyte数を返す</returns>
        public override int Read(byte[] buffer, int offset, int count)
        {
            var startPosition = Position;
            int readCount = baseStream.Read(buffer, offset, count);

            int keyIndex = (int)(startPosition % key.Length);
            for (int i = 0; i < readCount; i++)
            {
                int bufferIndex = i + offset;
                if (keyIndex == key.Length)
                {
                    keyIndex = 0;
                }
                buffer[bufferIndex] ^= key[keyIndex]; // xor
                keyIndex++;
            }
            return readCount;
        }

        public override void Write(byte[] buffer, int offset, int count)
        {
            if (buffer.Length < offset + count)
            {
                throw new ArgumentException("offset + count is larger than buffer.Length");
            }

            byte[] writeBuffer = new byte[count];
            var startPosition = Position;
            int keyIndex = (int)(startPosition % key.Length);
            for (int i = 0; i < count; i++)
            {
                int bufferIndex = i + offset;
                if (keyIndex == key.Length)
                {
                    keyIndex = 0;
                }

                writeBuffer[i] = buffer[bufferIndex];
                writeBuffer[i] ^= key[keyIndex];
                keyIndex++;
            }
            baseStream.Write(writeBuffer, 0, count);
        }
    }

baseStreamをラップし、Read, Writeの際に事前に与えていたkeyでxorする。

このクラスは以下のようにテストを行った。

    [Test]
    public void TestXORCryptStream_Read()
    {
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);

        using (var fs = new FileStream(xorEncryptedSamplePath, FileMode.Open))
        using (var cs = new XORCryptStream(fs, keyBytes))
        {
            // offsetが機能しているか
            byte[] buffer1 = new byte[3];
            int readCount1 = cs.Read(buffer1, 1, 2);
            Assert.AreEqual(2, readCount1);
            Assert.AreEqual(0, Convert.ToInt32(buffer1[0]));
            Assert.AreEqual(104, Convert.ToInt32(buffer1[1]));
            Assert.AreEqual(101, Convert.ToInt32(buffer1[2]));

            byte[] buffer2 = new byte[4];
            int readCount2 = cs.Read(buffer2, 2, 1);
            Assert.AreEqual(1, readCount2);
            Assert.AreEqual(0, Convert.ToInt32(buffer2[0]));
            Assert.AreEqual(0, Convert.ToInt32(buffer2[1]));
            Assert.AreEqual(108, Convert.ToInt32(buffer2[2]));
            Assert.AreEqual(0, Convert.ToInt32(buffer2[3]));
        }
    }

    [Test]
    public void TestXORCryptStream_Write()
    {
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);

        using (var ms = new MemoryStream())
        using (var cs = new XORCryptStream(ms, keyBytes))
        {
            byte[] encryptedHello = new byte[] { 89, 87, 95, 88, 14 };
            byte[] encryptedWorld = new byte[] { 0, 0, 21, 12, 22, 93, 86 };

            cs.Write(encryptedHello, 0, 5);
            cs.Write(encryptedWorld, 2, 5);

            Assert.AreEqual(10, ms.Length);
            ms.Seek(0L, SeekOrigin.Begin);
            byte[] buffer = new byte[10];
            ms.Read(buffer, 0, buffer.Length);
            Assert.AreEqual("helloworld", Encoding.UTF8.GetString(buffer));
        }
    }

また、ファイルの読み込みを行うために暗号化されたファイルを作っておく

    string xorEncryptedSamplePath = "Test/sample2.txt"; // helloworldを1234abcdをキーにxor暗号化したもの

...

    [Test]
    [Ignore("書き出すときだけ外す")]
    public void samplePathを暗号化して保存()
    {
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);
        List<byte> outputBytes = new List<byte>();

        using (var fs = new FileStream(samplePath, FileMode.Open))
        {
            for (int i = 0; i < fs.Length; i++)
            {
                byte[] buffer = new byte[8];
                // Positionを保持 (0スタート)
                var startPosition = fs.Position;
                // 8byteずつ読む
                int readCount = fs.Read(buffer, 0, buffer.Length); // byteなので 0 ~ 255
                for (int j = 0; j < readCount; j++)
                {
                    // byteの位置からkeyのindexを算出
                    var keyIndex = (startPosition + j) % keyBytes.Length;
                    buffer[j] ^= keyBytes[keyIndex]; // xor
                    outputBytes.Add(buffer[j]);
                }
            }
        }

        File.WriteAllBytes(xorEncryptedSamplePath, outputBytes.ToArray());
    }

そして、その生成したファイルを使って読み込みと書き込み

    [Test]
    public void Test5XOR復号化しながら読み込んだ文字列を出力()
    {
        var sb = new StringBuilder();
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);

        List<byte> bytes = new List<byte>();

        using (var fs = new FileStream(xorEncryptedSamplePath, FileMode.Open))
        using (var cs = new XORCryptStream(fs, keyBytes))
        {
            byte[] buffer = new byte[8];
            while (true){
                // Positionを保持 (0スタート)
                var startPosition = cs.Position;
                // 8byteずつ読む
                int readCount = cs.Read(buffer, 0, buffer.Length);
                if (readCount == 0)
                {
                    break;
                }

                for (int i = 0; i < readCount; i++)
                {
                    bytes.Add(buffer[i]);
                }
            }
        }
        string text = System.Text.Encoding.ASCII.GetString(bytes.ToArray());
        Assert.AreEqual("helloworld", text);
    }

    [Test]
    public void Test6XOR暗号化しながら書き込む()
    {
        string key = "1234abcd";
        byte[] keyBytes = Encoding.UTF8.GetBytes(key);

        using (var fileReader = new FileStream(samplePath, FileMode.Open))
        using (var fileWriter = new FileStream(test6ResultPath, FileMode.Create))
        using (var cs = new XORCryptStream(fileWriter, keyBytes))
        {
            byte[] buffer = new byte[8];
            while(true){
                // Positionを保持 (0スタート)
                var startPosition = fileReader.Position;
                // 8byteずつ読む
                int readCount = fileReader.Read(buffer, 0, buffer.Length); // byteなので 0 ~ 255
                if (readCount == 0)
                {
                    break;
                }
                cs.Write(buffer, 0, readCount);
            }
        }
        Assert.AreEqual(File.ReadAllText(xorEncryptedSamplePath), File.ReadAllText(test6ResultPath));
    }

これでStream, XORを試すことができた。

参考

Stream microsoft doc

Why not register and get more from Qiita?
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
Comments
No comments
Sign up for free and join this conversation.
If you already have a Qiita account
Why do not you register as a user and use Qiita more conveniently?
You need to log in to use this function. Qiita can be used more conveniently after logging in.
You seem to be reading articles frequently this month. Qiita can be used more conveniently after logging in.
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
ユーザーは見つかりませんでした