File size: 15,340 Bytes
d6c9678
385f196
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
d6c9678
918e0df
 
 
 
 
 
0ad99f2
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
918e0df
d6c9678
 
 
 
 
 
 
 
c88d3e9
 
 
 
918e0df
 
c88d3e9
918e0df
c88d3e9
918e0df
 
c88d3e9
918e0df
d6c9678
 
 
 
 
0ad99f2
d6c9678
385f196
d6c9678
 
918e0df
d6c9678
 
 
 
0ad99f2
d6c9678
385f196
d6c9678
 
918e0df
d6c9678
 
 
 
0ad99f2
918e0df
d6c9678
 
 
 
0ad99f2
918e0df
0ad99f2
d6c9678
 
 
 
 
 
 
 
 
 
 
 
 
 
385f196
 
 
 
d6c9678
 
d2635d4
 
 
82ec900
 
 
 
 
 
 
 
 
 
 
 
 
 
 
d6c9678
 
 
 
 
 
2a8204e
 
d6c9678
 
 
 
 
e8db196
22106af
d6c9678
 
 
 
 
 
a577d9a
d6c9678
 
 
a577d9a
 
d6c9678
 
 
 
 
 
 
 
b0eaf10
 
 
 
0d8b6ec
 
 
 
 
 
 
 
d0f2c73
 
 
 
 
d6c9678
 
d0f2c73
d6c9678
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
21fc066
 
 
 
d6c9678
 
 
 
 
 
 
 
 
 
 
 
9a6f968
d6c9678
0ad99f2
 
d6c9678
385f196
d6c9678
 
0ad99f2
 
d6c9678
 
b0eaf10
 
 
 
 
4e2aace
b0eaf10
 
 
4e2aace
b0eaf10
2e13456
 
d6c9678
 
 
 
 
 
 
 
2a8204e
 
d6c9678
614e632
d6c9678
 
 
 
 
 
 
21fc066
 
 
 
d6c9678
 
 
 
 
 
 
 
 
 
 
9a6f968
d6c9678
385f196
 
c88d3e9
385f196
 
 
 
c88d3e9
385f196
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
d6c9678
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
const API_BASE = import.meta.env.VITE_API_BASE || 'http://localhost:8000'
const AUTH_TOKEN_STORAGE_KEY = 'mesa_auth_token'

let AUTH_TOKEN = ''
if (typeof window !== 'undefined') {
  AUTH_TOKEN = window.localStorage.getItem(AUTH_TOKEN_STORAGE_KEY) || ''
}

export function setAuthToken(token) {
  const value = String(token || '').trim()
  AUTH_TOKEN = value
  if (typeof window === 'undefined') return
  if (value) {
    window.localStorage.setItem(AUTH_TOKEN_STORAGE_KEY, value)
  } else {
    window.localStorage.removeItem(AUTH_TOKEN_STORAGE_KEY)
  }
}

export function getAuthToken() {
  return AUTH_TOKEN
}

function authHeaders(extraHeaders = {}) {
  const headers = { ...extraHeaders }
  if (AUTH_TOKEN) headers['X-Auth-Token'] = AUTH_TOKEN
  return headers
}

function emitAuthRequired(detail = 'Login obrigatorio') {
  if (typeof window === 'undefined') return
  const event = new CustomEvent('mesa:auth-required', { detail: { message: detail } })
  window.dispatchEvent(event)
}

function describeFetchFailure(cause) {
  const name = String(cause?.name || '').trim()
  const message = String(cause?.message || '').trim()

  if (name === 'AbortError') {
    return 'a requisicao foi interrompida antes de concluir'
  }
  if (typeof navigator !== 'undefined' && navigator.onLine === false) {
    return 'sem conexao com a internet'
  }
  if (/failed to fetch|networkerror|load failed/i.test(message)) {
    return 'conexao interrompida durante a transferencia ou bloqueio de rede/CORS/proxy'
  }
  return message || 'falha de rede nao identificada'
}

function buildFetchError(path, stage, cause, status = null) {
  const endpoint = String(path || '').trim() || '(endpoint desconhecido)'
  const reason = describeFetchFailure(cause)
  const statusNumber = Number(status)
  const statusLabel = Number.isFinite(statusNumber) ? ` (HTTP ${statusNumber})` : ''
  const action = stage === 'body'
    ? `Falha ao receber dados de ${endpoint}${statusLabel}`
    : `Falha ao conectar com ${endpoint}`
  const error = new Error(`${action}. Motivo: ${reason}.`)
  error.path = endpoint
  error.stage = stage
  error.reason = reason
  if (Number.isFinite(statusNumber)) {
    error.status = statusNumber
  }
  error.cause = cause
  return error
}

async function fetchWithDiagnostics(path, options = {}) {
  try {
    return await fetch(`${API_BASE}${path}`, options)
  } catch (cause) {
    throw buildFetchError(path, 'request', cause)
  }
}

