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

org.ldp4j.application.kernel.session.WriteSessionService Maven / Gradle / Ivy

The newest version!
/**
 * #-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=#
 *   This file is part of the LDP4j Project:
 *     http://www.ldp4j.org/
 *
 *   Center for Open Middleware
 *     http://www.centeropenmiddleware.com/
 * #-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=#
 *   Copyright (C) 2014-2016 Center for Open Middleware.
 * #-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=#
 *   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.
 * #-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=#
 *   Artifact    : org.ldp4j.framework:ldp4j-application-kernel-core:0.2.2
 *   Bundle      : ldp4j-application-kernel-core-0.2.2.jar
 * #-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=#
 */
package org.ldp4j.application.kernel.session;

import static com.google.common.base.Preconditions.checkArgument;

import java.net.URI;
import java.util.Date;
import java.util.UUID;

import org.ldp4j.application.engine.context.EntityTag;
import org.ldp4j.application.ext.ResourceHandler;
import org.ldp4j.application.kernel.endpoint.Endpoint;
import org.ldp4j.application.kernel.endpoint.EndpointCreationException;
import org.ldp4j.application.kernel.endpoint.EndpointManagementService;
import org.ldp4j.application.kernel.endpoint.EndpointNotFoundException;
import org.ldp4j.application.kernel.resource.Resource;
import org.ldp4j.application.kernel.resource.ResourceId;
import org.ldp4j.application.kernel.resource.ResourceRepository;
import org.ldp4j.application.kernel.service.Service;
import org.ldp4j.application.kernel.service.ServiceBuilder;
import org.ldp4j.application.kernel.spi.RuntimeDelegate;
import org.ldp4j.application.kernel.template.TemplateManagementService;
import org.ldp4j.application.kernel.transaction.TransactionManager;
import org.ldp4j.application.session.ResourceSnapshot;
import org.ldp4j.application.session.WriteSession;
import org.ldp4j.application.session.WriteSessionException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public final class WriteSessionService implements Service {

	private static class WriteSessionServiceBuilder extends ServiceBuilder {

		private WriteSessionServiceBuilder() {
			super(WriteSessionService.class);
		}

		@Override
		public WriteSessionService build() {
			return
				new WriteSessionService(
					service(TemplateManagementService.class),
					service(EndpointManagementService.class));
		}

	}

	private final class ResourceProcessor implements UnitOfWork.Visitor {

		private final DelegatedWriteSession session;
		private final Date lastModified;

		private ResourceProcessor(final DelegatedWriteSession session) {
			this.lastModified = session.getLastModified();
			this.session = session;
		}

		@Override
		public void visitNew(final DelegatedResourceSnapshot obj) {
			createResource(obj.delegate(),this.lastModified,this.session.getDesiredPath(obj),this.session.getIndirectId(obj));
		}

		@Override
		public void visitDirty(final DelegatedResourceSnapshot obj) {
			modifyResource(obj.delegate(),this.lastModified);
		}

		@Override
		public void visitDeleted(final DelegatedResourceSnapshot obj) {
			deleteResource(obj.delegate(),this.lastModified);
		}
	}

	private final class RollbackResourceProcessor implements UnitOfWork.Visitor {

		private RollbackResourceProcessor() {
		}

		@Override
		public void visitNew(final DelegatedResourceSnapshot obj) {
			LOGGER.trace("Discarding creation of resource {}",obj.resourceId());
		}

		@Override
		public void visitDirty(final DelegatedResourceSnapshot obj) {
			LOGGER.trace("Discarding modifications for resource {}",obj.resourceId());
		}

		@Override
		public void visitDeleted(final DelegatedResourceSnapshot obj) {
			LOGGER.trace("Discarding deletion of resource {}",obj.resourceId());
		}
	}

	private static final Logger LOGGER=LoggerFactory.getLogger(WriteSessionService.class);

	private final EndpointManagementService endpointManagementService;

	private final TemplateManagementService templateManagementService;

	private final ResourceRepository resourceRepository;

	private final TransactionManager transactionManager;

	private WriteSessionService(final TemplateManagementService templateManagementService, final EndpointManagementService endointManagementService) {
		this.templateManagementService = templateManagementService;
		this.endpointManagementService = endointManagementService;
		this.resourceRepository=RuntimeDelegate.getInstance().getResourceRepository();
		this.transactionManager=RuntimeDelegate.getInstance().getTransactionManager();
	}

	public WriteSession createSession(final WriteSessionConfiguration configuration) {
		UnitOfWork.newCurrent();
		logLifecycleMessage("Created write session: %s",configuration);
		return new DelegatedWriteSession(configuration,this);
	}

	void terminateSession(final DelegatedWriteSession session) {
		try {
			switch(session.status()) {
				case ACTIVE:
					logLifecycleMessage("Force termination of active session...");
					session.discardChanges();
					break;
				case ABORTED:
					// Nothing to do
					break;
				case COMPLETED:
					// Nothing to do
					break;
				default:
					final String errorMessage = "Unsupported status "+session.status();
					LOGGER.error(errorMessage);
					throw new IllegalStateException(errorMessage);
			}
		} catch (final WriteSessionException e) {
			LOGGER.error("Could not force termination of active session",e);
		} finally {
			UnitOfWork.setCurrent(null);
		}
	}

	public ResourceSnapshot attach(final WriteSession writeSession, final Resource resource, final Class handlerClass) {
		return writeSession.find(ResourceSnapshot.class, resource.id().name(), handlerClass);
	}

	public Resource detach(final WriteSession writeSession, final ResourceSnapshot snapshot) {
		checkArgument(writeSession instanceof DelegatedWriteSession,"Invalid session");
		checkArgument(snapshot instanceof DelegatedResourceSnapshot,"Unknown resource '%s'",snapshot.name());
		final DelegatedResourceSnapshot delegatedSnapshot=(DelegatedResourceSnapshot)snapshot;
		return delegatedSnapshot.delegate();
	}

	void commitSession(final DelegatedWriteSession session) {
		logLifecycleMessage("Commiting session...");
		UnitOfWork.getCurrent().accept(new ResourceProcessor(session));
		this.transactionManager.currentTransaction().commit();
	}

	void rollbackSession(final DelegatedWriteSession session) {
		logLifecycleMessage("Rolling back session...");
		UnitOfWork.getCurrent().accept(new RollbackResourceProcessor());
		this.transactionManager.currentTransaction().rollback();
	}

	Resource resourceOfId(final ResourceId resourceId) {
		return this.resourceRepository.resourceById(resourceId,Resource.class);
	}

	TemplateManagementService templateManagementService() {
		return this.templateManagementService;
	}

	private void logLifecycleMessage(final String msg, final Object... args) {
		if(LOGGER.isDebugEnabled()) {
			LOGGER.debug(String.format(msg,args));
		}
	}

	/**
	 * TODO: Devise a mechanism for generating the entity tags using meaningful
	 * information from the client
	 */
	private EntityTag generateEntityTag(final Resource resource) { // NOSONAR
		return EntityTag.createStrong(UUID.randomUUID().toString());
	}

	private void createResource(final Resource resource, final Date lastModified, final String relativePath, final URI indirectId) {
		try {
			resource.setIndirectId(indirectId);
			this.resourceRepository.add(resource);
			final Endpoint newEndpoint=
				this.endpointManagementService.
					createEndpointForResource(
						resource,
						relativePath,
						generateEntityTag(resource),
						lastModified);
			if(LOGGER.isTraceEnabled()) {
				LOGGER.trace("Created "+resource);
				LOGGER.trace("Created "+newEndpoint);
			}
		} catch (final EndpointCreationException e) {
			throw new IllegalStateException(e);
		}
	}

	private void modifyResource(final Resource resource, final Date lastModified) {
		try {
			final Endpoint endpoint =
				this.endpointManagementService.
					modifyResourceEndpoint(
						resource,
						generateEntityTag(resource),
						lastModified);
			if(LOGGER.isTraceEnabled()) {
				LOGGER.trace("Modified "+resource);
				LOGGER.trace("Modified "+endpoint);
			}
		} catch (final EndpointNotFoundException e) {
			throw new IllegalStateException(e);
		}
	}

	private void deleteResource(final Resource resource, final Date lastModified) {
		try {
			this.resourceRepository.remove(resource);
			final Endpoint endpoint =
				this.endpointManagementService.
					deleteResourceEndpoint(
						resource,
						lastModified);
			if(LOGGER.isTraceEnabled()) {
				LOGGER.trace("Deleted "+resource);
				LOGGER.trace("Deleted "+endpoint);
			}
		} catch (final EndpointNotFoundException e) {
			throw new IllegalStateException(e);
		}
	}

	public static ServiceBuilder serviceBuilder() {
		return new WriteSessionServiceBuilder();
	}

	public static WriteSessionService defaultService() {
		return serviceBuilder().build();
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy