/*
* ====================================================================
* Copyright (c) 2004-2012 TMate Software Ltd. All rights reserved.
*
* This software is licensed as described in the file COPYING, which
* you should have received as part of this distribution. The terms
* are also available at http://svnkit.com/license.html
* If newer versions of this license are posted there, you may use a
* newer version instead, at your option.
* ====================================================================
*/
package org.tmatesoft.svn.core.internal.io.dav.handlers;
import org.tmatesoft.svn.core.SVNErrorCode;
import org.tmatesoft.svn.core.SVNErrorMessage;
import org.tmatesoft.svn.core.SVNException;
import org.tmatesoft.svn.core.SVNPropertyValue;
import org.tmatesoft.svn.core.internal.io.dav.DAVElement;
import org.tmatesoft.svn.core.internal.util.SVNXMLUtil;
import org.tmatesoft.svn.core.internal.wc.SVNErrorManager;
import org.tmatesoft.svn.core.io.ISVNEditor;
import org.tmatesoft.svn.util.SVNLogType;
import org.xml.sax.Attributes;
/**
* @author TMate Software Ltd.
* @version 1.3
*/
public class DAVReplayHandler extends DAVEditorHandler {
public static StringBuffer generateReplayRequest(long highRevision, long lowRevision, boolean sendDeltas) {
StringBuffer xmlBuffer = new StringBuffer();
SVNXMLUtil.addXMLHeader(xmlBuffer);
SVNXMLUtil.openNamespaceDeclarationTag(SVNXMLUtil.SVN_NAMESPACE_PREFIX, "replay-report",
SVN_NAMESPACES_LIST, SVNXMLUtil.PREFIX_MAP, xmlBuffer);
SVNXMLUtil.openCDataTag(SVNXMLUtil.SVN_NAMESPACE_PREFIX, "revision", String.valueOf(highRevision),
xmlBuffer);
SVNXMLUtil.openCDataTag(SVNXMLUtil.SVN_NAMESPACE_PREFIX, "low-water-mark", String.valueOf(lowRevision),
xmlBuffer);
SVNXMLUtil.openCDataTag(SVNXMLUtil.SVN_NAMESPACE_PREFIX, "send-deltas", sendDeltas ? "1" : "0",
xmlBuffer);
SVNXMLUtil.addXMLFooter(SVNXMLUtil.SVN_NAMESPACE_PREFIX, "replay-report", xmlBuffer);
return xmlBuffer;
}
protected static final DAVElement EDITOR_REPORT = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "editor-report");
protected static final DAVElement OPEN_ROOT = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "open-root");
protected static final DAVElement APPLY_TEXT_DELTA = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "apply-textdelta");
protected static final DAVElement CLOSE_FILE = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "close-file");
protected static final DAVElement CLOSE_DIRECTORY = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "close-directory");
protected static final DAVElement CHANGE_FILE_PROPERTY = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "change-file-prop");
protected static final DAVElement CHANGE_DIR_PROPERTY = DAVElement.getElement(DAVElement.SVN_NAMESPACE, "change-dir-prop");
protected static final String CHECKSUM_ATTR = "checksum";
protected static final String DEL_ATTR = "del";
public DAVReplayHandler(ISVNEditor editor, boolean fetchContent) {
super(null, null, editor, null, fetchContent, false, null);
}
protected void startElement(DAVElement parent, DAVElement element, Attributes attrs) throws SVNException {
if (element == TARGET_REVISION) {
String rev = attrs.getValue(REVISION_ATTR);
if (rev == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing revision attr in target-revision element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
try {
myEditor.targetRevision(Long.parseLong(rev));
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
} else if (element == OPEN_ROOT) {
String rev = attrs.getValue(REVISION_ATTR);
if (rev == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing revision attr in open-root element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
try {
myEditor.openRoot(Long.parseLong(rev));
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
myPath = "";
myIsDirectory = true;
}
} else if (element == DELETE_ENTRY) {
String path = attrs.getValue(NAME_ATTR);
String rev = attrs.getValue(REVISION_ATTR);
if (path == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing name attr in delete-entry element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else if (rev == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing rev attr in delete-entry element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
try {
myEditor.deleteEntry(path, Long.parseLong(rev));
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
} else if (element == OPEN_DIRECTORY || element == ADD_DIRECTORY) {
String path = attrs.getValue(NAME_ATTR);
String rev = attrs.getValue(REVISION_ATTR);
if (path == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing name attr in " + (element == OPEN_DIRECTORY ? "open-directory" : "add-directory") + " element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
long revision = -1;
if (rev != null) {
try {
revision =Long.parseLong(rev);
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
if (element == OPEN_DIRECTORY) {
myEditor.openDir(path, revision);
} else {
String copyFromPath = attrs.getValue(COPYFROM_PATH_ATTR);
String cfRevision = attrs.getValue(COPYFROM_REV_ATTR);
long copyFromRevision = -1;
if (cfRevision != null) {
try {
copyFromRevision = Long.parseLong(cfRevision);
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
myEditor.addDir(path, copyFromPath, copyFromRevision);
}
}
myPath = path;
myIsDirectory = true;
} else if (element == OPEN_FILE || element == ADD_FILE) {
String path = attrs.getValue(NAME_ATTR);
if (path == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing name attr in " + (element == OPEN_FILE ? "open-file" : "add-file") + " element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
}
if (element == ADD_FILE) {
String copyFromPath = attrs.getValue(COPYFROM_PATH_ATTR);
String cfRevision = attrs.getValue(COPYFROM_REV_ATTR);
long copyFromRevision = -1;
if (cfRevision != null) {
try {
copyFromRevision = Long.parseLong(cfRevision);
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
myEditor.addFile(path, copyFromPath, copyFromRevision);
} else {
String rev = attrs.getValue(REVISION_ATTR);
long revision = -1;
if (rev != null) {
try {
revision = Long.parseLong(rev);
} catch (NumberFormatException nfe) {
SVNErrorManager.error(SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, nfe), SVNLogType.NETWORK);
}
}
myEditor.openFile(path, revision);
}
myIsDirectory = false;
myPath = path;
} else if (element == APPLY_TEXT_DELTA) {
if (myIsDirectory) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Got apply-textdelta element without preceding add-file or open-file");
SVNErrorManager.error(err, SVNLogType.NETWORK);
}
String checksum = attrs.getValue(CHECKSUM_ATTR);
try {
myEditor.applyTextDelta(myPath, checksum);
setDeltaProcessing(true);
} catch (SVNException svne) {
//
}
} else if (element == CLOSE_FILE) {
if (myIsDirectory) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Got close-file element without preceding add-file or open-file");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
String checksum = attrs.getValue(CHECKSUM_ATTR);
myEditor.closeFile(myPath, checksum);
myIsDirectory = true;
}
} else if (element == CLOSE_DIRECTORY) {
if (!myIsDirectory) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Got close-directory element without ever opening a directory");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
myEditor.closeDir();
}
} else if (element == CHANGE_FILE_PROPERTY || element == CHANGE_DIR_PROPERTY) {
String name = attrs.getValue(NAME_ATTR);
if (name == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Missing name attr in " + (element == CHANGE_FILE_PROPERTY ? "change-file-prop" : "change-dir-prop") + " element");
SVNErrorManager.error(err, SVNLogType.NETWORK);
} else {
if (attrs.getValue(DEL_ATTR) != null) {
if (element == CHANGE_FILE_PROPERTY) {
myEditor.changeFileProperty(myPath, name, null);
} else {
myEditor.changeDirProperty(name, null);
}
myPropertyName = null;
} else {
myPropertyName = name;
}
}
}
}
protected void endElement(DAVElement parent, DAVElement element, StringBuffer cdata) throws SVNException {
if (element == APPLY_TEXT_DELTA) {
setDeltaProcessing(false);
} else if (element == CHANGE_FILE_PROPERTY || element == CHANGE_DIR_PROPERTY) {
if (cdata != null && !"".equals(cdata.toString()) && myPropertyName == null) {
SVNErrorMessage err = SVNErrorMessage.create(SVNErrorCode.RA_DAV_MALFORMED_DATA, "Got cdata content for a prop delete");
SVNErrorManager.error(err, SVNLogType.NETWORK);
}
if (myPropertyName != null) {
SVNPropertyValue propertyValue = createPropertyValueFromBase64(null, myPropertyName, cdata);
if (element == CHANGE_FILE_PROPERTY) {
myEditor.changeFileProperty(myPath, myPropertyName, propertyValue);
} else {
myEditor.changeDirProperty(myPropertyName, propertyValue);
}
}
}
}
}