Compare commits
	
		
			17 Commits
		
	
	
		
			master
			...
			60d7d7fe5c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 60d7d7fe5c | |||
| 5f024f6028 | |||
| 5ea803c8a4 | |||
| b199361bf1 | |||
| 26a445000a | |||
|  | c51228eff9 | ||
|  | 6de1f73d47 | ||
|  | 0c1eda8299 | ||
|  | 4e493f79d1 | ||
|  | 3644718b19 | ||
|  | 2efbad4fe5 | ||
|  | de24645436 | ||
|  | 5350966785 | ||
|  | 8e8c93ef2c | ||
|  | edab87d3f1 | ||
|  | c6b09aa075 | ||
|  | 498e77e9a1 | 
							
								
								
									
										21
									
								
								.cracorc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.cracorc.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | const { ServiceRegister } = require("configuration"); | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  |   devServer: (devServerConfig) => { | ||||||
|  |     devServerConfig.port = "auto"; | ||||||
|  |     devServerConfig.onListening = function (devServer) { | ||||||
|  |       if (!devServer) { | ||||||
|  |         throw new Error("webpack-dev-server is not defined"); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const port = devServer.listeningApp.address().port; | ||||||
|  |       const register = new ServiceRegister({ | ||||||
|  |         etcd: { hosts: ["http://rpi:2379"] }, | ||||||
|  |       }); | ||||||
|  |       register.register("fennec", `http://localhost:${port}/`); | ||||||
|  |       console.log("Listening on port:", port); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     return devServerConfig; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -21,3 +21,5 @@ | |||||||
| npm-debug.log* | npm-debug.log* | ||||||
| yarn-debug.log* | yarn-debug.log* | ||||||
| yarn-error.log* | yarn-error.log* | ||||||
|  |  | ||||||
|  | .vscode/chrome | ||||||
							
								
								
									
										5
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -8,10 +8,9 @@ | |||||||
|       "name": "chrome", |       "name": "chrome", | ||||||
|       "type": "chrome", |       "type": "chrome", | ||||||
|       "request": "launch", |       "request": "launch", | ||||||
|       "reAttach": true, |       "url": "http://fennec.localhost:7070/", | ||||||
|       "url": "http://fennec.localhost/", |  | ||||||
|       "webRoot": "${workspaceFolder}", |       "webRoot": "${workspaceFolder}", | ||||||
|       "userDataDir": "/Users/ivan/Projects/.chrome" |       "userDataDir": ".vscode/chrome" | ||||||
|     } |     } | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| module.exports = { | module.exports = { | ||||||
|   client: { |   client: { | ||||||
|     service: { |     service: { | ||||||
|       name: 'fennec-be', |       name: "fennec-be", | ||||||
|       url: 'http://api.fennec.localhost/graphql' |       url: "http://localhost:7122/graphql", | ||||||
|     } |     }, | ||||||
|   } |   }, | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| overwrite: true | overwrite: true | ||||||
| schema: "http://api.fennec.localhost/graphql" | schema: "http://localhost:7122/graphql" | ||||||
| # documents: "src/**/*.graphql" | # documents: "src/**/*.graphql" | ||||||
| generates: | generates: | ||||||
|   src/generated/graphql.tsx: |   src/generated/graphql.tsx: | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| module.exports = { | module.exports = { | ||||||
|   apps: [ |   apps: [ | ||||||
|     { |     { | ||||||
|       name: "fennec-bs", |       name: "fennec-fe", | ||||||
|       script: "serve", |       script: "serve", | ||||||
|       args: "", |       args: "", | ||||||
|       watch: false, |       watch: false, | ||||||
| @@ -9,7 +9,7 @@ module.exports = { | |||||||
|       log_date_format: "MM-DD HH:mm:ss.SSS Z", |       log_date_format: "MM-DD HH:mm:ss.SSS Z", | ||||||
|       env: { |       env: { | ||||||
|         PM2_SERVE_PATH: "./build", |         PM2_SERVE_PATH: "./build", | ||||||
|         PM2_SERVE_PORT: 7135, |         PM2_SERVE_PORT: 7123, | ||||||
|         PM2_SERVE_SPA: "true", |         PM2_SERVE_SPA: "true", | ||||||
|         PM2_SERVE_HOMEPAGE: "/index.html", |         PM2_SERVE_HOMEPAGE: "/index.html", | ||||||
|       }, |       }, | ||||||
|   | |||||||
							
								
								
									
										24971
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										24971
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										24
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,20 +1,22 @@ | |||||||
| { | { | ||||||
|   "name": "fennec-bs", |   "name": "fennec-bs", | ||||||
|   "version": "0.1.1", |   "version": "0.2.0", | ||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@apollo/client": "^3.3.15", |     "@apollo/client": "^3.3.15", | ||||||
|  |     "@craco/craco": "^6.3.0", | ||||||
|     "@curi/react-dom": "^2.0.4", |     "@curi/react-dom": "^2.0.4", | ||||||
|     "@curi/router": "^2.1.2", |     "@curi/router": "^2.1.2", | ||||||
|     "@date-io/date-fns": "^1.3.13", |     "@emotion/react": "^11.4.1", | ||||||
|  |     "@emotion/styled": "^11.3.0", | ||||||
|     "@fortawesome/fontawesome-svg-core": "^1.2.35", |     "@fortawesome/fontawesome-svg-core": "^1.2.35", | ||||||
|     "@fortawesome/free-solid-svg-icons": "^5.15.3", |     "@fortawesome/free-solid-svg-icons": "^5.15.3", | ||||||
|     "@fortawesome/react-fontawesome": "^0.1.14", |     "@fortawesome/react-fontawesome": "^0.1.14", | ||||||
|     "@hickory/browser": "^2.1.0", |     "@hickory/browser": "^2.1.0", | ||||||
|     "@material-ui/core": "^4.11.3", |     "@mui/icons-material": "^5.0.1", | ||||||
|     "@material-ui/icons": "^4.11.2", |     "@mui/lab": "^5.0.0-alpha.49", | ||||||
|     "@material-ui/lab": "*", |     "@mui/material": "^5.0.2", | ||||||
|     "@material-ui/pickers": "^3.3.10", |     "@mui/styles": "^5.0.1", | ||||||
|     "@testing-library/jest-dom": "^5.11.10", |     "@testing-library/jest-dom": "^5.11.10", | ||||||
|     "@testing-library/react": "^11.2.6", |     "@testing-library/react": "^11.2.6", | ||||||
|     "@testing-library/user-event": "^12.8.3", |     "@testing-library/user-event": "^12.8.3", | ||||||
| @@ -23,13 +25,14 @@ | |||||||
|     "@types/react": "^17.0.3", |     "@types/react": "^17.0.3", | ||||||
|     "@types/react-dom": "^17.0.3", |     "@types/react-dom": "^17.0.3", | ||||||
|     "apollo-link-scalars": "^2.1.3", |     "apollo-link-scalars": "^2.1.3", | ||||||
|  |     "configuration": "file:../configuration", | ||||||
|     "date-fns": "^2.21.1", |     "date-fns": "^2.21.1", | ||||||
|     "fontsource-roboto": "^4.0.0", |     "fontsource-roboto": "^4.0.0", | ||||||
|     "formik": "^2.2.6", |     "formik": "^2.2.6", | ||||||
|     "formik-material-ui": "^3.0.1", |     "formik-material-ui": "^3.0.1", | ||||||
|     "formik-material-ui-pickers": "^0.0.12", |     "formik-material-ui-pickers": "^0.0.12", | ||||||
|     "graphql": "^15.5.0", |     "graphql": "^15.5.0", | ||||||
|     "graphql-scalars": "^1.9.3", |     "graphql-scalars": "^1.10.0", | ||||||
|     "material-ui-confirm": "^2.1.2", |     "material-ui-confirm": "^2.1.2", | ||||||
|     "notistack": "^1.0.6", |     "notistack": "^1.0.6", | ||||||
|     "ramda": "^0.27.1", |     "ramda": "^0.27.1", | ||||||
| @@ -42,10 +45,9 @@ | |||||||
|     "yup": "^0.32.9" |     "yup": "^0.32.9" | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "PORT=7123 BROWSER=none react-scripts start", |     "start": "BROWSER=none PORT=auto craco start", | ||||||
|     "build": "react-scripts build", |     "build": "craco build", | ||||||
|     "test": "react-scripts test", |     "test": "craco test", | ||||||
|     "eject": "react-scripts eject", |  | ||||||
|     "prestart": "npm run graphql", |     "prestart": "npm run graphql", | ||||||
|     "graphql": "graphql-codegen --config codegen.yml" |     "graphql": "graphql-codegen --config codegen.yml" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ | |||||||
|       work correctly both with client-side routing and a non-root public URL. |       work correctly both with client-side routing and a non-root public URL. | ||||||
|       Learn how to configure a non-root public URL by running `npm run build`. |       Learn how to configure a non-root public URL by running `npm run build`. | ||||||
|     --> |     --> | ||||||
|     <title>React App</title> |     <title>Fennec</title> | ||||||
|   </head> |   </head> | ||||||
|   <body> |   <body> | ||||||
|     <noscript>You need to enable JavaScript to run this app.</noscript> |     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||||
|   | |||||||
| @@ -13,10 +13,10 @@ import { | |||||||
|   ListItemIcon, |   ListItemIcon, | ||||||
|   ListItemSecondaryAction, |   ListItemSecondaryAction, | ||||||
|   ListItemText, |   ListItemText, | ||||||
|   makeStyles, |  | ||||||
|   useTheme, |   useTheme, | ||||||
|   withStyles, | } from "@mui/material"; | ||||||
| } from "@material-ui/core"; | import { makeStyles } from "@mui/styles"; | ||||||
|  | import withStyles from "@mui/styles/withStyles"; | ||||||
| import { | import { | ||||||
|   Cancel, |   Cancel, | ||||||
|   CheckCircle, |   CheckCircle, | ||||||
| @@ -24,7 +24,7 @@ import { | |||||||
|   ShoppingCart, |   ShoppingCart, | ||||||
|   Stop, |   Stop, | ||||||
|   Timer, |   Timer, | ||||||
| } from "@material-ui/icons"; | } from "@mui/icons-material"; | ||||||
| import { format } from "date-fns"; | import { format } from "date-fns"; | ||||||
| import { useSnackbar } from "notistack"; | import { useSnackbar } from "notistack"; | ||||||
| import { complement, equals, find, propEq, takeWhile } from "ramda"; | import { complement, equals, find, propEq, takeWhile } from "ramda"; | ||||||
| @@ -133,8 +133,7 @@ const Item: FC<{ commit: Commit; pipeline: Pipeline }> = ({ | |||||||
| }) => { | }) => { | ||||||
|   const [isOpen, setOpen] = useState(() => false); |   const [isOpen, setOpen] = useState(() => false); | ||||||
|  |  | ||||||
|   const [createTask, { loading }] = |   const [createTask, { loading }] = useMutation< | ||||||
|     useMutation< |  | ||||||
|     { createPipelineTask: PipelineTask }, |     { createPipelineTask: PipelineTask }, | ||||||
|     { task: CreatePipelineTaskInput } |     { task: CreatePipelineTaskInput } | ||||||
|   >(CREATE_PIPELINE_TASK); |   >(CREATE_PIPELINE_TASK); | ||||||
| @@ -181,6 +180,7 @@ const Item: FC<{ commit: Commit; pipeline: Pipeline }> = ({ | |||||||
|               aria-label={pair[2]} |               aria-label={pair[2]} | ||||||
|               disabled={loading} |               disabled={loading} | ||||||
|               onClick={() => handleCreateTask(unit)} |               onClick={() => handleCreateTask(unit)} | ||||||
|  |               size="large" | ||||||
|             > |             > | ||||||
|               {pair[1]} |               {pair[1]} | ||||||
|             </IconButton> |             </IconButton> | ||||||
| @@ -257,7 +257,7 @@ const TaskItem: FC<{ task: PipelineTask }> = ({ task }) => { | |||||||
|       /> |       /> | ||||||
|       <ListItemSecondaryAction> |       <ListItemSecondaryAction> | ||||||
|         {task.status === TaskStatuses.Working && ( |         {task.status === TaskStatuses.Working && ( | ||||||
|           <IconButton edge="end" aria-label="stop" onClick={stop}> |           <IconButton edge="end" aria-label="stop" onClick={stop} size="large"> | ||||||
|             <Stop /> |             <Stop /> | ||||||
|           </IconButton> |           </IconButton> | ||||||
|         )} |         )} | ||||||
|   | |||||||
							
								
								
									
										55
									
								
								src/commons/auth/auth.provider.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								src/commons/auth/auth.provider.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | import { createContext, useContext, useState } from "react"; | ||||||
|  | import { FC } from "react"; | ||||||
|  | import { Login } from "./login"; | ||||||
|  |  | ||||||
|  | export interface AuthContext { | ||||||
|  |   accessToken: string | null; | ||||||
|  |   setAccessToken: (token: string) => void; | ||||||
|  |   setRefreshToken: (token: string) => void; | ||||||
|  |   refreshToken: string | undefined; | ||||||
|  |   login: (dto: any) => void; | ||||||
|  |   account?: any; | ||||||
|  |   setAccount: (dto: any) => void; | ||||||
|  |   logout: () => void; | ||||||
|  | } | ||||||
|  | const Context = createContext({} as AuthContext); | ||||||
|  |  | ||||||
|  | export const useAuth = () => useContext(Context); | ||||||
|  |  | ||||||
|  | export const AuthProvider: FC = ({ children }) => { | ||||||
|  |   const [accessToken, setAccessToken] = useState<string | null>( | ||||||
|  |     localStorage.getItem("accessToken") | ||||||
|  |   ); | ||||||
|  |   const [refreshToken, setRefreshToken] = useState<string>(); | ||||||
|  |   const [account, setAccount] = useState<any>(); | ||||||
|  |  | ||||||
|  |   const login = (dto: any) => { | ||||||
|  |     setAccessToken(dto.accessToken); | ||||||
|  |     setRefreshToken(dto.refreshToken); | ||||||
|  |     setAccount(dto.account); | ||||||
|  |     localStorage.setItem("accessToken", dto.accessToken); | ||||||
|  |   }; | ||||||
|  |   const logout = () => { | ||||||
|  |     setAccessToken(null); | ||||||
|  |     setRefreshToken(undefined); | ||||||
|  |     setAccount(undefined); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <Context.Provider | ||||||
|  |       value={{ | ||||||
|  |         accessToken, | ||||||
|  |         setAccessToken, | ||||||
|  |         refreshToken, | ||||||
|  |         setRefreshToken, | ||||||
|  |         login, | ||||||
|  |         account, | ||||||
|  |         setAccount, | ||||||
|  |         logout, | ||||||
|  |       }} | ||||||
|  |     > | ||||||
|  |       {children} | ||||||
|  |       {accessToken ? null : <Login />} | ||||||
|  |     </Context.Provider> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										69
									
								
								src/commons/auth/login.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/commons/auth/login.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | import makeStyles from "@mui/styles/makeStyles"; | ||||||
|  | import { FC, Fragment, useEffect, useRef } from "react"; | ||||||
|  | import { useAuth } from "./auth.provider"; | ||||||
|  | const useStyles = makeStyles((theme) => { | ||||||
|  |   debugger; | ||||||
|  |   return { | ||||||
|  |     iframe: { | ||||||
|  |       height: "300px", | ||||||
|  |       width: "500px", | ||||||
|  |       position: "absolute", | ||||||
|  |       top: "100px", | ||||||
|  |       left: "50%", | ||||||
|  |       transform: "translateX(-50%)", | ||||||
|  |       zIndex: theme.zIndex.modal, | ||||||
|  |       border: "none", | ||||||
|  |       boxShadow: theme.shadows[4], | ||||||
|  |     }, | ||||||
|  |     mask: { | ||||||
|  |       top: "0", | ||||||
|  |       left: "0", | ||||||
|  |       bottom: "0", | ||||||
|  |       right: "0", | ||||||
|  |       position: "absolute", | ||||||
|  |       backgroundColor: "rgba(0, 0, 0, 0.3)", | ||||||
|  |       zIndex: theme.zIndex.modal, | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | export const Login: FC = () => { | ||||||
|  |   const iframeRef = useRef<HTMLIFrameElement>(null); | ||||||
|  |   const { login } = useAuth(); | ||||||
|  |   useEffect(() => { | ||||||
|  |     const iframe = iframeRef.current; | ||||||
|  |     if (!iframe) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     let messagePort: MessagePort; | ||||||
|  |     const onLoad = (ev: MessageEvent) => { | ||||||
|  |       if (ev.data !== "init-channel") { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       messagePort = ev.ports?.[0] as MessagePort; | ||||||
|  |       messagePort.onmessage = (ev: MessageEvent) => { | ||||||
|  |         if (ev.data?.type === "logged") { | ||||||
|  |           login(ev.data.payload); | ||||||
|  |         } | ||||||
|  |       }; | ||||||
|  |     }; | ||||||
|  |     window.addEventListener("message", onLoad); | ||||||
|  |  | ||||||
|  |     return () => { | ||||||
|  |       window.removeEventListener("message", onLoad); | ||||||
|  |     }; | ||||||
|  |   }, [login]); | ||||||
|  |  | ||||||
|  |   const classes = useStyles(); | ||||||
|  |   return ( | ||||||
|  |     <Fragment> | ||||||
|  |       <div className={classes.mask} /> | ||||||
|  |       <iframe | ||||||
|  |         ref={iframeRef} | ||||||
|  |         className={classes.iframe} | ||||||
|  |         title="Auth" | ||||||
|  |         src="https://user.rpi.ivanli.cc/auth/login" | ||||||
|  |       ></iframe> | ||||||
|  |     </Fragment> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { Typography } from '@material-ui/core'; | import { Typography } from '@mui/material'; | ||||||
| import React, { FC } from 'react'; | import React, { FC } from 'react'; | ||||||
|  |  | ||||||
| export const ErrorPage: FC = ({children}) => { | export const ErrorPage: FC = ({children}) => { | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ import { | |||||||
|   InMemoryCache, |   InMemoryCache, | ||||||
|   split, |   split, | ||||||
|   ApolloProvider, |   ApolloProvider, | ||||||
|  |   fromPromise, | ||||||
|  |   FetchResult, | ||||||
| } from "@apollo/client"; | } from "@apollo/client"; | ||||||
| import { withScalars } from "apollo-link-scalars"; | import { withScalars } from "apollo-link-scalars"; | ||||||
| import { buildClientSchema, IntrospectionQuery } from "graphql"; | import { buildClientSchema, IntrospectionQuery } from "graphql"; | ||||||
| @@ -13,9 +15,16 @@ import { FC } from "react"; | |||||||
| import introspectionResult from "../../generated/graphql.schema.json"; | import introspectionResult from "../../generated/graphql.schema.json"; | ||||||
| import { onError } from "@apollo/client/link/error"; | import { onError } from "@apollo/client/link/error"; | ||||||
| import { WebSocketLink } from "@apollo/client/link/ws"; | import { WebSocketLink } from "@apollo/client/link/ws"; | ||||||
| import { getMainDefinition } from "@apollo/client/utilities"; | import { getMainDefinition, Observable } from "@apollo/client/utilities"; | ||||||
| import { useSnackbar } from "notistack"; | import { useSnackbar } from "notistack"; | ||||||
| import { deepOmit } from "../../utils/deep-omit"; | import { deepOmit } from "../../utils/deep-omit"; | ||||||
|  | import { useMemo } from "react"; | ||||||
|  | import { EventEmitter } from "events"; | ||||||
|  | import { useState } from "react"; | ||||||
|  | import { useAuth } from "../auth/auth.provider"; | ||||||
|  | import { useEffect } from "react"; | ||||||
|  | import { useRef } from "react"; | ||||||
|  | import { setContext } from "@apollo/client/link/context"; | ||||||
|  |  | ||||||
| const schema = buildClientSchema( | const schema = buildClientSchema( | ||||||
|   introspectionResult as unknown as IntrospectionQuery |   introspectionResult as unknown as IntrospectionQuery | ||||||
| @@ -37,8 +46,76 @@ const cleanTypeName = new ApolloLink((operation, forward) => { | |||||||
|   ); |   ); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const FennecApolloClientProvider: FC = ({ children }) => { | export const AppApolloClientProvider: FC = ({ children }) => { | ||||||
|   const { enqueueSnackbar } = useSnackbar(); |   const { enqueueSnackbar } = useSnackbar(); | ||||||
|  |   const { accessToken, logout } = useAuth(); | ||||||
|  |  | ||||||
|  |   const [loggedEventTarget] = useState(() => new EventEmitter()); | ||||||
|  |   const accessTokenRef = useRef(accessToken); | ||||||
|  |   const logoutRef = useRef(logout); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     accessTokenRef.current = accessToken; | ||||||
|  |     if (accessToken) { | ||||||
|  |       loggedEventTarget.emit("logged", accessToken); | ||||||
|  |     } | ||||||
|  |   }, [loggedEventTarget, accessToken]); | ||||||
|  |   useEffect(() => { | ||||||
|  |     logoutRef.current = logout; | ||||||
|  |   }, [logout]); | ||||||
|  |   const client = useMemo(() => { | ||||||
|  |     const authLink = onError( | ||||||
|  |       ({ graphQLErrors, networkError, operation, forward }) => { | ||||||
|  |         if (graphQLErrors) { | ||||||
|  |           for (const error of graphQLErrors) { | ||||||
|  |             if (error.extensions?.code === "401") { | ||||||
|  |               return fromPromise( | ||||||
|  |                 new Promise<Observable<FetchResult>>((resolve) => { | ||||||
|  |                   loggedEventTarget.once("logged", (accessToken: string) => { | ||||||
|  |                     const oldHeaders = operation.getContext().headers; | ||||||
|  |                     operation.setContext({ | ||||||
|  |                       headers: { | ||||||
|  |                         ...oldHeaders, | ||||||
|  |                         authorization: `Bearer ${accessToken}`, | ||||||
|  |                       }, | ||||||
|  |                     }); | ||||||
|  |                     const subscriptionClient = (wsLink as any) | ||||||
|  |                       .subscriptionClient; | ||||||
|  |                     subscriptionClient?.close(false, false); | ||||||
|  |                     resolve(forward(operation)); | ||||||
|  |                   }); | ||||||
|  |                   logoutRef.current(); | ||||||
|  |                 }) | ||||||
|  |               ).flatMap((v) => v); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         const httpResult = (networkError as any)?.result; | ||||||
|  |         if (httpResult?.statusCode === 401) { | ||||||
|  |           return fromPromise( | ||||||
|  |             new Promise<Observable<FetchResult>>((resolve) => { | ||||||
|  |               loggedEventTarget.once("logged", (accessToken: string) => { | ||||||
|  |                 const oldHeaders = operation.getContext().headers; | ||||||
|  |                 operation.setContext({ | ||||||
|  |                   headers: { | ||||||
|  |                     ...oldHeaders, | ||||||
|  |                     authorization: `Bearer ${accessToken}`, | ||||||
|  |                   }, | ||||||
|  |                 }); | ||||||
|  |                 resolve(forward(operation)); | ||||||
|  |               }); | ||||||
|  |               logoutRef.current(); | ||||||
|  |             }) | ||||||
|  |           ).flatMap((v) => v); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ).concat( | ||||||
|  |       setContext(() => ({ | ||||||
|  |         headers: { | ||||||
|  |           authorization: `Bearer ${accessTokenRef.current}`, | ||||||
|  |         }, | ||||||
|  |       })) | ||||||
|  |     ); | ||||||
|     const errorLink = onError(({ graphQLErrors, networkError }) => { |     const errorLink = onError(({ graphQLErrors, networkError }) => { | ||||||
|       if (graphQLErrors) { |       if (graphQLErrors) { | ||||||
|         graphQLErrors.forEach((error) => { |         graphQLErrors.forEach((error) => { | ||||||
| @@ -65,6 +142,9 @@ export const FennecApolloClientProvider: FC = ({ children }) => { | |||||||
|       }:${window.location.port}/api/graphql`, |       }:${window.location.port}/api/graphql`, | ||||||
|       options: { |       options: { | ||||||
|         reconnect: true, |         reconnect: true, | ||||||
|  |         connectionParams: () => ({ | ||||||
|  |           authorization: `Bearer ${accessTokenRef.current} `, | ||||||
|  |         }), | ||||||
|       }, |       }, | ||||||
|     }); |     }); | ||||||
|     const httpLink = new HttpLink({ |     const httpLink = new HttpLink({ | ||||||
| @@ -83,6 +163,7 @@ export const FennecApolloClientProvider: FC = ({ children }) => { | |||||||
|     ); |     ); | ||||||
|     const link = ApolloLink.from([ |     const link = ApolloLink.from([ | ||||||
|       errorLink, |       errorLink, | ||||||
|  |       authLink, | ||||||
|       withScalars({ schema, typesMap }) as unknown as ApolloLink, |       withScalars({ schema, typesMap }) as unknown as ApolloLink, | ||||||
|       cleanTypeName, |       cleanTypeName, | ||||||
|       splitLink, |       splitLink, | ||||||
| @@ -92,22 +173,12 @@ export const FennecApolloClientProvider: FC = ({ children }) => { | |||||||
|       ssrMode: typeof window === "undefined", |       ssrMode: typeof window === "undefined", | ||||||
|       link, |       link, | ||||||
|       cache: new InMemoryCache({ |       cache: new InMemoryCache({ | ||||||
|       typePolicies: { |         typePolicies: {}, | ||||||
|         // PipelineTaskLogs: { |  | ||||||
|         //   keyFields: ["unit"], |  | ||||||
|         // }, |  | ||||||
|         // PipelineTask: { |  | ||||||
|         //   fields: { |  | ||||||
|         //     logs: { |  | ||||||
|         //       merge(existing = [], incoming: any[]) { |  | ||||||
|         //         return [...existing, ...incoming]; |  | ||||||
|         //       }, |  | ||||||
|         //     }, |  | ||||||
|         //   }, |  | ||||||
|         // }, |  | ||||||
|       }, |  | ||||||
|       }), |       }), | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     return client; | ||||||
|  |   }, [enqueueSnackbar, loggedEventTarget]); | ||||||
|  |  | ||||||
|   return <ApolloProvider client={client}>{children}</ApolloProvider>; |   return <ApolloProvider client={client}>{children}</ApolloProvider>; | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -2,9 +2,9 @@ import { useApolloClient } from "@apollo/client"; | |||||||
| import { createRouterComponent } from "@curi/react-dom"; | import { createRouterComponent } from "@curi/react-dom"; | ||||||
| import { createRouter, announce } from "@curi/router"; | import { createRouter, announce } from "@curi/router"; | ||||||
| import { browser } from "@hickory/browser"; | import { browser } from "@hickory/browser"; | ||||||
| import { FC, ReactNode, useEffect, useMemo, useState } from "react"; | import { FC, useEffect, useState } from "react"; | ||||||
| import routes from "../../routes"; | import routes from "../../routes"; | ||||||
| import { LinearProgress } from "@material-ui/core"; | import { LinearProgress } from "@mui/material"; | ||||||
|  |  | ||||||
| const Component: FC = ({ children }) => { | const Component: FC = ({ children }) => { | ||||||
|   const client = useApolloClient(); |   const client = useApolloClient(); | ||||||
|   | |||||||
| @@ -1949,22 +1949,6 @@ | |||||||
|         "description": null, |         "description": null, | ||||||
|         "fields": null, |         "fields": null, | ||||||
|         "inputFields": [ |         "inputFields": [ | ||||||
|           { |  | ||||||
|             "name": "projectId", |  | ||||||
|             "description": null, |  | ||||||
|             "type": { |  | ||||||
|               "kind": "NON_NULL", |  | ||||||
|               "name": null, |  | ||||||
|               "ofType": { |  | ||||||
|                 "kind": "SCALAR", |  | ||||||
|                 "name": "String", |  | ||||||
|                 "ofType": null |  | ||||||
|               } |  | ||||||
|             }, |  | ||||||
|             "defaultValue": null, |  | ||||||
|             "isDeprecated": false, |  | ||||||
|             "deprecationReason": null |  | ||||||
|           }, |  | ||||||
|           { |           { | ||||||
|             "name": "branch", |             "name": "branch", | ||||||
|             "description": null, |             "description": null, | ||||||
|   | |||||||
| @@ -255,7 +255,6 @@ export enum TaskStatuses { | |||||||
| } | } | ||||||
|  |  | ||||||
| export type UpdatePipelineInput = { | export type UpdatePipelineInput = { | ||||||
|   projectId: Scalars['String']; |  | ||||||
|   branch: Scalars['String']; |   branch: Scalars['String']; | ||||||
|   name: Scalars['String']; |   name: Scalars['String']; | ||||||
|   workUnitMetadata: WorkUnitMetadataInput; |   workUnitMetadata: WorkUnitMetadataInput; | ||||||
|   | |||||||
| @@ -4,26 +4,45 @@ import "./index.css"; | |||||||
| import "fontsource-roboto"; | import "fontsource-roboto"; | ||||||
| import App from "./App"; | import App from "./App"; | ||||||
| import reportWebVitals from "./reportWebVitals"; | import reportWebVitals from "./reportWebVitals"; | ||||||
| import { FennecApolloClientProvider } from "./commons/graphql/client"; | import { AppApolloClientProvider } from "./commons/graphql/client"; | ||||||
| import { MuiPickersUtilsProvider } from "@material-ui/pickers"; |  | ||||||
| import DateFnsUtils from "@date-io/date-fns"; |  | ||||||
| import zhLocale from "date-fns/locale/zh-CN"; |  | ||||||
| import { ConfirmProvider } from "material-ui-confirm"; | import { ConfirmProvider } from "material-ui-confirm"; | ||||||
| import { SnackbarProvider } from "notistack"; | import { SnackbarProvider } from "notistack"; | ||||||
| import Router from './commons/route/router'; | import Router from "./commons/route/router"; | ||||||
|  | import { AuthProvider } from "./commons/auth/auth.provider"; | ||||||
|  | import { LocalizationProvider } from "@mui/lab"; | ||||||
|  | import AdapterDateFns from "@mui/lab/AdapterDateFns"; | ||||||
|  | import { | ||||||
|  |   ThemeProvider, | ||||||
|  |   Theme, | ||||||
|  |   StyledEngineProvider, | ||||||
|  |   createTheme, | ||||||
|  | } from "@mui/material/styles"; | ||||||
|  |  | ||||||
|  | declare module "@mui/styles/defaultTheme" { | ||||||
|  |   // eslint-disable-next-line @typescript-eslint/no-empty-interface | ||||||
|  |   interface DefaultTheme extends Theme {} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const theme = createTheme(); | ||||||
|  |  | ||||||
| ReactDOM.render( | ReactDOM.render( | ||||||
|   <React.StrictMode> |   <React.StrictMode> | ||||||
|     <ConfirmProvider> |     <ConfirmProvider> | ||||||
|  |       <StyledEngineProvider injectFirst> | ||||||
|  |         <ThemeProvider theme={theme}> | ||||||
|           <SnackbarProvider maxSnack={5}> |           <SnackbarProvider maxSnack={5}> | ||||||
|           <FennecApolloClientProvider> |             <AuthProvider> | ||||||
|             <MuiPickersUtilsProvider utils={DateFnsUtils} locale={zhLocale}> |               <AppApolloClientProvider> | ||||||
|  |                 <LocalizationProvider dateAdapter={AdapterDateFns}> | ||||||
|                   <Router> |                   <Router> | ||||||
|                     <App /> |                     <App /> | ||||||
|                   </Router> |                   </Router> | ||||||
|             </MuiPickersUtilsProvider> |                 </LocalizationProvider> | ||||||
|           </FennecApolloClientProvider> |               </AppApolloClientProvider> | ||||||
|  |             </AuthProvider> | ||||||
|           </SnackbarProvider> |           </SnackbarProvider> | ||||||
|  |         </ThemeProvider> | ||||||
|  |       </StyledEngineProvider> | ||||||
|     </ConfirmProvider> |     </ConfirmProvider> | ||||||
|   </React.StrictMode>, |   </React.StrictMode>, | ||||||
|   document.getElementById("root") |   document.getElementById("root") | ||||||
|   | |||||||
| @@ -1,21 +1,17 @@ | |||||||
| import React, { FC, useCallback, useRef, useState } from "react"; | import React, { FC, useCallback, useState } from "react"; | ||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
| import { | import { useTheme, Theme } from "@mui/material/styles"; | ||||||
|   createStyles, | import createStyles from '@mui/styles/createStyles'; | ||||||
|   makeStyles, | import makeStyles from '@mui/styles/makeStyles'; | ||||||
|   useTheme, | import Drawer from "@mui/material/Drawer"; | ||||||
|   Theme, | import AppBar from "@mui/material/AppBar"; | ||||||
| } from "@material-ui/core/styles"; | import Toolbar from "@mui/material/Toolbar"; | ||||||
| import Drawer from "@material-ui/core/Drawer"; | import CssBaseline from "@mui/material/CssBaseline"; | ||||||
| import AppBar from "@material-ui/core/AppBar"; | import Divider from "@mui/material/Divider"; | ||||||
| import Toolbar from "@material-ui/core/Toolbar"; | import IconButton from "@mui/material/IconButton"; | ||||||
| import CssBaseline from "@material-ui/core/CssBaseline"; | import MenuIcon from "@mui/icons-material/Menu"; | ||||||
| import Typography from "@material-ui/core/Typography"; | import ChevronLeftIcon from "@mui/icons-material/ChevronLeft"; | ||||||
| import Divider from "@material-ui/core/Divider"; | import ChevronRightIcon from "@mui/icons-material/ChevronRight"; | ||||||
| import IconButton from "@material-ui/core/IconButton"; |  | ||||||
| import MenuIcon from "@material-ui/icons/Menu"; |  | ||||||
| import ChevronLeftIcon from "@material-ui/icons/ChevronLeft"; |  | ||||||
| import ChevronRightIcon from "@material-ui/icons/ChevronRight"; |  | ||||||
| import { ProjectPanel } from "../projects/project-panel"; | import { ProjectPanel } from "../projects/project-panel"; | ||||||
| import { HeaderContainerProvider } from "./header-container"; | import { HeaderContainerProvider } from "./header-container"; | ||||||
| const drawerWidth = 240; | const drawerWidth = 240; | ||||||
| @@ -128,7 +124,7 @@ export const DefaultLayout: FC = ({ children }) => { | |||||||
|             className={clsx(classes.menuButton, { |             className={clsx(classes.menuButton, { | ||||||
|               [classes.hide]: open, |               [classes.hide]: open, | ||||||
|             })} |             })} | ||||||
|           > |             size="large"> | ||||||
|             <MenuIcon /> |             <MenuIcon /> | ||||||
|           </IconButton> |           </IconButton> | ||||||
|           <div className={classes.headerContaner} ref={onRefChange}></div> |           <div className={classes.headerContaner} ref={onRefChange}></div> | ||||||
| @@ -148,7 +144,7 @@ export const DefaultLayout: FC = ({ children }) => { | |||||||
|         }} |         }} | ||||||
|       > |       > | ||||||
|         <div className={classes.toolbar}> |         <div className={classes.toolbar}> | ||||||
|           <IconButton onClick={handleDrawerClose}> |           <IconButton onClick={handleDrawerClose} size="large"> | ||||||
|             {theme.direction === "rtl" ? ( |             {theme.direction === "rtl" ? ( | ||||||
|               <ChevronRightIcon /> |               <ChevronRightIcon /> | ||||||
|             ) : ( |             ) : ( | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| import { gql, useQuery, useSubscription } from "@apollo/client"; | import { gql, useQuery, useSubscription } from "@apollo/client"; | ||||||
| import { LinearProgress, makeStyles, Typography } from "@material-ui/core"; | import { LinearProgress, Typography } from "@mui/material"; | ||||||
|  | import makeStyles from '@mui/styles/makeStyles'; | ||||||
| import { format } from "date-fns"; | import { format } from "date-fns"; | ||||||
| import { FC, useState } from "react"; | import { FC, useState } from "react"; | ||||||
| import { ErrorPage } from "../commons/fallbacks/error-page"; | import { ErrorPage } from "../commons/fallbacks/error-page"; | ||||||
| @@ -56,9 +57,7 @@ const useStyles = makeStyles((theme) => ({ | |||||||
| })); | })); | ||||||
|  |  | ||||||
| export const PipelineTaskDetail: FC<Props> = ({ taskId }) => { | export const PipelineTaskDetail: FC<Props> = ({ taskId }) => { | ||||||
|   const [taskEvents, setTaskEvents] = useState( |   const [, setTaskEvents] = useState(() => new Array<PipelineTaskEvent>()); | ||||||
|     () => new Array<PipelineTaskEvent>() |  | ||||||
|   ); |  | ||||||
|   const { data, loading, error, client } = useQuery<{ |   const { data, loading, error, client } = useQuery<{ | ||||||
|     pipelineTask: PipelineTask; |     pipelineTask: PipelineTask; | ||||||
|   }>(PIPELINE_TASK, { |   }>(PIPELINE_TASK, { | ||||||
|   | |||||||
| @@ -1,33 +1,18 @@ | |||||||
| import { gql, Reference, useMutation, useQuery } from "@apollo/client"; | import { gql, Reference, useMutation } from "@apollo/client"; | ||||||
| import { useRouter } from "@curi/react-dom"; | import { useRouter } from "@curi/react-dom"; | ||||||
| import { | import { Button, Grid, IconButton, LinearProgress, Paper, Portal, Typography } from "@mui/material"; | ||||||
|   Button, | import makeStyles from '@mui/styles/makeStyles'; | ||||||
|   Grid, | import { Delete } from "@mui/icons-material"; | ||||||
|   IconButton, |  | ||||||
|   LinearProgress, |  | ||||||
|   makeStyles, |  | ||||||
|   Paper, |  | ||||||
|   Portal, |  | ||||||
|   Typography, |  | ||||||
| } from "@material-ui/core"; |  | ||||||
| import { Delete } from "@material-ui/icons"; |  | ||||||
| import { FormikHelpers, Formik, Form, Field } from "formik"; | import { FormikHelpers, Formik, Form, Field } from "formik"; | ||||||
| import { TextField, TextFieldProps } from "formik-material-ui"; | import { TextField, TextFieldProps } from "formik-material-ui"; | ||||||
| import { TextField as MuiTextField } from "@material-ui/core"; |  | ||||||
| import { useConfirm } from "material-ui-confirm"; | import { useConfirm } from "material-ui-confirm"; | ||||||
| import { useSnackbar } from "notistack"; | import { useSnackbar } from "notistack"; | ||||||
| import { not, omit } from "ramda"; | import { not, omit } from "ramda"; | ||||||
| import { ChangeEvent, FC } from "react"; | import { ChangeEvent, FC } from "react"; | ||||||
| import { | import { Pipeline, PipelineUnits } from "../generated/graphql"; | ||||||
|   Pipeline, |  | ||||||
|   WorkUnitMetadata, |  | ||||||
|   PipelineUnits, |  | ||||||
| } from "../generated/graphql"; |  | ||||||
| import { useHeaderContainer } from "../layouts"; | import { useHeaderContainer } from "../layouts"; | ||||||
| import { CREATE_PIPELINE, DELETE_PIPELINE, UPDATE_PIPELINE } from "./mutations"; | import { CREATE_PIPELINE, DELETE_PIPELINE, UPDATE_PIPELINE } from "./mutations"; | ||||||
| import { PIPELINE } from "./queries"; |  | ||||||
| import * as Yup from "yup"; | import * as Yup from "yup"; | ||||||
| import { useField } from "formik"; |  | ||||||
|  |  | ||||||
| type Values = Partial<Pipeline>; | type Values = Partial<Pipeline>; | ||||||
|  |  | ||||||
| @@ -108,7 +93,7 @@ export const PipelineEditor: FC<Props> = ({ pipeline }) => { | |||||||
|       if (isCreate) { |       if (isCreate) { | ||||||
|         await createPipeline({ |         await createPipeline({ | ||||||
|           variables: { |           variables: { | ||||||
|             input: values, |             pipeline: values, | ||||||
|           }, |           }, | ||||||
|         }).then(({ data }) => { |         }).then(({ data }) => { | ||||||
|           pipelineId = data!.createPipeline.id; |           pipelineId = data!.createPipeline.id; | ||||||
| @@ -183,16 +168,12 @@ export const PipelineEditor: FC<Props> = ({ pipeline }) => { | |||||||
|   return ( |   return ( | ||||||
|     <Paper className={classes.root}> |     <Paper className={classes.root}> | ||||||
|       <Portal container={headerContainer}> |       <Portal container={headerContainer}> | ||||||
|         <Grid container justify="space-between" alignItems="center"> |         <Grid container justifyContent="space-between" alignItems="center"> | ||||||
|           <Typography variant="h6" component="h1"> |           <Typography variant="h6" component="h1"> | ||||||
|             {isCreate ? "Create" : "Edit"} Pipeline |             {isCreate ? "Create" : "Edit"} Pipeline | ||||||
|           </Typography> |           </Typography> | ||||||
|           {isCreate ? null : ( |           {isCreate ? null : ( | ||||||
|             <IconButton |             <IconButton color="inherit" onClick={handleDelete} disabled={deleting} size="large"> | ||||||
|               color="inherit" |  | ||||||
|               onClick={handleDelete} |  | ||||||
|               disabled={deleting} |  | ||||||
|             > |  | ||||||
|               <Delete /> |               <Delete /> | ||||||
|             </IconButton> |             </IconButton> | ||||||
|           )} |           )} | ||||||
| @@ -273,11 +254,18 @@ const ScriptsField: FC<TextFieldProps> = ({ field, form, meta, ...props }) => { | |||||||
|       meta={meta} |       meta={meta} | ||||||
|       field={{ |       field={{ | ||||||
|         ...field, |         ...field, | ||||||
|  |         onBlur: (ev: React.FocusEvent) => { | ||||||
|  |           form.setFieldValue( | ||||||
|  |             field.name, | ||||||
|  |             field.value.filter((it: string) => !!it) | ||||||
|  |           ); | ||||||
|  |           return field.onBlur(ev); | ||||||
|  |         }, | ||||||
|         value: field.value?.join("\n") ?? "", |         value: field.value?.join("\n") ?? "", | ||||||
|         onChange: (ev: ChangeEvent<HTMLInputElement>) => |         onChange: (ev: ChangeEvent<HTMLInputElement>) => | ||||||
|           form.setFieldValue( |           form.setFieldValue( | ||||||
|             field.name, |             field.name, | ||||||
|             ev.target.value?.split("\n").map((it) => it.trim()) ?? [] |             ev.target.value?.split("\n").map((it) => it) ?? [] | ||||||
|           ), |           ), | ||||||
|       }} |       }} | ||||||
|     /> |     /> | ||||||
|   | |||||||
| @@ -7,12 +7,14 @@ import { | |||||||
|   ListItemText, |   ListItemText, | ||||||
|   ListItemSecondaryAction, |   ListItemSecondaryAction, | ||||||
|   IconButton, |   IconButton, | ||||||
| } from "@material-ui/core"; |   Menu, | ||||||
| import { FC, MouseEventHandler, useMemo } from "react"; |   MenuItem, | ||||||
|  |   PopoverPosition, | ||||||
|  | } from "@mui/material"; | ||||||
|  | import { FC, MouseEventHandler, useMemo, useState } from "react"; | ||||||
| import { Pipeline, Project } from "../generated/graphql"; | import { Pipeline, Project } from "../generated/graphql"; | ||||||
| import { CallMerge, Edit } from "@material-ui/icons"; | import { CallMerge, Edit } from "@mui/icons-material"; | ||||||
| import { clone } from "ramda"; | import { any, values } from "ramda"; | ||||||
| import { useEffect } from "react"; |  | ||||||
|  |  | ||||||
| interface Props { | interface Props { | ||||||
|   projectId: string; |   projectId: string; | ||||||
| @@ -32,7 +34,7 @@ const PIPELINES = gql` | |||||||
| `; | `; | ||||||
|  |  | ||||||
| export const PipelineList: FC<Props> = ({ projectId }) => { | export const PipelineList: FC<Props> = ({ projectId }) => { | ||||||
|   const { data, loading } = useQuery< |   const { data } = useQuery< | ||||||
|     { pipelines: Pipeline[]; project: Project }, |     { pipelines: Pipeline[]; project: Project }, | ||||||
|     { projectId: string } |     { projectId: string } | ||||||
|   >(PIPELINES, { |   >(PIPELINES, { | ||||||
| @@ -44,18 +46,49 @@ export const PipelineList: FC<Props> = ({ projectId }) => { | |||||||
|       project: data?.project, |       project: data?.project, | ||||||
|     })); |     })); | ||||||
|   }, [data]); |   }, [data]); | ||||||
|  |  | ||||||
|  |   const [contextMenu, setContextMenu] = useState<PopoverPosition>(); | ||||||
|  |  | ||||||
|  |   const handleContextMenu: MouseEventHandler = (event) => { | ||||||
|  |     event.preventDefault(); | ||||||
|  |     setContextMenu({ top: event.clientY - 4, left: event.clientX - 2 }); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleClose = () => { | ||||||
|  |     setContextMenu(undefined); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|  |     <> | ||||||
|       <List> |       <List> | ||||||
|         {pipelines?.map((pipeline) => ( |         {pipelines?.map((pipeline) => ( | ||||||
|           <Link |           <Link | ||||||
|             name="pipeline-commits" |             name="pipeline-commits" | ||||||
|             params={{ pipelineId: pipeline.id, projectId: projectId }} |             params={{ pipelineId: pipeline.id, projectId: projectId }} | ||||||
|             key={pipeline.id} |             key={pipeline.id} | ||||||
|  |             onContextMenu={handleContextMenu} | ||||||
|           > |           > | ||||||
|             <Item pipeline={pipeline} /> |             <Item pipeline={pipeline} /> | ||||||
|           </Link> |           </Link> | ||||||
|         ))} |         ))} | ||||||
|       </List> |       </List> | ||||||
|  |       <Menu | ||||||
|  |         keepMounted | ||||||
|  |         open={!!contextMenu} | ||||||
|  |         onClose={handleClose} | ||||||
|  |         anchorReference="anchorPosition" | ||||||
|  |         anchorPosition={ | ||||||
|  |           contextMenu && any(Boolean, values(contextMenu)) | ||||||
|  |             ? contextMenu | ||||||
|  |             : undefined | ||||||
|  |         } | ||||||
|  |       > | ||||||
|  |         <MenuItem onClick={handleClose}>Copy</MenuItem> | ||||||
|  |         <MenuItem onClick={handleClose}>Print</MenuItem> | ||||||
|  |         <MenuItem onClick={handleClose}>Highlight</MenuItem> | ||||||
|  |         <MenuItem onClick={handleClose}>Email</MenuItem> | ||||||
|  |       </Menu> | ||||||
|  |     </> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -82,7 +115,7 @@ const Item = ({ pipeline }: { pipeline: Pipeline }) => { | |||||||
|         } |         } | ||||||
|       /> |       /> | ||||||
|       <ListItemSecondaryAction> |       <ListItemSecondaryAction> | ||||||
|         <IconButton edge="end" aria-label="edit" onClick={modify}> |         <IconButton edge="end" aria-label="edit" onClick={modify} size="large"> | ||||||
|           <Edit /> |           <Edit /> | ||||||
|         </IconButton> |         </IconButton> | ||||||
|       </ListItemSecondaryAction> |       </ListItemSecondaryAction> | ||||||
|   | |||||||
| @@ -1,10 +1,13 @@ | |||||||
| import { Project } from "../generated/graphql"; | import { Project } from "../generated/graphql"; | ||||||
| import React, { FC, Fragment } from "react"; | import React, { FC, Fragment } from "react"; | ||||||
| import { IconButton, Grid, makeStyles, Paper, Portal, Typography } from "@material-ui/core"; | import { IconButton, Grid, Paper, Portal, Typography, Box } from "@mui/material"; | ||||||
|  | import makeStyles from '@mui/styles/makeStyles'; | ||||||
| import { useHeaderContainer } from "../layouts"; | import { useHeaderContainer } from "../layouts"; | ||||||
| import { PipelineList } from "../pipelines/pipeline-list"; | import { PipelineList } from "../pipelines/pipeline-list"; | ||||||
| import { Edit } from '@material-ui/icons'; | import { Edit } from '@mui/icons-material'; | ||||||
| import { Link } from '@curi/react-dom'; | import { Link } from '@curi/react-dom'; | ||||||
|  | import { Button } from "@mui/material"; | ||||||
|  | import { AddBox } from "@mui/icons-material"; | ||||||
|  |  | ||||||
| interface Props { | interface Props { | ||||||
|   project: Project; |   project: Project; | ||||||
| @@ -35,7 +38,7 @@ export const ProjectDetail: FC<Props> = ({ project, children }) => { | |||||||
|           container |           container | ||||||
|           spacing={3} |           spacing={3} | ||||||
|           direction="row" |           direction="row" | ||||||
|           justify="space-between" |           justifyContent="space-between" | ||||||
|           alignItems="center" |           alignItems="center" | ||||||
|         > |         > | ||||||
|           <Grid item> |           <Grid item> | ||||||
| @@ -48,7 +51,7 @@ export const ProjectDetail: FC<Props> = ({ project, children }) => { | |||||||
|           </Grid> |           </Grid> | ||||||
|           <Grid item> |           <Grid item> | ||||||
|             <Link name="edit-project" params={{ projectId: project.id }}> |             <Link name="edit-project" params={{ projectId: project.id }}> | ||||||
|               <IconButton color="inherit">{<Edit />}</IconButton> |               <IconButton color="inherit" size="large">{<Edit />}</IconButton> | ||||||
|             </Link> |             </Link> | ||||||
|           </Grid> |           </Grid> | ||||||
|         </Grid> |         </Grid> | ||||||
| @@ -62,6 +65,18 @@ export const ProjectDetail: FC<Props> = ({ project, children }) => { | |||||||
|       > |       > | ||||||
|         <Grid item xs={3} lg={2} style={{ height: "100%", display: "flex" }}> |         <Grid item xs={3} lg={2} style={{ height: "100%", display: "flex" }}> | ||||||
|           <Paper className={classes.pipelineListContainer}> |           <Paper className={classes.pipelineListContainer}> | ||||||
|  |             <Box m={2}> | ||||||
|  |               <Link name="create-pipeline" params={{ projectId: project.id }}> | ||||||
|  |                 <Button | ||||||
|  |                   variant="contained" | ||||||
|  |                   color="primary" | ||||||
|  |                   title="New Pipeline" | ||||||
|  |                   startIcon={<AddBox />} | ||||||
|  |                 > | ||||||
|  |                   New Pipeline | ||||||
|  |                 </Button> | ||||||
|  |               </Link> | ||||||
|  |             </Box> | ||||||
|             <PipelineList projectId={project.id} /> |             <PipelineList projectId={project.id} /> | ||||||
|           </Paper> |           </Paper> | ||||||
|         </Grid> |         </Grid> | ||||||
|   | |||||||
| @@ -1,14 +1,6 @@ | |||||||
| import { gql, Reference, useMutation } from "@apollo/client"; | import { gql, Reference, useMutation } from "@apollo/client"; | ||||||
| import { | import { Button, LinearProgress, Paper, Portal, Typography, Grid, IconButton } from "@mui/material"; | ||||||
|   Button, | import makeStyles from '@mui/styles/makeStyles'; | ||||||
|   LinearProgress, |  | ||||||
|   makeStyles, |  | ||||||
|   Paper, |  | ||||||
|   Portal, |  | ||||||
|   Typography, |  | ||||||
|   Grid, |  | ||||||
|   IconButton, |  | ||||||
| } from "@material-ui/core"; |  | ||||||
| import { Form, Formik, Field, FormikHelpers } from "formik"; | import { Form, Formik, Field, FormikHelpers } from "formik"; | ||||||
| import { TextField } from "formik-material-ui"; | import { TextField } from "formik-material-ui"; | ||||||
| import { not } from "ramda"; | import { not } from "ramda"; | ||||||
| @@ -17,7 +9,7 @@ import { Project } from "../generated/graphql"; | |||||||
| import * as Yup from "yup"; | import * as Yup from "yup"; | ||||||
| import { useRouter } from "@curi/react-dom"; | import { useRouter } from "@curi/react-dom"; | ||||||
| import { useHeaderContainer } from "../layouts"; | import { useHeaderContainer } from "../layouts"; | ||||||
| import DeleteIcon from "@material-ui/icons/Delete"; | import DeleteIcon from "@mui/icons-material/Delete"; | ||||||
| import { useConfirm } from "material-ui-confirm"; | import { useConfirm } from "material-ui-confirm"; | ||||||
| import { useSnackbar } from "notistack"; | import { useSnackbar } from "notistack"; | ||||||
|  |  | ||||||
| @@ -178,16 +170,12 @@ export const ProjectEditor: FC<Props> = ({ project }) => { | |||||||
|   return ( |   return ( | ||||||
|     <Paper className={classes.root}> |     <Paper className={classes.root}> | ||||||
|       <Portal container={headerContainer}> |       <Portal container={headerContainer}> | ||||||
|         <Grid container justify="space-between" alignItems="center"> |         <Grid container justifyContent="space-between" alignItems="center"> | ||||||
|           <Typography variant="h6" component="h1"> |           <Typography variant="h6" component="h1"> | ||||||
|             {isCreate ? "Create" : "Edit"} Project |             {isCreate ? "Create" : "Edit"} Project | ||||||
|           </Typography> |           </Typography> | ||||||
|           {isCreate ? null : ( |           {isCreate ? null : ( | ||||||
|             <IconButton |             <IconButton color="inherit" onClick={handleDelete} disabled={deleting} size="large"> | ||||||
|               color="inherit" |  | ||||||
|               onClick={handleDelete} |  | ||||||
|               disabled={deleting} |  | ||||||
|             > |  | ||||||
|               <DeleteIcon /> |               <DeleteIcon /> | ||||||
|             </IconButton> |             </IconButton> | ||||||
|           )} |           )} | ||||||
|   | |||||||
| @@ -1,11 +1,12 @@ | |||||||
| import { gql, useQuery } from "@apollo/client"; | import { gql, useQuery } from "@apollo/client"; | ||||||
| import { Link } from "@curi/react-dom"; | import { Link } from "@curi/react-dom"; | ||||||
| import { Box, List, ListItem, makeStyles, Theme } from "@material-ui/core"; | import { Box, List, ListItem, Theme } from "@mui/material"; | ||||||
|  | import makeStyles from '@mui/styles/makeStyles'; | ||||||
| import { FC } from "react"; | import { FC } from "react"; | ||||||
| import { Project } from "../generated/graphql"; | import { Project } from "../generated/graphql"; | ||||||
| import { ListItemText } from "@material-ui/core"; | import { ListItemText } from "@mui/material"; | ||||||
| import { Button } from "@material-ui/core"; | import { Button } from "@mui/material"; | ||||||
| import { AddBox } from "@material-ui/icons"; | import { AddBox } from "@mui/icons-material"; | ||||||
| import { ActiveLink } from "../commons/route/active-link"; | import { ActiveLink } from "../commons/route/active-link"; | ||||||
|  |  | ||||||
| const PROJECTS = gql` | const PROJECTS = gql` | ||||||
|   | |||||||
| @@ -1,16 +1,16 @@ | |||||||
| import { ApolloClient, InMemoryCache } from "@apollo/client"; | import { ApolloClient, InMemoryCache } from "@apollo/client"; | ||||||
| import { prepareRoutes } from "@curi/router"; | import { prepareRoutes } from "@curi/router"; | ||||||
| import { omit } from 'ramda'; | import { omit } from "ramda"; | ||||||
| import React from "react"; |  | ||||||
| import { ProjectDetail, ProjectEditor, PROJECT } from "./projects"; | import { ProjectDetail, ProjectEditor, PROJECT } from "./projects"; | ||||||
| import { COMMIT_LIST_QUERY } from './commons/graphql/queries'; | import { COMMIT_LIST_QUERY } from "./commons/graphql/queries"; | ||||||
| import { CommitList } from './commits/commit-list'; | import { CommitList } from "./commits/commit-list"; | ||||||
| import { PipelineTaskDetail } from './pipeline-tasks/pipeline-task-detail'; | import { PipelineTaskDetail } from "./pipeline-tasks/pipeline-task-detail"; | ||||||
| import { PipelineEditor } from "./pipelines/pipeline-editor"; | import { PipelineEditor } from "./pipelines/pipeline-editor"; | ||||||
| import { | import { | ||||||
|   CreatePipelineInput, |   CreatePipelineInput, | ||||||
|   CreateProjectInput, |   CreateProjectInput, | ||||||
|   Pipeline, |   Pipeline, | ||||||
|  |   PipelineUnits, | ||||||
|   Project, |   Project, | ||||||
| } from "./generated/graphql"; | } from "./generated/graphql"; | ||||||
| import { PIPELINE } from "./pipelines"; | import { PIPELINE } from "./pipelines"; | ||||||
| @@ -65,13 +65,20 @@ export default prepareRoutes([ | |||||||
|       matched, |       matched, | ||||||
|       { client }: { client: ApolloClient<InMemoryCache> } |       { client }: { client: ApolloClient<InMemoryCache> } | ||||||
|     ) { |     ) { | ||||||
|  |       const units = [ | ||||||
|  |         PipelineUnits.Checkout, | ||||||
|  |         PipelineUnits.InstallDependencies, | ||||||
|  |         PipelineUnits.Test, | ||||||
|  |         PipelineUnits.Deploy, | ||||||
|  |         PipelineUnits.CleanUp, | ||||||
|  |       ]; | ||||||
|       const input: CreatePipelineInput = { |       const input: CreatePipelineInput = { | ||||||
|         name: "", |         name: "", | ||||||
|         branch: "", |         branch: "", | ||||||
|         projectId: matched!.params.projectId, |         projectId: matched!.params.projectId, | ||||||
|         workUnitMetadata: { |         workUnitMetadata: { | ||||||
|           version: 1, |           version: 1, | ||||||
|           units: [], |           units: units.map((util) => ({ type: util, scripts: [] })), | ||||||
|         }, |         }, | ||||||
|       }; |       }; | ||||||
|       return { |       return { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user