All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.anotheria.anosite.photoserver.service.storage.StorageServiceImpl Maven / Gradle / Ivy

The newest version!
package net.anotheria.anosite.photoserver.service.storage;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import net.anotheria.anoprise.metafactory.MetaFactory;
import net.anotheria.anoprise.metafactory.MetaFactoryException;
import net.anotheria.anosite.photoserver.service.storage.event.EventAnnouncer;
import net.anotheria.anosite.photoserver.service.storage.persistence.PhotoNotFoundPersistenceServiceException;
import net.anotheria.anosite.photoserver.service.storage.persistence.StoragePersistenceService;
import net.anotheria.anosite.photoserver.service.storage.persistence.StoragePersistenceServiceException;
import net.anotheria.anosite.photoserver.service.storage.persistence.album.AlbumNotFoundPersistenceServiceException;
import net.anotheria.anosite.photoserver.service.storage.persistence.album.AlbumPersistenceService;
import net.anotheria.anosite.photoserver.service.storage.persistence.album.AlbumPersistenceServiceException;
import net.anotheria.anosite.photoserver.service.storage.persistence.album.DefaultAlbumNotFoundPersistenceServiceException;
import net.anotheria.anosite.photoserver.shared.ApprovalStatus;
import net.anotheria.anosite.photoserver.shared.PhotoServerConfig;
import net.anotheria.moskito.aop.annotation.Accumulate;
import net.anotheria.moskito.aop.annotation.Accumulates;
import net.anotheria.moskito.aop.annotation.Monitor;
import net.anotheria.util.StringUtils;
import net.anotheria.util.concurrency.IdBasedLock;
import net.anotheria.util.concurrency.IdBasedLockManager;
import net.anotheria.util.concurrency.SafeIdBasedLockManager;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.MarkerFactory;

/**
 * {@link StorageService} implementation.
 *
 * @author Alexandr Bolbat
 * @version $Id: $Id
 */
@Monitor(category = "service", subsystem = "photostorage")
@Accumulates({
		@Accumulate(valueName = "Avg", intervalName = "5m"),
		@Accumulate(valueName = "Avg", intervalName = "1h"),
		@Accumulate(valueName = "Req", intervalName = "5m"),
		@Accumulate(valueName = "Req", intervalName = "1h"),
		@Accumulate(valueName = "Err", intervalName = "5m"),
		@Accumulate(valueName = "Err", intervalName = "1h"),
		@Accumulate(valueName = "Time", intervalName = "5m"),
		@Accumulate(valueName = "Time", intervalName = "1h")
})
public class StorageServiceImpl implements StorageService {

	/**
	 * Logger instance for StorageServiceImpl class.
	 */
	private static final Logger LOG = LoggerFactory.getLogger(StorageServiceImpl.class);

	/**
	 * Lock manager for safe operations.
	 */
	private static final IdBasedLockManager LOCK_MANAGER = new SafeIdBasedLockManager<>();

	/**
	 * Album lock suffix.
	 */
	private static final String ALBUM = "_ALBUM";
	/**
	 * User lock suffix.
	 */
	private static final String USER = "_USER";
	/**
	 * Photo lock suffix.
	 */
	private static final String PHOTO = "_PHOTO";

	/**
	 * {@link StoragePersistenceService} instance.
	 */
	private final StoragePersistenceService persistenceService;

	/**
	 * {@link AlbumPersistenceService} instance.
	 */
	private final AlbumPersistenceService albumPersistenceService;

	/**
	 * {@link StorageServiceCache} cache.
	 */
	private final StorageServiceCache cache;

	/**
	 * {@link PhotoServerConfig} instance.
	 */
	private final PhotoServerConfig configuration;

	/**
	 * Events announcer.
	 */
	private final EventAnnouncer announcer;

	/**
	 * Constructor.
	 */
	protected StorageServiceImpl() {
		try {
			persistenceService = MetaFactory.get(StoragePersistenceService.class);
			albumPersistenceService = MetaFactory.get(AlbumPersistenceService.class);
		} catch (MetaFactoryException mfe) {
			LOG.error(MarkerFactory.getMarker("FATAL"), "Can't init StoragePersistenceService", mfe);
			throw new RuntimeException("Can't init StoragePersistenceService", mfe);
		}
		cache = new StorageServiceCache();
		announcer = new EventAnnouncer();
		configuration = PhotoServerConfig.getInstance();
	}

