feat: Nokia/Alcatel-Lucent OEM seed + scheduler
Add 41 Nokia OEM transceiver PIDs: SFP-1G/SFP-10G/SFP-25G/QSFP+-40G/ QSFP28-100G/QSFPDD-200G+400G + DWDM ZR/ZR+ + DAC. Includes legacy 3HExxxxxxxx alternate part numbers in notes field. Scheduler: daily 04:45.
This commit is contained in:
parent
51cee266f5
commit
b4c8b9b625
@ -222,6 +222,7 @@ export async function registerSchedules(boss: PgBoss): Promise<void> {
|
|||||||
await boss.schedule("scrape:catalog:arista-oem", "0 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:arista-oem", "0 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:juniper-oem", "15 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:juniper-oem", "15 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:hpe-aruba-oem", "30 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:hpe-aruba-oem", "30 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:nokia-oem", "45 4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
|
||||||
// ══════════════════════════════════════════════════════════════════════
|
// ══════════════════════════════════════════════════════════════════════
|
||||||
// VENDOR LISTS — every 12h
|
// VENDOR LISTS — every 12h
|
||||||
@ -505,6 +506,12 @@ export async function registerWorkers(boss: PgBoss): Promise<void> {
|
|||||||
await scrapeHpeArubaOem();
|
await scrapeHpeArubaOem();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:nokia-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Nokia OEM catalog seed`);
|
||||||
|
const { scrapeNokiaOem } = await import("./scrapers/nokia-oem");
|
||||||
|
await scrapeNokiaOem();
|
||||||
|
});
|
||||||
|
|
||||||
// ── Vendor lists ──────────────────────────────────────────────────────
|
// ── Vendor lists ──────────────────────────────────────────────────────
|
||||||
|
|
||||||
await boss.work("scrape:vendors:flexoptix", async () => {
|
await boss.work("scrape:vendors:flexoptix", async () => {
|
||||||
|
|||||||
153
packages/scraper/src/scrapers/nokia-oem.ts
Normal file
153
packages/scraper/src/scrapers/nokia-oem.ts
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
/**
|
||||||
|
* Nokia / Alcatel-Lucent OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Seeds Nokia-branded transceiver PIDs into the transceivers table.
|
||||||
|
* Nokia uses both a legacy Alcatel-Lucent naming scheme (SFP-1G-SX-C)
|
||||||
|
* and 3HExxxxxx part numbers. This seed covers the common public PIDs.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Nokia Transceiver Compatibility Guide (public, nokia.com)
|
||||||
|
* - Nokia 7750 SR / 7210 SAS / FP5 platform hardware guides
|
||||||
|
* - Alcatel-Lucent legacy SFP catalog
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/nokia-oem.ts
|
||||||
|
* Cron: daily at 04:45
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface NokiaPID {
|
||||||
|
pid: string;
|
||||||
|
alias?: string; // alternate part number (3HExxxxxxxx)
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const NOKIA_PIDS: NokiaPID[] = [
|
||||||
|
// ── 1G SFP ──────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "SFP-1G-SX-C", alias: "3HE00027BA", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX" },
|
||||||
|
{ pid: "SFP-1G-LX-C", alias: "3HE00028BA", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX" },
|
||||||
|
{ pid: "SFP-1G-LH-C", alias: "3HE00036BA", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 70000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-LH" },
|
||||||
|
{ pid: "SFP-1G-ZX-C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX" },
|
||||||
|
{ pid: "SFP-1G-T-C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T" },
|
||||||
|
|
||||||
|
// ── 10G SFP+ ────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "SFP-10G-SR-C", alias: "3HE04823AA", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR" },
|
||||||
|
{ pid: "SFP-10G-LR-C", alias: "3HE04824AA", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR" },
|
||||||
|
{ pid: "SFP-10G-ER-C", alias: "3HE04826AA", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
|
||||||
|
{ pid: "SFP-10G-ZR-C", alias: "3HE04827AA", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR" },
|
||||||
|
{ pid: "SFP-10G-LRM-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 220, reachLabel: "LRM", fiberType: "MMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LRM" },
|
||||||
|
{ pid: "SFP-10G-T-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 30, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "10GBASE-T" },
|
||||||
|
|
||||||
|
// ── 10G DWDM SFP+ (common in 7750 SR) ───────────────────────────────────
|
||||||
|
{ pid: "SFP-10G-DWDM-ZR-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "DWDM-ZR",fiberType:"SMF",connector: "LC", wavelengths: "C-band", notes: "DWDM tunable ZR" },
|
||||||
|
{ pid: "SFP-10G-DWDM-ER-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "DWDM-ER",fiberType:"SMF",connector: "LC", wavelengths: "C-band", notes: "DWDM fixed-channel ER" },
|
||||||
|
|
||||||
|
// ── 25G SFP28 ───────────────────────────────────────────────────────────
|
||||||
|
{ pid: "SFP-25G-SR-C", alias: "3HE12170AA", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR" },
|
||||||
|
{ pid: "SFP-25G-LR-C", alias: "3HE12171AA", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR" },
|
||||||
|
{ pid: "SFP-25G-ER-C", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 30000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1310nm" },
|
||||||
|
|
||||||
|
// ── 40G QSFP+ ───────────────────────────────────────────────────────────
|
||||||
|
{ pid: "QSFP+-40G-SR4-C", alias: "3HE07690AA", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4" },
|
||||||
|
{ pid: "QSFP+-40G-LR4-C", alias: "3HE07692AA", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4" },
|
||||||
|
{ pid: "QSFP+-40G-ER4-C", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 40000, reachLabel: "ER4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-ER4" },
|
||||||
|
{ pid: "QSFP+-40G-UNIV-C", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "UNIV", fiberType: "MMF", connector: "LC", wavelengths: "850nm", notes: "Universal SR4 2-fiber" },
|
||||||
|
|
||||||
|
// ── 100G QSFP28 ─────────────────────────────────────────────────────────
|
||||||
|
{ pid: "QSFP28-100G-SR4-C", alias: "3HE12138AA", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
|
||||||
|
{ pid: "QSFP28-100G-LR4-C", alias: "3HE12139AA", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
|
||||||
|
{ pid: "QSFP28-100G-ER4-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 30000, reachLabel: "ER4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm" },
|
||||||
|
{ pid: "QSFP28-100G-CWDM4-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4",fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
|
||||||
|
{ pid: "QSFP28-100G-DR-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
|
||||||
|
{ pid: "QSFP28-100G-FR-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "FR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-FR" },
|
||||||
|
{ pid: "QSFP28-100G-ZR4-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "ZR4", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Coherent 100G ZR4 DWDM" },
|
||||||
|
{ pid: "QSFP28-100G-LR4L-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "LR4L", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", notes: "100G LR4 Lite (Nokia)" },
|
||||||
|
|
||||||
|
// ── 200G QSFP-DD ────────────────────────────────────────────────────────
|
||||||
|
{ pid: "QSFPDD-200G-SR4-C", formFactor: "QSFP-DD",speedGbps: 200,speed: "200G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm" },
|
||||||
|
{ pid: "QSFPDD-200G-FR4-C", formFactor: "QSFP-DD",speedGbps: 200,speed: "200G", reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm" },
|
||||||
|
|
||||||
|
// ── 400G QSFP-DD ────────────────────────────────────────────────────────
|
||||||
|
{ pid: "QSFPDD-400G-SR8-C", alias: "3HE15771AA", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 100, reachLabel: "SR8", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "400GBASE-SR8" },
|
||||||
|
{ pid: "QSFPDD-400G-DR4-C", alias: "3HE15772AA", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
|
||||||
|
{ pid: "QSFPDD-400G-FR4-C", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "400GBASE-FR4" },
|
||||||
|
{ pid: "QSFPDD-400G-LR4-C", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "400GBASE-LR4" },
|
||||||
|
{ pid: "QSFPDD-400G-ZR-C", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "400ZR" },
|
||||||
|
{ pid: "QSFPDD-400G-ZRP-C", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G", reachMeters: 120000,reachLabel: "ZR+", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "OpenZR+" },
|
||||||
|
|
||||||
|
// ── DAC ─────────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "SFP-10G-DAC-1M-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC-1M",fiberType:"DAC", connector: "SFP+" },
|
||||||
|
{ pid: "SFP-10G-DAC-3M-C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC-3M",fiberType:"DAC", connector: "SFP+" },
|
||||||
|
{ pid: "QSFP+-40G-DAC-3M-C", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 3, reachLabel: "DAC-3M",fiberType:"DAC", connector: "QSFP+" },
|
||||||
|
{ pid: "QSFP28-100G-DAC-1M-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 1, reachLabel: "DAC-1M",fiberType:"DAC", connector: "QSFP28" },
|
||||||
|
{ pid: "QSFP28-100G-DAC-3M-C", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 3, reachLabel: "DAC-3M",fiberType:"DAC", connector: "QSFP28" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeNokiaOem(): Promise<void> {
|
||||||
|
console.log("=== Nokia / Alcatel-Lucent OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Nokia",
|
||||||
|
"oem",
|
||||||
|
"https://www.nokia.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0;
|
||||||
|
let updated = 0;
|
||||||
|
let errors = 0;
|
||||||
|
|
||||||
|
for (const p of NOKIA_PIDS) {
|
||||||
|
const slug = `nokia-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
|
||||||
|
// Store alias in notes if present
|
||||||
|
const notes = [p.notes, p.alias ? `Alt PN: ${p.alias}` : null].filter(Boolean).join(" | ") || null;
|
||||||
|
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, notes]
|
||||||
|
);
|
||||||
|
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=== Nokia OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}`);
|
||||||
|
console.log(` Updated: ${updated}`);
|
||||||
|
console.log(` Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${NOKIA_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeNokiaOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => {
|
||||||
|
console.error("Fatal:", err);
|
||||||
|
pool.end();
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user