2 # -*- coding: utf-8 -*-
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the Revised BSD License.
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 Revised BSD License for more details.
13 Copyright 2018-2024 Cool Dude 2k - http://idb.berlios.de/
14 Copyright 2018-2024 Game Maker 2k - http://intdb.sourceforge.net/
15 Copyright 2018-2024 Kazuki Przyborowski - https://github.com/KazukiPrzyborowski
17 $FileInfo: catfile.py - Last Update: 5/3/2024 Ver. 0.10.0 RC 1 - Author: cooldude2k $
20 from __future__
import absolute_import
, division
, print_function
, unicode_literals
;
21 import sys
, argparse
, pycatfile
, binascii
;
23 rarfile_support
= pycatfile
.rarfile_support
;
24 py7zr_support
= pycatfile
.py7zr_support
;
26 if(sys
.version
[0]=="2"):
28 from io
import StringIO
, BytesIO
;
31 from cStringIO
import StringIO
;
32 from cStringIO
import StringIO
as BytesIO
;
34 from StringIO
import StringIO
;
35 from StringIO
import StringIO
as BytesIO
;
36 elif(sys
.version
[0]>="3"):
37 from io
import StringIO
, BytesIO
;
42 from cStringIO
import StringIO
as BytesIO
;
48 from StringIO
import StringIO
as BytesIO
;
54 from io
import BytesIO
;
59 __project__
= pycatfile
.__project
__;
60 __program_name__
= pycatfile
.__program
_name
__;
61 __file_format_name__
= pycatfile
.__file
_format
_name
__;
62 __file_format_lower__
= pycatfile
.__file
_format
_lower
__;
63 __file_format_magic__
= pycatfile
.__file
_format
_magic
__;
64 __file_format_len__
= pycatfile
.__file
_format
_len
__;
65 __file_format_hex__
= pycatfile
.__file
_format
_hex
__;
66 __file_format_delimiter__
= pycatfile
.__file
_format
_delimiter
__;
67 __file_format_list__
= pycatfile
.__file
_format
_list
__;
68 __use_new_style__
= pycatfile
.__use
_new
_style
__;
69 __use_advanced_list__
= pycatfile
.__use
_advanced
_list
__;
70 __use_alt_inode__
= pycatfile
.__use
_alt
_inode
__;
71 __project_url__
= pycatfile
.__project
_url
__;
72 __version_info__
= pycatfile
.__version
_info
__;
73 __version_date_info__
= pycatfile
.__version
_date
_info
__;
74 __version_date__
= pycatfile
.__version
_date
__;
75 __version_date_plusrc__
= pycatfile
.__version
_date
_plusrc
__;
76 __version__
= pycatfile
.__version
__;
78 # Initialize the argument parser
79 argparser
= argparse
.ArgumentParser(description
="Manipulate concatenated files.", conflict_handler
="resolve", add_help
=True);
82 argparser
.add_argument("-V", "--version", action
="version", version
=__program_name__
+ " " + __version__
);
83 # Input and output specifications
84 argparser
.add_argument("-i", "--input", help="Specify the file(s) to concatenate or the concatenated file to extract.", required
=True);
85 argparser
.add_argument("-o", "--output", default
=None, help="Specify the name for the extracted or output concatenated files.");
87 argparser
.add_argument("-c", "--create", action
="store_true", help="Perform only the concatenation operation.");
88 argparser
.add_argument("-e", "--extract", action
="store_true", help="Perform only the extraction operation.");
89 argparser
.add_argument("-t", "--convert", action
="store_true", help="Convert a tar/zip/rar/7zip file to a concatenated file.");
90 argparser
.add_argument("-r", "--repack", action
="store_true", help="Re-concatenate files, fixing checksum errors if any.");
91 # File manipulation options
92 argparser
.add_argument("-F", "--format", default
=__file_format_list__
[0], help="Specify the format to use.");
93 argparser
.add_argument("-D", "--delimiter", default
=__file_format_list__
[5], help="Specify the delimiter to use.");
94 argparser
.add_argument("-m", "--formatver", default
=__file_format_list__
[6], help="Specify the format version.");
95 argparser
.add_argument("-l", "--list", action
="store_true", help="List files included in the concatenated file.");
97 argparser
.add_argument("-P", "--compression", default
="auto", help="Specify the compression method to use for concatenation.");
98 argparser
.add_argument("-L", "--level", default
=None, help="Specify the compression level for concatenation.");
99 # Checksum and validation
100 argparser
.add_argument("-v", "--validate", action
="store_true", help="Validate concatenated file checksums.");
101 argparser
.add_argument("-C", "--checksum", default
="crc32", help="Specify the type of checksum to use. The default is crc32.");
102 argparser
.add_argument("-s", "--skipchecksum", action
="store_true", help="Skip the checksum check of files.");
103 # Permissions and metadata
104 argparser
.add_argument("-p", "--preserve", action
="store_false", help="Do not preserve permissions and timestamps of files.");
106 argparser
.add_argument("-d", "--verbose", action
="store_true", help="Enable verbose mode to display various debugging information.");
107 argparser
.add_argument("-T", "--text", action
="store_true", help="Read file locations from a text file.");
108 # Parse the arguments
109 getargs
= argparser
.parse_args();
111 fname
= getargs
.format
;
112 fnamelower
= fname
.lower();
114 fnamelen
= len(fname
);
115 fnamehex
= binascii
.hexlify(fname
.encode("UTF-8")).decode("UTF-8");
116 fnamesty
= __use_new_style__
;
117 fnamelst
= __use_advanced_list__
;
118 fnameino
= __use_alt_inode__
;
119 fnamelist
= [fname
, fnamemagic
, fnamelower
, fnamelen
, fnamehex
, getargs
.delimiter
, getargs
.formatver
, fnamesty
, fnamelst
, fnameino
];
121 # Determine the primary action based on user input
122 actions
= ['create', 'extract', 'list', 'repack', 'validate'];
123 active_action
= next((action
for action
in actions
if getattr(getargs
, action
)), None);
125 # Execute the appropriate functions based on determined actions and arguments
127 if active_action
=='create':
129 checkcompressfile
= pycatfile
.CheckCompressionSubType(getargs
.input, fnamelist
, True);
130 if(checkcompressfile
=="catfile"):
131 tmpout
= pycatfile
.RePackArchiveFile(getargs
.input, getargs
.output
, getargs
.compression
, getargs
.level
, False, 0, 0, getargs
.checksum
, getargs
.skipchecksum
, [], fnamelist
, getargs
.verbose
, False);
133 tmpout
= pycatfile
.PackArchiveFileFromInFile(getargs
.input, getargs
.output
, getargs
.compression
, getargs
.level
, getargs
.checksum
, [], fnamelist
, getargs
.verbose
, False);
137 pycatfile
.PackArchiveFile(getargs
.input, getargs
.output
, getargs
.text
, getargs
.compression
, getargs
.level
, False, getargs
.checksum
, [], fnamelist
, getargs
.verbose
, False);
138 elif active_action
=='repack':
140 checkcompressfile
= pycatfile
.CheckCompressionSubType(getargs
.input, fnamelist
, True);
141 if(checkcompressfile
=="catfile"):
142 pycatfile
.RePackArchiveFile(getargs
.input, getargs
.output
, getargs
.compression
, getargs
.level
, False, 0, 0, getargs
.checksum
, getargs
.skipchecksum
, [], fnamelist
, getargs
.verbose
, False);
144 pycatfile
.PackArchiveFileFromInFile(getargs
.input, getargs
.output
, getargs
.compression
, getargs
.level
, getargs
.checksum
, [], fnamelist
, getargs
.verbose
, False);
148 pycatfile
.RePackArchiveFile(getargs
.input, getargs
.output
, getargs
.compression
, getargs
.level
, False, 0, 0, getargs
.checksum
, getargs
.skipchecksum
, [], fnamelist
, getargs
.verbose
, False);
149 elif active_action
=='extract':
151 checkcompressfile
= pycatfile
.CheckCompressionSubType(getargs
.input, fnamelist
, True);
153 if(checkcompressfile
=="catfile"):
154 tmpout
= pycatfile
.RePackArchiveFile(getargs
.input, tempout
, getargs
.compression
, getargs
.level
, False, 0, 0, getargs
.checksum
, getargs
.skipchecksum
, [], fnamelist
, getargs
.verbose
, False);
156 tmpout
= pycatfile
.PackArchiveFileFromInFile(getargs
.input, tempout
, getargs
.compression
, getargs
.level
, getargs
.checksum
, [], fnamelist
, getargs
.verbose
, False);
159 getargs
.input = tempout
;
160 pycatfile
.UnPackArchiveFile(getargs
.input, getargs
.output
, False, 0, 0, getargs
.skipchecksum
, fnamelist
, getargs
.verbose
, getargs
.preserve
, getargs
.preserve
, False);
161 elif active_action
=='list':
163 checkcompressfile
= pycatfile
.CheckCompressionSubType(getargs
.input, fnamelist
, True);
164 if(checkcompressfile
=="catfile"):
165 tmpout
= pycatfile
.ArchiveFileListFiles(getargs
.input, 0, 0, getargs
.skipchecksum
, fnamelist
, getargs
.verbose
, False);
167 tmpout
= pycatfile
.InFileListFiles(getargs
.input, getargs
.verbose
, fnamelist
, False);
171 pycatfile
.ArchiveFileListFiles(getargs
.input, 0, 0, getargs
.skipchecksum
, fnamelist
, getargs
.verbose
, False);
172 elif active_action
=='validate':
174 checkcompressfile
= pycatfile
.CheckCompressionSubType(getargs
.input, fnamelist
, True);
176 if(checkcompressfile
=="catfile"):
177 tmpout
= pycatfile
.RePackArchiveFile(getargs
.input, tempout
, getargs
.compression
, getargs
.level
, False, 0, 0, getargs
.checksum
, getargs
.skipchecksum
, [], fnamelist
, getargs
.verbose
, False);
179 tmpout
= pycatfile
.PackArchiveFileFromInFile(getargs
.input, tempout
, getargs
.compression
, getargs
.level
, getargs
.checksum
, [], fnamelist
, getargs
.verbose
, False);
180 getargs
.input = tempout
;
183 fvalid
= pycatfile
.ArchiveFileValidate(getargs
.input, fnamelist
, getargs
.verbose
, False);
184 if(not getargs
.verbose
):
186 logging
.basicConfig(format
="%(message)s", stream
=sys
.stdout
, level
=logging
.DEBUG
);
188 pycatfile
.VerbosePrintOut("File is valid: \n" + str(getargs
.input));
190 pycatfile
.VerbosePrintOut("File is invalid: \n" + str(getargs
.input));