com.mobius.software.telco.protocols.diameter.impl.DiameterLinkImpl Maven / Gradle / Ivy
package com.mobius.software.telco.protocols.diameter.impl;
/*
* Mobius Software LTD
* Copyright 2023, Mobius Software LTD and individual contributors
* by the @authors tag.
*
* This program is free software: you can redistribute it and/or modify
* under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation; either version 3 of
* the License, or (at your option) any later version.
*
* This program 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see
*/
import java.net.InetAddress;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.restcomm.protocols.api.Association;
import org.restcomm.protocols.api.AssociationListener;
import org.restcomm.protocols.api.AssociationType;
import org.restcomm.protocols.api.IpChannelType;
import org.restcomm.protocols.api.Management;
import org.restcomm.protocols.api.PayloadData;
import org.restcomm.protocols.api.Server;
import com.mobius.software.telco.protocols.diameter.AsyncCallback;
import com.mobius.software.telco.protocols.diameter.DiameterLink;
import com.mobius.software.telco.protocols.diameter.DiameterStack;
import com.mobius.software.telco.protocols.diameter.NetworkListener;
import com.mobius.software.telco.protocols.diameter.PeerStateEnum;
import com.mobius.software.telco.protocols.diameter.ResultCodes;
import com.mobius.software.telco.protocols.diameter.annotations.DiameterCommandDefinition;
import com.mobius.software.telco.protocols.diameter.commands.DiameterAnswer;
import com.mobius.software.telco.protocols.diameter.commands.DiameterErrorAnswer;
import com.mobius.software.telco.protocols.diameter.commands.DiameterMessage;
import com.mobius.software.telco.protocols.diameter.commands.DiameterRequest;
import com.mobius.software.telco.protocols.diameter.commands.commons.AccountingAnswer;
import com.mobius.software.telco.protocols.diameter.commands.commons.AccountingRequest;
import com.mobius.software.telco.protocols.diameter.commands.commons.CapabilitiesExchangeAnswer;
import com.mobius.software.telco.protocols.diameter.commands.commons.CapabilitiesExchangeRequest;
import com.mobius.software.telco.protocols.diameter.commands.commons.DeviceWatchdogAnswer;
import com.mobius.software.telco.protocols.diameter.commands.commons.DeviceWatchdogRequest;
import com.mobius.software.telco.protocols.diameter.commands.commons.DisconnectPeerAnswer;
import com.mobius.software.telco.protocols.diameter.commands.commons.DisconnectPeerRequest;
import com.mobius.software.telco.protocols.diameter.commands.commons.VendorSpecificAnswer;
import com.mobius.software.telco.protocols.diameter.commands.commons.VendorSpecificRequest;
import com.mobius.software.telco.protocols.diameter.exceptions.AvpNotSupportedException;
import com.mobius.software.telco.protocols.diameter.exceptions.DiameterException;
import com.mobius.software.telco.protocols.diameter.exceptions.MissingAvpException;
import com.mobius.software.telco.protocols.diameter.impl.commands.DiameterErrorAnswerImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.DiameterErrorAnswerWithSessionImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.CapabilitiesExchangeAnswerImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.CapabilitiesExchangeRequestImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.DeviceWatchdogAnswerImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.DeviceWatchdogRequestImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.DisconnectPeerAnswerImpl;
import com.mobius.software.telco.protocols.diameter.impl.commands.common.DisconnectPeerRequestImpl;
import com.mobius.software.telco.protocols.diameter.impl.primitives.common.AcctApplicationIdImpl;
import com.mobius.software.telco.protocols.diameter.parser.DiameterParser;
import com.mobius.software.telco.protocols.diameter.primitives.common.DisconnectCauseEnum;
import com.mobius.software.telco.protocols.diameter.primitives.common.VendorSpecificApplicationId;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
/**
*
* @author yulian oifa
*
*/
public class DiameterLinkImpl implements DiameterLink,AssociationListener
{
public static Logger logger=LogManager.getLogger(DiameterLinkImpl.class);
public static final Integer DIAMETER_SCTP_PROTOCOL_IDENTIFIER=46;
private DiameterParser parser;
private Management management;
private Association association;
private Server server;
private AtomicReference peerState = new AtomicReference(PeerStateEnum.IDLE);
private String linkId;
private String localHost;
private String localRealm;
private String destinationHost;
private String destinationRealm;
private InetAddress remoteAddress;
private InetAddress localAddress;
private AtomicInteger wheel=new AtomicInteger(0);
private Integer maxStreams = 1;
private List applicationIds = new ArrayList();
private List authApplicationIds = new ArrayList();
private List acctApplicationIds = new ArrayList();
private AtomicReference> remoteApplicationIds=new AtomicReference>();
private AtomicReference> remoteAuthApplicationIds=new AtomicReference>();
private AtomicReference> remoteAcctApplicationIds=new AtomicReference>();
private ConcurrentHashMap authApplicationPackages = new ConcurrentHashMap();
private ConcurrentHashMap acctApplicationPackages = new ConcurrentHashMap();
private DiameterStack stack;
private Boolean rejectUnmandatoryAvps;
private Long inactivityTimeout;
private Long reconnectTimeout;
private InactivityTimer inactivityTimer;
private DisconnectTimer disconnectTimer;
private ReconnectTimer reconnectTimer;
private AtomicLong lastActivity=new AtomicLong(System.currentTimeMillis());
private AtomicBoolean waitingForDWA=new AtomicBoolean(false);
private ConcurrentHashMap genericListeners;
public DiameterLinkImpl(DiameterStack stack, Management management, ConcurrentHashMap genericListeners, String linkId, InetAddress remoteAddress, Integer remotePort, InetAddress localAddress, Integer localPort, Boolean isServer, Boolean isSctp, String localHost, String localRealm, String destinationHost, String destinationRealm, Boolean rejectUnmandatoryAvps, Long inactivityTimeout, Long responseTimeout, Long reconnectTimeout) throws DiameterException
{
this.parser = new DiameterParser(stack.getClassLoader(), Arrays.asList(new Class>[] { DiameterErrorAnswerImpl.class , DiameterErrorAnswerWithSessionImpl.class }),Package.getPackage("com.mobius.software.telco.protocols.diameter.impl.primitives"));
this.genericListeners = genericListeners;
this.linkId = linkId;
if(remoteAddress==null)
throw new DiameterException("The remote address can not be null", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
if(localAddress==null)
throw new DiameterException("The local address can not be null", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
IpChannelType channelType=IpChannelType.SCTP;
if(isSctp==null || !isSctp)
channelType = IpChannelType.TCP;
this.management = management;
this.localHost = localHost;
this.localRealm = localRealm;
this.destinationHost = destinationHost;
this.destinationRealm = destinationRealm;
this.localAddress = localAddress;
this.remoteAddress = remoteAddress;
if(isServer!=null && isServer)
{
if(localPort==null || localPort<1)
throw new DiameterException("The local port can not be null and should be positive", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
if(remotePort!=null && remotePort<0)
throw new DiameterException("The remote port should be positive or null", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
for(Server server:management.getServers())
{
if(server.getIpChannelType().equals(channelType) && server.getHostAddress().equals(localAddress.getHostAddress()) && localPort.equals(server.getHostport()))
{
this.server = server;
break;
}
}
StringBuilder serverNameBuilder = new StringBuilder();
serverNameBuilder.append(channelType.name()).append("://").append(localAddress.getHostAddress()).append(":").append(localPort);
String serverName = serverNameBuilder.toString();
if(server == null)
{
try
{
server = management.addServer(serverName, localAddress.getHostAddress(), localPort, channelType, null);
management.startServer(serverName);
}
catch(Exception ex)
{
throw new DiameterException("An error occured while establishing a peer", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
}
}
try
{
this.association = management.addServerAssociation(remoteAddress.getHostAddress(), remotePort, serverName, linkId, channelType);
}
catch(Exception ex)
{
logger.warn("An error occured while adding server association," + ex.getMessage(),ex);
throw new DiameterException("An error occured while establishing a peer", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
}
}
else
{
if(localPort==null || localPort<0)
throw new DiameterException("The local port can not be null and should be zero or positive", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
if(remotePort==null || remotePort<1)
throw new DiameterException("The remote port can not be null and should be positive", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
try
{
association = management.addAssociation(localAddress.getHostAddress(), localPort, remoteAddress.getHostAddress(), remotePort, linkId, channelType, null);
}
catch(Exception ex)
{
throw new DiameterException("An error occured while establishing a peer", null, ResultCodes.DIAMETER_UNKNOWN_PEER, null);
}
}
association.setAssociationListener(this);
//registering common
@SuppressWarnings("unused")
Class> clazz = CapabilitiesExchangeRequestImpl.class;
@SuppressWarnings("unused")
Class> avpClass = AcctApplicationIdImpl.class;
parser.registerApplication(stack.getClassLoader(), Package.getPackage("com.mobius.software.telco.protocols.diameter.impl.commands.common"));
this.stack = stack;
this.rejectUnmandatoryAvps = rejectUnmandatoryAvps;
this.inactivityTimeout = inactivityTimeout;
this.reconnectTimeout = reconnectTimeout;
this.inactivityTimer = new InactivityTimer(this, lastActivity, waitingForDWA, inactivityTimeout, responseTimeout);
this.disconnectTimer = new DisconnectTimer(this);
this.reconnectTimer = new ReconnectTimer(this);
}
@Override
public void stop(Boolean remove) throws DiameterException
{
try
{
if(isStarted())
this.management.stopAssociation(association.getName());
if(remove)
this.management.removeAssociation(association.getName());
}
catch(Exception ex)
{
throw new DiameterException(ex.getMessage(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null);
}
}
@Override
public void start() throws DiameterException
{
try
{
this.management.startAssociation(association.getName());
}
catch(Exception ex)
{
throw new DiameterException(ex.getMessage(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null);
}
}
@Override
public boolean isStarted()
{
return association.isStarted();
}
@Override
public boolean isConnected()
{
return association.isConnected() && getPeerState().equals(PeerStateEnum.OPEN);
}
@Override
public boolean isUp()
{
return association.isUp();
}
@Override
public InetAddress getRemoteAddress()
{
return remoteAddress;
}
@Override
public Integer getRemotePort()
{
return association.getPeerPort();
}
@Override
public InetAddress getLocalAddress()
{
return localAddress;
}
@Override
public Integer getLocalPort()
{
return association.getHostPort();
}
@Override
public Boolean isServer()
{
return association.getAssociationType()!=AssociationType.CLIENT;
}
@Override
public Boolean isSctp()
{
return association.getIpChannelType()==IpChannelType.SCTP;
}
@Override
public String getLocalHost()
{
return localHost;
}
@Override
public String getLocalRealm()
{
return localRealm;
}
@Override
public String getDestinationHost()
{
return destinationHost;
}
@Override
public String getDestinationRealm()
{
return destinationRealm;
}
@Override
public void onCommunicationUp(Association association, int maxInboundStreams, int maxOutboundStreams)
{
this.maxStreams = maxOutboundStreams;
if(!isServer())
sendCER();
}
@Override
public void onCommunicationShutdown(Association association)
{
logger.info("Association ," + association + " shutdown");
setPeerState(PeerStateEnum.IDLE);
inactivityTimer.stop();
}
@Override
public void onCommunicationLost(Association association)
{
logger.info("Association ," + association + " communication lost");
setPeerState(PeerStateEnum.IDLE);
inactivityTimer.stop();
}
@Override
public void onCommunicationRestart(Association association)
{
logger.info("Association ," + association + " communication restart");
setPeerState(PeerStateEnum.IDLE);
inactivityTimer.stop();
}
@Override
public void inValidStreamId(PayloadData payloadData)
{
logger.info("Association ," + association + " invalid stream id");
}
@Override
public Boolean canSendMessage(DiameterMessage message)
{
if(logger.isDebugEnabled())
logger.debug("Checking if link " + getID() + " can send message " + message.getClass().getName());
if(!getPeerState().equals(PeerStateEnum.OPEN))
return false;
DiameterCommandDefinition commandDefintion = DiameterParser.getCommandDefinition(message.getClass());
if(commandDefintion == null)
return false;
if(message instanceof AccountingRequest || message instanceof AccountingAnswer)
{
if(logger.isDebugEnabled())
logger.debug("It is accounting message " + message.getClass().getName());
Boolean found=false;
List remoteIds = remoteAcctApplicationIds.get();
if(remoteIds!=null)
{
for(Long currApplicationId:remoteIds)
{
if(currApplicationId.equals(commandDefintion.applicationId()))
{
if(logger.isDebugEnabled())
logger.debug("Acct application id matching, application id " + currApplicationId);
found = true;
}
else
{
if(logger.isDebugEnabled())
logger.debug("Acct application id not matching, remote id " + currApplicationId + ",message acct id " + commandDefintion.applicationId());
}
}
}
else
{
if(logger.isDebugEnabled())
logger.debug("Remote acct application ids are empty for link " + getID());
}
if(!found)
return false;
}
else if(message instanceof VendorSpecificRequest || message instanceof VendorSpecificAnswer)
{
if(logger.isDebugEnabled())
logger.debug("It is vendor specific message " + message.getClass().getName());
VendorSpecificApplicationId appId = null;
if(message instanceof VendorSpecificRequest)
appId = ((VendorSpecificRequest)message).getVendorSpecificApplicationId();
else
appId = ((VendorSpecificAnswer)message).getVendorSpecificApplicationId();
Boolean found=false;
List remoteVendorIds = remoteApplicationIds.get();
if(remoteVendorIds!=null && appId!=null)
{
for(VendorSpecificApplicationId currApplicationId:remoteVendorIds)
{
if(sameVendorSpecificApplicationId(appId, currApplicationId))
{
found = true;
break;
}
else
{
if(logger.isDebugEnabled())
logger.debug("Vendor id not matching Ids remote vendor id[" + currApplicationId.getVendorId() + "," + currApplicationId.getAuthApplicationId() + "," + currApplicationId.getAcctApplicationId() + "],message vendor id[" + appId.getVendorId() + "," + appId.getAuthApplicationId() + "," + appId.getAcctApplicationId() + "]");
}
}
}
else
{
if(logger.isDebugEnabled())
logger.debug("No vendor specific application Ids for link " + getID());
}
if(!found && appId!=null)
{
if(logger.isDebugEnabled())
logger.debug("No vendor application id has been found trying to check based on auth/acct application id message " + message.getClass().getName());
if(appId.getAcctApplicationId()!=null)
{
if(logger.isDebugEnabled())
logger.debug("Checking based on acct application id message " + message.getClass().getName());
List remoteIds = remoteAcctApplicationIds.get();
if(remoteIds!=null)
{
for(Long currApplicationId:remoteIds)
{
if(currApplicationId.equals(commandDefintion.applicationId()))
{
if(logger.isDebugEnabled())
logger.debug("Acct application id matching, application id " + currApplicationId);
found = true;
}
else
{
if(logger.isDebugEnabled())
logger.debug("Acct application id not matching, remote id " + currApplicationId + ",message acct id " + commandDefintion.applicationId());
}
}
}
else
{
if(logger.isDebugEnabled())
logger.debug("Remote acct application ids are empty for link " + getID());
}
}
else if(appId.getAuthApplicationId()!=null)
{
if(logger.isDebugEnabled())
logger.debug("Checking based on auth application id message " + message.getClass().getName());
List remoteIds = remoteAuthApplicationIds.get();
if(remoteIds!=null)
{
for(Long currApplicationId:remoteIds)
{
if(currApplicationId.equals(commandDefintion.applicationId()))
{
if(logger.isDebugEnabled())
logger.debug("Auth application id matching, application id " + currApplicationId);
found = true;
}
else
{
if(logger.isDebugEnabled())
logger.debug("Auth application id not matching, remote id " + currApplicationId + ",message acct id " + commandDefintion.applicationId());
}
}
}
else
{
if(logger.isDebugEnabled())
logger.debug("Remote auth application ids are empty for link " + getID());
}
}
}
if(!found)
return false;
return true;
}
else
{
if(logger.isDebugEnabled())
logger.debug("It is authentication message " + message.getClass().getName());
Boolean found=false;
List remoteIds = remoteAuthApplicationIds.get();
if(remoteIds!=null && remoteIds.size()>0)
{
for(Long currApplicationId:remoteIds)
{
if(currApplicationId.equals(commandDefintion.applicationId()))
{
if(logger.isDebugEnabled())
logger.debug("Auth application id matching, application id " + currApplicationId);
found = true;
}
else
{
if(logger.isDebugEnabled())
logger.debug("Auth application id not matching, remote id " + currApplicationId + ",message acct id " + commandDefintion.applicationId());
}
}
}
else
{
if(logger.isDebugEnabled())
logger.debug("Remote auth application ids are empty for link " + getID());
}
if(!found)
return false;
}
return true;
}
public ByteBuf sendMessage(DiameterMessage message, AsyncCallback callback)
{
if(!getPeerState().equals(PeerStateEnum.OPEN))
{
callback.onError(new DiameterException("Invalid state for peer while sending message , current state " + getPeerState(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null));
return null;
}
DiameterCommandDefinition commandDefintion = DiameterParser.getCommandDefinition(message.getClass());
if(commandDefintion == null)
{
callback.onError(new DiameterException("Command not registered in parser for class " + message.getClass().getCanonicalName(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null));
return null;
}
if(!canSendMessage(message))
{
callback.onError(new DiameterException("Application id is not supported by peer, " + commandDefintion.applicationId(), null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null));
return null;
}
return sendMessageInternally(message, callback);
}
private ByteBuf sendMessageInternally(DiameterMessage message, AsyncCallback callback)
{
if(message.getOriginHost()==null)
{
try
{
message.setOriginHost(localHost);
}
catch(MissingAvpException ex)
{
//will not happen value is set
}
}
if(message.getOriginRealm()==null)
{
try
{
message.setOriginRealm(localRealm);
}
catch(MissingAvpException ex)
{
//will not happen value is set
}
}
if(message instanceof DiameterRequest)
{
DiameterRequest request = (DiameterRequest)message;
try
{
if(request.isDestinationHostRequred() && request.getDestinationHost()==null)
request.setDestinationHost(destinationHost);
}
catch(DiameterException ex)
{
//may be not supported
}
try
{
if(request.getDestinationRealm()==null)
request.setDestinationRealm(destinationRealm);
}
catch(DiameterException ex)
{
//may be not supported
}
}
stack.messageSent(message, linkId);
PayloadData payloadData = null;
ByteBuf buffer = null;
ByteBuf copiedBuffer = null;
try
{
buffer = parser.encode(message);
copiedBuffer = Unpooled.copiedBuffer(buffer);
payloadData = new PayloadData(buffer.readableBytes(), buffer, true, false, DIAMETER_SCTP_PROTOCOL_IDENTIFIER, wheel.incrementAndGet()%maxStreams);
}
catch(DiameterException ex)
{
callback.onError(ex);
return null;
}
try
{
association.send(payloadData);
lastActivity.set(System.currentTimeMillis());
callback.onSuccess();
}
catch(Exception ex)
{
callback.onError(new DiameterException(ex.getMessage(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null));
}
return copiedBuffer;
}
public void sendEncodedMessage(ByteBuf buffer, AsyncCallback callback)
{
try
{
PayloadData payloadData = new PayloadData(buffer.readableBytes(), buffer, true, false, DIAMETER_SCTP_PROTOCOL_IDENTIFIER, wheel.incrementAndGet()%maxStreams);
association.send(payloadData);
lastActivity.set(System.currentTimeMillis());
callback.onSuccess();
}
catch(Exception ex)
{
callback.onError(new DiameterException(ex.getMessage(), null, ResultCodes.DIAMETER_UNABLE_TO_COMPLY, null));
}
}
@Override
public void onPayload(Association association, PayloadData payloadData)
{
if(logger.isDebugEnabled())
logger.debug(String.format("Diameter Message received on link=%s %s", this.getID(), payloadData));
ByteBuf buffer = payloadData.getByteBuf();
while(buffer.readableBytes()>0)
{
Integer preMessageIndex = buffer.readerIndex();
try
{
DiameterMessage message = parser.decode(buffer, rejectUnmandatoryAvps);
if(message!=null)
stack.getQueue().offerLast(new MessageProcessingTask(stack, this, genericListeners, lastActivity, waitingForDWA, association, buffer, message, remoteApplicationIds, remoteAuthApplicationIds, remoteAcctApplicationIds));
}
catch(DiameterException ex)
{
logger.warn("An error occured while parsing incoming message " + ex.getMessage() + " from " + association, ex);
if(ex.getPartialMessage()!=null && ex.getPartialMessage() instanceof DiameterRequest)
{
try
{
sendError(ex);
}
catch(DiameterException ex2)
{
logger.warn("An error occured while sending error for incoming message " + ex2.getMessage() + " from " + association, ex2);
}
}
}
Integer currMessageIndex = buffer.readerIndex();
if(currMessageIndex.equals(preMessageIndex))
{
//we will loop if we will continue , lets drop the packet remainings
return;
}
}
}
public void registerApplication(List vendorApplicationIds, List authApplicationIds,List acctApplicationIds, Package providerPackageName, Package packageName) throws DiameterException
{
for(VendorSpecificApplicationId vendorApplicationId:vendorApplicationIds)
{
addApplication(vendorApplicationId);
if(vendorApplicationId.getAcctApplicationId()!=null)
acctApplicationPackages.put(vendorApplicationId.getAcctApplicationId(), providerPackageName);
if(vendorApplicationId.getAuthApplicationId()!=null)
authApplicationPackages.put(vendorApplicationId.getAuthApplicationId(), providerPackageName);
}
for(Long applicationId:authApplicationIds)
{
Boolean found = false;
for(VendorSpecificApplicationId vendorApplicationId:vendorApplicationIds)
{
if(vendorApplicationId.getAuthApplicationId()!=null && vendorApplicationId.getAuthApplicationId().equals(applicationId))
{
found = true;
break;
}
}
if(!found)
{
addAuthApplication(applicationId);
authApplicationPackages.put(applicationId, providerPackageName);
}
}
for(Long applicationId:acctApplicationIds)
{
Boolean found = false;
for(VendorSpecificApplicationId vendorApplicationId:vendorApplicationIds)
{
if(vendorApplicationId.getAcctApplicationId()!=null && vendorApplicationId.getAcctApplicationId().equals(applicationId))
{
found = true;
break;
}
}
if(!found)
{
addAcctApplication(applicationId);
acctApplicationPackages.put(applicationId, providerPackageName);
}
}
parser.registerApplication(stack.getClassLoader(), packageName);
stack.registerGlobalApplication(providerPackageName, packageName);
}
public static Boolean sameVendorSpecificApplicationId(VendorSpecificApplicationId oldId, VendorSpecificApplicationId applicationId)
{
Boolean sameInterface = true;
if(oldId.getVendorId()==null)
{
if(applicationId.getVendorId()!=null)
sameInterface = false;
}
else if(!oldId.getVendorId().equals(applicationId.getVendorId()))
sameInterface = false;
if(oldId.getAuthApplicationId()==null)
{
if(applicationId.getAuthApplicationId()!=null)
sameInterface = false;
}
else if(!oldId.getAuthApplicationId().equals(applicationId.getAuthApplicationId()))
sameInterface = false;
if(oldId.getAcctApplicationId()==null)
{
if(applicationId.getAcctApplicationId()!=null)
sameInterface = false;
}
else if(!oldId.getAcctApplicationId().equals(applicationId.getAcctApplicationId()))
sameInterface = false;
return sameInterface;
}
private void addApplication(VendorSpecificApplicationId applicationId) throws DiameterException
{
for(VendorSpecificApplicationId oldId:applicationIds)
{
Boolean sameInterface = sameVendorSpecificApplicationId(oldId, applicationId);
if(sameInterface)
throw new DiameterException("Application is already registred for this peer", null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null);
}
if(applicationId.getAuthApplicationId()!=null)
{
for(Long oldId:authApplicationIds)
{
if(oldId.equals(applicationId.getAuthApplicationId()))
throw new DiameterException("Application is already registred for this peer", null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null);
}
}
if(applicationId.getAcctApplicationId()!=null)
{
for(Long oldId:acctApplicationIds)
{
if(oldId.equals(applicationId.getAuthApplicationId()))
throw new DiameterException("Application is already registred for this peer", null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null);
}
}
applicationIds.add(applicationId);
if(applicationId.getAuthApplicationId()!=null)
authApplicationIds.add(applicationId.getAuthApplicationId());
if(applicationId.getAcctApplicationId()!=null)
acctApplicationIds.add(applicationId.getAcctApplicationId());
}
private void addAuthApplication(Long applicationId) throws DiameterException
{
for(Long oldId:authApplicationIds)
{
if(oldId.equals(applicationId))
throw new DiameterException("Application is already registred for this peer", null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null);
}
authApplicationIds.add(applicationId);
}
private void addAcctApplication(Long applicationId) throws DiameterException
{
for(Long oldId:acctApplicationIds)
{
if(oldId.equals(applicationId))
throw new DiameterException("Application is already registred for this peer", null, ResultCodes.DIAMETER_APPLICATION_UNSUPPORTED, null);
}
acctApplicationIds.add(applicationId);
}
public void sendCER()
{
List addresses = Arrays.asList(new InetAddress[] { localAddress });
CapabilitiesExchangeRequest cer = null;
try
{
cer = new CapabilitiesExchangeRequestImpl(localHost, localRealm, false, addresses, stack.getVendorID(), stack.getProductName());
Long hopIdentifier = stack.getNextHopByHopIdentifier();
cer.setHopByHopIdentifier(hopIdentifier);
cer.setEndToEndIdentifier(hopIdentifier);
cer.setOriginStateId(stack.getOriginalStateId());
cer.setFirmwareRevision(stack.getFirmwareRevision());
List vaiToSend=new ArrayList();
List acctToSend=new ArrayList();
List authToSend=new ArrayList();
List vendorsToSend=new ArrayList();
ConcurrentHashMap acctUsed=new ConcurrentHashMap();
ConcurrentHashMap authUsed=new ConcurrentHashMap();
for(VendorSpecificApplicationId vai:applicationIds)
{
if(vai.getVendorId()==null)
vaiToSend.add(vai);
else
{
Boolean found=false;
for(Long currVendor:vendorsToSend)
if(currVendor.equals(vai.getVendorId()))
{
found=true;
break;
}
if(!found)
vendorsToSend.add(vai.getVendorId());
if(vai.getAuthApplicationId()!=null)
{
authToSend.add(vai.getAuthApplicationId());
authUsed.put(vai.getAuthApplicationId(),true);
}
if(vai.getAcctApplicationId()!=null)
{
acctToSend.add(vai.getAcctApplicationId());
acctUsed.put(vai.getAcctApplicationId(),true);
}
}
}
for(Long acct:acctApplicationIds)
{
if(acctUsed.putIfAbsent(acct, true)==null)
acctToSend.add(acct);
}
for(Long auth:authApplicationIds)
if(authUsed.putIfAbsent(auth, true)==null)
authToSend.add(auth);
if(vaiToSend.size()>0)
cer.setVendorSpecificApplicationIds(vaiToSend);
if(authToSend.size()>0)
cer.setAuthApplicationIds(authToSend);
if(acctToSend.size()>0)
cer.setAcctApplicationIds(acctToSend);
if(vendorsToSend.size()>0)
cer.setSupportedVendorIds(vendorsToSend);
setPeerState(PeerStateEnum.CER_SENT);
sendMessageInternally(cer, new AsyncCallback()
{
@Override
public void onSuccess()
{
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending CER to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
resetReconnectTimer();
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending CER to " + association + " " + ex.getMessage(),ex);
}
}
public void sendCEA(CapabilitiesExchangeRequest request)
{
List addresses = Arrays.asList(new InetAddress[] { localAddress });
CapabilitiesExchangeAnswer cea = null;
try
{
cea = new CapabilitiesExchangeAnswerImpl(localHost, localRealm, false, ResultCodes.DIAMETER_SUCCESS, addresses, stack.getVendorID(), stack.getProductName());
cea.setHopByHopIdentifier(request.getHopByHopIdentifier());
cea.setEndToEndIdentifier(request.getEndToEndIdentifier());
cea.setOriginStateId(stack.getOriginalStateId());
cea.setFirmwareRevision(stack.getFirmwareRevision());
List vaiToSend=new ArrayList();
List acctToSend=new ArrayList();
List authToSend=new ArrayList();
List vendorsToSend=new ArrayList();
for(VendorSpecificApplicationId vai:applicationIds)
{
if(vai.getVendorId()==null)
vaiToSend.add(vai);
else
{
Boolean found=false;
for(Long currVendor:vendorsToSend)
if(currVendor.equals(vai.getVendorId()))
{
found=true;
break;
}
if(!found)
vendorsToSend.add(vai.getVendorId());
if(vai.getAuthApplicationId()!=null)
authToSend.add(vai.getAuthApplicationId());
if(vai.getAcctApplicationId()!=null)
acctToSend.add(vai.getAcctApplicationId());
}
}
for(Long acct:acctApplicationIds)
{
Boolean isNonVendor=true;
for(VendorSpecificApplicationId vai:applicationIds)
if(vai.getAcctApplicationId().equals(acct))
{
isNonVendor = false;
break;
}
if(isNonVendor)
acctToSend.add(acct);
}
for(Long auth:authApplicationIds)
{
Boolean isNonVendor=true;
for(VendorSpecificApplicationId vai:applicationIds)
if(vai.getAuthApplicationId().equals(auth))
{
isNonVendor = false;
break;
}
if(isNonVendor)
authToSend.add(auth);
}
if(vaiToSend.size()>0)
cea.setVendorSpecificApplicationIds(vaiToSend);
if(authToSend.size()>0)
cea.setAuthApplicationIds(authToSend);
if(acctToSend.size()>0)
cea.setAcctApplicationIds(acctToSend);
if(vendorsToSend.size()>0)
cea.setSupportedVendorIds(vendorsToSend);
setPeerState(PeerStateEnum.OPEN);
sendMessageInternally(cea, new AsyncCallback()
{
@Override
public void onSuccess()
{
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending CEA to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
resetReconnectTimer();
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending CER to " + association + " " + ex.getMessage(),ex);
}
}
public void sendDWR()
{
DeviceWatchdogRequest dwr = null;
try
{
dwr = new DeviceWatchdogRequestImpl(localHost, localRealm, false);
Long hopIdentifier = stack.getNextHopByHopIdentifier();
dwr.setHopByHopIdentifier(hopIdentifier);
dwr.setEndToEndIdentifier(hopIdentifier);
dwr.setOriginStateId(stack.getOriginalStateId());
sendMessageInternally(dwr, new AsyncCallback()
{
@Override
public void onSuccess()
{
//waiting for DWA
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending DWR to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
resetReconnectTimer();
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending DWR to " + association + " " + ex.getMessage(),ex);
}
}
public void sendDWA(DeviceWatchdogRequest request, long resultCode)
{
DeviceWatchdogAnswer dwa = null;
try
{
dwa = new DeviceWatchdogAnswerImpl(localHost, localRealm, false, resultCode);
dwa.setHopByHopIdentifier(request.getHopByHopIdentifier());
dwa.setEndToEndIdentifier(request.getEndToEndIdentifier());
dwa.setOriginStateId(stack.getOriginalStateId());
sendMessageInternally(dwa, new AsyncCallback()
{
@Override
public void onSuccess()
{
//waiting for DWA
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending DWA to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
resetReconnectTimer();
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending DWA to " + association + " " + ex.getMessage(),ex);
}
}
public void sendDPA(DisconnectPeerRequest request, long resultCode)
{
DisconnectPeerAnswer dpa = null;
try
{
dpa = new DisconnectPeerAnswerImpl(localHost, localRealm, false, resultCode);
dpa.setHopByHopIdentifier(request.getHopByHopIdentifier());
dpa.setEndToEndIdentifier(request.getEndToEndIdentifier());
sendMessageInternally(dpa, new AsyncCallback()
{
@Override
public void onSuccess()
{
//waiting for DWA
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending DPA to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
resetReconnectTimer();
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending DPA to " + association + " " + ex.getMessage(),ex);
}
}
public void sendDPR(DisconnectCauseEnum cause, AsyncCallback callback)
{
DisconnectPeerRequest dpr = null;
try
{
dpr = new DisconnectPeerRequestImpl(localHost, localRealm, false, cause);
Long hopIdentifier = stack.getNextHopByHopIdentifier();
dpr.setEndToEndIdentifier(hopIdentifier);
dpr.setHopByHopIdentifier(hopIdentifier);
setPeerState(PeerStateEnum.DPR_SENT);
sendMessageInternally(dpr, new AsyncCallback()
{
@Override
public void onSuccess()
{
disconnectTimer.startTimer(inactivityTimeout, callback);
stack.getPeriodicQueue().store(disconnectTimer.getRealTimestamp(), disconnectTimer);
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending DPR to " + association + " " + ex.getMessage(),ex);
setPeerState(PeerStateEnum.IDLE);
inactivityTimer.stop();
if(callback!=null)
callback.onError(ex);
}
});
}
catch(Exception ex)
{
logger.warn("An error occured while sending CER to " + association + " " + ex.getMessage(),ex);
}
}
public void sendError(DiameterException ex) throws MissingAvpException, AvpNotSupportedException
{
if(ex.getPartialMessage()==null || ex.getPartialMessage() instanceof DiameterAnswer)
return;
DiameterErrorAnswer answer;
String sessionId = null;
try
{
sessionId = ex.getPartialMessage().getSessionId();
}
catch(DiameterException ex2)
{
//may be without session
}
String originHost=null;
String originRealm=null;
Integer commandCode = ex.getCommandCode();
Long applicationID = ex.getApplicationID();
if(commandCode==null || applicationID==null)
{
DiameterCommandDefinition commandDef = DiameterParser.getCommandDefinition(ex.getPartialMessage().getClass());
if(commandDef!=null)
{
commandCode = commandDef.commandCode();
applicationID = commandDef.applicationId();
}
}
if(ex.getPartialMessage() instanceof DiameterRequest)
{
originHost = ((DiameterRequest)ex.getPartialMessage()).getDestinationHost();
originRealm=((DiameterRequest)ex.getPartialMessage()).getDestinationRealm();
}
if(originHost == null)
originHost = this.localHost;
if(originRealm == null)
originRealm = this.localRealm;
if(sessionId == null)
answer = new DiameterErrorAnswerImpl(applicationID, commandCode, originHost, originRealm, false, ex.getErrorCode());
else
answer = new DiameterErrorAnswerWithSessionImpl(applicationID, commandCode, originHost, originRealm, false, ex.getErrorCode(), sessionId);
answer.setErrorMessage(ex.getMessage());
answer.setErrorReportingHost(localHost);
answer.setEndToEndIdentifier(ex.getPartialMessage().getEndToEndIdentifier());
answer.setHopByHopIdentifier(ex.getPartialMessage().getHopByHopIdentifier());
sendMessageInternally(answer, new AsyncCallback()
{
@Override
public void onSuccess()
{
}
@Override
public void onError(DiameterException ex)
{
logger.warn("An error occured while sending error answer " + ex.getMessage() + " from " + association, ex);
}
});
}
@Override
public PeerStateEnum getPeerState()
{
return peerState.get();
}
@Override
public void setPeerState(PeerStateEnum peerState)
{
this.peerState.set(peerState);
}
@Override
public Package getPackage(Long applicationID, Boolean isAuth)
{
if(isAuth!=null && isAuth)
return authApplicationPackages.get(applicationID);
return acctApplicationPackages.get(applicationID);
}
@Override
public List getAuthApplicationIds()
{
return authApplicationIds;
}
@Override
public List getAcctApplicationIds()
{
return acctApplicationIds;
}
@Override
public List getVendorSpecificApplicationIds()
{
return applicationIds;
}
@Override
public void resetInactivityTimer()
{
inactivityTimer.resetTimer();
stack.getPeriodicQueue().store(inactivityTimer.getRealTimestamp(), inactivityTimer);
}
@Override
public void resetReconnectTimer()
{
reconnectTimer.startTimer(reconnectTimeout, null);
stack.getPeriodicQueue().store(reconnectTimer.getRealTimestamp(), reconnectTimer);
}
@Override
public void disconnectOperationCompleted()
{
disconnectTimer.execute();
}
@Override
public String getID()
{
return linkId;
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy