blob: 303c49776035ac5baa193487865e9ee728f1cb04 [file] [log] [blame]
// Copyright (C) 2009 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.testutil;
import static com.google.inject.Scopes.SINGLETON;
import com.google.gerrit.reviewdb.client.CurrentSchemaVersion;
import com.google.gerrit.reviewdb.client.SystemConfig;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.GerritPersonIdent;
import com.google.gerrit.server.GerritPersonIdentProvider;
import com.google.gerrit.server.config.AllProjectsName;
import com.google.gerrit.server.config.AllProjectsNameProvider;
import com.google.gerrit.server.config.AnonymousCowardName;
import com.google.gerrit.server.config.AnonymousCowardNameProvider;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.config.SitePath;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.schema.Current;
import com.google.gerrit.server.schema.DataSourceType;
import com.google.gerrit.server.schema.SchemaCreator;
import com.google.gerrit.server.schema.SchemaVersion;
import com.google.gwtorm.jdbc.Database;
import com.google.gwtorm.jdbc.SimpleDataSource;
import com.google.gwtorm.server.OrmException;
import com.google.gwtorm.server.SchemaFactory;
import com.google.inject.AbstractModule;
import com.google.inject.Guice;
import com.google.inject.Injector;
import com.google.inject.Key;
import junit.framework.TestCase;
import org.eclipse.jgit.errors.ConfigInvalidException;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.lib.PersonIdent;
import java.io.File;
import java.io.IOException;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;
import javax.sql.DataSource;
/**
* An in-memory test instance of {@link ReviewDb} database.
* <p>
* Test classes should create one instance of this class for each unique test
* database they want to use. When the tests needing this instance are complete,
* ensure that {@link #drop(InMemoryDatabase)} is called to free the resources so
* the JVM running the unit tests doesn't run out of heap space.
*/
public class InMemoryDatabase implements SchemaFactory<ReviewDb> {
private static int dbCnt;
private static synchronized DataSource newDataSource() throws SQLException {
final Properties p = new Properties();
p.setProperty("driver", org.h2.Driver.class.getName());
p.setProperty("url", "jdbc:h2:mem:" + "Test_" + (++dbCnt));
final DataSource dataSource = new SimpleDataSource(p);
return dataSource;
}
/** Drop the database from memory; does nothing if the instance was null. */
public static void drop(final InMemoryDatabase db) {
if (db != null) {
db.drop();
}
}
private Connection openHandle;
private Database<ReviewDb> database;
private boolean created;
private SchemaVersion schemaVersion;
private Injector injector;
public InMemoryDatabase() throws OrmException {
try {
final DataSource dataSource = newDataSource();
// Open one connection. This will peg the database into memory
// until someone calls drop on us, allowing subsequent connections
// opened against the same URL to go to the same set of tables.
//
openHandle = dataSource.getConnection();
// Build the access layer around the connection factory.
//
database = new Database<ReviewDb>(dataSource, ReviewDb.class);
injector = Guice.createInjector(new AbstractModule() {
@Override
protected void configure() {
install(new SchemaVersion.Module());
bind(File.class) //
.annotatedWith(SitePath.class) //
.toInstance(new File("."));
Config cfg = new Config();
cfg.setString("gerrit", null, "basePath", "git");
cfg.setString("gerrit", null, "allProjects", "Test-Projects");
cfg.setString("user", null, "name", "Gerrit Code Review");
cfg.setString("user", null, "email", "gerrit@localhost");
bind(Config.class) //
.annotatedWith(GerritServerConfig.class) //
.toInstance(cfg);
bind(PersonIdent.class) //
.annotatedWith(GerritPersonIdent.class) //
.toProvider(GerritPersonIdentProvider.class);
bind(AllProjectsName.class) //
.toProvider(AllProjectsNameProvider.class);
bind(GitRepositoryManager.class) //
.to(InMemoryRepositoryManager.class).in(SINGLETON);
bind(String.class) //
.annotatedWith(AnonymousCowardName.class) //
.toProvider(AnonymousCowardNameProvider.class);
bind(DataSourceType.class) //
.to(InMemoryH2Type.class);
}
});
schemaVersion = injector.getInstance(
Key.get(SchemaVersion.class, Current.class));
} catch (SQLException e) {
throw new OrmException(e);
}
}
public <T> T getInstance(Class<T> clazz) {
return injector.getInstance(clazz);
}
public Database<ReviewDb> getDatabase() {
return database;
}
@Override
public ReviewDb open() throws OrmException {
return getDatabase().open();
}
/** Ensure the database schema has been created and initialized. */
public InMemoryDatabase create() throws OrmException {
if (!created) {
created = true;
final ReviewDb c = open();
try {
try {
getInstance(SchemaCreator.class).create(c);
} catch (IOException e) {
throw new OrmException("Cannot create in-memory database", e);
} catch (ConfigInvalidException e) {
throw new OrmException("Cannot create in-memory database", e);
}
} finally {
c.close();
}
}
return this;
}
/** Drop this database from memory so it no longer exists. */
public void drop() {
if (openHandle != null) {
try {
openHandle.close();
} catch (SQLException e) {
System.err.println("WARNING: Cannot close database connection");
e.printStackTrace(System.err);
}
openHandle = null;
database = null;
}
}
public SystemConfig getSystemConfig() throws OrmException {
final ReviewDb c = open();
try {
return c.systemConfig().get(new SystemConfig.Key());
} finally {
c.close();
}
}
public CurrentSchemaVersion getSchemaVersion() throws OrmException {
final ReviewDb c = open();
try {
return c.schemaVersion().get(new CurrentSchemaVersion.Key());
} finally {
c.close();
}
}
public void assertSchemaVersion() throws OrmException {
final CurrentSchemaVersion act = getSchemaVersion();
TestCase.assertEquals(schemaVersion.getVersionNbr(), act.versionNbr);
}
}