blob: 85dbdeb2b4989203dc79bc412e1763461ec0550a [file] [log] [blame]
// Copyright (C) 2008 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.entities;
import static com.google.gerrit.entities.RefNames.REFS_DRAFT_COMMENTS;
import static com.google.gerrit.entities.RefNames.REFS_STARRED_CHANGES;
import static com.google.gerrit.entities.RefNames.REFS_USERS;
import com.google.auto.value.AutoValue;
import com.google.common.primitives.Ints;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.extensions.client.DiffPreferencesInfo;
import java.time.Instant;
import java.util.Optional;
/**
* Information about a single user.
*
* <p>A user may have multiple identities they can use to login to Gerrit (see ExternalId), but in
* such cases they always map back to a single Account entity.
*
* <p>Entities "owned" by an Account (that is, their primary key contains the {@link Account.Id} key
* as part of their key structure):
*
* <ul>
* <li>ExternalId: OpenID identities and email addresses known to be registered to this user.
* Multiple records can exist when the user has more than one public identity, such as a work
* and a personal email address.
* <li>AccountSshKey: user's public SSH keys, for authentication through the internal SSH daemon.
* One record per SSH key uploaded by the user, keys are checked in random order until a match
* is found.
* <li>{@link DiffPreferencesInfo}: user's preferences for rendering side-to-side and unified diff
* </ul>
*/
@AutoValue
public abstract class Account {
/** Placeholder for indicating an account-id that does not correspond to any local account */
public static final Id UNKNOWN_ACCOUNT_ID = id(0);
public static Id id(int id) {
return new AutoValue_Account_Id(id);
}
/** Key local to Gerrit to identify a user. */
@AutoValue
public abstract static class Id implements Comparable<Id> {
/** Parse an Account.Id out of a string representation. */
public static Optional<Id> tryParse(String str) {
return Optional.ofNullable(Ints.tryParse(str)).map(Account::id);
}
public static Id fromRef(String name) {
if (name == null) {
return null;
}
if (name.startsWith(REFS_USERS)) {
return fromRefPart(name.substring(REFS_USERS.length()));
} else if (name.startsWith(REFS_DRAFT_COMMENTS)) {
return parseAfterShardedRefPart(name.substring(REFS_DRAFT_COMMENTS.length()));
} else if (name.startsWith(REFS_STARRED_CHANGES)) {
return parseAfterShardedRefPart(name.substring(REFS_STARRED_CHANGES.length()));
}
return null;
}
/**
* Parse an Account.Id out of a part of a ref-name.
*
* @param name a ref name with the following syntax: {@code "34/1234..."}. We assume that the
* caller has trimmed any prefix.
*/
public static Id fromRefPart(String name) {
Integer id = RefNames.parseShardedRefPart(name);
return id != null ? Account.id(id) : null;
}
public static Id parseAfterShardedRefPart(String name) {
Integer id = RefNames.parseAfterShardedRefPart(name);
return id != null ? Account.id(id) : null;
}
/**
* Parse an Account.Id out of the last part of a ref name.
*
* <p>The input is a ref name of the form {@code ".../1234"}, where the suffix is a non-sharded
* account ID. Ref names using a sharded ID should use {@link #fromRefPart(String)} instead for
* greater safety.
*
* @param name ref name
* @return account ID, or null if not numeric.
*/
public static Id fromRefSuffix(String name) {
Integer id = RefNames.parseRefSuffix(name);
return id != null ? Account.id(id) : null;
}
abstract int id();
public int get() {
return id();
}
@Override
public final int compareTo(Id o) {
return Integer.compare(id(), o.id());
}
@Override
public final String toString() {
return Integer.toString(get());
}
}
public abstract Id id();
/** Date and time the user registered with the review server. */
public abstract Instant registeredOn();
/** Full name of the user ("Given-name Surname" style). */
@Nullable
public abstract String fullName();
/** Optional display name of the user to be shown in the UI. */
@Nullable
public abstract String displayName();
/** Email address the user prefers to be contacted through. */
@Nullable
public abstract String preferredEmail();
/**
* Is this user inactive? This is used to avoid showing some users (eg. former employees) in
* auto-suggest.
*/
public abstract boolean inactive();
/** The user-settable status of this account (e.g. busy, OOO, available) */
@Nullable
public abstract String status();
/** ID of the user branch from which the account was read. */
@Nullable
public abstract String metaId();
/**
* Create a new account.
*
* @param newId unique id, see {@link com.google.gerrit.server.notedb.Sequences#nextAccountId()}.
* @param registeredOn when the account was registered.
*/
public static Account.Builder builder(Account.Id newId, Instant registeredOn) {
return new AutoValue_Account.Builder()
.setInactive(false)
.setId(newId)
.setRegisteredOn(registeredOn);
}
/**
* Formats an account name.
*
* <p>The return value goes into NoteDb commits and audit logs, so it should not be changed.
*
* <p>This method deliberately does not use {@code Anonymous Coward} because it can be changed
* using a {@code gerrit.config} option which is a problem for NoteDb commits that still refer to
* a previously defined value.
*
* @return the fullname, if present, otherwise the preferred email, if present, as a last resort a
* generic string containing the accountId.
*/
public String getName() {
if (fullName() != null) {
return fullName();
}
if (preferredEmail() != null) {
return preferredEmail();
}
return getName(id());
}
public static String getName(Account.Id accountId) {
return "GerritAccount #" + accountId.get();
}
/**
* Get the name and email address.
*
* <p>Example output:
*
* <ul>
* <li>{@code A U. Thor <author@example.com>}: full populated
* <li>{@code A U. Thor (12)}: missing email address
* <li>{@code Anonymous Coward <author@example.com>}: missing name
* <li>{@code Anonymous Coward (12)}: missing name and email address
* </ul>
*/
public String getNameEmail(String anonymousCowardName) {
String name = fullName() != null ? fullName() : anonymousCowardName;
StringBuilder b = new StringBuilder();
b.append(name);
if (preferredEmail() != null) {
b.append(" <");
b.append(preferredEmail());
b.append(">");
} else {
b.append(" (");
b.append(id().get());
b.append(")");
}
return b.toString();
}
public boolean isActive() {
return !inactive();
}
public abstract Builder toBuilder();
@AutoValue.Builder
public abstract static class Builder {
public abstract Id id();
abstract Builder setId(Id id);
public abstract Instant registeredOn();
abstract Builder setRegisteredOn(Instant registeredOn);
@Nullable
public abstract String fullName();
public abstract Builder setFullName(String fullName);
@Nullable
public abstract String displayName();
public abstract Builder setDisplayName(String displayName);
@Nullable
public abstract String preferredEmail();
public abstract Builder setPreferredEmail(String preferredEmail);
public abstract boolean inactive();
public abstract Builder setInactive(boolean inactive);
public Builder setActive(boolean active) {
return setInactive(!active);
}
@Nullable
public abstract String status();
public abstract Builder setStatus(String status);
@Nullable
public abstract String metaId();
public abstract Builder setMetaId(@Nullable String metaId);
public abstract Account build();
}
@Override
public final String toString() {
return getName();
}
}