Before putting a simulation into production it's important to confirm its trustworthiness and authenticate that it works as intended. This process of quality assurance is made up of two separate procedures:
Verification is the practice of ensuring that the simulation you've built is what you expected to build.
Like in traditional software engineering, you can use unit tests and integration tests to ensure that a simulation follows the stated requirements of a project. You can provide a specific state to an agent and check that it changes in the way you expect it to.
While there are a lot of ways to add unit tests to a simulation, a straightforward approach is to create a behavior attached to an agent that will run comparisons at specific time steps. For instance, in the example below, a behavior would run a test case on the first time step and tenth time step, asserting that properties stored on the agent's state are the expected values.
// test_behavior.js
function testCaseOne(state) {
if (state.test_case_one == state.actual_value) {
console.info("Test Case One Passes");
} else {
throw "Test Case One Fails";
}
}
function testCaseTwo(state) {
if (state.test_case_two == state.actual_value) {
console.info("Test Case Two Passes");
} else {
throw "Test Case Two Fails";
}
}
const behavior = (state, context) => {
switch (context.step()) {
case 1:
testCaseOne(state);
break;
case 10:
testCaseTwo(state);
break;
}
};
While verification will ensure that the simulation meets the desired specification, you'll also want to ensure that the desired specifications map to the real world.
A common way to validate a simulation is by comparing the results of a simulation run against external data. You can use HASH's dataset features to add data and visualize it against a simulation run. For example, in the Multi-Stage Cell Replication simulation, the validate.js behavior takes an external dataset and saves the values for a given time step to a value on state to then be visualized as a metric.
// validate.js
const data = context.data()[context.globals().dataset][state.counter];
const [red_data, yellow_data, green_data] = [
parseFloat(data[1]),
parseFloat(data[2]),
parseFloat(data[3]),
];
state.red_data = red_data;
state.yellow_data = yellow_data;
state.green_data = green_data;
state.cells_data = red_data + yellow_data + green_data;
Additionally you can use Complex Metrics to score the error difference between a simulation and an external dataset, to get a more quantitative validation measure.
Previous