notification.rs 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. //! Purpose: Allow any user to ping a pre-selected group of people on GitHub via comments.
  2. //!
  3. //! The set of "teams" which can be pinged is intentionally restricted via configuration.
  4. //!
  5. //! Parsing is done in the `parser::command::ping` module.
  6. use crate::db::notifications;
  7. use crate::{
  8. github::{self, Event},
  9. handlers::Context,
  10. };
  11. use anyhow::Context as _;
  12. use std::collections::HashSet;
  13. use std::convert::{TryFrom, TryInto};
  14. use tracing as log;
  15. pub async fn handle(ctx: &Context, event: &Event) -> anyhow::Result<()> {
  16. let body = match event.comment_body() {
  17. Some(v) => v,
  18. // Skip events that don't have comment bodies associated
  19. None => return Ok(()),
  20. };
  21. if let Event::Issue(e) = event {
  22. if !matches!(
  23. e.action,
  24. github::IssuesAction::Opened | github::IssuesAction::Edited
  25. ) {
  26. // no change in issue's body for these events, so skip
  27. return Ok(());
  28. }
  29. }
  30. let short_description = match event {
  31. Event::Issue(e) => e.issue.title.clone(),
  32. Event::IssueComment(e) => format!("Comment on {}", e.issue.title),
  33. Event::Push(_) | Event::Create(_) => return Ok(()),
  34. };
  35. let mut caps = parser::get_mentions(body)
  36. .into_iter()
  37. .collect::<HashSet<_>>();
  38. // FIXME: Remove this hardcoding. Ideally we need organization-wide
  39. // configuration, but it's unclear where to put it.
  40. if event.issue().unwrap().repository().organization == "serde-rs" {
  41. // Only add dtolnay on new issues/PRs, not on comments to old PRs and
  42. // issues.
  43. if let Event::Issue(e) = event {
  44. if e.action == github::IssuesAction::Opened {
  45. caps.insert("dtolnay");
  46. }
  47. }
  48. }
  49. // Get the list of users already notified by a previous version of this
  50. // comment, so they don't get notified again
  51. let mut users_notified = HashSet::new();
  52. if let Some(from) = event.comment_from() {
  53. for login in parser::get_mentions(from).into_iter() {
  54. if let Some((Ok(users), _)) = id_from_user(ctx, login).await? {
  55. users_notified.extend(users.into_iter().map(|user| user.id.unwrap()));
  56. }
  57. }
  58. };
  59. // We've implicitly notified the user that is submitting the notification:
  60. // they already know that they left this comment.
  61. //
  62. // If the user intended to ping themselves, they can add the GitHub comment
  63. // via the Zulip interface.
  64. match event.user().get_id(&ctx.github).await {
  65. Ok(Some(id)) => {
  66. users_notified.insert(id.try_into().unwrap());
  67. }
  68. Ok(None) => {}
  69. Err(err) => {
  70. log::error!("Failed to query ID for {:?}: {:?}", event.user(), err);
  71. }
  72. }
  73. log::trace!("Captured usernames in comment: {:?}", caps);
  74. for login in caps {
  75. let (users, team_name) = match id_from_user(ctx, login).await? {
  76. Some((users, team_name)) => (users, team_name),
  77. None => continue,
  78. };
  79. let users = match users {
  80. Ok(users) => users,
  81. Err(err) => {
  82. log::error!("getting users failed: {:?}", err);
  83. continue;
  84. }
  85. };
  86. let client = ctx.db.get().await;
  87. for user in users {
  88. if !users_notified.insert(user.id.unwrap()) {
  89. // Skip users already associated with this event.
  90. continue;
  91. }
  92. if let Err(err) = notifications::record_username(&client, user.id.unwrap(), user.login)
  93. .await
  94. .context("failed to record username")
  95. {
  96. log::error!("record username: {:?}", err);
  97. }
  98. if let Err(err) = notifications::record_ping(
  99. &client,
  100. &notifications::Notification {
  101. user_id: user.id.unwrap(),
  102. origin_url: event.html_url().unwrap().to_owned(),
  103. origin_html: body.to_owned(),
  104. time: event.time().unwrap(),
  105. short_description: Some(short_description.clone()),
  106. team_name: team_name.clone(),
  107. },
  108. )
  109. .await
  110. .context("failed to record ping")
  111. {
  112. log::error!("record ping: {:?}", err);
  113. }
  114. }
  115. }
  116. Ok(())
  117. }
  118. async fn id_from_user(
  119. ctx: &Context,
  120. login: &str,
  121. ) -> anyhow::Result<Option<(anyhow::Result<Vec<github::User>>, Option<String>)>> {
  122. if login.contains('/') {
  123. // This is a team ping. For now, just add it to everyone's agenda on
  124. // that team, but also mark it as such (i.e., a team ping) for
  125. // potentially different prioritization and so forth.
  126. //
  127. // In order to properly handle this down the road, we will want to
  128. // distinguish between "everyone must pay attention" and "someone
  129. // needs to take a look."
  130. //
  131. // We may also want to be able to categorize into these buckets
  132. // *after* the ping occurs and is initially processed.
  133. let mut iter = login.split('/');
  134. let _rust_lang = iter.next().unwrap();
  135. let team = iter.next().unwrap();
  136. let team = match github::get_team(&ctx.github, team).await {
  137. Ok(Some(team)) => team,
  138. Ok(None) => {
  139. // If the team is in rust-lang*, then this is probably an error (potentially user
  140. // error, but should be investigated). Otherwise it's probably not going to be in
  141. // the team repository so isn't actually an error.
  142. if login.starts_with("rust") {
  143. log::error!("team ping ({}) failed to resolve to a known team", login);
  144. } else {
  145. log::info!("team ping ({}) failed to resolve to a known team", login);
  146. }
  147. return Ok(None);
  148. }
  149. Err(err) => {
  150. log::error!(
  151. "team ping ({}) failed to resolve to a known team: {:?}",
  152. login,
  153. err
  154. );
  155. return Ok(None);
  156. }
  157. };
  158. Ok(Some((
  159. team.members
  160. .into_iter()
  161. .map(|member| {
  162. let id = i64::try_from(member.github_id)
  163. .with_context(|| format!("user id {} out of bounds", member.github_id))?;
  164. Ok(github::User {
  165. id: Some(id),
  166. login: member.github,
  167. })
  168. })
  169. .collect::<anyhow::Result<Vec<github::User>>>(),
  170. Some(team.name),
  171. )))
  172. } else {
  173. let user = github::User {
  174. login: login.to_owned(),
  175. id: None,
  176. };
  177. let id = user
  178. .get_id(&ctx.github)
  179. .await
  180. .with_context(|| format!("failed to get user {} ID", user.login))?;
  181. let id = match id {
  182. Some(id) => id,
  183. // If the user was not in the team(s) then just don't record it.
  184. None => {
  185. log::trace!("Skipping {} because no id found", user.login);
  186. return Ok(None);
  187. }
  188. };
  189. let id = i64::try_from(id).with_context(|| format!("user id {} out of bounds", id));
  190. Ok(Some((
  191. id.map(|id| {
  192. vec![github::User {
  193. login: user.login.clone(),
  194. id: Some(id),
  195. }]
  196. }),
  197. None,
  198. )))
  199. }
  200. }