d
Altering data in server
We naturally want to store all notes that are created in our application in a backend server somewhere. The json-server package claims to be a so-called REST or RESTful API in its documentation:
Get a full fake REST API with zero coding in less than 30 seconds (seriously)
If we go by the original textbook definition, the json-server does not fit the description of a REST API, but the same is true of most APIs claiming to be RESTful.
We will take a closer look at REST in the next part of the course, but it's good to familiarize ourselves at this point with some of the conventions used by json-server and REST APIs at large. In particular, we will be taking a look at how routes aka. URLs and HTTP request types are conventionally used in REST.
REST
In REST terminology, we refer to individual data objects like the notes in our application as resources. Every resource has an associated unique address, aka. its URL. In the convention used by json-server, we would be able to locate an individual note at the resource URL notes/3, where 3 is the id of the resource. In this convention the notes url would point to a resource collection containing all notes.
Resources are fetched from the server with HTTP GET requests. E.g. an HTTP GET request to the URL notes/3 will return the note that has the id number 3. An HTTP GET request to the notes URL would return a list of all notes.
The creation of a new note resource in the REST convention followed by json-server is done with an HTTP POST request, that is made to the notes URL. The data for the new note resource is sent in the body of the request.
json-server requires that all data is sent in the JSON format. In practice this means that the data must be a correctly formatted string, and that the request must contain the Content-Type request header with the value application/json.
Sending Data to the Server
Let's make the following changes to the event handler responsible for creating a new note:
addNote = event => {
event.preventDefault()
const noteObject = {
content: newNote,
date: new Date(),
important: Math.random() > 0.5,
}
axios .post('http://localhost:3001/notes', noteObject) .then(response => { console.log(response) })}
We create a new object for the note but omit the id property, since it's better to let the server generate ids for our resources!
The object is sent to the server by using the axios post method. The registered event handler logs the response that is sent back from the server to the console.
When we try to create a new note, the following output pops up in console:
The newly created note resource is stored in the value of the data property of the response object.
Sometimes it can be useful to inspect HTTP requests in the Network tab of Chrome developer tools, which was used heavily at the beginning of part 0:
We can use the inspector to check that the headers sent in the POST request are what we expected them to be, and that their values are correct.
Since the data we sent in the POST request was a JavaScript object, axios automatically knew to set the appropriate application/json value for the Content-Type header.
The new note is not yet rendered to the screen. This is because we did not update the state of the App component when we created the new note. Let's fix this:
addNote = event => {
event.preventDefault()
const noteObject = {
content: newNote,
date: new Date(),
important: Math.random() > 0.5,
}
axios
.post('http://localhost:3001/notes', noteObject)
.then(response => {
setNotes(notes.concat(response.data)) setNewNote('') })
}
The new note returned by the backend server is added to the list of notes in our application's state in a familiar fashion, by using the setNotes function and then resetting the note creation form. An important detail to remember is that the concat method does not change the original state, but rather creates a new copy of the list.
Once the data returned by the server starts to have an effect on the behavior of our web applications, we are immediately faced by a whole new set of challenges that arise due to the nature of asynchronous communication among other things. This creates the demand for new debugging strategies. Logging to the console and other means of debugging become more and more important, and we have to develop an understanding of the inner workings of the JavaScript runtime and React components — guessing is no longer an option.
It's beneficial to inspect the state of the backend server e.g. through the browser:
This makes it possible to verify that all the data we intended to send was actually received by the server.
In the next part of the course we will learn to implement our own logic in the backend. We will then take a closer look at tools like postman that help us to debug our server applications. However, inspecting the state of the json-server through the browser is sufficient for our current needs.
NB: in the current version of our application the browser adds the creation date property to the note. Since the clock of the machine running the browser can be wrongly configured, it's much wiser to let the backend server generate this timestamp for us. This is in fact what we will do in the next part of the course.
The code for the current state of our application can be found in the part2-5 branch on github.
Changing the importance of notes
Let's add a button to every note that can be used for toggling its importance.
We make the following changes to the Note component:
const Note = ({ note, toggleImportance }) => {
const label = note.important
? 'make not important' : 'make important'
return (
<li>
{note.content}
<button onClick={toggleImportance}>{label}</button>
</li>
)
}
We add a button to the component and assign its event handler as the toggleImportance function passed in the component's props.
The App component defines an initial version of the toggleImportanceOf event handler function and passes it to every Note component:
const App = () => {
const [notes, setNotes] = useState([])
const [newNote, setNewNote] = useState('')
const [showAll, setShowAll] = useState(true)
// ...
const toggleImportanceOf = id => { console.log( 'importance of ' + id + ' needs to be toggled' ) }
const rows = () => notesToShow.map(note =>
<Note
key={note.id}
note={note}
toggleImportance={() => toggleImportanceOf(note.id)} />
)
return (
// ...
)
}
Notice how every note receives its own unique event handler function, as the id of every note is unique.
E.g. if node.id is 3, the event handler function returned by toggleImportance(note.id) will be:
() => { console.log('importance of 3 needs to be toggled') }
Short reminder. The string printed by the event handler is defined in a Java-like manner by plussing strings:
console.log('importance of ' + id + ' needs to be toggled')
The template string syntax added in ES6 can be used to write similar strings in a much nicer way:
console.log(`importance of ${id} needs to be toggled`)
We can now use the "dollar bracket"-syntax to add parts to the string that will evaluate JavaScript expressions, e.g. the value of a variable. Note that the quotation marks used in template strings differ from the quotation marks used in regular JavaScript strings.
Individual notes stored in the json-server backend can be modified in two different ways by making HTTP requests to the note's unique URL. We can either replace the entire note with an HTTP PUT request, or only change some of the note's properties with an HTTP PATCH request.
The final form of the event handler function is the following:
const toggleImportanceOf = id => {
const url = `http://localhost:3001/notes/${id}`
const note = notes.find(n => n.id === id)
const changedNote = { ...note, important: !note.important }
axios.put(url, changedNote).then(response => {
setNotes(notes.map(note => note.id !== id ? note : response.data))
})
}
Almost every line of code in the function body contains important details. The first line defines the unique url for each note resource based on its id.
The array find method is used to find the note we want to modify, and we then assign it to the note variable.
After this we create a new object that is an exact copy of the old note, excluding the important property. The code for creating the new object may seem a bit strange:
const changedNote = { ...note, important: !note.important }
The command uses the object spread syntax which is not yet a standard part of the JavaScript language specification.
In practice { ...note } creates a new object, with copies of all the properties from the note object. When we add properties inside the brackets after the spreaded object, e.g. { ...note, important: true }, then the value of the important property of the new object will be true. In our example the important property gets the negation of its previous value in the original object.
There's a few things to point out. Why did we make a copy of the note object we wanted to modify, when the following code also appears to work:
const note = notes.find(n => n.id === id)
note.important = !note.important
axios.put(url, note).then(response => {
// ...
This is not recommended because the variable note is a reference to an item in the notes array in the component's state, and as we recall we must never mutate state directly in React.
It's also worth noting that the new object changedNote is only a so-called shallow copy, meaning that the values of the new object are the same as the values of the old object. If the values of the old object were objects themselves, then the copied values in new object would reference the same objects that were in the old object.
The new note is then sent with a PUT request to the backend where it will replace the old object.
The callback function sets the component's notes state to a new array that contains all the items from the previous notes array, except for the old note which is replaced by the updated version of that note returned by the server:
axios.put(url, changedNote).then(response => {
setNotes(notes.map(note => note.id !== id ? note : response.data))
})
This is accomplished with the map method:
notes.map(note => note.id !== id ? note : response.data)
The map method creates a new array by mapping every item from the old array into an item in the new array. In our example, the new array is created conditionally so that if note.id !== id is true, we simply copy the item from the old array into the new array. If the condition is false, then the note object returned by the server is added to the array instead.
This map trick may seem a bit strange at first but it's worth spending some time wrapping your head around it. We will use this method many times throughout the course.
Extracting communication with the backend into a separate module
The App component has become quite bloated after we added the code for communicating with the backend server. In the spirit of the single responsibility principle, we deem it wise to extract this communication into its own module.
Let's create a src/services directory and add a file there called notes.js:
import axios from 'axios'
const baseUrl = 'http://localhost:3001/notes'
const getAll = () => {
return axios.get(baseUrl)
}
const create = newObject => {
return axios.post(baseUrl, newObject)
}
const update = (id, newObject) => {
return axios.put(`${baseUrl}/${id}`, newObject)
}
export default {
getAll: getAll,
create: create,
update: update
}
The module returns an object that has three functions (getAll, create ja update) for dealing with notes as its properties. The functions directly return the promises returned by the axios methods.
Komponentti App saa moduulin käyttöön import-lauseella:
import noteService from './services/notes'
const App = () => {
The functions of the module can be used directly with the imported variable noteService like this:
const App = () => {
// ...
useEffect(() => {
noteService .getAll() .then(response => { setNotes(response.data) }) }, [])
const toggleImportanceOf = id => {
const note = notes.find(n => n.id === id)
const changedNote = { ...note, important: !note.important }
noteService .update(id, changedNote) .then(response => { setNotes(notes.map(note => note.id !== id ? note : response.data)) }) }
const addNote = (event) => {
event.preventDefault()
const noteObject = {
content: newNote,
date: new Date().toISOString(),
important: Math.random() > 0.5
}
noteService .create(noteObject) .then(response => { setNotes(notes.concat(response.data)) setNewNote('') }) }
// ...
}
export default App
We could take our implementation one step further. When the App component uses the functions, it receives an object that contains the entire response for the HTTP request:
noteService
.getAll()
.then(response => {
setNotes(response.data)
})
The App component only uses the response.data property of the response object.
The module would be much nicer to use, if instead of the entire HTTP response we would only get the response data. Then using the module would look like this:
noteService
.getAll()
.then(initialNotes => {
setNotes(initialNotes)
})
We can achieve this by changing the code in the module as follows (the current code contains some copy-paste but we will tolerate that for now):
import axios from 'axios'
const baseUrl = 'http://localhost:3001/notes'
const getAll = () => {
const request = axios.get(baseUrl)
return request.then(response => response.data)
}
const create = newObject => {
const request = axios.post(baseUrl, newObject)
return request.then(response => response.data)
}
const update = (id, newObject) => {
const request = axios.put(`${baseUrl}/${id}`, newObject)
return request.then(response => response.data)
}
export default {
getAll: getAll,
create: create,
update: update
}
We no longer return the promise returned by axios directly. Instead, we assign the promise to the request variable and call its then method:
const getAll = () => {
const request = axios.get(baseUrl)
return request.then(response => response.data)
}
The last row in our function is simply a more compact expression of the same code as shown below:
const getAll = () => {
const request = axios.get(baseUrl)
return request.then(response => { return response.data })}
The modified getAll function still returns a promise, as the then method of a promise also returns a promise.
After defining the parameter of the then method to directly return response.data, we have gotten the getAll function to work like we wanted it to. When the HTTP request is successful, the promise returns the data sent back in the response from the backend.
We have to update the App component to work with the changes made to our module. We have to fix the callback functions given as parameters to the noteService object's methods, so that they use the directly returned response data:
const App = () => {
// ...
useEffect(() => {
noteService
.getAll()
.then(initialNotes => { setNotes(initialNotes) })
}, [])
const toggleImportanceOf = id => {
const note = notes.find(n => n.id === id)
const changedNote = { ...note, important: !note.important }
noteService
.update(id, changedNote)
.then(returnedNote => { setNotes(notes.map(note => note.id !== id ? note : returnedNote)) })
}
const addNote = (event) => {
event.preventDefault()
const noteObject = {
content: newNote,
date: new Date().toISOString(),
important: Math.random() > 0.5
}
noteService
.create(noteObject)
.then(returnedNote => { setNotes(notes.concat(returnedNote)) setNewNote('')
})
}
// ...
}
This is all quite complicated and attempting to explain it may just make it harder to understand. The internet is full of material discussing the topic, like this.
The "Async and performance" book from the You do not know JS book series explains the topic well, but the explanation is many pages long.
Promises are central to modern JavaScript development and it is highly recommended to invest a reasonable amount of time into understanding them.
Cleaner syntax for defining object literals
The module defining note related services currently exports an object with the properties getAll, create and update that are assigned to functions for handling notes.
The module definition was as follows:
import axios from 'axios'
const baseUrl = 'http://localhost:3001/notes'
const getAll = () => {
const request = axios.get(baseUrl)
return request.then(response => response.data)
}
const create = newObject => {
const request = axios.post(baseUrl, newObject)
return request.then(response => response.data)
}
const update = (id, newObject) => {
const request = axios.put(`${baseUrl}/${id}`, newObject)
return request.then(response => response.data)
}
export default {
getAll: getAll,
create: create,
update: update
}
The module exports the following, rather peculiar looking object:
{
getAll: getAll,
create: create,
update: update
}
The names to the left of the colon in the object definition are the keys of the object while the names to the right are variables that are defined inside of the module.
Since the name of thy keys and the assigned variables are the same, we can write the object definition with a more compact syntax:
{
getAll,
create,
update
}
As a result the module definition gets simplified into the following form:
import axios from 'axios'
const baseUrl = 'http://localhost:3001/notes'
const getAll = () => {
const request = axios.get(baseUrl)
return request.then(response => response.data)
}
const create = newObject => {
const request = axios.post(baseUrl, newObject)
return request.then(response => response.data)
}
const update = (id, newObject) => {
const request = axios.put(`${baseUrl}/${id}`, newObject)
return request.then(response => response.data)
}
export default { getAll, create, update }
In this more compact definition we are using a new feature that was added to JavaScript in the ES6 language specification. The feature specifically enables this more compact way of defining objects.
To demonstrate this feature, let's consider a situation where we have the following values assigned to variables:
const name = 'Leevi'
const age = 0
In older versions of JavaScript we had to define an object like this:
const person = {
name: name,
age: age
}
Because the name of the variables and the name of keys of the object are the same, it's enough to simply write the following in ES6 JavaScript:
const person = { name, age }
The result is identical in both expressions. They both create an object with a name property with the value Leevi and an age property with the value 0.
Promises and errors
If our application allowed users to delete notes, we could end up in a situation where a user tries to change the importance of a note that has already been deleted from the system.
Let's simulate this situation by making the getAll function of the note service return a "hardcoded" note that does not actually exist in the backend server:
const getAll = () => {
const request = axios.get(baseUrl)
const nonExisting = {
id: 10000,
content: 'Tätä muistiinpanoa ei ole palvelimelta',
date: '2017-12-10T17:30:31.098Z',
important: true,
}
return request.then(response => response.data.concat(nonExisting))
}
When we try to change the importance of the hardcoded note, we see the following error message in the console. The error says that the backend server responded to our HTTP PUT request with a status code 404 not found.
The application should be able to handle these types of error situations gracefully. Users won't be able to tell that an error has actually occurred unless they happen to have their console open. The only way the error can be seen in the application is that clicking the button has no effect on the importance of the note.
We had previously mentioned that a promise can be in one of three different states. When an HTTP request fails, the associated promise is rejected. In our current code we do not handle this rejection in any way.
The rejection of a promise is handled by providing the then method with a second callback function, that is called in the situation where the promise is rejected.
The more common way of adding a handler for rejected promises is to use the catch method.
In practice the error handler for rejected promises is defined like this:
axios
.get('http://example.com/probably_will_fail')
.then(response => {
console.log('success!')
})
.catch(error => {
console.log('fail')
})
If the request fails, the event handler registered with the catch method gets called.
The catch method is often utilized by placing it deeper within the promise chain.
When our application makes an HTTP request, we are in fact creating a promise chain:
axios
.put(`${baseUrl}/${id}`, newObject)
.then(response => response.data)
.then(changedNote => {
// ...
})
The catch method can be used to define a handler function at the end of a promise chain, that is called once any promise in the chain throws an error and the promise becomes rejected.
axios
.put(`${baseUrl}/${id}`, newObject)
.then(response => response.data)
.then(changedNote => {
// ...
})
.catch(error => {
console.log('fail')
})
Let's use this feature and register an error handler in the App component:
const toggleImportanceOf = id => {
const note = notes.find(n => n.id === id)
const changedNote = { ...note, important: !note.important }
noteService
.update(id, changedNote).then(returnedNote => {
setNotes(notes.map(note => note.id !== id ? note : returnedNote))
})
.catch(error => { alert( `muistiinpano '${note.content}' on jo valitettavasti poistettu palvelimelta` ) setNotes(notes.filter(n => n.id !== id)) })}
The error message is displayed to the user with the trusty old alert dialog popup, and the deleted note gets filtered from the state.
Removing an already deleted note from the application's state is done with the array filter method, that returns a new array that only contains the items from the list, that the function passed as a parameter returns true:
notes.filter(n => n.id !== id)
It's probably not a good idea to use alert in more serious React applications. We will soon learn a more advanced way of displaying messages and notifications to users. However, there are situations where a simple battle tested method like alert can function as a starting point. The more advanced method can then be added in later, if there's the time and energy for it.
The code for the current state of our application can be found in the part2-6 branch on github.