using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Threading.Tasks; using ZeroLevel.Services.PartitionStorage.Interfaces; using ZeroLevel.Services.Serialization; namespace ZeroLevel.Services.PartitionStorage { /// /// Responsible for building index files /// internal sealed class IndexBuilder { private const string INDEX_SUBFOLDER_NAME = "__indexes__"; private readonly IndexStepType _indexType; private readonly string _indexCatalog; private readonly string _dataCatalog; private readonly int _stepValue; private readonly IStoreKVSerializer Serializer; private readonly PhisicalFileAccessorCachee _phisicalFileAccessorCachee; public IndexBuilder(IndexStepType indexType, int stepValue, string dataCatalog, PhisicalFileAccessorCachee phisicalFileAccessorCachee, IStoreKVSerializer serializer) { _dataCatalog = dataCatalog; _indexCatalog = Path.Combine(dataCatalog, INDEX_SUBFOLDER_NAME); _indexType = indexType; _stepValue = stepValue; Serializer = serializer; _phisicalFileAccessorCachee = phisicalFileAccessorCachee; } /// /// Rebuild indexes for all files /// internal async Task RebuildIndex() { var files = Directory.GetFiles(_dataCatalog); if (files != null && files.Length > 0) { foreach (var file in files) { await RebuildFileIndex(Path.GetFileName(file)); } } } /// /// Rebuild index for the specified file /// internal async Task RebuildFileIndex(string file) { if (_indexType == IndexStepType.AbsoluteCount) { await RebuildFileIndexWithAbsoluteCountIndexes(file); } else { await RebuildFileIndexWithSteps(file); } } /// /// Delete the index for the specified file /// internal void DropFileIndex(string file) { var index_file = Path.Combine(_indexCatalog, Path.GetFileName(file)); _phisicalFileAccessorCachee.LockFile(index_file); try { if (File.Exists(index_file)) { File.Delete(index_file); } } finally { _phisicalFileAccessorCachee.UnlockFile(index_file); } } /// /// Rebuild index with specified number of steps for specified file /// private async Task RebuildFileIndexWithAbsoluteCountIndexes(string file) { if (false == Directory.Exists(_indexCatalog)) { Directory.CreateDirectory(_indexCatalog); } var dict = new Dictionary(); using (var reader = new MemoryStreamReader(new FileStream(Path.Combine(_dataCatalog, file), FileMode.Open, FileAccess.Read, FileShare.None))) { while (reader.EOS == false) { var pos = reader.Position; var k = await Serializer.KeyDeserializer.Invoke(reader); dict[k.Value] = pos; await Serializer.ValueDeserializer.Invoke(reader); } } if (dict.Count > _stepValue) { var step = (int)Math.Round(dict.Count / (float)_stepValue, MidpointRounding.ToZero); var index_file = Path.Combine(_indexCatalog, Path.GetFileName(file)); _phisicalFileAccessorCachee.LockFile(index_file); if (File.Exists(index_file)) { File.Delete(index_file); } try { var d_arr = dict.OrderBy(p => p.Key).ToArray(); using (var writer = new MemoryStreamWriter(new FileStream(index_file, FileMode.Create, FileAccess.Write, FileShare.None))) { for (int i = 0; i < _stepValue; i++) { var pair = d_arr[i * step]; await Serializer.KeySerializer.Invoke(writer, pair.Key); writer.WriteLong(pair.Value); } } } finally { _phisicalFileAccessorCachee.UnlockFile(index_file); } } } /// /// Rebuild index with specified step for keys /// private async Task RebuildFileIndexWithSteps(string file) { if (false == Directory.Exists(_indexCatalog)) { Directory.CreateDirectory(_indexCatalog); } using (var reader = new MemoryStreamReader(new FileStream(Path.Combine(_dataCatalog, file), FileMode.Open, FileAccess.Read, FileShare.None))) { var index_file = Path.Combine(_indexCatalog, Path.GetFileName(file)); _phisicalFileAccessorCachee.LockFile(index_file); if (File.Exists(index_file)) { File.Delete(index_file); } try { using (var writer = new MemoryStreamWriter(new FileStream(index_file, FileMode.Create, FileAccess.Write, FileShare.None))) { var counter = 1; while (reader.EOS == false) { counter--; var pos = reader.Position; var k = await Serializer.KeyDeserializer.Invoke(reader); await Serializer.ValueDeserializer.Invoke(reader); if (counter == 0) { await Serializer.KeySerializer.Invoke(writer, k.Value); writer.WriteLong(pos); counter = _stepValue; } } } } finally { _phisicalFileAccessorCachee.UnlockFile(index_file); } } } } }