Browse Source

Add support for getting the system wide configuration

These settings are stored in <prefix>/etc/gitconfig. The C Git
binary is installed in <prefix>/bin, so we look for the C Git
executable to find this location, first by looking at the PATH
environment variable and then by attemting to launch bash as
a login shell to find out.

Bug: 333216
Change-Id: I1bbee9fb123a81714a34a9cc242b92beacfbb4a8
Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
Signed-off-by: Robin Rosenberg <robin.rosenberg@dewire.com>
stable-0.11
Robin Rosenberg 14 years ago
parent
commit
797ebba307
  1. 43
      org.eclipse.jgit.junit/src/org/eclipse/jgit/junit/MockSystemReader.java
  2. 3
      org.eclipse.jgit.test/tst/org/eclipse/jgit/lib/ConfigTest.java
  3. 1
      org.eclipse.jgit/resources/org/eclipse/jgit/JGitText.properties
  4. 1
      org.eclipse.jgit/src/org/eclipse/jgit/JGitText.java
  5. 26
      org.eclipse.jgit/src/org/eclipse/jgit/storage/file/FileRepository.java
  6. 3
      org.eclipse.jgit/src/org/eclipse/jgit/util/FS.java
  7. 81
      org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX.java
  8. 2
      org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX_Java5.java
  9. 2
      org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX_Java6.java
  10. 20
      org.eclipse.jgit/src/org/eclipse/jgit/util/FS_Win32.java
  11. 44
      org.eclipse.jgit/src/org/eclipse/jgit/util/SystemReader.java

43
org.eclipse.jgit.junit/src/org/eclipse/jgit/junit/MockSystemReader.java

@ -45,39 +45,50 @@
package org.eclipse.jgit.junit; package org.eclipse.jgit.junit;
import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.TimeZone; import java.util.TimeZone;
import org.eclipse.jgit.errors.ConfigInvalidException; import org.eclipse.jgit.errors.ConfigInvalidException;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.lib.Constants; import org.eclipse.jgit.lib.Constants;
import org.eclipse.jgit.storage.file.FileBasedConfig; import org.eclipse.jgit.storage.file.FileBasedConfig;
import org.eclipse.jgit.util.FS; import org.eclipse.jgit.util.FS;
import org.eclipse.jgit.util.SystemReader; import org.eclipse.jgit.util.SystemReader;
public class MockSystemReader extends SystemReader { public class MockSystemReader extends SystemReader {
private final class MockConfig extends FileBasedConfig {
private MockConfig(File cfgLocation, FS fs) {
super(cfgLocation, fs);
}
@Override
public void load() throws IOException, ConfigInvalidException {
// Do nothing
}
@Override
public boolean isOutdated() {
return false;
}
}
final Map<String, String> values = new HashMap<String, String>(); final Map<String, String> values = new HashMap<String, String>();
FileBasedConfig userGitConfig; FileBasedConfig userGitConfig;
FileBasedConfig systemGitConfig;
public MockSystemReader() { public MockSystemReader() {
init(Constants.OS_USER_NAME_KEY); init(Constants.OS_USER_NAME_KEY);
init(Constants.GIT_AUTHOR_NAME_KEY); init(Constants.GIT_AUTHOR_NAME_KEY);
init(Constants.GIT_AUTHOR_EMAIL_KEY); init(Constants.GIT_AUTHOR_EMAIL_KEY);
init(Constants.GIT_COMMITTER_NAME_KEY); init(Constants.GIT_COMMITTER_NAME_KEY);
init(Constants.GIT_COMMITTER_EMAIL_KEY); init(Constants.GIT_COMMITTER_EMAIL_KEY);
userGitConfig = new FileBasedConfig(null, null) { userGitConfig = new MockConfig(null, null);
@Override systemGitConfig = new MockConfig(null, null);
public void load() throws IOException, ConfigInvalidException {
// Do nothing
}
@Override
public boolean isOutdated() {
return false;
}
};
} }
private void init(final String n) { private void init(final String n) {
@ -103,10 +114,17 @@ public class MockSystemReader extends SystemReader {
} }
@Override @Override
public FileBasedConfig openUserConfig(FS fs) { public FileBasedConfig openUserConfig(Config parent, FS fs) {
assert parent == null || parent == systemGitConfig;
return userGitConfig; return userGitConfig;
} }
@Override
public FileBasedConfig openSystemConfig(Config parent, FS fs) {
assert parent == null;
return systemGitConfig;
}
@Override @Override
public String getHostname() { public String getHostname() {
return "fake.host.example.com"; return "fake.host.example.com";
@ -121,4 +139,5 @@ public class MockSystemReader extends SystemReader {
public int getTimezone(long when) { public int getTimezone(long when) {
return TimeZone.getTimeZone("GMT-03:30").getOffset(when) / (60 * 1000); return TimeZone.getTimeZone("GMT-03:30").getOffset(when) / (60 * 1000);
} }
} }

3
org.eclipse.jgit.test/tst/org/eclipse/jgit/lib/ConfigTest.java

@ -120,7 +120,8 @@ public class ConfigTest extends TestCase {
final MockSystemReader mockSystemReader = new MockSystemReader(); final MockSystemReader mockSystemReader = new MockSystemReader();
SystemReader.setInstance(mockSystemReader); SystemReader.setInstance(mockSystemReader);
final String hostname = mockSystemReader.getHostname(); final String hostname = mockSystemReader.getHostname();
final Config userGitConfig = mockSystemReader.openUserConfig(FS.DETECTED); final Config userGitConfig = mockSystemReader.openUserConfig(null,
FS.DETECTED);
final Config localConfig = new Config(userGitConfig); final Config localConfig = new Config(userGitConfig);
mockSystemReader.clearProperties(); mockSystemReader.clearProperties();

1
org.eclipse.jgit/resources/org/eclipse/jgit/JGitText.properties

@ -393,6 +393,7 @@ startingReadStageWithoutWrittenRequestDataPendingIsNotSupported=Starting read st
statelessRPCRequiresOptionToBeEnabled=stateless RPC requires {0} to be enabled statelessRPCRequiresOptionToBeEnabled=stateless RPC requires {0} to be enabled
submodulesNotSupported=Submodules are not supported submodulesNotSupported=Submodules are not supported
symlinkCannotBeWrittenAsTheLinkTarget=Symlink "{0}" cannot be written as the link target cannot be read from within Java. symlinkCannotBeWrittenAsTheLinkTarget=Symlink "{0}" cannot be written as the link target cannot be read from within Java.
systemConfigFileInvalid=Systen wide config file {0} is invalid {1}
tagNameInvalid=tag name {0} is invalid tagNameInvalid=tag name {0} is invalid
tagOnRepoWithoutHEADCurrentlyNotSupported=Tag on repository without HEAD currently not supported tagOnRepoWithoutHEADCurrentlyNotSupported=Tag on repository without HEAD currently not supported
tSizeMustBeGreaterOrEqual1=tSize must be >= 1 tSizeMustBeGreaterOrEqual1=tSize must be >= 1

1
org.eclipse.jgit/src/org/eclipse/jgit/JGitText.java

@ -453,6 +453,7 @@ public class JGitText extends TranslationBundle {
/***/ public String statelessRPCRequiresOptionToBeEnabled; /***/ public String statelessRPCRequiresOptionToBeEnabled;
/***/ public String submodulesNotSupported; /***/ public String submodulesNotSupported;
/***/ public String symlinkCannotBeWrittenAsTheLinkTarget; /***/ public String symlinkCannotBeWrittenAsTheLinkTarget;
/***/ public String systemConfigFileInvalid;
/***/ public String tagNameInvalid; /***/ public String tagNameInvalid;
/***/ public String tagOnRepoWithoutHEADCurrentlyNotSupported; /***/ public String tagOnRepoWithoutHEADCurrentlyNotSupported;
/***/ public String tSizeMustBeGreaterOrEqual1; /***/ public String tSizeMustBeGreaterOrEqual1;

26
org.eclipse.jgit/src/org/eclipse/jgit/storage/file/FileRepository.java

@ -95,6 +95,8 @@ import org.eclipse.jgit.util.SystemReader;
* *
*/ */
public class FileRepository extends Repository { public class FileRepository extends Repository {
private final FileBasedConfig systemConfig;
private final FileBasedConfig userConfig; private final FileBasedConfig userConfig;
private final FileBasedConfig repoConfig; private final FileBasedConfig repoConfig;
@ -152,11 +154,14 @@ public class FileRepository extends Repository {
public FileRepository(final BaseRepositoryBuilder options) throws IOException { public FileRepository(final BaseRepositoryBuilder options) throws IOException {
super(options); super(options);
userConfig = SystemReader.getInstance().openUserConfig(getFS()); systemConfig = SystemReader.getInstance().openSystemConfig(null, getFS());
userConfig = SystemReader.getInstance().openUserConfig(systemConfig,
getFS());
repoConfig = new FileBasedConfig(userConfig, // repoConfig = new FileBasedConfig(userConfig, //
getFS().resolve(getDirectory(), "config"), // getFS().resolve(getDirectory(), "config"), //
getFS()); getFS());
loadSystemConfig();
loadUserConfig(); loadUserConfig();
loadRepoConfig(); loadRepoConfig();
@ -184,6 +189,18 @@ public class FileRepository extends Repository {
} }
} }
private void loadSystemConfig() throws IOException {
try {
systemConfig.load();
} catch (ConfigInvalidException e1) {
IOException e2 = new IOException(MessageFormat.format(JGitText
.get().systemConfigFileInvalid, systemConfig.getFile()
.getAbsolutePath(), e1));
e2.initCause(e1);
throw e2;
}
}
private void loadUserConfig() throws IOException { private void loadUserConfig() throws IOException {
try { try {
userConfig.load(); userConfig.load();
@ -285,6 +302,13 @@ public class FileRepository extends Repository {
* @return the configuration of this repository * @return the configuration of this repository
*/ */
public FileBasedConfig getConfig() { public FileBasedConfig getConfig() {
if (systemConfig.isOutdated()) {
try {
loadSystemConfig();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
if (userConfig.isOutdated()) { if (userConfig.isOutdated()) {
try { try {
loadUserConfig(); loadUserConfig();

3
org.eclipse.jgit/src/org/eclipse/jgit/util/FS.java

@ -252,4 +252,7 @@ public abstract class FS {
} }
return null; return null;
} }
/** @return the $prefix directory C Git would use. */
public abstract File gitPrefix();
} }

81
org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX.java

@ -0,0 +1,81 @@
/*
* Copyright (C) 2010, Robin Rosenberg
* and other copyright owners as documented in the project's IP log.
*
* This program and the accompanying materials are made available
* under the terms of the Eclipse Distribution License v1.0 which
* accompanies this distribution, is reproduced below, and is
* available at http://www.eclipse.org/org/documents/edl-v10.php
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or
* without modification, are permitted provided that the following
* conditions are met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
*
* - Neither the name of the Eclipse Foundation, Inc. nor the
* names of its contributors may be used to endorse or promote
* products derived from this software without specific prior
* written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.eclipse.jgit.util;
import java.io.File;
import java.nio.charset.Charset;
import java.security.AccessController;
import java.security.PrivilegedAction;
abstract class FS_POSIX extends FS {
@Override
public File gitPrefix() {
String path = SystemReader.getInstance().getenv("PATH");
File gitExe = searchPath(path, "git");
if (gitExe != null)
return gitExe.getParentFile().getParentFile();
if (isMacOS()) {
// On MacOSX, PATH is shorter when Eclipse is launched from the
// Finder than from a terminal. Therefore try to launch bash as a
// login shell and search using that.
//
String w = readPipe(userHome(), //
new String[] { "bash", "--login", "-c", "which git" }, //
Charset.defaultCharset().name());
return new File(w).getParentFile().getParentFile();
}
return null;
}
private static boolean isMacOS() {
final String osDotName = AccessController
.doPrivileged(new PrivilegedAction<String>() {
public String run() {
return System.getProperty("os.name");
}
});
return "Mac OS X".equals(osDotName);
}
}

2
org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX_Java5.java

@ -45,7 +45,7 @@ package org.eclipse.jgit.util;
import java.io.File; import java.io.File;
class FS_POSIX_Java5 extends FS { class FS_POSIX_Java5 extends FS_POSIX {
public boolean supportsExecute() { public boolean supportsExecute() {
return false; return false;
} }

2
org.eclipse.jgit/src/org/eclipse/jgit/util/FS_POSIX_Java6.java

@ -49,7 +49,7 @@ import java.io.File;
import java.lang.reflect.InvocationTargetException; import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; import java.lang.reflect.Method;
class FS_POSIX_Java6 extends FS { class FS_POSIX_Java6 extends FS_POSIX {
private static final Method canExecute; private static final Method canExecute;
private static final Method setExecute; private static final Method setExecute;

20
org.eclipse.jgit/src/org/eclipse/jgit/util/FS_Win32.java

@ -45,6 +45,7 @@
package org.eclipse.jgit.util; package org.eclipse.jgit.util;
import java.io.File; import java.io.File;
import java.nio.charset.Charset;
import java.security.AccessController; import java.security.AccessController;
import java.security.PrivilegedAction; import java.security.PrivilegedAction;
@ -76,4 +77,23 @@ class FS_Win32 extends FS {
public boolean retryFailedLockFileCommit() { public boolean retryFailedLockFileCommit() {
return true; return true;
} }
@Override
public File gitPrefix() {
String path = SystemReader.getInstance().getenv("PATH");
File gitExe = searchPath(path, "git.exe", "git.cmd");
if (gitExe != null)
return gitExe.getParentFile().getParentFile();
// This isn't likely to work, if bash is in $PATH, git should
// also be in $PATH. But its worth trying.
//
String w = readPipe(userHome(), //
new String[] { "bash", "--login", "-c", "which git" }, //
Charset.defaultCharset().name());
if (w != null)
return new File(w).getParentFile().getParentFile();
return null;
}
} }

44
org.eclipse.jgit/src/org/eclipse/jgit/util/SystemReader.java

@ -50,6 +50,7 @@ import java.net.InetAddress;
import java.net.UnknownHostException; import java.net.UnknownHostException;
import java.util.TimeZone; import java.util.TimeZone;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.storage.file.FileBasedConfig; import org.eclipse.jgit.storage.file.FileBasedConfig;
/** /**
@ -72,9 +73,28 @@ public abstract class SystemReader {
return System.getProperty(key); return System.getProperty(key);
} }
public FileBasedConfig openUserConfig(FS fs) { public FileBasedConfig openSystemConfig(Config parent, FS fs) {
File prefix = fs.gitPrefix();
if (prefix == null) {
return new FileBasedConfig(null, fs) {
public void load() {
// empty, do not load
}
public boolean isOutdated() {
// regular class would bomb here
return false;
}
};
}
File etc = fs.resolve(prefix, "etc");
File config = fs.resolve(etc, "gitconfig");
return new FileBasedConfig(parent, config, fs);
}
public FileBasedConfig openUserConfig(Config parent, FS fs) {
final File home = fs.userHome(); final File home = fs.userHome();
return new FileBasedConfig(new File(home, ".gitconfig"), fs); return new FileBasedConfig(parent, new File(home, ".gitconfig"), fs);
} }
public String getHostname() { public String getHostname() {
@ -136,12 +156,26 @@ public abstract class SystemReader {
public abstract String getProperty(String key); public abstract String getProperty(String key);
/** /**
* @param parent
* a config with values not found directly in the returned config
* @param fs * @param fs
* the file system abstraction which will be necessary to * the file system abstraction which will be necessary to perform
* perform certain file system operations. * certain file system operations.
* @return the git configuration found in the user home * @return the git configuration found in the user home
*/ */
public abstract FileBasedConfig openUserConfig(FS fs); public abstract FileBasedConfig openUserConfig(Config parent, FS fs);
/**
* @param parent
* a config with values not found directly in the returned
* config. Null is a reasonable value here.
* @param fs
* the file system abstraction which will be necessary to perform
* certain file system operations.
* @return the gitonfig configuration found in the system-wide "etc"
* directory
*/
public abstract FileBasedConfig openSystemConfig(Config parent, FS fs);
/** /**
* @return the current system time * @return the current system time

Loading…
Cancel
Save