- Notations
- Exploring the client
- Todo API: Redux
- Use GitHub Projects to support agile development
- Writing (and testing) a beautiful client side application
- Remember to test
- Questions
- Questions that you need to answer (as a team!) are indicated with question mark symbols (:question:)
- Tasks that specify work to do without a written response will be bulleted
- Remember to set up GitHub Projects with your stories and estimates
Write up your answers to these questions in a Google Doc and turn that in via Canvas on the assignment for this lab.
- Make sure that everyone in your group has edit privileges on the document.
- Make sure that the link you turn in gives the TA(s) and professor(s) at least comment privileges.
- Include the URL of the GitHub repository for your group at the top of the GDoc. This will make it easier for us to figure out which team is "Snoozing Llamas".
Definitely ask if you're ever confused about what you need to do for a given task, or what the answer to a question is, etc.
The client side of our project has changed since lab #2. The directory structure is mostly the same, but the client-side interface uses Angular to handle most of the creation of the elements of the user interface. Angular's template syntax extends HTML and JavaScript. The testing is handled in two new places:
- Angular spec files (e.g.,
user-list.component.spec.ts
) for unit testing in Angular. These use Jasmine and Karma. - E2E (end-to-end) tests in
client/e2e
. These use Cypress. (Note: the picture shows the Cypress spec files in a folder called "integration", but the default and what we will use is now "e2e".)
The starting code includes several
ways of using Angular components to display user data.
It includes two ways of organizing the user-list
information:
a grid-like arrangement of cards and a list.
The grid approach to organizing the user-list
information
leverages a user-card
component.
The user-card
component is also used (in a slightly different way) in the user-profile
component.
Karma tests for each component are named
almost the same as the component but include .cy
before the .ts
,
and Cypress E2E tests for user-list
are located in client/cypress/e2e
. There are supporting files for the E2E tests (called
page objects) in client/cypress/support
. The page objects are named similarly to the Cypress E2E tests, but also have .po
before the .ts
. The purpose of the page objects files is to help separate aspects of accessing elements on a page (like exactly what kind of component you are using) separate from the tests. That way, when you decide to use a simple button with a custom button (for example), your test will still just find the button. Specific information goes in the page object.
❓ Answer questions 1 through 4 in QUESTIONS below.
In Lab 2, you worked with your partner to implement an API for requesting
'to-dos' from a server. In this lab, you'll be using a to-do API that we provide
for you with the lab. The API meets the specifications of lab 2 and
can be found at localhost:4567/api/todos
when you are running your server
(go into the server directory of your project in a terminal window and execute ./gradlew run
).
We'll be using GitHub Projects to augment the standard GitHub issues system with nifty powers to aid in Agile estimating, planning, tracking, and development. The next two sections describe the software development tasks you need to complete for this lab, which all take the form of augmenting the server API with new functionality.
Before you actually start coding on any part of the lab, you should spend some time using issues and GitHub Projects to capture and estimate issues and do some planning.
- Go to the
Issues
tab for your repository - Near the green
New issue
button, there is a button-like thing that saysMilestones
(click it) - Click the green
New milestone
button - Create a milestone for the lab that uses the lab's due date
- If you'd like to make multiple, smaller milestones, you may do so
- You can write in other information if you'd like, but at least include the one milestone for the lab's due date
Once you have created a milestone, you will be ready to create a GitHub Projects board to act as your visual workspace that is connected to your GitHub repository.
⚠️ One thing you should not do is create separate tasks for things like unit tests or refactoring. Those activities should be "baked in" to your work flow, and not considered separate (and therefore to some degree optional) activities.
- Click the
Projects
tab on your GitHub repository - Use the green dropdown by the button to make that button read
New project
(then, clickNew project
) - In the popup dialog, choose the
Feature release
template and then click the greenCreate
button
The view that you see will have several views, each focused on a different way of thinking about the state of your project.
If you haven't already assigned estimates as you went along, now is a good time to think about how difficult you think each task will be and put estimates on each issue. Once you've created and estimated all the issues, you should think about which ones you think you can reasonably do in this lab. This could be all of them, but it doesn't have to be. You can always add issues to this as things progress, and in general customers would rather see the set of issues you expect to complete in an iteration increase rather than decrease, so being conservative in your initial planning is probably a Good Thing.
You should move the issues you really expect to do into the Ready
track, leaving all the other issues (that you may
or may not do) in the Backlog
track.
Now, you are ready to get started working on the coding part of this lab!
You'll want to keep an eye on your board throughout the lab, using it to guide your decisions about what to work on, updating issues as you make progress, etc. When you start work on an issue, move it to the "In progress" track.
Whenever you sit down to work on the project, you should be clearly working on a specific issue. If you feel like there's something that needs to be done but isn't in an issue, you should make an issue for that before you start working on it.
When you start work on a new issue, you should create a
feature branch for that issue, and commit your work on that
issue to that branch. Commit messages should refer to that
issue (by number, e.g., Issue #8
) so GitHub can auto-link
the commits to that issue for you.
When you feel like an issue is complete
- Move that card to the
In review
track. - If you haven't already done so, create a Pull Request from
your feature branch onto your
main
branch.
Then step away from that issue for a while, either by working on a different part of the lab, or by doing something unrelated to Software Design. Then come back back to that as a team and review the requirements described in the issue and compare them to the functionality you implemented. Is the issue done done? Are there solid and complete tests that back up the work? Can you break it? Have you tried? Would you bet your career (or at least your next raise) on this working in a customer demo or out in the field?
If you find bugs, document them, either in the existing issue, or through new issues. Then go back to working in the feature branch for that issue, and repeat the whole process.
Once the issue passes review, you should
- Merge the associated feature branch into master by accepting the (perhaps modified) pull request
- Move the issue to the
Done
track (or, fee free to create more tracks as you see fit) - There are ways to automate the moves through the tracks based on what's happening in GitHub, but we won't look at that in detail for this lab.
Now that we have a reliable way to request todo data from our server, we should write a nice client-side application to help us request and view this data.
- Use Angular to build a nice client-side interface which allows the user to:
- filter search results by status, owner, body text, and category
- see returned todo items in a useful, meaningful way including:
- choosing to limit the number of todos displayed
- sorting the todos by various fields
- combinations of sorting, filtering, and limiting
- Your new functionality should be contained in a 'todos' view,
with a 'todo-list' component and probably a service.
Note: You do NOT need to have multiple views of your todos like we provided you for users, and you should feel free to try out a different kind of component that you think would work well to display todos. Think about what information you would want to actually see about the todo to help you manage your tasks.
- You should make some decisions about when to request data from the API,
and when to simply use Angular's filtering tools to change how
the data is displayed.
- You have to use Angular's filtering at least once
- You have to use the server's filtering at least once
- You have to be able to use combinations of filters (implement and test this)
- ❓ Make note of why you choose to do each of those two things the way you did
❓ Answer Question 5 about your filtering in QUESTIONS
Your project should have tests that help you meaningfully practice using continuous integration. You should expand on these tests as appropriate so that your GitHub Actions checks are telling you valuable things about the health of your project.
- As you work, create a branch for a new feature,
move the story from the
icebox
(orbacklog
ornew
) toin-progress
in GitHub Projects, write unit tests (Karma) for the new Angular components and services you are adding and using, write new end-to-end tests (Cypress) for the new views, and address failing tests. - Use pull requests to review code and
merge things into
main
when a feature is working and is tested (with passing tests and decent coverage). Move your story todone
.
In general you'll want to write unit tests (using Karma) for small, focused bits of logic, often in an Angular service, but sometimes in a component.
E2E tests, on the other hand, are typically used to capture the desired functional behavior of specific features or stories.
❓ Answer Questions 6 and 7 about your Karma and E2E tests in QUESTIONS
- ❓ How does the navigation menu (with Home and Users) work in this project? Compare
server/src/main/java/umm3601/Server.java
andclient/src/app/app-routing.module.ts
. Both do a kind of routing that maps a "path" to something that "handles" that path.- What are the "paths" in each case? (Be specific.)
- Trace through an example of a path being handled by both Angular and Javalin.
- Where does the "path" come from? As a user, how might I enter or trigger a particular path?
- What kinds of things do Angular and Javalin map their paths to? (Be specific.)
- What do those targets "do" with that path?
- ❓ What does the
user.service.ts
do? Why is it not just done in theuser-list.component.ts
? - Look over the the test for calling
getUsers()
with anage
parameters inclient/src/app/users/user.service.spec.ts
.- ❓ Where do we tell the service that we want only users with age 25?
- ❓ Where do we tell the mock HTTP system how many requests to expect and what parameters those requests should have?
- ❓ Where do we specify the expected HTTP request type (PUT or GET or DELETE or whatever)?
- ❓ Where do we specify what value the mock HTTP system should return in response to the expected request.
- Look over the E2E test for testing the age filtering in
client/cypress/e2e/user-list.cy.ts
.- ❓ What is the
page
object? Where is that defined? - ❓ Where do we enter the value 27 in the age field?
- ❓ How does Cypress find the age field?
- ❓ How does Cypress update the value in the age field?
- ❓ How does Cypress get all the cards on this page? (Hint: Look at the definition of
getUserCards()
inclient/cypress/support/user-list.po.ts
.) - How does Cypress extract the user's names from the cards?
- ❓ What is the
- You need to use filtering in Angular and filtering on the server each at least one time. Our example filters users by company on the client side in Angular and filters users by role on the server side in Java. There is not just one right answer here, so think about the why.
- ❓ What is one thing you filtered in Angular and why did that approach make sense for that filter?
- ❓ What is one thing you filtered using the server and why did that approach make sense for that filter?
- ❓ What's one piece of "internal" functionality that you chose to
write a unit test (with Karma) for?
- ❓ Why did you choose to test that piece of functionality?
- ❓ What is the "it" for that test, i.e., what Angular tool/method are you testing?
- ❓ List the behaviors you tested via your E2E tests. For each behavior:
- ❓ Why did you test that particular behavior?
- ❓ What is the "it" for that test, i.e., what part of the web app are you testing? (You don't need to tell us how the test works since your code will do that.)