Skip to content

Commit

Permalink
feat: project applications controller/service layer (#6184)
Browse files Browse the repository at this point in the history
Just adding controller/service layer, connecting with schema.
Next PR will implement store and e2e tests.
  • Loading branch information
sjaanus authored Feb 9, 2024
1 parent 1b1bde8 commit 4972b96
Show file tree
Hide file tree
Showing 5 changed files with 123 additions and 35 deletions.
Original file line number Diff line number Diff line change
@@ -1,18 +1,18 @@
import { Response } from 'express';
import Controller from '../../controller';
import Controller from '../../routes/controller';
import {
IArchivedQuery,
IProjectParam,
IUnleashConfig,
IUnleashServices,
NONE,
serializeDates,
} from '../../../types';
import ProjectFeaturesController from '../../../features/feature-toggle/feature-toggle-controller';
import EnvironmentsController from '../../../features/project-environments/environments';
import ProjectHealthReport from './health-report';
import ProjectService from '../../../services/project-service';
import VariantsController from './variants';
} from '../../types';
import ProjectFeaturesController from '../feature-toggle/feature-toggle-controller';
import EnvironmentsController from '../project-environments/environments';
import ProjectHealthReport from '../../routes/admin-api/project/health-report';
import ProjectService from '../../services/project-service';
import VariantsController from '../../routes/admin-api/project/variants';
import {
createResponseSchema,
ProjectDoraMetricsSchema,
Expand All @@ -22,18 +22,22 @@ import {
projectsSchema,
ProjectsSchema,
projectOverviewSchema,
} from '../../../openapi';
import { getStandardResponses } from '../../../openapi/util/standard-responses';
import { OpenApiService, SettingService } from '../../../services';
import { IAuthRequest } from '../../unleash-types';
import { ProjectApiTokenController } from './api-token';
import ProjectArchiveController from './project-archive';
import { createKnexTransactionStarter } from '../../../db/transaction';
import { Db } from '../../../db/db';
import DependentFeaturesController from '../../../features/dependent-features/dependent-features-controller';
import { ProjectOverviewSchema } from '../../../openapi/spec/project-overview-schema';

export default class ProjectApi extends Controller {
} from '../../openapi';
import { getStandardResponses } from '../../openapi/util/standard-responses';
import { OpenApiService, SettingService } from '../../services';
import { IAuthRequest } from '../../routes/unleash-types';
import { ProjectApiTokenController } from '../../routes/admin-api/project/api-token';
import ProjectArchiveController from '../../routes/admin-api/project/project-archive';
import { createKnexTransactionStarter } from '../../db/transaction';
import { Db } from '../../db/db';
import DependentFeaturesController from '../dependent-features/dependent-features-controller';
import { ProjectOverviewSchema } from '../../openapi/spec/project-overview-schema';
import {
projectApplicationsSchema,
ProjectApplicationsSchema,
} from '../../openapi/spec/project-applications-schema';

export default class ProjectController extends Controller {
private projectService: ProjectService;

private settingService: SettingService;
Expand Down Expand Up @@ -129,6 +133,26 @@ export default class ProjectApi extends Controller {
],
});

this.route({
method: 'get',
path: '/:projectId/applications',
handler: this.getProjectApplications,
permission: NONE,
middleware: [
services.openApiService.validPath({
tags: ['Unstable'],
operationId: 'getProjectApplications',
summary: 'Get a list of all applications for a project.',
description:
'This endpoint returns an list of all the applications for a project.',
responses: {
200: createResponseSchema('projectApplicationsSchema'),
...getStandardResponses(401, 403, 404),
},
}),
],
});

this.use(
'/',
new ProjectFeaturesController(
Expand Down Expand Up @@ -229,4 +253,21 @@ export default class ProjectApi extends Controller {
dora,
);
}

async getProjectApplications(
req: IAuthRequest,
res: Response<ProjectApplicationsSchema>,
): Promise<void> {
const { projectId } = req.params;

const applications =
await this.projectService.getApplications(projectId);

this.openApiService.respondWithValidation(
200,
res,
projectApplicationsSchema.$id,
applications,
);
}
}
Original file line number Diff line number Diff line change
@@ -1,14 +1,14 @@
import dbInit, { ITestDb } from '../../../helpers/database-init';
import dbInit, { ITestDb } from '../../../test/e2e/helpers/database-init';
import {
IUnleashTest,
insertFeatureEnvironmentsLastSeen,
insertLastSeenAt,
setupAppWithCustomConfig,
} from '../../../helpers/test-helper';
import getLogger from '../../../../fixtures/no-logger';
} from '../../../test/e2e/helpers/test-helper';
import getLogger from '../../../test/fixtures/no-logger';

