hudson.remoting.ProxyInputStream Maven / Gradle / Ivy
/*
* The MIT License
*
* Copyright (c) 2004-2009, Sun Microsystems, Inc., Kohsuke Kawaguchi
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package hudson.remoting;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Serializable;
/**
* {@link InputStream} that reads bits from an exported
* {@link InputStream} on a remote machine.
*
*
* TODO: pre-fetch bytes in advance
*
* @author Kohsuke Kawaguchi
*/
final class ProxyInputStream extends InputStream {
private Channel channel;
private int oid;
/**
* Creates an already connected {@link ProxyOutputStream}.
*
* @param oid
* The object id of the exported {@link OutputStream}.
*/
public ProxyInputStream(Channel channel, int oid) throws IOException {
this.channel = channel;
this.oid = oid;
}
@Override
public int read() throws IOException {
try {
Buffer buf = new Chunk(oid, 1).call(channel);
if(buf.len==1)
// byte->int expansion needs to be done carefully becaue byte in Java is signed
// whose idea was it to make byte signed, anyway!?
return ((int)buf.buf[0])&0xFF;
else
return -1;
} catch (InterruptedException e) {
// pretend EOF
Thread.currentThread().interrupt(); // process interrupt later
close();
return -1;
}
}
@Override
public int read(byte b[], int off, int len) throws IOException {
try {
Buffer buf = new Chunk(oid,len).call(channel);
if(buf.len==-1) return -1;
System.arraycopy(buf.buf,0,b,off,buf.len);
return buf.len;
} catch (InterruptedException e) {
// pretend EOF
Thread.currentThread().interrupt(); // process interrupt later
close();
return -1;
}
}
@Override
public synchronized void close() throws IOException {
if(channel!=null) {
channel.send(new EOF(oid));
channel = null;
oid = -1;
}
}
private static final class Buffer implements Serializable {
byte[] buf;
int len;
public Buffer(int len) {
this.buf = new byte[len];
}
public void read(InputStream in) throws IOException {
len = in.read(buf,0,buf.length);
}
private static final long serialVersionUID = 1L;
}
/**
* Command to fetch bytes.
*/
private static final class Chunk extends Request {
private final int oid;
private final int len;
public Chunk(int oid, int len) {
this.oid = oid;
this.len = len;
}
protected Buffer perform(Channel channel) throws IOException {
InputStream in = (InputStream) channel.getExportedObject(oid);
Buffer buf = new Buffer(len);
buf.read(in);
return buf;
}
private static final long serialVersionUID = 1L;
}
/**
* {@link Command} for sending EOF.
*/
private static final class EOF extends Command {
private final int oid;
public EOF(int oid) {
this.oid = oid;
}
protected void execute(Channel channel) {
InputStream in = (InputStream) channel.getExportedObject(oid);
channel.unexport(oid);
try {
in.close();
} catch (IOException e) {
// ignore errors
}
}
public String toString() {
return "EOF("+oid+")";
}
private static final long serialVersionUID = 1L;
}
}