tikhomirov@239: /* tikhomirov@239: * Copyright (c) 2011 TMate Software Ltd tikhomirov@239: * tikhomirov@239: * This program is free software; you can redistribute it and/or modify tikhomirov@239: * it under the terms of the GNU General Public License as published by tikhomirov@239: * the Free Software Foundation; version 2 of the License. tikhomirov@239: * tikhomirov@239: * This program is distributed in the hope that it will be useful, tikhomirov@239: * but WITHOUT ANY WARRANTY; without even the implied warranty of tikhomirov@239: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the tikhomirov@239: * GNU General Public License for more details. tikhomirov@239: * tikhomirov@239: * For information on how to redistribute this software under tikhomirov@239: * the terms of a license other than GNU General Public License tikhomirov@239: * contact TMate Software at support@hg4j.com tikhomirov@239: */ tikhomirov@239: package org.tmatesoft.hg.internal; tikhomirov@239: tikhomirov@239: import java.io.BufferedReader; tikhomirov@239: import java.io.File; tikhomirov@239: import java.io.FileReader; tikhomirov@239: import java.io.IOException; tikhomirov@239: import java.util.Arrays; tikhomirov@239: import java.util.Collections; tikhomirov@239: import java.util.HashMap; tikhomirov@239: import java.util.LinkedList; tikhomirov@239: import java.util.List; tikhomirov@239: import java.util.Map; tikhomirov@239: tikhomirov@423: import org.tmatesoft.hg.repo.HgInvalidControlFileException; tikhomirov@239: import org.tmatesoft.hg.repo.HgRepository; tikhomirov@239: import org.tmatesoft.hg.repo.HgSubrepoLocation; tikhomirov@239: tikhomirov@239: /** tikhomirov@239: * tikhomirov@239: * @author Artem Tikhomirov tikhomirov@239: * @author TMate Software Ltd. tikhomirov@239: */ tikhomirov@239: public class SubrepoManager /* XXX RepoChangeNotifier, RepoChangeListener */{ tikhomirov@239: tikhomirov@239: private final HgRepository repo; tikhomirov@239: private List subRepos; tikhomirov@239: tikhomirov@239: public SubrepoManager(HgRepository hgRepo) { tikhomirov@239: assert hgRepo != null; tikhomirov@239: repo = hgRepo; tikhomirov@239: } tikhomirov@239: tikhomirov@348: private List readActualState() throws HgInvalidControlFileException { tikhomirov@239: File hgsubFile = new File(repo.getWorkingDir(), ".hgsub"); tikhomirov@239: if (!hgsubFile.canRead()) { tikhomirov@239: return Collections.emptyList(); tikhomirov@239: } tikhomirov@348: Map state; // path -> revision tikhomirov@348: File hgstateFile = null; tikhomirov@239: try { tikhomirov@348: hgstateFile = new File(repo.getWorkingDir(), ".hgsubstate"); tikhomirov@239: if (hgstateFile.canRead()) { tikhomirov@239: state = readState(new BufferedReader(new FileReader(hgstateFile))); tikhomirov@239: } else { tikhomirov@239: state = Collections.emptyMap(); tikhomirov@239: } tikhomirov@348: } catch (IOException ex) { tikhomirov@348: throw new HgInvalidControlFileException("Subrepo state read failed", ex, hgstateFile); tikhomirov@348: } tikhomirov@348: try { tikhomirov@239: BufferedReader br = new BufferedReader(new FileReader(hgsubFile)); tikhomirov@239: return readConfig(br, state); tikhomirov@239: } catch (IOException ex) { tikhomirov@348: throw new HgInvalidControlFileException("Subrepo state read failed", ex, hgsubFile); tikhomirov@239: } tikhomirov@239: } tikhomirov@239: tikhomirov@239: private List readConfig(BufferedReader br, Map substate) throws IOException { tikhomirov@239: try { tikhomirov@239: String line; tikhomirov@239: LinkedList res = new LinkedList(); tikhomirov@239: while ((line = br.readLine()) != null) { tikhomirov@239: int sep = line.indexOf('='); tikhomirov@239: if (sep == -1) { tikhomirov@239: continue; tikhomirov@239: } tikhomirov@239: // since both key and value are referenced from HgSubrepoLocation, doesn't make sense tikhomirov@239: // to have separate String instances (new String(line.substring())) tikhomirov@239: String key = line.substring(0, sep).trim(); tikhomirov@239: String value = line.substring(sep + 1).trim(); tikhomirov@239: if (value.length() == 0) { tikhomirov@239: // XXX log bad line? tikhomirov@239: continue; tikhomirov@239: } tikhomirov@239: HgSubrepoLocation.Kind kind = HgSubrepoLocation.Kind.Hg; tikhomirov@239: int kindEnd = value.indexOf(']', 1); tikhomirov@239: if (value.charAt(0) == '[' && kindEnd != -1) { tikhomirov@239: String kindStr = value.substring(1, kindEnd); tikhomirov@239: value = value.substring(kindEnd + 1); tikhomirov@239: if ("svn".equals(kindStr)) { tikhomirov@239: kind = HgSubrepoLocation.Kind.SVN; tikhomirov@239: } else if ("git".equals(kindStr)) { tikhomirov@239: kind = HgSubrepoLocation.Kind.Git; tikhomirov@239: } tikhomirov@239: } tikhomirov@239: // TODO respect paths mappings in config file tikhomirov@239: HgSubrepoLocation loc = new HgSubrepoLocation(repo, key, value, kind, substate.get(key)); tikhomirov@239: res.add(loc); tikhomirov@239: } tikhomirov@239: return Arrays.asList(res.toArray(new HgSubrepoLocation[res.size()])); tikhomirov@239: } finally { tikhomirov@239: br.close(); tikhomirov@239: } tikhomirov@239: } tikhomirov@239: tikhomirov@239: private Map readState(BufferedReader br) throws IOException { tikhomirov@239: HashMap rv = new HashMap(); tikhomirov@239: try { tikhomirov@239: String line; tikhomirov@239: while ((line = br.readLine()) != null) { tikhomirov@239: int sep = line.trim().indexOf(' '); tikhomirov@239: if (sep != -1) { tikhomirov@239: rv.put(line.substring(sep+1).trim(), line.substring(0, sep).trim()); tikhomirov@239: } tikhomirov@239: } tikhomirov@239: } finally { tikhomirov@239: br.close(); tikhomirov@239: } tikhomirov@239: return rv; tikhomirov@239: } tikhomirov@239: tikhomirov@348: /*public to allow access from HgRepository, otherwise package-local*/ tikhomirov@348: public void read() throws HgInvalidControlFileException { tikhomirov@348: subRepos = readActualState(); tikhomirov@348: } tikhomirov@348: tikhomirov@239: public List all(/*int revision, or TIP|WC*/) { tikhomirov@348: assert subRepos != null; tikhomirov@239: return subRepos; tikhomirov@239: } tikhomirov@239: }