How to Write Acceptance Tests

Поделиться
HTML-код
  • Опубликовано: 3 янв 2025

Комментарии • 50

  • @gpzim981
    @gpzim981 3 года назад +55

    Imagine the honour of working on a team lead by Dave Farley.

  • @seanregehr4921
    @seanregehr4921 3 года назад +39

    [User Stories] become [Acceptance Tests] which is [Behavior Driven Development] "Doing the RIGHT thing."
    [Code Functionality] becomes [Unit Testing] which is [Test Driven Development] "Doing the THING right."
    In both scenarios, tests are written to be loosely coupled, which enables scaling and future changes, without breaking anything (mostly). The tests only care that "the system works" and not 'how the system works'.

    • @calorus
      @calorus 2 года назад +3

      ^Excellent comment.

  • @ruixue6955
    @ruixue6955 2 года назад +8

    0:16 *how to write acceptance tests that don't break as your system changes*
    1:00 *acceptance tests are always written from the perspective of external user of the system* - try to separate how the system is working from what we would like to do. focus is on *what*
    3:14 1st thing to do: requirement from customers - User Story: pay for book with a credit card
    3:44 example: somebody buying a book
    4:20 example executable specification in DSL
    4:5
    6:17 next layer: *DSL for testing*
    7:07 example implementation code of DSL
    7:58 3rd layer: *protocol driver*
    8:25 job of the protocol driver:
    9:15 what for a minute what the protocol driver is: the only layer that understands how the system works
    9:28 hidden completely from the test cases
    9:48 bottom: System Under Test
    10:25 example demo
    10:42 protocol driver sample code
    12:40 recap

  • @PaulSebastianM
    @PaulSebastianM 6 месяцев назад +2

    Imagine the level of day to day happiness at work when Dave Farley is your team lead or principal engineer.

  • @gonzalowaszczuk638
    @gonzalowaszczuk638 3 года назад +5

    How would you handle side-effects and setup, to make sure the tests are reproduceable and independent? For example, who is in charge of making sure the "Continuous Delivery" book exists in the book store before running the acceptance test (because it would fail otherwise)? Who is in charge of resetting the state of the system after the test is finished? The protocol driver?

    • @ContinuousDelivery
      @ContinuousDelivery  3 года назад +8

      The test gets the system into the state ready for that test, I use an approach that I call "functional isolation" to mean that the data for each test doesn't leak to other tests, so no need for clean-up, other than dropping the whole system, or DB, at the end.
      I think that the development team owns all this stuff, and it is all completely automated.

    • @gonzalowaszczuk638
      @gonzalowaszczuk638 3 года назад +5

      @@ContinuousDelivery That sounds very interesting. Do you have a video/resource on the subject of "functional isolation"?

  • @maximilianosorich554
    @maximilianosorich554 2 года назад +1

    the visual robot example was very nice...

  • @TheJessejunior
    @TheJessejunior 3 года назад +2

    Very good! simple and efective! thanks again sr!

  • @NilsElHimoud
    @NilsElHimoud 4 года назад +2

    Thank YOU for sharing this.

  • @_djordje
    @_djordje 4 года назад +2

    Excellent video, thank you.

  • @roman_mf
    @roman_mf Год назад +2

    Hello Dave, thank you very much for all content that you produce. I like the idea of Channels. I went through the code on GH link that you have in the description, but there a couple of things I don't understand and would greatly appreciate if you can clear these up a little:
    1) You have several acceptance test examples, and each of them seem to duplicate channels in their @Channel annotations. What is the logic behind this?
    2) I see that you have a separate class BookShopDrivers that acts as a registry for all drivers that the bookshop application is using. However, driver() method always returns the first driver from the @Channel's arguments, which I see how it is connected with 1), because each new test adds new channel as the first item on the list, but again, I am trying to understand the logic behind all of this.

    • @ContinuousDelivery
      @ContinuousDelivery  Год назад +1

      “Channels” may be a misnomer, but I can’t think of a better name. The name made sense when we first came up with the idea. Then they represented “different channels of communication” with a single system, Web, public API, institutional Std API etc. Later, one of my clients used the same idea to represent an Old version of their system and a new, running the same tests against each system.
      Fundamentally, the “channels” are defining which Protocol Driver to choose.
      The example in the code that you point to is a sketch, not a fully working version of the channel idea, I don’t have a publishable version of the custom test-runner that you need to automate the switching of the protocol drivers work, so I didn’t bother finishing this code. It was written originally as an example for some one, so I didn’t need to take it any further, sorry for the confusion.

  • @miguelgarciadasilva
    @miguelgarciadasilva 2 года назад +5

    Great content as always! In my understanding, with this approach we have three layers of testing:
    - Unit testing: for the domain layer (in-process dependencies and mock external dependencies).
    - Integration testing: for the infrastructure layer (every adapter in isolation with technologies like testcontainers or MockServer).
    - Acceptance test: as an end to end testing for the use cases (using all the real collaborators deployed in an environment similar to pro).
    This is the point or I misunderstood something?

    • @ContinuousDelivery
      @ContinuousDelivery  2 года назад +6

      I usually say that 2 layers are mandatory, unit & acceptance. The acceptance tests as a kind of super-integration test. I'd say you need these for every change. Integration tests are valuable, but more about the context. They are useful for some kinds of code, and not needed for others, but mostly, yes, I think that is a reasonable summary.

  • @random6434
    @random6434 3 года назад +3

    So it turns out I've been writing my "unit tests" like this or something like this for years. Normally after writing a few tests I start gathering set up and assertions into some kind of "test environment" class just for DRY purposes.

  • @ziabasit8745
    @ziabasit8745 3 года назад +1

    Thank you for doing this Video!

  • @MrSkinbad
    @MrSkinbad 2 года назад +2

    Great content but I don't get the "robot in book store" analogy - it works as long as the UX for e-commerce websites are modelled around how physical stores operate. If your physical bookstore was a "smart" store where you just walked out with the stuff you want and are automatically billed the analogy falls down.
    More than that, I see how its nice to write the acceptance test cases in business language, but aren't you literally aiming to test the UI here? So surely this is where the abstraction should end. e.g. we would have different test cases for a mobile app vs a web app since the UI is surely different in some ways.
    Or is the suggestion actually that we would have abstracted test cases to such an extent that we can use the same cases for multiple UIs? I just wonder how realistic or useful this actually is?

    • @ContinuousDelivery
      @ContinuousDelivery  2 года назад +4

      In that case the business case has changed. Of course there are limits to how far you can take this approach, but the point is to model the outcome, and test for that, and that is ALWAYS a more stable, more generic thing than any specific implementation that you choose to deliver the outcome that you were aiming for.

  • @donnyroufs551
    @donnyroufs551 Год назад

    In a standard 2 tier setting (client and server) would you write acceptance tests on both the client and backend or truly do it from the client all the way to the backend(e2e)?

    • @ContinuousDelivery
      @ContinuousDelivery  Год назад +1

      It depends, is each of the pieces independently deployable. That is can I release each without testing with the other? If not, then yes, I would test them together.
      I define the scope of acceptance testing in my deployment pipeline as being aligned with whatever it is that I will deploy. I want to test everything that I deploy together, together before release, and acceptance testing is the most effective, most efficient way to do that.

    • @donnyroufs551
      @donnyroufs551 Год назад

      @@ContinuousDelivery Makes sense. How would you deal with replacing out-of-process dependencies? I assume you would need some kind of test client that knows about both the client and server e.g.
      TestClient -> server (responsible for setting state, mocking out-of-process dependencies)
      TestClient -> client
      Tests -> TestClient
      edit:
      Wouldn't your acceptance tests be kind of useless during the TDD cycle since both sides now need to be done for it to actually run? I have always relied on acceptance tests to tell me whether my API does what is expected, but this doesn't work the moment you write it end to end

  • @torkleyy9168
    @torkleyy9168 3 года назад +4

    Thank you for these excellent videos! I'm not sure I understood how BDD, Continuous Delivery and automated Acceptance Testing fit together, maybe you can clear things up.
    Given that
    work should be integrated regularly and
    acceptance tests should, too, be written before implementing the code
    does the DSL help anybody other than the developer making the change?
    It seems to me such a simplified DSL could be written by a less technical person, but with BDD that is discouraged.
    Now if that's true, it seems to me that you have to either
    1) completely implement your user story in one commit
    2) have your pipeline failing until the written scenarios are fully implemented
    3) collaborate on a branch to implement the story, risking the creation of long-lived feature branches
    It seems like wasted effort to create such a simple natural language DSL if it doesn't promote collaboration with e.g. QA engineers who can write the tests.

    • @ContinuousDelivery
      @ContinuousDelivery  3 года назад +5

      ...or 4) mark acceptance tests that represent features that are "in-development" as "in-development" and don't run them in the pipeline.
      I'd pick 4!

    • @torkleyy9168
      @torkleyy9168 3 года назад

      That makes sense thank you! And is there any room for tests not written by developers, or is that always a bad idea?

    • @ContinuousDelivery
      @ContinuousDelivery  3 года назад +1

      @@torkleyy9168 No it is ok, but the Dev team MUST own responsibility for the tests once they are written. They are the people who will change things that break the tests, so they should be first to see the breakage, and fix it immediately.

    • @andreasv9472
      @andreasv9472 Год назад

      @@ContinuousDelivery I still don't understand the difference between ATDD and BDD? Seems to be the same thing?

    • @ContinuousDelivery
      @ContinuousDelivery  Год назад +2

      It's a nuance, and probably doesn't matter much. I would make the distinction that BDD is about focusing your testing on evaluating the behaviour of the system. This can be useful whatever the nature of the test, BDD works for tiny, fine-grained TDD style tests or bigger, more complex, more whole-system functional tests.
      ATDD is the second one, but not the first.
      BDD was originally invented to cover the first case, to find a way to teach TDD better, but has become synonymous with the second, because of more heavy-weight tools like SpecFlow and Cucumber.
      So for practical purposes BDD == ATDD, but as someone who was in at the birth of BDD, I still find it's original aim useful and important.

  • @thigmotrope
    @thigmotrope 2 года назад

    So PageObjects are abstraction that lives in the protocol driver layer? If you only have one protocol that you have to support (e.g. web), the argument for the layers seems to weaken. I do think the dsl (bdd with mocha/chai) in the language of the problem domain is helpful the

    • @ContinuousDelivery
      @ContinuousDelivery  2 года назад

      Yes, page drivers are a form of protocol driver. I don't agree that a single protocol weakens the argument, the separation is still good, even if you only have one. I'd argue that it is probably a poor design if you only have one, because a better design would separate the concerns better, but even so, the layering is important. The DSL should be clean of system-implementation-specifcs, the protocol layer is where the translation happens.

  • @ML-hf6ii
    @ML-hf6ii 2 года назад

    love that channel

  • @TheVincent0268
    @TheVincent0268 3 месяца назад

    What to do with a legacy spaghetti system? How do you AT that?

    • @ContinuousDelivery
      @ContinuousDelivery  3 месяца назад

      I talk about some aspects of that in this video ruclips.net/video/_gs3MjJBEfs/видео.html

  • @OthmanAlikhan
    @OthmanAlikhan 3 года назад

    Thanks for the video =)

  • @reverendbluejeans1748
    @reverendbluejeans1748 2 года назад

    Is the e2e testing.

  • @mml1224
    @mml1224 3 года назад +2

    helpful, but hands moving very distracting

  • @stanislavcoros
    @stanislavcoros 3 года назад

    all what was said is normal logic. Nothing special. But some testers thinks they job is so out of this world like they think they are inhumans :D yet they are replacable by simple script

    • @ContinuousDelivery
      @ContinuousDelivery  3 года назад +9

      If all they do is translate some "test script" into key presses, then yes, but I think most testers bring more to the party than that.