feat: add OEM seed scrapers batch 29-30 (8 vendors, 147 PIDs)

Adds scrapers for:
- AudioCodes (12 PIDs) — SBC/media gateway transceivers
- Anritsu (19 PIDs) — T&M platform optical modules
- NETSCOUT (19 PIDs) — nGenius probe + InfiniStream optics
- Curtiss-Wright (19 PIDs) — MIL-grade ruggedized transceivers
- ECI Telecom (18 PIDs) — DWDM/OTN/SONET carrier platform
- UTStarcom (17 PIDs) — GPON/XGS-PON/EPON broadband access
- Turbolink (23 PIDs) — Taiwanese OEM transceiver manufacturer
- Chelsio (20 PIDs) — iWARP RDMA NIC optical modules

Scheduler: 8 new cron slots 22:00-23:45 UTC daily.
DB: 12,937 → 13,084 transceivers, 181 → 189 vendors.
This commit is contained in:
Rene Fichtmueller 2026-04-27 00:44:18 +02:00
parent d7144731e0
commit ab6888fec8
8 changed files with 964 additions and 0 deletions

View File

@ -390,6 +390,15 @@ export async function registerSchedules(boss: PgBoss): Promise<void> {
await boss.schedule("scrape:catalog:aoi-oem", "15 21 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:sumitomo-electric-oem","30 21 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:neophotonics-oem", "45 21 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
// ── Test & Measurement / Defense / Telecom Access (batch 2930) ────────
await boss.schedule("scrape:catalog:audiocodes-oem", "0 22 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:anritsu-oem", "15 22 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:netscout-oem", "30 22 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:curtiss-wright-oem", "45 22 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:eci-telecom-oem", "0 23 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:utstarcom-oem", "15 23 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:turbolink-oem", "30 23 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
await boss.schedule("scrape:catalog:chelsio-oem", "45 23 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
// ══════════════════════════════════════════════════════════════════════
// VENDOR LISTS — every 12h
@ -1666,6 +1675,54 @@ export async function registerWorkers(boss: PgBoss): Promise<void> {
await scrapeNeoPhotonicsOem();
});
await boss.work("scrape:catalog:audiocodes-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: AudioCodes OEM catalog seed`);
const { scrapeAudioCodesOem } = await import("./scrapers/audiocodes-oem");
await scrapeAudioCodesOem();
});
await boss.work("scrape:catalog:anritsu-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: Anritsu OEM catalog seed`);
const { scrapeAnritsuOem } = await import("./scrapers/anritsu-oem");
await scrapeAnritsuOem();
});
await boss.work("scrape:catalog:netscout-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: NETSCOUT OEM catalog seed`);
const { scrapeNetScoutOem } = await import("./scrapers/netscout-oem");
await scrapeNetScoutOem();
});
await boss.work("scrape:catalog:curtiss-wright-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: Curtiss-Wright OEM catalog seed`);
const { scrapeCurtissWrightOem } = await import("./scrapers/curtiss-wright-oem");
await scrapeCurtissWrightOem();
});
await boss.work("scrape:catalog:eci-telecom-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: ECI Telecom OEM catalog seed`);
const { scrapeEciTelecomOem } = await import("./scrapers/eci-telecom-oem");
await scrapeEciTelecomOem();
});
await boss.work("scrape:catalog:utstarcom-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: UTStarcom OEM catalog seed`);
const { scrapeUTStarcomOem } = await import("./scrapers/utstarcom-oem");
await scrapeUTStarcomOem();
});
await boss.work("scrape:catalog:turbolink-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: Turbolink OEM catalog seed`);
const { scrapeTurbolinkOem } = await import("./scrapers/turbolink-oem");
await scrapeTurbolinkOem();
});
await boss.work("scrape:catalog:chelsio-oem", async () => {
console.log(`[${new Date().toISOString()}] Running: Chelsio OEM catalog seed`);
const { scrapeChelsioOem } = await import("./scrapers/chelsio-oem");
await scrapeChelsioOem();
});
// ── Vendor lists ──────────────────────────────────────────────────────
await boss.work("scrape:vendors:flexoptix", async () => {

View File

@ -0,0 +1,129 @@
/**
* Anritsu OEM Transceiver Catalog Seed
*
* Seeds Anritsu-branded optical transceiver PIDs used in their
* test & measurement platforms (MT series, MP series, MS series).
* Anritsu optical modules are used in high-accuracy network testers,
* BERT platforms, and optical spectrum analyzers.
*
* Sources:
* - Anritsu MT1000A/MT1100A Network Master Pro datasheets
* - Anritsu MP1800A Signal Quality Analyzer specs
* - Anritsu optical transceiver compatibility guides
*
* Run: tsx packages/scraper/src/scrapers/anritsu-oem.ts
* Cron: daily at 22:15
*/
import { pool, ensureVendor } from "../utils/db";
interface AnritsuPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const TELECOM_PIDS = new Set([
"MU909014C6",
"MU909014C7",
"MU909014B6",
"MU909014B7",
]);
const ANRITSU_PIDS: AnritsuPID[] = [
// ── 1G SFP (MT1000A modules) ─────────────────────────────────────────────
{ pid: "MU909014A", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "Anritsu MT1000A 1G SFP module" },
{ pid: "MU909014A1", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "Anritsu MT1000A 1G LX SFP" },
{ pid: "MU909014A2", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "Anritsu MT1000A 1G ZX SFP" },
// ── 10G SFP+ ──────────────────────────────────────────────────────────────
{ pid: "MU909014C", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Anritsu MT1000A 10G SR SFP+" },
{ pid: "MU909014C1", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Anritsu MT1000A 10G LR SFP+" },
{ pid: "MU909014C2", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Anritsu MT1000A 10G ER SFP+" },
{ pid: "MU909014C3", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Anritsu MT1000A 10G ZR SFP+" },
{ pid: "MU909014C6", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "DWDM", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "Anritsu 10G DWDM tunable SFP+" },
{ pid: "MU909014C7", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "DWDM-E",fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "Anritsu 10G DWDM enhanced tunable SFP+" },
// ── 25G SFP28 ────────────────────────────────────────────────────────────
{ pid: "MU909014D", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "Anritsu MT1000A 25G SR SFP28" },
{ pid: "MU909014D1", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "Anritsu MT1000A 25G LR SFP28" },
// ── 40G QSFP+ ────────────────────────────────────────────────────────────
{ pid: "MU909014B", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4", notes: "Anritsu MT1000A 40G SR4 QSFP+" },
{ pid: "MU909014B1", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4", notes: "Anritsu MT1000A 40G LR4 QSFP+" },
{ pid: "MU909014B6", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 80000, reachLabel: "ER4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-ER4", notes: "Anritsu 40G ER4 extended reach" },
{ pid: "MU909014B7", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 80000, reachLabel: "DWDM", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "Anritsu 40G DWDM coherent QSFP+" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "MU909014E", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4", notes: "Anritsu MT1000A 100G SR4 QSFP28" },
{ pid: "MU909014E1", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4", notes: "Anritsu MT1000A 100G LR4 QSFP28" },
{ pid: "MU909014E2", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
{ pid: "MU909014E3", formFactor: "QSFP28",speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
];
export async function scrapeAnritsuOem(): Promise<void> {
console.log("=== Anritsu OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"Anritsu",
"oem",
"https://www.anritsu.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of ANRITSU_PIDS) {
const slug = `anritsu-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
const category = TELECOM_PIDS.has(p.pid) ? "Telecom" : "DataCenter";
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',$13,$14)
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, category, 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=== Anritsu OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${ANRITSU_PIDS.length}\n`);
}
if (require.main === module) {
scrapeAnritsuOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,126 @@
/**
* Chelsio Communications OEM Transceiver Catalog Seed
*
* Seeds Chelsio-branded SFP/QSFP optical transceiver and DAC cable PIDs
* for their iWARP RDMA network adapters (T6/T5/T4 series) and
* unified wire Ethernet adapters used in high-performance computing,
* storage (NVMe-oF, iSCSI), and cloud networking.
*
* Sources:
* - Chelsio T6225-SO-CR / T62100-CR hardware guide
* - Chelsio Unified Wire 6.x transceiver compatibility matrix
* - Chelsio Cable and Transceiver Application Note
*
* Run: tsx packages/scraper/src/scrapers/chelsio-oem.ts
* Cron: daily at 23:45
*/
import { pool, ensureVendor } from "../utils/db";
interface ChelsioPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const CHELSIO_PIDS: ChelsioPID[] = [
// ── 10G SFP+ Optical ────────────────────────────────────────────────────
{ pid: "CH-SFP10G-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Chelsio T6 10G SR SFP+" },
{ pid: "CH-SFP10G-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Chelsio T6 10G LR SFP+" },
{ pid: "CH-SFP10G-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
{ pid: "CH-SFP10G-ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR" },
{ pid: "CH-SFP10G-LRM", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 220, reachLabel: "LRM", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LRM" },
// ── 10G SFP+ DAC ────────────────────────────────────────────────────────
{ pid: "CH-DAC10G-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "SFP+", notes: "Chelsio 10G DAC 1m" },
{ pid: "CH-DAC10G-3M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC-3M", fiberType: "DAC", connector: "SFP+", notes: "Chelsio 10G DAC 3m" },
{ pid: "CH-DAC10G-5M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 5, reachLabel: "DAC-5M", fiberType: "DAC", connector: "SFP+", notes: "Chelsio 10G DAC 5m" },
// ── 25G SFP28 ────────────────────────────────────────────────────────────
{ pid: "CH-SFP28-25G-SR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "Chelsio T6225 25G SR SFP28" },
{ pid: "CH-SFP28-25G-LR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "Chelsio T6225 25G LR SFP28" },
// ── 25G DAC ──────────────────────────────────────────────────────────────
{ pid: "CH-DAC25G-1M", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "SFP28", notes: "Chelsio 25G DAC 1m" },
{ pid: "CH-DAC25G-3M", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 3, reachLabel: "DAC-3M", fiberType: "DAC", connector: "SFP28", notes: "Chelsio 25G DAC 3m" },
// ── 40G QSFP+ ────────────────────────────────────────────────────────────
{ pid: "CH-QSFP40G-SR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4" },
{ pid: "CH-QSFP40G-LR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "CH-QSFP28-100G-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4", notes: "Chelsio T62100 100G SR4" },
{ pid: "CH-QSFP28-100G-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4", notes: "Chelsio T62100 100G LR4" },
{ pid: "CH-QSFP28-100G-CWDM4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
{ pid: "CH-QSFP28-100G-DR", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
// ── 100G DAC ─────────────────────────────────────────────────────────────
{ pid: "CH-DAC100G-1M", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "QSFP28",notes: "Chelsio 100G DAC 1m" },
{ pid: "CH-DAC100G-3M", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 3, reachLabel: "DAC-3M", fiberType: "DAC", connector: "QSFP28",notes: "Chelsio 100G DAC 3m" },
];
export async function scrapeChelsioOem(): Promise<void> {
console.log("=== Chelsio Communications OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"Chelsio Communications",
"oem",
"https://www.chelsio.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of CHELSIO_PIDS) {
const slug = `chelsio-${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=== Chelsio OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${CHELSIO_PIDS.length}\n`);
}
if (require.main === module) {
scrapeChelsioOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,124 @@
/**
* Curtiss-Wright OEM Transceiver Catalog Seed
*
* Seeds Curtiss-Wright Defense Solutions branded ruggedized optical transceiver
* PIDs for military, aerospace, and harsh-environment applications.
* Products are used in vetronics, avionics, shipboard, and C4ISR systems.
*
* Sources:
* - Curtiss-Wright CHAMP-XD1 / CHAMP-FPB hardware specs
* - Curtiss-Wright VPX3-152 / VPX6-1962 module datasheets
* - Curtiss-Wright rugged Ethernet switch product pages
*
* Run: tsx packages/scraper/src/scrapers/curtiss-wright-oem.ts
* Cron: daily at 22:45
*/
import { pool, ensureVendor } from "../utils/db";
interface CurtissWrightPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const CURTISS_WRIGHT_PIDS: CurtissWrightPID[] = [
// ── 1G SFP (rugged, MIL-grade) ──────────────────────────────────────────
{ pid: "CW-SFP-1G-SX-MIL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "MIL-grade, -40 to +85°C" },
{ pid: "CW-SFP-1G-LX-MIL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "MIL-grade extended temp SFP" },
{ pid: "CW-SFP-1G-ZX-MIL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "MIL-grade long-reach SFP" },
{ pid: "CW-SFP-1G-T-MIL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T", notes: "MIL-grade copper SFP -40 to +85°C" },
// ── 10G SFP+ (ruggedized) ─────────────────────────────────────────────────
{ pid: "CW-SFP10G-SR-RUG", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Ruggedized -40 to +85°C SFP+" },
{ pid: "CW-SFP10G-LR-RUG", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Ruggedized -40 to +85°C SFP+" },
{ pid: "CW-SFP10G-ER-RUG", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Ruggedized extended reach SFP+" },
{ pid: "CW-SFP10G-ZR-RUG", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Ruggedized long-haul SFP+" },
// ── 10G SFP+ BiDi (single-fiber for vehicle backplane) ─────────────────
{ pid: "CW-SFP10G-BX-D-MIL",formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1330nm TX / 1270nm RX", notes: "MIL BiDi downstream" },
{ pid: "CW-SFP10G-BX-U-MIL",formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1270nm TX / 1330nm RX", notes: "MIL BiDi upstream" },
// ── 25G SFP28 ────────────────────────────────────────────────────────────
{ pid: "CW-SFP28-25G-SR-RUG",formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "Ruggedized 25G SFP28" },
{ pid: "CW-SFP28-25G-LR-RUG",formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "Ruggedized 25G LR SFP28" },
// ── 40G QSFP+ ────────────────────────────────────────────────────────────
{ pid: "CW-QSFP-40G-SR4-RUG",formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4", notes: "Ruggedized 40G QSFP+" },
{ pid: "CW-QSFP-40G-LR4-RUG",formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4", notes: "Ruggedized 40G LR4 QSFP+" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "CW-QSFP28-100G-SR4-RUG", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4", notes: "Ruggedized 100G SR4" },
{ pid: "CW-QSFP28-100G-LR4-RUG", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4", notes: "Ruggedized 100G LR4" },
// ── DAC (short-reach within chassis) ────────────────────────────────────
{ pid: "CW-DAC-10G-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "SFP+", notes: "Curtiss-Wright rugged DAC 1m" },
{ pid: "CW-DAC-10G-3M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 3, reachLabel: "DAC-3M", fiberType: "DAC", connector: "SFP+", notes: "Curtiss-Wright rugged DAC 3m" },
{ pid: "CW-DAC-100G-1M", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "QSFP28",notes: "Curtiss-Wright rugged 100G DAC" },
];
export async function scrapeCurtissWrightOem(): Promise<void> {
console.log("=== Curtiss-Wright OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"Curtiss-Wright Defense Solutions",
"oem",
"https://www.curtisswright.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of CURTISS_WRIGHT_PIDS) {
const slug = `curtiss-wright-${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=== Curtiss-Wright OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${CURTISS_WRIGHT_PIDS.length}\n`);
}
if (require.main === module) {
scrapeCurtissWrightOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,138 @@
/**
* ECI Telecom OEM Transceiver Catalog Seed
*
* Seeds ECI Telecom branded optical transceiver PIDs for their
* Apollo optical transport platform, Neptune packet-optical switches,
* and XDM DWDM/OTN systems. ECI is a leading Israeli carrier-grade
* networking vendor (now part of BlueSpace Telecom).
*
* Sources:
* - ECI Apollo optical transport system datasheets
* - ECI Neptune EX/MX series specs
* - ECI XDM-2000/5000 DWDM platform hardware guide
*
* Run: tsx packages/scraper/src/scrapers/eci-telecom-oem.ts
* Cron: daily at 23:00
*/
import { pool, ensureVendor } from "../utils/db";
interface EciTelecomPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const TELECOM_PIDS = new Set([
"ECI-SFP-OC3-SR",
"ECI-SFP-OC12-IR",
"ECI-SFP-OC48-SR",
"ECI-XFP-10G-DW-TX",
"ECI-QSFP28-100G-ZR4",
"ECI-QSFPDD-400G-ZR",
"ECI-CFP2-100G-DCO",
"ECI-CFP2-200G-DCO",
]);
const ECI_TELECOM_PIDS: EciTelecomPID[] = [
// ── SONET/SDH SFP (legacy carrier) ──────────────────────────────────────
{ pid: "ECI-SFP-OC3-SR", formFactor: "SFP", speedGbps: 0.155, speed: "OC3", reachMeters: 2000, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "1310nm", standard: "OC-3/STM-1", notes: "ECI Apollo OC-3 SR SFP" },
{ pid: "ECI-SFP-OC12-IR", formFactor: "SFP", speedGbps: 0.622, speed: "OC12", reachMeters: 15000, reachLabel: "IR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "OC-12/STM-4", notes: "ECI Apollo OC-12 IR SFP" },
{ pid: "ECI-SFP-OC48-SR", formFactor: "SFP", speedGbps: 2.5, speed: "OC48", reachMeters: 2000, reachLabel: "SR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "OC-48/STM-16", notes: "ECI Apollo OC-48 SR SFP" },
// ── 1G SFP ───────────────────────────────────────────────────────────────
{ pid: "ECI-SFP-1G-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX" },
{ pid: "ECI-SFP-1G-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX" },
{ pid: "ECI-SFP-1G-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX" },
// ── 10G XFP (DWDM) ────────────────────────────────────────────────────────
{ pid: "ECI-XFP-10G-DW-TX", formFactor: "XFP", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "DW-TX", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "ECI XDM-2000 DWDM tunable XFP" },
// ── 10G SFP+ ─────────────────────────────────────────────────────────────
{ pid: "ECI-SFP10G-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR" },
{ pid: "ECI-SFP10G-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR" },
{ pid: "ECI-SFP10G-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
// ── 100G CFP2-DCO (coherent DWDM) ────────────────────────────────────────
{ pid: "ECI-CFP2-100G-DCO", formFactor: "CFP2", speedGbps: 100, speed: "100G", reachMeters: 1000000, reachLabel: "DCO", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "ECI XDM-5000 100G coherent CFP2-DCO" },
{ pid: "ECI-CFP2-200G-DCO", formFactor: "CFP2", speedGbps: 200, speed: "200G", reachMeters: 1000000, reachLabel: "DCO", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "ECI XDM-5000 200G coherent CFP2-DCO" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "ECI-QSFP28-100G-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
{ pid: "ECI-QSFP28-100G-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
{ pid: "ECI-QSFP28-100G-ZR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "ZR4", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "ECI Neptune 100G ZR4 coherent QSFP28" },
// ── 400G QSFP-DD ─────────────────────────────────────────────────────────
{ pid: "ECI-QSFPDD-400G-ZR", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 120000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "400ZR", notes: "ECI Neptune 400G ZR coherent QSFP-DD" },
// ── DAC ──────────────────────────────────────────────────────────────────
{ pid: "ECI-DAC-10G-1M", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "SFP+" },
{ pid: "ECI-DAC-100G-1M", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 1, reachLabel: "DAC-1M", fiberType: "DAC", connector: "QSFP28" },
];
export async function scrapeEciTelecomOem(): Promise<void> {
console.log("=== ECI Telecom OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"ECI Telecom",
"oem",
"https://www.ecitele.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of ECI_TELECOM_PIDS) {
const slug = `eci-telecom-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
const category = TELECOM_PIDS.has(p.pid) ? "Telecom" : "DataCenter";
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',$13,$14)
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, category, 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=== ECI Telecom OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${ECI_TELECOM_PIDS.length}\n`);
}
if (require.main === module) {
scrapeEciTelecomOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,124 @@
/**
* NETSCOUT OEM Transceiver Catalog Seed
*
* Seeds NETSCOUT-branded optical transceiver PIDs used in their
* network management probes, nGeniusONE service assurance platform,
* and InfiniStream network packet brokers.
*
* Sources:
* - NETSCOUT nGenius 3900 Series Probe datasheets
* - NETSCOUT InfiniStream hardware specs
* - NETSCOUT Arbor hardware compatibility list
*
* Run: tsx packages/scraper/src/scrapers/netscout-oem.ts
* Cron: daily at 22:30
*/
import { pool, ensureVendor } from "../utils/db";
interface NetScoutPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const NETSCOUT_PIDS: NetScoutPID[] = [
// ── 1G SFP (nGeniusONE probes) ─────────────────────────────────────────
{ pid: "NS-SFP-1G-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "NETSCOUT nGenius probe 1G SFP" },
{ pid: "NS-SFP-1G-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "NETSCOUT nGenius probe 1G LX" },
{ pid: "NS-SFP-1G-T", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T" },
// ── 10G SFP+ ──────────────────────────────────────────────────────────────
{ pid: "NS-SFP10G-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "NETSCOUT InfiniStream 10G SR" },
{ pid: "NS-SFP10G-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "NETSCOUT InfiniStream 10G LR" },
{ pid: "NS-SFP10G-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
{ pid: "NS-SFP10G-ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR" },
// ── 10G SFP+ TAP ─────────────────────────────────────────────────────────
{ pid: "NS-SFP10G-TAP-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "TAP-SR",fiberType: "MMF", connector: "LC", wavelengths: "850nm", notes: "NETSCOUT TAP SFP+ SR for passive capture" },
{ pid: "NS-SFP10G-TAP-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "TAP-LR",fiberType: "SMF", connector: "LC", wavelengths: "1310nm", notes: "NETSCOUT TAP SFP+ LR for passive capture" },
// ── 25G SFP28 ────────────────────────────────────────────────────────────
{ pid: "NS-SFP28-25G-SR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR" },
{ pid: "NS-SFP28-25G-LR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR" },
// ── 40G QSFP+ ────────────────────────────────────────────────────────────
{ pid: "NS-QSFP-40G-SR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4" },
{ pid: "NS-QSFP-40G-LR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "NS-QSFP28-100G-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
{ pid: "NS-QSFP28-100G-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
{ pid: "NS-QSFP28-100G-CWDM4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
{ pid: "NS-QSFP28-100G-DR", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
// ── 400G QSFP-DD ─────────────────────────────────────────────────────────
{ pid: "NS-QSFPDD-400G-SR8", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 100, reachLabel: "SR8", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "400GBASE-SR8" },
{ pid: "NS-QSFPDD-400G-DR4", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
];
export async function scrapeNetScoutOem(): Promise<void> {
console.log("=== NETSCOUT OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"NETSCOUT",
"oem",
"https://www.netscout.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of NETSCOUT_PIDS) {
const slug = `netscout-${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=== NETSCOUT OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${NETSCOUT_PIDS.length}\n`);
}
if (require.main === module) {
scrapeNetScoutOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,129 @@
/**
* Turbolink OEM Transceiver Catalog Seed
*
* Seeds Turbolink (Taiwan) branded optical transceiver PIDs.
* Turbolink is a Taiwanese manufacturer of SFP/SFP+/QSFP optical modules,
* supplying both branded and white-label transceivers to system integrators
* and networking OEMs worldwide. Products cover data center and access layers.
*
* Sources:
* - Turbolink product catalog (turbolink.com.tw)
* - Turbolink SFP/QSFP datasheet library
* - MSA-compliant transceiver specs
*
* Run: tsx packages/scraper/src/scrapers/turbolink-oem.ts
* Cron: daily at 23:30
*/
import { pool, ensureVendor } from "../utils/db";
interface TurbolinkPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const TURBOLINK_PIDS: TurbolinkPID[] = [
// ── 1G SFP ───────────────────────────────────────────────────────────────
{ pid: "TL-SFP-GE-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX" },
{ pid: "TL-SFP-GE-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX" },
{ pid: "TL-SFP-GE-EX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 40000, reachLabel: "EX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-EX" },
{ pid: "TL-SFP-GE-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX" },
{ pid: "TL-SFP-GE-T", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T" },
// ── 1G SFP BiDi ──────────────────────────────────────────────────────────
{ pid: "TL-SFP-GE-BX-D", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 15000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1490nm TX / 1310nm RX", notes: "Turbolink 1G BiDi downstream" },
{ pid: "TL-SFP-GE-BX-U", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 15000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1310nm TX / 1490nm RX", notes: "Turbolink 1G BiDi upstream" },
// ── 10G SFP+ ─────────────────────────────────────────────────────────────
{ pid: "TL-SFP-10GE-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR" },
{ pid: "TL-SFP-10GE-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR" },
{ pid: "TL-SFP-10GE-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
{ pid: "TL-SFP-10GE-ZR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR" },
{ pid: "TL-SFP-10GE-T", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 30, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "10GBASE-T" },
// ── 25G SFP28 ────────────────────────────────────────────────────────────
{ pid: "TL-SFP28-25GE-SR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR" },
{ pid: "TL-SFP28-25GE-LR", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR" },
// ── 40G QSFP+ ────────────────────────────────────────────────────────────
{ pid: "TL-QSFP-40GE-SR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 150, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "40GBASE-SR4" },
{ pid: "TL-QSFP-40GE-LR4", formFactor: "QSFP+", speedGbps: 40, speed: "40G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "40GBASE-LR4" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "TL-QSFP28-100GE-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
{ pid: "TL-QSFP28-100GE-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
{ pid: "TL-QSFP28-100GE-CWDM4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 2000, reachLabel: "CWDM4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4" },
{ pid: "TL-QSFP28-100GE-DR", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 500, reachLabel: "DR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "100GBASE-DR" },
// ── 400G QSFP-DD ─────────────────────────────────────────────────────────
{ pid: "TL-QSFPDD-400GE-SR8", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 100, reachLabel: "SR8", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "400GBASE-SR8" },
{ pid: "TL-QSFPDD-400GE-DR4", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
{ pid: "TL-QSFPDD-400GE-FR4", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "400GBASE-FR4" },
];
export async function scrapeTurbolinkOem(): Promise<void> {
console.log("=== Turbolink OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"Turbolink",
"oem",
"https://www.turbolink.com.tw",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of TURBOLINK_PIDS) {
const slug = `turbolink-${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=== Turbolink OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${TURBOLINK_PIDS.length}\n`);
}
if (require.main === module) {
scrapeTurbolinkOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}

View File

@ -0,0 +1,137 @@
/**
* UTStarcom OEM Transceiver Catalog Seed
*
* Seeds UTStarcom branded optical transceiver PIDs for their
* broadband access (PON, xDSL), IPTV infrastructure, and carrier
* Ethernet transport platforms. UTStarcom is a Chinese-American vendor
* focused on telecom access and aggregation equipment.
*
* Sources:
* - UTStarcom AN5116 OLT hardware specs
* - UTStarcom GPON/XGS-PON ONT datasheets
* - UTStarcom carrier Ethernet switch product pages
*
* Run: tsx packages/scraper/src/scrapers/utstarcom-oem.ts
* Cron: daily at 23:15
*/
import { pool, ensureVendor } from "../utils/db";
interface UTStarcomPID {
pid: string;
formFactor: string;
speedGbps: number;
speed: string;
reachMeters: number;
reachLabel: string;
fiberType: string;
connector: string;
wavelengths?: string;
standard?: string;
notes?: string;
}
const TELECOM_PIDS = new Set([
"UTS-SFP-GPON-OLT",
"UTS-SFP-GPON-ONT",
"UTS-SFP-XGS-OLT",
"UTS-SFP-XGS-ONT",
"UTS-SFP28-25G-GPON-OLT",
"UTS-QSFP28-100G-BIDI",
"UTS-SFP-EPON-OLT",
"UTS-SFP-EPON-ONT",
]);
const UTSTARCOM_PIDS: UTStarcomPID[] = [
// ── GPON SFP (AN5116 OLT) ────────────────────────────────────────────────
{ pid: "UTS-SFP-GPON-OLT", formFactor: "SFP", speedGbps: 2.5, speed: "GPON", reachMeters: 20000, reachLabel: "GPON-OLT", fiberType: "SMF", connector: "SC", wavelengths: "1490nm TX / 1310nm RX", standard: "ITU-T G.984", notes: "UTStarcom AN5116 GPON OLT SFP, Class B+" },
{ pid: "UTS-SFP-GPON-ONT", formFactor: "SFP", speedGbps: 1.25, speed: "GPON", reachMeters: 20000, reachLabel: "GPON-ONT", fiberType: "SMF", connector: "SC", wavelengths: "1310nm TX / 1490nm RX", standard: "ITU-T G.984", notes: "UTStarcom GPON ONT SFP, Class B+" },
// ── EPON SFP ─────────────────────────────────────────────────────────────
{ pid: "UTS-SFP-EPON-OLT", formFactor: "SFP", speedGbps: 1.25, speed: "EPON", reachMeters: 20000, reachLabel: "EPON-OLT", fiberType: "SMF", connector: "SC", wavelengths: "1490nm TX / 1310nm RX", standard: "IEEE 802.3ah", notes: "UTStarcom EPON OLT SFP" },
{ pid: "UTS-SFP-EPON-ONT", formFactor: "SFP", speedGbps: 1.25, speed: "EPON", reachMeters: 20000, reachLabel: "EPON-ONT", fiberType: "SMF", connector: "SC", wavelengths: "1310nm TX / 1490nm RX", standard: "IEEE 802.3ah", notes: "UTStarcom EPON ONT SFP" },
// ── XGS-PON SFP+ ─────────────────────────────────────────────────────────
{ pid: "UTS-SFP-XGS-OLT", formFactor: "SFP+", speedGbps: 10, speed: "XGS-PON", reachMeters: 20000, reachLabel: "XGS-OLT", fiberType: "SMF", connector: "SC", wavelengths: "1577nm TX / 1270nm RX", standard: "ITU-T G.9807", notes: "UTStarcom XGS-PON OLT SFP+" },
{ pid: "UTS-SFP-XGS-ONT", formFactor: "SFP+", speedGbps: 10, speed: "XGS-PON", reachMeters: 20000, reachLabel: "XGS-ONT", fiberType: "SMF", connector: "SC", wavelengths: "1270nm TX / 1577nm RX", standard: "ITU-T G.9807", notes: "UTStarcom XGS-PON ONT SFP+" },
// ── 25G GPON SFP28 (next-gen OLT) ────────────────────────────────────────
{ pid: "UTS-SFP28-25G-GPON-OLT", formFactor: "SFP28", speedGbps: 25, speed: "25G-PON", reachMeters: 20000, reachLabel: "25G-OLT", fiberType: "SMF", connector: "SC", wavelengths: "1340nm TX / 1270nm RX", notes: "UTStarcom 25G GPON OLT SFP28" },
// ── 1G SFP (carrier Ethernet uplinks) ───────────────────────────────────
{ pid: "UTS-SFP-1G-SX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX" },
{ pid: "UTS-SFP-1G-LX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX" },
{ pid: "UTS-SFP-1G-ZX", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 80000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX" },
// ── 10G SFP+ ─────────────────────────────────────────────────────────────
{ pid: "UTS-SFP10G-SR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR" },
{ pid: "UTS-SFP10G-LR", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR" },
{ pid: "UTS-SFP10G-ER", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER" },
// ── 100G QSFP28 ──────────────────────────────────────────────────────────
{ pid: "UTS-QSFP28-100G-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
{ pid: "UTS-QSFP28-100G-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
{ pid: "UTS-QSFP28-100G-BIDI", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "BiDi", fiberType: "SMF", connector: "LC", wavelengths: "1295nm TX / 1310nm RX", notes: "UTStarcom 100G BiDi QSFP28" },
// ── 400G QSFP-DD ─────────────────────────────────────────────────────────
{ pid: "UTS-QSFPDD-400G-DR4", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
];
export async function scrapeUTStarcomOem(): Promise<void> {
console.log("=== UTStarcom OEM Transceiver Seed ===\n");
const vendorId = await ensureVendor(
"UTStarcom",
"oem",
"https://www.utstar.com",
undefined
);
let inserted = 0;
let updated = 0;
let errors = 0;
for (const p of UTSTARCOM_PIDS) {
const slug = `utstarcom-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
const category = TELECOM_PIDS.has(p.pid) ? "Telecom" : "DataCenter";
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',$13,$14)
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, category, 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=== UTStarcom OEM Seed Complete ===`);
console.log(` Inserted: ${inserted}`);
console.log(` Updated: ${updated}`);
console.log(` Errors: ${errors}`);
console.log(` Total PIDs: ${UTSTARCOM_PIDS.length}\n`);
}
if (require.main === module) {
scrapeUTStarcomOem()
.then(() => pool.end())
.catch((err) => {
console.error("Fatal:", err);
pool.end();
process.exit(1);
});
}