You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
sems/core/AmSipSubscription.h

310 lines
7.6 KiB

/*
* Copyright (C) 2012 Frafos GmbH
*
* This file is part of SEMS, a free SIP media server.
*
* SEMS is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version. This program is released under
* the GPL with the additional exemption that compiling, linking,
* and/or using OpenSSL is allowed.
*
* For a license to use the SEMS software under conditions
* other than those described here, or to purchase support for this
* software, please contact iptel.org by e-mail at the following addresses:
* info@iptel.org
*
* SEMS 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#ifndef _AmSipSubscription_h_
#define _AmSipSubscription_h_
#include "AmSipMsg.h"
#include "AmBasicSipDialog.h"
#include "AmEvent.h"
#include "AmEventQueue.h"
#include "AmAppTimer.h"
#include <map>
#include <list>
#include <memory>
using std::map;
using std::list;
class AmBasicSipDialog;
class AmEventQueue;
class AmSipSubscription;
class SingleSubscription;
struct SingleSubTimeoutEvent
: public AmEvent
{
string ltag;
int timer_id;
SingleSubscription* sub;
SingleSubTimeoutEvent(const string& ltag, int timer_id, SingleSubscription* sub)
: AmEvent(E_SIP_SUBSCRIPTION), ltag(ltag), timer_id(timer_id), sub(sub)
{}
};
/**
* \brief Single SIP Subscription
*
* This class contain only one SIP suscription,
* identified by its event package name, id and role.
*/
class SingleSubscription
{
public:
enum Role {
Subscriber=0,
Notifier
};
private:
class SubscriptionTimer
: public DirectAppTimer
{
SingleSubscription* sub;
int timer_id;
public:
SubscriptionTimer(SingleSubscription* sub, int timer_id)
: sub(sub), timer_id(timer_id)
{}
void fire(){
sub->onTimer(timer_id);
}
};
enum SubscriptionTimerId {
RFC6665_TIMER_N=0,
SUBSCRIPTION_EXPIRE
};
// state
unsigned int sub_state;
int pending_subscribe;
unsigned long expires;
// timers
SubscriptionTimer timer_n;
SubscriptionTimer timer_expires;
AmSipSubscription* subs;
void onTimer(int timer_id);
AmBasicSipDialog* dlg();
void requestFSM(const AmSipRequest& req);
friend class SubscriptionTimer;
public:
enum SubscriptionState {
SubState_init=0,
SubState_notify_wait,
SubState_pending,
SubState_active,
SubState_terminated
};
// identifiers
string event;
string id;
Role role;
SingleSubscription(AmSipSubscription* subs, Role role,
const string& event, const string& id);
virtual ~SingleSubscription();
bool onRequestIn(const AmSipRequest& req);
void onRequestSent(const AmSipRequest& req);
virtual void replyFSM(const AmSipRequest& req, const AmSipReply& reply);
unsigned int getState() { return sub_state; }
virtual void setState(unsigned int st);
unsigned long getExpires() { return expires; }
void setExpires(unsigned long exp);
void terminate();
bool terminated();
string to_str();
};
/**
* \brief SIP Subscription collection
*
* This class contains all the suscriptions sharing
* the same dialog.
*/
class AmSipSubscription
{
protected:
typedef list<SingleSubscription*> Subscriptions;
typedef map<unsigned int,Subscriptions::iterator> CSeqMap;
AmBasicSipDialog* dlg;
AmEventQueue* ev_q;
Subscriptions subs;
CSeqMap uas_cseq_map;
CSeqMap uac_cseq_map;
bool allow_subless_notify;
SingleSubscription* makeSubscription(const AmSipRequest& req, bool uac);
Subscriptions::iterator createSubscription(const AmSipRequest& req, bool uac);
Subscriptions::iterator matchSubscription(const AmSipRequest& req, bool uac);
Subscriptions::iterator findSubscription(SingleSubscription::Role role,
const string& event, const string& id);
void removeSubFromUACCSeqMap(Subscriptions::iterator sub);
void removeSubFromUASCSeqMap(Subscriptions::iterator sub);
virtual void removeSubscription(Subscriptions::iterator sub);
virtual SingleSubscription* newSingleSubscription(SingleSubscription::Role role,
const string& event,
const string& id);
friend class SingleSubscription;
public:
AmSipSubscription(AmBasicSipDialog* dlg, AmEventQueue* ev_q);
virtual ~AmSipSubscription();
virtual void allowUnsolicitedNotify(bool allow) {
allow_subless_notify = allow;
}
/**
* Is there at least one active subscription?
*/
bool isActive();
/**
* Terminate all subscriptions
*/
void terminate();
/**
* Check if a subscription exists
*/
bool subscriptionExists(SingleSubscription::Role role,
const string& event, const string& id);
bool onRequestIn(const AmSipRequest& req);
void onRequestSent(const AmSipRequest& req);
bool onReplyIn(const AmSipRequest& req, const AmSipReply& reply);
void onReplySent(const AmSipRequest& req, const AmSipReply& reply);
virtual void onNotify(const AmSipRequest& req, SingleSubscription* sub) {}
virtual void onFailureReply(const AmSipReply& reply, SingleSubscription* sub) {}
virtual void onTimeout(int timer_id, SingleSubscription* sub);
virtual void debug();
};
struct SIPSubscriptionEvent
: public AmEvent
{
enum SubscriptionStatus {
SubscribeActive=0,
SubscribeFailed,
SubscribeTerminated,
SubscribePending,
SubscriptionTimeout
};
string handle;
unsigned int code;
string reason;
SubscriptionStatus status;
unsigned int expires;
std::auto_ptr<AmMimeBody> notify_body;
SIPSubscriptionEvent(SubscriptionStatus status, const string& handle,
unsigned int expires = 0,
unsigned int code=0, const string& reason="");
const char* getStatusText();
};
struct AmSipSubscriptionInfo
{
string domain;
string user;
string from_user;
string pwd;
string proxy;
string event;
string accept;
string id;
AmSipSubscriptionInfo(const string& domain,
const string& user,
const string& from_user,
const string& pwd,
const string& proxy,
const string& event)
: domain(domain),user(user),
from_user(from_user),pwd(pwd),proxy(proxy),
event(event)
{ }
};
class AmSipSubscriptionDialog
: public AmBasicSipDialog,
public AmBasicSipEventHandler,
public AmSipSubscription
{
string event;
string event_id;
string accept;
string sess_link;
public:
AmSipSubscriptionDialog(const AmSipSubscriptionInfo& info,
const string& sess_link,
AmEventQueue* ev_q = NULL);
int subscribe(int expires);
string getDescription();
/** AmSipSubscription interface */
void onNotify(const AmSipRequest& req, SingleSubscription* sub);
void onFailureReply(const AmSipReply& reply, SingleSubscription* sub);
void onTimeout(int timer_id, SingleSubscription* sub);
/** AmBasicDialogEventHandler interface */
void onSipRequest(const AmSipRequest& req)
{ AmSipSubscription::onRequestIn(req); }
void onRequestSent(const AmSipRequest& req)
{ AmSipSubscription::onRequestSent(req); }
void onSipReply(const AmSipRequest& req, const AmSipReply& reply,
AmBasicSipDialog::Status old_status)
{ AmSipSubscription::onReplyIn(req,reply); }
void onReplySent(const AmSipRequest& req, const AmSipReply& reply)
{ AmSipSubscription::onReplySent(req,reply); }
};
#endif