diff --git a/packages/frontend/.env b/packages/frontend/.env
new file mode 100644
index 0000000..5b03837
--- /dev/null
+++ b/packages/frontend/.env
@@ -0,0 +1 @@
+REACT_APP_GQL_SERVER_URL = 'http://localhost:8000/graphql'
diff --git a/packages/frontend/.eslintrc.json b/packages/frontend/.eslintrc.json
index f57b9c1..96d5591 100644
--- a/packages/frontend/.eslintrc.json
+++ b/packages/frontend/.eslintrc.json
@@ -17,5 +17,7 @@
"plugin:@typescript-eslint/recommended",
"plugin:prettier/recommended"
],
- "rules": {}
+ "rules": {
+ "@typescript-eslint/no-explicit-any": "off"
+ }
}
diff --git a/packages/frontend/package.json b/packages/frontend/package.json
index fedec4b..bcefa1e 100644
--- a/packages/frontend/package.json
+++ b/packages/frontend/package.json
@@ -11,6 +11,7 @@
"@types/node": "^16.18.68",
"@types/react": "^18.2.42",
"@types/react-dom": "^18.2.17",
+ "assert": "^2.1.0",
"date-fns": "^3.0.1",
"downshift": "^8.2.3",
"luxon": "^3.4.4",
diff --git a/packages/frontend/src/context/GQLClientContext.tsx b/packages/frontend/src/context/GQLClientContext.tsx
new file mode 100644
index 0000000..d71db3d
--- /dev/null
+++ b/packages/frontend/src/context/GQLClientContext.tsx
@@ -0,0 +1,24 @@
+import React, { createContext, useContext, ReactNode } from 'react';
+import { GQLClient } from 'gql-client';
+
+const GQLClientContext = createContext({} as GQLClient);
+
+export const GQLClientProvider = ({
+ client,
+ children,
+}: {
+ children: ReactNode;
+ client: GQLClient;
+}) => (
+
+ {children}
+
+);
+
+export const useGQLClient = () => {
+ const client = useContext(GQLClientContext);
+ if (!client) {
+ throw new Error('useGQLClient must be used within a GQLClientProvider');
+ }
+ return client;
+};
diff --git a/packages/frontend/src/index.tsx b/packages/frontend/src/index.tsx
index af5a9a1..6317b83 100644
--- a/packages/frontend/src/index.tsx
+++ b/packages/frontend/src/index.tsx
@@ -1,21 +1,32 @@
import React from 'react';
import ReactDOM from 'react-dom/client';
+import assert from 'assert';
+import { Toaster } from 'react-hot-toast';
+import { GQLClient } from 'gql-client';
import { ThemeProvider } from '@material-tailwind/react';
import './index.css';
import App from './App';
import reportWebVitals from './reportWebVitals';
-import { Toaster } from 'react-hot-toast';
+import { GQLClientProvider } from './context/GQLClientContext';
const root = ReactDOM.createRoot(
document.getElementById('root') as HTMLElement,
);
+
+const gqlEndpoint = process.env.REACT_APP_GQL_SERVER_URL;
+assert(gqlEndpoint, 'GQL server URL not provided');
+
+const gqlClient = new GQLClient({ gqlEndpoint });
+
root.render(
-
-
+
+
+
+
,
);
diff --git a/packages/frontend/src/pages/index.tsx b/packages/frontend/src/pages/index.tsx
index 204b7a8..5045a83 100644
--- a/packages/frontend/src/pages/index.tsx
+++ b/packages/frontend/src/pages/index.tsx
@@ -1,12 +1,49 @@
-import React from 'react';
+import React, { useEffect, useState } from 'react';
import { Link } from 'react-router-dom';
import { Button, Typography, Chip } from '@material-tailwind/react';
import ProjectCard from '../components/projects/ProjectCard';
import projectsDetail from '../assets/projects.json';
+import { useGQLClient } from '../context/GQLClientContext';
const Projects = () => {
+ const client = useGQLClient();
+ const [projects, setProjects] = useState([]);
+
+ useEffect(() => {
+ const fetchOrganization = async () => {
+ const res = await client.getOrganizations();
+
+ // Note: select first organization as organization switching not yet implemented
+ const projects = res.organizations[0].projects;
+
+ const updatedProjects = projects.map((project: any) => {
+ return {
+ ...project,
+ // TODO: populate empty fields
+ icon: '',
+ title: '',
+ organization: '',
+ url: '',
+ domain: null,
+ createdBy: '',
+ source: '',
+ // TODO: populate from github API
+ latestCommit: {
+ message: '',
+ createdAt: '',
+ branch: '',
+ },
+ };
+ });
+
+ setProjects(updatedProjects);
+ };
+
+ fetchOrganization();
+ }, [client]);
+
return (
@@ -29,9 +66,10 @@ const Projects = () => {
- {projectsDetail.map((project, key) => {
- return
;
- })}
+ {projects.length !== 0 &&
+ projects.map((project, key) => {
+ return
;
+ })}
);
diff --git a/packages/gql-client/src/gql-client.ts b/packages/gql-client/src/gql-client.ts
index 6b0856e..b2f3627 100644
--- a/packages/gql-client/src/gql-client.ts
+++ b/packages/gql-client/src/gql-client.ts
@@ -1,6 +1,6 @@
import { ApolloClient, InMemoryCache, NormalizedCacheObject } from '@apollo/client';
-import { getUser } from './gql-queries';
+import { getUser, getOrganizations } from './gql-queries';
export interface GraphQLConfig {
gqlEndpoint: string;
@@ -23,4 +23,12 @@ export class GQLClient {
return data;
}
+
+ async getOrganizations () : Promise {
+ const { data } = await this.client.query({
+ query: getOrganizations
+ });
+
+ return data;
+ }
}
diff --git a/packages/gql-client/src/gql-queries.ts b/packages/gql-client/src/gql-queries.ts
index 93b1a31..565d795 100644
--- a/packages/gql-client/src/gql-queries.ts
+++ b/packages/gql-client/src/gql-queries.ts
@@ -11,3 +11,34 @@ query {
}
}
`;
+
+export const getOrganizations = gql`
+query {
+ organizations {
+ projects {
+ id
+ owner {
+ id
+ }
+ deployments {
+ id
+ }
+ name
+ repository
+ prodBranch
+ description
+ template
+ framework
+ webhooks
+ members {
+ id
+ }
+ environmentVariables {
+ id
+ }
+ createdAt
+ updatedAt
+ }
+ }
+}
+`;
diff --git a/yarn.lock b/yarn.lock
index 8e0d40f..f9b1a8b 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -4226,6 +4226,17 @@ asap@~2.0.6:
resolved "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz"
integrity sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==
+assert@^2.1.0:
+ version "2.1.0"
+ resolved "https://registry.yarnpkg.com/assert/-/assert-2.1.0.tgz#6d92a238d05dc02e7427c881fb8be81c8448b2dd"
+ integrity sha512-eLHpSK/Y4nhMJ07gDaAzoX/XAKS8PSaojml3M0DM4JpV1LAi5JOJ/p6H/XWrl8L+DzVEvVCW1z3vWAaB9oTsQw==
+ dependencies:
+ call-bind "^1.0.2"
+ is-nan "^1.3.2"
+ object-is "^1.1.5"
+ object.assign "^4.1.4"
+ util "^0.12.5"
+
ast-types-flow@^0.0.8:
version "0.0.8"
resolved "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.8.tgz"
@@ -7927,7 +7938,7 @@ ipaddr.js@^2.0.1:
resolved "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-2.1.0.tgz"
integrity sha512-LlbxQ7xKzfBusov6UMi4MFpEg0m+mAm9xyNGEduwXMEDuf4WfzB/RZwMVYEd7IKGvh4IUkEXYxtAVu9T3OelJQ==
-is-arguments@^1.1.1:
+is-arguments@^1.0.4, is-arguments@^1.1.1:
version "1.1.1"
resolved "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz"
integrity sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==
@@ -8043,7 +8054,7 @@ is-generator-fn@^2.0.0:
resolved "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz"
integrity sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==
-is-generator-function@^1.0.10:
+is-generator-function@^1.0.10, is-generator-function@^1.0.7:
version "1.0.10"
resolved "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz"
integrity sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==
@@ -8084,6 +8095,14 @@ is-module@^1.0.0:
resolved "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz"
integrity sha512-51ypPSPCoTEIN9dy5Oy+h4pShgJmPCygKfyRCISBI+JoWT/2oJvK8QPxmwv7b/p239jXrm9M1mlQbyKJ5A152g==
+is-nan@^1.3.2:
+ version "1.3.2"
+ resolved "https://registry.yarnpkg.com/is-nan/-/is-nan-1.3.2.tgz#043a54adea31748b55b6cd4e09aadafa69bd9e1d"
+ integrity sha512-E+zBKpQ2t6MEo1VsonYmluk9NxGrbzpeeLC2xIViuO2EjU2xsXsBPwTr3Ykv9l08UYEVEdWeRZNouaZqF6RN0w==
+ dependencies:
+ call-bind "^1.0.0"
+ define-properties "^1.1.3"
+
is-negative-zero@^2.0.2:
version "2.0.2"
resolved "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz"
@@ -8211,7 +8230,7 @@ is-text-path@^1.0.1:
dependencies:
text-extensions "^1.0.0"
-is-typed-array@^1.1.10, is-typed-array@^1.1.12, is-typed-array@^1.1.9:
+is-typed-array@^1.1.10, is-typed-array@^1.1.12, is-typed-array@^1.1.3, is-typed-array@^1.1.9:
version "1.1.12"
resolved "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.12.tgz"
integrity sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==
@@ -13592,6 +13611,17 @@ util.promisify@~1.0.0:
has-symbols "^1.0.1"
object.getownpropertydescriptors "^2.1.0"
+util@^0.12.5:
+ version "0.12.5"
+ resolved "https://registry.yarnpkg.com/util/-/util-0.12.5.tgz#5f17a6059b73db61a875668781a1c2b136bd6fbc"
+ integrity sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==
+ dependencies:
+ inherits "^2.0.3"
+ is-arguments "^1.0.4"
+ is-generator-function "^1.0.7"
+ is-typed-array "^1.1.3"
+ which-typed-array "^1.1.2"
+
utila@~0.4:
version "0.4.0"
resolved "https://registry.npmjs.org/utila/-/utila-0.4.0.tgz"
@@ -13950,7 +13980,7 @@ which-collection@^1.0.1:
is-weakmap "^2.0.1"
is-weakset "^2.0.1"
-which-typed-array@^1.1.11, which-typed-array@^1.1.13, which-typed-array@^1.1.9:
+which-typed-array@^1.1.11, which-typed-array@^1.1.13, which-typed-array@^1.1.2, which-typed-array@^1.1.9:
version "1.1.13"
resolved "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.13.tgz"
integrity sha512-P5Nra0qjSncduVPEAr7xhoF5guty49ArDTwzJ/yNuPIbZppyRxFQsRCWrocxIY+CnMVG+qfbU2FmDKyvSGClow==