codeberg_cli/actions/issue/
view.rsuse crate::render::comment::render_comment;
use crate::render::datetime::render_datetime_and_info;
use crate::render::option::option_display;
use crate::render::spinner::spin_until_ready;
use crate::render::ui::fuzzy_select_with_key;
use crate::types::api::state_type::ViewStateType;
use crate::types::context::BergContext;
use crate::{actions::GeneralArgs, types::git::OwnerRepo};
use anyhow::Context;
use forgejo_api::structs::{
Issue, IssueGetCommentsQuery, IssueListIssuesQuery, IssueListIssuesQueryState,
};
use crate::actions::text_manipulation::select_prompt_for;
use super::display_issue;
use clap::Parser;
#[derive(Parser, Debug)]
pub struct ViewIssueArgs {
#[arg(short, long, value_enum, default_value_t = ViewStateType::All)]
pub state: ViewStateType,
#[arg(short, long)]
pub comments: bool,
}
impl ViewIssueArgs {
pub async fn run(self, general_args: GeneralArgs) -> anyhow::Result<()> {
let _ = general_args;
let ctx = BergContext::new(self).await?;
let OwnerRepo { repo, owner } = ctx.owner_repo()?;
let state = match ctx.args.state {
ViewStateType::Closed => IssueListIssuesQueryState::Closed,
ViewStateType::Open => IssueListIssuesQueryState::Open,
ViewStateType::All => IssueListIssuesQueryState::All,
};
let issues_list = spin_until_ready(ctx.client.issue_list_issues(
owner.as_str(),
repo.as_str(),
IssueListIssuesQuery {
state: Some(state),
..Default::default()
},
))
.await?;
let selected_issue =
fuzzy_select_with_key(&issues_list, select_prompt_for("issue"), display_issue)?;
if ctx.args.comments {
spin_until_ready(present_issue_comments(&ctx, selected_issue)).await?;
} else {
present_issue_overview(&ctx, selected_issue);
}
Ok(())
}
}
fn present_issue_overview(ctx: &BergContext<ViewIssueArgs>, issue: &Issue) {
let days_passed_since_creation =
option_display(&issue.created_at.as_ref().map(render_datetime_and_info));
let mut table = ctx.make_table();
table
.set_header(vec![option_display(
&issue.id.as_ref().map(|id| format!("Issue #{}", id)),
)])
.add_row(vec![String::from("Title"), option_display(&issue.title)])
.add_row(vec![String::from("Created"), days_passed_since_creation])
.add_row(vec![
String::from("Labels"),
option_display(&issue.labels.as_ref().map(|labels| {
labels
.iter()
.map(|label| option_display(&label.name))
.collect::<Vec<_>>()
.join(", ")
})),
])
.add_row(vec![
String::from("Description"),
option_display(&issue.body),
]);
println!("{table}");
}
async fn present_issue_comments(
ctx: &BergContext<ViewIssueArgs>,
issue: &Issue,
) -> anyhow::Result<()> {
let id = issue.id.context("Issue has no Id")?;
let OwnerRepo { repo, owner } = ctx.owner_repo()?;
let (header, comments) = {
let comments_list = ctx
.client
.issue_get_comments(
owner.as_str(),
repo.as_str(),
id,
IssueGetCommentsQuery::default(),
)
.await?;
let header = format!(
"Issue #{} {}",
id,
if comments_list.is_empty() {
"(no comments)"
} else {
"comments"
}
);
(header, comments_list)
};
let mut table = ctx.make_table();
table
.add_row(vec![header])
.add_rows(comments.into_iter().filter_map(|comment| {
let username = comment.user.as_ref()?.login.as_ref()?.as_str();
let creation_time = comment.created_at.as_ref()?;
let comment = comment.body.as_ref()?;
let comment = render_comment(&ctx.config, username, creation_time, comment);
Some(vec![comment])
}));
println!("{table}");
Ok(())
}