async function responseBlobWithDiagnostics(response, path = '') {
  try {
    return await response.blob()
  } catch (cause) {
    throw buildFetchError(path, 'body', cause, response?.status)
  }
}

async function handleResponse(response, path = '') {
  if (!response.ok) {
    let detail = 'Erro inesperado'
    try {
      const data = await response.json()
      detail = data.detail || detail
    } catch {
      detail = response.statusText || detail
    }
    const detailMessage = typeof detail === 'string'
      ? detail
      : String(detail?.message || response.statusText || 'Erro inesperado')

    if (response.status === 401 && path !== '/api/auth/login') {
      setAuthToken('')
      emitAuthRequired(detailMessage)
    }
    const error = new Error(detailMessage)
    error.status = response.status
    error.path = path
    error.detail = detail
    throw error
  }
  return response
}

async function postJson(path, body) {
  const response = await fetchWithDiagnostics(path, {
    method: 'POST',
    headers: authHeaders({ 'Content-Type': 'application/json' }),
    body: JSON.stringify(body),
  })
  await handleResponse(response, path)
  return response.json()
}

async function postForm(path, formData) {
  const response = await fetchWithDiagnostics(path, {
    method: 'POST',
    headers: authHeaders(),
    body: formData,
  })
  await handleResponse(response, path)
  return response.json()
}

async function getJson(path) {
  const response = await fetchWithDiagnostics(path, { headers: authHeaders() })
  await handleResponse(response, path)
  return response.json()
}

async function getBlob(path) {
  const response = await fetchWithDiagnostics(path, { headers: authHeaders() })
  await handleResponse(response, path)
  return responseBlobWithDiagnostics(response, path)
}

export function downloadBlob(blob, fileName) {
  const url = URL.createObjectURL(blob)
  const a = document.createElement('a')
  a.href = url
  a.download = fileName
  document.body.appendChild(a)
  a.click()
  a.remove()
  URL.revokeObjectURL(url)
}

