Skip to content

Commit 90c87a4

Browse files
committed
FileChannels.contentEquals(FileChannel, FileChannel, int) can return
false when comparing a non-blocking channel - Add FileChannels.contentEquals(SeekableByteChannel, SeekableByteChannel, int)
1 parent 53674e9 commit 90c87a4

File tree

6 files changed

+345
-16
lines changed

6 files changed

+345
-16
lines changed

src/changes/changes.xml

+3
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,8 @@ The <action> type attribute can be add,update,fix,remove.
6262
<action dev="ggregory" type="fix" due-to="Gary Gregory">Avoid unnecessary boxing and unboxing of int values in UncheckedBufferedReader.read().</action>
6363
<action dev="ggregory" type="fix" due-to="Gary Gregory">Avoid unnecessary boxing and unboxing of int values in UncheckedFilterInputStream.available() and read().</action>
6464
<action dev="ggregory" type="fix" due-to="Gary Gregory">Avoid unnecessary boxing and unboxing of int values in UncheckedFilterReader.read().</action>
65+
<action dev="ggregory" type="fix" due-to="Gary Gregory">FileChannels.contentEquals(FileChannel, FileChannel, int) can return false when comparing a non-blocking channel.</action>
66+
<action dev="ggregory" type="fix" due-to="Gary Gregory">Deprecate FileChannels.contentEquals(FileChannel, FileChannel, int) in favor of FileChannels.contentEquals(SeekableByteChannel, SeekableByteChannel, int).</action>
6567
<!-- ADD -->
6668
<action dev="ggregory" type="add" issue="IO-860" due-to="Nico Strecker, Gary Gregory">Add ThrottledInputStream.Builder.setMaxBytes(long, ChronoUnit).</action>
6769
<action dev="ggregory" type="add" due-to="Gary Gregory">Add IOIterable.</action>
@@ -77,6 +79,7 @@ The <action> type attribute can be add,update,fix,remove.
7779
<action dev="ggregory" type="add" due-to="Gary Gregory">Add BrokenOutputStream.BrokenOutputStream(Function&lt;String&gt;, Throwable>) and deprecate Supplier&lt;String&gt; constructor.</action>
7880
<action dev="ggregory" type="add" due-to="Gary Gregory">Add IOBooleanSupplier.</action>
7981
<action dev="ggregory" type="add" due-to="Gary Gregory">Add Uncheck.getAsBoolean(IOBooleanSupplier).</action>
82+
<action dev="ggregory" type="fix" due-to="Gary Gregory">Add FileChannels.contentEquals(SeekableByteChannel, SeekableByteChannel, int).</action>
8083
<!-- UPDATE -->
8184
<action dev="ggregory" type="update" due-to="Dependabot, Gary Gregory">Bump commons.bytebuddy.version from 1.15.10 to 1.17.1 #710, #715, #720.</action>
8285
<action dev="ggregory" type="update" due-to="Gary Gregory">Bump commons-codec:commons-codec from 1.17.1 to 1.18.0. #717.</action>

src/main/java/org/apache/commons/io/RandomAccessFiles.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919

2020
import java.io.IOException;
2121
import java.io.RandomAccessFile;
22+
import java.nio.channels.SeekableByteChannel;
2223
import java.util.Objects;
2324

2425
import org.apache.commons.io.channels.FileChannels;
@@ -57,7 +58,7 @@ public static boolean contentEquals(final RandomAccessFile raf1, final RandomAcc
5758
// Dig in and to the work
5859
// We do not close FileChannels because that would close the owning RandomAccessFile.
5960
// Instead, the caller is assumed to manage the given RandomAccessFile objects.
60-
return FileChannels.contentEquals(raf1.getChannel(), raf2.getChannel(), IOUtils.DEFAULT_BUFFER_SIZE);
61+
return FileChannels.contentEquals((SeekableByteChannel) raf1.getChannel(), raf2.getChannel(), IOUtils.DEFAULT_BUFFER_SIZE);
6162
}
6263

6364
private static long length(final RandomAccessFile raf) throws IOException {

src/main/java/org/apache/commons/io/channels/FileChannels.java

+42-7
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
import java.io.IOException;
2121
import java.nio.ByteBuffer;
2222
import java.nio.channels.FileChannel;
23+
import java.nio.channels.SeekableByteChannel;
2324
import java.util.Objects;
2425

2526
import org.apache.commons.io.IOUtils;
@@ -39,8 +40,27 @@ public final class FileChannels {
3940
* @param byteBufferSize The two internal buffer capacities, in bytes.
4041
* @return true if the contents of both RandomAccessFiles are equal, false otherwise.
4142
* @throws IOException if an I/O error occurs.
43+
* @deprecated Use {@link #contentEquals(SeekableByteChannel, SeekableByteChannel, int)}.
4244
*/
45+
@Deprecated
4346
public static boolean contentEquals(final FileChannel channel1, final FileChannel channel2, final int byteBufferSize) throws IOException {
47+
return contentEquals((SeekableByteChannel) channel1, channel2, byteBufferSize);
48+
}
49+
50+
/**
51+
* Tests if two readable byte channel contents are equal starting at their respective current positions.
52+
* <p>
53+
* If a file channel is a non-blocking file channel, it may return 0 bytes read for any given call. In order to avoid waiting forever when trying again, a
54+
* timeout Duration can be specified, which when met, throws an IOException.
55+
* </p>
56+
*
57+
* @param channel1 A readable byte channel.
58+
* @param channel2 Another readable byte channel.
59+
* @param byteBufferSize The two internal buffer capacities, in bytes.
60+
* @return true if the contents of both RandomAccessFiles are equal, false otherwise.
61+
* @throws IOException if an I/O error occurs or the timeout is met.
62+
*/
63+
public static boolean contentEquals(final SeekableByteChannel channel1, final SeekableByteChannel channel2, final int byteBufferSize) throws IOException {
4464
// Short-circuit test
4565
if (Objects.equals(channel1, channel2)) {
4666
return true;
@@ -57,15 +77,30 @@ public static boolean contentEquals(final FileChannel channel1, final FileChanne
5777
// Dig in and do the work
5878
final ByteBuffer byteBuffer1 = ByteBuffer.allocateDirect(byteBufferSize);
5979
final ByteBuffer byteBuffer2 = ByteBuffer.allocateDirect(byteBufferSize);
80+
int numRead1 = 0;
81+
int numRead2 = 0;
82+
boolean read0On1 = false;
83+
boolean read0On2 = false;
6084
while (true) {
61-
final int read1 = channel1.read(byteBuffer1);
62-
final int read2 = channel2.read(byteBuffer2);
63-
byteBuffer1.clear();
64-
byteBuffer2.clear();
65-
if (read1 == IOUtils.EOF && read2 == IOUtils.EOF) {
85+
if (!read0On2) {
86+
numRead1 = channel1.read(byteBuffer1);
87+
byteBuffer1.clear();
88+
read0On1 = numRead1 == 0;
89+
}
90+
if (!read0On1) {
91+
numRead2 = channel2.read(byteBuffer2);
92+
byteBuffer2.clear();
93+
read0On2 = numRead2 == 0;
94+
}
95+
if (numRead1 == IOUtils.EOF && numRead2 == IOUtils.EOF) {
6696
return byteBuffer1.equals(byteBuffer2);
6797
}
68-
if (read1 != read2) {
98+
if (numRead1 == 0 || numRead2 == 0) {
99+
// 0 may be returned from a non-blocking channel.
100+
Thread.yield();
101+
continue;
102+
}
103+
if (numRead1 != numRead2) {
69104
return false;
70105
}
71106
if (!byteBuffer1.equals(byteBuffer2)) {
@@ -74,7 +109,7 @@ public static boolean contentEquals(final FileChannel channel1, final FileChanne
74109
}
75110
}
76111

77-
private static long size(final FileChannel channel) throws IOException {
112+
private static long size(final SeekableByteChannel channel) throws IOException {
78113
return channel != null ? channel.size() : 0;
79114
}
80115

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,138 @@
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one or more
3+
* contributor license agreements. See the NOTICE file distributed with
4+
* this work for additional information regarding copyright ownership.
5+
* The ASF licenses this file to You under the Apache License, Version 2.0
6+
* (the "License"); you may not use this file except in compliance with
7+
* the License. You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
18+
package org.apache.commons.io.channels;
19+
20+
import java.io.IOException;
21+
import java.nio.ByteBuffer;
22+
import java.nio.MappedByteBuffer;
23+
import java.nio.channels.FileChannel;
24+
import java.nio.channels.FileLock;
25+
import java.nio.channels.ReadableByteChannel;
26+
import java.nio.channels.WritableByteChannel;
27+
28+
/**
29+
* Proxies a FileChannel.
30+
*/
31+
class FileChannelProxy extends FileChannel {
32+
33+
FileChannel fileChannel;
34+
35+
FileChannelProxy(final FileChannel fileChannel) {
36+
this.fileChannel = fileChannel;
37+
}
38+
39+
@Override
40+
public boolean equals(final Object o) {
41+
return fileChannel.equals(o);
42+
}
43+
44+
@Override
45+
public void force(final boolean metaData) throws IOException {
46+
fileChannel.force(metaData);
47+
}
48+
49+
@Override
50+
public int hashCode() {
51+
return fileChannel.hashCode();
52+
}
53+
54+
@Override
55+
protected void implCloseChannel() throws IOException {
56+
fileChannel.close();
57+
}
58+
59+
@Override
60+
public FileLock lock(final long position, final long size, final boolean shared) throws IOException {
61+
return fileChannel.lock(position, size, shared);
62+
}
63+
64+
@Override
65+
public MappedByteBuffer map(final MapMode mode, final long position, final long size) throws IOException {
66+
return fileChannel.map(mode, position, size);
67+
}
68+
69+
@Override
70+
public long position() throws IOException {
71+
return fileChannel.position();
72+
}
73+
74+
@Override
75+
public FileChannel position(final long newPosition) throws IOException {
76+
return fileChannel.position(newPosition);
77+
}
78+
79+
@Override
80+
public int read(final ByteBuffer dst) throws IOException {
81+
return fileChannel.read(dst);
82+
}
83+
84+
@Override
85+
public int read(final ByteBuffer dst, final long position) throws IOException {
86+
return fileChannel.read(dst, position);
87+
}
88+
89+
@Override
90+
public long read(final ByteBuffer[] dsts, final int offset, final int length) throws IOException {
91+
return fileChannel.read(dsts, offset, length);
92+
}
93+
94+
@Override
95+
public long size() throws IOException {
96+
return fileChannel.size();
97+
}
98+
99+
@Override
100+
public String toString() {
101+
return fileChannel.toString();
102+
}
103+
104+
@Override
105+
public long transferFrom(final ReadableByteChannel src, final long position, final long count) throws IOException {
106+
return fileChannel.transferFrom(src, position, count);
107+
}
108+
109+
@Override
110+
public long transferTo(final long position, final long count, final WritableByteChannel target) throws IOException {
111+
return fileChannel.transferTo(position, count, target);
112+
}
113+
114+
@Override
115+
public FileChannel truncate(final long size) throws IOException {
116+
return fileChannel.truncate(size);
117+
}
118+
119+
@Override
120+
public FileLock tryLock(final long position, final long size, final boolean shared) throws IOException {
121+
return fileChannel.tryLock(position, size, shared);
122+
}
123+
124+
@Override
125+
public int write(final ByteBuffer src) throws IOException {
126+
return fileChannel.write(src);
127+
}
128+
129+
@Override
130+
public int write(final ByteBuffer src, final long position) throws IOException {
131+
return fileChannel.write(src, position);
132+
}
133+
134+
@Override
135+
public long write(final ByteBuffer[] srcs, final int offset, final int length) throws IOException {
136+
return fileChannel.write(srcs, offset, length);
137+
}
138+
}

0 commit comments

Comments
 (0)