@@ -1864,10 +1864,10 @@ public void Add(IStaticDataSource dataSource, ZipEntry entry)
1864
1864
1865
1865
// We don't currently support adding entries with AES encryption, so throw
1866
1866
// up front instead of failing or falling back to ZipCrypto later on
1867
- if ( entry . AESKeySize > 0 )
1868
- {
1869
- throw new NotSupportedException ( "Creation of AES encrypted entries is not supported" ) ;
1870
- }
1867
+ // if (entry.AESKeySize > 0)
1868
+ // {
1869
+ // throw new NotSupportedException("Creation of AES encrypted entries is not supported");
1870
+ // }
1871
1871
1872
1872
CheckSupportedCompressionMethod ( entry . CompressionMethod ) ;
1873
1873
CheckUpdating ( ) ;
@@ -2104,7 +2104,7 @@ private void WriteLocalEntryHeader(ZipUpdate update)
2104
2104
WriteLEShort ( entry . Version ) ;
2105
2105
WriteLEShort ( entry . Flags ) ;
2106
2106
2107
- WriteLEShort ( ( byte ) entry . CompressionMethod ) ;
2107
+ WriteLEShort ( ( byte ) entry . CompressionMethodForHeader ) ;
2108
2108
WriteLEInt ( ( int ) entry . DosTime ) ;
2109
2109
2110
2110
if ( ! entry . HasCrc )
@@ -2158,6 +2158,12 @@ private void WriteLocalEntryHeader(ZipUpdate update)
2158
2158
ed . Delete ( 1 ) ;
2159
2159
}
2160
2160
2161
+ // Write AES Data if needed
2162
+ if ( entry . AESKeySize > 0 )
2163
+ {
2164
+ AddExtraDataAES ( entry , ed ) ;
2165
+ }
2166
+
2161
2167
entry . ExtraData = ed . GetEntryData ( ) ;
2162
2168
2163
2169
WriteLEShort ( name . Length ) ;
@@ -2214,7 +2220,7 @@ private int WriteCentralDirectoryHeader(ZipEntry entry)
2214
2220
2215
2221
unchecked
2216
2222
{
2217
- WriteLEShort ( ( byte ) entry . CompressionMethod ) ;
2223
+ WriteLEShort ( ( byte ) entry . CompressionMethodForHeader ) ;
2218
2224
WriteLEInt ( ( int ) entry . DosTime ) ;
2219
2225
WriteLEInt ( ( int ) entry . Crc ) ;
2220
2226
}
@@ -2281,6 +2287,11 @@ private int WriteCentralDirectoryHeader(ZipEntry entry)
2281
2287
ed . Delete ( 1 ) ;
2282
2288
}
2283
2289
2290
+ if ( entry . AESKeySize > 0 )
2291
+ {
2292
+ AddExtraDataAES ( entry , ed ) ;
2293
+ }
2294
+
2284
2295
byte [ ] centralExtraData = ed . GetEntryData ( ) ;
2285
2296
2286
2297
WriteLEShort ( centralExtraData . Length ) ;
@@ -2335,6 +2346,22 @@ private int WriteCentralDirectoryHeader(ZipEntry entry)
2335
2346
return ZipConstants . CentralHeaderBaseSize + name . Length + centralExtraData . Length + rawComment . Length ;
2336
2347
}
2337
2348
2349
+ private static void AddExtraDataAES ( ZipEntry entry , ZipExtraData extraData )
2350
+ {
2351
+ // Vendor Version: AE-1 IS 1. AE-2 is 2. With AE-2 no CRC is required and 0 is stored.
2352
+ const int VENDOR_VERSION = 2 ;
2353
+ // Vendor ID is the two ASCII characters "AE".
2354
+ const int VENDOR_ID = 0x4541 ; //not 6965;
2355
+ extraData . StartNewEntry ( ) ;
2356
+ // Pack AES extra data field see http://www.winzip.com/aes_info.htm
2357
+ //extraData.AddLeShort(7); // Data size (currently 7)
2358
+ extraData . AddLeShort ( VENDOR_VERSION ) ; // 2 = AE-2
2359
+ extraData . AddLeShort ( VENDOR_ID ) ; // "AE"
2360
+ extraData . AddData ( entry . AESEncryptionStrength ) ; // 1 = 128, 2 = 192, 3 = 256
2361
+ extraData . AddLeShort ( ( int ) entry . CompressionMethod ) ; // The actual compression method used to compress the file
2362
+ extraData . AddNewEntry ( 0x9901 ) ;
2363
+ }
2364
+
2338
2365
#endregion Writing Values/Headers
2339
2366
2340
2367
private void PostUpdateCleanup ( )
@@ -2621,13 +2648,20 @@ private Stream GetOutputStream(ZipEntry entry)
2621
2648
switch ( entry . CompressionMethod )
2622
2649
{
2623
2650
case CompressionMethod . Stored :
2624
- result = new UncompressedStream ( result ) ;
2651
+ if ( ! entry . IsCrypted )
2652
+ {
2653
+ // If there is an encryption stream in use, that can be written to directly
2654
+ // otherwise, wrap it in an UncompressedStream instead of returning the base stream directly
2655
+ result = new UncompressedStream ( result ) ;
2656
+ }
2625
2657
break ;
2626
2658
2627
2659
case CompressionMethod . Deflated :
2628
2660
var dos = new DeflaterOutputStream ( result , new Deflater ( 9 , true ) )
2629
2661
{
2630
- IsStreamOwner = false
2662
+ // If there is an encryption stream in use, then we want that to be disposed when the deflator stream is disposed
2663
+ // If not, then we don't want it to dispose the base stream
2664
+ IsStreamOwner = entry . IsCrypted
2631
2665
} ;
2632
2666
result = dos ;
2633
2667
break ;
@@ -3667,9 +3701,16 @@ private Stream CreateAndInitDecryptionStream(Stream baseStream, ZipEntry entry)
3667
3701
3668
3702
private Stream CreateAndInitEncryptionStream ( Stream baseStream , ZipEntry entry )
3669
3703
{
3670
- CryptoStream result = null ;
3671
- if ( ( entry . Version < ZipConstants . VersionStrongEncryption )
3672
- || ( entry . Flags & ( int ) GeneralBitFlags . StrongEncryption ) == 0 )
3704
+ if ( entry . CompressionMethodForHeader == CompressionMethod . WinZipAES )
3705
+ {
3706
+ int blockSize = entry . AESKeySize / 8 ; // bits to bytes
3707
+
3708
+ var aesStream =
3709
+ new ZipAESEncryptionStream ( baseStream , rawPassword_ , entry . AESSaltLen , blockSize ) ;
3710
+
3711
+ return aesStream ;
3712
+ }
3713
+ else
3673
3714
{
3674
3715
var classicManaged = new PkzipClassicManaged ( ) ;
3675
3716
@@ -3681,7 +3722,7 @@ private Stream CreateAndInitEncryptionStream(Stream baseStream, ZipEntry entry)
3681
3722
3682
3723
// Closing a CryptoStream will close the base stream as well so wrap it in an UncompressedStream
3683
3724
// which doesnt do this.
3684
- result = new CryptoStream ( new UncompressedStream ( baseStream ) ,
3725
+ CryptoStream result = new CryptoStream ( new UncompressedStream ( baseStream ) ,
3685
3726
classicManaged . CreateEncryptor ( key , null ) , CryptoStreamMode . Write ) ;
3686
3727
3687
3728
if ( ( entry . Crc < 0 ) || ( entry . Flags & 8 ) != 0 )
@@ -3692,8 +3733,9 @@ private Stream CreateAndInitEncryptionStream(Stream baseStream, ZipEntry entry)
3692
3733
{
3693
3734
WriteEncryptionHeader ( result , entry . Crc ) ;
3694
3735
}
3736
+
3737
+ return result ;
3695
3738
}
3696
- return result ;
3697
3739
}
3698
3740
3699
3741
private static void CheckClassicPassword ( CryptoStream classicCryptoStream , ZipEntry entry )
0 commit comments