Redux Flashcards
What will we learn?
everything we need to know to build real, complex apps with Redux
work on a real bug tracking app with a node backend
by end of course, know and understand Redux inside/out
data:image/s3,"s3://crabby-images/71dc9/71dc91b458af0ad7418f96b9cda2a1baa5e4785c" alt=""
What is Redux?
data:image/s3,"s3://crabby-images/7cf7b/7cf7be624640e0ea8e03e37727e322963c0ebf01" alt=""
A state management library for Javascript applications
doesn’t care what library we use for UI development
instead of scattering app state across UI
store state in a single JS object (store)
single source of truth
like a “database” for front end
components get state from store
data is updated in a single place
can see how data changed, why, when and where it came from
data:image/s3,"s3://crabby-images/484f5/484f5642a2efc3be2e7e74c2d557812f6494145d" alt=""
Why do we need redux (state managment)?
data:image/s3,"s3://crabby-images/427ed/427ed49ac880d69966aae383befd1061d6a7a659" alt=""
complex UI, need to keep in sync
data updates from backend request, user interaction, etc. changes must be reflected immediately
data:image/s3,"s3://crabby-images/27720/27720cd1f74d649b3c493e743c6cc21a3d51ad75" alt=""
What are the pro’s of Redux?
debugging
transparent state changes
can reload state, view actions and see UI changes
cache/preserve page state
entire app state available on client in JS object (store)
don’t have to reload data from server
centralized state
data is available in one place, accessible by all parts of UI
data:image/s3,"s3://crabby-images/278df/278df986046addb7de5f5cc9caaae775ed2073ff" alt=""
When is redux NOT right for my project?
data:image/s3,"s3://crabby-images/16e7e/16e7e6b7c4cba511fef8898a1243fdc885084832" alt=""
must consider constraints and problem domain
A real software engineer is an active problem solver
not every application requires Redux
DON’T USE
if load data and display statically (not changing)
if it’s a tight budget
if the UI is very simple
**it will be complex and slow you downl
What is functional programming?
data:image/s3,"s3://crabby-images/d524f/d524fe7ccb9dd812ab3045034c39de81fe471603" alt=""
Redux is built on top of functional programming
A good background is critical to building redux apps
One of the programming paradigms
Each paradigm has rules for how to structure code to solve problems
invented in 1950s
decomposing a problem into small, re-usable functions
can compose functions to build complex pipelines
Pros
can run in parrallel
exploit multiple cores
Benefits:
more concise
easier to debug
easier to test
more scalable
data:image/s3,"s3://crabby-images/cbbd8/cbbd8880bafe85959b70c1292a775b4d49da7d51" alt=""
Why are functions first class citizens in Javascript?
can treat them like any other variables
assign them as variables
pass them as arguments
return them from other functions
data:image/s3,"s3://crabby-images/f51af/f51af00e3711e40828a138c0a0d45842ade54832" alt=""
What are higher order functions?
a function that…
takes a function as an argument
returns a function
both!
Why?
instead of working on strings, numbers or booleans
it goes higher to operates on functions
What is the problem with this approach?
data:image/s3,"s3://crabby-images/bec42/bec42ff917c953fbdc3f02ac0b52d58106dc02dc" alt=""
expression is read right to left
trim, make lowercase, wrap in div
so many parenthesis
Soln?
What is currying?
data:image/s3,"s3://crabby-images/c6f9b/c6f9b3b48928309a9a36ff4845799cca2b1d7db3" alt=""
Solves this problem:
functional pipeline
function in line requires two parameters (fns)
only have one fn as parameter
How?
allows us to take a fn with N arguments
convert it to an fn with 1 argument
What?
instead of separating args with commas “,”
separate with ( )
data:image/s3,"s3://crabby-images/08dc9/08dc90b7580180435de7b1cbb6e5892d9ecba326" alt=""
What is the challenge with this code?
data:image/s3,"s3://crabby-images/2c2bd/2c2bd45973677fda1e109dfca3f32104b2fe3657" alt=""
wrap function expects two arguments
if we pass type as argument
pipeline will throw an error
Why?
it requires a function as argument, not string
Soln?
currying
data:image/s3,"s3://crabby-images/8fa83/8fa839014667a466656aa2280304ed05eab23a36" alt=""
What are pure functions?
data:image/s3,"s3://crabby-images/c6800/c680035dcc0dbf9f8c964c223cbfc699cac90b3d" alt=""
a function that given the same args, produces the same result
data:image/s3,"s3://crabby-images/d7c8c/d7c8c772045727bcd456833977081571f59e58e4" alt=""
NOT:
random values
current date/time
global state
Why?
These change
Redux?
reducers must be pure
other fns don’t have to be
What are the benefits of pure functions?
data:image/s3,"s3://crabby-images/8e104/8e104d6e10b8013cb4858c029112386c098311d3" alt=""
Self documenting
everything a fn needs is clearly identified in parameter signature
easier to test
no global state
concurrency
because not using global state
can run in parallel
cacheable
can store result in cache and use in future
useful in programs with intensive computations
if we know certain computations produce same results
can optimize by storing in a cache
data:image/s3,"s3://crabby-images/92be8/92be842901a82c55a0a19ab53aa4d0ab5da8ad73" alt=""
What is immutability?
data:image/s3,"s3://crabby-images/72aac/72aacf035c7d9775ac48fe28b57fc62e102032b5" alt=""
Goes hand in hand with pure functions
data:image/s3,"s3://crabby-images/1e8bf/1e8bf59d947a1e7290a328886f0b87e3274e20be" alt=""
once we create an object, cannot be changed
must take a copy and change that copy
Ex. Strings are immutable, create copy, original string not affected
Objects and arrays can be modified directly, not immutable
Does “const” prevent mutability?
No
it prevents re-assignment
common misconception
Ex. cannot re-assign book to another object
data:image/s3,"s3://crabby-images/e11ea/e11ea5fb252f1d824ee5137f361437e9642954ee" alt=""
What are the benefits of immutability?
Predicatable apps
if we call a fn, pass an obj
obj won’t get changed, no suprises down road
Fast change detection
React needs to know when changed
creating a new obj, stored in memory
React compares objects by references
this is fast operation
in contrast, without immutability
React has to compare all properties in two objects to see it’s changed
slow operation
Concurrency
can run in parallel
not going to affect something that affects system as a whole
data:image/s3,"s3://crabby-images/d7dce/d7dce45fd508ff0a913d18092b9db8f00493467e" alt=""
What is the problem with this code?
Spread operator does a shallow copy
nested objects are copied by reference (not value)
Ex. Both person2 and person2copy have the same reference to “address” in memory (not copied by value) copied by reference
Soln?
Do a deep copy
set a new nested object using spread operator
data:image/s3,"s3://crabby-images/c4ba4/c4ba4b688075ded904e323e8063eaa5e6d457d09" alt=""
How do we copy an array and insert a number at a specific location?
data:image/s3,"s3://crabby-images/f9f6c/f9f6cc2bf38aac93cc6df165223d1e046af8d30d" alt=""
How do we update an array using immutability?
Hint:.map ( ) method
data:image/s3,"s3://crabby-images/856b4/856b49a8a9887dcd940245381c6d5d52401b5028" alt=""
How do we enforce immutability?
data:image/s3,"s3://crabby-images/638fb/638fbc1bf68d42b5ff2d08717829003c9049018e" alt=""
By default, Javascript doesn’t do this
Why?
It’s a multi paradigm language
it doesn’t enforce immutability without libraries
Soln?
Libraries that enforce real immutability
Different people love different tools
Immutable.js (by Facebook)
Immer (becoming trendy, Mosh loves it)
What is immutable.js?
data:image/s3,"s3://crabby-images/1845c/1845ca4fcd237cd564b83525d7c9824ac0ef2c0f" alt=""
Hint: npm i immutable
a library by Facebook
provides a bunch of immutable data structures
*Mosh prefer’s another library
How?
instead of a plan JS object
use one of the datastructures from this library
Problems?
have to learn a new API
cannot access obj properties with . or [] notation
have to use .get( ) method
integration with other libraries
hard to integrate with other libraries that expect plain JS objects
have to call .toJS( ) method to convert to plain JS objects
data:image/s3,"s3://crabby-images/eebb0/eebb01fe71aa42258d637af083ccef89f8cf93ff" alt=""
What is Immer?
Hint: Immutability library (npm i immer)
almost as popular as immutable.js
data:image/s3,"s3://crabby-images/27095/270959cc2bc5fcadb66b217f2566d357982ed6a4" alt=""
Pros
writing code as if mutating objects but actually not
better than using spread operator (get’s nasty with nested objs)
Write code to solve this?
Hint: function composition
data:image/s3,"s3://crabby-images/3b840/3b840158b426e1abef099db6d8d98e64fee5311c" alt=""
data:image/s3,"s3://crabby-images/c552a/c552aefffe087f65129745d27dcfca10c3df914c" alt=""
What are the three building blocks in Redux applications?
Store
single JS Object that includes application state
Actions
plain JS Objects that represent what happened
aka events (what happened)
Reducers
one or more functions responsible for updating a slice of Store (single JS object)
aka event handlers or processors
pure functions, don’t mutate arguments, no side effects
data:image/s3,"s3://crabby-images/2a060/2a060d3f45926350d974459515e984c7bf21787e" alt=""
How do Redux building blocks work together?
Action object is dispatched to Store
Store calles the Reducer
Ex. When user performs an action (add item to shopping cart)
Create an Action item and dispatch it
Store object has dispatch method that takes an action
Store object forwards action to reducer
Reducer is not called directly, we only work with the Store
Store is in charge of calling the reducer
Reducer computes new state and returns it to Store
Store sets state internally and notifies UI components directly
data:image/s3,"s3://crabby-images/b9751/b97512324b2daf4f93acfe21ee7a21e32d3b9508" alt=""
Write code to solve this?
Hint: immutabile objects / arrays
data:image/s3,"s3://crabby-images/28202/28202c520fdc95dfa1f7aa18918d82f781d5a0f7" alt=""
data:image/s3,"s3://crabby-images/eba1b/eba1b63b332548ff22e00773d99c3be25d4dd6b8" alt=""
What is a store?
data:image/s3,"s3://crabby-images/5bafa/5bafa9b8b0cda34beabffac44b5c1dcb4c501625" alt=""
simple API interface
JS Object
includes properties:
.dispatch
method for dispatching actions
.subscribe
method for subscribing to store
get notified everytime state of store changes
used by UI layer
.getState
get current state of store
to change state of store, have to dispatch an action
Store object has dispatch method that takes an action
Store object forwards action to reducer
Reducer is not called directly, we only work with the Store
Store is in charge of calling the reducer
Reducer computes new state and returns it to Store
Store sets state internally and notifies UI components directly
data:image/s3,"s3://crabby-images/4b77f/4b77f94f6885075bc46bf1ae75092482bf4c8efe" alt=""
What is a reducer?
A function that takes current instance of the store (JS state object) and returns an updated store
Returns?
using spread operator or immutable library to return an updated store
How?
supply store and an action as arguments
action is a plain JS object describing what just happened
lets reducer know what to update in store (JS state object)
Many?
each reducer is responsible for updating a specific part of the store
Ex. Many departments in a store, each department has a manager
data:image/s3,"s3://crabby-images/e46bf/e46bf6f696c63edfeb426a241f776e76d5c3587f" alt=""
What is an action?
argument to the reducer function
tells reducer what properties in store to update
actions that happen in our applications (events)
Ex. UserLogin, UserLogout
based on the type of action, reduce will know what parts of store (state) to update
data:image/s3,"s3://crabby-images/1ec02/1ec0209c1db782ab5d9108cb63bc0ecd383859a1" alt=""
Why is redux designed this way (ie. to dispatch actions)?
data:image/s3,"s3://crabby-images/a3ccf/a3ccf50f35048c281fe9a17073ead25fcafbb93b" alt=""
Dispatch is like entry point to store (JS state object)
every action is sent through same entry point
central place to control what should happen every time a user performs an action
can log every action
can implement undue/redue mechanism
data:image/s3,"s3://crabby-images/82864/828646398222aa0c0c7b75c4e9f2373568cc3a67" alt=""
What are the four steps to follow when building a redux app?
Design the Store
What do you want keep in the store?
Define Actions
What are actions user can perform in application?
Create Reducers
Take an action, return updated state
Setup Store based on Reducer
data:image/s3,"s3://crabby-images/be4fe/be4fed802df6b366d7d60aa92ffae19b6b3d168a" alt=""
How would we design the store of a real world application for tracking bugs?
An object with two properties (slices):
bugs - list of bugs
current user - initially null, then set when user login
*two slices, requires two reducers
data:image/s3,"s3://crabby-images/975c2/975c247186e6e47344062cfc4455fb335ce0207d" alt=""
What actions would a user perform in a real bug tracking application?
data:image/s3,"s3://crabby-images/46cec/46cec11c169f0a143f22a697f63ce58a7aa3b9f8" alt=""
An action is just a plain JS object that describes what happened
must have a type property (required)
Payload property contains minimum data including description or ID for an action
description can be any serializeable data type
because description must be able to store on disk, reload later
use strings so we can view description of what happend
can store form data in description, etc. (bug type, who logged, etc)
ADD_BUG
bugAdded
common conventions for type in Redux apps
(vs. numberss not descriptive)
Actions (events):
Add a bug
Change status of bug
filter list of bugs
change sort order
Mark as resolved
Delete a bug
data:image/s3,"s3://crabby-images/b59a1/b59a15d28467db405c952e50c6299ca02ee8c525" alt=""
How do we create a reducer?
data:image/s3,"s3://crabby-images/12c3c/12c3cd5e84ad6524fd423c798538f77f7fae5ce3" alt=""
data:image/s3,"s3://crabby-images/f19e7/f19e7d0647e530c9862cbf4fcbdb253c33d90fd6" alt=""
What’s the problem with this code?
data:image/s3,"s3://crabby-images/df7e1/df7e1462e6efc47df4575c0d1eace991bb518635" alt=""
In real app
payload contains minimum required information
Ex. bugAdded -> only need ID, not description too
Use immutable library in real app (not spread operator)
How do we implement this logic using switch and case?
data:image/s3,"s3://crabby-images/ab08c/ab08cd42e9f08cac337c17764d82961fed336408" alt=""
Reducer is a pure function
everything it needs is passed as arguments (only dependencies)
given the same input, returns same output
doesn’t touch DOM elements
doesn’t work with global state
No API calls
data:image/s3,"s3://crabby-images/471a4/471a47859b1f8b105e919795ffc774cb56840760" alt=""
How do we create a store?
Hint: single JS object that contains app state
data:image/s3,"s3://crabby-images/53f74/53f74668d7ea83651ea7fd5f6637398aa8df1050" alt=""
import {createStore} from ‘redux’;
import reducer from ’./reducer’;
data:image/s3,"s3://crabby-images/2b83a/2b83a945c008281f41a97314189c71c6c8839a39" alt=""
How do we create a store?
data:image/s3,"s3://crabby-images/cd9ea/cd9ea5201b59193006b1b22525441f09b69d9f6f" alt=""
data:image/s3,"s3://crabby-images/65182/651821db8740857e01f882fb19511867e60fe45a" alt=""
How do we dispatch an action?
data:image/s3,"s3://crabby-images/39515/39515c0fa6ad564bc129182609f35b045f521c45" alt=""
In real app,
when user clicks add button
we are going to dispatch an action (raise an event)
data:image/s3,"s3://crabby-images/f8110/f8110c7b635b85c220d4ecc8838fe00c13647e1c" alt=""
How do we subscribe to the store?
Hint: get current state of store everytime state changes
data:image/s3,"s3://crabby-images/04c86/04c86dfaafea14ff8ef5ba70dbd7e2022f3fe1f9" alt=""
store.subscribe ( )
takes a fn
called everytime a change happens in the UI
UI components should subscribe to store so they are notified when changes happen
data:image/s3,"s3://crabby-images/5d6a4/5d6a438a265263bac0fe4c9603814e98f5e91436" alt=""
What does the subscribe method return?
data:image/s3,"s3://crabby-images/8445e/8445e47de94cbaad6481749b7c156e05de99b20d" alt=""
a fn for unsubscribing from store
possible user navigates away from current page
in new page, don’t have UI component
don’t want to be subscribed to a UI component that is not visable
this can create memory leaks
data:image/s3,"s3://crabby-images/95047/95047c50a7ed8017ea6e3d4a24439fb67483b708" alt=""
What is the redux workflow?
Hint: code example
data:image/s3,"s3://crabby-images/725fc/725fcc07e6dc67c7d0cc1deea58b7358b8d9e961" alt=""
When we dispatch an action
store.dispatch( { action } )
store calls reducer
state = reducer (state = [] , action)
store gives reducer current state in store and action passed to store
reducer returns a new state to store
based on logic / type of action (switch case)
notifies subscribers of changed state
data:image/s3,"s3://crabby-images/4f218/4f2188ee3feb040e2dd43917a3cdf4618eac8bfd" alt=""
What’s the problem with this code?
data:image/s3,"s3://crabby-images/05cf2/05cf2228ac9f98b1ab029ee6f20e6626e7a57259" alt=""
we hard coded the action type (string)
if we change the action type name later
will have to change in multiple places or get a bug
Soln?
create a file actionTypes.js
data:image/s3,"s3://crabby-images/abb46/abb46797d4d710aafce7d90b4699b0f6e9f8bdb2" alt=""
What’s the problem with this implementation?
data:image/s3,"s3://crabby-images/880c6/880c60908c57061ebd7d421bea1bccc45623b965" alt=""
dispatching an action is not easy
have to type the entire structure
if re-using action in multiple places, will duplicate code
Soln?
Create a fn that creates action object
data:image/s3,"s3://crabby-images/3df3a/3df3a2a75062c551512a57da6f0b010c87686c8c" alt=""
How do we implement BUG_RESOLVED?
data:image/s3,"s3://crabby-images/c25b7/c25b76257ea4a91e9e80f0864761b4607d5bd1c1" alt=""
Create the action first
plan JS object w/ type + payload
store.dispatch ( action )
Create the reducer next
returns an updated state object to store
store.subscribe ( observer ) notifies observers of state change
Pass action to store
store.dispatch ( action )
data:image/s3,"s3://crabby-images/f862d/f862d0aca777dd2175602d5e184ceae4944a45db" alt=""
What is a good idea when learning a new tool?
Think about how that tool works and how it was built
Mosh is going to show us how Redux works / is built!
We are going to build Redux from scratch
How do we build the store from scratch?
state property is a private property
How do we create a private property?
data:image/s3,"s3://crabby-images/2f3ec/2f3ec24573e9503fe5c612c9d46aaa05ed711ee4" alt=""
How do we implement the dispatch method in our custom store?
data:image/s3,"s3://crabby-images/d414a/d414a5fc6eedcb588b97e56d8a26766c42059e83" alt=""
Call the reducer to get the new state
Notify the subscribers
data:image/s3,"s3://crabby-images/0a78f/0a78f70d604347123ba03357368d5cab9d4650fa" alt=""
How do we implement the subscribe method from scratch?
data:image/s3,"s3://crabby-images/05d88/05d88faadeddec4c3f9371829394e699ea8081e1" alt=""
How do we use Redux Dev Tools?
Hint: A powerful tool for debugging redux applications
data:image/s3,"s3://crabby-images/8c1dd/8c1dd73fe6d44da8934203ad0b74defae302d3b7" alt=""
We need to pass extension to createStore ( ) method
Add:
window.__REDUX_DEVTOOLS_EXTENSION__ && window.__REDUX_DEVTOOLS_EXTENSION__()
A powerful tool for debugging redux applications
data:image/s3,"s3://crabby-images/51500/515005df9ccebf3d7fa74bfc654f0dfc61cb2a75" alt=""
What is the log monitor?
Hint: Redux Dev Tools
a plain list of actions that have been applied
Action + state after action applied
data:image/s3,"s3://crabby-images/2b3c8/2b3c8926e75de9b107ef80aecac3da2361e70088" alt=""
What is the Chart field?
Hint: Redux Dev tools
show actions in a visual way
Ex. DollarShaveClub’s React/Redux Actions Chart
data:image/s3,"s3://crabby-images/fd5af/fd5af7cb2719ef8d4bd9f9dc3238b0d7bd152713" alt=""
What is the inspector field?
Hint: Redux Dev tools
data:image/s3,"s3://crabby-images/3c7f2/3c7f23f0a36b1c9e86b7bd25fab566e923f16ea9" alt=""
diff shows you how tab has changed
data:image/s3,"s3://crabby-images/a7d76/a7d762d2aed1156f245f76b0202ae13d0af9db76" alt=""
How can I identify the source of problem in my Redux app?
data:image/s3,"s3://crabby-images/0f995/0f9958bc50c190cc0ba5c1b0fe954a4390645654" alt=""
using Redux Dev Tools:
1. Look at actions dispatch - ensure right action was dispatched, if not dispatch that action
- Check to ensure action was carrying the right data, if not modify action creator (responsible for action)
3. Check state tab - ensure state was updated properly, if not modify reducer function (responsible for state)
data:image/s3,"s3://crabby-images/d4635/d4635a6fdb544626c9d0eca6904353909a21c2f4" alt=""
What is the test tab?
Hint: Redux Dev Tools
Generates basic boiler plate for testing action
data:image/s3,"s3://crabby-images/b4229/b42292805d18302341702c464723a817eb8d9122" alt=""
How do we enable tracing in redux dev tools?
Hint: $ npm i redux-devtools-extension
data:image/s3,"s3://crabby-images/fc486/fc486020d4d8c38952e0f20730b75130417fec55" alt=""
Where in source code a given action was dispatched
import in store module
replace window._REDUX_DEVTOOLS_EXTENSION
with devToolsEnhancer()
call this function, get a function returned
returns a storeEnhancer function
data:image/s3,"s3://crabby-images/d397f/d397ffd8fb2f90ea1db093527e2a6e0be405f77c" alt=""
How do we configure Redux Dev tools tracing function?
data:image/s3,"s3://crabby-images/7c686/7c686c5c80a3bdf528421c931242a09ebb5345cb" alt=""
maps line of code in bundle and source code
add devtool: “source-map” to webpack.config.js
app.js.map
settings -> code editor -> vsCode -> pwd (working directory path)
/Users/kingarsalon/Desktop/GitHub/redux/Source Code/redux-starter
data:image/s3,"s3://crabby-images/62899/62899cb9e04218bf3a472d0536bd3c4adec5b843" alt=""
What is exporting the store?
data:image/s3,"s3://crabby-images/91144/91144f5521308158be2c587eb7ff83d963d3fb4b" alt=""
Very powerful feature of redux
Can store all interactions of user with our application in a single JSON file
then, reload the state from that JSON file later
Why?
Easy debugging
don’t have to launch application and follow all steps leading to a bug
can simply reload app in same state as user
look at all actions
look at state updates
Log Rocket does this for every user in production
download from Redux Dev Tools
upload into Redux Dev Tools
What is one of the complaints about Redux?
data:image/s3,"s3://crabby-images/fd7fb/fd7fb3f2307ea28b28bcd1956a81b7bbc87862bd" alt=""
It makes you write so much boiler plate code
So…
Mosh is going to show us
How to write clean, concise redux code
data:image/s3,"s3://crabby-images/0754e/0754e2aac0c91e7c1b80e210dc8db5760cf40dbe" alt=""
How do we structure our redux code in a clean way?
Hint: completely isolate from UI code
data:image/s3,"s3://crabby-images/e5b65/e5b652aea06c0ba3585f3f0476109d8eeb708c13" alt=""
UI and state management are two different concerns
Move all redux code into a separate folder
next, group all files by feature (subdomain)
each subdomain should contain it’s own:
actions
actionTypes
reducers
Why?
could have hundreds of actions
break down into small subdomains or bounded contexts
Ex. Auth domain
all about authorization and authentication
actions - login, logout, changePassword
Ex. Bugs domain
data:image/s3,"s3://crabby-images/500b6/500b60a66a27f99da42d19d2349f9fb50aaa3ba7" alt=""
What is the dux pattern?
Hint: re_dux_
data:image/s3,"s3://crabby-images/89fb8/89fb89a8fab819b4a3263f4f27bc65dc39b79228" alt=""
a bundle of actions, actionTypes and reducer for a specific domain
Why?
instead of having separate files:
actionTypes.js, actions.js, reducer.js
combining all for particular domain into one file
makes it easier to edit, update redux domain
any changes to that domain, go to one file
data:image/s3,"s3://crabby-images/92625/926252ef6f94dff4ebef8704824d6884e7215cda" alt=""
What rules must we follow when implementing the dux (redux) pattern?
reducer
has to be a default export in the module
action creators
must export individual action creators
data:image/s3,"s3://crabby-images/dcb07/dcb0737ae73ac84d4309fd0a0113c5d313a57726" alt=""
What is the Redux Toolkit?
Hint: npm i @reduxjs/toolkit
data:image/s3,"s3://crabby-images/c9b5a/c9b5ab9392c6d365952400494488f59f210c9ea5" alt=""
Redux team officially recommends Redux Toolkit
to simplify your code
library provides a bunch of helper functions (utility functions) to simplify redux code
How can we refactor this code?
data:image/s3,"s3://crabby-images/655e4/655e4a080a4de1ca2a9eed0c1e57e8a470324f47" alt=""
npm i @reduxjs/toolkit
configureStore ( )
wraps createStore ( ) fn and development tools
includes:
reduxDevTools
we don’t have to manually import this anymore
utility functions to help with calling APIs
able to dispatch asynch actions (calling API)
without this function, will have to apply middleware to dispatch async functions for calling APIs
data:image/s3,"s3://crabby-images/79141/79141c4d75c174c9bbeeecdd6d6e2f3d255b2a53" alt=""
How do we create actions with less boiler plate?
Hint: npm i @reduxjs/toolkit
data:image/s3,"s3://crabby-images/88d9e/88d9ed84e96a397f367eb6e3c0b7a9c74fe6e27f" alt=""
every action creator returns an object
can use reduxtoolkit to simplify the code
createAction( )
returns an action creator
import { createAction } from “@reduxjs/toolkit”
How do we refactor this code using Redux toolkit?
Hint: refactor Action creators
data:image/s3,"s3://crabby-images/5c584/5c584ce54f04a4c217eb8c12a20359f16e95fb57" alt=""
createAction ( ) returns an action creator
call fn and pass payload (see index.js)
data:image/s3,"s3://crabby-images/18684/186848744dde997e618d382fff6f4161c10b5216" alt=""
What does the createAction ( ) fn return?
Hint: createActionCreator( )
data:image/s3,"s3://crabby-images/fc64e/fc64e7dc32e58fbb1a1462441009780e52e1d1f4" alt=""
utility function from react-dev-tools
createAction ( ) returns an action creator function
should be called createActionCreator ( ) !
returns an action creator function
pass args to create an action object
data:image/s3,"s3://crabby-images/cba37/cba37b0024d4ada0ed74716f456a43519f1e54cc" alt=""
How can we refactor this reducer using Redux toolkit?
data:image/s3,"s3://crabby-images/1ea08/1ea08e75de36cde10765a170aade2381369e9f1a" alt=""
don’t have to write immutable updates to state
can replace switch-case statements
under the hood, uses immer
code we write in reducer is automatically converted to immutable code!
data:image/s3,"s3://crabby-images/6c85e/6c85ee7d56e11df576e833a51ea1929984544075" alt=""
What is the problem with this code?
data:image/s3,"s3://crabby-images/f982e/f982e89248c26f501b08da511113a89988337c31" alt=""
The string we are passing to the actionCreator is hard coded
This dependency breaks the reducer when calling createReducer and passing an action with a hardcoded name that was changed
soln?
instead of hardcoding the name by passing a string arugment
can use [] to dynamically compute the action name (property)
action object contains a method that returns type property of the action
What is the create reducer method take as arguments?
Hint: redux toolkit createReducer ( )
createReducer ( ) two method arguments
Arg #1
initial state, alias that matches slice of state to be changed
Arg #2
{ pass an obj } - that maps
key = action names value = fns
event => eventHandler
What is Creating Slices?
Redux toolkit
createSlice( ) function
creates an action and a reducer together
cleaner, simplier code
createSlice ( ) method
takes a configuration object
with properties :
name:
initialState:
reducers: {
action: actionHandler}
How do we refactor this code?
data:image/s3,"s3://crabby-images/70e07/70e07150b95b00e5bbb7a8b49679ffec178cc876" alt=""
Hint: createSlice( ) redux toolkit
use CreateSlice ( ) method
internally the reducer property function calls
two functions - createAction ( ) , createReducer ( )
data:image/s3,"s3://crabby-images/e236b/e236bda8278d489607e82127567a41581e621a50" alt=""
What is the createSlice ( ) method?
data:image/s3,"s3://crabby-images/e9ea6/e9ea65ea114dad4ad5b017dc39c0fced39f837c3" alt=""
Redux tool kit
takes a configuration object
pass action as key (event)
pass reducer as value (eventHandler)
data:image/s3,"s3://crabby-images/01ea8/01ea829114ab3e7cb7238d70470330183ab98199" alt=""
How do we use the createSlice ( ) method to write clean, beautiful redux code?
data:image/s3,"s3://crabby-images/50bb5/50bb5359863f489d7787569ca9c0d0c445884370" alt=""
don’t have to explicitly create actions and actionhandlers
createSlice ( ) creates both for us
data:image/s3,"s3://crabby-images/6fad6/6fad6488112aa3eb64351ae1d267ae7e43dd497e" alt=""
What is a slice?
data:image/s3,"s3://crabby-images/0c1d8/0c1d8ae373431cd33048963a5f43b4e251aa4f01" alt=""
Redux Toolkit
creates action using createAction ( )
creates reducer using createReducer ( )
allows us to write mutable looking code
converts to immer (immutable code)
data:image/s3,"s3://crabby-images/f89e2/f89e2ef5bbd8da752bbec000770ff383fa336ff9" alt=""
Okay to have local state in our UI components or do we have to put all our state into redux store?
Two approaches to organizing data in redux apps:
- Global Data Only - in Redux store
if multiple components use data, store in Redux store
Ex. eCommerce - put user and shopping cart (global data) in redux store
everything else in outside of store
Mosh is not a fan because it doesn’t allow us to get all redux benefits
Using redux in small part of application
no time travel debugging
no testable code
no caching and persistence
Don’t use Redux in a small app with simple data flows
Therefore…
if all we do is use Redux to share global data, it’s a waste, better to use Context Objects to share global data in React
- Put everything in store - single, unified way to access data, comes at a cost
Benefits:
more consistent
more maintanable code
store as a cache
easier debugging using redux dev tools
code becomes easier to test
Mosh favors this approach
data:image/s3,"s3://crabby-images/f3b18/f3b18b1afcfcb43695ebdb564a70cc56dff193de" alt=""
What are the benefits of putting all the UI data in a redux store?
Hint: Exception - form state data
Don’t store form state data in Redux
Put everything in store
single, unified way to access data, comes at a cost
Benefits:
more consistent
more maintanable code
store as a cache
easier debugging using redux dev tools
code becomes easier to test
data:image/s3,"s3://crabby-images/2eed4/2eed4469706ca1ea108ad2d45fb31919c7ce46a2" alt=""
Why is it not a good approach to only store global data in Redux and other UI data locally?
Global Data Only - in Redux store
if multiple components use data, store in Redux store
Ex. eCommerce - put user and shopping cart (global data) in redux store
everything else in outside of store
Mosh is not a fan because it doesn’t allow us to get all redux benefits
Using redux in small part of application
no time travel debugging
no testable code
no caching and persistence
Don’t use Redux for small data apps
Why should we not store form state in Redux store?
Hint: Store form state locally in components
data:image/s3,"s3://crabby-images/41a3b/41a3bf4164a1c0c4ec966b43e966576fd419f96a" alt=""
Reasons:
- Temporary values in a form
have no impact on other UI until user submits form
okay to populate a form with data from store
don’t want to update the store as the user is typing
no value in keeping temporary data in the store
update store only when user submits
- Too many dispatches
we shouldn’t dispatch an action on every key stroke
causes performance issues
makes debugging more challenging
Ex. 50 actions as user was typing full name
Soln?
Store form state locally in components
data:image/s3,"s3://crabby-images/9311d/9311d84bcaf0af2bc35f4c1d76eed5d67c8eea15" alt=""
What is the best way to structure our store?
Hint: use an object
Use an object
each key represents an ID
each value represents an object
Benefits:
Can quickly lookup object by ID (accessing property)
this is a very fast operation
very clean and fast lookup
doesn’t preserve order
vs.
using an array, first find index (slow operation, iteration)
What are the Pros and Cons of structuring the Redux store using an object?
Pros
allow fast lookup (hashmap, key-value pairs)
if you need fast lookups, use an object
Cons
doesn’t preserve order
cannot re-order properties in an object
if you need ordered data, use an array
Or use a combination of two!
Ex. using an object to store bugs, array to store bugID’s
data:image/s3,"s3://crabby-images/609d6/609d67c8b14198d616432bd941ccd4e9b3c5940d" alt=""
How should we structure our store?
data:image/s3,"s3://crabby-images/68fb7/68fb7b833422099012e6ff7870b399978c92e341" alt=""
each slice represented as an array
wrap slices in a parent slice called (nested)
another high level slice for user (name, id , authtoken)
another high level slice called UI (state specific for components)
data:image/s3,"s3://crabby-images/b54d6/b54d637d4e5052401f814a8fe5cabe851e3cf84e" alt=""
How can we create a store with multiple slices (multiple reducers)?
data:image/s3,"s3://crabby-images/85fdf/85fdf10cd8929621dbe2c8b87603cab74beea4ca" alt=""
combineReducers ( ) from redux
combines our reducers into a single reducer we can pass to our store
data:image/s3,"s3://crabby-images/3f6bb/3f6bbc26e6bf4e68371200730d69a862e1ca98cf" alt=""
What are we doing essentially by nesting slices under entities?
data:image/s3,"s3://crabby-images/71d26/71d267aca5ecf43d12fc55cc0890b412da0759cc" alt=""
combining reducers
creating a hierarchy of reducer functions
root reducer passes action to entities
entities reducer passes actions to children reducers
Main Idea?
multiple reducers can handle same action
each reducer is responsible for updating only one property in the store
this keeps each reducer in a single responsibility
What does normalization mean?
data:image/s3,"s3://crabby-images/4b83f/4b83f67d3803a82f8183326b33ee4de85b3eecba" alt=""
We should not duplicate data in our store
remove nested data by replacing with identifiers
connect related data using identifiers
aim to keep data as flat as possible
get rid of duplicates, using identifiers
No nesting
can look up object by objectID if neccesary
Ex. project object in a few places of store - update name, have to update every instance of data or will have inconsistent data
data:image/s3,"s3://crabby-images/ff5cd/ff5cd903be83eeaffb55f2ce9cdcae89e0f103eb" alt=""
What is the library normalizr?
Hint: npm i normalizr
If your app talks to an API
received de-normalized, hierarchical data from server
use normlizr to normalize your data
data:image/s3,"s3://crabby-images/52400/52400e6ce1ce28097a3e5823509116a3462fec9e" alt=""
How do we compute derived data from data in our store?
Write a query using method chaining after
.getState( )
A derived data element is a data element derived from other data elements using a mathematical, logical, or other type of transformation, e.g. arithmetic formula, composition, aggregation.
data:image/s3,"s3://crabby-images/33a73/33a731479f456b236da8b412c379d929d741d0a6" alt=""
What is the problem with this approach?
data:image/s3,"s3://crabby-images/6136d/6136d26e266387709b4cb1a09c52b4f893ba2f9e" alt=""
In real application
logic for computing derived data can be complex
logic may be in a few different places
without a deriving class (selector)
these objects will break with changes
Soln?
Selector fn
takes state, returns a computed state
What are selectors?
Hint: computed state functions (query state logic)
data:image/s3,"s3://crabby-images/bf1de/bf1de916cc3fd49780b771d4c47154f6a50fc315" alt=""
functions that return computed data
take state as an argument
return a computed state
Naming conventions:
getUnresolvedBugs( ) *
mosh’s favorite, it’s shorter
unresolvedBugsSelector ( )
selectUnresolvedBugs ( )
Benefits:
single place to modify function
can re-use in multiple places in application
data:image/s3,"s3://crabby-images/7e837/7e8374ea54ffab1daddc7349960af6dadeceda83" alt=""
What is the problem with this code?
data:image/s3,"s3://crabby-images/1768c/1768c69d4f863084a9250fc13836469b0624a140" alt=""
Hint: .filter ( ) method returns a new array
$ npm i reselect
$ npm i reselect
createSelector ( ) method returns a memoized selector (cacheable)
selector won’t execute logic if it’s in cache
will return value from cache
.filter ( ) method returns a new array
we don’t want state to re-render everytime we query the store
Soln?
Memoization
memoize this function
a technique for optimizing expensive functions
can build a cache of input / output
before executing expensive logic, look at cache
if operation has previously been run, get data from cache
What is createSelector ( ) ?
from npm i reselect
a function that creates a memoized selector
Two arguments:
selector - get’s state and passes it to results function (query logic)
results function - selector passes state, logic is executed if value not in cache
if value is in cache, results function will get data from cache
How do we assign a bug to a user?
data:image/s3,"s3://crabby-images/dfda5/dfda586a095f6c2fac2cbb8cc857580b1bb38dc6" alt=""
data:image/s3,"s3://crabby-images/cf818/cf818ad3c6b82420ae32198c61551bdac021d120" alt=""
What is middleware in redux?
data:image/s3,"s3://crabby-images/106c6/106c6c7246fdad9361ca39c1c1bc74c926d1e497" alt=""
a curried function with three parameters
store => next => action
an advanced topic in Redux
building block that allows us to do side effects (like calling APIs)
How?
add functions that are executed everytime an action is dispatched
called middleware
peice of code that is executed after it’s dispatched, before reaching root reducer
Why?
Calling APIs
Error Reporting
Analytics
Authorization
data:image/s3,"s3://crabby-images/ad754/ad754508b832f55ca595cbfb4ddd131ee27ee985" alt=""
How can we parameterize a middleware function?
data:image/s3,"s3://crabby-images/40a8b/40a8b680f62d3bd71824fd38e1417544fa203e5f" alt=""
data:image/s3,"s3://crabby-images/f6946/f6946998c2d2533981755d01c28eda14787afc63" alt=""
How can we change our .dispatch ( ) method in our store to accept functions?
Hint: allows us to make Async API calls
data:image/s3,"s3://crabby-images/a469f/a469f745dd837c4643da63c242c64335e535c677" alt=""
By default
store.dispatch ( ) takes an object with type property ONLY
will give errors if we try passing a function or obj without type property
Soln
create a middleware function called func.js
handles calling the function if store.dispatch ( ) is passed a fn
allows us to make async API calls
How?
logic - if typeOf action is function, call function
otherwise, pass to next middleware or reducer (if no more middleware)
Why?
passing a function is used call asynchronous APIs that return promises
Redux Toolkit
creates this middleware func (thunk) for us!
Use redux toolkit
data:image/s3,"s3://crabby-images/a2bb2/a2bb2546fdb7d40960f39b1a381adc9d6aeaf994" alt=""
What is the problem with this code?
data:image/s3,"s3://crabby-images/f1d92/f1d9282b13875620cd4a23c3e46d3741914cd4fb" alt=""
In index.js have access to store.dispatch ( )
in other parts of our application we can not dispatch functions easily
if we try to import the store in other parts of application, a new store is created
How can we dispatch functions using the store in other parts of our application?
Soln?
pass a reference to the dispatch function when dispatching a function
data:image/s3,"s3://crabby-images/aa99c/aa99c6147c0db18e5bc87db7ed60ae2403fdcad2" alt=""
Why do we need to call APIs in our redux applications?
Hint: Getting or Saving Data with Backend
Almost every front end application needs
to talk to a backend for getting or saving data
Very important topic
Really important principles
Change how you think about software design
Where should we put code with side effects (API calls, global data, etc.)?
Hint: Action creators
data:image/s3,"s3://crabby-images/fe073/fe0737b6381ee82f4cb7f3f28f028ebc61b9f383" alt=""
In action creators
Why?
Reducers must be pure functions
this makes them easy to test
no API calls
no DOM manipulation
How?
With Thunk middleware (dispatch functions)
can return a function from an action creator
re-write actionCreator ( ) fn using => syntax
with dispatch and getState can do more processing
data:image/s3,"s3://crabby-images/36be3/36be37610dda36a463c5959a4aa3f6fb4652233c" alt=""
How can we refactor this action creator?
Hint: use arrow fn syntax
data:image/s3,"s3://crabby-images/5d73c/5d73caf282b0e48f3f049b2ae3a7b351c0d1c68e" alt=""
action creator is a fn with no parameters
returns a function with (dispatch, getState) as parameters
returns some logic
this is currying in action
data:image/s3,"s3://crabby-images/7a74e/7a74e5a09a1953c52b1f3fcb1911532e31b7090a" alt=""
How do we call an API using our action creator?
data:image/s3,"s3://crabby-images/b83aa/b83aafc68d46a0179da268d75c43951c28080df7" alt=""
Three things (basic pattern):
- Call API - axios, fetchApi, any library (async)
returns a promise
- Dispatch success action - promise resolves
reducer catches and updates state
Resolved: dispatch (success)
- Dispatch error action - promise rejects
Rejected: dispatch (error)
data:image/s3,"s3://crabby-images/c3c5d/c3c5d24f1533d085e5c86660ea25e5cc53dca03f" alt=""
What are the naming conventions for actions that return functions for calling APIs?
Hint: Past tense vs Present tense
Present
GET_BUGS_REQUEST
GET_BUGS_SUCCESS
GET_BUGS_FAIL
Past tense
bugsRequested
bugsRecieved
bugsRequestFailed
data:image/s3,"s3://crabby-images/37afe/37afedd9aeadb7e8a3237467c9a1eaff27ea2253" alt=""
What’s the problem with this pattern for calling APIs using actionCreators?
data:image/s3,"s3://crabby-images/c8da8/c8da8777c9c16a5910aabb2196b08c551810f237" alt=""
Very repetative
as we implement more features in App
each time we call an API
have to follow three steps
write code for calling, dispatching, error handling
chaining then and catch methods
or try catch blocks
Soln?
Middleware
can implement this pattern in middleware
data:image/s3,"s3://crabby-images/8b546/8b546a12a57f20796db99964a51ca49d37ba941d" alt=""
What kind of action object should our api middleware be able to handle?
data:image/s3,"s3://crabby-images/a769a/a769a769b5f431558847518bf104e42289b152f0" alt=""
serializable
passing strings vs functions
data:image/s3,"s3://crabby-images/ac4f7/ac4f79dc7d61d51426f2b26d12004af437f88b5f" alt=""
What are the problems with this implementation?
data:image/s3,"s3://crabby-images/44ae4/44ae4e480d6ad4a775db436e5f4fc654d1a2ed00" alt=""
hard coded action type’s
one in middleware
one in action
soln?
create an actionCreator( )
Inconsistent naming
better to have names match
Soln?
refactor
action creators!
access .type property in middleware fns
data:image/s3,"s3://crabby-images/3dc83/3dc83ad82b945c41f36546351bce4b6ca862abcd" alt=""
How can we restructure our store?
Hint: [] vs { }
data:image/s3,"s3://crabby-images/57fda/57fdaab83b99aa4a72a30edfa629109187d4a233" alt=""
restructure from array to object data type
add properties :
loading
display loading to user
lastFetch
used to determine cache integrity (up to datedness)
data:image/s3,"s3://crabby-images/a3228/a32285f93ca578d707f2e80585b6e4b2527084e1" alt=""
Why should we not have these details in our UI layer?
data:image/s3,"s3://crabby-images/ea66f/ea66ff268d37e38f30ea41a519bf8aada3d1d3b0" alt=""
have to repeat this action creation in multiple UI places
Ex. drop down, bugs page, etc.
breaks with changes
have to update multiple lines of code
Soln?
store.dispatch(loadBugs() )
details of endpoint, action, etc. abstracted
How do we design a feature to handling loading while app is getting data from server?
When call server
set loading to true
in UI show/hide spinner
Steps:
1. Define new action
bugsRequested in bugs slice automatically creates an action
2. Define new reducer
sets loading prop to true
fn in slice
3. Change middleware
generat action based on property (onStart) by loadBugs ( ) to middleware
dispatch new action before making an API call
data:image/s3,"s3://crabby-images/b11ec/b11ec31d3738fdd79f1cb2adff02ca42fcf8951f" alt=""
How do we handle the situation in which something goes wrong?
data:image/s3,"s3://crabby-images/1ddf6/1ddf67042af7da3d4b9009ca89701957c0959dd0" alt=""
create a new action and reducer
that set’s the loading property to false if an error occurs
data:image/s3,"s3://crabby-images/9c537/9c537382730625304858af662cd58535cbd75d98" alt=""
How do we solve this problem?
Hint: npm i moment
data:image/s3,"s3://crabby-images/625a9/625a9a018e84d280a76337592310e61c8bc11ab6" alt=""
Don’t want server to call API second time in 2 seconds
data:image/s3,"s3://crabby-images/6492b/6492bbe046d53f6981a49da1a9aa48da67302b9f" alt=""
Soln?
Caching
re-write loadBugs action creator to return a fn that includes dispatching new action
stamps currentTime
moment ( ) library returns current date time
can compare to current time if
in real app, extract logic into re-usable component
How do we save data to the server?
Hint: Http Post request
data:image/s3,"s3://crabby-images/7513d/7513d385a47a779383a704235bc358af780a19c0" alt=""
Soln?
dispatch API call, let API middleware handle everything
loadBugs ( ) is dispatching an API action (bugs.js)
addBug ( ) dispatches an API action (bugs.js)
consume using .dispatch ( API action)
Why?
Same steps as making an API call:
Make an API call
dispatch (success)
dispatch (error)
just let API middleware handle this logic
data:image/s3,"s3://crabby-images/22dbb/22dbbff9a6d305c49534eb20b2c85ef0d71302f0" alt=""
How do we resolve a bug?
create an actionCreator to create an API action
let API middlware handling calling server, handling errors
updates server
updates store
data:image/s3,"s3://crabby-images/d2796/d27965158ba9ec42149134a0c45445fa53f282e0" alt=""
What’s the problem with this code?
data:image/s3,"s3://crabby-images/e744d/e744d849b53030b0ee595ef0d87cf1163012e6b6" alt=""
exporting bugAdded, bugResolved, etc.
only place they’re used is in bugs.js module
Soln?
Don’t expose to outside
other modules cannot use internals of bugs module
data:image/s3,"s3://crabby-images/23619/236194419ceec308fe67a573d3ade37947ebc80d" alt=""
What is Mosh going to show us around testing?
Right way to do testing in Redux applications
there is a lot of bad information out there
Short / practical introduction to testing (included)
lot’s of misconceptions around automated testing
What is automated testing?
data:image/s3,"s3://crabby-images/6dc53/6dc5343d77fa7e9c544dae28f491361cb9e601c1" alt=""
Hint: npm i
jest
@types/jest
@bale/core
@bale/preset-env
babel-just
Writing code to test our code
can run hundreds or thousands of tests
far faster and easier than manually testing functions
always an element of manual testing
there is a balance between automated testing and manual testing
Unit Test
tested without external dependencies
fast to run, uses mock functions
focus of this course** (unit testing)
run during production
Integration
test app with external resources
slower but give greater confidence
run at certain phases like committing code, getting production ready
End-to End
launch application, drive through UI
slowest, most fragile tests
data:image/s3,"s3://crabby-images/90032/90032daf68399c1e88dffafc8d8d6a70a2176f68" alt=""
How do we write tests?
Hint: Write social tests not solidarity tests
data:image/s3,"s3://crabby-images/6cd2a/6cd2ad1489770aa15ad12a239f63b043f86e5dfe" alt=""
Define a group / suit of tests
describe ( ) function
first arg - name (group)
second arg - fn called by jest (includes test fns)
Name should be meaningful
“isEven should return true if given an even number”
function to define test
it ( ) given by jest
first arg - name of test
second arg - test fn (contains code of test)
Call function under test
call function you’re testing
create an expectation (make an assertion)
expect ( )
fn provided by jest
Use a matcher in assertion
.toEqual ( ) , NaN, contains, etc
data:image/s3,"s3://crabby-images/b2cd7/b2cd7966c5f5acfeae37dace4a8aac4f95b3ac69" alt=""
How can we unit test a redux application?
data:image/s3,"s3://crabby-images/483f0/483f0702c4fac100814485b00bb3648dfb4c5bc7" alt=""
Hint: Solidarity tests are poor way
as long as no external resources, it’s a unit test
Solidarity (lonely) tests
test individual building blocks in isolation
write separate tests for
actions creators, reducers, middleware
POOR WAY
Sadely, most books / courses teach this way
Why?
Coupled to implementation
Know too much about how app is structured
structured to implementation
when our implementation changes, tests break
slow us down, re-writing tests!
Soln?
Social tests
implementation details not important, study behavior
data:image/s3,"s3://crabby-images/60b1e/60b1e950e38eb3324bf3fbe035dd3687671f9420" alt=""
What should our unit tests test?
Hint: test behavior NOT implementation
data:image/s3,"s3://crabby-images/3a8b8/3a8b8d84463446b2191e2d316b46c0a34c634510" alt=""
Behavior of application
NOT implementation
Ex. Test timer or microwave
don’t care how current goes around, etc in microwave
only care about pressing start button
test behavior not implementation
solidarity tests are testing action creators, reducers, etc.
They know too much about implementation details
They’re too connected to implementation
can test a lot of building blocks, but doesn’t show if app work
Redux is about changing state of store
building blocks ensure app state changes accordingly
test all building blocks together!
Soln
Social Tests
data:image/s3,"s3://crabby-images/418e2/418e2648e1bd6a4ff933ce61b37365ed5522b9ef" alt=""
What are social tests?
data:image/s3,"s3://crabby-images/c1c6c/c1c6c2253abe7972960b454baa6cd8956cebdc38" alt=""
Proper way to test redux applications
dispatch an action, check state
what happens under hood, irrelevant to tests
Tests that involve multiple functions / objects working together
Benefits:
less fragile
don’t break
more reliable
test behavior of application
don’t care if an action, calls another action
don’t care if middleware catches an action, dispatches another
don’t care about reducers and how we combine them
data:image/s3,"s3://crabby-images/6cfdd/6cfdd492754f7849b8a28a18b7045e33318e5b55" alt=""
Are solidarity tests evil?
data:image/s3,"s3://crabby-images/18e67/18e673365a9e8fc6fcb6969aba037b0c7e79b06f" alt=""
Absolutely not
they’re great and have their own uses
if building block is complex
test it in isolation
Redux building blocks are simple
they don’t have algorithms or complex logic
no logic
action creators return an action
reducers return state
How do we write a solidarity test for adding a bug?
data:image/s3,"s3://crabby-images/1faab/1faabc80245a3970ca7a848b77e83e0570ef5039" alt=""
action creator addBug
returns an action object that represents API call
have to call it, look at object
API middleware takes object, on success
dispatches another action
reducer takes action, updates store
data:image/s3,"s3://crabby-images/99752/9975298ff8b2fe0b5e747dc6adec33dded8b493a" alt=""
What is the problem with writing solidarity tests?
data:image/s3,"s3://crabby-images/4a787/4a787a4211ae649b164f8d0d293906a1391dad45" alt=""
If we change the implementation details, test fails!
Solidarity tests:
break easily
don’t give confidence
data:image/s3,"s3://crabby-images/f2558/f255821e47d0cbe8e0c1f7f354aed92ce99209f5" alt=""
What’s the problem with solidarity tests?
Hint: Doesn’t test behavior of application
data:image/s3,"s3://crabby-images/1063d/1063d807c138e3c9f344a21ed7d77aa2a2aa5b82" alt=""
They don’t test application behavior
Ex. api middleware commented out, test still passing
data:image/s3,"s3://crabby-images/85951/85951081186236cb71bec90871938fa2c025f403" alt=""
How can we write a social test?
data:image/s3,"s3://crabby-images/da392/da3929893001af6c4368484d11f33813c313642e" alt=""
A test that excercises multple building blocks (units) together
doesn’t break with updated implementation
behavior is not changed
beauty of social tests
Ex. Change addBug ( ) implementation to omit API middleware
data:image/s3,"s3://crabby-images/1dd4c/1dd4c22f910490c40538346a161ab5ff7e9cab64" alt=""
How do we convert this test from an integration test to a unit test?
Hint: npm i axios-mock-adapter -D
data:image/s3,"s3://crabby-images/48229/48229810f6c55c5eaf68f9e8fd7f7efc265d5de6" alt=""
npm i axios-mock-adapter -D
creates mock axios HTTP objects
mock HTTP calls
Why?
calling external resources slows down our test
external resources might not be available when running tests
data:image/s3,"s3://crabby-images/52994/52994ffeea29ff6e1060407fc8c50c926997952c" alt=""
What are techniques for writing clean and maintainable tests?
data:image/s3,"s3://crabby-images/34d2d/34d2de125c3dc99763a8b1f3a9e425db923d7360" alt=""
tests are equally as important as source code
Many people don’t treat them that way!
spending time with messy test code
Soln?
write, clean maintainable testing code
common pattern (arrange, act, assert)
arrange
initialization code
act
triggering an action
assert
expectation code
What else?
beforeEach ( )
executes a function before each test
can initialize things before each test
data:image/s3,"s3://crabby-images/fff7f/fff7fb4533a25edd762e042a4618b2fc405c03b9" alt=""
How do we write clean, maintainable tests?
Hint: Arrange, Act, Assert
Arrange
setup the testing needs
Act
call function
Assert
make a statement about expected result
data:image/s3,"s3://crabby-images/f0e7d/f0e7d9b9780576d19ae6ef87ce07b3c0c5f8b9e0" alt=""
How do we see how much of our code is covered by tests?
Hint: ./node_modules/.bin/jest –coverage
data:image/s3,"s3://crabby-images/a1623/a1623bc9b13bf919046f7baa2dd21f3896189d44" alt=""
Jest –coverage
as we write more tests, have better coverage
coverage report
can immediately see which parts of code are covered or not (by tests)
data:image/s3,"s3://crabby-images/dccb4/dccb4bdddfeef48f1e1aef13169a71a2ecad1eac" alt=""
How can we make this test code cleaner?
data:image/s3,"s3://crabby-images/8179b/8179b47101659a4c4e1adc3563db8203145715db" alt=""
Object structure (store) is cluttering testing code
will need this store object in other tests
repeating code is un-necessary
Soln?
create helper function for creating object structure (store object)
createState( )
makes test clean and readable
data:image/s3,"s3://crabby-images/15932/1593201ba8985f5c3fe152a52f61b962bfb6eaae" alt=""
How do we write a test for testing the behavior of the resolveBug ( ) feature?
data:image/s3,"s3://crabby-images/ab06b/ab06b27be9d0ff767856cd1528c481727442a5be" alt=""
Totally okay to dispatch multiple actions
interacting with the system, looking at behavior
we are testing the behavior not the individual building blocks
Two Tests:
Happy Path
Negative Path
data:image/s3,"s3://crabby-images/180ad/180adf86d40fcdf590c92e37053958741045f16f" alt=""
How can we test our loadBugs ( ) feature?
Hint: challenging and important excercise
data:image/s3,"s3://crabby-images/84112/841129b5395d01f4f5f4c286cefbfa40693eabef" alt=""
most challenging and important excercises
Pay lots of attention
Lots of “aha” moments
Key Moment:
fakeAxios has two method signatures
call second signature to execute code before server responds
testing if using cache vs. server call
call method twice (dispatch loadBugs( ) x 2)
check to see if have single HTTP request to backend
fakeAxios.history returns an array of get requests (see docs)
can see if length of array equals 1
data:image/s3,"s3://crabby-images/8a0e9/8a0e905e780505e358c19e695aaa84ba5ed4292b" alt=""
What are we going to learn about Redux and React integration?
How to subscribe to redux store to get data
What is the proper way to connect a React app to Redux?
data:image/s3,"s3://crabby-images/1895b/1895bc3651ef879f15ae804a3bead3af865c1543" alt=""
using React Redux library
Mosh is going to show us the manual way
How do we connect our React app to Redux by scratch?
data:image/s3,"s3://crabby-images/c5efe/c5efe5a7f69e702f1e3c22322a008e6999d3f908" alt=""
import configureStore module
pass store down using context (not prop drilling)
can provide at top of tree, all others have access
part of React course
How do we get the bugs from the backend and display them on the page?
Two Steps:
- subscribe to store
get list of bugs from store, render on screen
takes a function
executed everytime an action is dispatched
get bugs in store, store them locally in component
- dispatch(loadBugs())
to get bugs from backend and put them in the store
data:image/s3,"s3://crabby-images/9af88/9af882b5de1ff94edf4d1dbb6db468bc4c040ffc" alt=""
What is the problem with this code?
data:image/s3,"s3://crabby-images/4eed6/4eed61136b7578f2313aae6db3b458cdd9e528f1" alt=""
It’s very tedious
have to write code in every component
componentDidMount ( ) subscribe to store
componentWillUnmount ( ) unsubscribe from store
if forget anything, will cause memory leaks
very tedious and error prone
Soln?
React Redux
takes care of all this complexity
How do I use React-Redux library to connect my app to redux?
storeContext no longer needed
no longer do I have to maintain local state
React Redux will handle subscribing and unsubscribing from the store
don’t need componentWillUnmount
this.props
data:image/s3,"s3://crabby-images/88917/88917a49b00fe81e0dc7338d97b3081da4890037" alt=""
What is the connect ( ) function in React-Redux?
called a container component
why?
because it returns a function that can pass bugs component
so that bugs doesn’t know anything about store (redux)
Takes two arguments:
1. A function that takes state, returns part of the store our component is interested
state.entities.bugs.list
takes state and returns part interested in
pass as props to component
const mapStateToProps = state => ({
bugs: state.entites.bugs.list })
- mapDispatchToProps - dispatching actions
takes dispatch function from store
maps to props for component
data:image/s3,"s3://crabby-images/cb94a/cb94a89ecd4fb8f04f90935ad4a557ea97d4e08a" alt=""
What are hooks?
kind of a new thing (feature) in react
covered in detail in React course (final section)
data:image/s3,"s3://crabby-images/70199/70199d43745054172a7581a311be4b6a84432b1c" alt=""
Should I connect a top-level component or many small components to the Redux store?
data:image/s3,"s3://crabby-images/1f04e/1f04ed4544818aa206cd8c80cd1fb4aebfd482a1" alt=""
If you connect a top-level component to the store, that component and all its children will get re-rendered every time the store gets updated.
Let’s say App component has two children: Bugs and Projects.
If you connect App component to the store, it gets notified every time the list of bugs or projects is modified. So if you add a bug, App component gets notified and the list of bugs as well as the list of projects will be re-rendered. This is unnecessary and can cause a performance penalty.
So, as a best practice, each component should independently subscribe to a small slice of the store it is interested in. This way it won’t be re-rendered if other slices of the store are updated.
data:image/s3,"s3://crabby-images/5d03b/5d03b0dbed9585d9190ad79cd67431ece9f0fe92" alt=""