DEV Community

Cover image for Good testing practices with πŸ¦” Angular Testing Library
Tim Deschryver for Angular

Posted on • Edited on • Originally published at timdeschryver.dev

Good testing practices with πŸ¦” Angular Testing Library

Angular Testing Library provides utility functions to interact with Angular components, in the same way as a user would. This brings more maintainability to our tests, gives us more confidence that our component does what it's supposed to do, and it improves the accessibility which is better for our users. All these benefits, plus you'll see that it's fun to write tests in this way.

The Angular Testing Library

Angular Testing Library is a part of the @testing-library family with πŸ¦‘ DOM Testing Library in the center of the family.
We're encouraging good testing practices across multiple frameworks and libraries by bringing a similar API to the table.
These tests can be written in the test runner of your liking.

We encourage:

  • maintainable tests: we do not want to test implementation details
  • confidence in our components: you interact with your components the same way as your end-users
  • accessibility: we want inclusive components

The more your tests resemble the way your software is used, the more confidence they can give you.

Getting Started

To get started, the first step is to install @testing-library/angular, after that we're good to go.

npm install --save-dev @testing-library/angular
Enter fullscreen mode Exit fullscreen mode

In this post, we'll take an introduction by writing tests for a feedback form, starting very simple and keep building on top of it.

The form we'll put under test has a required name field, a required rating field that must be between 0 and 10, a summary field, and a select box to select a t-shirt size. A form wouldn't be a form without a submit button, so let's add that too.

The code of the forms looks as follows.

export class FeedbackComponent {
  @Input() shirtSizes: string[] = []
  @Output() submitForm = new EventEmitter<Feedback>()

  form = this.formBuilder.group({
    name: ['', [Validators.required]],
    rating: ['', [Validators.required, Validators.min(0), Validators.max(10)]],
    description: [''],
    shirtSize: ['', [Validators.required]],
  })

  nameControl = this.form.get('name')
  ratingControl = this.form.get('rating')
  shirtSizeControl = this.form.get('shirtSize')

  constructor(private formBuilder: FormBuilder) {}

  submit() {
    if (this.form.valid) {
      this.submitForm.emit(this.form.value)
    }
  }
}
Enter fullscreen mode Exit fullscreen mode
<form [formGroup]="form" (ngSubmit)="submit()">
  <legend>Feedback form</legend>

  <mat-form-field>
    <mat-label>Name</mat-label>
    <input matInput type="text" formControlName="name" />
    <mat-error *ngIf="nameControl.hasError('required')">
      Name is required
    </mat-error>
  </mat-form-field>

  <mat-form-field>
    <mat-label>Rating</mat-label>
    <input matInput type="number" formControlName="rating" />
    <mat-error *ngIf="ratingControl.hasError('required')">
      Rating is required
    </mat-error>
    <mat-error
      *ngIf="ratingControl.hasError('min') || ratingControl.hasError('max')"
    >
      Rating must be between 0 and 10
    </mat-error>
  </mat-form-field>

  <mat-form-field>
    <mat-label>Description</mat-label>
    <textarea matInput formControlName="description"></textarea>
  </mat-form-field>

  <mat-form-field>
    <mat-label>T-shirt size</mat-label>
    <mat-select placeholder="Select" formControlName="shirtSize">
      <mat-option *ngFor="let size of shirtSizes" [value]="size"
        >{{ size }}</mat-option
      >
    </mat-select>
    <mat-error *ngIf="shirtSizeControl.hasError('required')">
      T-shirt size is required
    </mat-error>
  </mat-form-field>

  <button type="submit" mat-stroked-button color="primary">
    Submit your feedback
  </button>
</form>
Enter fullscreen mode Exit fullscreen mode

Our first test

To be able to test the feedback component we must be able to render the component, we can do this by using the render function.
The render function takes the component under test as the first argument and has an optional second argument for more options see RenderOptions, which we'll be covering soon.

import { render } from '@testing-library/angular'

it('should render the form' async () => {
  const component = await render(FeedbackComponent)
})
Enter fullscreen mode Exit fullscreen mode

It doesn't have to be more than this as setup to test a simple component

In this case, it throws an exception because we're making use of reactive forms and some of the Angular Material components.
To solve this we must provide both of the missing modules. To provide these modules we use the imports property on the RenderOptions, similar to how TestBed.configureTestingModule works.

it('should render the form', async () => {
  const component = await render(FeedbackComponent, {
    imports: [ReactiveFormsModule, MaterialModule],
  })
})
Enter fullscreen mode Exit fullscreen mode

Now, this test works.

Queries

The render function returns a RenderResult object which contains utility functions to test the component.

You'll notice that we test components in a similar way just as an an end-user would do.
We don't test implementation details but Angular Testing Library gives us an API to test the component from the outside, via the component DOM nodes.

To verify the nodes an end-user sees we use queries, these are available on the rendered component.

A query looks for the given text (as a string or RegExp) in the component, which is the first argument of the query. The optional second argument is TextMatch.

In our test, to verify that the form is rendered with the correct title we can use the getByText query.

it('should render the form', async () => {
  const component = await render(FeedbackComponent, {
    imports: [ReactiveFormsModule, MaterialModule],
  })

  component.getByText(/Feedback form/i)
})
Enter fullscreen mode Exit fullscreen mode

In the above snippet, you don't see an assertion. This is because the getBy and getAllBy queries throw an error when the query isn't able to find the given text in the document. If we don't want Angular Testing Library to throw an error we can use the queryBy and queryAllBy queries instead.

The error will print out the component's DOM elements with syntax highlighting

Setting @Input() and @Output() properties

With the component rendered, the next step is to provide the needed @Input() and @Output() properties.
To assign these properties, we can use componentProperties from the RenderOptions.
In the case of the feedback component, we assign shirtSizes to a collection of t-shirt sizes and we assign submitForm to be a spy. Later on, we can use this spy to the verify the form submission.

it('form should display error messages and submit if valid', async () => {
  const submitSpy = jasmine.createSpy('submit')
  const component = await render(FeedbackComponent, {
    imports: [ReactiveFormsModule, MaterialModule],
    componentProperties: {
      shirtSizes: ['XS', 'S', 'M', 'L', 'XL', 'XXL'],
      submitForm: {
        // because the output is an `EventEmitter` we must mock `emit`
        // the component uses `output.emit` to interact with the parent component
        emit: submitSpy,
      } as any,
    },
  })
})
Enter fullscreen mode Exit fullscreen mode

With this step, the component is ready to start writing tests.

Events

So far we've seen how we can assert our rendered components with query functions, but we also need a way to interact with our components.
We can do this by firing events.
Just like the query functions, these events are also available on the rendered component.

For the full list of supported events you can take a look at the source code. This post only covers the ones it needs to test the feedback component, but all of the events have a comparable API.

The first argument of an event is always the targeted DOM node, the optional second parameter is to provide extra information for the event. An example is which mouse button was pressed, or the text of an input event.

Good to know: an event will run a change detection cycle by calling detectChanges() after the event is fired.

Clicking on elements

To click on an element, we use the component.click() function.

it('form should display error messages and submit if valid', async () => {
  const submitSpy = jasmine.createSpy('submit')
  const component = await render(FeedbackComponent, {
    imports: [ReactiveFormsModule, MaterialModule],
    componentProperties: {
      shirtSizes: ['XS', 'S', 'M', 'L', 'XL', 'XXL'],
      submitForm: {
        emit: submitSpy,
      } as any,
    },
  })

  // first, get the submit button
  const submit = component.getByText(/Submit your feedback/i)

  // secondly, click on the button
  component.click(submit)

  expect(submitSpy).not.toHaveBeenCalled()
})
Enter fullscreen mode Exit fullscreen mode

Because we're able to click on the submit button now, we can also verify that the form has not been submitted because it's currently invalid.

We can use the second parameter to fire a right click:

component.click(submit, { button: 2 })
Enter fullscreen mode Exit fullscreen mode

Filling in input fields

To make the form valid, we must be able to fill in the fields.
We we can use this by using various events.

