/**
* Copyright 2010-2011 Voxeo Corporation
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
* file except in compliance with the License.
*
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed
* under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
package com.voxeo.moho.sip;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.StringTokenizer;
import javax.servlet.sip.Address;
import javax.servlet.sip.ServletParseException;
import javax.servlet.sip.SipServletRequest;
import javax.servlet.sip.SipServletResponse;
import javax.servlet.sip.SipURI;
import com.voxeo.moho.Endpoint;
import com.voxeo.moho.Framework;
import com.voxeo.moho.SignalException;
import com.voxeo.moho.common.event.MohoRegisterEvent;
import com.voxeo.moho.sip.SIPEndpoint;
import com.voxeo.moho.spi.ExecutionContext;
public class SIPRegisterEventImpl extends MohoRegisterEvent implements SIPRegisterEvent {
protected SipServletRequest _req;
protected ExecutionContext _ctx;
protected Contact[] _contacts;
protected Endpoint _endpoint;
protected long _creationTime;
protected String _domain;
public class ContactImpl implements SIPContact {
SIPEndpoint _endpoint;
int _expiration;
int _cseq;
String _callID;
ContactImpl(SIPEndpoint ep, int expiration, String callID, int cseq) {
_endpoint = ep;
_expiration = expiration;
_creationTime = System.currentTimeMillis();
_callID = callID;
_cseq = cseq;
}
@Override
public Endpoint getEndpoint() {
return _endpoint;
}
public void setExpiration(int exp) {
_expiration = exp;
}
@Override
public int getExpiration() {
return _expiration;
}
@Override
public String toString() {
return _endpoint.toString() + ";expiration=" + _expiration;
}
@Override
public int hashCode() {
return _endpoint.hashCode() + _expiration;
}
@Override
public boolean equals(Object o) {
if (o instanceof Contact) {
Contact c = (Contact) o;
if (!_endpoint.equals(c.getEndpoint())) {
return false;
}
if (_expiration != c.getExpiration()) {
return false;
}
return true;
}
return false;
}
@Override
public boolean isWildCard() {
return _endpoint.isWildCard();
}
@Override
public boolean isExpired() {
return (System.currentTimeMillis() - _creationTime) > (_expiration * 1000L);
}
@Override
public int getCSeq() {
return _cseq;
}
@Override
public String getCallID() {
return _callID;
}
}
public SIPRegisterEventImpl(final Framework source, final SipServletRequest req) {
super(source);
_req = req;
_ctx = (ExecutionContext) source.getApplicationContext();
_endpoint = new SIPEndpointImpl(_ctx, _req.getFrom());
_domain = ((SipURI)req.getRequestURI()).getHost();
}
@Override
public SipServletRequest getSipRequest() {
return _req;
}
@Override
public Endpoint getEndpoint() {
return _endpoint;
}
@Override
public synchronized Contact[] getContacts() {
if (_contacts == null) {
final List<Contact> retval = new ArrayList<Contact>();
try {
final ListIterator<Address> headers = _req.getAddressHeaders("Contact");
int expiration = _req.getExpires();
int cseq = Integer.parseInt(new StringTokenizer(_req.getHeader("CSeq").trim()).nextToken());;
while (headers.hasNext()) {
Address addr = headers.next();
SIPEndpoint ep = new SIPEndpointImpl(_ctx, addr);
int exp = addr.getExpires();
if (exp <= 0) {
exp = expiration;
}
Contact contact = new ContactImpl(ep, exp, _req.getCallId(), cseq);
retval.add(contact);
}
}
catch (final ServletParseException e) {
throw new IllegalArgumentException(e);
}
_contacts = retval.toArray(new Contact[retval.size()]);
}
return _contacts;
}
@Override
public synchronized void accept(final Contact[] contacts, final Map<String, String> headers) {
this.checkState();
_accepted = true;
final SipServletResponse res = _req.createResponse(SipServletResponse.SC_OK);
for (final Contact contact : contacts) {
res.addHeader("Contact", contact.toString());
}
SIPHelper.addHeaders(res, headers);
try {
res.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
@Override
public synchronized void reject(final Reason reason, final Map<String, String> headers) {
this.checkState();
_rejected = true;
final SipServletResponse res = _req.createResponse(reason == null ? Reason.DECLINE.getCode() : reason.getCode());
SIPHelper.addHeaders(res, headers);
try {
res.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
@Override
public synchronized void redirect(Endpoint o, Map<String, String> headers) throws SignalException {
checkState();
_redirected = true;
if (o instanceof SIPEndpoint) {
final SipServletResponse res = _req.createResponse(SipServletResponse.SC_MOVED_TEMPORARILY);
res.setHeader("Contact", ((SIPEndpoint) o).getURI().toString());
SIPHelper.addHeaders(res, headers);
try {
res.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
else {
throw new IllegalArgumentException("Unable to redirect the call to a non-SIP participant.");
}
}
@Override
public void proxyTo(boolean recordRoute, boolean parallel, Endpoint... destinations) throws SignalException {
this.proxyTo(recordRoute, parallel, null, destinations);
}
@Override
public synchronized void proxyTo(boolean recordRoute, boolean parallel, Map<String, String> headers, Endpoint... destinations) {
checkState();
_proxied = true;
SIPHelper.proxyTo(_ctx.getSipFactory(), _req, headers, recordRoute, parallel, destinations);
}
@Override
public String getDomain() {
return _domain;
}
}