diff --git a/api.saladeaula.digital/app/routes/orders/checkout.py b/api.saladeaula.digital/app/routes/orders/checkout.py index 852cfcb..398d88e 100644 --- a/api.saladeaula.digital/app/routes/orders/checkout.py +++ b/api.saladeaula.digital/app/routes/orders/checkout.py @@ -328,7 +328,7 @@ def _get_due_days( return dyn.collection.get_item( KeyPair( pk=str(org_id), - sk=SortKey('METADATA#PAYMENT_POLICY', path_spec='due_days'), + sk=SortKey('METADATA#BILLING', path_spec='due_days'), ), raise_on_error=False, default=default, diff --git a/api.saladeaula.digital/app/routes/orgs/__init__.py b/api.saladeaula.digital/app/routes/orgs/__init__.py index 9044b90..395c57f 100644 --- a/api.saladeaula.digital/app/routes/orgs/__init__.py +++ b/api.saladeaula.digital/app/routes/orgs/__init__.py @@ -43,6 +43,7 @@ def get_org(org_id: str): TransactKey(org_id) + SortKey('0') + SortKey('METADATA#ADDRESS', rename_key='address') + + SortKey('METADATA#BILLING', rename_key='billing') + SortKey('METADATA#SUBSCRIPTION', rename_key='subscription') + KeyPair( pk='SUBSCRIPTION#FROZEN', diff --git a/api.saladeaula.digital/app/routes/orgs/billing.py b/api.saladeaula.digital/app/routes/orgs/billing.py index 56643d8..eb2f48b 100644 --- a/api.saladeaula.digital/app/routes/orgs/billing.py +++ b/api.saladeaula.digital/app/routes/orgs/billing.py @@ -1,13 +1,19 @@ from datetime import date +from decimal import Decimal +from enum import Enum +from http import HTTPStatus from typing import Annotated from aws_lambda_powertools.event_handler.api_gateway import Router from aws_lambda_powertools.event_handler.exceptions import NotFoundError -from aws_lambda_powertools.event_handler.openapi.params import Query +from aws_lambda_powertools.event_handler.openapi.params import Body, Query +from layercake.dateutils import now from layercake.dynamodb import DynamoDBPersistenceLayer, KeyPair +from api_gateway import JSONResponse from boto3clients import dynamodb_client -from config import ORDER_TABLE +from config import ORDER_TABLE, USER_TABLE +from exceptions import OrgNotFoundError router = Router() dyn = DynamoDBPersistenceLayer(ORDER_TABLE, dynamodb_client) @@ -16,6 +22,11 @@ dyn = DynamoDBPersistenceLayer(ORDER_TABLE, dynamodb_client) class BillingNotFoundError(NotFoundError): ... +class PaymentMethod(str, Enum): + BANK_SLIP = 'BANK_SLIP' + MANUAL = 'MANUAL' + + @router.get('//billing') def billing( org_id: str, @@ -36,3 +47,30 @@ def billing( ), limit=150, ) + + +@router.put('//billing') +def update( + org_id: str, + due_days: Annotated[Decimal, Body(embed=True, ge=1, le=90)], + payment_method: Annotated[PaymentMethod, Body(embed=True)], +): + with dyn.transact_writer() as transact: + transact.condition( + key=KeyPair(org_id, '0'), + cond_expr='attribute_exists(sk)', + exc_cls=OrgNotFoundError, + table_name=USER_TABLE, + ) + transact.put( + item={ + 'id': org_id, + 'sk': 'METADATA#BILLING', + 'due_days': due_days, + 'payment_method': payment_method.value, + 'created_at': now(), + }, + table_name=USER_TABLE, + ) + + return JSONResponse(status_code=HTTPStatus.NO_CONTENT) diff --git a/api.saladeaula.digital/app/routes/orgs/subscription.py b/api.saladeaula.digital/app/routes/orgs/subscription.py index 6ca360b..2db95a4 100644 --- a/api.saladeaula.digital/app/routes/orgs/subscription.py +++ b/api.saladeaula.digital/app/routes/orgs/subscription.py @@ -1,4 +1,3 @@ -from enum import Enum from http import HTTPStatus from typing import Annotated @@ -20,17 +19,11 @@ router = Router() dyn = DynamoDBPersistenceLayer(USER_TABLE, dynamodb_client) -class PaymentMethod(str, Enum): - BANK_SLIP = 'BANK_SLIP' - MANUAL = 'MANUAL' - - @router.post('//subscription') def add( org_id: str, name: Annotated[str, Body(embed=True)], billing_day: Annotated[int, Body(embed=True, ge=1, le=31)], - payment_method: Annotated[PaymentMethod, Body(embed=True)], subscription_frozen: Annotated[bool, Body(embed=True)] = False, ): now_ = now() @@ -55,7 +48,6 @@ def add( 'id': org_id, 'sk': 'METADATA#SUBSCRIPTION', 'billing_day': billing_day, - 'payment_method': payment_method.value, 'created_at': now_, }, cond_expr='attribute_not_exists(sk)', @@ -88,7 +80,6 @@ def add( def edit( org_id: str, billing_day: Annotated[int, Body(embed=True, ge=1, le=31)], - payment_method: Annotated[PaymentMethod, Body(embed=True)], subscription_frozen: Annotated[bool, Body(embed=True)] = False, ): now_ = now() @@ -102,11 +93,9 @@ def edit( transact.update( key=KeyPair(org_id, 'METADATA#SUBSCRIPTION'), update_expr='SET billing_day = :billing_day, \ - payment_method = :payment_method, \ updated_at = :now', expr_attr_values={ ':billing_day': billing_day, - ':payment_method': payment_method.value, ':now': now_, }, cond_expr='attribute_exists(sk)', diff --git a/apps/admin.saladeaula.digital/app/conf.ts b/apps/admin.saladeaula.digital/app/conf.ts index 03cbf50..9855ed5 100644 --- a/apps/admin.saladeaula.digital/app/conf.ts +++ b/apps/admin.saladeaula.digital/app/conf.ts @@ -1,2 +1,3 @@ export const TZ = 'America/Sao_Paulo' export const INTERNAL_EMAIL_DOMAIN = 'users.noreply.saladeaula.digital' +export const RYBBIT_SITE_ID = '83748b35413d' diff --git a/apps/admin.saladeaula.digital/app/root.tsx b/apps/admin.saladeaula.digital/app/root.tsx index a10f369..3da0edf 100644 --- a/apps/admin.saladeaula.digital/app/root.tsx +++ b/apps/admin.saladeaula.digital/app/root.tsx @@ -12,12 +12,11 @@ import { import { loggingMiddleware } from '@repo/auth/middleware/logging' import { ThemeProvider } from '@repo/ui/components/theme-provider' import './app.css' +import { RYBBIT_SITE_ID } from './conf' export const middleware: Route.MiddlewareFunction[] = [loggingMiddleware] export function Layout({ children }: { children: React.ReactNode }) { - const rybbitSiteId = '83748b35413d' - return ( @@ -30,7 +29,7 @@ export function Layout({ children }: { children: React.ReactNode }) { {/* Rybbit Tracking Snippet */} diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id._index/route.tsx b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id._index/route.tsx index 7e9fee4..6c34d68 100644 --- a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id._index/route.tsx +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id._index/route.tsx @@ -41,7 +41,8 @@ export default function Route({}: Route.ComponentProps) { Editar empresa - Configurar as informações gerais para esta empresa. + Gerencie as informações cadastrais e configurações gerais desta + empresa. diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.address/route.tsx b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.address/route.tsx index 0bb2a26..a11d115 100644 --- a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.address/route.tsx +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.address/route.tsx @@ -36,8 +36,7 @@ export default function Route({}: Route.ComponentProps) { Editar endereço - Este endereço será usado automaticamente sempre que for necessário - informar um endereço. + Este endereço será usado automaticamente sempre que necessário. diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.billing/route.tsx b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.billing/route.tsx new file mode 100644 index 0000000..191e05a --- /dev/null +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.billing/route.tsx @@ -0,0 +1,161 @@ +import type { Route } from './+types/route' + +import { zodResolver } from '@hookform/resolvers/zod' +import { useForm } from 'react-hook-form' +import { toast } from 'sonner' +import { z } from 'zod' + +import { Button } from '@repo/ui/components/ui/button' +import { + Card, + CardContent, + CardDescription, + CardHeader, + CardTitle +} from '@repo/ui/components/ui/card' +import { FieldSet } from '@repo/ui/components/ui/field' +import { + Form, + FormControl, + FormField, + FormItem, + FormLabel, + FormMessage +} from '@repo/ui/components/ui/form' +import { Input } from '@repo/ui/components/ui/input' +import { + NativeSelect, + NativeSelectOption +} from '@repo/ui/components/ui/native-select' +import { Spinner } from '@repo/ui/components/ui/spinner' +import { HttpMethod, request as req } from '@repo/util/request' +import { useFetcher, useOutletContext } from 'react-router' +import type { Org } from '../_app.orgs.$id/data' + +const formSchema = z.object({ + due_days: z + .number({ error: 'Deve estar entre 1 e 90' }) + .min(1, { error: 'Deve ser igual 1 ou maior' }) + .max(90, { error: 'Deve ser menor ou igual a 90' }), + payment_method: z.enum(['BANK_SLIP', 'MANUAL'], { + error: 'Selecione uma opção' + }) +}) + +type Schema = z.infer + +export async function action({ params, request, context }: Route.ActionArgs) { + const r = await req({ + url: `orgs/${params.id}/billing`, + headers: new Headers({ 'Content-Type': 'application/json' }), + method: HttpMethod.PUT, + body: JSON.stringify(await request.json()), + request, + context + }) + + return { ok: r.ok } +} + +export default function Route({}: Route.ComponentProps) { + const fetcher = useFetcher() + const { org } = useOutletContext() as { org: Org } + const form = useForm({ + resolver: zodResolver(formSchema), + defaultValues: org?.billing + }) + const { handleSubmit, control, formState } = form + const onSubmit = async (data: Schema) => { + toast.success('As condições de pagamento foram atualizadas') + + fetcher.submit(JSON.stringify(data), { + method: 'PUT', + encType: 'application/json' + }) + } + + return ( +
+ + + + + Condições de pagamento + + + Estas condições serão usadas automaticamente sempre que um + pagamento for gerado. + + + +
+ ( + + + Prazo para vencimento{' '} + + (em dias) + + + + onChange(Number(e.target.value))} + {...field} + /> + + + + )} + /> + + ( + + Forma de pagamento + + + + Selecione + + + Boleto bancário + + + Depósito bancário + + + Pix + + + Cartão de crédito + + + + + + )} + /> +
+ + +
+
+
+ + ) +} diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.subscription/route.tsx b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.subscription/route.tsx index 37e637b..7bda588 100644 --- a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.subscription/route.tsx +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id.subscription/route.tsx @@ -26,10 +26,6 @@ import { } from '@repo/ui/components/ui/form' import { Input } from '@repo/ui/components/ui/input' import { Label } from '@repo/ui/components/ui/label' -import { - NativeSelect, - NativeSelectOption -} from '@repo/ui/components/ui/native-select' import { RadioGroup, RadioGroupItem } from '@repo/ui/components/ui/radio-group' import { Spinner } from '@repo/ui/components/ui/spinner' import { HttpMethod, request as req } from '@repo/util/request' @@ -42,9 +38,6 @@ const formSchema = z.object({ .number({ error: 'Deve estar entre 1 e 31' }) .min(1, { error: 'Deve ser igual 1 ou maior' }) .max(31, { error: 'Deve ser menor ou igual a 31' }), - payment_method: z.enum(['BANK_SLIP', 'MANUAL'], { - error: 'Selecione uma opção' - }), subscription_frozen: z.boolean().optional() }) @@ -88,7 +81,7 @@ export default function Route({}: Route.ComponentProps) { }, resolver: zodResolver(formSchema) }) - const { handleSubmit, formState, watch, reset } = form + const { handleSubmit, formState, watch, reset, control } = form const plan = watch('plan') const onSubmit = async ({ plan, ...data }: Schema) => { @@ -129,7 +122,7 @@ export default function Route({}: Route.ComponentProps) { ( @@ -166,7 +159,7 @@ export default function Route({}: Route.ComponentProps) { ( @@ -186,31 +179,7 @@ export default function Route({}: Route.ComponentProps) { /> ( - - Forma de pagamento - - - - Selecione - - - Boleto bancário - - - Depósito bancário - - - - - - )} - /> - - ( diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/data.ts b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/data.ts index 4377bfb..046329b 100644 --- a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/data.ts +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/data.ts @@ -2,7 +2,11 @@ import type { Org as Org_ } from '../_app.orgs._index/columns' export type Subscription = { billing_day: number - payment_method: 'BANK_SLIP' | 'MANUAL' +} + +export type Billing = { + due_days: number + payment_methot: 'BANK_SLIP' | 'MANUAL' } export type Address = { @@ -14,4 +18,5 @@ export type Org = Org_ & { address?: Address subscription?: Subscription subscription_frozen?: boolean + billing: Billing } diff --git a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/route.tsx b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/route.tsx index a1b05b4..bf5b044 100644 --- a/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/route.tsx +++ b/apps/insights.saladeaula.digital/app/routes/_app.orgs.$id/route.tsx @@ -26,7 +26,8 @@ import type { Org } from './data' const links = [ { to: '', title: 'Perfil', end: true }, { to: 'address', title: 'Endereço' }, - { to: 'subscription', title: 'Plano' } + { to: 'subscription', title: 'Plano' }, + { to: 'billing', title: 'Pagamentos' } ] export function meta() { diff --git a/apps/studio.saladeaula.digital/app/routes/edit.tsx b/apps/studio.saladeaula.digital/app/routes/edit.tsx index 30960a4..b110ac5 100644 --- a/apps/studio.saladeaula.digital/app/routes/edit.tsx +++ b/apps/studio.saladeaula.digital/app/routes/edit.tsx @@ -388,7 +388,7 @@ function Editing() { {...field} /> - Não listar no catálogo de cursos + Ocultar curso do catálogo público )} />