it('form should display error messages and submit if valid', async () => {
  const submitSpy = jasmine.createSpy('submit')
  const component = await render(FeedbackComponent, {
    imports: [ReactiveFormsModule, MaterialModule],
    componentProperties: {
      shirtSizes: ['XS', 'S', 'M', 'L', 'XL', 'XXL'],
      submitForm: {
        emit: submitSpy,
      } as any,
    },
  })

  // use queries to find form fields
  const name = component.getByLabelText(/name/i)
  const rating = component.getByLabelText(/rating/i)
  const description = component.getByLabelText(/description/i)
  const shirtSize = component.getByLabelText(/t-shirt size/i)
  const submit = component.getByText(/submit your feedback/i)

  const inputValues = {
    name: 'Tim',
    rating: 7,
    description: 'I really like @testing-library β™₯',
    shirtSize: 'M',
  }

  component.click(submit)
  expect(submitSpy).not.toHaveBeenCalled()

  // fill in the name field with the `input` event
  // the second argument sets the value of the target, similar to the JavaScript API
  component.input(name, {
    target: {
      value: inputValues.name,
    },
  })

  // an easier way to accomplish the same result is to use the `type` event
  component.type(rating, inputValues.rating)
  component.type(description, inputValues.description)

  // to select a value from the select component, we first have to click on the selectbox before clicking on the select option
  component.click(shirtSize)
  // because the select options aren't rendered in the component, we have to look for them outside our component
  // we use `getByText` (exported from `@testing-library/angular`) to search for our select option on the document's body
  component.click(getByText(document.body, 'L'))

  // an easier way to select options is to use the `selectOptions` event
  component.selectOptions(shirtSize, inputValues.shirtSize)

  component.click(submit)
  // our form is valid, so now we can verify it has been called with the form value
  expect(submitSpy).toHaveBeenCalledWith(inputValues)
})
Enter fullscreen mode Exit fullscreen mode

Just like before, we can get our form fields by using queries.
This time we get the form fields by their corresponding label, this has the benefit that we're creating accessible forms.

The getByLabelText and queryByLabelText are also looking at aria-labels to find an element

To fill in form fields you notice we have two ways of doing it, the first one is via the input event, the second one with the type event.
With input we just fire the JavaScript input event to set the form value, via the second parameter we assign the event's value to the value in our test case.
type is a new introduced (user-)event (introduced in v7.2). Besides writing text in the form field, type will also simulate the same events an end-user makes while interacting with our form to fill in the form field. This means that it also fires other events like keydown and keyup.

Because we're using the Angular Material select element, we can't set the value with the input event.
That's why we must click on the select element before we can select a select option by clicking on it the option.
Here again, we can use the new selectOptions (user-)event (introduced in v7.4) to select options from a select element.
The selectOptions (user-)event will not only work for Angular Material, but it also works for native select elements.
selectOptions is a user event, thus it will also fire other events to simulate an end-user's interaction with the select element.

As you can notice we have 2 different types of events, the JavaScript events, and the user events.
The difference between the two, is that a JavaScript event will only fire the event and nothing more.
Whereas a user event will fire multiple events to replicate a real behavior while interacting with the component.
For now, only type and selectOptions are implemented as user events.

Invalid controls

So far we have a working feedback component, but how could we test the validation messages?
We already have seen how we can verify our rendered component with queries and we've also seen how we interact with the component by firing events, this means we got all the tools in our toolbelt to test invalid controls in the feedback form.

If we blank out an input field, we should see a validation message.
This looks as follows.

component.type(name, '')
component.getByText('Name is required')
expect(name.getAttribute('aria-invalid')).toBe('true')

component.type(name, 'Bob')
expect(component.queryByText('Name is required')).toBeNull()
expect(name.getAttribute('aria-invalid')).toBe('false')

component.type(rating, 15)
component.queryByText('Rating must be between 0 and 10')
expect(rating.getAttribute('aria-invalid')).toBe('true')

component.type(rating, inputValues.rating)
expect(rating.getAttribute('aria-invalid')).toBe('false')
Enter fullscreen mode Exit fullscreen mode

Because a query returns the DOM node, we can use the DOM node to verify the control is valid or invalid.

Working with containers and components

The current test covers our feedback component, which is just a component.
For some scenarios this might be a good thing to do, but more often I'm of the opinion that these tests add no value.
What I like to do is to test container components. Because a container consists of one or more components, these components will also be tested during the test of the container. Otherwise you will usually end with the same test twice, and with a double of the maintaining work.

To keep it simple, we simply wrap the form component in a container. The container has a service injected to provide the t-shirt sizes, and the service also has a submit function.

@Component({
  selector: 'feedback-container',
  template: `
    <feedback-form
      [shirtSizes]="service.shirtSizes$ | async"
      (submitForm)="service.submit($event)"
    ></feedback-form>
  `,
})
export class FeedbackContainer {
  constructor(public service: FeedbackService) {}
}
Enter fullscreen mode Exit fullscreen mode

