From f3fa048bf758e0d574567c809cf6cc88ee69b6fb Mon Sep 17 00:00:00 2001 From: Nikita Mazur Date: Thu, 18 Jun 2020 17:48:39 +0300 Subject: [PATCH 1/2] Add access lib to repo --- packages/access/README.md | 287 ++++++++++++++++++++++++++++ packages/access/index.js | 4 + packages/access/package.json | 21 ++ packages/access/src/CheckAccess.jsx | 46 +++++ packages/access/yarn.lock | 73 +++++++ 5 files changed, 431 insertions(+) create mode 100644 packages/access/README.md create mode 100644 packages/access/index.js create mode 100644 packages/access/package.json create mode 100644 packages/access/src/CheckAccess.jsx create mode 100644 packages/access/yarn.lock diff --git a/packages/access/README.md b/packages/access/README.md new file mode 100644 index 0000000..64c411d --- /dev/null +++ b/packages/access/README.md @@ -0,0 +1,287 @@ +## The goal + +working with REST-api and redux, using common practice, we always create almost same actions and reducers to send HTTP request to different endpoints. This will lead to problem that our projects will always have lot of duplicated code. For example + +``` +// action types +const FETCH_USERS_LIST = Symbol('FETCH_USERS_LIST') +const FETCH_USER = Symbol('FETCH_USER') +const SAVE_USER = Symbol('SAVE_USER') +const CREATE_USER = Symbol('CREATE_USER') + +// action creators +function fetchUsersList() { + return { + type: FETCH_USERS_LIST + } +} + +function fetchUser(id) { + return { + type: FETCH_USER, + payload: id, + } +} + +function saveUser(id, data) { + return { + type: SAVE_USER, + payload: data, + meta: {id} + } +} + +function createUser(data) { + return { + type: CREATE_USER, + payload: data, + } +} + +// reducers +function users(state = {}, action) { + switch(action.type) { + case FETCH_USERS_LIST: + return {...state, isLoading: true} + case SAVE_USER: + return {...state, ...action.payload, isLoading: false} + } +} + +// epics +// TODO +... +``` +this example does not contains error handling, caching data, authorization, options, filters ... +And basically we always copy paste this code from file to file and rename function names and constants values + +``` +fetchUsers +fetchBooks +fetchGroups +fetchComputers +fetchOrders +etc +``` + +this package will help u stop to duplicating code and start thinking about more interesting staff in your projects + +1. REST API CRUD: +``` +endpoint: /api/v1/users/:id + +GET /api/v1/users/ - get users list +POST /api/v1/users/ - create new user +GET /api/v1/users/1 - get user details +PATCH /api/v1/users/1 - update user +POST /api/v1/users/1 - recreate user +PUT /api/v1/users/1 - полностью перезаписать юзера (обычно не используем) +DELETE /api/v1/users/1 - delete user +OPTIONS /api/v1/users/ - get metadata +``` + +2. endpoint CRUD +``` +endpoint: /api/v1/some/custom/endpoint +``` +And this basically same as previous, but it has only 1 single resource. + +resources will give u ability to add all data and methods to communicate with REST API almost with 1 line of code. +``` +connectResource([resource]) + +// where: +resource = { + namespace: 'internalResourceName', + endpoint: '/some/endpoint/with/:placeholders', + forceUpdates: true|false (default false), + withNavigation: true|false (default false), + reducer: 'object|paginationList|none|replace|custom function' (default 'object'), + queries: [], +} +``` + +And in props u will have next data: +``` +props.internalResourceName = { + data: { /* ... resource data from API ... */ }, + isLoading: true|false, + options: { /* parsed OPTIONS from API */ }, + errors: null, // or object with errors { }, + + // actions + fetch: func, // GET request, useful when no prefetch + fetchOptions: func, // OPTIONS request + create: func, // POST request + save: func, // PATCH request + update: func, // PATCH request, alias for save + remove: func, // DELETE request + replace: func, // PUT request if you need it + setData: func, // you can update data in store manually, but please be carefull with this action + setLoading: func, // you can updates isLoading in store manually, but please be carefull with this action + setErrors: func, // you can updates errors in store manually, but please be carefull with this action + setFilters: func, // you can updates current filters in store manually, but please be carefull with this action + filters: {}, // current applied filters +} +``` + + +## Options + +all options could be defined with connectResource (on initialization level) and then u can override all options when u will call a function + + +#### `namespace : String` [required] + +property name for resource binding. e.g. for `namespace: 'test'` you will get resource in `props.test`. And all data will be saved in store under the "test" key. + + +#### `endpoint : String` [optional] [default: value of namespace option] + +will be set to `namespace` if ommited. resource endpoint name - where to get data from server. +can contains placeholders (see doc for `path-to-regexp` package for additional information). all you current props will be forwarded to `path-to-regexp`, plus one additional property `id` wich will be get by `idKey` (see below) + +#### `forceUpdates : Boolean` [optional] [default: false] +By default resources will trigger all circle methods + 1. toggle loading, clear errors, safe filters + 2. send HTTP request + 3. toggle loading, set errors(on catch) + + In case u want just send HTTP request and then send another one for better performance it is better to set this param to false and call this methods by your own + +#### `queries : Object` [optional] [default: {}] + +used with list resources. representing initial query for fetch request. +* By default it will skip queries for all API requests except GET + +#### `reducer : String|Function` [optional] [default: 'object'}] +one of possible reducer functions that u can use. +Possible variants: +- **object**: Object.assign (prevState, nextState) +- **paginationList**: reducer for lists dataStructures `{count:int,results:[]}` +- **none**: return prevState +- **replace** return nextState +or u can define your custom function + + + +### Examples + +``` +class App extends Component { + componentDidmount() { + //get users + this.props.users.fetch() + //get books + this.props.books.fetch({offset: 0, limit: 25}) + //get book by id + this.props.books.fetch({id: 'harry_potter'}) + //but u can to more! + this.props.books.fetch({id: 'harry_potter'}, {namespace: 'test', endpoint: 'cars'}) //that will send get request to cars endpoint and save data to test in redux + or + this.props.books.setLoading(true); //set loading + this.props.books.fetch(null, {endpoint: 'users/me', reducer: 'none', forceUpdates: true}) + .then((me)=>this.props.books.fetch({id: me.uuid})) //get book with same uuid as user + + } + render(){...} +} + + + + connectResource([ + { + namespace: 'books', + endpoint: 'books/:id?', + queries: ['offset', 'limit'], + reducer: 'paginationList' + }, + 'users' //u can use even just a string in case your enpoint == namespace and all other configs are default + ])(App) +``` + +#### connect single resource +``` +connectResource({ + namespace: 'books', + endpoint: 'books/:id?' +}) +or +connectResource('users') +``` +#### connect multiple resource +``` +connectResource([{ + namespace: 'books', + endpoint: 'books/:id?' +}, 'users']) +``` + +#### customResource +In case u need more complex logic rather then sending 1 HTTP request, but u still want to have all abilities that brings `resources`. +customResource will return HOC which will add same props as connectResource, and add 1 more function `this.props[namespace].customFetch` and this customFetch will work same as .fetch but instead of sending predefined single HTTP request it will run your own async task. So it might be very useful in your projects to have standard resource abilities + your own async job + +``` +function myCustomFetch(API, payload, meta) { + return new Promise(function(resolve,reject){ + setTimeout(()=>resolve({succes: true}),1000) + }) +} + +const customConnect = customResource(myCustomFetch) +``` + +then u can use this HOC with your components +``` +class Test extends Component { + componentDidMount(){ + this.props.test.customFetch({},{}) + } + ... +} +export default customConnect({namespace: 'test', endpoint: 'test'})(Test) +//or +export default customConnect('test')(Test) +///but not +customConnect([{...}, 'test'])(Test) +``` + +customConnect HOC will not support array of resources because it is single resource HOC + + +# HOCS +This package will also provide a standard list of React Hight Order Components: + - [prefetchResources](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/Prefetch.md) + - [withReduxForm](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithReduxForm.md) + - [withInfinityList](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithInfinityList.md) + - [withFinalForm](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithFinalForm.md) + - navigationToProps + + +# Hook + +## useResource +Hook that will create resource in Redux and return same props as connectResources +Accepts same configuration as connectResources + +``` +import { useResource } from '@ds-frontend/resource' + +function MyReactComponent () { + const { data, isLoading, errors, fetch } = useResource('users/me') + useEffect(()=>{ + const request = fetch() + return request.cancel + }, []) + if(isLoading) { + return 'Loading...' + } + if(errors){ + return 'Oooops something went wrong please contact local police office' + } + return + +} +``` + + \ No newline at end of file diff --git a/packages/access/index.js b/packages/access/index.js new file mode 100644 index 0000000..fca20b9 --- /dev/null +++ b/packages/access/index.js @@ -0,0 +1,4 @@ +import CheckAccessProvider, { CheckAccess } from './src/CheckAccess' + + +export { CheckAccess, CheckAccessProvider } diff --git a/packages/access/package.json b/packages/access/package.json new file mode 100644 index 0000000..509780e --- /dev/null +++ b/packages/access/package.json @@ -0,0 +1,21 @@ +{ + "name": "@ds-frontend/access", + "version": "2.0.0-alpha", + "description": "React component for condition rendering", + "keywords": [ + "react", + "redux", + "access" + ], + "author": "DjangoStars (https://github.com/django-stars/)", + "contributors": [ + "Nikita Mazur " + ], + "dependencies": { + "prop-types": "^15.7.2" + }, + "peerDependencies": { + "react": "^16.13.1", + "react-redux": "^7.2.0" + } +} diff --git a/packages/access/src/CheckAccess.jsx b/packages/access/src/CheckAccess.jsx new file mode 100644 index 0000000..7a532f3 --- /dev/null +++ b/packages/access/src/CheckAccess.jsx @@ -0,0 +1,46 @@ +import { connect } from 'react-redux' +import { createContext } from 'react' +import PropTypes from 'prop-types' + +const F_PUBLIC = 2 ** 0 +export const CheckAccessContext = createContext() + +CheckAccessProvider.propTypes = { + level: PropTypes.number.isRequired, + children: PropTypes.node.isRequired, +} + +function CheckAccessProvider({ children, level }) { + return ( + + {children} + + ) +} + +export default connect( + (state, props) => ({ + level: props.userLevelSelector ? props.userLevelSelector(state) : F_PUBLIC, + }) +)(CheckAccessProvider) + +CheckAccess.propTypes = { + access: PropTypes.number, + fallback: PropTypes.node, + children: PropTypes.node.isRequired, +} + +CheckAccess.defaultProps = { + access: F_PUBLIC, + fallback: null, +} + +export function CheckAccess({ access = F_PUBLIC, children, fallback }) { + return ( + + {level => { + return level & access ? children : fallback + }} + + ) +} diff --git a/packages/access/yarn.lock b/packages/access/yarn.lock new file mode 100644 index 0000000..9f5df32 --- /dev/null +++ b/packages/access/yarn.lock @@ -0,0 +1,73 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@babel/runtime@^7.5.5": + version "7.10.2" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.10.2.tgz#d103f21f2602497d38348a32e008637d506db839" + integrity sha512-6sF3uQw2ivImfVIl62RZ7MXhO2tap69WeWK57vAaimT6AZbE4FbqjdEJIN1UqoD6wI6B+1n9UiagafH1sxjOtg== + dependencies: + regenerator-runtime "^0.13.4" + +hoist-non-react-statics@^3.3.0: + version "3.3.2" + resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz#ece0acaf71d62c2969c2ec59feff42a4b1a85b45" + integrity sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw== + dependencies: + react-is "^16.7.0" + +"js-tokens@^3.0.0 || ^4.0.0": + version "4.0.0" + resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" + integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== + +loose-envify@^1.1.0, loose-envify@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf" + integrity sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q== + dependencies: + js-tokens "^3.0.0 || ^4.0.0" + +object-assign@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" + integrity sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM= + +prop-types@^15.6.2, prop-types@^15.7.2: + version "15.7.2" + resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.7.2.tgz#52c41e75b8c87e72b9d9360e0206b99dcbffa6c5" + integrity sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ== + dependencies: + loose-envify "^1.4.0" + object-assign "^4.1.1" + react-is "^16.8.1" + +react-is@^16.7.0, react-is@^16.8.1, react-is@^16.9.0: + version "16.13.1" + resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4" + integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ== + +react-redux@^7.2.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/react-redux/-/react-redux-7.2.0.tgz#f970f62192b3981642fec46fd0db18a074fe879d" + integrity sha512-EvCAZYGfOLqwV7gh849xy9/pt55rJXPwmYvI4lilPM5rUT/1NxuuN59ipdBksRVSvz0KInbPnp4IfoXJXCqiDA== + dependencies: + "@babel/runtime" "^7.5.5" + hoist-non-react-statics "^3.3.0" + loose-envify "^1.4.0" + prop-types "^15.7.2" + react-is "^16.9.0" + +react@^16.13.1: + version "16.13.1" + resolved "https://registry.yarnpkg.com/react/-/react-16.13.1.tgz#2e818822f1a9743122c063d6410d85c1e3afe48e" + integrity sha512-YMZQQq32xHLX0bz5Mnibv1/LHb3Sqzngu7xstSM+vrkE5Kzr9xE0yMByK5kMoTK30YVJE61WfbxIFFvfeDKT1w== + dependencies: + loose-envify "^1.1.0" + object-assign "^4.1.1" + prop-types "^15.6.2" + +regenerator-runtime@^0.13.4: + version "0.13.5" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.13.5.tgz#d878a1d094b4306d10b9096484b33ebd55e26697" + integrity sha512-ZS5w8CpKFinUzOwW3c83oPeVXoNsrLsaCoLtJvAClH135j/R77RuymhiSErhm2lKcwSCIpmvIWSbDkIfAqKQlA== From 7d2c2af3e6bd0682030c0fa0f662e9bd3a15cc08 Mon Sep 17 00:00:00 2001 From: Nikita Mazur Date: Thu, 30 Jul 2020 14:07:58 +0300 Subject: [PATCH 2/2] Add README file to access component --- packages/access/README.md | 323 +++++++------------------------------- 1 file changed, 59 insertions(+), 264 deletions(-) diff --git a/packages/access/README.md b/packages/access/README.md index 64c411d..5afc5da 100644 --- a/packages/access/README.md +++ b/packages/access/README.md @@ -1,287 +1,82 @@ -## The goal -working with REST-api and redux, using common practice, we always create almost same actions and reducers to send HTTP request to different endpoints. This will lead to problem that our projects will always have lot of duplicated code. For example +### CheckAccessProvider +A simple provider based on the Redux library, used to store access to components and application pages -``` -// action types -const FETCH_USERS_LIST = Symbol('FETCH_USERS_LIST') -const FETCH_USER = Symbol('FETCH_USER') -const SAVE_USER = Symbol('SAVE_USER') -const CREATE_USER = Symbol('CREATE_USER') +Props: -// action creators -function fetchUsersList() { - return { - type: FETCH_USERS_LIST - } -} +| Property | type | Description | +| --------------- | --------------------- | --------------------- | +| userLevelSelector | Integer | Selector that returns integer value for comparison with the access flag | -function fetchUser(id) { - return { - type: FETCH_USER, - payload: id, - } -} +Usage: -function saveUser(id, data) { - return { - type: SAVE_USER, - payload: data, - meta: {id} - } -} +```javascript +import { Provider } from 'react-redux' +import CheckAccessProvider from '@ds-frontend/access' -function createUser(data) { - return { - type: CREATE_USER, - payload: data, - } -} +import userLevelSelector from '...path' -// reducers -function users(state = {}, action) { - switch(action.type) { - case FETCH_USERS_LIST: - return {...state, isLoading: true} - case SAVE_USER: - return {...state, ...action.payload, isLoading: false} - } +function App(){ + return ( + + + // ...childrens + + + ) } - -// epics -// TODO -... -``` -this example does not contains error handling, caching data, authorization, options, filters ... -And basically we always copy paste this code from file to file and rename function names and constants values - ``` -fetchUsers -fetchBooks -fetchGroups -fetchComputers -fetchOrders -etc -``` - -this package will help u stop to duplicating code and start thinking about more interesting staff in your projects +Than you need to create **userLevelSelector** file. This file should describe the logic of all permissions for each role. For this we prefer to use the **reselect** library -1. REST API CRUD: -``` -endpoint: /api/v1/users/:id +```javascript +import isEmpty from 'lodash/isEmpty' +import get from 'lodash/get' +import { createSelector } from 'reselect' -GET /api/v1/users/ - get users list -POST /api/v1/users/ - create new user -GET /api/v1/users/1 - get user details -PATCH /api/v1/users/1 - update user -POST /api/v1/users/1 - recreate user -PUT /api/v1/users/1 - полностью перезаписать юзера (обычно не используем) -DELETE /api/v1/users/1 - delete user -OPTIONS /api/v1/users/ - get metadata -``` +export const F_PUBLIC = 2 ** 0 +export const F_PROTECTED = 2 ** 1 +export const F_UNAUTHORISED = 2 ** 2 +export const F_CHIEF = 2 ** 52 - 1 -2. endpoint CRUD -``` -endpoint: /api/v1/some/custom/endpoint -``` -And this basically same as previous, but it has only 1 single resource. +export const userLevelSelector = createSelector( + // base permissions + (state) => isEmpty(get(state, 'session.data')) ? F_UNAUTHORISED : F_PROTECTED, -resources will give u ability to add all data and methods to communicate with REST API almost with 1 line of code. -``` -connectResource([resource]) + // collect all user permissions + (...args) => args.reduce((level, flag) => level | flag, F_PUBLIC), +) -// where: -resource = { - namespace: 'internalResourceName', - endpoint: '/some/endpoint/with/:placeholders', - forceUpdates: true|false (default false), - withNavigation: true|false (default false), - reducer: 'object|paginationList|none|replace|custom function' (default 'object'), - queries: [], -} ``` +*NOTE: F_CHIEF have full access to application. It should contains all flags. the value should be next exponent minus one. The maximum exponent can be 52, because the MAX_SAFE_INTEGER is (2 ** 53)* -And in props u will have next data: -``` -props.internalResourceName = { - data: { /* ... resource data from API ... */ }, - isLoading: true|false, - options: { /* parsed OPTIONS from API */ }, - errors: null, // or object with errors { }, - - // actions - fetch: func, // GET request, useful when no prefetch - fetchOptions: func, // OPTIONS request - create: func, // POST request - save: func, // PATCH request - update: func, // PATCH request, alias for save - remove: func, // DELETE request - replace: func, // PUT request if you need it - setData: func, // you can update data in store manually, but please be carefull with this action - setLoading: func, // you can updates isLoading in store manually, but please be carefull with this action - setErrors: func, // you can updates errors in store manually, but please be carefull with this action - setFilters: func, // you can updates current filters in store manually, but please be carefull with this action - filters: {}, // current applied filters -} -``` - - -## Options - -all options could be defined with connectResource (on initialization level) and then u can override all options when u will call a function - - -#### `namespace : String` [required] - -property name for resource binding. e.g. for `namespace: 'test'` you will get resource in `props.test`. And all data will be saved in store under the "test" key. - - -#### `endpoint : String` [optional] [default: value of namespace option] - -will be set to `namespace` if ommited. resource endpoint name - where to get data from server. -can contains placeholders (see doc for `path-to-regexp` package for additional information). all you current props will be forwarded to `path-to-regexp`, plus one additional property `id` wich will be get by `idKey` (see below) - -#### `forceUpdates : Boolean` [optional] [default: false] -By default resources will trigger all circle methods - 1. toggle loading, clear errors, safe filters - 2. send HTTP request - 3. toggle loading, set errors(on catch) - - In case u want just send HTTP request and then send another one for better performance it is better to set this param to false and call this methods by your own - -#### `queries : Object` [optional] [default: {}] - -used with list resources. representing initial query for fetch request. -* By default it will skip queries for all API requests except GET +### CheckAccess +React component (Consumer) for condition rendering -#### `reducer : String|Function` [optional] [default: 'object'}] -one of possible reducer functions that u can use. -Possible variants: -- **object**: Object.assign (prevState, nextState) -- **paginationList**: reducer for lists dataStructures `{count:int,results:[]}` -- **none**: return prevState -- **replace** return nextState -or u can define your custom function +Props: +| Property | type | Description | +| --------------- | --------------------- | --------------------- | +| access | Integer | Acess Level | +| fallback | React Element | React Componet to render if condition is `false` | +| children | Node | Will render children component if condition is `true` | +Usage: -### Examples +```javascript +import { CheckAccess } from '@ds-frontend/access' -``` -class App extends Component { - componentDidmount() { - //get users - this.props.users.fetch() - //get books - this.props.books.fetch({offset: 0, limit: 25}) - //get book by id - this.props.books.fetch({id: 'harry_potter'}) - //but u can to more! - this.props.books.fetch({id: 'harry_potter'}, {namespace: 'test', endpoint: 'cars'}) //that will send get request to cars endpoint and save data to test in redux - or - this.props.books.setLoading(true); //set loading - this.props.books.fetch(null, {endpoint: 'users/me', reducer: 'none', forceUpdates: true}) - .then((me)=>this.props.books.fetch({id: me.uuid})) //get book with same uuid as user - - } - render(){...} -} - - - - connectResource([ - { - namespace: 'books', - endpoint: 'books/:id?', - queries: ['offset', 'limit'], - reducer: 'paginationList' - }, - 'users' //u can use even just a string in case your enpoint == namespace and all other configs are default - ])(App) -``` - -#### connect single resource -``` -connectResource({ - namespace: 'books', - endpoint: 'books/:id?' -}) -or -connectResource('users') -``` -#### connect multiple resource -``` -connectResource([{ - namespace: 'books', - endpoint: 'books/:id?' -}, 'users']) -``` - -#### customResource -In case u need more complex logic rather then sending 1 HTTP request, but u still want to have all abilities that brings `resources`. -customResource will return HOC which will add same props as connectResource, and add 1 more function `this.props[namespace].customFetch` and this customFetch will work same as .fetch but instead of sending predefined single HTTP request it will run your own async task. So it might be very useful in your projects to have standard resource abilities + your own async job - -``` -function myCustomFetch(API, payload, meta) { - return new Promise(function(resolve,reject){ - setTimeout(()=>resolve({succes: true}),1000) - }) -} - -const customConnect = customResource(myCustomFetch) -``` - -then u can use this HOC with your components -``` -class Test extends Component { - componentDidMount(){ - this.props.test.customFetch({},{}) - } - ... +import { F_PROTECTED } from 'path/userLevelSelector' + + +export default function SessionText() { + return ( + Unauthorised} + > + Authorised + + ) } -export default customConnect({namespace: 'test', endpoint: 'test'})(Test) -//or -export default customConnect('test')(Test) -///but not -customConnect([{...}, 'test'])(Test) ``` - -customConnect HOC will not support array of resources because it is single resource HOC - - -# HOCS -This package will also provide a standard list of React Hight Order Components: - - [prefetchResources](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/Prefetch.md) - - [withReduxForm](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithReduxForm.md) - - [withInfinityList](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithInfinityList.md) - - [withFinalForm](https://github.com/django-stars/ds-frontend-resource/blob/master/src/hocs/WithFinalForm.md) - - navigationToProps - - -# Hook - -## useResource -Hook that will create resource in Redux and return same props as connectResources -Accepts same configuration as connectResources - -``` -import { useResource } from '@ds-frontend/resource' - -function MyReactComponent () { - const { data, isLoading, errors, fetch } = useResource('users/me') - useEffect(()=>{ - const request = fetch() - return request.cancel - }, []) - if(isLoading) { - return 'Loading...' - } - if(errors){ - return 'Oooops something went wrong please contact local police office' - } - return - -} -``` - - \ No newline at end of file +Since our flag depends on whether our session is active or not, a text `Authorised` will be displayed during an active session. Otherwise, the text `Unauthorised` will be displayed