+
{
width: window.innerWidth * 0.9,
}}
>
-
+
Total Requested Attestations
-
+
{
bottom: 5,
}}
>
-
-
+
+
-
+
@@ -114,13 +99,10 @@ const Dashboard = () => {
position: 'relative',
}}
>
-
+
Total Attestations not Approved
-
+
{kpi.attestationsNotApproved}
@@ -133,13 +115,10 @@ const Dashboard = () => {
position: 'relative',
}}
>
-
+
Total Attestations revoked
-
+
{kpi.attestationsRevoked}
@@ -150,13 +129,10 @@ const Dashboard = () => {
position: 'relative',
}}
>
-
+
Total Claimers
-
+
{kpi.totalClaimers}
diff --git a/frontend/src/components/index.ts b/frontend/src/components/index.ts
new file mode 100644
index 0000000..b307d2c
--- /dev/null
+++ b/frontend/src/components/index.ts
@@ -0,0 +1,5 @@
+import Dashboard from './Dashboard'
+import AttestationList from './AttestationList'
+import AttestationCreate from './AttestationAdd'
+
+export { Dashboard, AttestationList, AttestationCreate }
diff --git a/frontend/src/index.tsx b/frontend/src/index.tsx
index 42abbd1..dff2a06 100644
--- a/frontend/src/index.tsx
+++ b/frontend/src/index.tsx
@@ -1,9 +1,10 @@
-import React from "react"
-import ReactDOM from "react-dom/client"
-import { App } from "./App"
+/* eslint-disable @typescript-eslint/no-non-null-assertion */
+import React from 'react'
+import ReactDOM from 'react-dom/client'
+import { App } from './App'
-ReactDOM.createRoot(document.getElementById("root")!).render(
+ReactDOM.createRoot(document.getElementById('root')!).render(
-
+ ,
)
diff --git a/frontend/src/layout/AppBar.tsx b/frontend/src/layout/AppBar.tsx
index 0f92d50..f26467f 100644
--- a/frontend/src/layout/AppBar.tsx
+++ b/frontend/src/layout/AppBar.tsx
@@ -1,17 +1,13 @@
import { AppBar, TitlePortal } from 'react-admin'
import { Box, useMediaQuery, Theme } from '@mui/material'
-const CustomAppBar = () => {
- const isLargeEnough = useMediaQuery((theme) =>
- theme.breakpoints.up('sm')
- )
+export default function CustomAppBar() {
+ const isLargeEnough = useMediaQuery((theme) => theme.breakpoints.up('sm'))
return (
-
+
- {isLargeEnough && }
+ {isLargeEnough && }
)
}
-
-export default CustomAppBar
diff --git a/frontend/src/layout/Layout.tsx b/frontend/src/layout/Layout.tsx
index f1b41b7..5eeda47 100644
--- a/frontend/src/layout/Layout.tsx
+++ b/frontend/src/layout/Layout.tsx
@@ -1,8 +1,6 @@
-import { Layout, LayoutProps } from "react-admin"
-import AppBar from "./AppBar"
-import Menu from "./Menu"
+import { Layout, LayoutProps } from 'react-admin'
+import AppBar from './AppBar'
+import Menu from './Menu'
// eslint-disable-next-line react/display-name
-export default (props: LayoutProps) => (
-
-)
+export default (props: LayoutProps) =>
diff --git a/frontend/src/layout/Login.tsx b/frontend/src/layout/Login.tsx
index 8e93d50..6e6da63 100644
--- a/frontend/src/layout/Login.tsx
+++ b/frontend/src/layout/Login.tsx
@@ -5,7 +5,7 @@ import { Utils } from '@kiltprotocol/sdk-js'
import Box from '@mui/material/Box'
import authProvider from '../api/authProvider'
-const Login = () => {
+export default function Login() {
const handleSubmit = useCallback((clientId: string) => {
const nonce = Utils.UUID.generate()
const state = Utils.UUID.generate()
@@ -13,7 +13,7 @@ const Login = () => {
const url = new URL(import.meta.env.VITE_AUTH_URL)
url.searchParams.append('response_type', 'id_token')
url.searchParams.append('client_id', clientId as string)
- url.searchParams.append('redirect_uri', window.location.origin + '/#/login')
+ url.searchParams.append('redirect_uri', `${window.location.origin}/#/login`)
url.searchParams.append('scope', 'openid')
url.searchParams.append('state', state)
url.searchParams.append('nonce', nonce)
@@ -64,19 +64,19 @@ const Login = () => {
)
}
-
-export default Login
diff --git a/frontend/src/layout/Menu.tsx b/frontend/src/layout/Menu.tsx
index d52d682..6bc0442 100644
--- a/frontend/src/layout/Menu.tsx
+++ b/frontend/src/layout/Menu.tsx
@@ -1,8 +1,8 @@
-import Box from "@mui/material/Box"
-import AppsIcon from "@mui/icons-material/Apps"
-import { DashboardMenuItem, useSidebarState, MenuItemLink } from "react-admin"
+import Box from '@mui/material/Box'
+import AppsIcon from '@mui/icons-material/Apps'
+import { DashboardMenuItem, useSidebarState, MenuItemLink } from 'react-admin'
-const Menu = () => {
+export default function Menu() {
const [open] = useSidebarState()
return (
@@ -12,7 +12,7 @@ const Menu = () => {
marginTop: 1,
marginBottom: 1,
transition: (theme) =>
- theme.transitions.create("width", {
+ theme.transitions.create('width', {
easing: theme.transitions.easing.sharp,
duration: theme.transitions.duration.leavingScreen,
}),
@@ -22,11 +22,9 @@ const Menu = () => {
}
/>
)
}
-
-export default Menu
diff --git a/frontend/src/layout/SubMenu.tsx b/frontend/src/layout/SubMenu.tsx
index 98c8b2f..4188e0d 100644
--- a/frontend/src/layout/SubMenu.tsx
+++ b/frontend/src/layout/SubMenu.tsx
@@ -1,25 +1,18 @@
-import { ReactElement, ReactNode } from "react"
-import {
- List,
- MenuItem,
- ListItemIcon,
- Typography,
- Collapse,
- Tooltip,
-} from "@mui/material"
-import ExpandMore from "@mui/icons-material/ExpandMore"
-import { useTranslate, useSidebarState } from "react-admin"
+import { ReactElement, ReactNode } from 'react'
+import { List, MenuItem, ListItemIcon, Typography, Collapse, Tooltip } from '@mui/material'
+import ExpandMore from '@mui/icons-material/ExpandMore'
+import { useTranslate, useSidebarState } from 'react-admin'
interface Props {
- dense: boolean;
- handleToggle: () => void;
- icon: ReactElement;
- isOpen: boolean;
- name: string;
- children: ReactNode;
+ dense: boolean
+ handleToggle: () => void
+ icon: ReactElement
+ isOpen: boolean
+ name: string
+ children: ReactNode
}
-const SubMenu = (props: Props) => {
+export default function SubMenu(props: Props) {
const { handleToggle, isOpen, name, icon, children, dense } = props
const translate = useTranslate()
@@ -27,9 +20,7 @@ const SubMenu = (props: Props) => {
const header = (
)
}
-
-export default SubMenu
diff --git a/frontend/src/layout/index.ts b/frontend/src/layout/index.ts
index e39d19f..3cfd284 100644
--- a/frontend/src/layout/index.ts
+++ b/frontend/src/layout/index.ts
@@ -1,6 +1,6 @@
-import AppBar from "./AppBar"
-import Layout from "./Layout"
-import Login from "./Login"
-import Menu from "./Menu"
+import AppBar from './AppBar'
+import Layout from './Layout'
+import Login from './Login'
+import Menu from './Menu'
export { AppBar, Layout, Login, Menu }
diff --git a/frontend/src/layout/themes.ts b/frontend/src/layout/themes.ts
index 478331a..b04ad72 100644
--- a/frontend/src/layout/themes.ts
+++ b/frontend/src/layout/themes.ts
@@ -1,14 +1,14 @@
-import { defaultTheme } from "react-admin"
+import { defaultTheme } from 'react-admin'
export const darkTheme = {
palette: {
primary: {
- main: "#90caf9",
+ main: '#90caf9',
},
secondary: {
- main: "#FBBA72",
+ main: '#FBBA72',
},
- mode: "dark" as const, // Switching the dark mode on is a single property value change.
+ mode: 'dark' as const, // Switching the dark mode on is a single property value change.
},
sidebar: {
width: 200,
@@ -18,9 +18,9 @@ export const darkTheme = {
RaMenuItemLink: {
styleOverrides: {
root: {
- borderLeft: "3px solid #000",
- "&.RaMenuItemLink-active": {
- borderLeft: "3px solid #90caf9",
+ borderLeft: '3px solid #000',
+ '&.RaMenuItemLink-active': {
+ borderLeft: '3px solid #90caf9',
},
},
},
@@ -28,8 +28,8 @@ export const darkTheme = {
MuiAppBar: {
styleOverrides: {
colorSecondary: {
- color: "#ffffffb3",
- backgroundColor: "#616161e6",
+ color: '#ffffffb3',
+ backgroundColor: '#616161e6',
},
},
},
@@ -39,18 +39,18 @@ export const darkTheme = {
export const lightTheme = {
palette: {
primary: {
- main: "#4f3cc9",
+ main: '#4f3cc9',
},
secondary: {
- light: "#5f5fc4",
- main: "#283593",
- dark: "#001064",
- contrastText: "#fff",
+ light: '#5f5fc4',
+ main: '#283593',
+ dark: '#001064',
+ contrastText: '#fff',
},
background: {
- default: "#fcfcfe",
+ default: '#fcfcfe',
},
- mode: "light" as const,
+ mode: 'light' as const,
},
shape: {
borderRadius: 10,
@@ -63,9 +63,9 @@ export const lightTheme = {
RaMenuItemLink: {
styleOverrides: {
root: {
- borderLeft: "3px solid #fff",
- "&.RaMenuItemLink-active": {
- borderLeft: "3px solid #4f3cc9",
+ borderLeft: '3px solid #fff',
+ '&.RaMenuItemLink-active': {
+ borderLeft: '3px solid #4f3cc9',
},
},
},
@@ -73,36 +73,36 @@ export const lightTheme = {
MuiPaper: {
styleOverrides: {
elevation1: {
- boxShadow: "none",
+ boxShadow: 'none',
},
root: {
- border: "1px solid #e0e0e3",
- backgroundClip: "padding-box",
+ border: '1px solid #e0e0e3',
+ backgroundClip: 'padding-box',
},
},
},
MuiAppBar: {
styleOverrides: {
colorSecondary: {
- color: "#808080",
- backgroundColor: "#fff",
+ color: '#808080',
+ backgroundColor: '#fff',
},
},
},
MuiLinearProgress: {
styleOverrides: {
colorPrimary: {
- backgroundColor: "#f5f5f5",
+ backgroundColor: '#f5f5f5',
},
barColorPrimary: {
- backgroundColor: "#d7d7d7",
+ backgroundColor: '#d7d7d7',
},
},
},
MuiTableRow: {
styleOverrides: {
root: {
- "&:last-child td": { border: 0 },
+ '&:last-child td': { border: 0 },
},
},
},
diff --git a/frontend/src/session.ts b/frontend/src/session.ts
deleted file mode 100644
index 6e016c3..0000000
--- a/frontend/src/session.ts
+++ /dev/null
@@ -1,135 +0,0 @@
-import { DidUri } from "@kiltprotocol/sdk-js";
-import { useState, useEffect } from "react";
-import { getAxiosClient } from "./api/dataProvider";
-
-interface EncryptedMessage {
- receiverKeyUri: string;
- senderKeyUri: string;
- ciphertext: string;
- nonce: string;
- receivedAt?: number;
-}
-
-interface PubSubSession {
- listen: (
- callback: (message: EncryptedMessage) => Promise
- ) => Promise;
- close: () => Promise;
- send: (message: EncryptedMessage) => Promise;
- encryptionKeyUri: string;
- encryptedChallenge: string;
- nonce: string;
-}
-
-export interface InjectedWindowProvider {
- startSession: (
- dAppName: string,
- dAppEncryptionKeyUri: string,
- challenge: string
- ) => Promise;
- name: string;
- version: string;
- specVersion: "3.0";
- signWithDid: (
- data: string,
- didKeyUri: DidUri
- ) => Promise<{ didKeyUri: string; signature: string }>;
- getDidList: () => Promise>;
-}
-
-export const apiWindow = window as unknown as {
- kilt: Record;
-};
-
-export function useCompatibleExtensions() {
- const [extensions, setExtensions] = useState(getCompatibleExtensions());
- useEffect(() => {
- function handler() {
- setExtensions(getCompatibleExtensions());
- }
- window.dispatchEvent(new CustomEvent("kilt-dapp#initialized"));
- window.addEventListener("kilt-extension#initialized", handler);
- return () =>
- window.removeEventListener("kilt-extension#initialized", handler);
- }, []);
-
- return { extensions };
-}
-
-export function getCompatibleExtensions(): Array {
- return Object.entries(apiWindow.kilt)
- .filter(([, provider]) => provider.specVersion.startsWith("3."))
- .map(([name]) => name);
-}
-
-export async function requestAttestation(
- provider: InjectedWindowProvider,
- attestationId: string
-): Promise {
- if (!provider) {
- throw new Error("No provider");
- }
-
- const apiURL = import.meta.env.VITE_SIMPLE_REST_URL;
- const challengeUrl = apiURL + "/challenge";
-
-
- const client = await getAxiosClient();
-
- const get_challenge_response = await client.get(challengeUrl);
-
- if (get_challenge_response.status !== 200) {
- throw new Error("No valid challenge received");
- }
-
- const challenge = get_challenge_response.data;
- const session = await provider.startSession(
- challenge.dAppName,
- challenge.dAppEncryptionKeyUri,
- challenge.challenge
- );
-
- // post challenge and receive encrypted Message.
- const post_session_response = await client.post(challengeUrl, session);
-
- if (post_session_response.status !== 200) {
- throw new Error("No valid Session.");
- }
-
- const session_reference = post_session_response.data;
-
- const termsRequestData = {
- challenge: session_reference,
- attestationId,
- };
-
- const credentialUrl = apiURL + "/credential";
-
- const get_terms_response = await client.post(
- credentialUrl + "/terms/" + session_reference + "/" + attestationId,
- termsRequestData
- );
-
- const getCredentialRequestFromExtension = await new Promise(
- async (resolve, reject) => {
- try {
- await session.listen(async (credentialRequest) => {
- resolve(credentialRequest);
- });
- await session.send(get_terms_response.data);
- } catch (e) {
- reject(e);
- }
- }
- );
-
- let attestation_message = await client.post(
- credentialUrl + "/" + session_reference + "/" + attestationId,
- getCredentialRequestFromExtension
- );
-
- if (attestation_message.status !== 200) {
- throw new Error("No valid attestation message");
- }
-
-}
diff --git a/frontend/src/users.json b/frontend/src/users.json
deleted file mode 100644
index fc14c48..0000000
--- a/frontend/src/users.json
+++ /dev/null
@@ -1,18 +0,0 @@
-{
- "users": [
- {
- "id": 1,
- "username": "janedoe",
- "password": "password",
- "fullName": "Jane Doe",
- "avatar": ""
- },
- {
- "id": 2,
- "username": "johndoe",
- "password": "password",
- "fullName": "John Doe",
- "avatar": ""
- }
- ]
-}
diff --git a/frontend/src/utils/types.ts b/frontend/src/utils/types.ts
index 592cd11..3d8604e 100644
--- a/frontend/src/utils/types.ts
+++ b/frontend/src/utils/types.ts
@@ -4,7 +4,7 @@ import { UUID } from 'crypto'
export interface AttestationRequest {
approved: boolean
revoked: boolean
- marked_approve: boolean,
+ marked_approve: boolean
claimer: DidUri
createdAt: string
credential: ICredential
diff --git a/frontend/src/utils/utils.ts b/frontend/src/utils/utils.ts
index dcabb58..292fef5 100644
--- a/frontend/src/utils/utils.ts
+++ b/frontend/src/utils/utils.ts
@@ -1,14 +1,12 @@
-import { ICType, CType, connect } from "@kiltprotocol/sdk-js"
-import authProvider from "../api/authProvider"
+import { ICType, CType, connect } from '@kiltprotocol/sdk-js'
+import authProvider from '../api/authProvider'
-export async function fetchCType(
- ctypeId: ICType["$id"]
-): Promise {
+export async function fetchCType(ctypeId: ICType['$id']): Promise {
await connect(import.meta.env.VITE_WSS_ENDPOINT)
return CType.fetchFromChain(ctypeId)
}
export function isUserAdmin() {
const role = authProvider.getRole()
- return role === "admin"
+ return role === 'admin'
}