@platform-mesh/portal-server-lib 0.6.10 → 0.6.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,10 +1,8 @@
1
1
  export * from './account-entity-context-provider.service.js';
2
+ export * from './auth-config-provider.js';
3
+ export * from './logout-callback.service.js';
2
4
  export * from './pm-portal-context.service.js';
3
5
  export * from './pm-request-context-provider.js';
4
- export * from './auth-config-provider.js';
5
6
  export * from './service-providers/content-configuration-service-providers.service.js';
6
7
  export * from './service-providers/kubernetes-service-providers.service.js';
7
- export * from './auth-callback-provider.js';
8
- export * from './logout-callback.service.js';
9
8
  export * from './services/kcp-k8s.service.js';
10
- export * from './services/iam-graphql.service.js';
@@ -1,11 +1,9 @@
1
1
  export * from './account-entity-context-provider.service.js';
2
+ export * from './auth-config-provider.js';
3
+ export * from './logout-callback.service.js';
2
4
  export * from './pm-portal-context.service.js';
3
5
  export * from './pm-request-context-provider.js';
4
- export * from './auth-config-provider.js';
5
6
  export * from './service-providers/content-configuration-service-providers.service.js';
6
7
  export * from './service-providers/kubernetes-service-providers.service.js';
7
- export * from './auth-callback-provider.js';
8
- export * from './logout-callback.service.js';
9
8
  export * from './services/kcp-k8s.service.js';
10
- export * from './services/iam-graphql.service.js';
11
9
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/portal-options/index.ts"],"names":[],"mappings":"AAAA,cAAc,8CAA8C,CAAC;AAC7D,cAAc,gCAAgC,CAAC;AAC/C,cAAc,kCAAkC,CAAC;AACjD,cAAc,2BAA2B,CAAC;AAC1C,cAAc,wEAAwE,CAAC;AACvF,cAAc,6DAA6D,CAAC;AAC5E,cAAc,6BAA6B,CAAC;AAC5C,cAAc,8BAA8B,CAAC;AAE7C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,mCAAmC,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/portal-options/index.ts"],"names":[],"mappings":"AAAA,cAAc,8CAA8C,CAAC;AAC7D,cAAc,2BAA2B,CAAC;AAC1C,cAAc,8BAA8B,CAAC;AAC7C,cAAc,gCAAgC,CAAC;AAC/C,cAAc,kCAAkC,CAAC;AACjD,cAAc,wEAAwE,CAAC;AACvF,cAAc,6DAA6D,CAAC;AAE5E,cAAc,+BAA+B,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@platform-mesh/portal-server-lib",
3
- "version": "0.6.10",
3
+ "version": "0.6.11",
4
4
  "author": "Platform Mesh",
5
5
  "license": "Apache-2.0",
6
6
  "publishConfig": {
@@ -1,11 +1,9 @@
1
1
  export * from './account-entity-context-provider.service.js';
2
+ export * from './auth-config-provider.js';
3
+ export * from './logout-callback.service.js';
2
4
  export * from './pm-portal-context.service.js';
3
5
  export * from './pm-request-context-provider.js';
4
- export * from './auth-config-provider.js';
5
6
  export * from './service-providers/content-configuration-service-providers.service.js';
6
7
  export * from './service-providers/kubernetes-service-providers.service.js';
7
- export * from './auth-callback-provider.js';
8
- export * from './logout-callback.service.js';
9
8
 
10
9
  export * from './services/kcp-k8s.service.js';
11
- export * from './services/iam-graphql.service.js';
@@ -335,6 +335,35 @@ describe('KcpKubernetesService', () => {
335
335
  );
336
336
  });
337
337
 
338
+ it('executes post middleware', async () => {
339
+ const svc = new KcpKubernetesService();
340
+ const gvr: K8sResourceDescriptor = {
341
+ group: 'apps',
342
+ version: 'v1',
343
+ plural: 'deployments',
344
+ name: 'my-deployment',
345
+ };
346
+ const context: K8sRequestContext = {
347
+ organization: 'org1',
348
+ 'core_platform-mesh_io_account': 'acc1',
349
+ };
350
+
351
+ let postResult: any;
352
+ mockListClusterCustomObject.mockImplementation(async (_gvr, options) => {
353
+ const middleware = options.middleware[0];
354
+ const mockContext = {
355
+ setUrl: jest.fn(),
356
+ };
357
+ await middleware.options.pre(mockContext);
358
+ postResult = await middleware.options.post(mockContext);
359
+ return { data: {} };
360
+ });
361
+
362
+ await svc.listClusterCustomObject(gvr, context);
363
+
364
+ expect(postResult).toBeDefined();
365
+ });
366
+
338
367
  it('builds correct URL path in middleware', async () => {
339
368
  const svc = new KcpKubernetesService();
340
369
  const gvr: K8sResourceDescriptor = {
@@ -579,6 +608,30 @@ describe('KcpKubernetesService', () => {
579
608
  );
580
609
  });
581
610
 
611
+ it('executes post middleware', async () => {
612
+ const svc = new KcpKubernetesService();
613
+ const orgName = 'post-org';
614
+
615
+ let postResult: any;
616
+ mockReadNamespacedSecret.mockImplementation(async (_params, options) => {
617
+ const middleware = options.middleware[0];
618
+ const mockContext = {
619
+ setUrl: jest.fn(),
620
+ };
621
+ await middleware.options.pre(mockContext);
622
+ postResult = await middleware.options.post(mockContext);
623
+ return {
624
+ data: {
625
+ client_secret: Buffer.from('test').toString('base64'),
626
+ },
627
+ };
628
+ });
629
+
630
+ await svc.getClientSecret(orgName);
631
+
632
+ expect(postResult).toBeDefined();
633
+ });
634
+
582
635
  it('throws error when secret retrieval fails', async () => {
583
636
  const svc = new KcpKubernetesService();
584
637
  const orgName = 'fail-org';
@@ -1,10 +0,0 @@
1
- import { IAMGraphQlService } from './services/iam-graphql.service.js';
2
- import { AuthCallback, AuthTokenData } from '@openmfp/portal-server-lib';
3
- import { Request, Response } from 'express';
4
- export declare class AuthCallbackProvider implements AuthCallback {
5
- private iamService;
6
- private logger;
7
- constructor(iamService: IAMGraphQlService);
8
- handleSuccess(request: Request, response: Response, authTokenResponse: AuthTokenData): Promise<void>;
9
- handleFailure(request: Request, response: Response): Promise<void>;
10
- }
@@ -1,36 +0,0 @@
1
- var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
- return c > 3 && r && Object.defineProperty(target, key, r), r;
6
- };
7
- var __metadata = (this && this.__metadata) || function (k, v) {
8
- if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
- };
10
- var AuthCallbackProvider_1;
11
- import { IAMGraphQlService } from './services/iam-graphql.service.js';
12
- import { Injectable, Logger } from '@nestjs/common';
13
- let AuthCallbackProvider = AuthCallbackProvider_1 = class AuthCallbackProvider {
14
- iamService;
15
- logger = new Logger(AuthCallbackProvider_1.name);
16
- constructor(iamService) {
17
- this.iamService = iamService;
18
- }
19
- async handleSuccess(request, response, authTokenResponse) {
20
- try {
21
- await this.iamService.addUser(authTokenResponse.id_token, request, response);
22
- }
23
- catch (e) {
24
- this.logger.error(e);
25
- }
26
- }
27
- async handleFailure(request, response) {
28
- return Promise.resolve();
29
- }
30
- };
31
- AuthCallbackProvider = AuthCallbackProvider_1 = __decorate([
32
- Injectable(),
33
- __metadata("design:paramtypes", [IAMGraphQlService])
34
- ], AuthCallbackProvider);
35
- export { AuthCallbackProvider };
36
- //# sourceMappingURL=auth-callback-provider.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"auth-callback-provider.js","sourceRoot":"","sources":["../../src/portal-options/auth-callback-provider.ts"],"names":[],"mappings":";;;;;;;;;;AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,mCAAmC,CAAC;AACtE,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAK7C,IAAM,oBAAoB,4BAA1B,MAAM,oBAAoB;IAGX;IAFZ,MAAM,GAAW,IAAI,MAAM,CAAC,sBAAoB,CAAC,IAAI,CAAC,CAAC;IAE/D,YAAoB,UAA6B;QAA7B,eAAU,GAAV,UAAU,CAAmB;IAAG,CAAC;IAErD,KAAK,CAAC,aAAa,CACjB,OAAgB,EAChB,QAAkB,EAClB,iBAAgC;QAEhC,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,iBAAiB,CAAC,QAAQ,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;QAC/E,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACvB,CAAC;IACH,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,OAAgB,EAAE,QAAkB;QACtD,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;IAC3B,CAAC;CACF,CAAA;AApBY,oBAAoB;IADhC,UAAU,EAAE;qCAIqB,iBAAiB;GAHtC,oBAAoB,CAoBhC"}
@@ -1,7 +0,0 @@
1
- import { PMRequestContextProvider } from '../pm-request-context-provider.js';
2
- import type { Request, Response } from 'express';
3
- export declare class IAMGraphQlService {
4
- private requestContextProvider;
5
- constructor(requestContextProvider: PMRequestContextProvider);
6
- addUser(token: string, request: Request, response: Response): Promise<void>;
7
- }
@@ -1,40 +0,0 @@
1
- var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
- return c > 3 && r && Object.defineProperty(target, key, r), r;
6
- };
7
- var __metadata = (this && this.__metadata) || function (k, v) {
8
- if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
- };
10
- import { PMRequestContextProvider } from '../pm-request-context-provider.js';
11
- import { MUTATION_LOGIN } from './queries.js';
12
- import { Injectable } from '@nestjs/common';
13
- import { GraphQLClient } from 'graphql-request';
14
- let IAMGraphQlService = class IAMGraphQlService {
15
- requestContextProvider;
16
- constructor(requestContextProvider) {
17
- this.requestContextProvider = requestContextProvider;
18
- }
19
- async addUser(token, request, response) {
20
- const requestContext = await this.requestContextProvider.getContextValues(request, response);
21
- const iamUrl = requestContext.iamServiceApiUrl;
22
- const client = new GraphQLClient(iamUrl, {
23
- headers: {
24
- Authorization: `Bearer ${token}`,
25
- },
26
- });
27
- try {
28
- await client.request(MUTATION_LOGIN);
29
- }
30
- catch (e) {
31
- console.error(e);
32
- }
33
- }
34
- };
35
- IAMGraphQlService = __decorate([
36
- Injectable(),
37
- __metadata("design:paramtypes", [PMRequestContextProvider])
38
- ], IAMGraphQlService);
39
- export { IAMGraphQlService };
40
- //# sourceMappingURL=iam-graphql.service.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"iam-graphql.service.js","sourceRoot":"","sources":["../../../src/portal-options/services/iam-graphql.service.ts"],"names":[],"mappings":";;;;;;;;;AAAA,OAAO,EAAE,wBAAwB,EAAE,MAAM,mCAAmC,CAAC;AAC7E,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAE5C,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAGzC,IAAM,iBAAiB,GAAvB,MAAM,iBAAiB;IACR;IAApB,YAAoB,sBAAgD;QAAhD,2BAAsB,GAAtB,sBAAsB,CAA0B;IAAG,CAAC;IAExE,KAAK,CAAC,OAAO,CACX,KAAa,EACb,OAAgB,EAChB,QAAkB;QAElB,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,CACvE,OAAO,EACP,QAAQ,CACT,CAAC;QACF,MAAM,MAAM,GAAG,cAAc,CAAC,gBAAgB,CAAC;QAC/C,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,MAAM,EAAE;YACvC,OAAO,EAAE;gBACP,aAAa,EAAE,UAAU,KAAK,EAAE;aACjC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC;YACH,MAAM,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QACvC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACnB,CAAC;IACH,CAAC;CACF,CAAA;AAzBY,iBAAiB;IAD7B,UAAU,EAAE;qCAEiC,wBAAwB;GADzD,iBAAiB,CAyB7B"}
@@ -1,85 +0,0 @@
1
- import { AuthCallbackProvider } from './auth-callback-provider.js';
2
- import { IAMGraphQlService } from './services/iam-graphql.service.js';
3
- import { Test, TestingModule } from '@nestjs/testing';
4
- import type { AuthTokenData } from '@openmfp/portal-server-lib';
5
- import type { Request, Response } from 'express';
6
- import { mock } from 'jest-mock-extended';
7
-
8
- jest.mock('@kubernetes/client-node', () => {
9
- class KubeConfig {
10
- loadFromDefault = jest.fn();
11
- loadFromFile = jest.fn();
12
- getCurrentCluster = jest.fn().mockReturnValue({
13
- server: 'https://k8s.example.com/base',
14
- name: 'test-cluster',
15
- });
16
- makeApiClient = jest.fn();
17
- addUser = jest.fn();
18
- addContext = jest.fn();
19
- setCurrentContext = jest.fn();
20
- }
21
- class CustomObjectsApi {}
22
- return { KubeConfig, CustomObjectsApi };
23
- });
24
-
25
- describe('AuthCallbackProvider', () => {
26
- let callback: AuthCallbackProvider;
27
- let iamServiceMock: IAMGraphQlService;
28
-
29
- beforeEach(async () => {
30
- iamServiceMock = mock<IAMGraphQlService>();
31
- const module: TestingModule = await Test.createTestingModule({
32
- providers: [
33
- AuthCallbackProvider,
34
- {
35
- provide: IAMGraphQlService,
36
- useValue: iamServiceMock,
37
- },
38
- ],
39
- }).compile();
40
-
41
- callback = module.get<AuthCallbackProvider>(AuthCallbackProvider);
42
- });
43
-
44
- it('should be defined', () => {
45
- expect(callback).toBeDefined();
46
- });
47
-
48
- it('should create a user', async () => {
49
- const req = mock<Request>();
50
- const res = mock<Response>();
51
-
52
- await callback.handleSuccess(req, res, {
53
- id_token: 'idtoken',
54
- } as AuthTokenData);
55
-
56
- expect(iamServiceMock.addUser).toHaveBeenCalledTimes(1);
57
- expect(iamServiceMock.addUser).toHaveBeenCalledWith('idtoken', req, res);
58
- });
59
-
60
- it('should log error if addUser throws', async () => {
61
- const req = mock<Request>();
62
- const res = mock<Response>();
63
- const error = new Error('boom');
64
- (iamServiceMock.addUser as jest.Mock).mockRejectedValueOnce(error);
65
-
66
- const errorSpy = jest
67
-
68
- .spyOn((callback as any).logger, 'error')
69
- .mockImplementation(() => undefined as unknown as never);
70
-
71
- await callback.handleSuccess(req, res, {
72
- id_token: 'bad',
73
- } as AuthTokenData);
74
-
75
- expect(iamServiceMock.addUser).toHaveBeenCalledTimes(1);
76
- expect(errorSpy).toHaveBeenCalledWith(error);
77
- });
78
-
79
- it('should resolve handleFailure without action', async () => {
80
- const req = mock<Request>();
81
- const res = mock<Response>();
82
-
83
- await expect(callback.handleFailure(req, res)).resolves.toBeUndefined();
84
- });
85
- });
@@ -1,27 +0,0 @@
1
- import { IAMGraphQlService } from './services/iam-graphql.service.js';
2
- import { Injectable, Logger } from '@nestjs/common';
3
- import { AuthCallback, AuthTokenData } from '@openmfp/portal-server-lib';
4
- import { Request, Response } from 'express';
5
-
6
- @Injectable()
7
- export class AuthCallbackProvider implements AuthCallback {
8
- private logger: Logger = new Logger(AuthCallbackProvider.name);
9
-
10
- constructor(private iamService: IAMGraphQlService) {}
11
-
12
- async handleSuccess(
13
- request: Request,
14
- response: Response,
15
- authTokenResponse: AuthTokenData,
16
- ): Promise<void> {
17
- try {
18
- await this.iamService.addUser(authTokenResponse.id_token, request, response);
19
- } catch (e) {
20
- this.logger.error(e);
21
- }
22
- }
23
-
24
- async handleFailure(request: Request, response: Response): Promise<void> {
25
- return Promise.resolve();
26
- }
27
- }
@@ -1,77 +0,0 @@
1
- import { PMRequestContextProvider } from '../pm-request-context-provider.js';
2
- import { IAMGraphQlService } from './iam-graphql.service.js';
3
- import { MUTATION_LOGIN } from './queries.js';
4
- import { GraphQLClient } from 'graphql-request';
5
- import { mock } from 'jest-mock-extended';
6
-
7
- jest.mock('@kubernetes/client-node', () => {
8
- class KubeConfig {
9
- loadFromDefault = jest.fn();
10
- loadFromFile = jest.fn();
11
- getCurrentCluster = jest.fn().mockReturnValue({
12
- server: 'https://k8s.example.com/base',
13
- name: 'test-cluster',
14
- });
15
- makeApiClient = jest.fn();
16
- addUser = jest.fn();
17
- addContext = jest.fn();
18
- setCurrentContext = jest.fn();
19
- }
20
- class CustomObjectsApi {}
21
- return { KubeConfig, CustomObjectsApi };
22
- });
23
-
24
- describe('IAMGraphQlService', () => {
25
- const mockIamServiceApiUrl = 'http://localhost:8080/query';
26
- let service: IAMGraphQlService;
27
- const gqlClient = {
28
- request: jest.fn(),
29
- } as unknown as GraphQLClient;
30
-
31
- const requestContextProvider = mock<PMRequestContextProvider>({
32
- getContextValues: jest
33
- .fn()
34
- .mockResolvedValue({ iamServiceApiUrl: mockIamServiceApiUrl }),
35
- });
36
-
37
- let GraphQLClientMock: jest.SpyInstance;
38
-
39
- beforeEach(() => {
40
- jest.resetAllMocks();
41
-
42
- // Re-apply RequestContextProvider mock after resetAllMocks
43
- (
44
- requestContextProvider.getContextValues as unknown as jest.Mock
45
- ).mockResolvedValue({
46
- iamServiceApiUrl: mockIamServiceApiUrl,
47
- });
48
-
49
- // Mock GraphQLClient constructor to return our mocked client
50
- GraphQLClientMock = jest
51
- .spyOn<any, any>(require('graphql-request'), 'GraphQLClient')
52
- .mockImplementation(() => gqlClient);
53
-
54
- service = new IAMGraphQlService(requestContextProvider as any);
55
- });
56
-
57
- it('should call mutation addUser', async () => {
58
- (gqlClient.request as jest.Mock).mockResolvedValue('');
59
-
60
- const response = await service.addUser('token', {} as any, {} as any);
61
-
62
- expect(GraphQLClientMock).toHaveBeenCalledWith(mockIamServiceApiUrl, {
63
- headers: { Authorization: 'Bearer token' },
64
- });
65
- expect(gqlClient.request).toHaveBeenCalledWith(MUTATION_LOGIN);
66
- expect(response).toBe(undefined);
67
- });
68
-
69
- it('should call mutation addUser and log error', async () => {
70
- console.error = jest.fn();
71
- (gqlClient.request as jest.Mock).mockRejectedValue('error');
72
-
73
- const response = await service.addUser('token', {} as any, {} as any);
74
- expect(response).toBe(undefined);
75
- expect(console.error).toHaveBeenCalledWith('error');
76
- });
77
- });
@@ -1,33 +0,0 @@
1
- import { PMRequestContextProvider } from '../pm-request-context-provider.js';
2
- import { MUTATION_LOGIN } from './queries.js';
3
- import { Injectable } from '@nestjs/common';
4
- import type { Request, Response } from 'express';
5
- import { GraphQLClient } from 'graphql-request';
6
-
7
- @Injectable()
8
- export class IAMGraphQlService {
9
- constructor(private requestContextProvider: PMRequestContextProvider) {}
10
-
11
- async addUser(
12
- token: string,
13
- request: Request,
14
- response: Response,
15
- ): Promise<void> {
16
- const requestContext = await this.requestContextProvider.getContextValues(
17
- request,
18
- response,
19
- );
20
- const iamUrl = requestContext.iamServiceApiUrl;
21
- const client = new GraphQLClient(iamUrl, {
22
- headers: {
23
- Authorization: `Bearer ${token}`,
24
- },
25
- });
26
-
27
- try {
28
- await client.request(MUTATION_LOGIN);
29
- } catch (e) {
30
- console.error(e);
31
- }
32
- }
33
- }