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

net.shibboleth.ext.spring.resource.SVNResource Maven / Gradle / Ivy

/*
 * Licensed to the University Corporation for Advanced Internet Development, 
 * Inc. (UCAID) under one or more contributor license agreements.  See the 
 * NOTICE file distributed with this work for additional information regarding
 * copyright ownership. The UCAID licenses this file to You 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 net.shibboleth.ext.spring.resource;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;

import net.shibboleth.utilities.java.support.component.AbstractIdentifiedInitializableComponent;
import net.shibboleth.utilities.java.support.primitive.StringSupport;

import org.joda.time.DateTime;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.BeanCreationException;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.core.io.Resource;
import org.tmatesoft.svn.core.SVNDepth;
import org.tmatesoft.svn.core.SVNException;
import org.tmatesoft.svn.core.SVNURL;
import org.tmatesoft.svn.core.internal.io.dav.DAVRepositoryFactory;
import org.tmatesoft.svn.core.internal.io.fs.FSRepositoryFactory;
import org.tmatesoft.svn.core.internal.io.svn.SVNRepositoryFactoryImpl;
import org.tmatesoft.svn.core.wc.ISVNStatusHandler;
import org.tmatesoft.svn.core.wc.SVNClientManager;
import org.tmatesoft.svn.core.wc.SVNRevision;
import org.tmatesoft.svn.core.wc.SVNStatus;

/**
 * Implementation of a Spring {@link Resource} that communications with a Subversion server (via a spool directory).
 * 
 * This resource will fetch the given resource as follows:
 * 
    *
  • If the revision is a positive number the resource will fetch the resource once during construction time and will * never attempt to fetch it again.
  • *
  • If the revision number is zero or less, signaling the HEAD revision, every call this resource will cause the * resource to check to see if the current working copy is the same as the revision in the remote repository. If it is * not the new revision will be retrieved.
  • *
* * The behavior of multiple {@link SVNResource} operating on the same local copy are undefined. */ public class SVNResource extends AbstractIdentifiedInitializableComponent implements Resource, BeanNameAware, InitializingBean, net.shibboleth.utilities.java.support.resource.Resource { /** Class logger. */ private final Logger log = LoggerFactory.getLogger(SVNResource.class); /** SVN Client manager. */ private final SVNClientManager clientManager; /** URL to the remote repository. */ private SVNURL remoteRepository; /** Directory where the working copy will be kept. */ private File workingCopyDirectory; /** Revision of the working copy. */ private SVNRevision retrievalRevision; /** File, within the working copy, represented by this resource. */ private String resourceFileName; /** Time the resource file was last modified. */ private DateTime lastModified; /** * Constructor. * * @param svnClientMgr manager used to create SVN clients * @param repositoryUrl URL of the remote repository * @param workingCopy directory that will serve as the root of the local working copy * @param workingRevision revision of the resource to retrieve or -1 for HEAD revision * @param resourceFile file, within the working copy, represented by this resource * * @throws BeanCreationException thrown if there is a problem initializing the SVN resource */ public SVNResource(SVNClientManager svnClientMgr, SVNURL repositoryUrl, File workingCopy, long workingRevision, String resourceFile) { DAVRepositoryFactory.setup(); SVNRepositoryFactoryImpl.setup(); FSRepositoryFactory.setup(); if (svnClientMgr == null) { log.error("SVN client manager may not be null"); throw new BeanCreationException("SVN client manager may not be null"); } clientManager = svnClientMgr; if (repositoryUrl == null) { throw new BeanCreationException("SVN repository URL may not be null"); } remoteRepository = repositoryUrl; try { checkWorkingCopyDirectory(workingCopy); workingCopyDirectory = workingCopy; } catch (IOException e) { throw new BeanCreationException(e.getMessage()); } if (workingRevision < 0) { this.retrievalRevision = SVNRevision.HEAD; } else { this.retrievalRevision = SVNRevision.create(workingRevision); } setFilename(resourceFile); try { checkoutOrUpdateResource(); if (!getFile().exists()) { log.error("Resource file " + resourceFile + " does not exist in SVN working copy directory " + workingCopy.getAbsolutePath()); } } catch (IOException e) { throw new BeanCreationException(e.getMessage()); } } /** * {@inheritDoc} */ @Override public String getFilename() { return resourceFileName; } /** * Set the resourceFileName. * * @param fileName the name */ public void setFilename(String fileName) { resourceFileName = StringSupport.trimOrNull(fileName); if (resourceFileName == null) { log.error("SVN working copy resource file name may not be null or empty"); throw new BeanCreationException("SVN working copy resource file name may not be null or empty"); } } /** * Helper function to generate the full path for injecting into URLs and URIs. * * @return the directory path (from svn) pus the file. */ protected String getFullPath() { final StringBuffer buffer = new StringBuffer(remoteRepository.getPath().length() + 1 + getFilename().length()); buffer.append(remoteRepository.getPath()).append('/').append(getFilename()); return buffer.toString(); } /** * Helper function to return a complex scheme - a conjunction of 'svn' and the underlying protocol. * * @return 'svn+'protocol */ protected String getProtocol() { final StringBuffer buffer = new StringBuffer(4 + remoteRepository.getProtocol().length()); buffer.append("svn+").append(remoteRepository.getProtocol()); return buffer.toString(); } /** * Checks that the given file exists, or can be created, is a directory, and is read/writable by this process. * * @param directory the directory to check * * @throws IOException thrown if the file is invalid */ protected void checkWorkingCopyDirectory(File directory) throws IOException { if (directory == null) { log.error("SVN working copy directory cannot be null"); throw new IOException("SVN working copy directory cannot be null"); } if (!directory.exists()) { boolean created = directory.mkdirs(); if (!created) { final String msg = "SVN working copy directory " + directory.getAbsolutePath() + " does not exist and could not be created"; log.error(msg); throw new IOException(msg); } } if (!directory.isDirectory()) { final String msg = "SVN working copy location " + directory.getAbsolutePath() + " is not a directory"; log.error(msg); throw new IOException(msg); } if (!directory.canRead()) { final String msg = "SVN working copy directory " + directory.getAbsolutePath() + " cannot be read by this process"; log.error(msg); throw new IOException(msg); } if (!directory.canWrite()) { final String msg = "SVN working copy directory " + directory.getAbsolutePath() + " cannot be written to by this process"; log.error(msg); throw new IOException(msg); } } /** * Checks out the resource specified by the {@link #remoteRepository} in to the working copy * {@link #workingCopyDirectory}. If the working copy is empty than an SVN checkout is performed if the working copy * already exists then an SVN update is performed. * * @throws IOException thrown if there is a problem communicating with the remote repository, the revision does not * exist, or the working copy is unusable */ protected void checkoutOrUpdateResource() throws IOException { log.debug("checking out or updating working copy"); SVNRevision newRevision; if (!workingCopyDirectoryExists()) { log.debug("Working copy does not yet exist, checking it out to {}", workingCopyDirectory.getAbsolutePath()); newRevision = checkoutResourceDirectory(); } else { if (retrievalRevision != SVNRevision.HEAD) { lastModified = getLastModificationForRevision(SVNRevision.HEAD); log.debug("Working copy exists and version is pegged at {}, no need to update", retrievalRevision.toString()); return; } log.debug("Working copy exists, updating to latest version."); newRevision = updateResourceDirectory(); } if (getFile().exists()) { log.debug("Determining last modification date of revision {}", newRevision.getNumber()); lastModified = getLastModificationForRevision(newRevision); } } /** * Checks to see if the working copy directory exists. * * @return true if the working copy directory exists, false otherwise */ private boolean workingCopyDirectoryExists() { final File svnMetadataDir = new File(workingCopyDirectory, ".svn"); return svnMetadataDir.exists(); } /** * Fetches the content from the SVN repository and creates the local working copy. * * @return the revision of the fetched content * * @throws IOException thrown if there is a problem checking out the content from the repository */ private SVNRevision checkoutResourceDirectory() throws IOException { try { final long newRevision = clientManager.getUpdateClient().doCheckout(remoteRepository, workingCopyDirectory, retrievalRevision, retrievalRevision, SVNDepth.INFINITY, true); log.debug( "Checked out revision {} from remote repository {} and stored it in local working directory {}", new Object[] {newRevision, remoteRepository.toDecodedString(), workingCopyDirectory.getAbsolutePath(),}); return SVNRevision.create(newRevision); } catch (final SVNException e) { final String errMsg = "Unable to check out revsion " + retrievalRevision.toString() + " from remote repository " + remoteRepository.toDecodedString() + " to local working directory " + workingCopyDirectory.getAbsolutePath(); log.error(errMsg, e); throw new IOException(errMsg, e); } } /** * Gets the last modified time for the given revision. * * @param revision revision to get the last modified date for * * @return the last modified time * * @throws IOException thrown if there is a problem getting the last modified time */ private DateTime getLastModificationForRevision(SVNRevision revision) throws IOException { try { final SVNStatusHandler handler = new SVNStatusHandler(); clientManager.getStatusClient().doStatus(getFile(), revision, SVNDepth.INFINITY, true, true, false, false, handler, null); final SVNStatus status = handler.getStatus(); // We want the date when this was committed return new DateTime(status.getCommittedDate()); } catch (final SVNException e) { final String errMsg = "Unable to check status of resource " + resourceFileName + " within working directory " + workingCopyDirectory.getAbsolutePath(); log.error(errMsg, e); throw new IOException(errMsg, e); } } /** * Updates an existing local working copy from the repository. * * @return the revision of the fetched content * * @throws IOException thrown if there is a problem updating the working copy */ private SVNRevision updateResourceDirectory() throws IOException { try { final long newRevision = clientManager.getUpdateClient().doUpdate(workingCopyDirectory, retrievalRevision, SVNDepth.INFINITY, true, true); log.debug("Updated local working directory {} to revision {} from remote repository {}", new Object[] { workingCopyDirectory.getAbsolutePath(), newRevision, remoteRepository.toDecodedString(),}); return SVNRevision.create(newRevision); } catch (final SVNException e) { final String errMsg = "Unable to update working copy of resoure " + remoteRepository.toDecodedString() + " in working copy " + workingCopyDirectory.getAbsolutePath() + " to revsion " + retrievalRevision.toString(); log.error(errMsg, e); throw new IOException(errMsg, e); } } /** {@inheritDoc} */ @Override public InputStream getInputStream() throws IOException { checkoutOrUpdateResource(); try { return new FileInputStream(getFile()); } catch (final IOException e) { log.error("Unable to read resource file {} from local working copy {}", resourceFileName, workingCopyDirectory.getAbsolutePath(), e); throw e; } } /** {@inheritDoc} */ @Override public boolean exists() { try { checkoutOrUpdateResource(); return getFile().exists(); } catch (IOException e) { return false; } } /** * {@inheritDoc} We always say yes. */ @Override public boolean isReadable() { return exists(); } /** * {@inheritDoc} Our stream can be read multiple times. */ @Override public boolean isOpen() { return false; } /** {@inheritDoc} */ @Override public URL getURL() throws IOException { return new URL(getProtocol(), remoteRepository.getHost(), remoteRepository.getPort(), getFullPath()); } /** {@inheritDoc} */ @Override public URI getURI() throws IOException { try { return new URI(getProtocol(), null, remoteRepository.getHost(), remoteRepository.getPort(), getFullPath(), null, null); } catch (URISyntaxException e) { throw new IOException(e); } } /** {@inheritDoc} */ @Override public File getFile() throws IOException { return new File(workingCopyDirectory, resourceFileName); } /** {@inheritDoc} */ @Override public long contentLength() throws IOException { return getFile().getTotalSpace(); } /** {@inheritDoc} */ @Override public long lastModified() throws IOException { checkoutOrUpdateResource(); return lastModified.getMillis(); } /** {@inheritDoc} */ @Override public Resource createRelative(String relativePath) throws IOException { throw new IOException("Cannot support relative open on SVN resources"); } /** {@inheritDoc} */ @Override public net.shibboleth.utilities.java.support.resource.Resource createRelativeResource(String relativePath) throws IOException { throw new IOException("Cannot support relative open on SVN resources"); } /** {@inheritDoc} */ @Override public String getDescription() { StringBuffer sb = new StringBuffer("SVN Resource: "); return sb.append(getFullPath()).toString(); } /** {@inheritDoc} */ @Override public void afterPropertiesSet() throws Exception { initialize(); } /** {@inheritDoc} */ @Override public void setBeanName(String name) { // For some reason Spring will call this after initialization. if (!isInitialized()) { setId(name); } } /** Simple {@link ISVNStatusHandler} implementation that just stores and returns the status. */ private class SVNStatusHandler implements ISVNStatusHandler { /** Current status of the resource. */ private SVNStatus status; /** * Gets the current status of the resource. * * @return current status of the resource */ public SVNStatus getStatus() { return status; } /** {@inheritDoc} */ @Override public void handleStatus(SVNStatus currentStatus) throws SVNException { status = currentStatus; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy