mirror of
https://github.com/bisq-network/bisq.git
synced 2025-02-24 15:10:44 +01:00
Implement AsyncFileChannelWriter
This commit is contained in:
parent
97cee35e39
commit
6707af5f7c
2 changed files with 152 additions and 0 deletions
|
@ -0,0 +1,55 @@
|
|||
/*
|
||||
* This file is part of Bisq.
|
||||
*
|
||||
* Bisq is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU Affero General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* Bisq is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
||||
* License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package bisq.persistence;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.channels.AsynchronousFileChannel;
|
||||
import java.nio.channels.CompletionHandler;
|
||||
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
|
||||
public class AsyncFileChannelWriter implements AsyncFileWriter {
|
||||
private final AsynchronousFileChannel fileChannel;
|
||||
|
||||
public AsyncFileChannelWriter(AsynchronousFileChannel fileChannel) {
|
||||
this.fileChannel = fileChannel;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CompletableFuture<Integer> write(byte[] data, int offset) {
|
||||
var byteBuffer = ByteBuffer.wrap(data);
|
||||
var completableFuture = new CompletableFuture<Integer>();
|
||||
var completionHandler = createCompletionHandler(completableFuture);
|
||||
fileChannel.write(byteBuffer, offset, null, completionHandler);
|
||||
return completableFuture;
|
||||
}
|
||||
|
||||
private CompletionHandler<Integer, Object> createCompletionHandler(CompletableFuture<Integer> completableFuture) {
|
||||
return new CompletionHandler<>() {
|
||||
@Override
|
||||
public void completed(Integer writtenData, Object o) {
|
||||
completableFuture.complete(writtenData);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void failed(Throwable throwable, Object o) {
|
||||
completableFuture.completeExceptionally(throwable);
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
|
@ -0,0 +1,97 @@
|
|||
/*
|
||||
* This file is part of Bisq.
|
||||
*
|
||||
* Bisq is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU Affero General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* Bisq is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
||||
* License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package bisq.persistence;
|
||||
|
||||
import java.nio.channels.AsynchronousFileChannel;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.Path;
|
||||
import java.nio.file.StandardOpenOption;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.Random;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.TimeoutException;
|
||||
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.junit.jupiter.api.io.TempDir;
|
||||
|
||||
import static org.hamcrest.CoreMatchers.is;
|
||||
import static org.hamcrest.MatcherAssert.assertThat;
|
||||
|
||||
public class AsyncFileChannelWriterTests {
|
||||
private Path filePath;
|
||||
private AsynchronousFileChannel fileChannel;
|
||||
private AsyncFileChannelWriter asyncFileChannelWriter;
|
||||
|
||||
@BeforeEach
|
||||
void setup(@TempDir Path tempDir) throws IOException {
|
||||
filePath = tempDir.resolve("file");
|
||||
fileChannel = AsynchronousFileChannel.open(filePath, StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE);
|
||||
asyncFileChannelWriter = new AsyncFileChannelWriter(fileChannel);
|
||||
}
|
||||
|
||||
@Test
|
||||
void writeData() throws IOException, InterruptedException, ExecutionException, TimeoutException {
|
||||
byte[] expected = new byte[1050];
|
||||
new Random().nextBytes(expected);
|
||||
|
||||
CompletableFuture<Integer> completableFuture = asyncFileChannelWriter.write(expected, 0);
|
||||
|
||||
int writtenBytes = completableFuture.get(30, TimeUnit.SECONDS);
|
||||
while (writtenBytes < expected.length) {
|
||||
completableFuture = asyncFileChannelWriter.write(expected, writtenBytes);
|
||||
writtenBytes += completableFuture.get(30, TimeUnit.SECONDS);
|
||||
}
|
||||
|
||||
assertThat(writtenBytes, is(expected.length));
|
||||
|
||||
fileChannel.close();
|
||||
byte[] actual = Files.readAllBytes(filePath);
|
||||
|
||||
assertThat(expected, is(actual));
|
||||
}
|
||||
|
||||
@Test
|
||||
void writeDataAtOffset() throws IOException, InterruptedException, ExecutionException, TimeoutException {
|
||||
final int startOffset = 100;
|
||||
byte[] data = new byte[1024];
|
||||
new Random().nextBytes(data);
|
||||
|
||||
CompletableFuture<Integer> completableFuture = asyncFileChannelWriter.write(data, startOffset);
|
||||
|
||||
int writtenBytes = startOffset + completableFuture.get(30, TimeUnit.SECONDS);
|
||||
while (writtenBytes < data.length) {
|
||||
completableFuture = asyncFileChannelWriter.write(data, writtenBytes);
|
||||
writtenBytes += completableFuture.get(30, TimeUnit.SECONDS);
|
||||
}
|
||||
|
||||
assertThat(writtenBytes - startOffset, is(data.length));
|
||||
|
||||
fileChannel.close();
|
||||
byte[] readFromFile = Files.readAllBytes(filePath);
|
||||
assertThat(readFromFile.length, is(startOffset + data.length));
|
||||
|
||||
byte[] readFromFileWithoutOffset = Arrays.copyOfRange(readFromFile, 100, readFromFile.length);
|
||||
assertThat(readFromFileWithoutOffset, is(data));
|
||||
}
|
||||
}
|
Loading…
Add table
Reference in a new issue