001/* 002 * Licensed to the Apache Software Foundation (ASF) under one 003 * or more contributor license agreements. See the NOTICE file 004 * distributed with this work for additional information 005 * regarding copyright ownership. The ASF licenses this file 006 * to you under the Apache License, Version 2.0 (the 007 * "License"); you may not use this file except in compliance 008 * with the License. You may obtain a copy of the License at 009 * 010 * https://www.apache.org/licenses/LICENSE-2.0 011 * 012 * Unless required by applicable law or agreed to in writing, 013 * software distributed under the License is distributed on an 014 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 015 * KIND, either express or implied. See the License for the 016 * specific language governing permissions and limitations 017 * under the License. 018 */ 019package org.apache.commons.compress.archivers.zip; 020 021import java.nio.file.attribute.FileTime; 022import java.util.Date; 023import java.util.Objects; 024import java.util.zip.ZipException; 025 026import org.apache.commons.io.file.attribute.FileTimes; 027 028/** 029 * NTFS extra field that was thought to store various attributes but in reality only stores timestamps. 030 * 031 * <pre> 032 * 4.5.5 -NTFS Extra Field (0x000a): 033 * 034 * The following is the layout of the NTFS attributes 035 * "extra" block. (Note: At this time the Mtime, Atime 036 * and Ctime values MAY be used on any WIN32 system.) 037 * 038 * Note: all fields stored in Intel low-byte/high-byte order. 039 * 040 * Value Size Description 041 * ----- ---- ----------- 042 * (NTFS) 0x000a 2 bytes Tag for this "extra" block type 043 * TSize 2 bytes Size of the total "extra" block 044 * Reserved 4 bytes Reserved for future use 045 * Tag1 2 bytes NTFS attribute tag value #1 046 * Size1 2 bytes Size of attribute #1, in bytes 047 * (var) Size1 Attribute #1 data 048 * . 049 * . 050 * . 051 * TagN 2 bytes NTFS attribute tag value #N 052 * SizeN 2 bytes Size of attribute #N, in bytes 053 * (var) SizeN Attribute #N data 054 * 055 * For NTFS, values for Tag1 through TagN are as follows: 056 * (currently only one set of attributes is defined for NTFS) 057 * 058 * Tag Size Description 059 * ----- ---- ----------- 060 * 0x0001 2 bytes Tag for attribute #1 061 * Size1 2 bytes Size of attribute #1, in bytes 062 * Mtime 8 bytes File last modification time 063 * Atime 8 bytes File last access time 064 * Ctime 8 bytes File creation time 065 * </pre> 066 * 067 * @since 1.11 068 * @NotThreadSafe 069 */ 070public class X000A_NTFS implements ZipExtraField { 071 072 /** 073 * The header ID for this extra field. 074 * 075 * @since 1.23 076 */ 077 public static final ZipShort HEADER_ID = new ZipShort(0x000a); 078 079 private static final ZipShort TIME_ATTR_TAG = new ZipShort(0x0001); 080 private static final ZipShort TIME_ATTR_SIZE = new ZipShort(3 * 8); 081 082 private static ZipEightByteInteger dateToZip(final Date d) { 083 if (d == null) { 084 return null; 085 } 086 return new ZipEightByteInteger(FileTimes.toNtfsTime(d)); 087 } 088 089 private static ZipEightByteInteger fileTimeToZip(final FileTime time) { 090 if (time == null) { 091 return null; 092 } 093 return new ZipEightByteInteger(FileTimes.toNtfsTime(time)); 094 } 095 096 private static Date zipToDate(final ZipEightByteInteger z) { 097 if (z == null || ZipEightByteInteger.ZERO.equals(z)) { 098 return null; 099 } 100 return FileTimes.ntfsTimeToDate(z.getLongValue()); 101 } 102 103 private static FileTime zipToFileTime(final ZipEightByteInteger z) { 104 if (z == null || ZipEightByteInteger.ZERO.equals(z)) { 105 return null; 106 } 107 return FileTimes.ntfsTimeToFileTime(z.getLongValue()); 108 } 109 110 private ZipEightByteInteger modifyTime = ZipEightByteInteger.ZERO; 111 112 private ZipEightByteInteger accessTime = ZipEightByteInteger.ZERO; 113 114 private ZipEightByteInteger createTime = ZipEightByteInteger.ZERO; 115 116 @Override 117 public boolean equals(final Object o) { 118 if (o instanceof X000A_NTFS) { 119 final X000A_NTFS xf = (X000A_NTFS) o; 120 121 return Objects.equals(modifyTime, xf.modifyTime) && Objects.equals(accessTime, xf.accessTime) && Objects.equals(createTime, xf.createTime); 122 } 123 return false; 124 } 125 126 /** 127 * Gets the access time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 128 * 129 * @return access time as a {@link FileTime} or null. 130 * @since 1.23 131 */ 132 public FileTime getAccessFileTime() { 133 return zipToFileTime(accessTime); 134 } 135 136 /** 137 * Gets the access time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 138 * 139 * @return access time as java.util.Date or null. 140 */ 141 public Date getAccessJavaTime() { 142 return zipToDate(accessTime); 143 } 144 145 /** 146 * Gets the "File last access time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists in 147 * the ZIP entry. 148 * 149 * @return File last access time 150 */ 151 public ZipEightByteInteger getAccessTime() { 152 return accessTime; 153 } 154 155 /** 156 * Gets the actual data to put into central directory data - without Header-ID or length specifier. 157 * 158 * @return the central directory data 159 */ 160 @Override 161 public byte[] getCentralDirectoryData() { 162 return getLocalFileDataData(); 163 } 164 165 /** 166 * Gets the length of the extra field in the local file data - without Header-ID or length specifier. 167 * 168 * <p> 169 * For X5455 the central length is often smaller than the local length, because central cannot contain access or create timestamps. 170 * </p> 171 * 172 * @return a {@code ZipShort} for the length of the data of this extra field 173 */ 174 @Override 175 public ZipShort getCentralDirectoryLength() { 176 return getLocalFileDataLength(); 177 } 178 179 /** 180 * Gets the create time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 181 * 182 * @return create time as a {@link FileTime} or null. 183 * @since 1.23 184 */ 185 public FileTime getCreateFileTime() { 186 return zipToFileTime(createTime); 187 } 188 189 /** 190 * Gets the create time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 191 * 192 * @return create time as java.util.Date or null. 193 */ 194 public Date getCreateJavaTime() { 195 return zipToDate(createTime); 196 } 197 198 /** 199 * Gets the "File creation time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists in the 200 * ZIP entry. 201 * 202 * @return File creation time 203 */ 204 public ZipEightByteInteger getCreateTime() { 205 return createTime; 206 } 207 208 /** 209 * Gets the Header-ID. 210 * 211 * @return the value for the header id for this extrafield 212 */ 213 @Override 214 public ZipShort getHeaderId() { 215 return HEADER_ID; 216 } 217 218 /** 219 * Gets the actual data to put into local file data - without Header-ID or length specifier. 220 * 221 * @return get the data 222 */ 223 @Override 224 public byte[] getLocalFileDataData() { 225 final byte[] data = new byte[getLocalFileDataLength().getValue()]; 226 int pos = 4; 227 System.arraycopy(TIME_ATTR_TAG.getBytes(), 0, data, pos, 2); 228 pos += 2; 229 System.arraycopy(TIME_ATTR_SIZE.getBytes(), 0, data, pos, 2); 230 pos += 2; 231 System.arraycopy(modifyTime.getBytes(), 0, data, pos, 8); 232 pos += 8; 233 System.arraycopy(accessTime.getBytes(), 0, data, pos, 8); 234 pos += 8; 235 System.arraycopy(createTime.getBytes(), 0, data, pos, 8); 236 return data; 237 } 238 239 /** 240 * Gets the length of the extra field in the local file data - without Header-ID or length specifier. 241 * 242 * @return a {@code ZipShort} for the length of the data of this extra field 243 */ 244 @Override 245 public ZipShort getLocalFileDataLength() { 246 return new ZipShort(4 /* reserved */ 247 + 2 /* Tag#1 */ 248 + 2 /* Size#1 */ 249 + 3 * 8 /* time values */); 250 } 251 252 /** 253 * Gets the modify time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 254 * 255 * @return modify time as a {@link FileTime} or null. 256 * @since 1.23 257 */ 258 public FileTime getModifyFileTime() { 259 return zipToFileTime(modifyTime); 260 } 261 262 /** 263 * Gets the modify time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. 264 * 265 * @return modify time as java.util.Date or null. 266 */ 267 public Date getModifyJavaTime() { 268 return zipToDate(modifyTime); 269 } 270 271 /** 272 * Gets the "File last modification time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists 273 * in the ZIP entry. 274 * 275 * @return File last modification time 276 */ 277 public ZipEightByteInteger getModifyTime() { 278 return modifyTime; 279 } 280 281 @Override 282 public int hashCode() { 283 int hc = -123; 284 if (modifyTime != null) { 285 hc ^= modifyTime.hashCode(); 286 } 287 if (accessTime != null) { 288 // Since accessTime is often same as modifyTime, 289 // this prevents them from XOR negating each other. 290 hc ^= Integer.rotateLeft(accessTime.hashCode(), 11); 291 } 292 if (createTime != null) { 293 hc ^= Integer.rotateLeft(createTime.hashCode(), 22); 294 } 295 return hc; 296 } 297 298 /** 299 * Doesn't do anything special since this class always uses the same parsing logic for both central directory and local file data. 300 */ 301 @Override 302 public void parseFromCentralDirectoryData(final byte[] buffer, final int offset, final int length) throws ZipException { 303 reset(); 304 parseFromLocalFileData(buffer, offset, length); 305 } 306 307 /** 308 * Populate data from this array as if it was in local file data. 309 * 310 * @param data an array of bytes 311 * @param offset the start offset 312 * @param length the number of bytes in the array from offset 313 * @throws java.util.zip.ZipException on error 314 */ 315 @Override 316 public void parseFromLocalFileData(final byte[] data, int offset, final int length) throws ZipException { 317 final int len = offset + length; 318 319 // skip reserved 320 offset += 4; 321 322 while (offset + 4 <= len) { 323 final ZipShort tag = new ZipShort(data, offset); 324 offset += 2; 325 if (tag.equals(TIME_ATTR_TAG)) { 326 readTimeAttr(data, offset, len - offset); 327 break; 328 } 329 final ZipShort size = new ZipShort(data, offset); 330 offset += 2 + size.getValue(); 331 } 332 } 333 334 private void readTimeAttr(final byte[] data, int offset, final int length) { 335 if (length >= 2 + 3 * 8) { 336 final ZipShort tagValueLength = new ZipShort(data, offset); 337 if (TIME_ATTR_SIZE.equals(tagValueLength)) { 338 offset += 2; 339 modifyTime = new ZipEightByteInteger(data, offset); 340 offset += 8; 341 accessTime = new ZipEightByteInteger(data, offset); 342 offset += 8; 343 createTime = new ZipEightByteInteger(data, offset); 344 } 345 } 346 } 347 348 /** 349 * Reset state back to newly constructed state. Helps us make sure parse() calls always generate clean results. 350 */ 351 private void reset() { 352 this.modifyTime = ZipEightByteInteger.ZERO; 353 this.accessTime = ZipEightByteInteger.ZERO; 354 this.createTime = ZipEightByteInteger.ZERO; 355 } 356 357 /** 358 * Sets the access time. 359 * 360 * @param time access time as a {@link FileTime} 361 * @since 1.23 362 */ 363 public void setAccessFileTime(final FileTime time) { 364 setAccessTime(fileTimeToZip(time)); 365 } 366 367 /** 368 * Sets the access time as a java.util.Date of this ZIP entry. 369 * 370 * @param d access time as java.util.Date 371 */ 372 public void setAccessJavaTime(final Date d) { 373 setAccessTime(dateToZip(d)); 374 } 375 376 /** 377 * Sets the File last access time of this ZIP entry using a ZipEightByteInteger object. 378 * 379 * @param t ZipEightByteInteger of the access time 380 */ 381 public void setAccessTime(final ZipEightByteInteger t) { 382 accessTime = t == null ? ZipEightByteInteger.ZERO : t; 383 } 384 385 /** 386 * Sets the create time. 387 * 388 * @param time create time as a {@link FileTime} 389 * @since 1.23 390 */ 391 public void setCreateFileTime(final FileTime time) { 392 setCreateTime(fileTimeToZip(time)); 393 } 394 395 /** 396 * <p> 397 * Sets the create time as a java.util.Date of this ZIP entry. Supplied value is truncated to per-second precision (milliseconds zeroed-out). 398 * </p> 399 * <p> 400 * Note: the setters for flags and timestamps are decoupled. Even if the timestamp is not-null, it will only be written out if the corresponding bit in the 401 * flags is also set. 402 * </p> 403 * 404 * @param d create time as java.util.Date 405 */ 406 public void setCreateJavaTime(final Date d) { 407 setCreateTime(dateToZip(d)); 408 } 409 410 /** 411 * Sets the File creation time of this ZIP entry using a ZipEightByteInteger object. 412 * 413 * @param t ZipEightByteInteger of the create time 414 */ 415 public void setCreateTime(final ZipEightByteInteger t) { 416 createTime = t == null ? ZipEightByteInteger.ZERO : t; 417 } 418 419 /** 420 * Sets the modify time. 421 * 422 * @param time modify time as a {@link FileTime} 423 * @since 1.23 424 */ 425 public void setModifyFileTime(final FileTime time) { 426 setModifyTime(fileTimeToZip(time)); 427 } 428 429 /** 430 * Sets the modify time as a java.util.Date of this ZIP entry. 431 * 432 * @param d modify time as java.util.Date 433 */ 434 public void setModifyJavaTime(final Date d) { 435 setModifyTime(dateToZip(d)); 436 } 437 438 /** 439 * Sets the File last modification time of this ZIP entry using a ZipEightByteInteger object. 440 * 441 * @param t ZipEightByteInteger of the modify time 442 */ 443 public void setModifyTime(final ZipEightByteInteger t) { 444 modifyTime = t == null ? ZipEightByteInteger.ZERO : t; 445 } 446 447 /** 448 * Returns a String representation of this class useful for debugging purposes. 449 * 450 * @return A String representation of this class useful for debugging purposes. 451 */ 452 @Override 453 public String toString() { 454 // @formatter:off 455 return new StringBuilder() 456 .append("0x000A Zip Extra Field:") 457 .append(" Modify:[") 458 .append(getModifyFileTime()) 459 .append("] ") 460 .append(" Access:[") 461 .append(getAccessFileTime()) 462 .append("] ") 463 .append(" Create:[") 464 .append(getCreateFileTime()) 465 .append("] ") 466 .toString(); 467 // @formatter:on 468 } 469}