All Downloads are FREE. Search and download functionalities are using the official Maven repository.

gov.nist.javax.sip.stack.sctp.SCTPMessageChannel Maven / Gradle / Ivy

There is a newer version: 1.3.0-91
Show newest version
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 as could not acquire semaphore or the valve dropped the message.");
                }
            }

        }
    }

    /**
     * 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