feat(scraper): batch 36 OEM seeds — EnGenius, Palo Alto Networks, Brocade, Foundry Networks
Added 4 new OEM transceiver catalog seed scrapers (72 PIDs total): - engenius-oem.ts: 18 PIDs — ECS switch series 1G–100G SFP/SFP+/SFP28/QSFP28 + DAC/AOC - paloalto-networks-oem.ts: 18 PIDs — PA-3200/5200/7000/5450 NGFW SFP/SFP+/SFP28/QSFP28 + DAC - brocade-legacy-oem.ts: 18 PIDs — ICX/FCX/VDX/MLX E1MG/10G-SFPP family, market_status=Legacy - foundry-networks-oem.ts: 18 PIDs — FastIron/NetIron FDR- series incl. XFP, market_status=Legacy Scheduler: wired at 04:30/04:45/05:00/05:15 UTC. All 72 PIDs seeded clean.
This commit is contained in:
parent
e9b8cb95db
commit
e684d3d1c3
@ -420,6 +420,12 @@ export async function registerSchedules(boss: PgBoss): Promise<void> {
|
|||||||
await boss.schedule("scrape:catalog:emcore-oem", "0 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:emcore-oem", "0 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:reflex-photonics-oem", "15 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:reflex-photonics-oem", "15 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
|
||||||
|
// ── Batch 36: EnGenius, Palo Alto Networks, Brocade (Legacy), Foundry ─
|
||||||
|
await boss.schedule("scrape:catalog:engenius-oem", "30 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:paloalto-networks-oem", "45 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:brocade-legacy-oem", "0 5 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:foundry-networks-oem", "15 5 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
|
||||||
// ══════════════════════════════════════════════════════════════════════
|
// ══════════════════════════════════════════════════════════════════════
|
||||||
// VENDOR LISTS — every 12h
|
// VENDOR LISTS — every 12h
|
||||||
// ══════════════════════════════════════════════════════════════════════
|
// ══════════════════════════════════════════════════════════════════════
|
||||||
@ -1852,6 +1858,31 @@ export async function registerWorkers(boss: PgBoss): Promise<void> {
|
|||||||
await scrapeReflexPhotonicsOem();
|
await scrapeReflexPhotonicsOem();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// ── Batch 36 workers ─────────────────────────────────────────────────
|
||||||
|
await boss.work("scrape:catalog:engenius-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: EnGenius OEM catalog seed`);
|
||||||
|
const { scrapeEngeniusOem } = await import("./scrapers/engenius-oem");
|
||||||
|
await scrapeEngeniusOem();
|
||||||
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:paloalto-networks-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Palo Alto Networks OEM catalog seed`);
|
||||||
|
const { scrapePaloaltoNetworksOem } = await import("./scrapers/paloalto-networks-oem");
|
||||||
|
await scrapePaloaltoNetworksOem();
|
||||||
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:brocade-legacy-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Brocade (Legacy) OEM catalog seed`);
|
||||||
|
const { scrapeBrocadeLegacyOem } = await import("./scrapers/brocade-legacy-oem");
|
||||||
|
await scrapeBrocadeLegacyOem();
|
||||||
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:foundry-networks-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Foundry Networks (Legacy) OEM catalog seed`);
|
||||||
|
const { scrapeFoundryNetworksOem } = await import("./scrapers/foundry-networks-oem");
|
||||||
|
await scrapeFoundryNetworksOem();
|
||||||
|
});
|
||||||
|
|
||||||
// ── Vendor lists ──────────────────────────────────────────────────────
|
// ── Vendor lists ──────────────────────────────────────────────────────
|
||||||
|
|
||||||
await boss.work("scrape:vendors:flexoptix", async () => {
|
await boss.work("scrape:vendors:flexoptix", async () => {
|
||||||
|
|||||||
108
packages/scraper/src/scrapers/brocade-legacy-oem.ts
Normal file
108
packages/scraper/src/scrapers/brocade-legacy-oem.ts
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/**
|
||||||
|
* Brocade (Legacy) OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Brocade Communications (acquired by Broadcom/Extreme Networks in 2017) made
|
||||||
|
* FC SAN and Ethernet switches. Their ICX/FCX/VDX/MLX series are still widely
|
||||||
|
* deployed. Brocade-branded SFPs remain in service and appear frequently in
|
||||||
|
* legacy BoMs and upgrade requests.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Brocade ICX/FCX switch transceiver catalog
|
||||||
|
* - Brocade MLX/XMR router SFP/QSFP compatibility guide
|
||||||
|
* - Brocade VDX/FCoE SFP+ catalog
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/brocade-legacy-oem.ts
|
||||||
|
* Cron: daily at 05:00
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface BrocadePID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const BROCADE_PIDS: BrocadePID[] = [
|
||||||
|
// ── SFP 1G — ICX/FCX GigE ────────────────────────────────────────────
|
||||||
|
{ pid: "E1MG-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "Brocade 1G SFP SX 550m MM — ICX/FCX series" },
|
||||||
|
{ pid: "E1MG-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "Brocade 1G SFP LX 10km SM" },
|
||||||
|
{ pid: "E1MG-EX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "EX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-EX", notes: "Brocade 1G SFP EX 40km SM" },
|
||||||
|
{ pid: "E1MG-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "Brocade 1G SFP ZX 80km SM" },
|
||||||
|
{ pid: "E1MG-TX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "CU", connector: "RJ-45", standard: "1000BASE-T", notes: "Brocade 1G copper SFP RJ45" },
|
||||||
|
{ pid: "E1MG-LHA", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 70000, reachLabel: "LHA", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", notes: "Brocade 1G SFP LH 70km SM" },
|
||||||
|
// ── SFP 1G BiDi — ICX uplink point-to-point ──────────────────────────
|
||||||
|
{ pid: "E1MG-BXU-A", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1310nm TX / 1490nm RX", notes: "Brocade 1G BiDi SFP BX-U 1310/1490" },
|
||||||
|
{ pid: "E1MG-BXD-A", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1490nm TX / 1310nm RX", notes: "Brocade 1G BiDi SFP BX-D 1490/1310" },
|
||||||
|
// ── SFP+ 10G — ICX-7750/FCX/VDX ─────────────────────────────────────
|
||||||
|
{ pid: "10G-SFPP-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Brocade 10G SFP+ SR 300m" },
|
||||||
|
{ pid: "10G-SFPP-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Brocade 10G SFP+ LR 10km" },
|
||||||
|
{ pid: "10G-SFPP-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Brocade 10G SFP+ ER 40km" },
|
||||||
|
{ pid: "10G-SFPP-ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Brocade 10G SFP+ ZR 80km" },
|
||||||
|
{ pid: "10G-SFPP-BXU", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1270nm TX / 1330nm RX", notes: "Brocade 10G BiDi SFP+ BX-U" },
|
||||||
|
{ pid: "10G-SFPP-BXD", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1330nm TX / 1270nm RX", notes: "Brocade 10G BiDi SFP+ BX-D" },
|
||||||
|
// ── QSFP+ 40G — MLX/XMR router uplinks ──────────────────────────────
|
||||||
|
{ pid: "40G-QSFP-SR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4", notes: "Brocade 40G QSFP+ SR4 150m MPO" },
|
||||||
|
{ pid: "40G-QSFP-LR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000,reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4", notes: "Brocade 40G QSFP+ LR4 10km LC" },
|
||||||
|
// ── DAC ───────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "10G-SFPP-TWX-0101",formFactor:"SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "Brocade 10G SFP+ twinax DAC 1m" },
|
||||||
|
{ pid: "10G-SFPP-TWX-0301",formFactor:"SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "Brocade 10G SFP+ twinax DAC 3m" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeBrocadeLegacyOem(): Promise<void> {
|
||||||
|
console.log("=== Brocade (Legacy) OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Brocade",
|
||||||
|
"oem",
|
||||||
|
"https://www.broadcom.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of BROCADE_PIDS) {
|
||||||
|
const slug = `brocade-legacy-${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,'Legacy','DataCenter',$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=== Brocade (Legacy) OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${BROCADE_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeBrocadeLegacyOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
108
packages/scraper/src/scrapers/engenius-oem.ts
Normal file
108
packages/scraper/src/scrapers/engenius-oem.ts
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/**
|
||||||
|
* EnGenius OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* EnGenius (brand of Senao International) focuses on enterprise switching,
|
||||||
|
* wireless APs, and outdoor CPE for SMB/WISP deployments. Their ECS managed
|
||||||
|
* switch family (ECS1552FP, ECS4100, ECS5512-54P) supports SFP/SFP+/QSFP28.
|
||||||
|
* EnGenius-branded SFPs are sold separately from Senao OEM modules.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - EnGenius ECS switch SFP compatibility matrix
|
||||||
|
* - EnGenius Smart/Cloud switch accessory guide 2023
|
||||||
|
* - EnGenius SFP/SFP+ module datasheet
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/engenius-oem.ts
|
||||||
|
* Cron: daily at 04:30
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface EngeniusPID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ENGENIUS_PIDS: EngeniusPID[] = [
|
||||||
|
// ── SFP 1G — ECS1552FP/ECS4100 GigE ports ────────────────────────────
|
||||||
|
{ pid: "ECS-SFP-1G-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "EnGenius 1G SFP 850nm 550m MM for ECS switches" },
|
||||||
|
{ pid: "ECS-SFP-1G-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "EnGenius 1G SFP 1310nm 10km SM" },
|
||||||
|
{ pid: "ECS-SFP-1G-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "EnGenius 1G SFP 1550nm 80km SM" },
|
||||||
|
{ pid: "ECS-SFP-1G-T", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "CU", connector: "RJ-45", standard: "1000BASE-T", notes: "EnGenius 1G copper SFP RJ45" },
|
||||||
|
{ pid: "ECS-SFP-1G-BXU20", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 20000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1310nm TX / 1550nm RX", notes: "EnGenius 1G BiDi SFP Tx1310 upstream 20km" },
|
||||||
|
{ pid: "ECS-SFP-1G-BXD20", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 20000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1550nm TX / 1310nm RX", notes: "EnGenius 1G BiDi SFP Tx1550 downstream 20km" },
|
||||||
|
// ── SFP+ 10G — ECS4100-28T/ECS5512 uplinks ───────────────────────────
|
||||||
|
{ pid: "ECS-SFP-10G-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "EnGenius 10G SFP+ 850nm 300m" },
|
||||||
|
{ pid: "ECS-SFP-10G-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "EnGenius 10G SFP+ 1310nm 10km" },
|
||||||
|
{ pid: "ECS-SFP-10G-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "EnGenius 10G SFP+ 1550nm 40km" },
|
||||||
|
{ pid: "ECS-SFP-10G-BXU20",formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 20000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1270nm TX / 1330nm RX", notes: "EnGenius 10G BiDi SFP+ Tx1270 20km" },
|
||||||
|
{ pid: "ECS-SFP-10G-BXD20",formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 20000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1330nm TX / 1270nm RX", notes: "EnGenius 10G BiDi SFP+ Tx1330 20km" },
|
||||||
|
// ── SFP28 25G — ECS5512-54P 25G ports ────────────────────────────────
|
||||||
|
{ pid: "ECS-SFP28-25G-SR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "EnGenius 25G SFP28 850nm 100m" },
|
||||||
|
{ pid: "ECS-SFP28-25G-LR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "EnGenius 25G SFP28 1310nm 10km" },
|
||||||
|
// ── QSFP28 100G — ECS5512-54P core uplinks ───────────────────────────
|
||||||
|
{ pid: "ECS-QSFP28-100G-SR4",formFactor:"QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4",notes: "EnGenius 100G QSFP28 SR4 MPO 100m" },
|
||||||
|
{ pid: "ECS-QSFP28-100G-LR4",formFactor:"QSFP28",speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-LR4",notes: "EnGenius 100G QSFP28 LR4 LC 10km" },
|
||||||
|
// ── DAC / AOC ─────────────────────────────────────────────────────────
|
||||||
|
{ pid: "ECS-DAC-10G-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "EnGenius 10G SFP+ DAC twinax 1m" },
|
||||||
|
{ pid: "ECS-DAC-10G-3M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "EnGenius 10G SFP+ DAC twinax 3m" },
|
||||||
|
{ pid: "ECS-AOC-10G-10M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10, reachLabel: "AOC", fiberType: "MMF", connector: "LC", wavelengths: "850nm", notes: "EnGenius 10G SFP+ AOC active optical 10m" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeEngeniusOem(): Promise<void> {
|
||||||
|
console.log("=== EnGenius OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"EnGenius",
|
||||||
|
"oem",
|
||||||
|
"https://www.engeniusnetworks.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of ENGENIUS_PIDS) {
|
||||||
|
const slug = `engenius-${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','DataCenter',$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=== EnGenius OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${ENGENIUS_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeEngeniusOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
106
packages/scraper/src/scrapers/foundry-networks-oem.ts
Normal file
106
packages/scraper/src/scrapers/foundry-networks-oem.ts
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
/**
|
||||||
|
* Foundry Networks (Legacy) OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Foundry Networks (acquired by Brocade in 2008) made FastIron and NetIron
|
||||||
|
* switches/routers. Their SFP/SFP+ PIDs are referenced in enterprise upgrades
|
||||||
|
* and legacy BoMs. The E1MG/10G-SFPP family carries over from Foundry heritage.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Foundry FastIron switch hardware installation guide
|
||||||
|
* - Foundry NetIron CES/CER router SFP catalog
|
||||||
|
* - Brocade/Foundry compatibility matrices for SFP modules
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/foundry-networks-oem.ts
|
||||||
|
* Cron: daily at 05:15
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface FoundryPID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FOUNDRY_PIDS: FoundryPID[] = [
|
||||||
|
// ── 100M/1G SFP — FastIron GigE ──────────────────────────────────────
|
||||||
|
{ pid: "FDR-SFP-FX", formFactor: "SFP", speedGbps: 0.1,speed: "100M", reachMeters: 2000, reachLabel: "FX", fiberType: "MMF", connector: "LC", wavelengths: "1310nm", standard: "100BASE-FX", notes: "Foundry 100M SFP FX 2km MM" },
|
||||||
|
{ pid: "FDR-SFP-LX10", formFactor: "SFP", speedGbps: 0.1,speed: "100M", reachMeters: 10000, reachLabel: "LX10", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100BASE-LX10",notes: "Foundry 100M SFP LX10 10km SM" },
|
||||||
|
{ pid: "FDR-E1MG-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "Foundry 1G SFP SX 550m — FastIron edge" },
|
||||||
|
{ pid: "FDR-E1MG-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "Foundry 1G SFP LX 10km SM" },
|
||||||
|
{ pid: "FDR-E1MG-EX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "EX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-EX", notes: "Foundry 1G SFP EX 40km SM" },
|
||||||
|
{ pid: "FDR-E1MG-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "Foundry 1G SFP ZX 80km SM" },
|
||||||
|
{ pid: "FDR-E1MG-TX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "CU", connector: "RJ-45", standard: "1000BASE-T", notes: "Foundry 1G copper SFP RJ45" },
|
||||||
|
// ── SFP+ 10G — NetIron CES/CER and FastIron SX/TurboIron 24X ─────────
|
||||||
|
{ pid: "FDR-10G-SFPP-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Foundry 10G SFP+ SR 300m MM" },
|
||||||
|
{ pid: "FDR-10G-SFPP-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Foundry 10G SFP+ LR 10km SM" },
|
||||||
|
{ pid: "FDR-10G-SFPP-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Foundry 10G SFP+ ER 40km SM" },
|
||||||
|
{ pid: "FDR-10G-SFPP-ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Foundry 10G SFP+ ZR 80km SM" },
|
||||||
|
// ── XFP 10G — NetIron XMR/MLX ────────────────────────────────────────
|
||||||
|
{ pid: "FDR-XFP-10G-SR", formFactor: "XFP", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Foundry 10G XFP SR 300m for NetIron" },
|
||||||
|
{ pid: "FDR-XFP-10G-LR", formFactor: "XFP", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Foundry 10G XFP LR 10km SM" },
|
||||||
|
{ pid: "FDR-XFP-10G-ER", formFactor: "XFP", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Foundry 10G XFP ER 40km SM" },
|
||||||
|
{ pid: "FDR-XFP-10G-ZR", formFactor: "XFP", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Foundry 10G XFP ZR 80km SM" },
|
||||||
|
// ── DAC ───────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "FDR-DAC-SFP-1G-1M", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "Foundry 1G SFP DAC twinax 1m" },
|
||||||
|
{ pid: "FDR-DAC-10G-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "Foundry 10G SFP+ DAC twinax 1m" },
|
||||||
|
{ pid: "FDR-DAC-10G-3M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "Foundry 10G SFP+ DAC twinax 3m" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeFoundryNetworksOem(): Promise<void> {
|
||||||
|
console.log("=== Foundry Networks (Legacy) OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Foundry Networks",
|
||||||
|
"oem",
|
||||||
|
"https://www.broadcom.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of FOUNDRY_PIDS) {
|
||||||
|
const slug = `foundry-networks-${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,'Legacy','DataCenter',$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=== Foundry Networks (Legacy) OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${FOUNDRY_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeFoundryNetworksOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
108
packages/scraper/src/scrapers/paloalto-networks-oem.ts
Normal file
108
packages/scraper/src/scrapers/paloalto-networks-oem.ts
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/**
|
||||||
|
* Palo Alto Networks OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Palo Alto Networks next-gen firewalls (PA-series) and switches include
|
||||||
|
* SFP/SFP+/SFP28/QSFP28 interfaces. PAN-branded transceiver accessories
|
||||||
|
* are required for TAC support on PA-3200, PA-5200, PA-7000 series.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Palo Alto Networks PA-series hardware reference guides
|
||||||
|
* - PAN transceiver compatibility matrix (doc: PAN-OS 11.x)
|
||||||
|
* - Palo Alto Networks SFP accessory catalog
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/paloalto-networks-oem.ts
|
||||||
|
* Cron: daily at 04:45
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface PanPID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PAN_PIDS: PanPID[] = [
|
||||||
|
// ── SFP 1G — PA-200/PA-500/PA-800 management ports ────────────────────
|
||||||
|
{ pid: "PAN-SFP-1G-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "PAN 1G SFP 850nm 550m MM for PA-series firewalls" },
|
||||||
|
{ pid: "PAN-SFP-1G-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "PAN 1G SFP 1310nm 10km SM" },
|
||||||
|
{ pid: "PAN-SFP-1G-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "PAN 1G SFP 1550nm 80km SM" },
|
||||||
|
{ pid: "PAN-SFP-1G-T", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "CU", connector: "RJ-45", standard: "1000BASE-T", notes: "PAN 1G copper SFP RJ45" },
|
||||||
|
// ── SFP+ 10G — PA-3200/PA-5200 data plane interfaces ─────────────────
|
||||||
|
{ pid: "PAN-SFP-PLUS-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "PAN 10G SFP+ 850nm 300m for PA-3200/5200" },
|
||||||
|
{ pid: "PAN-SFP-PLUS-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "PAN 10G SFP+ 1310nm 10km SM" },
|
||||||
|
{ pid: "PAN-SFP-PLUS-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "PAN 10G SFP+ 1550nm 40km SM" },
|
||||||
|
{ pid: "PAN-SFP-PLUS-BXU", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1270nm TX / 1330nm RX", notes: "PAN 10G BiDi SFP+ Tx1270 upstream" },
|
||||||
|
{ pid: "PAN-SFP-PLUS-BXD", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1330nm TX / 1270nm RX", notes: "PAN 10G BiDi SFP+ Tx1330 downstream" },
|
||||||
|
// ── SFP28 25G — PA-5450 and PA-7000 line cards ────────────────────────
|
||||||
|
{ pid: "PAN-SFP28-25G-SR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "PAN 25G SFP28 for PA-5450/PA-7000" },
|
||||||
|
{ pid: "PAN-SFP28-25G-LR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "PAN 25G SFP28 1310nm 10km SM" },
|
||||||
|
// ── QSFP+ 40G — PA-5200/PA-7000 high-throughput ──────────────────────
|
||||||
|
{ pid: "PAN-QSFP-40G-SR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4", notes: "PAN 40G QSFP+ SR4 MPO 150m" },
|
||||||
|
{ pid: "PAN-QSFP-40G-LR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4", notes: "PAN 40G QSFP+ LR4 LC 10km" },
|
||||||
|
// ── QSFP28 100G — PA-7000/PA-5450 100G interfaces ────────────────────
|
||||||
|
{ pid: "PAN-QSFP28-100G-SR4", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4",notes: "PAN 100G QSFP28 SR4 MPO for PA-7000" },
|
||||||
|
{ pid: "PAN-QSFP28-100G-LR4", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-LR4",notes: "PAN 100G QSFP28 LR4 LC 10km" },
|
||||||
|
// ── DAC — rack-internal NGFW to switch wiring ─────────────────────────
|
||||||
|
{ pid: "PAN-DAC-SFP-PLUS-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "PAN 10G SFP+ DAC 1m passive" },
|
||||||
|
{ pid: "PAN-DAC-SFP-PLUS-3M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "PAN 10G SFP+ DAC 3m passive" },
|
||||||
|
{ pid: "PAN-DAC-QSFP28-100G-1M",formFactor:"QSFP28",speedGbps:100, speed: "100G", reachMeters: 1, reachLabel: "DAC", fiberType: "DAC", connector: "DAC", notes: "PAN 100G QSFP28 DAC 1m passive" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapePaloaltoNetworksOem(): Promise<void> {
|
||||||
|
console.log("=== Palo Alto Networks OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Palo Alto Networks",
|
||||||
|
"oem",
|
||||||
|
"https://www.paloaltonetworks.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of PAN_PIDS) {
|
||||||
|
const slug = `paloalto-networks-${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','DataCenter',$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=== Palo Alto Networks OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${PAN_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapePaloaltoNetworksOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user