org.duracloud.sync.endpoint.DuraStoreSyncEndpoint Maven / Gradle / Ivy
/*
* The contents of this file are subject to the license and copyright
* detailed in the LICENSE and NOTICE files at the root of the source
* tree and available online at
*
* http://duracloud.org/license/
*/
package org.duracloud.sync.endpoint;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.nio.charset.CharsetEncoder;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.Map;
import org.apache.commons.lang3.event.EventListenerSupport;
import org.duracloud.client.ContentStore;
import org.duracloud.common.util.ContentIdUtil;
import org.duracloud.common.util.DateUtil;
import org.duracloud.error.ContentStoreException;
import org.duracloud.storage.util.StorageProviderUtil;
import org.duracloud.sync.config.SyncToolConfig;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Endpoint which pushes files to DuraCloud.
*
* @author: Bill Branan
* Date: Mar 17, 2010
*/
public class DuraStoreSyncEndpoint implements SyncEndpoint {
private final Logger logger =
LoggerFactory.getLogger(DuraStoreSyncEndpoint.class);
private ContentStore contentStore;
private String username;
private String spaceId;
private boolean syncDeletes;
private boolean syncUpdates;
private boolean renameUpdates;
private boolean jumpStart;
private String updateSuffix;
private String storeId;
private String prefix;
EventListenerSupport listenerList;
public DuraStoreSyncEndpoint(ContentStore contentStore,
String username,
String spaceId,
boolean syncDeletes,
boolean syncUpdates,
boolean renameUpdates,
boolean jumpStart,
String updateSuffix,
String prefix) {
this.contentStore = contentStore;
this.username = username;
this.storeId = this.contentStore.getStoreId();
this.spaceId = spaceId;
this.syncDeletes = syncDeletes;
this.syncUpdates = syncUpdates;
this.renameUpdates = renameUpdates;
this.jumpStart = jumpStart;
this.updateSuffix = updateSuffix;
this.prefix = prefix;
this.listenerList = new EventListenerSupport<>(EndPointListener.class);
logger.info("Sync endpoint ready to transfer to space:" + spaceId +
" in store: " + storeId + " with config: " +
" syncDeletes:" + syncDeletes + ", syncUpdates:" + syncUpdates +
", renameUpdates:" + renameUpdates + ", jumpStart:" + jumpStart +
", updateSuffix:" + updateSuffix + ", prefix:" + prefix);
ensureSpaceExists();
}
public DuraStoreSyncEndpoint(ContentStore contentStore,
String username,
String spaceId,
boolean syncDeletes,
boolean jumpStart) {
this(contentStore,
username,
spaceId,
syncDeletes,
true,
false,
jumpStart,
SyncToolConfig.DEFAULT_UPDATE_SUFFIX,
null);
}
protected String getUsername() {
return this.username;
}
private void ensureSpaceExists() {
try {
contentStore.getSpaceACLs(spaceId);
} catch (ContentStoreException e) {
throw new RuntimeException("Could not connect to space with ID '" +
spaceId + "'.");
}
}
@Override
public boolean syncFile(MonitoredFile syncFile, File watchDir) {
SyncResultType result =
syncFileAndReturnDetailedResult(syncFile, watchDir);
return (result != SyncResultType.FAILED);
}
@Override
public SyncResultType syncFileAndReturnDetailedResult(MonitoredFile syncFile,
File watchDir) {
SyncResultType result = SyncResultType.ALREADY_IN_SYNC;
String contentId =
ContentIdUtil.getContentId(syncFile.getFile(), watchDir, prefix);
String absPath = syncFile.getAbsolutePath();
logger.debug("Syncing file " + absPath +
" to DuraCloud with ID " + contentId);
try {
if (jumpStart) { // Skip all of the usual checks, just push the file
if (syncFile.exists()) {
doAddContent(syncFile, contentId, absPath);
return SyncResultType.ADDED;
}
}
Map contentProperties =
getContentProperties(spaceId, contentId);
boolean dcFileExists = (null != contentProperties);
if (syncFile.exists()) {
if (dcFileExists) { // File was updated
String dcChecksum =
contentProperties.get(ContentStore.CONTENT_CHECKSUM);
if (dcChecksum.equals(syncFile.getChecksum())) {
logger.debug("Checksum for local file {} matches " +
"file in DuraCloud, no update needed.",
absPath);
} else {
if (syncUpdates) {
logger.debug("Local file {} changed, updating DuraCloud.",
absPath);
if (renameUpdates) {
// create backup of original using current timestamp
// in backup content id. I'm using current timestamp
// since original timestamp is just a date without
// timestamp info. Plus I think it is more intuitive
// to have the timestamp reflect the moment when the
// backup file was created. -dbernstein
String timeStamp = DateUtil.nowPlain();
String backupContentId = contentId + this.updateSuffix + "." + timeStamp;
logger.info("Renaming {} to {} to prevent it " +
"from being overwritten",
contentId, backupContentId);
this.contentStore.copyContent(this.spaceId,
contentId,
this.spaceId,
backupContentId);
this.listenerList.fire()
.contentBackedUp(this.storeId,
this.spaceId,
contentId,
backupContentId,
absPath);
}
addUpdateContent(contentId, syncFile, absPath);
this.listenerList
.fire().contentUpdated(this.storeId, this.spaceId,
contentId, absPath);
result = SyncResultType.UPDATED;
} else {
logger.debug("Local file {} changed, but sync updates options ", absPath);
this.listenerList.fire().contentUpdateIgnored(this.storeId,
this.spaceId,
contentId,
absPath);
result = SyncResultType.UPDATE_IGNORED;
}
}
} else { // File was added
doAddContent(syncFile, contentId, absPath);
result = SyncResultType.ADDED;
}
} else { // File was deleted (does not exist locally)
if (syncDeletes) {
if (dcFileExists) {
result = deleteContent(spaceId, contentId, absPath);
} else if (null != prefix) {
// Check for dc file without prefix
String noPrefixContentId =
contentId.substring(prefix.length());
if (null != getContentProperties(spaceId, noPrefixContentId)) {
result = deleteContent(spaceId, noPrefixContentId, absPath);
}
}
} else {
logger.debug("Ignoring delete of file {}", absPath);
}
}
} catch (ContentStoreException e) {
throw new RuntimeException(e);
}
return result;
}
protected void doAddContent(MonitoredFile syncFile,
String contentId,
String absPath) throws ContentStoreException {
logger.debug("Local file {} added, moving to DuraCloud.", absPath);
addUpdateContent(contentId, syncFile, syncFile.getAbsolutePath());
this.listenerList.fire().contentAdded(this.storeId, this.spaceId,
contentId, absPath);
}
protected Map getContentProperties(String spaceId,
String contentId) {
Map props = null;
try {
props = contentStore.getContentProperties(spaceId, contentId);
} catch (ContentStoreException e) {
logger.debug("Content properties do not exist for content item " +
"{} in space {}", contentId, spaceId);
}
return props;
}
private SyncResultType deleteContent(String spaceId,
String contentId,
String absPath)
throws ContentStoreException {
logger.debug("Local file {} deleted, removing from DuraCloud.", absPath);
deleteContent(spaceId, contentId);
return SyncResultType.DELETED;
}
@Override
public void deleteContent(String spaceId, String contentId)
throws ContentStoreException {
logger.info("Deleting {} from DuraCloud space {}", contentId, spaceId);
contentStore.deleteContent(spaceId, contentId);
this.listenerList.fire().contentDeleted(this.storeId, this.spaceId, contentId);
}
private void addUpdateContent(String contentId,
MonitoredFile syncFile,
String absPath)
throws ContentStoreException {
logger.info("Adding local file {} to DuraCloud space {}" +
" with content ID {}", absPath, spaceId, contentId);
addUpdateContent(contentId, syncFile);
}
protected void addUpdateContent(String contentId, MonitoredFile syncFile)
throws ContentStoreException {
InputStream syncStream = syncFile.getStream();
Map props = createProps(syncFile.getAbsolutePath(), this.username);
try {
contentStore.addContent(spaceId,
contentId,
syncStream,
syncFile.length(),
syncFile.getMimetype(),
syncFile.getChecksum(),
props);
} finally {
try {
syncStream.close();
} catch (IOException e) {
logger.error("Error attempting to close stream for file " +
contentId + ": " + e.getMessage(), e);
}
}
}
protected Map createProps(String absolutePath, String username) {
Map props = StorageProviderUtil.createContentProperties(absolutePath, username);
removePropsWithNonUSASCIINamesOrValues(props);
return props;
}
private void removePropsWithNonUSASCIINamesOrValues(Map props) {
for (String key : new ArrayList<>(props.keySet())) {
if (!isAllUSASCII(key) || !isAllUSASCII(props.get(key))) {
props.remove(key);
}
}
}
private boolean isAllUSASCII(String value) {
if (value != null) {
CharsetEncoder encoder =
Charset.forName("US-ASCII").newEncoder();
return encoder.canEncode(value);
}
return true;
}
public Iterator getFilesList() {
Iterator spaceContents;
try {
spaceContents = contentStore.getSpaceContents(spaceId);
} catch (ContentStoreException e) {
throw new RuntimeException("Unable to get list of files from " +
"DuraStore due to: " + e.getMessage());
}
return spaceContents;
}
protected ContentStore getContentStore() {
return contentStore;
}
protected String getSpaceId() {
return spaceId;
}
@Override
public void addEndPointListener(EndPointListener listener) {
this.listenerList.addListener(listener);
}
@Override
public void removeEndPointListener(EndPointListener listener) {
this.listenerList.removeListener(listener);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy