Getting started with testeranto
So you want to add testeranto to your project?
1) add the testeranto
package to your project
npm install testeranto
or
yarn install testeranto
2) initializing your project
This command will scaffold out a config file and some output directories.
yarn tsx node_modules/testeranto/bin/init-docs.js
3) Create the subject of your test.
Every test needs a subject- the thing-being-tested, rather than the test itself. Any piece of ecmascript can be the subject of a testeranto test. For this example, lets create a Rectangle
class as the subject.
class Rectangle {
height: number;
width: number;
constructor(height: number = 2, width: number = 2) {
this.height = height;
this.width = width;
}
getHeight() {
return this.height;
}
getWidth() {
return this.width;
}
setHeight(height: number): void {
this.height = height;
}
setWidth(width: number): void {
this.width = width;
}
area(): number {
return this.width * this.height;
}
circumference(): number {
return 2 * (this.width + this.height);
}
}
export default Rectangle;
4) the test "shape"
Every testeranto test has an "shape" describing the necessary type signatures. In this example, we have the file Rectangle.test.shape.ts
export type IRectangleTestShape = {
iinput: Rectangle;
isubject: Rectangle;
istore: Rectangle;
iselection: Rectangle;
when: (rectangle: Rectangle) => any;
then: unknown;
given: (x) => (y) => unknown;
suites: {
Default: [string];
};
givens: {
Default;
WidthOfOneAndHeightOfOne;
WidthAndHeightOf: [number, number];
};
whens: {
HeightIsPubliclySetTo: [number];
WidthIsPubliclySetTo: [number];
setWidth: [number];
setHeight: [number];
};
thens: {
AreaPlusCircumference: [number];
circumference: [number];
getWidth: [number];
getHeight: [number];
area: [number];
prototype: [string];
};
checks: {
Default;
WidthOfOneAndHeightOfOne;
WidthAndHeightOf: [number, number];
};
};
5) The test "specification"
Every test has as specification which acts is the pure logic of a BDD test. This piece of code is gherkin-like Behavior Driven DSL which described the behavior of a piece of code, but none of it's implementation details. It is designed to be comprehensible to non-coding stakeholders.
export const RectangleTesterantoBaseTestSpecification: ITestSpecification<
IRectangleTestShape
> = (Suite, Given, When, Then, Check) => {
return [
Suite.Default(
"Testing the Rectangle class",
{
test0: Given.Default(
["https://api.github.com/repos/adamwong246/testeranto/issues/8"],
[When.setWidth(4), When.setHeight(9)],
[Then.getWidth(4), Then.getHeight(9)]
),
},
),
];
};
6) The test "interface"
Inverse of the "specifcation" is the interface. The "interface" is only implementation details, but no BDD logic.
export const RectangleTesterantoBaseInterface: IPartialInterface<IRectangleTestShape> =
{
beforeEach: async (subject, initializer, art, tr, initialValues) => {
return subject;
},
andWhen: async function (renderer, actioner) {
actioner(renderer);
return renderer;
},
butThen: (s, t, tr) => {
return t(s);
},
};
7) The test "implementation"
Every test has an implementation which acts is the "glue" between the interface and the specification.
export const RectangleTesterantoBaseTestImplementation: ITestImplementation<IRectangleTestShape, Rectangle> = {
suites: {
Default: "a default suite",
},
givens: {
Default: () => new Rectangle(2, 2),
WidthOfOneAndHeightOfOne: () => new Rectangle(1, 1),
WidthAndHeightOf: (width: number, height: number) => new Rectangle(width, height),
},
whens: {
HeightIsPubliclySetTo: (height: number) => async (rectangle: Rectangle, utils: PM) => {
rectangle.setHeight(height);
return rectangle;
},
WidthIsPubliclySetTo: (width: number) => async (rectangle: Rectangle, utils: PM) => {
rectangle.setWidth(width);
return rectangle;
},
setWidth: (width: number) => async (rectangle: Rectangle, utils: PM) => {
rectangle.setWidth(width);
return rectangle;
},
setHeight: (height: number) => async (rectangle: Rectangle, utils: PM) => {
rectangle.setHeight(height);
return rectangle;
},
},
thens: {
AreaPlusCircumference: (combined: number) => (rectangle: Rectangle) => {
assert.equal(rectangle.area() + rectangle.circumference(), combined);
return rectangle;
},
getWidth: (expectedWidth: number) => (rectangle: Rectangle) => {
assert.equal(rectangle.getWidth(), expectedWidth);
return rectangle;
},
getHeight: (expectedHeight: number) => (rectangle: Rectangle) => {
assert.equal(rectangle.getHeight(), expectedHeight);
return rectangle;
},
area: (area: number) => (rectangle: Rectangle) => {
assert.equal(rectangle.area(), area);
return rectangle;
},
prototype: (name: string) => (rectangle: Rectangle) => {
assert.equal(Object.getPrototypeOf(rectangle), Rectangle.prototype);
return rectangle;
},
circumference: (circumference: number) => (rectangle: Rectangle) => {
assert.equal(rectangle.circumference(), circumference);
return rectangle;
},
},
checks: {
/* @ts-ignore:next-line */
AnEmptyState: () => {
return {};
},
},
};
8) Create a test entrypoint
Every testeranto test has an entrypoint. In this example, we have the file Rectangle.test.web.ts
import Rectangle from "../Rectangle";
import { RectangleTesterantoBaseTestSpecification } from "../Rectangle.test.specification";
import { RectangleTesterantoBaseTestImplementation } from "../Rectangle.test.implementation";
import { RectangleTesterantoBasePrototype } from "../Rectangle.test";
export default Testeranto(
RectangleTesterantoBasePrototype,
RectangleTesterantoBaseTestSpecification,
RectangleTesterantoBaseTestImplementation,
{
beforeEach: async (
rectangleProto: Rectangle,
init: (c?: any) => (x: any) => (y: any) => Rectangle,
artificer: ITestArtificer,
tr: ITTestResourceConfiguration,
initialValues: any,
pm: PM_Web
): Promise<Rectangle> => {
pm.writeFileSync("beforeEachLog", "bar");
return rectangleProto;
},
afterAll: async (store, artificer, utils) => {
return new Promise(async (res, rej) => {
res(store);
});
},
andWhen: async function (
s: Rectangle,
whenCB: (s: Rectangle) => Promise<Rectangle>,
tr: ITTestResourceConfiguration,
utils: PM_Web
): Promise<Rectangle> {
return whenCB(s);
},
},
{
ports: 0,
}
);
9) Add your test to the config file
You need to Register your test in the config file
This will register our test to be run in the browser, with no needed ports and no sidecars.
...
tests: [
...
["./src/Rectangle/Rectangle.test.web.ts", "web", { ports: 0 }, []],
]
10) build your tests
tsx node_modules/testeranto/dist/prebuild/build-tests.mjs testeranto.mts
11) run your tests
tsx node_modules/testeranto/dist/prebuild/run-tests.mjs testeranto.mts
12) examine your test results
Running the tests should produce a series of file in docs
.
13) Run the aider prompt to auto-magically fix tests
aider --model deepseek --api-key deepseek=YOUR_SECRET_KEY --load docs/web/src/Rectangle/Rectangle.test.electron/prompt.txt
Aider should launch and being fixing any bugs!