mirror of
https://github.com/snowball-tools/snowballtools-base
synced 2025-01-09 20:38:05 +00:00
Save commit message in DB and show in deployments list (#67)
* Display commit message in projects and deployments * Handle if current deployment not present * Rename types file --------- Co-authored-by: neeraj <neeraj.rtly@gmail.com>
This commit is contained in:
parent
adba64fd2b
commit
db3b9148b6
@ -59,6 +59,9 @@ export class Deployment {
|
|||||||
@Column('varchar')
|
@Column('varchar')
|
||||||
commitHash!: string;
|
commitHash!: string;
|
||||||
|
|
||||||
|
@Column('varchar')
|
||||||
|
commitMessage!: string;
|
||||||
|
|
||||||
@Column('varchar')
|
@Column('varchar')
|
||||||
url!: string;
|
url!: string;
|
||||||
|
|
||||||
|
@ -90,6 +90,7 @@ type Deployment {
|
|||||||
domain: Domain
|
domain: Domain
|
||||||
branch: String!
|
branch: String!
|
||||||
commitHash: String!
|
commitHash: String!
|
||||||
|
commitMessage: String!
|
||||||
url: String!
|
url: String!
|
||||||
environment: Environment!
|
environment: Environment!
|
||||||
isCurrent: Boolean!
|
isCurrent: Boolean!
|
||||||
|
@ -218,7 +218,8 @@ export class Service {
|
|||||||
branch: oldDeployment.branch,
|
branch: oldDeployment.branch,
|
||||||
environment: Environment.Production,
|
environment: Environment.Production,
|
||||||
domain: prodBranchDomains[0],
|
domain: prodBranchDomains[0],
|
||||||
commitHash: oldDeployment.commitHash
|
commitHash: oldDeployment.commitHash,
|
||||||
|
commitMessage: oldDeployment.commitMessage
|
||||||
});
|
});
|
||||||
|
|
||||||
return newDeployement;
|
return newDeployement;
|
||||||
@ -252,6 +253,7 @@ export class Service {
|
|||||||
project: data.project,
|
project: data.project,
|
||||||
branch: data.branch,
|
branch: data.branch,
|
||||||
commitHash: data.commitHash,
|
commitHash: data.commitHash,
|
||||||
|
commitMessage: data.commitMessage,
|
||||||
environment: data.environment,
|
environment: data.environment,
|
||||||
isCurrent: data.isCurrent,
|
isCurrent: data.isCurrent,
|
||||||
status: DeploymentStatus.Building,
|
status: DeploymentStatus.Building,
|
||||||
@ -300,7 +302,8 @@ export class Service {
|
|||||||
branch: project.prodBranch,
|
branch: project.prodBranch,
|
||||||
environment: Environment.Production,
|
environment: Environment.Production,
|
||||||
domain: null,
|
domain: null,
|
||||||
commitHash: latestCommit.sha
|
commitHash: latestCommit.sha,
|
||||||
|
commitMessage: latestCommit.commit.message
|
||||||
});
|
});
|
||||||
|
|
||||||
const { registryRecordId, registryRecordData } = await this.registry.createApplicationDeploymentRequest(
|
const { registryRecordId, registryRecordData } = await this.registry.createApplicationDeploymentRequest(
|
||||||
@ -370,7 +373,8 @@ export class Service {
|
|||||||
isCurrent: true,
|
isCurrent: true,
|
||||||
environment: Environment.Production,
|
environment: Environment.Production,
|
||||||
domain: oldDeployment.domain,
|
domain: oldDeployment.domain,
|
||||||
commitHash: oldDeployment.commitHash
|
commitHash: oldDeployment.commitHash,
|
||||||
|
commitMessage: oldDeployment.commitMessage
|
||||||
});
|
});
|
||||||
|
|
||||||
return newDeployement;
|
return newDeployement;
|
||||||
|
27
packages/backend/test/fixtures/deployments.json
vendored
27
packages/backend/test/fixtures/deployments.json
vendored
@ -11,6 +11,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "main",
|
"branch": "main",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-ffhae3zq.snowball.xyz"
|
"url": "testProject-ffhae3zq.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -25,6 +26,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-vehagei8.snowball.xyz"
|
"url": "testProject-vehagei8.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -39,6 +41,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-qmgekyte.snowball.xyz"
|
"url": "testProject-qmgekyte.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -53,6 +56,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "prod",
|
"branch": "prod",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-f8wsyim6.snowball.xyz"
|
"url": "testProject-f8wsyim6.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -67,6 +71,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "main",
|
"branch": "main",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-2-eO8cckxk.snowball.xyz"
|
"url": "testProject-2-eO8cckxk.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -81,6 +86,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-2-yaq0t5yw.snowball.xyz"
|
"url": "testProject-2-yaq0t5yw.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -95,11 +101,12 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "testProject-2-hwwr6sbx.snowball.xyz"
|
"url": "testProject-2-hwwr6sbx.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"projectIndex": 2,
|
"projectIndex": 2,
|
||||||
"domainIndex":6,
|
"domainIndex":9,
|
||||||
"createdByIndex": 2,
|
"createdByIndex": 2,
|
||||||
"id":"ndxje48a",
|
"id":"ndxje48a",
|
||||||
"status": "Building",
|
"status": "Building",
|
||||||
@ -109,6 +116,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "main",
|
"branch": "main",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "iglootools-ndxje48a.snowball.xyz"
|
"url": "iglootools-ndxje48a.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -123,6 +131,7 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "iglootools-gtgpgvei.snowball.xyz"
|
"url": "iglootools-gtgpgvei.snowball.xyz"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -137,6 +146,22 @@
|
|||||||
"registryRecordData": {},
|
"registryRecordData": {},
|
||||||
"branch": "test",
|
"branch": "test",
|
||||||
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
|
"url": "iglootools-b4bpthjr.snowball.xyz"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"projectIndex": 3,
|
||||||
|
"domainIndex": 6,
|
||||||
|
"createdByIndex": 2,
|
||||||
|
"id":"b4bpthjr",
|
||||||
|
"status": "Building",
|
||||||
|
"environment": "Production",
|
||||||
|
"isCurrent": true,
|
||||||
|
"registryRecordId": "pbafyreihvzya6ovp4yfpkqnddkui2iw7t6hbhwq74lbqs7bhobvmfhrowo",
|
||||||
|
"registryRecordData": {},
|
||||||
|
"branch": "test",
|
||||||
|
"commitHash": "d5dfd7b827226b0d09d897346d291c256e113e00",
|
||||||
|
"commitMessage": "subscription added",
|
||||||
"url": "iglootools-b4bpthjr.snowball.xyz"
|
"url": "iglootools-b4bpthjr.snowball.xyz"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -34,5 +34,11 @@
|
|||||||
"name": "saugatt.com",
|
"name": "saugatt.com",
|
||||||
"status": "Pending",
|
"status": "Pending",
|
||||||
"branch": "test"
|
"branch": "test"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"projectIndex": 3,
|
||||||
|
"name": "iglootools-2.com",
|
||||||
|
"status": "Pending",
|
||||||
|
"branch": "test"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
import { Project } from 'gql-client';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Menu,
|
Menu,
|
||||||
@ -10,11 +11,10 @@ import {
|
|||||||
Avatar,
|
Avatar,
|
||||||
} from '@material-tailwind/react';
|
} from '@material-tailwind/react';
|
||||||
|
|
||||||
import { relativeTimeISO } from '../../utils/time';
|
import { relativeTimeMs } from '../../utils/time';
|
||||||
import { ProjectDetails } from '../../types/project';
|
|
||||||
|
|
||||||
interface ProjectCardProps {
|
interface ProjectCardProps {
|
||||||
project: ProjectDetails;
|
project: Project;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ProjectCard: React.FC<ProjectCardProps> = ({ project }) => {
|
const ProjectCard: React.FC<ProjectCardProps> = ({ project }) => {
|
||||||
@ -42,13 +42,21 @@ const ProjectCard: React.FC<ProjectCardProps> = ({ project }) => {
|
|||||||
</Menu>
|
</Menu>
|
||||||
</div>
|
</div>
|
||||||
<div className="border-t-2 border-solid p-4 bg-gray-50">
|
<div className="border-t-2 border-solid p-4 bg-gray-50">
|
||||||
<Typography variant="small" color="gray">
|
{project.deployments.length > 0 ? (
|
||||||
^ {project.latestCommit.message}
|
<>
|
||||||
</Typography>
|
<Typography variant="small" color="gray">
|
||||||
<Typography variant="small" color="gray">
|
^ {project.deployments[0].commitMessage}
|
||||||
{relativeTimeISO(project.latestCommit.createdAt)} on ^
|
</Typography>
|
||||||
{project.latestCommit.branch}
|
<Typography variant="small" color="gray">
|
||||||
</Typography>
|
{relativeTimeMs(project.deployments[0].createdAt)} on ^
|
||||||
|
{project.deployments[0].branch}
|
||||||
|
</Typography>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<Typography variant="small" color="gray">
|
||||||
|
No Production deployment
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -4,7 +4,7 @@ import { useNavigate, useParams } from 'react-router-dom';
|
|||||||
import { Chip, IconButton } from '@material-tailwind/react';
|
import { Chip, IconButton } from '@material-tailwind/react';
|
||||||
|
|
||||||
import { relativeTimeISO } from '../../../utils/time';
|
import { relativeTimeISO } from '../../../utils/time';
|
||||||
import { GitRepositoryDetails } from '../../../types/project';
|
import { GitRepositoryDetails } from '../../../types';
|
||||||
import { useGQLClient } from '../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../context/GQLClientContext';
|
||||||
|
|
||||||
interface ProjectRepoCardProps {
|
interface ProjectRepoCardProps {
|
||||||
|
@ -7,7 +7,7 @@ import { Button, Typography, Option, Select } from '@material-tailwind/react';
|
|||||||
|
|
||||||
import SearchBar from '../../SearchBar';
|
import SearchBar from '../../SearchBar';
|
||||||
import ProjectRepoCard from './ProjectRepoCard';
|
import ProjectRepoCard from './ProjectRepoCard';
|
||||||
import { GitOrgDetails, GitRepositoryDetails } from '../../../types/project';
|
import { GitOrgDetails, GitRepositoryDetails } from '../../../types';
|
||||||
|
|
||||||
const DEFAULT_SEARCHED_REPO = '';
|
const DEFAULT_SEARCHED_REPO = '';
|
||||||
const REPOS_PER_PAGE = 5;
|
const REPOS_PER_PAGE = 5;
|
||||||
|
@ -3,7 +3,7 @@ import React from 'react';
|
|||||||
import { Typography, IconButton } from '@material-tailwind/react';
|
import { Typography, IconButton } from '@material-tailwind/react';
|
||||||
|
|
||||||
import { relativeTimeISO } from '../../../utils/time';
|
import { relativeTimeISO } from '../../../utils/time';
|
||||||
import { GitCommitDetails } from '../../../types/project';
|
import { GitCommitDetails } from '../../../types';
|
||||||
|
|
||||||
interface ActivityCardProps {
|
interface ActivityCardProps {
|
||||||
activity: GitCommitDetails;
|
activity: GitCommitDetails;
|
||||||
|
@ -1,6 +1,12 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import toast from 'react-hot-toast';
|
import toast from 'react-hot-toast';
|
||||||
import { Environment, Project, Domain, DeploymentStatus } from 'gql-client';
|
import {
|
||||||
|
Environment,
|
||||||
|
Project,
|
||||||
|
Domain,
|
||||||
|
DeploymentStatus,
|
||||||
|
Deployment,
|
||||||
|
} from 'gql-client';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Menu,
|
Menu,
|
||||||
@ -16,13 +22,12 @@ import { relativeTimeMs } from '../../../../utils/time';
|
|||||||
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
||||||
import DeploymentDialogBodyCard from './DeploymentDialogBodyCard';
|
import DeploymentDialogBodyCard from './DeploymentDialogBodyCard';
|
||||||
import AssignDomainDialog from './AssignDomainDialog';
|
import AssignDomainDialog from './AssignDomainDialog';
|
||||||
import { DeploymentDetails } from '../../../../types/project';
|
|
||||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||||
import { SHORT_COMMIT_HASH_LENGTH } from '../../../../constants';
|
import { SHORT_COMMIT_HASH_LENGTH } from '../../../../constants';
|
||||||
|
|
||||||
interface DeployDetailsCardProps {
|
interface DeployDetailsCardProps {
|
||||||
deployment: DeploymentDetails;
|
deployment: Deployment;
|
||||||
currentDeployment: DeploymentDetails;
|
currentDeployment: Deployment;
|
||||||
onUpdate: () => Promise<void>;
|
onUpdate: () => Promise<void>;
|
||||||
project: Project;
|
project: Project;
|
||||||
prodBranchDomains: Domain[];
|
prodBranchDomains: Domain[];
|
||||||
@ -103,7 +108,7 @@ const DeploymentDetailsCard = ({
|
|||||||
<Typography color="gray">^ {deployment.branch}</Typography>
|
<Typography color="gray">^ {deployment.branch}</Typography>
|
||||||
<Typography color="gray">
|
<Typography color="gray">
|
||||||
^ {deployment.commitHash.substring(0, SHORT_COMMIT_HASH_LENGTH)}{' '}
|
^ {deployment.commitHash.substring(0, SHORT_COMMIT_HASH_LENGTH)}{' '}
|
||||||
{deployment.commit.message}
|
{deployment.commitMessage}
|
||||||
</Typography>
|
</Typography>
|
||||||
</div>
|
</div>
|
||||||
<div className="col-span-1 flex items-center">
|
<div className="col-span-1 flex items-center">
|
||||||
@ -149,7 +154,8 @@ const DeploymentDetailsCard = ({
|
|||||||
onClick={() => setRollbackDeployment(!rollbackDeployment)}
|
onClick={() => setRollbackDeployment(!rollbackDeployment)}
|
||||||
disabled={
|
disabled={
|
||||||
deployment.isCurrent ||
|
deployment.isCurrent ||
|
||||||
deployment.environment !== Environment.Production
|
deployment.environment !== Environment.Production ||
|
||||||
|
!Boolean(currentDeployment)
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
^ Rollback to this version
|
^ Rollback to this version
|
||||||
@ -213,44 +219,46 @@ const DeploymentDetailsCard = ({
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</ConfirmDialog>
|
</ConfirmDialog>
|
||||||
<ConfirmDialog
|
{Boolean(currentDeployment) && (
|
||||||
dialogTitle="Rollback to this deployment?"
|
<ConfirmDialog
|
||||||
handleOpen={() => setRollbackDeployment((preVal) => !preVal)}
|
dialogTitle="Rollback to this deployment?"
|
||||||
open={rollbackDeployment}
|
handleOpen={() => setRollbackDeployment((preVal) => !preVal)}
|
||||||
confirmButtonTitle="Rollback"
|
open={rollbackDeployment}
|
||||||
color="blue"
|
confirmButtonTitle="Rollback"
|
||||||
handleConfirm={async () => {
|
color="blue"
|
||||||
await rollbackDeploymentHandler();
|
handleConfirm={async () => {
|
||||||
setRollbackDeployment((preVal) => !preVal);
|
await rollbackDeploymentHandler();
|
||||||
}}
|
setRollbackDeployment((preVal) => !preVal);
|
||||||
>
|
}}
|
||||||
<div className="flex flex-col gap-2">
|
>
|
||||||
<Typography variant="small">
|
<div className="flex flex-col gap-2">
|
||||||
Upon confirmation, this deployment will replace your current
|
<Typography variant="small">
|
||||||
deployment
|
Upon confirmation, this deployment will replace your current
|
||||||
</Typography>
|
deployment
|
||||||
<DeploymentDialogBodyCard
|
</Typography>
|
||||||
deployment={currentDeployment}
|
<DeploymentDialogBodyCard
|
||||||
chip={{
|
deployment={currentDeployment}
|
||||||
value: 'Live Deployment',
|
chip={{
|
||||||
color: 'green',
|
value: 'Live Deployment',
|
||||||
}}
|
color: 'green',
|
||||||
/>
|
}}
|
||||||
<DeploymentDialogBodyCard
|
/>
|
||||||
deployment={deployment}
|
<DeploymentDialogBodyCard
|
||||||
chip={{
|
deployment={deployment}
|
||||||
value: 'New Deployment',
|
chip={{
|
||||||
color: 'orange',
|
value: 'New Deployment',
|
||||||
}}
|
color: 'orange',
|
||||||
/>
|
}}
|
||||||
<Typography variant="small">
|
/>
|
||||||
These domains will point to your new deployment:
|
<Typography variant="small">
|
||||||
</Typography>
|
These domains will point to your new deployment:
|
||||||
<Typography variant="small" color="blue">
|
</Typography>
|
||||||
^ {currentDeployment.domain?.name}
|
<Typography variant="small" color="blue">
|
||||||
</Typography>
|
^ {currentDeployment.domain?.name}
|
||||||
</div>
|
</Typography>
|
||||||
</ConfirmDialog>
|
</div>
|
||||||
|
</ConfirmDialog>
|
||||||
|
)}
|
||||||
<AssignDomainDialog
|
<AssignDomainDialog
|
||||||
open={assignDomainDialog}
|
open={assignDomainDialog}
|
||||||
handleOpen={() => setAssignDomainDialog(!assignDomainDialog)}
|
handleOpen={() => setAssignDomainDialog(!assignDomainDialog)}
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { Deployment } from 'gql-client';
|
||||||
|
|
||||||
import { Typography, Chip, Card } from '@material-tailwind/react';
|
import { Typography, Chip, Card } from '@material-tailwind/react';
|
||||||
import { color } from '@material-tailwind/react/types/components/chip';
|
import { color } from '@material-tailwind/react/types/components/chip';
|
||||||
import { DeploymentDetails } from '../../../../types/project';
|
|
||||||
import { relativeTimeMs } from '../../../../utils/time';
|
import { relativeTimeMs } from '../../../../utils/time';
|
||||||
import { SHORT_COMMIT_HASH_LENGTH } from '../../../../constants';
|
import { SHORT_COMMIT_HASH_LENGTH } from '../../../../constants';
|
||||||
|
|
||||||
interface DeploymentDialogBodyCardProps {
|
interface DeploymentDialogBodyCardProps {
|
||||||
deployment: DeploymentDetails;
|
deployment: Deployment;
|
||||||
chip?: {
|
chip?: {
|
||||||
value: string;
|
value: string;
|
||||||
color?: color;
|
color?: color;
|
||||||
@ -34,7 +34,7 @@ const DeploymentDialogBodyCard = ({
|
|||||||
<Typography variant="small">
|
<Typography variant="small">
|
||||||
^ {deployment.branch} ^{' '}
|
^ {deployment.branch} ^{' '}
|
||||||
{deployment.commitHash.substring(0, SHORT_COMMIT_HASH_LENGTH)}{' '}
|
{deployment.commitHash.substring(0, SHORT_COMMIT_HASH_LENGTH)}{' '}
|
||||||
{deployment.commit.message}
|
{deployment.commitMessage}
|
||||||
</Typography>
|
</Typography>
|
||||||
<Typography variant="small">
|
<Typography variant="small">
|
||||||
^ {relativeTimeMs(deployment.createdAt)} ^ {deployment.createdBy.name}
|
^ {relativeTimeMs(deployment.createdAt)} ^ {deployment.createdBy.name}
|
||||||
|
@ -3,7 +3,7 @@ import { UseFormRegister } from 'react-hook-form';
|
|||||||
|
|
||||||
import { Typography, Input, IconButton } from '@material-tailwind/react';
|
import { Typography, Input, IconButton } from '@material-tailwind/react';
|
||||||
|
|
||||||
import { EnvironmentVariablesFormValues } from '../../../../types/project';
|
import { EnvironmentVariablesFormValues } from '../../../../types';
|
||||||
|
|
||||||
interface AddEnvironmentVariableRowProps {
|
interface AddEnvironmentVariableRowProps {
|
||||||
onDelete: () => void;
|
onDelete: () => void;
|
||||||
|
@ -12,7 +12,7 @@ import {
|
|||||||
Card,
|
Card,
|
||||||
} from '@material-tailwind/react';
|
} from '@material-tailwind/react';
|
||||||
|
|
||||||
import { RepositoryDetails } from '../../../../types/project';
|
import { RepositoryDetails } from '../../../../types';
|
||||||
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
||||||
import EditDomainDialog from './EditDomainDialog';
|
import EditDomainDialog from './EditDomainDialog';
|
||||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||||
|
@ -15,7 +15,7 @@ import {
|
|||||||
Option,
|
Option,
|
||||||
} from '@material-tailwind/react';
|
} from '@material-tailwind/react';
|
||||||
|
|
||||||
import { RepositoryDetails } from '../../../../types/project';
|
import { RepositoryDetails } from '../../../../types';
|
||||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||||
|
|
||||||
const DEFAULT_REDIRECT_OPTIONS = ['none'];
|
const DEFAULT_REDIRECT_OPTIONS = ['none'];
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import { GitSelect } from '../../../../types/project';
|
import { GitSelect } from '../../../../types';
|
||||||
|
|
||||||
const GitSelectionSection = ({
|
const GitSelectionSection = ({
|
||||||
gitSelectionHandler,
|
gitSelectionHandler,
|
||||||
|
@ -2,7 +2,7 @@ import React, { useState } from 'react';
|
|||||||
|
|
||||||
import { Button, Typography } from '@material-tailwind/react';
|
import { Button, Typography } from '@material-tailwind/react';
|
||||||
|
|
||||||
import { GitRepositoryDetails } from '../../../../types/project';
|
import { GitRepositoryDetails } from '../../../../types';
|
||||||
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
||||||
|
|
||||||
const RepoConnectedSection = ({
|
const RepoConnectedSection = ({
|
||||||
|
@ -1,9 +1,3 @@
|
|||||||
export const COMMIT_DETAILS = {
|
|
||||||
message: 'subscription added',
|
|
||||||
createdAt: '2023-12-11T04:20:00',
|
|
||||||
branch: 'main',
|
|
||||||
};
|
|
||||||
|
|
||||||
export const ORGANIZATION_ID = '2379cf1f-a232-4ad2-ae14-4d881131cc26';
|
export const ORGANIZATION_ID = '2379cf1f-a232-4ad2-ae14-4d881131cc26';
|
||||||
|
|
||||||
export const GIT_TEMPLATE_LINK =
|
export const GIT_TEMPLATE_LINK =
|
||||||
|
@ -1,32 +1,22 @@
|
|||||||
import React, { useCallback, useEffect, useState } from 'react';
|
import React, { useCallback, useEffect, useState } from 'react';
|
||||||
import { Link, useParams } from 'react-router-dom';
|
import { Link, useParams } from 'react-router-dom';
|
||||||
|
import { Project } from 'gql-client';
|
||||||
|
|
||||||
import { Button, Typography, Chip } from '@material-tailwind/react';
|
import { Button, Typography, Chip } from '@material-tailwind/react';
|
||||||
|
|
||||||
import ProjectCard from '../../components/projects/ProjectCard';
|
import ProjectCard from '../../components/projects/ProjectCard';
|
||||||
import { useGQLClient } from '../../context/GQLClientContext';
|
import { useGQLClient } from '../../context/GQLClientContext';
|
||||||
import { ProjectDetails } from '../../types/project';
|
|
||||||
import { COMMIT_DETAILS } from '../../constants';
|
|
||||||
|
|
||||||
const Projects = () => {
|
const Projects = () => {
|
||||||
const client = useGQLClient();
|
const client = useGQLClient();
|
||||||
const { orgSlug } = useParams();
|
const { orgSlug } = useParams();
|
||||||
const [projects, setProjects] = useState<ProjectDetails[]>([]);
|
const [projects, setProjects] = useState<Project[]>([]);
|
||||||
|
|
||||||
const fetchProjects = useCallback(async () => {
|
const fetchProjects = useCallback(async () => {
|
||||||
const { projectsInOrganization } = await client.getProjectsInOrganization(
|
const { projectsInOrganization } = await client.getProjectsInOrganization(
|
||||||
orgSlug!,
|
orgSlug!,
|
||||||
);
|
);
|
||||||
|
setProjects(projectsInOrganization);
|
||||||
const updatedProjects = projectsInOrganization.map((project) => {
|
|
||||||
return {
|
|
||||||
...project,
|
|
||||||
// TODO: Populate from github API
|
|
||||||
latestCommit: COMMIT_DETAILS,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
setProjects(updatedProjects);
|
|
||||||
}, [orgSlug]);
|
}, [orgSlug]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import React, { useCallback, useEffect, useMemo, useState } from 'react';
|
import React, { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
import { Domain } from 'gql-client';
|
import { Deployment, Domain } from 'gql-client';
|
||||||
import { useOutletContext } from 'react-router-dom';
|
import { useOutletContext } from 'react-router-dom';
|
||||||
|
|
||||||
import { Button, Typography } from '@material-tailwind/react';
|
import { Button, Typography } from '@material-tailwind/react';
|
||||||
@ -9,12 +9,8 @@ import FilterForm, {
|
|||||||
FilterValue,
|
FilterValue,
|
||||||
StatusOptions,
|
StatusOptions,
|
||||||
} from '../../../../components/projects/project/deployments/FilterForm';
|
} from '../../../../components/projects/project/deployments/FilterForm';
|
||||||
import {
|
import { OutletContextType } from '../../../../types';
|
||||||
DeploymentDetails,
|
|
||||||
OutletContextType,
|
|
||||||
} from '../../../../types/project';
|
|
||||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||||
import { COMMIT_DETAILS } from '../../../../constants';
|
|
||||||
|
|
||||||
const DEFAULT_FILTER_VALUE: FilterValue = {
|
const DEFAULT_FILTER_VALUE: FilterValue = {
|
||||||
searchedBranch: '',
|
searchedBranch: '',
|
||||||
@ -27,19 +23,12 @@ const DeploymentsTabPanel = () => {
|
|||||||
const { project } = useOutletContext<OutletContextType>();
|
const { project } = useOutletContext<OutletContextType>();
|
||||||
|
|
||||||
const [filterValue, setFilterValue] = useState(DEFAULT_FILTER_VALUE);
|
const [filterValue, setFilterValue] = useState(DEFAULT_FILTER_VALUE);
|
||||||
const [deployments, setDeployments] = useState<DeploymentDetails[]>([]);
|
const [deployments, setDeployments] = useState<Deployment[]>([]);
|
||||||
const [prodBranchDomains, setProdBranchDomains] = useState<Domain[]>([]);
|
const [prodBranchDomains, setProdBranchDomains] = useState<Domain[]>([]);
|
||||||
|
|
||||||
const fetchDeployments = async () => {
|
const fetchDeployments = async () => {
|
||||||
const { deployments } = await client.getDeployments(project.id);
|
const { deployments } = await client.getDeployments(project.id);
|
||||||
const updatedDeployments = deployments.map((deployment) => {
|
setDeployments(deployments);
|
||||||
return {
|
|
||||||
...deployment,
|
|
||||||
author: '',
|
|
||||||
commit: COMMIT_DETAILS,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
setDeployments(updatedDeployments);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const fetchProductionBranchDomains = useCallback(async () => {
|
const fetchProductionBranchDomains = useCallback(async () => {
|
||||||
@ -60,7 +49,7 @@ const DeploymentsTabPanel = () => {
|
|||||||
});
|
});
|
||||||
}, [deployments]);
|
}, [deployments]);
|
||||||
|
|
||||||
const filteredDeployments = useMemo<DeploymentDetails[]>(() => {
|
const filteredDeployments = useMemo<Deployment[]>(() => {
|
||||||
return deployments.filter((deployment) => {
|
return deployments.filter((deployment) => {
|
||||||
// Searched branch filter
|
// Searched branch filter
|
||||||
const branchMatch =
|
const branchMatch =
|
||||||
@ -81,7 +70,7 @@ const DeploymentsTabPanel = () => {
|
|||||||
new Date(deployment.updatedAt) <= filterValue.updateAtRange!.to!);
|
new Date(deployment.updatedAt) <= filterValue.updateAtRange!.to!);
|
||||||
|
|
||||||
return branchMatch && statusMatch && dateMatch;
|
return branchMatch && statusMatch && dateMatch;
|
||||||
}) as DeploymentDetails[];
|
});
|
||||||
}, [filterValue, deployments]);
|
}, [filterValue, deployments]);
|
||||||
|
|
||||||
const handleResetFilters = useCallback(() => {
|
const handleResetFilters = useCallback(() => {
|
||||||
|
@ -7,7 +7,7 @@ import { Typography, Button, Chip, Avatar } from '@material-tailwind/react';
|
|||||||
import ActivityCard from '../../../../components/projects/project/ActivityCard';
|
import ActivityCard from '../../../../components/projects/project/ActivityCard';
|
||||||
import { relativeTimeMs } from '../../../../utils/time';
|
import { relativeTimeMs } from '../../../../utils/time';
|
||||||
import { useOctokit } from '../../../../context/OctokitContext';
|
import { useOctokit } from '../../../../context/OctokitContext';
|
||||||
import { GitCommitDetails, OutletContextType } from '../../../../types/project';
|
import { GitCommitDetails, OutletContextType } from '../../../../types';
|
||||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||||
|
|
||||||
const COMMITS_PER_PAGE = 4;
|
const COMMITS_PER_PAGE = 4;
|
||||||
|
@ -3,7 +3,7 @@ import { Link, Outlet, useLocation, useOutletContext } from 'react-router-dom';
|
|||||||
|
|
||||||
import { Tabs, TabsHeader, TabsBody, Tab } from '@material-tailwind/react';
|
import { Tabs, TabsHeader, TabsBody, Tab } from '@material-tailwind/react';
|
||||||
|
|
||||||
import { OutletContextType } from '../../../../types/project';
|
import { OutletContextType } from '../../../../types';
|
||||||
|
|
||||||
const tabsData = [
|
const tabsData = [
|
||||||
{
|
{
|
||||||
|
@ -7,7 +7,7 @@ import { Button, Typography } from '@material-tailwind/react';
|
|||||||
import DomainCard from '../../../../../components/projects/project/settings/DomainCard';
|
import DomainCard from '../../../../../components/projects/project/settings/DomainCard';
|
||||||
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
||||||
import repositories from '../../../../../assets/repositories.json';
|
import repositories from '../../../../../assets/repositories.json';
|
||||||
import { OutletContextType } from '../../../../../types/project';
|
import { OutletContextType } from '../../../../../types';
|
||||||
|
|
||||||
const Domains = () => {
|
const Domains = () => {
|
||||||
const client = useGQLClient();
|
const client = useGQLClient();
|
||||||
|
@ -17,7 +17,7 @@ import AddEnvironmentVariableRow from '../../../../../components/projects/projec
|
|||||||
import DisplayEnvironmentVariables from '../../../../../components/projects/project/settings/DisplayEnvironmentVariables';
|
import DisplayEnvironmentVariables from '../../../../../components/projects/project/settings/DisplayEnvironmentVariables';
|
||||||
import HorizontalLine from '../../../../../components/HorizontalLine';
|
import HorizontalLine from '../../../../../components/HorizontalLine';
|
||||||
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
||||||
import { EnvironmentVariablesFormValues } from '../../../../../types/project';
|
import { EnvironmentVariablesFormValues } from '../../../../../types';
|
||||||
|
|
||||||
export const EnvironmentVariablesTabPanel = () => {
|
export const EnvironmentVariablesTabPanel = () => {
|
||||||
const { id } = useParams();
|
const { id } = useParams();
|
||||||
|
@ -10,7 +10,7 @@ import DeleteProjectDialog from '../../../../../components/projects/project/sett
|
|||||||
import ConfirmDialog from '../../../../../components/shared/ConfirmDialog';
|
import ConfirmDialog from '../../../../../components/shared/ConfirmDialog';
|
||||||
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
||||||
import AsyncSelect from '../../../../../components/shared/AsyncSelect';
|
import AsyncSelect from '../../../../../components/shared/AsyncSelect';
|
||||||
import { OutletContextType } from '../../../../../types/project';
|
import { OutletContextType } from '../../../../../types';
|
||||||
|
|
||||||
const CopyIcon = ({ value }: { value: string }) => {
|
const CopyIcon = ({ value }: { value: string }) => {
|
||||||
return (
|
return (
|
||||||
|
@ -7,7 +7,7 @@ import { Button, Input, Switch, Typography } from '@material-tailwind/react';
|
|||||||
|
|
||||||
import WebhookCard from '../../../../../components/projects/project/settings/WebhookCard';
|
import WebhookCard from '../../../../../components/projects/project/settings/WebhookCard';
|
||||||
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
||||||
import { OutletContextType } from '../../../../../types/project';
|
import { OutletContextType } from '../../../../../types';
|
||||||
|
|
||||||
type UpdateProdBranchValues = {
|
type UpdateProdBranchValues = {
|
||||||
prodBranch: string;
|
prodBranch: string;
|
||||||
|
@ -8,7 +8,7 @@ import { Chip, Button, Typography } from '@material-tailwind/react';
|
|||||||
import MemberCard from '../../../../../components/projects/project/settings/MemberCard';
|
import MemberCard from '../../../../../components/projects/project/settings/MemberCard';
|
||||||
import AddMemberDialog from '../../../../../components/projects/project/settings/AddMemberDialog';
|
import AddMemberDialog from '../../../../../components/projects/project/settings/AddMemberDialog';
|
||||||
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
import { useGQLClient } from '../../../../../context/GQLClientContext';
|
||||||
import { OutletContextType } from '../../../../../types/project';
|
import { OutletContextType } from '../../../../../types';
|
||||||
|
|
||||||
const FIRST_MEMBER_CARD = 0;
|
const FIRST_MEMBER_CARD = 0;
|
||||||
|
|
||||||
|
@ -1,13 +1,4 @@
|
|||||||
import { Project, Deployment } from 'gql-client';
|
import { Project } from 'gql-client';
|
||||||
|
|
||||||
export interface ProjectDetails extends Project {
|
|
||||||
latestCommit: Commit;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface DeploymentDetails extends Deployment {
|
|
||||||
commit: Commit;
|
|
||||||
author: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface GitOrgDetails {
|
export interface GitOrgDetails {
|
||||||
id: number;
|
id: number;
|
||||||
@ -55,12 +46,6 @@ export enum GitSelect {
|
|||||||
NONE = 'none',
|
NONE = 'none',
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Commit {
|
|
||||||
message: string;
|
|
||||||
createdAt: string;
|
|
||||||
branch: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type OutletContextType = {
|
export type OutletContextType = {
|
||||||
project: Project;
|
project: Project;
|
||||||
onUpdate: () => Promise<void>;
|
onUpdate: () => Promise<void>;
|
@ -84,6 +84,7 @@ query ($organizationSlug: String!) {
|
|||||||
status
|
status
|
||||||
updatedAt
|
updatedAt
|
||||||
commitHash
|
commitHash
|
||||||
|
commitMessage
|
||||||
createdAt
|
createdAt
|
||||||
environment
|
environment
|
||||||
domain {
|
domain {
|
||||||
@ -125,6 +126,7 @@ query ($projectId: String!) {
|
|||||||
}
|
}
|
||||||
branch
|
branch
|
||||||
commitHash
|
commitHash
|
||||||
|
commitMessage
|
||||||
url
|
url
|
||||||
environment
|
environment
|
||||||
isCurrent
|
isCurrent
|
||||||
|
@ -62,6 +62,7 @@ export type Deployment = {
|
|||||||
domain: Domain
|
domain: Domain
|
||||||
branch: string
|
branch: string
|
||||||
commitHash: string
|
commitHash: string
|
||||||
|
commitMessage: string
|
||||||
url: string
|
url: string
|
||||||
environment: Environment
|
environment: Environment
|
||||||
isCurrent: boolean
|
isCurrent: boolean
|
||||||
|
Loading…
Reference in New Issue
Block a user