summaryrefslogtreecommitdiffstats
path: root/org.eclipse.jgit.iplog/src
diff options
context:
space:
mode:
authorShawn O. Pearce <spearce@spearce.org>2010-01-28 11:13:11 -0800
committerShawn O. Pearce <spearce@spearce.org>2010-01-29 07:23:54 -0800
commit1e48c338dc237d5d4e314646d7b4d775249065c9 (patch)
tree83bfe50bbe62a6ed9f57ce8eae4530c4f682164a /org.eclipse.jgit.iplog/src
parentbaaa78f1f05d197ba2efefb713f194cd3f09725d (diff)
downloadjgit-1e48c338dc237d5d4e314646d7b4d775249065c9.tar.gz
jgit-1e48c338dc237d5d4e314646d7b4d775249065c9.zip
Generate an Eclipse IP log with jgit eclipse-iplog
The new plugin contains the bulk of the logic to scan a Git repository, and query IPZilla, in order to produce an XML formatted IP log for the requested revision of any Git based project. This plugin is suitable for embedding into a servlet container, or into the Eclipse workbench. The command line pgm package knows how to invoke this plugin through the eclipse-iplog subcommand, permitting storage of the resulting log as a local XML file. Change-Id: If01d9d98d07096db6980292bd5f91618c55d00be Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
Diffstat (limited to 'org.eclipse.jgit.iplog/src')
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CQ.java161
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CSV.java127
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Committer.java205
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Contributor.java98
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IPZillaQuery.java281
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogGenerator.java604
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogMeta.java199
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Project.java119
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SimpleCookieManager.java124
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SingleContribution.java112
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/gsql_query.txt22
-rw-r--r--org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/sample_gerrit_committers2
12 files changed, 2054 insertions, 0 deletions
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CQ.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CQ.java
new file mode 100644
index 0000000000..c2a4fe1195
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CQ.java
@@ -0,0 +1,161 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.util.Comparator;
+
+/**
+ * A contribution questionnaire stored in IPzilla.
+ *
+ * @see <a href="http://wiki.eclipse.org/IPzilla">IPzilla - Eclipsepedia</a>
+ * @see <a href="https://dev.eclipse.org/ipzilla/">IPzilla - Login</a>
+ */
+class CQ {
+ /** Sorts CQs by their unique number. */
+ static final Comparator<CQ> COMPARATOR = new Comparator<CQ>() {
+ public int compare(CQ a, CQ b) {
+ int cmp = state(a) - state(b);
+ if (cmp == 0)
+ cmp = compare(a.getID(), b.getID());
+ return cmp;
+ }
+
+ private int state(CQ a) {
+ if ("approved".equals(a.getState()))
+ return 1;
+ return 50;
+ }
+
+ private int compare(long a, long b) {
+ return a < b ? -1 : a == b ? 0 : 1;
+ }
+ };
+
+ private final long id;
+
+ private String description;
+
+ private String license;
+
+ private String use;
+
+ private String state;
+
+ private String comments;
+
+ /**
+ * @param id
+ */
+ CQ(final long id) {
+ this.id = id;
+ }
+
+ /** @return unique id number of the contribution questionnaire. */
+ long getID() {
+ return id;
+ }
+
+ /** @return short description of this CQ record. */
+ String getDescription() {
+ return description;
+ }
+
+ void setDescription(String description) {
+ this.description = description;
+ }
+
+ /** @return the license the contribution is under. */
+ String getLicense() {
+ return license;
+ }
+
+ void setLicense(String license) {
+ this.license = license;
+ }
+
+ /** @return how this code is used by the project, e.g. "unmodified binary". */
+ String getUse() {
+ return use;
+ }
+
+ void setUse(String use) {
+ this.use = use;
+ }
+
+ /** @return TODO find out what state is */
+ String getState() {
+ return state;
+ }
+
+ void setState(String state) {
+ this.state = state;
+ }
+
+ /** @return any additional comments about this particular CQ. */
+ String getComments() {
+ return comments;
+ }
+
+ void setComments(String comments) {
+ this.comments = comments;
+ }
+
+ @Override
+ public int hashCode() {
+ return (int) getID();
+ }
+
+ @Override
+ public boolean equals(Object other) {
+ if (other instanceof CQ) {
+ return ((CQ) other).getID() == getID();
+ }
+ return false;
+ }
+
+ @Override
+ public String toString() {
+ return "CQ " + getID();
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CSV.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CSV.java
new file mode 100644
index 0000000000..8c57edb00f
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/CSV.java
@@ -0,0 +1,127 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+
+/** A crude CSV file parser. */
+class CSV {
+ private final BufferedReader in;
+
+ private List<String> columns;
+
+ CSV(BufferedReader br) throws IOException {
+ in = br;
+ columns = readLine();
+ }
+
+ Map<String, String> next() throws IOException {
+ List<String> row = readLine();
+ if (columns == null || row == null)
+ return null;
+
+ Map<String, String> r = new LinkedHashMap<String, String>();
+ for (int col = 0; col < columns.size(); col++)
+ r.put(columns.get(col), row.get(col));
+ return r;
+ }
+
+ private List<String> readLine() throws IOException {
+ String line = in.readLine();
+ if (line == null || line.length() == 0)
+ return null;
+
+ ArrayList<String> row;
+ if (columns != null)
+ row = new ArrayList<String>(columns.size());
+ else
+ row = new ArrayList<String>();
+
+ int p = 0;
+ while (p < line.length()) {
+ if (line.charAt(p) == '"') {
+ p++; // skip the opening quote.
+
+ StringBuilder b = new StringBuilder();
+ SCAN: while (p < line.length()) {
+ char c = line.charAt(p);
+ switch (c) {
+ case '"':
+ p++;
+ break SCAN;
+
+ case '\\':
+ b.append(line.charAt(p + 1));
+ p += 2;
+ break;
+
+ default:
+ b.append(c);
+ p++;
+ break;
+ }
+ }
+ if (p < line.length() && line.charAt(p) != ',')
+ throw new IOException("CSV parsing error: " + line);
+ row.add(b.toString());
+ p++; // skip the trailing comma (if present)
+
+ } else if (line.charAt(p) == ',') {
+ row.add("");
+ p++;
+
+ } else {
+ int comma = line.indexOf(',', p);
+ row.add(line.substring(p, comma));
+ p = comma + 1;
+ }
+ }
+ return row;
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Committer.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Committer.java
new file mode 100644
index 0000000000..fe84a08ecf
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Committer.java
@@ -0,0 +1,205 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.Date;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+
+/** A project committer. */
+class Committer {
+ /** Sorts committers by their name first name, then last name. */
+ static final Comparator<Committer> COMPARATOR = new Comparator<Committer>() {
+ public int compare(Committer a, Committer b) {
+ int cmp = a.firstName.compareTo(b.firstName);
+ if (cmp == 0)
+ cmp = a.lastName.compareTo(b.lastName);
+ return cmp;
+ }
+ };
+
+ private final String id;
+
+ private String firstName;
+
+ private String lastName;
+
+ private String affiliation;
+
+ private boolean hasCommits;
+
+ private String comments;
+
+ private final Set<String> emailAddresses = new HashSet<String>();
+
+ private final List<ActiveRange> active = new ArrayList<ActiveRange>(2);
+
+ /**
+ * @param id
+ * unique identity of the committer
+ */
+ Committer(String id) {
+ this.id = id;
+ }
+
+ /** @return unique identity of this committer in the foundation database. */
+ String getID() {
+ return id;
+ }
+
+ /** @return first name of the committer; their given name. */
+ String getFirstName() {
+ return firstName;
+ }
+
+ void setFirstName(String firstName) {
+ this.firstName = firstName;
+ }
+
+ /** @return last name of the committer; their surname or family name. */
+ String getLastName() {
+ return lastName;
+ }
+
+ void setLastName(String lastName) {
+ this.lastName = lastName;
+ }
+
+ /** @return the organization the committer is affiliated with. */
+ String getAffiliation() {
+ return affiliation;
+ }
+
+ void setAffiliation(String affiliation) {
+ this.affiliation = affiliation;
+ }
+
+ /** @return true if this committer is still an active member of the project. */
+ boolean isActive() {
+ if (active.isEmpty())
+ return false;
+ ActiveRange last = active.get(active.size() - 1);
+ return last.end == null;
+ }
+
+ /** @return true if this committer has commits in the project. */
+ boolean hasCommits() {
+ return hasCommits;
+ }
+
+ void setHasCommits(boolean hasCommits) {
+ this.hasCommits = hasCommits;
+ }
+
+ /** @return any additional comments about this committer. */
+ String getComments() {
+ return comments;
+ }
+
+ void setComments(String comments) {
+ this.comments = comments;
+ }
+
+ void addEmailAddress(String email) {
+ emailAddresses.add(email);
+ }
+
+ void addActiveRange(ActiveRange r) {
+ active.add(r);
+ Collections.sort(active, new Comparator<ActiveRange>() {
+ public int compare(ActiveRange a, ActiveRange b) {
+ return a.begin.compareTo(b.begin);
+ }
+ });
+ }
+
+ /**
+ * @param when
+ * @return true if the event occurred while an active committer.
+ */
+ boolean inRange(Date when) {
+ for (ActiveRange ar : active) {
+ if (ar.contains(when))
+ return true;
+ }
+ return false;
+ }
+
+ @Override
+ public String toString() {
+ return "Committer " + getFirstName() + " " + getLastName();
+ }
+
+ /** Date period during which the committer was active. */
+ static class ActiveRange {
+ private final Date begin;
+
+ private final Date end;
+
+ /**
+ * @param begin
+ * @param end
+ */
+ ActiveRange(Date begin, Date end) {
+ this.begin = begin;
+ this.end = end;
+ }
+
+ /**
+ * @param when
+ * @return true if {@code when} is within this date span.
+ */
+ boolean contains(Date when) {
+ if (when.compareTo(begin) < 0)
+ return false;
+ if (end == null)
+ return true;
+ return when.compareTo(end) < 0;
+ }
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Contributor.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Contributor.java
new file mode 100644
index 0000000000..cf1f7c1013
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Contributor.java
@@ -0,0 +1,98 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.List;
+
+/** A project contributor (non-committer). */
+class Contributor {
+ /** Sorts contributors by their name first name, then last name. */
+ static final Comparator<Contributor> COMPARATOR = new Comparator<Contributor>() {
+ public int compare(Contributor a, Contributor b) {
+ return a.name.compareTo(b.name);
+ }
+ };
+
+ private final String id;
+
+ private final String name;
+
+ private final List<SingleContribution> contributions = new ArrayList<SingleContribution>();
+
+ /**
+ * @param name
+ */
+ Contributor(String id, String name) {
+ this.id = id;
+ this.name = name;
+ }
+
+ /** @return unique identity of this contributor in the foundation database. */
+ String getID() {
+ return id;
+ }
+
+ /** @return name of the contributor. */
+ String getName() {
+ return name;
+ }
+
+ /** @return all known contributions. */
+ Collection<SingleContribution> getContributions() {
+ return Collections.unmodifiableCollection(contributions);
+ }
+
+ void add(SingleContribution bug) {
+ contributions.add(bug);
+ }
+
+ @Override
+ public String toString() {
+ return "Contributor " + getName();
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IPZillaQuery.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IPZillaQuery.java
new file mode 100644
index 0000000000..74bc10c50a
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IPZillaQuery.java
@@ -0,0 +1,281 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.io.OutputStream;
+import java.io.UnsupportedEncodingException;
+import java.net.ConnectException;
+import java.net.CookieHandler;
+import java.net.HttpURLConnection;
+import java.net.MalformedURLException;
+import java.net.Proxy;
+import java.net.ProxySelector;
+import java.net.URISyntaxException;
+import java.net.URL;
+import java.net.URLEncoder;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.TreeSet;
+
+import org.eclipse.jgit.util.HttpSupport;
+
+/** A crude interface to query IPzilla. */
+class IPZillaQuery {
+ private static final String RE_EPL = "^.*(Eclipse Public License|EPL).*$";
+
+ private final URL base;
+
+ private final String username;
+
+ private final String password;
+
+ private final ProxySelector proxySelector = ProxySelector.getDefault();
+
+ IPZillaQuery(URL base, String username, String password) {
+ this.base = base;
+ this.username = username;
+ this.password = password;
+ }
+
+ Set<CQ> getCQs(Collection<Project> projects) throws IOException {
+ try {
+ login();
+ Set<CQ> cqs = new HashSet<CQ>();
+ for (Project project : projects)
+ cqs.addAll(queryOneProject(project));
+ return cqs;
+ } finally {
+ // Kill the IPzilla session and log us out from there.
+ logout();
+ }
+ }
+
+ private Set<CQ> queryOneProject(Project project) throws IOException {
+ Map<String, String> p = new LinkedHashMap<String, String>();
+ p.put("bugidtype", "include");
+ p.put("chfieldto", "Now");
+ p.put("component", project.getID());
+ p.put("field-1-0-0", "component");
+ p.put("type-1-0-0", "anyexact");
+ p.put("value-1-0-0", project.getID());
+ p.put("ctype", "csv");
+
+ StringBuilder req = new StringBuilder();
+ for (Map.Entry<String, String> e : p.entrySet()) {
+ if (req.length() > 0)
+ req.append('&');
+ req.append(URLEncoder.encode(e.getKey(), "UTF-8"));
+ req.append('=');
+ req.append(URLEncoder.encode(e.getValue(), "UTF-8"));
+ }
+ URL csv = new URL(new URL(base, "buglist.cgi").toString() + "?" + req);
+
+ req = new StringBuilder();
+ for (String name : new String[] { "bug_severity", "bug_status",
+ "resolution", "short_desc", "cf_license", "keywords" }) {
+ if (req.length() > 0)
+ req.append("%20");
+ req.append(name);
+ }
+ setCookie(csv, "COLUMNLIST", req.toString());
+
+ HttpURLConnection conn = open(csv);
+ if (HttpSupport.response(conn) != HttpURLConnection.HTTP_OK) {
+ throw new IOException("Query " + csv + " failed: "
+ + conn.getResponseCode() + " " + conn.getResponseMessage());
+ }
+
+ BufferedReader br = reader(conn);
+ try {
+ Set<CQ> cqs = new HashSet<CQ>();
+ CSV in = new CSV(br);
+ Map<String, String> row;
+ while ((row = in.next()) != null) {
+ CQ cq = parseOneCQ(row);
+ if (cq != null)
+ cqs.add(cq);
+ }
+ return cqs;
+ } finally {
+ br.close();
+ }
+ }
+
+ private BufferedReader reader(HttpURLConnection conn)
+ throws UnsupportedEncodingException, IOException {
+ String encoding = conn.getContentEncoding();
+ InputStream in = conn.getInputStream();
+ if (encoding != null && !encoding.equals(""))
+ return new BufferedReader(new InputStreamReader(in, encoding));
+ return new BufferedReader(new InputStreamReader(in));
+ }
+
+ private void login() throws MalformedURLException,
+ UnsupportedEncodingException, ConnectException, IOException {
+ final URL login = new URL(base, "index.cgi");
+ StringBuilder req = new StringBuilder();
+ req.append("Bugzilla_login=");
+ req.append(URLEncoder.encode(username, "UTF-8"));
+ req.append('&');
+ req.append("Bugzilla_password=");
+ req.append(URLEncoder.encode(password, "UTF-8"));
+ byte[] reqbin = req.toString().getBytes("UTF-8");
+
+ HttpURLConnection c = open(login);
+ c.setDoOutput(true);
+ c.setFixedLengthStreamingMode(reqbin.length);
+ c.setRequestProperty(HttpSupport.HDR_CONTENT_TYPE,
+ "application/x-www-form-urlencoded");
+ OutputStream out = c.getOutputStream();
+ out.write(reqbin);
+ out.close();
+
+ if (HttpSupport.response(c) != HttpURLConnection.HTTP_OK) {
+ throw new IOException("Login as " + username + " to " + login
+ + " failed: " + c.getResponseCode() + " "
+ + c.getResponseMessage());
+ }
+ }
+
+ private void logout() throws MalformedURLException, ConnectException,
+ IOException {
+ HttpSupport.response(open(new URL(base, "relogin.cgi")));
+ }
+
+ private HttpURLConnection open(URL url) throws ConnectException,
+ IOException {
+ Proxy proxy = HttpSupport.proxyFor(proxySelector, url);
+ HttpURLConnection c = (HttpURLConnection) url.openConnection(proxy);
+ c.setUseCaches(false);
+ return c;
+ }
+
+ private void setCookie(URL url, String name, String value)
+ throws IOException {
+ Map<String, List<String>> cols = new HashMap<String, List<String>>();
+ cols.put("Set-Cookie", Collections.singletonList(name + "=" + value));
+ try {
+ CookieHandler.getDefault().put(url.toURI(), cols);
+ } catch (URISyntaxException e) {
+ IOException err = new IOException("Invalid URI format:" + url);
+ err.initCause(e);
+ throw err;
+ }
+ }
+
+ private CQ parseOneCQ(Map<String, String> row) {
+ long id = Long.parseLong(row.get("bug_id"));
+ String state = row.get("bug_severity");
+ String bug_status = row.get("bug_status");
+ String resolution = row.get("resolution");
+ String short_desc = row.get("short_desc");
+ String license = row.get("cf_license");
+
+ Set<String> keywords = new TreeSet<String>();
+ for (String w : row.get("keywords").split(", *"))
+ keywords.add(w);
+
+ // Skip any CQs that were not accepted.
+ //
+ if ("closed".equalsIgnoreCase(state)
+ || "rejected".equalsIgnoreCase(state)
+ || "withdrawn".equalsIgnoreCase(state))
+ return null;
+
+ // Skip any CQs under the EPL without nonepl keyword
+ // Skip any CQs with the EPL keyword
+ //
+ if (!keywords.contains("nonepl") && license.matches(RE_EPL))
+ return null;
+ if (keywords.contains("epl"))
+ return null;
+
+ // Work around CQs that were closed in the wrong state.
+ //
+ if ("new".equalsIgnoreCase(state)
+ || "under_review".equalsIgnoreCase(state)
+ || state.startsWith("awaiting_")) {
+ if ("RESOLVED".equalsIgnoreCase(bug_status)
+ || "CLOSED".equalsIgnoreCase(bug_status)) {
+ if ("FIXED".equalsIgnoreCase(resolution))
+ state = "approved";
+ else
+ return null;
+ }
+ }
+
+ StringBuilder use = new StringBuilder();
+ for (String n : new String[] { "unmodified", "modified", "source",
+ "binary" }) {
+ if (keywords.contains(n)) {
+ if (use.length() > 0)
+ use.append(' ');
+ use.append(n);
+ }
+ }
+ if (keywords.contains("sourceandbinary")) {
+ if (use.length() > 0)
+ use.append(' ');
+ use.append("source & binary");
+ }
+
+ CQ cq = new CQ(id);
+ cq.setDescription(short_desc);
+ cq.setLicense(license);
+ cq.setState(state);
+ if (use.length() > 0)
+ cq.setUse(use.toString().trim());
+ return cq;
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogGenerator.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogGenerator.java
new file mode 100644
index 0000000000..1f60ee5628
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogGenerator.java
@@ -0,0 +1,604 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.io.BufferedReader;
+import java.io.File;
+import java.io.FileReader;
+import java.io.IOException;
+import java.io.OutputStream;
+import java.text.ParseException;
+import java.text.SimpleDateFormat;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.Date;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.TreeMap;
+import java.util.TreeSet;
+
+import javax.xml.parsers.DocumentBuilderFactory;
+import javax.xml.parsers.ParserConfigurationException;
+import javax.xml.transform.OutputKeys;
+import javax.xml.transform.Transformer;
+import javax.xml.transform.TransformerConfigurationException;
+import javax.xml.transform.TransformerException;
+import javax.xml.transform.TransformerFactory;
+import javax.xml.transform.dom.DOMSource;
+import javax.xml.transform.stream.StreamResult;
+
+import org.eclipse.jgit.diff.Edit;
+import org.eclipse.jgit.diff.EditList;
+import org.eclipse.jgit.diff.MyersDiff;
+import org.eclipse.jgit.diff.RawText;
+import org.eclipse.jgit.errors.ConfigInvalidException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.iplog.Committer.ActiveRange;
+import org.eclipse.jgit.lib.BlobBasedConfig;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.MutableObjectId;
+import org.eclipse.jgit.lib.ObjectLoader;
+import org.eclipse.jgit.lib.PersonIdent;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.lib.WindowCursor;
+import org.eclipse.jgit.revwalk.FooterKey;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevTree;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.treewalk.NameConflictTreeWalk;
+import org.eclipse.jgit.treewalk.TreeWalk;
+import org.eclipse.jgit.treewalk.filter.TreeFilter;
+import org.eclipse.jgit.util.RawParseUtils;
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+
+/**
+ * Creates an Eclipse IP log in XML format.
+ *
+ * @see <a href="http://www.eclipse.org/projects/xml/iplog.xsd">IP log XSD</a>
+ */
+public class IpLogGenerator {
+ private static final String IPLOG_NS = "http://www.eclipse.org/projects/xml/iplog";
+
+ private static final String IPLOG_PFX = "iplog:";
+
+ private static final String INDENT = "{http://xml.apache.org/xslt}indent-amount";
+
+ private static final FooterKey BUG = new FooterKey("Bug");
+
+ /** Projects indexed by their ID string, e.g. {@code technology.jgit}. */
+ private final Map<String, Project> projects = new TreeMap<String, Project>();
+
+ /** Known committers, indexed by their foundation ID. */
+ private final Map<String, Committer> committersById = new HashMap<String, Committer>();
+
+ /** Known committers, indexed by their email address. */
+ private final Map<String, Committer> committersByEmail = new HashMap<String, Committer>();
+
+ /** Discovered contributors. */
+ private final Map<String, Contributor> contributorsByName = new HashMap<String, Contributor>();
+
+ /** All known CQs matching the projects we care about. */
+ private final Set<CQ> cqs = new HashSet<CQ>();
+
+ /** Root commits which were scanned to gather project data. */
+ private final Set<RevCommit> commits = new HashSet<RevCommit>();
+
+ private String characterEncoding = "UTF-8";
+
+ private Repository db;
+
+ private RevWalk rw;
+
+ private NameConflictTreeWalk tw;
+
+ private final WindowCursor curs = new WindowCursor();
+
+ private final MutableObjectId idbuf = new MutableObjectId();
+
+ private Document doc;
+
+ /** Create an empty generator. */
+ public IpLogGenerator() {
+ // Do nothing.
+ }
+
+ /**
+ * Set the character encoding used to write the output file.
+ *
+ * @param encodingName
+ * the character set encoding name.
+ */
+ public void setCharacterEncoding(String encodingName) {
+ characterEncoding = encodingName;
+ }
+
+ /**
+ * Scan a Git repository's history to compute the changes within it.
+ *
+ * @param repo
+ * the repository to scan.
+ * @param startCommit
+ * commit the IP log is needed for.
+ * @param version
+ * symbolic label for the version.
+ * @throws IOException
+ * the repository cannot be read.
+ * @throws ConfigInvalidException
+ * the {@code .eclipse_iplog} file present at the top level of
+ * {@code startId} is not a valid configuration file.
+ */
+ public void scan(Repository repo, RevCommit startCommit, String version)
+ throws IOException, ConfigInvalidException {
+ try {
+ db = repo;
+ rw = new RevWalk(db);
+ tw = new NameConflictTreeWalk(db);
+
+ RevCommit c = rw.parseCommit(startCommit);
+
+ loadEclipseIpLog(version, c);
+ loadCommitters(repo);
+ scanProjectCommits(c);
+ commits.add(c);
+ } finally {
+ WindowCursor.release(curs);
+ db = null;
+ rw = null;
+ tw = null;
+ }
+ }
+
+ private void loadEclipseIpLog(String version, RevCommit commit)
+ throws IOException, ConfigInvalidException {
+ TreeWalk log = TreeWalk.forPath(db, IpLogMeta.IPLOG_CONFIG_FILE, commit
+ .getTree());
+ if (log == null)
+ return;
+
+ IpLogMeta meta = new IpLogMeta();
+ try {
+ meta.loadFrom(new BlobBasedConfig(null, db, log.getObjectId(0)));
+ } catch (ConfigInvalidException e) {
+ throw new ConfigInvalidException("Configuration file "
+ + log.getPathString() + " in commit " + commit.name()
+ + " is invalid", e);
+ }
+
+ for (Project p : meta.getProjects()) {
+ p.setVersion(version);
+ projects.put(p.getName(), p);
+ }
+ cqs.addAll(meta.getCQs());
+ }
+
+ private void loadCommitters(Repository repo) throws IOException {
+ SimpleDateFormat dt = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
+ File list = new File(repo.getDirectory(), "gerrit_committers");
+ BufferedReader br = new BufferedReader(new FileReader(list));
+ String line;
+
+ while ((line = br.readLine()) != null) {
+ String[] field = line.trim().split(" *\\| *");
+ String user = field[1];
+ String name = field[2];
+ String email = field[3];
+ Date begin = parseDate(dt, field[4]);
+ Date end = parseDate(dt, field[5]);
+
+ if (user.startsWith("username:"))
+ user = user.substring("username:".length());
+
+ Committer who = committersById.get(user);
+ if (who == null) {
+ who = new Committer(user);
+ int sp = name.indexOf(' ');
+ if (0 < sp) {
+ who.setFirstName(name.substring(0, sp).trim());
+ who.setLastName(name.substring(sp + 1).trim());
+ } else {
+ who.setFirstName(name);
+ who.setLastName(null);
+ }
+ committersById.put(who.getID(), who);
+ }
+
+ who.addEmailAddress(email);
+ who.addActiveRange(new ActiveRange(begin, end));
+ committersByEmail.put(email, who);
+ }
+ }
+
+ private Date parseDate(SimpleDateFormat dt, String value)
+ throws IOException {
+ if ("NULL".equals(value) || "".equals(value) || value == null)
+ return null;
+ int dot = value.indexOf('.');
+ if (0 < dot)
+ value = value.substring(0, dot);
+ try {
+ return dt.parse(value);
+ } catch (ParseException e) {
+ IOException err = new IOException("Invalid date: " + value);
+ err.initCause(e);
+ throw err;
+ }
+ }
+
+ private void scanProjectCommits(RevCommit start) throws IOException {
+ rw.reset();
+ rw.markStart(start);
+
+ RevCommit commit;
+ while ((commit = rw.next()) != null) {
+ final PersonIdent author = commit.getAuthorIdent();
+ final Date when = author.getWhen();
+
+ Committer who = committersByEmail.get(author.getEmailAddress());
+ if (who != null && who.inRange(when)) {
+ // Commit was written by the committer while they were
+ // an active committer on the project.
+ //
+ who.setHasCommits(true);
+ continue;
+ }
+
+ // Commit from a non-committer contributor.
+ //
+ final int cnt = commit.getParentCount();
+ if (2 <= cnt) {
+ // Avoid a pointless merge attributed to a non-committer.
+ // Skip this commit if every file matches at least one
+ // of the parent commits exactly, if so then the blame
+ // for code in that file can be fully passed onto that
+ // parent and this non-committer isn't responsible.
+ //
+ tw.setFilter(TreeFilter.ANY_DIFF);
+ tw.setRecursive(true);
+
+ RevTree[] trees = new RevTree[1 + cnt];
+ trees[0] = commit.getTree();
+ for (int i = 0; i < cnt; i++)
+ trees[i + 1] = commit.getParent(i).getTree();
+ tw.reset(trees);
+
+ boolean matchAll = true;
+ while (tw.next()) {
+ boolean matchOne = false;
+ for (int i = 1; i <= cnt; i++) {
+ if (tw.getRawMode(0) == tw.getRawMode(i)
+ && tw.idEqual(0, i)) {
+ matchOne = true;
+ break;
+ }
+ }
+ if (!matchOne) {
+ matchAll = false;
+ break;
+ }
+ }
+ if (matchAll)
+ continue;
+ }
+
+ Contributor contributor = contributorsByName.get(author.getName());
+ if (contributor == null) {
+ String id = author.getEmailAddress();
+ String name = author.getName();
+ contributor = new Contributor(id, name);
+ contributorsByName.put(name, contributor);
+ }
+
+ String id = commit.name();
+ String subj = commit.getShortMessage();
+ SingleContribution item = new SingleContribution(id, when, subj);
+
+ List<String> bugs = commit.getFooterLines(BUG);
+ if (1 == bugs.size()) {
+ item.setBugID(bugs.get(0));
+
+ } else if (2 <= bugs.size()) {
+ StringBuilder tmp = new StringBuilder();
+ for (String bug : bugs) {
+ if (tmp.length() > 0)
+ tmp.append(",");
+ tmp.append(bug);
+ }
+ item.setBugID(tmp.toString());
+ }
+
+ if (2 <= cnt) {
+ item.setSize("(merge)");
+ contributor.add(item);
+ continue;
+ }
+
+ int addedLines = 0;
+ if (1 == cnt) {
+ final RevCommit parent = commit.getParent(0);
+ tw.setFilter(TreeFilter.ANY_DIFF);
+ tw.setRecursive(true);
+ tw.reset(new RevTree[] { parent.getTree(), commit.getTree() });
+ while (tw.next()) {
+ if (tw.getFileMode(1).getObjectType() != Constants.OBJ_BLOB)
+ continue;
+
+ byte[] oldImage;
+ if (tw.getFileMode(0).getObjectType() == Constants.OBJ_BLOB)
+ oldImage = openBlob(0);
+ else
+ oldImage = new byte[0];
+
+ EditList edits = new MyersDiff(new RawText(oldImage),
+ new RawText(openBlob(1))).getEdits();
+ for (Edit e : edits)
+ addedLines += e.getEndB() - e.getBeginB();
+ }
+
+ } else { // no parents, everything is an addition
+ tw.setFilter(TreeFilter.ALL);
+ tw.setRecursive(true);
+ tw.reset(commit.getTree());
+ while (tw.next()) {
+ if (tw.getFileMode(0).getObjectType() == Constants.OBJ_BLOB) {
+ byte[] buf = openBlob(0);
+ for (int ptr = 0; ptr < buf.length;) {
+ ptr += RawParseUtils.nextLF(buf, ptr);
+ addedLines++;
+ }
+ }
+ }
+ }
+
+ if (addedLines < 0)
+ throw new IOException("Incorrectly scanned " + commit.name());
+ if (1 == addedLines)
+ item.setSize("+1 line");
+ else
+ item.setSize("+" + addedLines + " lines");
+ contributor.add(item);
+ }
+ }
+
+ private byte[] openBlob(int side) throws IOException {
+ tw.getObjectId(idbuf, side);
+ ObjectLoader ldr = db.openObject(curs, idbuf);
+ if (ldr == null)
+ throw new MissingObjectException(idbuf.copy(), Constants.OBJ_BLOB);
+ return ldr.getCachedBytes();
+ }
+
+ /**
+ * Dump the scanned information into an XML file.
+ *
+ * @param out
+ * the file stream to write to. The caller is responsible for
+ * closing the stream upon completion.
+ * @throws IOException
+ * the stream cannot be written.
+ */
+ public void writeTo(OutputStream out) throws IOException {
+ try {
+ TransformerFactory factory = TransformerFactory.newInstance();
+ Transformer s = factory.newTransformer();
+ s.setOutputProperty(OutputKeys.ENCODING, characterEncoding);
+ s.setOutputProperty(OutputKeys.METHOD, "xml");
+ s.setOutputProperty(OutputKeys.INDENT, "yes");
+ s.setOutputProperty(INDENT, "2");
+ s.transform(new DOMSource(toXML()), new StreamResult(out));
+ } catch (ParserConfigurationException e) {
+ IOException err = new IOException("Cannot serialize XML");
+ err.initCause(e);
+ throw err;
+
+ } catch (TransformerConfigurationException e) {
+ IOException err = new IOException("Cannot serialize XML");
+ err.initCause(e);
+ throw err;
+
+ } catch (TransformerException e) {
+ IOException err = new IOException("Cannot serialize XML");
+ err.initCause(e);
+ throw err;
+ }
+ }
+
+ private Document toXML() throws ParserConfigurationException {
+ DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
+ factory.setNamespaceAware(true);
+ doc = factory.newDocumentBuilder().newDocument();
+
+ Element root = createElement("iplog");
+ doc.appendChild(root);
+
+ if (projects.size() == 1) {
+ Project soleProject = projects.values().iterator().next();
+ root.setAttribute("name", soleProject.getID());
+ }
+
+ Set<String> licenses = new TreeSet<String>();
+ for (Project project : sort(projects, Project.COMPARATOR)) {
+ root.appendChild(createProject(project));
+ licenses.addAll(project.getLicenses());
+ }
+ for (RevCommit c : sort(commits))
+ root.appendChild(createCommitMeta(c));
+ for (String name : sort(licenses))
+ root.appendChild(createLicense(name));
+
+ if (!cqs.isEmpty())
+ appendBlankLine(root);
+ for (CQ cq : sort(cqs, CQ.COMPARATOR))
+ root.appendChild(createCQ(cq));
+
+ if (!committersByEmail.isEmpty())
+ appendBlankLine(root);
+ for (Committer committer : sort(committersById, Committer.COMPARATOR))
+ root.appendChild(createCommitter(committer));
+
+ for (Contributor c : sort(contributorsByName, Contributor.COMPARATOR)) {
+ appendBlankLine(root);
+ root.appendChild(createContributor(c));
+ }
+
+ return doc;
+ }
+
+ private void appendBlankLine(Element root) {
+ root.appendChild(doc.createTextNode("\n\n "));
+ }
+
+ private Element createProject(Project p) {
+ Element project = createElement("project");
+ required(project, "id", p.getID());
+ required(project, "name", p.getName());
+ optional(project, "comments", p.getComments());
+ optional(project, "version", p.getVersion());
+ return project;
+ }
+
+ private Element createCommitMeta(RevCommit c) {
+ Element meta = createElement("meta");
+ required(meta, "key", "git-commit");
+ required(meta, "value", c.name());
+ return meta;
+ }
+
+ private Element createLicense(String name) {
+ Element license = createElement("license");
+ required(license, "id", name);
+ optional(license, "description", null);
+ optional(license, "comments", null);
+ return license;
+ }
+
+ private Element createCQ(CQ cq) {
+ Element r = createElement("cq");
+ required(r, "id", Long.toString(cq.getID()));
+ required(r, "description", cq.getDescription());
+ optional(r, "license", cq.getLicense());
+ optional(r, "use", cq.getUse());
+ optional(r, "state", cq.getState());
+ optional(r, "comments", cq.getComments());
+ return r;
+ }
+
+ private Element createCommitter(Committer who) {
+ Element r = createElement("committer");
+ required(r, "id", who.getID());
+ required(r, "firstName", who.getFirstName());
+ required(r, "lastName", who.getLastName());
+ optional(r, "affiliation", who.getAffiliation());
+ required(r, "active", Boolean.toString(who.isActive()));
+ required(r, "hasCommits", Boolean.toString(who.hasCommits()));
+ optional(r, "comments", who.getComments());
+ return r;
+ }
+
+ private Element createContributor(Contributor c) {
+ Element r = createElement("contributor");
+ required(r, "id", c.getID());
+ required(r, "name", c.getName());
+
+ for (SingleContribution s : sort(c.getContributions(),
+ SingleContribution.COMPARATOR))
+ r.appendChild(createContribution(s));
+
+ return r;
+ }
+
+ private Element createContribution(SingleContribution s) {
+ Element r = createElement("bug");
+ required(r, "id", s.getID());
+ optional(r, "bug-id", s.getBugID());
+ required(r, "size", s.getSize());
+ required(r, "type", "A"); // assume attachment type
+ required(r, "created", format(s.getCreated()));
+ required(r, "summary", s.getSummary());
+ return r;
+ }
+
+ private String format(Date created) {
+ return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(created);
+ }
+
+ private Element createElement(String name) {
+ return doc.createElementNS(IPLOG_NS, IPLOG_PFX + name);
+ }
+
+ private void required(Element r, String name, String value) {
+ if (value == null)
+ value = "";
+ r.setAttribute(name, value);
+ }
+
+ private void optional(Element r, String name, String value) {
+ if (value != null && value.length() > 0)
+ r.setAttribute(name, value);
+ }
+
+ private static <T, Q extends Comparator<T>> Iterable<T> sort(
+ Collection<T> objs, Q cmp) {
+ ArrayList<T> sorted = new ArrayList<T>(objs);
+ Collections.sort(sorted, cmp);
+ return sorted;
+ }
+
+ private static <T, Q extends Comparator<T>> Iterable<T> sort(
+ Map<?, T> objs, Q cmp) {
+ return sort(objs.values(), cmp);
+ }
+
+ @SuppressWarnings("unchecked")
+ private static <T extends Comparable> Iterable<T> sort(Collection<T> objs) {
+ ArrayList<T> sorted = new ArrayList<T>(objs);
+ Collections.sort(sorted);
+ return sorted;
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogMeta.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogMeta.java
new file mode 100644
index 0000000000..38b7d417b1
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/IpLogMeta.java
@@ -0,0 +1,199 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.io.File;
+import java.io.IOException;
+import java.net.URL;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+
+import org.eclipse.jgit.errors.ConfigInvalidException;
+import org.eclipse.jgit.lib.Config;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.FileBasedConfig;
+import org.eclipse.jgit.lib.LockFile;
+
+/**
+ * Manages the {@code .eclipse_iplog} file in a project.
+ */
+public class IpLogMeta {
+ /** Default name of the {@code .eclipse_iplog} file. */
+ public static final String IPLOG_CONFIG_FILE = ".eclipse_iplog";
+
+ private static final String S_PROJECT = "project";
+
+ private static final String S_CQ = "CQ";
+
+ private static final String K_NAME = "name";
+
+ private static final String K_COMMENTS = "comments";
+
+ private static final String K_LICENSE = "license";
+
+ private static final String K_DESCRIPTION = "description";
+
+ private static final String K_USE = "use";
+
+ private static final String K_STATE = "state";
+
+ private List<Project> projects = new ArrayList<Project>();
+
+ private Set<CQ> cqs = new HashSet<CQ>();
+
+ List<Project> getProjects() {
+ return projects;
+ }
+
+ Set<CQ> getCQs() {
+ return cqs;
+ }
+
+ void loadFrom(Config cfg) {
+ projects.clear();
+ cqs.clear();
+
+ for (String id : cfg.getSubsections(S_PROJECT)) {
+ String name = cfg.getString(S_PROJECT, id, K_NAME);
+ Project project = new Project(id, name);
+ project.setComments(cfg.getString(S_PROJECT, id, K_COMMENTS));
+
+ for (String license : cfg.getStringList(S_PROJECT, id, K_LICENSE))
+ project.addLicense(license);
+ projects.add(project);
+ }
+
+ for (String id : cfg.getSubsections(S_CQ)) {
+ CQ cq = new CQ(Long.parseLong(id));
+ cq.setDescription(cfg.getString(S_CQ, id, K_DESCRIPTION));
+ cq.setLicense(cfg.getString(S_CQ, id, K_LICENSE));
+ cq.setUse(cfg.getString(S_CQ, id, K_USE));
+ cq.setState(cfg.getString(S_CQ, id, K_STATE));
+ cq.setComments(cfg.getString(S_CQ, id, K_COMMENTS));
+ cqs.add(cq);
+ }
+ }
+
+ /**
+ * Query the Eclipse Foundation's IPzilla database for CQ records.
+ * <p>
+ * Updates the local {@code .eclipse_iplog} configuration file with current
+ * information by deleting CQs which are no longer relevant, and adding or
+ * updating any CQs which currently exist in the database.
+ *
+ * @param file
+ * local file to update with current CQ records.
+ * @param base
+ * base https:// URL of the IPzilla server.
+ * @param username
+ * username to login to IPzilla as. Must be a Bugzilla username
+ * of someone authorized to query the project's IPzilla records.
+ * @param password
+ * password for {@code username}.
+ * @throws IOException
+ * IPzilla cannot be queried, or the local file cannot be read
+ * from or written to.
+ * @throws ConfigInvalidException
+ * the local file cannot be read, as it is not a valid
+ * configuration file format.
+ */
+ public void syncCQs(File file, URL base, String username, String password)
+ throws IOException, ConfigInvalidException {
+ if (!file.getParentFile().exists())
+ file.getParentFile().mkdirs();
+
+ LockFile lf = new LockFile(file);
+ if (!lf.lock())
+ throw new IOException("Cannot lock " + file);
+ try {
+ FileBasedConfig cfg = new FileBasedConfig(file);
+ cfg.load();
+ loadFrom(cfg);
+
+ IPZillaQuery ipzilla = new IPZillaQuery(base, username, password);
+ Set<CQ> current = ipzilla.getCQs(projects);
+
+ for (CQ cq : sort(current, CQ.COMPARATOR)) {
+ String id = Long.toString(cq.getID());
+
+ set(cfg, S_CQ, id, K_DESCRIPTION, cq.getDescription());
+ set(cfg, S_CQ, id, K_LICENSE, cq.getLicense());
+ set(cfg, S_CQ, id, K_USE, cq.getUse());
+ set(cfg, S_CQ, id, K_STATE, cq.getState());
+ set(cfg, S_CQ, id, K_COMMENTS, cq.getComments());
+ }
+
+ for (CQ cq : cqs) {
+ if (!current.contains(cq))
+ cfg.unsetSection(S_CQ, Long.toString(cq.getID()));
+ }
+
+ lf.write(Constants.encode(cfg.toText()));
+ if (!lf.commit())
+ throw new IOException("Cannot write " + file);
+ } finally {
+ lf.unlock();
+ }
+ }
+
+ private static void set(Config cfg, String section, String subsection,
+ String key, String value) {
+ if (value == null || "".equals(value))
+ cfg.unset(section, subsection, key);
+ else
+ cfg.setString(section, subsection, key, value);
+ }
+
+ private static <T, Q extends Comparator<T>> Iterable<T> sort(
+ Collection<T> objs, Q cmp) {
+ ArrayList<T> sorted = new ArrayList<T>(objs);
+ Collections.sort(sorted, cmp);
+ return sorted;
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Project.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Project.java
new file mode 100644
index 0000000000..6495d38ff5
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/Project.java
@@ -0,0 +1,119 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.Set;
+import java.util.TreeSet;
+
+/** Description of a project. */
+class Project {
+ /** Sorts projects by unique identities. */
+ static final Comparator<Project> COMPARATOR = new Comparator<Project>() {
+ public int compare(Project a, Project b) {
+ return a.getID().compareTo(b.getID());
+ }
+ };
+
+ private final String id;
+
+ private final String name;
+
+ private String comments;
+
+ private final Set<String> licenses = new TreeSet<String>();
+
+ private String version;
+
+ /**
+ * @param id
+ * @param name
+ */
+ Project(String id, String name) {
+ this.id = id;
+ this.name = name;
+ }
+
+ /** @return unique identity of this project. */
+ String getID() {
+ return id;
+ }
+
+ /** @return name of this project. */
+ String getName() {
+ return name;
+ }
+
+ /** @return any additional comments about this project. */
+ String getComments() {
+ return comments;
+ }
+
+ void setComments(String comments) {
+ this.comments = comments;
+ }
+
+ /** @return the licenses this project is released under. */
+ Set<String> getLicenses() {
+ return Collections.unmodifiableSet(licenses);
+ }
+
+ void addLicense(String licenseName) {
+ licenses.add(licenseName);
+ }
+
+ String getVersion() {
+ return version;
+ }
+
+ void setVersion(String v) {
+ version = v;
+ }
+
+ @Override
+ public String toString() {
+ return "Project " + getID() + " (" + getName() + ")";
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SimpleCookieManager.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SimpleCookieManager.java
new file mode 100644
index 0000000000..dca6f15054
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SimpleCookieManager.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.io.IOException;
+import java.net.CookieHandler;
+import java.net.URI;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+/**
+ * Dumb implementation of a CookieManager for the JRE.
+ * <p>
+ * Cookies are keyed only by the host name in the URI. Cookie attributes like
+ * domain and path are ignored to simplify the implementation.
+ * <p>
+ * If we are running on Java 6 or later we should favor using the standard
+ * {@code java.net.CookieManager} class instead.
+ */
+public class SimpleCookieManager extends CookieHandler {
+ private Map<String, Map<String, String>> byHost = new HashMap<String, Map<String, String>>();
+
+ @Override
+ public Map<String, List<String>> get(URI uri,
+ Map<String, List<String>> requestHeaders) throws IOException {
+ String host = hostOf(uri);
+
+ Map<String, String> map = byHost.get(host);
+ if (map == null || map.isEmpty())
+ return requestHeaders;
+
+ Map<String, List<String>> r = new HashMap<String, List<String>>();
+ r.putAll(requestHeaders);
+ StringBuilder buf = new StringBuilder();
+ for (Map.Entry<String, String> e : map.entrySet()) {
+ if (buf.length() > 0)
+ buf.append("; ");
+ buf.append(e.getKey());
+ buf.append('=');
+ buf.append(e.getValue());
+ }
+ r.put("Cookie", Collections.singletonList(buf.toString()));
+ return Collections.unmodifiableMap(r);
+ }
+
+ @Override
+ public void put(URI uri, Map<String, List<String>> responseHeaders)
+ throws IOException {
+ List<String> list = responseHeaders.get("Set-Cookie");
+ if (list == null || list.isEmpty()) {
+ return;
+ }
+
+ String host = hostOf(uri);
+ Map<String, String> map = byHost.get(host);
+ if (map == null) {
+ map = new HashMap<String, String>();
+ byHost.put(host, map);
+ }
+
+ for (String hdr : list) {
+ String attributes[] = hdr.split(";");
+ String nameValue = attributes[0].trim();
+ int eq = nameValue.indexOf('=');
+ String name = nameValue.substring(0, eq);
+ String value = nameValue.substring(eq + 1);
+
+ map.put(name, value);
+ }
+ }
+
+ private String hostOf(URI uri) {
+ StringBuilder key = new StringBuilder();
+ key.append(uri.getScheme());
+ key.append(':');
+ key.append(uri.getHost());
+ if (0 < uri.getPort())
+ key.append(':' + uri.getPort());
+ return key.toString();
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SingleContribution.java b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SingleContribution.java
new file mode 100644
index 0000000000..2cd5562a10
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/SingleContribution.java
@@ -0,0 +1,112 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * 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.iplog;
+
+import java.util.Comparator;
+import java.util.Date;
+
+/** A single contribution by a {@link Contributor}. */
+class SingleContribution {
+ /** Sorts contributors by their name first name, then last name. */
+ public static final Comparator<SingleContribution> COMPARATOR = new Comparator<SingleContribution>() {
+ public int compare(SingleContribution a, SingleContribution b) {
+ return a.created.compareTo(b.created);
+ }
+ };
+
+ private final String id;
+
+ private String summary;
+
+ private Date created;
+
+ private String bugId;
+
+ private String size;
+
+ /**
+ * @param id
+ * @param created
+ * @param summary
+ */
+ SingleContribution(String id, Date created, String summary) {
+ this.id = id;
+ this.summary = summary;
+ this.created = created;
+ }
+
+ /** @return unique identity of the contribution. */
+ String getID() {
+ return id;
+ }
+
+ /** @return date the contribution was created. */
+ Date getCreated() {
+ return created;
+ }
+
+ /** @return summary of the contribution. */
+ String getSummary() {
+ return summary;
+ }
+
+ /** @return Bugzilla bug id */
+ String getBugID() {
+ return bugId;
+ }
+
+ void setBugID(String id) {
+ if (id.startsWith("https://bugs.eclipse.org/"))
+ id = id.substring("https://bugs.eclipse.org/".length());
+ bugId = id;
+ }
+
+ String getSize() {
+ return size;
+ }
+
+ void setSize(String sz) {
+ size = sz;
+ }
+}
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/gsql_query.txt b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/gsql_query.txt
new file mode 100644
index 0000000000..441cc978d7
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/gsql_query.txt
@@ -0,0 +1,22 @@
+# Query for Gerrit Code Review gsql to produce the .git/gerrit_committers
+# file for a project. Needing to do this manually is a horrible hack.
+
+SELECT a.account_id,
+ u.external_id,
+ a.full_name,
+ b.email_address,
+ r.added_on,
+ r.removed_on
+FROM accounts a,
+ account_external_ids b,
+ account_groups g,
+ account_group_members_audit r,
+ account_external_ids u
+WHERE a.account_id = b.account_id
+ AND b.email_address IS NOT NULL
+ AND r.account_id = a.account_id
+ AND r.group_id = g.group_id
+ AND u.account_id = a.account_id
+ AND u.external_id like 'username:%'
+ AND g.name = 'technology.jgit-committers'
+ORDER BY a.full_name, r.added_on;
diff --git a/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/sample_gerrit_committers b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/sample_gerrit_committers
new file mode 100644
index 0000000000..0e0b47eb04
--- /dev/null
+++ b/org.eclipse.jgit.iplog/src/org/eclipse/jgit/iplog/sample_gerrit_committers
@@ -0,0 +1,2 @@
+ 1 | username:spearce | Shawn Pearce | sop@google.com | 2009-09-29 16:47:03.0 | NULL
+ 1 | username:spearce | Shawn Pearce | spearce@spearce.org | 2009-09-29 16:47:03.0 | NULL