|
@@ -1,182 +1,160 @@
|
|
|
use crate::{
|
|
|
config::MajorChangeConfig,
|
|
|
- github::{self, Event, IssuesAction},
|
|
|
- handlers::{Context, Handler},
|
|
|
+ github::{Event, Issue, IssuesAction, IssuesEvent, Label},
|
|
|
+ handlers::Context,
|
|
|
interactions::ErrorComment,
|
|
|
};
|
|
|
use anyhow::Context as _;
|
|
|
-use futures::future::{BoxFuture, FutureExt};
|
|
|
use parser::command::second::SecondCommand;
|
|
|
-use parser::command::{Command, Input};
|
|
|
|
|
|
-#[derive(Copy, Clone, PartialEq, Eq)]
|
|
|
-pub(super) enum Invocation {
|
|
|
- Second,
|
|
|
+#[derive(Copy, Clone, PartialEq, Eq, Debug)]
|
|
|
+pub enum Invocation {
|
|
|
NewProposal,
|
|
|
AcceptedProposal,
|
|
|
}
|
|
|
|
|
|
-pub(super) struct MajorChangeHandler;
|
|
|
-
|
|
|
-impl Handler for MajorChangeHandler {
|
|
|
- type Input = Invocation;
|
|
|
- type Config = MajorChangeConfig;
|
|
|
-
|
|
|
- fn parse_input(
|
|
|
- &self,
|
|
|
- ctx: &Context,
|
|
|
- event: &Event,
|
|
|
- _: Option<&Self::Config>,
|
|
|
- ) -> Result<Option<Self::Input>, String> {
|
|
|
- let body = if let Some(b) = event.comment_body() {
|
|
|
- b
|
|
|
- } else {
|
|
|
- // not interested in other events
|
|
|
- return Ok(None);
|
|
|
- };
|
|
|
-
|
|
|
- match event {
|
|
|
- Event::Issue(e) => {
|
|
|
- // If we were labeled with accepted, then issue that event
|
|
|
- if e.action == IssuesAction::Labeled
|
|
|
- && e.label
|
|
|
- .as_ref()
|
|
|
- .map_or(false, |l| l.name == "major-change-accepted")
|
|
|
- {
|
|
|
- return Ok(Some(Invocation::AcceptedProposal));
|
|
|
- }
|
|
|
-
|
|
|
- // Opening an issue with a label assigned triggers both
|
|
|
- // "Opened" and "Labeled" events.
|
|
|
- //
|
|
|
- // We want to treat reopened issues as new proposals but if the
|
|
|
- // issues is freshly opened, we only want to trigger once;
|
|
|
- // currently we do so on the label event.
|
|
|
- if (e.action == IssuesAction::Reopened
|
|
|
- && e.issue.labels().iter().any(|l| l.name == "major-change"))
|
|
|
- || (e.action == IssuesAction::Labeled
|
|
|
- && e.label.as_ref().map_or(false, |l| l.name == "major-change"))
|
|
|
- {
|
|
|
- return Ok(Some(Invocation::NewProposal));
|
|
|
- }
|
|
|
-
|
|
|
- // All other issue events are ignored
|
|
|
- return Ok(None);
|
|
|
- }
|
|
|
- Event::IssueComment(_) => {}
|
|
|
- }
|
|
|
-
|
|
|
- let mut input = Input::new(&body, &ctx.username);
|
|
|
- let command = input.parse_command();
|
|
|
-
|
|
|
- if let Some(previous) = event.comment_from() {
|
|
|
- let mut prev_input = Input::new(&previous, &ctx.username);
|
|
|
- let prev_command = prev_input.parse_command();
|
|
|
- if command == prev_command {
|
|
|
- return Ok(None);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- match command {
|
|
|
- Command::Second(Ok(SecondCommand)) => Ok(Some(Invocation::Second)),
|
|
|
- _ => Ok(None),
|
|
|
- }
|
|
|
+pub(super) fn parse_input(
|
|
|
+ _ctx: &Context,
|
|
|
+ event: &IssuesEvent,
|
|
|
+ _config: Option<&MajorChangeConfig>,
|
|
|
+) -> Result<Option<Invocation>, String> {
|
|
|
+ // If we were labeled with accepted, then issue that event
|
|
|
+ if event.action == IssuesAction::Labeled
|
|
|
+ && event
|
|
|
+ .label
|
|
|
+ .as_ref()
|
|
|
+ .map_or(false, |l| l.name == "major-change-accepted")
|
|
|
+ {
|
|
|
+ return Ok(Some(Invocation::AcceptedProposal));
|
|
|
}
|
|
|
|
|
|
- fn handle_input<'a>(
|
|
|
- &self,
|
|
|
- ctx: &'a Context,
|
|
|
- config: &'a Self::Config,
|
|
|
- event: &'a Event,
|
|
|
- cmd: Self::Input,
|
|
|
- ) -> BoxFuture<'a, anyhow::Result<()>> {
|
|
|
- handle_input(ctx, config, event, cmd).boxed()
|
|
|
+ // Opening an issue with a label assigned triggers both
|
|
|
+ // "Opened" and "Labeled" events.
|
|
|
+ //
|
|
|
+ // We want to treat reopened issues as new proposals but if the
|
|
|
+ // issues is freshly opened, we only want to trigger once;
|
|
|
+ // currently we do so on the label event.
|
|
|
+ if (event.action == IssuesAction::Reopened
|
|
|
+ && event
|
|
|
+ .issue
|
|
|
+ .labels()
|
|
|
+ .iter()
|
|
|
+ .any(|l| l.name == "major-change"))
|
|
|
+ || (event.action == IssuesAction::Labeled
|
|
|
+ && event
|
|
|
+ .label
|
|
|
+ .as_ref()
|
|
|
+ .map_or(false, |l| l.name == "major-change"))
|
|
|
+ {
|
|
|
+ return Ok(Some(Invocation::NewProposal));
|
|
|
}
|
|
|
+
|
|
|
+ // All other issue events are ignored
|
|
|
+ return Ok(None);
|
|
|
}
|
|
|
|
|
|
-async fn handle_input(
|
|
|
+pub(super) async fn handle_input(
|
|
|
ctx: &Context,
|
|
|
config: &MajorChangeConfig,
|
|
|
- event: &Event,
|
|
|
+ event: &IssuesEvent,
|
|
|
cmd: Invocation,
|
|
|
+) -> anyhow::Result<()> {
|
|
|
+ if !event
|
|
|
+ .issue
|
|
|
+ .labels()
|
|
|
+ .iter()
|
|
|
+ .any(|l| l.name == "major-change")
|
|
|
+ {
|
|
|
+ let cmnt = ErrorComment::new(
|
|
|
+ &event.issue,
|
|
|
+ "This is not a major change (it lacks the `major-change` label).",
|
|
|
+ );
|
|
|
+ cmnt.post(&ctx.github).await?;
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+ let zulip_msg = match cmd {
|
|
|
+ Invocation::NewProposal => format!(
|
|
|
+ "A new proposal has been announced: [#{}]({}). It will be \
|
|
|
+ announced at the next meeting to try and draw attention to it, \
|
|
|
+ but usually MCPs are not discussed during triage meetings. If \
|
|
|
+ you think this would benefit from discussion amongst the \
|
|
|
+ team, consider proposing a design meeting.",
|
|
|
+ event.issue.number, event.issue.html_url,
|
|
|
+ ),
|
|
|
+ Invocation::AcceptedProposal => format!(
|
|
|
+ "This proposal has been accepted: [#{}]({}).",
|
|
|
+ event.issue.number, event.issue.html_url,
|
|
|
+ ),
|
|
|
+ };
|
|
|
+ handle(
|
|
|
+ ctx,
|
|
|
+ config,
|
|
|
+ &event.issue,
|
|
|
+ zulip_msg,
|
|
|
+ config.meeting_label.clone(),
|
|
|
+ cmd == Invocation::NewProposal,
|
|
|
+ )
|
|
|
+ .await
|
|
|
+}
|
|
|
+
|
|
|
+pub(super) async fn handle_command(
|
|
|
+ ctx: &Context,
|
|
|
+ config: &MajorChangeConfig,
|
|
|
+ event: &Event,
|
|
|
+ _cmd: SecondCommand,
|
|
|
) -> anyhow::Result<()> {
|
|
|
let issue = event.issue().unwrap();
|
|
|
|
|
|
- let (zulip_msg, label_to_add) = match cmd {
|
|
|
- Invocation::Second => {
|
|
|
- if !issue.labels().iter().any(|l| l.name == "major-change") {
|
|
|
- let cmnt = ErrorComment::new(
|
|
|
- &issue,
|
|
|
- "This is not a major change (it lacks the `major-change` label).",
|
|
|
- );
|
|
|
- cmnt.post(&ctx.github).await?;
|
|
|
- return Ok(());
|
|
|
- }
|
|
|
-
|
|
|
- let is_team_member =
|
|
|
- if let Err(_) | Ok(false) = event.user().is_team_member(&ctx.github).await {
|
|
|
- false
|
|
|
- } else {
|
|
|
- true
|
|
|
- };
|
|
|
-
|
|
|
- if !is_team_member {
|
|
|
- let cmnt = ErrorComment::new(&issue, "Only team members can second issues.");
|
|
|
- cmnt.post(&ctx.github).await?;
|
|
|
- return Ok(());
|
|
|
- }
|
|
|
-
|
|
|
- (format!(
|
|
|
- "@*{}*: Proposal [#{}]({}) has been seconded, and will be approved in 10 days if no objections are raised.",
|
|
|
- config.zulip_ping,
|
|
|
- issue.number,
|
|
|
- event.html_url().unwrap()
|
|
|
- ), config.second_label.clone())
|
|
|
- }
|
|
|
- Invocation::NewProposal => {
|
|
|
- if !issue.labels().iter().any(|l| l.name == "major-change") {
|
|
|
- let cmnt = ErrorComment::new(
|
|
|
- &issue,
|
|
|
- "This is not a major change (it lacks the `major-change` label).",
|
|
|
- );
|
|
|
- cmnt.post(&ctx.github).await?;
|
|
|
- return Ok(());
|
|
|
- }
|
|
|
- (
|
|
|
- format!(
|
|
|
- "A new proposal has been announced: [#{}]({}). It will be
|
|
|
- announced at the next meeting to try and draw attention to it,
|
|
|
- but usually MCPs are not discussed during triage meetings. If
|
|
|
- you think this would benefit from discussion amongst the
|
|
|
- team, consider proposing a design meeting.",
|
|
|
- issue.number,
|
|
|
- event.html_url().unwrap()
|
|
|
- ),
|
|
|
- config.meeting_label.clone(),
|
|
|
- )
|
|
|
- }
|
|
|
- Invocation::AcceptedProposal => {
|
|
|
- if !issue.labels().iter().any(|l| l.name == "major-change") {
|
|
|
- let cmnt = ErrorComment::new(
|
|
|
- &issue,
|
|
|
- "This is not a major change (it lacks the `major-change` label).",
|
|
|
- );
|
|
|
- cmnt.post(&ctx.github).await?;
|
|
|
- return Ok(());
|
|
|
- }
|
|
|
- (
|
|
|
- format!(
|
|
|
- "This proposal has been accepted: [#{}]({}).",
|
|
|
- issue.number,
|
|
|
- event.html_url().unwrap()
|
|
|
- ),
|
|
|
- config.meeting_label.clone(),
|
|
|
- )
|
|
|
- }
|
|
|
- };
|
|
|
+ if !issue.labels().iter().any(|l| l.name == "major-change") {
|
|
|
+ let cmnt = ErrorComment::new(
|
|
|
+ &issue,
|
|
|
+ "This is not a major change (it lacks the `major-change` label).",
|
|
|
+ );
|
|
|
+ cmnt.post(&ctx.github).await?;
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+
|
|
|
+ let is_team_member = event
|
|
|
+ .user()
|
|
|
+ .is_team_member(&ctx.github)
|
|
|
+ .await
|
|
|
+ .ok()
|
|
|
+ .unwrap_or(false);
|
|
|
+
|
|
|
+ if !is_team_member {
|
|
|
+ let cmnt = ErrorComment::new(&issue, "Only team members can second issues.");
|
|
|
+ cmnt.post(&ctx.github).await?;
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+
|
|
|
+ let zulip_msg = format!(
|
|
|
+ "@*{}*: Proposal [#{}]({}) has been seconded, and will be approved in 10 days if no objections are raised.",
|
|
|
+ config.zulip_ping,
|
|
|
+ issue.number,
|
|
|
+ event.html_url().unwrap()
|
|
|
+ );
|
|
|
|
|
|
+ handle(
|
|
|
+ ctx,
|
|
|
+ config,
|
|
|
+ issue,
|
|
|
+ zulip_msg,
|
|
|
+ config.second_label.clone(),
|
|
|
+ false,
|
|
|
+ )
|
|
|
+ .await
|
|
|
+}
|
|
|
+
|
|
|
+async fn handle(
|
|
|
+ ctx: &Context,
|
|
|
+ config: &MajorChangeConfig,
|
|
|
+ issue: &Issue,
|
|
|
+ zulip_msg: String,
|
|
|
+ label_to_add: String,
|
|
|
+ new_proposal: bool,
|
|
|
+) -> anyhow::Result<()> {
|
|
|
let mut labels = issue.labels().to_owned();
|
|
|
- labels.push(github::Label { name: label_to_add });
|
|
|
+ labels.push(Label { name: label_to_add });
|
|
|
let github_req = issue.set_labels(&ctx.github, labels);
|
|
|
|
|
|
let mut zulip_topic = format!(" {}", issue.zulip_topic_reference());
|
|
@@ -195,7 +173,7 @@ async fn handle_input(
|
|
|
content: &zulip_msg,
|
|
|
};
|
|
|
|
|
|
- if cmd == Invocation::NewProposal {
|
|
|
+ if new_proposal {
|
|
|
let topic_url = zulip_req.url();
|
|
|
let comment = format!(
|
|
|
"This issue is not meant to be used for technical discussion. \
|