/** * 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.media; import java.util.concurrent.ExecutionException; 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.mediagroup.MediaGroup; import javax.media.mscontrol.mediagroup.Recorder; 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; public class RecordingImpl<T extends EventSource> implements Recording<T> { private static final Logger LOG = Logger.getLogger(RecordingImpl.class); protected MediaGroup _group; protected SettableResultFuture<RecordCompleteEvent<T>> _future = new SettableResultFuture<RecordCompleteEvent<T>>(); final Lock lock = new ReentrantLock(); private Condition pauseActionResult = lock.newCondition(); private Condition resumeActionResult = lock.newCondition(); private boolean _normalDisconnected = false; protected RecordingImpl(final MediaGroup group) { _group = group; } protected void done(final RecordCompleteEvent<T> event) { _future.setResult(event); } protected void done(final MediaException exception) { _future.setException(exception); } protected boolean paused = false; protected boolean pauseResult = false; @Override public void pause() { lock.lock(); try { if (!_future.isDone() && !paused) { _group.triggerAction(Recorder.PAUSE); 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) { _group.triggerAction(Recorder.RESUME); while (!resumeResult&& !_future.isDone()) { try { resumeActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } resumeResult = false; } } finally { lock.unlock(); } } protected boolean resumeResult = false; protected void resumeActionDone() { lock.lock(); resumeResult = true; paused = false; try { resumeActionResult.signalAll(); } finally { lock.unlock(); } } @Override public void stop() { if (!_future.isDone()) { _group.triggerAction(Recorder.STOP); try { _future.get(10, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } catch (ExecutionException e) { // ignore } catch (TimeoutException e) { LOG.error("Timeout when waiting record complete for MediaGroup " + _group, e); } } } @Override public boolean cancel(final boolean mayInterruptIfRunning) { return _future.cancel(mayInterruptIfRunning); } @Override public RecordCompleteEvent<T> get() throws InterruptedException, ExecutionException { return _future.get(); } @Override public RecordCompleteEvent<T> get(final long timeout, final 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(); } public synchronized boolean isPending() { return !_future.isDone(); } public void normalDisconnect(boolean normal) { _normalDisconnected = true; } public boolean isNormalDisconnect() { return _normalDisconnected; } }