/** * Phoenix Contact OEM Transceiver Catalog Seed * * Seeds transceiver PIDs for Phoenix Contact FL SFP/SFP+ industrial modules * used in FL SWITCH, FL COMSERVER, and Axioline industrial automation devices. * * Sources: * - Phoenix Contact FL SFP product datasheets (phoenixcontact.com) * - Phoenix Contact Industrial Ethernet catalog * * Run: tsx packages/scraper/src/scrapers/phoenix-contact-oem.ts * Cron: daily at 15:15 */ import { pool, ensureVendor } from "../utils/db"; interface PhoenixContactPID { pid: string; formFactor: string; speedGbps: number; speed: string; reachMeters: number; reachLabel: string; fiberType: string; connector: string; wavelengths?: string; standard?: string; notes?: string; } const PHOENIX_CONTACT_PIDS: PhoenixContactPID[] = [ // ── 1G SFP ────────────────────────────────────────────────────────────── { pid: "FL SFP 1G SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "Phoenix Contact FL SFP 1G SX industrial" }, { pid: "FL SFP 1G LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "Phoenix Contact FL SFP 1G LX industrial" }, { pid: "FL SFP 1G LH", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "Phoenix Contact FL SFP 1G LH 80km" }, { pid: "FL SFP 1G BIDI TX1310", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 20000, reachLabel: "BiDi-TX1310", fiberType: "SMF", connector: "LC", wavelengths: "TX1310/RX1550nm", notes: "Phoenix Contact FL SFP 1G BiDi TX1310 single fiber" }, { pid: "FL SFP 1G BIDI TX1550", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 20000, reachLabel: "BiDi-TX1550", fiberType: "SMF", connector: "LC", wavelengths: "TX1550/RX1310nm", notes: "Phoenix Contact FL SFP 1G BiDi TX1550 single fiber" }, { pid: "FL SFP 1G BIDI TX1490", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "BiDi-TX1490", fiberType: "SMF", connector: "LC", wavelengths: "TX1490/RX1310nm", notes: "Phoenix Contact FL SFP 1G BiDi TX1490 40km" }, // ── 10G SFP+ ──────────────────────────────────────────────────────────── { pid: "FL SFP PLUS 10G SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Phoenix Contact FL SFP+ 10G SR industrial" }, { pid: "FL SFP PLUS 10G LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Phoenix Contact FL SFP+ 10G LR industrial" }, { pid: "FL SFP PLUS 10G ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Phoenix Contact FL SFP+ 10G ER industrial" }, { pid: "FL SFP PLUS 10G ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Phoenix Contact FL SFP+ 10G ZR 80km industrial" }, // ── 1G CWDM SFP (factory/substation rings) ────────────────────────────── { pid: "FL SFP CWDM 1470", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1470", fiberType: "SMF", connector: "LC", wavelengths: "1470nm", notes: "Phoenix Contact FL SFP CWDM 1470nm 40km" }, { pid: "FL SFP CWDM 1490", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1490", fiberType: "SMF", connector: "LC", wavelengths: "1490nm", notes: "Phoenix Contact FL SFP CWDM 1490nm 40km" }, { pid: "FL SFP CWDM 1510", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1510", fiberType: "SMF", connector: "LC", wavelengths: "1510nm", notes: "Phoenix Contact FL SFP CWDM 1510nm 40km" }, { pid: "FL SFP CWDM 1530", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1530", fiberType: "SMF", connector: "LC", wavelengths: "1530nm", notes: "Phoenix Contact FL SFP CWDM 1530nm 40km" }, { pid: "FL SFP CWDM 1550", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1550", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", notes: "Phoenix Contact FL SFP CWDM 1550nm 40km" }, { pid: "FL SFP CWDM 1570", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1570", fiberType: "SMF", connector: "LC", wavelengths: "1570nm", notes: "Phoenix Contact FL SFP CWDM 1570nm 40km" }, { pid: "FL SFP CWDM 1590", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1590", fiberType: "SMF", connector: "LC", wavelengths: "1590nm", notes: "Phoenix Contact FL SFP CWDM 1590nm 40km" }, { pid: "FL SFP CWDM 1610", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "CWDM-1610", fiberType: "SMF", connector: "LC", wavelengths: "1610nm", notes: "Phoenix Contact FL SFP CWDM 1610nm 40km" }, ]; export async function scrapePhoenixContactOem(): Promise { console.log("=== Phoenix Contact OEM Transceiver Seed ===\n"); const vendorId = await ensureVendor( "Phoenix Contact", "oem", "https://www.phoenixcontact.com", undefined ); let inserted = 0; let updated = 0; let errors = 0; for (const p of PHOENIX_CONTACT_PIDS) { const slug = `phoenix-contact-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`; try { const res = await pool.query( `INSERT INTO transceivers (slug, part_number, vendor_id, form_factor, speed, speed_gbps, reach_meters, reach_label, fiber_type, connector, wavelengths, dom_support, ieee_reference, market_status, category, notes) VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9,$10,$11,true,$12,'Mainstream','Industrial',$13) ON CONFLICT (slug) DO UPDATE SET speed_gbps = EXCLUDED.speed_gbps, reach_meters = CASE WHEN EXCLUDED.reach_meters > 0 THEN EXCLUDED.reach_meters ELSE transceivers.reach_meters END, fiber_type = CASE WHEN EXCLUDED.fiber_type <> '' THEN EXCLUDED.fiber_type ELSE transceivers.fiber_type END, wavelengths = COALESCE(EXCLUDED.wavelengths, transceivers.wavelengths), updated_at = NOW() RETURNING (xmax = 0) as was_inserted`, [slug, p.pid, vendorId, p.formFactor, p.speed, p.speedGbps, p.reachMeters, p.reachLabel, p.fiberType, p.connector, p.wavelengths ?? null, p.standard ?? null, p.notes ?? null] ); if (res.rows[0]?.was_inserted) inserted++; else updated++; } catch (err) { console.warn(` Skip ${p.pid}: ${(err as Error).message.slice(0, 80)}`); errors++; } } console.log(`\n=== Phoenix Contact OEM Seed Complete ===`); console.log(` Inserted: ${inserted}`); console.log(` Updated: ${updated}`); console.log(` Errors: ${errors}`); console.log(` Total PIDs: ${PHOENIX_CONTACT_PIDS.length}\n`); } if (require.main === module) { scrapePhoenixContactOem() .then(() => pool.end()) .catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); }); }