forked from cerc-io/snowballtools-base
add version backend (temp) (#201)
* add version backend (temp) * staging router endpoint * remove frontend
This commit is contained in:
parent
bfb4a3f30b
commit
dc7b251988
@ -25,8 +25,15 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
return service.getProjectById(projectId);
|
return service.getProjectById(projectId);
|
||||||
},
|
},
|
||||||
|
|
||||||
projectsInOrganization: async (_: any, { organizationSlug }: {organizationSlug: string }, context: any) => {
|
projectsInOrganization: async (
|
||||||
return service.getProjectsInOrganization(context.user, organizationSlug);
|
_: any,
|
||||||
|
{ organizationSlug }: { organizationSlug: string },
|
||||||
|
context: any,
|
||||||
|
) => {
|
||||||
|
return service.getProjectsInOrganization(
|
||||||
|
context.user,
|
||||||
|
organizationSlug,
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
deployments: async (_: any, { projectId }: { projectId: string }) => {
|
deployments: async (_: any, { projectId }: { projectId: string }) => {
|
||||||
@ -35,7 +42,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
environmentVariables: async (
|
environmentVariables: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId }: { projectId: string }
|
{ projectId }: { projectId: string },
|
||||||
) => {
|
) => {
|
||||||
return service.getEnvironmentVariablesByProjectId(projectId);
|
return service.getEnvironmentVariablesByProjectId(projectId);
|
||||||
},
|
},
|
||||||
@ -44,7 +51,11 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
return service.getProjectMembersByProjectId(projectId);
|
return service.getProjectMembersByProjectId(projectId);
|
||||||
},
|
},
|
||||||
|
|
||||||
searchProjects: async (_: any, { searchText }: { searchText: string }, context: any) => {
|
searchProjects: async (
|
||||||
|
_: any,
|
||||||
|
{ searchText }: { searchText: string },
|
||||||
|
context: any,
|
||||||
|
) => {
|
||||||
return service.searchProjects(context.user, searchText);
|
return service.searchProjects(context.user, searchText);
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -52,11 +63,11 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
filter
|
filter,
|
||||||
}: { projectId: string; filter?: FindOptionsWhere<Domain> }
|
}: { projectId: string; filter?: FindOptionsWhere<Domain> },
|
||||||
) => {
|
) => {
|
||||||
return service.getDomainsByProjectId(projectId, filter);
|
return service.getDomainsByProjectId(projectId, filter);
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
// TODO: Return error in GQL response
|
// TODO: Return error in GQL response
|
||||||
@ -64,10 +75,13 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
removeProjectMember: async (
|
removeProjectMember: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectMemberId }: { projectMemberId: string },
|
{ projectMemberId }: { projectMemberId: string },
|
||||||
context: any
|
context: any,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.removeProjectMember(context.user, projectMemberId);
|
return await service.removeProjectMember(
|
||||||
|
context.user,
|
||||||
|
projectMemberId,
|
||||||
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
@ -78,13 +92,13 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectMemberId,
|
projectMemberId,
|
||||||
data
|
data,
|
||||||
}: {
|
}: {
|
||||||
projectMemberId: string;
|
projectMemberId: string;
|
||||||
data: {
|
data: {
|
||||||
permissions: Permission[];
|
permissions: Permission[];
|
||||||
};
|
};
|
||||||
}
|
},
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateProjectMember(projectMemberId, data);
|
return await service.updateProjectMember(projectMemberId, data);
|
||||||
@ -98,14 +112,14 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
data: {
|
data: {
|
||||||
email: string;
|
email: string;
|
||||||
permissions: Permission[];
|
permissions: Permission[];
|
||||||
};
|
};
|
||||||
}
|
},
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.addProjectMember(projectId, data));
|
return Boolean(await service.addProjectMember(projectId, data));
|
||||||
@ -119,15 +133,15 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
data: { environments: string[]; key: string; value: string }[];
|
data: { environments: string[]; key: string; value: string }[];
|
||||||
}
|
},
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(
|
return Boolean(
|
||||||
await service.addEnvironmentVariables(projectId, data)
|
await service.addEnvironmentVariables(projectId, data),
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
@ -139,16 +153,16 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
environmentVariableId,
|
environmentVariableId,
|
||||||
data
|
data,
|
||||||
}: {
|
}: {
|
||||||
environmentVariableId: string;
|
environmentVariableId: string;
|
||||||
data: DeepPartial<EnvironmentVariable>;
|
data: DeepPartial<EnvironmentVariable>;
|
||||||
}
|
},
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateEnvironmentVariable(
|
return await service.updateEnvironmentVariable(
|
||||||
environmentVariableId,
|
environmentVariableId,
|
||||||
data
|
data,
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
@ -158,7 +172,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
removeEnvironmentVariable: async (
|
removeEnvironmentVariable: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ environmentVariableId }: { environmentVariableId: string }
|
{ environmentVariableId }: { environmentVariableId: string },
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.removeEnvironmentVariable(environmentVariableId);
|
return await service.removeEnvironmentVariable(environmentVariableId);
|
||||||
@ -171,10 +185,12 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
updateDeploymentToProd: async (
|
updateDeploymentToProd: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ deploymentId }: { deploymentId: string },
|
{ deploymentId }: { deploymentId: string },
|
||||||
context: any
|
context: any,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.updateDeploymentToProd(context.user, deploymentId));
|
return Boolean(
|
||||||
|
await service.updateDeploymentToProd(context.user, deploymentId),
|
||||||
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
@ -185,9 +201,9 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
organizationSlug,
|
organizationSlug,
|
||||||
data
|
data,
|
||||||
}: { organizationSlug: string; data: DeepPartial<Project> },
|
}: { organizationSlug: string; data: DeepPartial<Project> },
|
||||||
context: any
|
context: any,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.addProject(context.user, organizationSlug, data);
|
return await service.addProject(context.user, organizationSlug, data);
|
||||||
@ -199,7 +215,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
updateProject: async (
|
updateProject: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId, data }: { projectId: string; data: DeepPartial<Project> }
|
{ projectId, data }: { projectId: string; data: DeepPartial<Project> },
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateProject(projectId, data);
|
return await service.updateProject(projectId, data);
|
||||||
@ -212,10 +228,12 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
redeployToProd: async (
|
redeployToProd: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ deploymentId }: { deploymentId: string },
|
{ deploymentId }: { deploymentId: string },
|
||||||
context: any
|
context: any,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.redeployToProd(context.user, deploymentId));
|
return Boolean(
|
||||||
|
await service.redeployToProd(context.user, deploymentId),
|
||||||
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
@ -244,8 +262,8 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
deploymentId
|
deploymentId,
|
||||||
}: { deploymentId: string; projectId: string }
|
}: { deploymentId: string; projectId: string },
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.rollbackDeployment(projectId, deploymentId);
|
return await service.rollbackDeployment(projectId, deploymentId);
|
||||||
@ -257,9 +275,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
deleteDeployment: async (
|
deleteDeployment: async (
|
||||||
_: any,
|
_: any,
|
||||||
{
|
{ deploymentId }: { deploymentId: string },
|
||||||
deploymentId
|
|
||||||
}: { deploymentId: string; }
|
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.deleteDeployment(deploymentId);
|
return await service.deleteDeployment(deploymentId);
|
||||||
@ -271,7 +287,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
addDomain: async (
|
addDomain: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId, data }: { projectId: string; data: { name: string } }
|
{ projectId, data }: { projectId: string; data: { name: string } },
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.addDomain(projectId, data));
|
return Boolean(await service.addDomain(projectId, data));
|
||||||
@ -283,7 +299,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
updateDomain: async (
|
updateDomain: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ domainId, data }: { domainId: string; data: DeepPartial<Domain> }
|
{ domainId, data }: { domainId: string; data: DeepPartial<Domain> },
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateDomain(domainId, data);
|
return await service.updateDomain(domainId, data);
|
||||||
@ -296,7 +312,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
authenticateGitHub: async (
|
authenticateGitHub: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ code }: { code: string },
|
{ code }: { code: string },
|
||||||
context: any
|
context: any,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.authenticateGitHub(code, context.user);
|
return await service.authenticateGitHub(code, context.user);
|
||||||
@ -308,12 +324,14 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
unauthenticateGitHub: async (_: any, __: object, context: any) => {
|
unauthenticateGitHub: async (_: any, __: object, context: any) => {
|
||||||
try {
|
try {
|
||||||
return service.unauthenticateGitHub(context.user, { gitHubToken: null });
|
return service.unauthenticateGitHub(context.user, {
|
||||||
|
gitHubToken: null,
|
||||||
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
9
packages/backend/src/routes/staging.ts
Normal file
9
packages/backend/src/routes/staging.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import { Router } from 'express';
|
||||||
|
|
||||||
|
const router = Router();
|
||||||
|
|
||||||
|
router.get('/version', async (req, res) => {
|
||||||
|
return res.send({ version: '0.0.2' });
|
||||||
|
});
|
||||||
|
|
||||||
|
export default router;
|
@ -17,6 +17,7 @@ import { ServerConfig } from './config';
|
|||||||
import { DEFAULT_GQL_PATH } from './constants';
|
import { DEFAULT_GQL_PATH } from './constants';
|
||||||
import githubRouter from './routes/github';
|
import githubRouter from './routes/github';
|
||||||
import authRouter from './routes/auth';
|
import authRouter from './routes/auth';
|
||||||
|
import stagingRouter from './routes/staging';
|
||||||
import { Service } from './service';
|
import { Service } from './service';
|
||||||
|
|
||||||
const log = debug('snowball:server');
|
const log = debug('snowball:server');
|
||||||
@ -109,6 +110,7 @@ export const createAndStartServer = async (
|
|||||||
app.set('service', service);
|
app.set('service', service);
|
||||||
app.use('/auth', authRouter);
|
app.use('/auth', authRouter);
|
||||||
app.use('/api/github', githubRouter);
|
app.use('/api/github', githubRouter);
|
||||||
|
app.use('/staging', stagingRouter);
|
||||||
|
|
||||||
httpServer.listen(port, host, () => {
|
httpServer.listen(port, host, () => {
|
||||||
log(`Server is listening on ${host}:${port}${server.graphqlPath}`);
|
log(`Server is listening on ${host}:${port}${server.graphqlPath}`);
|
||||||
|
@ -15,7 +15,12 @@ import { Permission, ProjectMember } from './entity/ProjectMember';
|
|||||||
import { User } from './entity/User';
|
import { User } from './entity/User';
|
||||||
import { Registry } from './registry';
|
import { Registry } from './registry';
|
||||||
import { GitHubConfig, RegistryConfig } from './config';
|
import { GitHubConfig, RegistryConfig } from './config';
|
||||||
import { AppDeploymentRecord, AppDeploymentRemovalRecord, GitPushEventPayload, PackageJSON } from './types';
|
import {
|
||||||
|
AppDeploymentRecord,
|
||||||
|
AppDeploymentRemovalRecord,
|
||||||
|
GitPushEventPayload,
|
||||||
|
PackageJSON,
|
||||||
|
} from './types';
|
||||||
import { Role } from './entity/UserOrganization';
|
import { Role } from './entity/UserOrganization';
|
||||||
|
|
||||||
const log = debug('snowball:service');
|
const log = debug('snowball:service');
|
||||||
@ -71,33 +76,32 @@ export class Service {
|
|||||||
// Fetch deployments in building state
|
// Fetch deployments in building state
|
||||||
const deployments = await this.db.getDeployments({
|
const deployments = await this.db.getDeployments({
|
||||||
where: {
|
where: {
|
||||||
status: DeploymentStatus.Building
|
status: DeploymentStatus.Building,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (deployments.length) {
|
if (deployments.length) {
|
||||||
log(
|
log(
|
||||||
`Found ${deployments.length} deployments in ${DeploymentStatus.Building} state`
|
`Found ${deployments.length} deployments in ${DeploymentStatus.Building} state`,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Calculate a timestamp for one hour ago
|
// Calculate a timestamp for one hour ago
|
||||||
const anHourAgo = Date.now() - HOUR;
|
const anHourAgo = Date.now() - HOUR;
|
||||||
|
|
||||||
// Filter out deployments started more than an hour ago and mark them as Error
|
// Filter out deployments started more than an hour ago and mark them as Error
|
||||||
const oldDeploymentsToUpdate = deployments.filter(
|
const oldDeploymentsToUpdate = deployments
|
||||||
deployment => (Number(deployment.updatedAt) < anHourAgo)
|
.filter((deployment) => Number(deployment.updatedAt) < anHourAgo)
|
||||||
)
|
|
||||||
.map((deployment) => {
|
.map((deployment) => {
|
||||||
return this.db.updateDeploymentById(deployment.id, {
|
return this.db.updateDeploymentById(deployment.id, {
|
||||||
status: DeploymentStatus.Error,
|
status: DeploymentStatus.Error,
|
||||||
isCurrent: false
|
isCurrent: false,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// If there are old deployments to update, log and perform the updates
|
// If there are old deployments to update, log and perform the updates
|
||||||
if (oldDeploymentsToUpdate.length > 0) {
|
if (oldDeploymentsToUpdate.length > 0) {
|
||||||
log(
|
log(
|
||||||
`Cleaning up ${oldDeploymentsToUpdate.length} deployments stuck in ${DeploymentStatus.Building} state for over an hour`
|
`Cleaning up ${oldDeploymentsToUpdate.length} deployments stuck in ${DeploymentStatus.Building} state for over an hour`,
|
||||||
);
|
);
|
||||||
await Promise.all(oldDeploymentsToUpdate);
|
await Promise.all(oldDeploymentsToUpdate);
|
||||||
}
|
}
|
||||||
@ -125,17 +129,18 @@ export class Service {
|
|||||||
// Fetch deployments in deleting state
|
// Fetch deployments in deleting state
|
||||||
const deployments = await this.db.getDeployments({
|
const deployments = await this.db.getDeployments({
|
||||||
where: {
|
where: {
|
||||||
status: DeploymentStatus.Deleting
|
status: DeploymentStatus.Deleting,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (deployments.length) {
|
if (deployments.length) {
|
||||||
log(
|
log(
|
||||||
`Found ${deployments.length} deployments in ${DeploymentStatus.Deleting} state`
|
`Found ${deployments.length} deployments in ${DeploymentStatus.Deleting} state`,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Fetch ApplicationDeploymentRemovalRecords for deployments
|
// Fetch ApplicationDeploymentRemovalRecords for deployments
|
||||||
const records = await this.registry.getDeploymentRemovalRecords(deployments);
|
const records =
|
||||||
|
await this.registry.getDeploymentRemovalRecords(deployments);
|
||||||
log(`Found ${records.length} ApplicationDeploymentRemovalRecords`);
|
log(`Found ${records.length} ApplicationDeploymentRemovalRecords`);
|
||||||
|
|
||||||
// Update deployments for which ApplicationDeploymentRemovalRecords were returned
|
// Update deployments for which ApplicationDeploymentRemovalRecords were returned
|
||||||
@ -153,16 +158,16 @@ export class Service {
|
|||||||
* Update deployments with ApplicationDeploymentRecord data
|
* Update deployments with ApplicationDeploymentRecord data
|
||||||
*/
|
*/
|
||||||
async updateDeploymentsWithRecordData(
|
async updateDeploymentsWithRecordData(
|
||||||
records: AppDeploymentRecord[]
|
records: AppDeploymentRecord[],
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
// Get deployments for ApplicationDeploymentRecords
|
// Get deployments for ApplicationDeploymentRecords
|
||||||
const deployments = await this.db.getDeployments({
|
const deployments = await this.db.getDeployments({
|
||||||
where: records.map((record) => ({
|
where: records.map((record) => ({
|
||||||
applicationRecordId: record.attributes.application
|
applicationRecordId: record.attributes.application,
|
||||||
})),
|
})),
|
||||||
order: {
|
order: {
|
||||||
createdAt: 'DESC'
|
createdAt: 'DESC',
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
// Get project IDs of deployments that are in production environment
|
// Get project IDs of deployments that are in production environment
|
||||||
@ -174,13 +179,13 @@ export class Service {
|
|||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
},
|
},
|
||||||
new Set<string>()
|
new Set<string>(),
|
||||||
);
|
);
|
||||||
|
|
||||||
// Set old deployments isCurrent to false
|
// Set old deployments isCurrent to false
|
||||||
await this.db.updateDeploymentsByProjectIds(
|
await this.db.updateDeploymentsByProjectIds(
|
||||||
Array.from(productionDeploymentProjectIds),
|
Array.from(productionDeploymentProjectIds),
|
||||||
{ isCurrent: false }
|
{ isCurrent: false },
|
||||||
);
|
);
|
||||||
|
|
||||||
const recordToDeploymentsMap = deployments.reduce(
|
const recordToDeploymentsMap = deployments.reduce(
|
||||||
@ -188,7 +193,7 @@ export class Service {
|
|||||||
acc[deployment.applicationRecordId] = deployment;
|
acc[deployment.applicationRecordId] = deployment;
|
||||||
return acc;
|
return acc;
|
||||||
},
|
},
|
||||||
{}
|
{},
|
||||||
);
|
);
|
||||||
|
|
||||||
// Update deployment data for ApplicationDeploymentRecords
|
// Update deployment data for ApplicationDeploymentRecords
|
||||||
@ -200,11 +205,11 @@ export class Service {
|
|||||||
applicationDeploymentRecordData: record.attributes,
|
applicationDeploymentRecordData: record.attributes,
|
||||||
url: record.attributes.url,
|
url: record.attributes.url,
|
||||||
status: DeploymentStatus.Ready,
|
status: DeploymentStatus.Ready,
|
||||||
isCurrent: deployment.environment === Environment.Production
|
isCurrent: deployment.environment === Environment.Production,
|
||||||
});
|
});
|
||||||
|
|
||||||
log(
|
log(
|
||||||
`Updated deployment ${deployment.id} with URL ${record.attributes.url}`
|
`Updated deployment ${deployment.id} with URL ${record.attributes.url}`,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -218,17 +223,23 @@ export class Service {
|
|||||||
records: AppDeploymentRemovalRecord[],
|
records: AppDeploymentRemovalRecord[],
|
||||||
deployments: Deployment[],
|
deployments: Deployment[],
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
const removedApplicationDeploymentRecordIds = records.map(record => record.attributes.deployment);
|
const removedApplicationDeploymentRecordIds = records.map(
|
||||||
|
(record) => record.attributes.deployment,
|
||||||
|
);
|
||||||
|
|
||||||
// Get removed deployments for ApplicationDeploymentRecords
|
// Get removed deployments for ApplicationDeploymentRecords
|
||||||
const removedDeployments = deployments.filter(deployment => removedApplicationDeploymentRecordIds.includes(deployment.applicationDeploymentRecordId!))
|
const removedDeployments = deployments.filter((deployment) =>
|
||||||
|
removedApplicationDeploymentRecordIds.includes(
|
||||||
|
deployment.applicationDeploymentRecordId!,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
const recordToDeploymentsMap = removedDeployments.reduce(
|
const recordToDeploymentsMap = removedDeployments.reduce(
|
||||||
(acc: { [key: string]: Deployment }, deployment) => {
|
(acc: { [key: string]: Deployment }, deployment) => {
|
||||||
acc[deployment.applicationDeploymentRecordId!] = deployment;
|
acc[deployment.applicationDeploymentRecordId!] = deployment;
|
||||||
return acc;
|
return acc;
|
||||||
},
|
},
|
||||||
{}
|
{},
|
||||||
);
|
);
|
||||||
|
|
||||||
// Update deployment data for ApplicationDeploymentRecords and delete
|
// Update deployment data for ApplicationDeploymentRecords and delete
|
||||||
@ -241,10 +252,10 @@ export class Service {
|
|||||||
});
|
});
|
||||||
|
|
||||||
log(
|
log(
|
||||||
`Updated deployment ${deployment.id} with ApplicationDeploymentRemovalRecord ${record.id}`
|
`Updated deployment ${deployment.id} with ApplicationDeploymentRemovalRecord ${record.id}`,
|
||||||
);
|
);
|
||||||
|
|
||||||
await this.db.deleteDeploymentById(deployment.id)
|
await this.db.deleteDeploymentById(deployment.id);
|
||||||
});
|
});
|
||||||
|
|
||||||
await Promise.all(deploymentUpdatePromises);
|
await Promise.all(deploymentUpdatePromises);
|
||||||
@ -253,41 +264,41 @@ export class Service {
|
|||||||
async getUser(userId: string): Promise<User | null> {
|
async getUser(userId: string): Promise<User | null> {
|
||||||
return this.db.getUser({
|
return this.db.getUser({
|
||||||
where: {
|
where: {
|
||||||
id: userId
|
id: userId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUserByEmail(email: string): Promise<User | null> {
|
async getUserByEmail(email: string): Promise<User | null> {
|
||||||
return await this.db.getUser({
|
return await this.db.getUser({
|
||||||
where: {
|
where: {
|
||||||
email
|
email,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUserBySubOrgId(subOrgId: string): Promise<User | null> {
|
async getUserBySubOrgId(subOrgId: string): Promise<User | null> {
|
||||||
return await this.db.getUser({
|
return await this.db.getUser({
|
||||||
where: {
|
where: {
|
||||||
subOrgId
|
subOrgId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUserByEthAddress(ethAddress: string): Promise<User | null> {
|
async getUserByEthAddress(ethAddress: string): Promise<User | null> {
|
||||||
return await this.db.getUser({
|
return await this.db.getUser({
|
||||||
where: {
|
where: {
|
||||||
ethAddress
|
ethAddress,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async createUser(params: {
|
async createUser(params: {
|
||||||
name: string
|
name: string;
|
||||||
email: string
|
email: string;
|
||||||
subOrgId: string
|
subOrgId: string;
|
||||||
ethAddress: string
|
ethAddress: string;
|
||||||
turnkeyWalletId: string
|
turnkeyWalletId: string;
|
||||||
}): Promise<User> {
|
}): Promise<User> {
|
||||||
const [org] = await this.db.getOrganizations({});
|
const [org] = await this.db.getOrganizations({});
|
||||||
assert(org, 'No organizations exists in database');
|
assert(org, 'No organizations exists in database');
|
||||||
@ -305,7 +316,7 @@ export class Service {
|
|||||||
await this.db.addUserOrganization({
|
await this.db.addUserOrganization({
|
||||||
member: user,
|
member: user,
|
||||||
organization: org,
|
organization: org,
|
||||||
role: Role.Owner
|
role: Role.Owner,
|
||||||
});
|
});
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
@ -315,7 +326,7 @@ export class Service {
|
|||||||
const user = await this.db.getUser({ where: { id: userId } });
|
const user = await this.db.getUser({ where: { id: userId } });
|
||||||
assert(
|
assert(
|
||||||
user && user.gitHubToken,
|
user && user.gitHubToken,
|
||||||
'User needs to be authenticated with GitHub token'
|
'User needs to be authenticated with GitHub token',
|
||||||
);
|
);
|
||||||
|
|
||||||
return new Octokit({ auth: user.gitHubToken });
|
return new Octokit({ auth: user.gitHubToken });
|
||||||
@ -331,8 +342,14 @@ export class Service {
|
|||||||
return dbProject;
|
return dbProject;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectsInOrganization (user: User, organizationSlug: string): Promise<Project[]> {
|
async getProjectsInOrganization(
|
||||||
const dbProjects = await this.db.getProjectsInOrganization(user.id, organizationSlug);
|
user: User,
|
||||||
|
organizationSlug: string,
|
||||||
|
): Promise<Project[]> {
|
||||||
|
const dbProjects = await this.db.getProjectsInOrganization(
|
||||||
|
user.id,
|
||||||
|
organizationSlug,
|
||||||
|
);
|
||||||
return dbProjects;
|
return dbProjects;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -342,7 +359,7 @@ export class Service {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async getEnvironmentVariablesByProjectId(
|
async getEnvironmentVariablesByProjectId(
|
||||||
projectId: string
|
projectId: string,
|
||||||
): Promise<EnvironmentVariable[]> {
|
): Promise<EnvironmentVariable[]> {
|
||||||
const dbEnvironmentVariables =
|
const dbEnvironmentVariables =
|
||||||
await this.db.getEnvironmentVariablesByProjectId(projectId);
|
await this.db.getEnvironmentVariablesByProjectId(projectId);
|
||||||
@ -350,7 +367,7 @@ export class Service {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async getProjectMembersByProjectId(
|
async getProjectMembersByProjectId(
|
||||||
projectId: string
|
projectId: string,
|
||||||
): Promise<ProjectMember[]> {
|
): Promise<ProjectMember[]> {
|
||||||
const dbProjectMembers =
|
const dbProjectMembers =
|
||||||
await this.db.getProjectMembersByProjectId(projectId);
|
await this.db.getProjectMembersByProjectId(projectId);
|
||||||
@ -358,13 +375,16 @@ export class Service {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async searchProjects(user: User, searchText: string): Promise<Project[]> {
|
async searchProjects(user: User, searchText: string): Promise<Project[]> {
|
||||||
const dbProjects = await this.db.getProjectsBySearchText(user.id, searchText);
|
const dbProjects = await this.db.getProjectsBySearchText(
|
||||||
|
user.id,
|
||||||
|
searchText,
|
||||||
|
);
|
||||||
return dbProjects;
|
return dbProjects;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDomainsByProjectId(
|
async getDomainsByProjectId(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
filter?: FindOptionsWhere<Domain>
|
filter?: FindOptionsWhere<Domain>,
|
||||||
): Promise<Domain[]> {
|
): Promise<Domain[]> {
|
||||||
const dbDomains = await this.db.getDomainsByProjectId(projectId, filter);
|
const dbDomains = await this.db.getDomainsByProjectId(projectId, filter);
|
||||||
return dbDomains;
|
return dbDomains;
|
||||||
@ -372,7 +392,7 @@ export class Service {
|
|||||||
|
|
||||||
async updateProjectMember(
|
async updateProjectMember(
|
||||||
projectMemberId: string,
|
projectMemberId: string,
|
||||||
data: { permissions: Permission[] }
|
data: { permissions: Permission[] },
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
return this.db.updateProjectMemberById(projectMemberId, data);
|
return this.db.updateProjectMemberById(projectMemberId, data);
|
||||||
}
|
}
|
||||||
@ -382,36 +402,39 @@ export class Service {
|
|||||||
data: {
|
data: {
|
||||||
email: string;
|
email: string;
|
||||||
permissions: Permission[];
|
permissions: Permission[];
|
||||||
}
|
},
|
||||||
): Promise<ProjectMember> {
|
): Promise<ProjectMember> {
|
||||||
// TODO: Send invitation
|
// TODO: Send invitation
|
||||||
let user = await this.db.getUser({
|
let user = await this.db.getUser({
|
||||||
where: {
|
where: {
|
||||||
email: data.email
|
email: data.email,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!user) {
|
if (!user) {
|
||||||
user = await this.db.addUser({
|
user = await this.db.addUser({
|
||||||
email: data.email
|
email: data.email,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const newProjectMember = await this.db.addProjectMember({
|
const newProjectMember = await this.db.addProjectMember({
|
||||||
project: {
|
project: {
|
||||||
id: projectId
|
id: projectId,
|
||||||
},
|
},
|
||||||
permissions: data.permissions,
|
permissions: data.permissions,
|
||||||
isPending: true,
|
isPending: true,
|
||||||
member: {
|
member: {
|
||||||
id: user.id
|
id: user.id,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return newProjectMember;
|
return newProjectMember;
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeProjectMember (user: User, projectMemberId: string): Promise<boolean> {
|
async removeProjectMember(
|
||||||
|
user: User,
|
||||||
|
projectMemberId: string,
|
||||||
|
): Promise<boolean> {
|
||||||
const member = await this.db.getProjectMemberById(projectMemberId);
|
const member = await this.db.getProjectMemberById(projectMemberId);
|
||||||
|
|
||||||
if (String(member.member.id) === user.id) {
|
if (String(member.member.id) === user.id) {
|
||||||
@ -430,7 +453,7 @@ export class Service {
|
|||||||
|
|
||||||
async addEnvironmentVariables(
|
async addEnvironmentVariables(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
data: { environments: string[]; key: string; value: string }[]
|
data: { environments: string[]; key: string; value: string }[],
|
||||||
): Promise<EnvironmentVariable[]> {
|
): Promise<EnvironmentVariable[]> {
|
||||||
const formattedEnvironmentVariables = data
|
const formattedEnvironmentVariables = data
|
||||||
.map((environmentVariable) => {
|
.map((environmentVariable) => {
|
||||||
@ -440,38 +463,41 @@ export class Service {
|
|||||||
value: environmentVariable.value,
|
value: environmentVariable.value,
|
||||||
environment: environment as Environment,
|
environment: environment as Environment,
|
||||||
project: Object.assign(new Project(), {
|
project: Object.assign(new Project(), {
|
||||||
id: projectId
|
id: projectId,
|
||||||
})
|
}),
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.flat();
|
.flat();
|
||||||
|
|
||||||
const savedEnvironmentVariables = await this.db.addEnvironmentVariables(
|
const savedEnvironmentVariables = await this.db.addEnvironmentVariables(
|
||||||
formattedEnvironmentVariables
|
formattedEnvironmentVariables,
|
||||||
);
|
);
|
||||||
return savedEnvironmentVariables;
|
return savedEnvironmentVariables;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateEnvironmentVariable(
|
async updateEnvironmentVariable(
|
||||||
environmentVariableId: string,
|
environmentVariableId: string,
|
||||||
data: DeepPartial<EnvironmentVariable>
|
data: DeepPartial<EnvironmentVariable>,
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
return this.db.updateEnvironmentVariable(environmentVariableId, data);
|
return this.db.updateEnvironmentVariable(environmentVariableId, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeEnvironmentVariable(
|
async removeEnvironmentVariable(
|
||||||
environmentVariableId: string
|
environmentVariableId: string,
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
return this.db.deleteEnvironmentVariable(environmentVariableId);
|
return this.db.deleteEnvironmentVariable(environmentVariableId);
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDeploymentToProd (user: User, deploymentId: string): Promise<Deployment> {
|
async updateDeploymentToProd(
|
||||||
|
user: User,
|
||||||
|
deploymentId: string,
|
||||||
|
): Promise<Deployment> {
|
||||||
const oldDeployment = await this.db.getDeployment({
|
const oldDeployment = await this.db.getDeployment({
|
||||||
where: { id: deploymentId },
|
where: { id: deploymentId },
|
||||||
relations: {
|
relations: {
|
||||||
project: true
|
project: true,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!oldDeployment) {
|
if (!oldDeployment) {
|
||||||
@ -480,20 +506,18 @@ export class Service {
|
|||||||
|
|
||||||
const prodBranchDomains = await this.db.getDomainsByProjectId(
|
const prodBranchDomains = await this.db.getDomainsByProjectId(
|
||||||
oldDeployment.project.id,
|
oldDeployment.project.id,
|
||||||
{ branch: oldDeployment.project.prodBranch }
|
{ branch: oldDeployment.project.prodBranch },
|
||||||
);
|
);
|
||||||
|
|
||||||
const octokit = await this.getOctokit(user.id);
|
const octokit = await this.getOctokit(user.id);
|
||||||
|
|
||||||
const newDeployment = await this.createDeployment(user.id,
|
const newDeployment = await this.createDeployment(user.id, octokit, {
|
||||||
octokit,
|
|
||||||
{
|
|
||||||
project: oldDeployment.project,
|
project: oldDeployment.project,
|
||||||
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
|
commitMessage: oldDeployment.commitMessage,
|
||||||
});
|
});
|
||||||
|
|
||||||
return newDeployment;
|
return newDeployment;
|
||||||
@ -502,11 +526,11 @@ export class Service {
|
|||||||
async createDeployment(
|
async createDeployment(
|
||||||
userId: string,
|
userId: string,
|
||||||
octokit: Octokit,
|
octokit: Octokit,
|
||||||
data: DeepPartial<Deployment>
|
data: DeepPartial<Deployment>,
|
||||||
): Promise<Deployment> {
|
): Promise<Deployment> {
|
||||||
assert(data.project?.repository, 'Project repository not found');
|
assert(data.project?.repository, 'Project repository not found');
|
||||||
log(
|
log(
|
||||||
`Creating deployment in project ${data.project.name} from branch ${data.branch}`
|
`Creating deployment in project ${data.project.name} from branch ${data.branch}`,
|
||||||
);
|
);
|
||||||
const [owner, repo] = data.project.repository.split('/');
|
const [owner, repo] = data.project.repository.split('/');
|
||||||
|
|
||||||
@ -514,7 +538,7 @@ export class Service {
|
|||||||
owner,
|
owner,
|
||||||
repo,
|
repo,
|
||||||
path: 'package.json',
|
path: 'package.json',
|
||||||
ref: data.commitHash
|
ref: data.commitHash,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!packageJSONData) {
|
if (!packageJSONData) {
|
||||||
@ -526,10 +550,12 @@ export class Service {
|
|||||||
|
|
||||||
assert(packageJSON.name, "name field doesn't exist in package.json");
|
assert(packageJSON.name, "name field doesn't exist in package.json");
|
||||||
|
|
||||||
const repoUrl = (await octokit.rest.repos.get({
|
const repoUrl = (
|
||||||
|
await octokit.rest.repos.get({
|
||||||
owner,
|
owner,
|
||||||
repo
|
repo,
|
||||||
})).data.html_url;
|
})
|
||||||
|
).data.html_url;
|
||||||
|
|
||||||
// TODO: Set environment variables for each deployment (environment variables can`t be set in application record)
|
// TODO: Set environment variables for each deployment (environment variables can`t be set in application record)
|
||||||
const { applicationRecordId, applicationRecordData } =
|
const { applicationRecordId, applicationRecordData } =
|
||||||
@ -538,17 +564,20 @@ export class Service {
|
|||||||
packageJSON,
|
packageJSON,
|
||||||
appType: data.project!.template!,
|
appType: data.project!.template!,
|
||||||
commitHash: data.commitHash!,
|
commitHash: data.commitHash!,
|
||||||
repoUrl
|
repoUrl,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Update previous deployment with prod branch domain
|
// Update previous deployment with prod branch domain
|
||||||
// TODO: Fix unique constraint error for domain
|
// TODO: Fix unique constraint error for domain
|
||||||
if (data.domain) {
|
if (data.domain) {
|
||||||
await this.db.updateDeployment({
|
await this.db.updateDeployment(
|
||||||
domainId: data.domain.id
|
{
|
||||||
}, {
|
domainId: data.domain.id,
|
||||||
domain: null
|
},
|
||||||
});
|
{
|
||||||
|
domain: null,
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const newDeployment = await this.db.addDeployment({
|
const newDeployment = await this.db.addDeployment({
|
||||||
@ -562,53 +591,67 @@ export class Service {
|
|||||||
applicationRecordData,
|
applicationRecordData,
|
||||||
domain: data.domain,
|
domain: data.domain,
|
||||||
createdBy: Object.assign(new User(), {
|
createdBy: Object.assign(new User(), {
|
||||||
id: userId
|
id: userId,
|
||||||
})
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
log(`Created deployment ${newDeployment.id} and published application record ${applicationRecordId}`);
|
log(
|
||||||
|
`Created deployment ${newDeployment.id} and published application record ${applicationRecordId}`,
|
||||||
|
);
|
||||||
|
|
||||||
const environmentVariables = await this.db.getEnvironmentVariablesByProjectId(data.project.id!, { environment: Environment.Production });
|
const environmentVariables =
|
||||||
|
await this.db.getEnvironmentVariablesByProjectId(data.project.id!, {
|
||||||
|
environment: Environment.Production,
|
||||||
|
});
|
||||||
|
|
||||||
const environmentVariablesObj = environmentVariables.reduce((acc, env) => {
|
const environmentVariablesObj = environmentVariables.reduce(
|
||||||
|
(acc, env) => {
|
||||||
acc[env.key] = env.value;
|
acc[env.key] = env.value;
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, {} as { [key: string]: string });
|
},
|
||||||
|
{} as { [key: string]: string },
|
||||||
|
);
|
||||||
|
|
||||||
// To set project DNS
|
// To set project DNS
|
||||||
if (data.environment === Environment.Production) {
|
if (data.environment === Environment.Production) {
|
||||||
// On deleting deployment later, project DNS deployment is also deleted
|
// On deleting deployment later, project DNS deployment is also deleted
|
||||||
// So publish project DNS deployment first so that ApplicationDeploymentRecord for the same is available when deleting deployment later
|
// So publish project DNS deployment first so that ApplicationDeploymentRecord for the same is available when deleting deployment later
|
||||||
await this.registry.createApplicationDeploymentRequest(
|
await this.registry.createApplicationDeploymentRequest({
|
||||||
{
|
|
||||||
deployment: newDeployment,
|
deployment: newDeployment,
|
||||||
appName: repo,
|
appName: repo,
|
||||||
repository: repoUrl,
|
repository: repoUrl,
|
||||||
environmentVariables: environmentVariablesObj,
|
environmentVariables: environmentVariablesObj,
|
||||||
dns: `${newDeployment.project.name}`
|
dns: `${newDeployment.project.name}`,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const { applicationDeploymentRequestId, applicationDeploymentRequestData } = await this.registry.createApplicationDeploymentRequest(
|
const { applicationDeploymentRequestId, applicationDeploymentRequestData } =
|
||||||
{
|
await this.registry.createApplicationDeploymentRequest({
|
||||||
deployment: newDeployment,
|
deployment: newDeployment,
|
||||||
appName: repo,
|
appName: repo,
|
||||||
repository: repoUrl,
|
repository: repoUrl,
|
||||||
environmentVariables: environmentVariablesObj,
|
environmentVariables: environmentVariablesObj,
|
||||||
dns: `${newDeployment.project.name}-${newDeployment.id}`
|
dns: `${newDeployment.project.name}-${newDeployment.id}`,
|
||||||
});
|
});
|
||||||
|
|
||||||
await this.db.updateDeploymentById(newDeployment.id, { applicationDeploymentRequestId, applicationDeploymentRequestData });
|
await this.db.updateDeploymentById(newDeployment.id, {
|
||||||
|
applicationDeploymentRequestId,
|
||||||
|
applicationDeploymentRequestData,
|
||||||
|
});
|
||||||
|
|
||||||
return newDeployment;
|
return newDeployment;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addProject (user: User, organizationSlug: string, data: DeepPartial<Project>): Promise<Project | undefined> {
|
async addProject(
|
||||||
|
user: User,
|
||||||
|
organizationSlug: string,
|
||||||
|
data: DeepPartial<Project>,
|
||||||
|
): Promise<Project | undefined> {
|
||||||
const organization = await this.db.getOrganization({
|
const organization = await this.db.getOrganization({
|
||||||
where: {
|
where: {
|
||||||
slug: organizationSlug
|
slug: organizationSlug,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
if (!organization) {
|
if (!organization) {
|
||||||
throw new Error('Organization does not exist');
|
throw new Error('Organization does not exist');
|
||||||
@ -620,26 +663,23 @@ export class Service {
|
|||||||
const [owner, repo] = project.repository.split('/');
|
const [owner, repo] = project.repository.split('/');
|
||||||
|
|
||||||
const {
|
const {
|
||||||
data: [latestCommit]
|
data: [latestCommit],
|
||||||
} = await octokit.rest.repos.listCommits({
|
} = await octokit.rest.repos.listCommits({
|
||||||
owner,
|
owner,
|
||||||
repo,
|
repo,
|
||||||
sha: project.prodBranch,
|
sha: project.prodBranch,
|
||||||
per_page: 1
|
per_page: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create deployment with prod branch and latest commit
|
// Create deployment with prod branch and latest commit
|
||||||
await this.createDeployment(user.id,
|
await this.createDeployment(user.id, octokit, {
|
||||||
octokit,
|
|
||||||
{
|
|
||||||
project,
|
project,
|
||||||
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
|
commitMessage: latestCommit.commit.message,
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
await this.createRepoHook(octokit, project);
|
await this.createRepoHook(octokit, project);
|
||||||
|
|
||||||
@ -655,11 +695,11 @@ export class Service {
|
|||||||
config: {
|
config: {
|
||||||
url: new URL(
|
url: new URL(
|
||||||
'api/github/webhook',
|
'api/github/webhook',
|
||||||
this.config.gitHubConfig.webhookUrl
|
this.config.gitHubConfig.webhookUrl,
|
||||||
).href,
|
).href,
|
||||||
content_type: 'json'
|
content_type: 'json',
|
||||||
},
|
},
|
||||||
events: ['push']
|
events: ['push'],
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// https://docs.github.com/en/rest/repos/webhooks?apiVersion=2022-11-28#create-a-repository-webhook--status-codes
|
// https://docs.github.com/en/rest/repos/webhooks?apiVersion=2022-11-28#create-a-repository-webhook--status-codes
|
||||||
@ -668,7 +708,7 @@ export class Service {
|
|||||||
err instanceof RequestError &&
|
err instanceof RequestError &&
|
||||||
err.status === 422 &&
|
err.status === 422 &&
|
||||||
(err.response?.data as any).errors.some(
|
(err.response?.data as any).errors.some(
|
||||||
(err: any) => err.message === GITHUB_UNIQUE_WEBHOOK_ERROR
|
(err: any) => err.message === GITHUB_UNIQUE_WEBHOOK_ERROR,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
@ -687,9 +727,11 @@ export class Service {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
log(`Handling GitHub push event from repository: ${repository.full_name}, branch: ${ref}`);
|
log(
|
||||||
|
`Handling GitHub push event from repository: ${repository.full_name}, branch: ${ref}`,
|
||||||
|
);
|
||||||
const projects = await this.db.getProjects({
|
const projects = await this.db.getProjects({
|
||||||
where: { repository: repository.full_name }
|
where: { repository: repository.full_name },
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!projects.length) {
|
if (!projects.length) {
|
||||||
@ -703,7 +745,7 @@ export class Service {
|
|||||||
for await (const project of projects) {
|
for await (const project of projects) {
|
||||||
const octokit = await this.getOctokit(project.ownerId);
|
const octokit = await this.getOctokit(project.ownerId);
|
||||||
const [domain] = await this.db.getDomainsByProjectId(project.id, {
|
const [domain] = await this.db.getDomainsByProjectId(project.id, {
|
||||||
branch
|
branch,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create deployment with branch and latest commit in GitHub data
|
// Create deployment with branch and latest commit in GitHub data
|
||||||
@ -716,14 +758,14 @@ export class Service {
|
|||||||
: Environment.Preview,
|
: Environment.Preview,
|
||||||
domain,
|
domain,
|
||||||
commitHash: headCommit.id,
|
commitHash: headCommit.id,
|
||||||
commitMessage: headCommit.message
|
commitMessage: headCommit.message,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateProject(
|
async updateProject(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
data: DeepPartial<Project>
|
data: DeepPartial<Project>,
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
return this.db.updateProjectById(projectId, data);
|
return this.db.updateProjectById(projectId, data);
|
||||||
}
|
}
|
||||||
@ -736,13 +778,13 @@ export class Service {
|
|||||||
async deleteDomain(domainId: string): Promise<boolean> {
|
async deleteDomain(domainId: string): Promise<boolean> {
|
||||||
const domainsRedirectedFrom = await this.db.getDomains({
|
const domainsRedirectedFrom = await this.db.getDomains({
|
||||||
where: {
|
where: {
|
||||||
redirectToId: domainId
|
redirectToId: domainId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (domainsRedirectedFrom.length > 0) {
|
if (domainsRedirectedFrom.length > 0) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
'Cannot delete domain since it has redirects from other domains'
|
'Cannot delete domain since it has redirects from other domains',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -754,11 +796,11 @@ export class Service {
|
|||||||
relations: {
|
relations: {
|
||||||
project: true,
|
project: true,
|
||||||
domain: true,
|
domain: true,
|
||||||
createdBy: true
|
createdBy: true,
|
||||||
},
|
},
|
||||||
where: {
|
where: {
|
||||||
id: deploymentId
|
id: deploymentId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (oldDeployment === null) {
|
if (oldDeployment === null) {
|
||||||
@ -767,16 +809,14 @@ export class Service {
|
|||||||
|
|
||||||
const octokit = await this.getOctokit(user.id);
|
const octokit = await this.getOctokit(user.id);
|
||||||
|
|
||||||
const newDeployment = await this.createDeployment(user.id,
|
const newDeployment = await this.createDeployment(user.id, octokit, {
|
||||||
octokit,
|
|
||||||
{
|
|
||||||
project: oldDeployment.project,
|
project: oldDeployment.project,
|
||||||
// TODO: Put isCurrent field in project
|
// TODO: Put isCurrent field in project
|
||||||
branch: oldDeployment.branch,
|
branch: oldDeployment.branch,
|
||||||
environment: Environment.Production,
|
environment: Environment.Production,
|
||||||
domain: oldDeployment.domain,
|
domain: oldDeployment.domain,
|
||||||
commitHash: oldDeployment.commitHash,
|
commitHash: oldDeployment.commitHash,
|
||||||
commitMessage: oldDeployment.commitMessage
|
commitMessage: oldDeployment.commitMessage,
|
||||||
});
|
});
|
||||||
|
|
||||||
return newDeployment;
|
return newDeployment;
|
||||||
@ -784,19 +824,19 @@ export class Service {
|
|||||||
|
|
||||||
async rollbackDeployment(
|
async rollbackDeployment(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
deploymentId: string
|
deploymentId: string,
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
// TODO: Implement transactions
|
// TODO: Implement transactions
|
||||||
const oldCurrentDeployment = await this.db.getDeployment({
|
const oldCurrentDeployment = await this.db.getDeployment({
|
||||||
relations: {
|
relations: {
|
||||||
domain: true
|
domain: true,
|
||||||
},
|
},
|
||||||
where: {
|
where: {
|
||||||
project: {
|
project: {
|
||||||
id: projectId
|
id: projectId,
|
||||||
|
},
|
||||||
|
isCurrent: true,
|
||||||
},
|
},
|
||||||
isCurrent: true
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!oldCurrentDeployment) {
|
if (!oldCurrentDeployment) {
|
||||||
@ -805,12 +845,12 @@ export class Service {
|
|||||||
|
|
||||||
const oldCurrentDeploymentUpdate = await this.db.updateDeploymentById(
|
const oldCurrentDeploymentUpdate = await this.db.updateDeploymentById(
|
||||||
oldCurrentDeployment.id,
|
oldCurrentDeployment.id,
|
||||||
{ isCurrent: false, domain: null }
|
{ isCurrent: false, domain: null },
|
||||||
);
|
);
|
||||||
|
|
||||||
const newCurrentDeploymentUpdate = await this.db.updateDeploymentById(
|
const newCurrentDeploymentUpdate = await this.db.updateDeploymentById(
|
||||||
deploymentId,
|
deploymentId,
|
||||||
{ isCurrent: true, domain: oldCurrentDeployment?.domain }
|
{ isCurrent: true, domain: oldCurrentDeployment?.domain },
|
||||||
);
|
);
|
||||||
|
|
||||||
return newCurrentDeploymentUpdate && oldCurrentDeploymentUpdate;
|
return newCurrentDeploymentUpdate && oldCurrentDeploymentUpdate;
|
||||||
@ -819,11 +859,11 @@ export class Service {
|
|||||||
async deleteDeployment(deploymentId: string): Promise<boolean> {
|
async deleteDeployment(deploymentId: string): Promise<boolean> {
|
||||||
const deployment = await this.db.getDeployment({
|
const deployment = await this.db.getDeployment({
|
||||||
where: {
|
where: {
|
||||||
id: deploymentId
|
id: deploymentId,
|
||||||
},
|
},
|
||||||
relations: {
|
relations: {
|
||||||
project: true
|
project: true,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (deployment && deployment.applicationDeploymentRecordId) {
|
if (deployment && deployment.applicationDeploymentRecordId) {
|
||||||
@ -831,31 +871,39 @@ export class Service {
|
|||||||
if (deployment.isCurrent) {
|
if (deployment.isCurrent) {
|
||||||
const currentDeploymentURL = `https://${deployment.project.subDomain}`;
|
const currentDeploymentURL = `https://${deployment.project.subDomain}`;
|
||||||
|
|
||||||
const deploymentRecords = await this.registry.getDeploymentRecordsByFilter({
|
const deploymentRecords =
|
||||||
|
await this.registry.getDeploymentRecordsByFilter({
|
||||||
application: deployment.applicationRecordId,
|
application: deployment.applicationRecordId,
|
||||||
url: currentDeploymentURL
|
url: currentDeploymentURL,
|
||||||
})
|
});
|
||||||
|
|
||||||
if (!deploymentRecords.length) {
|
if (!deploymentRecords.length) {
|
||||||
log(`No ApplicationDeploymentRecord found for URL ${currentDeploymentURL} and ApplicationDeploymentRecord id ${deployment.applicationDeploymentRecordId}`);
|
log(
|
||||||
|
`No ApplicationDeploymentRecord found for URL ${currentDeploymentURL} and ApplicationDeploymentRecord id ${deployment.applicationDeploymentRecordId}`,
|
||||||
|
);
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.registry.createApplicationDeploymentRemovalRequest({ deploymentId: deploymentRecords[0].id });
|
await this.registry.createApplicationDeploymentRemovalRequest({
|
||||||
|
deploymentId: deploymentRecords[0].id,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const result = await this.registry.createApplicationDeploymentRemovalRequest({ deploymentId: deployment.applicationDeploymentRecordId });
|
const result =
|
||||||
|
await this.registry.createApplicationDeploymentRemovalRequest({
|
||||||
|
deploymentId: deployment.applicationDeploymentRecordId,
|
||||||
|
});
|
||||||
|
|
||||||
await this.db.updateDeploymentById(
|
await this.db.updateDeploymentById(deployment.id, {
|
||||||
deployment.id,
|
|
||||||
{
|
|
||||||
status: DeploymentStatus.Deleting,
|
status: DeploymentStatus.Deleting,
|
||||||
applicationDeploymentRemovalRequestId: result.applicationDeploymentRemovalRequestId,
|
applicationDeploymentRemovalRequestId:
|
||||||
applicationDeploymentRemovalRequestData: result.applicationDeploymentRemovalRequestData
|
result.applicationDeploymentRemovalRequestId,
|
||||||
}
|
applicationDeploymentRemovalRequestData:
|
||||||
);
|
result.applicationDeploymentRemovalRequestData,
|
||||||
|
});
|
||||||
|
|
||||||
return (result !== undefined || result !== null);
|
return result !== undefined || result !== null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@ -863,7 +911,7 @@ export class Service {
|
|||||||
|
|
||||||
async addDomain(
|
async addDomain(
|
||||||
projectId: string,
|
projectId: string,
|
||||||
data: { name: string }
|
data: { name: string },
|
||||||
): Promise<{
|
): Promise<{
|
||||||
primaryDomain: Domain;
|
primaryDomain: Domain;
|
||||||
redirectedDomain: Domain;
|
redirectedDomain: Domain;
|
||||||
@ -877,7 +925,7 @@ export class Service {
|
|||||||
const primaryDomainDetails = {
|
const primaryDomainDetails = {
|
||||||
...data,
|
...data,
|
||||||
branch: currentProject.prodBranch,
|
branch: currentProject.prodBranch,
|
||||||
project: currentProject
|
project: currentProject,
|
||||||
};
|
};
|
||||||
|
|
||||||
const savedPrimaryDomain = await this.db.addDomain(primaryDomainDetails);
|
const savedPrimaryDomain = await this.db.addDomain(primaryDomainDetails);
|
||||||
@ -888,27 +936,27 @@ export class Service {
|
|||||||
name: domainArr.length > 1 ? domainArr[1] : `www.${domainArr[0]}`,
|
name: domainArr.length > 1 ? domainArr[1] : `www.${domainArr[0]}`,
|
||||||
branch: currentProject.prodBranch,
|
branch: currentProject.prodBranch,
|
||||||
project: currentProject,
|
project: currentProject,
|
||||||
redirectTo: savedPrimaryDomain
|
redirectTo: savedPrimaryDomain,
|
||||||
};
|
};
|
||||||
|
|
||||||
const savedRedirectedDomain = await this.db.addDomain(
|
const savedRedirectedDomain = await this.db.addDomain(
|
||||||
redirectedDomainDetails
|
redirectedDomainDetails,
|
||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
primaryDomain: savedPrimaryDomain,
|
primaryDomain: savedPrimaryDomain,
|
||||||
redirectedDomain: savedRedirectedDomain
|
redirectedDomain: savedRedirectedDomain,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDomain(
|
async updateDomain(
|
||||||
domainId: string,
|
domainId: string,
|
||||||
data: DeepPartial<Domain>
|
data: DeepPartial<Domain>,
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
const domain = await this.db.getDomain({
|
const domain = await this.db.getDomain({
|
||||||
where: {
|
where: {
|
||||||
id: domainId
|
id: domainId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (domain === null) {
|
if (domain === null) {
|
||||||
@ -916,16 +964,16 @@ export class Service {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const newDomain = {
|
const newDomain = {
|
||||||
...data
|
...data,
|
||||||
};
|
};
|
||||||
|
|
||||||
const domainsRedirectedFrom = await this.db.getDomains({
|
const domainsRedirectedFrom = await this.db.getDomains({
|
||||||
where: {
|
where: {
|
||||||
project: {
|
project: {
|
||||||
id: domain.projectId
|
id: domain.projectId,
|
||||||
|
},
|
||||||
|
redirectToId: domain.id,
|
||||||
},
|
},
|
||||||
redirectToId: domain.id
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// If there are domains redirecting to current domain, only branch of current domain can be updated
|
// If there are domains redirecting to current domain, only branch of current domain can be updated
|
||||||
@ -936,8 +984,8 @@ export class Service {
|
|||||||
if (data.redirectToId) {
|
if (data.redirectToId) {
|
||||||
const redirectedDomain = await this.db.getDomain({
|
const redirectedDomain = await this.db.getDomain({
|
||||||
where: {
|
where: {
|
||||||
id: data.redirectToId
|
id: data.redirectToId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (redirectedDomain === null) {
|
if (redirectedDomain === null) {
|
||||||
@ -946,7 +994,7 @@ export class Service {
|
|||||||
|
|
||||||
if (redirectedDomain.redirectToId) {
|
if (redirectedDomain.redirectToId) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
'Unable to redirect to the domain because it is already redirecting elsewhere. Redirects cannot be chained.'
|
'Unable to redirect to the domain because it is already redirecting elsewhere. Redirects cannot be chained.',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -958,9 +1006,14 @@ export class Service {
|
|||||||
return updateResult;
|
return updateResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
async authenticateGitHub (code:string, user: User): Promise<{token: string}> {
|
async authenticateGitHub(
|
||||||
const { authentication: { token } } = await this.oauthApp.createToken({
|
code: string,
|
||||||
code
|
user: User,
|
||||||
|
): Promise<{ token: string }> {
|
||||||
|
const {
|
||||||
|
authentication: { token },
|
||||||
|
} = await this.oauthApp.createToken({
|
||||||
|
code,
|
||||||
});
|
});
|
||||||
|
|
||||||
await this.db.updateUser(user, { gitHubToken: token });
|
await this.db.updateUser(user, { gitHubToken: token });
|
||||||
@ -968,7 +1021,10 @@ export class Service {
|
|||||||
return { token };
|
return { token };
|
||||||
}
|
}
|
||||||
|
|
||||||
async unauthenticateGitHub (user: User, data: DeepPartial<User>): Promise<boolean> {
|
async unauthenticateGitHub(
|
||||||
|
user: User,
|
||||||
|
data: DeepPartial<User>,
|
||||||
|
): Promise<boolean> {
|
||||||
return this.db.updateUser(user, data);
|
return this.db.updateUser(user, data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
2
packages/gql-client/dist/index.js.map
vendored
2
packages/gql-client/dist/index.js.map
vendored
File diff suppressed because one or more lines are too long
5
packages/gql-client/dist/index.mjs
vendored
5
packages/gql-client/dist/index.mjs
vendored
@ -20,7 +20,10 @@ var __async = (__this, __arguments, generator) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// src/client.ts
|
// src/client.ts
|
||||||
import { ApolloClient, InMemoryCache } from "@apollo/client";
|
import {
|
||||||
|
ApolloClient,
|
||||||
|
InMemoryCache
|
||||||
|
} from "@apollo/client";
|
||||||
|
|
||||||
// src/queries.ts
|
// src/queries.ts
|
||||||
import { gql } from "@apollo/client";
|
import { gql } from "@apollo/client";
|
||||||
|
2
packages/gql-client/dist/index.mjs.map
vendored
2
packages/gql-client/dist/index.mjs.map
vendored
File diff suppressed because one or more lines are too long
@ -1,8 +1,13 @@
|
|||||||
import { ApolloClient, DefaultOptions, InMemoryCache, NormalizedCacheObject } from '@apollo/client';
|
import {
|
||||||
|
ApolloClient,
|
||||||
|
DefaultOptions,
|
||||||
|
InMemoryCache,
|
||||||
|
NormalizedCacheObject,
|
||||||
|
} from "@apollo/client";
|
||||||
|
|
||||||
import * as queries from './queries';
|
import * as queries from "./queries";
|
||||||
import * as types from './types';
|
import * as types from "./types";
|
||||||
import * as mutations from './mutations';
|
import * as mutations from "./mutations";
|
||||||
|
|
||||||
export interface GraphQLConfig {
|
export interface GraphQLConfig {
|
||||||
gqlEndpoint: string;
|
gqlEndpoint: string;
|
||||||
@ -11,13 +16,13 @@ export interface GraphQLConfig {
|
|||||||
// TODO: check options
|
// TODO: check options
|
||||||
const defaultOptions: DefaultOptions = {
|
const defaultOptions: DefaultOptions = {
|
||||||
watchQuery: {
|
watchQuery: {
|
||||||
fetchPolicy: 'no-cache',
|
fetchPolicy: "no-cache",
|
||||||
errorPolicy: 'ignore'
|
errorPolicy: "ignore",
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
fetchPolicy: 'no-cache',
|
fetchPolicy: "no-cache",
|
||||||
errorPolicy: 'all'
|
errorPolicy: "all",
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export class GQLClient {
|
export class GQLClient {
|
||||||
@ -28,13 +33,13 @@ export class GQLClient {
|
|||||||
uri: config.gqlEndpoint,
|
uri: config.gqlEndpoint,
|
||||||
cache: new InMemoryCache(),
|
cache: new InMemoryCache(),
|
||||||
defaultOptions,
|
defaultOptions,
|
||||||
credentials: 'include'
|
credentials: "include",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUser(): Promise<types.GetUserResponse> {
|
async getUser(): Promise<types.GetUserResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getUser
|
query: queries.getUser,
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
@ -44,19 +49,21 @@ export class GQLClient {
|
|||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getProject,
|
query: queries.getProject,
|
||||||
variables: {
|
variables: {
|
||||||
projectId
|
projectId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectsInOrganization (organizationSlug: string) : Promise<types.GetProjectsInOrganizationResponse> {
|
async getProjectsInOrganization(
|
||||||
|
organizationSlug: string
|
||||||
|
): Promise<types.GetProjectsInOrganizationResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getProjectsInOrganization,
|
query: queries.getProjectsInOrganization,
|
||||||
variables: {
|
variables: {
|
||||||
organizationSlug
|
organizationSlug,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
@ -64,179 +71,216 @@ export class GQLClient {
|
|||||||
|
|
||||||
async getOrganizations(): Promise<types.GetOrganizationsResponse> {
|
async getOrganizations(): Promise<types.GetOrganizationsResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getOrganizations
|
query: queries.getOrganizations,
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDeployments (projectId: string) : Promise<types.GetDeploymentsResponse> {
|
async getDeployments(
|
||||||
|
projectId: string
|
||||||
|
): Promise<types.GetDeploymentsResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getDeployments,
|
query: queries.getDeployments,
|
||||||
variables: {
|
variables: {
|
||||||
projectId
|
projectId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getEnvironmentVariables (projectId: string) : Promise<types.GetEnvironmentVariablesResponse> {
|
async getEnvironmentVariables(
|
||||||
|
projectId: string
|
||||||
|
): Promise<types.GetEnvironmentVariablesResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getEnvironmentVariables,
|
query: queries.getEnvironmentVariables,
|
||||||
variables: {
|
variables: {
|
||||||
projectId
|
projectId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectMembers (projectId: string) : Promise<types.GetProjectMembersResponse> {
|
async getProjectMembers(
|
||||||
|
projectId: string
|
||||||
|
): Promise<types.GetProjectMembersResponse> {
|
||||||
const result = await this.client.query({
|
const result = await this.client.query({
|
||||||
query: queries.getProjectMembers,
|
query: queries.getProjectMembers,
|
||||||
variables: {
|
variables: {
|
||||||
projectId
|
projectId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addProjectMember (projectId: string, data: types.AddProjectMemberInput) : Promise<types.AddProjectMemberResponse> {
|
async addProjectMember(
|
||||||
|
projectId: string,
|
||||||
|
data: types.AddProjectMemberInput
|
||||||
|
): Promise<types.AddProjectMemberResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.addProjectMember,
|
mutation: mutations.addProjectMember,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateProjectMember (projectMemberId: string, data: types.UpdateProjectMemberInput): Promise<types.UpdateProjectMemberResponse> {
|
async updateProjectMember(
|
||||||
|
projectMemberId: string,
|
||||||
|
data: types.UpdateProjectMemberInput
|
||||||
|
): Promise<types.UpdateProjectMemberResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.updateProjectMember,
|
mutation: mutations.updateProjectMember,
|
||||||
variables: {
|
variables: {
|
||||||
projectMemberId,
|
projectMemberId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeProjectMember (projectMemberId: string): Promise<types.RemoveProjectMemberResponse> {
|
async removeProjectMember(
|
||||||
|
projectMemberId: string
|
||||||
|
): Promise<types.RemoveProjectMemberResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.removeProjectMember,
|
mutation: mutations.removeProjectMember,
|
||||||
variables: {
|
variables: {
|
||||||
projectMemberId
|
projectMemberId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchProjects (searchText: string) : Promise<types.SearchProjectsResponse> {
|
async searchProjects(
|
||||||
|
searchText: string
|
||||||
|
): Promise<types.SearchProjectsResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.searchProjects,
|
query: queries.searchProjects,
|
||||||
variables: {
|
variables: {
|
||||||
searchText
|
searchText,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addEnvironmentVariables (projectId: string, data: types.AddEnvironmentVariableInput[]): Promise<types.AddEnvironmentVariablesResponse> {
|
async addEnvironmentVariables(
|
||||||
|
projectId: string,
|
||||||
|
data: types.AddEnvironmentVariableInput[]
|
||||||
|
): Promise<types.AddEnvironmentVariablesResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.addEnvironmentVariables,
|
mutation: mutations.addEnvironmentVariables,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateEnvironmentVariable (environmentVariableId: string, data: types.UpdateEnvironmentVariableInput): Promise<types.UpdateEnvironmentVariableResponse> {
|
async updateEnvironmentVariable(
|
||||||
|
environmentVariableId: string,
|
||||||
|
data: types.UpdateEnvironmentVariableInput
|
||||||
|
): Promise<types.UpdateEnvironmentVariableResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.updateEnvironmentVariable,
|
mutation: mutations.updateEnvironmentVariable,
|
||||||
variables: {
|
variables: {
|
||||||
environmentVariableId,
|
environmentVariableId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeEnvironmentVariable (environmentVariableId: string): Promise<types.RemoveEnvironmentVariableResponse> {
|
async removeEnvironmentVariable(
|
||||||
|
environmentVariableId: string
|
||||||
|
): Promise<types.RemoveEnvironmentVariableResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.removeEnvironmentVariable,
|
mutation: mutations.removeEnvironmentVariable,
|
||||||
variables: {
|
variables: {
|
||||||
environmentVariableId
|
environmentVariableId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDeploymentToProd (deploymentId: string): Promise<types.UpdateDeploymentToProdResponse> {
|
async updateDeploymentToProd(
|
||||||
|
deploymentId: string
|
||||||
|
): Promise<types.UpdateDeploymentToProdResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.updateDeploymentToProd,
|
mutation: mutations.updateDeploymentToProd,
|
||||||
variables: {
|
variables: {
|
||||||
deploymentId
|
deploymentId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addProject (organizationSlug: string, data: types.AddProjectInput): Promise<types.AddProjectResponse> {
|
async addProject(
|
||||||
|
organizationSlug: string,
|
||||||
|
data: types.AddProjectInput
|
||||||
|
): Promise<types.AddProjectResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.addProject,
|
mutation: mutations.addProject,
|
||||||
variables: {
|
variables: {
|
||||||
organizationSlug,
|
organizationSlug,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateProject (projectId: string, data: types.UpdateProjectInput): Promise<types.UpdateProjectResponse> {
|
async updateProject(
|
||||||
|
projectId: string,
|
||||||
|
data: types.UpdateProjectInput
|
||||||
|
): Promise<types.UpdateProjectResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.updateProjectMutation,
|
mutation: mutations.updateProjectMutation,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDomain (domainId: string, data: types.UpdateDomainInput): Promise<types.UpdateDomainResponse> {
|
async updateDomain(
|
||||||
|
domainId: string,
|
||||||
|
data: types.UpdateDomainInput
|
||||||
|
): Promise<types.UpdateDomainResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.updateDomainMutation,
|
mutation: mutations.updateDomainMutation,
|
||||||
variables: {
|
variables: {
|
||||||
domainId,
|
domainId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async redeployToProd (deploymentId: string): Promise<types.RedeployToProdResponse> {
|
async redeployToProd(
|
||||||
|
deploymentId: string
|
||||||
|
): Promise<types.RedeployToProdResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.redeployToProd,
|
mutation: mutations.redeployToProd,
|
||||||
variables: {
|
variables: {
|
||||||
deploymentId
|
deploymentId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
@ -246,8 +290,8 @@ export class GQLClient {
|
|||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.deleteProject,
|
mutation: mutations.deleteProject,
|
||||||
variables: {
|
variables: {
|
||||||
projectId
|
projectId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
@ -257,66 +301,79 @@ export class GQLClient {
|
|||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.deleteDomain,
|
mutation: mutations.deleteDomain,
|
||||||
variables: {
|
variables: {
|
||||||
domainId
|
domainId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async rollbackDeployment (projectId: string, deploymentId: string): Promise<types.RollbackDeploymentResponse> {
|
async rollbackDeployment(
|
||||||
|
projectId: string,
|
||||||
|
deploymentId: string
|
||||||
|
): Promise<types.RollbackDeploymentResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.rollbackDeployment,
|
mutation: mutations.rollbackDeployment,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
deploymentId
|
deploymentId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteDeployment (deploymentId: string): Promise<types.DeleteDeploymentResponse> {
|
async deleteDeployment(
|
||||||
|
deploymentId: string
|
||||||
|
): Promise<types.DeleteDeploymentResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.deleteDeployment,
|
mutation: mutations.deleteDeployment,
|
||||||
variables: {
|
variables: {
|
||||||
deploymentId
|
deploymentId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addDomain (projectId: string, data: types.AddDomainInput): Promise<types.AddDomainResponse> {
|
async addDomain(
|
||||||
|
projectId: string,
|
||||||
|
data: types.AddDomainInput
|
||||||
|
): Promise<types.AddDomainResponse> {
|
||||||
const result = await this.client.mutate({
|
const result = await this.client.mutate({
|
||||||
mutation: mutations.addDomain,
|
mutation: mutations.addDomain,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
data
|
data,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return result.data;
|
return result.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDomains (projectId: string, filter?: types.FilterDomainInput): Promise<types.GetDomainsResponse> {
|
async getDomains(
|
||||||
|
projectId: string,
|
||||||
|
filter?: types.FilterDomainInput
|
||||||
|
): Promise<types.GetDomainsResponse> {
|
||||||
const { data } = await this.client.query({
|
const { data } = await this.client.query({
|
||||||
query: queries.getDomains,
|
query: queries.getDomains,
|
||||||
variables: {
|
variables: {
|
||||||
projectId,
|
projectId,
|
||||||
filter
|
filter,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
async authenticateGitHub (code: string): Promise<types.AuthenticateGitHubResponse> {
|
async authenticateGitHub(
|
||||||
|
code: string
|
||||||
|
): Promise<types.AuthenticateGitHubResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.authenticateGitHub,
|
mutation: mutations.authenticateGitHub,
|
||||||
variables: {
|
variables: {
|
||||||
code
|
code,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
@ -324,7 +381,7 @@ export class GQLClient {
|
|||||||
|
|
||||||
async unauthenticateGithub(): Promise<types.UnauthenticateGitHubResponse> {
|
async unauthenticateGithub(): Promise<types.UnauthenticateGitHubResponse> {
|
||||||
const { data } = await this.client.mutate({
|
const { data } = await this.client.mutate({
|
||||||
mutation: mutations.unauthenticateGitHub
|
mutation: mutations.unauthenticateGitHub,
|
||||||
});
|
});
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
Loading…
Reference in New Issue
Block a user