| // Copyright (C) 2013 The Android Open Source Project |
| // |
| // Licensed 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 com.google.gerrit.server.securestore; |
| |
| import com.google.gerrit.common.FileUtil; |
| import com.google.gerrit.common.Nullable; |
| import com.google.gerrit.server.config.SitePaths; |
| import com.google.inject.Inject; |
| import com.google.inject.ProvisionException; |
| import com.google.inject.Singleton; |
| import java.io.File; |
| import java.io.IOException; |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| import org.eclipse.jgit.errors.ConfigInvalidException; |
| import org.eclipse.jgit.internal.storage.file.LockFile; |
| import org.eclipse.jgit.lib.Constants; |
| import org.eclipse.jgit.storage.file.FileBasedConfig; |
| import org.eclipse.jgit.util.FS; |
| |
| @Singleton |
| public class DefaultSecureStore extends SecureStore { |
| private final FileBasedConfig sec; |
| private final Map<String, FileBasedConfig> pluginSec; |
| private final SitePaths site; |
| |
| @Inject |
| DefaultSecureStore(SitePaths site) { |
| this.site = site; |
| sec = new FileBasedConfig(site.secure_config.toFile(), FS.DETECTED); |
| try { |
| sec.load(); |
| } catch (IOException | ConfigInvalidException e) { |
| throw new RuntimeException("Cannot load secure.config", e); |
| } |
| this.pluginSec = new HashMap<>(); |
| } |
| |
| @Override |
| public String[] getList(String section, String subsection, String name) { |
| return sec.getStringList(section, subsection, name); |
| } |
| |
| @Nullable |
| @Override |
| public synchronized String[] getListForPlugin( |
| String pluginName, String section, String subsection, String name) { |
| FileBasedConfig cfg = null; |
| if (pluginSec.containsKey(pluginName)) { |
| cfg = pluginSec.get(pluginName); |
| } else { |
| String filename = pluginName + ".secure.config"; |
| File pluginConfigFile = site.etc_dir.resolve(filename).toFile(); |
| if (pluginConfigFile.exists()) { |
| cfg = new FileBasedConfig(pluginConfigFile, FS.DETECTED); |
| try { |
| cfg.load(); |
| pluginSec.put(pluginName, cfg); |
| } catch (IOException | ConfigInvalidException e) { |
| throw new RuntimeException("Cannot load " + filename, e); |
| } |
| } |
| } |
| return cfg != null ? cfg.getStringList(section, subsection, name) : null; |
| } |
| |
| @Override |
| public void setList(String section, String subsection, String name, List<String> values) { |
| if (values != null) { |
| sec.setStringList(section, subsection, name, values); |
| } else { |
| sec.unset(section, subsection, name); |
| } |
| save(); |
| } |
| |
| @Override |
| public void unset(String section, String subsection, String name) { |
| sec.unset(section, subsection, name); |
| save(); |
| } |
| |
| @Override |
| public Iterable<EntryKey> list() { |
| List<EntryKey> result = new ArrayList<>(); |
| for (String section : sec.getSections()) { |
| for (String subsection : sec.getSubsections(section)) { |
| for (String name : sec.getNames(section, subsection)) { |
| result.add(new EntryKey(section, subsection, name)); |
| } |
| } |
| for (String name : sec.getNames(section)) { |
| result.add(new EntryKey(section, null, name)); |
| } |
| } |
| return result; |
| } |
| |
| @Override |
| public boolean isOutdated() { |
| return sec.isOutdated(); |
| } |
| |
| @Override |
| public void reload() { |
| try { |
| sec.load(); |
| } catch (IOException | ConfigInvalidException e) { |
| throw new ProvisionException("Couldn't reload secure.config", e); |
| } |
| } |
| |
| private void save() { |
| try { |
| saveSecure(sec); |
| } catch (IOException e) { |
| throw new RuntimeException("Cannot save secure.config", e); |
| } |
| } |
| |
| private static void saveSecure(FileBasedConfig sec) throws IOException { |
| if (FileUtil.modified(sec)) { |
| final byte[] out = Constants.encode(sec.toText()); |
| final File path = sec.getFile(); |
| final LockFile lf = new LockFile(path); |
| if (!lf.lock()) { |
| throw new IOException("Cannot lock " + path); |
| } |
| try { |
| FileUtil.chmod(0600, new File(path.getParentFile(), path.getName() + ".lock")); |
| lf.write(out); |
| if (!lf.commit()) { |
| throw new IOException("Cannot commit write to " + path); |
| } |
| } finally { |
| lf.unlock(); |
| } |
| } |
| } |
| } |