Skip to content

Commit

Permalink
Merge pull request #998 from Vizzuality/feat/add-cloning-project-and-…
Browse files Browse the repository at this point in the history
…scenario-e2e-tests

feat : add clone project and clone scenario e2e test
  • Loading branch information
angelhigueraacid authored Apr 20, 2022
2 parents be639c2 + 6956e42 commit dcfb69f
Show file tree
Hide file tree
Showing 4 changed files with 498 additions and 10 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -52,15 +52,19 @@ export class ExportConfigReader {
);
if (isLeft(exportConfigOrError)) return left(archiveCorrupted);

const exportConfigSnapshot: ExportConfigContent = JSON.parse(
exportConfigOrError.right,
);
try {
const exportConfigSnapshot: ExportConfigContent = JSON.parse(
exportConfigOrError.right,
);

const exportConfig = this.convertToClass(exportConfigSnapshot);
const exportConfig = this.convertToClass(exportConfigSnapshot);

const validationErrors = await validate(exportConfig);
if (validationErrors.length > 0) return left(invalidFiles);
const validationErrors = await validate(exportConfig);
if (validationErrors.length > 0) return left(invalidFiles);

return right(exportConfig);
return right(exportConfig);
} catch (error) {
return left(archiveCorrupted);
}
}
}
207 changes: 207 additions & 0 deletions api/apps/api/test/project/clone-project.e2e-spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,207 @@
import { ProjectRoles } from '@marxan-api/modules/access-control/projects-acl/dto/user-role-project.dto';
import { UsersProjectsApiEntity } from '@marxan-api/modules/access-control/projects-acl/entity/users-projects.api.entity';
import { ApiEventsService } from '@marxan-api/modules/api-events';
import { ExportEntity } from '@marxan-api/modules/clone/export/adapters/entities/exports.api.entity';
import { CompleteExportPiece } from '@marxan-api/modules/clone/export/application/complete-export-piece.command';
import { ExportRepository } from '@marxan-api/modules/clone/export/application/export-repository.port';
import { Export, ExportId } from '@marxan-api/modules/clone/export/domain';
import { ImportEntity } from '@marxan-api/modules/clone/import/adapters/entities/imports.api.entity';
import { CompleteImportPiece } from '@marxan-api/modules/clone/import/application/complete-import-piece.command';
import { AllPiecesImported } from '@marxan-api/modules/clone/import/domain';
import { SchedulePieceImport } from '@marxan-api/modules/clone/infra/import/schedule-piece-import.command';
import { API_EVENT_KINDS } from '@marxan/api-events';
import { CloningFilesRepository } from '@marxan/cloning-files-repository';
import { ClonePiece, ComponentId, ResourceKind } from '@marxan/cloning/domain';
import { ClonePieceUrisResolver } from '@marxan/cloning/infrastructure/clone-piece-data';
import {
exportVersion,
ProjectExportConfigContent,
} from '@marxan/cloning/infrastructure/clone-piece-data/export-config';
import { FixtureType } from '@marxan/utils/tests/fixture-type';
import { CommandBus, CqrsModule } from '@nestjs/cqrs';
import { getRepositoryToken } from '@nestjs/typeorm';
import { isLeft } from 'fp-ts/lib/These';
import { Readable } from 'stream';
import * as request from 'supertest';
import { Connection, Repository } from 'typeorm';
import { GivenProjectExists } from '../steps/given-project';
import { GivenUserExists } from '../steps/given-user-exists';
import { GivenUserIsLoggedIn } from '../steps/given-user-is-logged-in';
import { bootstrapApplication } from '../utils/api-application';
import { EventBusTestUtils } from '../utils/event-bus.test.utils';
import { OrganizationsTestUtils } from '../utils/organizations.test.utils';
import { ProjectsTestUtils } from '../utils/projects.test.utils';

let fixtures: FixtureType<typeof getFixtures>;

beforeEach(async () => {
fixtures = await getFixtures();
});

afterEach(async () => {
await fixtures?.cleanup();
});

test('should permit cloning a project', async () => {
await fixtures.GivenProjectWasCreated();
await fixtures.GivenCloneWasRequested();

await fixtures.WhenImportIsReady();
await fixtures.ThenImportIsCompleted();
});

