package com.voxeo.moho.media; import java.util.concurrent.ExecutionException; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import java.util.concurrent.locks.Condition; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; import javax.media.mscontrol.networkconnection.NetworkConnection; import org.apache.log4j.Logger; import com.voxeo.moho.MediaException; import com.voxeo.moho.common.util.SettableResultFuture; import com.voxeo.moho.event.EventSource; import com.voxeo.moho.event.RecordCompleteEvent; import com.voxeo.moho.media.GenericMediaService.MaxCallRecordDurationTask; import com.voxeo.moho.media.dialect.MediaDialect; public class CallRecordingImpl<T extends EventSource> implements Recording<T> { private static final Logger LOG = Logger.getLogger(CallRecordingImpl.class); protected NetworkConnection _nc; protected MediaDialect _dialect; protected boolean _normalDisconnected = false; protected ScheduledFuture maxDurationTimerFuture; protected MaxCallRecordDurationTask maxDurationTask; protected SettableResultFuture<RecordCompleteEvent<T>> _future = new SettableResultFuture<RecordCompleteEvent<T>>(); protected boolean maxDurationStop; final Lock lock = new ReentrantLock(); private Condition pauseActionResult = lock.newCondition(); private Condition resumeActionResult = lock.newCondition(); protected boolean paused = false; protected boolean pauseResult = false; protected boolean resumeResult = false; public CallRecordingImpl(NetworkConnection nc, MediaDialect dialect) { super(); this._nc = nc; this._dialect = dialect; } protected void done(final RecordCompleteEvent<T> event) { _future.setResult(event); } protected void done(final MediaException exception) { _future.setException(exception); } @Override public void stop() { if (!_future.isDone()) { _dialect.stopCallRecord(_nc); try { _future.get(10, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } catch (ExecutionException e) { // ignore } catch (TimeoutException e) { LOG.error("Timeout when waiting call record complete for NetworkConnection " + _nc, e); } } } @Override public boolean cancel(boolean mayInterruptIfRunning) { return _future.cancel(mayInterruptIfRunning); } @Override public RecordCompleteEvent<T> get() throws InterruptedException, ExecutionException { return _future.get(); } @Override public RecordCompleteEvent<T> get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return _future.get(timeout, unit); } @Override public boolean isCancelled() { return _future.isCancelled(); } @Override public boolean isDone() { return _future.isDone(); } @Override public void pause() { lock.lock(); try { if (!_future.isDone() && !paused) { _dialect.pauseCallRecord(_nc); while (!pauseResult && !_future.isDone()) { try { pauseActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } pauseResult = false; } } finally { lock.unlock(); } } protected void pauseActionDone() { lock.lock(); pauseResult = true; paused = true; try { pauseActionResult.signalAll(); } finally { lock.unlock(); } } @Override public void resume() { lock.lock(); try { if (!_future.isDone() && paused) { _dialect.resumeCallRecord(_nc); while (!resumeResult && !_future.isDone()) { try { resumeActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } resumeResult = false; } } finally { lock.unlock(); } } protected void resumeActionDone() { lock.lock(); resumeResult = true; paused = false; try { resumeActionResult.signalAll(); } finally { lock.unlock(); } } public void normalDisconnect(boolean normal) { _normalDisconnected = true; } public boolean isNormalDisconnect() { return _normalDisconnected; } public ScheduledFuture getMaxDurationTimerFuture() { return maxDurationTimerFuture; } public void setMaxDurationTimerFuture(ScheduledFuture maxDurationTimerFuture) { this.maxDurationTimerFuture = maxDurationTimerFuture; } public MaxCallRecordDurationTask getMaxDurationTask() { return maxDurationTask; } public void setMaxDurationTask(MaxCallRecordDurationTask maxDurationTask) { this.maxDurationTask = maxDurationTask; } public boolean isMaxDurationStop() { return maxDurationStop; } public void setMaxDurationStop(boolean maxDurationStop) { this.maxDurationStop = maxDurationStop; } }