Compare commits
2 Commits
8397571c11
...
1b37af82e3
Author | SHA1 | Date |
---|---|---|
|
1b37af82e3 | 3 months ago |
|
86aababf33 | 3 months ago |
@ -1,104 +0,0 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { Buses, UnitCommitmentScenario } from "../fixtures";
|
||||
import { ValidationError } from "../Validation/validate";
|
||||
import { generateTimeslots } from "../../components/Common/Forms/DataTable";
|
||||
|
||||
export const generateUniqueName = (container: any, prefix: string): string => {
|
||||
let counter = 1;
|
||||
let name = `${prefix}${counter}`;
|
||||
while (name in container) {
|
||||
counter++;
|
||||
name = `${prefix}${counter}`;
|
||||
}
|
||||
return name;
|
||||
};
|
||||
|
||||
export const createBus = (scenario: UnitCommitmentScenario) => {
|
||||
const name = generateUniqueName(scenario.Buses, "b");
|
||||
const timeslots = generateTimeslots(scenario);
|
||||
return {
|
||||
...scenario,
|
||||
Buses: {
|
||||
...scenario.Buses,
|
||||
[name]: {
|
||||
"Load (MW)": Array(timeslots.length).fill(0),
|
||||
},
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
export const changeBusData = (
|
||||
bus: string,
|
||||
field: string,
|
||||
newValueStr: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
// Load (MW)
|
||||
const match = field.match(/Load \(MW\) (\d+):(\d+)/);
|
||||
if (match) {
|
||||
const newValueFloat = parseFloat(newValueStr);
|
||||
if (isNaN(newValueFloat)) {
|
||||
return [scenario, { message: `Invalid value: ${newValueStr}` }];
|
||||
}
|
||||
|
||||
// Convert HH:MM to offset
|
||||
const hours = parseInt(match[1]!, 10);
|
||||
const min = parseInt(match[2]!, 10);
|
||||
const idx = (hours * 60 + min) / scenario.Parameters["Time step (min)"];
|
||||
|
||||
const newLoad = [...scenario.Buses[bus]!["Load (MW)"]];
|
||||
newLoad[idx] = newValueFloat;
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Buses: {
|
||||
...scenario.Buses,
|
||||
[bus]: {
|
||||
"Load (MW)": newLoad,
|
||||
},
|
||||
},
|
||||
},
|
||||
null,
|
||||
];
|
||||
}
|
||||
|
||||
throw Error(`Unknown field: ${field}`);
|
||||
};
|
||||
|
||||
export const deleteBus = (bus: string, scenario: UnitCommitmentScenario) => {
|
||||
const { [bus]: _, ...newBuses } = scenario.Buses;
|
||||
return {
|
||||
...scenario,
|
||||
Buses: newBuses,
|
||||
};
|
||||
};
|
||||
|
||||
export const renameBus = (
|
||||
oldName: string,
|
||||
newName: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
if (newName in scenario.Buses) {
|
||||
return [scenario, { message: `Bus ${newName} already exists` }];
|
||||
}
|
||||
const newBuses: Buses = Object.keys(scenario.Buses).reduce((acc, val) => {
|
||||
if (val === oldName) {
|
||||
acc[newName] = scenario.Buses[val]!;
|
||||
} else {
|
||||
acc[val] = scenario.Buses[val]!;
|
||||
}
|
||||
return acc;
|
||||
}, {} as Buses);
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Buses: newBuses,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
@ -0,0 +1,70 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { Buses, UnitCommitmentScenario } from "../fixtures";
|
||||
import { ValidationError } from "../Validation/validate";
|
||||
import { generateTimeslots } from "../../components/Common/Forms/DataTable";
|
||||
import {
|
||||
changeData,
|
||||
generateUniqueName,
|
||||
renameItemInObject,
|
||||
} from "./commonOps";
|
||||
import { BusesColumnSpec } from "../../components/CaseBuilder/Buses/Buses";
|
||||
|
||||
export const createBus = (scenario: UnitCommitmentScenario) => {
|
||||
const name = generateUniqueName(scenario.Buses, "b");
|
||||
const timeslots = generateTimeslots(scenario);
|
||||
return {
|
||||
...scenario,
|
||||
Buses: {
|
||||
...scenario.Buses,
|
||||
[name]: {
|
||||
"Load (MW)": Array(timeslots.length).fill(0),
|
||||
},
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
export const changeBusData = (
|
||||
bus: string,
|
||||
field: string,
|
||||
newValueStr: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const [newBus, err] = changeData(
|
||||
field,
|
||||
newValueStr,
|
||||
scenario.Buses[bus]!,
|
||||
BusesColumnSpec,
|
||||
scenario,
|
||||
);
|
||||
if (err) return [scenario, err];
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Buses: {
|
||||
...scenario.Buses,
|
||||
[bus]: newBus,
|
||||
} as Buses,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const deleteBus = (bus: string, scenario: UnitCommitmentScenario) => {
|
||||
const { [bus]: _, ...newBuses } = scenario.Buses;
|
||||
return { ...scenario, Buses: newBuses };
|
||||
};
|
||||
|
||||
export const renameBus = (
|
||||
oldName: string,
|
||||
newName: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const [newBuses, err] = renameItemInObject(oldName, newName, scenario.Buses);
|
||||
if (err) return [scenario, err];
|
||||
return [{ ...scenario, Buses: newBuses }, null];
|
||||
};
|
@ -0,0 +1,157 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { ValidationError } from "../Validation/validate";
|
||||
import { UnitCommitmentScenario } from "../fixtures";
|
||||
import { ColumnSpec } from "../../components/Common/Forms/DataTable";
|
||||
|
||||
export const renameItemInObject = <T>(
|
||||
oldName: string,
|
||||
newName: string,
|
||||
container: { [key: string]: T },
|
||||
): [{ [key: string]: T }, ValidationError | null] => {
|
||||
if (newName in container) {
|
||||
return [container, { message: `${newName} already exists` }];
|
||||
}
|
||||
const newContainer = Object.keys(container).reduce(
|
||||
(acc, val) => {
|
||||
if (val === oldName) {
|
||||
acc[newName] = container[val]!;
|
||||
} else {
|
||||
acc[val] = container[val]!;
|
||||
}
|
||||
return acc;
|
||||
},
|
||||
{} as { [key: string]: T },
|
||||
);
|
||||
return [newContainer, null];
|
||||
};
|
||||
|
||||
export const generateUniqueName = (container: any, prefix: string): string => {
|
||||
let counter = 1;
|
||||
let name = `${prefix}${counter}`;
|
||||
while (name in container) {
|
||||
counter++;
|
||||
name = `${prefix}${counter}`;
|
||||
}
|
||||
return name;
|
||||
};
|
||||
|
||||
const parseNumber = (valueStr: string): [number, ValidationError | null] => {
|
||||
const valueFloat = parseFloat(valueStr);
|
||||
if (isNaN(valueFloat)) {
|
||||
return [0, { message: `"${valueStr}" is not a valid number` }];
|
||||
} else {
|
||||
return [valueFloat, null];
|
||||
}
|
||||
};
|
||||
|
||||
export const changeStringData = (
|
||||
field: string,
|
||||
newValue: string,
|
||||
container: { [key: string]: any },
|
||||
): [{ [key: string]: any }, ValidationError | null] => {
|
||||
return [
|
||||
{
|
||||
...container,
|
||||
[field]: newValue,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const changeBusRefData = (
|
||||
field: string,
|
||||
newValue: string,
|
||||
container: { [key: string]: any },
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [{ [key: string]: any }, ValidationError | null] => {
|
||||
if (!(newValue in scenario.Buses)) {
|
||||
return [scenario, { message: `Bus "${newValue}" does not exist` }];
|
||||
}
|
||||
return changeStringData(field, newValue, container);
|
||||
};
|
||||
|
||||
export const changeNumberData = (
|
||||
field: string,
|
||||
newValueStr: string,
|
||||
container: { [key: string]: any },
|
||||
): [{ [key: string]: any }, ValidationError | null] => {
|
||||
// Parse value
|
||||
const [newValueFloat, err] = parseNumber(newValueStr);
|
||||
if (err) return [container, err];
|
||||
|
||||
// Build the new object
|
||||
return [
|
||||
{
|
||||
...container,
|
||||
[field]: newValueFloat,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const changeNumberVecData = (
|
||||
field: string,
|
||||
time: string,
|
||||
newValueStr: string,
|
||||
container: { [key: string]: any },
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [{ [key: string]: any }, ValidationError | null] => {
|
||||
// Parse value
|
||||
const [newValueFloat, err] = parseNumber(newValueStr);
|
||||
if (err) return [container, err];
|
||||
|
||||
// Convert HH:MM to offset
|
||||
const hours = parseInt(time.split(":")[0]!, 10);
|
||||
const min = parseInt(time.split(":")[1]!, 10);
|
||||
const idx = (hours * 60 + min) / scenario.Parameters["Time step (min)"];
|
||||
|
||||
// Build the new vector
|
||||
const newVec = [...container[field]];
|
||||
newVec[idx] = newValueFloat;
|
||||
return [
|
||||
{
|
||||
...container,
|
||||
[field]: newVec,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const changeData = (
|
||||
field: string,
|
||||
newValueStr: string,
|
||||
container: { [key: string]: any },
|
||||
colSpecs: ColumnSpec[],
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [{ [key: string]: any }, ValidationError | null] => {
|
||||
const match = field.match(/^([^0-9]+)(\d+:\d+)?$/);
|
||||
const fieldName = match![1]!.trim();
|
||||
const fieldTime = match![2];
|
||||
for (const spec of colSpecs) {
|
||||
if (spec.title !== fieldName) continue;
|
||||
switch (spec.type) {
|
||||
case "string":
|
||||
return changeStringData(fieldName, newValueStr, container);
|
||||
case "busRef":
|
||||
return changeBusRefData(fieldName, newValueStr, container, scenario);
|
||||
case "number":
|
||||
return changeNumberData(fieldName, newValueStr, container);
|
||||
case "number[]":
|
||||
return changeNumberVecData(
|
||||
fieldName,
|
||||
fieldTime!,
|
||||
newValueStr,
|
||||
container,
|
||||
scenario,
|
||||
);
|
||||
default:
|
||||
throw Error(`Unknown type: ${spec.type}`);
|
||||
}
|
||||
}
|
||||
throw Error(`Unknown field: ${fieldName}`);
|
||||
};
|
@ -0,0 +1,127 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { TEST_DATA_1, TEST_DATA_BLANK } from "../fixtures.test";
|
||||
import assert from "node:assert";
|
||||
import {
|
||||
changeProfiledUnitData,
|
||||
createProfiledUnit,
|
||||
deleteGenerator,
|
||||
renameGenerator,
|
||||
} from "./generatorOps";
|
||||
|
||||
test("createProfiledUnit", () => {
|
||||
const [newScenario, err] = createProfiledUnit(TEST_DATA_1);
|
||||
assert(err === null);
|
||||
assert.deepEqual(newScenario.Generators, {
|
||||
pu1: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 12.5,
|
||||
"Maximum power (MW)": [10, 12, 13, 15, 20],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
pu2: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 120,
|
||||
"Maximum power (MW)": [50, 50, 50, 50, 50],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
pu3: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 0,
|
||||
"Maximum power (MW)": [0, 0, 0, 0, 0],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test("createProfiledUnit with blank file", () => {
|
||||
const [, err] = createProfiledUnit(TEST_DATA_BLANK);
|
||||
assert(err !== null);
|
||||
assert.equal(err.message, "Profiled unit requires an existing bus.");
|
||||
});
|
||||
|
||||
test("changeProfiledUnitData", () => {
|
||||
let scenario = TEST_DATA_1;
|
||||
let err = null;
|
||||
[scenario, err] = changeProfiledUnitData(
|
||||
"pu1",
|
||||
"Cost ($/MW)",
|
||||
"99",
|
||||
scenario,
|
||||
);
|
||||
assert.equal(err, null);
|
||||
[scenario, err] = changeProfiledUnitData(
|
||||
"pu1",
|
||||
"Maximum power (MW) 03:00",
|
||||
"99",
|
||||
scenario,
|
||||
);
|
||||
assert.equal(err, null);
|
||||
[scenario, err] = changeProfiledUnitData("pu2", "Bus", "b3", scenario);
|
||||
assert.equal(err, null);
|
||||
assert.deepEqual(scenario.Generators, {
|
||||
pu1: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 99,
|
||||
"Maximum power (MW)": [10, 12, 13, 99, 20],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
pu2: {
|
||||
Bus: "b3",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 120,
|
||||
"Maximum power (MW)": [50, 50, 50, 50, 50],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test("changeProfiledUnitData with invalid bus", () => {
|
||||
let scenario = TEST_DATA_1;
|
||||
let err = null;
|
||||
[scenario, err] = changeProfiledUnitData("pu1", "Bus", "b99", scenario);
|
||||
assert(err !== null);
|
||||
assert.equal(err.message, 'Bus "b99" does not exist');
|
||||
});
|
||||
|
||||
test("deleteGenerator", () => {
|
||||
const newScenario = deleteGenerator("pu1", TEST_DATA_1);
|
||||
assert.deepEqual(newScenario.Generators, {
|
||||
pu2: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 120,
|
||||
"Maximum power (MW)": [50, 50, 50, 50, 50],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test("renameGenerator", () => {
|
||||
const [newScenario, err] = renameGenerator("pu1", "pu5", TEST_DATA_1);
|
||||
assert(err === null);
|
||||
assert.deepEqual(newScenario.Generators, {
|
||||
pu5: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 12.5,
|
||||
"Maximum power (MW)": [10, 12, 13, 15, 20],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
pu2: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 120,
|
||||
"Maximum power (MW)": [50, 50, 50, 50, 50],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
});
|
||||
});
|
@ -0,0 +1,90 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { Generators, UnitCommitmentScenario } from "../fixtures";
|
||||
import { generateTimeslots } from "../../components/Common/Forms/DataTable";
|
||||
import { ValidationError } from "../Validation/validate";
|
||||
import {
|
||||
changeData,
|
||||
generateUniqueName,
|
||||
renameItemInObject,
|
||||
} from "./commonOps";
|
||||
import { ProfiledUnitsColumnSpec } from "../../components/CaseBuilder/ProfiledUnits/ProfiledUnits";
|
||||
|
||||
export const createProfiledUnit = (
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const busNames = Object.keys(scenario.Buses);
|
||||
if (busNames.length === 0) {
|
||||
return [scenario, { message: "Profiled unit requires an existing bus." }];
|
||||
}
|
||||
const timeslots = generateTimeslots(scenario);
|
||||
const name = generateUniqueName(scenario.Generators, "pu");
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Generators: {
|
||||
...scenario.Generators,
|
||||
[name]: {
|
||||
Bus: busNames[0]!,
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 0,
|
||||
"Minimum power (MW)": Array(timeslots.length).fill(0),
|
||||
"Maximum power (MW)": Array(timeslots.length).fill(0),
|
||||
},
|
||||
},
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const changeProfiledUnitData = (
|
||||
generator: string,
|
||||
field: string,
|
||||
newValueStr: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const [newGen, err] = changeData(
|
||||
field,
|
||||
newValueStr,
|
||||
scenario.Generators[generator]!,
|
||||
ProfiledUnitsColumnSpec,
|
||||
scenario,
|
||||
);
|
||||
if (err) return [scenario, err];
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Generators: {
|
||||
...scenario.Generators,
|
||||
[generator]: newGen,
|
||||
} as Generators,
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
||||
|
||||
export const deleteGenerator = (
|
||||
name: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): UnitCommitmentScenario => {
|
||||
const { [name]: _, ...newGenerators } = scenario.Generators;
|
||||
return { ...scenario, Generators: newGenerators };
|
||||
};
|
||||
|
||||
export const renameGenerator = (
|
||||
oldName: string,
|
||||
newName: string,
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const [newGen, err] = renameItemInObject(
|
||||
oldName,
|
||||
newName,
|
||||
scenario.Generators,
|
||||
);
|
||||
if (err) return [scenario, err];
|
||||
return [{ ...scenario, Generators: newGen }, null];
|
||||
};
|
@ -1,36 +0,0 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { TEST_DATA_1, TEST_DATA_BLANK } from "../fixtures.test";
|
||||
import assert from "node:assert";
|
||||
import { createProfiledUnit } from "./profiledUnitOps";
|
||||
|
||||
test("createUnit", () => {
|
||||
const [newScenario, err] = createProfiledUnit(TEST_DATA_1);
|
||||
assert(err === null);
|
||||
assert.deepEqual(newScenario.Generators, {
|
||||
pu1: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 12.5,
|
||||
"Maximum power (MW)": [10, 12, 13, 15, 20],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
pu2: {
|
||||
Bus: "b1",
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 0,
|
||||
"Maximum power (MW)": [0, 0, 0, 0, 0],
|
||||
"Minimum power (MW)": [0, 0, 0, 0, 0],
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
test("createUnit with blank file", () => {
|
||||
const [newScenario, err] = createProfiledUnit(TEST_DATA_BLANK);
|
||||
assert(err !== null);
|
||||
assert.equal(err.message, "Profiled unit requires an existing bus.");
|
||||
});
|
@ -1,37 +0,0 @@
|
||||
/*
|
||||
* UnitCommitment.jl: Optimization Package for Security-Constrained Unit Commitment
|
||||
* Copyright (C) 2020-2025, UChicago Argonne, LLC. All rights reserved.
|
||||
* Released under the modified BSD license. See COPYING.md for more details.
|
||||
*/
|
||||
|
||||
import { UnitCommitmentScenario } from "../fixtures";
|
||||
import { generateTimeslots } from "../../components/Common/Forms/DataTable";
|
||||
import { generateUniqueName } from "./busOperations";
|
||||
import { ValidationError } from "../Validation/validate";
|
||||
|
||||
export const createProfiledUnit = (
|
||||
scenario: UnitCommitmentScenario,
|
||||
): [UnitCommitmentScenario, ValidationError | null] => {
|
||||
const busNames = Object.keys(scenario.Buses);
|
||||
if (busNames.length === 0) {
|
||||
return [scenario, { message: "Profiled unit requires an existing bus." }];
|
||||
}
|
||||
const timeslots = generateTimeslots(scenario);
|
||||
const name = generateUniqueName(scenario.Generators, "pu");
|
||||
return [
|
||||
{
|
||||
...scenario,
|
||||
Generators: {
|
||||
...scenario.Generators,
|
||||
[name]: {
|
||||
Bus: busNames[0]!,
|
||||
Type: "Profiled",
|
||||
"Cost ($/MW)": 0,
|
||||
"Minimum power (MW)": Array(timeslots.length).fill(0),
|
||||
"Maximum power (MW)": Array(timeslots.length).fill(0),
|
||||
},
|
||||
},
|
||||
},
|
||||
null,
|
||||
];
|
||||
};
|
Loading…
Reference in new issue