import { IProjectStore } from '../../../../../lib/types';
import { DEFAULT_ENV } from '../../../../../lib/util';
import { IProjectStore } from '../../types';
import { DEFAULT_ENV } from '../../util';

let app: IUnleashTest;
let db: ITestDb;
Expand Down Expand Up @@ -143,9 +143,11 @@ test('response for default project should include created_at', async () => {
.expect(200);
expect(body.createdAt).toBeDefined();
});

test('response for project overview should include feature type counts', async () => {
await app.createFeature({ name: 'my-new-release-toggle', type: 'release' });
await app.createFeature({
name: 'my-new-release-toggle',
type: 'release',
});
await app.createFeature({
name: 'my-new-development-toggle',
type: 'development',
Expand All @@ -156,8 +158,14 @@ test('response for project overview should include feature type counts', async (
.expect(200);
expect(body).toMatchObject({
featureTypeCounts: [
{ type: 'development', count: 1 },
{ type: 'release', count: 1 },
{
type: 'development',
count: 1,
},
{
type: 'release',
count: 1,
},
],
});
});
Expand Down Expand Up @@ -278,3 +286,12 @@ test('response should include last seen at per environment for multiple environm

expect(body.features[1].lastSeenAt).toBe('2023-10-01T12:34:56.000Z');
});

test('should return empty list of applications', async () => {
const { body } = await app.request
.get('/api/admin/projects/default/applications')
.expect('Content-Type', /json/)
.expect(200);

expect(body).toMatchObject([]);
});
2 changes: 2 additions & 0 deletions src/lib/openapi/spec/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -175,3 +175,5 @@ export * from './feature-type-count-schema';
export * from './feature-search-response-schema';
export * from './inactive-user-schema';
export * from './inactive-users-schema';
export * from './project-application-schema';
export * from './project-applications-schema';
7 changes: 5 additions & 2 deletions src/lib/routes/admin-api/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ import UserAdminController from './user-admin';
import EmailController from './email';
import UserFeedbackController from './user-feedback';
import UserSplashController from './user-splash';
import ProjectApi from './project/project-api';
import ProjectController from '../../features/project/project-controller';
import { EnvironmentsController } from './environments';
import ConstraintsController from './constraints';
import PatController from './user/pat';
Expand Down Expand Up @@ -117,7 +117,10 @@ class AdminApi extends Controller {
'/feedback',
new UserFeedbackController(config, services).router,
);
this.app.use('/projects', new ProjectApi(config, services, db).router);
this.app.use(
'/projects',
new ProjectController(config, services, db).router,
);
this.app.use(
'/environments',
new EnvironmentsController(config, services).router,
Expand Down
35 changes: 30 additions & 5 deletions src/lib/services/project-service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,10 @@ import { calculateAverageTimeToProd } from '../features/feature-toggle/time-to-p
import { IProjectStatsStore } from '../types/stores/project-stats-store-type';
import { uniqueByKey } from '../util/unique';
import { BadDataError, PermissionError } from '../error';
import { ProjectDoraMetricsSchema } from '../openapi';
import {
ProjectDoraMetricsSchema,
ProjectApplicationsSchema,
} from '../openapi';
import { checkFeatureNamingData } from '../features/feature-naming-pattern/feature-naming-validation';
import { IPrivateProjectChecker } from '../features/private-project/privateProjectCheckerType';
import EventService from '../features/events/event-service';
Expand All @@ -89,7 +92,14 @@ interface ICalculateStatus {
updates: IProjectStats;
}

function includes(list: number[], { id }: { id: number }): boolean {
function includes(
list: number[],
{
id,
}: {
id: number;
},
): boolean {
return list.some((l) => l === id);
}

Expand Down Expand Up @@ -887,7 +897,16 @@ export default class ProjectService {
featureToggleNames,
);

return { features: toggleAverage, projectAverage: projectAverage };
return {
features: toggleAverage,
projectAverage: projectAverage,
};
}

async getApplications(
projectId: string,
): Promise<ProjectApplicationsSchema> {
return [];
}

async changeRole(
Expand Down Expand Up @@ -1091,7 +1110,10 @@ export default class ProjectService {
const [projectActivityCurrentWindow, projectActivityPastWindow] =
await Promise.all([
this.eventStore.queryCount([
{ op: 'where', parameters: { project: projectId } },
{
op: 'where',
parameters: { project: projectId },
},
{
op: 'beforeDate',
parameters: {
Expand All @@ -1101,7 +1123,10 @@ export default class ProjectService {
},
]),
this.eventStore.queryCount([
{ op: 'where', parameters: { project: projectId } },
{
op: 'where',
parameters: { project: projectId },
},
{
op: 'betweenDate',
parameters: {
Expand Down

0 comments on commit 4972b96

Please sign in to comment.