gov.nist.javax.sip.stack.sctp.SCTPMessageChannel Maven / Gradle / Ivy
package gov.nist.javax.sip.stack.sctp;
import gov.nist.core.CommonLogger;
import gov.nist.core.LogWriter;
import gov.nist.core.ServerLogger;
import gov.nist.core.StackLogger;
import gov.nist.javax.sip.header.CSeq;
import gov.nist.javax.sip.header.CallID;
import gov.nist.javax.sip.header.From;
import gov.nist.javax.sip.header.RequestLine;
import gov.nist.javax.sip.header.StatusLine;
import gov.nist.javax.sip.header.To;
import gov.nist.javax.sip.header.Via;
import gov.nist.javax.sip.message.SIPMessage;
import gov.nist.javax.sip.message.SIPRequest;
import gov.nist.javax.sip.message.SIPResponse;
import gov.nist.javax.sip.parser.ParseExceptionListener;
import gov.nist.javax.sip.parser.StringMsgParser;
import gov.nist.javax.sip.stack.MessageChannel;
import gov.nist.javax.sip.stack.SIPClientTransaction;
import gov.nist.javax.sip.stack.SIPServerTransaction;
import gov.nist.javax.sip.stack.SIPTransaction;
import gov.nist.javax.sip.stack.SIPTransactionStack;
import gov.nist.javax.sip.stack.ServerRequestInterface;
import gov.nist.javax.sip.stack.ServerResponseInterface;
import java.io.IOException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.text.ParseException;
import com.sun.nio.sctp.MessageInfo;
import com.sun.nio.sctp.SctpChannel;
/**
* SCTP message channel
*
* @author Jeroen van Bemmel
*/
final class SCTPMessageChannel extends MessageChannel
implements ParseExceptionListener, Comparable {
private static StackLogger logger = CommonLogger.getLogger(SCTPMessageChannel.class);
private final SCTPMessageProcessor processor;
private InetSocketAddress peerAddress; // destination address
private InetSocketAddress peerSrcAddress;
private final SctpChannel channel;
private final SelectionKey key;
private final MessageInfo messageInfo; // outgoing SCTP options, cached
private long rxTime; //< Time first byte of message was received
// XXX Hardcoded, enough? TODO make stack property
private final ByteBuffer rxBuffer = ByteBuffer.allocateDirect( 10000 );
private final StringMsgParser parser = new StringMsgParser(); // Parser instance
// for outgoing connections
SCTPMessageChannel( SCTPMessageProcessor p, InetSocketAddress dest ) throws IOException {
this.processor = p;
this.messageProcessor = p; // super class
this.peerAddress = dest;
this.peerSrcAddress = dest; // assume the same, override upon packet
this.messageInfo = MessageInfo.createOutgoing( dest, 0 );
messageInfo.unordered( true );
this.channel = SctpChannel.open( dest, 1, 1 );
channel.configureBlocking( false );
this.key = processor.registerChannel( this, channel );
}
// For incoming connections
SCTPMessageChannel( SCTPMessageProcessor p, SctpChannel c ) throws IOException {
this.processor = p;
this.messageProcessor = p; // super class
SocketAddress a = c.getRemoteAddresses().iterator().next();
this.peerAddress = (InetSocketAddress) a;
this.peerSrcAddress = (InetSocketAddress) a;
this.messageInfo = MessageInfo.createOutgoing( a, 0 );
messageInfo.unordered( true );
this.channel = c;
channel.configureBlocking( false );
this.key = processor.registerChannel( this, channel );
}
@Override
public void close() {
try {
this.key.cancel();
channel.close();
} catch (IOException e) {
e.printStackTrace();
} finally {
processor.removeChannel( this );
}
}
void closeNoRemove() {
try {
this.key.cancel();
channel.close();
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public String getKey() {
// Note: could put this in super class
return getKey( this.getPeerInetAddress(), this.getPeerPort(), this.getTransport() );
}
@Override
public String getPeerAddress() {
return peerAddress.getHostString();
}
@Override
protected InetAddress getPeerInetAddress() {
return peerAddress.getAddress();
}
@Override
public InetAddress getPeerPacketSourceAddress() {
return peerSrcAddress.getAddress();
}
@Override
public int getPeerPacketSourcePort() {
return peerSrcAddress.getPort();
}
@Override
public int getPeerPort() {
return peerAddress.getPort();
}
@Override
protected String getPeerProtocol() {
return "sctp"; // else something really is weird ;)
}
@Override
public SIPTransactionStack getSIPStack() {
return processor.getSIPStack();
}
@Override
public String getTransport() {
return "sctp";
}
@Override
public String getViaHost() {
return processor.getSavedIpAddress();
}
@Override
public int getViaPort() {
return processor.getPort();
}
@Override
public boolean isReliable() {
return true;
}
@Override
public boolean isSecure() {
return false;
}
@Override
public void sendMessage(SIPMessage sipMessage) throws IOException {
byte[] msg = sipMessage.encodeAsBytes( this.getTransport() );
this.sendMessage( msg, this.getPeerInetAddress(), this.getPeerPort(), false );
}
@Override
protected void sendMessage(byte[] message, InetAddress receiverAddress,
int receiverPort, boolean reconnectFlag) throws IOException {
assert( receiverAddress.equals( peerAddress.getAddress() ) );
assert( receiverPort == peerAddress.getPort() );
// XX ignoring 'reconnect' for now
int nBytes = channel.send( ByteBuffer.wrap(message), messageInfo );
if ( logger.isLoggingEnabled( LogWriter.TRACE_DEBUG ) ) {
logger.logDebug( "SCTP bytes sent:" + nBytes );
}
}
/**
* Called by SCTPMessageProcessor when one or more bytes are available for reading
* @throws IOException
*/
void readMessages() throws IOException {
if (rxTime==0) {
rxTime = System.currentTimeMillis();
}
MessageInfo info = channel.receive( rxBuffer, null, null );
if (info==null) {
// happens a lot, some sort of keep-alive?
if ( logger.isLoggingEnabled( LogWriter.TRACE_DEBUG ) ) {
logger.logDebug( "SCTP read-event but no message" );
}
return;
} else if (info.bytes()==-1) {
logger.logWarning( "SCTP peer closed, closing too..." );
this.close();
return;
} else if ( !info.isComplete() ) {
if ( logger.isLoggingEnabled( LogWriter.TRACE_DEBUG ) ) {
logger.logDebug( "SCTP incomplete message; bytes=" + info.bytes() );
}
return;
} else {
if ( logger.isLoggingEnabled( LogWriter.TRACE_DEBUG ) ) {
logger.logDebug( "SCTP message now complete; bytes=" + info.bytes() );
}
}
// Assume it is 1 full message, not multiple messages
byte[] msg = new byte[ rxBuffer.position() ];
rxBuffer.flip();
rxBuffer.get( msg );
rxBuffer.compact();
try {
SIPMessage m = parser.parseSIPMessage( msg, true, true, this );
this.processMessage( m, rxTime );
rxTime = 0; // reset for next message
} catch (ParseException e) {
if ( logger.isLoggingEnabled( LogWriter.TRACE_DEBUG ) ) {
logger.logDebug( "Invalid message bytes=" + msg.length + ":" + new String(msg), e);
}
this.close();
throw new IOException( "Error parsing incoming SCTP message", e );
}
}
/**
* Actually proces the parsed message.
* @param sipMessage
*
* JvB: copied from UDPMessageChannel, TODO restructure
*/
private void processMessage( SIPMessage sipMessage, long rxTime ) {
SIPTransactionStack sipStack = processor.getSIPStack();
sipMessage.setRemoteAddress(this.peerAddress.getAddress());
sipMessage.setRemotePort(this.getPeerPort());
sipMessage.setLocalPort(this.getPort());
sipMessage.setLocalAddress(this.getMessageProcessor().getIpAddress());
if (sipMessage instanceof SIPRequest) {
SIPRequest sipRequest = (SIPRequest) sipMessage;
// This is a request - process it.
// So far so good -- we will commit this message if
// all processing is OK.
if (logger.isLoggingEnabled(ServerLogger.TRACE_MESSAGES)) {
sipStack.getServerLogger().logMessage(sipMessage, this
.getPeerHostPort().toString(), this.getHost() + ":"
+ this.getPort(), false, rxTime);
}
ServerRequestInterface sipServerRequest = sipStack
.newSIPServerRequest(sipRequest, this);
// Drop it if there is no request returned
if (sipServerRequest == null) {
if (logger.isLoggingEnabled()) {
logger
.logWarning("Null request interface returned -- dropping request");
}
return;
}
if (logger.isLoggingEnabled(LogWriter.TRACE_DEBUG)) {
logger.logDebug("About to process "
+ sipRequest.getFirstLine() + "/" + sipServerRequest);
}
try {
sipServerRequest.processRequest(sipRequest, this);
} finally {
if (sipServerRequest instanceof SIPTransaction) {
SIPServerTransaction sipServerTx = (SIPServerTransaction) sipServerRequest;
if (!sipServerTx.passToListener()) {
((SIPTransaction) sipServerRequest).releaseSem();
}
}
}
if (logger.isLoggingEnabled(LogWriter.TRACE_DEBUG))
logger.logDebug("Done processing "
+ sipRequest.getFirstLine() + "/" + sipServerRequest);
// So far so good -- we will commit this message if
// all processing is OK.
} else {
// Handle a SIP Reply message.
SIPResponse sipResponse = (SIPResponse) sipMessage;
try {
sipResponse.checkHeaders();
} catch (ParseException ex) {
if (logger.isLoggingEnabled())
logger
.logError("Dropping Badly formatted response message >>> "
+ sipResponse);
return;
}
ServerResponseInterface sipServerResponse = sipStack
.newSIPServerResponse(sipResponse, this);
if (sipServerResponse != null) {
try {
if (sipServerResponse instanceof SIPClientTransaction
&& !((SIPClientTransaction) sipServerResponse)
.checkFromTag(sipResponse)) {
if (logger.isLoggingEnabled())
logger
.logError("Dropping response message with invalid tag >>> "
+ sipResponse);
return;
}
sipServerResponse.processResponse(sipResponse, this);
} finally {
if (sipServerResponse instanceof SIPTransaction
&& !((SIPTransaction) sipServerResponse)
.passToListener())
((SIPTransaction) sipServerResponse).releaseSem();
}
// Normal processing of message.
} else {
if (logger.isLoggingEnabled(LogWriter.TRACE_DEBUG)) {
logger.logDebug("null sipServerResponse!");
}
}
}
}
/**
* Implementation of the ParseExceptionListener interface.
*
* @param ex
* Exception that is given to us by the parser.
* @throws ParseException
* If we choose to reject the header or message.
*
* JvB: copied from UDPMessageChannel, TODO restructure!
*/
public void handleException(ParseException ex, SIPMessage sipMessage,
Class hdrClass, String header, String message)
throws ParseException {
if (logger.isLoggingEnabled(LogWriter.TRACE_DEBUG))
logger.logDebug("Parsing Exception: " , ex);
// Log the bad message for later reference.
if ((hdrClass != null)
&& (hdrClass.equals(From.class) || hdrClass.equals(To.class)
|| hdrClass.equals(CSeq.class)
|| hdrClass.equals(Via.class)
|| hdrClass.equals(CallID.class)
|| hdrClass.equals(RequestLine.class) || hdrClass
.equals(StatusLine.class))) {
if (logger.isLoggingEnabled(LogWriter.TRACE_DEBUG))
logger.logError("BAD MESSAGE!" + message);
// JvB: send a 400 response for requests (except ACK)
// Currently only UDP, @todo also other transports
String msgString = sipMessage.toString();
if (!msgString.startsWith("SIP/") && !msgString.startsWith("ACK ")) {
if (channel != null) {
if (logger.isLoggingEnabled(LogWriter.TRACE_ERROR)) {
logger
.logError("Malformed mandatory headers: closing socket! :"
+ channel.toString());
}
try {
channel.close();
} catch (IOException ie) {
if (logger.isLoggingEnabled(LogWriter.TRACE_ERROR)) {
logger.logError("Exception while closing socket! :"
+ channel.toString() + ":" + ie.toString());
}
}
}
}
throw ex;
} else {
sipMessage.addUnparsed(header);
}
}
public int compareTo(SCTPMessageChannel o) {
return this.hashCode() - o.hashCode();
}
@Override
protected void uncache() {
processor.removeChannel( this );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy