diff --git a/.gitignore b/.gitignore index eac6aedb..862c9e2c 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,6 @@ .env +.env.dev +.env.prod data # compiled output /dist diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 00000000..2bdf0e03 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,3 @@ +[submodule "apps/server/src/ee"] + path = apps/server/src/ee + url = https://github.com/docmost/ee diff --git a/apps/client/package.json b/apps/client/package.json index cd1248c7..2a0ec761 100644 --- a/apps/client/package.json +++ b/apps/client/package.json @@ -16,12 +16,12 @@ "@emoji-mart/data": "^1.2.1", "@emoji-mart/react": "^1.1.1", "@excalidraw/excalidraw": "^0.17.6", - "@mantine/core": "^7.14.2", - "@mantine/form": "^7.14.2", - "@mantine/hooks": "^7.14.2", - "@mantine/modals": "^7.14.2", - "@mantine/notifications": "^7.14.2", - "@mantine/spotlight": "^7.14.2", + "@mantine/core": "^7.17.0", + "@mantine/form": "^7.17.0", + "@mantine/hooks": "^7.17.0", + "@mantine/modals": "^7.17.0", + "@mantine/notifications": "^7.17.0", + "@mantine/spotlight": "^7.17.0", "@tabler/icons-react": "^3.22.0", "@tanstack/react-query": "^5.61.4", "axios": "^1.7.9", @@ -30,7 +30,7 @@ "file-saver": "^2.0.5", "i18next": "^23.14.0", "i18next-http-backend": "^2.6.1", - "jotai": "^2.10.3", + "jotai": "^2.12.1", "jotai-optics": "^0.4.0", "js-cookie": "^3.0.5", "katex": "0.16.21", diff --git a/apps/client/src/App.tsx b/apps/client/src/App.tsx index 95389e06..c806f852 100644 --- a/apps/client/src/App.tsx +++ b/apps/client/src/App.tsx @@ -18,10 +18,18 @@ import { ErrorBoundary } from "react-error-boundary"; import InviteSignup from "@/pages/auth/invite-signup.tsx"; import ForgotPassword from "@/pages/auth/forgot-password.tsx"; import PasswordReset from "./pages/auth/password-reset"; +import Billing from "@/ee/billing/pages/billing.tsx"; +import CloudLogin from "@/ee/pages/cloud-login.tsx"; +import CreateWorkspace from "@/ee/pages/create-workspace.tsx"; +import { isCloud } from "@/lib/config.ts"; import { useTranslation } from "react-i18next"; +import Security from "@/ee/security/pages/security.tsx"; +import License from "@/ee/licence/pages/license.tsx"; +import { useRedirectToCloudSelect } from "@/ee/hooks/use-redirect-to-cloud-select.tsx"; export default function App() { const { t } = useTranslation(); + useRedirectToCloudSelect(); return ( <> @@ -29,15 +37,24 @@ export default function App() { } /> } /> } /> - } /> } /> } /> + {!isCloud() && ( + } /> + )} + + {isCloud() && ( + <> + } /> + } /> + + )} + } /> }> } /> - } /> } /> } /> } /> + } /> + {!isCloud() && } />} + {isCloud() && } />} diff --git a/apps/client/src/components/common/copy.tsx b/apps/client/src/components/common/copy.tsx new file mode 100644 index 00000000..efae5750 --- /dev/null +++ b/apps/client/src/components/common/copy.tsx @@ -0,0 +1,31 @@ +import { ActionIcon, CopyButton, Tooltip } from "@mantine/core"; +import { IconCheck, IconCopy } from "@tabler/icons-react"; +import React from "react"; +import { useTranslation } from "react-i18next"; + +interface CopyProps { + text: string; +} +export default function CopyTextButton({ text }: CopyProps) { + const { t } = useTranslation(); + + return ( + + {({ copied, copy }) => ( + + + {copied ? : } + + + )} + + ); +} diff --git a/apps/client/src/components/icons/google-icon.tsx b/apps/client/src/components/icons/google-icon.tsx new file mode 100644 index 00000000..e30be880 --- /dev/null +++ b/apps/client/src/components/icons/google-icon.tsx @@ -0,0 +1,33 @@ +import { rem } from "@mantine/core"; + +interface Props { + size?: number | string; +} + +export function GoogleIcon({ size }: Props) { + return ( + + + + + + + ); +} diff --git a/apps/client/src/components/icons/openid-icon.tsx b/apps/client/src/components/icons/openid-icon.tsx new file mode 100644 index 00000000..3252d47c --- /dev/null +++ b/apps/client/src/components/icons/openid-icon.tsx @@ -0,0 +1,20 @@ +import { rem } from "@mantine/core"; + +interface Props { + size?: number | string; +} + +export function OpenIdIcon({ size }: Props) { + return ( + + + + ); +} diff --git a/apps/client/src/components/layouts/global/app-header.tsx b/apps/client/src/components/layouts/global/app-header.tsx index 04b2b09d..30d03b53 100644 --- a/apps/client/src/components/layouts/global/app-header.tsx +++ b/apps/client/src/components/layouts/global/app-header.tsx @@ -1,19 +1,21 @@ -import {Group, Text, Tooltip} from "@mantine/core"; +import { Badge, Group, Text, Tooltip } from "@mantine/core"; import classes from "./app-header.module.css"; import React from "react"; import TopMenu from "@/components/layouts/global/top-menu.tsx"; -import {Link} from "react-router-dom"; +import { Link } from "react-router-dom"; import APP_ROUTE from "@/lib/app-route.ts"; -import {useAtom} from "jotai/index"; +import { useAtom } from "jotai"; import { desktopSidebarAtom, mobileSidebarAtom, } from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts"; -import {useToggleSidebar} from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts"; +import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts"; import SidebarToggle from "@/components/ui/sidebar-toggle-button.tsx"; import { useTranslation } from "react-i18next"; +import useTrial from "@/ee/hooks/use-trial.tsx"; +import { isCloud } from "@/lib/config.ts"; -const links = [{link: APP_ROUTE.HOME, label: "Home"}]; +const links = [{ link: APP_ROUTE.HOME, label: "Home" }]; export function AppHeader() { const { t } = useTranslation(); @@ -22,6 +24,7 @@ export function AppHeader() { const [desktopOpened] = useAtom(desktopSidebarAtom); const toggleDesktop = useToggleSidebar(desktopSidebarAtom); + const { isTrial, trialDaysLeft } = useTrial(); const isHomeRoute = location.pathname.startsWith("/home"); @@ -38,7 +41,6 @@ export function AppHeader() { {!isHomeRoute && ( <> - @@ -75,8 +77,21 @@ export function AppHeader() { - - + + {isCloud() && isTrial && trialDaysLeft !== 0 && ( + + {trialDaysLeft === 1 + ? "1 day left" + : `${trialDaysLeft} days left`} + + )} + diff --git a/apps/client/src/components/layouts/global/global-app-shell.tsx b/apps/client/src/components/layouts/global/global-app-shell.tsx index e5fe224b..4b5c0269 100644 --- a/apps/client/src/components/layouts/global/global-app-shell.tsx +++ b/apps/client/src/components/layouts/global/global-app-shell.tsx @@ -1,23 +1,26 @@ import { AppShell, Container } from "@mantine/core"; -import React, { useCallback, useEffect, useRef, useState } from "react"; +import React, { useEffect, useRef, useState } from "react"; import { useLocation } from "react-router-dom"; import SettingsSidebar from "@/components/settings/settings-sidebar.tsx"; import { useAtom } from "jotai"; import { asideStateAtom, desktopSidebarAtom, - mobileSidebarAtom, sidebarWidthAtom, + mobileSidebarAtom, + sidebarWidthAtom, } from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts"; import { SpaceSidebar } from "@/features/space/components/sidebar/space-sidebar.tsx"; import { AppHeader } from "@/components/layouts/global/app-header.tsx"; import Aside from "@/components/layouts/global/aside.tsx"; import classes from "./app-shell.module.css"; +import { useTrialEndAction } from "@/ee/hooks/use-trial-end-action.tsx"; export default function GlobalAppShell({ children, }: { children: React.ReactNode; }) { + useTrialEndAction(); const [mobileOpened] = useAtom(mobileSidebarAtom); const [desktopOpened] = useAtom(desktopSidebarAtom); const [{ isAsideOpen }] = useAtom(asideStateAtom); @@ -37,7 +40,9 @@ export default function GlobalAppShell({ const resize = React.useCallback( (mouseMoveEvent) => { if (isResizing) { - const newWidth = mouseMoveEvent.clientX - sidebarRef.current.getBoundingClientRect().left; + const newWidth = + mouseMoveEvent.clientX - + sidebarRef.current.getBoundingClientRect().left; if (newWidth < 220) { setSidebarWidth(220); return; @@ -49,7 +54,7 @@ export default function GlobalAppShell({ setSidebarWidth(newWidth); } }, - [isResizing] + [isResizing], ); useEffect(() => { @@ -94,7 +99,11 @@ export default function GlobalAppShell({ {!isHomeRoute && ( - +
{isSpaceRoute && } {isSettingsRoute && } diff --git a/apps/client/src/components/layouts/global/top-menu.tsx b/apps/client/src/components/layouts/global/top-menu.tsx index 28c64660..52cabb5a 100644 --- a/apps/client/src/components/layouts/global/top-menu.tsx +++ b/apps/client/src/components/layouts/global/top-menu.tsx @@ -33,13 +33,13 @@ export default function TopMenu() { - {workspace.name} + {workspace?.name} diff --git a/apps/client/src/components/settings/settings-queries.tsx b/apps/client/src/components/settings/settings-queries.tsx new file mode 100644 index 00000000..a37ecd8b --- /dev/null +++ b/apps/client/src/components/settings/settings-queries.tsx @@ -0,0 +1,51 @@ +import { queryClient } from "@/main.tsx"; +import { + getBilling, + getBillingPlans, +} from "@/ee/billing/services/billing-service.ts"; +import { getSpaces } from "@/features/space/services/space-service.ts"; +import { getGroups } from "@/features/group/services/group-service.ts"; +import { QueryParams } from "@/lib/types.ts"; +import { getWorkspaceMembers } from "@/features/workspace/services/workspace-service.ts"; +import { getLicenseInfo } from "@/ee/licence/services/license-service.ts"; + +export const prefetchWorkspaceMembers = () => { + const params = { limit: 100, page: 1, query: "" } as QueryParams; + queryClient.prefetchQuery({ + queryKey: ["workspaceMembers", params], + queryFn: () => getWorkspaceMembers(params), + }); +}; + +export const prefetchSpaces = () => { + queryClient.prefetchQuery({ + queryKey: ["spaces", { page: 1 }], + queryFn: () => getSpaces({ page: 1 }), + }); +}; + +export const prefetchGroups = () => { + queryClient.prefetchQuery({ + queryKey: ["groups", { page: 1 }], + queryFn: () => getGroups({ page: 1 }), + }); +}; + +export const prefetchBilling = () => { + queryClient.prefetchQuery({ + queryKey: ["billing"], + queryFn: () => getBilling(), + }); + + queryClient.prefetchQuery({ + queryKey: ["billing-plans"], + queryFn: () => getBillingPlans(), + }); +}; + +export const prefetchLicense = () => { + queryClient.prefetchQuery({ + queryKey: ["license"], + queryFn: () => getLicenseInfo(), + }); +}; diff --git a/apps/client/src/components/settings/settings-sidebar.tsx b/apps/client/src/components/settings/settings-sidebar.tsx index 8ffb7c4a..a6cf9474 100644 --- a/apps/client/src/components/settings/settings-sidebar.tsx +++ b/apps/client/src/components/settings/settings-sidebar.tsx @@ -1,5 +1,5 @@ import React, { useEffect, useState } from "react"; -import { Group, Text, ScrollArea, ActionIcon, rem } from "@mantine/core"; +import { Group, Text, ScrollArea, ActionIcon } from "@mantine/core"; import { IconUser, IconSettings, @@ -8,15 +8,33 @@ import { IconUsersGroup, IconSpaces, IconBrush, + IconCoin, + IconLock, + IconKey, } from "@tabler/icons-react"; import { Link, useLocation, useNavigate } from "react-router-dom"; import classes from "./settings.module.css"; import { useTranslation } from "react-i18next"; +import { isCloud } from "@/lib/config.ts"; +import useUserRole from "@/hooks/use-user-role.tsx"; +import { useAtom } from "jotai/index"; +import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts"; +import { + prefetchBilling, + prefetchGroups, + prefetchLicense, + prefetchSpaces, + prefetchWorkspaceMembers, +} from "@/components/settings/settings-queries.tsx"; interface DataItem { label: string; icon: React.ElementType; path: string; + isCloud?: boolean; + isEnterprise?: boolean; + isAdmin?: boolean; + isSelfhosted?: boolean; } interface DataGroup { @@ -45,10 +63,35 @@ const groupedData: DataGroup[] = [ icon: IconUsers, path: "/settings/members", }, + { + label: "Billing", + icon: IconCoin, + path: "/settings/billing", + isCloud: true, + isAdmin: true, + }, + { + label: "Security & SSO", + icon: IconLock, + path: "/settings/security", + isCloud: true, + isEnterprise: true, + isAdmin: true, + }, { label: "Groups", icon: IconUsersGroup, path: "/settings/groups" }, { label: "Spaces", icon: IconSpaces, path: "/settings/spaces" }, ], }, + { + heading: "System", + items: [ + { + label: "License & Edition", + icon: IconKey, + path: "/settings/license", + }, + ], + }, ]; export default function SettingsSidebar() { @@ -56,29 +99,92 @@ export default function SettingsSidebar() { const location = useLocation(); const [active, setActive] = useState(location.pathname); const navigate = useNavigate(); + const { isAdmin } = useUserRole(); + const [workspace] = useAtom(workspaceAtom); useEffect(() => { setActive(location.pathname); }, [location.pathname]); - const menuItems = groupedData.map((group) => ( -
- - {t(group.heading)} - - {group.items.map((item) => ( - - - {t(item.label)} - - ))} -
- )); + const canShowItem = (item: DataItem) => { + if (item.isCloud && item.isEnterprise) { + if (!(isCloud() || workspace?.hasLicenseKey)) return false; + return item.isAdmin ? isAdmin : true; + } + + if (item.isCloud) { + return isCloud() ? (item.isAdmin ? isAdmin : true) : false; + } + + if (item.isSelfhosted) { + return !isCloud() ? (item.isAdmin ? isAdmin : true) : false; + } + + if (item.isEnterprise) { + return workspace?.hasLicenseKey ? (item.isAdmin ? isAdmin : true) : false; + } + + if (item.isAdmin) { + return isAdmin; + } + + return true; + }; + + const menuItems = groupedData.map((group) => { + if (group.heading === "System" && (!isAdmin || isCloud())) { + return null; + } + + return ( +
+ + {t(group.heading)} + + {group.items.map((item) => { + if (!canShowItem(item)) { + return null; + } + + let prefetchHandler: any; + switch (item.label) { + case "Members": + prefetchHandler = prefetchWorkspaceMembers; + break; + case "Spaces": + prefetchHandler = prefetchSpaces; + break; + case "Groups": + prefetchHandler = prefetchGroups; + break; + case "Billing": + prefetchHandler = prefetchBilling; + break; + case "License & Edition": + if (workspace?.hasLicenseKey) { + prefetchHandler = prefetchLicense; + } + break; + default: + break; + } + + return ( + + + {t(item.label)} + + ); + })} +
+ ); + }); return (
@@ -95,9 +201,8 @@ export default function SettingsSidebar() { {menuItems} -
+
+ + {isCloud() && ( +
+ + help@docmost.com + +
+ )}
); } diff --git a/apps/client/src/components/settings/settings.module.css b/apps/client/src/components/settings/settings.module.css index 546ff351..48de4c79 100644 --- a/apps/client/src/components/settings/settings.module.css +++ b/apps/client/src/components/settings/settings.module.css @@ -58,7 +58,7 @@ align-items: center; } -.version { +.text { padding-left: var(--mantine-spacing-xs) ; padding-top: 10px; } diff --git a/apps/client/src/ee/LICENSE b/apps/client/src/ee/LICENSE new file mode 100644 index 00000000..2a9b8b4c --- /dev/null +++ b/apps/client/src/ee/LICENSE @@ -0,0 +1 @@ +Files in this directory are subject to the Docmost Enterprise Software license. \ No newline at end of file diff --git a/apps/client/src/ee/billing/components/billing-details.tsx b/apps/client/src/ee/billing/components/billing-details.tsx new file mode 100644 index 00000000..9ecd1558 --- /dev/null +++ b/apps/client/src/ee/billing/components/billing-details.tsx @@ -0,0 +1,130 @@ +import { + useBillingPlans, + useBillingQuery, +} from "@/ee/billing/queries/billing-query.ts"; +import { Group, Text, SimpleGrid, Paper } from "@mantine/core"; +import classes from "./billing.module.css"; +import { format } from "date-fns"; +import { formatInterval } from "@/ee/billing/utils.ts"; + +export default function BillingDetails() { + const { data: billing } = useBillingQuery(); + const { data: plans } = useBillingPlans(); + + if (!billing || !plans) { + return null; + } + + return ( +
+ + + +
+ + Plan + + + { + plans.find( + (plan) => plan.productId === billing.stripeProductId, + )?.name + } + +
+
+
+ + + +
+ + Billing Period + + + {formatInterval(billing.interval)} + +
+
+
+ + + +
+ + {billing.cancelAtPeriodEnd + ? "Cancellation date" + : "Renewal date"} + + + {format(billing.periodEndAt, "dd MMM, yyyy")} + +
+
+
+
+ + + + +
+ + Seat count + + + {billing.quantity} + +
+
+
+ + + +
+ + Total + + + {(billing.amount / 100) * billing.quantity}{" "} + {billing.currency.toUpperCase()} + + + ${billing.amount / 100} /user/{billing.interval} + +
+
+
+
+
+ ); +} diff --git a/apps/client/src/ee/billing/components/billing-incomplete.tsx b/apps/client/src/ee/billing/components/billing-incomplete.tsx new file mode 100644 index 00000000..d2e6b42f --- /dev/null +++ b/apps/client/src/ee/billing/components/billing-incomplete.tsx @@ -0,0 +1,13 @@ +import { Alert } from "@mantine/core"; +import React from "react"; + +export default function BillingIncomplete() { + return ( + <> + + Your subscription is in an incomplete state. Please refresh this page if + you recently made your payment. + + + ); +} diff --git a/apps/client/src/ee/billing/components/billing-plans.tsx b/apps/client/src/ee/billing/components/billing-plans.tsx new file mode 100644 index 00000000..3ff655d6 --- /dev/null +++ b/apps/client/src/ee/billing/components/billing-plans.tsx @@ -0,0 +1,115 @@ +import { + Button, + Card, + List, + SegmentedControl, + ThemeIcon, + Title, + Text, + Group, +} from "@mantine/core"; +import { useState } from "react"; +import { IconCheck } from "@tabler/icons-react"; +import { useBillingPlans } from "@/ee/billing/queries/billing-query.ts"; +import { getCheckoutLink } from "@/ee/billing/services/billing-service.ts"; + +export default function BillingPlans() { + const { data: plans } = useBillingPlans(); + const [interval, setInterval] = useState("yearly"); + + if (!plans) { + return null; + } + + const handleCheckout = async (priceId: string) => { + try { + const checkoutLink = await getCheckoutLink({ + priceId: priceId, + }); + window.location.href = checkoutLink.url; + } catch (err) { + console.error("Failed to get checkout link", err); + } + }; + + return ( + + {plans.map((plan) => { + const price = + interval === "monthly" ? plan.price.monthly : plan.price.yearly; + const priceId = interval === "monthly" ? plan.monthlyId : plan.yearlyId; + const yearlyMonthPrice = parseInt(plan.price.yearly) / 12; + + return ( + + + + + {plan.name} + + + {interval === "monthly" && ( + <> + ${price}{" "} + + /user/month + + + )} + {interval === "yearly" && ( + <> + ${yearlyMonthPrice}{" "} + + /user/month + + + )} +
+ + billed {interval} + +
+ + + + + + + + + + } + > + {plan.features.map((feature, index) => ( + {feature} + ))} + + +
+ ); + })} +
+ ); +} diff --git a/apps/client/src/ee/billing/components/billing-trial.tsx b/apps/client/src/ee/billing/components/billing-trial.tsx new file mode 100644 index 00000000..4628ae00 --- /dev/null +++ b/apps/client/src/ee/billing/components/billing-trial.tsx @@ -0,0 +1,32 @@ +import { Alert } from "@mantine/core"; +import { useBillingQuery } from "@/ee/billing/queries/billing-query.ts"; +import useTrial from "@/ee/hooks/use-trial.tsx"; + +export default function BillingTrial() { + const { data: billing, isLoading } = useBillingQuery(); + const { trialDaysLeft } = useTrial(); + + if (isLoading) { + return null; + } + + return ( + <> + {trialDaysLeft > 0 && !billing && ( + + You have {trialDaysLeft} {trialDaysLeft === 1 ? "day" : "days"} left + in your 14-day trial. Please subscribe to a plan before your trial + ends. + + )} + + {trialDaysLeft === 0 || + (trialDaysLeft === null && !billing && ( + + Your 14-day trial has come to an end. Please subscribe to a plan to + continue using this service. + + ))} + + ); +} diff --git a/apps/client/src/ee/billing/components/billing.module.css b/apps/client/src/ee/billing/components/billing.module.css new file mode 100644 index 00000000..50398f33 --- /dev/null +++ b/apps/client/src/ee/billing/components/billing.module.css @@ -0,0 +1,10 @@ +.root { + padding-top: var(--mantine-spacing-xs); + padding-bottom: var(--mantine-spacing-xs); +} + +.label { + font-family: + Greycliff CF, + var(--mantine-font-family); +} \ No newline at end of file diff --git a/apps/client/src/ee/billing/components/manage-billing.tsx b/apps/client/src/ee/billing/components/manage-billing.tsx new file mode 100644 index 00000000..2424d1e6 --- /dev/null +++ b/apps/client/src/ee/billing/components/manage-billing.tsx @@ -0,0 +1,34 @@ +import { Button, Group, Text } from "@mantine/core"; +import React from "react"; +import { getBillingPortalLink } from "@/ee/billing/services/billing-service.ts"; + +export default function ManageBilling() { + const handleBillingPortal = async () => { + try { + const portalLink = await getBillingPortalLink(); + window.location.href = portalLink.url; + } catch (err) { + console.error("Failed to get billing portal link", err); + } + }; + + return ( + <> + +
+ + Manage subscription + + + Manage your your subscription, invoices, update payment details, and + more. + +
+ + +
+ + ); +} diff --git a/apps/client/src/ee/billing/pages/billing.tsx b/apps/client/src/ee/billing/pages/billing.tsx new file mode 100644 index 00000000..a389a1e5 --- /dev/null +++ b/apps/client/src/ee/billing/pages/billing.tsx @@ -0,0 +1,41 @@ +import { Helmet } from "react-helmet-async"; +import { getAppName } from "@/lib/config.ts"; +import SettingsTitle from "@/components/settings/settings-title.tsx"; +import BillingPlans from "@/ee/billing/components/billing-plans.tsx"; +import BillingTrial from "@/ee/billing/components/billing-trial.tsx"; +import ManageBilling from "@/ee/billing/components/manage-billing.tsx"; +import { Divider } from "@mantine/core"; +import React from "react"; +import BillingDetails from "@/ee/billing/components/billing-details.tsx"; +import { useBillingQuery } from "@/ee/billing/queries/billing-query.ts"; +import useUserRole from "@/hooks/use-user-role.tsx"; + +export default function Billing() { + const { data: billing, isError: isBillingError } = useBillingQuery(); + const { isAdmin } = useUserRole(); + + if (!isAdmin) { + return null; + } + + return ( + <> + + Billing - {getAppName()} + + + + + + + {isBillingError && } + + {billing && ( + <> + + + + )} + + ); +} diff --git a/apps/client/src/ee/billing/queries/billing-query.ts b/apps/client/src/ee/billing/queries/billing-query.ts new file mode 100644 index 00000000..261102f5 --- /dev/null +++ b/apps/client/src/ee/billing/queries/billing-query.ts @@ -0,0 +1,20 @@ +import { useQuery, UseQueryResult } from "@tanstack/react-query"; +import { + getBilling, + getBillingPlans, +} from "@/ee/billing/services/billing-service.ts"; +import { IBilling, IBillingPlan } from "@/ee/billing/types/billing.types.ts"; + +export function useBillingQuery(): UseQueryResult { + return useQuery({ + queryKey: ["billing"], + queryFn: () => getBilling(), + }); +} + +export function useBillingPlans(): UseQueryResult { + return useQuery({ + queryKey: ["billing-plans"], + queryFn: () => getBillingPlans(), + }); +} diff --git a/apps/client/src/ee/billing/services/billing-service.ts b/apps/client/src/ee/billing/services/billing-service.ts new file mode 100644 index 00000000..c76f4ea5 --- /dev/null +++ b/apps/client/src/ee/billing/services/billing-service.ts @@ -0,0 +1,29 @@ +import api from "@/lib/api-client.ts"; +import { + IBilling, + IBillingPlan, + IBillingPortal, + ICheckoutLink, +} from "@/ee/billing/types/billing.types.ts"; + +export async function getBilling(): Promise { + const req = await api.post("/billing/info"); + return req.data; +} + +export async function getBillingPlans(): Promise { + const req = await api.post("/billing/plans"); + return req.data; +} + +export async function getCheckoutLink(data: { + priceId: string; +}): Promise { + const req = await api.post("/billing/checkout", data); + return req.data; +} + +export async function getBillingPortalLink(): Promise { + const req = await api.post("/billing/portal"); + return req.data; +} diff --git a/apps/client/src/ee/billing/types/billing.types.ts b/apps/client/src/ee/billing/types/billing.types.ts new file mode 100644 index 00000000..89d936f0 --- /dev/null +++ b/apps/client/src/ee/billing/types/billing.types.ts @@ -0,0 +1,49 @@ +export enum BillingPlan { + STANDARD = "standard", +} + +export interface IBilling { + id: string; + stripeSubscriptionId: string; + stripeCustomerId: string; + status: string; + quantity: number; + amount: number; + interval: string; + currency: string; + metadata: Record; + stripePriceId: string; + stripeItemId: string; + stripeProductId: string; + periodStartAt: Date; + periodEndAt: Date; + cancelAtPeriodEnd: boolean; + cancelAt: Date; + canceledAt: Date; + workspaceId: string; + createdAt: Date; + updatedAt: Date; + deletedAt: Date; +} + +export interface ICheckoutLink { + url: string; +} + +export interface IBillingPortal { + url: string; +} + +export interface IBillingPlan { + name: string; + description: string; + productId: string; + monthlyId: string; + yearlyId: string; + currency: string; + price: { + monthly: string; + yearly: string; + }; + features: string[]; +} diff --git a/apps/client/src/ee/billing/utils.ts b/apps/client/src/ee/billing/utils.ts new file mode 100644 index 00000000..bf41dff7 --- /dev/null +++ b/apps/client/src/ee/billing/utils.ts @@ -0,0 +1,17 @@ +import { differenceInCalendarDays } from "date-fns"; + +export function formatInterval(interval: string): string { + if (interval === "month") { + return "monthly"; + } + if (interval === "year") { + return "yearly"; + } +} + +export function getTrialDaysLeft(trialEndAt: Date) { + if (!trialEndAt) return null; + + const daysLeft = differenceInCalendarDays(trialEndAt, new Date()); + return daysLeft > 0 ? daysLeft : 0; +} diff --git a/apps/client/src/ee/cloud/query/cloud-query.ts b/apps/client/src/ee/cloud/query/cloud-query.ts new file mode 100644 index 00000000..367d89e3 --- /dev/null +++ b/apps/client/src/ee/cloud/query/cloud-query.ts @@ -0,0 +1,13 @@ +import { useQuery, UseQueryResult } from "@tanstack/react-query"; +import { IWorkspace } from "@/features/workspace/types/workspace.types.ts"; +import { getJoinedWorkspaces } from "@/ee/cloud/service/cloud-service.ts"; + +export function useJoinedWorkspacesQuery(): UseQueryResult< + Partial, + Error +> { + return useQuery({ + queryKey: ["joined-workspaces"], + queryFn: () => getJoinedWorkspaces(), + }); +} diff --git a/apps/client/src/ee/cloud/service/cloud-service.ts b/apps/client/src/ee/cloud/service/cloud-service.ts new file mode 100644 index 00000000..e544733e --- /dev/null +++ b/apps/client/src/ee/cloud/service/cloud-service.ts @@ -0,0 +1,7 @@ +import { IWorkspace } from "@/features/workspace/types/workspace.types.ts"; +import api from "@/lib/api-client.ts"; + +export async function getJoinedWorkspaces(): Promise> { + const req = await api.post>("/workspace/joined"); + return req.data; +} diff --git a/apps/client/src/ee/components/cloud-login-form.tsx b/apps/client/src/ee/components/cloud-login-form.tsx new file mode 100644 index 00000000..cb1af8a7 --- /dev/null +++ b/apps/client/src/ee/components/cloud-login-form.tsx @@ -0,0 +1,96 @@ +import * as z from "zod"; +import { useForm, zodResolver } from "@mantine/form"; +import { + Container, + Title, + TextInput, + Button, + Box, + Text, + Anchor, + Divider, +} from "@mantine/core"; +import classes from "../../features/auth/components/auth.module.css"; +import { getCheckHostname } from "@/features/workspace/services/workspace-service.ts"; +import { useState } from "react"; +import { getSubdomainHost } from "@/lib/config.ts"; +import { Link } from "react-router-dom"; +import APP_ROUTE from "@/lib/app-route.ts"; +import { useTranslation } from "react-i18next"; +import JoinedWorkspaces from "@/ee/components/joined-workspaces.tsx"; +import { useJoinedWorkspacesQuery } from "@/ee/cloud/query/cloud-query.ts"; + +const formSchema = z.object({ + hostname: z.string().min(1, { message: "subdomain is required" }), +}); + +export function CloudLoginForm() { + const { t } = useTranslation(); + const [isLoading, setIsLoading] = useState(false); + const { data: joinedWorkspaces } = useJoinedWorkspacesQuery(); + + const form = useForm({ + validate: zodResolver(formSchema), + initialValues: { + hostname: "", + }, + }); + + async function onSubmit(data: { hostname: string }) { + setIsLoading(true); + + try { + const checkHostname = await getCheckHostname(data.hostname); + window.location.href = checkHostname.hostname; + } catch (err) { + if (err?.status === 404) { + form.setFieldError("hostname", "We could not find this workspace"); + } else { + form.setFieldError("hostname", "An error occurred"); + } + } + + setIsLoading(false); + } + + return ( +
+ + + + {t("Login")} + + + + + {joinedWorkspaces?.length > 0 && ( + + )} + +
+ .{getSubdomainHost()}} + rightSectionWidth={150} + withErrorStyles={false} + {...form.getInputProps("hostname")} + /> + + +
+
+ + + {t("Don't have a workspace?")}{" "} + + {t("Create new workspace")} + + +
+ ); +} diff --git a/apps/client/src/ee/components/joined-workspaces.module.css b/apps/client/src/ee/components/joined-workspaces.module.css new file mode 100644 index 00000000..74871e66 --- /dev/null +++ b/apps/client/src/ee/components/joined-workspaces.module.css @@ -0,0 +1,13 @@ +.workspace { + display: block; + width: 100%; + padding: var(--mantine-spacing-xs); + margin-bottom: var(--mantine-spacing-xs); + color: light-dark(var(--mantine-color-black), var(--mantine-color-dark-0)); + border: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5)); + border-radius: var(--mantine-spacing-xs); + + @mixin hover { + background-color: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-8)); + } +} \ No newline at end of file diff --git a/apps/client/src/ee/components/joined-workspaces.tsx b/apps/client/src/ee/components/joined-workspaces.tsx new file mode 100644 index 00000000..7129aa00 --- /dev/null +++ b/apps/client/src/ee/components/joined-workspaces.tsx @@ -0,0 +1,49 @@ +import { Group, Text, UnstyledButton } from "@mantine/core"; +import { useJoinedWorkspacesQuery } from "../cloud/query/cloud-query"; +import { CustomAvatar } from "@/components/ui/custom-avatar.tsx"; +import classes from "./joined-workspaces.module.css"; +import { IconChevronRight } from "@tabler/icons-react"; +import { getHostnameUrl } from "@/ee/utils.ts"; +import { Link } from "react-router-dom"; +import { IWorkspace } from "@/features/workspace/types/workspace.types.ts"; + +export default function JoinedWorkspaces() { + const { data, isLoading } = useJoinedWorkspacesQuery(); + if (isLoading || !data || data?.length === 0) { + return null; + } + + return ( + <> + {data.map((workspace: Partial, index) => ( + + + + +
+ + {workspace?.name} + + + + {getHostnameUrl(workspace?.hostname)?.split("//")[1]} + +
+ + +
+
+ ))} + + ); +} diff --git a/apps/client/src/ee/components/manage-hostname.tsx b/apps/client/src/ee/components/manage-hostname.tsx new file mode 100644 index 00000000..c7a595ff --- /dev/null +++ b/apps/client/src/ee/components/manage-hostname.tsx @@ -0,0 +1,119 @@ +import { Button, Group, Text, Modal, TextInput } from "@mantine/core"; +import * as z from "zod"; +import { useState } from "react"; +import { useDisclosure } from "@mantine/hooks"; +import * as React from "react"; +import { useForm, zodResolver } from "@mantine/form"; +import { notifications } from "@mantine/notifications"; +import { useTranslation } from "react-i18next"; +import { getSubdomainHost } from "@/lib/config.ts"; +import { IWorkspace } from "@/features/workspace/types/workspace.types.ts"; +import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts"; +import { getHostnameUrl } from "@/ee/utils.ts"; +import { useAtom } from "jotai/index"; +import { + currentUserAtom, + workspaceAtom, +} from "@/features/user/atoms/current-user-atom.ts"; +import useUserRole from "@/hooks/use-user-role.tsx"; +import { RESET } from "jotai/utils"; + +export default function ManageHostname() { + const { t } = useTranslation(); + const [opened, { open, close }] = useDisclosure(false); + const [workspace] = useAtom(workspaceAtom); + const { isAdmin } = useUserRole(); + + return ( + +
+ {t("Hostname")} + + {workspace?.hostname}.{getSubdomainHost()} + +
+ + {isAdmin && ( + + )} + + + + +
+ ); +} + +const formSchema = z.object({ + hostname: z.string().min(4), +}); + +type FormValues = z.infer; + +interface ChangeHostnameFormProps { + onClose?: () => void; +} +function ChangeHostnameForm({ onClose }: ChangeHostnameFormProps) { + const { t } = useTranslation(); + const [isLoading, setIsLoading] = useState(false); + const [currentUser, setCurrentUser] = useAtom(currentUserAtom); + + const form = useForm({ + validate: zodResolver(formSchema), + initialValues: { + hostname: currentUser?.workspace?.hostname, + }, + }); + + async function handleSubmit(data: Partial) { + setIsLoading(true); + + if (data.hostname === currentUser?.workspace?.hostname) { + onClose(); + return; + } + + try { + await updateWorkspace({ + hostname: data.hostname, + }); + setCurrentUser(RESET); + window.location.href = getHostnameUrl(data.hostname.toLowerCase()); + } catch (err) { + notifications.show({ + message: err?.response?.data?.message, + color: "red", + }); + } + setIsLoading(false); + } + + return ( +
+ .{getSubdomainHost()}} + rightSectionWidth={150} + withErrorStyles={false} + width={200} + {...form.getInputProps("hostname")} + /> + + + + + + ); +} diff --git a/apps/client/src/ee/components/sso-cloud-signup.tsx b/apps/client/src/ee/components/sso-cloud-signup.tsx new file mode 100644 index 00000000..c8657955 --- /dev/null +++ b/apps/client/src/ee/components/sso-cloud-signup.tsx @@ -0,0 +1,25 @@ +import { Button, Divider, Stack } from "@mantine/core"; +import { getGoogleSignupUrl } from "@/ee/security/sso.utils.ts"; +import { GoogleIcon } from "@/components/icons/google-icon.tsx"; + +export default function SsoCloudSignup() { + const handleSsoLogin = () => { + window.location.href = getGoogleSignupUrl(); + }; + + return ( + <> + + + + + + ); +} diff --git a/apps/client/src/ee/components/sso-login.tsx b/apps/client/src/ee/components/sso-login.tsx new file mode 100644 index 00000000..8de93c29 --- /dev/null +++ b/apps/client/src/ee/components/sso-login.tsx @@ -0,0 +1,57 @@ +import { useWorkspacePublicDataQuery } from "@/features/workspace/queries/workspace-query.ts"; +import { Button, Divider, Stack } from "@mantine/core"; +import { IconLock } from "@tabler/icons-react"; +import { IAuthProvider } from "@/ee/security/types/security.types.ts"; +import { buildSsoLoginUrl } from "@/ee/security/sso.utils.ts"; +import { SSO_PROVIDER } from "@/ee/security/contants.ts"; +import { GoogleIcon } from "@/components/icons/google-icon.tsx"; +import { isCloud } from "@/lib/config.ts"; + +export default function SsoLogin() { + const { data, isLoading } = useWorkspacePublicDataQuery(); + + if (!data?.authProviders || data?.authProviders?.length === 0) { + return null; + } + + const handleSsoLogin = (provider: IAuthProvider) => { + window.location.href = buildSsoLoginUrl({ + providerId: provider.id, + type: provider.type, + workspaceId: data.id, + }); + }; + + return ( + <> + {(isCloud() || data.hasLicenseKey) && ( + <> + + {data.authProviders.map((provider) => ( +
+ +
+ ))} +
+ + {!data.enforceSso && ( + + )} + + )} + + ); +} diff --git a/apps/client/src/ee/hooks/use-plan.tsx b/apps/client/src/ee/hooks/use-plan.tsx new file mode 100644 index 00000000..52790178 --- /dev/null +++ b/apps/client/src/ee/hooks/use-plan.tsx @@ -0,0 +1,15 @@ +import { useAtom } from "jotai"; +import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts"; +import { BillingPlan } from "@/ee/billing/types/billing.types.ts"; + +export const usePlan = () => { + const [workspace] = useAtom(workspaceAtom); + + const isStandard = + typeof workspace?.plan === "string" && + workspace?.plan.toLowerCase() === BillingPlan.STANDARD.toLowerCase(); + + return { isStandard }; +}; + +export default usePlan; diff --git a/apps/client/src/ee/hooks/use-redirect-to-cloud-select.tsx b/apps/client/src/ee/hooks/use-redirect-to-cloud-select.tsx new file mode 100644 index 00000000..a7d78047 --- /dev/null +++ b/apps/client/src/ee/hooks/use-redirect-to-cloud-select.tsx @@ -0,0 +1,20 @@ +import { useEffect } from "react"; +import { useLocation, useNavigate } from "react-router-dom"; +import { getAppUrl, getServerAppUrl, isCloud } from "@/lib/config.ts"; +import APP_ROUTE from "@/lib/app-route.ts"; + +export const useRedirectToCloudSelect = () => { + const navigate = useNavigate(); + const pathname = useLocation().pathname; + + useEffect(() => { + const pathsToRedirect = ["/login", "/home"]; + if (isCloud() && pathsToRedirect.includes(pathname)) { + const frontendUrl = getAppUrl(); + const serverUrl = getServerAppUrl(); + if (frontendUrl === serverUrl) { + navigate(APP_ROUTE.AUTH.SELECT_WORKSPACE); + } + } + }, [navigate]); +}; diff --git a/apps/client/src/ee/hooks/use-trial-end-action.tsx b/apps/client/src/ee/hooks/use-trial-end-action.tsx new file mode 100644 index 00000000..5ca08dbf --- /dev/null +++ b/apps/client/src/ee/hooks/use-trial-end-action.tsx @@ -0,0 +1,36 @@ +import { useEffect } from "react"; +import { useLocation, useNavigate } from "react-router-dom"; +import { isCloud } from "@/lib/config.ts"; +import APP_ROUTE from "@/lib/app-route.ts"; +import useUserRole from "@/hooks/use-user-role.tsx"; +import { notifications } from "@mantine/notifications"; +import useTrial from "@/ee/hooks/use-trial.tsx"; + +export const useTrialEndAction = () => { + const navigate = useNavigate(); + const pathname = useLocation().pathname; + const { isAdmin } = useUserRole(); + const { trialDaysLeft } = useTrial(); + + useEffect(() => { + if (isCloud() && trialDaysLeft === 0) { + if (!pathname.startsWith("/settings")) { + notifications.show({ + position: "top-right", + color: "red", + title: "Your 14-day trial has ended", + message: + "Please upgrade to a paid plan or contact your workspace admin.", + autoClose: false, + }); + + // only admins can access the billing page + if (isAdmin) { + navigate(APP_ROUTE.SETTINGS.WORKSPACE.BILLING); + } else { + navigate(APP_ROUTE.SETTINGS.ACCOUNT.PROFILE); + } + } + } + }, [navigate]); +}; diff --git a/apps/client/src/ee/hooks/use-trial.tsx b/apps/client/src/ee/hooks/use-trial.tsx new file mode 100644 index 00000000..2ae68af2 --- /dev/null +++ b/apps/client/src/ee/hooks/use-trial.tsx @@ -0,0 +1,16 @@ +import { useAtom } from "jotai"; +import { currentUserAtom } from "@/features/user/atoms/current-user-atom.ts"; +import { getTrialDaysLeft } from "@/ee/billing/utils.ts"; +import { ICurrentUser } from "@/features/user/types/user.types.ts"; + +export const useTrial = () => { + const [currentUser] = useAtom(currentUserAtom); + const workspace = currentUser?.workspace; + + const trialDaysLeft = getTrialDaysLeft(workspace?.trialEndAt); + const isTrial = !!workspace?.trialEndAt && trialDaysLeft !== null; + + return { isTrial: isTrial, trialDaysLeft: trialDaysLeft }; +}; + +export default useTrial; diff --git a/apps/client/src/ee/licence/components/activate-license-modal.tsx b/apps/client/src/ee/licence/components/activate-license-modal.tsx new file mode 100644 index 00000000..9d81f29e --- /dev/null +++ b/apps/client/src/ee/licence/components/activate-license-modal.tsx @@ -0,0 +1,89 @@ +import * as z from "zod"; +import React from "react"; +import { Button, Group, Modal, Textarea } from "@mantine/core"; +import { useForm, zodResolver } from "@mantine/form"; +import { useTranslation } from "react-i18next"; +import { useActivateMutation } from "@/ee/licence/queries/license-query.ts"; +import { useDisclosure } from "@mantine/hooks"; +import { useAtom } from "jotai"; +import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts"; +import RemoveLicense from "@/ee/licence/components/remove-license.tsx"; + +export default function ActivateLicense() { + const { t } = useTranslation(); + const [opened, { open, close }] = useDisclosure(false); + const [workspace] = useAtom(workspaceAtom); + + return ( + + + + {workspace?.hasLicenseKey && } + + + + + + ); +} + +const formSchema = z.object({ + licenseKey: z.string().min(1), +}); + +type FormValues = z.infer; + +interface ActivateLicenseFormProps { + onClose?: () => void; +} +export function ActivateLicenseForm({ onClose }: ActivateLicenseFormProps) { + const { t } = useTranslation(); + const activateLicenseMutation = useActivateMutation(); + + const form = useForm({ + validate: zodResolver(formSchema), + initialValues: { + licenseKey: "", + }, + }); + + async function handleSubmit(data: { licenseKey: string }) { + await activateLicenseMutation.mutateAsync(data.licenseKey); + form.reset(); + onClose(); + } + + return ( +
+