In the test for the FeedbackContainer, we now have to declare FeedbackComponent and provide a stubbed version of FeedbackService.
To do this, we can use a similar API as TestBed.configureTestingModule and use the declarations and providers properties on the RenderOptions.

Besides the setup our test looks the same.
In the test below, I choose to write the test in a more compact way which I find to come in handy for bigger forms.

it('form should display error messages and submit if valid (container)', async () => {
  const submitSpy = jasmine.createSpy('submit')
  const component = await render(FeedbackContainer, {
    declarations: [FeedbackComponent],
    imports: [ReactiveFormsModule, MaterialModule],
    providers: [
      {
        provide: FeedbackService,
        useValue: {
          shirtSizes$: of(['XS', 'S', 'M', 'L', 'XL', 'XXL']),
          submit: submitSpy,
        },
      },
    ],
  })

  const submit = component.getByText('Submit your feedback')
  const inputValues = [
    { value: 'Tim', label: /name/i, name: 'name' },
    { value: 7, label: /rating/i, name: 'rating' },
    {
      value: 'I really like @testing-library β™₯',
      label: /description/i,
      name: 'description',
    },
    { value: 'M', label: /T-shirt size/i, name: 'shirtSize' },
  ]

  inputValues.forEach(({ value, label }) => {
    const control = component.getByLabelText(label)
    if (control.tagName === 'MAT-SELECT') {
      component.selectOptions(control, value as string)
    } else {
      component.type(control, value)
    }
  })
  component.click(submit)

  expect(submitSpy).toHaveBeenCalledWith(
    inputValues.reduce((form, { value, name }) => {
      form[name] = value
      return form
    }, {}),
  )
})
Enter fullscreen mode Exit fullscreen mode

A couple of tips for writing tests

For End to End testing with Cypress use πŸ… Cypress Testing Library

Because it's part of @testing-library, you can use a similar API while using Cypress.
This library exports the same utility functions from DOM Testing Library as Cypress commands.

More info and examples can be found at @testing-library/cypress.

Use @testing-library/jest-dom to make assertions human readable

This is only applicable if you're using Jest as your test runner.
This library has useful utility Jest matchers, for example toBeValid(), toBeVisible(), toHaveFormValues(), and more.

More info and examples can be found at @testing-library/jest-dom.

Prefer one test over multiple tests

As you have noticed in the snippets used in this article, they are all part of one single test.
This goes against a popular principle that you should only have one assert for a test.
I usually have one arrange and multiple acts and asserts in one test.

"Think of a test case workflow for a manual tester and try to make each of your test cases include all parts to that workflow. This often results in multiple actions and assertions which is fine."

Read more about this practice can be found at Write fewer, longer tests by Kent C. Dodds.

Don't use beforeEach

Using beforeEach might be useful for certain test cases, but in most cases, I prefer to use a simple setup function.
I find it more readable, plus it's more flexible if you want to use a different setup in various tests, for example:

it('should show the dashboard for an admin', () => {
  const { component, handleClick } = setup({ name: 'Tim', roles: ['admin'] })
  ...
})

it('should show the dashboard for an employee', () => {
  const { component, handleClick } = setup({ name: 'Alicia', roles: ['employee'] })
  ...
})

function setup(user, handleClick = jest.fn()) {
  const component = render(DashboardComponent, {
    componentProperties: {
      user,
      handleClick,
    },
  })

  return {
    component,
    handleClick,
  }
}
Enter fullscreen mode Exit fullscreen mode

Example code

The code from this article is available on GitHub.

Once you know queries and how to fire events, you're all set to test your components.
The only difference between the test case in this post and other test cases lies on how to set up the component with the render function, you can find more examples in the Angular Testing Library repository.

Top comments (3)

Collapse
 
jwp profile image
John Peters

Hey Tim, how do we run these tests? My current setup is Karma and Jasmine, do we just type in ng test? It doesn't seem to work.

Collapse
 
nicoalani profile image
Nico Alani -βœ¨πŸ›Έβ˜„οΈ

I'm interested in this.. any notice?. Thanks in advance! πŸ™Œ

Collapse
 
timdeschryver profile image
Tim Deschryver

Oh sorry, I missed the original reply.
You can run just by using the angular jest preset, if you want to run jest tests with ng test, you will need to use a jest builder.