org.restcomm.connect.sms.smpp.SmppClientOpsThread Maven / Gradle / Ivy
The newest version!
/*
* TeleStax, Open Source Cloud Communications
* Copyright 2011-2013, Telestax Inc and individual contributors
* by the @authors tag.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.restcomm.connect.sms.smpp;
import java.io.IOException;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import javax.servlet.ServletException;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.LogManager;
import org.restcomm.connect.sms.smpp.dlr.provider.TelestaxDlrParser;
import org.restcomm.connect.sms.smpp.dlr.spi.DLRPayload;
import org.restcomm.connect.sms.smpp.dlr.spi.DlrParser;
import com.cloudhopper.commons.charset.Charset;
import com.cloudhopper.commons.charset.CharsetUtil;
import com.cloudhopper.smpp.PduAsyncResponse;
import com.cloudhopper.smpp.SmppConstants;
import com.cloudhopper.smpp.SmppSession;
import com.cloudhopper.smpp.SmppSessionConfiguration;
import com.cloudhopper.smpp.SmppSessionHandler;
import com.cloudhopper.smpp.impl.DefaultSmppClient;
import com.cloudhopper.smpp.impl.DefaultSmppSession;
import com.cloudhopper.smpp.pdu.DeliverSm;
import com.cloudhopper.smpp.pdu.EnquireLink;
import com.cloudhopper.smpp.pdu.EnquireLinkResp;
import com.cloudhopper.smpp.pdu.PduRequest;
import com.cloudhopper.smpp.pdu.PduResponse;
import com.cloudhopper.smpp.tlv.Tlv;
import com.cloudhopper.smpp.type.Address;
import com.cloudhopper.smpp.type.RecoverablePduException;
import com.cloudhopper.smpp.type.UnrecoverablePduException;
import akka.actor.ActorRef;
/**
* @author amit bhayani
* @author [email protected]
*/
public class SmppClientOpsThread implements Runnable {
private static final Logger logger = LogManager.getLogger(SmppClientOpsThread.class);
private static final long SCHEDULE_CONNECT_DELAY = 1000 * 30; // 30 sec
private List pendingChanges = new CopyOnWriteArrayList();
private Object waitObject = new Object();
private final DefaultSmppClient clientBootstrap;
private static SmppSession getSmppSession;
//FIXME: like getSmppSession, this is bad design
//this assumes singular SMPP connection all the time,so it works
private static Charset outboundEncoding;
private static Charset inboundEncoding;
private static boolean messagePayloadFlag;
private static boolean autoDetectDcsFlag;
protected volatile boolean started = true;
private static int sipPort;
private final ActorRef smppMessageHandler;
public SmppClientOpsThread(DefaultSmppClient clientBootstrap, int sipPort, final ActorRef smppMessageHandler) {
this.clientBootstrap = clientBootstrap;
this.sipPort = sipPort;
this.smppMessageHandler = smppMessageHandler;
}
protected void setStarted(boolean started) {
this.started = started;
synchronized (this.waitObject) {
this.waitObject.notify();
}
}
protected void scheduleConnect(Smpp esme) {
synchronized (this.pendingChanges) {
this.pendingChanges.add(new ChangeRequest(esme,
ChangeRequest.CONNECT, System.currentTimeMillis()
+ SCHEDULE_CONNECT_DELAY));
}
synchronized (this.waitObject) {
this.waitObject.notify();
}
}
protected void scheduleEnquireLink(Smpp esme) {
synchronized (this.pendingChanges) {
this.pendingChanges.add(new ChangeRequest(esme,
ChangeRequest.ENQUIRE_LINK, System.currentTimeMillis()
+ esme.getEnquireLinkDelay()));
}
synchronized (this.waitObject) {
this.waitObject.notify();
}
}
@Override
public void run() {
if (logger.isInfoEnabled()) {
logger.info("SmppClientOpsThread started.");
}
while (this.started) {
try {
synchronized (this.pendingChanges) {
Iterator changes = pendingChanges.iterator();
while (changes.hasNext()) {
ChangeRequest change = changes.next();
switch (change.getType()) {
case ChangeRequest.CONNECT:
if (!change.getSmpp().isStarted()) {
pendingChanges.remove(change);
// changes.remove();
} else {
if (change.getExecutionTime() <= System
.currentTimeMillis()) {
pendingChanges.remove(change);
// changes.remove();
initiateConnection(change.getSmpp());
}
}
break;
case ChangeRequest.ENQUIRE_LINK:
if (!change.getSmpp().isStarted()) {
pendingChanges.remove(change);
// changes.remove();
} else {
if (change.getExecutionTime() <= System
.currentTimeMillis()) {
pendingChanges.remove(change);
// changes.remove();
enquireLink(change.getSmpp());
}
}
break;
}
}
}
synchronized (this.waitObject) {
this.waitObject.wait(5000);
}
} catch (InterruptedException e) {
logger.error("Error while looping SmppClientOpsThread thread",
e);
}
}
if (logger.isInfoEnabled()) {
logger.info("SmppClientOpsThread for stopped.");
}
}
private void enquireLink(Smpp esme) {
SmppSession smppSession = esme.getSmppSession();
if (!esme.isStarted()) {
return;
}
if (smppSession != null && smppSession.isBound()) {
try {
EnquireLinkResp enquireLinkResp1 = smppSession.enquireLink(
new EnquireLink(), 10000);
// all ok lets scehdule another ENQUIRE_LINK
this.scheduleEnquireLink(esme);
return;
} catch (RecoverablePduException e) {
logger.warn(
String.format(
"RecoverablePduException while sending the ENQURE_LINK for ESME SystemId=%s",
esme.getSystemId()), e);
// Recoverabel exception is ok
// all ok lets schedule another ENQUIRE_LINK
this.scheduleEnquireLink(esme);
return;
} catch (Exception e) {
logger.error(
String.format(
"Exception while trying to send ENQUIRE_LINK for ESME SystemId=%s",
esme.getSystemId()), e);
// For all other exceptions lets close session and re-try
// connect
smppSession.close();
this.scheduleConnect(esme);
}
} else {
// This should never happen
logger.warn(String
.format("Sending ENQURE_LINK fialed for ESME SystemId=%s as SmppSession is =%s !",
esme.getSystemId(), (smppSession == null ? null
: smppSession.getStateName())));
if (smppSession != null) {
smppSession.close();
}
this.scheduleConnect(esme);
}
}
private void initiateConnection(Smpp esme) {
// If Esme is stopped, don't try to initiate connect
if (!esme.isStarted()) {
return;
}
SmppSession smppSession = esme.getSmppSession();
if ((smppSession != null && smppSession.isBound())
|| (smppSession != null && smppSession.isBinding())) {
// If process has already begun lets not do it again
return;
}
SmppSession session0 = null;
try {
SmppSessionConfiguration config0 = new SmppSessionConfiguration();
config0.setWindowSize(esme.getWindowSize());
config0.setName(esme.getSystemId());
config0.setType(esme.getSmppBindType());
config0.setSystemType(esme.getSystemType());
config0.setHost(esme.getPeerIp());
config0.setPort(esme.getPeerPort());
config0.setConnectTimeout(esme.getConnectTimeout());
config0.setSystemId(esme.getSystemId());
config0.setPassword(esme.getPassword());
//this will disable Enquire Link heartbeat logs printed to the console
//TODO this can be put into the restcomm.xml for Admin use
config0.getLoggingOptions().setLogBytes(false);
config0.getLoggingOptions().setLogPdu(false);
// to enable monitoring (request expiration)
config0.setRequestExpiryTimeout(esme.getRequestExpiryTimeout());
config0.setWindowMonitorInterval(esme.getWindowMonitorInterval());
config0.setCountersEnabled(esme.isCountersEnabled());
Address address = esme.getAddress();
config0.setAddressRange(address);
SmppSessionHandler sessionHandler = new ClientSmppSessionHandler(
esme);
session0 = clientBootstrap.bind(config0, sessionHandler);
//getting session to be used to process SMS received from Restcomm
getSmppSession = session0;
// Set in ESME
esme.setSmppSession((DefaultSmppSession) session0);
// Finally set Enquire Link schedule
this.scheduleEnquireLink(esme);
} catch (Exception e) {
logger.error(
String.format(
"Exception when trying to bind client SMPP connection for ESME systemId=%s",
esme.getSystemId()), e);
if (session0 != null) {
session0.close();
}
this.scheduleConnect(esme);
}
}
//*************Inner Class******************
protected class ClientSmppSessionHandler implements SmppSessionHandler {
//private final Smpp esme ;
private Smpp esme = null;
/**
* @param esme
*/
public ClientSmppSessionHandler(Smpp esme) {
super();
this.esme = esme;
inboundEncoding = esme.getInboundDefaultEncoding();
outboundEncoding = esme.getOutboundDefaultEncoding();
messagePayloadFlag = esme.getMessagePayloadFlag();
autoDetectDcsFlag = esme.getAutoDetectDcsFlag();
}
@Override
public String lookupResultMessage(int arg0) {
// TODO Auto-generated method stub
return null;
}
@Override
public String lookupTlvTagName(short arg0) {
// TODO Auto-generated method stub
return null;
}
@Override
public void fireChannelUnexpectedlyClosed() {
logger.error("ChannelUnexpectedlyClosed for Smpp "
+ this.esme.getName()
+ " Closing Smpp session and restrting BIND process again");
this.esme.getSmppSession().close();
// Schedule the connection again
scheduleConnect(this.esme);
}
@Override
public void fireExpectedPduResponseReceived (
PduAsyncResponse pduAsyncResponse) {
// TODO : SMPP Response received. Does RestComm need confirmation
// for this?
logger.info("ExpectedPduResponseReceived received for Smpp "
+ this.esme.getName() + " PduAsyncResponse="
+ pduAsyncResponse);
//forward this to smppMessageHandle so we can potentially update the message status
smppMessageHandler.tell(pduAsyncResponse, null);
}
@Override
public void firePduRequestExpired(PduRequest pduRequest) {
// TODO : SMPP request Expired. RestComm needs to notify Application
// about SMS failure
logger.warn("PduRequestExpired for Smpp " + this.esme.getName()
+ " PduRequest=" + pduRequest);
}
@Override
public PduResponse firePduRequestReceived(PduRequest pduRequest) {
PduResponse response = pduRequest.createResponse();
if (!pduRequest.toString().toLowerCase().contains("enquire_link")) {
try {
DeliverSm deliverSm = (DeliverSm) pduRequest;
byte dcs = deliverSm.getDataCoding();
String destSmppAddress = deliverSm.getDestAddress().getAddress();
String sourceSmppAddress = deliverSm.getSourceAddress().getAddress();
//this default esme encoding only applies to DCS==0
Charset encoding = esme.getInboundDefaultEncoding();
if(dcs == SmppConstants.DATA_CODING_UCS2) {
encoding = CharsetUtil.CHARSET_UCS_2;
}
byte[] pduMessage = deliverSm.getShortMessage();
int smsLength = deliverSm.getShortMessageLength();
byte esmClass = deliverSm.getEsmClass();
byte msgType = (byte)(((esmClass) >> 2) & 0x0f);
if(logger.isDebugEnabled()) {
logger.debug("deliverSm="+deliverSm.toString()+" smsLength="+smsLength+" esmClass="+Byte.toString(esmClass)+" msgType="+Byte.toString(msgType));
}
//check message_payload
if(smsLength > 0) {
if(logger.isInfoEnabled()) {
logger.info("Using message from message body " + Arrays.toString(pduMessage));
}
} else {
Tlv msgPayload = deliverSm.getOptionalParameter(SmppConstants.TAG_MESSAGE_PAYLOAD);
if(msgPayload!=null) {
pduMessage = msgPayload.getValue();
if(logger.isInfoEnabled()) {
logger.info("Using message from TAG_MESSAGE_PAYLOAD " + Arrays.toString(pduMessage));
}
} else {
//TODO: what else do we do?
logger.error("incoming message has no message body nor message_payload");
}
}
String decodedPduMessage = CharsetUtil.decode(pduMessage, encoding);
//send received SMPP PDU message to restcomm only if not DLR
if (msgType == 0x0) {
try {
sendSmppMessageToRestcomm(decodedPduMessage, destSmppAddress, sourceSmppAddress, encoding);
} catch (IOException | ServletException e) {
logger.error("Exception while trying to dispatch incoming SMPP message to Restcomm: " + e);
}
} else {
if(logger.isInfoEnabled()) {
logger.info("Message is not normal request, esmClass:" + esmClass +", Using message from message body " + decodedPduMessage);
}
DlrParser dlrParser = new TelestaxDlrParser();
DLRPayload dlrPayload = dlrParser.parseMessage(decodedPduMessage);
smppMessageHandler.tell(dlrPayload, null);
}
} catch (Exception e) {
logger.error("Exception while trying to process incoming SMPP message to Restcomm: " + e);
}
}
return response;
}
@Override
public void fireRecoverablePduException(RecoverablePduException e) {
logger.warn("RecoverablePduException received for Smpp "
+ this.esme.getName(), e);
}
@Override
public void fireUnexpectedPduResponseReceived(PduResponse pduResponse) {
logger.warn("UnexpectedPduResponseReceived received for Smpp "
+ this.esme.getName() + " PduResponse=" + pduResponse);
}
@Override
public void fireUnknownThrowable(Throwable e) {
logger.error("UnknownThrowable for Smpp " + this.esme.getName()
+ " Closing Smpp session and restarting BIND process again",
e);
// TODO is this ok?
this.esme.getSmppSession().close();
// Schedule the connection again
scheduleConnect(this.esme);
}
@Override
public void fireUnrecoverablePduException(UnrecoverablePduException e) {
logger.error(
"UnrecoverablePduException for Smpp "
+ this.esme.getName()
+ " Closing Smpp session and restrting BIND process again",
e);
this.esme.getSmppSession().close();
// Schedule the connection again
scheduleConnect(this.esme);
}
}
//smpp session to be used for sending SMS from Restcomm to smpp endpoint
public static SmppSession getSmppSession() {
return getSmppSession;
}
public void sendSmppMessageToRestcomm(String smppMessage, String smppTo, String smppFrom, Charset charset) throws IOException, ServletException {
SmppSession smppSession = SmppClientOpsThread.getSmppSession();
String to = smppTo;
String from = smppFrom;
String inboundMessage = smppMessage;
SmppInboundMessageEntity smppInboundMessage = new SmppInboundMessageEntity(to, from, inboundMessage, charset);
smppMessageHandler.tell(smppInboundMessage, null);
}
public static Charset getInboundDefaultEncoding() {
return inboundEncoding;
}
public static Charset getOutboundDefaultEncoding() {
return outboundEncoding;
}
public static boolean getMessagePayloadFlag() {
return messagePayloadFlag;
}
public static boolean getAutoDetectDcsFlag() {
return autoDetectDcsFlag;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy