package org.jruby.ext.socket;
import org.jruby.Ruby;
import org.jruby.RubyArray;
import org.jruby.RubyClass;
import org.jruby.RubySymbol;
import org.jruby.anno.JRubyClass;
import org.jruby.anno.JRubyMethod;
import org.jruby.runtime.Arity;
import org.jruby.runtime.ObjectAllocator;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import org.jruby.util.TypeConverter;
import org.jruby.util.io.BadDescriptorException;
import org.jruby.util.io.Sockaddr;
import java.net.InetSocketAddress;
@JRubyClass(name="IPSocket", parent="BasicSocket")
public class RubyIPSocket extends RubyBasicSocket {
static void createIPSocket(Ruby runtime) {
RubyClass rb_cIPSocket = runtime.defineClass("IPSocket", runtime.getClass("BasicSocket"), IPSOCKET_ALLOCATOR);
rb_cIPSocket.defineAnnotatedMethods(RubyIPSocket.class);
rb_cIPSocket.undefineMethod("initialize");
runtime.getObject().setConstant("IPsocket",rb_cIPSocket);
}
private static ObjectAllocator IPSOCKET_ALLOCATOR = new ObjectAllocator() {
public IRubyObject allocate(Ruby runtime, RubyClass klass) {
return new RubyIPSocket(runtime, klass);
}
};
public RubyIPSocket(Ruby runtime, RubyClass type) {
super(runtime, type);
}
@JRubyMethod(name = "addr")
public IRubyObject addr(ThreadContext context) {
return addrCommon(context, !context.getRuntime().isDoNotReverseLookupEnabled());
}
@JRubyMethod(name = "addr")
public IRubyObject addr(ThreadContext context, IRubyObject reverse) {
return addrCommon(context, reverse);
}
@JRubyMethod(name = "peeraddr")
public IRubyObject peeraddr(ThreadContext context) {
return peeraddrCommon(context, !context.getRuntime().isDoNotReverseLookupEnabled());
}
@JRubyMethod(name = "peeraddr")
public IRubyObject peeraddr(ThreadContext context, IRubyObject reverse) {
return peeraddrCommon(context, reverse);
}
@JRubyMethod(meta = true)
public static IRubyObject getaddress(ThreadContext context, IRubyObject self, IRubyObject hostname) {
return SocketUtils.getaddress(context, hostname);
}
@JRubyMethod
public IRubyObject recvfrom(ThreadContext context, IRubyObject _length) {
Ruby runtime = context.runtime;
IRubyObject result = recv(context, _length);
InetSocketAddress sender = getInetRemoteSocket();
int port;
String hostName;
String hostAddress;
if (sender == null) {
port = 0;
hostName = hostAddress = "0.0.0.0";
} else {
port = sender.getPort();
hostName = sender.getHostName();
hostAddress = sender.getAddress().getHostAddress();
}
IRubyObject addressArray = context.runtime.newArrayNoCopy(
runtime.newString("AF_INET"),
runtime.newFixnum(port),
runtime.newString(hostName),
runtime.newString(hostAddress)
);
return runtime.newArray(result, addressArray);
}
@JRubyMethod
public IRubyObject recvfrom(ThreadContext context, IRubyObject _length, IRubyObject _flags) {
return recvfrom(context, _length);
}
@JRubyMethod(name = "getpeereid", notImplemented = true)
public IRubyObject getpeereid(ThreadContext context) {
throw context.runtime.newNotImplementedError("getpeereid not implemented");
}
@Override
protected IRubyObject getSocknameCommon(ThreadContext context, String caller) {
InetSocketAddress sock = getInetSocketAddress();
return Sockaddr.packSockaddrFromAddress(context, sock);
}
@Override
public IRubyObject getpeername(ThreadContext context) {
InetSocketAddress sock = getInetRemoteSocket();
return Sockaddr.packSockaddrFromAddress(context, sock);
}
private IRubyObject addrCommon(ThreadContext context, IRubyObject reverse) {
Boolean doReverse = doReverseLookup(context, reverse);
if (doReverse == null) doReverse = false;
return addrCommon(context, doReverse);
}
private IRubyObject addrCommon(ThreadContext context, boolean reverse) {
InetSocketAddress address = getInetSocketAddress();
checkAddress(context, address);
return addrFor(context, address, reverse);
}
private void checkAddress(ThreadContext context, InetSocketAddress address) {
if (address == null) {
throw context.runtime.newErrnoENOTSOCKError("Not socket or not connected");
}
}
private IRubyObject peeraddrCommon(ThreadContext context, IRubyObject reverse) {
Boolean doReverse = doReverseLookup(context, reverse);
if (doReverse == null) doReverse = !context.runtime.isDoNotReverseLookupEnabled();
return peeraddrCommon(context, doReverse);
}
private IRubyObject peeraddrCommon(ThreadContext context, boolean reverse) {
InetSocketAddress address = getInetRemoteSocket();
checkAddress(context, address);
return addrFor(context, address, reverse);
}
public static Boolean doReverseLookup(ThreadContext context, IRubyObject noreverse) {
if (noreverse == context.tru) {
return false;
} else if (noreverse == context.fals) {
return true;
} else if (noreverse == context.nil) {
return null;
} else {
Ruby runtime = context.runtime;
TypeConverter.checkType(context, noreverse, runtime.getSymbol());
switch (noreverse.toString()) {
case "numeric": return true;
case "hostname": return false;
default: throw runtime.newArgumentError("invalid reverse_lookup flag: " + noreverse);
}
}
}
@Deprecated
public IRubyObject addr() {
return addr(getRuntime().getCurrentContext());
}
@Deprecated
public IRubyObject peeraddr() {
return peeraddr(getRuntime().getCurrentContext());
}
@Deprecated
public static IRubyObject getaddress(IRubyObject recv, IRubyObject hostname) {
return getaddress(recv.getRuntime().getCurrentContext(), recv, hostname);
}
@Deprecated
public IRubyObject recvfrom(ThreadContext context, IRubyObject[] args) {
switch (args.length) {
case 1:
return recvfrom(context, args[0]);
case 2:
return recvfrom(context, args[0], args[1]);
default:
Arity.raiseArgumentError(context.runtime, args, 1, 2);
return null;
}
}
}