export const api = {
  authLogin: (usuario, matricula) => postJson('/api/auth/login', { usuario, matricula }),
  authMe: () => getJson('/api/auth/me'),
  authLogout: () => postJson('/api/auth/logout', {}),

  createSession: () => postJson('/api/sessions', {}),

  pesquisaAdminConfig: () => getJson('/api/pesquisa/admin-config'),
  pesquisaAdminConfigSalvar: (campos = {}) => postJson('/api/pesquisa/admin-config', { campos }),

  pesquisarModelos(filtros = {}) {
    const params = new URLSearchParams()
    Object.entries(filtros).forEach(([key, value]) => {
      if (value === null || value === undefined) return
      const text = String(value).trim()
      if (!text) return
      params.append(key, text)
    })
    const query = params.toString()
    return getJson(query ? `/api/pesquisa/modelos?${query}` : '/api/pesquisa/modelos')
  },
  pesquisarMapaModelos(modelosIds = []) {
    return postJson('/api/pesquisa/mapa-modelos', { modelos_ids: modelosIds })
  },

  uploadElaboracaoFile(sessionId, file) {
    const form = new FormData()
    form.append('session_id', sessionId)
    form.append('file', file)
    return postForm('/api/elaboracao/upload', form)
  },
  elaboracaoRepositorioModelos: () => getJson('/api/elaboracao/repositorio-modelos'),
  elaboracaoRepositorioCarregar: (sessionId, modeloId) => postJson('/api/elaboracao/repositorio-carregar', { session_id: sessionId, modelo_id: modeloId }),

  confirmSheet: (sessionId, sheetName) => postJson('/api/elaboracao/confirm-sheet', { session_id: sessionId, sheet_name: sheetName }),
  mapCoords: (sessionId, colLat, colLon) => postJson('/api/elaboracao/map-coords', { session_id: sessionId, col_lat: colLat, col_lon: colLon }),
  geocodificar: (sessionId, colCdlog, colNum, auto200) => postJson('/api/elaboracao/geocodificar', { session_id: sessionId, col_cdlog: colCdlog, col_num: colNum, auto_200: auto200 }),
  geocodificarCorrecoes: (sessionId, correcoes, auto200) => postJson('/api/elaboracao/geocodificar-correcoes', { session_id: sessionId, correcoes, auto_200: auto200 }),
  geocodificarReiniciar: (sessionId) => postJson('/api/elaboracao/geocodificar-reiniciar', { session_id: sessionId }),
  geocodificarExcluirCoords: (sessionId) => postJson('/api/elaboracao/geocodificar-excluir-coords', { session_id: sessionId }),

  applySelection(payload) {
    return postJson('/api/elaboracao/apply-selection', payload)
  },
  classifyElaboracaoX: (sessionId, colunasX) => postJson('/api/elaboracao/classify-x', { session_id: sessionId, colunas_x: colunasX }),

  searchTransformations: (sessionId, grauCoef, grauF, transformacoesFixas = {}, transformacaoYFixa = 'Livre') => postJson('/api/elaboracao/search-transformations', {
    session_id: sessionId,
    grau_min_coef: grauCoef,
    grau_min_f: grauF,
    transformacoes_fixas: transformacoesFixas,
    transformacao_y_fixa: transformacaoYFixa,
  }),

  adoptSuggestion: (sessionId, indice) => postJson('/api/elaboracao/adopt-suggestion', { session_id: sessionId, indice }),

  fitModel(payload) {
    return postJson('/api/elaboracao/fit-model', payload)
  },

  updateModelDispersao: (sessionId, payload) => postJson('/api/elaboracao/model-dispersao', {
    session_id: sessionId,
    ...(payload || {}),
  }),
  getDispersaoInterativo: (sessionId, alvo) => postJson('/api/elaboracao/dispersao-interativo', {
    session_id: sessionId,
    alvo,
  }),
  getDiagnosticoInterativo: (sessionId, grafico) => postJson('/api/elaboracao/diagnostico-interativo', {
    session_id: sessionId,
    grafico,
  }),
  previewTransformElab: (sessionId, transformacaoY, transformacoesX) => postJson('/api/elaboracao/transform-preview', {
    session_id: sessionId,
    transformacao_y: transformacaoY,
    transformacoes_x: transformacoesX,
  }),

  applyOutlierFilters: (sessionId, filtros) => postJson('/api/elaboracao/outliers/apply-filters', { session_id: sessionId, filtros }),
  applyOutlierFiltersRecursive: (sessionId, filtros) => postJson('/api/elaboracao/outliers/apply-filters-recursive', { session_id: sessionId, filtros }),
  restartOutlierIteration: (sessionId, outliersTexto, reincluirTexto, grauCoef, grauF) => postJson('/api/elaboracao/outliers/restart', {
    session_id: sessionId,
    outliers_texto: outliersTexto,
    reincluir_texto: reincluirTexto,
    grau_min_coef: grauCoef,
    grau_min_f: grauF,
  }),
  outlierSummary: (sessionId, outliersTexto, reincluirTexto) => postJson('/api/elaboracao/outliers/summary', {
    session_id: sessionId,
    outliers_texto: outliersTexto,
    reincluir_texto: reincluirTexto,
  }),
  clearOutlierHistory: (sessionId) => postJson('/api/elaboracao/outliers/clear-history', { session_id: sessionId }),

  evaluationFieldsElab: (sessionId) => postJson('/api/elaboracao/evaluation/fields', { session_id: sessionId }),
  evaluationCalculateElab: (sessionId, valoresX, indiceBase) => postJson('/api/elaboracao/evaluation/calculate', {
    session_id: sessionId,
    valores_x: valoresX,
    indice_base: indiceBase,
  }),
  evaluationKnnDetailsElab: (sessionId, valoresX) => postJson('/api/elaboracao/evaluation/knn-details', {
    session_id: sessionId,
    valores_x: valoresX,
  }),
  evaluationClearElab: (sessionId) => postJson('/api/elaboracao/evaluation/clear', { session_id: sessionId }),
  evaluationDeleteElab: (sessionId, indice, indiceBase) => postJson('/api/elaboracao/evaluation/delete', {
    session_id: sessionId,
    indice,
    indice_base: indiceBase,
  }),
  getAvaliadores: () => getJson('/api/elaboracao/avaliadores'),
  evaluationBaseElab: (sessionId, indiceBase) => postJson('/api/elaboracao/evaluation/base', {
    session_id: sessionId,
    indice_base: indiceBase,
  }),
  exportEvaluationElab: async (sessionId) => getBlob(`/api/elaboracao/evaluation/export?session_id=${encodeURIComponent(sessionId)}`),
  exportEquationElab: async (sessionId, mode = 'excel') => getBlob(`/api/elaboracao/equation/export?session_id=${encodeURIComponent(sessionId)}&mode=${encodeURIComponent(String(mode || 'excel'))}`),
  exportModel: async (sessionId, nomeArquivo, elaborador) => {
    const path = '/api/elaboracao/export-model'
    const response = await fetchWithDiagnostics(path, {
      method: 'POST',
      headers: authHeaders({ 'Content-Type': 'application/json' }),
      body: JSON.stringify({ session_id: sessionId, nome_arquivo: nomeArquivo, elaborador }),
    })
    await handleResponse(response, path)
    return responseBlobWithDiagnostics(response, path)
  },
  exportBase: (sessionId, filtered = true) => getBlob(`/api/elaboracao/export-base?session_id=${encodeURIComponent(sessionId)}&filtered=${String(filtered)}`),
  updateElaboracaoMap: (sessionId, variavelMapa, modoMapa = 'pontos') => postJson('/api/elaboracao/map/update', {
    session_id: sessionId,
    variavel_mapa: variavelMapa,
    modo_mapa: modoMapa,
  }),
  updateElaboracaoResiduosMap: (sessionId, variavelMapa, modoMapa = 'pontos', escalaExtremoAbs = null) => postJson('/api/elaboracao/residuos/map/update', {
    session_id: sessionId,
    variavel_mapa: variavelMapa,
    modo_mapa: modoMapa,
    escala_extremo_abs: escalaExtremoAbs,
  }),
  previewMarketDateColumn: (sessionId, colunaData) => postJson('/api/elaboracao/market-date/preview', { session_id: sessionId, coluna_data: colunaData }),
  applyMarketDateColumn: (sessionId, colunaData) => postJson('/api/elaboracao/market-date/apply', { session_id: sessionId, coluna_data: colunaData }),
  getContext: (sessionId) => getJson(`/api/elaboracao/context?session_id=${encodeURIComponent(sessionId)}`),

  uploadVisualizacaoFile(sessionId, file) {
    const form = new FormData()
    form.append('session_id', sessionId)
    form.append('file', file)
    return postForm('/api/visualizacao/upload', form)
  },
  visualizacaoRepositorioModelos: () => getJson('/api/visualizacao/repositorio-modelos'),
  visualizacaoRepositorioCarregar: (sessionId, modeloId) => postJson('/api/visualizacao/repositorio-carregar', { session_id: sessionId, modelo_id: modeloId }),
  exibirVisualizacao: (sessionId) => postJson('/api/visualizacao/exibir', { session_id: sessionId }),
  evaluationContextViz: (sessionId) => postJson('/api/visualizacao/evaluation/context', { session_id: sessionId }),
  updateVisualizacaoMap: (sessionId, variavelMapa) => postJson('/api/visualizacao/map/update', { session_id: sessionId, variavel_mapa: variavelMapa }),
  evaluationFieldsViz: (sessionId) => postJson('/api/visualizacao/evaluation/fields', { session_id: sessionId }),
  evaluationCalculateViz: (sessionId, valoresX, indiceBase) => postJson('/api/visualizacao/evaluation/calculate', {
    session_id: sessionId,
    valores_x: valoresX,
    indice_base: indiceBase,
  }),
  evaluationKnnDetailsViz: (sessionId, valoresX) => postJson('/api/visualizacao/evaluation/knn-details', {
    session_id: sessionId,
    valores_x: valoresX,
  }),
  evaluationClearViz: (sessionId) => postJson('/api/visualizacao/evaluation/clear', { session_id: sessionId }),
  evaluationDeleteViz: (sessionId, indice, indiceBase) => postJson('/api/visualizacao/evaluation/delete', {
    session_id: sessionId,
    indice,
    indice_base: indiceBase,
  }),
  evaluationBaseViz: (sessionId, indiceBase) => postJson('/api/visualizacao/evaluation/base', {
    session_id: sessionId,
    indice_base: indiceBase,
  }),
  exportEvaluationViz: (sessionId) => getBlob(`/api/visualizacao/evaluation/export?session_id=${encodeURIComponent(sessionId)}`),
  exportEquationViz: (sessionId, mode = 'excel') => getBlob(`/api/visualizacao/equation/export?session_id=${encodeURIComponent(sessionId)}&mode=${encodeURIComponent(String(mode || 'excel'))}`),
  clearVisualizacao: (sessionId) => postJson('/api/visualizacao/clear', { session_id: sessionId }),

  repositorioListar: () => getJson('/api/repositorio/modelos'),
  repositorioUpload(files = [], { confirmarSubstituicao = false } = {}) {
    const form = new FormData()
    files.forEach((file) => {
      form.append('files', file)
    })
    form.append('confirmar_substituicao', confirmarSubstituicao ? 'true' : 'false')
    return postForm('/api/repositorio/upload', form)
  },
  repositorioDelete: (modelosIds = []) => postJson('/api/repositorio/delete', { modelos_ids: modelosIds }),

  logsStatus: () => getJson('/api/logs/status'),
  logsEvents({ scope = '', usuario = '', limit = 200 } = {}) {
    const params = new URLSearchParams()
    const scopeText = String(scope || '').trim()
    const usuarioText = String(usuario || '').trim()
    const limitNumber = Number(limit)
    if (scopeText) params.set('scope', scopeText)
    if (usuarioText) params.set('usuario', usuarioText)
    if (Number.isFinite(limitNumber) && limitNumber > 0) params.set('limit', String(Math.floor(limitNumber)))
    const query = params.toString()
    return getJson(query ? `/api/logs/events?${query}` : '/api/logs/events')
  },
}