feat: add Finisar, Acacia, Inphi OEM scrapers (batch 34)
- finisar-oem: 17 PIDs (FTLX/FTLC historical BoM series, 1G-100G, widely referenced) - acacia-oem: 14 PIDs (AC400/AC1200 coherent CFP2-DCO/QSFP-DD/OSFP up to 1.2T) - inphi-oem: 13 PIDs (ColorZ/COLORZ-II DWDM QSFP28/QSFP-DD + 800G OSFP) - scheduler: wired all 3 at 02:45/03:00/03:15 UTC
This commit is contained in:
parent
1023b24fd0
commit
32d3ded169
@ -410,6 +410,9 @@ export async function registerSchedules(boss: PgBoss): Promise<void> {
|
|||||||
await boss.schedule("scrape:catalog:radiflow-oem", "0 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:radiflow-oem", "0 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:dragonwave-oem", "15 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:dragonwave-oem", "15 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
await boss.schedule("scrape:catalog:teledyne-lecroy-oem", "30 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
await boss.schedule("scrape:catalog:teledyne-lecroy-oem", "30 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:finisar-oem", "45 2 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:acacia-oem", "0 3 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
await boss.schedule("scrape:catalog:inphi-oem", "15 3 * * *", {}, { retryLimit: 2, expireInSeconds: 3600 });
|
||||||
|
|
||||||
// ══════════════════════════════════════════════════════════════════════
|
// ══════════════════════════════════════════════════════════════════════
|
||||||
// VENDOR LISTS — every 12h
|
// VENDOR LISTS — every 12h
|
||||||
@ -1800,6 +1803,24 @@ export async function registerWorkers(boss: PgBoss): Promise<void> {
|
|||||||
await scrapeTeledyneLeCroyOem();
|
await scrapeTeledyneLeCroyOem();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:finisar-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Finisar OEM catalog seed`);
|
||||||
|
const { scrapeFinisarOem } = await import("./scrapers/finisar-oem");
|
||||||
|
await scrapeFinisarOem();
|
||||||
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:acacia-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Acacia Communications OEM catalog seed`);
|
||||||
|
const { scrapeAcaciaOem } = await import("./scrapers/acacia-oem");
|
||||||
|
await scrapeAcaciaOem();
|
||||||
|
});
|
||||||
|
|
||||||
|
await boss.work("scrape:catalog:inphi-oem", async () => {
|
||||||
|
console.log(`[${new Date().toISOString()}] Running: Inphi (Marvell) OEM catalog seed`);
|
||||||
|
const { scrapeInphiOem } = await import("./scrapers/inphi-oem");
|
||||||
|
await scrapeInphiOem();
|
||||||
|
});
|
||||||
|
|
||||||
// ── Vendor lists ──────────────────────────────────────────────────────
|
// ── Vendor lists ──────────────────────────────────────────────────────
|
||||||
|
|
||||||
await boss.work("scrape:vendors:flexoptix", async () => {
|
await boss.work("scrape:vendors:flexoptix", async () => {
|
||||||
|
|||||||
111
packages/scraper/src/scrapers/acacia-oem.ts
Normal file
111
packages/scraper/src/scrapers/acacia-oem.ts
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
/**
|
||||||
|
* Acacia Communications OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Seeds Acacia Communications (acquired by Cisco in 2021) branded coherent
|
||||||
|
* optical module PIDs. Acacia pioneered silicon photonics-based coherent DSP
|
||||||
|
* modules including the AC400, AC1200, and Pico platforms used in DCI,
|
||||||
|
* metro, and long-haul optical transport networks.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Acacia AC400 CFP2-DCO 400G coherent module datasheet
|
||||||
|
* - Acacia AC1200 1.2T coherent module product brief
|
||||||
|
* - Acacia Pico 100G coherent CFP datasheet
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/acacia-oem.ts
|
||||||
|
* Cron: daily at 03:00
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface AcaciaPID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ACACIA_PIDS: AcaciaPID[] = [
|
||||||
|
// ── 100G Coherent (Pico / CFP) ───────────────────────────────────────
|
||||||
|
{ pid: "AC-PICO-100G-CFP", formFactor: "CFP", speedGbps: 100, speed: "100G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia Pico 100G coherent CFP, long-haul" },
|
||||||
|
{ pid: "AC-PICO-100G-CFP2", formFactor: "CFP2", speedGbps: 100, speed: "100G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia Pico 100G coherent CFP2" },
|
||||||
|
{ pid: "AC-PICO-100G-QSFP28", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "100G-ZR", notes: "Acacia Pico 100G ZR QSFP28" },
|
||||||
|
|
||||||
|
// ── 200G Coherent ────────────────────────────────────────────────────
|
||||||
|
{ pid: "AC-CFP2-DCO-200G", formFactor: "CFP2-DCO",speedGbps: 200,speed: "200G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia 200G CFP2-DCO coherent" },
|
||||||
|
|
||||||
|
// ── 400G Coherent (AC400) ─────────────────────────────────────────────
|
||||||
|
{ pid: "AC400-CFP2-DCO", formFactor: "CFP2-DCO",speedGbps: 400,speed: "400G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia AC400 400G CFP2-DCO — industry milestone" },
|
||||||
|
{ pid: "AC400-QSFPDD-ZR", formFactor: "QSFP-DD", speedGbps: 400,speed: "400G", reachMeters: 120000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "400ZR", notes: "Acacia 400G ZR QSFP-DD" },
|
||||||
|
{ pid: "AC400-QSFPDD-ZRP", formFactor: "QSFP-DD", speedGbps: 400,speed: "400G", reachMeters: 1000000, reachLabel: "ZR+", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia 400G ZR+ QSFP-DD extended reach" },
|
||||||
|
|
||||||
|
// ── 400G DataCenter (non-coherent) ───────────────────────────────────
|
||||||
|
{ pid: "AC-QSFPDD-400G-DR4", formFactor: "QSFP-DD", speedGbps: 400,speed: "400G", reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
|
||||||
|
{ pid: "AC-QSFPDD-400G-FR4", formFactor: "QSFP-DD", speedGbps: 400,speed: "400G", reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "400GBASE-FR4" },
|
||||||
|
{ pid: "AC-QSFPDD-400G-LR4", formFactor: "QSFP-DD", speedGbps: 400,speed: "400G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "400GBASE-LR4" },
|
||||||
|
|
||||||
|
// ── 800G Coherent (AC1200 platform) ──────────────────────────────────
|
||||||
|
{ pid: "AC1200-800G-OSFP", formFactor: "OSFP", speedGbps: 800, speed: "800G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia AC1200 800G coherent OSFP" },
|
||||||
|
{ pid: "AC1200-1200G-OSFP", formFactor: "OSFP", speedGbps: 1200,speed: "1.2T", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Acacia AC1200 1.2T coherent OSFP — highest capacity" },
|
||||||
|
|
||||||
|
// ── 100G standard (non-coherent, Acacia silicon photonics) ───────────
|
||||||
|
{ pid: "AC-QSFP28-100G-SR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
|
||||||
|
{ pid: "AC-QSFP28-100G-LR4", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeAcaciaOem(): Promise<void> {
|
||||||
|
console.log("=== Acacia Communications OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Acacia Communications (Cisco)",
|
||||||
|
"oem",
|
||||||
|
"https://acacia-inc.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of ACACIA_PIDS) {
|
||||||
|
const slug = `acacia-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
|
||||||
|
const category = p.speedGbps >= 200 && p.reachMeters > 10000 ? "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=== Acacia OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${ACACIA_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeAcaciaOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
118
packages/scraper/src/scrapers/finisar-oem.ts
Normal file
118
packages/scraper/src/scrapers/finisar-oem.ts
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
/**
|
||||||
|
* Finisar OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Seeds historical Finisar branded optical transceiver PIDs. Finisar was
|
||||||
|
* one of the largest optical transceiver makers before being acquired by
|
||||||
|
* II-VI (now Coherent) in 2019. The FTLX/FTLC PID families remain widely
|
||||||
|
* referenced in legacy BoMs and compatibility matrices.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Finisar FTLX product series datasheets
|
||||||
|
* - Finisar FTLC 100G QSFP28 product guide
|
||||||
|
* - Finisar FCLF/FCMJ legacy SFP catalog
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/finisar-oem.ts
|
||||||
|
* Cron: daily at 02:45
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface FinisarPID {
|
||||||
|
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([
|
||||||
|
"FTLX3612M357",
|
||||||
|
"FTLX1612M357",
|
||||||
|
"FTLC1121RDNM",
|
||||||
|
]);
|
||||||
|
|
||||||
|
const FINISAR_PIDS: FinisarPID[] = [
|
||||||
|
// ── 1G SFP (FCLF / FCMJ series) ─────────────────────────────────────
|
||||||
|
{ pid: "FCLF8521P2BTL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 550, reachLabel: "SX", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "1000BASE-SX", notes: "Finisar 1G SX SFP" },
|
||||||
|
{ pid: "FCLF8522P2BTL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 10000, reachLabel: "LX", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "1000BASE-LX", notes: "Finisar 1G LX SFP" },
|
||||||
|
{ pid: "FCLF8524P2BTL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 70000, reachLabel: "ZX", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "1000BASE-ZX", notes: "Finisar 1G ZX SFP" },
|
||||||
|
{ pid: "FCLF8519P2BTL", formFactor: "SFP", speedGbps: 1, speed: "1G", reachMeters: 100, reachLabel: "T", fiberType: "DAC", connector: "RJ45", standard: "1000BASE-T", notes: "Finisar copper SFP" },
|
||||||
|
|
||||||
|
// ── 10G SFP+ (FTLX series) ────────────────────────────────────────────
|
||||||
|
{ pid: "FTLX8571D3BCL", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 300, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "10GBASE-SR", notes: "Finisar 10G SR SFP+ (most-referenced)" },
|
||||||
|
{ pid: "FTLX1471D3BCL", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "10GBASE-LR", notes: "Finisar 10G LR SFP+" },
|
||||||
|
{ pid: "FTLX1471D3BNL", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 40000, reachLabel: "ER", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ER", notes: "Finisar 10G ER SFP+" },
|
||||||
|
{ pid: "FTLX1471D3BCL-Z",formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "1550nm", standard: "10GBASE-ZR", notes: "Finisar 10G ZR SFP+" },
|
||||||
|
{ pid: "FTLX3612M357", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 80000, reachLabel: "DWDM", fiberType: "SMF", connector: "LC", wavelengths: "C-band DWDM", notes: "Finisar DWDM XFP/SFP+ tunable" },
|
||||||
|
|
||||||
|
// ── 10G BiDi ─────────────────────────────────────────────────────────
|
||||||
|
{ pid: "FTLX1412M3BCL", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-D", fiberType: "SMF", connector: "LC", wavelengths: "1330nm TX / 1270nm RX", notes: "Finisar 10G BiDi DS" },
|
||||||
|
{ pid: "FTLX1411M3BCL", formFactor: "SFP+", speedGbps: 10, speed: "10G", reachMeters: 10000, reachLabel: "BX-U", fiberType: "SMF", connector: "LC", wavelengths: "1270nm TX / 1330nm RX", notes: "Finisar 10G BiDi US" },
|
||||||
|
|
||||||
|
// ── 25G SFP28 ────────────────────────────────────────────────────────
|
||||||
|
{ pid: "FTLF8536P4BCL", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 100, reachLabel: "SR", fiberType: "MMF", connector: "LC", wavelengths: "850nm", standard: "25GBASE-SR", notes: "Finisar 25G SR SFP28" },
|
||||||
|
{ pid: "FTLF1536P4BCL", formFactor: "SFP28", speedGbps: 25, speed: "25G", reachMeters: 10000, reachLabel: "LR", fiberType: "SMF", connector: "LC", wavelengths: "1310nm", standard: "25GBASE-LR", notes: "Finisar 25G LR SFP28" },
|
||||||
|
|
||||||
|
// ── 100G QSFP28 (FTLC series) ────────────────────────────────────────
|
||||||
|
{ pid: "FTLC9551REPM", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4", notes: "Finisar 100G SR4 QSFP28" },
|
||||||
|
{ pid: "FTLC1121SDNM", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4", notes: "Finisar 100G LR4 QSFP28" },
|
||||||
|
{ pid: "FTLC1121RDNM", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 80000, reachLabel: "ER4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-ER4", notes: "Finisar 100G ER4 QSFP28" },
|
||||||
|
{ pid: "FTLC9152RGPL", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 2000, reachLabel: "CWDM4",fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "100GBASE-CWDM4",notes: "Finisar 100G CWDM4 QSFP28" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeFinisarOem(): Promise<void> {
|
||||||
|
console.log("=== Finisar OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Finisar (II-VI/Coherent)",
|
||||||
|
"oem",
|
||||||
|
"https://www.ii-vi.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of FINISAR_PIDS) {
|
||||||
|
const slug = `finisar-${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=== Finisar OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${FINISAR_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeFinisarOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
106
packages/scraper/src/scrapers/inphi-oem.ts
Normal file
106
packages/scraper/src/scrapers/inphi-oem.ts
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
/**
|
||||||
|
* Inphi Corporation OEM Transceiver Catalog Seed
|
||||||
|
*
|
||||||
|
* Seeds Inphi (acquired by Marvell in 2021) branded optical transceiver PIDs.
|
||||||
|
* Inphi pioneered the ColorZ DWDM pluggable platform and COLORZ II 400G
|
||||||
|
* coherent modules, enabling DWDM over standard QSFP28/QSFP-DD form factors
|
||||||
|
* for metro and DCI applications.
|
||||||
|
*
|
||||||
|
* Sources:
|
||||||
|
* - Inphi ColorZ 100G DWDM QSFP28 datasheet
|
||||||
|
* - Inphi COLORZ II 400G QSFP-DD product brief
|
||||||
|
* - Inphi Canopus 200G CFP2-DCO coherent specs
|
||||||
|
*
|
||||||
|
* Run: tsx packages/scraper/src/scrapers/inphi-oem.ts
|
||||||
|
* Cron: daily at 03:15
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { pool, ensureVendor } from "../utils/db";
|
||||||
|
|
||||||
|
interface InphiPID {
|
||||||
|
pid: string;
|
||||||
|
formFactor: string;
|
||||||
|
speedGbps: number;
|
||||||
|
speed: string;
|
||||||
|
reachMeters: number;
|
||||||
|
reachLabel: string;
|
||||||
|
fiberType: string;
|
||||||
|
connector: string;
|
||||||
|
wavelengths?: string;
|
||||||
|
standard?: string;
|
||||||
|
notes?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const INPHI_PIDS: InphiPID[] = [
|
||||||
|
// ── ColorZ 100G DWDM QSFP28 ──────────────────────────────────────────
|
||||||
|
{ pid: "COLORZ-100G-CH17", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "DWDM-C17", fiberType: "SMF", connector: "LC", wavelengths: "1563.86nm (C17)", notes: "Inphi ColorZ 100G Ch17 DWDM QSFP28" },
|
||||||
|
{ pid: "COLORZ-100G-CH21", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "DWDM-C21", fiberType: "SMF", connector: "LC", wavelengths: "1560.61nm (C21)", notes: "Inphi ColorZ 100G Ch21" },
|
||||||
|
{ pid: "COLORZ-100G-CH33", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "DWDM-C33", fiberType: "SMF", connector: "LC", wavelengths: "1550.92nm (C33)", notes: "Inphi ColorZ 100G Ch33" },
|
||||||
|
{ pid: "COLORZ-100G-TUNE", formFactor: "QSFP28", speedGbps: 100, speed: "100G", reachMeters: 80000, reachLabel: "DWDM-TUNE",fiberType: "SMF", connector: "LC", wavelengths: "C-band tunable", notes: "Inphi ColorZ 100G Tunable QSFP28 — landmark DCI product" },
|
||||||
|
|
||||||
|
// ── ColorZ II 400G DWDM QSFP-DD ──────────────────────────────────────
|
||||||
|
{ pid: "COLORZII-400G-ZR", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 120000, reachLabel: "ZR", fiberType: "SMF", connector: "LC", wavelengths: "C-band", standard: "400ZR", notes: "Inphi COLORZ II 400G ZR QSFP-DD" },
|
||||||
|
{ pid: "COLORZII-400G-ZRP", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 1000000, reachLabel: "ZR+", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Inphi COLORZ II 400G ZR+ extended reach" },
|
||||||
|
{ pid: "COLORZII-400G-TUNE", formFactor: "QSFP-DD",speedGbps: 400, speed: "400G", reachMeters: 80000, reachLabel: "DWDM-TUNE",fiberType: "SMF", connector: "LC", wavelengths: "C-band tunable", notes: "Inphi COLORZ II 400G tunable" },
|
||||||
|
|
||||||
|
// ── Canopus 200G CFP2-DCO ─────────────────────────────────────────────
|
||||||
|
{ pid: "CANOPUS-200G-CFP2", formFactor: "CFP2-DCO",speedGbps: 200,speed: "200G", reachMeters: 1000000, reachLabel: "LH", fiberType: "SMF", connector: "LC", wavelengths: "C-band", notes: "Inphi Canopus 200G CFP2-DCO coherent" },
|
||||||
|
|
||||||
|
// ── Standard non-coherent (Inphi silicon photonics) ──────────────────
|
||||||
|
{ pid: "INPHI-QSFP28-100G-SR4", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 100, reachLabel: "SR4", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "100GBASE-SR4" },
|
||||||
|
{ pid: "INPHI-QSFP28-100G-LR4", formFactor: "QSFP28",speedGbps: 100,speed: "100G",reachMeters: 10000, reachLabel: "LR4", fiberType: "SMF", connector: "LC", wavelengths: "1295-1310nm", standard: "100GBASE-LR4" },
|
||||||
|
{ pid: "INPHI-QSFPDD-400G-DR4", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G",reachMeters: 500, reachLabel: "DR4", fiberType: "SMF", connector: "MPO", wavelengths: "1310nm", standard: "400GBASE-DR4" },
|
||||||
|
{ pid: "INPHI-QSFPDD-400G-FR4", formFactor: "QSFP-DD",speedGbps: 400,speed: "400G",reachMeters: 2000, reachLabel: "FR4", fiberType: "SMF", connector: "LC", wavelengths: "1271-1331nm", standard: "400GBASE-FR4" },
|
||||||
|
{ pid: "INPHI-OSFP-800G-SR8", formFactor: "OSFP", speedGbps: 800,speed: "800G",reachMeters: 100, reachLabel: "SR8", fiberType: "MMF", connector: "MPO", wavelengths: "850nm", standard: "800GBASE-SR8", notes: "Inphi/Marvell 800G SR8" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export async function scrapeInphiOem(): Promise<void> {
|
||||||
|
console.log("=== Inphi (Marvell) OEM Transceiver Seed ===\n");
|
||||||
|
|
||||||
|
const vendorId = await ensureVendor(
|
||||||
|
"Inphi (Marvell)",
|
||||||
|
"oem",
|
||||||
|
"https://www.marvell.com",
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
let inserted = 0, updated = 0, errors = 0;
|
||||||
|
|
||||||
|
for (const p of INPHI_PIDS) {
|
||||||
|
const slug = `inphi-${p.pid.toLowerCase().replace(/[^a-z0-9]+/g, "-")}`;
|
||||||
|
const category = p.reachMeters > 10000 ? "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=== Inphi OEM Seed Complete ===`);
|
||||||
|
console.log(` Inserted: ${inserted}, Updated: ${updated}, Errors: ${errors}`);
|
||||||
|
console.log(` Total PIDs: ${INPHI_PIDS.length}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
scrapeInphiOem()
|
||||||
|
.then(() => pool.end())
|
||||||
|
.catch((err) => { console.error("Fatal:", err); pool.end(); process.exit(1); });
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user