major_change.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. use crate::{
  2. config::MajorChangeConfig,
  3. github::{Event, Issue, IssuesAction, IssuesEvent, Label, ZulipGitHubReference},
  4. handlers::Context,
  5. interactions::ErrorComment,
  6. };
  7. use anyhow::Context as _;
  8. use parser::command::second::SecondCommand;
  9. use tracing as log;
  10. #[derive(Clone, PartialEq, Eq, Debug)]
  11. pub enum Invocation {
  12. NewProposal,
  13. AcceptedProposal,
  14. Rename { prev_issue: ZulipGitHubReference },
  15. }
  16. pub(super) async fn parse_input(
  17. _ctx: &Context,
  18. event: &IssuesEvent,
  19. config: Option<&MajorChangeConfig>,
  20. ) -> Result<Option<Invocation>, String> {
  21. let config = if let Some(config) = config {
  22. config
  23. } else {
  24. return Ok(None);
  25. };
  26. let enabling_label = config.enabling_label.as_str();
  27. if event.action == IssuesAction::Edited {
  28. if let Some(changes) = &event.changes {
  29. if let Some(previous_title) = &changes.title {
  30. let prev_issue = ZulipGitHubReference {
  31. number: event.issue.number,
  32. title: previous_title.from.clone(),
  33. repository: event.issue.repository().clone(),
  34. };
  35. if event
  36. .issue
  37. .labels()
  38. .iter()
  39. .any(|l| l.name == enabling_label)
  40. {
  41. return Ok(Some(Invocation::Rename { prev_issue }));
  42. } else {
  43. // Ignore renamed issues without primary label (e.g., major-change)
  44. // to avoid warning about the feature not being enabled.
  45. return Ok(None);
  46. }
  47. }
  48. } else {
  49. log::warn!("Did not note changes in edited issue?");
  50. return Ok(None);
  51. }
  52. }
  53. // If we were labeled with accepted, then issue that event
  54. if matches!(&event.action, IssuesAction::Labeled { label } if label.name == config.accept_label)
  55. {
  56. return Ok(Some(Invocation::AcceptedProposal));
  57. }
  58. // Opening an issue with a label assigned triggers both
  59. // "Opened" and "Labeled" events.
  60. //
  61. // We want to treat reopened issues as new proposals but if the
  62. // issue is freshly opened, we only want to trigger once;
  63. // currently we do so on the label event.
  64. if matches!(event.action, IssuesAction::Reopened if event.issue.labels().iter().any(|l| l.name == enabling_label))
  65. || matches!(&event.action, IssuesAction::Labeled { label } if label.name == enabling_label)
  66. {
  67. return Ok(Some(Invocation::NewProposal));
  68. }
  69. // All other issue events are ignored
  70. return Ok(None);
  71. }
  72. pub(super) async fn handle_input(
  73. ctx: &Context,
  74. config: &MajorChangeConfig,
  75. event: &IssuesEvent,
  76. cmd: Invocation,
  77. ) -> anyhow::Result<()> {
  78. if !event
  79. .issue
  80. .labels()
  81. .iter()
  82. .any(|l| l.name == config.enabling_label)
  83. {
  84. let cmnt = ErrorComment::new(
  85. &event.issue,
  86. format!(
  87. "This issue is not ready for proposals; it lacks the `{}` label.",
  88. config.enabling_label
  89. ),
  90. );
  91. cmnt.post(&ctx.github).await?;
  92. return Ok(());
  93. }
  94. let zulip_msg = match cmd {
  95. Invocation::NewProposal => format!(
  96. "A new proposal has been announced: [{} #{}]({}). It will be \
  97. announced at the next meeting to try and draw attention to it, \
  98. but usually MCPs are not discussed during triage meetings. If \
  99. you think this would benefit from discussion amongst the \
  100. team, consider proposing a design meeting.",
  101. event.issue.title, event.issue.number, event.issue.html_url,
  102. ),
  103. Invocation::AcceptedProposal => format!(
  104. "This proposal has been accepted: [#{}]({}).",
  105. event.issue.number, event.issue.html_url,
  106. ),
  107. Invocation::Rename { prev_issue } => {
  108. let issue = &event.issue;
  109. let prev_topic = zulip_topic_from_issue(&prev_issue);
  110. let partial_issue = issue.to_zulip_github_reference();
  111. let new_topic = zulip_topic_from_issue(&partial_issue);
  112. let zulip_send_req = crate::zulip::MessageApiRequest {
  113. recipient: crate::zulip::Recipient::Stream {
  114. id: config.zulip_stream,
  115. topic: &prev_topic,
  116. },
  117. content: "The associated GitHub issue has been renamed. Renaming this Zulip topic.",
  118. };
  119. let zulip_send_res = zulip_send_req
  120. .send(&ctx.github.raw())
  121. .await
  122. .context("zulip post failed")?;
  123. let zulip_send_res: crate::zulip::MessageApiResponse = zulip_send_res.json().await?;
  124. let zulip_update_req = crate::zulip::UpdateMessageApiRequest {
  125. message_id: zulip_send_res.message_id,
  126. topic: Some(&new_topic),
  127. propagate_mode: Some("change_all"),
  128. content: None,
  129. };
  130. zulip_update_req
  131. .send(&ctx.github.raw())
  132. .await
  133. .context("zulip message update failed")?;
  134. // after renaming the zulip topic, post an additional comment under the old topic with a url to the new, renamed topic
  135. // this is necessary due to the lack of topic permalinks, see https://github.com/zulip/zulip/issues/15290
  136. let new_topic_url = crate::zulip::Recipient::Stream {
  137. id: config.zulip_stream,
  138. topic: &new_topic,
  139. }
  140. .url();
  141. let breadcrumb_comment = format!(
  142. "The associated GitHub issue has been renamed. Please see the [renamed Zulip topic]({}).",
  143. new_topic_url
  144. );
  145. let zulip_send_breadcrumb_req = crate::zulip::MessageApiRequest {
  146. recipient: crate::zulip::Recipient::Stream {
  147. id: config.zulip_stream,
  148. topic: &prev_topic,
  149. },
  150. content: &breadcrumb_comment,
  151. };
  152. zulip_send_breadcrumb_req
  153. .send(&ctx.github.raw())
  154. .await
  155. .context("zulip post failed")?;
  156. return Ok(());
  157. }
  158. };
  159. handle(
  160. ctx,
  161. config,
  162. &event.issue,
  163. zulip_msg,
  164. config.meeting_label.clone(),
  165. cmd == Invocation::NewProposal,
  166. )
  167. .await
  168. }
  169. pub(super) async fn handle_command(
  170. ctx: &Context,
  171. config: &MajorChangeConfig,
  172. event: &Event,
  173. _cmd: SecondCommand,
  174. ) -> anyhow::Result<()> {
  175. let issue = event.issue().unwrap();
  176. if !issue
  177. .labels()
  178. .iter()
  179. .any(|l| l.name == config.enabling_label)
  180. {
  181. let cmnt = ErrorComment::new(
  182. &issue,
  183. &format!(
  184. "This issue cannot be seconded; it lacks the `{}` label.",
  185. config.enabling_label
  186. ),
  187. );
  188. cmnt.post(&ctx.github).await?;
  189. return Ok(());
  190. }
  191. let is_team_member = event
  192. .user()
  193. .is_team_member(&ctx.github)
  194. .await
  195. .ok()
  196. .unwrap_or(false);
  197. if !is_team_member {
  198. let cmnt = ErrorComment::new(&issue, "Only team members can second issues.");
  199. cmnt.post(&ctx.github).await?;
  200. return Ok(());
  201. }
  202. let zulip_msg = format!(
  203. "@*{}*: Proposal [#{}]({}) has been seconded, and will be approved in 10 days if no objections are raised.",
  204. config.zulip_ping,
  205. issue.number,
  206. event.html_url().unwrap()
  207. );
  208. handle(
  209. ctx,
  210. config,
  211. issue,
  212. zulip_msg,
  213. config.second_label.clone(),
  214. false,
  215. )
  216. .await
  217. }
  218. async fn handle(
  219. ctx: &Context,
  220. config: &MajorChangeConfig,
  221. issue: &Issue,
  222. zulip_msg: String,
  223. label_to_add: String,
  224. new_proposal: bool,
  225. ) -> anyhow::Result<()> {
  226. let github_req = issue.add_labels(&ctx.github, vec![Label { name: label_to_add }]);
  227. let partial_issue = issue.to_zulip_github_reference();
  228. let zulip_topic = zulip_topic_from_issue(&partial_issue);
  229. let zulip_req = crate::zulip::MessageApiRequest {
  230. recipient: crate::zulip::Recipient::Stream {
  231. id: config.zulip_stream,
  232. topic: &zulip_topic,
  233. },
  234. content: &zulip_msg,
  235. };
  236. if new_proposal {
  237. let topic_url = zulip_req.url();
  238. let comment = format!(
  239. "This issue is not meant to be used for technical discussion. \
  240. There is a Zulip [stream] for that. Use this issue to leave \
  241. procedural comments, such as volunteering to review, indicating that you \
  242. second the proposal (or third, etc), or raising a concern that you would \
  243. like to be addressed. \
  244. \n\n \
  245. Concerns or objections to the proposal should be discussed on Zulip and formally registered \
  246. here by adding a comment with the following syntax: \
  247. \n \
  248. ``` \
  249. \n \
  250. @dragonosbot concern reason-for-concern \
  251. \n \
  252. <description of the concern> \
  253. \n \
  254. ``` \
  255. \n \
  256. Concerns can be lifted with: \
  257. \n \
  258. ``` \
  259. \n \
  260. @dragonosbot resolve reason-for-concern \
  261. \n \
  262. ``` \
  263. \n\n \
  264. See documentation at [https://forge.rust-lang.org](https://forge.rust-lang.org/compiler/mcp.html#what-kinds-of-comments-should-go-on-the-tracking-issue-in-compiler-team-repo) \
  265. \n\n{} \
  266. \n\n[stream]: {}",
  267. config.open_extra_text.as_deref().unwrap_or_default(),
  268. topic_url
  269. );
  270. issue
  271. .post_comment(&ctx.github, &comment)
  272. .await
  273. .context("post major change comment")?;
  274. }
  275. let zulip_req = zulip_req.send(&ctx.github.raw());
  276. let (gh_res, zulip_res) = futures::join!(github_req, zulip_req);
  277. zulip_res.context("zulip post failed")?;
  278. gh_res.context("label setting failed")?;
  279. Ok(())
  280. }
  281. fn zulip_topic_from_issue(issue: &ZulipGitHubReference) -> String {
  282. // Concatenate the issue title and the topic reference, truncating such that
  283. // the overall length does not exceed 60 characters (a Zulip limitation).
  284. let topic_ref = issue.zulip_topic_reference();
  285. // Skip chars until the last characters that can be written:
  286. // Maximum 60, minus the reference, minus the elipsis and the space
  287. let mut chars = issue
  288. .title
  289. .char_indices()
  290. .skip(60 - topic_ref.chars().count() - 2);
  291. match chars.next() {
  292. Some((len, _)) if chars.next().is_some() => {
  293. format!("{}… {}", &issue.title[..len], topic_ref)
  294. }
  295. _ => format!("{} {}", issue.title, topic_ref),
  296. }
  297. }