	/** {@inheritDoc} */
	@Override
	public AlbumBO getAlbum(final long albumId) throws StorageServiceException {
		try {
			AlbumBO album = cache.getCachedAlbumById(albumId);
			if (album != null)
				return album;

			album = albumPersistenceService.getAlbum(albumId);
			// put to cache
			cache.updateItem(album);
			return album;
		} catch (AlbumNotFoundPersistenceServiceException e) {
			throw new AlbumNotFoundServiceException(albumId);
		} catch (AlbumPersistenceServiceException e) {
			String message = "getAlbum(" + albumId + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	@Override
	public String getAlbumOwnerId(long albumId) throws StorageServiceException {
		return cache.getAlbumOwnerId(albumId);
	}

	/** {@inheritDoc} */
	@Override
	public List getAlbums(final String userId) throws StorageServiceException {
		try {
			List result = cache.getAllAlbums(userId);
			if (result != null)
				return result;

			result = albumPersistenceService.getAlbums(userId);

			// cache data
			cache.cacheAlbums(userId, result);
			return result;
		} catch (AlbumPersistenceServiceException e) {
			String message = "getAlbums(" + userId + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	@Override
	public boolean hasPhotos(String userId) {
		if (!cache.isAllUserAlbumsLoaded(userId)) {
			List result = new ArrayList<>();
			try {
				result = albumPersistenceService.getAlbums(userId);
			} catch (AlbumPersistenceServiceException e) {
				String message = "hasPhotos(" + userId + ") fail.";
				LOG.warn(message, e);
			}
			cache.cacheAlbums(userId, result);
		}
		return cache.hasPhotos(userId);
	}

	/** {@inheritDoc} */
	@Override
	public AlbumBO getDefaultAlbum(final String userId) throws StorageServiceException {
		IdBasedLock lock = LOCK_MANAGER.obtainLock(userId + USER);
		lock.lock();
		try {
			AlbumBO defaultAlbum = cache.getDefaultAlbum(userId);
			if (defaultAlbum != null)
				return defaultAlbum;

			return albumPersistenceService.getDefaultAlbum(userId);
		} catch (DefaultAlbumNotFoundPersistenceServiceException e) {
			AlbumBO defaultAlbum = new AlbumBO();
			defaultAlbum.setUserId(userId);
			defaultAlbum.setDefault(true);

			return createAlbumInternally(defaultAlbum);
		} catch (AlbumPersistenceServiceException e) {
			String message = "getDefaultAlbum(" + userId + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public AlbumBO createAlbum(final AlbumBO album) throws StorageServiceException {
		if (album == null)
			throw new IllegalArgumentException("Null album");
		if (album.isDefault())
			throw new StorageServiceException("For retrieving default album use getDefaultAlbum(userId) method.");

		return createAlbumInternally(album);
	}

	/**
	 * Creates album.
	 *
	 * @param album
	 *            {@link AlbumBO}
	 * @return created {@link AlbumBO}
	 * @throws StorageServiceException
	 *             on errors
	 */
	private AlbumBO createAlbumInternally(final AlbumBO album) throws StorageServiceException {
		IdBasedLock lock = LOCK_MANAGER.obtainLock(album.getId() + ALBUM);
		lock.lock();
		try {
			AlbumBO created = albumPersistenceService.createAlbum(album);
			// cache
			cache.updateItem(created);
			return created;
		} catch (AlbumPersistenceServiceException e) {
			String message = "createAlbumInternally(" + album + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public AlbumBO updateAlbum(final AlbumBO album) throws StorageServiceException {
		if (album == null)
			throw new IllegalArgumentException("Null album");

		IdBasedLock lock = LOCK_MANAGER.obtainLock(album.getId() + ALBUM);
		lock.lock();
		try {
			albumPersistenceService.updateAlbum(album);

			// removing album from cache! -- afterwards it will be placed back!
			cache.removeItem(album);
			return getAlbum(album.getId());
		} catch (AlbumPersistenceServiceException e) {
			String message = "updateAlbum(" + album + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public AlbumBO removeAlbum(final long albumId) throws StorageServiceException {
		AlbumBO albumToRemove = getAlbum(albumId);
		List albumPhotos = getPhotos(albumToRemove.getUserId(), albumToRemove.getId());
		if (!albumPhotos.isEmpty())
			throw new AlbumWithPhotosServiceException(albumId);

		IdBasedLock lock = LOCK_MANAGER.obtainLock(albumId + ALBUM);
		lock.lock();
		try {
			albumPersistenceService.deleteAlbum(albumId);

			// removing album from cache!
			cache.removeItem(albumToRemove);
			return albumToRemove;
		} catch (AlbumPersistenceServiceException e) {
			String message = "removeAlbum(" + albumId + ") fail.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO getPhoto(final long photoId) throws StorageServiceException {
		try {
			PhotoBO photo = cache.getPhotoById(photoId);
			if (photo != null)
				return photo;

			photo = persistenceService.getPhoto(photoId);

			// put to cache
			cache.updateItem(photo);
			return photo;
		} catch (PhotoNotFoundPersistenceServiceException e) {
			LOG.warn("getPhoto(" + photoId + ") failed. Photo not found in persistence.", e);
			throw new PhotoNotFoundServiceException(photoId);
		} catch (StoragePersistenceServiceException e) {
			String message = "getPhoto(" + photoId + ") failed. Underlying StoragePersistenceService thrown exception.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO getDefaultPhoto(String ownerId) throws StorageServiceException {
		return getDefaultPhoto(getDefaultAlbum(ownerId));
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO getDefaultPhoto(String userId, long albumId) throws StorageServiceException {
		AlbumBO album = getAlbum(albumId);
		if (album.getUserId().equals(userId))
			return getDefaultPhoto(album);
		// this is actually more runtime exception! then Checked!
		throw new StorageServiceException("Album[" + albumId + "] does not belongs to User[" + userId + "]");

	}

	/** {@inheritDoc} */
	@Override
	public List getPhotos(final String userId, final long albumId) throws StorageServiceException {
		try {
			AlbumBO album = getAlbum(albumId);
			if (album == null)
				return Collections.emptyList();

			List result = cache.getAllAlbumPhotos(userId, albumId);
			if (result != null)
				return result;

			result = persistenceService.getUserPhotos(userId, albumId);
			cache.addAlbumPhotosToCache(userId, albumId, result);
			return result;
		} catch (StoragePersistenceServiceException e) {
			String message = "getPhotos(" + userId + ", " + albumId + ") failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public List getPhotos(final String userId, final List photoIDs) throws StorageServiceException {
		if (photoIDs == null || photoIDs.isEmpty())
			throw new IllegalArgumentException("Null photos id's");

		try {
			return persistenceService.getUserPhotos(userId, photoIDs);
		} catch (StoragePersistenceServiceException e) {
			String message = "getPhotos" + userId + ", " + photoIDs + ") failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public List getWaitingApprovalPhotos(int photosAmount) throws StorageServiceException {
		if (photosAmount < 0)
			throw new IllegalArgumentException("Illegal photos amount selected amount[" + photosAmount + "]");
		try {
			List result = new ArrayList(photosAmount);
			List fromService = persistenceService.getPhotosWithStatus(photosAmount, ApprovalStatus.WAITING_APPROVAL);

			// sort by user
			while (fromService.size() > 0 && result.size() < photosAmount) {
				PhotoBO photo = fromService.remove(0);
				String userID = photo.getUserId();
				result.add(photo);
				// get all photos of the same user
				for (Iterator i = fromService.iterator(); i.hasNext() && result.size() < photosAmount;) {
					PhotoBO toCheck = i.next();
					if (toCheck.getUserId().equals(userID)) {
						result.add(toCheck);
						i.remove();
					}
				}
			}

			return result;
		} catch (StoragePersistenceServiceException e) {
			String message = "getWaitingApprovalPhotos(" + photosAmount + ") failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public int getWaitingApprovalPhotosCount() throws StorageServiceException {
		try {
			return persistenceService.getPhotosWithStatusCount(ApprovalStatus.WAITING_APPROVAL);
		} catch (StoragePersistenceServiceException e) {
			String message = "getWaitingApprovalPhotosCount() failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO createPhoto(final PhotoBO photo) throws StorageServiceException {
		if (photo == null)
			throw new IllegalArgumentException("Null photo");

		AlbumBO photoAlbum = getAlbum(photo.getAlbumId());

		IdBasedLock lock = LOCK_MANAGER.obtainLock(photo.getId() + PHOTO);
		lock.lock();
		try {
			PhotoBO clonedPhoto = photo.clone();
			clonedPhoto.setModificationTime(System.currentTimeMillis());
			clonedPhoto.setApprovalStatus(ApprovalStatus.WAITING_APPROVAL);

			PhotoBO result = persistenceService.createPhoto(clonedPhoto);

			// put to cache
			cache.updateItem(result);
			announcer.photoCreated(result);

			// updating album photos order - adding new photo to end of the order and saving album
			photoAlbum.addPhotoToPhotoOrder(result.getId());
			updateAlbum(photoAlbum);
			return result;
		} catch (StoragePersistenceServiceException e) {
			String message = "createPhoto(" + photo + ") failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO updatePhoto(final PhotoBO photo) throws StorageServiceException {
		if (photo == null)
			throw new IllegalArgumentException("Null photo");

		IdBasedLock lock = LOCK_MANAGER.obtainLock(photo.getId() + PHOTO);
		lock.lock();
		try {
			PhotoBO oldPhoto = getPhoto(photo.getId());
			photo.setModificationTime(System.currentTimeMillis());
			persistenceService.updatePhoto(photo);

			// remove photo from cache!
			cache.removeItem(photo);

			PhotoBO result = getPhoto(photo.getId());
			announcer.photoUpdated(result, oldPhoto);
			return result;
		} catch (PhotoNotFoundPersistenceServiceException e) {
			LOG.warn("updatePhoto(" + photo + ") failed. Photo not found in persistence.", e);
			throw new PhotoNotFoundServiceException(photo.getId());
		} catch (StoragePersistenceServiceException e) {
			String message = "updatePhoto(" + photo + ") failed. Underlying StoragePersistenceService failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public synchronized void updatePhotoApprovalStatuses(final Map statuses) throws StorageServiceException {
		if (statuses == null || statuses.isEmpty())
			throw new IllegalArgumentException("Illegal statuses, incoming param");
		try {
			// map with photo id to UserId mapping!
			Map photoIdToUserId = new HashMap<>();
			// map with current statuses of photos!
			Map previousStatuses = new HashMap<>();
			// real status update map! We should not execute update - if status which should be set - is already there :)
			Map statusesToUpdate = new HashMap<>();

			for (Map.Entry entry : statuses.entrySet()) {
				PhotoBO photo = getPhoto(entry.getKey());
				if (photo.getApprovalStatus() != entry.getValue()) {
					photoIdToUserId.put(entry.getKey(), photo.getUserId());
					previousStatuses.put(entry.getKey(), photo.getApprovalStatus());
					statusesToUpdate.put(entry.getKey(), entry.getValue());
				}
			}
			if (statusesToUpdate.isEmpty())
				return;

			persistenceService.updatePhotoApprovalStatuses(statusesToUpdate);

			// update data in cache!!!
			cache.updatePhotoApprovalsStatuses(statusesToUpdate);

			// sending events!
			for (Map.Entry entry : statuses.entrySet()) {
				String userId = photoIdToUserId.get(entry.getKey());
				ApprovalStatus previous = previousStatuses.get(entry.getKey());
				if (!StringUtils.isEmpty(userId) && previous != null && entry.getValue() != null)
					announcer.photoStatusChanged(userId, entry.getKey(), entry.getValue(), previous);
			}

		} catch (StoragePersistenceServiceException e) {
			String message = "updatePhotoApprovalStatuses(" + statuses + ") failed. Underlying StoragePersistenceService failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public Map getAlbumPhotosApprovalStatus(long albumId) throws StorageServiceException {
		try {
			// try from cache!
			AlbumBO album = getAlbum(albumId);

			List albumPhotos = cache.getAllAlbumPhotos(album.getUserId(), albumId);
			if (albumPhotos != null) {
				// build result
				Map result = new HashMap<>();
				for (PhotoBO photo : albumPhotos)
					result.put(photo.getId(), photo.getApprovalStatus());

				return result;
			}

			return persistenceService.getAlbumPhotosApprovalStatus(albumId);
		} catch (StoragePersistenceServiceException e) {
			String message = "getAlbumPhotosApprovalStatus(" + albumId + ") failed. Underlying StoragePersistenceService failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		}
	}

	/** {@inheritDoc} */
	@Override
	public void removePhoto(final long photoId) throws StorageServiceException {
		IdBasedLock lock = LOCK_MANAGER.obtainLock(photoId + PHOTO);
		lock.lock();
		try {
			final PhotoBO photo = persistenceService.getPhoto(photoId);
			persistenceService.deletePhoto(photoId);

			cache.removeItem(photo);
			announcer.photoDeleted(photoId, photo.getUserId());

			// removing photo from photo album photos order list
			AlbumBO photoAlbum = getAlbum(photo.getAlbumId());
			photoAlbum.removePhotofromPhotoOrder(photoId);
			updateAlbum(photoAlbum);
		} catch (PhotoNotFoundPersistenceServiceException e) {
			LOG.warn("removePhoto(" + photoId + ") failed. Photo not found in persistence.", e);
			throw new PhotoNotFoundServiceException(photoId);
		} catch (StoragePersistenceServiceException e) {
			String message = "removePhoto(" + photoId + ") failed. Underlying StoragePersistenceService failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/** {@inheritDoc} */
	@Override
	public PhotoBO movePhoto(long photoId, long newAlbumId) throws StorageServiceException {
		IdBasedLock lock = LOCK_MANAGER.obtainLock(photoId + PHOTO);
		lock.lock();
		try {
			PhotoBO oldPhoto = getPhoto(photoId);
			AlbumBO newAlbum = getAlbum(newAlbumId);

			persistenceService.movePhoto(photoId, newAlbum.getId(), System.currentTimeMillis());

			// remove photo from cache!
			cache.removeItem(oldPhoto);

			PhotoBO result = getPhoto(photoId);
			announcer.photoUpdated(result, oldPhoto);
			return result;
		} catch (PhotoNotFoundPersistenceServiceException e) {
			LOG.warn("movePhoto(" + photoId + ") failed. Photo not found in persistence.", e);
			throw new PhotoNotFoundServiceException(photoId);
		} catch (StoragePersistenceServiceException e) {
			String message = "movePhoto(" + photoId + ") failed. Underlying StoragePersistenceService failed.";
			LOG.warn(message, e);
			throw new StorageServiceException(message, e);
		} finally {
			lock.unlock();
		}
	}

	/**
	 * Return id of the Default photo, which belongs to selected album.
	 *
	 * @param album
	 *            {@link AlbumBO}
	 * @return id of the default photo if such exists
	 * @throws StorageServiceException
	 *             DefaultPhotoNotFoundServiceException - if no such photo found, or on other errors
	 */
	private PhotoBO getDefaultPhoto(AlbumBO album) throws StorageServiceException {
		List orderedPhotoIdsList = album.getPhotosOrder();

		// reading ordered photos! Searching and returning Default photo - if such will be found!
		for (Long photoId : orderedPhotoIdsList) {
			// just an additional NULL check!
			if (photoId == null) {
				LOG.warn("NULL photo id detected inside ordered photos list! album[" + album.getId() + "]");
				continue;
			}
			try {
				PhotoBO photo = getPhoto(photoId);
				// if approving is disabled - returning photo!
				if (!configuration.isPhotoApprovingEnabled())
					return photo;
				// if Approving is Enabled and photo is Approved returning it
				if (configuration.isPhotoApprovingEnabled() && ApprovalStatus.APPROVED == photo.getApprovalStatus())
					return photo;

			} catch (PhotoNotFoundServiceException phfSe) {
				LOG.warn("Corrupted data detected! photo[" + photoId + "] does not exists! But present inside photos order. SKIPPING!" + phfSe.getMessage());
			}
		}

		List allPhotos = getPhotos(album.getUserId(), album.getId());
		// checking that some photos really exists!
		if (allPhotos.isEmpty())
			throw new DefaultPhotoNotFoundServiceException(album.getId(), " Album does not contains photos.");

		if (!configuration.isPhotoApprovingEnabled()) {
			LOG.debug("PhotoApproving is disabled! there is nothing to filter out!");
			return allPhotos.get(0);
		}

		// looking for first approved photo!
		for (PhotoBO photo : allPhotos)
			if (ApprovalStatus.APPROVED == photo.getApprovalStatus())
				return photo;
		// Exception in case - When there is no any approved photo found!
		throw new DefaultPhotoNotFoundServiceException(album.getId(), " Album does not contains approved photos.");
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy