use std::io::IsTerminal;
use anyhow::Context;
use chrono::Utc;
use colored::Colorize;
use serde::Serialize;
use crate::api::{DocumentationSnippet, LibrarySearchResult, RespostaDocumentacao};
use crate::i18n::{idioma_atual, t, Idioma, Mensagem};
use crate::storage::ChaveArmazenada;
fn simbolo_ou_ascii<'a>(unicode: &'a str, ascii: &'a str) -> &'a str {
static USAR_ASCII: std::sync::OnceLock<bool> = std::sync::OnceLock::new();
let usar_ascii = *USAR_ASCII.get_or_init(|| {
!std::io::stdout().is_terminal()
|| std::env::var("NO_COLOR").is_ok()
|| std::env::var("TERM").map(|t| t == "dumb").unwrap_or(false)
});
if usar_ascii {
ascii
} else {
unicode
}
}
#[derive(Serialize)]
struct EventoNdjson<'a, T: Serialize> {
#[serde(rename = "type")]
tipo: &'a str,
timestamp: String,
#[serde(flatten)]
dados: T,
}
pub fn emitir_ndjson<T: Serialize>(tipo: &str, dados: &T) {
let evento = EventoNdjson {
tipo,
timestamp: Utc::now().to_rfc3339(),
dados,
};
if let Ok(json) = serde_json::to_string(&evento) {
println!("{json}");
}
}
pub fn exibir_bibliotecas_formatado(resultados: &[LibrarySearchResult]) {
if resultados.is_empty() {
println!("{}", t(Mensagem::NenhumaBibliotecaEncontrada).yellow());
return;
}
println!("{}", t(Mensagem::BibliotecasEncontradas).green().bold());
println!("{}", simbolo_ou_ascii("─", "-").repeat(60).dimmed());
for (i, lib) in resultados.iter().enumerate() {
let numero = format!("{}.", i + 1);
let titulo = if let Some(score) = lib.trust_score {
format!(
"{} {} ({} {:.1}/10)",
numero.cyan(),
lib.title.bold(),
t(Mensagem::ConfiancaScore),
score
)
} else {
format!("{} {}", numero.cyan(), lib.title.bold())
};
println!("{}", titulo);
println!(" {}", lib.id.dimmed());
if let Some(desc) = &lib.description {
println!(" {}", desc.italic());
}
println!();
}
}
pub fn exibir_dica_biblioteca_nao_encontrada() {
eprintln!("{}", t(Mensagem::BibliotecaNaoEncontradaApi).yellow());
}
pub fn exibir_documentacao_formatada(doc: &RespostaDocumentacao) {
let snippets = match &doc.snippets {
Some(s) if !s.is_empty() => s,
_ => {
println!("{}", t(Mensagem::NenhumaDocumentacaoEncontrada).yellow());
return;
}
};
println!("{}", t(Mensagem::TituloDocumentacao).green().bold());
println!("{}", simbolo_ou_ascii("─", "-").repeat(60).dimmed());
for snippet in snippets {
exibir_snippet(snippet);
}
}
fn exibir_snippet(snippet: &DocumentationSnippet) {
if let Some(titulo_pagina) = &snippet.page_title {
println!("{}", format!("## {}", titulo_pagina).green().bold());
}
if let Some(titulo_codigo) = &snippet.code_title {
println!(
"{}",
format!("{} {}", simbolo_ou_ascii("▸", ">"), titulo_codigo).cyan()
);
}
if let Some(descricao) = &snippet.code_description {
println!(" {}", descricao.dimmed().italic());
}
if let Some(blocos) = &snippet.code_list {
for bloco in blocos {
println!("```{}", bloco.language);
println!("{}", bloco.code);
println!("```");
}
}
if let Some(source) = &snippet.code_id {
println!("{}", source.blue().bold().dimmed());
}
println!();
}
pub fn exibir_chaves_mascaradas(chaves: &[ChaveArmazenada], mascarar: impl Fn(&str) -> String) {
println!(
"{}",
format!("{} {}", chaves.len(), t(Mensagem::ContadorChaves))
.green()
.bold()
);
println!("{}", simbolo_ou_ascii("─", "-").repeat(60).dimmed());
let rotulo_adicionada = match idioma_atual() {
Idioma::English => "added:",
Idioma::Portugues => "adicionada:",
};
for (i, chave) in chaves.iter().enumerate() {
println!(
" {} {} {}",
format!("[{}]", i + 1).cyan(),
mascarar(&chave.value).bold(),
format!(
"({} {})",
rotulo_adicionada,
formatar_added_at_display(&chave.added_at)
)
.dimmed()
);
}
}
pub fn formatar_added_at_display(iso: &str) -> String {
chrono::DateTime::parse_from_rfc3339(iso)
.map(|dt| dt.format("%Y-%m-%d %H:%M:%S").to_string())
.unwrap_or_else(|_| iso.to_string())
}
pub fn exibir_nenhuma_chave() {
println!("{}", t(Mensagem::NenhumaChaveArmazenada).yellow());
println!("{}", t(Mensagem::UsarKeysAdd).cyan());
}
pub fn exibir_nenhuma_chave_para_remover() {
println!("{}", t(Mensagem::NenhumaChaveParaRemover).yellow());
}
pub fn exibir_indice_invalido(_indice: usize, total: usize) {
println!(
"{}",
format!("{} {}.", t(Mensagem::IndiceInvalido), total).red()
);
}
pub fn exibir_chave_adicionada(caminho: &std::path::Path) {
println!(
"{} {}",
t(Mensagem::ChaveAdicionada),
caminho.display().to_string().green()
);
}
pub fn exibir_chave_ja_existia() {
println!("{}", t(Mensagem::ChaveJaExistia).yellow());
}
pub fn exibir_chave_invalida_vazia() {
eprintln!("{}", t(Mensagem::ChaveVaziaOuInvalida).red());
}
pub fn exibir_aviso_formato_chave() {
eprintln!("{}", t(Mensagem::AvisoFormatoChave).yellow());
}
pub fn exibir_chave_removida(chave_mascarada: &str) {
println!(
"{} {}",
chave_mascarada.bold(),
t(Mensagem::ChaveRemovidaSucesso)
);
}
pub fn exibir_operacao_cancelada() {
println!("{}", t(Mensagem::OperacaoCancelada).yellow());
}
pub fn exibir_chaves_removidas() {
println!("{}", t(Mensagem::TodasChavesRemovidas).green());
}
pub fn exibir_xdg_nao_suportado() {
println!("{}", t(Mensagem::SistemaXdgNaoSuportado).red());
}
pub fn exibir_json_array_vazio() {
println!("[]");
}
pub fn exibir_json_bruto(json: &str) {
println!("{}", json);
}
pub fn exibir_caminho_config(caminho: &std::path::Path) {
println!("{}", caminho.display());
}
pub fn exibir_chave_exportada(valor: &str) {
println!("CONTEXT7_API={}", valor);
}
pub fn exibir_json_resultados(json: &str) {
println!("{}", json);
}
pub fn exibir_texto_plano(texto: &str) {
println!("{}", texto);
}
pub fn confirmar_clear() -> anyhow::Result<bool> {
use std::io::Write;
print!("{}", t(Mensagem::ConfirmarRemoverTodas));
std::io::stdout()
.flush()
.context("Falha ao limpar buffer de saída")?;
let mut entrada = String::new();
std::io::stdin()
.read_line(&mut entrada)
.context("Falha ao ler confirmação do usuário")?;
Ok(matches!(
entrada.trim().to_lowercase().as_str(),
"s" | "sim" | "y" | "yes"
))
}
pub fn exibir_importacao_concluida(importadas: usize, total: usize) {
println!(
"{}",
format!(
"{}/{} {}",
importadas,
total,
t(Mensagem::ChavesImportadasSucesso)
)
.green()
);
}
#[cfg(test)]
mod testes {
use super::formatar_added_at_display;
#[test]
fn testa_formatar_added_at_rfc3339_com_nanossegundos() {
let resultado = formatar_added_at_display("2026-04-09T13:34:59.060818734+00:00");
assert_eq!(resultado, "2026-04-09 13:34:59");
assert!(
!resultado.contains('T'),
"Resultado não deve conter 'T': {resultado}"
);
assert!(
!resultado.contains('.'),
"Resultado não deve conter nanossegundos: {resultado}"
);
assert!(
!resultado.contains("+00:00"),
"Resultado não deve conter offset de timezone: {resultado}"
);
}
#[test]
fn testa_formatar_added_at_rfc3339_sem_nanossegundos() {
let resultado = formatar_added_at_display("2026-01-01T00:00:00+00:00");
assert_eq!(resultado, "2026-01-01 00:00:00");
}
#[test]
fn testa_formatar_added_at_rfc3339_offset_nao_utc() {
let resultado = formatar_added_at_display("2026-04-09T10:00:00-03:00");
assert_eq!(resultado, "2026-04-09 10:00:00");
assert!(
!resultado.contains("-03:00"),
"Resultado não deve conter offset de timezone: {resultado}"
);
}
#[test]
fn testa_formatar_added_at_fallback_string_invalida() {
let resultado = formatar_added_at_display("lixo-nao-e-data");
assert_eq!(
resultado, "lixo-nao-e-data",
"String inválida deve ser retornada sem modificação"
);
}
#[test]
fn testa_formatar_added_at_string_vazia() {
let resultado = formatar_added_at_display("");
assert_eq!(
resultado, "",
"String vazia deve ser retornada sem modificação"
);
}
#[test]
fn testa_formatar_added_at_formato_saida_legivel() {
let resultado = formatar_added_at_display("2026-04-09T13:34:59.123456789+00:00");
assert_eq!(
resultado.len(),
19,
"Formato de saída deve ter 19 caracteres, obteve: '{resultado}'"
);
assert!(
resultado.contains(' '),
"Resultado deve conter espaço separando data e hora: {resultado}"
);
}
}