X000A_NTFS.java
/*
* 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
*
* http://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;
@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
}
}