Skip to content

egbakou/reedsolomon

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

55 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

IconReedSolomon.NET

CI NuGet

Reed-Solomon Erasure coding implementation in C#/.NET.

This is a .NET Standard 2.1 port of the JavaReedSolomon library released by Backblaze, without additional optimizations.

What is Reed-Solomon?

Reed-Solomon Erasure Coding is a method of data protection that uses a specific type of error-correcting code. It is designed to provide protection against data loss in case of failures or errors. It works by dividing data into smaller blocks and then encoding each block with extra information. This extra information allows the system to reconstruct the original data even if some of the blocks are lost or corrupted. Reed-Solomon Erasure Coding is commonly used in object storage systems, distributed file systems, and other types of storage systems where data durability and reliability are important.

Here is a well-explained article about the topic by Backblaze : https://www.backblaze.com/blog/reed-solomon/

Motivation

I had a school project last year about creating an object storage system. I choose not to use Azure Blob Storage or Amazon S3 as I had exhausted my student credits. I researched various open-source alternatives such as minio, nextcloud, and seaweedfs, but ultimately decided to develop my own S3 compatible storage system from scratch. I set up four virtual machines, each with two disks. However, the virtual machines were in an unstable environment and I aimed to avoid data replication across them. Through my research, I discovered that minio uses the Reed-Solomon Erasure Coding method. Azure Blob Storage also uses Erasure Coding, but I am not sure if they use the Reed-Solomon version. This is how the project came to realization.

I followed the Java implementation released by Backblaze.

Installation

ReedSolomon.NET can be found here on NuGet and can be installed by copying and pasting the following command into your Package Manager Console within Visual Studio (Tools > NuGet Package Manager > Package Manager Console).

Install-Package ReedSolomon.NET

Alternatively, you can install ReedSolomon.NET via the CLI with the following command:

dotnet add package ReedSolomon.NET

Usage

Take a look at the ReedSolomon.NET.Sample project in the src folder.

You will find these three files: SampleEncoder, SampleDecoder, and RemoteDecoder. They show, in a very simple way, how to break a file into data chunks and parity chunks, and then how to take a subset of the shards and reconstruct the original file.

  • SampleEncoder.cs

Break a file into data chunks and parity chunks.

  • SampleDecoder.cs

Reconstruct the original file from data chunks and/or parity chunks.

In the illustration below, at least 4 "data chunks + Parity chunks" must be available, otherwise, the reconstruction will not be possible.

Performance

The performance of the inner loop depends on the specific processor you're running on. There are twelve different permutations of the loop in this library, and the ReedSolomonBenchmark class will tell you which one is faster for your particular application.

These are the speeds I got running the benchmark on an AMD Ryzen 5 computer.

InputOutputByteTableCodingLoop 304.8 MB/s
OutputInputByteTableCodingLoop 296.8 MB/s
InputOutputByteExpCodingLoop 258.3 MB/s
OutputByteInputExpCodingLoop 254.5 MB/s
OutputInputByteExpCodingLoop 241.6 MB/s
ByteOutputInputTableCodingLoop 234.6 MB/s
ByteOutputInputExpCodingLoop 234.2 MB/s
OutputByteInputTableCodingLoop 232.5 MB/s
InputByteOutputTableCodingLoop 152.0 MB/s
ByteInputOutputTableCodingLoop 145.5 MB/s
InputByteOutputExpCodingLoop 134.3 MB/s
ByteInputOutputExpCodingLoop 127.1 MB/s

Limitation

  • File size can't exceed 2GB due to the byte array limitation.

When I was working on my project, the maximum size of any file was 1GB, so I didn't care about this.

If you want to improve this project, check out the NativeMemoryArray repo to remove this limitation and feel free to make a PR.

  • See the Go implementation to improve the performance of this project.

  • You must run the benchmark on your hardware to detect the best inner loop to use.

Shortcomings

  • If the shard number isn't the same for the decoder as in the encoder, invalid output will be generated.
  • If values have changed in a shard, the original file cannot be reconstructed.
  • If two shards have been swapped, reconstruction will always fail. You need to supply the shards in the same order as they were given to you.

The solution for this is to save a metadata file containing:

  • File size.

  • The number of data/parity shards.

  • HASH of each shard.

  • Order of the shards.

If you save these properties, you should be able to detect file corruption in a shard and be able to reconstruct your data if you have the needed number of shards left.

For security reasons, you can also encrypt each shard.

Links

License

This project is licensed under the terms of the MIT license.