diff --git a/apps/remix/.bin/stripe-dev.sh b/apps/remix/.bin/stripe-dev.sh index 67d349ded..fc8ac7485 100755 --- a/apps/remix/.bin/stripe-dev.sh +++ b/apps/remix/.bin/stripe-dev.sh @@ -73,5 +73,12 @@ if [ -z "$NEXT_PRIVATE_STRIPE_WEBHOOK_SECRET" ]; then echo "╚═════════════════════════════════════════════════════════════════════╝" fi +NEXT_PUBLIC_WEBAPP_URL=$(load_env_var "NEXT_PUBLIC_WEBAPP_URL") + +if [ -z "$NEXT_PUBLIC_WEBAPP_URL" ]; then + NEXT_PUBLIC_WEBAPP_URL="http://localhost:3000" + echo "[INFO]: NEXT_PUBLIC_WEBAPP_URL not set, defaulting to $NEXT_PUBLIC_WEBAPP_URL" +fi + echo "[INFO]: Starting Stripe webhook listener..." -stripe listen --api-key "$NEXT_PRIVATE_STRIPE_API_KEY" --forward-to http://localhost:3000/api/stripe/webhook +stripe listen --api-key "$NEXT_PRIVATE_STRIPE_API_KEY" --forward-to "$NEXT_PUBLIC_WEBAPP_URL/api/stripe/webhook" diff --git a/apps/remix/app/components/dialogs/admin-organisation-sync-subscription-dialog.tsx b/apps/remix/app/components/dialogs/admin-organisation-sync-subscription-dialog.tsx new file mode 100644 index 000000000..57f3c3cb0 --- /dev/null +++ b/apps/remix/app/components/dialogs/admin-organisation-sync-subscription-dialog.tsx @@ -0,0 +1,155 @@ +import { AppError } from '@documenso/lib/errors/app-error'; +import { trpc } from '@documenso/trpc/react'; +import { Button } from '@documenso/ui/primitives/button'; +import { Checkbox } from '@documenso/ui/primitives/checkbox'; +import { + Dialog, + DialogContent, + DialogDescription, + DialogFooter, + DialogHeader, + DialogTitle, + DialogTrigger, +} from '@documenso/ui/primitives/dialog'; +import { Form, FormControl, FormField, FormItem } from '@documenso/ui/primitives/form/form'; +import { useToast } from '@documenso/ui/primitives/use-toast'; +import { zodResolver } from '@hookform/resolvers/zod'; +import { Trans, useLingui } from '@lingui/react/macro'; +import { useEffect, useState } from 'react'; +import { useForm } from 'react-hook-form'; +import { useNavigate } from 'react-router'; +import { z } from 'zod'; + +export type AdminOrganisationSyncSubscriptionDialogProps = { + organisationId: string; + trigger?: React.ReactNode; +}; + +const ZAdminOrganisationSyncSubscriptionFormSchema = z.object({ + syncClaims: z.boolean(), +}); + +type TAdminOrganisationSyncSubscriptionFormSchema = z.infer; + +export const AdminOrganisationSyncSubscriptionDialog = ({ + organisationId, + trigger, +}: AdminOrganisationSyncSubscriptionDialogProps) => { + const [open, setOpen] = useState(false); + + const { t } = useLingui(); + const { toast } = useToast(); + + const navigate = useNavigate(); + + const form = useForm({ + resolver: zodResolver(ZAdminOrganisationSyncSubscriptionFormSchema), + defaultValues: { + syncClaims: false, + }, + }); + + const { mutateAsync: syncSubscription } = trpc.admin.organisation.subscription.sync.useMutation(); + + const onFormSubmit = async (values: TAdminOrganisationSyncSubscriptionFormSchema) => { + try { + await syncSubscription({ + organisationId, + syncClaims: values.syncClaims, + }); + + toast({ + title: t`Subscription synced`, + description: t`The organisation subscription has been synced with Stripe.`, + duration: 5000, + }); + + await navigate(0); + + setOpen(false); + } catch (err) { + const error = AppError.parseError(err); + + console.error(error); + + toast({ + title: t`Failed to sync subscription`, + description: error.message, + variant: 'destructive', + duration: 10000, + }); + } + }; + + useEffect(() => { + if (!open) { + form.reset(); + } + }, [open, form]); + + return ( + !form.formState.isSubmitting && setOpen(value)}> + + {trigger ?? ( + + )} + + + + + + Sync Stripe subscription + + + + Fetch the latest subscription data from Stripe and apply it to this organisation. + + + +
+ +
+ ( + + + + + + + + )} + /> + + + + + + +
+
+ +
+
+ ); +}; diff --git a/apps/remix/app/components/dialogs/admin-swap-subscription-dialog.tsx b/apps/remix/app/components/dialogs/admin-swap-subscription-dialog.tsx index 72c0b3326..90489bf8f 100644 --- a/apps/remix/app/components/dialogs/admin-swap-subscription-dialog.tsx +++ b/apps/remix/app/components/dialogs/admin-swap-subscription-dialog.tsx @@ -67,7 +67,7 @@ export const AdminSwapSubscriptionDialog = ({ const selectedOrg = eligibleOrgs.find((org) => org.id === selectedOrgId); - const { mutateAsync: swapSubscription } = trpc.admin.organisation.swapSubscription.useMutation(); + const { mutateAsync: swapSubscription } = trpc.admin.organisation.subscription.swap.useMutation(); const onSubmit = async () => { if (!selectedOrgId) { diff --git a/apps/remix/app/components/general/organisation-usage-reset-button.tsx b/apps/remix/app/components/general/organisation-usage-reset-button.tsx index 73340b122..7451f6e1c 100644 --- a/apps/remix/app/components/general/organisation-usage-reset-button.tsx +++ b/apps/remix/app/components/general/organisation-usage-reset-button.tsx @@ -14,7 +14,7 @@ export const OrganisationUsageResetButton = ({ organisationId, counter }: Organi const { toast } = useToast(); const { revalidate } = useRevalidator(); - const { mutateAsync: reset, isPending } = trpc.admin.organisation.resetMonthlyStat.useMutation({ + const { mutateAsync: reset, isPending } = trpc.admin.organisation.stats.reset.useMutation({ onSuccess: async () => { toast({ title: t`Counter reset.` }); await revalidate(); diff --git a/apps/remix/app/components/tables/admin-organisation-stats-table.tsx b/apps/remix/app/components/tables/admin-organisation-stats-table.tsx index 97275e1c1..0e8bf7003 100644 --- a/apps/remix/app/components/tables/admin-organisation-stats-table.tsx +++ b/apps/remix/app/components/tables/admin-organisation-stats-table.tsx @@ -41,7 +41,7 @@ export const AdminOrganisationStatsTable = () => { const orderByColumn = parseOrderByColumn(searchParams?.get('orderByColumn') ?? undefined); const orderByDirection = parseOrderByDirection(searchParams?.get('orderByDirection') ?? undefined); - const { data, isLoading, isLoadingError } = trpc.admin.organisation.findStats.useQuery({ + const { data, isLoading, isLoadingError } = trpc.admin.organisation.stats.find.useQuery({ query: parsedSearchParams.query, page: parsedSearchParams.page, perPage: parsedSearchParams.perPage, diff --git a/apps/remix/app/routes/_authenticated+/admin+/organisations.$id.tsx b/apps/remix/app/routes/_authenticated+/admin+/organisations.$id.tsx index a0c5cf37a..9824390d1 100644 --- a/apps/remix/app/routes/_authenticated+/admin+/organisations.$id.tsx +++ b/apps/remix/app/routes/_authenticated+/admin+/organisations.$id.tsx @@ -40,6 +40,7 @@ import type { z } from 'zod'; import { AdminOrganisationDeleteDialog } from '~/components/dialogs/admin-organisation-delete-dialog'; import { AdminOrganisationMemberDeleteDialog } from '~/components/dialogs/admin-organisation-member-delete-dialog'; import { AdminOrganisationMemberUpdateDialog } from '~/components/dialogs/admin-organisation-member-update-dialog'; +import { AdminOrganisationSyncSubscriptionDialog } from '~/components/dialogs/admin-organisation-sync-subscription-dialog'; import { DetailsCard, DetailsValue } from '~/components/general/admin-details'; import { AdminGlobalSettingsSection } from '~/components/general/admin-global-settings-section'; import { ClaimLimitFields } from '~/components/general/claim-limit-fields'; @@ -377,7 +378,16 @@ export default function OrganisationGroupSettingsPage({ params, loaderData }: Ro )} {organisation.subscription && ( -
+
+ + Sync Stripe subscription + + } + /> +