X-Git-Url: https://git.pterodactylus.net/?p=Sone.git;a=blobdiff_plain;f=src%2Fmain%2Fjava%2Fnet%2Fpterodactylus%2Fsone%2Fdatabase%2Fmemory%2FMemoryDatabase.java;h=df3951110ea5e2a9f89a1fbab36ce831bc6fa852;hp=eed00eab1e3607a08b69a2fdb047f7230b9d72cc;hb=90d8e2eacb9888735865d8f890d105885959ca65;hpb=60fda3f6fd8cd72151338c831f509dd8d9d0f9ff
diff --git a/src/main/java/net/pterodactylus/sone/database/memory/MemoryDatabase.java b/src/main/java/net/pterodactylus/sone/database/memory/MemoryDatabase.java
index eed00ea..df39511 100644
--- a/src/main/java/net/pterodactylus/sone/database/memory/MemoryDatabase.java
+++ b/src/main/java/net/pterodactylus/sone/database/memory/MemoryDatabase.java
@@ -1,5 +1,5 @@
/*
- * Sone - MemoryPostDatabase.java - Copyright © 2013 David Roden
+ * Sone - MemoryDatabase.java - Copyright © 2013 David Roden
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -17,28 +17,50 @@
package net.pterodactylus.sone.database.memory;
+import static com.google.common.base.Optional.fromNullable;
import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Predicates.not;
+import static com.google.common.collect.FluentIterable.from;
+import static java.util.Collections.emptyList;
+import static net.pterodactylus.sone.data.Sone.LOCAL_SONE_FILTER;
+import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
+import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
+import java.util.List;
import java.util.Map;
import java.util.Set;
-import java.util.UUID;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
+import net.pterodactylus.sone.data.Album;
+import net.pterodactylus.sone.data.Image;
import net.pterodactylus.sone.data.Post;
+import net.pterodactylus.sone.data.PostReply;
import net.pterodactylus.sone.data.Sone;
-import net.pterodactylus.sone.data.impl.AbstractPostBuilder;
-import net.pterodactylus.sone.database.PostBuilder;
+import net.pterodactylus.sone.data.impl.DefaultSoneBuilder;
+import net.pterodactylus.sone.database.Database;
+import net.pterodactylus.sone.database.DatabaseException;
import net.pterodactylus.sone.database.PostDatabase;
-import net.pterodactylus.sone.database.SoneProvider;
+import net.pterodactylus.sone.database.SoneBuilder;
+import net.pterodactylus.sone.freenet.wot.Identity;
import net.pterodactylus.util.config.Configuration;
import net.pterodactylus.util.config.ConfigurationException;
+import com.google.common.base.Function;
import com.google.common.base.Optional;
+import com.google.common.collect.ArrayListMultimap;
+import com.google.common.collect.HashMultimap;
+import com.google.common.collect.ListMultimap;
+import com.google.common.collect.Maps;
+import com.google.common.collect.Multimap;
+import com.google.common.collect.SetMultimap;
+import com.google.common.collect.SortedSetMultimap;
+import com.google.common.collect.TreeMultimap;
+import com.google.common.util.concurrent.AbstractService;
import com.google.inject.Inject;
/**
@@ -46,122 +68,306 @@ import com.google.inject.Inject;
*
* @author David âBombeâ Roden
*/
-public class MemoryPostDatabase implements PostDatabase {
+public class MemoryDatabase extends AbstractService implements Database {
/** The lock. */
private final ReadWriteLock lock = new ReentrantReadWriteLock();
- /** The Sone provider. */
- private final SoneProvider soneProvider;
+ /** The configuration. */
+ private final Configuration configuration;
+
+ private final Map identities = Maps.newHashMap();
+ private final Map sones = new HashMap();
/** All posts by their ID. */
private final Map allPosts = new HashMap();
/** All posts by their Sones. */
- private final Map> sonePosts = new HashMap>();
+ private final Multimap sonePosts = HashMultimap.create();
+ private final SetMultimap likedPostsBySone = HashMultimap.create();
+ private final SetMultimap postLikingSones = HashMultimap.create();
/** All posts by their recipient. */
- private final Map> recipientPosts = new HashMap>();
+ private final Multimap recipientPosts = HashMultimap.create();
/** Whether posts are known. */
private final Set knownPosts = new HashSet();
+ /** All post replies by their ID. */
+ private final Map allPostReplies = new HashMap();
+ private final SetMultimap likedPostRepliesBySone = HashMultimap.create();
+ private final SetMultimap postReplyLikingSones = HashMultimap.create();
+
+ /** Replies sorted by Sone. */
+ private final SortedSetMultimap sonePostReplies = TreeMultimap.create(new Comparator() {
+
+ @Override
+ public int compare(String leftString, String rightString) {
+ return leftString.compareTo(rightString);
+ }
+ }, PostReply.TIME_COMPARATOR);
+
+ /** Replies by post. */
+ private final SortedSetMultimap postReplies = TreeMultimap.create(new Comparator() {
+
+ @Override
+ public int compare(String leftString, String rightString) {
+ return leftString.compareTo(rightString);
+ }
+ }, PostReply.TIME_COMPARATOR);
+
+ /** Whether post replies are known. */
+ private final Set knownPostReplies = new HashSet();
+
+ private final Map allAlbums = new HashMap();
+ private final ListMultimap albumChildren = ArrayListMultimap.create();
+ private final ListMultimap albumImages = ArrayListMultimap.create();
+
+ private final Map allImages = new HashMap();
+
/**
* Creates a new memory database.
*
- * @param soneProvider
- * The Sone provider
+ * @param configuration
+ * The configuration for loading and saving elements
*/
@Inject
- public MemoryPostDatabase(SoneProvider soneProvider) {
- this.soneProvider = soneProvider;
+ public MemoryDatabase(Configuration configuration) {
+ this.configuration = configuration;
+ }
+
+ //
+ // DATABASE METHODS
+ //
+
+ @Override
+ public void save() throws DatabaseException {
+ saveKnownPosts();
+ saveKnownPostReplies();
+ }
+
+ //
+ // SERVICE METHODS
+ //
+
+ @Override
+ protected void doStart() {
+ loadKnownPosts();
+ loadKnownPostReplies();
+ notifyStarted();
+ }
+
+ @Override
+ protected void doStop() {
+ try {
+ save();
+ notifyStopped();
+ } catch (DatabaseException de1) {
+ notifyFailed(de1);
+ }
+ }
+
+ @Override
+ public Optional getIdentity(String identityId) {
+ lock.readLock().lock();
+ try {
+ return fromNullable(identities.get(identityId));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public void storeIdentity(Identity identitiy) {
+ lock.writeLock().lock();
+ try {
+ identities.put(identitiy.getId(), identitiy);
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public Function> getSone() {
+ return new Function>() {
+ @Override
+ public Optional apply(String soneId) {
+ return (soneId == null) ? Optional.absent() : getSone(soneId);
+ }
+ };
+ }
+
+ @Override
+ public Optional getSone(String soneId) {
+ lock.readLock().lock();
+ try {
+ return fromNullable(sones.get(soneId));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public Collection getSones() {
+ lock.readLock().lock();
+ try {
+ return Collections.unmodifiableCollection(sones.values());
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public Collection getLocalSones() {
+ lock.readLock().lock();
+ try {
+ return from(getSones()).filter(LOCAL_SONE_FILTER).toSet();
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public Collection getRemoteSones() {
+ lock.readLock().lock();
+ try {
+ return from(getSones()).filter(not(LOCAL_SONE_FILTER)).toSet();
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public SoneBuilder newSoneBuilder() {
+ return new DefaultSoneBuilder(this) {
+ @Override
+ public Sone build(Optional soneCreated) throws IllegalStateException {
+ Sone sone = super.build(soneCreated);
+ lock.writeLock().lock();
+ try {
+ sones.put(sone.getId(), sone);
+ } finally {
+ lock.writeLock().unlock();
+ }
+ return sone;
+ }
+ };
}
//
// POSTPROVIDER METHODS
//
- /**
- * {@inheritDocs}
- */
+ @Override
+ public Function> getPost() {
+ return new Function>() {
+ @Override
+ public Optional apply(String postId) {
+ return (postId == null) ? Optional.absent() : getPost(postId);
+ }
+ };
+ }
+
@Override
public Optional getPost(String postId) {
lock.readLock().lock();
try {
- return Optional.fromNullable(allPosts.get(postId));
+ return fromNullable(allPosts.get(postId));
} finally {
lock.readLock().unlock();
}
}
- /**
- * {@inheritDocs}
- */
@Override
public Collection getPosts(String soneId) {
- return new HashSet(getPostsFrom(soneId));
+ lock.readLock().lock();
+ try {
+ return new HashSet(sonePosts.get(soneId));
+ } finally {
+ lock.readLock().unlock();
+ }
}
- /**
- * {@inheritDocs}
- */
@Override
public Collection getDirectedPosts(String recipientId) {
lock.readLock().lock();
try {
Collection posts = recipientPosts.get(recipientId);
- return (posts == null) ? Collections. emptySet() : new HashSet(posts);
+ return (posts == null) ? Collections.emptySet() : new HashSet(posts);
} finally {
lock.readLock().unlock();
}
}
- //
- // POSTBUILDERFACTORY METHODS
- //
+ @Override
+ public void likePost(Post post, Sone localSone) {
+ lock.writeLock().lock();
+ try {
+ likedPostsBySone.put(localSone.getId(), post.getId());
+ postLikingSones.put(post.getId(), localSone.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
- /**
- * {@inheritDocs}
- */
@Override
- public PostBuilder newPostBuilder() {
- return new MemoryPostBuilder(soneProvider);
+ public void unlikePost(Post post, Sone localSone) {
+ lock.writeLock().lock();
+ try {
+ likedPostsBySone.remove(localSone.getId(), post.getId());
+ postLikingSones.remove(post.getId(), localSone.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ public boolean isLiked(Post post, Sone sone) {
+ lock.readLock().lock();
+ try {
+ return likedPostsBySone.containsEntry(sone.getId(), post.getId());
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public Set getLikes(Post post) {
+ lock.readLock().lock();
+ try {
+ return from(postLikingSones.get(post.getId())).transform(getSone()).transformAndConcat(this.unwrap()).toSet();
+ } finally {
+ lock.readLock().unlock();
+ }
}
//
// POSTSTORE METHODS
//
- /**
- * {@inheritDocs}
- */
@Override
public void storePost(Post post) {
checkNotNull(post, "post must not be null");
lock.writeLock().lock();
try {
allPosts.put(post.getId(), post);
- getPostsFrom(post.getSone().getId()).add(post);
+ sonePosts.put(post.getSone().getId(), post);
if (post.getRecipientId().isPresent()) {
- getPostsTo(post.getRecipientId().get()).add(post);
+ recipientPosts.put(post.getRecipientId().get(), post);
}
} finally {
lock.writeLock().unlock();
}
}
- /**
- * {@inheritDocs}
- */
@Override
public void removePost(Post post) {
checkNotNull(post, "post must not be null");
lock.writeLock().lock();
try {
allPosts.remove(post.getId());
- getPostsFrom(post.getSone().getId()).remove(post);
+ sonePosts.remove(post.getSone().getId(), post);
if (post.getRecipientId().isPresent()) {
- getPostsTo(post.getRecipientId().get()).remove(post);
+ recipientPosts.remove(post.getRecipientId().get(), post);
}
post.getSone().removePost(post);
} finally {
@@ -169,9 +375,6 @@ public class MemoryPostDatabase implements PostDatabase {
}
}
- /**
- * {@inheritDocs}
- */
@Override
public void storePosts(Sone sone, Collection posts) throws IllegalArgumentException {
checkNotNull(sone, "sone must not be null");
@@ -185,20 +388,20 @@ public class MemoryPostDatabase implements PostDatabase {
lock.writeLock().lock();
try {
/* remove all posts by the Sone. */
- getPostsFrom(sone.getId()).clear();
+ sonePosts.removeAll(sone.getId());
for (Post post : posts) {
allPosts.remove(post.getId());
if (post.getRecipientId().isPresent()) {
- getPostsTo(post.getRecipientId().get()).remove(post);
+ recipientPosts.remove(post.getRecipientId().get(), post);
}
}
/* add new posts. */
- getPostsFrom(sone.getId()).addAll(posts);
+ sonePosts.putAll(sone.getId(), posts);
for (Post post : posts) {
allPosts.put(post.getId(), post);
if (post.getRecipientId().isPresent()) {
- getPostsTo(post.getRecipientId().get()).add(post);
+ recipientPosts.put(post.getRecipientId().get(), post);
}
}
} finally {
@@ -206,20 +409,17 @@ public class MemoryPostDatabase implements PostDatabase {
}
}
- /**
- * {@inheritDocs}
- */
@Override
public void removePosts(Sone sone) {
checkNotNull(sone, "sone must not be null");
lock.writeLock().lock();
try {
/* remove all posts by the Sone. */
- getPostsFrom(sone.getId()).clear();
+ sonePosts.removeAll(sone.getId());
for (Post post : sone.getPosts()) {
allPosts.remove(post.getId());
if (post.getRecipientId().isPresent()) {
- getPostsTo(post.getRecipientId().get()).remove(post);
+ recipientPosts.remove(post.getRecipientId().get(), post);
}
}
} finally {
@@ -228,46 +428,333 @@ public class MemoryPostDatabase implements PostDatabase {
}
//
- // POSTDATABASE METHODS
+ // POSTREPLYPROVIDER METHODS
+ //
+
+ @Override
+ public Optional getPostReply(String id) {
+ lock.readLock().lock();
+ try {
+ return fromNullable(allPostReplies.get(id));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public List getReplies(String postId) {
+ lock.readLock().lock();
+ try {
+ if (!postReplies.containsKey(postId)) {
+ return emptyList();
+ }
+ return new ArrayList(postReplies.get(postId));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public void likePostReply(PostReply postReply, Sone localSone) {
+ lock.writeLock().lock();
+ try {
+ likedPostRepliesBySone.put(localSone.getId(), postReply.getId());
+ postReplyLikingSones.put(postReply.getId(), localSone.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void unlikePostReply(PostReply postReply, Sone localSone) {
+ lock.writeLock().lock();
+ try {
+ likedPostRepliesBySone.remove(localSone.getId(), postReply.getId());
+ postReplyLikingSones.remove(postReply.getId(), localSone.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public boolean isLiked(PostReply postReply, Sone sone) {
+ lock.readLock().lock();
+ try {
+ return postReplyLikingSones.containsEntry(postReply.getId(), sone.getId());
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public Set getLikes(PostReply postReply) {
+ lock.readLock().lock();
+ try {
+ return from(postReplyLikingSones.get(postReply.getId())).transform(getSone()).transformAndConcat(this.unwrap()).toSet();
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ //
+ // POSTREPLYSTORE METHODS
//
/**
- * {@inheritDocs}
+ * Returns whether the given post reply is known.
+ *
+ * @param postReply
+ * The post reply
+ * @return {@code true} if the given post reply is known, {@code false}
+ * otherwise
*/
+ public boolean isPostReplyKnown(PostReply postReply) {
+ lock.readLock().lock();
+ try {
+ return knownPostReplies.contains(postReply.getId());
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
@Override
- public void loadKnownPosts(Configuration configuration, String prefix) {
+ public void setPostReplyKnown(PostReply postReply) {
lock.writeLock().lock();
try {
- int postCounter = 0;
- while (true) {
- String knownPostId = configuration.getStringValue(prefix + postCounter++ + "/ID").getValue(null);
- if (knownPostId == null) {
- break;
- }
- knownPosts.add(knownPostId);
+ knownPostReplies.add(postReply.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void storePostReply(PostReply postReply) {
+ lock.writeLock().lock();
+ try {
+ allPostReplies.put(postReply.getId(), postReply);
+ postReplies.put(postReply.getPostId(), postReply);
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void storePostReplies(Sone sone, Collection postReplies) {
+ checkNotNull(sone, "sone must not be null");
+ /* verify that all posts are from the same Sone. */
+ for (PostReply postReply : postReplies) {
+ if (!sone.equals(postReply.getSone())) {
+ throw new IllegalArgumentException(String.format("PostReply from different Sone found: %s", postReply));
+ }
+ }
+
+ lock.writeLock().lock();
+ try {
+ /* remove all post replies of the Sone. */
+ for (PostReply postReply : getRepliesFrom(sone.getId())) {
+ removePostReply(postReply);
+ }
+ for (PostReply postReply : postReplies) {
+ allPostReplies.put(postReply.getId(), postReply);
+ sonePostReplies.put(postReply.getSone().getId(), postReply);
+ this.postReplies.put(postReply.getPostId(), postReply);
}
} finally {
lock.writeLock().unlock();
}
}
- /**
- * {@inheritDocs}
- */
@Override
- public void saveKnownPosts(Configuration configuration, String prefix) throws ConfigurationException {
+ public void removePostReply(PostReply postReply) {
+ lock.writeLock().lock();
+ try {
+ allPostReplies.remove(postReply.getId());
+ postReplies.remove(postReply.getPostId(), postReply);
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void removePostReplies(Sone sone) {
+ checkNotNull(sone, "sone must not be null");
+
+ lock.writeLock().lock();
+ try {
+ for (PostReply postReply : sone.getReplies()) {
+ removePostReply(postReply);
+ }
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ //
+ // ALBUMPROVDER METHODS
+ //
+
+ @Override
+ public Optional getAlbum(String albumId) {
lock.readLock().lock();
try {
- int postCounter = 0;
- for (String knownPostId : knownPosts) {
- configuration.getStringValue(prefix + postCounter++ + "/ID").setValue(knownPostId);
+ return fromNullable(allAlbums.get(albumId));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public List getAlbums(Album parent) {
+ lock.readLock().lock();
+ try {
+ return from(albumChildren.get(parent.getId())).transformAndConcat(getAlbum()).toList();
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public void moveUp(Album album) {
+ lock.writeLock().lock();
+ try {
+ List albums = albumChildren.get(album.getParent().getId());
+ int currentIndex = albums.indexOf(album.getId());
+ if (currentIndex == 0) {
+ return;
+ }
+ albums.remove(album.getId());
+ albums.add(currentIndex - 1, album.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void moveDown(Album album) {
+ lock.writeLock().lock();
+ try {
+ List albums = albumChildren.get(album.getParent().getId());
+ int currentIndex = albums.indexOf(album.getId());
+ if (currentIndex == (albums.size() - 1)) {
+ return;
}
- configuration.getStringValue(prefix + postCounter + "/ID").setValue(null);
+ albums.remove(album.getId());
+ albums.add(currentIndex + 1, album.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ //
+ // ALBUMSTORE METHODS
+ //
+
+ @Override
+ public void storeAlbum(Album album) {
+ lock.writeLock().lock();
+ try {
+ allAlbums.put(album.getId(), album);
+ if (!album.isRoot()) {
+ albumChildren.put(album.getParent().getId(), album.getId());
+ }
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void removeAlbum(Album album) {
+ lock.writeLock().lock();
+ try {
+ allAlbums.remove(album.getId());
+ albumChildren.remove(album.getParent().getId(), album.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ //
+ // IMAGEPROVIDER METHODS
+ //
+
+ @Override
+ public Optional getImage(String imageId) {
+ lock.readLock().lock();
+ try {
+ return fromNullable(allImages.get(imageId));
+ } finally {
+ lock.readLock().unlock();
+ }
+ }
+
+ @Override
+ public List getImages(Album parent) {
+ lock.readLock().lock();
+ try {
+ return from(albumImages.get(parent.getId())).transformAndConcat(getImage()).toList();
} finally {
lock.readLock().unlock();
}
}
+ @Override
+ public void moveUp(Image image) {
+ lock.writeLock().lock();
+ try {
+ List images = albumImages.get(image.getAlbum().getId());
+ int currentIndex = images.indexOf(image.getId());
+ if (currentIndex == 0) {
+ return;
+ }
+ images.remove(image.getId());
+ images.add(currentIndex - 1, image.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void moveDown(Image image) {
+ lock.writeLock().lock();
+ try {
+ List images = albumChildren.get(image.getAlbum().getId());
+ int currentIndex = images.indexOf(image.getId());
+ if (currentIndex == (images.size() - 1)) {
+ return;
+ }
+ images.remove(image.getId());
+ images.add(currentIndex + 1, image.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ //
+ // IMAGESTORE METHODS
+ //
+
+ @Override
+ public void storeImage(Image image) {
+ lock.writeLock().lock();
+ try {
+ allImages.put(image.getId(), image);
+ albumImages.put(image.getAlbum().getId(), image.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
+ @Override
+ public void removeImage(Image image) {
+ lock.writeLock().lock();
+ try {
+ allImages.remove(image.getId());
+ albumImages.remove(image.getAlbum().getId(), image.getId());
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
//
// PACKAGE-PRIVATE METHODS
//
@@ -276,7 +763,7 @@ public class MemoryPostDatabase implements PostDatabase {
* Returns whether the given post is known.
*
* @param post
- * The post
+ * The post
* @return {@code true} if the post is known, {@code false} otherwise
*/
boolean isPostKnown(Post post) {
@@ -292,9 +779,9 @@ public class MemoryPostDatabase implements PostDatabase {
* Sets whether the given post is known.
*
* @param post
- * The post
+ * The post
* @param known
- * {@code true} if the post is known, {@code false} otherwise
+ * {@code true} if the post is known, {@code false} otherwise
*/
void setPostKnown(Post post, boolean known) {
lock.writeLock().lock();
@@ -313,96 +800,126 @@ public class MemoryPostDatabase implements PostDatabase {
// PRIVATE METHODS
//
+ /** Loads the known posts. */
+ private void loadKnownPosts() {
+ lock.writeLock().lock();
+ try {
+ int postCounter = 0;
+ while (true) {
+ String knownPostId = configuration.getStringValue("KnownPosts/" + postCounter++ + "/ID").getValue(null);
+ if (knownPostId == null) {
+ break;
+ }
+ knownPosts.add(knownPostId);
+ }
+ } finally {
+ lock.writeLock().unlock();
+ }
+ }
+
/**
- * Gets all posts for the given Sone, creating a new collection if there is
- * none yet.
+ * Saves the known posts to the configuration.
*
- * @param soneId
- * The ID of the Sone to get the posts for
- * @return All posts
+ * @throws DatabaseException
+ * if a configuration error occurs
*/
- private Collection getPostsFrom(String soneId) {
- Collection posts = null;
+ private void saveKnownPosts() throws DatabaseException {
lock.readLock().lock();
try {
- posts = sonePosts.get(soneId);
+ int postCounter = 0;
+ for (String knownPostId : knownPosts) {
+ configuration.getStringValue("KnownPosts/" + postCounter++ + "/ID").setValue(knownPostId);
+ }
+ configuration.getStringValue("KnownPosts/" + postCounter + "/ID").setValue(null);
+ } catch (ConfigurationException ce1) {
+ throw new DatabaseException("Could not save database.", ce1);
} finally {
lock.readLock().unlock();
}
- if (posts != null) {
- return posts;
- }
-
- posts = new HashSet();
- lock.writeLock().lock();
- try {
- sonePosts.put(soneId, posts);
- } finally {
- lock.writeLock().unlock();
- }
-
- return posts;
}
/**
- * Gets all posts that are directed the given Sone, creating a new
- * collection if there is none yet.
+ * Returns all replies by the given Sone.
*
- * @param recipientId
- * The ID of the Sone to get the posts for
- * @return All posts
+ * @param id
+ * The ID of the Sone
+ * @return The post replies of the Sone, sorted by time (newest first)
*/
- private Collection getPostsTo(String recipientId) {
- Collection posts = null;
+ private Collection getRepliesFrom(String id) {
lock.readLock().lock();
try {
- posts = recipientPosts.get(recipientId);
+ if (sonePostReplies.containsKey(id)) {
+ return Collections.unmodifiableCollection(sonePostReplies.get(id));
+ }
+ return Collections.emptySet();
} finally {
lock.readLock().unlock();
}
- if (posts != null) {
- return posts;
- }
+ }
- posts = new HashSet();
+ /** Loads the known post replies. */
+ private void loadKnownPostReplies() {
lock.writeLock().lock();
try {
- recipientPosts.put(recipientId, posts);
+ int replyCounter = 0;
+ while (true) {
+ String knownReplyId = configuration.getStringValue("KnownReplies/" + replyCounter++ + "/ID").getValue(null);
+ if (knownReplyId == null) {
+ break;
+ }
+ knownPostReplies.add(knownReplyId);
+ }
} finally {
lock.writeLock().unlock();
}
-
- return posts;
}
/**
- * {@link PostBuilder} implementation that creates a {@link MemoryPost}.
+ * Saves the known post replies to the configuration.
*
- * @author David âBombeâ Roden
+ * @throws DatabaseException
+ * if a configuration error occurs
*/
- private class MemoryPostBuilder extends AbstractPostBuilder {
-
- /**
- * Creates a new memory post builder.
- *
- * @param soneProvider
- * The Sone provider
- */
- public MemoryPostBuilder(SoneProvider soneProvider) {
- super(soneProvider);
+ private void saveKnownPostReplies() throws DatabaseException {
+ lock.readLock().lock();
+ try {
+ int replyCounter = 0;
+ for (String knownReplyId : knownPostReplies) {
+ configuration.getStringValue("KnownReplies/" + replyCounter++ + "/ID").setValue(knownReplyId);
+ }
+ configuration.getStringValue("KnownReplies/" + replyCounter + "/ID").setValue(null);
+ } catch (ConfigurationException ce1) {
+ throw new DatabaseException("Could not save database.", ce1);
+ } finally {
+ lock.readLock().unlock();
}
+ }
- /**
- * {@inheritDocs}
- */
- @Override
- public Post build() throws IllegalStateException {
- validate();
- Post post = new MemoryPost(MemoryPostDatabase.this, soneProvider, randomId ? UUID.randomUUID().toString() : id, senderId, recipientId, currentTime ? System.currentTimeMillis() : time, text);
- post.setKnown(isPostKnown(post));
- return post;
- }
+ private Function> getAlbum() {
+ return new Function>() {
+ @Override
+ public Iterable apply(String input) {
+ return (input == null) ? Collections.emptyList() : getAlbum(input).asSet();
+ }
+ };
+ }
+
+ private Function> getImage() {
+ return new Function>() {
+ @Override
+ public Iterable apply(String input) {
+ return (input == null) ? Collections.emptyList() : getImage(input).asSet();
+ }
+ };
+ }
+ private static Function, Iterable> unwrap() {
+ return new Function, Iterable>() {
+ @Override
+ public Iterable apply(Optional input) {
+ return (input == null) ? Collections.emptyList() : input.asSet();
+ }
+ };
}
}