mirror of
https://github.com/michivonah/nextjs.git
synced 2025-12-22 22:16:28 +01:00
implement error handling
This commit is contained in:
parent
39005d46a5
commit
7d2dea886c
5 changed files with 144 additions and 8 deletions
68
README.md
68
README.md
|
|
@ -570,6 +570,74 @@ export default async function Page(props: {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Error handling
|
||||||
|
Error handling works in `TypeScript` the same way like in JS.
|
||||||
|
```tsx
|
||||||
|
try{
|
||||||
|
await sql`DELETE FROM invoices WHERE id = ${id}`;
|
||||||
|
} catch (error){
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
> Redirects have to be outside of a try & catch block, becuase a redirect would cause to trigger a error each time it's executed.
|
||||||
|
|
||||||
|
A error also can be thrown manually by using `throw`.
|
||||||
|
```tsx
|
||||||
|
throw new Error('Failed to delete invoice.');
|
||||||
|
```
|
||||||
|
|
||||||
|
To show an error also to the user instead only on the server you can add a `error.tsx` file. This file needs to be a client component.
|
||||||
|
|
||||||
|
Here is a simple example for a such error page:
|
||||||
|
```tsx
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useEffect } from 'react';
|
||||||
|
|
||||||
|
export default function Error({
|
||||||
|
error,
|
||||||
|
reset,
|
||||||
|
}: {
|
||||||
|
error: Error & { digest?: string };
|
||||||
|
reset: () => void;
|
||||||
|
}) {
|
||||||
|
useEffect(() => {
|
||||||
|
// Optionally log the error to an error reporting service
|
||||||
|
console.error(error);
|
||||||
|
}, [error]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<main className="flex h-full flex-col items-center justify-center">
|
||||||
|
<h2 className="text-center">Something went wrong!</h2>
|
||||||
|
<button
|
||||||
|
className="mt-4 rounded-md bg-blue-500 px-4 py-2 text-sm text-white transition-colors hover:bg-blue-400"
|
||||||
|
onClick={
|
||||||
|
// Attempt to recover by trying to re-render the invoices route
|
||||||
|
() => reset()
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Try again
|
||||||
|
</button>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
It's also possible to create separate error pages for specific errors. An example would be a not found page.
|
||||||
|
|
||||||
|
A redirect to a not found page can be implemented with `notFound` from `next/navigation`.
|
||||||
|
|
||||||
|
```tsx
|
||||||
|
import { notFound } from 'next/navigation';
|
||||||
|
|
||||||
|
if(!invoice){
|
||||||
|
notFound();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Therefor the file `not-found.tsx` should be created with the UI you want to show, when something is not found.
|
||||||
|
Thoose specific error pages will be shown before the general error page.
|
||||||
|
|
||||||
|
|
||||||
## Ressources
|
## Ressources
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,18 @@
|
||||||
|
import Link from 'next/link';
|
||||||
|
import { FaceFrownIcon } from '@heroicons/react/24/outline';
|
||||||
|
|
||||||
|
export default function NotFound() {
|
||||||
|
return (
|
||||||
|
<main className="flex h-full flex-col items-center justify-center gap-2">
|
||||||
|
<FaceFrownIcon className="w-10 text-gray-400" />
|
||||||
|
<h2 className="text-xl font-semibold">404 Not Found</h2>
|
||||||
|
<p>Could not find the requested invoice.</p>
|
||||||
|
<Link
|
||||||
|
href="/dashboard/invoices"
|
||||||
|
className="mt-4 rounded-md bg-blue-500 px-4 py-2 text-sm text-white transition-colors hover:bg-blue-400"
|
||||||
|
>
|
||||||
|
Go Back
|
||||||
|
</Link>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
import Form from '@/app/ui/invoices/edit-form';
|
import Form from '@/app/ui/invoices/edit-form';
|
||||||
import Breadcrumbs from '@/app/ui/invoices/breadcrumbs';
|
import Breadcrumbs from '@/app/ui/invoices/breadcrumbs';
|
||||||
import { fetchCustomers, fetchInvoiceById } from '@/app/lib/data';
|
import { fetchCustomers, fetchInvoiceById } from '@/app/lib/data';
|
||||||
|
import { notFound } from 'next/navigation';
|
||||||
|
|
||||||
export default async function Page(props: {
|
export default async function Page(props: {
|
||||||
params: Promise<{
|
params: Promise<{
|
||||||
|
|
@ -14,6 +15,10 @@ export default async function Page(props: {
|
||||||
fetchCustomers(),
|
fetchCustomers(),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
if(!invoice){
|
||||||
|
notFound();
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<main>
|
<main>
|
||||||
<Breadcrumbs
|
<Breadcrumbs
|
||||||
|
|
|
||||||
31
dashboard-app-course/app/dashboard/invoices/error.tsx
Normal file
31
dashboard-app-course/app/dashboard/invoices/error.tsx
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useEffect } from 'react';
|
||||||
|
|
||||||
|
export default function Error({
|
||||||
|
error,
|
||||||
|
reset,
|
||||||
|
}: {
|
||||||
|
error: Error & { digest?: string };
|
||||||
|
reset: () => void;
|
||||||
|
}) {
|
||||||
|
useEffect(() => {
|
||||||
|
// Optionally log the error to an error reporting service
|
||||||
|
console.error(error);
|
||||||
|
}, [error]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<main className="flex h-full flex-col items-center justify-center">
|
||||||
|
<h2 className="text-center">Something went wrong!</h2>
|
||||||
|
<button
|
||||||
|
className="mt-4 rounded-md bg-blue-500 px-4 py-2 text-sm text-white transition-colors hover:bg-blue-400"
|
||||||
|
onClick={
|
||||||
|
// Attempt to recover by trying to re-render the invoices route
|
||||||
|
() => reset()
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Try again
|
||||||
|
</button>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
@ -27,10 +27,14 @@ export async function createInvoice(formData: FormData){
|
||||||
const amountInCents = amount * 100;
|
const amountInCents = amount * 100;
|
||||||
const date = new Date().toISOString().split('T')[0];
|
const date = new Date().toISOString().split('T')[0];
|
||||||
|
|
||||||
|
try{
|
||||||
await sql`
|
await sql`
|
||||||
INSERT INTO invoices (customer_Id, amount, status, date)
|
INSERT INTO invoices (customer_Id, amount, status, date)
|
||||||
VALUES (${customerId}, ${amountInCents}, ${status}, ${date})
|
VALUES (${customerId}, ${amountInCents}, ${status}, ${date})
|
||||||
`;
|
`;
|
||||||
|
} catch(error){
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
|
||||||
revalidatePath('/dashboard/invoices');
|
revalidatePath('/dashboard/invoices');
|
||||||
redirect('/dashboard/invoices');
|
redirect('/dashboard/invoices');
|
||||||
|
|
@ -46,17 +50,27 @@ export async function updateInvoice(id: string, formData: FormData){
|
||||||
});
|
});
|
||||||
|
|
||||||
const amountInCents = amount * 100;
|
const amountInCents = amount * 100;
|
||||||
|
|
||||||
|
try{
|
||||||
await sql`
|
await sql`
|
||||||
UPDATE invoices
|
UPDATE invoices
|
||||||
SET customer_id = ${customerId}, amount = ${amountInCents}, status = ${status}
|
SET customer_id = ${customerId}, amount = ${amountInCents}, status = ${status}
|
||||||
WHERE id = ${id}
|
WHERE id = ${id}
|
||||||
`;
|
`;
|
||||||
|
} catch(error){
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
|
||||||
revalidatePath('/dashboard/invoices');
|
revalidatePath('/dashboard/invoices');
|
||||||
redirect('/dashboard/invoices');
|
redirect('/dashboard/invoices');
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteInvoice(id: string){
|
export async function deleteInvoice(id: string){
|
||||||
|
try{
|
||||||
await sql`DELETE FROM invoices WHERE id = ${id}`;
|
await sql`DELETE FROM invoices WHERE id = ${id}`;
|
||||||
|
} catch (error){
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
|
||||||
revalidatePath('/dashboard/invoices');
|
revalidatePath('/dashboard/invoices');
|
||||||
}
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue