microsoft.exchange.webservices.data.StreamingSubscriptionConnection Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of exchange-ws-api Show documentation
Show all versions of exchange-ws-api Show documentation
The source came from http://archive.msdn.microsoft.com/ewsjavaapi
Support for Maven has been added.
/**************************************************************************
* copyright file="StreamingSubscriptionConnection.java" company="Microsoft"
* Copyright (c) Microsoft Corporation. All rights reserved.
*
* Defines the StreamingSubscriptionConnection class.
**************************************************************************/
package microsoft.exchange.webservices.data;
import java.io.Closeable;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Represents a connection to an ongoing stream of events.
*/
public final class StreamingSubscriptionConnection implements
Closeable,HangingServiceRequestBase.IHandleResponseObject,
HangingServiceRequestBase.IHangingRequestDisconnectHandler{
/**
* Mapping of streaming id to subscriptions currently on the connection.
*/
private Map subscriptions;
/**
* connection lifetime, in minutes
*/
private int connectionTimeout;
/**
* ExchangeService instance used to make the EWS call.
*/
private ExchangeService session;
/**
* Value indicating whether the class is disposed.
*/
private boolean isDisposed;
/**
* Currently used instance of a GetStreamingEventsRequest connected to EWS.
*/
private GetStreamingEventsRequest currentHangingRequest;
public interface INotificationEventDelegate {
/**
* Represents a delegate that is invoked when
* notifications are received from the server
* @param sender The StreamingSubscriptionConnection
* instance that received the events.
* @param args The event data.
*/
void notificationEventDelegate(Object sender,
NotificationEventArgs args);
}
/***
* Notification events Occurs when
* notifications are received from the server.
*/
private List onNotificationEvent =
new ArrayList();
/***
* Set event to happen when property Notify.
*
* @param notificationEvent
* notification event
*/
public void addOnNotificationEvent(
INotificationEventDelegate notificationEvent) {
onNotificationEvent.add(notificationEvent);
}
/***
* Remove the event from happening when property Notify.
*
* @param notificationEvent
* notification event
*/
public void removeNotificationEvent(
INotificationEventDelegate notificationEvent) {
onNotificationEvent.remove(notificationEvent);
}
/**
* * Clears notification events list.
*/
public void clearNotificationEvent() {
onNotificationEvent.clear();
}
public interface ISubscriptionErrorDelegate {
/**
* Represents a delegate that is invoked when
* an error occurs within a streaming subscription connection.
* @param sender The StreamingSubscriptionConnection
* instance within which the error occurred.
* @param args The event data.
*/
void subscriptionErrorDelegate(Object sender,
SubscriptionErrorEventArgs args);
}
/***
* Subscription events Occur when a subscription encounters an error.
*/
private List onSubscriptionError =
new ArrayList();
/***
* Set event to happen when property subscriptionError.
*
* @param subscriptionError
* subscription event
*/
public void addOnSubscriptionError(
ISubscriptionErrorDelegate subscriptionError) {
onSubscriptionError.add(subscriptionError);
}
/***
* Remove the event from happening when property subscription.
*
* @param subscriptionError
* subscription event
*/
public void removeSubscriptionError(
ISubscriptionErrorDelegate subscriptionError) {
onSubscriptionError.remove(subscriptionError);
}
/**
* * Clears subscription events list.
*/
public void clearSubscriptionError() {
onSubscriptionError.clear();
}
/***
* Disconnect events Occurs when a streaming subscription
* connection is disconnected from the server.
*/
private List onDisconnect =
new ArrayList();
/***
* Set event to happen when property disconnect.
*
* @param disconnect
* disconnect event
*/
public void addOnDisconnect(
ISubscriptionErrorDelegate disconnect) {
onDisconnect.add(disconnect);
}
/***
* Remove the event from happening when property disconnect.
*
* @param disconnect
* disconnect event
*/
public void removeDisconnect(
ISubscriptionErrorDelegate disconnect) {
onDisconnect.remove(disconnect);
}
/**
* * Clears disconnect events list.
*/
public void clearDisconnect() {
onDisconnect.clear();
}
/**
* Initializes a new instance of the StreamingSubscriptionConnection class.
* @param service The ExchangeService instance this
* connection uses to connect to the server.
* @param lifetime The maximum time, in minutes, the connection
* will remain open. Lifetime must be between 1 and 30.
* @throws Exception
*/
public StreamingSubscriptionConnection(
ExchangeService service,
int lifetime) throws Exception {
EwsUtilities.validateParam(service, "service");
EwsUtilities.validateClassVersion(
service,
ExchangeVersion.Exchange2010_SP1,
this.getClass().getName());
if (lifetime < 1 || lifetime > 30) {
throw new ArgumentOutOfRangeException("lifetime");
}
this.session = service;
this.subscriptions = new HashMap();
this.connectionTimeout = lifetime;
}
/**
* Initializes a new instance of the StreamingSubscriptionConnection class.
* @param service The ExchangeService instance this
* connection uses to connect to the server.
* @param subscriptions Iterable subcriptions
* @param lifetime The maximum time, in minutes, the connection
* will remain open. Lifetime must be between 1 and 30.
* @throws Exception
*/
public StreamingSubscriptionConnection(
ExchangeService service,
Iterable subscriptions,
int lifetime) throws Exception {
this(service, lifetime);
EwsUtilities.validateParamCollection(subscriptions.iterator(),
"subscriptions");
for(StreamingSubscription subscription : subscriptions) {
this.subscriptions.put(subscription.getId(), subscription);
}
}
/**
* Adds a subscription to this connection.
* @param subscription The subscription to add.
* @throws Exception
* @exception Thrown when AddSubscription is called while connected.
*/
public void addSubscription(StreamingSubscription subscription)
throws Exception {
this.throwIfDisposed();
EwsUtilities.validateParam(subscription, "subscription");
this.validateConnectionState(false, Strings.
CannotAddSubscriptionToLiveConnection);
synchronized (this)
{
if (this.subscriptions.containsKey(subscription.getId()))
{
return;
}
this.subscriptions.put(subscription.getId(), subscription);
}
}
/**
* Removes the specified streaming subscription from the connection.
* @param subscription The subscription to remove.
* @throws Exception
* @exception Thrown when RemoveSubscription is called while connected.
*/
public void removeSubscription(StreamingSubscription subscription)
throws Exception {
this.throwIfDisposed();
EwsUtilities.validateParam(subscription, "subscription");
this.validateConnectionState(false, Strings.
CannotRemoveSubscriptionFromLiveConnection);
synchronized (this)
{
this.subscriptions.remove(subscription.getId());
}
}
/**
* Opens this connection so it starts receiving events
* from the server.This results in a long-standing call to EWS.
* @throws Exception
* @throws ServiceLocalException
* @exception Thrown when Open is called while connected.
*/
public void open() throws ServiceLocalException, Exception
{
synchronized (this)
{
this.throwIfDisposed();
this.validateConnectionState(false, Strings.
CannotCallConnectDuringLiveConnection);
if (this.subscriptions.size() == 0)
{
throw new ServiceLocalException(Strings.
NoSubscriptionsOnConnection);
}
this.currentHangingRequest = new GetStreamingEventsRequest(
this.session,
this,
this.subscriptions.keySet(),
this.connectionTimeout);
this.currentHangingRequest.addOnDisconnectEvent(this);
this.currentHangingRequest.internalExecute();
}
}
/**
* Called when the request is disconnected.
* @param sender The sender.
* @param args The Microsoft.Exchange.WebServices.Data.
* HangingRequestDisconnectEventArgs instance containing the event data.
*/
private void onRequestDisconnect(Object sender,
HangingRequestDisconnectEventArgs args)
{
this.internalOnDisconnect(args.getException());
}
/**
* Closes this connection so it stops receiving events
* from the server.This terminates a long-standing call to EWS.
* @exception Thrown when Close is called while not connected.
*/
public void close()
{
synchronized (this)
{
try {
this.throwIfDisposed();
this.validateConnectionState(true, Strings.
CannotCallDisconnectWithNoLiveConnection);
// Further down in the stack, this will result in a
//call to our OnRequestDisconnect event handler,
// doing the necessary cleanup.
this.currentHangingRequest.disconnect();
} catch (Exception e) {
e.printStackTrace();
}
}
}
/**
* Internal helper method called when the request disconnects.
* @param ex The exception that caused the disconnection. May be null.
*/
private void internalOnDisconnect(Exception ex)
{
if (!onDisconnect.isEmpty()) {
for (ISubscriptionErrorDelegate disconnect : onDisconnect) {
disconnect.subscriptionErrorDelegate(this,
new SubscriptionErrorEventArgs(null, ex));
}
}
this.currentHangingRequest = null;
}
/**
* Gets a value indicating whether this connection is opened
* @throws Exception
*/
public boolean getIsOpen() throws Exception
{
this.throwIfDisposed();
if (this.currentHangingRequest == null)
{
return false;
}
else
{
return this.currentHangingRequest.isConnected();
}
}
/**
* Validates the state of the connection.
* @param isConnectedExpected Value indicating whether we expect to be currently connected.
* @param errorMessage The error message.
* @throws Exception
*/
private void validateConnectionState(boolean isConnectedExpected,
String errorMessage) throws Exception
{
if ((isConnectedExpected && !this.getIsOpen()) ||
(!isConnectedExpected && this.getIsOpen()))
{
throw new ServiceLocalException(errorMessage);
}
}
/**
* Handles the service response object.
* @param response The response.
* @throws ArgumentException
*/
private void handleServiceResponseObject(Object response)
throws ArgumentException
{
GetStreamingEventsResponse gseResponse =
(GetStreamingEventsResponse)response ;
if (gseResponse == null)
{
throw new ArgumentException();
}
else
{
if (gseResponse.getResult() == ServiceResult.Success ||
gseResponse.getResult() == ServiceResult.Warning)
{
if (gseResponse.getResults().getNotifications().size() > 0)
{
// We got notifications; dole them out.
this.issueNotificationEvents(gseResponse);
}
else
{
//// This was just a heartbeat, nothing to do here.
}
}
else if (gseResponse.getResult() == ServiceResult.Error)
{
if (gseResponse.getErrorSubscriptionIds() == null ||
gseResponse.getErrorSubscriptionIds().size() == 0)
{
// General error
this.issueGeneralFailure(gseResponse);
}
else
{
// subscription-specific errors
this.issueSubscriptionFailures(gseResponse);
}
}
}
}
/**
* Issues the subscription failures.
* @param gseResponse The GetStreamingEvents response.
*/
private void issueSubscriptionFailures(GetStreamingEventsResponse
gseResponse)
{
ServiceResponseException exception =
new ServiceResponseException(gseResponse);
for(String id : gseResponse.getErrorSubscriptionIds())
{
StreamingSubscription subscription;
synchronized (this.subscriptions)
{
EwsUtilities.EwsAssert(
this.subscriptions.containsKey(id),
"StreamingSubscriptionConnection." +
"IssueSubscriptionFailures",
"Got a subscription id for a subscription " +
"not on the connection's list.");
subscription = this.subscriptions.get(id);
}
SubscriptionErrorEventArgs eventArgs =
new SubscriptionErrorEventArgs(
subscription,
exception);
if (!onSubscriptionError.isEmpty()) {
for (ISubscriptionErrorDelegate subError : onSubscriptionError) {
subError.subscriptionErrorDelegate(this, eventArgs);
}
}
if (gseResponse.getErrorCode() != ServiceError.ErrorMissedNotificationEvents)
{
// We are no longer servicing the subscription.
this.subscriptions.remove(id);
}
}
}
/**
* Issues the general failure.
* @param gseResponse The GetStreamingEvents response.
*/
private void issueGeneralFailure(GetStreamingEventsResponse gseResponse)
{
SubscriptionErrorEventArgs eventArgs = new SubscriptionErrorEventArgs(
null,
new ServiceResponseException(gseResponse));
if (!onSubscriptionError.isEmpty()) {
for (ISubscriptionErrorDelegate subError : onSubscriptionError) {
subError.subscriptionErrorDelegate(this, eventArgs);
}
}
}
/**
* Issues the notification events.
* @param gseResponse The GetStreamingEvents response.
*/
private void issueNotificationEvents(GetStreamingEventsResponse gseResponse)
{
for(GetStreamingEventsResults.NotificationGroup events : gseResponse.getResults().getNotifications())
{
StreamingSubscription subscription;
synchronized (this.subscriptions)
{
EwsUtilities.EwsAssert(
this.subscriptions.containsKey(events.subscriptionId),
"StreamingSubscriptionConnection." +
"IssueNotificationEvents",
"Got a subscription id for a subscription " +
"not on the connection's list.");
subscription = this.subscriptions.get(events.subscriptionId);
}
NotificationEventArgs eventArgs = new NotificationEventArgs(
subscription,
events.events);
if (!onNotificationEvent.isEmpty()) {
for (INotificationEventDelegate notifyEvent : onNotificationEvent) {
notifyEvent.notificationEventDelegate(this, eventArgs);
}
}
}
}
/**
* Finalizes an instance of the StreamingSubscriptionConnection class.
*/
@Override
protected void finalize() throws Throwable {
this.dispose(false);
}
/**
* Frees resources associated with this StreamingSubscriptionConnection.
*/
public void dispose()
{
this.dispose(true);
}
/**
* Performs application-defined tasks associated
* with freeing, releasing, or resetting unmanaged resources.
* @param suppressFinalizer Value indicating whether to
* suppress the garbage collector's finalizer.
*/
@SuppressWarnings("deprecation")
private void dispose(boolean suppressFinalizer)
{
if (suppressFinalizer)
{
System.runFinalizersOnExit(false);
}
synchronized (this)
{
if (!this.isDisposed)
{
if (this.currentHangingRequest != null)
{
this.currentHangingRequest = null;
}
this.subscriptions = null;
this.session = null;
this.isDisposed = true;
}
}
}
/**
* Throws if disposed.
* @throws Exception
*/
private void throwIfDisposed() throws Exception
{
if (this.isDisposed)
{
throw new Exception(this.getClass().getName());
}
}
@Override
public void handleResponseObject(Object response) throws ArgumentException {
this.handleServiceResponseObject(response);
}
@Override
public void hangingRequestDisconnectHandler(Object sender,
HangingRequestDisconnectEventArgs args) {
this.onRequestDisconnect(sender, args);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy