import {readFileSync, writeFileSync} from "fs"; import {Utils} from "../../../Utils"; import {TagRenderingConfigJson} from "../../../Models/ThemeConfig/Json/TagRenderingConfigJson"; import ScriptUtils from "../../../scripts/ScriptUtils"; import {LayerConfigJson} from "../../../Models/ThemeConfig/Json/LayerConfigJson"; function colonSplit(value: string): string[] { return value.split(";").map(v => v.replace(/"/g, '').trim().toLowerCase()).filter(s => s !== ""); } function loadCsv(file): { key: string, image: string, description: Map, countryWhiteList?: string[], countryBlackList?: string[], commonVoltages?: number[], commonCurrents?: number[], commonOutputs?: string[], associatedVehicleTypes?:string[] }[] { const entries: string[] = Utils.NoNull(readFileSync(file, "utf8").split("\n").map(str => str.trim())) const header = entries.shift().split(",") return Utils.NoNull(entries.map(entry => { const values = entry.split(",").map(str => str.trim()) if (values[0] === undefined || values[0] === "") { return undefined; } const v = {} const colonSeperated = ["commonVoltages", "commonOutputs", "commonCurrents", "countryWhiteList","countryBlackList","associatedVehicleTypes"] const descriptionTranslations = new Map() for (let j = 0; j < header.length; j++) { const key = header[j]; if (key.startsWith("description")) { const language = key.substring("description:".length) descriptionTranslations.set(language, values[j]) } if (colonSeperated.indexOf(key) >= 0) { v[key] = colonSplit(values[j]) } else { v[key] = values[j] } } v["description"] = descriptionTranslations return v; })) } // SMall script to output the properties of the types.csv as json to add in the tagRenderingGroup. Should be copied manually function run(file, protojson) { const overview_question_answers = [] const questions: (TagRenderingConfigJson & {"id": string})[] = [] const filterOptions: { question: any, osmTags?: string } [] = [ { question: { en: "All connectors", nl: "Alle types" } } ] const entries = loadCsv(file) for (let i = 0; i < entries.length; i++){ const e = entries[i]; const txt = { en: `
${e.description.get("en")}
`, nl: `
${e.description.get("nl")} 0 && e.countryBlackList.length > 0){ throw "Error for type "+e.key+": don't defined both a whitelist and a blacklist" } if (e.countryWhiteList.length > 0) { // This is a 'hideInAnswer', thus _reverse_ logic! const countries = e.countryWhiteList.map(country => "_country!=" + country) //HideInAnswer if it is in the wrong country json["hideInAnswer"] = {or: countries} }else if (e.countryBlackList .length > 0) { const countries = e.countryBlackList.map(country => "_country=" + country) //HideInAnswer if it is in the wrong country json["hideInAnswer"] = {or: countries} } if(e.associatedVehicleTypes?.length > 0 && e.associatedVehicleTypes.indexOf("*") < 0){ // This plug only occurs if some vehicle specific vehicle type is present. // IF all of the needed vehicle types are explicitly NO, then we hide this type as well let hideInAnswer : any = {and: [].concat(...e.associatedVehicleTypes.map(neededVehicle => [neededVehicle+"~*", neededVehicle+"!=yes"]))} if(json["hideInAnswer"] !== undefined){ hideInAnswer = {or: [json["hideInAnswer"], hideInAnswer]} } json["hideInAnswer"] = hideInAnswer } overview_question_answers.push(json) // We add a second time for any amount to trigger a visualisation; but this is not an answer option const no_ask_json = { if: { and: [`${e.key}~*`, `${e.key}!=1`] }, then: txt, hideInAnswer: true } overview_question_answers.push(no_ask_json) const descrWithImage_en = `${e.description.get("en")} ` const descrWithImage_nl = `${e.description.get("nl")} ` questions.push({ "id":"plugs-"+i, question: { en: `How much plugs of type ${descrWithImage_en} are available here?`, nl: `Hoeveel stekkers van type ${descrWithImage_nl} heeft dit oplaadpunt?`, }, render: { en: `There are ${descrWithImage_en} plugs of type ${e.description.get("en")} available here`, nl: `Hier zijn ${descrWithImage_nl} stekkers van het type ${e.description.get("nl")}` }, freeform: { key: e.key, type: "pnat" }, condition: { and: [`${e.key}~*`, `${e.key}!=0`] } }) questions.push({ "id":"voltage-"+i, question: { en: `What voltage do the plugs with ${descrWithImage_en} offer?`, nl: `Welke spanning levert de stekker van type ${descrWithImage_nl}` }, render: { en: `${descrWithImage_en} outputs {${e.key}:voltage} volt`, nl: `${descrWithImage_nl} heeft een spanning van {${e.key}:voltage} volt` }, freeform: { key: `${e.key}:voltage`, type: "pfloat" }, mappings: e.commonVoltages.map(voltage => { return { if: `socket:${e.key}:voltage=${voltage} V`, then: { en: `${descrWithImage_en} outputs ${voltage} volt`, nl: `${descrWithImage_nl} heeft een spanning van ${voltage} volt` } } }), condition: { and: [`${e.key}~*`, `${e.key}!=0`] } }) questions.push({ "id":"current-"+i, question: { en: `What current do the plugs with ${descrWithImage_en} offer?`, nl: `Welke stroom levert de stekker van type ${descrWithImage_nl}?`, }, render: { en: `${descrWithImage_en} outputs at most {${e.key}:current}A`, nl: `${descrWithImage_nl} levert een stroom van maximaal {${e.key}:current}A` }, freeform: { key: `${e.key}:current`, type: "pfloat" }, mappings: e.commonCurrents.map(current => { return { if: `socket:${e.key}:current=${current} A`, then: { en: `${descrWithImage_en} outputs at most ${current} A`, nl: `${descrWithImage_nl} levert een stroom van maximaal ${current} A` } } }), condition: { and: [`${e.key}~*`, `${e.key}!=0`] } }) questions.push({ "id":"power-output-"+i, question: { en: `What power output does a single plug of type ${descrWithImage_en} offer?`, nl: `Welk vermogen levert een enkele stekker van type ${descrWithImage_nl}?`, }, render: { en: `${descrWithImage_en} outputs at most {${e.key}:output}`, nl: `${descrWithImage_nl} levert een vermogen van maximaal {${e.key}:output}` }, freeform: { key: `${e.key}:output`, type: "pfloat" }, mappings: e.commonOutputs.map(output => { return { if: `socket:${e.key}:output=${output}`, then: { en: `${descrWithImage_en} outputs at most ${output}`, nl: `${descrWithImage_nl} levert een vermogen van maximaal ${output}` } } }), condition: { and: [`${e.key}~*`, `${e.key}!=0`] } }) filterOptions.push({ question: { en: `Has a ${descrWithImage_en} connector`, nl: `Heeft een ${descrWithImage_nl}` }, osmTags: `${e.key}~*` }) } const toggles = { "id":"Available_charging_stations (generated)", "question": { "en": "Which charging stations are available here?" }, "multiAnswer": true, "mappings": overview_question_answers } questions.unshift(toggles) const stringified = questions.map(q => JSON.stringify(q, null, " ")) let protoString = readFileSync(protojson, "utf8") protoString = protoString.replace("{\"id\": \"$$$\"}", stringified.join(",\n")) const proto = JSON.parse(protoString) proto.tagRenderings.forEach(tr => { if(typeof tr === "string"){ return; } if(tr["id"] === undefined || typeof tr["id"] !== "string"){ console.error(tr) throw "Every tagrendering should have an id, acting as comment" } }) proto["filter"].push({ id:"connection_type", options: filterOptions }) const extraUnits = [ { appliesToKey: entries.map(e => e.key + ":voltage"), applicableUnits: [{ canonicalDenomination: 'V', alternativeDenomination: ["v", "volt", "voltage",'V','Volt'], human: { en: "Volts", nl: "volt" } }], eraseInvalidValues: true }, { appliesToKey: entries.map(e => e.key + ":current"), applicableUnits: [{ canonicalDenomination: 'A', alternativeDenomination: ["a", "amp", "amperage",'A'], human: { en: "A", nl: "A" } }], eraseInvalidValues: true }, { appliesToKey: entries.map(e => e.key + ":output"), applicableUnits: [{ canonicalDenomination: 'kW', alternativeDenomination: ["kilowatt"], human: { en: "kilowatt", nl: "kilowatt" } }, { canonicalDenomination: 'mW', alternativeDenomination: ["megawatt"], human: { en: "megawatt", nl: "megawatt" } }], eraseInvalidValues: true }, ]; if(proto["units"] == undefined){ proto["units"] = [] } proto["units"].push(...extraUnits) // mergeTranslations("charging_station.json",proto) writeFileSync("charging_station.json", JSON.stringify(proto, undefined, " ")) } async function queryTagInfo(file, type, clean: ((s: string) => string)) { for (const e of loadCsv(file)) { const value = await ScriptUtils.TagInfoHistogram(e.key + ":" + type) const result = value.data const counts = new Map() // console.log(result) for (const r of result) { let key = r.value; key = clean(key) key.trim(); if (key.indexOf('-') >= 0) { continue } if (r.fraction < 0.05) { continue } if (counts.has(key)) { counts.set(key, counts.get(key) + r.count) } else { counts.set(key, r.count) } } const countsArray = Array.from(counts.keys()) countsArray.sort() console.log(`${e.key}:${type} = ${countsArray.join(";")}`) } } /** * Adds the translations into the 'newConfig' object * @param origPath * @param newConfig */ function mergeTranslations(origPath, newConfig: LayerConfigJson){ const oldFile = JSON.parse(readFileSync(origPath, "utf-8")) const newFile = newConfig const renderingsOld = oldFile.tagRenderings delete oldFile.tagRenderings const newRenderings = newFile.tagRenderings Utils.Merge(oldFile, newFile) for (const oldRendering of renderingsOld) { const oldRenderingName = oldRendering["id"] if(oldRenderingName === undefined){ continue } const applicable = newRenderings.filter(r => r["id"] === oldRenderingName)[0] if(applicable === undefined){ continue; } Utils.Merge(oldRendering, applicable) } } try { console.log("Generating the charging_station.json file") run("types.csv", "charging_station.protojson") /*/ queryTagInfo("types.csv","voltage", s => s.trim()) queryTagInfo("types.csv", "current", s => s.trim()) queryTagInfo("types.csv", "output", s => { if(s.endsWith("kW")){ s = s.substring(0, s.length - 2) } s = s.trim() return s + " kW" }) //*/ } catch (e) { console.error(e) }