Skip to content

Commit

Permalink
Merge branch 'main' into luna/new-banner
Browse files Browse the repository at this point in the history
  • Loading branch information
dasfmi authored Jul 22, 2024
2 parents 6cf4a8b + c1b8e11 commit 5d88654
Show file tree
Hide file tree
Showing 8 changed files with 108 additions and 97 deletions.
119 changes: 54 additions & 65 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,60 +26,57 @@ For more information, check out the [official documentation](https://axiom.co/do

This library allows you to send Web Vitals as well as structured logs from your Next.js application to Axiom.

## Installation
> Using the Pages Router? Use version `0.*` which continues to receive security patches. Here's the [README for `0.x`](https://github.com/axiomhq/next-axiom/blob/v0.x/README.md).
> **Note**
> Using the Pages Router? Use version `0.*`, which will continue to get security patches. Here's the [README for `0.x`](https://github.com/axiomhq/next-axiom/blob/v0.x/README.md).
## Prerequisites

In your Next.js project, install next-axiom:
- [Create an Axiom account](https://app.axiom.co/).
- [Create a dataset in Axiom](/docs/reference/datasets) where you send your data.
- [Create an API token in Axiom](/docs/reference/tokens) with permissions to create, read, update, and delete datasets.
- [A new or existing Next.js app](https://nextjs.org/).

```sh
npm install --save next-axiom
```
## Install next-axiom

In the `next.config.ts` file, wrap your Next.js config in `withAxiom` as follows:
1. In your terminal, go to the root folder of your Next.js app, and then run `npm install --save next-axiom` to install the latest version of next-axiom.
2. Add the following environment variables to your Next.js app. For more information, see the [Vercel documentation](https://vercel.com/docs/projects/environment-variables).
- `NEXT_PUBLIC_AXIOM_DATASET` is the name of the Axiom dataset where you want to send data.
- `NEXT_PUBLIC_AXIOM_TOKEN` is the Axiom API token you have generated.
3. In the `next.config.ts` file, wrap your Next.js configuration in `withAxiom`:

```js
const { withAxiom } = require('next-axiom');

module.exports = withAxiom({
// ... your existing config
// Your existing configuration.
});
```

If you are using the [Vercel integration](https://www.axiom.co/vercel),
no further configuration is required.

Otherwise create a dataset and an API token in [Axiom settings](https://app.axiom.co/settings/profile), then export them as environment variables `NEXT_PUBLIC_AXIOM_DATASET` and `NEXT_PUBLIC_AXIOM_TOKEN`.


## Capture traffic requests

Create or edit the `middleware.ts` in the root directory of your app:
To capture traffic requests, create a `middleware.ts` file in the root folder of your Next.js app:

```typescript
```ts
import { Logger } from 'next-axiom'
import { NextResponse } from 'next/server'
import type { NextFetchEvent, NextRequest } from 'next/server'

export async function middleware(request: NextRequest, event: NextFetchEvent) {
const logger = new Logger({ source: 'middleware' });
const logger = new Logger({ source: 'middleware' }); // traffic, request
logger.middleware(request)

event.waitUntil(logger.flush())
return NextResponse.next()
}


// For more information, see Matching Paths below
export const config = {
}
```
### Web Vitals
## Web Vitals
Go to `app/layout.tsx` and add the Web Vitals component:
To send Web Vitals to Axiom, add the `AxiomWebVitals` component from next-axiom to the `app/layout.tsx` file:
```tsx
```ts
import { AxiomWebVitals } from 'next-axiom';

export default function RootLayout() {
Expand All @@ -93,25 +90,24 @@ export default function RootLayout() {
}
```
> **Note**: WebVitals are only sent from production deployments.
Web Vitals are only sent from production deployments.
### Logs
## Logs
Send logs to Axiom from different parts of your application. Each log function call takes a message and an optional fields object.
Send logs to Axiom from different parts of your app. Each log function call takes a message and an optional `fields` object.
```typescript
log.debug('Login attempt', { user: 'j_doe', status: 'success' }); // results in {"message": "Login attempt", "fields": {"user": "j_doe", "status": "success"}}
```ts
log.debug('Login attempt', { user: 'j_doe', status: 'success' }); // Results in {"message": "Login attempt", "fields": {"user": "j_doe", "status": "success"}}
log.info('Payment completed', { userID: '123', amount: '25USD' });
log.warn('API rate limit exceeded', { endpoint: '/users/1', rateLimitRemaining: 0 });
log.error('System Error', { code: '500', message: 'Internal server error' });
```
#### Route Handlers
### Route handlers
Wrapping your Route Handlers in `withAxiom` will add a logger to your
request and automatically log exceptions:
Wrap your route handlers in `withAxiom` to add a logger to your request and log exceptions automatically:
```typescript
```ts
import { withAxiom, AxiomRequest } from 'next-axiom';

export const GET = withAxiom((req: AxiomRequest) => {
Expand All @@ -125,11 +121,11 @@ export const GET = withAxiom((req: AxiomRequest) => {
});
```
#### Client Components
### Client components
For Client Components, you can add a logger to your component with `useLogger`:
To send logs from client components, add `useLogger` from next-axiom to your component:
```tsx
```ts
'use client';
import { useLogger } from 'next-axiom';

Expand All @@ -140,11 +136,11 @@ export default function ClientComponent() {
}
```
#### Server Components
### Server components
For Server Components, create a logger and make sure to call flush before returning:
To send logs from server components, add `Logger` from next-axiom to your component, and call flush before returning:
```tsx
```ts
import { Logger } from 'next-axiom';

export default async function ServerComponent() {
Expand All @@ -158,42 +154,31 @@ export default async function ServerComponent() {
}
```
#### Log Levels
### Log levels
The log level defines the lowest level of logs sent to Axiom.
The default is debug, resulting in all logs being sent.
Available levels are (from lowest to highest): `debug`, `info`, `warn`, `error`
The log level defines the lowest level of logs sent to Axiom. Choose one of the following levels (from lowest to highest):
For example, if you don't want debug logs to be sent to Axiom:
- `debug` is the default setting. It means that you send all logs to Axiom.
- `info`
- `warn`
- `error` means that you only send the highest-level logs to Axiom.
- `off` means that you don't send any logs to Axiom.
For example, to send all logs except for debug logs to Axiom:
```sh
export NEXT_PUBLIC_AXIOM_LOG_LEVEL=info
```
You can also disable logging completely by setting the log level to `off`.

#### Custom printing to console

Printing logs to the console can be customized through the `prettyPrint` configuration. It is a function taking a `LogEvent` as parameter.
The following example prints the event message without any formatting:

```typescript
const logger = new Logger({
prettyPrint: (event) => {
console.log(event.message);
},
});
```

### Capturing Errors
## Capture errors
To capture routing errors we can use the [Error Handling](https://nextjs.org/docs/app/building-your-application/routing/error-handling) mechanism of Next.
To capture routing errors, use the [error handling mechanism of Next.js](https://nextjs.org/docs/app/building-your-application/routing/error-handling):
Create or edit the `error.tsx` file under your `/app` directory. Inside your component function use the logger to ingest the error to Axiom.
1. Go to the `app` folder.
2. Create an `error.tsx` file.
3. Inside your component function, add `useLogger` from next-axiom to send the error to Axiom. For example:
Example:

```typescript
```ts
"use client";

import NavTable from "@/components/NavTable";
Expand Down Expand Up @@ -227,8 +212,12 @@ export default function ErrorPage({
);

return (
<div>
<div className="p-8">
Ops! An Error has occurred:{" "}
<p className="text-red-400 px-8 py-2 text-lg">`{error.message}`</p>
<div className="w-1/3 mt-8">
<NavTable />
</div>
</div>
);
}
Expand Down
2 changes: 2 additions & 0 deletions examples/logger/.env-example
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
NEXT_PUBLIC_AXIOM_DATASET=""
NEXT_PUBLIC_AXIOM_TOKEN=""
2 changes: 2 additions & 0 deletions examples/logger/.gitignore
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.

.env

# dependencies
/node_modules
/.pnp
Expand Down
5 changes: 5 additions & 0 deletions examples/logger/app/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,10 @@ function Home() {

logger.info('Hello from client', { foo: 'bar' });

const logFromEventHandler = () => {
logger.info('Hello from event handler', { foo: 'bar' });
};

return (
<main className={styles.main}>
<h1>
Expand All @@ -17,6 +21,7 @@ function Home() {
<h1>
<Link href="/worker">Worker</Link>
</h1>
<button onClick={logFromEventHandler}>Log from event handler</button>
</main>
);
}
Expand Down
44 changes: 33 additions & 11 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@
"typescript": "^5.1.6"
},
"dependencies": {
"remeda": "^1.29.0",
"use-deep-compare": "^1.2.1",
"vitest": "^1.2.2",
"whatwg-fetch": "^3.6.2"
}
Expand Down
19 changes: 11 additions & 8 deletions src/hooks.ts
Original file line number Diff line number Diff line change
@@ -1,18 +1,21 @@
import { usePathname } from 'next/navigation';
import { Logger, LoggerConfig } from './logger';
import { useEffect, useMemo } from 'react';
import { useDeepMemo } from './util';
import { useDeepCompareMemo } from 'use-deep-compare';

export function useLogger(config: LoggerConfig = {}): Logger {
const path = usePathname();

const memoizedConfig = useDeepMemo({
...config,
args: {
...(config.args ?? {}),
path,
},
});
const memoizedConfig = useDeepCompareMemo(
() => ({
...config,
args: {
...(config.args ?? {}),
path,
},
}),
[config, path]
);

const logger = useMemo(() => new Logger(memoizedConfig), [memoizedConfig]);

Expand Down
12 changes: 0 additions & 12 deletions src/util.ts

This file was deleted.

0 comments on commit 5d88654

Please sign in to comment.