Skip to content

Commit

Permalink
feat(Rest): Add Rest visual entity
Browse files Browse the repository at this point in the history
This commit adds support for the Rest DSL from Apache Camel.

fix: KaotoIO#1505
fix: KaotoIO#54
  • Loading branch information
lordrip committed Dec 16, 2024
1 parent ee781f4 commit f5f4738
Show file tree
Hide file tree
Showing 9 changed files with 414 additions and 4 deletions.
6 changes: 4 additions & 2 deletions packages/ui/src/models/camel/camel-route-resource.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import { CamelYamlDsl, RouteDefinition } from '@kaoto/camel-catalog/types';
import { TileFilter } from '../../components/Catalog';
import { YamlCamelResourceSerializer } from '../../serializers';
import { CamelResourceSerializer } from '../../serializers/camel-resource-serializer';
import { createCamelPropertiesSorter, isDefined } from '../../utils';
import { CatalogKind } from '../catalog-kind';
import { AddStepMode, BaseVisualCamelEntityConstructor } from '../visualization/base-visual-entity';
Expand All @@ -11,6 +13,7 @@ import { CamelInterceptVisualEntity } from '../visualization/flows/camel-interce
import { CamelOnCompletionVisualEntity } from '../visualization/flows/camel-on-completion-visual-entity';
import { CamelOnExceptionVisualEntity } from '../visualization/flows/camel-on-exception-visual-entity';
import { CamelRestConfigurationVisualEntity } from '../visualization/flows/camel-rest-configuration-visual-entity';
import { CamelRestVisualEntity } from '../visualization/flows/camel-rest-visual-entity';
import { CamelRouteConfigurationVisualEntity } from '../visualization/flows/camel-route-configuration-visual-entity';
import { NonVisualEntity } from '../visualization/flows/non-visual-entity';
import { CamelComponentFilterService } from '../visualization/flows/support/camel-component-filter.service';
Expand All @@ -20,8 +23,6 @@ import { BeansEntity, isBeans } from '../visualization/metadata';
import { BaseVisualCamelEntityDefinition, BeansAwareResource, CamelResource } from './camel-resource';
import { BaseCamelEntity, EntityType } from './entities';
import { SourceSchemaType } from './source-schema-type';
import { CamelResourceSerializer } from '../../serializers/camel-resource-serializer';
import { YamlCamelResourceSerializer } from '../../serializers';

export class CamelRouteResource implements CamelResource, BeansAwareResource {
static readonly SUPPORTED_ENTITIES: { type: EntityType; group: string; Entity: BaseVisualCamelEntityConstructor }[] =
Expand All @@ -39,6 +40,7 @@ export class CamelRouteResource implements CamelResource, BeansAwareResource {
{ type: EntityType.OnException, group: 'Error Handling', Entity: CamelOnExceptionVisualEntity },
{ type: EntityType.ErrorHandler, group: 'Error Handling', Entity: CamelErrorHandlerVisualEntity },
{ type: EntityType.RestConfiguration, group: 'Rest', Entity: CamelRestConfigurationVisualEntity },
{ type: EntityType.Rest, group: 'Rest', Entity: CamelRestVisualEntity },
];
static readonly PARAMETERS_ORDER = ['id', 'description', 'uri', 'parameters', 'steps'];
private static readonly ERROR_RELATED_ENTITIES = [EntityType.OnException, EntityType.ErrorHandler];
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -139,8 +139,8 @@ export class CamelRestConfigurationVisualEntity implements BaseVisualCamelEntity
return restConfigurationGroupNode;
}

