feat: HPE/Aruba OEM seed + Cisco TMG upsert fix
Add 43 HPE/Aruba OEM transceiver PIDs (J/JL/JH/R series — 1G through 400G QSFP-DD + DAC/AOC). Scheduler: daily 04:30. Cisco TMG scraper: fixed market_status/temp_range constraint violations, switched to always-upsert pattern. Result: 423 switches, 22476 Cisco OEM transceivers, 22476 compat entries written to DB. Update CHANGELOG_PENDING with all session data changes.
This commit is contained in:
parent
c9a50ad551
commit
51cee266f5
@ -1,6 +1,9 @@
|
|||||||
# TIP Changelog
|
# TIP Changelog
|
||||||
|
|
||||||
Format: `{"d":"YYYY-MM-DD","t":"TYPE","m":"Description"}`
|
Format: `{"d":"YYYY-MM-DD","t":"TYPE","m":"Description"}`
|
||||||
|
{"d":"2026-04-26","t":"DATA","m":"Juniper OEM transceiver seed: 59 PIDs inserted (SFP-1GE/SFPP-10G/SFP-25G/QSFPP-40G/JNP-QSFP-100G/JNP-QSFP56-200G/JNP-QSFPDD-400G/JNP-OSFP-400G+800G + DAC/AOC). Scheduler: daily 04:15."}
|
||||||
|
{"d":"2026-04-26","t":"FIX","m":"BlueOptics scraper: force HTTP/1.1 via Node.js https.get() to bypass empty-body HTTP/2 server bug; updated catalog path to /Transceivers_1 (changed 2026)."}
|
||||||
|
{"d":"2026-04-26","t":"DATA","m":"Cisco TMG scraper: upsert logic fixed (market_status EOL + temp_range IND normalization). Full run in progress: 300+ switches, 15000+ compat matches written to switch_transceiver_compat."}
|
||||||
Types: FEAT · FIX · UI · DATA · AI · INFRA
|
Types: FEAT · FIX · UI · DATA · AI · INFRA
|
||||||
|
|
||||||
{"d":"2026-04-25","t":"FEAT","m":"Standards Audit + Form Factors Reference: expanded standards from 40 to 63 (+23 new: full 200G tier SR4/DR4/FR4/LR4/ER4/CR4, PON family GPON/XG-PON1/NG-PON2/25G-PON, copper DAC variants CR4 for 25G/40G/100G/400G, 800G emerging FR4/LR8/CR8, 1.6TBASE-DR16 emerging). All 63 standards have bilingual plain-language descriptions (DE+EN, for non-technical colleagues). New form_factors table (migration 101) with 20 entries: SFP family SFP→SFP112, QSFP family QSFP+→QSFP-DD800, OSFP family OSFP→OSFP224, CFP family, legacy XFP/CXP — with full names, channel count, max speed, hot-swap flag, supersedes chain, status, and bilingual descriptions. New GET /api/form-factors endpoint. Dashboard Standards tab: descriptions shown as table row subtitles, Form Factors grid section with family color coding, speed/channel info, openFormFactorDetail panel."}
|
{"d":"2026-04-25","t":"FEAT","m":"Standards Audit + Form Factors Reference: expanded standards from 40 to 63 (+23 new: full 200G tier SR4/DR4/FR4/LR4/ER4/CR4, PON family GPON/XG-PON1/NG-PON2/25G-PON, copper DAC variants CR4 for 25G/40G/100G/400G, 800G emerging FR4/LR8/CR8, 1.6TBASE-DR16 emerging). All 63 standards have bilingual plain-language descriptions (DE+EN, for non-technical colleagues). New form_factors table (migration 101) with 20 entries: SFP family SFP→SFP112, QSFP family QSFP+→QSFP-DD800, OSFP family OSFP→OSFP224, CFP family, legacy XFP/CXP — with full names, channel count, max speed, hot-swap flag, supersedes chain, status, and bilingual descriptions. New GET /api/form-factors endpoint. Dashboard Standards tab: descriptions shown as table row subtitles, Form Factors grid section with family color coding, speed/channel info, openFormFactorDetail panel."}
|
||||||
|
|||||||
@ -218,9 +218,10 @@ export async function registerSchedules(boss: PgBoss): Promise<void> {
|
|||||||
await boss.schedule("scrape:catalog:smartoptics", "10 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:smartoptics", "10 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:hubersuhner", "25 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:hubersuhner", "25 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:eoptolink", "40 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:eoptolink", "40 */4 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
// OEM vendor seed catalogs — daily at 04:00/04:15 (stable data, rarely changes)
|
// OEM vendor seed catalogs — daily at 04:00+ (stable data, rarely changes)
|
||||||
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 });
|
||||||
|
|
||||||
// ══════════════════════════════════════════════════════════════════════
|
// ══════════════════════════════════════════════════════════════════════
|
||||||
// VENDOR LISTS — every 12h
|
// VENDOR LISTS — every 12h
|
||||||
@ -498,6 +499,12 @@ export async function registerWorkers(boss: PgBoss): Promise<void> {
|
|||||||
await scrapeJuniperOem();
|
await scrapeJuniperOem();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:hpe-aruba-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: HPE/Aruba OEM catalog seed`);
|
||||||
|
const { scrapeHpeArubaOem } = await import("./scrapers/hpe-aruba-oem");
|
||||||
|
await scrapeHpeArubaOem();
|
||||||
|
});
|
||||||
|
|
||||||
// ── Vendor lists ──────────────────────────────────────────────────────
|
// ── Vendor lists ──────────────────────────────────────────────────────
|
||||||
|
|
||||||
await boss.work("scrape:vendors:flexoptix", async () => {
|
await boss.work("scrape:vendors:flexoptix", async () => {
|
||||||
|
|||||||
146
packages/scraper/src/scrapers/hpe-aruba-oem.ts
Normal file
146
packages/scraper/src/scrapers/hpe-aruba-oem.ts
Normal file
@ -0,0 +1,146 @@
|
|||||||
|
/**
|
||||||
|
* HPE / Aruba OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Seeds HPE-branded transceiver PIDs (J / JL / JH / R series)
|
||||||
|
* into the transceivers table.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - HPE Aruba Networking Transceiver Guide (public, arubanetworks.com)
|
||||||
|
* - HPE QuickSpecs / product pages (hpe.com)
|
||||||
|
* - PIDs verified against hpe.com/h20195/v2/getpdf.aspx transceiver guides
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/hpe-aruba-oem.ts
|
||||||
|
* Cron: daily at 04:30
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface HpePID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const HPE_PIDS: HpePID[] = [
|
||||||
|
// ── 1G SFP ──────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "J4858C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX" },
|
||||||
|
{ pid: "J4859C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX" },
|
||||||
|
{ pid: "J4860C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 70000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-LH" },
|
||||||
|
{ pid: "J4861C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 70000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX" },
|
||||||
|
{ pid: "J8177C", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T", notes: "Copper SFP" },
|
||||||
|
{ pid: "J4862B", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm" },
|
||||||
|
{ pid: "JL490A", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T", notes: "Aruba CX SFP RJ45" },
|
||||||
|
|
||||||
|
// ── 10G SFP+ ────────────────────────────────────────────────────────────
|
||||||
|
{ pid: "J9151E", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR" },
|
||||||
|
{ pid: "J9150A", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR" },
|
||||||
|
{ pid: "J9152A", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
|
||||||
|
{ pid: "J9153A", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 220, reachLabel: "LRM", fiberType: "MMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LRM" },
|
||||||
|
{ pid: "JL432A", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 30, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "10GBASE-T", notes: "Aruba CX SFP+ RJ45" },
|
||||||
|
{ pid: "J9285D", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 7, reachLabel: "DAC-7M",fiberType: "DAC",connector: "SFP+", notes: "Direct Attach 7m" },
|
||||||
|
{ pid: "J9283B", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC-3M",fiberType: "DAC",connector: "SFP+", notes: "Direct Attach 3m" },
|
||||||
|
{ pid: "J9281D", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC-1M",fiberType: "DAC",connector: "SFP+", notes: "Direct Attach 1m" },
|
||||||
|
{ pid: "J9286B", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 5, reachLabel: "AOC-5M",fiberType: "MMF",connector: "SFP+", notes: "Active Optical Cable 5m" },
|
||||||
|
|
||||||
|
// ── 25G SFP28 ───────────────────────────────────────────────────────────
|
||||||
|
{ pid: "JL484A", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR" },
|
||||||
|
{ pid: "JL485A", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR" },
|
||||||
|
{ pid: "R0M46A", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 30000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", notes: "Aruba CX 25G ER" },
|
||||||
|
{ pid: "JL488A", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 3, reachLabel: "DAC-3M",fiberType: "DAC",connector: "SFP28",notes: "25G DAC 3m" },
|
||||||
|
{ pid: "JL487A", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 1, reachLabel: "DAC-1M",fiberType: "DAC",connector: "SFP28",notes: "25G DAC 1m" },
|
||||||
|
|
||||||
|
// ── 40G QSFP+ ───────────────────────────────────────────────────────────
|
||||||
|
{ pid: "JH231A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4" },
|
||||||
|
{ pid: "JH232A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4" },
|
||||||
|
{ pid: "JH233A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "BiDi",fiberType: "MMF", connector: "LC", wavelengths: "832/853nm", notes: "40G BiDi (2-fiber)" },
|
||||||
|
{ pid: "JH236A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 3, reachLabel: "DAC-3M",fiberType: "DAC",connector: "QSFP+",notes: "40G DAC 3m" },
|
||||||
|
{ pid: "JH237A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 7, reachLabel: "DAC-7M",fiberType: "DAC",connector: "QSFP+",notes: "40G DAC 7m" },
|
||||||
|
{ pid: "JH238A", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 5, reachLabel: "AOC-5M",fiberType: "MMF",connector: "QSFP+",notes: "40G AOC 5m" },
|
||||||
|
|
||||||
|
// ── 100G QSFP28 ─────────────────────────────────────────────────────────
|
||||||
|
{ pid: "JL309A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
|
||||||
|
{ pid: "JL310A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
|
||||||
|
{ pid: "JL311A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 30000, reachLabel: "ER4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm" },
|
||||||
|
{ pid: "R0M47A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4",fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
|
||||||
|
{ pid: "R0M48A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
|
||||||
|
{ pid: "R0M49A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "FR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-FR" },
|
||||||
|
{ pid: "JL312A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 3, reachLabel: "DAC-3M",fiberType: "DAC",connector: "QSFP28",notes: "100G DAC 3m" },
|
||||||
|
{ pid: "JL313A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 1, reachLabel: "DAC-1M",fiberType: "DAC",connector: "QSFP28",notes: "100G DAC 1m" },
|
||||||
|
{ pid: "JL314A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 3, reachLabel: "AOC-3M",fiberType: "MMF",connector: "QSFP28",notes: "100G AOC 3m" },
|
||||||
|
{ pid: "R0M50A", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", notes: "Aruba CX 100G SR4 BIDI" },
|
||||||
|
|
||||||
|
// ── 400G QSFP-DD ────────────────────────────────────────────────────────
|
||||||
|
{ pid: "R9B33A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 100, reachLabel: "SR8", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "400GBASE-SR8" },
|
||||||
|
{ pid: "R9B34A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
|
||||||
|
{ pid: "R9B35A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "400GBASE-FR4" },
|
||||||
|
{ pid: "R9B36A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "400GBASE-LR4" },
|
||||||
|
{ pid: "R9B37A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 3, reachLabel: "DAC-3M",fiberType:"DAC", connector: "QSFP-DD",notes: "400G DAC 3m" },
|
||||||
|
{ pid: "R9B38A", formFactor: "QSFP-DD",speedGbps:400, speed: "400G", reachMeters: 1, reachLabel: "DAC-1M",fiberType:"DAC", connector: "QSFP-DD",notes: "400G DAC 1m" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeHpeArubaOem(): Promise<void> {
|
||||||
|
console.log("=== HPE/Aruba OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
// HPE has "HPE Aruba Networking" in the vendor table
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"HPE Aruba Networking",
|
||||||
|
"oem",
|
||||||
|
"https://www.hpe.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0;
|
||||||
|
let updated = 0;
|
||||||
|
let errors = 0;
|
||||||
|
|
||||||
|
for (const p of HPE_PIDS) {
|
||||||
|
const slug = `hpe-${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=== HPE/Aruba OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}`);
|
||||||
|
console.log(` Updated: ${updated}`);
|
||||||
|
console.log(` Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${HPE_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeHpeArubaOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => {
|
||||||
|
console.error("Fatal:", err);
|
||||||
|
pool.end();
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user