123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- //! Allow users to add summary comments in Issues & Pull Requests.
- //!
- //! Users can make a new summary entry by commenting the following:
- //!
- //! ```md
- //! @dragonosbot note summary-title
- //! ```
- //!
- //! If this is the first summary entry, rustbot will amend the original post (the top-level comment) to add a "Notes" section. The section should **not** be edited by hand.
- //!
- //! ```md
- //! <!-- TRIAGEBOT_SUMMARY_START -->
- //!
- //! ### Summary Notes
- //!
- //! - ["summary-title" by @username](link-to-comment)
- //!
- //! Generated by triagebot, see [help](https://forge.rust-lang.org/triagebot/note.html) for how to add more
- //! <!-- TRIAGEBOT_SUMMARY_END -->
- //! ```
- //!
- //! If this is *not* the first summary entry, rustbot will simply append the new entry to the existing notes section:
- //!
- //! ```md
- //! <!-- TRIAGEBOT_SUMMARY_START -->
- //!
- //! ### Summary Notes
- //!
- //! - ["first-note" by @username](link-to-comment)
- //! - ["second-note" by @username](link-to-comment)
- //! - ["summary-title" by @username](link-to-comment)
- //!
- //! <!-- TRIAGEBOT_SUMMARY_END -->
- //! ```
- //!
- use crate::{config::NoteConfig, github::Event, handlers::Context, interactions::EditIssueBody};
- use itertools::Itertools;
- use parser::command::note::NoteCommand;
- use std::{cmp::Ordering, collections::HashMap};
- use tracing as log;
- #[derive(Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize, Clone)]
- struct NoteDataEntry {
- title: String,
- comment_url: String,
- author: String,
- }
- impl NoteDataEntry {
- pub fn to_markdown(&self) -> String {
- format!(
- "\n- [\"{title}\" by @{author}]({comment_url})",
- title = self.title,
- author = self.author,
- comment_url = self.comment_url
- )
- }
- }
- impl Ord for NoteDataEntry {
- fn cmp(&self, other: &Self) -> Ordering {
- self.comment_url.cmp(&other.comment_url)
- }
- }
- impl PartialOrd for NoteDataEntry {
- fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
- Some(self.cmp(other))
- }
- }
- #[derive(Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize, Default)]
- struct NoteData {
- entries_by_url: HashMap<String, NoteDataEntry>,
- }
- impl NoteData {
- pub fn get_url_from_title(&self, title: &str) -> Option<String> {
- let tmp = self.entries_by_url.clone();
- tmp.iter().sorted().find_map(|(key, val)| {
- if val.title == title {
- Some(key.to_owned())
- } else {
- None
- }
- })
- }
- pub fn remove_by_title(&mut self, title: &str) -> Option<NoteDataEntry> {
- if let Some(url_to_remove) = self.get_url_from_title(title) {
- if let Some(entry) = self.entries_by_url.remove(&url_to_remove) {
- log::debug!("SUCCESSFULLY REMOVED ENTRY: {:#?}", &entry);
- Some(entry)
- } else {
- log::debug!("UNABLE TO REMOVE ENTRY WITH URL: {:?}", &url_to_remove);
- None
- }
- } else {
- log::debug!("UNABLE TO REMOVE ENTRY WITH TITLE: {:?}", title);
- None
- }
- }
- pub fn to_markdown(&self) -> String {
- if self.entries_by_url.is_empty() {
- return String::new();
- }
- let mut text = String::from("\n### Summary Notes\n");
- for (_, entry) in self.entries_by_url.iter().sorted() {
- text.push_str(&entry.to_markdown());
- }
- text.push_str("\n\nGenerated by triagebot, see [help](https://forge.rust-lang.org/triagebot/note.html) for how to add more");
- text
- }
- }
- pub(super) async fn handle_command(
- ctx: &Context,
- _config: &NoteConfig,
- event: &Event,
- cmd: NoteCommand,
- ) -> anyhow::Result<()> {
- let issue = event.issue().unwrap();
- let e = EditIssueBody::new(&issue, "SUMMARY");
- let mut current: NoteData = e.current_data().unwrap_or_default();
- let comment_url = String::from(event.html_url().unwrap());
- let author = event.user().login.to_owned();
- match &cmd {
- NoteCommand::Summary { title } => {
- let title = title.to_owned();
- if let Some(existing_entry) = current.entries_by_url.get_mut(&comment_url) {
- existing_entry.title = title;
- log::debug!("Updated existing entry: {:#?}", existing_entry);
- } else {
- let new_entry = NoteDataEntry {
- title,
- comment_url: comment_url.clone(),
- author,
- };
- log::debug!("New Note Entry: {:#?}", new_entry);
- current.entries_by_url.insert(comment_url, new_entry);
- log::debug!("Entries by URL: {:#?}", current.entries_by_url);
- }
- }
- NoteCommand::Remove { title } => {
- if let Some(entry) = current.remove_by_title(title) {
- log::debug!("SUCCESSFULLY REMOVED ENTRY: {:#?}", entry);
- } else {
- log::debug!("UNABLE TO REMOVE ENTRY");
- }
- }
- }
- let new_markdown = current.to_markdown();
- log::debug!("New MD: {:#?}", new_markdown);
- e.apply(&ctx.github, new_markdown, current).await?;
- Ok(())
- }
|