// todo: use import assertions once they're supported by Node.js & ESLint // https://github.com/tc39/proposal-import-assertions import {createRequire} from 'module'; const require = createRequire(import.meta.url); import {parseHook} from '../../lib/profile-hooks.js'; import {parseAndAddLocationDHID} from '../vbb/parse-loc-dhid.js'; import {parseLocation as _parseLocation} from '../../parse/location.js'; import {parseArrival as _parseArrival} from '../../parse/arrival.js'; import {parseDeparture as _parseDeparture} from '../../parse/departure.js'; import {parseStopover as _parseStopover} from '../../parse/stopover.js'; import {parseJourneyLeg as _parseJourneyLeg} from '../../parse/journey-leg.js'; const baseProfile = require('./base.json'); import {products} from './products.js'; // todo: there's also a referenced icon `{"res":"occup_fig_{low,mid}"}` const addOccupancy = (item, occupancyCodes) => { const remIdx = (item.remarks || []) .findIndex(r => r.code && occupancyCodes.has(r.code)); if (remIdx < 0) { return; } const rem = item.remarks[remIdx]; item.occupancy = occupancyCodes.get(rem.code); item.remarks = [ ...item.remarks.slice(0, remIdx), ...item.remarks.slice(remIdx + 1), ]; }; const stopoverOccupancyCodes = new Map([ ['text.occup.loc.max.11', 'low'], ['text.occup.loc.max.12', 'medium'], ['text.occup.loc.max.13', 'high'], ]); const journeyLegOccupancyCodes = new Map([ ['text.occup.jny.max.11', 'low'], ['text.occup.jny.max.12', 'medium'], ['text.occup.jny.max.13', 'high'], ]); const parseLocation = ({parsed}, l) => { parseAndAddLocationDHID(parsed, l); return parsed; }; // todo: S45, S46? const ringbahnClockwise = /^ringbahn s\s?41$/i; const ringbahnAnticlockwise = /^ringbahn s\s?42$/i; const parseDepartureRenameRingbahn = ({parsed}, dep) => { if (parsed.line && parsed.line.product === 'suburban') { const d = parsed.direction && parsed.direction.trim(); if (ringbahnClockwise.test(d)) { parsed.direction = 'Ringbahn S41 ⟳'; } else if (ringbahnAnticlockwise.test(d)) { parsed.direction = 'Ringbahn S42 ⟲'; } } return parsed; }; const parseArrivalRenameRingbahn = ({parsed}, arr) => { if (parsed.line && parsed.line.product === 'suburban') { const p = parsed.provenance && parsed.provenance.trim(); if (ringbahnClockwise.test(p)) { parsed.provenance = 'Ringbahn S41 ⟳'; } else if (ringbahnAnticlockwise.test(p)) { parsed.provenance = 'Ringbahn S42 ⟲'; } } return parsed; }; const parseArrDepWithOccupancy = ({parsed}, d) => { addOccupancy(parsed, stopoverOccupancyCodes); return parsed; }; const parseStopoverWithOccupancy = ({parsed}, st, date) => { addOccupancy(parsed, stopoverOccupancyCodes); return parsed; }; const parseJourneyLegWithBerlkönig = (ctx, leg, date) => { if (leg.type === 'KISS') { const icon = ctx.common.icons[leg.icoX]; if (icon && icon.type === 'prod_berl') { const res = _parseJourneyLeg(ctx, { ...leg, type: 'WALK', }, date); delete res.walking; const mcp = leg.dep.mcp || {}; const mcpData = mcp.mcpData || {}; // todo: mcp.lid // todo: mcpData.occupancy, mcpData.type // todo: journey.trfRes.bkgData res.line = { type: 'line', id: null, // todo // todo: fahrtNr? name: mcpData.providerName, public: true, mode: 'taxi', product: 'berlkoenig', // todo: operator }; return res; } } return _parseJourneyLeg(ctx, leg, date); }; const parseJourneyLegWithOccupancy = ({parsed}, leg, date) => { if (leg.type === 'JNY') { addOccupancy(parsed, journeyLegOccupancyCodes); } return parsed; }; // use the Berlkönig ride sharing service? // todo: https://github.com/alexander-albers/tripkit/issues/26#issuecomment-825437320 const requestJourneysWithBerlkoenig = ({opt}, query) => { if ('numF' in query && opt.berlkoenig) { // todo: check if this is still true throw new Error('The `berlkoenig` and `results` options are mutually exclusive.'); } query.jnyFltrL.push({type: 'GROUP', mode: 'INC', value: 'OEV'}); if (opt.berlkoenig) { query.jnyFltrL.push({type: 'GROUP', mode: 'INC', value: 'BERLKOENIG'}); } query.gisFltrL = [{meta: 'foot_speed_normal', type: 'M', mode: 'FB'}]; return query; }; // todo: adapt/extend `vbb-parse-ticket` to support the BVG markup const profile = { ...baseProfile, locale: 'de-DE', timezone: 'Europe/Berlin', transformJourneysQuery: requestJourneysWithBerlkoenig, products, parseLocation: parseHook(_parseLocation, parseLocation), parseArrival: parseHook( parseHook(_parseArrival, parseArrivalRenameRingbahn), parseArrDepWithOccupancy, ), parseDeparture: parseHook( parseHook(_parseDeparture, parseDepartureRenameRingbahn), parseArrDepWithOccupancy, ), parseStopover: parseHook(_parseStopover, parseStopoverWithOccupancy), parseJourneyLeg: parseHook( parseJourneyLegWithBerlkönig, parseJourneyLegWithOccupancy, ), refreshJourneyUseOutReconL: true, trip: true, radar: true, refreshJourney: true, reachableFrom: true, }; export { profile, };