feat(backend/frontend): integrar dados do Lattes no ranking

- Adicionar endpoint /consultor/{id}/lattes para buscar producoes
- Incluir id_lattes e titulacoes na resposta do ranking paginado
- Adicionar LattesSchema no backend
- Adicionar funcao getLattes no servico frontend
- Simplificar botao Producoes removendo estado loading desnecessario
This commit is contained in:
Frederico Castro
2025-12-26 23:35:32 -03:00
parent 962cea0fd9
commit 9d3b4d37b7
5 changed files with 249 additions and 4 deletions

1
backend/.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
instantclient_23_7/

View File

@@ -236,13 +236,16 @@ async def ranking_paginado(
if not d.get("idiomas"):
faltando_idiomas.append((c.id_pessoa, d))
if faltando_idiomas:
ids = [item[0] for item in faltando_idiomas]
faltando_lattes = [(c.id_pessoa, d) for c, d in consultores_dados if not d.get("lattes")]
ids_buscar = list(set([item[0] for item in faltando_idiomas] + [item[0] for item in faltando_lattes]))
if ids_buscar:
docs = await es_client.buscar_por_ids(
ids,
source_fields=["id", "dadosPessoais", "idiomas", "atuacoes", "formacoes"],
ids_buscar,
source_fields=["id", "dadosPessoais", "idiomas", "atuacoes", "formacoes", "identificadorLattes", "titulacoes"],
)
docs_map = {int(doc.get("id")): doc for doc in docs if doc.get("id")}
for id_pessoa, detalhes in faltando_idiomas:
doc = docs_map.get(int(id_pessoa))
if not doc:
@@ -264,6 +267,32 @@ async def ranking_paginado(
if titulacao:
detalhes["titulacao"] = titulacao
for id_pessoa, detalhes in faltando_lattes:
doc = docs_map.get(int(id_pessoa))
if not doc:
continue
id_lattes_obj = doc.get("identificadorLattes")
titulacoes_raw = doc.get("titulacoes", [])
if id_lattes_obj and id_lattes_obj.get("descricao"):
id_lattes = id_lattes_obj.get("descricao")
titulacoes_formatadas = []
for t in titulacoes_raw:
grau_obj = t.get("grauAcademico", {})
ies_obj = t.get("ies", {})
titulacoes_formatadas.append({
"grau": grau_obj.get("nome", ""),
"ano": t.get("ano"),
"ies_nome": ies_obj.get("nome"),
"ies_sigla": ies_obj.get("sigla"),
"area": t.get("areaConhecimento", {}).get("nome"),
"pais": "Brasil",
})
detalhes["lattes"] = {
"id_lattes": id_lattes,
"url": f"http://lattes.cnpq.br/{id_lattes}",
"titulacoes": titulacoes_formatadas,
}
for c, d in consultores_dados:
tipos_atuacao = RankingMapper._extrair_tipos_atuacao(d)
consultores_schema.append(
@@ -294,6 +323,7 @@ async def ranking_paginado(
idiomas=d.get("idiomas"),
titulacao=d.get("titulacao"),
pontuacao=d.get("pontuacao"),
lattes=d.get("lattes"),
)
)
@@ -436,6 +466,53 @@ async def obter_consultor_raw(
raise HTTPException(status_code=500, detail=str(e))
@router.get("/consultor/{id_pessoa}/lattes")
async def obter_lattes(
id_pessoa: int,
es_client: ElasticsearchClient = Depends(get_es_client),
):
docs = await es_client.buscar_por_ids(
[id_pessoa],
source_fields=["id", "identificadorLattes", "titulacoes"],
)
if not docs:
return {"encontrado": False, "motivo": "Consultor não encontrado"}
doc = docs[0]
id_lattes_obj = doc.get("identificadorLattes")
if not id_lattes_obj or not id_lattes_obj.get("descricao"):
return {"encontrado": False, "motivo": "Currículo Lattes não cadastrado"}
id_lattes = id_lattes_obj.get("descricao")
titulacoes_raw = doc.get("titulacoes", [])
titulacoes = []
for t in titulacoes_raw:
grau_obj = t.get("grauAcademico", {})
ies_obj = t.get("ies", {})
area_obj = t.get("areaConhecimento", {})
titulacoes.append({
"grau": grau_obj.get("nome", ""),
"ano": t.get("ano"),
"inicio": t.get("inicio"),
"fim": t.get("fim"),
"ies_nome": ies_obj.get("nome"),
"ies_sigla": ies_obj.get("sigla"),
"area": area_obj.get("nome"),
"area_avaliacao": area_obj.get("areaAvaliacao", {}).get("nome") if area_obj.get("areaAvaliacao") else None,
"programa": t.get("programa", {}).get("nome") if t.get("programa") else None,
})
return {
"encontrado": True,
"id_lattes": id_lattes,
"url": f"http://lattes.cnpq.br/{id_lattes}",
"titulacoes": titulacoes,
"orientacoes_lattes": len([t for t in titulacoes if t.get("grau") in ["Doutorado", "Mestrado"]]),
}
@router.get("/consultor/{id_pessoa}/pdf")
async def exportar_ficha_pdf(
id_pessoa: int,

View File

@@ -3,6 +3,12 @@ from typing import Optional, List
from datetime import datetime
class LattesSchema(BaseModel):
id_lattes: Optional[str] = None
url: Optional[str] = None
titulacoes: Optional[list] = None
class ConsultorRankingResumoSchema(BaseModel):
id_pessoa: int
nome: str
@@ -30,6 +36,7 @@ class ConsultorRankingResumoSchema(BaseModel):
idiomas: Optional[list] = None
titulacao: Optional[str] = None
pontuacao: Optional[dict] = None
lattes: Optional[LattesSchema] = None
class RankingPaginadoResponseSchema(BaseModel):

View File

@@ -697,6 +697,8 @@ const ItemDetalheModal = ({ item, tipo, onClose }) => {
const getTitulo = () => {
switch (tipo) {
case 'titulacao': return 'Titulação';
case 'producoes_lattes': return 'Produções Lattes';
case 'vinculo': return 'Vínculo de Consultoria';
case 'coordenacao': return 'Coordenação CAPES';
case 'premiacao': return 'Premiação';
@@ -710,6 +712,8 @@ const ItemDetalheModal = ({ item, tipo, onClose }) => {
const getIcone = () => {
switch (tipo) {
case 'titulacao': return '🎓';
case 'producoes_lattes': return '📚';
case 'vinculo': return '💼';
case 'coordenacao': return '🎯';
case 'premiacao': return '🏆';
@@ -723,6 +727,90 @@ const ItemDetalheModal = ({ item, tipo, onClose }) => {
const renderContent = () => {
switch (tipo) {
case 'titulacao': {
return (
<div className="modal-detalhe-content">
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Grau</span>
<span className="modal-detalhe-value">{item.grau || 'N/A'}</span>
</div>
{item.area && (
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Área</span>
<span className="modal-detalhe-value">{item.area}</span>
</div>
)}
{(item.ies_sigla || item.ies_nome) && (
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Instituição</span>
<span className="modal-detalhe-value">
{item.ies_sigla && item.ies_nome ? `${item.ies_sigla} - ${item.ies_nome}` : (item.ies_sigla || item.ies_nome)}
</span>
</div>
)}
{item.pais && (
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">País</span>
<span className="modal-detalhe-value">{item.pais}</span>
</div>
)}
{item.ano && (
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Ano</span>
<span className="modal-detalhe-value">{item.ano}</span>
</div>
)}
</div>
);
}
case 'producoes_lattes': {
const producoes = item.producoes || item.producoes_recentes || [];
return (
<div className="modal-detalhe-content">
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Total</span>
<span className="modal-detalhe-value">{item.total_producoes ?? 0}</span>
</div>
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Bibliográfica</span>
<span className="modal-detalhe-value">{item.producao_bibliografica ?? 0}</span>
</div>
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Técnica</span>
<span className="modal-detalhe-value">{item.producao_tecnica ?? 0}</span>
</div>
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Orientações</span>
<span className="modal-detalhe-value">{item.orientacoes_lattes ?? 0}</span>
</div>
{item.data_atualizacao_lattes && (
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Atualização</span>
<span className="modal-detalhe-value">{formatDate(item.data_atualizacao_lattes)}</span>
</div>
)}
<div className="modal-detalhe-row">
<span className="modal-detalhe-label">Produções recentes</span>
<span className="modal-detalhe-value">{producoes.length}</span>
</div>
{producoes.length > 0 ? (
<ul className="modal-list" style={{ marginTop: '0.6rem' }}>
{producoes.map((prod, idx) => (
<li key={idx} className="modal-item">
<span className="modal-item-main">{prod.titulo || 'Sem título'}</span>
<span className="muted">{prod.ano || '-'}</span>
{prod.natureza && <span className="badge">{prod.natureza}</span>}
</li>
))}
</ul>
) : (
<p className="modal-empty">Nenhuma produção encontrada.</p>
)}
</div>
);
}
case 'vinculo': {
const periodo = item.periodo || {};
const isAtivo = periodo.ativo ?? !periodo.fim;
@@ -1366,6 +1454,7 @@ const ConsultorCard = memo(({ consultor, highlight, selecionado, onToggleSelecio
const [tipoAtuacaoModal, setTipoAtuacaoModal] = useState(null);
const [seloModal, setSeloModal] = useState(null);
const [itemDetalhe, setItemDetalhe] = useState(null);
const [loadingLattes, setLoadingLattes] = useState(false);
const [pontuacaoModal, setPontuacaoModal] = useState(null);
const [showInsights, setShowInsights] = useState(false);
const cardRef = useRef(null);
@@ -1618,6 +1707,71 @@ const ConsultorCard = memo(({ consultor, highlight, selecionado, onToggleSelecio
</div>
)}
{consultor.lattes?.id_lattes && (
<div className="detail-section lattes-section">
<h4>Curriculo Lattes</h4>
<div className="lattes-content">
<a
href={consultor.lattes.url}
target="_blank"
rel="noopener noreferrer"
className="lattes-link"
>
<span className="lattes-icon">📄</span>
<span className="lattes-id">{consultor.lattes.id_lattes}</span>
<span className="lattes-external"></span>
</a>
<div className="titulacoes-resumo">
{consultor.lattes.titulacoes
?.sort((a, b) => {
const ordem = { 'Pos-Doutorado': 1, 'Doutorado': 2, 'Mestrado': 3, 'Graduacao': 4 };
return (ordem[a.grau] || 99) - (ordem[b.grau] || 99);
})
.map((t, idx) => (
<span
key={idx}
className="titulacao-badge titulacao-clicavel"
onClick={(e) => {
e.stopPropagation();
setItemDetalhe({
item: t,
tipo: 'titulacao'
});
}}
title={`${t.grau}${t.area ? ` em ${t.area}` : ''}${t.ies_nome ? ` - ${t.ies_nome}` : ''}${t.pais ? ` (${t.pais})` : ''}`}
>
{t.grau}{t.ies_sigla ? ` (${t.ies_sigla})` : ''}{t.ano ? ` - ${t.ano}` : ''}
</span>
))}
<span
className="titulacao-badge titulacao-clicavel lattes-producoes"
title="Ver producoes do Lattes"
onClick={async (e) => {
e.stopPropagation();
try {
const data = await rankingService.getLattes(consultor.id_pessoa);
if (data.encontrado) {
setItemDetalhe({
item: { ...data, filtro: null, titulo: 'Producoes Lattes' },
tipo: 'producoes_lattes'
});
} else if (data.motivo) {
alert(data.motivo);
} else {
alert('Nenhuma producao encontrada no Lattes');
}
} catch (err) {
alert('Erro ao buscar Lattes: ' + err.message);
}
}}
>
📚 Producoes
</span>
</div>
</div>
</div>
)}
{consultoria?.vinculos?.length > 0 && (
<div className="extra-details">
<h4>Vinculos de Consultoria</h4>

View File

@@ -92,6 +92,7 @@ export const rankingService = {
membros_banca: c.membros_banca || [],
idiomas: c.idiomas || [],
titulacao: c.titulacao || '',
lattes: c.lattes || null,
};
});
@@ -175,6 +176,11 @@ export const rankingService = {
const response = await api.get('/consultores/areas-avaliacao');
return response.data;
},
async getLattes(idPessoa) {
const response = await api.get(`/consultor/${idPessoa}/lattes`);
return response.data;
},
};
export default api;