Anki/ts/reviewer/lib.test.ts
Damien Elmes 9f55cf26fc
Switch to SvelteKit (#3077)
* Update to latest Node LTS

* Add sveltekit

* Split tslib into separate @generated and @tslib components

SvelteKit's path aliases don't support multiple locations, so our old
approach of using @tslib to refer to both ts/lib and out/ts/lib will no
longer work. Instead, all generated sources and their includes are
placed in a separate out/ts/generated folder, and imported via @generated
instead. This also allows us to generate .ts files, instead of needing
to output separate .d.ts and .js files.

* Switch package.json to module type

* Avoid usage of baseUrl

Incompatible with SvelteKit

* Move sass into ts; use relative links

SvelteKit's default sass support doesn't allow overriding loadPaths

* jest->vitest, graphs example working with yarn dev

* most pages working in dev mode

* Some fixes after rebasing

* Fix/silence some svelte-check errors

* Get image-occlusion working with Fabric types

* Post-rebase lock changes

* Editor is now checked

* SvelteKit build integrated into ninja

* Use the new SvelteKit entrypoint for pages like congrats/deck options/etc

* Run eslint once for ts/**; fix some tests

* Fix a bunch of issues introduced when rebasing over latest main

* Run eslint fix

* Fix remaining eslint+pylint issues; tests now all pass

* Fix some issues with a clean build

* Latest bufbuild no longer requires @__PURE__ hack

* Add a few missed dependencies

* Add yarn.bat to fix Windows build

* Fix pages failing to show when ANKI_API_PORT not defined

* Fix svelte-check and vitest on Windows

* Set node path in ./yarn

* Move svelte-kit output to ts/.svelte-kit

Sadly, I couldn't figure out a way to store it in out/ if out/ is
a symlink, as it breaks module resolution when SvelteKit is run.

* Allow HMR inside Anki

* Skip SvelteKit build when HMR is defined

* Fix some post-rebase issues

I should have done a normal merge instead.
2024-03-31 09:16:31 +01:00

148 lines
5.6 KiB
TypeScript

// Copyright: Ankitects Pty Ltd and contributors
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
import { SchedulingStatesWithContext } from "@generated/anki/frontend_pb";
import { SchedulingContext, SchedulingStates } from "@generated/anki/scheduler_pb";
import { expect, test } from "vitest";
import { applyStateTransform } from "./answering";
/* eslint
@typescript-eslint/no-explicit-any: "off",
*/
function exampleInput(): SchedulingStatesWithContext {
return SchedulingStatesWithContext.fromJson(
{
"states": {
"current": {
"normal": {
"review": {
"scheduledDays": 1,
"elapsedDays": 2,
"easeFactor": 1.850000023841858,
"lapses": 4,
"leeched": false,
},
},
"customData": "{\"v\":\"v3.20.0\",\"seed\":2104,\"d\":5.39,\"s\":11.06}",
},
"again": {
"normal": {
"relearning": {
"review": {
"scheduledDays": 1,
"elapsedDays": 0,
"easeFactor": 1.649999976158142,
"lapses": 5,
"leeched": false,
},
"learning": {
"remainingSteps": 1,
"scheduledSecs": 600,
},
},
},
},
"hard": {
"normal": {
"review": {
"scheduledDays": 2,
"elapsedDays": 0,
"easeFactor": 1.7000000476837158,
"lapses": 4,
"leeched": false,
},
},
},
"good": {
"normal": {
"review": {
"scheduledDays": 4,
"elapsedDays": 0,
"easeFactor": 1.850000023841858,
"lapses": 4,
"leeched": false,
},
},
},
"easy": {
"normal": {
"review": {
"scheduledDays": 6,
"elapsedDays": 0,
"easeFactor": 2,
"lapses": 4,
"leeched": false,
},
},
},
},
"context": { "deckName": "hello", "seed": 123 },
},
);
}
test("can change oneof", () => {
let states = exampleInput().states!;
const jsonStates = states.toJson({ "emitDefaultValues": true }) as any;
// again should be a relearning state
const inner = states.again?.kind?.value?.kind;
assert(inner?.case === "relearning");
expect(inner.value.learning?.remainingSteps).toBe(1);
// change it to a review state
jsonStates.again.normal = { "review": jsonStates.again.normal.relearning.review };
states = SchedulingStates.fromJson(jsonStates);
const inner2 = states.again?.kind?.value?.kind;
assert(inner2?.case === "review");
// however, it's not valid to have multiple oneofs set
jsonStates.again.normal = { "review": jsonStates.again.normal.review, "learning": {} };
expect(() => {
SchedulingStates.fromJson(jsonStates);
}).toThrow();
});
test("no-op transform", async () => {
const input = exampleInput();
const output = await applyStateTransform(input, async (states: any, customData, ctx) => {
expect(ctx.deckName).toBe("hello");
expect(customData.easy.seed).toBe(2104);
expect(states!.again!.normal!.relearning!.learning!.remainingSteps).toBe(1);
});
// the input only has customData set on `current`, so we need to update it
// before we compare the two as equal
input.states!.again!.customData = input.states!.current!.customData;
input.states!.hard!.customData = input.states!.current!.customData;
input.states!.good!.customData = input.states!.current!.customData;
input.states!.easy!.customData = input.states!.current!.customData;
expect(output).toStrictEqual(input.states);
});
test("custom data change", async () => {
const output = await applyStateTransform(exampleInput(), async (_states: any, customData, _ctx) => {
customData.easy = { foo: "hello world" };
});
expect(output!.hard!.customData).not.toMatch(/hello world/);
expect(output!.easy!.customData).toBe("{\"foo\":\"hello world\"}");
});
test("adjust interval", async () => {
const output = await applyStateTransform(exampleInput(), async (states: any, _customData, _ctx) => {
states.good.normal.review.scheduledDays = 10;
});
const kind = output.good?.kind?.value?.kind;
assert(kind?.case === "review");
expect(kind.value.scheduledDays).toBe(10);
});
test("default context values exist", async () => {
const ctx = SchedulingContext.fromBinary(new Uint8Array());
expect(ctx.deckName).toBe("");
expect(ctx.seed).toBe(0n);
});
function assert(condition: boolean): asserts condition {
if (!condition) {
throw new Error();
}
}