REST API /projects/$SUGGEST

The /projects/ URL now accepts a prefix string as part of the URL,
making it usable by the suggestion client UI to limit results.
With this change, the project suggestion service can be removed
and replaced by the REST API.

"GET /projects/platform/" will list all projects that start with
the specified platform/ prefix.

Change-Id: I6f8b9302166f59d03d58a85292cbf052e8419a78
This commit is contained in:
Shawn O. Pearce 2012-04-07 13:47:18 -07:00
parent e96071a099
commit 5cd0577828
9 changed files with 78 additions and 68 deletions

View File

@ -81,11 +81,17 @@ Line-feeds are escaped to allow ls-project to keep the
the 'READ' access right is not assigned to the calling user
account).
--limit::
Cap the number of results to the first N matches.
HTTP
----
This command is also available over HTTP, as `/projects/` for
anonymous access and `/a/projects/` for authenticated access.
Named options are available as query parameters.
Named options are available as query parameters. Results can
be limited to projects matching a prefix by supplying the prefix
as part of the URL, for example `/projects/external/` lists only
projects whose name start with the string `external/`.
Over HTTP the `json_compact` output format is assumed if the client
explicitly asks for JSON using HTTP header `Accept: application/json`.
@ -102,10 +108,16 @@ EXAMPLES
List visible projects:
=====
$ ssh -p 29418 review.example.com gerrit ls-projects
platform/manifest
tools/gerrit
tools/gwtorm
$ curl http://review.example.com/projects/
platform/manifest
tools/gerrit
tools/gwtorm
$ curl http://review.example.com/projects/tools/
tools/gerrit
tools/gwtorm
=====

View File

