All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.account.AccountConfig Maven / Gradle / Ivy

There is a newer version: 3.11.0
Show newest version
// Copyright (C) 2017 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.account;

import static com.google.common.base.Preconditions.checkState;

import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.RefNames;
import com.google.gerrit.server.git.MetaDataUpdate;
import com.google.gerrit.server.git.ValidationError;
import com.google.gerrit.server.git.VersionedMetaData;
import com.google.gerrit.server.mail.send.OutgoingEmailValidator;
import com.google.gwtorm.server.OrmDuplicateKeyException;
import java.io.IOException;
import java.sql.Timestamp;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.jgit.errors.ConfigInvalidException;
import org.eclipse.jgit.lib.CommitBuilder;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.lib.PersonIdent;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevSort;

/**
 * ‘account.config’ file in the user branch in the All-Users repository that contains the properties
 * of the account.
 *
 * 

The 'account.config' file is a git config file that has one 'account' section with the * properties of the account: * *

 *   [account]
 *     active = false
 *     fullName = John Doe
 *     preferredEmail = [email protected]
 *     status = Overloaded with reviews
 * 
* *

All keys are optional. This means 'account.config' may not exist on the user branch if no * properties are set. * *

Not setting a key and setting a key to an empty string are treated the same way and result in * a {@code null} value. * *

If no value for 'active' is specified, by default the account is considered as active. * *

The commit date of the first commit on the user branch is used as registration date of the * account. The first commit may be an empty commit (if no properties were set and 'account.config' * doesn't exist). */ public class AccountConfig extends VersionedMetaData implements ValidationError.Sink { public static final String ACCOUNT_CONFIG = "account.config"; public static final String ACCOUNT = "account"; public static final String KEY_ACTIVE = "active"; public static final String KEY_FULL_NAME = "fullName"; public static final String KEY_PREFERRED_EMAIL = "preferredEmail"; public static final String KEY_STATUS = "status"; @Nullable private final OutgoingEmailValidator emailValidator; private final Account.Id accountId; private final String ref; private boolean isLoaded; private Account account; private Timestamp registeredOn; private List validationErrors; public AccountConfig(@Nullable OutgoingEmailValidator emailValidator, Account.Id accountId) { this.emailValidator = emailValidator; this.accountId = accountId; this.ref = RefNames.refsUsers(accountId); } @Override protected String getRefName() { return ref; } /** * Get the loaded account. * * @return loaded account. * @throws IllegalStateException if the account was not loaded yet */ public Account getAccount() { checkLoaded(); return account; } /** * Sets the account. This means the loaded account will be overwritten with the given account. * *

Changing the registration date of an account is not supported. * * @param account account that should be set * @throws IllegalStateException if the account was not loaded yet */ public void setAccount(Account account) { checkLoaded(); this.account = account; this.registeredOn = account.getRegisteredOn(); } /** * Creates a new account. * * @return the new account * @throws OrmDuplicateKeyException if the user branch already exists */ public Account getNewAccount() throws OrmDuplicateKeyException { checkLoaded(); if (revision != null) { throw new OrmDuplicateKeyException(String.format("account %s already exists", accountId)); } this.registeredOn = TimeUtil.nowTs(); this.account = new Account(accountId, registeredOn); return account; } @Override protected void onLoad() throws IOException, ConfigInvalidException { if (revision != null) { rw.markStart(revision); rw.sort(RevSort.REVERSE); registeredOn = new Timestamp(rw.next().getCommitTime() * 1000L); Config cfg = readConfig(ACCOUNT_CONFIG); account = parse(cfg); account.setMetaId(revision.name()); } isLoaded = true; } private Account parse(Config cfg) { Account account = new Account(accountId, registeredOn); account.setActive(cfg.getBoolean(ACCOUNT, null, KEY_ACTIVE, true)); account.setFullName(get(cfg, KEY_FULL_NAME)); String preferredEmail = get(cfg, KEY_PREFERRED_EMAIL); account.setPreferredEmail(preferredEmail); if (emailValidator != null && !emailValidator.isValid(preferredEmail)) { error( new ValidationError( ACCOUNT_CONFIG, String.format("Invalid preferred email: %s", preferredEmail))); } account.setStatus(get(cfg, KEY_STATUS)); return account; } @Override public RevCommit commit(MetaDataUpdate update) throws IOException { RevCommit c = super.commit(update); account.setMetaId(c.name()); return c; } @Override protected boolean onSave(CommitBuilder commit) throws IOException, ConfigInvalidException { checkLoaded(); if (revision != null) { commit.setMessage("Update account\n"); } else if (account != null) { commit.setMessage("Create account\n"); commit.setAuthor(new PersonIdent(commit.getAuthor(), registeredOn)); commit.setCommitter(new PersonIdent(commit.getCommitter(), registeredOn)); } Config cfg = readConfig(ACCOUNT_CONFIG); writeToConfig(account, cfg); saveConfig(ACCOUNT_CONFIG, cfg); return true; } public static void writeToConfig(Account account, Config cfg) { setActive(cfg, account.isActive()); set(cfg, KEY_FULL_NAME, account.getFullName()); set(cfg, KEY_PREFERRED_EMAIL, account.getPreferredEmail()); set(cfg, KEY_STATUS, account.getStatus()); } /** * Sets/Unsets {@code account.active} in the given config. * *

{@code account.active} is set to {@code false} if the account is inactive. * *

If the account is active {@code account.active} is unset since {@code true} is the default * if this field is missing. * * @param cfg the config * @param value whether the account is active */ private static void setActive(Config cfg, boolean value) { if (!value) { cfg.setBoolean(ACCOUNT, null, KEY_ACTIVE, false); } else { cfg.unset(ACCOUNT, null, KEY_ACTIVE); } } /** * Sets/Unsets the given key in the given config. * *

The key unset if the value is {@code null}. * * @param cfg the config * @param key the key * @param value the value */ private static void set(Config cfg, String key, String value) { if (!Strings.isNullOrEmpty(value)) { cfg.setString(ACCOUNT, null, key, value); } else { cfg.unset(ACCOUNT, null, key); } } /** * Gets the given key from the given config. * *

Empty values are returned as {@code null} * * @param cfg the config * @param key the key * @return the value, {@code null} if key was not set or key was set to empty string */ private static String get(Config cfg, String key) { return Strings.emptyToNull(cfg.getString(ACCOUNT, null, key)); } private void checkLoaded() { checkState(isLoaded, "account not loaded yet"); } /** * Get the validation errors, if any were discovered during load. * * @return list of errors; empty list if there are no errors. */ public List getValidationErrors() { if (validationErrors != null) { return ImmutableList.copyOf(validationErrors); } return ImmutableList.of(); } @Override public void error(ValidationError error) { if (validationErrors == null) { validationErrors = new ArrayList<>(4); } validationErrors.add(error); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy