-
Notifications
You must be signed in to change notification settings - Fork 350
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Mock Widget for cleaning up Tests #2072
Conversation
@@ -235,6 +235,14 @@ export type PerseusTableRubric = { | |||
|
|||
export type PerseusTableUserInput = ReadonlyArray<ReadonlyArray<string>>; | |||
|
|||
export type PerseusMockWidgetRubric = { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note to self: I should ensure these are ordered alphabetically
@@ -330,6 +331,8 @@ export type MoleculeRendererWidget = WidgetOptions<'molecule-renderer', PerseusM | |||
export type RefTargetWidget = WidgetOptions<'passage-ref-target', PerseusPassageRefTargetWidgetOptions>; | |||
// prettier-ignore | |||
export type VideoWidget = WidgetOptions<'video', PerseusVideoWidgetOptions>; | |||
// prettier-ignore | |||
export type MockWidget = WidgetOptions<'mock-widget', MockWidgetOptions>; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note to self: I should ensure these are ordered alphabetically
npm Snapshot: PublishedGood news!! We've packaged up the latest commit from this PR (4f3fce4) and published it to npm. You Example: yarn add @khanacademy/perseus@PR2072 If you are working in Khan Academy's webapp, you can run: ./dev/tools/bump_perseus_version.sh -t PR2072 |
Size Change: -2 B (0%) Total Size: 1.45 MB
ℹ️ View Unchanged
|
@@ -0,0 +1,38 @@ | |||
import KhanAnswerTypes from "../../util/answer-types"; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will probably end up getting moved, or possible removed, as part of the Server Side Scoring project - which is fine!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I love this! Moving away from using production widgets for tests that aren't testing that widget should make for much less churn as we maintain/improve each widget!
|
||
/** | ||
* This is a Mock Perseus widget, which is used for our various rendering tests | ||
* across both Perseus and Webapp. It is a simple widget that renders an interactable |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you remove the reference to webapp (private repo)? I think it's enough to say it's for unit testing (both internally and in consuming projects)?
* | ||
* Please use this widget for all tests that are not specifically testing the | ||
* functionality of a particular widget, such as testing the rendering components. | ||
* This allows us to more easily update our widget schemas without needing to |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
* This allows us to more easily update our widget schemas without needing to | |
* This allows us to more easily update our widget schemas and behaviour without needing to |
arg2: string, | ||
arg3?: () => unknown | null | undefined, | ||
) => void = (path, newValue, cb) => { | ||
/* c8 ignore next */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we can remove this comment. We originally added these comments in a few, select places where we couldn't (or chose not to) unit test a line or block of code... this comment prevents the block from counting towards code coverage. I don't think we'll be too worried about test coverage of this mock widget as a whole.
getInputPaths: () => ReadonlyArray<ReadonlyArray<string>> = () => { | ||
// The widget itself is an input, so we return a single empty list to | ||
// indicate this. | ||
/* c8 ignore next */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
/* c8 ignore next */ |
|
||
const styles = StyleSheet.create({ | ||
widgetContainer: { | ||
color: "red", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good idea to make this kinda blatantly non-production!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice. I like the idea of moving these together under this folder.
// Act | ||
const input = "40"; | ||
const textbox = screen.getByRole("textbox"); | ||
await userEvent.click(textbox); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You can remove the .click()
call when using .type()
(it does that for you unless explicitly disabled)
packages/perseus/src/index.ts
Outdated
@@ -223,6 +223,7 @@ export type { | |||
CollinearTuple, | |||
MathFormat, | |||
InputNumberWidget, // TODO(jeremy): remove? | |||
NumericInputWidget, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is needing this export? In a perfect world, we wouldn't export any of our widgets and usage of them would all be funnelled through a Renderer
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh I should have caught this, thank you!
I changed some testdata
in perseus-editor
from InputNumber
to NumericInput
. It's nice to have the widget types there, but completely unneccessary.
In fact, that change also means we no longer need to export InputNumberWidget
either, so I can remove both! :) (I made sure to check Webapp and we're not using it there either.)
import type {PerseusRenderer} from "../../perseus-types"; | ||
|
||
export default { | ||
question: { | ||
content: "[[☃ mock-widget 1]] [[☃ mock-widget 2]]", | ||
images: {}, | ||
widgets: { | ||
"mock-widget 1": { | ||
type: "mock-widget", | ||
graded: true, | ||
options: { | ||
value: "5", | ||
}, | ||
}, | ||
"mock-widget 2": { | ||
type: "mock-widget", | ||
graded: true, | ||
options: { | ||
value: "6", | ||
}, | ||
}, | ||
}, | ||
} as PerseusRenderer, | ||
answerArea: { | ||
calculator: false, | ||
}, | ||
hints: [] as ReadonlyArray<any>, | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of casting the question
to PerseusRenderer
, can you type the entire thing as a PerseusItem
using satisfies
?
import type {PerseusRenderer} from "../../perseus-types"; | |
export default { | |
question: { | |
content: "[[☃ mock-widget 1]] [[☃ mock-widget 2]]", | |
images: {}, | |
widgets: { | |
"mock-widget 1": { | |
type: "mock-widget", | |
graded: true, | |
options: { | |
value: "5", | |
}, | |
}, | |
"mock-widget 2": { | |
type: "mock-widget", | |
graded: true, | |
options: { | |
value: "6", | |
}, | |
}, | |
}, | |
} as PerseusRenderer, | |
answerArea: { | |
calculator: false, | |
}, | |
hints: [] as ReadonlyArray<any>, | |
}; | |
import type {PerseusItem} from "../../perseus-types"; | |
export default { | |
question: { | |
content: "[[☃ mock-widget 1]] [[☃ mock-widget 2]]", | |
images: {}, | |
widgets: { | |
"mock-widget 1": { | |
type: "mock-widget", | |
graded: true, | |
options: { | |
value: "5", | |
}, | |
}, | |
"mock-widget 2": { | |
type: "mock-widget", | |
graded: true, | |
options: { | |
value: "6", | |
}, | |
}, | |
}, | |
}, | |
answerArea: { | |
calculator: false, | |
}, | |
hints: [] as ReadonlyArray<any>, | |
} satisfies PerseusItem; |
static: true, | ||
}, | ||
}, | ||
}); | ||
const cb = jest.fn(); | ||
|
||
// Act | ||
act(() => renderer.setInputValue(["input-number 2"], "1000", cb)); | ||
act(() => renderer.setInputValue(["mock-widget 2"], "1000", cb)); | ||
act(() => jest.runOnlyPendingTimers()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
With a simpler widget, we may be able to remove these jest.runOnlyPendingTimers()
calls. Might be worth a try to simplify these tests slightly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's actually one of the few calls that I cannot remove the jest call from, but I WAS able to remove them from 6 other locations in this file, plus the call in afterEach.
The tests all seems to run reliably without the ones I removed, but the remaining ones are all necessary for tests to pass, usually due to the use of _.defer() in how we handle blurs in the renderer.
@@ -153,13 +156,13 @@ describe("Perseus API", function () { | |||
|
|||
describe("CSS ClassNames", function () { | |||
describe("perseus-focused", function () { | |||
it("should be on an input-number exactly when focused", async function () { | |||
it("should be on an mock-widget exactly when focused", async function () { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: "an" => "a"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for all the work to clean up these loose ends (aka input number widget)!!
ac1bec6
to
830c45c
Compare
Summary:
This work was done as part of the Numeric Input project.
This PR introduces a new Mock Widget to be used for any tests that doesn't require testing the capabilities of a specific widget. I have updated as many tests and/or testdatas as I could find, except for a few key situations:
I have also updated as many occurrences of any remaining uses of
input-number
in our tests tonumeric-input
, in order to help smooth the path forward for the eventual return to the Input Number Conversion project.I have opted not to include the widget in our extra-widgets / general registration for now as it would require that I also create an editor for it, but a caveat of this approach means that this widget is not accessible for Webapp tests. I would be happy to hear any thoughts regarding this approach, as I wasn't sure if I should be globally registering this widget.
Issue: LEMS-2615
Test plan: