/*
 *  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.util.Date;
import java.util.zip.ZipException;

NTFS extra field that was thought to store various attributes but in reality only stores timestamps.
   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
Since:1.11
@NotThreadSafe
/** * 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 { private 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 ZipEightByteInteger modifyTime = ZipEightByteInteger.ZERO; private ZipEightByteInteger accessTime = ZipEightByteInteger.ZERO; private ZipEightByteInteger createTime = ZipEightByteInteger.ZERO;
The Header-ID.
Returns:the value for the header id for this extrafield
/** * The Header-ID. * * @return the value for the header id for this extrafield */
@Override public ZipShort getHeaderId() { return HEADER_ID; }
Length of the extra field in the local file data - without Header-ID or length specifier.
Returns:a ZipShort for the length of the data of this extra field
/** * Length of the extra field in the local file data - without * Header-ID or length specifier. * * @return a <code>ZipShort</code> 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 */); }
Length of the extra field in the local file data - without Header-ID or length specifier.

For X5455 the central length is often smaller than the local length, because central cannot contain access or create timestamps.

Returns:a ZipShort for the length of the data of this extra field
/** * 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</code> for the length of the data of this extra field */
@Override public ZipShort getCentralDirectoryLength() { return getLocalFileDataLength(); }
The actual data to put into local file data - without Header-ID or length specifier.
Returns:get the data
/** * 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; }
The actual data to put into central directory data - without Header-ID or length specifier.
Returns:the central directory data
/** * 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(); }
Populate data from this array as if it was in local file data.
Params:
  • data – an array of bytes
  • offset – the start offset
  • length – the number of bytes in the array from offset
Throws:
/** * 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(); } }
Doesn't do anything special since this class always uses the same parsing logic for both central directory and local file data.
/** * 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); }
Returns the "File last modification time" of this zip entry as a ZipEightByteInteger object, or ZipEightByteInteger.ZERO if no such timestamp exists in the zip entry.
Returns:File last modification time
/** * Returns 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; }
Returns the "File last access time" of this zip entry as a ZipEightByteInteger object, or ZipEightByteInteger.ZERO if no such timestamp exists in the zip entry.
Returns:File last access time
/** * Returns 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; }
Returns the "File creation time" of this zip entry as a ZipEightByteInteger object, or ZipEightByteInteger.ZERO if no such timestamp exists in the zip entry.
Returns:File creation time
/** * Returns 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; }
Returns the modify time as a java.util.Date of this zip entry, or null if no such timestamp exists in the zip entry.
Returns:modify time as java.util.Date or null.
/** * Returns 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); }
Returns the access time as a java.util.Date of this zip entry, or null if no such timestamp exists in the zip entry.
Returns:access time as java.util.Date or null.
/** * Returns 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); }
Returns the create time as a a java.util.Date of this zip entry, or null if no such timestamp exists in the zip entry.
Returns:create time as java.util.Date or null.
/** * Returns the create time as a 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); }
Sets the File last modification time of this zip entry using a ZipEightByteInteger object.
Params:
  • t – ZipEightByteInteger of the modify time
/** * 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; }
Sets the File last access time of this zip entry using a ZipEightByteInteger object.
Params:
  • t – ZipEightByteInteger of the access time
/** * 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 File creation time of this zip entry using a ZipEightByteInteger object.
Params:
  • t – ZipEightByteInteger of the create time
/** * 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 as a java.util.Date of this zip entry.
Params:
  • d – modify time as java.util.Date
/** * 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 access time as a java.util.Date of this zip entry.
Params:
  • d – access time as java.util.Date
/** * 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 create time as a java.util.Date of this zip entry. Supplied value is truncated to per-second precision (milliseconds zeroed-out).

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.

Params:
  • d – create time as java.util.Date
/** * <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)); }
Returns a String representation of this class useful for debugging purposes.
Returns:A String representation of this class useful for debugging purposes.
/** * 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() { final StringBuilder buf = new StringBuilder(); buf.append("0x000A Zip Extra Field:") .append(" Modify:[").append(getModifyJavaTime()).append("] ") .append(" Access:[").append(getAccessJavaTime()).append("] ") .append(" Create:[").append(getCreateJavaTime()).append("] "); return buf.toString(); } @Override public boolean equals(final Object o) { if (o instanceof X000A_NTFS) { final X000A_NTFS xf = (X000A_NTFS) o; return (modifyTime == xf.modifyTime || (modifyTime != null && modifyTime.equals(xf.modifyTime))) && (accessTime == xf.accessTime || (accessTime != null && accessTime.equals(xf.accessTime))) && (createTime == xf.createTime || (createTime != null && createTime.equals(xf.createTime))); } return false; } @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; }
Reset state back to newly constructed state. Helps us make sure parse() calls always generate clean results.
/** * 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; } 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); } } } // https://msdn.microsoft.com/en-us/library/windows/desktop/ms724290%28v=vs.85%29.aspx // A file time is a 64-bit value that represents the number of // 100-nanosecond intervals that have elapsed since 12:00 // A.M. January 1, 1601 Coordinated Universal Time (UTC). // this is the offset of Windows time 0 to Unix epoch in 100-nanosecond intervals private static final long EPOCH_OFFSET = -116444736000000000L; private static ZipEightByteInteger dateToZip(final Date d) { if (d == null) { return null; } return new ZipEightByteInteger((d.getTime() * 10000L) - EPOCH_OFFSET); } private static Date zipToDate(final ZipEightByteInteger z) { if (z == null || ZipEightByteInteger.ZERO.equals(z)) { return null; } final long l = (z.getLongValue() + EPOCH_OFFSET) / 10000L; return new Date(l); } }