Merge changes Ic9315314,I8c336be2
* changes: Add acceptance test for new plugin and extension API Make acceptance test framework request scope aware
This commit is contained in:
@@ -0,0 +1,186 @@
|
||||
// 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.acceptance;
|
||||
|
||||
import com.google.common.collect.Maps;
|
||||
import com.google.gerrit.reviewdb.server.ReviewDb;
|
||||
import com.google.gerrit.server.CurrentUser;
|
||||
import com.google.gerrit.server.RequestCleanup;
|
||||
import com.google.gerrit.server.config.RequestScopedReviewDbProvider;
|
||||
import com.google.gerrit.server.util.RequestContext;
|
||||
import com.google.gerrit.server.util.ThreadLocalRequestContext;
|
||||
import com.google.gerrit.server.util.ThreadLocalRequestScopePropagator;
|
||||
import com.google.gerrit.server.util.TimeUtil;
|
||||
import com.google.gwtorm.server.SchemaFactory;
|
||||
import com.google.inject.Inject;
|
||||
import com.google.inject.Key;
|
||||
import com.google.inject.OutOfScopeException;
|
||||
import com.google.inject.Provider;
|
||||
import com.google.inject.Scope;
|
||||
import com.google.inject.util.Providers;
|
||||
|
||||
import java.util.Map;
|
||||
|
||||
/** Guice scopes for state during an Acceptance Test connection. */
|
||||
public class AcceptanceTestRequestScope {
|
||||
private static final Key<RequestCleanup> RC_KEY =
|
||||
Key.get(RequestCleanup.class);
|
||||
|
||||
private static final Key<RequestScopedReviewDbProvider> DB_KEY =
|
||||
Key.get(RequestScopedReviewDbProvider.class);
|
||||
|
||||
public class Context implements RequestContext {
|
||||
private final RequestCleanup cleanup = new RequestCleanup();
|
||||
private final Map<Key<?>, Object> map = Maps.newHashMap();
|
||||
private final SchemaFactory<ReviewDb> schemaFactory;
|
||||
private final SshSession session;
|
||||
private final CurrentUser user;
|
||||
|
||||
final long created;
|
||||
volatile long started;
|
||||
volatile long finished;
|
||||
|
||||
private Context(SchemaFactory<ReviewDb> sf, SshSession s,
|
||||
CurrentUser u, long at) {
|
||||
schemaFactory = sf;
|
||||
session = s;
|
||||
user = u;
|
||||
created = started = finished = at;
|
||||
map.put(RC_KEY, cleanup);
|
||||
map.put(DB_KEY, new RequestScopedReviewDbProvider(
|
||||
schemaFactory,
|
||||
Providers.of(cleanup)));
|
||||
}
|
||||
|
||||
private Context(Context p, SshSession s, CurrentUser c) {
|
||||
this(p.schemaFactory, s, c, p.created);
|
||||
started = p.started;
|
||||
finished = p.finished;
|
||||
}
|
||||
|
||||
SshSession getSession() {
|
||||
return session;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CurrentUser getCurrentUser() {
|
||||
if (user == null) {
|
||||
throw new IllegalStateException("user == null, forgot to set it?");
|
||||
}
|
||||
return user;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Provider<ReviewDb> getReviewDbProvider() {
|
||||
return (RequestScopedReviewDbProvider) map.get(DB_KEY);
|
||||
}
|
||||
|
||||
synchronized <T> T get(Key<T> key, Provider<T> creator) {
|
||||
@SuppressWarnings("unchecked")
|
||||
T t = (T) map.get(key);
|
||||
if (t == null) {
|
||||
t = creator.get();
|
||||
map.put(key, t);
|
||||
}
|
||||
return t;
|
||||
}
|
||||
}
|
||||
|
||||
static class ContextProvider implements Provider<Context> {
|
||||
@Override
|
||||
public Context get() {
|
||||
return requireContext();
|
||||
}
|
||||
}
|
||||
|
||||
static class SshSessionProvider implements Provider<SshSession> {
|
||||
@Override
|
||||
public SshSession get() {
|
||||
return requireContext().getSession();
|
||||
}
|
||||
}
|
||||
|
||||
static class Propagator extends ThreadLocalRequestScopePropagator<Context> {
|
||||
private final AcceptanceTestRequestScope atrScope;
|
||||
|
||||
@Inject
|
||||
Propagator(AcceptanceTestRequestScope atrScope, ThreadLocalRequestContext local,
|
||||
Provider<RequestScopedReviewDbProvider> dbProviderProvider) {
|
||||
super(REQUEST, current, local, dbProviderProvider);
|
||||
this.atrScope = atrScope;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Context continuingContext(Context ctx) {
|
||||
// The cleanup is not chained, since the RequestScopePropagator executors
|
||||
// the Context's cleanup when finished executing.
|
||||
return atrScope.newContinuingContext(ctx);
|
||||
}
|
||||
}
|
||||
|
||||
private static final ThreadLocal<Context> current =
|
||||
new ThreadLocal<Context>();
|
||||
|
||||
private static Context requireContext() {
|
||||
final Context ctx = current.get();
|
||||
if (ctx == null) {
|
||||
throw new OutOfScopeException("Not in command/request");
|
||||
}
|
||||
return ctx;
|
||||
}
|
||||
|
||||
private final ThreadLocalRequestContext local;
|
||||
|
||||
@Inject
|
||||
AcceptanceTestRequestScope(ThreadLocalRequestContext local) {
|
||||
this.local = local;
|
||||
}
|
||||
|
||||
public Context newContext(SchemaFactory<ReviewDb> sf, SshSession s, CurrentUser user) {
|
||||
return new Context(sf, s, user, TimeUtil.nowMs());
|
||||
}
|
||||
|
||||
private Context newContinuingContext(Context ctx) {
|
||||
return new Context(ctx, ctx.getSession(), ctx.getCurrentUser());
|
||||
}
|
||||
|
||||
public Context set(Context ctx) {
|
||||
Context old = current.get();
|
||||
current.set(ctx);
|
||||
local.setContext(ctx);
|
||||
return old;
|
||||
}
|
||||
|
||||
/** Returns exactly one instance per command executed. */
|
||||
static final Scope REQUEST = new Scope() {
|
||||
public <T> Provider<T> scope(final Key<T> key, final Provider<T> creator) {
|
||||
return new Provider<T>() {
|
||||
public T get() {
|
||||
return requireContext().get(key, creator);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return String.format("%s[%s]", creator, REQUEST);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "Acceptance Test Scope.REQUEST";
|
||||
}
|
||||
};
|
||||
}
|
||||
@@ -58,4 +58,8 @@ public class TestAccount {
|
||||
server.getHttpAddress().getAddress().getHostAddress(),
|
||||
server.getHttpAddress().getPort());
|
||||
}
|
||||
|
||||
public Account.Id getId() {
|
||||
return id;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,6 @@
|
||||
include_defs('//gerrit-acceptance-tests/tests.defs')
|
||||
|
||||
acceptance_tests(
|
||||
srcs = ['ReviewIT.java'],
|
||||
deps = ['//gerrit-acceptance-tests/src/test/java/com/google/gerrit/acceptance/git:util'],
|
||||
)
|
||||
@@ -0,0 +1,117 @@
|
||||
// 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.acceptance.api.revision;
|
||||
|
||||
import static com.google.gerrit.acceptance.git.GitUtil.cloneProject;
|
||||
import static com.google.gerrit.acceptance.git.GitUtil.createProject;
|
||||
import static com.google.gerrit.acceptance.git.GitUtil.initSsh;
|
||||
|
||||
import com.google.common.collect.Maps;
|
||||
import com.google.gerrit.acceptance.AbstractDaemonTest;
|
||||
import com.google.gerrit.acceptance.AcceptanceTestRequestScope;
|
||||
import com.google.gerrit.acceptance.AccountCreator;
|
||||
import com.google.gerrit.acceptance.SshSession;
|
||||
import com.google.gerrit.acceptance.TestAccount;
|
||||
import com.google.gerrit.acceptance.git.PushOneCommit;
|
||||
import com.google.gerrit.extensions.api.GerritApi;
|
||||
import com.google.gerrit.extensions.api.changes.ReviewInput;
|
||||
import com.google.gerrit.extensions.restapi.RestApiException;
|
||||
import com.google.gerrit.reviewdb.client.Project;
|
||||
import com.google.gerrit.reviewdb.server.ReviewDb;
|
||||
import com.google.gerrit.server.IdentifiedUser;
|
||||
import com.google.gwtorm.server.SchemaFactory;
|
||||
import com.google.inject.Inject;
|
||||
|
||||
import org.eclipse.jgit.api.Git;
|
||||
import org.eclipse.jgit.api.errors.GitAPIException;
|
||||
import org.junit.After;
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
public class ReviewIT extends AbstractDaemonTest {
|
||||
|
||||
@Inject
|
||||
private AccountCreator accounts;
|
||||
|
||||
@Inject
|
||||
private SchemaFactory<ReviewDb> reviewDbProvider;
|
||||
|
||||
@Inject
|
||||
private GerritApi gApi;
|
||||
|
||||
@Inject
|
||||
AcceptanceTestRequestScope atrScope;
|
||||
|
||||
@Inject
|
||||
private IdentifiedUser.GenericFactory identifiedUserFactory;
|
||||
|
||||
private TestAccount admin;
|
||||
private Git git;
|
||||
private ReviewDb db;
|
||||
|
||||
@Before
|
||||
public void setUp() throws Exception {
|
||||
admin = accounts.admin();
|
||||
initSsh(admin);
|
||||
Project.NameKey project = new Project.NameKey("p");
|
||||
SshSession sshSession = new SshSession(server, admin);
|
||||
createProject(sshSession, project.get());
|
||||
git = cloneProject(sshSession.getUrl() + "/" + project.get());
|
||||
atrScope.set(atrScope.newContext(reviewDbProvider, sshSession,
|
||||
identifiedUserFactory.create(admin.getId())));
|
||||
db = reviewDbProvider.open();
|
||||
}
|
||||
|
||||
@After
|
||||
public void cleanup() {
|
||||
db.close();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void reviewTriplet() throws GitAPIException,
|
||||
IOException, RestApiException {
|
||||
PushOneCommit.Result r = createChange();
|
||||
gApi.changes()
|
||||
.id("p~master~" + r.getChangeId())
|
||||
.revision(r.getCommit().name())
|
||||
.review(makeReview());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void reviewId() throws GitAPIException,
|
||||
IOException, RestApiException {
|
||||
PushOneCommit.Result r = createChange();
|
||||
gApi.changes()
|
||||
.id(r.getChangeId())
|
||||
.revision(r.getCommit().name())
|
||||
.review(makeReview());
|
||||
}
|
||||
|
||||
private PushOneCommit.Result createChange() throws GitAPIException,
|
||||
IOException {
|
||||
PushOneCommit push = new PushOneCommit(db, admin.getIdent());
|
||||
return push.to(git, "refs/for/master");
|
||||
}
|
||||
|
||||
private static ReviewInput makeReview() {
|
||||
ReviewInput in = new ReviewInput();
|
||||
in.message = "Looks good!";
|
||||
in.labels = Maps.newHashMap();
|
||||
in.labels.put("Code-Review", (short) 2);
|
||||
return in;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user