| // 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.server.schema; |
| |
| import com.google.common.collect.ImmutableList; |
| import com.google.gerrit.common.Version; |
| import com.google.gerrit.common.data.AccessSection; |
| import com.google.gerrit.common.data.GlobalCapability; |
| import com.google.gerrit.common.data.LabelType; |
| import com.google.gerrit.common.data.LabelValue; |
| import com.google.gerrit.common.data.Permission; |
| import com.google.gerrit.common.data.PermissionRule; |
| import com.google.gerrit.reviewdb.client.AccountGroup; |
| import com.google.gerrit.reviewdb.client.AccountGroupName; |
| import com.google.gerrit.reviewdb.client.CurrentSchemaVersion; |
| import com.google.gerrit.reviewdb.client.Project; |
| import com.google.gerrit.reviewdb.client.Project.InheritableBoolean; |
| 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.account.GroupUUID; |
| import com.google.gerrit.server.config.AllProjectsName; |
| import com.google.gerrit.server.config.SitePath; |
| import com.google.gerrit.server.config.SitePaths; |
| import com.google.gerrit.server.extensions.events.GitReferenceUpdated; |
| import com.google.gerrit.server.git.GitRepositoryManager; |
| import com.google.gerrit.server.git.MetaDataUpdate; |
| import com.google.gerrit.server.git.ProjectConfig; |
| import com.google.gwtorm.jdbc.JdbcExecutor; |
| import com.google.gwtorm.jdbc.JdbcSchema; |
| import com.google.gwtorm.server.OrmException; |
| import com.google.inject.Inject; |
| |
| import org.eclipse.jgit.errors.ConfigInvalidException; |
| import org.eclipse.jgit.errors.RepositoryNotFoundException; |
| import org.eclipse.jgit.lib.Constants; |
| import org.eclipse.jgit.lib.PersonIdent; |
| import org.eclipse.jgit.lib.RefUpdate; |
| import org.eclipse.jgit.lib.Repository; |
| |
| import java.io.File; |
| import java.io.IOException; |
| import java.util.Collections; |
| |
| /** Creates the current database schema and populates initial code rows. */ |
| public class SchemaCreator { |
| private final @SitePath |
| File site_path; |
| |
| private final GitRepositoryManager mgr; |
| private final AllProjectsName allProjectsName; |
| private final PersonIdent serverUser; |
| private final DataSourceType dataSourceType; |
| |
| private final int versionNbr; |
| |
| private AccountGroup admin; |
| private AccountGroup anonymous; |
| private AccountGroup registered; |
| private AccountGroup owners; |
| |
| @Inject |
| public SchemaCreator(SitePaths site, |
| @Current SchemaVersion version, |
| GitRepositoryManager mgr, |
| AllProjectsName allProjectsName, |
| @GerritPersonIdent PersonIdent au, |
| DataSourceType dst) { |
| this(site.site_path, version, mgr, allProjectsName, au, dst); |
| } |
| |
| public SchemaCreator(@SitePath File site, |
| @Current SchemaVersion version, |
| GitRepositoryManager gitMgr, |
| AllProjectsName ap, |
| @GerritPersonIdent PersonIdent au, |
| DataSourceType dst) { |
| site_path = site; |
| mgr = gitMgr; |
| allProjectsName = ap; |
| serverUser = au; |
| dataSourceType = dst; |
| versionNbr = version.getVersionNbr(); |
| } |
| |
| public void create(final ReviewDb db) throws OrmException, IOException, |
| ConfigInvalidException { |
| final JdbcSchema jdbc = (JdbcSchema) db; |
| final JdbcExecutor e = new JdbcExecutor(jdbc); |
| try { |
| jdbc.updateSchema(e); |
| } finally { |
| e.close(); |
| } |
| |
| final CurrentSchemaVersion sVer = CurrentSchemaVersion.create(); |
| sVer.versionNbr = versionNbr; |
| db.schemaVersion().insert(Collections.singleton(sVer)); |
| |
| initSystemConfig(db); |
| initAllProjects(); |
| dataSourceType.getIndexScript().run(db); |
| } |
| |
| private AccountGroup newGroup(ReviewDb c, String name, AccountGroup.UUID uuid) |
| throws OrmException { |
| if (uuid == null) { |
| uuid = GroupUUID.make(name, serverUser); |
| } |
| return new AccountGroup( // |
| new AccountGroup.NameKey(name), // |
| new AccountGroup.Id(c.nextAccountGroupId()), // |
| uuid); |
| } |
| |
| private SystemConfig initSystemConfig(final ReviewDb c) throws OrmException { |
| admin = newGroup(c, "Administrators", null); |
| admin.setDescription("Gerrit Site Administrators"); |
| admin.setType(AccountGroup.Type.INTERNAL); |
| c.accountGroups().insert(Collections.singleton(admin)); |
| c.accountGroupNames().insert( |
| Collections.singleton(new AccountGroupName(admin))); |
| |
| anonymous = |
| newGroup(c, "Anonymous Users", AccountGroup.ANONYMOUS_USERS); |
| anonymous.setDescription("Any user, signed-in or not"); |
| anonymous.setOwnerGroupUUID(admin.getGroupUUID()); |
| anonymous.setType(AccountGroup.Type.SYSTEM); |
| c.accountGroups().insert(Collections.singleton(anonymous)); |
| c.accountGroupNames().insert( |
| Collections.singleton(new AccountGroupName(anonymous))); |
| |
| registered = |
| newGroup(c, "Registered Users", AccountGroup.REGISTERED_USERS); |
| registered.setDescription("Any signed-in user"); |
| registered.setOwnerGroupUUID(admin.getGroupUUID()); |
| registered.setType(AccountGroup.Type.SYSTEM); |
| c.accountGroups().insert(Collections.singleton(registered)); |
| c.accountGroupNames().insert( |
| Collections.singleton(new AccountGroupName(registered))); |
| |
| final AccountGroup batchUsers = newGroup(c, "Non-Interactive Users", null); |
| batchUsers.setDescription("Users who perform batch actions on Gerrit"); |
| batchUsers.setOwnerGroupUUID(admin.getGroupUUID()); |
| batchUsers.setType(AccountGroup.Type.INTERNAL); |
| c.accountGroups().insert(Collections.singleton(batchUsers)); |
| c.accountGroupNames().insert( |
| Collections.singleton(new AccountGroupName(batchUsers))); |
| |
| owners = newGroup(c, "Project Owners", AccountGroup.PROJECT_OWNERS); |
| owners.setDescription("Any owner of the project"); |
| owners.setOwnerGroupUUID(admin.getGroupUUID()); |
| owners.setType(AccountGroup.Type.SYSTEM); |
| c.accountGroups().insert(Collections.singleton(owners)); |
| c.accountGroupNames().insert( |
| Collections.singleton(new AccountGroupName(owners))); |
| |
| final SystemConfig s = SystemConfig.create(); |
| try { |
| s.sitePath = site_path.getCanonicalPath(); |
| } catch (IOException e) { |
| s.sitePath = site_path.getAbsolutePath(); |
| } |
| c.systemConfig().insert(Collections.singleton(s)); |
| return s; |
| } |
| |
| private void initAllProjects() throws IOException, ConfigInvalidException { |
| Repository git = null; |
| try { |
| git = mgr.openRepository(allProjectsName); |
| initAllProjects(git); |
| } catch (RepositoryNotFoundException notFound) { |
| // A repository may be missing if this project existed only to store |
| // inheritable permissions. For example 'All-Projects'. |
| try { |
| git = mgr.createRepository(allProjectsName); |
| initAllProjects(git); |
| final RefUpdate u = git.updateRef(Constants.HEAD); |
| u.link(GitRepositoryManager.REF_CONFIG); |
| } catch (RepositoryNotFoundException err) { |
| final String name = allProjectsName.get(); |
| throw new IOException("Cannot create repository " + name, err); |
| } |
| } finally { |
| if (git != null) { |
| git.close(); |
| } |
| } |
| } |
| |
| private void initAllProjects(Repository git) throws IOException, |
| ConfigInvalidException { |
| MetaDataUpdate md = |
| new MetaDataUpdate(GitReferenceUpdated.DISABLED, allProjectsName, git); |
| md.getCommitBuilder().setAuthor(serverUser); |
| md.getCommitBuilder().setCommitter(serverUser); |
| |
| ProjectConfig config = ProjectConfig.read(md); |
| Project p = config.getProject(); |
| p.setDescription("Access inherited by all other projects."); |
| p.setRequireChangeID(InheritableBoolean.TRUE); |
| p.setUseContentMerge(InheritableBoolean.TRUE); |
| p.setUseContributorAgreements(InheritableBoolean.FALSE); |
| p.setUseSignedOffBy(InheritableBoolean.FALSE); |
| |
| AccessSection cap = config.getAccessSection(AccessSection.GLOBAL_CAPABILITIES, true); |
| AccessSection all = config.getAccessSection(AccessSection.ALL, true); |
| AccessSection heads = config.getAccessSection(AccessSection.HEADS, true); |
| AccessSection tags = config.getAccessSection("refs/tags/*", true); |
| AccessSection meta = config.getAccessSection(GitRepositoryManager.REF_CONFIG, true); |
| AccessSection magic = config.getAccessSection("refs/for/" + AccessSection.ALL, true); |
| |
| grant(config, cap, GlobalCapability.ADMINISTRATE_SERVER, admin); |
| grant(config, all, Permission.READ, admin, anonymous); |
| |
| LabelType cr = initCodeReviewLabel(config); |
| grant(config, heads, cr, -1, 1, registered); |
| grant(config, heads, cr, -2, 2, admin, owners); |
| grant(config, heads, Permission.CREATE, admin, owners); |
| grant(config, heads, Permission.PUSH, admin, owners); |
| grant(config, heads, Permission.SUBMIT, admin, owners); |
| grant(config, heads, Permission.FORGE_AUTHOR, registered); |
| grant(config, heads, Permission.FORGE_COMMITTER, admin, owners); |
| grant(config, heads, Permission.EDIT_TOPIC_NAME, true, admin, owners); |
| |
| grant(config, tags, Permission.PUSH_TAG, admin, owners); |
| grant(config, tags, Permission.PUSH_SIGNED_TAG, admin, owners); |
| |
| grant(config, magic, Permission.PUSH, registered); |
| grant(config, magic, Permission.PUSH_MERGE, registered); |
| |
| meta.getPermission(Permission.READ, true).setExclusiveGroup(true); |
| grant(config, meta, Permission.READ, admin, owners); |
| grant(config, meta, cr, -2, 2, admin, owners); |
| grant(config, meta, Permission.PUSH, admin, owners); |
| grant(config, meta, Permission.SUBMIT, admin, owners); |
| |
| md.setMessage("Initialized Gerrit Code Review " + Version.getVersion()); |
| config.commit(md); |
| } |
| |
| private PermissionRule grant(ProjectConfig config, AccessSection section, |
| String permission, AccountGroup group1, AccountGroup... groupList) { |
| return grant(config, section, permission, false, group1, groupList); |
| } |
| |
| private PermissionRule grant(ProjectConfig config, AccessSection section, |
| String permission, boolean force, AccountGroup group1, |
| AccountGroup... groupList) { |
| Permission p = section.getPermission(permission, true); |
| PermissionRule rule = rule(config, group1); |
| rule.setForce(force); |
| p.add(rule); |
| for (AccountGroup group : groupList) { |
| rule = rule(config, group); |
| rule.setForce(force); |
| p.add(rule); |
| } |
| return rule; |
| } |
| |
| private void grant(ProjectConfig config, |
| AccessSection section, LabelType type, |
| int min, int max, AccountGroup... groupList) { |
| String name = Permission.LABEL + type.getName(); |
| Permission p = section.getPermission(name, true); |
| for (AccountGroup group : groupList) { |
| PermissionRule r = rule(config, group); |
| r.setRange(min, max); |
| p.add(r); |
| } |
| } |
| |
| private PermissionRule rule(ProjectConfig config, AccountGroup group) { |
| return new PermissionRule(config.resolve(group)); |
| } |
| |
| public static LabelType initCodeReviewLabel(ProjectConfig c) { |
| LabelType type = new LabelType("Code-Review", ImmutableList.of( |
| new LabelValue((short) 2, "Looks good to me, approved"), |
| new LabelValue((short) 1, "Looks good to me, but someone else must approve"), |
| new LabelValue((short) 0, "No score"), |
| new LabelValue((short) -1, "I would prefer that you didn't submit this"), |
| new LabelValue((short) -2, "Do not submit"))); |
| type.setAbbreviatedName("CR"); |
| type.setCopyMinScore(true); |
| c.getLabelSections().put(type.getName(), type); |
| return type; |
| } |
| } |