/**
* Copyright 2010 Google Inc.
*
* 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 org.waveprotocol.box.server.waveserver;
import com.google.common.base.Function;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Iterators;
import com.google.common.collect.MapMaker;
import com.google.common.util.concurrent.ListenableFuture;
import org.waveprotocol.box.server.persistence.PersistenceException;
import org.waveprotocol.wave.model.id.WaveId;
import org.waveprotocol.wave.model.id.WaveletId;
import org.waveprotocol.wave.model.id.WaveletName;
import java.util.Iterator;
import java.util.concurrent.ConcurrentMap;
/**
* The wavelets in a wave.
*
* @author soren@google.com (Soren Lassen)
*/
final class Wave implements Iterable<WaveletContainer> {
private class WaveletCreator<T extends WaveletContainer> implements Function<WaveletId, T> {
private final WaveletContainer.Factory<T> factory;
private final String waveDomain;
public WaveletCreator(WaveletContainer.Factory<T> factory, String waveDomain) {
this.factory = factory;
this.waveDomain = waveDomain;
}
@Override
public T apply(WaveletId waveletId) {
return factory.create(notifiee, WaveletName.of(waveId, waveletId), waveDomain);
}
}
private final WaveId waveId;
/** Future providing already-existing wavelets in storage. */
private final ListenableFuture<ImmutableSet<WaveletId>> lookedupWavelets;
private final ConcurrentMap<WaveletId, LocalWaveletContainer> localWavelets;
private final ConcurrentMap<WaveletId, RemoteWaveletContainer> remoteWavelets;
private final WaveletNotificationSubscriber notifiee;
/**
* Creates a wave. The {@code lookupWavelets} future is examined only when a
* query is first made.
*/
public Wave(WaveId waveId,
ListenableFuture<ImmutableSet<WaveletId>> lookedupWavelets,
WaveletNotificationSubscriber notifiee, LocalWaveletContainer.Factory localFactory,
RemoteWaveletContainer.Factory remoteFactory,
String waveDomain) {
this.waveId = waveId;
this.lookedupWavelets = lookedupWavelets;
this.notifiee = notifiee;
this.localWavelets = new MapMaker().makeComputingMap(
new WaveletCreator<LocalWaveletContainer>(localFactory, waveDomain));
this.remoteWavelets = new MapMaker().makeComputingMap(
new WaveletCreator<RemoteWaveletContainer>(remoteFactory, waveDomain));
}
@Override
public Iterator<WaveletContainer> iterator() {
return Iterators.unmodifiableIterator(
Iterables.concat(localWavelets.values(), remoteWavelets.values()).iterator());
}
LocalWaveletContainer getLocalWavelet(WaveletId waveletId)
throws WaveletStateException {
return getWavelet(waveletId, localWavelets);
}
RemoteWaveletContainer getRemoteWavelet(WaveletId waveletId)
throws WaveletStateException {
return getWavelet(waveletId, remoteWavelets);
}
LocalWaveletContainer getOrCreateLocalWavelet(WaveletId waveletId) {
return localWavelets.get(waveletId);
}
RemoteWaveletContainer getOrCreateRemoteWavelet(WaveletId waveletId) {
return remoteWavelets.get(waveletId);
}
private <T extends WaveletContainer> T getWavelet(WaveletId waveletId,
ConcurrentMap<WaveletId, T> waveletsMap) throws WaveletStateException {
ImmutableSet<WaveletId> storedWavelets;
try {
storedWavelets =
FutureUtil.getResultOrPropagateException(lookedupWavelets, PersistenceException.class);
} catch (PersistenceException e) {
throw new WaveletStateException(
"Failed to lookup wavelet " + WaveletName.of(waveId, waveletId), e);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new WaveletStateException(
"Interrupted looking up wavelet " + WaveletName.of(waveId, waveletId), e);
}
// Since waveletsMap is a computing map, we must call containsKey(waveletId)
// to tell if waveletId is mapped, we cannot test if get(waveletId) returns null.
if (storedWavelets != null && !storedWavelets.contains(waveletId)
&& !waveletsMap.containsKey(waveletId)) {
return null;
} else {
T wavelet = waveletsMap.get(waveletId);
Preconditions.checkNotNull(wavelet, "computingMap returned null");
return wavelet;
}
}
ListenableFuture<ImmutableSet<WaveletId>> getLookedupWavelets() {
return lookedupWavelets;
}
}