| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * https://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| package org.apache.commons.compress.archivers.zip; |
| |
| import java.nio.file.attribute.FileTime; |
| import java.util.Date; |
| import java.util.Objects; |
| import java.util.zip.ZipException; |
| |
| import org.apache.commons.io.file.attribute.FileTimes; |
| |
| /** |
| * NTFS extra field that was thought to store various attributes but in reality only stores timestamps. |
| * |
| * <pre> |
| * 4.5.5 -NTFS Extra Field (0x000a): |
| * |
| * The following is the layout of the NTFS attributes |
| * "extra" block. (Note: At this time the Mtime, Atime |
| * and Ctime values MAY be used on any WIN32 system.) |
| * |
| * Note: all fields stored in Intel low-byte/high-byte order. |
| * |
| * Value Size Description |
| * ----- ---- ----------- |
| * (NTFS) 0x000a 2 bytes Tag for this "extra" block type |
| * TSize 2 bytes Size of the total "extra" block |
| * Reserved 4 bytes Reserved for future use |
| * Tag1 2 bytes NTFS attribute tag value #1 |
| * Size1 2 bytes Size of attribute #1, in bytes |
| * (var) Size1 Attribute #1 data |
| * . |
| * . |
| * . |
| * TagN 2 bytes NTFS attribute tag value #N |
| * SizeN 2 bytes Size of attribute #N, in bytes |
| * (var) SizeN Attribute #N data |
| * |
| * For NTFS, values for Tag1 through TagN are as follows: |
| * (currently only one set of attributes is defined for NTFS) |
| * |
| * Tag Size Description |
| * ----- ---- ----------- |
| * 0x0001 2 bytes Tag for attribute #1 |
| * Size1 2 bytes Size of attribute #1, in bytes |
| * Mtime 8 bytes File last modification time |
| * Atime 8 bytes File last access time |
| * Ctime 8 bytes File creation time |
| * </pre> |
| * |
| * @since 1.11 |
| * @NotThreadSafe |
| */ |
| public class X000A_NTFS implements ZipExtraField { |
| |
| /** |
| * The header ID for this extra field. |
| * |
| * @since 1.23 |
| */ |
| public static final ZipShort HEADER_ID = new ZipShort(0x000a); |
| |
| private static final ZipShort TIME_ATTR_TAG = new ZipShort(0x0001); |
| private static final ZipShort TIME_ATTR_SIZE = new ZipShort(3 * 8); |
| |
| private static ZipEightByteInteger dateToZip(final Date d) { |
| if (d == null) { |
| return null; |
| } |
| return new ZipEightByteInteger(FileTimes.toNtfsTime(d)); |
| } |
| |
| private static ZipEightByteInteger fileTimeToZip(final FileTime time) { |
| if (time == null) { |
| return null; |
| } |
| return new ZipEightByteInteger(FileTimes.toNtfsTime(time)); |
| } |
| |
| private static Date zipToDate(final ZipEightByteInteger z) { |
| if (z == null || ZipEightByteInteger.ZERO.equals(z)) { |
| return null; |
| } |
| return FileTimes.ntfsTimeToDate(z.getLongValue()); |
| } |
| |
| private static FileTime zipToFileTime(final ZipEightByteInteger z) { |
| if (z == null || ZipEightByteInteger.ZERO.equals(z)) { |
| return null; |
| } |
| return FileTimes.ntfsTimeToFileTime(z.getLongValue()); |
| } |
| |
| private ZipEightByteInteger modifyTime = ZipEightByteInteger.ZERO; |
| |
| private ZipEightByteInteger accessTime = ZipEightByteInteger.ZERO; |
| |
| private ZipEightByteInteger createTime = ZipEightByteInteger.ZERO; |
| |
| /** |
| * Constructs a new instance. |
| */ |
| public X000A_NTFS() { |
| // Default constructor |
| } |
| |
| @Override |
| public boolean equals(final Object o) { |
| if (o instanceof X000A_NTFS) { |
| final X000A_NTFS xf = (X000A_NTFS) o; |
| |
| return Objects.equals(modifyTime, xf.modifyTime) && Objects.equals(accessTime, xf.accessTime) && Objects.equals(createTime, xf.createTime); |
| } |
| return false; |
| } |
| |
| /** |
| * Gets the access time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return access time as a {@link FileTime} or null. |
| * @since 1.23 |
| */ |
| public FileTime getAccessFileTime() { |
| return zipToFileTime(accessTime); |
| } |
| |
| /** |
| * Gets the access time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return access time as java.util.Date or null. |
| */ |
| public Date getAccessJavaTime() { |
| return zipToDate(accessTime); |
| } |
| |
| /** |
| * Gets the "File last access time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists in |
| * the ZIP entry. |
| * |
| * @return File last access time. |
| */ |
| public ZipEightByteInteger getAccessTime() { |
| return accessTime; |
| } |
| |
| /** |
| * Gets the actual data to put into central directory data - without Header-ID or length specifier. |
| * |
| * @return the central directory data. |
| */ |
| @Override |
| public byte[] getCentralDirectoryData() { |
| return getLocalFileDataData(); |
| } |
| |
| /** |
| * Gets the length of the extra field in the local file data - without Header-ID or length specifier. |
| * |
| * <p> |
| * For X5455 the central length is often smaller than the local length, because central cannot contain access or create timestamps. |
| * </p> |
| * |
| * @return a {@code ZipShort} for the length of the data of this extra field. |
| */ |
| @Override |
| public ZipShort getCentralDirectoryLength() { |
| return getLocalFileDataLength(); |
| } |
| |
| /** |
| * Gets the create time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return create time as a {@link FileTime} or null. |
| * @since 1.23 |
| */ |
| public FileTime getCreateFileTime() { |
| return zipToFileTime(createTime); |
| } |
| |
| /** |
| * Gets the create time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return create time as java.util.Date or null. |
| */ |
| public Date getCreateJavaTime() { |
| return zipToDate(createTime); |
| } |
| |
| /** |
| * Gets the "File creation time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists in the |
| * ZIP entry. |
| * |
| * @return File creation time. |
| */ |
| public ZipEightByteInteger getCreateTime() { |
| return createTime; |
| } |
| |
| /** |
| * Gets the Header-ID. |
| * |
| * @return the value for the header id for this extrafield. |
| */ |
| @Override |
| public ZipShort getHeaderId() { |
| return HEADER_ID; |
| } |
| |
| /** |
| * Gets the actual data to put into local file data - without Header-ID or length specifier. |
| * |
| * @return get the data. |
| */ |
| @Override |
| public byte[] getLocalFileDataData() { |
| final byte[] data = new byte[getLocalFileDataLength().getValue()]; |
| int pos = 4; |
| System.arraycopy(TIME_ATTR_TAG.getBytes(), 0, data, pos, 2); |
| pos += 2; |
| System.arraycopy(TIME_ATTR_SIZE.getBytes(), 0, data, pos, 2); |
| pos += 2; |
| System.arraycopy(modifyTime.getBytes(), 0, data, pos, 8); |
| pos += 8; |
| System.arraycopy(accessTime.getBytes(), 0, data, pos, 8); |
| pos += 8; |
| System.arraycopy(createTime.getBytes(), 0, data, pos, 8); |
| return data; |
| } |
| |
| /** |
| * Gets the length of the extra field in the local file data - without Header-ID or length specifier. |
| * |
| * @return a {@code ZipShort} for the length of the data of this extra field. |
| */ |
| @Override |
| public ZipShort getLocalFileDataLength() { |
| return new ZipShort(4 /* reserved */ |
| + 2 /* Tag#1 */ |
| + 2 /* Size#1 */ |
| + 3 * 8 /* time values */); |
| } |
| |
| /** |
| * Gets the modify time as a {@link FileTime} of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return modify time as a {@link FileTime} or null. |
| * @since 1.23 |
| */ |
| public FileTime getModifyFileTime() { |
| return zipToFileTime(modifyTime); |
| } |
| |
| /** |
| * Gets the modify time as a java.util.Date of this ZIP entry, or null if no such timestamp exists in the ZIP entry. |
| * |
| * @return modify time as java.util.Date or null. |
| */ |
| public Date getModifyJavaTime() { |
| return zipToDate(modifyTime); |
| } |
| |
| /** |
| * Gets the "File last modification time" of this ZIP entry as a ZipEightByteInteger object, or {@link ZipEightByteInteger#ZERO} if no such timestamp exists |
| * in the ZIP entry. |
| * |
| * @return File last modification time. |
| */ |
| public ZipEightByteInteger getModifyTime() { |
| return modifyTime; |
| } |
| |
| @Override |
| public int hashCode() { |
| int hc = -123; |
| if (modifyTime != null) { |
| hc ^= modifyTime.hashCode(); |
| } |
| if (accessTime != null) { |
| // Since accessTime is often same as modifyTime, |
| // this prevents them from XOR negating each other. |
| hc ^= Integer.rotateLeft(accessTime.hashCode(), 11); |
| } |
| if (createTime != null) { |
| hc ^= Integer.rotateLeft(createTime.hashCode(), 22); |
| } |
| return hc; |
| } |
| |
| /** |
| * Doesn't do anything special since this class always uses the same parsing logic for both central directory and local file data. |
| */ |
| @Override |
| public void parseFromCentralDirectoryData(final byte[] buffer, final int offset, final int length) throws ZipException { |
| reset(); |
| parseFromLocalFileData(buffer, offset, length); |
| } |
| |
| /** |
| * Populate data from this array as if it was in local file data. |
| * |
| * @param data an array of bytes. |
| * @param offset the start offset. |
| * @param length the number of bytes in the array from offset. |
| * @throws java.util.zip.ZipException on error. |
| */ |
| @Override |
| public void parseFromLocalFileData(final byte[] data, int offset, final int length) throws ZipException { |
| final int len = offset + length; |
| |
| // skip reserved |
| offset += 4; |
| |
| while (offset + 4 <= len) { |
| final ZipShort tag = new ZipShort(data, offset); |
| offset += 2; |
| if (tag.equals(TIME_ATTR_TAG)) { |
| readTimeAttr(data, offset, len - offset); |
| break; |
| } |
| final ZipShort size = new ZipShort(data, offset); |
| offset += 2 + size.getValue(); |
| } |
| } |
| |
| private void readTimeAttr(final byte[] data, int offset, final int length) { |
| if (length >= 2 + 3 * 8) { |
| final ZipShort tagValueLength = new ZipShort(data, offset); |
| if (TIME_ATTR_SIZE.equals(tagValueLength)) { |
| offset += 2; |
| modifyTime = new ZipEightByteInteger(data, offset); |
| offset += 8; |
| accessTime = new ZipEightByteInteger(data, offset); |
| offset += 8; |
| createTime = new ZipEightByteInteger(data, offset); |
| } |
| } |
| } |
| |
| /** |
| * Reset state back to newly constructed state. Helps us make sure parse() calls always generate clean results. |
| */ |
| private void reset() { |
| this.modifyTime = ZipEightByteInteger.ZERO; |
| this.accessTime = ZipEightByteInteger.ZERO; |
| this.createTime = ZipEightByteInteger.ZERO; |
| } |
| |
| /** |
| * Sets the access time. |
| * |
| * @param time access time as a {@link FileTime}. |
| * @since 1.23 |
| */ |
| public void setAccessFileTime(final FileTime time) { |
| setAccessTime(fileTimeToZip(time)); |
| } |
| |
| /** |
| * Sets the access time as a java.util.Date of this ZIP entry. |
| * |
| * @param d access time as java.util.Date. |
| */ |
| public void setAccessJavaTime(final Date d) { |
| setAccessTime(dateToZip(d)); |
| } |
| |
| /** |
| * Sets the File last access time of this ZIP entry using a ZipEightByteInteger object. |
| * |
| * @param t ZipEightByteInteger of the access time. |
| */ |
| public void setAccessTime(final ZipEightByteInteger t) { |
| accessTime = t == null ? ZipEightByteInteger.ZERO : t; |
| } |
| |
| /** |
| * Sets the create time. |
| * |
| * @param time create time as a {@link FileTime}. |
| * @since 1.23 |
| */ |
| public void setCreateFileTime(final FileTime time) { |
| setCreateTime(fileTimeToZip(time)); |
| } |
| |
| /** |
| * <p> |
| * Sets the create time as a java.util.Date of this ZIP entry. Supplied value is truncated to per-second precision (milliseconds zeroed-out). |
| * </p> |
| * <p> |
| * 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 |
| * flags is also set. |
| * </p> |
| * |
| * @param d create time as java.util.Date. |
| */ |
| public void setCreateJavaTime(final Date d) { |
| setCreateTime(dateToZip(d)); |
| } |
| |
| /** |
| * Sets the File creation time of this ZIP entry using a ZipEightByteInteger object. |
| * |
| * @param t ZipEightByteInteger of the create time. |
| */ |
| public void setCreateTime(final ZipEightByteInteger t) { |
| createTime = t == null ? ZipEightByteInteger.ZERO : t; |
| } |
| |
| /** |
| * Sets the modify time. |
| * |
| * @param time modify time as a {@link FileTime}. |
| * @since 1.23 |
| */ |
| public void setModifyFileTime(final FileTime time) { |
| setModifyTime(fileTimeToZip(time)); |
| } |
| |
| /** |
| * Sets the modify time as a java.util.Date of this ZIP entry. |
| * |
| * @param d modify time as java.util.Date. |
| */ |
| public void setModifyJavaTime(final Date d) { |
| setModifyTime(dateToZip(d)); |
| } |
| |
| /** |
| * Sets the File last modification time of this ZIP entry using a ZipEightByteInteger object. |
| * |
| * @param t ZipEightByteInteger of the modify time. |
| */ |
| public void setModifyTime(final ZipEightByteInteger t) { |
| modifyTime = t == null ? ZipEightByteInteger.ZERO : t; |
| } |
| |
| /** |
| * Returns a String representation of this class useful for debugging purposes. |
| * |
| * @return A String representation of this class useful for debugging purposes. |
| */ |
| @Override |
| public String toString() { |
| // @formatter:off |
| return new StringBuilder() |
| .append("0x000A Zip Extra Field:") |
| .append(" Modify:[") |
| .append(getModifyFileTime()) |
| .append("] ") |
| .append(" Access:[") |
| .append(getAccessFileTime()) |
| .append("] ") |
| .append(" Create:[") |
| .append(getCreateFileTime()) |
| .append("] ") |
| .toString(); |
| // @formatter:on |
| } |
| } |