toJSON(): unknown {
return this.restConfigurationDef;
toJSON(): { restConfiguration: RestConfiguration } {
return { restConfiguration: this.restConfigurationDef.restConfiguration };
}

private getValidatorFunction(
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,201 @@
import catalogLibrary from '@kaoto/camel-catalog/index.json';
import { CatalogLibrary, Rest } from '@kaoto/camel-catalog/types';
import { restStub } from '../../../stubs/rest';
import { getFirstCatalogMap } from '../../../stubs/test-load-catalog';
import { EntityType } from '../../camel/entities';
import { CatalogKind } from '../../catalog-kind';
import { CamelCatalogService } from './camel-catalog.service';
import { CamelRestVisualEntity } from './camel-rest-visual-entity';
import { KaotoSchemaDefinition } from '../../kaoto-schema';

describe('CamelRestVisualEntity', () => {
const REST_ID_REGEXP = /^rest-[a-zA-Z0-9]{4}$/;
let restDef: { rest: Rest };
let restSchema: KaotoSchemaDefinition['schema'];

beforeAll(async () => {
const catalogsMap = await getFirstCatalogMap(catalogLibrary as CatalogLibrary);
CamelCatalogService.setCatalogKey(CatalogKind.Entity, catalogsMap.entitiesCatalog);
restSchema = catalogsMap.entitiesCatalog[EntityType.Rest].propertiesSchema as KaotoSchemaDefinition['schema'];
});

afterAll(() => {
CamelCatalogService.clearCatalogs();
});

beforeEach(() => {
restDef = {
rest: {
...restStub.rest,
},
};
});

describe('isApplicable', () => {
it.each([
[true, { rest: {} }],
[true, { rest: { bindingMode: 'off' } }],
[true, restStub],
[false, { from: { id: 'from-1234', steps: [] } }],
[false, { rest: { bindingMode: 'off' }, anotherProperty: true }],
])('should return %s for %s', (result, definition) => {
expect(CamelRestVisualEntity.isApplicable(definition)).toEqual(result);
});
});

describe('constructor', () => {
it('should set id to generated id', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.id).toMatch(REST_ID_REGEXP);
});
});

it('should return id', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getId()).toMatch(REST_ID_REGEXP);
});

it('should set id', () => {
const entity = new CamelRestVisualEntity(restDef);
const newId = 'newId';
entity.setId(newId);

expect(entity.getId()).toEqual(newId);
});

it('should return node label', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getNodeLabel()).toEqual('rest');
});

it('should return tooltip content', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getTooltipContent()).toEqual('rest');
});

describe('getComponentSchema', () => {
it('should return entity current definition', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getComponentSchema().definition).toEqual(restDef.rest);
});

it('should return schema from store', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getComponentSchema().schema).toEqual(restSchema);
});
});

describe('updateModel', () => {
it('should update model', () => {
const entity = new CamelRestVisualEntity(restDef);
const path = 'rest.bindingMode';
const value = 'json';

entity.updateModel(path, value);

expect(restDef.rest.bindingMode).toEqual(value);
});

it('should not update model if path is not defined', () => {
const entity = new CamelRestVisualEntity(restDef);
const value = 'json_xml';

entity.updateModel(undefined, value);

expect(restDef.rest.bindingMode).toEqual('auto');
});

it('should reset the rest object if it is not defined', () => {
const entity = new CamelRestVisualEntity(restDef);

entity.updateModel('rest', {});

expect(restDef.rest).toEqual({});
});
});

it('return no interactions', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.getNodeInteraction()).toEqual({
canHavePreviousStep: false,
canHaveNextStep: false,
canHaveChildren: false,
canHaveSpecialChildren: false,
canRemoveStep: false,
canReplaceStep: false,
canRemoveFlow: true,
canBeDisabled: false,
});
});

describe('getNodeValidationText', () => {
it('should return undefined for valid definitions', () => {
const entity = new CamelRestVisualEntity({
rest: {
...restDef.rest,
bindingMode: 'json',
},
});

expect(entity.getNodeValidationText()).toBeUndefined();
});

it('should not modify the original definition when validating', () => {
const originalRestDef: Rest = { ...restDef.rest };
const entity = new CamelRestVisualEntity(restDef);

entity.getNodeValidationText();

expect(restDef.rest).toEqual(originalRestDef);
});

it('should return errors when there is an invalid property', () => {
const invalidRestDef: Rest = {
...restDef.rest,
bindingMode: 'true' as unknown as Rest['bindingMode'],
openApi: 'true' as unknown as Rest['openApi'],
};
const entity = new CamelRestVisualEntity({ rest: invalidRestDef });

expect(entity.getNodeValidationText()).toEqual(`'/bindingMode' must be equal to one of the allowed values,
'/openApi' must be object`);
});
});

describe('toVizNode', () => {
it('should return visualization node', () => {
const entity = new CamelRestVisualEntity(restDef);

const vizNode = entity.toVizNode();

expect(vizNode.data).toEqual({
componentName: undefined,
entity,
icon: '',
isGroup: true,
path: 'rest',
processorName: 'rest',
});
});

it('should return hardcoded schema title', () => {
const entity = new CamelRestVisualEntity(restDef);
const vizNode = entity.toVizNode();

expect(vizNode.getTitle()).toEqual('Rest');
});
});

it('should serialize the rest definition', () => {
const entity = new CamelRestVisualEntity(restDef);

expect(entity.toJSON()).toEqual(restDef);
});
});
Loading

0 comments on commit f5f4738

Please sign in to comment.