export const getFixtures = async () => {
const app = await bootstrapApplication([CqrsModule], [EventBusTestUtils]);
const eventBusTestUtils = app.get(EventBusTestUtils);
eventBusTestUtils.startInspectingEvents();
const commandBus = app.get(CommandBus);
const exportRepo = app.get(ExportRepository);
const apiEvents = app.get(ApiEventsService);
const fileRepo = app.get(CloningFilesRepository);
const userProjectsRepo = app.get<Repository<UsersProjectsApiEntity>>(
getRepositoryToken(UsersProjectsApiEntity),
);

const ownerToken = await GivenUserIsLoggedIn(app, 'aa');
const ownerUserId = await GivenUserExists(app, 'aa');

let projectId: string;
let organizationId: string;
let exportId: ExportId;
let clonedProjectId: string;

const piecesUris: Record<string, string> = {};
const savePiecesFiles = async (exportInstance: Export) => {
await Promise.all(
exportInstance.toSnapshot().exportPieces.map(async (piece, i) => {
let content = `${piece.piece}`;
if (piece.piece === ClonePiece.ExportConfig) {
const exportConfigContent: ProjectExportConfigContent = {
isCloning: true,
version: exportVersion,
name: 'random name',
description: 'random desc',
resourceKind: ResourceKind.Project,
resourceId: projectId,
pieces: {
project: exportInstance
.toSnapshot()
.exportPieces.map((exportPice) => exportPice.piece),
scenarios: {},
},
scenarios: [],
};
content = JSON.stringify(exportConfigContent);
}

const result = await fileRepo.save(Readable.from(content));

if (isLeft(result)) {
throw new Error(`Error while saving ${piece.id} file`);
}

piecesUris[piece.id] = result.right;
}),
);
};

const completeSchedueleImportPieces = async () => {
commandBus.subscribe((command) => {
if (command instanceof SchedulePieceImport) {
commandBus.execute(
new CompleteImportPiece(command.importId, command.componentId),
);
}
});
};

return {
cleanup: async () => {
const connection = app.get<Connection>(Connection);
const exportRepo = connection.getRepository(ExportEntity);
const importRepo = connection.getRepository(ImportEntity);

await exportRepo.delete({});
await importRepo.delete({});
await ProjectsTestUtils.deleteProject(app, ownerToken, projectId);
await userProjectsRepo.save({
projectId: clonedProjectId,
userId: ownerUserId,
roleName: ProjectRoles.project_owner,
});
await ProjectsTestUtils.deleteProject(app, ownerToken, clonedProjectId);
await OrganizationsTestUtils.deleteOrganization(
app,
ownerToken,
organizationId,
);
eventBusTestUtils.stopInspectingEvents();
await app.close();
},
GivenProjectWasCreated: async () => {
const result = await GivenProjectExists(app, ownerToken);
projectId = result.projectId;
organizationId = result.organizationId;
},
GivenCloneWasRequested: async () => {
completeSchedueleImportPieces();

const response = await request(app.getHttpServer())
.post(`/api/v1/projects/${projectId}/clone`)
.set('Authorization', `Bearer ${ownerToken}`)
.send({ scenarioIds: [] })
.expect(201);

exportId = new ExportId(response.body.exportId);
clonedProjectId = response.body.projectId;

const exportInstance = await exportRepo.find(exportId);

expect(exportInstance).toBeDefined();

await savePiecesFiles(exportInstance!);

exportInstance!.toSnapshot().exportPieces.forEach((exportedPiece) => {
commandBus.execute(
new CompleteExportPiece(
exportId,
new ComponentId(exportedPiece.id),
ClonePieceUrisResolver.resolveFor(
exportedPiece.piece,
piecesUris[exportedPiece.id],
),
),
);
});
},
WhenImportIsReady: async () => {
return eventBusTestUtils.waitUntilEventIsPublished(AllPiecesImported);
},
ThenImportIsCompleted: async () => {
return new Promise<void>((resolve, reject) => {
const findApiEventInterval = setInterval(async () => {
try {
await apiEvents.getLatestEventForTopic({
topic: clonedProjectId,
kind: API_EVENT_KINDS.project__import__finished__v1__alpha,
});
await apiEvents.getLatestEventForTopic({
topic: projectId,
kind: API_EVENT_KINDS.project__export__finished__v1__alpha,
});
await apiEvents.getLatestEventForTopic({
topic: clonedProjectId,
kind: API_EVENT_KINDS.project__clone__finished__v1__alpha,
});
clearInterval(findApiEventInterval);
resolve();
} catch (error) {}
}, 150);
setTimeout(() => {
clearInterval(findApiEventInterval);
reject('Import and Clone API event were not found');
}, 6000);
});
},
};
};
6 changes: 3 additions & 3 deletions api/apps/api/test/project/import-project.e2e-spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@ import {
exportVersion,
ProjectExportConfigContent,
} from '@marxan/cloning/infrastructure/clone-piece-data/export-config';
import { FileRepository } from '@marxan/files-repository';
import { FixtureType } from '@marxan/utils/tests/fixture-type';
import { CommandBus, CqrsModule } from '@nestjs/cqrs';
import * as archiver from 'archiver';
Expand All @@ -26,6 +25,7 @@ import { GivenUserIsLoggedIn } from '../steps/given-user-is-logged-in';
import { bootstrapApplication } from '../utils/api-application';
import { EventBusTestUtils } from '../utils/event-bus.test.utils';
import { ApiEventByTopicAndKind } from '@marxan-api/modules/api-events/api-event.topic+kind.api.entity';
import { CloningFilesRepository } from '@marxan/cloning-files-repository';

let fixtures: FixtureType<typeof getFixtures>;

Expand Down Expand Up @@ -53,7 +53,7 @@ export const getFixtures = async () => {
const commandBus = app.get(CommandBus);
const importRepo = app.get(ImportRepository);
const apiEvents = app.get(ApiEventsService);
const fileRepository = app.get(FileRepository);
const fileRepository = app.get(CloningFilesRepository);

const ownerToken = await GivenUserIsLoggedIn(app, 'aa');
const oldProjectId = v4();
Expand Down Expand Up @@ -142,7 +142,7 @@ export const getFixtures = async () => {
clearInterval(findApiEventInterval);
resolve(event);
} catch (error) {}
}, 1500);
}, 150);
const apiEventTimeOut = setTimeout(() => {
clearInterval(findApiEventInterval);
reject('Import API event was not found');
Expand Down
Loading

0 comments on commit dcfb69f

Please sign in to comment.