-
Notifications
You must be signed in to change notification settings - Fork 40.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Ignore system timezone when applying outputTimestamp to entries
Update `JarWriter` so that entry times are set with the default TimeZone offset removed. The Javadoc for `ZipEntry.setTime` states: The file entry is "encoded in standard `MS-DOS date and time format`. The default TimeZone is used to convert the epoch time to the MS-DOS data and time. Removing the offset from our UTC time before calling `entry.setTime()` ensures that we get consistent bytes in the zip file when the output stream reapplies the offset during write. Fixes gh-34424
- Loading branch information
Showing
6 changed files
with
149 additions
and
7 deletions.
There are no files selected for viewing
58 changes: 58 additions & 0 deletions
58
...ader-tools/src/main/java/org/springframework/boot/loader/tools/DefaultTimeZoneOffset.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
/* | ||
* Copyright 2012-2023 the original author or authors. | ||
* | ||
* Licensed 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.springframework.boot.loader.tools; | ||
|
||
import java.nio.file.attribute.FileTime; | ||
import java.util.TimeZone; | ||
import java.util.zip.ZipEntry; | ||
|
||
/** | ||
* Utility class that can be used change a UTC time based on the | ||
* {@link java.util.TimeZone#getDefault() default TimeZone}. This is required because | ||
* {@link ZipEntry#setTime(long)} expects times in the default timezone and not UTC. | ||
* | ||
* @author Phillip Webb | ||
*/ | ||
class DefaultTimeZoneOffset { | ||
|
||
static final DefaultTimeZoneOffset INSTANCE = new DefaultTimeZoneOffset(TimeZone.getDefault()); | ||
|
||
private final TimeZone defaultTimeZone; | ||
|
||
DefaultTimeZoneOffset(TimeZone defaultTimeZone) { | ||
this.defaultTimeZone = defaultTimeZone; | ||
} | ||
|
||
/** | ||
* Remove the default offset from the given time. | ||
* @param time the time to remove the default offset from | ||
* @return the time with the default offset removed | ||
*/ | ||
FileTime removeFrom(FileTime time) { | ||
return FileTime.fromMillis(removeFrom(time.toMillis())); | ||
} | ||
|
||
/** | ||
* Remove the default offset from the given time. | ||
* @param time the time to remove the default offset from | ||
* @return the time with the default offset removed | ||
*/ | ||
long removeFrom(long time) { | ||
return time - this.defaultTimeZone.getOffset(time); | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
77 changes: 77 additions & 0 deletions
77
...tools/src/test/java/org/springframework/boot/loader/tools/DefaultTimeZoneOffsetTests.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,77 @@ | ||
/* | ||
* Copyright 2012-2023 the original author or authors. | ||
* | ||
* Licensed 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.springframework.boot.loader.tools; | ||
|
||
import java.time.OffsetDateTime; | ||
import java.time.ZoneOffset; | ||
import java.util.Calendar; | ||
import java.util.TimeZone; | ||
|
||
import org.apache.commons.compress.archivers.zip.ZipArchiveOutputStream; | ||
import org.junit.jupiter.api.Test; | ||
|
||
import static org.assertj.core.api.Assertions.assertThat; | ||
|
||
/** | ||
* Tests for {@link DefaultTimeZoneOffset} | ||
* | ||
* @author Phillip Webb | ||
*/ | ||
class DefaultTimeZoneOffsetTests { | ||
|
||
// gh-34424 | ||
|
||
@Test | ||
void removeFromWithLongInDifferentTimeZonesReturnsSameValue() { | ||
long time = OffsetDateTime.of(2000, 1, 1, 0, 0, 0, 0, ZoneOffset.UTC).toInstant().toEpochMilli(); | ||
TimeZone timeZone1 = TimeZone.getTimeZone("GMT"); | ||
TimeZone timeZone2 = TimeZone.getTimeZone("GMT+8"); | ||
TimeZone timeZone3 = TimeZone.getTimeZone("GMT-8"); | ||
long result1 = new DefaultTimeZoneOffset(timeZone1).removeFrom(time); | ||
long result2 = new DefaultTimeZoneOffset(timeZone2).removeFrom(time); | ||
long result3 = new DefaultTimeZoneOffset(timeZone3).removeFrom(time); | ||
long dosTime1 = toDosTime(Calendar.getInstance(timeZone1), result1); | ||
long dosTime2 = toDosTime(Calendar.getInstance(timeZone2), result2); | ||
long dosTime3 = toDosTime(Calendar.getInstance(timeZone3), result3); | ||
assertThat(dosTime1).isEqualTo(dosTime2).isEqualTo(dosTime3); | ||
} | ||
|
||
@Test | ||
void removeFromWithFileTimeReturnsFileTime() { | ||
long time = OffsetDateTime.of(2000, 1, 1, 0, 0, 0, 0, ZoneOffset.UTC).toInstant().toEpochMilli(); | ||
long result = new DefaultTimeZoneOffset(TimeZone.getTimeZone("GMT+8")).removeFrom(time); | ||
assertThat(result).isNotEqualTo(time).isEqualTo(946656000000L); | ||
} | ||
|
||
/** | ||
* Identical functionality to package-private | ||
* org.apache.commons.compress.archivers.zip.ZipUtil.toDosTime(Calendar, long, byte[], | ||
* int) method used by {@link ZipArchiveOutputStream} to convert times. | ||
* @param calendar the source calendar | ||
* @param time the time to convert | ||
* @return the DOS time | ||
*/ | ||
private long toDosTime(Calendar calendar, long time) { | ||
calendar.setTimeInMillis(time); | ||
final int year = calendar.get(Calendar.YEAR); | ||
final int month = calendar.get(Calendar.MONTH) + 1; | ||
return ((year - 1980) << 25) | (month << 21) | (calendar.get(Calendar.DAY_OF_MONTH) << 16) | ||
| (calendar.get(Calendar.HOUR_OF_DAY) << 11) | (calendar.get(Calendar.MINUTE) << 5) | ||
| (calendar.get(Calendar.SECOND) >> 1); | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters