From fd5b7d20a8e795d133b93fc46f3df75e06c77114 Mon Sep 17 00:00:00 2001 From: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com> Date: Wed, 29 Nov 2023 20:33:08 +0530 Subject: [PATCH] dev: instance setup workflow (#2935) * chore: instance type updated * chore: instance not ready screen added * chore: instance layout added * chore: instance magic sign in endpoint and type added * chore: instance admin password endpoint added * chore: instance setup page added * chore: instance setup form added * chore: instance layout updated * fix: instance admin workflow setup * fix: admin workflow setup --------- Co-authored-by: sriram veeraghanta --- web/components/account/sign-in-forms/root.tsx | 2 +- web/components/common/index.ts | 1 + .../common/latest-feature-block.tsx | 35 ++++ web/components/instance/index.ts | 4 + web/components/instance/not-ready-view.tsx | 34 ++++ web/components/instance/setup-done-view.tsx | 43 +++++ .../instance/setup-form/email-code-form.tsx | 167 ++++++++++++++++++ .../instance/setup-form/email-form.tsx | 105 +++++++++++ web/components/instance/setup-form/index.ts | 3 + .../instance/setup-form/password-form.tsx | 120 +++++++++++++ web/components/instance/setup-form/root.tsx | 70 ++++++++ web/components/instance/setup-view.tsx | 38 ++++ web/layouts/admin-layout/layout.tsx | 15 ++ web/layouts/instance-layout/index.tsx | 52 ++++++ .../settings-layout/profile/sidebar.tsx | 27 ++- web/lib/app-provider.tsx | 39 ++-- web/pages/_app.tsx | 10 +- web/public/instance-not-ready.svg | 57 ++++++ web/public/instance-setup-done.svg | 60 +++++++ .../instance/plane-instance-not-ready.webp | Bin 0 -> 45894 bytes web/services/auth.service.ts | 30 ++++ web/services/instance.service.ts | 10 +- web/store/instance/instance.store.ts | 18 ++ web/types/instance.d.ts | 2 + web/types/users.d.ts | 1 + 25 files changed, 905 insertions(+), 38 deletions(-) create mode 100644 web/components/common/latest-feature-block.tsx create mode 100644 web/components/instance/not-ready-view.tsx create mode 100644 web/components/instance/setup-done-view.tsx create mode 100644 web/components/instance/setup-form/email-code-form.tsx create mode 100644 web/components/instance/setup-form/email-form.tsx create mode 100644 web/components/instance/setup-form/index.ts create mode 100644 web/components/instance/setup-form/password-form.tsx create mode 100644 web/components/instance/setup-form/root.tsx create mode 100644 web/components/instance/setup-view.tsx create mode 100644 web/layouts/instance-layout/index.tsx create mode 100644 web/public/instance-not-ready.svg create mode 100644 web/public/instance-setup-done.svg create mode 100644 web/public/instance/plane-instance-not-ready.webp diff --git a/web/components/account/sign-in-forms/root.tsx b/web/components/account/sign-in-forms/root.tsx index 49089b5bd..d2d6ef753 100644 --- a/web/components/account/sign-in-forms/root.tsx +++ b/web/components/account/sign-in-forms/root.tsx @@ -31,7 +31,7 @@ export const SignInRoot: React.FC = (props) => { return ( <> -
+
{signInStep === ESignInSteps.EMAIL && ( setSignInStep(step)} diff --git a/web/components/common/index.ts b/web/components/common/index.ts index 57f8b6cca..04aaa8512 100644 --- a/web/components/common/index.ts +++ b/web/components/common/index.ts @@ -1,2 +1,3 @@ export * from "./product-updates-modal"; export * from "./empty-state"; +export * from "./latest-feature-block"; diff --git a/web/components/common/latest-feature-block.tsx b/web/components/common/latest-feature-block.tsx new file mode 100644 index 000000000..46309f6da --- /dev/null +++ b/web/components/common/latest-feature-block.tsx @@ -0,0 +1,35 @@ +import Image from "next/image"; +import { useTheme } from "next-themes"; +// icons +import { Lightbulb } from "lucide-react"; +// images +import signInIssues from "public/onboarding/onboarding-issues.svg"; + +export const LatestFeatureBlock = () => { + const { resolvedTheme } = useTheme(); + + return ( + <> +
+ +

+ Try the latest features, like Tiptap editor, to write compelling responses.{" "} + {}}> + See new features + +

+
+
+ Plane Issues +
+ + ); +}; diff --git a/web/components/instance/index.ts b/web/components/instance/index.ts index 0a8f87052..0d19304bb 100644 --- a/web/components/instance/index.ts +++ b/web/components/instance/index.ts @@ -8,3 +8,7 @@ export * from "./github-config-form"; export * from "./google-config-form"; export * from "./image-config-form"; export * from "./instance-admin-restriction"; +export * from "./not-ready-view"; +export * from "./setup-view"; +export * from "./setup-done-view"; +export * from "./setup-form"; diff --git a/web/components/instance/not-ready-view.tsx b/web/components/instance/not-ready-view.tsx new file mode 100644 index 000000000..501ab4d41 --- /dev/null +++ b/web/components/instance/not-ready-view.tsx @@ -0,0 +1,34 @@ +import React from "react"; +import Image from "next/image"; +import { useTheme } from "next-themes"; +// image +import instanceNotReady from "public/instance/plane-instance-not-ready.webp"; +import PlaneWhiteLogo from "public/plane-logos/white-horizontal-with-blue-logo.svg"; +import PlaneDarkLogo from "public/plane-logos/black-horizontal-with-blue-logo.svg"; + +export const InstanceNotReady = () => { + const { resolvedTheme } = useTheme(); + + const planeLogo = resolvedTheme === "dark" ? PlaneWhiteLogo : PlaneDarkLogo; + + return ( +
+
+
+
+
+ image +
+
+ image +
+
+

Your Plane instance isn’t ready yet

+

Ask your Instance Admin to complete set-up first.

+
+
+
+
+
+ ); +}; diff --git a/web/components/instance/setup-done-view.tsx b/web/components/instance/setup-done-view.tsx new file mode 100644 index 000000000..4885e006a --- /dev/null +++ b/web/components/instance/setup-done-view.tsx @@ -0,0 +1,43 @@ +import React from "react"; +import Image from "next/image"; + +// ui +import { Button } from "@plane/ui"; +import { UserCog2 } from "lucide-react"; +// image +import instanceSetupDone from "public/instance-setup-done.svg"; +import PlaneLogo from "public/plane-logos/blue-without-text.png"; + +export const InstanceSetupDone = () => ( +
+
+
+
+
+ image + To the stratosphere now! +
+ +
+ image +
+ +
+ + Your instance is now ready for more security, more controls, and more intelligence. + + + +
+ Use this wisely. Remember, with great power comes great responsibility.🕷️ +
+
+
+
+
+
+); diff --git a/web/components/instance/setup-form/email-code-form.tsx b/web/components/instance/setup-form/email-code-form.tsx new file mode 100644 index 000000000..6264ba20f --- /dev/null +++ b/web/components/instance/setup-form/email-code-form.tsx @@ -0,0 +1,167 @@ +import { FC } from "react"; +import { useForm, Controller } from "react-hook-form"; +// ui +import { Input, Button } from "@plane/ui"; +// icons +import { XCircle } from "lucide-react"; +// services +import { AuthService } from "services/auth.service"; +const authService = new AuthService(); +// hooks +import useToast from "hooks/use-toast"; +import useTimer from "hooks/use-timer"; + +export interface InstanceSetupEmailCodeFormValues { + email: string; + token: string; +} + +export interface IInstanceSetupEmailCodeForm { + email: string; + handleNextStep: () => void; + moveBack: () => void; +} + +export const InstanceSetupEmailCodeForm: FC = (props) => { + const { handleNextStep, email, moveBack } = props; + // form info + const { + control, + handleSubmit, + reset, + formState: { isSubmitting }, + } = useForm({ + defaultValues: { + email, + token: "", + }, + }); + // hooks + const { setToastAlert } = useToast(); + const { timer, setTimer } = useTimer(30); + // computed + const isResendDisabled = timer > 0 || isSubmitting; + + const handleEmailCodeFormSubmit = (formValues: InstanceSetupEmailCodeFormValues) => + authService + .instanceMagicSignIn({ key: `magic_${formValues.email}`, token: formValues.token }) + .then(() => { + reset(); + handleNextStep(); + }) + .catch((err) => { + setToastAlert({ + title: "Oops!", + type: "error", + message: err?.error, + }); + }); + + const resendMagicCode = () => { + setTimer(30); + authService + .instanceAdminEmailCode({ email }) + .then(() => { + // setCodeResending(false); + setTimer(30); + }) + .catch((err) => { + setToastAlert({ + title: "Oops!", + type: "error", + message: err?.error, + }); + }); + }; + + return ( +
+
+

+ Let’s secure your instance +

+
+

Paste the code you got at

+ {email} + below. +
+ +
+ + /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/.test( + value + ) || "Email address is not valid", + }} + render={({ field: { value, onChange } }) => ( +
+ + moveBack()} + /> +
+ )} + /> +
+ {timer > 0 ? ( + Request new code in {timer}s + ) : isSubmitting ? ( + "Sending new code..." + ) : ( +
+ +
+ )} +
+ ( +
+ +
+ )} + /> + + +
+
+
+ ); +}; diff --git a/web/components/instance/setup-form/email-form.tsx b/web/components/instance/setup-form/email-form.tsx new file mode 100644 index 000000000..b2c336695 --- /dev/null +++ b/web/components/instance/setup-form/email-form.tsx @@ -0,0 +1,105 @@ +import { FC } from "react"; +import { useForm, Controller } from "react-hook-form"; +// ui +import { Input, Button } from "@plane/ui"; +// icons +import { XCircle } from "lucide-react"; +// services +import { AuthService } from "services/auth.service"; +const authService = new AuthService(); +// hooks +import useToast from "hooks/use-toast"; + +export interface InstanceSetupEmailFormValues { + email: string; +} + +export interface IInstanceSetupEmailForm { + handleNextStep: (email: string) => void; +} + +export const InstanceSetupEmailForm: FC = (props) => { + const { handleNextStep } = props; + // form info + const { + control, + handleSubmit, + setValue, + reset, + formState: { isSubmitting }, + } = useForm({ + defaultValues: { + email: "", + }, + }); + // hooks + const { setToastAlert } = useToast(); + + const handleEmailFormSubmit = (formValues: InstanceSetupEmailFormValues) => + authService + .instanceAdminEmailCode({ email: formValues.email }) + .then(() => { + reset(); + handleNextStep(formValues.email); + }) + .catch((err) => { + setToastAlert({ + title: "Oops!", + type: "error", + message: err?.error, + }); + }); + + return ( +
+
+

+ Let’s secure your instance +

+

+ Explore privacy options. Get AI features. Secure access.
Takes 2 minutes. +

+ +
+ + /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/.test( + value + ) || "Email address is not valid", + }} + render={({ field: { value, onChange } }) => ( +
+ + {value.length > 0 && ( + setValue("email", "")} + /> + )} +
+ )} + /> +

+ Use your email address if you are the instance admin.
Use your admin’s e-mail if you are not. +

+ + +
+
+
+ ); +}; diff --git a/web/components/instance/setup-form/index.ts b/web/components/instance/setup-form/index.ts new file mode 100644 index 000000000..f7a18d5b6 --- /dev/null +++ b/web/components/instance/setup-form/index.ts @@ -0,0 +1,3 @@ +export * from "./email-code-form"; +export * from "./email-form"; +export * from "./root"; diff --git a/web/components/instance/setup-form/password-form.tsx b/web/components/instance/setup-form/password-form.tsx new file mode 100644 index 000000000..f963b960e --- /dev/null +++ b/web/components/instance/setup-form/password-form.tsx @@ -0,0 +1,120 @@ +import React from "react"; +import { useForm, Controller } from "react-hook-form"; +// ui +import { Input, Button } from "@plane/ui"; +// icons +import { XCircle } from "lucide-react"; +// services +import { AuthService } from "services/auth.service"; +const authService = new AuthService(); + +export interface InstanceSetupPasswordFormValues { + email: string; + password: string; +} + +export interface IInstanceSetupPasswordForm { + email: string; + onNextStep: () => void; + resetSteps: () => void; +} + +export const InstanceSetupPasswordForm: React.FC = (props) => { + const { onNextStep, email, resetSteps } = props; + // form info + const { + control, + handleSubmit, + formState: { errors, isSubmitting }, + } = useForm({ + defaultValues: { + email, + password: "", + }, + mode: "onChange", + reValidateMode: "onChange", + }); + + const handlePasswordSubmit = (formData: InstanceSetupPasswordFormValues) => + authService.setInstanceAdminPassword({ password: formData.password }).then(() => { + onNextStep(); + }); + + return ( +
+
+

+ Moving to the runway +

+

+ {"Let's set a password so you can do away with codes."} +

+ +
+ + /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/.test( + value + ) || "Email address is not valid", + }} + render={({ field: { value, onChange } }) => ( +
+ + resetSteps()} + /> +
+ )} + /> + +
+ ( +
+ +
+ )} + /> +
+

+ {"Whatever you choose now will be your account's password"} +

+ +
+
+
+ ); +}; diff --git a/web/components/instance/setup-form/root.tsx b/web/components/instance/setup-form/root.tsx new file mode 100644 index 000000000..ebc6c5649 --- /dev/null +++ b/web/components/instance/setup-form/root.tsx @@ -0,0 +1,70 @@ +import { useState } from "react"; +// components +import { InstanceSetupEmailCodeForm } from "./email-code-form"; +import { InstanceSetupEmailForm } from "./email-form"; +import { InstanceSetupPasswordForm } from "./password-form"; +import { LatestFeatureBlock } from "components/common"; +import { InstanceSetupDone } from "components/instance"; + +export enum EInstanceSetupSteps { + EMAIL = "EMAIL", + VERIFY_CODE = "VERIFY_CODE", + PASSWORD = "PASSWORD", + DONE = "DONE", +} + +export const InstanceSetupFormRoot = () => { + // states + const [setupStep, setSetupStep] = useState(EInstanceSetupSteps.EMAIL); + const [email, setEmail] = useState(""); + + return ( + <> + {setupStep === EInstanceSetupSteps.DONE ? ( +
+ +
+ ) : ( +
+
+
+ {setupStep === EInstanceSetupSteps.EMAIL && ( + { + setEmail(email); + setSetupStep(EInstanceSetupSteps.VERIFY_CODE); + }} + /> + )} + + {setupStep === EInstanceSetupSteps.VERIFY_CODE && ( + { + setSetupStep(EInstanceSetupSteps.PASSWORD); + }} + moveBack={() => { + setSetupStep(EInstanceSetupSteps.EMAIL); + }} + /> + )} + + {setupStep === EInstanceSetupSteps.PASSWORD && ( + { + setSetupStep(EInstanceSetupSteps.DONE); + }} + resetSteps={() => { + setSetupStep(EInstanceSetupSteps.EMAIL); + }} + /> + )} +
+ +
+
+ )} + + ); +}; diff --git a/web/components/instance/setup-view.tsx b/web/components/instance/setup-view.tsx new file mode 100644 index 000000000..07d4455dd --- /dev/null +++ b/web/components/instance/setup-view.tsx @@ -0,0 +1,38 @@ +import { useEffect, useCallback } from "react"; +import { observer } from "mobx-react-lite"; +import Image from "next/image"; +// components +import { InstanceSetupFormRoot } from "components/instance"; +// hooks +import { useMobxStore } from "lib/mobx/store-provider"; +// images +import BluePlaneLogoWithoutText from "public/plane-logos/blue-without-text.png"; + +export const InstanceSetupView = observer(() => { + // store + const { + user: { fetchCurrentUser }, + } = useMobxStore(); + + const mutateUserInfo = useCallback(() => { + fetchCurrentUser(); + }, [fetchCurrentUser]); + + useEffect(() => { + mutateUserInfo(); + }, [mutateUserInfo]); + + return ( + <> +
+
+
+ Plane Logo + Plane +
+
+ +
+ + ); +}); diff --git a/web/layouts/admin-layout/layout.tsx b/web/layouts/admin-layout/layout.tsx index cfe178e97..1a5c188eb 100644 --- a/web/layouts/admin-layout/layout.tsx +++ b/web/layouts/admin-layout/layout.tsx @@ -4,6 +4,9 @@ import { AdminAuthWrapper, UserAuthWrapper } from "layouts/auth-layout"; // components import { InstanceAdminSidebar } from "./sidebar"; import { InstanceAdminHeader } from "./header"; +import { InstanceSetupView } from "components/instance"; +// store +import { useMobxStore } from "lib/mobx/store-provider"; export interface IInstanceAdminLayout { children: ReactNode; @@ -11,6 +14,18 @@ export interface IInstanceAdminLayout { export const InstanceAdminLayout: FC = (props) => { const { children } = props; + // store + const { + instance: { instance }, + user: { currentUser }, + } = useMobxStore(); + // fetch + + console.log("instance", instance); + + if (instance?.is_setup_done === false) { + return ; + } return ( <> diff --git a/web/layouts/instance-layout/index.tsx b/web/layouts/instance-layout/index.tsx new file mode 100644 index 000000000..75aa374c1 --- /dev/null +++ b/web/layouts/instance-layout/index.tsx @@ -0,0 +1,52 @@ +import { FC, ReactNode, useEffect } from "react"; + +import useSWR from "swr"; + +// route +import { useRouter } from "next/router"; +// store +import { observer } from "mobx-react-lite"; +import { useMobxStore } from "lib/mobx/store-provider"; +// components +import { Spinner } from "@plane/ui"; +import { InstanceNotReady } from "components/instance"; + +type Props = { + children: ReactNode; +}; + +const InstanceLayout: FC = observer(({ children }) => { + // store + const { + instance: { fetchInstanceInfo, instance, createInstance }, + } = useMobxStore(); + + const router = useRouter(); + const isGodMode = router.pathname.includes("god-mode"); + + useSWR("INSTANCE_INFO", () => fetchInstanceInfo()); + + useEffect(() => { + if (instance?.is_activated === false) { + createInstance(); + } + }, [instance?.is_activated, createInstance]); + + return ( +
+ {instance ? ( + !instance.is_setup_done && !isGodMode ? ( + + ) : ( + children + ) + ) : ( +
+ +
+ )} +
+ ); +}); + +export default InstanceLayout; diff --git a/web/layouts/settings-layout/profile/sidebar.tsx b/web/layouts/settings-layout/profile/sidebar.tsx index 513bf72b8..487c53078 100644 --- a/web/layouts/settings-layout/profile/sidebar.tsx +++ b/web/layouts/settings-layout/profile/sidebar.tsx @@ -102,21 +102,20 @@ export const ProfileLayoutSidebar = observer(() => { } ${sidebarCollapsed ? "left-0" : "-left-full md:left-0"}`} >
-
- - + + + - - - - {!sidebarCollapsed && ( -

Profile settings

- )} -
+ + {!sidebarCollapsed && ( +

Profile settings

+ )} + +
{!sidebarCollapsed && (
Your account
diff --git a/web/lib/app-provider.tsx b/web/lib/app-provider.tsx index 3a822401c..ba882f174 100644 --- a/web/lib/app-provider.tsx +++ b/web/lib/app-provider.tsx @@ -3,8 +3,15 @@ import dynamic from "next/dynamic"; import Router from "next/router"; import NProgress from "nprogress"; import { observer } from "mobx-react-lite"; +import { ThemeProvider } from "next-themes"; // mobx store provider import { useMobxStore } from "lib/mobx/store-provider"; +// constants +import { THEMES } from "constants/themes"; +// layouts +import InstanceLayout from "layouts/instance-layout"; +// contexts +import { ToastContextProvider } from "contexts/toast.context"; // dynamic imports const StoreWrapper = dynamic(() => import("lib/wrappers/store-wrapper"), { ssr: false }); const PosthogWrapper = dynamic(() => import("lib/wrappers/posthog-wrapper"), { ssr: false }); @@ -29,18 +36,24 @@ export const AppProvider: FC = observer((props) => { } = useMobxStore(); return ( - - - - {children} - - - + + + + + + + {children} + + + + + + ); }); diff --git a/web/pages/_app.tsx b/web/pages/_app.tsx index 9f2cf995b..c6fb89772 100644 --- a/web/pages/_app.tsx +++ b/web/pages/_app.tsx @@ -1,7 +1,6 @@ import { ReactElement } from "react"; import Head from "next/head"; import { AppProps } from "next/app"; -import { ThemeProvider } from "next-themes"; // styles import "styles/globals.css"; import "styles/editor.css"; @@ -9,10 +8,7 @@ import "styles/table.css"; import "styles/command-pallette.css"; import "styles/nprogress.css"; import "styles/react-datepicker.css"; -// contexts -import { ToastContextProvider } from "contexts/toast.context"; // constants -import { THEMES } from "constants/themes"; import { SITE_TITLE } from "constants/seo-variables"; // mobx store provider import { MobxStoreProvider } from "lib/mobx/store-provider"; @@ -34,11 +30,7 @@ function MyApp({ Component, pageProps }: AppPropsWithLayout) { {SITE_TITLE} - - - {getLayout()} - - + {getLayout()} ); diff --git a/web/public/instance-not-ready.svg b/web/public/instance-not-ready.svg new file mode 100644 index 000000000..393187bdb --- /dev/null +++ b/web/public/instance-not-ready.svg @@ -0,0 +1,57 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/web/public/instance-setup-done.svg b/web/public/instance-setup-done.svg new file mode 100644 index 000000000..3474f7f38 --- /dev/null +++ b/web/public/instance-setup-done.svg @@ -0,0 +1,60 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/web/public/instance/plane-instance-not-ready.webp b/web/public/instance/plane-instance-not-ready.webp new file mode 100644 index 0000000000000000000000000000000000000000..a0efca52c85b06c27136c032b6d84adff96f05fc GIT binary patch literal 45894 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 bT!XEBss9Hmi^{f7kbk~e{_C>(ziR&j&=8$T literal 0 HcmV?d00001 diff --git a/web/services/auth.service.ts b/web/services/auth.service.ts index 76d2cedc8..43511d33d 100644 --- a/web/services/auth.service.ts +++ b/web/services/auth.service.ts @@ -80,6 +80,18 @@ export class AuthService extends APIService { }); } + async setInstanceAdminPassword(data: any): Promise { + return this.post("/api/licenses/instances/admins/set-password/", data) + .then((response) => { + this.setAccessToken(response?.data?.access_token); + this.setRefreshToken(response?.data?.refresh_token); + return response?.data; + }) + .catch((error) => { + throw error?.response?.data; + }); + } + async socialAuth(data: any): Promise { return this.post("/api/social-auth/", data, { headers: {} }) .then((response) => { @@ -100,6 +112,14 @@ export class AuthService extends APIService { }); } + async instanceAdminEmailCode(data: any): Promise { + return this.post("/api/licenses/instances/admins/magic-generate/", data, { headers: {} }) + .then((response) => response?.data) + .catch((error) => { + throw error?.response?.data; + }); + } + async magicSignIn(data: IMagicSignInData): Promise { return await this.post("/api/magic-sign-in/", data, { headers: {} }) .then((response) => { @@ -114,6 +134,16 @@ export class AuthService extends APIService { }); } + async instanceMagicSignIn(data: any): Promise { + const response = await this.post("/api/licenses/instances/admins/magic-sign-in/", data); + if (response?.status === 200) { + this.setAccessToken(response?.data?.access_token); + this.setRefreshToken(response?.data?.refresh_token); + return response?.data; + } + throw response.response.data; + } + async signOut(): Promise { return this.post("/api/sign-out/", { refresh_token: this.getRefreshToken() }) .then((response) => { diff --git a/web/services/instance.service.ts b/web/services/instance.service.ts index 6c1eb3556..4be5311bc 100644 --- a/web/services/instance.service.ts +++ b/web/services/instance.service.ts @@ -15,7 +15,15 @@ export class InstanceService extends APIService { } async getInstanceInfo(): Promise { - return this.get("/api/licenses/instances/") + return this.get("/api/licenses/instances/", { headers: {} }) + .then((response) => response.data) + .catch((error) => { + throw error; + }); + } + + async createInstance(): Promise { + return this.post("/api/licenses/instances/", {}, { headers: {} }) .then((response) => response.data) .catch((error) => { throw error; diff --git a/web/store/instance/instance.store.ts b/web/store/instance/instance.store.ts index 644128e68..81db85ae8 100644 --- a/web/store/instance/instance.store.ts +++ b/web/store/instance/instance.store.ts @@ -17,6 +17,7 @@ export interface IInstanceStore { formattedConfig: IFormattedInstanceConfiguration | null; // action fetchInstanceInfo: () => Promise; + createInstance: () => Promise; fetchInstanceAdmins: () => Promise; updateInstanceInfo: (data: Partial) => Promise; fetchInstanceConfigurations: () => Promise; @@ -45,6 +46,7 @@ export class InstanceStore implements IInstanceStore { formattedConfig: computed, // actions fetchInstanceInfo: action, + createInstance: action, fetchInstanceAdmins: action, updateInstanceInfo: action, fetchInstanceConfigurations: action, @@ -84,6 +86,22 @@ export class InstanceStore implements IInstanceStore { } }; + /** + * Creating new Instance In case of no instance found + */ + createInstance = async () => { + try { + const instance = await this.instanceService.createInstance(); + runInAction(() => { + this.instance = instance; + }); + return instance; + } catch (error) { + console.log("Error while creating the instance"); + throw error; + } + }; + /** * fetch instance admins from API */ diff --git a/web/types/instance.d.ts b/web/types/instance.d.ts index b97f9cd7e..e11b6add8 100644 --- a/web/types/instance.d.ts +++ b/web/types/instance.d.ts @@ -16,6 +16,8 @@ export interface IInstance { is_support_required: boolean; created_by: string | null; updated_by: string | null; + is_activated: boolean; + is_setup_done: boolean; } export interface IInstanceConfiguration { diff --git a/web/types/users.d.ts b/web/types/users.d.ts index ba5527528..0d81ba2a7 100644 --- a/web/types/users.d.ts +++ b/web/types/users.d.ts @@ -16,6 +16,7 @@ export interface IUser { is_onboarded: boolean; is_password_autoset: boolean; is_tour_completed: boolean; + is_password_autoset: boolean; mobile_number: string | null; role: string | null; onboarding_step: {