@ -26,9 +26,6 @@ import java.util.List;
@RpcImpl(version = Version.V2_0)
public interface SuggestService extends RemoteJsonService {
void suggestProjectNameKey(String query, int limit,
AsyncCallback<List<Project.NameKey>> callback);
void suggestAccount(String query, Boolean enabled, int limit,
AsyncCallback<List<AccountInfo>> callback);

View File

@ -16,8 +16,11 @@ package com.google.gerrit.client.projects;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gwt.core.client.JavaScriptObject;
import com.google.gwt.user.client.ui.SuggestOracle;
public class ProjectInfo extends JavaScriptObject {
public class ProjectInfo
extends JavaScriptObject
implements SuggestOracle.Suggestion {
public final Project.NameKey name_key() {
return new Project.NameKey(name());
}
@ -25,6 +28,19 @@ public class ProjectInfo extends JavaScriptObject {
public final native String name() /*-{ return this.name; }-*/;
public final native String description() /*-{ return this.description; }-*/;
@Override
public final String getDisplayString() {
if (description() != null) {
return name() + " (" + description() + ")";
}
return name();
}
@Override
public final String getReplacementString() {
return name();
}
protected ProjectInfo() {
}
}

View File

@ -17,6 +17,7 @@ package com.google.gerrit.client.projects;
import com.google.gerrit.client.rpc.NativeMap;
import com.google.gerrit.client.rpc.RestApi;
import com.google.gwtjsonrpc.common.AsyncCallback;
import com.google.gwt.http.client.URL;
/** Projects available from {@code /projects/}. */
public class ProjectMap extends NativeMap<ProjectInfo> {
@ -36,6 +37,14 @@ public class ProjectMap extends NativeMap<ProjectInfo> {
.send(NativeMap.copyKeysIntoChildren(callback));
}
public static void suggest(String prefix, int limit, AsyncCallback<ProjectMap> cb) {
new RestApi("/projects/" + URL.encode(prefix).replaceAll("[?]", "%3F"))
.addParameterRaw("type", "ALL")
.addParameter("n", limit)
.addParameterTrue("d") // description
.send(NativeMap.copyKeysIntoChildren(cb));
}
protected ProjectMap() {
}
}

View File

@ -15,49 +15,25 @@
package com.google.gerrit.client.ui;
import com.google.gerrit.client.RpcStatus;
import com.google.gerrit.client.projects.ProjectMap;
import com.google.gerrit.client.rpc.GerritCallback;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gwt.user.client.ui.SuggestOracle;
import com.google.gwtexpui.safehtml.client.HighlightSuggestOracle;
import java.util.ArrayList;
import java.util.List;
/** Suggestion Oracle for Project.NameKey entities. */
public class ProjectNameSuggestOracle extends HighlightSuggestOracle {
@Override
public void onRequestSuggestions(final Request req, final Callback callback) {
RpcStatus.hide(new Runnable() {
@Override
public void run() {
SuggestUtil.SVC.suggestProjectNameKey(req.getQuery(), req.getLimit(),
new GerritCallback<List<Project.NameKey>>() {
public void onSuccess(final List<Project.NameKey> result) {
final ArrayList<ProjectNameSuggestion> r =
new ArrayList<ProjectNameSuggestion>(result.size());
for (final Project.NameKey p : result) {
r.add(new ProjectNameSuggestion(p));
}
callback.onSuggestionsReady(req, new Response(r));
ProjectMap.suggest(req.getQuery(), req.getLimit(),
new GerritCallback<ProjectMap>() {
@Override
public void onSuccess(ProjectMap map) {
callback.onSuggestionsReady(req, new Response(map.values().asList()));
}
});
}
});
}
private static class ProjectNameSuggestion implements
SuggestOracle.Suggestion {
private final Project.NameKey key;
ProjectNameSuggestion(final Project.NameKey k) {
key = k;
}
public String getDisplayString() {
return key.get();
}
public String getReplacementString() {
return key.get();
}
}
}

View File

@ -74,7 +74,7 @@ class UrlModule extends ServletModule {
filter("/a/*").through(RequireIdentifiedUserFilter.class);
serveRegex("^/(?:a/)?accounts/self/capabilities$").with(AccountCapabilitiesServlet.class);
serveRegex("^/(a/)?projects/$").with(ListProjectsServlet.class);
serveRegex("^/(?:a/)?projects/(.*)?$").with(ListProjectsServlet.class);
}
private Key<HttpServlet> notFound() {

View File

@ -39,8 +39,6 @@ import com.google.gerrit.server.patch.AddReviewer;
import com.google.gerrit.server.project.ChangeControl;
import com.google.gerrit.server.project.NoSuchChangeException;
import com.google.gerrit.server.project.NoSuchProjectException;
import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.project.ProjectControl;
import com.google.gwtjsonrpc.common.AsyncCallback;
import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject;
@ -60,8 +58,6 @@ class SuggestServiceImpl extends BaseServiceImplementation implements
private static final String MAX_SUFFIX = "\u9fa5";
private final Provider<ReviewDb> reviewDbProvider;
private final ProjectControl.Factory projectControlFactory;
private final ProjectCache projectCache;
private final AccountCache accountCache;
private final GroupControl.Factory groupControlFactory;
private final GroupMembers.Factory groupMembersFactory;
@ -74,8 +70,7 @@ class SuggestServiceImpl extends BaseServiceImplementation implements
@Inject
SuggestServiceImpl(final Provider<ReviewDb> schema,
final ProjectControl.Factory projectControlFactory,
final ProjectCache projectCache, final AccountCache accountCache,
final AccountCache accountCache,
final GroupControl.Factory groupControlFactory,
final GroupMembers.Factory groupMembersFactory,
final Provider<CurrentUser> currentUser,
@ -85,8 +80,6 @@ class SuggestServiceImpl extends BaseServiceImplementation implements
@GerritServerConfig final Config cfg, final GroupCache groupCache) {
super(schema, currentUser);
this.reviewDbProvider = schema;
this.projectControlFactory = projectControlFactory;
this.projectCache = projectCache;
this.accountCache = accountCache;
this.groupControlFactory = groupControlFactory;
this.groupMembersFactory = groupMembersFactory;
@ -111,28 +104,6 @@ class SuggestServiceImpl extends BaseServiceImplementation implements
}
}
public void suggestProjectNameKey(final String query, final int limit,
final AsyncCallback<List<Project.NameKey>> callback) {
final int max = 10;
final int n = limit <= 0 ? max : Math.min(limit, max);
final List<Project.NameKey> r = new ArrayList<Project.NameKey>(n);
for (final Project.NameKey nameKey : projectCache.byName(query)) {
final ProjectControl ctl;
try {
ctl = projectControlFactory.validateFor(nameKey);
} catch (NoSuchProjectException e) {
continue;
}
r.add(ctl.getProject().getNameKey());
if (r.size() == n) {
break;
}
}
callback.onSuccess(r);
}
private interface VisibilityControl {
boolean isVisible(Account account) throws OrmException;
}

View File

@ -14,6 +14,7 @@
package com.google.gerrit.httpd.rpc.project;
import com.google.common.base.Strings;
import com.google.gerrit.httpd.RestApiServlet;
import com.google.gerrit.server.OutputFormat;
import com.google.gerrit.server.project.ListProjects;
@ -23,6 +24,7 @@ import com.google.inject.Singleton;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.net.URLDecoder;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@ -43,6 +45,9 @@ public class ListProjectsServlet extends RestApiServlet {
protected void doGet(HttpServletRequest req, HttpServletResponse res)
throws IOException {
ListProjects impl = factory.get();
if (!Strings.isNullOrEmpty(req.getPathInfo())) {
impl.setMatchPrefix(URLDecoder.decode(req.getPathInfo(), "UTF-8"));
}
if (acceptsJson(req)) {
impl.setFormat(OutputFormat.JSON_COMPACT);
}

View File

@ -16,6 +16,7 @@ package com.google.gerrit.server.project;
import com.google.common.collect.Maps;
import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.reviewdb.client.Project.NameKey;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.OutputFormat;
import com.google.gerrit.server.git.GitRepositoryManager;
@ -100,6 +101,11 @@ public class ListProjects {
@Option(name = "--all", usage = "display all projects that are accessible by the calling user")
private boolean all;
@Option(name = "--limit", aliases = {"-n"}, metaVar = "CNT", usage = "maximum number of projects to list")
private int limit;
private String matchPrefix;
@Inject
protected ListProjects(CurrentUser currentUser, ProjectCache projectCache,
GitRepositoryManager repoManager,
@ -131,6 +137,11 @@ public class ListProjects {
return this;
}
public ListProjects setMatchPrefix(String prefix) {
this.matchPrefix = prefix;
return this;
}
public void display(OutputStream out) {
final PrintWriter stdout;
try {
@ -140,13 +151,14 @@ public class ListProjects {
throw new RuntimeException("JVM lacks UTF-8 encoding", e);
}
int found = 0;
Map<String, ProjectInfo> output = Maps.newTreeMap();
Map<String, String> hiddenNames = Maps.newHashMap();
final TreeMap<Project.NameKey, ProjectNode> treeMap =
new TreeMap<Project.NameKey, ProjectNode>();
try {
for (final Project.NameKey projectName : projectCache.all()) {
for (final Project.NameKey projectName : scan()) {
final ProjectState e = projectCache.get(projectName);
if (e == null) {
// If we can't get it from the cache, pretend its not present.
@ -233,6 +245,10 @@ public class ListProjects {
continue;
}
if (limit > 0 && ++found > limit) {
break;
}
if (format.isJson()) {
output.put(info.name, info);
continue;
@ -270,6 +286,14 @@ public class ListProjects {
}
}
private Iterable<NameKey> scan() {
if (matchPrefix != null) {
return projectCache.byName(matchPrefix);
} else {
return projectCache.all();
}
}
private void printProjectTree(final PrintWriter stdout,
final TreeMap<Project.NameKey, ProjectNode> treeMap) {
final SortedSet<ProjectNode> sortedNodes = new TreeSet<ProjectNode>();