feat(address): collecteur BAN + intégration enrich/address + PDF

- Géocodage via api-adresse.data.gouv.fr (BAN)
- Endpoint enrich/address: collecte réelle + génération PDF
- Intégration UI existante (Collecter/Voir PDF/JSON)
This commit is contained in:
4NK IA 2025-09-18 16:09:28 +00:00
parent 8e3daad446
commit 0c9d01404f
2 changed files with 88 additions and 18 deletions

View File

@ -0,0 +1,61 @@
/**
* Collecteur Adresses
* Géocodage via BAN (api-adresse.data.gouv.fr) et scaffold pour risques
*/
const fetch = require('node-fetch')
const USER_AGENT = '4NK-IA-Front/1.0 (Document Analysis Tool)'
const REQUEST_TIMEOUT_MS = 12000
async function geocodeBAN(address) {
const q = [address.street, address.postalCode, address.city, address.country]
.filter(Boolean)
.join(' ')
const url = `https://api-adresse.data.gouv.fr/search/?q=${encodeURIComponent(q)}&limit=1`
const start = Date.now()
try {
const res = await fetch(url, {
headers: { 'User-Agent': USER_AGENT, 'Accept': 'application/json' },
timeout: REQUEST_TIMEOUT_MS,
})
if (!res.ok) throw new Error(`HTTP ${res.status}`)
const json = await res.json()
const f = json.features && json.features[0]
if (!f) {
return { success: true, duration: Date.now() - start, score: 0, lat: null, lon: null, label: null }
}
return {
success: true,
duration: Date.now() - start,
score: f.properties && typeof f.properties.score === 'number' ? f.properties.score : null,
lat: f.geometry?.coordinates ? f.geometry.coordinates[1] : null,
lon: f.geometry?.coordinates ? f.geometry.coordinates[0] : null,
label: f.properties?.label || null,
}
} catch (e) {
return { success: false, duration: Date.now() - start, error: e.message }
}
}
async function collectAddressData(address) {
// Étape 1: Géocodage BAN
const geocode = await geocodeBAN(address)
// Étape 2: Risques (placeholder, à compléter avec GéoRisque/GéoFoncier)
const risks = []
return {
success: geocode.success,
geocode,
risks,
timestamp: new Date().toISOString(),
sources: ['ban']
}
}
module.exports = {
collectAddressData,
}

View File

@ -20,6 +20,7 @@ const pdf = require('pdf-parse')
const { searchBodaccGelAvoirs, generateBodaccSummary } = require('./collectors/bodaccCollector')
const { searchCompanyInfo, generateCompanySummary } = require('./collectors/inforgreffeCollector')
const { generatePersonPdf, generateCompanyPdf, generateAddressPdf } = require('./collectors/pdfGenerator')
const { collectAddressData } = require('./collectors/addressCollector')
const app = express()
const PORT = process.env.PORT || 3001
@ -2617,16 +2618,16 @@ app.post('/api/folders/:folderHash/files/:fileHash/enrich/:kind', async (req, re
if (fs.existsSync(cacheFile)) {
const cacheData = JSON.parse(fs.readFileSync(cacheFile, 'utf8'))
const persons = cacheData.entities?.persons || []
for (const person of persons) {
if (person.lastName) {
console.log(`[Enrich] Recherche Bodacc pour: ${person.firstName} ${person.lastName}`)
result = await searchBodaccGelAvoirs(person.lastName, person.firstName)
if (result.success) {
const summary = generateBodaccSummary(result.results, person.lastName, person.firstName)
result.summary = summary
// Génération du PDF
try {
await generatePersonPdf(person, result, pdfPath)
@ -2645,12 +2646,12 @@ app.post('/api/folders/:folderHash/files/:fileHash/enrich/:kind', async (req, re
if (fs.existsSync(cacheFile)) {
const cacheData = JSON.parse(fs.readFileSync(cacheFile, 'utf8'))
const companies = cacheData.entities?.companies || []
for (const company of companies) {
if (company.name) {
console.log(`[Enrich] Recherche Inforgreffe pour: ${company.name}`)
result = await searchCompanyInfo(company.name, company.siren)
if (result.success) {
// Génération du PDF
try {
@ -2665,17 +2666,25 @@ app.post('/api/folders/:folderHash/files/:fileHash/enrich/:kind', async (req, re
}
}
} else if (kind === 'address') {
// Placeholder pour les adresses (géocodage à implémenter)
result = {
success: true,
duration: 1000,
message: 'Enrichissement adresse en cours de développement',
data: { placeholder: true }
// Géocodage réel via BAN
const cacheFile = path.join(cachePath, `${fileHash}.json`)
let addressData = { street: '', city: '', postalCode: '', country: 'France' }
if (fs.existsSync(cacheFile)) {
const cacheData = JSON.parse(fs.readFileSync(cacheFile, 'utf8'))
const addresses = cacheData.entities?.addresses || []
if (addresses.length > 0) {
addressData = {
street: addresses[0].street || '',
city: addresses[0].city || '',
postalCode: addresses[0].postalCode || '',
country: addresses[0].country || 'France',
}
}
}
// Génération du PDF placeholder
result = await collectAddressData(addressData)
// Génération du PDF avec données géocodées
try {
const addressData = { street: 'Adresse', city: 'Ville' }
await generateAddressPdf(addressData, result, pdfPath)
pdfGenerated = true
} catch (pdfError) {
@ -2690,7 +2699,7 @@ app.post('/api/folders/:folderHash/files/:fileHash/enrich/:kind', async (req, re
startedAt: status.startedAt,
finishedAt: new Date().toISOString(),
message: result?.success ? 'Collecte terminée' : 'Erreur de collecte',
sources: result?.sources ? Object.keys(result.sources) :
sources: result?.sources ? Object.keys(result.sources) :
(kind === 'person') ? ['bodacc_gel_avoirs'] :
(kind === 'company') ? ['kbis_inforgreffe', 'societe_com'] :
['cadastre', 'georisque', 'geofoncier'],
@ -2698,12 +2707,12 @@ app.post('/api/folders/:folderHash/files/:fileHash/enrich/:kind', async (req, re
pdfGenerated
}
fs.writeFileSync(statusPath, JSON.stringify(done, null, 2))
console.log(`[Enrich] Terminé pour ${kind}:`, done.state)
} catch (error) {
console.error(`[Enrich] Erreur enrichissement ${kind}:`, error.message)
const errorStatus = {
kind,
state: 'error',