/usr/bin
#!/usr/bin/perl # zipdetails # # Display info on the contents of a Zip file # BEGIN { pop @INC if $INC[-1] eq '.' } use strict; use warnings ; use IO::File; use Encode; use Getopt::Long; # Compression types use constant ZIP_CM_STORE => 0 ; use constant ZIP_CM_IMPLODE => 6 ; use constant ZIP_CM_DEFLATE => 8 ; use constant ZIP_CM_BZIP2 => 12 ; use constant ZIP_CM_LZMA => 14 ; use constant ZIP_CM_PPMD => 98 ; # General Purpose Flag use constant ZIP_GP_FLAG_ENCRYPTED_MASK => (1 << 0) ; use constant ZIP_GP_FLAG_STREAMING_MASK => (1 << 3) ; use constant ZIP_GP_FLAG_PATCHED_MASK => (1 << 5) ; use constant ZIP_GP_FLAG_STRONG_ENCRYPTED_MASK => (1 << 6) ; use constant ZIP_GP_FLAG_LZMA_EOS_PRESENT => (1 << 1) ; use constant ZIP_GP_FLAG_LANGUAGE_ENCODING => (1 << 11) ; # Internal File Attributes use constant ZIP_IFA_TEXT_MASK => 1; # Signatures for each of the headers use constant ZIP_LOCAL_HDR_SIG => 0x04034b50; use constant ZIP_DATA_HDR_SIG => 0x08074b50; use constant ZIP_CENTRAL_HDR_SIG => 0x02014b50; use constant ZIP_END_CENTRAL_HDR_SIG => 0x06054b50; use constant ZIP64_END_CENTRAL_REC_HDR_SIG => 0x06064b50; use constant ZIP64_END_CENTRAL_LOC_HDR_SIG => 0x07064b50; use constant ZIP_ARCHIVE_EXTRA_DATA_SIG => 0x08064b50; use constant ZIP_DIGITAL_SIGNATURE_SIG => 0x05054b50; use constant ZIP_ARCHIVE_EXTRA_DATA_RECORD_SIG => 0x08064b50; # Extra sizes use constant ZIP_EXTRA_HEADER_SIZE => 2 ; use constant ZIP_EXTRA_MAX_SIZE => 0xFFFF ; use constant ZIP_EXTRA_SUBFIELD_ID_SIZE => 2 ; use constant ZIP_EXTRA_SUBFIELD_LEN_SIZE => 2 ; use constant ZIP_EXTRA_SUBFIELD_HEADER_SIZE => ZIP_EXTRA_SUBFIELD_ID_SIZE + ZIP_EXTRA_SUBFIELD_LEN_SIZE; use constant ZIP_EXTRA_SUBFIELD_MAX_SIZE => ZIP_EXTRA_MAX_SIZE - ZIP_EXTRA_SUBFIELD_HEADER_SIZE; my %ZIP_CompressionMethods = ( 0 => 'Stored', 1 => 'Shrunk', 2 => 'Reduced compression factor 1', 3 => 'Reduced compression factor 2', 4 => 'Reduced compression factor 3', 5 => 'Reduced compression factor 4', 6 => 'Imploded', 7 => 'Reserved for Tokenizing compression algorithm', 8 => 'Deflated', 9 => 'Enhanced Deflating using Deflate64(tm)', 10 => 'PKWARE Data Compression Library Imploding', 11 => 'Reserved by PKWARE', 12 => 'BZIP2 ', 13 => 'Reserved by PKWARE', 14 => 'LZMA', 15 => 'Reserved by PKWARE', 16 => 'IBM z/OS CMPSC Compression', 17 => 'Reserved by PKWARE', 18 => 'File is compressed using IBM TERSE (new)', 19 => 'IBM LZ77 z Architecture (PFS)', 93 => 'Zstandard', 94 => 'MP3', 95 => 'XZ', 96 => 'WinZip JPEG Compression', 97 => 'WavPack compressed data', 98 => 'PPMd version I, Rev 1', 99 => 'AES Encryption', ); my %OS_Lookup = ( 0 => "MS-DOS", 1 => "Amiga", 2 => "OpenVMS", 3 => "Unix", 4 => "VM/CMS", 5 => "Atari ST", 6 => "HPFS (OS/2, NT 3.x)", 7 => "Macintosh", 8 => "Z-System", 9 => "CP/M", 10 => "Windoxs NTFS or TOPS-20", 11 => "MVS or NTFS", 12 => "VSE or SMS/QDOS", 13 => "Acorn RISC OS", 14 => "VFAT", 15 => "alternate MVS", 16 => "BeOS", 17 => "Tandem", 18 => "OS/400", 19 => "OS/X (Darwin)", 30 => "AtheOS/Syllable", ); my %Lookup = ( ZIP_LOCAL_HDR_SIG, \&LocalHeader, ZIP_DATA_HDR_SIG, \&DataHeader, ZIP_CENTRAL_HDR_SIG, \&CentralHeader, ZIP_END_CENTRAL_HDR_SIG, \&EndCentralHeader, ZIP64_END_CENTRAL_REC_HDR_SIG, \&Zip64EndCentralHeader, ZIP64_END_CENTRAL_LOC_HDR_SIG, \&Zip64EndCentralLocator, # TODO - Archive Encryption Headers & digital signature #ZIP_ARCHIVE_EXTRA_DATA_RECORD_SIG #ZIP_DIGITAL_SIGNATURE_SIG #ZIP_ARCHIVE_EXTRA_DATA_SIG ); my %Extras = ( 0x0001, ['ZIP64', \&decode_Zip64], 0x0007, ['AV Info', undef], 0x0008, ['Extended Language Encoding', undef], 0x0009, ['OS/2 extended attributes', undef], 0x000a, ['NTFS FileTimes', \&decode_NTFS_Filetimes], 0x000c, ['OpenVMS', undef], 0x000d, ['Unix', undef], 0x000e, ['Stream & Fork Descriptors', undef], 0x000f, ['Patch Descriptor', undef], 0x0014, ['PKCS#7 Store for X.509 Certificates', undef], 0x0015, ['X.509 Certificate ID and Signature for individual file', undef], 0x0016, ['X.509 Certificate ID for Central Directory', undef], 0x0017, ['Strong Encryption Header', undef], 0x0018, ['Record Management Controls', undef], 0x0019, ['PKCS#7 Encryption Recipient Certificate List', undef], 0x0020, ['Reserved for Timestamp record', undef], 0x0021, ['Policy Decryption Key Record', undef], 0x0022, ['Smartcrypt Key Provider Record', undef], 0x0023, ['Smartcrypt Policy Key Data Record', undef], # The Header ID mappings defined by Info-ZIP and third parties are: 0x0065, ['IBM S/390 attributes - uncompressed', \&decodeMVS], 0x0066, ['IBM S/390 attributes - compressed', undef], 0x07c8, ['Info-ZIP Macintosh (old, J. Lee)', undef], 0x2605, ['ZipIt Macintosh (first version)', undef], 0x2705, ['ZipIt Macintosh v 1.3.5 and newer (w/o full filename)', undef], 0x2805, ['ZipIt Macintosh v 1.3.5 and newer ', undef], 0x334d, ["Info-ZIP Macintosh (new, D. Haase's 'Mac3' field)", undef], 0x4154, ['Tandem NSK', undef], 0x4341, ['Acorn/SparkFS (David Pilling)', undef], 0x4453, ['Windows NT security descriptor', \&decode_NT_security], 0x4690, ['POSZIP 4690', undef], 0x4704, ['VM/CMS', undef], 0x470f, ['MVS', undef], 0x4854, ['Theos, old inofficial port', undef], 0x4b46, ['FWKCS MD5 (see below)', undef], 0x4c41, ['OS/2 access control list (text ACL)', undef], 0x4d49, ['Info-ZIP OpenVMS (obsolete)', undef], 0x4d63, ['Macintosh SmartZIP, by Macro Bambini', undef], 0x4f4c, ['Xceed original location extra field', undef], 0x5356, ['AOS/VS (binary ACL)', undef], 0x5455, ['Extended Timestamp', \&decode_UT], 0x554e, ['Xceed unicode extra field', \&decode_Xceed_unicode], 0x5855, ['Info-ZIP Unix (original; also OS/2, NT, etc.)', \&decode_UX], 0x5a4c, ['ZipArchive Unicode Filename', undef], 0x5a4d, ['ZipArchive Offsets Array', undef], 0x6375, ['Info-ZIP Unicode Comment', \&decode_up ], 0x6542, ['BeOS (BeBox, PowerMac, etc.)', undef], 0x6854, ['Theos', undef], 0x7075, ['Info-ZIP Unicode Path', \&decode_up ], 0x756e, ['ASi Unix', undef], 0x7441, ['AtheOS (AtheOS/Syllable attributes)', undef], 0x7855, ['Unix Extra type 2', \&decode_Ux], 0x7875, ['Unix Extra Type 3', \&decode_ux], 0x9901, ['AES Encryption', \&decode_AES], 0xa11e, ['Data Stream Alignment', undef], 0xA220, ['Open Packaging Growth Hint', \&decode_GrowthHint ], 0xCAFE, ['Java Executable', \&decode_Java_exe], 0xfb4a, ['SMS/QDOS', undef], ); my $VERSION = "2.02" ; my $FH; my $ZIP64 = 0 ; my $NIBBLES = 8; my $LocalHeaderCount = 0; my $CentralHeaderCount = 0; my $START; my $OFFSET = U64->new( 0 ); my $TRAILING = 0 ; my $PAYLOADLIMIT = 256; # U64->new( 256 ); my $ZERO = U64->new( 0 ); my $SEEN = Seen->new(); sub prOff { my $offset = shift; my $s = offset($OFFSET); $OFFSET->add($offset); return $s; } sub offset { my $v = shift ; if (ref $v eq 'U64') { my $hi = $v->getHigh(); my $lo = $v->getLow(); if ($hi) { my $hiNib = $NIBBLES - 8 ; sprintf("%0${hiNib}X", $hi) . sprintf("%08X", $lo); } else { sprintf("%0${NIBBLES}X", $lo); } } else { sprintf("%0${NIBBLES}X", $v); } } my ($OFF, $LENGTH, $CONTENT, $TEXT, $VALUE) ; my $FMT1 ; my $FMT2 ; sub setupFormat { my $wantVerbose = shift ; my $nibbles = shift; my $width = '@' . ('>' x ($nibbles -1)); my $space = " " x length($width); my $fmt ; if ($wantVerbose) { $FMT1 = " format STDOUT = $width $width ^<<<<<<<<<<<^<<<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< \$OFF, \$LENGTH, \$CONTENT, \$TEXT, \$VALUE $space $space ^<<<<<<<<<<<^<<<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<~~ \$CONTENT, \$TEXT, \$VALUE . "; $FMT2 = " format STDOUT = $width $width ^<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< \$OFF, \$LENGTH, \$CONTENT, \$TEXT, \$VALUE $space $space ^<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<~~ \$CONTENT, \$TEXT, \$VALUE . " ; } else { $FMT1 = " format STDOUT = $width ^<<<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< \$OFF, \$TEXT, \$VALUE $space ^<<<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<~~ \$TEXT, \$VALUE . "; $FMT2 = " format STDOUT = $width ^<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< \$OFF, \$TEXT, \$VALUE $space ^<<<<<<<<<<<<<<<<<< ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<~~ \$TEXT, \$VALUE . " ; } eval "$FMT1"; $| = 1; } sub mySpr { my $format = shift ; return "" if ! defined $format; return $format unless @_ ; return sprintf $format, @_ ; } sub out0 { my $size = shift; my $text = shift; my $format = shift; $OFF = prOff($size); $LENGTH = offset($size) ; $CONTENT = '...'; $TEXT = $text; $VALUE = mySpr $format, @_; write; skip($FH, $size); } sub xDump { my $input = shift; $input =~ tr/\0-\37\177-\377/./; return $input; } sub hexDump { my $input = shift; my $out = unpack('H*', $input) ; $out =~ s#(..)# $1#g ; $out =~ s/^ //; $out = uc $out; return $out; } sub out { my $data = shift; my $text = shift; my $format = shift; my $size = length($data) ; $OFF = prOff($size); $LENGTH = offset($size) ; $CONTENT = hexDump($data); $TEXT = $text; $VALUE = mySpr $format, @_; no warnings; write; } sub out1 { my $text = shift; my $format = shift; $OFF = ''; $LENGTH = '' ; $CONTENT = ''; $TEXT = $text; $VALUE = mySpr $format, @_; write; } sub out2 { my $data = shift ; my $text = shift ; my $format = shift; my $size = length($data) ; $OFF = prOff($size); $LENGTH = offset($size); $CONTENT = hexDump($data); $TEXT = $text; $VALUE = mySpr $format, @_; no warnings; eval "$FMT2"; write ; eval "$FMT1"; } sub Value { my $letter = shift; my @value = @_; if ($letter eq 'C') { return Value_C(@value) } elsif ($letter eq 'v') { return Value_v(@value) } elsif ($letter eq 'V') { return Value_V(@value) } elsif ($letter eq 'VV') { return Value_VV(@value) } } sub outer { my $name = shift ; my $unpack = shift ; my $size = shift ; my $cb1 = shift ; my $cb2 = shift ; myRead(my $buff, $size); my (@value) = unpack $unpack, $buff; my $hex = Value($unpack, @value); if (defined $cb1) { my $v ; if (ref $cb1 eq 'CODE') { $v = $cb1->(@value) ; } else { $v = $cb1 ; } $v = "'" . $v unless $v =~ /^'/; $v .= "'" unless $v =~ /'$/; $hex .= " $v" ; } out $buff, $name, $hex ; $cb2->(@value) if defined $cb2 ; return $value[0]; } sub out_C { my $name = shift ; my $cb1 = shift ; my $cb2 = shift ; outer($name, 'C', 1, $cb1, $cb2); } sub out_v { my $name = shift ; my $cb1 = shift ; my $cb2 = shift ; outer($name, 'v', 2, $cb1, $cb2); } sub out_V { my $name = shift ; my $cb1 = shift ; my $cb2 = shift ; outer($name, 'V', 4, $cb1, $cb2); } sub out_VV { my $name = shift ; my $cb1 = shift ; my $cb2 = shift ; outer($name, 'VV', 8, $cb1, $cb2); } # sub outSomeData # { # my $size = shift; # my $message = shift; # my $size64 = U64::mkU64($size); # if ($size64->gt($ZERO)) { # my $size32 = $size64->getLow(); # if ($size64->gt($PAYLOADLIMIT) ) { # out0 $size32, $message; # } else { # myRead(my $buffer, $size32 ); # out $buffer, $message, xDump $buffer ; # } # } # } sub outSomeData { my $size = shift; my $message = shift; if ($size > 0) { if ($size > $PAYLOADLIMIT) { my $before = $FH->tell(); out0 $size, $message; # printf "outSomeData %X %X $size %X\n", $before, $FH->tell(), $size; } else { myRead(my $buffer, $size ); out $buffer, $message, xDump $buffer ; } } } sub unpackValue_C { Value_v(unpack "C", $_[0]); } sub Value_C { sprintf "%02X", $_[0]; } sub unpackValue_v { Value_v(unpack "v", $_[0]); } sub Value_v { sprintf "%04X", $_[0]; } sub unpackValue_V { Value_V(unpack "V", $_[0]); } sub Value_V { my $v = defined $_[0] ? $_[0] : 0; sprintf "%08X", $v; } sub unpackValue_VV { my ($lo, $hi) = unpack ("V V", $_[0]); Value_VV($lo, $hi); } sub Value_U64 { my $u64 = shift ; Value_VV($u64->getLow(), $u64->getHigh()); } sub Value_VV { my $lo = defined $_[0] ? $_[0] : 0; my $hi = defined $_[1] ? $_[1] : 0; if ($hi == 0) { sprintf "%016X", $lo; } else { sprintf("%08X", $hi) . sprintf "%08X", $lo; } } sub Value_VV64 { my $buffer = shift; # This needs perl 5.10 # return unpack "Q<", $buffer; my ($lo, $hi) = unpack ("V V" , $buffer); no warnings 'uninitialized'; return $hi * (0xFFFFFFFF+1) + $lo; } sub read_U64 { my $b ; myRead($b, 8); my ($lo, $hi) = unpack ("V V" , $b); no warnings 'uninitialized'; return ($b, U64->new( $hi, $lo) ); } sub read_VV { my $b ; myRead($b, 8); my ($lo, $hi) = unpack ("V V" , $b); no warnings 'uninitialized'; return ($b, $hi * (0xFFFFFFFF+1) + $lo); } sub read_V { my $b ; myRead($b, 4); return ($b, unpack ("V", $b)); } sub read_v { my $b ; myRead($b, 2); return ($b, unpack "v", $b); } sub read_C { my $b ; myRead($b, 1); return ($b, unpack "C", $b); } sub seekTo { my $offset = shift ; my $loc = shift ; $loc = SEEK_SET if ! defined $loc ; $FH->seek($offset, $loc); $OFFSET = new U64($offset); } sub scanForSignature { my %sigs = map { $_ => 1 } map { substr $_, 2, 2 } # don't want the initial "PK" map { pack "V", $_ } ( ZIP_LOCAL_HDR_SIG , ZIP_DATA_HDR_SIG , ZIP_CENTRAL_HDR_SIG , ZIP_END_CENTRAL_HDR_SIG , ZIP64_END_CENTRAL_REC_HDR_SIG , ZIP64_END_CENTRAL_LOC_HDR_SIG , # ZIP_ARCHIVE_EXTRA_DATA_SIG , # ZIP_DIGITAL_SIGNATURE_SIG , # ZIP_ARCHIVE_EXTRA_DATA_RECORD_SIG , ); my $start = $FH->tell(); my $last = ''; my $buffer ; while ($FH->read($buffer, 1024 * 1000)) { my $combine = substr($last, -4) . $buffer ; $last = $buffer; my $ix = index($combine, "PK") ; next if $ix == -1; my $rest = substr($combine, $ix+2, 2); next unless length $rest == 2 && $sigs{$rest} ; my $v = unpack("v", $rest) ; # possible match my $here = $FH->tell(); seekTo($here - length($combine) + $ix); return 1; } return 0; } my $is64In32 = 0; my $opt_verbose = 0; my $opt_scan = 0; $Getopt::Long::bundling = 1 ; GetOptions("h" => \&Usage, "v" => \$opt_verbose, "scan" => \$opt_scan, "version" => sub { print "$VERSION\n"; exit }, ) or Usage("Invalid command line option\n"); Usage("No zipfile") unless @ARGV == 1; my $filename = shift @ARGV; die "$filename does not exist\n" unless -e $filename ; die "$filename not a standard file\n" unless -f $filename ; $FH = IO::File->new( "<$filename" ) or die "Cannot open $filename: $!\n"; my $FILELEN = -s $filename ; $TRAILING = -s $filename ; $NIBBLES = U64::nibbles(-s $filename) ; #$NIBBLES = int ($NIBBLES / 4) + ( ($NIBBLES % 4) ? 1 : 0 ); #$NIBBLES = 4 * $NIBBLES; # Minimum of 4 nibbles $NIBBLES = 4 if $NIBBLES < 4 ; die "$filename too short to be a zip file\n" if $FILELEN < 22 ; setupFormat($opt_verbose, $NIBBLES); if(0) { # Sanity check that this is a Zip file my ($buffer, $signature) = read_V(); warn "$filename doesn't look like a zip file\n" if $signature != ZIP_LOCAL_HDR_SIG ; $FH->seek(0, SEEK_SET) ; } my @Messages = (); if ($opt_scan) { while(scanForSignature()) { my $here = $FH->tell(); my ($buffer, $signature) = read_V(); my $handler = $Lookup{$signature}; $handler->($signature, $buffer); seekTo($here + 4) ; } dislayMessages(); exit; } our ($CdExists, $CdOffset, @CentralDirectory) = scanCentralDirectory($FH); die "No Central Directory records found\n" if ! $CdExists ; $OFFSET->reset(); $FH->seek(0, SEEK_SET) ; outSomeData($START, "PREFIX DATA") if defined $START && $START > 0 ; my $skippedFrom = 0 ; my $skippedContent = 0 ; while (1) { last if $FH->eof(); my $here = $FH->tell(); if ($here >= $TRAILING) { print "\n" ; outSomeData($FILELEN - $TRAILING, "TRAILING DATA"); last; } my ($buffer, $signature) = read_V(); my $handler = $Lookup{$signature}; if (!defined $handler) { if (@CentralDirectory) { # Should be at offset that central directory says my $locOffset = $CentralDirectory[0][0]; my $delta = $locOffset - $here ; if ($here + 4 == $locOffset ) { for (0 .. 3) { $FH->ungetc(ord(substr($buffer, $_, 1))) } outSomeData($delta, "UNEXPECTED PADDING"); next; } } if ($here < $CdOffset) { # next # if scanForSignature() ; $skippedFrom = $FH->tell() ; $skippedContent = $CdOffset - $skippedFrom ; print "\nWARNING!\nZip local header not found.\n"; printf "Skipping 0x%x bytes to Central Directory...\n", $skippedContent; push @Messages, sprintf("Expected Zip header not found at offset 0x%X, ", $skippedFrom) . sprintf("skipped 0x%X bytes\n", $skippedContent); seekTo($CdOffset); next; } else { printf "\n\nUnexpected END at offset %08X, value %s\n", $here, Value_V($signature); last; } } $ZIP64 = 0 if $signature != ZIP_DATA_HDR_SIG ; $handler->($signature, $buffer); } dislayMessages(); exit ; sub dislayMessages { if (@Messages) { my $count = scalar @Messages ; print "\nWARNINGS\n\n"; print "* $_\n" for @Messages ; } print "Done\n"; } sub compressionMethod { my $id = shift ; Value_v($id) . " '" . ($ZIP_CompressionMethods{$id} || "Unknown Method") . "'" ; } sub LocalHeader { my $signature = shift ; my $data = shift ; my $from_offset = $FH->tell() - 4; print "\n"; ++ $LocalHeaderCount; out $data, "LOCAL HEADER #" . sprintf("%X", $LocalHeaderCount) , Value_V($signature); my $buffer; my ($loc, $CDcompressedLength) ; ($loc, $CDcompressedLength) = @{ shift @CentralDirectory } if ! $opt_scan ; out_C "Extract Zip Spec", \&decodeZipVer; out_C "Extract OS", \&decodeOS; my ($bgp, $gpFlag) = read_v(); my ($bcm, $compressedMethod) = read_v(); out $bgp, "General Purpose Flag", Value_v($gpFlag) ; GeneralPurposeBits($compressedMethod, $gpFlag); out $bcm, "Compression Method", compressionMethod($compressedMethod) ; out_V "Last Mod Time", sub { scalar getTime(_dosToUnixTime($_[0])) }; my $crc = out_V "CRC"; my $compressedLength = out_V "Compressed Length"; my $uncompressedLength = out_V "Uncompressed Length"; my $filenameLength = out_v "Filename Length"; my $extraLength = out_v "Extra Length"; my $filename ; myRead($filename, $filenameLength); outputFilename($filename); my $cl64 = U64->new( $compressedLength ); my %ExtraContext = (); if ($extraLength) { my @z64 = ($uncompressedLength, $compressedLength, 1, 1); $ExtraContext{Zip64} = \@z64 ; $ExtraContext{InCentralDir} = 0; walkExtra($extraLength, \%ExtraContext); } my $size = 0; $size = printAes(\%ExtraContext) if $compressedMethod == 99 ; $size += printLzmaProperties() if $compressedMethod == ZIP_CM_LZMA ; $CDcompressedLength = $compressedLength if $opt_scan ; # $CDcompressedLength->subtract($size) # if $size ; $CDcompressedLength -= $size; # if ($CDcompressedLength->getHigh() || $CDcompressedLength->getLow()) { if ($CDcompressedLength) { outSomeData($CDcompressedLength, "PAYLOAD") ; } if ($compressedMethod == 99) { my $auth ; myRead($auth, 10); out $auth, "AES Auth", hexDump($auth); } # $SEEN->save("LOCAL HEADER #" . sprintf("%X", $LocalHeaderCount), $filename, $from_offset, $from_offset + $CDcompressedLength) } sub outputFilename { my $filename = shift; if (length $filename > 256) { my $f = substr($filename, 0, 256) ; out $f, "Filename", "'". $f . "' ..."; } else { out $filename, "Filename", "'". $filename . "'"; } } sub CentralHeader { my $signature = shift ; my $data = shift ; my $from_offset = $FH->tell() - 4; ++ $CentralHeaderCount; print "\n"; out $data, "CENTRAL HEADER #" . sprintf("%X", $CentralHeaderCount) . "", Value_V($signature); my $buffer; out_C "Created Zip Spec", \&decodeZipVer; out_C "Created OS", \&decodeOS; out_C "Extract Zip Spec", \&decodeZipVer; out_C "Extract OS", \&decodeOS; my ($bgp, $gpFlag) = read_v(); my ($bcm, $compressedMethod) = read_v(); out $bgp, "General Purpose Flag", Value_v($gpFlag) ; GeneralPurposeBits($compressedMethod, $gpFlag); out $bcm, "Compression Method", compressionMethod($compressedMethod) ; out_V "Last Mod Time", sub { scalar getTime(_dosToUnixTime($_[0])) }; my $crc = out_V "CRC"; my $compressedLength = out_V "Compressed Length"; my $uncompressedLength = out_V "Uncompressed Length"; my $filenameLength = out_v "Filename Length"; my $extraLength = out_v "Extra Length"; my $comment_length = out_v "Comment Length"; my $disk_start = out_v "Disk Start"; my $int_file_attrib = out_v "Int File Attributes"; out1 "[Bit 0]", $int_file_attrib & 1 ? "1 Text Data" : "0 'Binary Data'"; my $ext_file_attrib = out_V "Ext File Attributes"; out1 "[Bit 0]", "Read-Only" if $ext_file_attrib & 0x01 ; out1 "[Bit 1]", "Hidden" if $ext_file_attrib & 0x02 ; out1 "[Bit 2]", "System" if $ext_file_attrib & 0x04 ; out1 "[Bit 3]", "Label" if $ext_file_attrib & 0x08 ; out1 "[Bit 4]", "Directory" if $ext_file_attrib & 0x10 ; out1 "[Bit 5]", "Archive" if $ext_file_attrib & 0x20 ; my $lcl_hdr_offset = out_V "Local Header Offset"; my $filename ; myRead($filename, $filenameLength); outputFilename($filename); my %ExtraContext = (); if ($extraLength) { my @z64 = ($uncompressedLength, $compressedLength, $lcl_hdr_offset, $disk_start); $ExtraContext{Zip64} = \@z64 ; $ExtraContext{InCentralDir} = 1; walkExtra($extraLength, \%ExtraContext); } if ($comment_length) { my $comment ; myRead($comment, $comment_length); out $comment, "Comment", "'". $comment . "'"; } $SEEN->save("CENTRAL HEADER ref Local #" . sprintf("%X", $CentralHeaderCount), $filename, $lcl_hdr_offset, $lcl_hdr_offset + $compressedLength) ; # $SEEN->save("CENTRAL HEADER #" . sprintf("%X", $CentralHeaderCount), $filename, $from_offset, $FH-tell()); } sub decodeZipVer { my $ver = shift ; my $sHi = int($ver /10) ; my $sLo = $ver % 10 ; #out1 "Zip Spec", "$sHi.$sLo"; "$sHi.$sLo"; } sub decodeOS { my $ver = shift ; $OS_Lookup{$ver} || "Unknown" ; } sub Zip64EndCentralHeader { my $signature = shift ; my $data = shift ; print "\n"; out $data, "ZIP64 END CENTRAL DIR RECORD", Value_V($signature); my $buff; myRead($buff, 8); out $buff, "Size of record", unpackValue_VV($buff); my $size = Value_VV64($buff); out_C "Created Zip Spec", \&decodeZipVer; out_C "Created OS", \&decodeOS; out_C "Extract Zip Spec", \&decodeZipVer; out_C "Extract OS", \&decodeOS; out_V "Number of this disk"; out_V "Central Dir Disk no"; out_VV "Entries in this disk"; out_VV "Total Entries"; out_VV "Size of Central Dir"; out_VV "Offset to Central dir"; # TODO - if ($size != 44) { push @Messages, "Unsupported Size field in Zip64EndCentralHeader: should be 44, got $size\n" } } sub Zip64EndCentralLocator { my $signature = shift ; my $data = shift ; print "\n"; out $data, "ZIP64 END CENTRAL DIR LOCATOR", Value_V($signature); out_V "Central Dir Disk no"; out_VV "Offset to Central dir"; out_V "Total no of Disks"; } sub EndCentralHeader { my $signature = shift ; my $data = shift ; print "\n"; out $data, "END CENTRAL HEADER", Value_V($signature); out_v "Number of this disk"; out_v "Central Dir Disk no"; out_v "Entries in this disk"; out_v "Total Entries"; out_V "Size of Central Dir"; out_V "Offset to Central Dir"; my $comment_length = out_v "Comment Length"; if ($comment_length) { my $comment ; myRead($comment, $comment_length); out $comment, "Comment", "'$comment'"; } } sub DataHeader { my $signature = shift ; my $data = shift ; print "\n"; out $data, "STREAMING DATA HEADER", Value_V($signature); out_V "CRC"; if ($ZIP64) { out_VV "Compressed Length" ; out_VV "Uncompressed Length" ; } else { out_V "Compressed Length" ; out_V "Uncompressed Length" ; } } sub GeneralPurposeBits { my $method = shift; my $gp = shift; out1 "[Bit 0]", "1 'Encryption'" if $gp & ZIP_GP_FLAG_ENCRYPTED_MASK; my %lookup = ( 0 => "Normal Compression", 1 => "Maximum Compression", 2 => "Fast Compression", 3 => "Super Fast Compression"); if ($method == ZIP_CM_DEFLATE) { my $mid = ($gp >> 1) & 0x03 ; out1 "[Bits 1-2]", "$mid '$lookup{$mid}'"; } if ($method == ZIP_CM_LZMA) { if ($gp & ZIP_GP_FLAG_LZMA_EOS_PRESENT) { out1 "[Bit 1]", "1 'LZMA EOS Marker Present'" ; } else { out1 "[Bit 1]", "0 'LZMA EOS Marker Not Present'" ; } } if ($method == ZIP_CM_IMPLODE) # Imploding { out1 "[Bit 1]", ($gp & (1 << 1) ? "1 '8k" : "0 '4k") . " Sliding Dictionary'" ; out1 "[Bit 2]", ($gp & (2 << 1) ? "1 '3" : "0 '2" ) . " Shannon-Fano Trees'" ; } out1 "[Bit 3]", "1 'Streamed'" if $gp & ZIP_GP_FLAG_STREAMING_MASK; out1 "[Bit 4]", "1 'Enhanced Deflating'" if $gp & 1 << 4; out1 "[Bit 5]", "1 'Compressed Patched'" if $gp & 1 << 5 ; out1 "[Bit 6]", "1 'Strong Encryption'" if $gp & ZIP_GP_FLAG_STRONG_ENCRYPTED_MASK; out1 "[Bit 11]", "1 'Language Encoding'" if $gp & ZIP_GP_FLAG_LANGUAGE_ENCODING; out1 "[Bit 12]", "1 'Pkware Enhanced Compression'" if $gp & 1 <<12 ; out1 "[Bit 13]", "1 'Encrypted Central Dir'" if $gp & 1 <<13 ; return (); } sub seekSet { my $fh = $_[0] ; my $size = $_[1]; use Fcntl qw(SEEK_SET); if (ref $size eq 'U64') { seek($fh, $size->get64bit(), SEEK_SET); } else { seek($fh, $size, SEEK_SET); } } sub skip { my $fh = $_[0] ; my $size = $_[1]; use Fcntl qw(SEEK_CUR); if (ref $size eq 'U64') { seek($fh, $size->get64bit(), SEEK_CUR); } else { seek($fh, $size, SEEK_CUR); } } sub myRead { my $got = \$_[0] ; my $size = $_[1]; my $wantSize = $size; $$got = ''; if ($size == 0) { return ; } if ($size > 0) { my $buff ; my $status = $FH->read($buff, $size); return $status if $status < 0; $$got .= $buff ; } my $len = length $$got; die "Truncated file (got $len, wanted $wantSize): $!\n" if length $$got != $wantSize; } sub walkExtra { my $XLEN = shift; my $context = shift; my $buff ; my $offset = 0 ; my $id; my $subLen; my $payload ; my $count = 0 ; while ($offset < $XLEN) { ++ $count; # Detect if there is not enough data for an extra ID and length. # Android zipalign and zipflinger are prime candidates for these # non-standard extra sub-fields. my $remaining = $XLEN - $offset; if ($remaining < ZIP_EXTRA_SUBFIELD_HEADER_SIZE) { # There is not enough. Consume whatever is there and return so parsing # can continue. myRead($payload, $remaining); my $data = hexDump($payload); out $payload, "Malformed Extra Data", $data; return undef; } myRead($id, ZIP_EXTRA_SUBFIELD_ID_SIZE); $offset += ZIP_EXTRA_SUBFIELD_ID_SIZE; my $lookID = unpack "v", $id ; my ($who, $decoder) = @{ defined $Extras{$lookID} ? $Extras{$lookID} : ['', undef] }; #my ($who, $decoder) = @{ $Extras{unpack "v", $id} || ['', undef] }; $who = "$id: $who" if $id =~ /\w\w/ ; $who = "'$who'"; out $id, "Extra ID #" . Value_v($count), unpackValue_v($id) . " $who" ; myRead($buff, ZIP_EXTRA_SUBFIELD_LEN_SIZE); $offset += ZIP_EXTRA_SUBFIELD_LEN_SIZE; $subLen = unpack("v", $buff); out2 $buff, "Length", Value_v($subLen) ; return undef if $offset + $subLen > $XLEN ; if (! defined $decoder) { if ($subLen) { myRead($payload, $subLen); my $data = hexDump($payload); out2 $payload, "Extra Payload", $data; } } else { $decoder->($subLen, $context) ; } $offset += $subLen ; } return undef ; } sub full32 { return $_[0] == 0xFFFFFFFF ; } sub decode_Zip64 { my $len = shift; my $context = shift; my $z64Data = $context->{Zip64}; $ZIP64 = 1; if (full32 $z64Data->[0] ) { out_VV " Uncompressed Size"; } if (full32 $z64Data->[1] ) { out_VV " Compressed Size"; } if (full32 $z64Data->[2] ) { out_VV " Offset to Local Dir"; } if ($z64Data->[3] == 0xFFFF ) { out_V " Disk Number"; } } sub Ntfs2Unix { my $v = shift; my $u64 = shift; # NTFS offset is 19DB1DED53E8000 my $hex = Value_U64($u64) ; my $NTFS_OFFSET = U64->new( 0x19DB1DE, 0xD53E8000 ); $u64->subtract($NTFS_OFFSET); my $elapse = $u64->get64bit(); my $ns = ($elapse % 10000000) * 100; $elapse = int ($elapse/10000000); return "$hex '" . localtime($elapse) . " " . sprintf("%0dns'", $ns); } sub decode_NTFS_Filetimes { my $len = shift; my $context = shift; out_V " Reserved"; out_v " Tag1"; out_v " Size1" ; my ($m, $s1) = read_U64; out $m, " Mtime", Ntfs2Unix($m, $s1); my ($c, $s2) = read_U64; out $c, " Ctime", Ntfs2Unix($m, $s2); my ($a, $s3) = read_U64; out $m, " Atime", Ntfs2Unix($m, $s3); } sub getTime { my $time = shift ; return "'" . localtime($time) . "'" ; } sub decode_UT { my $len = shift; my $context = shift; my ($data, $flags) = read_C(); my $f = Value_C $flags; $f .= " mod" if $flags & 1; $f .= " access" if $flags & 2; $f .= " change" if $flags & 4; out $data, " Flags", "'$f'"; -- $len; if ($flags & 1) { my ($data, $time) = read_V(); out2 $data, "Mod Time", Value_V($time) . " " . getTime($time) ; $len -= 4 ; } if ($flags & 2 && $len > 0 ) { my ($data, $time) = read_V(); out2 $data, "Access Time", Value_V($time) . " " . getTime($time) ; $len -= 4 ; } if ($flags & 4 && $len > 0) { my ($data, $time) = read_V(); out2 $data, "Change Time", Value_V($time) . " " . getTime($time) ; } } sub decode_AES { my $len = shift; my $context = shift; return if $len == 0 ; my %lookup = ( 1 => "AE-1", 2 => "AE-2"); out_v " Vendor Version", sub { $lookup{$_[0]} || "Unknown" } ; my $id ; myRead($id, 2); out $id, " Vendor ID", unpackValue_v($id) . " '$id'"; my %strengths = (1 => "128-bit encryption key", 2 => "192-bit encryption key", 3 => "256-bit encryption key", ); my $strength = out_C " Encryption Strength", sub {$strengths{$_[0]} || "Unknown" } ; my ($bmethod, $method) = read_v(); out $bmethod, " Compression Method", compressionMethod($method) ; $context->{AesStrength} = $strength ; } sub decode_GrowthHint { my $len = shift; my $context = shift; my $inCentralHdr = $context->{InCentralDir} ; return if $len == 0 ; out_v " Signature" ; out_v " Initial Value"; my $padding; myRead($padding, $len - 4); my $data = hexDump($padding); out2 $padding, "Padding", $data; } sub decode_UX { my $len = shift; my $context = shift; my $inCentralHdr = $context->{InCentralDir} ; return if $len == 0 ; my ($data, $time) = read_V(); out2 $data, "Access Time", Value_V($time) . " " . getTime($time) ; ($data, $time) = read_V(); out2 $data, "Mod Time", Value_V($time) . " " . getTime($time) ; if (! $inCentralHdr ) { out_v " UID" ; out_v " GID"; } } sub decode_Ux { my $len = shift; my $context = shift; return if $len == 0 ; out_v " UID" ; out_v " GID"; } sub decodeLitteEndian { my $value = shift ; if (length $value == 4) { return Value_V unpack ("V", $value) } else { # TODO - fix this die "unsupported\n"; } my $got = 0 ; my $shift = 0; #hexDump #reverse #my @a =unpack "C*", $value; #@a = reverse @a; #hexDump(@a); for (reverse unpack "C*", $value) { $got = ($got << 8) + $_ ; } return $got ; } sub decode_ux { my $len = shift; my $context = shift; return if $len == 0 ; out_C " Version" ; my $uidSize = out_C " UID Size"; myRead(my $data, $uidSize); out2 $data, "UID", decodeLitteEndian($data); my $gidSize = out_C " GID Size"; myRead($data, $gidSize); out2 $data, "GID", decodeLitteEndian($data); } sub decode_Java_exe { my $len = shift; my $context = shift; } sub decode_up { my $len = shift; my $context = shift; out_C " Version"; out_V " NameCRC32"; myRead(my $data, $len - 5); out $data, " UnicodeName", $data; } sub decode_Xceed_unicode { my $len = shift; my $context = shift; my $data ; # guess the fields used for this one myRead($data, 4); out $data, " ID", $data; out_v " Length"; out_v " Null"; myRead($data, $len - 8); out $data, " UTF16LE Name", decode("UTF16LE", $data); } sub decode_NT_security { my $len = shift; my $context = shift; my $inCentralHdr = $context->{InCentralDir} ; out_V " Uncompressed Size" ; if (! $inCentralHdr) { out_C " Version" ; out_v " Type"; out_V " NameCRC32" ; my $plen = $len - 4 - 1 - 2 - 4; myRead(my $payload, $plen); out $plen, " Extra Payload", hexDump($payload); } } sub decodeMVS { my $len = shift; my $context = shift; # data in Big-Endian myRead(my $data, $len); my $ID = unpack("N", $data); if ($ID == 0xE9F3F9F0) { out($data, " ID", "'Z390'"); substr($data, 0, 4) = ''; } out($data, " Extra Payload", hexDump($data)); } sub printAes { my $context = shift ; my %saltSize = ( 1 => 8, 2 => 12, 3 => 16, ); myRead(my $salt, $saltSize{$context->{AesStrength} }); out $salt, "AES Salt", hexDump($salt); myRead(my $pwv, 2); out $pwv, "AES Pwd Ver", hexDump($pwv); return $saltSize{$context->{AesStrength}} + 2 + 10; } sub printLzmaProperties { my $len = 0; my $b1; my $b2; my $buffer; myRead($b1, 2); my ($verHi, $verLow) = unpack ("CC", $b1); out $b1, "LZMA Version", sprintf("%02X%02X", $verHi, $verLow) . " '$verHi.$verLow'"; my $LzmaPropertiesSize = out_v "LZMA Properties Size"; $len += 4; my $LzmaInfo = out_C "LZMA Info", sub { $_[0] == 93 ? "(Default)" : ""}; my $PosStateBits = 0; my $LiteralPosStateBits = 0; my $LiteralContextBits = 0; $PosStateBits = int($LzmaInfo / (9 * 5)); $LzmaInfo -= $PosStateBits * 9 * 5; $LiteralPosStateBits = int($LzmaInfo / 9); $LiteralContextBits = $LzmaInfo - $LiteralPosStateBits * 9; out1 " PosStateBits", $PosStateBits; out1 " LiteralPosStateBits", $LiteralPosStateBits; out1 " LiteralContextBits", $LiteralContextBits; out_V "LZMA Dictionary Size"; # TODO - assumption that this is 5 $len += $LzmaPropertiesSize; skip($FH, $LzmaPropertiesSize - 5) if $LzmaPropertiesSize != 5 ; return $len; } sub scanCentralDirectory { my $fh = shift; my $here = $fh->tell(); # Use cases # 1 32-bit CD # 2 64-bit CD my @CD = (); my $offset = findCentralDirectoryOffset($fh); return () if ! defined $offset; $fh->seek($offset, SEEK_SET) ; # Now walk the Central Directory Records my $buffer ; while ($fh->read($buffer, 46) == 46 && unpack("V", $buffer) == ZIP_CENTRAL_HDR_SIG) { my $compressedLength = unpack("V", substr($buffer, 20, 4)); my $uncompressedLength = unpack("V", substr($buffer, 24, 4)); my $filename_length = unpack("v", substr($buffer, 28, 2)); my $extra_length = unpack("v", substr($buffer, 30, 2)); my $comment_length = unpack("v", substr($buffer, 32, 2)); my $locHeaderOffset = unpack("V", substr($buffer, 42, 4)); skip($fh, $filename_length ) ; if ($extra_length) { $fh->read(my $extraField, $extra_length) ; # $self->smartReadExact(\$extraField, $extra_length); # Check for Zip64 # my $zip64Extended = findID("\x01\x00", $extraField); my $zip64Extended = findID(0x0001, $extraField); if ($zip64Extended) { if ($uncompressedLength == 0xFFFFFFFF) { $uncompressedLength = Value_VV64 substr($zip64Extended, 0, 8, ""); # $uncompressedLength = unpack "Q<", substr($zip64Extended, 0, 8, ""); } if ($compressedLength == 0xFFFFFFFF) { $compressedLength = Value_VV64 substr($zip64Extended, 0, 8, ""); # $compressedLength = unpack "Q<", substr($zip64Extended, 0, 8, ""); } if ($locHeaderOffset == 0xFFFFFFFF) { $locHeaderOffset = Value_VV64 substr($zip64Extended, 0, 8, ""); # $locHeaderOffset = unpack "Q<", substr($zip64Extended, 0, 8, ""); } } } my $got = [$locHeaderOffset, $compressedLength] ; # my $v64 = U64->new( $compressedLength ); # my $loc64 = U64->new( $locHeaderOffset ); # my $got = [$loc64, $v64] ; # if (full32 $compressedLength || full32 $locHeaderOffset) { # $fh->read($buffer, $extra_length) ; # # TODO - fix this # die "xxx $offset $comment_length $filename_length $extra_length" . length($buffer) # if length($buffer) != $extra_length; # $got = get64Extra($buffer, full32($uncompressedLength), # $v64, # $loc64); # # If not Zip64 extra field, assume size is 0xFFFFFFFF # #$v64 = $got if defined $got; # } # else { # skip($fh, $extra_length) ; # } skip($fh, $comment_length ) ; push @CD, $got ; } $fh->seek($here, SEEK_SET) ; # @CD = sort { $a->[0]->cmp($b->[0]) } @CD ; @CD = sort { $a->[0] <=> $b->[0] } @CD ; # Set the first Local File Header offset. $START = $CD[0]->[0] if @CD ; return (1, $offset, @CD); } sub offsetFromZip64 { my $fh = shift ; my $here = shift; $fh->seek($here - 20, SEEK_SET) # TODO - fix this or die "xx $!" ; my $buffer; my $got = 0; ($got = $fh->read($buffer, 20)) == 20 # TODO - fix this or die "xxx $here $got $!" ; if ( unpack("V", $buffer) == ZIP64_END_CENTRAL_LOC_HDR_SIG ) { my $cd64 = Value_VV64 substr($buffer, 8, 8); $fh->seek($cd64, SEEK_SET) ; $fh->read($buffer, 4) == 4 # TODO - fix this or die "xxx" ; if ( unpack("V", $buffer) == ZIP64_END_CENTRAL_REC_HDR_SIG ) { $fh->read($buffer, 8) == 8 # TODO - fix this or die "xxx" ; my $size = Value_VV64($buffer); $fh->read($buffer, $size) == $size # TODO - fix this or die "xxx" ; my $cd64 = Value_VV64 substr($buffer, 36, 8); return $cd64 ; } die "Cannot find 'Zip64 end of central directory record': 0x06054b50\nTry running with --scan option.\n" ; } die "Cannot find signature for 'Zip64 end of central directory locator': 0x07064b50 \nTry running with --scan option.\n" ; } use constant Pack_ZIP_END_CENTRAL_HDR_SIG => pack("V", ZIP_END_CENTRAL_HDR_SIG); sub findCentralDirectoryOffset { my $fh = shift ; # Most common use-case is where there is no comment, so # know exactly where the end of central directory record # should be. $fh->seek(-22, SEEK_END) ; my $here = $fh->tell(); my $is64bit = $here > 0xFFFFFFFF; my $over64bit = $here & (~ 0xFFFFFFFF); my $buffer; $fh->read($buffer, 22) == 22 # TODO - fix this or die "xxx" ; my $zip64 = 0; my $centralDirOffset ; if ( unpack("V", $buffer) == ZIP_END_CENTRAL_HDR_SIG ) { $centralDirOffset = unpack("V", substr($buffer, 16, 4)); } else { $fh->seek(0, SEEK_END) ; my $fileLen = $fh->tell(); my $want = 0 ; while(1) { $want += 1024 * 32; my $seekTo = $fileLen - $want; if ($seekTo < 0 ) { $seekTo = 0; $want = $fileLen ; } $fh->seek( $seekTo, SEEK_SET) # TODO - fix this or die "xxx $!" ; my $got; ($got = $fh->read($buffer, $want)) == $want # TODO - fix this or die "xxx $got $!" ; my $pos = rindex( $buffer, Pack_ZIP_END_CENTRAL_HDR_SIG); if ($pos >= 0 && $want - $pos > 22) { $here = $seekTo + $pos ; $centralDirOffset = unpack("V", substr($buffer, $pos + 16, 4)); my $commentLength = unpack("V", substr($buffer, $pos + 20, 2)); $commentLength = 0 if ! defined $commentLength ; my $expectedEof = $fileLen - $want + $pos + 22 + $commentLength ; # check for trailing data after end of zip if ($expectedEof < $fileLen ) { $TRAILING = $expectedEof ; } last ; } return undef if $want == $fileLen; } } if (full32 $centralDirOffset) { $centralDirOffset = offsetFromZip64($fh, $here) } elsif ($is64bit) { # use-case is where a 64-bit zip file doesn't use the 64-bit # extensions. print "EOCD not 64-bit $centralDirOffset ($here)\n" ; push @Messages, sprintf "Zip file > 4Gig. Expected 'Offset to Central Dir' to be 0xFFFFFFFF, got 0x%X\n", $centralDirOffset; $centralDirOffset += $over64bit; $is64In32 = 1; } return $centralDirOffset ; } sub findID { my $id_want = shift ; my $data = shift; my $XLEN = length $data ; my $offset = 0 ; while ($offset < $XLEN) { return undef if $offset + ZIP_EXTRA_SUBFIELD_HEADER_SIZE > $XLEN ; my $id = substr($data, $offset, ZIP_EXTRA_SUBFIELD_ID_SIZE); $id = unpack("v", $id); $offset += ZIP_EXTRA_SUBFIELD_ID_SIZE; my $subLen = unpack("v", substr($data, $offset, ZIP_EXTRA_SUBFIELD_LEN_SIZE)); $offset += ZIP_EXTRA_SUBFIELD_LEN_SIZE ; return undef if $offset + $subLen > $XLEN ; return substr($data, $offset, $subLen) if $id eq $id_want ; $offset += $subLen ; } return undef ; } sub _dosToUnixTime { my $dt = shift; my $year = ( ( $dt >> 25 ) & 0x7f ) + 80; my $mon = ( ( $dt >> 21 ) & 0x0f ) - 1; my $mday = ( ( $dt >> 16 ) & 0x1f ); my $hour = ( ( $dt >> 11 ) & 0x1f ); my $min = ( ( $dt >> 5 ) & 0x3f ); my $sec = ( ( $dt << 1 ) & 0x3e ); use POSIX 'mktime'; my $time_t = mktime( $sec, $min, $hour, $mday, $mon, $year, 0, 0, -1 ); return 0 if ! defined $time_t; return $time_t; } { package U64; use constant MAX32 => 0xFFFFFFFF ; use constant HI_1 => MAX32 + 1 ; use constant LOW => 0 ; use constant HIGH => 1; sub new { my $class = shift ; my $high = 0 ; my $low = 0 ; if (@_ == 2) { $high = shift ; $low = shift ; } elsif (@_ == 1) { my $value = shift ; if ($value > MAX32) { $high = $value >> 32 ; $low = $value & MAX32; } else { $low = $value ; } } bless [$low, $high], $class; } sub newUnpack_V64 { my $string = shift; my ($low, $hi) = unpack "V V", $string ; bless [ $low, $hi ], "U64"; } sub newUnpack_V32 { my $string = shift; my $low = unpack "V", $string ; bless [ $low, 0 ], "U64"; } sub reset { my $self = shift; $self->[HIGH] = $self->[LOW] = 0; } sub clone { my $self = shift; bless [ @$self ], ref $self ; } sub mkU64 { my $value = shift; return $value if ref $value eq 'U64'; bless [ $value, 0 ], "U64" ; } sub getHigh { my $self = shift; return $self->[HIGH]; } sub getLow { my $self = shift; return $self->[LOW]; } sub get32bit { my $self = shift; return $self->[LOW]; } sub get64bit { my $self = shift; # Not using << here because the result will still be # a 32-bit value on systems where int size is 32-bits return $self->[HIGH] * HI_1 + $self->[LOW]; } sub add { my $self = shift; my $value = shift; if (ref $value eq 'U64') { $self->[HIGH] += $value->[HIGH] ; $value = $value->[LOW]; } my $available = MAX32 - $self->[LOW] ; if ($value > $available) { ++ $self->[HIGH] ; $self->[LOW] = $value - $available - 1; } else { $self->[LOW] += $value ; } } sub subtract { my $self = shift; my $value = shift; if (ref $value eq 'U64') { if ($value->[HIGH]) { die "unsupport subtract option" if $self->[HIGH] == 0 || $value->[HIGH] > $self->[HIGH] ; $self->[HIGH] -= $value->[HIGH] ; } $value = $value->[LOW] ; } if ($value > $self->[LOW]) { -- $self->[HIGH] ; $self->[LOW] = MAX32 - $value + $self->[LOW] + 1; } else { $self->[LOW] -= $value; } } sub rshift { my $self = shift; my $count = shift; for (1 .. $count) { $self->[LOW] >>= 1; $self->[LOW] |= 0x80000000 if $self->[HIGH] & 1 ; $self->[HIGH] >>= 1; } } sub is64bit { my $self = shift; return $self->[HIGH] > 0 ; } sub getPacked_V64 { my $self = shift; return pack "V V", @$self ; } sub getPacked_V32 { my $self = shift; return pack "V", $self->[LOW] ; } sub pack_V64 { my $low = shift; return pack "V V", $low, 0; } sub max32 { my $self = shift; return $self->[HIGH] == 0 && $self->[LOW] == MAX32; } sub stringify { my $self = shift; return "High [$self->[HIGH]], Low [$self->[LOW]]"; } sub equal { my $self = shift; my $other = shift; return $self->[LOW] == $other->[LOW] && $self->[HIGH] == $other->[HIGH] ; } sub gt { my $self = shift; my $other = shift; return $self->cmp($other) > 0 ; } sub cmp { my $self = shift; my $other = shift ; if ($self->[LOW] == $other->[LOW]) { return $self->[HIGH] - $other->[HIGH] ; } else { return $self->[LOW] - $other->[LOW] ; } } sub nibbles { my @nibbles = ( [ 16 => HI_1 * 0x10000000 ], [ 15 => HI_1 * 0x1000000 ], [ 14 => HI_1 * 0x100000 ], [ 13 => HI_1 * 0x10000 ], [ 12 => HI_1 * 0x1000 ], [ 11 => HI_1 * 0x100 ], [ 10 => HI_1 * 0x10 ], [ 9 => HI_1 * 0x1 ], [ 8 => 0x10000000 ], [ 7 => 0x1000000 ], [ 6 => 0x100000 ], [ 5 => 0x10000 ], [ 4 => 0x1000 ], [ 3 => 0x100 ], [ 2 => 0x10 ], [ 1 => 0x1 ], ); my $value = shift ; for my $pair (@nibbles) { my ($count, $limit) = @{ $pair }; return $count if $value >= $limit ; } } } { package Seen; sub new { my $class = shift ; my %object = ( overlaps => [], detail => [], ) ; bless \%object, $class; } sub save { my $self = shift ; my $hdrId = shift; my $name = shift; my $from_offset = shift; my $to_offset = shift ; for my $entry ( @{ $self->{detail} } ) { if ( $from_offset == $entry->{from} && $to_offset == $entry->{to} ) { warn "$hdrId: '$name' matches with $entry->{str}\n" } elsif ( ($from_offset >= $entry->{from} && $from_offset <= $entry->{to} ) || ($to_offset >= $entry->{from} && $to_offset <= $entry->{to} ) ) { # die "overlap!" # push @{ $self->{overlap} }, die "$hdrId: '$name' overlaps with $entry->{str}\n" } } warn "ADD $from_offset $to_offset $hdrId: $name\n"; push @{ $self->{detail} }, { from => $from_offset, to => $to_offset, id => $hdrId, name => $name, str => "$hdrId: '$name'", } ; } sub summary { my $self = shift ; for my $outer ( sort { $a->{from} <=> $b->{from} } @{ $self->{detail} } ) { my $from_offset = $outer->{from}; my $to_offset = $outer->{to}; my $hdrId = $outer->{id}; my $name = $outer->{name}; for my $entry ( sort { $a->{from} <=> $b->{from} } @{ $self->{detail} } ) { # next if if ( ($from_offset >= $entry->{from} && $from_offset <= $entry->{to} ) || ($to_offset >= $entry->{from} && $to_offset <= $entry->{to} ) ) { # die "overlap!" die $entry->{str} . " overlaps with $hdrId: '$name'\n" } } } } } sub Usage { if (@_) { warn "$_\n" for @_ ; warn "\n"; } die <<EOM; zipdetails [OPTIONS] file Display details about the internal structure of a Zip file. This is zipdetails version $VERSION OPTIONS -h display help --scan enable scannng mode. Blindly scan the file looking for zip headers Expect false-positives. -v Verbose - output more stuff --version Print version number Copyright (c) 2011-2021 Paul Marquess. All rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. EOM } __END__ =head1 NAME zipdetails - display the internal structure of zip files =head1 SYNOPSIS zipdetails [-v][--scan] zipfile.zip zipdetails -h zipdetails --version =head1 DESCRIPTION Zipdetails displays information about the internal record structure of zip files. It is not concerned with displaying any details of the compressed data stored in the zip file. The program assumes prior understanding of the internal structure of a Zip file. You should have a copy of the Zip APPNOTE file at hand to help understand the output from this program (L<SEE ALSO> for details). =head2 Default Behaviour By default the program expects to be given a well-formed zip file. It will navigate the Zip file by first parsing the zip central directory at the end of the file. If that is found, it will then walk through the zip records starting at the beginning of the file. Any badly formed zip data structures encountered are likely to terminate the program. If the program finds any structural problems with the zip file it will print a summary at the end of the output report. The set of error cases reported is very much a work in progress, so don't rely on this feature to find all the possible errors in a zip file. If you have suggestions for use-cases where this could be enhanced please consider creating an enhancement request (see L<"SUPPORT">). =head2 Scan-Mode If you do have a potentially corrupt zip file, particulatly where the central directory at the end of the file is absent/incomplete, you can try usng the C<--scan> option to search for zip records that are still present. When Scan-mode is enabled, the program will walk the zip file from the start blindly looking for the 4-byte signatures that preceed each of the zip data structures. If it finds any of the recognised signatures it will attempt to dump the associated zip record. For very large zip files, this operation can take a long time to run. Note that the 4-byte signatures used in zip files can sometimes match with random data stored in the zip file, so care is needed interpreting the results. =head2 OPTIONS =over 5 =item -h Display help =item --scan Walk the zip file loking for possible zip records. Can be error-prone. See L<"Scan-Mode"> =item -v Enable Verbose mode. See L<"Verbose Output">. =item --version Display version number of the program and exit. =back =head2 Default Output By default zipdetails will output the details of the zip file in three columns. =over 5 =item Column 1 This contains the offset from the start of the file in hex. =item Column 2 This contains a textual description of the field. =item Column 3 If the field contains a numeric value it will be displayed in hex. Zip stores most numbers in little-endian format - the value displayed will have the little-endian encoding removed. Next, is an optional description of what the value means. =back =head2 Verbose Output If the C<-v> option is present, column 1 is expanded to include =over 5 =item * The offset from the start of the file in hex. =item * The length of the field in hex. =item * A hex dump of the bytes in field in the order they are stored in the zip file. =back =head1 LIMITATIONS The following zip file features are not supported by this program: =over 5 =item * Multi-part archives. =item * The strong encryption features defined in the "APPNOTE" document. =back =head1 TODO Error handling is a work in progress. If the program encounters a problem reading a zip file it is likely to terminate with an unhelpful error message. =head1 SUPPORT General feedback/questions/bug reports should be sent to L<https://github.com/pmqs/IO-Compress/issues> (preferred) or L<https://rt.cpan.org/Public/Dist/Display.html?Name=IO-Compress>. =head1 SEE ALSO The primary reference for Zip files is the "APPNOTE" document available at L<http://www.pkware.com/documents/casestudies/APPNOTE.TXT>. An alternative reference is the Info-Zip appnote. This is available from L<ftp://ftp.info-zip.org/pub/infozip/doc/> The C<zipinfo> program that comes with the info-zip distribution (L<http://www.info-zip.org/>) can also display details of the structure of a zip file. See also L<Archive::Zip::SimpleZip>, L<IO::Compress::Zip>, L<IO::Uncompress::Unzip>. =head1 AUTHOR Paul Marquess F<pmqs@cpan.org>. =head1 COPYRIGHT Copyright (c) 2011-2021 Paul Marquess. All rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
.
Edit
..
Edit
2to3
Edit
AlLicenseCP
Edit
CSPUpdate
Edit
CSPUpdateFallback
Edit
GET
Edit
HEAD
Edit
Mail
Edit
POST
Edit
[
Edit
ab
Edit
aclocal
Edit
aclocal-1.16
Edit
acyclic
Edit
addr2line
Edit
aec
Edit
alias
Edit
alt-java
Edit
alt-mysql-reconfigure
Edit
alt-php-mysql-reconfigure
Edit
alt-php-mysql-reconfigure.py
Edit
amuFormat.sh
Edit
animate
Edit
appstream-compose
Edit
appstream-util
Edit
apropos
Edit
apropos.man-db
Edit
ar
Edit
arch
Edit
aria_chk
Edit
aria_dump_log
Edit
aria_ftdump
Edit
aria_pack
Edit
aria_read_log
Edit
arpaname
Edit
arping
Edit
as
Edit
aserver
Edit
aspell
Edit
at
Edit
atq
Edit
atrm
Edit
attr
Edit
audit2allow
Edit
audit2why
Edit
aulast
Edit
aulastlog
Edit
ausyscall
Edit
authselect
Edit
autoconf
Edit
autoexpect
Edit
autoheader
Edit
autom4te
Edit
automake
Edit
automake-1.16
Edit
autopoint
Edit
autoreconf
Edit
autoscan
Edit
autoupdate
Edit
auvirt
Edit
awk
Edit
b2sum
Edit
base32
Edit
base64
Edit
basename
Edit
basenc
Edit
bash
Edit
bashbug
Edit
bashbug-64
Edit
batch
Edit
bc
Edit
bcomps
Edit
bdf2gdfont.pl
Edit
bg
Edit
bison
Edit
bond2team
Edit
bootconfig
Edit
bootctl
Edit
brotli
Edit
bsqldb
Edit
bsqlodbc
Edit
btop
Edit
bundle
Edit
bundler
Edit
bunzip2
Edit
busctl
Edit
bwrap
Edit
bzcat
Edit
bzcmp
Edit
bzdiff
Edit
bzegrep
Edit
bzfgrep
Edit
bzgrep
Edit
bzip2
Edit
bzip2recover
Edit
bzless
Edit
bzmore
Edit
c++
Edit
c++filt
Edit
c89
Edit
c99
Edit
ca-legacy
Edit
cagefs.server
Edit
cagefs_enter
Edit
cagefs_enter.proxied
Edit
cal
Edit
canberra-boot
Edit
canberra-gtk-play
Edit
captoinfo
Edit
cat
Edit
catchsegv
Edit
catman
Edit
cc
Edit
ccomps
Edit
cd
Edit
centrino-decode
Edit
certbot
Edit
certbot-3
Edit
certutil
Edit
chacl
Edit
chage
Edit
chardetect
Edit
chattr
Edit
chcat
Edit
chcon
Edit
checkipv2
Edit
checkmodule
Edit
checkpolicy
Edit
checksctp
Edit
checkstatus
Edit
chfn
Edit
chgrp
Edit
chmem
Edit
chmod
Edit
choom
Edit
chown
Edit
chronyc
Edit
chrt
Edit
chsh
Edit
chvt
Edit
cifsiostat
Edit
circo
Edit
cksum
Edit
cl-linksafe-apply-group
Edit
cl-linksafe-reconfigure
Edit
cl-phpextdesc
Edit
cl-quota
Edit
cl-syncpkgs
Edit
cldetect
Edit
cldiag
Edit
clear
Edit
clockdiff
Edit
cloudlinux-awp-admin
Edit
cloudlinux-awp-daemon
Edit
cloudlinux-awp-user
Edit
cloudlinux_domains_collector
Edit
clsupergid_process
Edit
cluster
Edit
clwpos-admin
Edit
clwpos-daemon
Edit
clwpos-erase
Edit
clwpos-user
Edit
cmp
Edit
cmsutil
Edit
col
Edit
colcrt
Edit
colrm
Edit
column
Edit
comm
Edit
command
Edit
compare
Edit
compile_et
Edit
composite
Edit
config_data
Edit
conjure
Edit
consolehelper
Edit
convert
Edit
coredumpctl
Edit
corelist
Edit
cp
Edit
cpan
Edit
cpan-mirrors
Edit
cpansign
Edit
cpapi1
Edit
cpapi2
Edit
cpapi3
Edit
cpio
Edit
cpp
Edit
cpupower
Edit
crb
Edit
crc32
Edit
crlutil
Edit
cronnext
Edit
crontab
Edit
crontab.cagefs
Edit
cspdaemon
Edit
cspfwd
Edit
csplit
Edit
curl
Edit
curl-config
Edit
curve_keygen
Edit
cut
Edit
cvtsudoers
Edit
cxpm
Edit
cyrusbdb2current
Edit
da-addsudoer
Edit
da-removesudoer
Edit
datacopy
Edit
date
Edit
dbilogstrip
Edit
dbiprof
Edit
dbus-broker
Edit
dbus-broker-launch
Edit
dbus-monitor
Edit
dbus-send
Edit
dbus-update-activation-environment
Edit
dbus-uuidgen
Edit
dc
Edit
dconf
Edit
dd
Edit
deallocvt
Edit
debuginfo-install
Edit
debuginfod-find
Edit
defncopy
Edit
delv
Edit
df
Edit
dict_util
Edit
dict_util_string
Edit
diff
Edit
diff3
Edit
diffimg
Edit
dig
Edit
dijkstra
Edit
dir
Edit
dircolors
Edit
dirmngr
Edit
dirmngr-client
Edit
dirname
Edit
dislocate
Edit
display
Edit
distro
Edit
dltest
Edit
dmesg
Edit
dnf
Edit
dnf-3
Edit
dnsdomainname
Edit
dnstap-read
Edit
domainname
Edit
dot
Edit
dot2gxl
Edit
dotty
Edit
doveadm
Edit
doveconf
Edit
dovecot-sysreport
Edit
dpkg
Edit
dpkg-deb
Edit
dpkg-divert
Edit
dpkg-maintscript-helper
Edit
dpkg-query
Edit
dpkg-realpath
Edit
dpkg-split
Edit
dpkg-statoverride
Edit
dpkg-trigger
Edit
dracut
Edit
dsync
Edit
dtrace
Edit
du
Edit
dumpkeys
Edit
dwp
Edit
dwz
Edit
ea-php56
Edit
ea-php56-pear
Edit
ea-php56-pecl
Edit
ea-php74
Edit
ea-php74-pear
Edit
ea-php74-pecl
Edit
ea-php81
Edit
ea-php81-pear
Edit
ea-php81-pecl
Edit
ea-php82
Edit
ea-php82-pear
Edit
ea-php82-pecl
Edit
ea-php83
Edit
ea-php83-pear
Edit
ea-php83-pecl
Edit
ea-php84
Edit
ea-php84-pear
Edit
ea-php84-pecl
Edit
ea-wappspector
Edit
echo
Edit
ed
Edit
edgepaint
Edit
egrep
Edit
eject
Edit
elfedit
Edit
enc2xs
Edit
encguess
Edit
enchant
Edit
enchant-lsmod
Edit
env
Edit
envml
Edit
envsubst
Edit
eps2eps
Edit
eqn
Edit
erb
Edit
event_rpcgen.py
Edit
evmctl
Edit
ex
Edit
exempi
Edit
exiv2
Edit
expand
Edit
expect
Edit
expr
Edit
factor
Edit
fallocate
Edit
false
Edit
fbsvcmgr
Edit
fbtracemgr
Edit
fc
Edit
fc-cache
Edit
fc-cache-64
Edit
fc-cat
Edit
fc-conflist
Edit
fc-list
Edit
fc-match
Edit
fc-pattern
Edit
fc-query
Edit
fc-scan
Edit
fc-validate
Edit
fdp
Edit
fg
Edit
fgconsole
Edit
fgrep
Edit
filan
Edit
file
Edit
fincore
Edit
find
Edit
find-repos-of-install
Edit
findmnt
Edit
fips-finish-install
Edit
fips-mode-setup
Edit
firewall-cmd
Edit
firewall-offline-cmd
Edit
fisql
Edit
flatpak
Edit
flatpak-bisect
Edit
flatpak-coredumpctl
Edit
flex
Edit
flex++
Edit
flock
Edit
fmt
Edit
fold
Edit
free
Edit
freebcp
Edit
freetype-config
Edit
fribidi
Edit
ftp
Edit
ftp-rfc
Edit
funzip
Edit
fuse2fs
Edit
fusermount
Edit
futurize
Edit
futurize-3
Edit
futurize-3.9
Edit
g++
Edit
g13
Edit
galera_new_cluster
Edit
galera_recovery
Edit
gapplication
Edit
garb-systemd
Edit
garbd
Edit
gawk
Edit
gbak
Edit
gc
Edit
gcc
Edit
gcc-ar
Edit
gcc-nm
Edit
gcc-ranlib
Edit
gcov
Edit
gcov-dump
Edit
gcov-tool
Edit
gdbm_dump
Edit
gdbm_load
Edit
gdbmtool
Edit
gdbus
Edit
gdbus-codegen
Edit
gdk-pixbuf-query-loaders-64
Edit
gdk-pixbuf-thumbnailer
Edit
gem
Edit
gencat
Edit
genisoimage
Edit
genl-ctrl-list
Edit
geoiplookup
Edit
geoiplookup6
Edit
geosop
Edit
geqn
Edit
getconf
Edit
getcontrolpaneluserspackages
Edit
getent
Edit
getfacl
Edit
getfattr
Edit
gethostip
Edit
getkeycodes
Edit
getopt
Edit
getopts
Edit
gettext
Edit
gettext.sh
Edit
gettextize
Edit
gfix
Edit
ghostscript
Edit
gif2h5
Edit
gio
Edit
gio-querymodules-64
Edit
git
Edit
git-receive-pack
Edit
git-shell
Edit
git-upload-archive
Edit
git-upload-pack
Edit
glances
Edit
glib-compile-resources
Edit
glib-compile-schemas
Edit
glib-genmarshal
Edit
glib-gettextize
Edit
glib-mkenums
Edit
gm
Edit
gmake
Edit
gml2gv
Edit
gneqn
Edit
gnroff
Edit
gobject-query
Edit
gpasswd
Edit
gpg
Edit
gpg-agent
Edit
gpg-card
Edit
gpg-connect-agent
Edit
gpg-error
Edit
gpg-error-config
Edit
gpg-wks-client
Edit
gpg-wks-server
Edit
gpg2
Edit
gpgconf
Edit
gpgme-json
Edit
gpgparsemail
Edit
gpgrt-config
Edit
gpgsplit
Edit
gpgtar
Edit
gpgv
Edit
gpgv2
Edit
gpic
Edit
gpio-event-mon
Edit
gpio-hammer
Edit
gpio-watch
Edit
gpre
Edit
gprof
Edit
gr2fonttest
Edit
graphml2gv
Edit
grep
Edit
gresource
Edit
groff
Edit
grops
Edit
grotty
Edit
groups
Edit
grpc_cpp_plugin
Edit
grpc_csharp_plugin
Edit
grpc_node_plugin
Edit
grpc_objective_c_plugin
Edit
grpc_php_plugin
Edit
grpc_python_plugin
Edit
grpc_ruby_plugin
Edit
grub2-editenv
Edit
grub2-file
Edit
grub2-fstest
Edit
grub2-glue-efi
Edit
grub2-kbdcomp
Edit
grub2-menulst2cfg
Edit
grub2-mkfont
Edit
grub2-mkimage
Edit
grub2-mklayout
Edit
grub2-mknetdir
Edit
grub2-mkpasswd-pbkdf2
Edit
grub2-mkrelpath
Edit
grub2-mkrescue
Edit
grub2-mkstandalone
Edit
grub2-mount
Edit
grub2-render-label
Edit
grub2-script-check
Edit
grub2-syslinux2cfg
Edit
gs
Edit
gsbj
Edit
gsdj
Edit
gsdj500
Edit
gsec
Edit
gsettings
Edit
gsf-office-thumbnailer
Edit
gslj
Edit
gslp
Edit
gsnd
Edit
gsoelim
Edit
gsplit
Edit
gst-inspect-1.0
Edit
gst-launch-1.0
Edit
gst-stats-1.0
Edit
gst-typefind-1.0
Edit
gstat-fb
Edit
gtar
Edit
gtbl
Edit
gtester
Edit
gtester-report
Edit
gtk-launch
Edit
gtk-query-immodules-2.0-64
Edit
gtk-query-immodules-3.0-64
Edit
gtk-update-icon-cache
Edit
gtroff
Edit
gunzip
Edit
gv2gml
Edit
gv2gxl
Edit
gvcolor
Edit
gvgen
Edit
gvmap
Edit
gvmap.sh
Edit
gvpack
Edit
gvpr
Edit
gxl2dot
Edit
gxl2gv
Edit
gzexe
Edit
gzip
Edit
h2ph
Edit
h2xs
Edit
h52gif
Edit
h5clear
Edit
h5copy
Edit
h5debug
Edit
h5diff
Edit
h5dump
Edit
h5format_convert
Edit
h5import
Edit
h5jam
Edit
h5ls
Edit
h5mkgrp
Edit
h5perf_serial
Edit
h5repack
Edit
h5repart
Edit
h5stat
Edit
h5unjam
Edit
h5watch
Edit
hardlink
Edit
hash
Edit
hb-ot-shape-closure
Edit
hb-shape
Edit
hb-subset
Edit
hb-view
Edit
head
Edit
hexdump
Edit
host
Edit
hostid
Edit
hostname
Edit
hostnamectl
Edit
htdbm
Edit
htdigest
Edit
html2text
Edit
htop
Edit
htpasswd
Edit
httxt2dbm
Edit
hunspell
Edit
i360-check-flow
Edit
i360-exclcomp
Edit
i360-pkg-watcher
Edit
i360-renameat2
Edit
i360-storage
Edit
i360-storage-hdb
Edit
i360-storage-mkcustom
Edit
i360-storage-new
Edit
i360-storage-replacehdb-v2
Edit
i360-storage.immunity
Edit
i360-storage.v2
Edit
i360-sync-config
Edit
i360-sync-config.sh
Edit
i360_wafd_check
Edit
i360_wafd_list
Edit
i386
Edit
iconv
Edit
icu-config
Edit
icu-config-64
Edit
icuinfo
Edit
id
Edit
identify
Edit
idiag-socket-details
Edit
idn
Edit
ifnames
Edit
iio_event_monitor
Edit
iio_generic_buffer
Edit
im360-k8s-syncer
Edit
ima-add-sigs
Edit
ima-setup
Edit
import
Edit
imunify-agent-proxy
Edit
imunify-antivirus
Edit
imunify-fgw-dump
Edit
imunify-resident
Edit
imunify-service
Edit
imunify360-agent
Edit
imunify360-command-wrapper
Edit
imunify360-dos-protection
Edit
info
Edit
infocmp
Edit
infotocap
Edit
innochecksum
Edit
inotifywait
Edit
inotifywatch
Edit
install
Edit
instmodsh
Edit
intel-speed-select
Edit
ionice
Edit
iostat
Edit
ipcmk
Edit
ipcrm
Edit
ipcs
Edit
iptc
Edit
irqtop
Edit
isodebug
Edit
isodump
Edit
isohybrid
Edit
isoinfo
Edit
isosize
Edit
isovfy
Edit
ispell
Edit
isppackagesreducer
Edit
isql
Edit
isql-fb
Edit
iusql
Edit
java
Edit
jcmd
Edit
jetapps
Edit
jetbackup5
Edit
jetbackup5api
Edit
jetmongo
Edit
jjs
Edit
jobs
Edit
join
Edit
journalctl
Edit
jq
Edit
json_pp
Edit
json_reformat
Edit
json_verify
Edit
json_xs
Edit
jws
Edit
kbd_mode
Edit
kbdinfo
Edit
kbdrate
Edit
kdumpctl
Edit
kernel-install
Edit
keyctl
Edit
keytool
Edit
kibitz
Edit
kill
Edit
killall
Edit
kmod
Edit
krb5-config
Edit
kvm_stat
Edit
last
Edit
lastb
Edit
lastlog
Edit
lchfn
Edit
lchsh
Edit
ld
Edit
ld.bfd
Edit
ld.gold
Edit
ld.so
Edit
ldd
Edit
lefty
Edit
less
Edit
lessecho
Edit
lesskey
Edit
lesspipe.sh
Edit
letsencrypt
Edit
lex
Edit
lexgrog
Edit
libnetcfg
Edit
libpng-config
Edit
libpng16-config
Edit
libtool
Edit
libtoolize
Edit
libwmf-fontmap
Edit
link
Edit
linux-boot-prober
Edit
linux32
Edit
linux64
Edit
ln
Edit
lneato
Edit
loadkeys
Edit
loadunimap
Edit
locale
Edit
localectl
Edit
localedef
Edit
logger
Edit
login
Edit
loginctl
Edit
logname
Edit
logresolve
Edit
look
Edit
lpunlock
Edit
ls
Edit
lsattr
Edit
lsblk
Edit
lscpu
Edit
lsgpio
Edit
lsiio
Edit
lsinitrd
Edit
lsipc
Edit
lsirq
Edit
lslocks
Edit
lslogins
Edit
lsmem
Edit
lsns
Edit
lsof
Edit
lsphp
Edit
lsscsi
Edit
lto-dump
Edit
lua
Edit
luac
Edit
lve_bash
Edit
lve_ksh
Edit
lve_pdksh
Edit
lve_suwrapper
Edit
lve_tcsh
Edit
lve_umount
Edit
lve_wrapper
Edit
lve_zsh
Edit
lvemanager-service
Edit
lveutils-panel-cron
Edit
lwp-download
Edit
lwp-dump
Edit
lwp-mirror
Edit
lwp-request
Edit
lz
Edit
m4
Edit
mail
Edit
mailx
Edit
mailx.s-nail
Edit
make
Edit
make-dummy-cert
Edit
man
Edit
man-recode
Edit
man.man-db
Edit
mandb
Edit
manpath
Edit
mapscrn
Edit
mariadb
Edit
mariadb-access
Edit
mariadb-admin
Edit
mariadb-binlog
Edit
mariadb-check
Edit
mariadb-config
Edit
mariadb-conv
Edit
mariadb-convert-table-format
Edit
mariadb-dump
Edit
mariadb-dumpslow
Edit
mariadb-embedded
Edit
mariadb-find-rows
Edit
mariadb-fix-extensions
Edit
mariadb-hotcopy
Edit
mariadb-import
Edit
mariadb-install-db
Edit
mariadb-plugin
Edit
mariadb-secure-installation
Edit
mariadb-service-convert
Edit
mariadb-setpermission
Edit
mariadb-show
Edit
mariadb-slap
Edit
mariadb-tzinfo-to-sql
Edit
mariadb-upgrade
Edit
mariadb-waitpid
Edit
mariadb_config
Edit
mariadbd-multi
Edit
mariadbd-safe
Edit
mariadbd-safe-helper
Edit
matdump
Edit
mattrib
Edit
mbadblocks
Edit
mcat
Edit
mcd
Edit
mcheck
Edit
mclasserase
Edit
mcomp
Edit
mcookie
Edit
mcopy
Edit
md5sum
Edit
mdb_copy
Edit
mdb_dump
Edit
mdb_load
Edit
mdb_stat
Edit
mdel
Edit
mdeltree
Edit
mdig
Edit
mdir
Edit
mdu
Edit
memdiskfind
Edit
mesg
Edit
mformat
Edit
migration_ve1_to_v2.py
Edit
minfo
Edit
mirror_server
Edit
mirror_server_stop
Edit
mkdir
Edit
mkfifo
Edit
mkfontdir
Edit
mkfontscale
Edit
mkhybrid
Edit
mkisofs
Edit
mkmanifest
Edit
mknod
Edit
mkpasswd-expect
Edit
mktemp
Edit
mlabel
Edit
mm2gv
Edit
mmd
Edit
mmdblookup
Edit
mmount
Edit
mmove
Edit
modulecmd
Edit
modulemd-validator
Edit
modutil
Edit
mogrify
Edit
montage
Edit
more
Edit
mount
Edit
mountpoint
Edit
mpartition
Edit
mpstat
Edit
mrd
Edit
mren
Edit
msgattrib
Edit
msgcat
Edit
msgcmp
Edit
msgcomm
Edit
msgconv
Edit
msgen
Edit
msgexec
Edit
msgfilter
Edit
msgfmt
Edit
msgfmt.py
Edit
msgfmt3.9.py
Edit
msgfmt3.py
Edit
msggrep
Edit
msginit
Edit
msgmerge
Edit
msgunfmt
Edit
msguniq
Edit
mshortname
Edit
mshowfat
Edit
msql2mysql
Edit
mtools
Edit
mtoolstest
Edit
mtype
Edit
mv
Edit
mxtar
Edit
my_print_defaults
Edit
myisam_ftdump
Edit
myisamchk
Edit
myisamlog
Edit
myisampack
Edit
mysql
Edit
mysql_config
Edit
mysql_embedded
Edit
mysql_find_rows
Edit
mysql_fix_extensions
Edit
mysql_install_db
Edit
mysql_plugin
Edit
mysql_tzinfo_to_sql
Edit
mysql_upgrade
Edit
mysql_waitpid
Edit
mysqlaccess
Edit
mysqladmin
Edit
mysqlbinlog
Edit
mysqlcheck
Edit
mysqld_multi
Edit
mysqld_safe
Edit
mysqld_safe_helper
Edit
mysqldump
Edit
mysqlimport
Edit
mysqlshow
Edit
mysqlslap
Edit
mytop
Edit
mzip
Edit
nail
Edit
named-rrchecker
Edit
namei
Edit
nano
Edit
nbackup
Edit
nc
Edit
ncurses6-config
Edit
ncursesw6-config
Edit
ndptool
Edit
neato
Edit
needs-restarting
Edit
neqn
Edit
netcat
Edit
netstat
Edit
newgidmap
Edit
newgrp
Edit
newuidmap
Edit
nf-ct-add
Edit
nf-ct-events
Edit
nf-ct-list
Edit
nf-exp-add
Edit
nf-exp-delete
Edit
nf-exp-list
Edit
nf-log
Edit
nf-monitor
Edit
nf-queue
Edit
ngettext
Edit
nice
Edit
nisdomainname
Edit
nl
Edit
nl-addr-add
Edit
nl-addr-delete
Edit
nl-addr-list
Edit
nl-class-add
Edit
nl-class-delete
Edit
nl-class-list
Edit
nl-classid-lookup
Edit
nl-cls-add
Edit
nl-cls-delete
Edit
nl-cls-list
Edit
nl-fib-lookup
Edit
nl-link-enslave
Edit
nl-link-ifindex2name
Edit
nl-link-list
Edit
nl-link-name2ifindex
Edit
nl-link-release
Edit
nl-link-set
Edit
nl-link-stats
Edit
nl-list-caches
Edit
nl-list-sockets
Edit
nl-monitor
Edit
nl-neigh-add
Edit
nl-neigh-delete
Edit
nl-neigh-list
Edit
nl-neightbl-list
Edit
nl-nh-list
Edit
nl-pktloc-lookup
Edit
nl-qdisc-add
Edit
nl-qdisc-delete
Edit
nl-qdisc-list
Edit
nl-route-add
Edit
nl-route-delete
Edit
nl-route-get
Edit
nl-route-list
Edit
nl-rule-list
Edit
nl-tctree-list
Edit
nl-util-addr
Edit
nload
Edit
nm
Edit
nm-online
Edit
nmcli
Edit
nmtui
Edit
nmtui-connect
Edit
nmtui-edit
Edit
nmtui-hostname
Edit
node
Edit
nohup
Edit
nop
Edit
notify-send
Edit
npm
Edit
nproc
Edit
npx
Edit
nroff
Edit
nsenter
Edit
nslookup
Edit
nss-policy-check
Edit
nsupdate
Edit
numfmt
Edit
objcopy
Edit
objdump
Edit
od
Edit
odbc_config
Edit
odbcinst
Edit
openssl
Edit
openvt
Edit
orc-bugreport
Edit
os-prober
Edit
osage
Edit
osinfo-db-export
Edit
osinfo-db-import
Edit
osinfo-db-path
Edit
osinfo-db-validate
Edit
osinfo-detect
Edit
osinfo-install-script
Edit
osinfo-query
Edit
osql
Edit
p11-kit
Edit
pack200
Edit
package-cleanup
Edit
package_reinstaller.py
Edit
page_owner_sort
Edit
pango-list
Edit
pango-segmentation
Edit
pango-view
Edit
paperconf
Edit
passenger
Edit
passenger-config
Edit
passmass
Edit
passwd
Edit
paste
Edit
pasteurize
Edit
pasteurize-3
Edit
pasteurize-3.9
Edit
patch
Edit
patchwork
Edit
pathchk
Edit
pathfix.py
Edit
pathfix3.9.py
Edit
pcre-config
Edit
pcre2-config
Edit
pdf2dsc
Edit
pdf2ps
Edit
pdfimage
Edit
pdns_control
Edit
pdnsutil
Edit
peekfd
Edit
perl
Edit
perl5.32.1
Edit
perlbug
Edit
perldoc
Edit
perlivp
Edit
perlml
Edit
perlthanks
Edit
perror
Edit
pf2afm
Edit
pfbtopfa
Edit
pflags
Edit
pftp
Edit
pgrep
Edit
php
Edit
pic
Edit
piconv
Edit
pidof
Edit
pidstat
Edit
pidwait
Edit
pigz
Edit
ping
Edit
pinky
Edit
pip
Edit
pip-3
Edit
pip-3.9
Edit
pip3
Edit
pip3.9
Edit
pipewire
Edit
pipewire-aes67
Edit
pipewire-avb
Edit
pipewire-pulse
Edit
pipewire-vulkan
Edit
pitchplay
Edit
pk12util
Edit
pkaction
Edit
pkcheck
Edit
pkexec
Edit
pkg-config
Edit
pkgconf
Edit
pkill
Edit
pkla-admin-identities
Edit
pkla-check-authorization
Edit
pkttyagent
Edit
pl2pm
Edit
pldd
Edit
plesk_configure
Edit
pmap
Edit
png-fix-itxt
Edit
pngfix
Edit
pod2html
Edit
pod2man
Edit
pod2text
Edit
pod2usage
Edit
podchecker
Edit
post-grohtml
Edit
powernow-k8-decode
Edit
pphs
Edit
pr
Edit
pre-grohtml
Edit
precat
Edit
preconv
Edit
preunzip
Edit
prezip
Edit
prezip-bin
Edit
printafm
Edit
printenv
Edit
printf
Edit
prlimit
Edit
procan
Edit
protoc
Edit
protoc-c
Edit
protoc-gen-c
Edit
prove
Edit
prtstat
Edit
prune
Edit
ps
Edit
ps2ascii
Edit
ps2epsi
Edit
ps2pdf
Edit
ps2pdf12
Edit
ps2pdf13
Edit
ps2pdf14
Edit
ps2pdfwr
Edit
ps2ps
Edit
ps2ps2
Edit
psfaddtable
Edit
psfgettable
Edit
psfstriptable
Edit
psfxtable
Edit
pslog
Edit
pstree
Edit
pstree.x11
Edit
ptar
Edit
ptardiff
Edit
ptargrep
Edit
ptx
Edit
pv
Edit
pw-jack
Edit
pwd
Edit
pwdx
Edit
pwmake
Edit
pwscore
Edit
pydoc
Edit
pydoc3
Edit
pydoc3.9
Edit
pygettext.py
Edit
pygettext3.9.py
Edit
pygettext3.py
Edit
pyinotify
Edit
python-config
Edit
python-html2text
Edit
python3
Edit
python3-config
Edit
python3-html2text
Edit
python3.9
Edit
python3.9-config
Edit
python3.9-x86_64-config
Edit
qli
Edit
quota
Edit
quotasync
Edit
ranlib
Edit
rdoc
Edit
re2c
Edit
re2go
Edit
read
Edit
readelf
Edit
readlink
Edit
readmult
Edit
realpath
Edit
recode
Edit
recode-sr-latin
Edit
red
Edit
remove_reseller
Edit
rename
Edit
renew-dummy-cert
Edit
renice
Edit
replace
Edit
repo-graph
Edit
repoclosure
Edit
repodiff
Edit
repomanage
Edit
repoquery
Edit
reposync
Edit
repotrack
Edit
rescan-scsi-bus.sh
Edit
reset
Edit
resizecons
Edit
resolve_stack_dump
Edit
resolveip
Edit
restore_infected
Edit
rev
Edit
rftp
Edit
rhn_register
Edit
ri
Edit
rlogin-cwd
Edit
rm
Edit
rmdir
Edit
rmid
Edit
rmiregistry
Edit
rnano
Edit
rpcbind
Edit
rpcinfo
Edit
rpm
Edit
rpm2archive
Edit
rpm2cpio
Edit
rpmdb
Edit
rpmkeys
Edit
rpmquery
Edit
rpmverify
Edit
rrdcached
Edit
rrdcgi
Edit
rrdcreate
Edit
rrdinfo
Edit
rrdtool
Edit
rrdupdate
Edit
rsync
Edit
rsync-ssl
Edit
ruby
Edit
run-parts
Edit
run-with-aspell
Edit
runcon
Edit
rvi
Edit
rview
Edit
s-nail
Edit
sadf
Edit
sar
Edit
scalar
Edit
sccmap
Edit
scl
Edit
scl_enabled
Edit
scl_source
Edit
scp
Edit
screen
Edit
script
Edit
scriptlive
Edit
scriptreplay
Edit
scsi-rescan
Edit
scsi_logging_level
Edit
scsi_mandat
Edit
scsi_readcap
Edit
scsi_ready
Edit
scsi_satl
Edit
scsi_start
Edit
scsi_stop
Edit
scsi_temperature
Edit
sctp_darn
Edit
sctp_status
Edit
sctp_test
Edit
sdiff
Edit
secon
Edit
sed
Edit
sedismod
Edit
sedispol
Edit
selectorctl
Edit
semodule_expand
Edit
semodule_link
Edit
semodule_package
Edit
semodule_unpackage
Edit
seq
Edit
sestatus
Edit
setarch
Edit
setfacl
Edit
setfattr
Edit
setfont
Edit
setkeycodes
Edit
setleds
Edit
setmetamode
Edit
setpriv
Edit
setsid
Edit
setterm
Edit
setup-nsssysinit
Edit
setup-nsssysinit.sh
Edit
setvtrgb
Edit
sfdp
Edit
sftp
Edit
sg
Edit
sg_bg_ctl
Edit
sg_compare_and_write
Edit
sg_copy_results
Edit
sg_dd
Edit
sg_decode_sense
Edit
sg_emc_trespass
Edit
sg_format
Edit
sg_get_config
Edit
sg_get_elem_status
Edit
sg_get_lba_status
Edit
sg_ident
Edit
sg_inq
Edit
sg_logs
Edit
sg_luns
Edit
sg_map
Edit
sg_map26
Edit
sg_modes
Edit
sg_opcodes
Edit
sg_persist
Edit
sg_prevent
Edit
sg_raw
Edit
sg_rbuf
Edit
sg_rdac
Edit
sg_read
Edit
sg_read_attr
Edit
sg_read_block_limits
Edit
sg_read_buffer
Edit
sg_read_long
Edit
sg_readcap
Edit
sg_reassign
Edit
sg_referrals
Edit
sg_rep_pip
Edit
sg_rep_zones
Edit
sg_requests
Edit
sg_reset
Edit
sg_reset_wp
Edit
sg_rmsn
Edit
sg_rtpg
Edit
sg_safte
Edit
sg_sanitize
Edit
sg_sat_identify
Edit
sg_sat_phy_event
Edit
sg_sat_read_gplog
Edit
sg_sat_set_features
Edit
sg_scan
Edit
sg_seek
Edit
sg_senddiag
Edit
sg_ses
Edit
sg_ses_microcode
Edit
sg_start
Edit
sg_stpg
Edit
sg_stream_ctl
Edit
sg_sync
Edit
sg_test_rwbuf
Edit
sg_timestamp
Edit
sg_turs
Edit
sg_unmap
Edit
sg_verify
Edit
sg_vpd
Edit
sg_wr_mode
Edit
sg_write_buffer
Edit
sg_write_long
Edit
sg_write_same
Edit
sg_write_verify
Edit
sg_write_x
Edit
sg_xcopy
Edit
sg_zone
Edit
sginfo
Edit
sgm_dd
Edit
sgp_dd
Edit
sh
Edit
sha1hmac
Edit
sha1sum
Edit
sha224hmac
Edit
sha224sum
Edit
sha256hmac
Edit
sha256sum
Edit
sha384hmac
Edit
sha384sum
Edit
sha512hmac
Edit
sha512sum
Edit
shasum
Edit
showconsolefont
Edit
showkey
Edit
shred
Edit
shuf
Edit
signver
Edit
size
Edit
skill
Edit
slabinfo
Edit
slabtop
Edit
sleep
Edit
slencheck
Edit
sm3hmac
Edit
snice
Edit
socat
Edit
soelim
Edit
soelim.groff
Edit
sort
Edit
sotruss
Edit
spell
Edit
splain
Edit
split
Edit
sprof
Edit
sqlite3
Edit
ssh
Edit
ssh-add
Edit
ssh-agent
Edit
ssh-copy-id
Edit
ssh-keygen
Edit
ssh-keyscan
Edit
ssltap
Edit
sss_ssh_authorizedkeys
Edit
sss_ssh_knownhostsproxy
Edit
stat
Edit
stdbuf
Edit
strace
Edit
strace-log-merge
Edit
stream
Edit
streamzip
Edit
strings
Edit
strip
Edit
stty
Edit
su
Edit
sudo
Edit
sudoedit
Edit
sudoreplay
Edit
sum
Edit
sw-engine
Edit
sxpm
Edit
sync
Edit
syslinux
Edit
systemctl
Edit
systemd-analyze
Edit
systemd-ask-password
Edit
systemd-cat
Edit
systemd-cgls
Edit
systemd-cgtop
Edit
systemd-creds
Edit
systemd-cryptenroll
Edit
systemd-delta
Edit
systemd-detect-virt
Edit
systemd-dissect
Edit
systemd-escape
Edit
systemd-firstboot
Edit
systemd-hwdb
Edit
systemd-id128
Edit
systemd-inhibit
Edit
systemd-machine-id-setup
Edit
systemd-mount
Edit
systemd-notify
Edit
systemd-path
Edit
systemd-repart
Edit
systemd-run
Edit
systemd-socket-activate
Edit
systemd-stdio-bridge
Edit
systemd-sysext
Edit
systemd-sysusers
Edit
systemd-tmpfiles
Edit
systemd-tty-ask-password-agent
Edit
systemd-umount
Edit
tabs
Edit
tac
Edit
tail
Edit
tapestat
Edit
tar
Edit
taskset
Edit
tbl
Edit
tclsh
Edit
tclsh8.6
Edit
tcptraceroute
Edit
tdspool
Edit
team2bond
Edit
teamd
Edit
teamdctl
Edit
teamnl
Edit
tee
Edit
telnet
Edit
test
Edit
text2pdf
Edit
tgz
Edit
tic
Edit
timed-read
Edit
timed-run
Edit
timedatectl
Edit
timeout
Edit
tload
Edit
tmon
Edit
tmpwatch
Edit
toe
Edit
top
Edit
touch
Edit
tput
Edit
tr
Edit
tracepath
Edit
traceroute
Edit
traceroute6
Edit
tracker3
Edit
tred
Edit
troff
Edit
true
Edit
truncate
Edit
trust
Edit
tset
Edit
tsort
Edit
tsql
Edit
tty
Edit
turbostat
Edit
twopi
Edit
type
Edit
tzselect
Edit
uapi
Edit
udevadm
Edit
ul
Edit
ulimit
Edit
ulockmgr_server
Edit
umask
Edit
uml_switchd
Edit
umount
Edit
unalias
Edit
uname
Edit
uname26
Edit
unbuffer
Edit
unexpand
Edit
unflatten
Edit
unicode_start
Edit
unicode_stop
Edit
uniq
Edit
unlink
Edit
unpack200
Edit
unpigz
Edit
unshare
Edit
unxz
Edit
unzip
Edit
unzipsfx
Edit
update-ca-trust
Edit
update-crypto-policies
Edit
update-gtk-immodules
Edit
update-mime-database
Edit
update_cpanelv2
Edit
upower
Edit
uptime
Edit
userdel.cagefs
Edit
users
Edit
usleep
Edit
utmpdump
Edit
uuidgen
Edit
uuidparse
Edit
uz
Edit
vdir
Edit
vi
Edit
view
Edit
vimdot
Edit
vlock
Edit
vmstat
Edit
w
Edit
wait
Edit
wall
Edit
watch
Edit
watchgnupg
Edit
wc
Edit
wdctl
Edit
weather
Edit
wget
Edit
whatis
Edit
whatis.man-db
Edit
whereis
Edit
which
Edit
whiptail
Edit
who
Edit
whoami
Edit
wireplumber
Edit
withsctp
Edit
wmf2eps
Edit
wmf2fig
Edit
wmf2gd
Edit
wmf2svg
Edit
wmf2x
Edit
word-list-compress
Edit
wpctl
Edit
wpexec
Edit
write
Edit
wsrep_sst_backup
Edit
wsrep_sst_common
Edit
wsrep_sst_mariabackup
Edit
wsrep_sst_mysqldump
Edit
wsrep_sst_rsync
Edit
wsrep_sst_rsync_wan
Edit
x265
Edit
x86_64
Edit
x86_64-redhat-linux-c++
Edit
x86_64-redhat-linux-g++
Edit
x86_64-redhat-linux-gcc
Edit
x86_64-redhat-linux-gcc-11
Edit
x86_64-redhat-linux-gnu-pkg-config
Edit
x86_energy_perf_policy
Edit
xargs
Edit
xdg-dbus-proxy
Edit
xgettext
Edit
xkibitz
Edit
xml2-config
Edit
xmlcatalog
Edit
xmllint
Edit
xmlwf
Edit
xslt-config
Edit
xsltproc
Edit
xsubpp
Edit
xz
Edit
xzcat
Edit
xzcmp
Edit
xzdec
Edit
xzdiff
Edit
xzegrep
Edit
xzfgrep
Edit
xzgrep
Edit
xzless
Edit
xzmore
Edit
yat2m
Edit
yes
Edit
ypdomainname
Edit
yum
Edit
yum-builddep
Edit
yum-config-manager
Edit
yum-debug-dump
Edit
yum-debug-restore
Edit
yum-groups-manager
Edit
yumdownloader
Edit
zcat
Edit
zcmp
Edit
zdiff
Edit
zdump
Edit
zegrep
Edit
zfgrep
Edit
zforce
Edit
zgrep
Edit
zip
Edit
zipcloak
Edit
zipdetails
Edit
zipgrep
Edit
zipinfo
Edit
zipnote
Edit
zipsplit
Edit
zless
Edit
zmore
Edit
znew
Edit
zone2json
Edit
zone2sql
Edit
zsoelim
Edit