forked from svn2github/freearc
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathArhiveDirectory.hs
413 lines (348 loc) · 22.9 KB
/
ArhiveDirectory.hs
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
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
----------------------------------------------------------------------------------------------------
---- Работа с оглавлением архива. ------
---- Этот модуль содержит процедуры для: ------
---- * чтения структуры входного архива (т.е. каталогов и других служебных блоков) ------
---- * записи и чтения каталогов архива ------
----------------------------------------------------------------------------------------------------
module ArhiveDirectory where
import Prelude hiding (catch)
import Control.Monad
import Data.HashTable as Hash
import Data.List
import Foreign.Marshal.Pool
import System.Mem
import GHC.PArr
import Utils
import Errors
import Files
import qualified ByteStream
import FileInfo
import Compression (CRC, isFakeCompressor)
import UI (debugLog)
import Options
import ArhiveStructure
----------------------------------------------------------------------------------------------------
---- Чтение структуры входного архива --------------------------------------------------------------
----------------------------------------------------------------------------------------------------
-- |Вся необходимая информация о входном архиве
data ArchiveInfo = ArchiveInfo
{ arcArchive :: Archive -- открытый файл архива
, arcFooter :: FooterBlock -- FOOTER BLOCK архива
, arcDirectory :: [CompressedFile] -- файлы, содержащиеся в архиве
, arcDataBlocks :: [ArchiveBlock] -- список солид-блоков
, arcDirBytes :: FileSize -- размер служебных блоков в распакованном виде
, arcDirCBytes :: FileSize -- размер служебных блоков в упакованном виде
, arcDataBytes :: FileSize -- размер данных в распакованном виде
, arcDataCBytes :: FileSize -- размер данных в упакованном виде
, arcPhantom :: Bool -- True, если архива на самом деле нет (используется для main_archive)
}
-- Процедуры, упрощающие работу с архивами
arcGetPos = archiveGetPos . arcArchive
arcSeek = archiveSeek . arcArchive
arcComment = ftComment . arcFooter
-- |Фантомный, несуществующий архив, необходимый для применения в некоторых операциях
-- (слияние списков файлов, закрытие входных архивов)
phantomArc = (dirlessArchive (error "phantomArc:arcArchive") (FooterBlock [] False "" "" 0)) {arcPhantom = True}
-- |Архив без каталога файлов - используется только для вызова writeSFX из runArchiveRecovery
dirlessArchive archive footer = ArchiveInfo archive footer [] [] (error "emptyArchive:arcDirBytes") (error "emptyArchive:arcDirCBytes") (error "emptyArchive:arcDataBytes") (error "emptyArchive:arcDataCBytes") False
-- |Закрыть архивный файл, если только это не фантомный архив
arcClose arc = unless (arcPhantom arc) $ do archiveClose (arcArchive arc)
{-# NOINLINE archiveReadInfo #-}
-- |Прочитать каталог архива
archiveReadInfo command -- выполняемая команда со всеми её опциями
arc_basedir -- базовый каталог внутри архива ("" для команд добавления)
disk_basedir -- базовый каталог на диске ("" для команд добавления/листинга)
filter_f -- предикат для фильтрации списка файлов в архиве
processFooterInfo -- процедура, выполняемая на данных из FOOTER_BLOCK
arcname = do -- имя файла, содержащего архив
-- Прочитать FOOTER_BLOCK и выполнить на нём переданную процедуру
(archive,footer) <- if opt_broken_archive command /= "-"
then findBlocksInBrokenArchive arcname
else archiveReadFooter command arcname
processFooterInfo archive footer
-- Прочитаем содержимое блоков каталога, описанных в FOOTER_BLOCK
let dir_blocks = filter ((DIR_BLOCK==).blType) (ftBlocks footer)
files <- foreach dir_blocks $ \block -> do
withPool $ \pool -> do
(buf,size) <- archiveBlockReadAll pool (opt_decryption_info command) block
archiveReadDir arc_basedir disk_basedir (opt_dir_exclude_path command) archive (blPos block) filter_f (return (buf,size))
let data_blocks = concatMap fst files
directory = concatMap snd files
-- Добавим в arcinfo информацию о списке файлов в архиве
return ArchiveInfo { arcArchive = archive
, arcFooter = footer
, arcDirectory = directory
, arcDataBlocks = data_blocks
, arcDirBytes = sum (map blOrigSize dir_blocks)
, arcDirCBytes = sum (map blCompSize dir_blocks)
, arcDataBytes = sum (map blOrigSize data_blocks)
, arcDataCBytes = sum (map blCompSize data_blocks)
, arcPhantom = False
}
{-# NOINLINE archiveReadFooter #-}
-- |Прочитать финальный блок архива
archiveReadFooter command -- выполняемая команда со всеми её опциями
arcname = do -- имя файла, содержащего архив
archive <- archiveOpen arcname
arcsize <- archiveGetSize archive
let scan_bytes = min aSCAN_MAX arcsize -- сканируем 4096 байт в конце архива, если столько найдётся :)
withPool $ \pool -> do
-- Прочитаем 4096 байт в конце архива, которые должны содержать дескриптор FOOTER_BLOCK'а
buf <- archiveMallocReadBuf pool archive (arcsize-scan_bytes) (i scan_bytes)
-- Найдём и декодируем последний дескриптор архива (это должен быть дескриптор FOOTER_BLOCK'а)
res <- archiveFindBlockDescriptor archive (arcsize-scan_bytes) buf (i scan_bytes) (i scan_bytes)
case res of
Left msg -> registerError msg
Right footer_descriptor -> do
-- Прочитаем FOOTER_BLOCK, описываемый этим дескриптором, целиком в буфер и декодируем его содержимое
footer <- archiveReadFooterBlock footer_descriptor (opt_decryption_info command)
return (archive,footer)
----------------------------------------------------------------------------------------------------
---- Запись блока каталога -------------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
{-# NOINLINE archiveWriteDir #-}
-- |Закодировать `dirdata` и послать полученные данные на дальнейшую обработку с помощью `sendBuf`
archiveWriteDir dirdata -- список пар (block :: ArchiveBlock, directory :: [FileWithCRC])
arcpos -- позиция в архиве, где начинается этот каталог
(receiveBuf -- "(buf,size) <- receiveBuf" получает для работы очередной буфер размером `size`
,sendBuf) -- "sendBuf buf size len" посылает сформированные в буфере данные на выход
= do
debugLog "\n Writing directory"
let blocks = map fst dirdata :: [ArchiveBlock] -- список солид-блоков, попавших в данный каталог
crcfilelist = concatMap snd dirdata :: [FileWithCRC] -- объединённый список файлов - в том порядке, в каком они расположены в блоках!
filelist = map fwFileInfo crcfilelist :: [FileInfo] -- информация о самих файлах
-- 0. Cоздадим выходной буфер, использующий для общения с внешним миром функции `receiveBuf` и `sendBuf`
stream <- ByteStream.create receiveBuf sendBuf (return ())
let write :: (ByteStream.BufferData a) => a -> IO () -- shortcuts для функций записи в буфер
write = ByteStream.write stream
writeLength = ByteStream.writeInteger stream . length
writeList :: (ByteStream.BufferData a) => [a] -> IO ()
writeList = ByteStream.writeList stream
writeIntegers = mapM_ (ByteStream.writeInteger stream)
writeTagged tag x = write tag >> write x -- запись с тегами - для опциональных полей
writeTaggedList tag xs = write tag >> writeList xs
-- 1. Закодируем описания блоков архива и кол-во файлов в каждом из них
writeLength dirdata -- кол-во блоков. Для каждого блока записывается:
mapM_ (writeLength.snd) dirdata -- кол-во файлов
writeList$ map (blCompressor ) blocks -- метод сжатия
writeList$ map (blEncodePosRelativeTo arcpos) blocks -- относительная позиция блока в файле архива
writeList$ map (blCompSize ) blocks -- размер блока в упакованном виде
-- 2. Запишем в архив список имён каталогов
-- Получим список имён каталогов и номера каталогов, соответствующие файлам в filelist
(n, dirnames, dir_numbers) <- enumDirectories filelist
debugLog$ " Found "++show n++" directory names"
writeLength dirnames -- временно, для обхода проблемы с Compressor==[String]
writeList dirnames
-- 3. Закодируем отдельно каждое оставшееся поле в CompressedFile/FileInfo
-- to do: добавить RLE-кодирование полей?
writeList$ map (fpBasename.fiStoredName) filelist -- имена файлов
writeIntegers dir_numbers -- номера каталогов
writeList$ map fiSize filelist -- размеры файлов
writeList$ map fiTime filelist -- времена создания
writeList$ map fiIsDir filelist -- признаки каталога
-- cfArcBlock и cfPos кодируются неявно, путём сортировки по этим двум полям
writeList$ map fwCRC crcfilelist -- CRC
-- 4. Опциональные поля, префиксируются своими тегами, в конце - тег окончания опциональных полей
write aTAG_END -- пока опциональных полей нет, нам остаётся только сразу записать тег их окончания
-- 5. Вотысё! :)
ByteStream.closeOut stream
-- Это приводит к вылету Arc.exe!!! - when (length filelist >= 10000) performGC -- Соберём мусор, если блок содержит достаточно много файлов
debugLog " Directory written"
-- Создание по списку файлов - списка уникальных каталогов + номер каталога для каждого файла в списке
enumDirectories filelist = do
-- Для каждого Stored имени файла мы ищем имя с тем же каталогом в хеш-таблице `table`.
-- Если оно найдено, то мы получаем из хеш-таблицы номер этого каталога,
-- а если нет - вставляем это имя в хеш-таблицу с очередным порядковым номером, которые
-- генерятся через переменную n, и добавляем имя каталога в список `dirnames`.
-- Таким образом, хеш-таблица `table` отображает имена каталогов в их номера
-- в создаваемом списке всех каталогов `dirnames`.
table <- Hash.new (==) fpHash -- отображает каталоги в их номера
-- Возвратить для списка файлов количество уникальных имён каталогов, их полный список,
-- и номер каталога для каждого файла (например, [0,1,0,0,2] для a\1 b\1 a\2 a\3 c\1)
let go [] dirnames dir_numbers n = return (n, reverse dirnames, reverse dir_numbers)
go (fileinfo:rest) dirnames dir_numbers n = do
let storedName = fiStoredName fileinfo -- имя, предназначенное для сохранения в архиве
dirname = fpParent storedName -- каталог, к которому принадлежит файл
x <- Hash.lookup table dirname -- есть ли уже в хеше этот каталог?
case x of -- Если нет, то
Nothing -> do Hash.insert table dirname n -- занести в хеш номер каталога
-- Добавить имя каталога в список имён каталогов,
-- номер каталога в список номеров каталога для каждого файла,
-- и инкрементировать счётчик каталогов
go rest (fpDirectory storedName:dirnames) (n:dir_numbers) $! n+1
Just x -> do go rest dirnames (x:dir_numbers) n
--
go filelist [] [] (0::FileCount)
----------------------------------------------------------------------------------------------------
---- Чтение блока каталога -------------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
{-# NOINLINE archiveReadDir #-}
-- |Прочитать каталог, записанный функцией `archiveWriteDir`
archiveReadDir arc_basedir -- базовый каталог в архиве
disk_basedir -- базовый каталог на диске
ep -- исключить каталоги из имён/разрешить абсолютные пути
archive -- файл архива
arcpos -- позиция в архиве, где начинается этот каталог
filter_f -- предикат фильтрации файлов
receiveBuf -- "(buf,size) <- receiveBuf" получает для работы очередной буфер размером `size`
= do
debugLog " Decoding directory"
-- 0. Cоздадим входной буфер, использующий для общения с внешним миром функцию `receiveBuf`
stream <- ByteStream.open receiveBuf (\a b c->return ()) (return ())
let read :: (ByteStream.BufferData a) => IO a -- shortcuts для функций чтения из буфера
read = ByteStream.read stream
readList :: (ByteStream.BufferData a) => Int -> IO [a]
readList = ByteStream.readList stream
readInteger = ByteStream.readInteger stream
readLength = readInteger
readIntegers n = replicateM n readInteger
-- 1. Прочитаем описания блоков архива
num_of_blocks <- readLength -- кол-во блоков
-- Для каждого блока прочитаем:
num_of_files <- readIntegers num_of_blocks -- кол-во файлов
blCompressors <- readList num_of_blocks -- метод сжатия
blOffsets <- readList num_of_blocks -- относительную позицию блока в файле архива
blCompSizes <- readList num_of_blocks -- размер блока в упакованном виде
-- 2. Прочитаем имена каталогов
total_dirs <- readLength -- Сколько всего имён каталогов сохранено в этом оглавлении архива
storedName <- readList total_dirs >>== toP -- Массив имён каталогов
-- 3. Прочитаем списки данных для каждого поля в CompressedFile/FileInfo
let total_files = sum num_of_files -- суммарное кол-во файлов в каталоге
names <- readList total_files -- Имена файлов (без имени каталога)
dir_numbers <- readIntegers total_files -- Номер каталога для каждого из файлов
sizes <- readList total_files -- Размеры файлов
times <- readList total_files -- Время модификации файлов
dir_flags <- readList total_files -- Булевские флаги "это каталог?"
crcs <- readList total_files -- CRC файлов
-- 4. Дополнительные поля, префиксируются своими тегами, в конце - тег окончания дополнительных полей
{-repeat_while (read) (/=aTAG_END) $ \tag -> do
(isMandatory::Bool) <- read
when isMandatory $ do
registerError$ GENERAL_ERROR ("can't skip mandatory field TAG="++show tag++" in archive directory")
readInteger >>= ByteStream.skipBytes stream -- пропустить данные этого поля
return ()
-}
-- 5. Вотысё! :)
ByteStream.closeIn stream
debugLog " Directory decoded"
------------------------------------------------------------------------------------------------
-- Теперь построим каталог по прочитанным данным -----------------------------------------------
------------------------------------------------------------------------------------------------
-- Массивы, содержащие информацию о каталогах
let drop_arc_basedir = if arc_basedir>"" then drop (length arc_basedir + 1) else id
make_disk_name = case ep of -- Превращает имя в архиве в имя на диске
0 -> const "" -- команда "e" -> использовать только базовое имя
3 -> id -- опция -ep3 -> использовать полное имя
_ -> stripRoot -- по умолчанию -> обрезать "d:\" часть
-- Массивы, отображающие номер каталога в его Filtered/Disk name (массив для Stored name построен сразу при чтении)
filteredName = fmap (drop_arc_basedir) storedName
diskName = fmap ((disk_basedir </>) . make_disk_name) filteredName
-- Массивы, отображающие номер каталога в структуру PackedFilePath
storedInfo = fmap packParentDirPath storedName
filteredInfo = fmap packParentDirPath filteredName
diskInfo = fmap packParentDirPath diskName
-- Для каждого каталога - булевский флаг: начинается ли его имя с базового каталога ("-ap")
dirIncludedArray = fmap (arc_basedir `isParentDirOf`) storedName
dirIncluded = if arc_basedir=="" then const True else (dirIncludedArray!:)
-- Список структур Maybe FileInfo (Nothing для тех файлов, которые не принадлежат
-- базовому каталогу ("-ap") или не проходят через предикат фильтрации файлов)
let make_fi dir name size time dir_flag =
if dirIncluded dir && filter_f fileinfo then Just fileinfo else Nothing
where fileinfo = FileInfo { fiFilteredName = if arc_basedir>"" then fiFilteredName else fiStoredName
, fiDiskName = if disk_basedir>"" || ep/=3 then fiDiskName else fiFilteredName
, fiStoredName = fiStoredName
, fiSize = size
, fiTime = time
, fiAttr = 0
, fiIsDir = dir_flag
, fiGroup = fiUndefinedGroup
}
fiStoredName = packFilePathPacked2 stored (fpPackedFullname stored) name
fiFilteredName = packFilePathPacked2 filtered (fpPackedFullname filtered) name
fiDiskName = packFilePathPacked2 disk (fpPackedFullname disk) name
stored = storedInfo !:dir
filtered = filteredInfo!:dir
disk = diskInfo !:dir
-- Составим структуры FileInfo из отдельных полей, прочитанных из архива
let fileinfos = zipWith5 make_fi dir_numbers names sizes times dir_flags
-- Реконструируем дескрипторы блоков данных.
-- Сначала разобьём список длин файлов на подсписки, относящиеся к отдельным блокам.
-- Это позволит нам вычислить суммарный объём файлов в каждом из блоков
let filesizes = splitByLens num_of_files sizes
let blocks = map (tupleToDataBlock archive arcpos) $
zip5 blCompressors
blOffsets
(map sum filesizes)
blCompSizes
num_of_files
-- Размножим ссылки на дескрипторы блоков данных, чтобы хватило на все файлы :)
let arcblocks = concat [ replicate files_in_block blockDescriptor
| files_in_block <- num_of_files -- кол-во файлов в очередном блоке данных
| blockDescriptor <- blocks -- дескриптор очередного блока
]
-- Позиция файла в блоке равна суммарной длине предыдущих файлов в этом блоке.
-- filesizes - список списков длин файлов, относящихся к каждому блоку.
-- Для того, чтобы получить из него позицию файла внутри блока, мы просто считаем
-- "сканирующую сумму". Добавляем [0] в начало каждого списка позиций,
-- чтобы получить позиции ПЕРЕД файлами, а не после них :)
-- Одним словом, если num_of_files = [1..4]
-- и sizes = [1..10]
-- то filesizes = [[1],[2,3],[4,5,6],[7,8, 9,10]]
-- и positions = [ 0, 0,2, 0,4,9, 0,7,15,24]
let positions = concatMap scanningSum filesizes
scanningSum [] = []
scanningSum xs = 0 : scanl1 (+) (init xs)
-- Теперь у нас готовы все компоненты для создания списка файлов, содержащихся в этом каталоге
let files = [ CompressedFile fileinfo arcblock pos crc
| (Just fileinfo, arcblock, pos, crc) <- zip4 fileinfos arcblocks positions crcs
]
return $! evalList files -- Переведём созданный список файлов в вычисленное состояние
when (total_files >= 10000) performGC -- Соберём мусор, если блок содержит достаточно много файлов
debugLog " Directory built"
return (blocks, files)
-- let f CompressedFile{cfFileInfo=FileInfo{fiFilteredName=PackedFilePath{fpParent=PackedFilePath{fpParent=RootDir}}}} = True
-- f _ = False
----------------------------------------------------------------------------------------------------
---- Упаковываемый файл (или с диска, или из уже существующего архива) -----------------------------
----------------------------------------------------------------------------------------------------
-- |File to compress: either file on disk or compressed file in existing archive
data FileToCompress
= DiskFile
{ cfFileInfo :: !FileInfo
}
| CompressedFile
{ cfFileInfo :: !FileInfo
, cfArcBlock :: ArchiveBlock -- Archive datablock which contains file data
, cfPos :: FileSize -- Starting byte of file data in datablock
, cfCRC :: {-# UNPACK #-} !CRC -- File's CRC
}
-- |Assign type synonym because variant label can't be used in another types declarations
type CompressedFile = FileToCompress
-- |Проверка того, что упаковываемый файл - из уже существующего архива, а не с диска
isCompressedFile CompressedFile{} = True
isCompressedFile DiskFile{} = False
-- |Алгоритм сжатия, использованный для данного (сжатого) файла
cfCompressor = blCompressor.cfArcBlock
-- |Это сжатый файл, использующий фейковый метод компрессии?
isCompressedFake file = isCompressedFile file && isFakeCompressor (cfCompressor file)
-- |Это запаролированный файл?
cfIsEncrypted = blIsEncrypted . cfArcBlock
-- |Определить тип файла по группе, если она не проставлена - вычислить по имени
cfType command file | group/=fiUndefinedGroup = opt_group2type command group
| otherwise = opt_find_type command fi
where fi = cfFileInfo file
group = fiGroup fi
----------------------------------------------------------------------------------------------------
---- Файл и его CRC - используется для передачи результатов упаковки -------------------------------
----------------------------------------------------------------------------------------------------
-- |File and it's CRC
data FileWithCRC = FileWithCRC { fwCRC :: {-# UNPACK #-} !CRC
, fwType :: {-# UNPACK #-} !FileType
, fwFileInfo :: !FileInfo
}
data FileType = FILE_ON_DISK | FILE_IN_ARCHIVE deriving (Eq)
-- |Проверка того, что упакованный файл - из исходного архива, а не с диска
isFILE_ON_DISK fw = fwType fw == FILE_ON_DISK
-- |Convert FileToCompress to FileWithCRC
fileWithCRC (DiskFile fi) = FileWithCRC 0 FILE_ON_DISK fi
fileWithCRC (CompressedFile fi _ _ crc) = FileWithCRC crc FILE_IN_ARCHIVE fi