| // Licensed to the Apache Software Foundation (ASF) under one or more |
| // contributor license agreements. See the NOTICE file distributed with |
| // this work for additional information regarding copyright ownership. |
| // The ASF licenses this file to You under the Apache License, Version 2.0 |
| // (the "License"); you may not use this file except in compliance with |
| // the License. You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| using System; |
| using System.IO; |
| using Apache.Arrow.Ipc; |
| using ZstdSharp; |
| |
| namespace Apache.Arrow.Compression |
| { |
| internal sealed class ZstdCompressionCodec : ITryCompressionCodec |
| { |
| private readonly Decompressor _decompressor; |
| private readonly Compressor _compressor; |
| |
| public ZstdCompressionCodec(int? compressionLevel = null) |
| { |
| if (compressionLevel.HasValue && |
| (compressionLevel.Value < Compressor.MinCompressionLevel || |
| compressionLevel.Value > Compressor.MaxCompressionLevel)) |
| { |
| throw new ArgumentException( |
| $"Zstd compression level must be between {Compressor.MinCompressionLevel} and {Compressor.MaxCompressionLevel}", |
| nameof(compressionLevel)); |
| } |
| |
| _decompressor = new Decompressor(); |
| _compressor = new Compressor(compressionLevel ?? Compressor.DefaultCompressionLevel); |
| } |
| |
| public int Decompress(ReadOnlyMemory<byte> source, Memory<byte> destination) |
| { |
| return _decompressor.Unwrap(source.Span, destination.Span); |
| } |
| |
| public void Compress(ReadOnlyMemory<byte> source, Stream destination) |
| { |
| using var compressor = new CompressionStream( |
| destination, _compressor, preserveCompressor: true, leaveOpen: true); |
| compressor.Write(source.Span); |
| } |
| |
| public bool TryCompress(ReadOnlyMemory<byte> source, Memory<byte> destination, out int bytesWritten) |
| { |
| return _compressor.TryWrap(source.Span, destination.Span, out bytesWritten); |
| } |
| |
| public void Dispose() |
| { |
| _decompressor.Dispose(); |
| _compressor.Dispose(); |
| } |
| } |
| } |