connectResources is a function that will return HOC to pass all necessary props to your React component for async HTTP request.
connectResources
accept 1 argument ResourceConfig or an Array of ResourceConfig
Copy import connectResources from '@cranium/resource'
connectResources ( ResourceConfig | [ ] ResourceConfig )
# Basic usageCopy import { connectResources } from '@cranium/resource'
function MyReactComponent ( { users } ) {
const {
data ,
isLoading ,
options ,
errors ,
filters ,
fetch ,
fetchOptions ,
create ,
replace ,
update ,
remove ,
setData ,
setLoading ,
setErrors ,
setFilters ,
clear ,
} = users
return ...
}
export default connectResources ( 'users' ) ( MyReactComponent )
You will have same property name as argument in connectResources
.
Like <MyReactComponent>
has property users
because of connectResources('users')
# Redux1 of biggest redux problem is that it is required to create reducers to handle changes for each part of store data.
That's why Redux applications on start have store with lot of empty objects, like:
Copy {
users : {
data : null ,
errors : null ,
isLoading : false
} ,
user : {
data : null ,
errors : null ,
isLoading : false
} ,
cars : {
data : null ,
errors : null ,
isLoading : false
} ,
pets : {
data : null ,
errors : null ,
isLoading : false
}
}
resources
module has empty store on start. And data in store will appear while sending HTTP requests. Also by default resources
module clears redux store on component unmount, so that you will see only that data in redux store that is currently using to render current components.
Initial app store Render component wrapped with connectResources
Send first HTTP request redux store
Copy {
users : {
data : { count : 12 , results : [ ] } ,
errors : null ,
isLoading : false
}
}
Unmount component In this way Redux store will be readable and understandable. There will not be any unused data, so ot will be more optimized
# Sending HTTP requests# GET requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , fetch } = users
useEffect ( ( ) => {
fetch ( )
} , [ ] )
if ( isLoading ) {
return 'Loading...'
}
if ( errors ) {
return 'Oooops... Server is temporary unavailable'
}
return data . map ( user => < User { ... data } key = { user . uuid } / > )
}
export default connectResources ( 'users' ) ( UserList )
# GET request with dynamic URLCopy import { connectResources } from '@cranium/resource'
function UserList ( { user } ) {
const { data , isLoading , errors , fetch } = user
useEffect ( ( ) => {
fetch ( { uuid : 'taras' } )
} , [ ] )
if ( isLoading ) {
return 'Loading...'
}
if ( errors ) {
return 'Oooops... Server is temporary unavailable'
}
return < h1 > Welcome { data . fullName } < h1 >
}
export default connectResources ( {
namespace : 'user' ,
endpoint : 'users/:uuid'
} ) ( UserList )
# GET request with query paramsCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , fetch } = users
useEffect ( ( ) => {
fetch ( { offset : 0 , limit : 10 } )
} , [ ] )
if ( isLoading ) {
return 'Loading...'
}
if ( errors ) {
return 'Oooops... Server is temporary unavailable'
}
return data . map ( user => < User { ... data } key = { user . uuid } / > )
}
export default connectResources ( {
namespace : 'users' ,
endpoint : 'users' ,
queries : [ 'offset' , 'limit' ]
} ) ( UserList )
Please note that it is important to describe all possible queries
.
This is made to build query string only with values that current API support. Like swagger schema.
This optimization will add more safety to your code
Copy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , fetch } = users
useEffect ( ( ) => {
fetch ( { offset : 0 , limit : 10 , fakeData : 'fake' } )
} , [ ] )
}
export default connectResources ( {
namespace : 'users' ,
endpoint : 'users' ,
queries : [ 'offset' , 'limit' ]
} ) ( UserList )
# OPTIONS requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { options , isLoading , errors , fetchOptions } = users
useEffect ( ( ) => {
fetchOptions ( )
} , [ ] )
if ( isLoading ) {
return 'Loading...'
}
if ( errors ) {
return 'Oooops... Server is temporary unavailable'
}
return < p > { options . count } < / p >
}
export default connectResources ( 'users' ) ( UserList )
# POST requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , create } = users
useEffect ( ( ) => {
create ( { name : 'wonder woman' , world : 'DC' } )
} , [ ] )
if ( isLoading ) {
return 'creating Wonder Woman'
}
if ( errors ) {
return 'Oooops... Can not create super hero'
}
return < p > { data . name } is created < / p >
}
export default connectResources ( 'users' ) ( UserList )
# DELETE requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { isLoading , errors , remove } = users
useEffect ( ( ) => {
remove ( { hero : 'super-man' } )
} , [ ] )
if ( isLoading ) {
return 'deleting Super Man'
}
if ( errors ) {
return 'Oooops... Super Man is stronger than our API, please try again'
}
return < p > Super Man has gone < / p >
}
export default connectResources ( 'users/:hero' ) ( UserList )
# PUT requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { isLoading , errors , replace } = users
useEffect ( ( ) => {
replace ( { hero : 'wonder-woman' , isSingle : true } )
} , [ ] )
if ( isLoading ) {
return 'updating Wonder Woman'
}
if ( errors ) {
return 'Oooops... Yaron Varsano is stronger than our API, please try again'
}
return < p > Wonder Woman is now single < / p >
}
export default connectResources ( 'users/:hero' ) ( UserList )
# PATCH requestCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { isLoading , errors , update } = users
useEffect ( ( ) => {
update ( { hero : 'wonder-woman' , isSingle : true } )
} , [ ] )
if ( isLoading ) {
return 'updating Wonder Woman'
}
if ( errors ) {
return 'Oooops... Yaron Varsano is stronger than our API, please try again'
}
return < p > Wonder Woman is now single < / p >
}
export default connectResources ( 'users/:hero' ) ( UserList )
To send PUT, PATCH, POST request with dynamic url, add URL params to body request
Copy
replace ( { uuid : '123-1233-123' , name : 'test' } )
update ( { uuid : '123-1233-123' , name : 'test' } )
create ( { uuid : '123-1233-123' , name : 'test' } )
# Terminating requestsEvery async action returns Promise with 1 extra method cancel
that calls AbortController abort
function. Read more
Copy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , fetch } = users
useEffect ( ( ) => {
const request = fetch ( )
return request . cancel
} , [ ] )
}
export default connectResources ( 'users' ) ( UserList )
# Handling HTTP requestsCopy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { data , isLoading , errors , fetch } = users
useEffect ( ( ) => {
fetch ( )
. then ( ( data ) => console . log ( data ) )
. catch ( console . warn )
. finally ( ( ) => alert ( 'done' ) )
} , [ ] )
}
export default connectResources ( 'users' ) ( UserList )
# Changing store data# setDataCopy import { connectResources } from '@cranium/resource'
function MyComponent ( { users } ) {
const { data , setData } = users
setData ( { wife : 'Wonder Woman' } )
return < p > Your wife is : { data . wife } < / p >
}
export default connectResources ( 'users' ) ( MyComponent )
# setLoadingCopy import { connectResources } from '@cranium/resource'
function MyComponent ( { users } ) {
const { isLoading , setLoading } = users
setLoading ( true )
return < p > { isLoading ? 'Loading...' : 'Loaded' } < / p >
}
export default connectResources ( 'users' ) ( MyComponent )
# setErrorsCopy import { connectResources } from '@cranium/resource'
function MyComponent ( { users } ) {
const { errors , setErrors } = users
setErrors ( 'Ooops...' )
return < p > { errors } < / p >
}
export default connectResources ( 'users' ) ( MyComponent )
# setFiltersCopy import { connectResources } from '@cranium/resource'
function MyComponent ( { users } ) {
const { filters , setFilters } = users
setFilters ( { offset : 0 , limit : 12 } )
return < p > offset : { filters . offset } , limit { filters . limit } < / p >
}
export default connectResources ( 'users' ) ( MyComponent )
# Several resourcesTo have several resources in 1 Component, you can define an Array of Resources to connectResources
Copy function MyComponent ( { cars , pets } ) {
}
connectResources ( [
{ namespace : 'cars' } ,
'pets'
] ) ( MyComponent )
Pay attention that all resources should have unique namespace, otherwise it will just override each other
Pay that is it much better to split your code. That's why mostly it is not a good choice to use several resources in 1 React Component.
BAD my-component.js
Copy
connectResources ( [
{ namespace : 'cars' } ,
{ namespace : 'pets' }
] )
function MyComponent ( { cars , pets } ) {
return (
< Pets { ... pets } />
< Cars { ... cars } />
)
}
GOOD with-pets.js
Copy export default connectResources ( { namespace : 'pets' } )
with-cars.js
Copy export default connectResources ( { namespace : 'cars' } )
pets.js
Copy import PetsView from './pets-view'
import withPets from './with-pets'
export default withPets ( PetsView )
cars.js
Copy import CarsView from './cars-view'
import withCars from './with-cars'
export default withCars ( CarsView )
my-component.js
Copy
import Pets from './pets'
import Cars from './cars'
export default function MyComponent ( ) {
return (
< >
< Pets />
< Cars />
</ >
)
}
# filtersFilters is json object that contains Query string and Url params from latest HTTP request.
Copy import { connectResources } from '@cranium/resource'
function UserList ( { users } ) {
const { filters , isLoading , errors , fetch } = users
useEffect ( ( ) => {
fetch ( { offset : 0 , limit : 10 , world : 'dc' } )
} , [ ] )
if ( isLoading ) {
return 'Loading...'
}
if ( errors ) {
return 'Oooops... Server is temporary unavailable'
}
return (
< >
< p > world : { filters . world } </ p >
< p > offset : { filters . offset } </ p >
< p > limit : { limit . limit } </ p >
</ >
)
}
export default connectResources ( {
namespace : 'users' ,
endpoint : 'users/:world' ,
queries : [ 'offset' , 'limit' ]
} ) ( UserList )
# Clear resource data (clear
)To delete resource data from redux use clear
action
Copy import { connectResources } from '@cranium/resource'
function MyComponent ( { users } ) {
const { data , setData , clear } = users
setData ( { wife : 'Wonder Woman' } )
clear ( )
}
export default connectResources ( 'users' ) ( MyComponent )
clear
action will fully remove data from Redux store by namespace
key