LECTURER:
Good to go. Alright. Welcome, everybody. This session is about Cypress. If you can't tell, I'm wearing a Cypress shirt there. And a little bit about me. I'm Alex Finnarn, and I kind of like to call myself a whatever and pragmatist because I work on whatever end of the stack and I just like saying that more than full stack. And then I try to be very pragmatic. So I like to keep it simple. And Cypress is a pretty simple solution to use, which I'll go into. That's my Twitter handle. If you wanna follow me on Twitter. And then I currently work at CivicActions and we do work with government clients. A lot of Drupal stuff kind of getting into more front end React stuff as well. And then Contribution Day. So they talked about this and the opening, but it's Friday 10am to 4pm. All are welcome. Yes, good thing to go to. OK. And so now when I was deciding on how to arrange this talk, I thought I would do it in the most brilliant way or you might see laziest way possible, but I'll go with brilliant.
And I just broke it into four different sections. So we'll actually go through building up suspense and excitement before we get to the last part Cypress. And so I started out with a quote, slide, I like quote, slides to introduce things. And so I was gonna come up with my own definition of test, but I thought that was a little pretentious. So I lifted one off of the Internet. This is a site called Wordnik. I never heard of it before. I kind of like the definition I found there determining presence, quality or truth of something. And then I also thought it was fun. At the end, they said a trial to sometimes test writing can be boring, so you can pretend like you're interrogating your test and say things like, Your Honor, this form input data is that order. I do declare. OK, that was a joke. I'll probably not pepper in more tough crowd today. I promise you sometimes I get laughs with that. And so before I get too far into it, I have to show the contractually obligated slide. Anytime you talk about testing, the testing pyramid comes up.
And so this was originally from a book called Seceding with Agile from a guy named Mike Cohen. But I lifted this particular one from Martin Fowler, very smart guy. He wrote an article called The Practical Test Pyramid, which I would encourage you all to read. And basically, the way the testing pyramid works, there's, people sometimes have many more levels, but they usually break it down in unit tests, UI test or I call it indent test and then service test or a bunch of other things in the middle people like to argue about. The main idea is that unit tests are way more isolated and they run faster and then what we'll be talking about indent test, you're testing the full app. They're slower, they're usually brittle, but Cypress has some nice features to make them less brittle and run them faster and parallel and things like that. And so like I said, our example for indent testing, a Cypress unit testing, you can use PHP unit, the PHP unit will do all kinds of testing. And then in the middle there be dragons.
So I'm not gonna go into like all the spy stuffs and mocks all that stuff. I just kind of keep it simple. But people argue for days on that type of stuff. And then here's another version of the testing pyramid. I got this from this course here which was actually it's kind of a departure into software quality assurance, and it's an interesting course you can take. It's like Netflix style subscription, but they've added another level. And I thought it was interesting that they kind of split it out to like who the tests are for. And ideally, you don't wanna like, you know, silo the test, but you're probably not gonna show unit tests to your clients. But the dream is that they're involved in an acceptance testing, kind of like we're gonna get into a little bit later. And then here's another chart from the same course. And oftentimes you can see test bases kind of upside down. And when you think about and the money is there, it'll cost you because it would be hard to maintain. When you think about Drupal, a lot of times you're testing, you're using a lot of contributed code, so you're maybe not at the unit level.
So sometimes this is how my test looked and I thought, Oh yeah, this is great. And then, yeah, and Kool-Aid, man. And so clearly I'm joking here, but you can kind of have a cavalier attitude I did before about why would you need more unit tests than acceptance? But I think you can push them down levels which I'll talk about a little bit later. And then the last slide and test, this is kind of how I got introduced to intent testing behaviour driven development. I took over a test suite, was written in Gherkin. And so you have this concept of giving when then and you also see a range act and assert. So you wanna set up the test with some kind of condition logging in. Then you do actions, usually more than probably one, and then assertions which could be one or many. And so now we'll get into Drupal, which is why we're we're all here and you know, things that you have in your Drupal site. But for a quote slide, I thought I'd just take a trip down memory lane and that's kind of the first quote I remember from Drupal come for the code, stay for the community.
Although when I was looking back on the Drupal.org site it showed up as come for the software safe for the community and then now I see that's on all of their theme again. So it's a very nice slogan. And then I just decided for whatever reason to put on like maybe who that slogan is for. So I think that's kind of, you know, for PHP developers getting in Drupal maybe mid 2000s late, everyone was kind of building their own system and Drupal was just have a nice community for that. That's kind of what I got into Drupal. And then as far as marketing stuff, this was about ten years ago. I found Dree's presentation at Drupal Down Under where you started marketing Drupal more. And this you're saying that it's mid-market wasn't enterprise yet and that's kind of contrasting it with WordPress, a small market and then we get a little more commercial here and five years later DrupalCon Baltimore, Drupal is for ambitious digital experiences. Kind of reminds me of like the Zoolander, like Blue Steel, ambitious digital experiences.
And that was a course for C-level executives, like site core and whatnot. And then now I'm glad that the slogan is back to kind of the roots for ambitious site builders. And I kind of see Drupal as competition more. It's like web flow bubble, things like that and getting more into the project browser. I think you'll see more along the lines of that. That's my quote slide. And then what is your Drupal site have or has users? And so I'll go into a custom command for this. But you know, that's one of the first things you figure out when you're writing a lot of intent test is how to kind of customize that and make it more efficient. You know, you have WYSIWYG and more. So I know ckader5 is not an iframe, I think. So it's a little easier to test than ckader4. And then you have things like autocomplete fields which can kind of be tricky or, you know, for searches and things. You have iframes. So just admittedly Cypress isn't that great for iframes. I've had solutions like there, I'm testing just iframe payment solution.
You might be familiar with it. Use that on your sites. And that worked well but other times and I'll get into that later. iFrame just really hard to do. Some other solutions have that done better. And then you can even test things like JSON:API, you have a decoupled site or you're exposing your API. Cypress can be used for that, might not be the best tool, but if you want to squeeze in a test there request or something like that, you can do that as well. And then a little bit about the Drupal test. So this is just a screenshot of drush generate command and you can see the five different types of Drupal tests going upwards from unit test to kernel test to browser to webdriver and Nightwatch I think are kind of on the same level. Webdriver uses selenium and then Nightwatch is kind of its own thing. And then a little bit about Drupal CI. I'm not super familiar with that, but I know there is a GitLab acceleration session I think tomorrow. So it's still on drupal.org but it's moving to GitLab CI.
I'm not sure the stage of that you can find out more probably in that talk. You can run locally what's on drupal.org via that run test shell script which is actually a PHP file and script. And I think Nightwatch is probably the closest comparison to Cypress. Like I said, the Webdriver functional JavaScript test traditionally use selenium and this is a nice add on if you have to do this locally. It came out like last fall, but I almost made an add on as well 'cause it was kind of hard to run the different types of tests locally, but most good Drupal contributor made that and it's a good add on. And then he also made something called Drupal test Trace, which you can use if you have content on your site and you're testing it with a database backup, which a lot of people are what I do. So that's a bit on Drupal site, but a lot on it. And then this quote, I was just kind of searching for easy way or whatever, and the universe sent me a Franz Kafka quote, a bold thing. And he said at one point in the struggle between yourself in the world, second the world.
And so I'm gonna tell you a troubled tale of where I chose the world. And in that case, it was Behat and selenium. And I'm going to contrast that with Cyprus. So this is like I said, I kind of got into doing more Internet testing with Behat and I was using this nice Drupal extension and you know what can go wrong with that nice DrupalCon there? It's one of my favorite DrupalCons. It's a nice trucker hat. And so here's what I had in composer file there. You can see there's several dependencies there to maintain where Cypress, I'm kind of just maintaining one. And then a little bit later, you can see I was cleaning it up. I got rid of the Drupal extension because it's useful, but I kind of was fighting with the step definitions and whatnot. So sort of similar if you're making a component library on the front end, maybe more, you get into it the kind of reduce dependencies like that. So it's a little better with maintaining. And then here's an example of a configuration file and it is cut off.
And that's intentional 'cause it's a sad slide and sad slides can have cut off text. And so just to point out, you know, there's a note there about having to do something or the test will fail because of some issue in that. And so it's just silly that I had to have a lot of this configuration because I was connecting to Source Labs to run the test because just maintaining the Selenium server and browser versions was tough. So I was using that in slide, a lot harder to use in the Cypress test run in my opinion. And then here, maybe this is a little overengineered, but I had gotten to three different Behat YAML files here for different situations and I just wanted to point out the commit message has something about testing the port a different proxies. I don't remember what that was, but it probably took me days trying to debug something on a remote and Sauce labs where you can barely see it and you have to look at weird debugging things. So I've never had a problem with ports or anything like that with Cypress.
It's been, you know, very easy to run the test. And so here when I was using Behat, you have your step definition. And so I was putting this after like all of the commands and it basically just arbitrarily waits for two seconds or those other conditions. So Cypress has better retryability and I know I haven't used Behat for a while. So it might have gotten a lot better, but that was always an annoying thing was waiting for the dang stuff to appear on the screen and test failing. And here's another step definitions just when you click the element and I was never good with regex so I just was glad I could copy that and work. But I wanna point out locality of behaviour which is a concept. There's an essay there, but the idea is that you're trying to keep your code, you know, all located so you don't have to go around to many different files to look at everything. And so with beheading the step definitions and gherkin, you're always kind of going back because the idea is that you can translate the two and it gets messy which I'll show in a little second.
But this was harder to keep where the code is. And if something fails, you know how to fix it. Just going around to different locations. And here's kind of the dream of the Behat scenario. It's very legible given when then there's variables there. You can probably know what happens as it runs through. It'll go through the example and use that and you would have more conditions there. Has got a tag at the top so you can run various tests and groups. It looks all nice, right? And then then it turns into this. And so it's another sad slide that has cut off text because slides can have cut off text. AndmI don't know why, but there was a coworker that liked to write in all caps. Maybe they were frustrated with Behat there. And so what you'll get into is you'll get into like, how do you select this element, and I'm not gonna get into page objects, but that is something you can try to use to abstract. You know, having a key for the value of each CSS selector. But Cypress doesn't really recommend doing that.
And then down here, you can see because it's hard to overwrite the step definition, you can do the object oriented, but the regex and whatnot, you can't do that. So it'll just error out because you have the same type of stuff definitions. So you end up having to like prefix steps, which is kind of look stupid. And you know, cypress you can overwrite them better. And so now we finally get to Cypress, so hooray. And for a quote slide, I didn't have to try that hard 'cause Cypress is pretty good at marketing itself. So this comes off of a great page there where they tell you why to use cypress and some things about their best practices and philosophies. And yeah, we're trying to address the key pain points which I just went over retryability, you know, keeping the test runner in line and different things like that. And they're trying to just make it really easy for you to get into testing. They also have like training resources which could do a really good job of. And then this is just what happens when you first open Cypress.
So there's a test runner you can open and there you can see I'm specifying some options to the CLI command to end to end tests and to specify the browser. And then the next part you'll get to is if you don't specify that in a CLI command and you just open it as two different types of testing. So I won't get into component testing, but Cypress is trying to get more into that. So it'll be, you know, React Vue things like that, you would use component testing. Two notes. Someone did create a twig component testing library so you can mount Twig files or are you doing something with twig components and test Cypress that way. I haven't really tried it out, but there's a link for you. But what we'll be doing is end to end testing apart and then it nicely lets you choose a browser. So if you notice there's no safari on there, I have a mac that should show up if there is support, but they're still working on that. So that's kind of if you need Cross-browser support, that's one of the things where selenium is a bit older and has more stable support.
You can start. Oh yeah, and that's a Docker image. So if you're using CI, it's good because it'll install the tools that are needed to run it in a CI environment which can be kind of tricky if you're trying to set that up yourself. So I encourage you to use that if you're testing Cypress and putting it in your CI pipeline. And then here's what some code looks like. So it's pretty simple and legible, even though it is JavaScript code. So you generally have describe it syntax. So describe, you generally group the test like what types of tests are they? And then it is the individual test there and it's just a function that runs. So here we are using the different Cypress commands, they're all chained off of CI.whatever, and you can see that a lot of them are changeable. So you can have a very kind of fluid syntax. And here we're just logging in using CSS selectors. So you can see that sometimes and I'll go into like what's a good best practice methodology for selecting things. But it can get tricky there.
And then we're visiting and then asserting something. So you can see this is kind of the range active search paradigm split out there. And I believe it's called mocha syntax I think is the describe it, but a lot of different testing frameworks use this now. And then here's a custom command. So there's a lot on this slide I'll break down. But you know, we just logged in in the previous slide, but you don't really wanna do that in all your tests. You just have a lot of them if you're logging in. A lot of Drupal sites have a lot of different users. And so here what we're doing is adding a custom command. So you put this in a support file and it loads this before the test run. So it's available when you're writing the test like down there. So it's called log in and then you can pass in any number of parameters. Here we have two user password. And so that's making a request which is faster than visit because it's not going to load the whole page, posting it and to that form. And then also to note is the session command which I've linked down below the site.
And what this will do is if it hasn't run, it'll look for a key which I'm just using the user there. And if it finds it then it will go ahead and restore that from cache. And you also want to use this option probably called cache across specs because if you don't do that when a test file, like I said, the authentication test, it goes to another test group, it'll raise that and sort of have to log in again. So I usually just cache it and it saves. I mean, it depends on your test run, how many tests, but it'll save you quite a bit of time to do it this way rather than logging in and out. That's a bit on custom commands. It's good to add a few, but you don't want to add like too many. Like I said, locality behavior if you're Cypress has a pretty nice API. So if you're making a bunch of commands for a bunch of things and you get into the kind of tricky things you get into with like, don't repeat yourself and then you do it too soon and then you're adding a bunch of parameters. So login commands, other common things, but I don't have too many custom commands usually.
And here's an example of a test just testing content creation. And so you can see there are some of the commands we had in the previous slides. Here is a select file, so that'll just help you upload the file pretty easy API. You just got to select an element and then point it to the file. Here, we're filling out a body field which you can see the selectors get a little hairy. It's interesting for the different CK editor buttons. The thing I found most semantic was the tooltip text actually, but otherwise, the CSS and the HTML attributes are a little hairy. And then another thing to notice, Cypress has a lot of great plugins, so this is Cypress real events. You have a lot of these like if there's a type command, you have a real type command. And so Cypress will simulate the events. Sometimes that can cause issues. So if you're having that issue, you can use Cypress real events or you can just switch to it all the time so you don't have to go back if you're having a problem. But it will actually, do the event in a native way so it's more reliable.
And then this is just the tag autocomplete field. You can type in and there are some special characters like Enter Tab. So I just type in Foobar and enters and then we have some tags and that's just saving the article. So here a little bit on best practice selecting elements as I saw in the last slide that it can get kind of hairy. So what Cypress recommends, and I do too is to use things like data attributes. So they like to use data PSI. I would use data test ID 'cause maybe you're not gonna use Cypress at some point, but they of course want to you to use data type because probably one more reason not to switch. But you'll see commonly like classes and IDs if you want to refactor something, you know, and that might break. So down here, we have, this is gonna be more brittle, let's say, for whatever reason decide to change SCS test will break. But if you tell the developers like, hey, when you see these data test ID attributes or data PSI, you know, don't don't change these and you can make them as semantic as you want.
So then there's more. There's a lot more to the best practices if you follow that selecting elements link. And then here we have assertions. So this is just asserting the content creation in the last slide. So you have commands like contains which will just look at what you've gathered here and it will just take the text from everything and look, does that have it in there? So some people like to be more specific with what the text is or assertions, but I'll use that if it just makes sense. It's pretty legible. There's asserting the image. Here we have the CK editor output, so I just have the assertion should have HTML. And so these are very readable try assertions and there's a lot of them. So you can have should not have HTML and various things like that. And then here's testing the tags and I'm using this function EQ. Cypress actually will get the elements and a jQuery element. So I believe it's using kind of just the jQuery EQ function, but that would just loop through the array and you know, I'm just making sure the same order and contains the text there.
And then you can flip the assertion. So you can say saiga and then should or you can say expect whatever your value is and you can even do the assertions in that method notation rather than the string notation. But I would just in your dev team, pick a style, you know, write it down and stick to it. You don't wanna have people going back and forth, but that will happen if you don't come to any agreement on what to do for standards. And then aliases. Aliases are are fun. Cypress is JavaScript. It functions in an async nature. And so sometimes it can be hard to get the variable and scope that you want. So one thing you can use is Cyprus aliases. So this just comes from the Cyprus docs. So here in Cyprus has lifecycle hooks, so that's important to know you can do before the whole test runs, before each test runs, before or after the whole suite or after each test runs. And what this is doing here is getting the button text. But then since you were not using an alias there, if you're trying to use it in a test, it doesn't work 'cause it's not in scope.
So this way, we'll get an element. You can use another command to just call any method. I see some people like writing it this way. You could also do it in JavaScript, but it's more the syntax of Cypress and then as keyword or as command will create the alias. So now we have an alias called text and now you can use it down in the test. It's actually just off of the current objects of this. So that will be a property, but you can also use it in this syntax. And what this will do when you get it was will rerun the command. So it's not a stale reference. So that's nice but that's a bit on aliases and then you can use aliases with the intercept command. And this is one of the coolest things about Cyprus and I know other frameworks can do it. But when I learned it, I thought, wow, this is cool beans here. And so this example we have, let's just pretend like we have a coupled application or something like that and we're getting data from an API. One common thing you'll have is, is the stuff there when the test runs, you know, how long do I have to wait for it?
That's a common source of errors with JavaScript not running. So what this does is it spies on any request because browser or Cypress is attached to the browser, so it runs. And so to look for any of these requests, turn them into aliases and then we can wait for them any number of aliases before we run our assertion. So that's a lot better than saying, you know, say wait five seconds or something arbitrarily like that, and then you can also attach a fixture. So here I'm just, you know, maybe you have you don't wanna test all of the users, but you wanna supply a test list of users. But here, it lets the API request goes through and just gets back the individual ID or something like that. Another cool thing you can do with intercepts is if you have a production environment with different headers on there and your code reacts that way, you kind of like, well, how do I do that if I'm not in production where you can actually capture the response and before it goes there and add a header. So I had an example where there was a certain header added, I don't remember why, but the code looked for it and made sure that it was multi-lingual feature.
And so I was able to add this and then the code could read it without having to change my application code which is really nice. And then, you know, down here we have the intercept command with CK editor, so or layout Builder. There's a lot of Ajax-y things that you might wanna spy on that stuff so that when you're trying to assert later, you know it's there. And then for people that like to use like a click browser extension, Cypress is working on stuff like that. So that's a screenshot of this extension that runs on the Chrome for some protocol and Chrome. And so it's using that. You can see that there's puppeteer also mentioned in there, but Cyprus added their hat to the ring and so you can use that like if you've used a selenium, sometimes they have a browser extension where you can just click around and then it will capture that and you can put that into a test. It's kind of nice to get started. And now that we have gone through Cypress features, I just wanna go and see some opinions of mine.
So this gets into the opinion part. And so as I say, like, you know, when you write test, now you know how to write them like, but where do you put them and all that type of thing. That can be confusing if you're a new developer on board and where do you put the test? And if you don't have that figured out, people just put them in all sorts of places and then you have to refactor that later. And so I think tests and then tests can kind of come from two places, features and then bugs. And so here's just an example of some way that you could arrange your end to end testing folder and so you could arrange it like this. So you could have at the top level regressions which of course come from bugs. So maybe you wanna isolate these out at first because they're really important. You might even have like a hotfix in your dev environment, but you don't know where the test really goes. And I think like any regression should be put into other tests because it's something user is doing. So it's not really isolated to a bug, it's part of a user story, probably.
Features, you can have user stories related to different roles. Those are different roles on the Drupal site there. And so you could arrange your tests that way and make sure that you're trying to think of the best way to arrange your features, however that's set out. You can also use like a user guide if you have that for your site users and try to map that, might be an easy way to organize it. And then here's what the creator of Cypress, you gave a talk and that's was a screenshot of his directory. So he said to break it up into models and you see it's not kind of doesn't really follow it. But anyway, you can split it up into models or user roles and then you can also put test in and use a skip method and that old Cypress will just skip that test like you would think. So it's a good way to add tests. Maybe you could, QA could stub out the test developer config figure, fill it in later. You can also, as you're writing features, you know, maybe during sprint planning, go through the steps and fill it in as you develop.
And then like I said, I think you wanna take regressions and kind of push them down into features. And then if you are looking at your features and try and refactor those, maybe try to push those down into integration tests because if you're repeating different things, you know, end to end testing can be expensive and integration test would generally run faster. So it's always kind of look how you can push things down. So that's a little methodology for you can think about. And here are some examples of tests that I've pulled from different repos I've worked on. And so this kind of like is this a good test or bad test and kind of musing on it. So here, you'll see this is kind of like a regression test. Oftentimes, I'll put in a specific bug, maybe a link to the testing or JIRA or whatever you have. So it has more information contextually for someone going back to see it. Here are basically just looking for a preview node. But is it good? I don't really think so because like this is probably part of some kind of user story somewhere.
So it's OK to have the test like maybe this was a bug you had where you kind of get the preview and you just wanted to make sure that that was working in your CI. But I would go back and refactor this and look for why is someone previewing a node? There's probably a reason and stick it into that test because this is a Cyprus best practice. Well, I'll get to that in a little bit, but not to split apart test too much. And so here we have a test where it is kind of split apart. You know, you can assert these things in one block because every time you have an IT function, Cypress will blow away the session cookie and all that type of stuff and reset itself. So it adds extra time. And a lot of times unit tests you're breaking up because you wanna know where it failed. But Cypress has very good reporting, so you always know where it failed. Takes videos and screenshots so it's less useful to break them up religiously one test per thing you're asserting. And then this also has a potential bug in it.
I don't know if anybody can spot it, but one thing to do is to not rely on state from a previous test in the next test. So here we have, I see this a lot and things I worked on. And so here we're asserting this and then visiting another page. So if this fails, then I won't be on the Bell page. And let's say I wanna have this assertion so this will fail. And then if that's relying on being on the about page, that'll fail. And so you wanna set things up maybe before each hook, before the test, and you just don't wanna rely on state like that between tests. So here's this test sort of cleaned up once again, and it's not a very good test because it's just flippantly like someone had to verify CSS and stuff. But this I think is more readable and, you know, it's testing similar things. So I just don't think you have to break it up like the previous slide. And then here's some tests I wrote. So I think they're better because I wrote them. But one common thing that you'll have to do, you can think of tests like, do you have a listing and then a detail with a different like I was saying, models before, this is a content type.
And so you might wanna think a test that way, like collections and then individual items. This is a test. I had a slide there, the iframe payment form. So this test as I wrote, that involves that and it's got several things in it. You know, here are some sort of variables like, you know, there's makes a single so there's a way to make multiple, there's other tests like that, test some form validation. So a little messy. But while you're in there, you know, filling out the form, if you have some validations, I think it's fine to include it rather than make another test just simply for that. And then here we had a feature where the form would be filled out if you went to it so you could share a link and it would fill out things like donation amount and giving frequency. And so I'm only really testing the first part and that, but I think that's OK because we're also testing the donation workflow and this previous test. So you don't have to like go all the way to the end if you have another test that complements it because then you're just duplicating it.
And one of the biggest complaints when you get enough test is it takes too long to run. And then here's another example of a test. So here you can import data and then dynamically create tests which is kind of interesting. And so you can use some of the data here to write the name of the test. Even this just authentication test. But is this a good test to be in Cypress? Probably not, because you're not really testing anything with user interaction. You know, you can probably push this down to a lower level and this is using PHP unit and browser test based. And so you also get things like creating users and other stuff like that. You have to create sort of a bridge if you're using Cypress in the JavaScript. So maybe another reason to use integration test 'cause you get some of those functions for free. And then, like I said, there's tons of plugins. So this is a link to the plugins page. You can go through it. There's like, I don't know how many of I'll just say 100 because it sounds like a good.
But there's many different categories and I'll go over some of them a little bit later. And then last slide on Cypress. So I'm not just gonna tell you it's all puppies and rainbows. There are some downsides there. I just wanna honest and fair. And so still no iframe support. So that links to an iframe issue. It's been open for a while. There's a lot of comments. People don't have this fixed yet. And so like I said, there's even plugins for it and sometimes it works fine. And I've had it work, many months, years and then other times you test an iframe and you just struggle with it because what Cypress does is it takes a DOM snapshot. And so what I've seen and do is go in there, it'll take an empty snapshot of iframe, iframe loads other things and then you can actually see it in the test runner, the next tick or whatever. It has all this stuff there. And it's frustrating 'cause you can see it, it shows up. But trying to get Cypress to wait for that or get the correct DOM snapshot can be tricky.
And then this comes from a quality assurance Reddit actually. So I kind of stepped into this a little bit outside of my zone. But they go into if you follow that link and a lot more detail. I still don't know what key cloaks are, but Cypress isn't too great with iframes. Multiple tabs. If you have anything with multiple tabs, you probably run into trouble multiple windows, things like selenium server and the Webdriver API just have kind of better support for that. And then this can be a pro or con, it's limited to JavaScript and typescript. And so if you're doing like a couple Drupal, you can actually do things like spy on the functions. So you can have a code coverage report which is kind of nice, but if you're using other languages like PHP and you're kind of trying to one foot in each language can be less useful. But if you're doing the couple Drupal, yeah, that can be a pro. And then Playwright is something I wanna look into more. It's what started a fork of Puppeteer which came up in a previous slide.
And this link will go to a post where someone explains in great detail why they prefer Playwright over Cypress and the reasons so. But they weren't shy in their opinion on that. So I definitely wanna look into Playwright and kind of compare it and then also Nightwatch. Mainly looking to Cypress and it's just been so easy for me to use test runner knock on wood has never failed for me and that's just a blessing. So that's kind of the end of that part. And now I will go show you some, just drag it over. I'm going to storm and show you some actual test and then I'll open it up for QA. So I guess I'll just go into this. This is a plug in here. So what common plug ins might you wanna use? This is actually test a lighthouse. So you can work that in there if you have a kind of performance budget. And what this does is just goes to the home page and then you have an object which you can set different thresholds. And if it goes under any of those thresholds, the test will fail. So that's a good way to keep honest with your performance.
Here's another plug in. It's Cypress X, uses X core. So what this is doing is just going around and it goes to the homepage again, you have to inject X and then you can check to see if it's accessible. And oftentimes, you'll have kind of these are rules that are in X. You can go there and look at that documentation. But sometimes you'll have things that are hard to get accessibility wise, but you have an escape hatch there to include excluded rules. And then I won't really go into this, but this is on the X core plugin page. And you'll wanna do this probably if you use this plug in and I just copied it right from the docs. But what it does is when you're running the test, it'll output a very nice little table of the violations. Otherwise, you have to go into the test runner. So that's a nice addition that they added that plug in. And then I often will with Drupal put things into a module and you can have config and adding pages or users, you can do this many, many different ways. But a lot of times I end up developing a testing module and do creating things when you install it.
Some people use Drush commands. You can kind of pick and choose. And then I'll try to show you the experimental studio which is kind of a click thing, a different thing than the Chrome extension. So I guess they're working on multiple ways of doing it now. But here you can set up configuration and these are just ways that make some of the plugins work. There's the authentication test. Oh, yeah, I'm gonna, that's where I'll try to put the experimental studio output when I try to run that. See if it actually works. And then, OK, yeah this, I was proud of this. So when you put the test up on your CLA pipeline, one of the common CI pipeline, one of the common things I've run into, you'll have a lot of tests. You know, you wanna group the test and maybe you only run one or on certain tests on pull requests and then other tests when you merge everything in. So I was trying to figure out how to do that on GitHub actions and this is what I came up with. So just just break it down. It took me a little while and you know, you can see the PR command is like 50 commits as I'm just trying to go through this.
So what this does is it goes through, it sets a variable just has tags false. So and this is I guess I'm looking for a commit message to something special in the commit message to see then if it has tags and which tags there. This will take the most recent log commit. And it's important to add the no merges because the most recent commit on GitHub actions is actually the merge commit. And so you won't see any message there which threw me for a loop for a while. It's like why is there no message? And you also have to put fetch dev zero so it gets more than the most recent commit. And then it reads it into array. It looks for the hash signal. So this is if you have a git commit message, whatever it says and at the end there's a hash symbol. And so that's what it's breaking it down upon sanity check, just making sure that there's something there. And then at the end, I'm making sure there's an add tag and that is in. Here you can add tags at several different levels, so you can tag the describe block.
Here's content. And then I was also using a smoke tag. So sometimes you just have like, we really wanna know if these tests work. And so that if it finds those things with at symbols, it'll put them into a string and then cause, set these variables. So it's true. And then installing DDEV, you can set this up different with an image or something like that. Setting up a Drupal website. Here I'm installing the testing module I was kind of showing before the install page there, no dependencies. And then here we get down to the actual environment. And so since I've up here put them in the GitHub environment variable, I can access them down here with the same keys. And so I'm doing it running different tests. If you really just wanna speed it up and you're doing stuff with content or whatever, it'll just run those tests with the commit message and then if it doesn't have anything in there, it'll run the smoke test. So that's a decent way to provide some things if you really wanna get the test run done, because sometimes you can get up to like hours depending on how many types you have.
And another thing that can help is splitting it up. So this is actually a pretty easy way of doing parallelization on GitHub actions. And this is what I would run after it's merged in to running more tests because you're not trying to review anything. So it's a good time to run all the tests, see if anything breaks. And so this is doing is building, is adding containers. And so you can add as many as you want. Here it's just two. And then same a lot of steps here. But then you have the, it's a plugin called Cypress Split. And what this does is help you parallelize it. And so all you have to do is use this GitHub action and use the strategy job total. So that's like the, that relates to the container. And this will go through and then take how many tests you have, divide them. So you have 50 tests, you have two containers of 25 each. And so let's say your original test run was like 40 minutes and there should be 20 each and you can keep going. And that's a nice way to cut down on your test run time because if you had more tests, it would just keep getting longer.
The last thing over here. Here are just some of the plugins that I have. I don't have a lot of training to make this kind of an example repository on my GitHub profile and keep adding examples. But there you see the X, the lighthouse audit. We had cypress grep is what splits it up into tags which showed real events we talked about that. And then Cypress split runs it in parallel and then Cypress slow down is actually one you can use for examples like I'm going to try to show here, but it'll pop up a test runner and it just slows down so people are better able to see, better able to see it. So now I'll just open the test runner and hopefully this pops up in the same window. Oh, no, it didn't. So just drag that over. And here's the test runner. They updated the UI, so it looks better. But here you have the different specs you can run. And I'll just show content creation here. Click on it. And here I've slowed it down because it would be like super fast. You can see it's creating the session because it hasn't yet.
Otherwise, it would be restored. Here is just going through. You can nicely see what's going on. And once again, this is using the slowdown so slower. Normally would be. But if you're debugging, that can be useful. This is the same test we went over. And then at the end it's still little assertions. And then, yeah, now it's green. And so if it failed, it would be red X. But you can go along and then you see a DOM Snapshot of each of the stages. So if I wanted to see contains article, why isn't there, you have the snapshot and you can go inspect the element. And now when you go over to the console, it has it right there for you. I don't know if I go click again. Here, it even has mouse events and there's the DOM node. But it's really nice there. Where was the... Yeah. And then you can even see before and after. So it's kind of hard to see, but before we don't have it selected. After there, shows up. So a lot of this stuff can be very useful if you're debugging test. And you know, it's just a lot harder for me to do with source labs or remote thing.
So that's that. And then I'll go into authentication. Here is not slowing down, so a lot faster. You can see those are just normal test run. And so let's say I wanted to add a command here. So for fun, I would just add a command to this test. So it reruns it first to go through all the steps and now you see there's additional prompts we have here. And so I can right click on that text. And they're still working on this. So it's it's very rough. But I can then assert that the text is there. And so now it has the steps. And so if I then save the command, it'll run through it again just to make sure it works hooray. And now we have an authentication this stuff. And so this is the last step that ran. And here it just gives you a comment. As you can see, you might have to redo the selectors and whatnot. So I oftentimes will just like go through a test, go through, and then you'll have to kind of like take it and massage it a lot. But if you're new to testing and you just wanna try, Cypress out it's a pretty interesting way to start your test and look at that stuff.
So that's basically been on the test runner. They do have a CI service. They kind of try to push you in 'cause they got to make money some way. And so this will connect to the cloud and helps you run in parallel and a lot of things. And then I guess you can debug the sessions remotely. That's like one of their newest features. You know, I don't pay for that stuff. And so there is even like somebody made like a Cypress dashboard that's like open source. You can download that and put it up on Heroku or whatever. But yeah, that's about it. So I guess I'll open it up for questions. If anybody has questions. You guys, I can ask you questions.
STUDENT:
Sure. That's a nice switch.
LECTURER:
Yeah, I guess I will stop the recording then for the people that will watch it later.