-
-
Notifications
You must be signed in to change notification settings - Fork 240
/
Copy pathCompactor.vb
148 lines (98 loc) · 5.2 KB
/
Compactor.vb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
Imports System.IO
Imports System.Runtime.InteropServices
Imports System.Threading
Public Class Compactor : Implements IDisposable
Public Sub New(folder As String, cLevel As CompressionAlgorithm, excludedFilesTypes As String())
If Not verifyFolder(folder).isValid Then Return
_workingDir = folder
_excludedFileTypes = excludedFilesTypes
_WOFCompressionLevel = cLevel
_EFInfo = New WOF_FILE_COMPRESSION_INFO_V1 With {.Algorithm = _WOFCompressionLevel, .Flags = 0}
_EFInfoPtr = Marshal.AllocHGlobal(Marshal.SizeOf(_EFInfo))
Marshal.StructureToPtr(_EFInfo, _EFInfoPtr, True)
End Sub
Private _workingDir As String
Private _excludedFileTypes() As String
Private _WOFCompressionLevel As CompressionAlgorithm
Private _EFInfo As WOF_FILE_COMPRESSION_INFO_V1
Private _EFInfoPtr As IntPtr
Private _pauseSemaphore As New SemaphoreSlim(1, 2)
Private _processedFilesBytes As Long = 0
Private _cancellationTokenSource As New CancellationTokenSource
Public Async Function RunCompactAsync(Optional progressMonitor As IProgress(Of (percentageProgress As Integer, currentFile As String)) = Nothing, Optional MaxParallelism As Integer = 1) As Task(Of Boolean)
If _cancellationTokenSource.IsCancellationRequested Then Return False
Dim FilesList = Await BuildWorkingFilesList()
Dim totalFilesSize As Long = FilesList.Sum(Function(f) f.Item2)
_processedFilesBytes = 0
If MaxParallelism <= 0 Then MaxParallelism = Environment.ProcessorCount
Dim sw As New Stopwatch
sw.Start()
Dim paraOptions As New ParallelOptions With {.MaxDegreeOfParallelism = MaxParallelism}
Await Parallel.ForEachAsync(FilesList, paraOptions,
Function(file, _ctx) As ValueTask
If _ctx.IsCancellationRequested Then Return ValueTask.FromCanceled(_ctx)
Return New ValueTask(PauseAndProcessFile(file.Item1, _cancellationTokenSource.Token, file.Item2, totalFilesSize, progressMonitor))
End Function).ConfigureAwait(False)
sw.Stop()
Debug.WriteLine($"Completed in {sw.Elapsed.TotalSeconds} s")
If _cancellationTokenSource.IsCancellationRequested Then Return False
Return True
End Function
Private Async Function PauseAndProcessFile(file As String, _ctx As CancellationToken, fileSize As Long, totalFilesSize As Long, Optional progressMonitor As IProgress(Of (percentageProgress As Integer, currentFile As String)) = Nothing) As Task
If _ctx.IsCancellationRequested Then Return
Try
Await _pauseSemaphore.WaitAsync(_ctx).ConfigureAwait(False)
_pauseSemaphore.Release()
Catch ex As OperationCanceledException
Return
End Try
If _ctx.IsCancellationRequested Then Return
Dim res = WOFCompressFile(file)
Interlocked.Add(_processedFilesBytes, fileSize)
Dim incremented = _processedFilesBytes
progressMonitor?.Report((CInt(((incremented / totalFilesSize) * 100)), file))
End Function
Public Sub PauseCompression()
_pauseSemaphore.Wait()
End Sub
Public Sub ResumeCompression()
If _pauseSemaphore.CurrentCount = 0 Then _pauseSemaphore.Release()
End Sub
Public Sub Cancel()
ResumeCompression()
_cancellationTokenSource.Cancel()
End Sub
Private Function WOFCompressFile(path As String)
Dim length As ULong = Marshal.SizeOf(_EFInfoPtr)
Try
Using fs As FileStream = New FileStream(path, FileMode.Open)
Dim hFile = fs.SafeFileHandle.DangerousGetHandle()
Dim res = WofSetFileDataLocation(hFile, WOF_PROVIDER_FILE, _EFInfoPtr, length)
Return res
End Using
Catch ex As Exception
Debug.WriteLine(ex.Message)
Return Nothing
End Try
End Function
Private Async Function BuildWorkingFilesList() As Task(Of IEnumerable(Of (String, Long)))
Dim clusterSize As Integer = GetClusterSize(_workingDir)
Dim _filesList As New Concurrent.ConcurrentBag(Of (String, Long))
'TODO: if the user has already analysed within the last minute, then skip creating a new one and use the old one
Dim ax As New Analyser(_workingDir)
Dim ret = Await ax.AnalyseFolder(Nothing)
Parallel.ForEach(ax.FileCompressionDetailsList, Sub(fl)
Dim ft = fl.FileInfo
If Not _excludedFileTypes.Contains(ft.Extension) AndAlso ft.Length > clusterSize AndAlso fl.CompressionMode <> _WOFCompressionLevel Then _filesList.Add((fl.FileName, fl.UncompressedSize))
End Sub)
Return _filesList.ToList
End Function
Public Sub Dispose() Implements IDisposable.Dispose
_cancellationTokenSource.Dispose()
_pauseSemaphore.Dispose()
If _EFInfoPtr <> IntPtr.Zero Then
Marshal.FreeHGlobal(_EFInfoPtr)
_EFInfoPtr = IntPtr.Zero
End If
End Sub
End Class