Skip to content

Commit

Permalink
Merge pull request #31 from wri/feat/TM-1531-return-name-site
Browse files Browse the repository at this point in the history
[TM-1531] Add site name in entity and return entityname in delayed job
  • Loading branch information
egrojMonroy authored Dec 24, 2024
2 parents d1cb1ea + be56314 commit 84badc0
Show file tree
Hide file tree
Showing 7 changed files with 180 additions and 104 deletions.
68 changes: 39 additions & 29 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,32 +1,36 @@
# terramatch-microservices

Repository for the Microservices API backend of the TerraMatch service

# Requirements:
* Node v20.11.1. Using [NVM](https://github.com/nvm-sh/nvm?tab=readme-ov-file) is recommended.
* [Docker](https://www.docker.com/)
* [CDK CLI](https://docs.aws.amazon.com/cdk/v2/guide/getting_started.html) (install globally)
* [AWS SAM CLI](https://docs.aws.amazon.com/serverless-application-model/latest/developerguide/install-sam-cli.html)
* [NX](https://nx.dev/getting-started/installation#installing-nx-globally) (install globally)
* [NestJS](https://docs.nestjs.com/) (install globally, useful for development)

- Node v20.11.1. Using [NVM](https://github.com/nvm-sh/nvm?tab=readme-ov-file) is recommended.
- [Docker](https://www.docker.com/)
- [CDK CLI](https://docs.aws.amazon.com/cdk/v2/guide/getting_started.html) (install globally)
- [AWS SAM CLI](https://docs.aws.amazon.com/serverless-application-model/latest/developerguide/install-sam-cli.html)
- [NX](https://nx.dev/getting-started/installation#installing-nx-globally) (install globally)
- [NestJS](https://docs.nestjs.com/) (install globally, useful for development)

# Building and starting the apps
* Copy `.env.local.sample` to `.env`
* On Linux systems, the DOCKER_HOST value should be `unix:///var/run/docker.sock` instead of what's in the sample.
* To run all services:
* `nx run-many -t serve`
* The default maximum number of services it can run in parallel is 3. To run all of the services at once, use something like
`nx run-many --parallel=100 -t serve`, or you can cherry-pick which services you want to run instead with
`nx run-many -t serve --projects user-service jobs-service`.
* Some useful targets have been added to the root `package.json` for service sets. For instance, to run just the services needed
by the TM React front end, use `npm run fe-services`, or to run all use `npm run all`.
* In `.env` in your `wri-terramatch-website` repository, set your BE connection URL correctly by noting the config
in `.env.local.sample` for local development.
* The `NEXT_PUBLIC_API_BASE_URL` still points at the PHP BE directly
* New `NEXT_PUBLIC_<SERVICE>_URL` values are needed for each service you're running locally. This will typically match
the services defined in `V3_NAMESPACES` in `src/generated/v3/utils.ts`.

- Copy `.env.local.sample` to `.env`
- On Linux systems, the DOCKER_HOST value should be `unix:///var/run/docker.sock` instead of what's in the sample.
- To run all services:
- `nx run-many -t serve`
- The default maximum number of services it can run in parallel is 3. To run all of the services at once, use something like
`nx run-many --parallel=100 -t serve`, or you can cherry-pick which services you want to run instead with
`nx run-many -t serve --projects user-service job-service`.
- Some useful targets have been added to the root `package.json` for service sets. For instance, to run just the services needed
by the TM React front end, use `npm run fe-services`, or to run all use `npm run all`.
- In `.env` in your `wri-terramatch-website` repository, set your BE connection URL correctly by noting the config
in `.env.local.sample` for local development.
- The `NEXT_PUBLIC_API_BASE_URL` still points at the PHP BE directly
- New `NEXT_PUBLIC_<SERVICE>_URL` values are needed for each service you're running locally. This will typically match
the services defined in `V3_NAMESPACES` in `src/generated/v3/utils.ts`.

# Deployment
Deployment is handled via manual trigger of GitHub actions. There is one for services, and one for the ApiGateway. The

Deployment is handled via manual trigger of GitHub actions. There is one for services, and one for the ApiGateway. The
ApiGateway only needs to be redeployed if its code changes; it does not need to be redeployed for updates to individual services
to take effect.

Expand Down Expand Up @@ -62,10 +66,12 @@ and main branches.
need to remain a manual process.
# Database work
For now, Laravel is the source of truth for all things related to the DB schema. As such, TypeORM is not allowed to modify the
schema, and is expected to interface with exactly the schema that is managed by Laravel. This note is included in user.entity.ts,
For now, Laravel is the source of truth for all things related to the DB schema. As such, TypeORM is not allowed to modify the
schema, and is expected to interface with exactly the schema that is managed by Laravel. This note is included in user.entity.ts,
and should hold true for all models created in this codebase until this codebase can take over as the source of truth for DB
schema:
```
// Note: this has some additional typing information (like width: 1 on bools and type: timestamps on
// CreateDateColumn) to make the types generated here match what is generated by Laravel exactly.
Expand All @@ -78,25 +84,29 @@ This codebase connects to the database running in the `wri-terramatch-api` docke
file included in this repo is used only for setting up the database needed for running unit tests in Github Actions.
# Testing
To set up the local testing database, run the `./bin/setup-test-database.sh` script. This script assumes that the
`wri-terramatch-api` project is checked out in the same parent directory as this one. The script may be run
`wri-terramatch-api` project is checked out in the same parent directory as this one. The script may be run
again at any time to clear out the test database records and schema.
`setup-jest.ts` is responsible for creating the Sequelize connection for all tests. Via the `sync` command, it also
creates database tables according to the schema declared in the `entity.ts` files in this codebase. Care should be
taken to make sure that the schema is set up in this codebase such that the database tables are created with the same
types and indices as in the primary database controlled by the Laravel backend.
types and indices as in the primary database controlled by the Laravel backend.
Factories may be used to create entries in the database for testing. See `user.factory.ts`, and uses of `UserFactory` for
Factories may be used to create entries in the database for testing. See `user.factory.ts`, and uses of `UserFactory` for
an example.
To run the tests for a single app/library:
* `nx test user-service` or `nx test common`
- `nx test user-service` or `nx test common`
To run the tests for the whole codebase:
* `nx run-many -t test --passWithNoTests`
- `nx run-many -t test --passWithNoTests`
For checking coverage, simply pass the `--coverage` flag:
* `nx test user-service --coverage` or `nx run-many -t test --passWithNoTests --coverage`
- `nx test user-service --coverage` or `nx run-many -t test --passWithNoTests --coverage`
For apps/libraries that have tests defined, the coverage thresholds are set for the whole project in `jest.preset.js`
141 changes: 80 additions & 61 deletions apps/job-service/src/jobs/delayed-jobs.controller.spec.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,12 @@
import { Test, TestingModule } from '@nestjs/testing';
import { DelayedJobsController } from './delayed-jobs.controller';
import { DelayedJob } from '@terramatch-microservices/database/entities';
import { DelayedJobBulkUpdateBodyDto, DelayedJobAttributes, DelayedJobData } from './dto/delayed-job-update.dto';
import { v4 as uuidv4 } from 'uuid';
import { NotFoundException } from '@nestjs/common';
import { plainToClass } from 'class-transformer';
import { validate } from 'class-validator';

describe('DelayedJobsController', () => {
import { Test, TestingModule } from "@nestjs/testing";
import { DelayedJobsController } from "./delayed-jobs.controller";
import { DelayedJob } from "@terramatch-microservices/database/entities";
import { DelayedJobBulkUpdateBodyDto, DelayedJobData } from "./dto/delayed-job-update.dto";
import { v4 as uuidv4 } from "uuid";
import { Logger, NotFoundException } from "@nestjs/common";
import { plainToClass } from "class-transformer";
import { validate } from "class-validator";
describe("DelayedJobsController", () => {
let controller: DelayedJobsController;

beforeEach(async () => {
Expand All @@ -26,84 +25,103 @@ describe('DelayedJobsController', () => {
afterEach(() => {
jest.restoreAllMocks();
});

describe('getRunningJobs', () => {
it('should return a list of running jobs for the authenticated user', async () => {
describe("getRunningJobs", () => {
it("should return a job with entity_name if metadata exists", async () => {
const authenticatedUserId = 130999;

const job = await DelayedJob.create({
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: 'completed',
status: "completed",
metadata: { entity_name: "TestEntity" }
});
const request = {
authenticatedUserId,
};


const request = { authenticatedUserId };
const result = await controller.getRunningJobs(request);

const data = Array.isArray(result.data) ? result.data : [result.data];

expect(data).toHaveLength(1);
expect(data[0].id).toBe(job.uuid);

const entityName = data[0].attributes.entityName;
expect(entityName).toBe("TestEntity");
});
it('should return an empty list when there are no running jobs', async () => {

it("should return a job with null entity_name if metadata does not have entity_name", async () => {
const authenticatedUserId = 130999;

const job = await DelayedJob.create({
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: "completed",
metadata: {}
});

const request = { authenticatedUserId };

const result = await controller.getRunningJobs(request);
expect(result.data).toHaveLength(0);

const data = Array.isArray(result.data) ? result.data : [result.data];

expect(data).toHaveLength(1);
expect(data[0].id).toBe(job.uuid);

const entityName = data[0].attributes.entityName;
expect(entityName).toBeUndefined();
});
});

describe('findOne', () => {
it('should return a job by UUID', async () => {
describe("findOne", () => {
it("should return a job by UUID with entity_name", async () => {
const authenticatedUserId = 130999;
const job = await DelayedJob.create({
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: 'completed'
status: "completed",
metadata: { entity_name: "TestEntity" } // Adding entity_name
});

const result = await controller.findOne(job.uuid);
const jobData = Array.isArray(result.data) ? result.data[0] : result.data;
expect(jobData.id).toBe(job.uuid);
});
it('should throw NotFoundException when the job does not exist', async () => {
it("should throw NotFoundException when the job does not exist", async () => {
const nonExistentUuid = uuidv4();

await expect(controller.findOne(nonExistentUuid)).rejects.toThrow(NotFoundException);
});

});

describe('bulkUdpateJobs', () => {
it('should successfully bulk update jobs to acknowledged', async () => {
describe("bulkUdpateJobs", () => {
it("should successfully bulk update jobs to acknowledged with entity_name", async () => {
const authenticatedUserId = 130999;
const job1 = await DelayedJob.create({
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: 'completed'
status: "completed",
metadata: { entity_name: "TestEntity1" } // Adding entity_name
});
const job2 = await DelayedJob.create({
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: 'failed'
status: "failed",
metadata: { entity_name: "TestEntity2" } // Adding entity_name
});

const payload: DelayedJobBulkUpdateBodyDto = {
data: [
{
type: 'delayedJobs',
type: "delayedJobs",
uuid: job1.uuid,
attributes: { isAcknowledged: true }
},
{
type: 'delayedJobs',
type: "delayedJobs",
uuid: job2.uuid,
attributes: { isAcknowledged: true }
}
Expand All @@ -113,31 +131,31 @@ describe('DelayedJobsController', () => {
const request = { authenticatedUserId };

const result = await controller.bulkUpdateJobs(payload, request);

expect(result.data).toHaveLength(2);
expect(result.data[0].id).toBe(job1.uuid);
expect(result.data[1].id).toBe(job2.uuid);
expect(result.data[0].attributes.entityName).toBe("TestEntity1"); // Check entity_name for job1
expect(result.data[1].attributes.entityName).toBe("TestEntity2"); // Check entity_name for job2

const updatedJob1 = await DelayedJob.findOne({ where: { uuid: job1.uuid } });
const updatedJob2 = await DelayedJob.findOne({ where: { uuid: job2.uuid } });
expect(updatedJob1.isAcknowledged).toBe(true);
expect(updatedJob2.isAcknowledged).toBe(true);
});

it('should throw NotFoundException for non-existent job', async () => {
it("should throw NotFoundException for non-existent job", async () => {
const payload: DelayedJobBulkUpdateBodyDto = {
data: [
{
type: 'delayedJobs',
uuid: 'non-existent-uuid',
type: "delayedJobs",
uuid: "non-existent-uuid",
attributes: { isAcknowledged: true }
}
]
};
const request = { authenticatedUserId: 130999 };

await expect(controller.bulkUpdateJobs(payload, request))
.rejects.toThrow(NotFoundException);
await expect(controller.bulkUpdateJobs(payload, request)).rejects.toThrow(NotFoundException);
});

it('should not update jobs with status "pending"', async () => {
Expand All @@ -146,62 +164,63 @@ describe('DelayedJobsController', () => {
uuid: uuidv4(),
createdBy: authenticatedUserId,
isAcknowledged: false,
status: 'pending'
status: "pending",
metadata: { entity_name: "TestEntityPending" } // Adding entity_name
});

const payload: DelayedJobBulkUpdateBodyDto = {
data: [
{
type: 'delayedJobs',
type: "delayedJobs",
uuid: pendingJob.uuid,
attributes: { isAcknowledged: true }
}
]
};
const request = { authenticatedUserId };

await expect(controller.bulkUpdateJobs(payload, request))
.rejects.toThrow(NotFoundException);
await expect(controller.bulkUpdateJobs(payload, request)).rejects.toThrow(NotFoundException);
});

});
describe('DelayedJobAttributes', () => {
it('should require an array of DelayedJobData', async () => {

describe("DelayedJobAttributes", () => {
it("should require an array of DelayedJobData", async () => {
const invalidData = {
data: 'not an array'
data: "not an array"
};

const invalidInstance = plainToClass(DelayedJobBulkUpdateBodyDto, invalidData);
const invalidResult = await validate(invalidInstance);

expect(invalidResult).toHaveLength(1);
expect(invalidResult[0].constraints).toHaveProperty('isArray');
expect(invalidResult[0].constraints).toHaveProperty("isArray");
});
it('should validate nested DelayedJobAttributes', async () => {

it("should validate nested DelayedJobAttributes", async () => {
const validData = {
type: 'delayedJobs',
type: "delayedJobs",
uuid: uuidv4(),
attributes: { isAcknowledged: true }
};

const invalidData = {
type: 'delayedJobs',
type: "delayedJobs",
uuid: uuidv4(),
attributes: {
isAcknowledged: 'not a boolean'
attributes: {
isAcknowledged: "not a boolean"
}
};

const validInstance = plainToClass(DelayedJobData, validData);
const validResult = await validate(validInstance);
expect(validResult).toHaveLength(0);
const invalidInstance = plainToClass(DelayedJobData, invalidData);
const invalidResult = await validate(invalidInstance);
expect(invalidResult).toHaveLength(1);
expect(invalidResult[0].property).toBe('attributes');
expect(invalidResult[0].property).toBe("attributes");
const nestedErrors = invalidResult[0].children;
expect(nestedErrors).toHaveLength(1);
expect(nestedErrors[0].constraints).toHaveProperty('isBoolean');
expect(nestedErrors[0].constraints).toHaveProperty("isBoolean");
});
});
});
Loading

0 comments on commit 84badc0

Please sign in to comment.