Skip to content
Snippets Groups Projects
Verified Commit 9b4e0454 authored by Konstantin Tsabolov's avatar Konstantin Tsabolov
Browse files

feat: implement schemas endpoints in schema manager

parent ad2758be
No related branches found
No related tags found
1 merge request!17Schemas and credential definitions. Refs #9
import type { TestingModule } from '@nestjs/testing'; import type { TestingModule } from '@nestjs/testing';
import type { EventAnonCredsSchemasRegisterInput } from '@ocm/shared';
import { Test } from '@nestjs/testing'; import { Test } from '@nestjs/testing';
import { import {
...@@ -31,23 +32,20 @@ describe('SchemasService', () => { ...@@ -31,23 +32,20 @@ describe('SchemasService', () => {
describe('getAll', () => { describe('getAll', () => {
it('should return the data from NATS client', (done) => { it('should return the data from NATS client', (done) => {
const unsubscribe$ = new Subject<void>(); const unsubscribe$ = new Subject<void>();
const payload = { const tenantId = 'mocked tenantId';
tenantId: 'mocked tenantId',
endpoint: EventAnonCredsSchemasGetAll.token,
};
const expectedResult: EventAnonCredsSchemasGetAll['data'] = []; const expectedResult: EventAnonCredsSchemasGetAll['data'] = [];
natsClientMock.send.mockReturnValueOnce( natsClientMock.send.mockReturnValueOnce(
of(new EventAnonCredsSchemasGetAll([], payload.tenantId)), of(new EventAnonCredsSchemasGetAll([], tenantId)),
); );
service service
.getAll(payload) .getAll(tenantId)
.pipe(takeUntil(unsubscribe$)) .pipe(takeUntil(unsubscribe$))
.subscribe((result) => { .subscribe((result) => {
expect(natsClientMock.send).toHaveBeenCalledWith( expect(natsClientMock.send).toHaveBeenCalledWith(
{ endpoint: EventAnonCredsSchemasGetAll.token }, EventAnonCredsSchemasGetAll.token,
payload, { tenantId },
); );
expect(result).toStrictEqual(expectedResult); expect(result).toStrictEqual(expectedResult);
...@@ -63,10 +61,8 @@ describe('SchemasService', () => { ...@@ -63,10 +61,8 @@ describe('SchemasService', () => {
describe('getById', () => { describe('getById', () => {
it('should return the data from NATS client', (done) => { it('should return the data from NATS client', (done) => {
const unsubscribe$ = new Subject<void>(); const unsubscribe$ = new Subject<void>();
const payload = { const tenantId = 'mocked tenantId';
tenantId: 'mocked tenantId', const schemaId = 'mocked id';
schemaId: 'mocked id',
};
const expectedResult: EventAnonCredsSchemasGetById['data'] = { const expectedResult: EventAnonCredsSchemasGetById['data'] = {
issuerId: 'mocked issuerDid', issuerId: 'mocked issuerDid',
name: 'mocked name', name: 'mocked name',
...@@ -75,16 +71,16 @@ describe('SchemasService', () => { ...@@ -75,16 +71,16 @@ describe('SchemasService', () => {
}; };
natsClientMock.send.mockReturnValueOnce( natsClientMock.send.mockReturnValueOnce(
of(new EventAnonCredsSchemasGetById(expectedResult, payload.tenantId)), of(new EventAnonCredsSchemasGetById(expectedResult, tenantId)),
); );
service service
.getById(payload) .getById(tenantId, schemaId)
.pipe(takeUntil(unsubscribe$)) .pipe(takeUntil(unsubscribe$))
.subscribe((result) => { .subscribe((result) => {
expect(natsClientMock.send).toHaveBeenCalledWith( expect(natsClientMock.send).toHaveBeenCalledWith(
{ endpoint: EventAnonCredsSchemasGetById.token }, EventAnonCredsSchemasGetById.token,
payload, { tenantId, schemaId },
); );
expect(result).toStrictEqual(expectedResult); expect(result).toStrictEqual(expectedResult);
...@@ -100,31 +96,31 @@ describe('SchemasService', () => { ...@@ -100,31 +96,31 @@ describe('SchemasService', () => {
describe('register', () => { describe('register', () => {
it('should return the data from NATS client', (done) => { it('should return the data from NATS client', (done) => {
const unsubscribe$ = new Subject<void>(); const unsubscribe$ = new Subject<void>();
const payload = { const tenantId = 'mocked tenantId';
tenantId: 'mocked tenantId', const payload: Omit<EventAnonCredsSchemasRegisterInput, 'tenantId'> = {
issuerDid: 'mocked issuerDid', issuerDid: 'mocked issuerDid',
name: 'mocked name', name: 'mocked name',
version: '1.0.0', version: '1.0.0',
attributeNames: ['mocked attribute1', 'mocked attribute2'], attributeNames: ['mocked attribute1', 'mocked attribute2'],
}; };
const expectedResult: EventAnonCredsSchemasRegister['data'] = { const expectedResult: EventAnonCredsSchemasRegister['data'] = {
issuerId: 'mocked issuerDid', issuerId: 'mocked issuerId',
name: 'mocked name', name: 'mocked name',
version: '1.0.0', version: '1.0.0',
attrNames: ['mocked attribute1', 'mocked attribute2'], attrNames: ['mocked attribute1', 'mocked attribute2'],
}; };
natsClientMock.send.mockReturnValueOnce( natsClientMock.send.mockReturnValueOnce(
of(new EventAnonCredsSchemasRegister(expectedResult, payload.tenantId)), of(new EventAnonCredsSchemasRegister(expectedResult, tenantId)),
); );
service service
.register(payload) .register(tenantId, payload)
.pipe(takeUntil(unsubscribe$)) .pipe(takeUntil(unsubscribe$))
.subscribe((result) => { .subscribe((result) => {
expect(natsClientMock.send).toHaveBeenCalledWith( expect(natsClientMock.send).toHaveBeenCalledWith(
{ endpoint: EventAnonCredsSchemasRegister.token }, EventAnonCredsSchemasRegister.token,
payload, { ...payload, tenantId },
); );
expect(result).toStrictEqual(expectedResult); expect(result).toStrictEqual(expectedResult);
......
import { ApiProperty } from '@nestjs/swagger';
import { IsNotEmpty, IsString } from 'class-validator';
export class TenantIdParam {
@IsString()
@IsNotEmpty()
@ApiProperty({
description: 'The tenant ID to use for the request',
format: 'string',
})
public tenantId: string;
}
...@@ -13,27 +13,24 @@ import { ...@@ -13,27 +13,24 @@ import {
Param, Param,
Post, Post,
Query, Query,
UseInterceptors,
UsePipes, UsePipes,
ValidationPipe, ValidationPipe,
Version, Version,
} from '@nestjs/common'; } from '@nestjs/common';
import { import { ApiOperation, ApiResponse, ApiTags } from '@nestjs/swagger';
ApiBody, import { MultitenancyParams } from '@ocm/shared';
ApiOperation,
ApiParam,
ApiQuery,
ApiResponse,
ApiTags,
} from '@nestjs/swagger';
import { Observable, of, switchMap } from 'rxjs'; import { Observable, of, switchMap } from 'rxjs';
import { ResponseFormatInterceptor } from '../common/response-format.interceptor.js';
import { GetByIdParams } from './dto/get-by-id.dto.js'; import { GetByIdParams } from './dto/get-by-id.dto.js';
import { RegisterSchemaPayload } from './dto/register-schema.dto.js'; import { RegisterSchemaPayload } from './dto/register-schema.dto.js';
import { TenantIdParam } from './dto/tenant-id.dto.js';
import { SchemasService } from './schemas.service.js'; import { SchemasService } from './schemas.service.js';
@Controller('schemas') @Controller()
@UsePipes(new ValidationPipe({ transform: true, whitelist: true })) @UsePipes(new ValidationPipe({ transform: true, whitelist: true }))
@UseInterceptors(ResponseFormatInterceptor)
@ApiTags('Schemas') @ApiTags('Schemas')
export class SchemasController { export class SchemasController {
public constructor(private readonly schemasService: SchemasService) {} public constructor(private readonly schemasService: SchemasService) {}
...@@ -44,34 +41,64 @@ export class SchemasController { ...@@ -44,34 +41,64 @@ export class SchemasController {
summary: 'Fetch a list of schemas', summary: 'Fetch a list of schemas',
description: 'This call provides a list of schemas for a given tenant', description: 'This call provides a list of schemas for a given tenant',
}) })
@ApiQuery({ name: 'tenantId', required: true })
@ApiResponse({ @ApiResponse({
status: HttpStatus.OK, status: HttpStatus.OK,
description: 'Schemas fetched successfully', description: 'Schemas fetched successfully',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Schemas fetched successfully': {
value: {
statusCode: 200,
message: 'Schemas fetched successfully',
data: [],
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.NOT_FOUND, status: HttpStatus.NOT_FOUND,
description: 'Tenant not found', description: 'Tenant not found',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Tenant not found': {
value: {
statusCode: 404,
message: 'Tenant not found',
data: null,
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.INTERNAL_SERVER_ERROR, status: HttpStatus.INTERNAL_SERVER_ERROR,
description: 'Internal server error', description: 'Internal server error',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Internal server error': {
value: {
statusCode: 500,
message: 'Internal server error',
error: 'Internal Server Error',
},
},
},
},
}, },
}) })
public getAll( public getAll(
@Query() { tenantId }: TenantIdParam, @Query() { tenantId }: MultitenancyParams,
): Observable<EventAnonCredsSchemasGetAll['data']> { ): Observable<EventAnonCredsSchemasGetAll['data']> {
return this.schemasService.getAll({ return this.schemasService.getAll(tenantId);
tenantId,
});
} }
@Version('1') @Version('1')
...@@ -81,53 +108,92 @@ export class SchemasController { ...@@ -81,53 +108,92 @@ export class SchemasController {
description: description:
'This call allows you to retrieve schema data for a given tenant by specifying the `schemaId`.', 'This call allows you to retrieve schema data for a given tenant by specifying the `schemaId`.',
}) })
@ApiParam({ name: 'schemaId', required: true })
@ApiQuery({ name: 'tenantId', required: true })
@ApiResponse({ @ApiResponse({
status: HttpStatus.OK, status: HttpStatus.OK,
description: 'Schema fetched successfully', description: 'Schema fetched successfully',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Schema fetched successfully': {
value: {
statusCode: 200,
message: 'Schema fetched successfully',
data: {
id: '71b784a3',
},
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.NOT_FOUND, status: HttpStatus.NOT_FOUND,
description: 'Tenant not found', description: 'Tenant not found',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Tenant not found': {
value: {
statusCode: 404,
message: 'Tenant not found',
data: null,
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.NOT_FOUND, status: HttpStatus.NOT_FOUND,
description: 'Schema not found', description: 'Schema not found',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Schema not found': {
value: {
statusCode: 404,
message: 'Schema not found',
data: null,
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.INTERNAL_SERVER_ERROR, status: HttpStatus.INTERNAL_SERVER_ERROR,
description: 'Internal server error', description: 'Internal server error',
content: { content: {
// TBD 'application/json': {
schema: {},
examples: {
'Internal server error': {
value: {
statusCode: 500,
message: 'Internal server error',
error: 'Internal Server Error',
},
},
},
},
}, },
}) })
public getById( public getById(
@Param() { schemaId }: GetByIdParams, @Param() { schemaId }: GetByIdParams,
@Query() { tenantId }: TenantIdParam, @Query() { tenantId }: MultitenancyParams,
): Observable<EventAnonCredsSchemasGetById['data']> { ): Observable<EventAnonCredsSchemasGetById['data']> {
return this.schemasService return this.schemasService.getById(tenantId, schemaId).pipe(
.getById({ switchMap((schema) => {
tenantId, if (schema === null) {
schemaId, throw new NotFoundException(`Schema with id ${schemaId} not found`);
}) }
.pipe( return of(schema);
switchMap((schema) => { }),
if (schema === null) { );
throw new NotFoundException(`Schema with id ${schemaId} not found`);
}
return of(schema);
}),
);
} }
@Version('1') @Version('1')
...@@ -137,50 +203,102 @@ export class SchemasController { ...@@ -137,50 +203,102 @@ export class SchemasController {
description: description:
'This call provides the capability to create new schema on ledger by name, author, version, schema attributes and type. Later this schema can be used to issue new credential definition. This call returns an information about created schema.', 'This call provides the capability to create new schema on ledger by name, author, version, schema attributes and type. Later this schema can be used to issue new credential definition. This call returns an information about created schema.',
}) })
@ApiQuery({ name: 'tenantId', required: true })
@ApiBody({ type: RegisterSchemaPayload })
@ApiResponse({ @ApiResponse({
status: HttpStatus.CREATED, status: HttpStatus.CREATED,
description: 'Schema registered successfully', description: 'Schema registered successfully',
content: { content: {
'application/json': {}, 'application/json': {
schema: {},
examples: {
'Schema registered successfully': {
value: {
statusCode: 201,
message: 'Schema registered successfully',
data: {
id: '71b784a3',
},
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.NOT_FOUND, status: HttpStatus.NOT_FOUND,
description: 'Tenant not found', description: 'Tenant not found',
content: { content: {
'application/json': {}, 'application/json': {
schema: {},
examples: {
'Tenant not found': {
value: {
statusCode: 404,
message: 'Tenant not found',
data: null,
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.BAD_REQUEST, status: HttpStatus.BAD_REQUEST,
description: 'All fields are required for schema registration', description: 'All fields are required for schema registration',
content: { content: {
'application/json': {}, 'application/json': {
schema: {},
examples: {
'All fields are required for schema registration': {
value: {
statusCode: 400,
message: 'All fields are required for schema registration',
error: 'Bad Request',
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.CONFLICT, status: HttpStatus.CONFLICT,
description: 'Schema already exists', description: 'Schema already exists',
content: { content: {
'application/json': {}, 'application/json': {
schema: {},
examples: {
'Schema already exists': {
value: {
statusCode: 409,
message: 'Schema already exists',
error: 'Conflict',
},
},
},
},
}, },
}) })
@ApiResponse({ @ApiResponse({
status: HttpStatus.INTERNAL_SERVER_ERROR, status: HttpStatus.INTERNAL_SERVER_ERROR,
description: 'Internal server error', description: 'Internal server error',
content: { content: {
'application/json': {}, 'application/json': {
schema: {},
examples: {
'Internal server error': {
value: {
statusCode: 500,
message: 'Internal server error',
error: 'Internal Server Error',
},
},
},
},
}, },
}) })
public register( public register(
@Query() { tenantId }: TenantIdParam, @Query() { tenantId }: MultitenancyParams,
@Body() payload: RegisterSchemaPayload, @Body() payload: RegisterSchemaPayload,
): Observable<EventAnonCredsSchemasRegister['data']> { ): Observable<EventAnonCredsSchemasRegister['data']> {
return this.schemasService.register({ return this.schemasService.register(tenantId, payload);
...payload,
tenantId,
});
} }
} }
...@@ -22,40 +22,36 @@ export class SchemasService { ...@@ -22,40 +22,36 @@ export class SchemasService {
) {} ) {}
public getAll( public getAll(
payload: EventAnonCredsSchemasGetAllInput, tenantId: string,
): Observable<EventAnonCredsSchemasGetAll['data']> { ): Observable<EventAnonCredsSchemasGetAll['data']> {
const pattern = { endpoint: EventAnonCredsSchemasGetAll.token };
return this.natsClient return this.natsClient
.send<EventAnonCredsSchemasGetAll, EventAnonCredsSchemasGetAllInput>( .send<EventAnonCredsSchemasGetAll, EventAnonCredsSchemasGetAllInput>(
pattern, EventAnonCredsSchemasGetAll.token,
payload, { tenantId },
) )
.pipe(map((result) => result.data)); .pipe(map((result) => result.data));
} }
public getById( public getById(
payload: EventAnonCredsSchemasGetByIdInput, tenantId: string,
schemaId: EventAnonCredsSchemasGetByIdInput['schemaId'],
): Observable<EventAnonCredsSchemasGetById['data']> { ): Observable<EventAnonCredsSchemasGetById['data']> {
const pattern = { endpoint: EventAnonCredsSchemasGetById.token };
return this.natsClient return this.natsClient
.send<EventAnonCredsSchemasGetById, EventAnonCredsSchemasGetByIdInput>( .send<EventAnonCredsSchemasGetById, EventAnonCredsSchemasGetByIdInput>(
pattern, EventAnonCredsSchemasGetById.token,
payload, { tenantId, schemaId },
) )
.pipe(map((result) => result.data)); .pipe(map((result) => result.data));
} }
public register( public register(
payload: EventAnonCredsSchemasRegisterInput, tenantId: string,
payload: Omit<EventAnonCredsSchemasRegisterInput, 'tenantId'>,
): Observable<EventAnonCredsSchemasRegister['data']> { ): Observable<EventAnonCredsSchemasRegister['data']> {
const pattern = { endpoint: EventAnonCredsSchemasRegister.token };
return this.natsClient return this.natsClient
.send<EventAnonCredsSchemasRegister, EventAnonCredsSchemasRegisterInput>( .send<EventAnonCredsSchemasRegister, EventAnonCredsSchemasRegisterInput>(
pattern, EventAnonCredsSchemasRegister.token,
payload, { ...payload, tenantId },
) )
.pipe(map((result) => result.data)); .pipe(map((result) => result.data));
} }
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment