/*
* Copyright (C) 2008 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.
*/
import java.io.File;
import java.io.IOException;
import java.util.SortedSet;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.Collection;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Pattern;
/**
* Generates an Eclipse project.
*/
public class Eclipse {
/**
* Generates an Eclipse .classpath file from the given configuration.
*/
public static void generateFrom(Configuration c) throws IOException {
StringBuilder classpath = new StringBuilder();
classpath.append("\n"
+ "\n");
/*
* If the user has a file named "path-precedence" in their project's
* root directory, we'll order source roots based on how they match
* regular expressions in that file. Source roots that match earlier
* patterns will come sooner in configuration file.
*/
List patterns = new ArrayList();
File precedence = new File("path-precedence");
if (precedence.exists()) {
Configuration.parseFile(precedence, patterns);
} else {
// Put ./out at the bottom by default.
patterns.add(Pattern.compile("^(?!out/)"));
}
// Everything not matched by the user's precedence spec.
patterns.add(Pattern.compile(".*"));
List buckets = new ArrayList(patterns.size());
for (Pattern pattern : patterns) {
buckets.add(new Bucket(pattern));
}
// Put source roots in respective buckets.
OUTER: for (File sourceRoot : c.sourceRoots) {
// Trim preceding "./" from path.
String path = sourceRoot.getPath().substring(2);
for (Bucket bucket : buckets) {
if (bucket.matches(path)) {
bucket.sourceRoots.add(sourceRoot);
continue OUTER;
}
}
}
// Output source roots to configuration file.
for (Bucket bucket : buckets) {
for (File sourceRoot : bucket.sourceRoots) {
classpath.append(" 0) {
classpath.append(" excluding=\"")
.append(excluding).append("\"");
}
classpath.append(" path=\"")
.append(trimmed(sourceRoot)).append("\"/>\n");
}
}
// Output .jar entries.
for (File jar : c.jarFiles) {
classpath.append(" \n");
}
/*
* Output directory. Unfortunately, Eclipse forces us to put it
* somewhere under the project directory.
*/
classpath.append(" \n");
classpath.append("\n");
Files.toFile(classpath.toString(), new File(".classpath"));
}
/**
* Constructs the "excluding" argument for a given source root.
*/
private static CharSequence constructExcluding(File sourceRoot,
Configuration c) {
StringBuilder classpath = new StringBuilder();
String path = sourceRoot.getPath();
// Exclude nested source roots.
SortedSet nextRoots = c.sourceRoots.tailSet(sourceRoot);
int count = 0;
for (File nextRoot : nextRoots) {
// The first root is this root.
if (count == 0) {
count++;
continue;
}
String nextPath = nextRoot.getPath();
if (!nextPath.startsWith(path)) {
break;
}
if (count > 1) {
classpath.append('|');
}
classpath.append(nextPath.substring(path.length() + 1))
.append('/');
count++;
}
// Exclude excluded directories under this source root.
SortedSet excludedDirs = c.excludedDirs.tailSet(sourceRoot);
for (File excludedDir : excludedDirs) {
String excludedPath = excludedDir.getPath();
if (!excludedPath.startsWith(path)) {
break;
}
if (count > 1) {
classpath.append('|');
}
classpath.append(excludedPath.substring(path.length() + 1))
.append('/');
count++;
}
return classpath;
}
/**
* Returns the trimmed path.
*/
private static String trimmed(File file) {
return file.getPath().substring(2);
}
/**
* A precedence bucket for source roots.
*/
private static class Bucket {
private final Pattern pattern;
private final List sourceRoots = new ArrayList();
private Bucket(Pattern pattern) {
this.pattern = pattern;
}
private boolean matches(String path) {
return pattern.matcher(path).find();
}
}
}