123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800 |
- //! Handles PR and issue assignment.
- //!
- //! This supports several ways for setting issue/PR assignment:
- //!
- //! * `@dragonosbot assign @gh-user`: Assigns to the given user.
- //! * `@dragonosbot claim`: Assigns to the comment author.
- //! * `@dragonosbot release-assignment`: Removes the commenter's assignment.
- //! * `r? @user`: Assigns to the given user (PRs only).
- //!
- //! This is capable of assigning to any user, even if they do not have write
- //! access to the repo. It does this by fake-assigning the bot and adding a
- //! "claimed by" section to the top-level comment.
- //!
- //! Configuration is done with the `[assign]` table.
- //!
- //! This also supports auto-assignment of new PRs. Based on rules in the
- //! `assign.owners` config, it will auto-select an assignee based on the files
- //! the PR modifies.
- use crate::{
- config::AssignConfig,
- github::{self, Event, FileDiff, Issue, IssuesAction, Selection},
- handlers::{Context, GithubClient, IssuesEvent},
- interactions::EditIssueBody,
- };
- use anyhow::{bail, Context as _};
- use parser::command::assign::AssignCommand;
- use parser::command::{Command, Input};
- use rand::seq::IteratorRandom;
- use rust_team_data::v1::Teams;
- use std::collections::{HashMap, HashSet};
- use std::fmt;
- use tracing as log;
- #[cfg(test)]
- mod tests {
- mod tests_candidates;
- mod tests_from_diff;
- }
- const NEW_USER_WELCOME_MESSAGE: &str = "Thanks for the pull request, and welcome! \
- The Rust team is excited to review your changes, and you should hear from {who} \
- some time within the next two weeks.";
- const CONTRIBUTION_MESSAGE: &str = "Please see [the contribution \
- instructions]({contributing_url}) for more information. Namely, in order to ensure the \
- minimum review times lag, PR authors and assigned reviewers should ensure that the review \
- label (`S-waiting-on-review` and `S-waiting-on-author`) stays updated, invoking these commands \
- when appropriate:
- - `@{bot} author`: the review is finished, PR author should check the comments and take action accordingly
- - `@{bot} review`: the author is ready for a review, this PR will be queued again in the reviewer's queue";
- const WELCOME_WITH_REVIEWER: &str = "@{assignee} (or someone else)";
- const WELCOME_WITHOUT_REVIEWER: &str = "@Mark-Simulacrum (NB. this repo may be misconfigured)";
- const RETURNING_USER_WELCOME_MESSAGE: &str = "r? @{assignee}
- {bot} has assigned @{assignee}.
- They will have a look at your PR within the next two weeks and either review your PR or \
- reassign to another reviewer.
- Use r? to explicitly pick a reviewer";
- const RETURNING_USER_WELCOME_MESSAGE_NO_REVIEWER: &str =
- "@{author}: no appropriate reviewer found, use r? to override";
- const ON_VACATION_WARNING: &str = "{username} is on vacation. Please do not assign them to PRs.";
- const NON_DEFAULT_BRANCH: &str =
- "Pull requests are usually filed against the {default} branch for this repo, \
- but this one is against {target}. \
- Please double check that you specified the right target!";
- const SUBMODULE_WARNING_MSG: &str = "These commits modify **submodules**.";
- fn on_vacation_msg(user: &str) -> String {
- ON_VACATION_WARNING.replace("{username}", user)
- }
- #[derive(Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
- struct AssignData {
- user: Option<String>,
- }
- /// Input for auto-assignment when a PR is created.
- pub(super) struct AssignInput {}
- /// Prepares the input when a new PR is opened.
- pub(super) async fn parse_input(
- _ctx: &Context,
- event: &IssuesEvent,
- config: Option<&AssignConfig>,
- ) -> Result<Option<AssignInput>, String> {
- let config = match config {
- Some(config) => config,
- None => return Ok(None),
- };
- if config.owners.is_empty()
- || !matches!(event.action, IssuesAction::Opened)
- || !event.issue.is_pr()
- {
- return Ok(None);
- }
- Ok(Some(AssignInput {}))
- }
- /// Handles the work of setting an assignment for a new PR and posting a
- /// welcome message.
- pub(super) async fn handle_input(
- ctx: &Context,
- config: &AssignConfig,
- event: &IssuesEvent,
- _input: AssignInput,
- ) -> anyhow::Result<()> {
- let Some(diff) = event.issue.diff(&ctx.github).await? else {
- bail!(
- "expected issue {} to be a PR, but the diff could not be determined",
- event.issue.number
- )
- };
- // Don't auto-assign or welcome if the user manually set the assignee when opening.
- if event.issue.assignees.is_empty() {
- let (assignee, from_comment) = determine_assignee(ctx, event, config, &diff).await?;
- if assignee.as_deref() == Some("ghost") {
- // "ghost" is GitHub's placeholder account for deleted accounts.
- // It is used here as a convenient way to prevent assignment. This
- // is typically used for rollups or experiments where you don't
- // want any assignments or noise.
- return Ok(());
- }
- let welcome = if ctx
- .github
- .is_new_contributor(&event.repository, &event.issue.user.login)
- .await
- {
- let who_text = match &assignee {
- Some(assignee) => WELCOME_WITH_REVIEWER.replace("{assignee}", assignee),
- None => WELCOME_WITHOUT_REVIEWER.to_string(),
- };
- let mut welcome = NEW_USER_WELCOME_MESSAGE.replace("{who}", &who_text);
- if let Some(contrib) = &config.contributing_url {
- welcome.push_str("\n\n");
- welcome.push_str(
- &CONTRIBUTION_MESSAGE
- .replace("{contributing_url}", contrib)
- .replace("{bot}", &ctx.username),
- );
- }
- Some(welcome)
- } else if !from_comment {
- let welcome = match &assignee {
- Some(assignee) => RETURNING_USER_WELCOME_MESSAGE
- .replace("{assignee}", assignee)
- .replace("{bot}", &ctx.username),
- None => RETURNING_USER_WELCOME_MESSAGE_NO_REVIEWER
- .replace("{author}", &event.issue.user.login),
- };
- Some(welcome)
- } else {
- // No welcome is posted if they are not new and they used `r?` in the opening body.
- None
- };
- if let Some(assignee) = assignee {
- set_assignee(&event.issue, &ctx.github, &assignee).await;
- }
- if let Some(welcome) = welcome {
- if let Err(e) = event.issue.post_comment(&ctx.github, &welcome).await {
- log::warn!(
- "failed to post welcome comment to {}: {e}",
- event.issue.global_id()
- );
- }
- }
- }
- // Compute some warning messages to post to new PRs.
- let mut warnings = Vec::new();
- if config.warn_non_default_branch {
- warnings.extend(non_default_branch(event));
- }
- warnings.extend(modifies_submodule(diff));
- if !warnings.is_empty() {
- let warnings: Vec<_> = warnings
- .iter()
- .map(|warning| format!("* {warning}"))
- .collect();
- let warning = format!(":warning: **Warning** :warning:\n\n{}", warnings.join("\n"));
- event.issue.post_comment(&ctx.github, &warning).await?;
- };
- Ok(())
- }
- /// Finds the `r?` command in the PR body.
- ///
- /// Returns the name after the `r?` command, or None if not found.
- fn find_assign_command(ctx: &Context, event: &IssuesEvent) -> Option<String> {
- let mut input = Input::new(&event.issue.body, vec![&ctx.username]);
- input.find_map(|command| match command {
- Command::Assign(Ok(AssignCommand::ReviewName { name })) => Some(name),
- _ => None,
- })
- }
- fn is_self_assign(assignee: &str, pr_author: &str) -> bool {
- assignee.to_lowercase() == pr_author.to_lowercase()
- }
- /// Returns a message if the PR is opened against the non-default branch.
- fn non_default_branch(event: &IssuesEvent) -> Option<String> {
- let target_branch = &event.issue.base.as_ref().unwrap().git_ref;
- let default_branch = &event.repository.default_branch;
- if target_branch == default_branch {
- return None;
- }
- Some(
- NON_DEFAULT_BRANCH
- .replace("{default}", default_branch)
- .replace("{target}", target_branch),
- )
- }
- /// Returns a message if the PR modifies a git submodule.
- fn modifies_submodule(diff: &[FileDiff]) -> Option<String> {
- let re = regex::Regex::new(r"\+Subproject\scommit\s").unwrap();
- if diff.iter().any(|fd| re.is_match(&fd.diff)) {
- Some(SUBMODULE_WARNING_MSG.to_string())
- } else {
- None
- }
- }
- /// Sets the assignee of a PR, alerting any errors.
- async fn set_assignee(issue: &Issue, github: &GithubClient, username: &str) {
- // Don't re-assign if already assigned, e.g. on comment edit
- if issue.contain_assignee(&username) {
- log::trace!(
- "ignoring assign PR {} to {}, already assigned",
- issue.global_id(),
- username,
- );
- return;
- }
- if let Err(err) = issue.set_assignee(github, &username).await {
- log::warn!(
- "failed to set assignee of PR {} to {}: {:?}",
- issue.global_id(),
- username,
- err
- );
- if let Err(e) = issue
- .post_comment(
- github,
- &format!(
- "Failed to set assignee to `{username}`: {err}\n\
- \n\
- > **Note**: Only org members with at least the repository \"read\" role, \
- users with write permissions, or people who have commented on the PR may \
- be assigned."
- ),
- )
- .await
- {
- log::warn!("failed to post error comment: {e}");
- }
- }
- }
- /// Determines who to assign the PR to based on either an `r?` command, or
- /// based on which files were modified.
- ///
- /// Returns `(assignee, from_comment)` where `assignee` is who to assign to
- /// (or None if no assignee could be found). `from_comment` is a boolean
- /// indicating if the assignee came from an `r?` command (it is false if
- /// determined from the diff).
- async fn determine_assignee(
- ctx: &Context,
- event: &IssuesEvent,
- config: &AssignConfig,
- diff: &[FileDiff],
- ) -> anyhow::Result<(Option<String>, bool)> {
- let teams = crate::team_data::teams(&ctx.github).await?;
- if let Some(name) = find_assign_command(ctx, event) {
- if is_self_assign(&name, &event.issue.user.login) {
- return Ok((Some(name.to_string()), true));
- }
- // User included `r?` in the opening PR body.
- match find_reviewer_from_names(&teams, config, &event.issue, &[name]) {
- Ok(assignee) => return Ok((Some(assignee), true)),
- Err(e) => {
- event
- .issue
- .post_comment(&ctx.github, &e.to_string())
- .await?;
- // Fall through below for normal diff detection.
- }
- }
- }
- // Errors fall-through to try fallback group.
- match find_reviewers_from_diff(config, diff) {
- Ok(candidates) if !candidates.is_empty() => {
- match find_reviewer_from_names(&teams, config, &event.issue, &candidates) {
- Ok(assignee) => return Ok((Some(assignee), false)),
- Err(FindReviewerError::TeamNotFound(team)) => log::warn!(
- "team {team} not found via diff from PR {}, \
- is there maybe a misconfigured group?",
- event.issue.global_id()
- ),
- // TODO: post a comment on the PR if the reviewers were filtered due to being on vacation
- Err(
- e @ FindReviewerError::NoReviewer { .. }
- | e @ FindReviewerError::AllReviewersFiltered { .. },
- ) => log::trace!(
- "no reviewer could be determined for PR {}: {e}",
- event.issue.global_id()
- ),
- }
- }
- // If no owners matched the diff, fall-through.
- Ok(_) => {}
- Err(e) => {
- log::warn!(
- "failed to find candidate reviewer from diff due to error: {e}\n\
- Is the triagebot.toml misconfigured?"
- );
- }
- }
- if let Some(fallback) = config.adhoc_groups.get("fallback") {
- match find_reviewer_from_names(&teams, config, &event.issue, fallback) {
- Ok(assignee) => return Ok((Some(assignee), false)),
- Err(e) => {
- log::trace!(
- "failed to select from fallback group for PR {}: {e}",
- event.issue.global_id()
- );
- }
- }
- }
- Ok((None, false))
- }
- /// Returns a list of candidate reviewers to use based on which files were changed.
- ///
- /// May return an error if the owners map is misconfigured.
- ///
- /// Beware this may return an empty list if nothing matches.
- fn find_reviewers_from_diff(
- config: &AssignConfig,
- diff: &[FileDiff],
- ) -> anyhow::Result<Vec<String>> {
- // Map of `owners` path to the number of changes found in that path.
- // This weights the reviewer choice towards places where the most edits are done.
- let mut counts: HashMap<&str, u32> = HashMap::new();
- // Iterate over the diff, counting the number of modified lines in each
- // file, and tracks those in the `counts` map.
- for file_diff in diff {
- // List of the longest `owners` patterns that match the current path. This
- // prefers choosing reviewers from deeply nested paths over those defined
- // for top-level paths, under the assumption that they are more
- // specialized.
- //
- // This is a list to handle the situation if multiple paths of the same
- // length match.
- let mut longest_owner_patterns = Vec::new();
- // Find the longest `owners` entries that match this path.
- let mut longest = HashMap::new();
- for owner_pattern in config.owners.keys() {
- let ignore = ignore::gitignore::GitignoreBuilder::new("/")
- .add_line(None, owner_pattern)
- .with_context(|| format!("owner file pattern `{owner_pattern}` is not valid"))?
- .build()?;
- if ignore
- .matched_path_or_any_parents(&file_diff.path, false)
- .is_ignore()
- {
- let owner_len = owner_pattern.split('/').count();
- longest.insert(owner_pattern, owner_len);
- }
- }
- let max_count = longest.values().copied().max().unwrap_or(0);
- longest_owner_patterns.extend(
- longest
- .iter()
- .filter(|(_, count)| **count == max_count)
- .map(|x| *x.0),
- );
- // Give some weight to these patterns to start. This helps with
- // files modified without any lines changed.
- for owner_pattern in &longest_owner_patterns {
- *counts.entry(owner_pattern).or_default() += 1;
- }
- // Count the modified lines.
- for line in file_diff.diff.lines() {
- if (!line.starts_with("+++") && line.starts_with('+'))
- || (!line.starts_with("---") && line.starts_with('-'))
- {
- for owner_path in &longest_owner_patterns {
- *counts.entry(owner_path).or_default() += 1;
- }
- }
- }
- }
- // Use the `owners` entry with the most number of modifications.
- let max_count = counts.values().copied().max().unwrap_or(0);
- let max_paths = counts
- .iter()
- .filter(|(_, count)| **count == max_count)
- .map(|(path, _)| path);
- let mut potential: Vec<_> = max_paths
- .flat_map(|owner_path| &config.owners[*owner_path])
- .map(|owner| owner.to_string())
- .collect();
- // Dedupe. This isn't strictly necessary, as `find_reviewer_from_names` will deduplicate.
- // However, this helps with testing.
- potential.sort();
- potential.dedup();
- Ok(potential)
- }
- /// Handles a command posted in a comment.
- pub(super) async fn handle_command(
- ctx: &Context,
- config: &AssignConfig,
- event: &Event,
- cmd: AssignCommand,
- ) -> anyhow::Result<()> {
- let is_team_member = if let Err(_) | Ok(false) = event.user().is_team_member(&ctx.github).await
- {
- false
- } else {
- true
- };
- // Don't handle commands in comments from the bot. Some of the comments it
- // posts contain commands to instruct the user, not things that the bot
- // should respond to.
- if event.user().login == ctx.username.as_str() {
- return Ok(());
- }
- let issue = event.issue().unwrap();
- if issue.is_pr() {
- if !issue.is_open() {
- issue
- .post_comment(&ctx.github, "Assignment is not allowed on a closed PR.")
- .await?;
- return Ok(());
- }
- let username = match cmd {
- AssignCommand::Own => event.user().login.clone(),
- AssignCommand::User { username } => {
- // Allow users on vacation to assign themselves to a PR, but not anyone else.
- if config.is_on_vacation(&username)
- && event.user().login.to_lowercase() != username.to_lowercase()
- {
- // This is a comment, so there must already be a reviewer assigned. No need to assign anyone else.
- issue
- .post_comment(&ctx.github, &on_vacation_msg(&username))
- .await?;
- return Ok(());
- }
- username
- }
- AssignCommand::Release => {
- log::trace!(
- "ignoring release on PR {:?}, must always have assignee",
- issue.global_id()
- );
- return Ok(());
- }
- AssignCommand::ReviewName { name } => {
- if config.owners.is_empty() {
- // To avoid conflicts with the highfive bot while transitioning,
- // r? is ignored if `owners` is not configured in triagebot.toml.
- return Ok(());
- }
- if matches!(
- event,
- Event::Issue(IssuesEvent {
- action: IssuesAction::Opened,
- ..
- })
- ) {
- // Don't handle r? comments on new PRs. Those will be
- // handled by the new PR trigger (which also handles the
- // welcome message).
- return Ok(());
- }
- if is_self_assign(&name, &event.user().login) {
- name.to_string()
- } else {
- let teams = crate::team_data::teams(&ctx.github).await?;
- // remove "t-" or "T-" prefixes before checking if it's a team name
- let team_name = name.trim_start_matches("t-").trim_start_matches("T-");
- // Determine if assignee is a team. If yes, add the corresponding GH label.
- if teams.teams.get(team_name).is_some() {
- let t_label = format!("T-{}", &team_name);
- if let Err(err) = issue
- .add_labels(&ctx.github, vec![github::Label { name: t_label }])
- .await
- {
- if let Some(github::UnknownLabels { .. }) = err.downcast_ref() {
- log::warn!("Error assigning label: {}", err);
- } else {
- return Err(err);
- }
- }
- }
- match find_reviewer_from_names(&teams, config, issue, &[team_name.to_string()])
- {
- Ok(assignee) => assignee,
- Err(e) => {
- issue.post_comment(&ctx.github, &e.to_string()).await?;
- return Ok(());
- }
- }
- }
- }
- };
- set_assignee(issue, &ctx.github, &username).await;
- return Ok(());
- }
- let e = EditIssueBody::new(&issue, "ASSIGN");
- let to_assign = match cmd {
- AssignCommand::Own => event.user().login.clone(),
- AssignCommand::User { username } => {
- if !is_team_member && username != event.user().login {
- bail!("Only Rust team members can assign other users");
- }
- username.clone()
- }
- AssignCommand::Release => {
- if let Some(AssignData {
- user: Some(current),
- }) = e.current_data()
- {
- if current == event.user().login || is_team_member {
- issue.remove_assignees(&ctx.github, Selection::All).await?;
- e.apply(&ctx.github, String::new(), AssignData { user: None })
- .await?;
- return Ok(());
- } else {
- bail!("Cannot release another user's assignment");
- }
- } else {
- let current = &event.user().login;
- if issue.contain_assignee(current) {
- issue
- .remove_assignees(&ctx.github, Selection::One(¤t))
- .await?;
- e.apply(&ctx.github, String::new(), AssignData { user: None })
- .await?;
- return Ok(());
- } else {
- bail!("Cannot release unassigned issue");
- }
- };
- }
- AssignCommand::ReviewName { .. } => bail!("r? is only allowed on PRs."),
- };
- // Don't re-assign if aleady assigned, e.g. on comment edit
- if issue.contain_assignee(&to_assign) {
- log::trace!(
- "ignoring assign issue {} to {}, already assigned",
- issue.global_id(),
- to_assign,
- );
- return Ok(());
- }
- let data = AssignData {
- user: Some(to_assign.clone()),
- };
- e.apply(&ctx.github, String::new(), &data).await?;
- match issue.set_assignee(&ctx.github, &to_assign).await {
- Ok(()) => return Ok(()), // we are done
- Err(github::AssignmentError::InvalidAssignee) => {
- issue
- .set_assignee(&ctx.github, &ctx.username)
- .await
- .context("self-assignment failed")?;
- let cmt_body = format!(
- "This issue has been assigned to @{} via [this comment]({}).",
- to_assign,
- event.html_url().unwrap()
- );
- e.apply(&ctx.github, cmt_body, &data).await?;
- }
- Err(e) => return Err(e.into()),
- }
- Ok(())
- }
- #[derive(PartialEq, Debug)]
- enum FindReviewerError {
- /// User specified something like `r? foo/bar` where that team name could
- /// not be found.
- TeamNotFound(String),
- /// No reviewer could be found.
- ///
- /// This could happen if there is a cyclical group or other misconfiguration.
- /// `initial` is the initial list of candidate names.
- NoReviewer { initial: Vec<String> },
- /// All potential candidates were excluded. `initial` is the list of
- /// candidate names that were used to seed the selection. `filtered` is
- /// the users who were prevented from being assigned. One example where
- /// this happens is if the given name was for a team where the PR author
- /// is the only member.
- AllReviewersFiltered {
- initial: Vec<String>,
- filtered: Vec<String>,
- },
- }
- impl std::error::Error for FindReviewerError {}
- impl fmt::Display for FindReviewerError {
- fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
- match self {
- FindReviewerError::TeamNotFound(team) => {
- write!(
- f,
- "Team or group `{team}` not found.\n\
- \n\
- rust-lang team names can be found at https://github.com/rust-lang/team/tree/master/teams.\n\
- Reviewer group names can be found in `triagebot.toml` in this repo."
- )
- }
- FindReviewerError::NoReviewer { initial } => {
- write!(
- f,
- "No reviewers could be found from initial request `{}`\n\
- This repo may be misconfigured.\n\
- Use r? to specify someone else to assign.",
- initial.join(",")
- )
- }
- FindReviewerError::AllReviewersFiltered { initial, filtered } => {
- write!(
- f,
- "Could not assign reviewer from: `{}`.\n\
- User(s) `{}` are either the PR author, already assigned, or on vacation, \
- and there are no other candidates.\n\
- Use r? to specify someone else to assign.",
- initial.join(","),
- filtered.join(","),
- )
- }
- }
- }
- }
- /// Finds a reviewer to assign to a PR.
- ///
- /// The `names` is a list of candidate reviewers `r?`, such as `compiler` or
- /// `@octocat`, or names from the owners map. It can contain GitHub usernames,
- /// auto-assign groups, or rust-lang team names. It must have at least one
- /// entry.
- fn find_reviewer_from_names(
- teams: &Teams,
- config: &AssignConfig,
- issue: &Issue,
- names: &[String],
- ) -> Result<String, FindReviewerError> {
- let candidates = candidate_reviewers_from_names(teams, config, issue, names)?;
- // This uses a relatively primitive random choice algorithm.
- // GitHub's CODEOWNERS supports much more sophisticated options, such as:
- //
- // - Round robin: Chooses reviewers based on who's received the least
- // recent review request, focusing on alternating between all members of
- // the team regardless of the number of outstanding reviews they
- // currently have.
- // - Load balance: Chooses reviewers based on each member's total number
- // of recent review requests and considers the number of outstanding
- // reviews for each member. The load balance algorithm tries to ensure
- // that each team member reviews an equal number of pull requests in any
- // 30 day period.
- //
- // Additionally, with CODEOWNERS, users marked as "Busy" in the GitHub UI
- // will not be selected for reviewer. There are several other options for
- // configuring CODEOWNERS as well.
- //
- // These are all ideas for improving the selection here. However, I'm not
- // sure they are really worth the effort.
- Ok(candidates
- .into_iter()
- .choose(&mut rand::thread_rng())
- .expect("candidate_reviewers_from_names always returns at least one entry")
- .to_string())
- }
- /// Returns a list of candidate usernames to choose as a reviewer.
- fn candidate_reviewers_from_names<'a>(
- teams: &'a Teams,
- config: &'a AssignConfig,
- issue: &Issue,
- names: &'a [String],
- ) -> Result<HashSet<&'a str>, FindReviewerError> {
- // Set of candidate usernames to choose from. This uses a set to
- // deduplicate entries so that someone in multiple teams isn't
- // over-weighted.
- let mut candidates: HashSet<&str> = HashSet::new();
- // Keep track of groups seen to avoid cycles and avoid expanding the same
- // team multiple times.
- let mut seen = HashSet::new();
- // This is a queue of potential groups or usernames to expand. The loop
- // below will pop from this and then append the expanded results of teams.
- // Usernames will be added to `candidates`.
- let mut group_expansion: Vec<&str> = names.iter().map(|n| n.as_str()).collect();
- // Keep track of which users get filtered out for a better error message.
- let mut filtered = Vec::new();
- let repo = issue.repository();
- let org_prefix = format!("{}/", repo.organization);
- // Don't allow groups or teams to include the current author or assignee.
- let mut filter = |name: &&str| -> bool {
- let name_lower = name.to_lowercase();
- let ok = name_lower != issue.user.login.to_lowercase()
- && !config.is_on_vacation(name)
- && !issue
- .assignees
- .iter()
- .any(|assignee| name_lower == assignee.login.to_lowercase());
- if !ok {
- filtered.push(name.to_string());
- }
- ok
- };
- // Loop over groups to recursively expand them.
- while let Some(group_or_user) = group_expansion.pop() {
- let group_or_user = group_or_user.strip_prefix('@').unwrap_or(group_or_user);
- // Try ad-hoc groups first.
- // Allow `rust-lang/compiler` to match `compiler`.
- let maybe_group = group_or_user
- .strip_prefix(&org_prefix)
- .unwrap_or(group_or_user);
- if let Some(group_members) = config.adhoc_groups.get(maybe_group) {
- // If a group has already been expanded, don't expand it again.
- if seen.insert(maybe_group) {
- group_expansion.extend(
- group_members
- .iter()
- .map(|member| member.as_str())
- .filter(&mut filter),
- );
- }
- continue;
- }
- // Check for a team name.
- // Allow either a direct team name like `rustdoc` or a GitHub-style
- // team name of `rust-lang/rustdoc` (though this does not check if
- // that is a real GitHub team name).
- //
- // This ignores subteam relationships (it only uses direct members).
- let maybe_team = group_or_user
- .strip_prefix("rust-lang/")
- .unwrap_or(group_or_user);
- if let Some(team) = teams.teams.get(maybe_team) {
- candidates.extend(
- team.members
- .iter()
- .map(|member| member.github.as_str())
- .filter(&mut filter),
- );
- continue;
- }
- if group_or_user.contains('/') {
- return Err(FindReviewerError::TeamNotFound(group_or_user.to_string()));
- }
- // Assume it is a user.
- if filter(&group_or_user) {
- candidates.insert(group_or_user);
- }
- }
- if candidates.is_empty() {
- let initial = names.iter().cloned().collect();
- if filtered.is_empty() {
- Err(FindReviewerError::NoReviewer { initial })
- } else {
- Err(FindReviewerError::AllReviewersFiltered { initial, filtered })
- }
- } else {
- Ok(candidates)
- }
- }
|