/**
* 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.Map;
import javax.servlet.sip.ServletParseException;
import javax.servlet.sip.SipServletRequest;
import javax.servlet.sip.SipServletResponse;
import javax.servlet.sip.SipSession;
import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import com.voxeo.moho.Call;
import com.voxeo.moho.Endpoint;
import com.voxeo.moho.SignalException;
import com.voxeo.moho.Subscription;
import com.voxeo.moho.Subscription.Type;
import com.voxeo.moho.common.event.MohoNotifyEvent;
import com.voxeo.moho.event.EventSource;
public class SIPNotifyEventImpl<T extends EventSource> extends MohoNotifyEvent<T> implements SIPNotifyEvent<T> {
protected SipServletRequest _req;
protected SIPNotifyEventImpl(final T source, final SipServletRequest req) {
super(source);
_req = req;
}
@Override
public SipServletRequest getSipRequest() {
return _req;
}
@Override
public synchronized void accept(final Map<String, String> headers) throws SignalException, IllegalStateException {
this.checkState();
_accepted = true;
final SipServletResponse res = _req.createResponse(SipServletResponse.SC_OK);
SIPHelper.addHeaders(res, headers);
try {
res.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
@Override
public void forwardTo(final Call call) throws SignalException {
this.forwardTo(call, null);
}
@Override
public synchronized void forwardTo(final Call call, final Map<String, String> headers) throws SignalException {
if (!(call instanceof SIPCall)) {
throw new UnsupportedOperationException("Cannot forward to non-SIPCall.");
}
if (_req.isInitial()) {
throw new IllegalArgumentException("Cannot forward initial SIP request.");
}
final SIPCallImpl scall = (SIPCallImpl) call;
if (!scall.isAnswered()) {
throw new IllegalStateException("Cannot forward to no-answered call.");
}
this.checkState();
_forwarded = true;
final SipSession session = scall.getSipSession();
final SipServletRequest req = session.createRequest(_req.getMethod());
SIPHelper.addHeaders(req, headers);
SIPHelper.copyContent(_req, req);
SIPHelper.linkSIPMessage(_req, req);
try {
req.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
@Override
public void forwardTo(final Endpoint endpoint) throws SignalException, IllegalStateException {
this.forwardTo(endpoint, null);
}
@Override
public synchronized void forwardTo(final Endpoint endpoint, final Map<String, String> headers)
throws SignalException, IllegalStateException {
throw new UnsupportedOperationException();
}
@Override
public Type getEventType() {
String event = null;
try {
event = _req.getParameterableHeader("Event").getValue();
}
catch (final ServletParseException e) {
throw new IllegalArgumentException(e);
}
if (event.equalsIgnoreCase(Subscription.Type.DIALOG.name())) {
return Subscription.Type.DIALOG;
}
else if (event.equalsIgnoreCase(Subscription.Type.PRESENCE.name())) {
return Subscription.Type.PRESENCE;
}
else if (event.equalsIgnoreCase(Subscription.Type.REFER.name())) {
return Subscription.Type.REFER;
}
return null;
}
@Override
public SubscriptionState getSubscriptionState() {
String state = null;
try {
state = _req.getParameterableHeader("Subscription-State").getValue();
}
catch (final ServletParseException e) {
throw new IllegalArgumentException(e);
}
if (state.equalsIgnoreCase(MohoNotifyEvent.SubscriptionState.ACTIVE.name())) {
return MohoNotifyEvent.SubscriptionState.ACTIVE;
}
else if (state.equalsIgnoreCase(MohoNotifyEvent.SubscriptionState.PENDING.name())) {
return MohoNotifyEvent.SubscriptionState.PENDING;
}
else if (state.equalsIgnoreCase(MohoNotifyEvent.SubscriptionState.TERMINATED.name())) {
return MohoNotifyEvent.SubscriptionState.TERMINATED;
}
return null;
}
@Override
public String getResourceState() {
if (getEventType() == Subscription.Type.DIALOG) {
// TODO improve the content parser
Document doc = null;
try {
doc = DocumentHelper.parseText(new String(_req.getRawContent(), "UTF-8"));
}
catch (final Exception e) {
throw new IllegalArgumentException(e);
}
// rfc4235 defined dialog state: Trying, Proceeding, Early, Confirmed,
// Terminated
return doc.getRootElement().element("dialog").element("state").getText();
}
else if (getEventType() == Subscription.Type.PRESENCE) {
// TODO improve the content parser
Document doc = null;
try {
doc = DocumentHelper.parseText(new String(_req.getRawContent(), "UTF-8"));
}
catch (final Exception e) {
throw new IllegalArgumentException(e);
}
// rfc3863 defined "open" and "close".
return doc.getRootElement().element("tuple").element("status").elementText("basic");
}
else if (getEventType() == Subscription.Type.REFER) {
// TODO
return null;
}
return null;
}
@Override
public synchronized void reject(Reason reason, Map<String, String> headers) throws SignalException {
this.checkState();
_rejected = true;
if (reason == null) {
reason = Reason.DECLINE;
}
final SipServletResponse res = _req.createResponse(reason.getCode());
SIPHelper.addHeaders(res, headers);
try {
res.send();
}
catch (final IOException e) {
throw new SignalException(e);
}
}
@Override
public void proxyTo(boolean recordRoute, boolean parallel, Endpoint... destinations) throws SignalException {
proxyTo(recordRoute, parallel, null, destinations);
}
@Override
public synchronized void proxyTo(boolean recordRoute, boolean parallel, Map<String, String> headers, Endpoint... destinations) {
this.checkState();
_proxied = true;
SIPHelper.proxyTo(getSource().getApplicationContext().getSipFactory(), _req, headers, recordRoute, parallel, destinations);
}
}