Research
Security News
Malicious npm Packages Inject SSH Backdoors via Typosquatted Libraries
Socket’s threat research team has detected six malicious npm packages typosquatting popular libraries to insert SSH backdoors.
@decathlon/moon
Advanced tools
A featured, production ready caching REST client for every React UI
The power of react-query with your favorite HTTP Client
Moon is a featured, production ready caching REST client based on react-query for every React UI. It allows you to manage queries life cycle with an api and easily build React UI components that fetch data via a HTTP client (Axios, fetch... ). The only thing you have to do is transmit the configuration. Moon does the rest ! 🚀
Moon client can be used in any React app where you want to use data. It's:
npm install @decathlon/moon @decathlon/moon-axios react-query axios --save
You get started by create REST links. A link is an object which need an id and an HTTP client config like the AxiosConfig (that extends the Moon's ClientConfig) of your REST server (for more information about the REST link config please see the Moon config section).
To connect Moon to your React app, you will need to use the MoonProvider component exported from @decathlon/moon
. The MoonProvider is a React's Context.Provider. It wraps your React app and places the client and the store (the query cache of the react-query) on the context, which allows you to access it from anywhere in your component tree. You also need to add the HTTP client factory (clientFactory). Here we have added the Axios factory to create an axios client for each link. You can also add a client factory for each link. This is useful for using multiple data sources or for mocking data during development phase (tests, mvp...).
import { MoonProvider } from "@decathlon/moon";
import axiosClientFactory from "@decathlon/moon-axios";
const links = [
{
id: "FOO",
config: { baseURL: "http://foo.com" }, // the Client config,
// clientFactory?: ClientFactory<C, R, I>
}
];
const App = () => {
return (
<MoonProvider links={links} clientFactory={axiosClientFactory}>
<MyComponent />
</MoonProvider>
);
};
Once your MoonProvider is hooked up, you're ready to start requesting data with the Query component or with the useQuery hook!
import { Query } from "@decathlon/moon";
const MyComponent = () => {
return (
<Query<QueryVariables, QueryResponse, QueryError>
id="queryId"
source="FOO"
endPoint="/users"
variables={{ foo: "bar" }}
fetchPolicy={FetchPolicy.CacheFirst} // please see the fetchPolicy query prop
>
{({ isLoading, data, error }) => {
if (isLoading) return <span> Loading ...</span>;
return <span>{error ? error.message : "success"}</span>;
}}
</Query>
);
};
Congrats 🎉, you just made your first query with the Query component!
The same query with the useQuery hook
import { useQuery } from "@decathlon/moon";
const MyComponent = () => {
const [{ isLoading, error }, { refetch }] = useQuery<QueryVariables, QueryData, QueryError>({
id: "queryId",
source: "FOO",
endPoint: "/users",
variables: { foo: "bar" },
fetchPolicy: FetchPolicy.CacheFirst // please see the fetchPolicy query prop
// options: {...} // the http client config
// queryConfig: {...} // the react-query config
});
if (isLoading) return <span> Loading ...</span>;
return <span>{error ? error.message : "success"}</span>;
};
Internally useQuery use the react-query's useQuery hook connected to your HTTP client with a configuration allowing better cache management (fetch policy) and better referencing (management of query identifiers adapted to the use of HTTP clients, useQueryState/useQueryResult...) of requests for REST clients.
import { useInfiniteQuery } from "@decathlon/moon";
interface QueryData {
comments: any;
nextId: string | null
}
interface PageVariables {
cursor: string
}
const MyComponent = () => {
const [{ isLoading, error, data }] = useInfiniteQuery<QueryVariables, PageVariables, QueryData, QueryError>({
source: "FOO",
endPoint: "/comments",
variables: { user: "bar" },
queryConfig: {
getNextPageParam: lastPage => {
return lastPage.nextId ? { cursor: lastPage.nextId } : undefined;
}
}
});
if (isLoading) return <span>Loading ...</span>;
if (error) return <span>{error.message}</span>;
return data.pages.map((page, i) => (
<React.Fragment key={i}>
{page.comments.map(comment => (
<Comment {...comment} />
))}
</React.Fragment>
));
};
Internally useInfiniteQuery use the react-query's useInfiniteQuery hook connected to your HTTP like the moon useQuery.
Now that we've learned how to fetch data with the Query/useQuery component/hook, the next step is to learn how to mutate that data with mutations. For that we need to use the Mutation/useMutation component/hook.
import { Mutation } from '@decathlon/moon';
const MyComponent = () => {
return (
<Mutation<MutationVariables, MutationResponse, MutationError> source="FOO" endPoint="/users" variables={{ foo: "bar" }}>
{({ data, error, actions: { mutate } }) => {
const result = data ? <span>{data.status && "Success"}</span> : <div onClick={()=>mutate()}>Go</div>;
return error ? <span>{error.message}</span> : result;
}}
</Mutation>
);
};
The same mutation with useMutation:
import { useQuery } from '@decathlon/moon';
const MyComponent = () => {
const variables = React.useMemo(() => ({ foo: "bar" }), [...]);
const [{ error, data }, { mutate }] = useMutation<MutationResponse, MutationVariables>({
source: "FOO",
endPoint: "/users",
variables: { foo: "bar" },
// type: ..., // the mutation type (POST, PUT...)
// options: {...} // the http client config
// mutationConfig: {...} // the react-query config
});
const result = data ? <span>{data.status && "Success"}</span> : <div onClick={()=>mutate()}>Go</div>;
return error ? <span>{error.message}</span> : result;
};
Internally useMutation use the react-query's useMutation connected to your HTTP client.
Sometimes we need to retrieve the state/result of a query in another component. useQueryResult/useQueriesResult/useQueryState/useQueriesStates allows you to do this. For that, it is enough to give him the id/ids of the query/queries:
Updated when the query state is changed. The optional stateToProps function is used for selecting the part of the data from the query state that the connected component needs.
import { useQueryState } from '@decathlon/moon';
const MyComponent = () => {
const stateToProps = (queryState) => queryState // optional
const isInfinite = true // optional - true if the it's an infinite query (default value === false)
const { isFetching } = useQueryState("queryId", stateToProps, isInfinite);
return <span>{isFetching ? "Loading..." : "success"}</span>;
};
The first prop is the query id used by the query. If the query is defined without an id then the id generated by default must be used. To generate an identifier, you must use the getQueryId utility. The default id is generated from the source, the endPoint and the variables props of the query.
import { useQueryState, getQueryId } from "@decathlon/moon";
const MyComponent = () => {
const queryId = getQueryId({ source: "FOO", endPoint: "/users", variables: { foo: "bar" } });
const { isFetching } = useQueryState(queryId);
return <span>{isFetching ? "Loading..." : "success"}</span>;
};
Updated when one of the query states is changed.The optional statesToProps function is used for selecting the part of the data from the query state that the connected component needs.
import { useQueriesStates } from '@decathlon/moon';
const MyComponent = () => {
const statesToProps = (queriesStates) => queriesStates
const { queryId: { isFetching } } = useQueriesStates(["queryId"], statesToProps);
return <span>{isFetching ? "Loading..." : "success"}</span>;
};
Updated only when the query result is changed. .The optional resultToProps function is used for selecting the part of the data from the query result that the connected component needs.
import { useQueryResult } from '@decathlon/moon';
const MyComponent = () => {
const resultToProps = (queryResult) => queryResult // optional
const isInfinite = true // optional - true if the it's an infinite query (default value === false)
const result = useQueryResult("queryId", resultToProps, isInfinite);
return <span>{...result...}</span>;
};
Updated only when one of the query results is changed. The optional statesToProps function is used for selecting the part of the data from the queries results that the connected component needs.
import { useQueriesResults } from '@decathlon/moon';
const MyComponent = () => {
const resultsToProps = (queriesResults) => queriesResults
const { queryId: queryIdResult } = useQueriesResults(["queryId"], statesToProps);
return <span>{...queryIdResult...}</span>;
};
You can use the moon client directly like this:
import { useMoon } from '@decathlon/moon';
const MyComponent = () => {
const { client, store } = useMoon();
client.query(...);
client.mutate(...);
// the store is the queryClient of the react-query API
};
Same as useMoon hook.
import { withMoon } from '@decathlon/moon';
interface Props extends IMoonContextValue {
prop: string;
}
const MyComponent: React.FunctionComponent<Props>= ({ client, store, prop }) => {
...
};
export withMoon<Props>(MyComponent);
Same as useQueryResult hook.
import { withQueryResult } from '@decathlon/moon';
interface Props {
queryResult: QueryState<QueryResponse, QueryError>;
}
const MyComponent: React.FunctionComponent<Props> = ({ queryResult }) => {
...
};
export default withQueryResult<Props, QueryResponse, /* QueryResultProps */>(queryId, /* resultToProps */)(MyComponent);
Same as useQueriesResults hook.
import { withQueriesResults } from '@decathlon/moon';
interface Props {
queriesResults: {
queryId: QueryState<QueryResponse, QueryError>;
queryId2: QueryState<QueryResponse, QueryError>;
}
}
const MyComponent: React.FunctionComponent<Props> = ({ queriesResults: { queryId, queryId2 } }) => {
...
};
export default withQueriesResults<Props, QueryResponse, /* QueryResultProps */>([queryId, queryId2], /* resultToProps */)(MyComponent);
interface IMoonProviderProps {
// The links ( HTTP clients config)
links: ILink[];
// The global Moon client factory (like the moon-axios Axios client for moon https://github.com/dktunited/moon-axios)
clientFactory: ClientFactory;
// The react-query QueryClient object
store?: QueryClient;
// The react-query initial cache state (please see https://react-query.tanstack.com/docs/api#hydrationdehydrate for more details)
hydrate?: HydrateProps;
}
This the Typescript interface of the Query/useQuery component/hook.
export interface IQueryProps<QueryVariables = any, QueryResponse = any, QueryError = any, QueryConfig = any> {
id?: string;
/** The Link id of the http client. */
source: string;
/** The REST end point. */
endPoint?: string;
/** The variables of your query. */
variables?: QueryVariables;
/**
* The fetch policy is an option which allows you to
* specify how you want your component to interact with
* the Moon data cache. Default value: FetchPolicy.CacheAndNetwork */
fetchPolicy?: FetchPolicy;
/** The http client options of your query. */
options?: QueryConfig;
/** The react-query config. Please see the react-query QueryConfig for more details. */
queryConfig?: ReactQueryConfig<QueryResponse, QueryError>;
}
The fetch policy is an option which allows you to specify how you want your component to interact with the Moon data cache. By default your component will try to read from the cache first, and if the full data for your query is in the cache then Moon simply returns the data from the cache. If the full data for your query is not in the cache then Moon will execute your request using your network interface. By changing this option you can change this behavior.
Valid fetchPolicy values are:
export interface IInfiniteQueryProps<QueryVariables = any, QueryResponse = any, QueryError = any, QueryConfig = any> {
id?: string;
/** The Link id of the http client. */
source?: string;
/** The REST end point. */
endPoint?: string;
/** The variables of your query. */
variables?: QueryVariables;
/** The http client options of your query. */
options?: QueryConfig;
/** The react-query config. Please see the react-query QueryConfig for more details. */
queryConfig?: ReactQueryConfig<QueryResponse | undefined, QueryError>;
}
This the Typescript interface of the Mutation/useMutation component/hook.
export interface IMutationProps<
MutationVariables = any,
MutationResponse = any,
MutationError = any,
MutationClientConfig = any
> {
/** The link id of the http client */
source: string;
/** The REST end point */
endPoint?: string;
/** The variables of your mutation */
variables?: MutationVariables;
/** The mutation method. Default value: MutateType.Post */
type?: MutateType;
/** The http client options of your mutation. */
options?: MutationClientConfig;
/** The react-query config. Please see the react-query MutationConfig for more details. */
mutationConfig?: MutationConfig<MutationResponse, MutationError, MutationVariables, unknown>;
}
For each Moon link we can add interceptors (middleware: language, api token, success Handler....) for the request and/or the response like this:
import { AxiosRequestConfig } from "axios";
import axiosClientFactory from "@decathlon/moon-axios";
function successHandler(response: AxiosResponse){...};
function setLanguage(config: AxiosRequestConfig): AxiosRequestConfig | Promise<AxiosRequestConfig> {
return {
...config,
headers: {
...config.headers,
"Accept-Language": "en"
}
};
}
const requestInterceptors = [{ onFulfilled: setLanguage }];
const responseInterceptors = [{ onFulfilled: successHandler }];
const links = [
{
id: "FOO",
config: { baseURL: "http://foo.com" },
interceptors: { request: requestInterceptors, response: responseInterceptors },
clientFactory: axiosClientFactory
},
{
id: "BAR",
config: { baseURL: "http://bar.com" }
clientFactory: yourClientFactory // must extends the Moon ClientInstace
}
];
git clone ...
cd moon
npm ci
npm run test
PRs are welcome! You noticed a bug, a possible improvement or whatever? Any help is always appreciated, so don't hesitate opening one!
Be sure to check out the contributing guidelines to fasten up the merging process.
See also the list of contributors who participated in this project.
This project is licensed under the Apache-2.0 License - see the LICENSE.md file for details
FAQs
A featured, production ready caching REST client for every React UI
We found that @decathlon/moon demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 4 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Research
Security News
Socket’s threat research team has detected six malicious npm packages typosquatting popular libraries to insert SSH backdoors.
Security News
MITRE's 2024 CWE Top 25 highlights critical software vulnerabilities like XSS, SQL Injection, and CSRF, reflecting shifts due to a refined ranking methodology.
Security News
In this segment of the Risky Business podcast, Feross Aboukhadijeh and Patrick Gray discuss the challenges of tracking malware discovered in open source softare.