[[Main]] Here on this page we will publish different user qeustions and answers to them that a visitor may find interesting for himself. <> = Q: I'm encountering performance issues refreshing a working copy accessed through a symbolic link. = '''For example I have a symbolic link named 'myProjects' in my home directory which links to /media/projects. When I use the command 'svn status myProjects/projectName' the operation is a lot slower than when I use 'svn status /media/projects/projectName'. Is this a known issue, can it be resolved? ''' A: This is a known problem specific for older versions of [[SVNKit]]. In older versions [[SVNKit]] used expensive native program calls (ls) to resolve symbolic links. In case some of the working copy directory parents are symbolic links, such calls are performed for every file in the working copy (because their absolute and canonical path differs and there is no way to say whether a file is a symbolic link or not, but calling ls command for it). However since relatively younger versions [[SVNKit]] has been using the [[JNA]] library to fulfill some native operations which can not be implemented in native Java, and symbolic links are among them. If you are using the latest version of [[SVNKit]] and experiencing such a trouble, it may mean that [[SVNKit]] can not find the [[JNA]] library and falls back to using expensive Process.exec() calls (ls program). Please, check that [[JNA]] is present on your classpath along with [[SVNKit]]. If you prefer to use an older version of [[SVNKit]] however, there are two workarounds for this problem: 1. Put your working copy or refer to it by it's "real" path, not by the one that includes symbolic link, that is what you already doing. 2. Disable versioned symbolic links support in [[SVNKit]]. To do that set the following System property: {{{ -Dsvnkit.symlinks=false }}} (above is an option that you should add to the command line you're using to start your application) at runtime you may call: {{{#!java SVNFileType.setSymlinkSupportEnabled(false); }}} <> ---- = Q: Can the current SVNKit version be forced to create pre-1.5 format working copies? = ''' My problem is I need to support the older working copy format (pre svn 1.5.0) and I assumed the new SVNKit automatically generates (and upgrades) working copies to this latest format. I thought the most recent 1.2.x release would do the trick. ''' A: Yes, you can configure [[SVNKit]] to use the pre-1.5 WC format by putting the following code somewhere in the startup function: {{{#!java SVNAdminAreaFactory.setSelector(new ISVNAdminAreaFactorySelector() { public Collection getEnabledFactories(File path, Collection factories, boolean writeAccess) throws SVNException { Collection enabledFactories = new TreeSet(); for (Iterator factoriesIter = factories.iterator(); factoriesIter.hasNext(); ) { SVNAdminAreaFactory factory = (SVNAdminAreaFactory)factoriesIter.next(); int version = factory.getSupportedVersion(); if (version == SVNAdminAreaFactory.WC_FORMAT_13 || version == SVNAdminAreaFactory.WC_FORMAT_14) { enabledFactories.add(factory); } } return enabledFactories; } }); }}} As you can see in this code snippet {{{ISVNAdminAreaFactorySelector}}} gives you an ability to switch off the 1.5 fromat admin area factory. There is also a system property {{{svnkit.upgradeWC}}}, when set to {{{false}}} it will disable upgrade of existing working copies to the new format, but it still will be used for new directories, so first option (providing custom interface implementation) is preferable. Since 1.2.0 version [[SVNKit]] provides an ability to downgrade working copy format. The following code snippet shows a way to achieve that: {{{#!java SVNClientManager clientManager = SVNClientManager.newInstance(); SVNWCClient wcClient = clientManager.getWCClient(); File wc = new File("/path/to/working/copy"); wcClient.doSetWCFormat(wc, SVNAdminAreaFactory.WC_FORMAT_14); }}} ---- = Q: I wonder if the methods of SVNRepository (like checkout() an update()) are atomic? = ''' Is it possible (not probable) that another SVN client commit changes just while (not before or after) I am checking out or updating through these methods ? ''' A: [[SVNKit]] commit operations are atomic. That means that if you have two or more different clients committing (no matter whether they are all [[SVNKit]] clients or only one of them), they will be synchronized on the back end. In case of {{{svn://}}} and {{{http://}}} protocols commit finalizations are handled by the server itself (svnserve or mod_dav_svn), [[SVNKit]] client only sends necessary data and commands to the server, but the final step - writing to the repository - is undertaken by the server as you call {{{ISVNEditor.closeEdit()}}}. In case of the {{{file:///}}} protocol [[SVNKit]] behaves like Subversion - it gets a write lock on a repository before making a transaction immutable the same way Subversion does it. When updating via {{{svn://}}} or {{{http://}}}: on the back end it is handled by the server itself, [[SVNKit]] receives data and commands to update, but what is happening on the back end, [[SVNKit]] doesn't matter (if something is wrong it's up to the server to deal with it). {{{file:///}}} protocol update is implemented in a similar manner as in Subversion - it's a read operation and does not need any locks. Once it got the revision to update against it doesn't matter whether any commits are being performed: commits can create new repository files, but can not modify or delete existing ones, - you know each revision creates a new rev file in an FSFS repository. So your update will be absolutely consistent, although it doesn't mean that you will always get the latest revision when updating to HEAD revision - it may occur that after you started an update against a HEAD revision, someone started a new commit. But I believe it's true for Subversion, too. So, the answer to your question is yes, it's possible and it's ok. ---- = Q: In case of heavy editing through calls of the editor's methods it is possible, that another SVN client took a commit meanwhile? = ''' I am concerned about the use of the SVNCommitEditor object: In case of heavy editing through calls of the editor's methods it is possible, that another SVN client took a commit meanwhile? Does this cause trouble for the editor itself? I think a final (and atomic) commit is done by the editor through the call of its .closeEdit() method, not during the editing, or isn't it? To which revision does this refer, the HEAD revision at initialization of the editor or the HEAD revision at the .closeEdit() call? ''' A: No, it doesn't cause any trouble for the editor itself. Until a call to {{{ISVNEditor.closeEdit()}}} is made each transaction lives in it's own directory on the back end. And even if someone committed a new revision after you started your transaction but before you finished it, when you call {{{ISVNEditor.closeEdit()}}} [[SVNKit]] (in {{{file:///}}}) tries to merge all changes committed ever since. And if it sees that someone has changed the same files it aborts, notifying a user with an exception. However if it succeeds to merge all changes committed since the transaction was started a new revision it results to will also contain all previous modifications. It means it "refers" to the real HEAD revision, which may be equal or greater than transaction start point revision. That is a new transaction will be absolutely consistent. In other protocols the picture is the same but this work is handled by the server itself. ---- = Q: Is there any way to distinguish svn moves from svn copies using the low level API? = A: Yes, there is. Here is a short example demonstrating how to do that: {{{#!java public class Tests { public static void main(String[] args) { try { DAVRepositoryFactory.setup(); SVNRepositoryFactoryImpl.setup(); FSRepositoryFactory.setup(); String url = args[0]; SVNRepository repository = SVNRepositoryFactory.create(SVNURL.parseURIEncoded(url)); repository.log(null, 0, -1, true, false, -1, false, null, new ISVNLogEntryHandler() { public void handleLogEntry(SVNLogEntry logEntry) throws SVNException { Map changedPaths = logEntry.getChangedPaths(); for (Iterator changedPathsIter = changedPaths.keySet().iterator(); changedPathsIter.hasNext();) { String changedPath = (String) changedPathsIter.next(); SVNLogEntryPath logEntryPath = (SVNLogEntryPath) changedPaths.get(changedPath); if (logEntryPath.getType() == SVNLogEntryPath.TYPE_ADDED && logEntryPath.getCopyPath() != null) { SVNLogEntryPath copySourcePath = (SVNLogEntryPath) changedPaths.get(logEntryPath.getCopyPath()); if (copySourcePath != null && copySourcePath.getType() == SVNLogEntryPath.TYPE_DELETED) { System.out.println("Path " + changedPath + " was moved from " + logEntryPath.getCopyPath() + " in revision " + logEntry.getRevision()); } } } } }); System.exit(0); } catch (Exception e) { e.printStackTrace(); } } } }}} ---- = Q: Do you have an idea how to get the revision numbers and the corresponding dates of a file? = A: Here is a short example demonstrating how to do that with the low-level API: {{{#!java public class Tests { public static void main(String[] args) { try { DAVRepositoryFactory.setup(); SVNRepositoryFactoryImpl.setup(); FSRepositoryFactory.setup(); String url = args[0]; String filePathRelativeToURL = args[1]; SVNRepository repository = SVNRepositoryFactory.create(SVNURL.parseURIEncoded(url)); final Map fileRevisionsToDates = new HashMap(); repository.getFileRevisions(filePathRelativeToURL, 0, -1, false, new ISVNFileRevisionHandler() { public void applyTextDelta(String path, String baseChecksum) throws SVNException { } public void closeRevision(String token) throws SVNException { } public void openRevision(SVNFileRevision fileRevision) throws SVNException { fileRevisionsToDates.put(new Long(fileRevision.getRevision()), fileRevision.getRevisionProperties().getStringValue(SVNRevisionProperty.DATE)); } public OutputStream textDeltaChunk(String path, SVNDiffWindow diffWindow) throws SVNException { return null; } public void textDeltaEnd(String path) throws SVNException { } }); for (Iterator revisionsIter = fileRevisionsToDates.keySet().iterator(); revisionsIter.hasNext();) { Long revision = (Long) revisionsIter.next(); String dateStamp = (String) fileRevisionsToDates.get(revision); System.out.println("file revision: " + revision + ", date stamp: " + dateStamp); } System.exit(0); } catch (Exception e) { e.printStackTrace(); } } } }}} ---- = Q: What is the difference between the two "standalone" downloads - one_with_jna and one_with_no_jna? = A: SVNKit uses JNA library to optionally perform certain low-level file system operations using native OS calls. In particular, JNA is used to improve performance of operations related to symbolic links. Also, it enables credentials encryption on Windows (to store and read encrypted cached credentials to and from SVN_CONFIG/auth/ files). The reason we provide a distribution without JNA library included is that JNA library is LGPL licensed and some of the users would like to have an ability to download SVNKit with no LGPL components included. ---- = Q: What is Trilead library for? = A: Trilead library is used to establish SSH connections, i.e. to support working with repository over svn+ssh protocol. ---- = Q: Is it possible to create a repository on a remote svn server? = ''' I know it is possible to use SVNKit to create a brand-new repository on the local system like this: ''' {{{#!java try { String tgtPath = "C:/repos/root/path"; SVNURL tgtURL = SVNRepositoryFactory.createLocalRepository( new File( tgtPath ), true , false ); } catch ( SVNException e ) { //handle exception } }}} ''' But is it possible to create a repository on a remote system? ''' A: In general it is possible if, for example, you have SSH access to the remote computer and could run the svnadmin or jsvnadmin program on the remote side. Or you may write a servlet that will run on the server side and handle repository creation request - use SVNKit to create local repository on the server computer. Otherwise, Subversion server (svnserve, apache modules) doesn't support such a feature (a command to create a brand-new repository on the server side), and so SVNKit by itself couldn't help with that task. ---- = Q: How to merge without specifying revisions (making use of the merge-tracking feature)? = ''' How would I merge all changes from one branch to another without specifying revisions (except maybe HEAD)? What I want is the equivalent of the following (assuming SVN 1.5 and SVNKit 1.2): ''' {{{ svn merge http://localhost/MyProject/trunk c:\myWorkingCopy }}} A: To get the same results as the command that you're asking about would give, you should call: {{{#!java SVNDiffClient.doMerge(SVNURL url1, SVNRevision pegRevision, Collection rangesToMerge, File dstPath, SVNDepth depth, boolean useAncestry, boolean force, boolean dryRun, boolean recordOnly) }}} in the following way: {{{#!java SVNRevisionRange range = new SVNRevisionRange(SVNRevision.create(1), SVNRevision.HEAD); diffClient.doMerge(SVNURL.parseURIEncoded("http://localhost/MyProject/trunk"), SVNRevision.HEAD, Collections.singleton(range), new File("c:\\myWorkingCopy"),SVNDepth.UNKNOWN, true, false, false, false); }}} ---- = Q: What could be a possible reason for the 'svn: Malformed reply from SOCKS server' error ? = ''' I've tried various of the sample code snippets from the wiki, but the application hangs for a couple of minutes and then reports "error while listing entries: svn: Malformed reply from SOCKS server". Has anyone got a solution to this problem please? I googled a bit and it seems this may be an issue with Java itself and using a proxy. ''' A: The second part of the error message (that follows "svn:") is an error message taken from an IOException instance that was thrown when a socket connection was attempted to be established. Java may use browser settings by default. Try either changing JVM settings in Java control panel or disabling proxy in your default web browser. In case this resolves the problem, then the reason of the error is most probably not SVNKit-related. ---- = Q: How to prevent users from updating revision properties when specifying read-only URLs? = ''' I'm able to update a revision property even when specifying a URL which is read-only for me. How can I change this behaviour? ''' A: In case you use a path-based authentication and a certain user has read-only access to some parts of a repository and write access to other parts (paths), that user still could change revision properties in the repository - this is how the Subversion server works. To disallow certain users changing revision properties you may check a user name in the pre-revprop-change hook. You may find more information on hooks at [[http://svnbook.red-bean.com/en/1.5/svn.ref.reposhooks.pre-revprop-change.html]]. I.e. you may check a user name in the pre-revprop-change hook script and exit with an error code other than 0 to disable revision property changes for those users. ---- = Q: Problem on Windows Vista, connecting a repository through the svn protocol: org.tmatesoft.svn.core.SVNException: svn: connection refused by the server = ''' I'm running a Subversion server (svnserve) on Windows Vista. When I try to connect to it through the svn:// protocol with SVNKit using a URL that starts with svn://localhost, I get the error "Conncetion refused by the server". However if I do the same thing with a native svn client or TortoiseSVN, everything works fine. Also if I use 127.0.0.1 ip address in the URL instead of localhost, SVNKit also connects to the repository. What am I doing wrong? ''' A: This can be a hosts problem. With some jdk's localhost resolves to an IPv6 address instead of IPv4. Try adding '127.0.0.1 localhost' line to your hosts file which on Windows Vista can be found under C:\Windows\System32\drivers\etc\. = Q: Recursive repository properties fetch\list operation is slow. Is it possible to make it work faster? = ''' Trying to get entries and their properties from a repository recursively I found that it was extremely slow. I am using the HTTP connection. {{{#!java repository.getDir(path, revision, properties, handler); repository.getFile(path, revision, properties, null); }}} Is there another way to retrieve entries\entry properties with better performance? ''' A: When one needs to get entry contents or properties in a single operation, then it makes sense to use SVNRepository.status(...) call. Instead of fetching properties recursively with getDir/getFile calls, you may try using SVNRepository.status method to get properties for all files in repository: {{{#!java final long rev = repos.getLatestRevision(); ISVNReporterBaton reporter = new ISVNReporterBaton() { public void report(ISVNReporter reporter) throws SVNException { reporter.setPath("", null, rev, true); reporter.finishReport(); } }; ISVNEditor editor = new ISVNEditor() { public void addDir(...) { //save info about the directory here } public void addFile(...) { //save info about the file here } public void changeDirProperty(...) { //save current (added) directory property here } public void changeFileProperty(...){ //save current (added) file property here } .... }; repos.status(rev, null, true, reporter, editor); }}} Status operation above does the same as checkout, but doesn't send file contents, only properties that you may collect in changeFile/DirProperty methods. Above approach will provide better performance, especially when working with remote repositories - single "connection" will be established, instead of one for each getFile/getDir calls. It may not provide as much improvement for "file" protocol, but there should be some. Look at the full example code [[http://svn.svnkit.com/repos/svnkit/tags/1.3.5/doc/examples/src/org/tmatesoft/svn/examples/repository/FetchPropertiesRecursivelyWithStatus.java|here]].