develop #3
@ -10,6 +10,7 @@
|
|||||||
"seed": "node -r dotenv/config ./src/bootstrap/db/seed.js"
|
"seed": "node -r dotenv/config ./src/bootstrap/db/seed.js"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@heroicons/react": "^2.1.5",
|
||||||
"@radix-ui/react-icons": "^1.3.1",
|
"@radix-ui/react-icons": "^1.3.1",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
@ -21,7 +22,8 @@
|
|||||||
"reflect-metadata": "^0.2.2",
|
"reflect-metadata": "^0.2.2",
|
||||||
"tailwind-merge": "^2.5.4",
|
"tailwind-merge": "^2.5.4",
|
||||||
"tailwindcss-animate": "^1.0.7",
|
"tailwindcss-animate": "^1.0.7",
|
||||||
"tsyringe": "^4.8.0"
|
"tsyringe": "^4.8.0",
|
||||||
|
"zod": "^3.23.8"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^20",
|
"@types/node": "^20",
|
||||||
|
BIN
public/customers/amy-burns.png
Normal file
After Width: | Height: | Size: 7.8 KiB |
BIN
public/customers/balazs-orban.png
Normal file
After Width: | Height: | Size: 7.8 KiB |
BIN
public/customers/delba-de-oliveira.png
Normal file
After Width: | Height: | Size: 5.7 KiB |
BIN
public/customers/emil-kowalski.png
Normal file
After Width: | Height: | Size: 4.0 KiB |
BIN
public/customers/evil-rabbit.png
Normal file
After Width: | Height: | Size: 1019 B |
BIN
public/customers/guillermo-rauch.png
Normal file
After Width: | Height: | Size: 6.1 KiB |
BIN
public/customers/hector-simpson.png
Normal file
After Width: | Height: | Size: 5.5 KiB |
BIN
public/customers/jared-palmer.png
Normal file
After Width: | Height: | Size: 6.3 KiB |
BIN
public/customers/lee-robinson.png
Normal file
After Width: | Height: | Size: 5.5 KiB |
BIN
public/customers/michael-novotny.png
Normal file
After Width: | Height: | Size: 9.7 KiB |
BIN
public/customers/steph-dietz.png
Normal file
After Width: | Height: | Size: 7.0 KiB |
BIN
public/customers/steven-tey.png
Normal file
After Width: | Height: | Size: 7.2 KiB |
@ -1,3 +1,8 @@
|
|||||||
|
import { LatestInvoicesSkeleton, RevenueChartSkeleton } from "@/app/components/skeleton/skeletons";
|
||||||
|
import CardWrapper from "@/app/dashboard/components/cards";
|
||||||
|
import LatestInvoices from "@/app/dashboard/components/latest-invoices";
|
||||||
|
import RevenueChart from "@/app/dashboard/components/revenue-chart";
|
||||||
|
import { Suspense } from "react";
|
||||||
|
|
||||||
export default async function Dashboard() {
|
export default async function Dashboard() {
|
||||||
|
|
||||||
@ -7,15 +12,15 @@ export default async function Dashboard() {
|
|||||||
Dashboard
|
Dashboard
|
||||||
</h1>
|
</h1>
|
||||||
<div className="grid gap-6 sm:grid-cols-2 lg:grid-cols-4">
|
<div className="grid gap-6 sm:grid-cols-2 lg:grid-cols-4">
|
||||||
{/* <CardWrapper /> */}
|
<CardWrapper />
|
||||||
</div>
|
</div>
|
||||||
<div className="mt-6 grid grid-cols-1 gap-6 md:grid-cols-4 lg:grid-cols-8">
|
<div className="mt-6 grid grid-cols-1 gap-6 md:grid-cols-4 lg:grid-cols-8">
|
||||||
{/* <Suspense fallback={<RevenueChartSkeleton />}>
|
<Suspense fallback={<RevenueChartSkeleton />}>
|
||||||
<RevenueChart />
|
<RevenueChart />
|
||||||
</Suspense> */}
|
</Suspense>
|
||||||
{/* <Suspense fallback={<LatestInvoicesSkeleton />}>
|
<Suspense fallback={<LatestInvoicesSkeleton />}>
|
||||||
<LatestInvoices />
|
<LatestInvoices />
|
||||||
</Suspense> */}
|
</Suspense>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
)
|
)
|
||||||
|
62
src/app/dashboard/components/cards.tsx
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
import {
|
||||||
|
BanknotesIcon,
|
||||||
|
ClockIcon,
|
||||||
|
UserGroupIcon,
|
||||||
|
InboxIcon,
|
||||||
|
} from '@heroicons/react/24/outline';
|
||||||
|
import { fetchCardData } from '@/app/lib/data';
|
||||||
|
|
||||||
|
const iconMap = {
|
||||||
|
collected: BanknotesIcon,
|
||||||
|
customers: UserGroupIcon,
|
||||||
|
pending: ClockIcon,
|
||||||
|
invoices: InboxIcon,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function CardWrapper() {
|
||||||
|
const {
|
||||||
|
numberOfInvoices,
|
||||||
|
numberOfCustomers,
|
||||||
|
totalPaidInvoices,
|
||||||
|
totalPendingInvoices,
|
||||||
|
} = await fetchCardData();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Card title="Collected" value={totalPaidInvoices} type="collected" />
|
||||||
|
<Card title="Pending" value={totalPendingInvoices} type="pending" />
|
||||||
|
<Card title="Total Invoices" value={numberOfInvoices} type="invoices" />
|
||||||
|
<Card
|
||||||
|
title="Total Customers"
|
||||||
|
value={numberOfCustomers}
|
||||||
|
type="customers"
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Card({
|
||||||
|
title,
|
||||||
|
value,
|
||||||
|
type,
|
||||||
|
}: {
|
||||||
|
title: string;
|
||||||
|
value: number | string;
|
||||||
|
type: 'invoices' | 'customers' | 'pending' | 'collected';
|
||||||
|
}) {
|
||||||
|
const Icon = iconMap[type];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="rounded-xl bg-gray-50 p-2 shadow-sm">
|
||||||
|
<div className="flex p-4">
|
||||||
|
{Icon ? <Icon className="h-5 w-5 text-gray-700" /> : null}
|
||||||
|
<h3 className="ml-2 text-sm font-medium">{title}</h3>
|
||||||
|
</div>
|
||||||
|
<p
|
||||||
|
className="rounded-xl bg-white px-4 py-8 text-center text-2xl"
|
||||||
|
>
|
||||||
|
{value}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
60
src/app/dashboard/components/latest-invoices.tsx
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import { ArrowPathIcon } from '@heroicons/react/24/outline';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { fetchLatestInvoices } from '@/app/lib/data';
|
||||||
|
export default async function LatestInvoices() {
|
||||||
|
const latestInvoices = await fetchLatestInvoices();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex w-full flex-col md:col-span-4">
|
||||||
|
<h2 className="mb-4 text-xl md:text-2xl">
|
||||||
|
Latest Invoices
|
||||||
|
</h2>
|
||||||
|
<div className="flex grow flex-col justify-between rounded-xl bg-gray-50 p-4">
|
||||||
|
|
||||||
|
<div className="bg-white px-6">
|
||||||
|
{latestInvoices.map((invoice, i) => {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={invoice.id}
|
||||||
|
className={clsx(
|
||||||
|
'flex flex-row items-center justify-between py-4',
|
||||||
|
{
|
||||||
|
'border-t': i !== 0,
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<Image
|
||||||
|
src={invoice.image_url}
|
||||||
|
alt={`${invoice.name}'s profile picture`}
|
||||||
|
className="mr-4 rounded-full"
|
||||||
|
width={32}
|
||||||
|
height={32}
|
||||||
|
/>
|
||||||
|
<div className="min-w-0">
|
||||||
|
<p className="truncate text-sm font-semibold md:text-base">
|
||||||
|
{invoice.name}
|
||||||
|
</p>
|
||||||
|
<p className="hidden text-sm text-gray-500 sm:block">
|
||||||
|
{invoice.email}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p
|
||||||
|
className="truncate text-sm font-medium md:text-base"
|
||||||
|
>
|
||||||
|
{invoice.amount}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center pb-2 pt-6">
|
||||||
|
<ArrowPathIcon className="h-5 w-5 text-gray-500" />
|
||||||
|
<h3 className="ml-2 text-sm text-gray-500 ">Updated just now</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
53
src/app/dashboard/components/revenue-chart.tsx
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import { generateYAxis } from '@/app/lib/utils';
|
||||||
|
import { CalendarIcon } from '@heroicons/react/24/outline';
|
||||||
|
import { fetchRevenue } from '@/app/lib/data';
|
||||||
|
|
||||||
|
export default async function RevenueChart() {
|
||||||
|
const revenue = await fetchRevenue();
|
||||||
|
|
||||||
|
const chartHeight = 350;
|
||||||
|
|
||||||
|
const { yAxisLabels, topLabel } = generateYAxis(revenue);
|
||||||
|
|
||||||
|
if (!revenue || revenue.length === 0) {
|
||||||
|
return <p className="mt-4 text-gray-400">No data available.</p>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-full md:col-span-4">
|
||||||
|
<h2 className={` mb-4 text-xl md:text-2xl`}>
|
||||||
|
Recent Revenue
|
||||||
|
</h2>
|
||||||
|
<div className="rounded-xl bg-gray-50 p-4">
|
||||||
|
<div className="sm:grid-cols-13 mt-0 grid grid-cols-12 items-end gap-2 rounded-md bg-white p-4 md:gap-4">
|
||||||
|
<div
|
||||||
|
className="mb-6 hidden flex-col justify-between text-sm text-gray-400 sm:flex"
|
||||||
|
style={{ height: `${chartHeight}px` }}
|
||||||
|
>
|
||||||
|
{yAxisLabels.map((label) => (
|
||||||
|
<p key={label}>{label}</p>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{revenue.map((month) => (
|
||||||
|
<div key={month.month} className="flex flex-col items-center gap-2">
|
||||||
|
<div
|
||||||
|
className="w-full rounded-md bg-blue-300"
|
||||||
|
style={{
|
||||||
|
height: `${(chartHeight / topLabel) * month.revenue}px`,
|
||||||
|
}}
|
||||||
|
></div>
|
||||||
|
<p className="-rotate-90 text-sm text-gray-400 sm:rotate-0">
|
||||||
|
{month.month}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center pb-2 pt-6">
|
||||||
|
<CalendarIcon className="h-5 w-5 text-gray-500" />
|
||||||
|
<h3 className="ml-2 text-sm text-gray-500 ">Last 12 months</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
117
src/app/lib/actions.ts
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
'use server';
|
||||||
|
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { revalidatePath } from 'next/cache';
|
||||||
|
import { redirect } from 'next/navigation';
|
||||||
|
import { sql } from '@/bootstrap/db/db';
|
||||||
|
|
||||||
|
const FormSchema = z.object({
|
||||||
|
id: z.string(),
|
||||||
|
customerId: z.string({
|
||||||
|
invalid_type_error: 'Please select a customer.',
|
||||||
|
}),
|
||||||
|
amount: z.coerce.number().gt(0, { message: 'Please enter an amount greater than $0.' }),
|
||||||
|
status: z.enum(['pending', 'paid'], {
|
||||||
|
invalid_type_error: 'Please select an invoice status.',
|
||||||
|
}),
|
||||||
|
date: z.string(),
|
||||||
|
});
|
||||||
|
const CreateInvoice = FormSchema.omit({ id: true, date: true });
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// This is temporary
|
||||||
|
export type State = {
|
||||||
|
errors?: {
|
||||||
|
customerId?: string[];
|
||||||
|
amount?: string[];
|
||||||
|
status?: string[];
|
||||||
|
};
|
||||||
|
message?: string | null;
|
||||||
|
};
|
||||||
|
|
||||||
|
export async function createInvoice(_prevState: State, formData: FormData) {
|
||||||
|
// Validate form fields using Zod
|
||||||
|
console.log('ffor', formData)
|
||||||
|
const validatedFields = CreateInvoice.safeParse({
|
||||||
|
customerId: formData?.get('customerId') || undefined,
|
||||||
|
amount: formData?.get('amount') || undefined,
|
||||||
|
status: formData?.get('status') || undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
// If form validation fails, return errors early. Otherwise, continue.
|
||||||
|
if (!validatedFields.success) {
|
||||||
|
return {
|
||||||
|
errors: validatedFields.error.flatten().fieldErrors,
|
||||||
|
message: 'Missing Fields. Failed to Create Invoice.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prepare data for insertion into the database
|
||||||
|
const { customerId, amount, status } = validatedFields.data;
|
||||||
|
const amountInCents = amount * 100;
|
||||||
|
const date = new Date().toISOString().split('T')[0];
|
||||||
|
|
||||||
|
// Insert data into the database
|
||||||
|
try {
|
||||||
|
await sql`
|
||||||
|
INSERT INTO invoices (customer_id, amount, status, date)
|
||||||
|
VALUES (${customerId}, ${amountInCents}, ${status}, ${date})
|
||||||
|
`;
|
||||||
|
} catch {
|
||||||
|
// If a database error occurs, return a more specific error.
|
||||||
|
return {
|
||||||
|
message: 'Database Error: Failed to Create Invoice.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Revalidate the cache for the invoices page and redirect the user.
|
||||||
|
revalidatePath('/dashboard/invoices');
|
||||||
|
redirect('/dashboard/invoices');
|
||||||
|
}
|
||||||
|
|
||||||
|
const UpdateInvoice = FormSchema.omit({ id: true, date: true });
|
||||||
|
export async function updateInvoice(
|
||||||
|
id: string,
|
||||||
|
_prevState: State,
|
||||||
|
formData: FormData,
|
||||||
|
) {
|
||||||
|
const validatedFields = UpdateInvoice.safeParse({
|
||||||
|
customerId: formData.get('customerId'),
|
||||||
|
amount: formData.get('amount'),
|
||||||
|
status: formData.get('status'),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!validatedFields.success) {
|
||||||
|
return {
|
||||||
|
errors: validatedFields.error.flatten().fieldErrors,
|
||||||
|
message: 'Missing Fields. Failed to Update Invoice.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { customerId, amount, status } = validatedFields.data;
|
||||||
|
const amountInCents = amount * 100;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await sql`
|
||||||
|
UPDATE invoices
|
||||||
|
SET customer_id = ${customerId}, amount = ${amountInCents}, status = ${status}
|
||||||
|
WHERE id = ${id}
|
||||||
|
`;
|
||||||
|
} catch {
|
||||||
|
return { message: 'Database Error: Failed to Update Invoice.' };
|
||||||
|
}
|
||||||
|
|
||||||
|
revalidatePath('/dashboard/invoices');
|
||||||
|
redirect('/dashboard/invoices');
|
||||||
|
}
|
||||||
|
export async function deleteInvoice(id: string) {
|
||||||
|
try {
|
||||||
|
await sql`DELETE FROM invoices WHERE id = ${id}`;
|
||||||
|
revalidatePath('/dashboard/invoices');
|
||||||
|
return { message: 'Deleted Invoice.' };
|
||||||
|
} catch {
|
||||||
|
return { message: 'Database Error: Failed to Delete Invoice.' };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
254
src/app/lib/data.ts
Normal file
@ -0,0 +1,254 @@
|
|||||||
|
import { sql } from '@/bootstrap/db/db';
|
||||||
|
import {
|
||||||
|
CustomerField,
|
||||||
|
CustomersTableType,
|
||||||
|
InvoiceForm,
|
||||||
|
InvoicesTable,
|
||||||
|
User,
|
||||||
|
Revenue,
|
||||||
|
Invoice,
|
||||||
|
Customer,
|
||||||
|
LatestInvoice,
|
||||||
|
} from './definitions';
|
||||||
|
import { formatCurrency } from './utils';
|
||||||
|
import postgres from 'postgres';
|
||||||
|
import { connection } from 'next/server';
|
||||||
|
|
||||||
|
export async function fetchRevenue() {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Artificially delay a response for demo purposes.
|
||||||
|
// Don't do this in production :)
|
||||||
|
|
||||||
|
console.log('Fetching revenue data...');
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 3000));
|
||||||
|
|
||||||
|
const data = await sql`SELECT * FROM revenue`;
|
||||||
|
|
||||||
|
console.log('Data fetch completed after 3 seconds.');
|
||||||
|
|
||||||
|
return data as postgres.RowList<Revenue[]>;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch revenue data.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchLatestInvoices() {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
|
||||||
|
try {
|
||||||
|
const data = await sql`
|
||||||
|
SELECT invoices.amount, customers.name, customers.image_url, customers.email, invoices.id
|
||||||
|
FROM invoices
|
||||||
|
JOIN customers ON invoices.customer_id = customers.id
|
||||||
|
ORDER BY invoices.date DESC
|
||||||
|
LIMIT 5` as postgres.RowList<LatestInvoice[]>;
|
||||||
|
|
||||||
|
const latestInvoices = data.map((invoice) => ({
|
||||||
|
...invoice,
|
||||||
|
amount: formatCurrency(+invoice.amount),
|
||||||
|
}));
|
||||||
|
return latestInvoices;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch the latest invoices.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchCardData() {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
|
||||||
|
try {
|
||||||
|
// You can probably combine these into a single SQL query
|
||||||
|
// However, we are intentionally splitting them to demonstrate
|
||||||
|
// how to initialize multiple queries in parallel with JS.
|
||||||
|
const invoiceCountPromise = sql`SELECT COUNT(*) FROM invoices`;
|
||||||
|
const customerCountPromise = sql`SELECT COUNT(*) FROM customers`;
|
||||||
|
const invoiceStatusPromise = sql`SELECT
|
||||||
|
SUM(CASE WHEN status = 'paid' THEN amount ELSE 0 END) AS "paid",
|
||||||
|
SUM(CASE WHEN status = 'pending' THEN amount ELSE 0 END) AS "pending"
|
||||||
|
FROM invoices`;
|
||||||
|
|
||||||
|
const data = await Promise.all([
|
||||||
|
invoiceCountPromise,
|
||||||
|
customerCountPromise,
|
||||||
|
invoiceStatusPromise,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const invoices = data[0] as postgres.RowList<Invoice[]>
|
||||||
|
const customres = data[1] as postgres.RowList<Customer[]>
|
||||||
|
const invoiceStatus = data[2] as postgres.RowList<({paid: string, pending: string})[]>
|
||||||
|
const numberOfInvoices = Number(invoices.count ?? '0');
|
||||||
|
const numberOfCustomers = Number(customres.count ?? '0');
|
||||||
|
const totalPaidInvoices = formatCurrency(Number(invoiceStatus.at(0)?.paid ?? '0'));
|
||||||
|
const totalPendingInvoices = formatCurrency(Number(invoiceStatus.at(0)?.pending ?? '0'));
|
||||||
|
|
||||||
|
return {
|
||||||
|
numberOfCustomers,
|
||||||
|
numberOfInvoices,
|
||||||
|
totalPaidInvoices,
|
||||||
|
totalPendingInvoices,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch card data.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const ITEMS_PER_PAGE = 6;
|
||||||
|
export async function fetchFilteredInvoices(
|
||||||
|
query: string,
|
||||||
|
currentPage: number,
|
||||||
|
) {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
|
||||||
|
const offset = (currentPage - 1) * ITEMS_PER_PAGE;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const invoices = await sql`
|
||||||
|
SELECT
|
||||||
|
invoices.id,
|
||||||
|
invoices.amount,
|
||||||
|
invoices.date,
|
||||||
|
invoices.status,
|
||||||
|
customers.name,
|
||||||
|
customers.email,
|
||||||
|
customers.image_url
|
||||||
|
FROM invoices
|
||||||
|
JOIN customers ON invoices.customer_id = customers.id
|
||||||
|
WHERE
|
||||||
|
customers.name ILIKE ${`%${query}%`} OR
|
||||||
|
customers.email ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.amount::text ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.date::text ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.status ILIKE ${`%${query}%`}
|
||||||
|
ORDER BY invoices.date DESC
|
||||||
|
LIMIT ${ITEMS_PER_PAGE} OFFSET ${offset}
|
||||||
|
` as postgres.RowList<InvoicesTable[]>;
|
||||||
|
|
||||||
|
return invoices;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch invoices.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchInvoicesPages(query: string) {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
try {
|
||||||
|
const count = await sql`SELECT COUNT(*)
|
||||||
|
FROM invoices
|
||||||
|
JOIN customers ON invoices.customer_id = customers.id
|
||||||
|
WHERE
|
||||||
|
customers.name ILIKE ${`%${query}%`} OR
|
||||||
|
customers.email ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.amount::text ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.date::text ILIKE ${`%${query}%`} OR
|
||||||
|
invoices.status ILIKE ${`%${query}%`}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const totalPages = Math.ceil(Number(count.at(0)?.count) / ITEMS_PER_PAGE);
|
||||||
|
return totalPages;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch total number of invoices.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchInvoiceById(id: string) {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
try {
|
||||||
|
const data = await sql`
|
||||||
|
SELECT
|
||||||
|
invoices.id,
|
||||||
|
invoices.customer_id,
|
||||||
|
invoices.amount,
|
||||||
|
invoices.status
|
||||||
|
FROM invoices
|
||||||
|
WHERE invoices.id = ${id};
|
||||||
|
` as postgres.RowList<InvoiceForm[]>;
|
||||||
|
|
||||||
|
const invoice = data.map((invoice) => ({
|
||||||
|
...invoice,
|
||||||
|
// Convert amount from cents to dollars
|
||||||
|
amount: invoice.amount / 100,
|
||||||
|
}));
|
||||||
|
|
||||||
|
return invoice[0];
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Database Error:', error);
|
||||||
|
throw new Error('Failed to fetch invoice.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchCustomers() {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
try {
|
||||||
|
const data = await sql`
|
||||||
|
SELECT
|
||||||
|
id,
|
||||||
|
name
|
||||||
|
FROM customers
|
||||||
|
ORDER BY name ASC
|
||||||
|
` as postgres.RowList<CustomerField[]>;
|
||||||
|
|
||||||
|
return data;
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Database Error:', err);
|
||||||
|
throw new Error('Failed to fetch all customers.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchFilteredCustomers(query: string) {
|
||||||
|
// This is equivalent to in fetch(..., {cache: 'no-store'}).
|
||||||
|
connection()
|
||||||
|
try {
|
||||||
|
const data = await sql`
|
||||||
|
SELECT
|
||||||
|
customers.id,
|
||||||
|
customers.name,
|
||||||
|
customers.email,
|
||||||
|
customers.image_url,
|
||||||
|
COUNT(invoices.id) AS total_invoices,
|
||||||
|
SUM(CASE WHEN invoices.status = 'pending' THEN invoices.amount ELSE 0 END) AS total_pending,
|
||||||
|
SUM(CASE WHEN invoices.status = 'paid' THEN invoices.amount ELSE 0 END) AS total_paid
|
||||||
|
FROM customers
|
||||||
|
LEFT JOIN invoices ON customers.id = invoices.customer_id
|
||||||
|
WHERE
|
||||||
|
customers.name ILIKE ${`%${query}%`} OR
|
||||||
|
customers.email ILIKE ${`%${query}%`}
|
||||||
|
GROUP BY customers.id, customers.name, customers.email, customers.image_url
|
||||||
|
ORDER BY customers.name ASC
|
||||||
|
` as postgres.RowList<CustomersTableType[]>;
|
||||||
|
|
||||||
|
const customers = data.map((customer) => ({
|
||||||
|
...customer,
|
||||||
|
total_pending: formatCurrency(customer.total_pending),
|
||||||
|
total_paid: formatCurrency(customer.total_paid),
|
||||||
|
}));
|
||||||
|
|
||||||
|
return customers;
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Database Error:', err);
|
||||||
|
throw new Error('Failed to fetch customer table.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getUser(email: string) {
|
||||||
|
try {
|
||||||
|
const user = await sql`SELECT * FROM users WHERE email=${email}`;
|
||||||
|
return user.at(0) as User;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch user:', error);
|
||||||
|
throw new Error('Failed to fetch user.');
|
||||||
|
}
|
||||||
|
}
|
86
src/app/lib/definitions.ts
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
// This file contains type definitions for your data.
|
||||||
|
// It describes the shape of the data, and what data type each property should accept.
|
||||||
|
// For simplicity of teaching, we're manually defining these types.
|
||||||
|
// However, these types are generated automatically if you're using an ORM such as Prisma.
|
||||||
|
export type User = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
password: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Customer = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
image_url: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Invoice = {
|
||||||
|
id: string; // Will be created on the database
|
||||||
|
customer_id: string;
|
||||||
|
amount: number; // Stored in cents
|
||||||
|
status: 'pending' | 'paid';
|
||||||
|
date: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Revenue = {
|
||||||
|
month: string;
|
||||||
|
revenue: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type LatestInvoice = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
image_url: string;
|
||||||
|
email: string;
|
||||||
|
amount: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
// The database returns a number for amount, but we later format it to a string with the formatCurrency function
|
||||||
|
export type LatestInvoiceRaw = Omit<LatestInvoice, 'amount'> & {
|
||||||
|
amount: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InvoicesTable = {
|
||||||
|
id: string;
|
||||||
|
customer_id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
image_url: string;
|
||||||
|
date: string;
|
||||||
|
amount: number;
|
||||||
|
status: 'pending' | 'paid';
|
||||||
|
};
|
||||||
|
|
||||||
|
export type CustomersTableType = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
image_url: string;
|
||||||
|
total_invoices: number;
|
||||||
|
total_pending: number;
|
||||||
|
total_paid: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type FormattedCustomersTable = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
image_url: string;
|
||||||
|
total_invoices: number;
|
||||||
|
total_pending: string;
|
||||||
|
total_paid: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type CustomerField = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InvoiceForm = {
|
||||||
|
id: string;
|
||||||
|
customer_id: string;
|
||||||
|
amount: number;
|
||||||
|
status: 'pending' | 'paid';
|
||||||
|
};
|
188
src/app/lib/placeholder-data.js
Normal file
@ -0,0 +1,188 @@
|
|||||||
|
// This file contains placeholder data that you'll be replacing with real data in the Data Fetching chapter:
|
||||||
|
// https://nextjs.org/learn/dashboard-app/fetching-data
|
||||||
|
const users = [
|
||||||
|
{
|
||||||
|
id: '410544b2-4001-4271-9855-fec4b6a6442a',
|
||||||
|
name: 'User',
|
||||||
|
email: 'user@nextmail.com',
|
||||||
|
password: '123456',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const customers = [
|
||||||
|
{
|
||||||
|
id: '3958dc9e-712f-4377-85e9-fec4b6a6442a',
|
||||||
|
name: 'Delba de Oliveira',
|
||||||
|
email: 'delba@oliveira.com',
|
||||||
|
image_url: '/customers/delba-de-oliveira.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '3958dc9e-742f-4377-85e9-fec4b6a6442a',
|
||||||
|
name: 'Lee Robinson',
|
||||||
|
email: 'lee@robinson.com',
|
||||||
|
image_url: '/customers/lee-robinson.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '3958dc9e-737f-4377-85e9-fec4b6a6442a',
|
||||||
|
name: 'Hector Simpson',
|
||||||
|
email: 'hector@simpson.com',
|
||||||
|
image_url: '/customers/hector-simpson.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '50ca3e18-62cd-11ee-8c99-0242ac120002',
|
||||||
|
name: 'Steven Tey',
|
||||||
|
email: 'steven@tey.com',
|
||||||
|
image_url: '/customers/steven-tey.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '3958dc9e-787f-4377-85e9-fec4b6a6442a',
|
||||||
|
name: 'Steph Dietz',
|
||||||
|
email: 'steph@dietz.com',
|
||||||
|
image_url: '/customers/steph-dietz.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '76d65c26-f784-44a2-ac19-586678f7c2f2',
|
||||||
|
name: 'Michael Novotny',
|
||||||
|
email: 'michael@novotny.com',
|
||||||
|
image_url: '/customers/michael-novotny.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'd6e15727-9fe1-4961-8c5b-ea44a9bd81aa',
|
||||||
|
name: 'Evil Rabbit',
|
||||||
|
email: 'evil@rabbit.com',
|
||||||
|
image_url: '/customers/evil-rabbit.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '126eed9c-c90c-4ef6-a4a8-fcf7408d3c66',
|
||||||
|
name: 'Emil Kowalski',
|
||||||
|
email: 'emil@kowalski.com',
|
||||||
|
image_url: '/customers/emil-kowalski.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'CC27C14A-0ACF-4F4A-A6C9-D45682C144B9',
|
||||||
|
name: 'Amy Burns',
|
||||||
|
email: 'amy@burns.com',
|
||||||
|
image_url: '/customers/amy-burns.png',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: '13D07535-C59E-4157-A011-F8D2EF4E0CBB',
|
||||||
|
name: 'Balazs Orban',
|
||||||
|
email: 'balazs@orban.com',
|
||||||
|
image_url: '/customers/balazs-orban.png',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const invoices = [
|
||||||
|
{
|
||||||
|
customer_id: customers[0].id,
|
||||||
|
amount: 15795,
|
||||||
|
status: 'pending',
|
||||||
|
date: '2022-12-06',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[1].id,
|
||||||
|
amount: 20348,
|
||||||
|
status: 'pending',
|
||||||
|
date: '2022-11-14',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[4].id,
|
||||||
|
amount: 3040,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2022-10-29',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[3].id,
|
||||||
|
amount: 44800,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-09-10',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[5].id,
|
||||||
|
amount: 34577,
|
||||||
|
status: 'pending',
|
||||||
|
date: '2023-08-05',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[7].id,
|
||||||
|
amount: 54246,
|
||||||
|
status: 'pending',
|
||||||
|
date: '2023-07-16',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[6].id,
|
||||||
|
amount: 666,
|
||||||
|
status: 'pending',
|
||||||
|
date: '2023-06-27',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[3].id,
|
||||||
|
amount: 32545,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-06-09',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[4].id,
|
||||||
|
amount: 1250,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-06-17',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[5].id,
|
||||||
|
amount: 8546,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-06-07',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[1].id,
|
||||||
|
amount: 500,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-08-19',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[5].id,
|
||||||
|
amount: 8945,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-06-03',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[2].id,
|
||||||
|
amount: 8945,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-06-18',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[0].id,
|
||||||
|
amount: 8945,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2023-10-04',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
customer_id: customers[2].id,
|
||||||
|
amount: 1000,
|
||||||
|
status: 'paid',
|
||||||
|
date: '2022-06-05',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const revenue = [
|
||||||
|
{ month: 'Jan', revenue: 2000 },
|
||||||
|
{ month: 'Feb', revenue: 1800 },
|
||||||
|
{ month: 'Mar', revenue: 2200 },
|
||||||
|
{ month: 'Apr', revenue: 2500 },
|
||||||
|
{ month: 'May', revenue: 2300 },
|
||||||
|
{ month: 'Jun', revenue: 3200 },
|
||||||
|
{ month: 'Jul', revenue: 3500 },
|
||||||
|
{ month: 'Aug', revenue: 3700 },
|
||||||
|
{ month: 'Sep', revenue: 2500 },
|
||||||
|
{ month: 'Oct', revenue: 2800 },
|
||||||
|
{ month: 'Nov', revenue: 3000 },
|
||||||
|
{ month: 'Dec', revenue: 4800 },
|
||||||
|
];
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
users,
|
||||||
|
customers,
|
||||||
|
invoices,
|
||||||
|
revenue,
|
||||||
|
};
|
69
src/app/lib/utils.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import { Revenue } from './definitions';
|
||||||
|
|
||||||
|
export const formatCurrency = (amount: number) => {
|
||||||
|
return (amount / 100).toLocaleString('en-US', {
|
||||||
|
style: 'currency',
|
||||||
|
currency: 'USD',
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const formatDateToLocal = (
|
||||||
|
dateStr: string,
|
||||||
|
locale: string = 'en-US',
|
||||||
|
) => {
|
||||||
|
const date = new Date(dateStr);
|
||||||
|
const options: Intl.DateTimeFormatOptions = {
|
||||||
|
day: 'numeric',
|
||||||
|
month: 'short',
|
||||||
|
year: 'numeric',
|
||||||
|
};
|
||||||
|
const formatter = new Intl.DateTimeFormat(locale, options);
|
||||||
|
return formatter.format(date);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const generateYAxis = (revenue: Revenue[]) => {
|
||||||
|
// Calculate what labels we need to display on the y-axis
|
||||||
|
// based on highest record and in 1000s
|
||||||
|
const yAxisLabels = [];
|
||||||
|
const highestRecord = Math.max(...revenue.map((month) => month.revenue));
|
||||||
|
const topLabel = Math.ceil(highestRecord / 1000) * 1000;
|
||||||
|
|
||||||
|
for (let i = topLabel; i >= 0; i -= 1000) {
|
||||||
|
yAxisLabels.push(`$${i / 1000}K`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { yAxisLabels, topLabel };
|
||||||
|
};
|
||||||
|
|
||||||
|
export const generatePagination = (currentPage: number, totalPages: number) => {
|
||||||
|
// If the total number of pages is 7 or less,
|
||||||
|
// display all pages without any ellipsis.
|
||||||
|
if (totalPages <= 7) {
|
||||||
|
return Array.from({ length: totalPages }, (_, i) => i + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the current page is among the first 3 pages,
|
||||||
|
// show the first 3, an ellipsis, and the last 2 pages.
|
||||||
|
if (currentPage <= 3) {
|
||||||
|
return [1, 2, 3, '...', totalPages - 1, totalPages];
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the current page is among the last 3 pages,
|
||||||
|
// show the first 2, an ellipsis, and the last 3 pages.
|
||||||
|
if (currentPage >= totalPages - 2) {
|
||||||
|
return [1, 2, '...', totalPages - 2, totalPages - 1, totalPages];
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the current page is somewhere in the middle,
|
||||||
|
// show the first page, an ellipsis, the current page and its neighbors,
|
||||||
|
// another ellipsis, and the last page.
|
||||||
|
return [
|
||||||
|
1,
|
||||||
|
'...',
|
||||||
|
currentPage - 1,
|
||||||
|
currentPage,
|
||||||
|
currentPage + 1,
|
||||||
|
'...',
|
||||||
|
totalPages,
|
||||||
|
];
|
||||||
|
};
|
13
src/bootstrap/db/db.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import postgres from "postgres";
|
||||||
|
|
||||||
|
|
||||||
|
const envs = process.env;
|
||||||
|
const dbConfigs = {
|
||||||
|
host: envs.POSTGRES_HOST,
|
||||||
|
port: Number(envs.POSTGRES_PORT),
|
||||||
|
username: envs.POSTGRES_USER,
|
||||||
|
password: envs.POSTGRES_PASS,
|
||||||
|
database: envs.POSTGRES_DB,
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sql = postgres(dbConfigs);
|
10
yarn.lock
@ -46,6 +46,11 @@
|
|||||||
resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.1.tgz#de633db3ec2ef6a3c89e2f19038063e8a122e2c2"
|
resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.1.tgz#de633db3ec2ef6a3c89e2f19038063e8a122e2c2"
|
||||||
integrity sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==
|
integrity sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==
|
||||||
|
|
||||||
|
"@heroicons/react@^2.1.5":
|
||||||
|
version "2.1.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/@heroicons/react/-/react-2.1.5.tgz#1e13f34976cc542deae92353c01c8b3d7942e9ba"
|
||||||
|
integrity sha512-FuzFN+BsHa+7OxbvAERtgBTNeZpUjgM/MIizfVkSCL2/edriN0Hx/DWRCR//aPYwO5QX/YlgLGXk+E3PcfZwjA==
|
||||||
|
|
||||||
"@humanwhocodes/config-array@^0.13.0":
|
"@humanwhocodes/config-array@^0.13.0":
|
||||||
version "0.13.0"
|
version "0.13.0"
|
||||||
resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.13.0.tgz#fb907624df3256d04b9aa2df50d7aa97ec648748"
|
resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.13.0.tgz#fb907624df3256d04b9aa2df50d7aa97ec648748"
|
||||||
@ -3161,3 +3166,8 @@ yocto-queue@^0.1.0:
|
|||||||
version "0.1.0"
|
version "0.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
|
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
|
||||||
integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
|
integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
|
||||||
|
|
||||||
|
zod@^3.23.8:
|
||||||
|
version "3.23.8"
|
||||||
|
resolved "https://registry.yarnpkg.com/zod/-/zod-3.23.8.tgz#e37b957b5d52079769fb8097099b592f0ef4067d"
|
||||||
|
integrity sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==
|
||||||
|