com.hierynomus.smbj.share.FileInputStream Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of smbj Show documentation
Show all versions of smbj Show documentation
SMB2 protocol library for communication with Windows servers
/*
* Copyright (C)2016 - SMBJ Contributors
*
* 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
*
* 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 com.hierynomus.smbj.share;
import com.hierynomus.mserref.NtStatus;
import com.hierynomus.mssmb2.SMBApiException;
import com.hierynomus.mssmb2.messages.SMB2ReadResponse;
import com.hierynomus.protocol.commons.concurrent.Futures;
import com.hierynomus.protocol.transport.TransportException;
import com.hierynomus.smbj.ProgressListener;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.io.InputStream;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
class FileInputStream extends InputStream {
private final long readTimeout;
private File file;
private long offset = 0;
private int curr = 0;
private byte[] buf;
private ProgressListener progressListener;
private boolean isClosed;
private Future nextResponse;
private static final Logger logger = LoggerFactory.getLogger(FileInputStream.class);
private int bufferSize;
FileInputStream(File file, int bufferSize, long readTimeout, ProgressListener progressListener) {
this.file = file;
this.bufferSize = bufferSize;
this.progressListener = progressListener;
this.readTimeout = readTimeout;
}
@Override
public int read() throws IOException {
if (buf == null || curr >= buf.length) {
loadBuffer();
}
if (isClosed) {
return -1;
}
return buf[curr++] & 0xFF;
}
@Override
public int read(byte b[]) throws IOException {
return read(b, 0, b.length);
}
@Override
public int read(byte b[], int off, int len) throws IOException {
if (buf == null || curr >= buf.length) {
loadBuffer();
}
if (isClosed) {
return -1;
}
int l = buf.length - curr > len ? len : buf.length - curr;
System.arraycopy(buf, curr, b, off, l);
curr += l;
return l;
}
@Override
public void close() {
isClosed = true;
file = null;
buf = null;
}
@Override
public int available() {
return 0;
}
@Override
public long skip(long n) {
if (buf == null) {
offset += n;
} else if (curr + n < buf.length) {
curr += n;
} else {
offset += (curr + n) - buf.length;
buf = null;
nextResponse = null;
}
return n;
}
private void loadBuffer() throws IOException {
if (isClosed) {
return;
}
if (nextResponse == null) {
nextResponse = sendRequest();
}
SMB2ReadResponse res = Futures.get(nextResponse, readTimeout, TimeUnit.MILLISECONDS, TransportException.Wrapper);
if (res.getHeader().getStatusCode() == NtStatus.STATUS_SUCCESS.getValue()) {
buf = res.getData();
curr = 0;
offset += res.getDataLength();
if (progressListener != null) {
progressListener.onProgressChanged(res.getDataLength(), offset);
}
}
// According to MS-SMB2 2.2.20 and 3.3.5.12 the server should terminate the last SMB2 READ Response with STATUS_END_OF_FILE,
// however at least the IBM implementation does not do that and only returns a '0' data length on the response.
// Treat this corner case as an EOF marker to fix unbounded loops.
if (res.getHeader().getStatusCode() == NtStatus.STATUS_END_OF_FILE.getValue() || res.getDataLength() == 0) {
logger.debug("EOF, {} bytes read", offset);
isClosed = true;
return;
}
if (res.getHeader().getStatusCode() != NtStatus.STATUS_SUCCESS.getValue()) {
throw new SMBApiException(res.getHeader(), "Read failed for " + this);
}
nextResponse = sendRequest();
}
private Future sendRequest() {
return file.readAsync(offset, bufferSize);
}
}