This repository contains the Full Stack Open course part5 exercises
All exercises are saved in a separate commit.
Playwright tests repository for Exercises 5.17.-5.23
We will now create a frontend for the blog list backend we created in the last part. You can use this application from GitHub as the base of your solution. You need to connect your backend with a proxy as shown in part 3.
It is enough to submit your finished solution. You can commit after each exercise, but that is not necessary.
The first few exercises revise everything we have learned about React so far. They can be challenging, especially if your backend is incomplete. It might be best to use the backend that we marked as the answer for part 4.
While doing the exercises, remember all of the debugging methods we have talked about, especially keeping an eye on the console.
Warning: If you notice you are mixing in the async/await
and then
commands, it's 99.9% certain you are doing something wrong. Use either or, never both.
Clone the application from GitHub with the command:
git clone https://github.com/fullstack-hy2020/bloglist-frontend.git
Remove the git configuration of the cloned application
cd bloglist-frontend // go to cloned repository
rm -rf .git
The application is started the usual way, but you have to install its dependencies first:
npm install
npm run dev
Implement login functionality to the frontend. The token returned with a successful login is saved to the application's state user.
If a user is not logged in, only the login form is visible.
If the user is logged-in, the name of the user and a list of blogs is shown.
User details of the logged-in user do not have to be saved to the local storage yet.
NB You can implement the conditional rendering of the login form like this for example:
if (user === null) {
return (
<div>
<h2>Log in to application</h2>
<form>
//...
</form>
</div>
)
}
return (
<div>
<h2>blogs</h2>
{blogs.map(blog =>
<Blog key={blog.id} blog={blog} />
)}
</div>
)
}
Make the login 'permanent' by using the local storage. Also, implement a way to log out.
Ensure the browser does not remember the details of the user after logging out.
Expand your application to allow a logged-in user to add new blogs:
Implement notifications that inform the user about successful and unsuccessful operations at the top of the page. For example, when a new blog is added, the following notification can be shown:
Failed login can show the following notification:
The notifications must be visible for a few seconds. It is not compulsory to add colors.
Change the form for creating blog posts so that it is only displayed when appropriate. Use functionality similar to what was shown earlier in this part of the course material. If you wish to do so, you can use the Togglable component defined in part 5.
By default the form is not visible
It expands when button create new blog is clicked
The form hides again after a new blog is created.
Separate the form for creating a new blog into its own component (if you have not already done so), and move all the states required for creating a new blog to this component.
The component must work like the NoteForm component from the material of this part.
Let's add a button to each blog, which controls whether all of the details about the blog are shown or not.
Full details of the blog open when the button is clicked.
And the details are hidden when the button is clicked again.
At this point, the like button does not need to do anything.
The application shown in the picture has a bit of additional CSS to improve its appearance.
It is easy to add styles to the application as shown in part 2 using inline styles:
const Blog = ({ blog }) => {
const blogStyle = {
paddingTop: 10,
paddingLeft: 2,
border: "solid",
borderWidth: 1,
marginBottom: 5,
};
return (
<div style={blogStyle}>
<div>
{blog.title} {blog.author}
</div>
// ...
</div>
);
};
NB: Even though the functionality implemented in this part is almost identical to the functionality provided by the Togglable component, it can't be used directly to achieve the desired behavior. The easiest solution would be to add a state to the blog component that controls if the details are being displayed or not.
Implement the functionality for the like button. Likes are increased by making an HTTP PUT request to the unique address of the blog post in the backend.
Since the backend operation replaces the entire blog post, you will have to send all of its fields in the request body. If you wanted to add a like to the following blog post:
{
_id: "5a43fde2cbd20b12a2c34e91",
user: {
_id: "5a43e6b6c37f3d065eaaa581",
username: "mluukkai",
name: "Matti Luukkainen"
},
likes: 0,
author: "Joel Spolsky",
title: "The Joel Test: 12 Steps to Better Code",
url: "https://www.joelonsoftware.com/2000/08/09/the-joel-test-12-steps-to-better-code/"
},
You would have to make an HTTP PUT request to the address /api/blogs/5a43fde2cbd20b12a2c34e91 with the following request data:
{
"user": "5a43e6b6c37f3d065eaaa581",
"likes": 1,
"author": "Joel Spolsky",
"title": "The Joel Test: 12 Steps to Better Code",
"url": "https://www.joelonsoftware.com/2000/08/09/the-joel-test-12-steps-to-better-code/"
}
The backend has to be updated too to handle the user reference.
We notice that something is wrong. When a blog is liked in the app, the name of the user that added the blog is not shown in its details:
When the browser is reloaded, the information of the person is displayed. This is not acceptable, find out where the problem is and make the necessary correction.
Of course, it is possible that you have already done everything correctly and the problem does not occur in your code. In that case, you can move on.
Modify the application to sort the blog posts by the number of likes. The Sorting can be done with the array sort method.
Add a new button for deleting blog posts. Also, implement the logic for deleting blog posts in the frontend.
Your application could look something like this:
The confirmation dialog for deleting a blog post is easy to implement with the window.confirm function.
Show the button for deleting a blog post only if the blog post was added by the user.
Define PropTypes for one of the components of your application, and add ESlint to the project. Define the configuration according to your liking. Fix all of the linter errors.
Vite has installed ESlint to the project by default, so all that's left for you to do is define your desired configuration in the .eslintrc.cjs file.
Make a test, which checks that the component displaying a blog renders the blog's title and author, but does not render its URL or number of likes by default.
Add CSS classes to the component to help the testing as necessary.
Make a test, which checks that the blog's URL and number of likes are shown when the button controlling the shown details has been clicked.
Make a test, which ensures that if the like button is clicked twice, the event handler the component received as props is called twice.
Make a test for the new blog form. The test should check, that the form calls the event handler it received as props with the right details when a new blog is created.
In the last exercises of this part, let's do some E2E tests for the blog application. The material above should be enough to do most of the exercises. However, you should definitely read Playwright's documentation and API description, at least the sections mentioned at the end of the previous chapter.
Create a new npm project for tests and configure Playwright there.
Make a test to ensure that the application displays the login form by default.
The body of the test should be as follows:
const { test, expect, beforeEach, describe } = require("@playwright/test");
describe("Blog app", () => {
beforeEach(async ({ page }) => {
await page.goto("http://localhost:5173");
});
test("Login form is shown", async ({ page }) => {
// ...
});
});
Do the tests for login. Test both successful and failed login. For tests, create a user in the beforeEach
block.
The body of the tests expands as follows
const { test, expect, beforeEach, describe } = require("@playwright/test");
describe("Blog app", () => {
beforeEach(async ({ page, request }) => {
// empty the db here
// create a user for the backend here
// ...
});
test("Login form is shown", async ({ page }) => {
// ...
});
describe("Login", () => {
test("succeeds with correct credentials", async ({ page }) => {
// ...
});
test("fails with wrong credentials", async ({ page }) => {
// ...
});
});
});
The beforeEach
block must empty the database using, for example, the reset method we used in the material.
Create a test that verifies that a logged in user can create a blog. The body of the test may look like the following
describe("When logged in", () => {
beforeEach(async ({ page }) => {
// ...
});
test("a new blog can be created", async ({ page }) => {
// ...
});
});
The test should ensure that the created blog is visible in the list of blogs.
Do a test that makes sure the blog can be liked.
Make a test that ensures that the user who added the blog can delete the blog. If you use the window.confirm
dialog in the delete operation, you may have to Google how to use the dialog in the Playwright tests.
Make a test that ensures that only the user who added the blog sees the blog's delete button.
Do a test that ensures that the blogs are arranged in the order according to the likes, the blog with the most likes first.
This task is significantly more challenging than the previous ones.
This was the last task of the section and it's time to push the code to GitHub and mark the completed tasks in the exercise submission system.