Official React hooks SDK for the Newsdata.io News
API. Drop-in useLatestNews, useArchiveNews, useCryptoNews,
useMarketNews, useNewsCount, … hooks plus a <NewsDataProvider> to share
one client. Built on the same proven core as the Node client — validation,
typed errors, retries with exponential backoff — and ships first-class
TypeScript definitions.
Zero runtime dependencies, no build step, React 18+ as a peer dependency.
npm install newsdataapi
# react is a peer dependency
npm install reactimport { NewsDataProvider, useLatestNews } from 'newsdataapi';
function Headlines() {
const { data, error, isLoading } = useLatestNews({
q: 'bitcoin',
country: ['us', 'gb'],
language: 'en',
});
if (isLoading) return <p>Loading…</p>;
if (error) return <p>Error: {error.message}</p>;
return (
<ul>
{data.results.map((article) => (
<li key={article.article_id}>
<a href={article.link}>{article.title}</a>
</li>
))}
</ul>
);
}
export default function App() {
return (
<NewsDataProvider apiKey={process.env.REACT_APP_NEWSDATA_API_KEY}>
<Headlines />
</NewsDataProvider>
);
}| Hook | Endpoint | Notes |
|---|---|---|
useLatestNews(params) |
/1/latest |
Real-time news |
useArchiveNews(params) |
/1/archive |
Historical news |
useNewsSources(params) |
/1/sources |
Available sources (single page) |
useCryptoNews(params) |
/1/crypto |
Cryptocurrency news |
useMarketNews(params) |
/1/market |
Market / financial news |
useNewsCount(params) |
/1/count |
Aggregate counts (requires from_date, to_date) |
useCryptoCount(params) |
/1/crypto/count |
Aggregate crypto counts |
useMarketCount(params) |
/1/market/count |
Aggregate market counts |
Every hook has the same shape:
const { data, error, isLoading, refetch } = useLatestNews(params, options);data— the API response (nulluntil the first fetch resolves)error— a typedNewsdataError(nullon success)isLoading—truewhile a request is in flightrefetch()— re-run the request; returns the underlyingPromise
useLatestNews(params, { enabled: false }) // skip the request until enabledoptions.enabled (default true) defers fetching — handy when params aren't
ready (e.g. waiting on user input).
Values can be a single string or an array of strings (sent comma-joined),
and parameter names are case-insensitive — qInTitle and qintitle are
equivalent:
useLatestNews({ country: ['us', 'gb'], language: 'en', size: 20 });Inline param objects are safe — the hook compares by value, not reference, so re-renders only re-fetch when the values change.
Two ways to wire it up:
// 1. Let the provider construct the client.
<NewsDataProvider apiKey="..." options={{ timeout: 10_000, maxRetries: 3 }}>
<App />
</NewsDataProvider>
// 2. Or pass your own pre-built client (full control over its lifecycle).
import { NewsDataApiClient } from 'newsdataapi';
const client = new NewsDataApiClient(apiKey);
<NewsDataProvider client={client}>
<App />
</NewsDataProvider>Anywhere inside the provider you can grab the client directly:
import { useNewsDataClient } from 'newsdataapi';
function ExportButton() {
const client = useNewsDataClient();
return <button onClick={() => client.archiveApi({ q: 'x' }).then(save)}>Export</button>;
}All hook errors are instances of the typed hierarchy from the core SDK:
import {
NewsdataValidationError, NewsdataAuthError, NewsdataRateLimitError,
NewsdataApiError, NewsdataNetworkError,
} from 'newsdataapi';
if (error instanceof NewsdataRateLimitError) {
console.log('retry after', error.retryAfter, 'seconds');
}NewsdataError (catch-all base)
├── NewsdataValidationError (.param)
├── NewsdataApiError (.statusCode, .responseBody)
│ ├── NewsdataAuthError (401 / 403)
│ ├── NewsdataRateLimitError (429; .retryAfter)
│ └── NewsdataServerError (5xx)
└── NewsdataNetworkError (.cause)
Validation errors are thrown before the request is sent (no API quota
spent) — e.g. setting q and qInTitle together, an unsupported parameter
for that endpoint, or missing from_date/to_date on a count endpoint.
You can also use the underlying client without React — same surface as
newsdata-nodejs-client:
import { NewsDataApiClient } from 'newsdataapi';
const client = new NewsDataApiClient(apiKey, { timeout: 10_000 });
// scroll: follow nextPage cursors and merge.
const merged = await client.latestApi({ q: 'news', scroll: true, maxResult: 200 });
// paginate: async generator, one page at a time.
for await (const page of client.latestApi({ q: 'news', paginate: true, maxPages: 5 })) {
console.log(page.results.length);
}npm install
npm test # node:test, 34 tests, no API key required