@c8y/ngx-components 1021.73.8 → 1021.74.1

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,6 +1,6 @@
1
1
  import { Component } from '@angular/core';
2
2
  import { ActivatedRoute } from '@angular/router';
3
- import { ActionBarItemComponent, AlertService, BuiltInActionType, C8yTranslateDirective, C8yTranslatePipe, DataGridModule, EmptyComponent, EmptyStateComponent, gettext, IconDirective, ModalService, Status, TitleComponent } from '@c8y/ngx-components';
3
+ import { ActionBarItemComponent, AlertService, BuiltInActionType, C8yTranslateDirective, C8yTranslatePipe, ContextRouteService, DataGridModule, EmptyComponent, EmptyStateComponent, gettext, IconDirective, ModalService, Status, TitleComponent } from '@c8y/ngx-components';
4
4
  import { RemoteAccessService } from '@c8y/ngx-components/remote-access/data';
5
5
  import { RemoteAccessConnectButtonComponent } from './remote-access-connect-button/remote-access-connect-button.component';
6
6
  import { BsModalService } from 'ngx-bootstrap/modal';
@@ -14,13 +14,14 @@ import * as i3 from "@c8y/ngx-components";
14
14
  import * as i4 from "ngx-bootstrap/modal";
15
15
  import * as i5 from "@ngx-translate/core";
16
16
  export class RemoteAccessConfigurationListComponent {
17
- constructor(remoteAccess, activatedRoute, modalService, alert, bsModalService, translateService) {
17
+ constructor(remoteAccess, activatedRoute, modalService, alert, bsModalService, translateService, ctx) {
18
18
  this.remoteAccess = remoteAccess;
19
19
  this.activatedRoute = activatedRoute;
20
20
  this.modalService = modalService;
21
21
  this.alert = alert;
22
22
  this.bsModalService = bsModalService;
23
23
  this.translateService = translateService;
24
+ this.ctx = ctx;
24
25
  this.rows = [];
25
26
  this.columns = [
26
27
  new IconDeviceGridColumn({
@@ -113,8 +114,9 @@ export class RemoteAccessConfigurationListComponent {
113
114
  };
114
115
  this.protocolProviders = [];
115
116
  this.gridTitle = gettext('Remote access endpoints');
116
- this.protocolProviders = this.remoteAccess.getProtocolProviders();
117
- this.deviceId = this.activatedRoute.parent.snapshot.params.id;
117
+ this.device = this.ctx.getContextData(this.activatedRoute)?.contextData;
118
+ this.deviceId = this.device?.id;
119
+ this.protocolProviders = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);
118
120
  const connectColum = this.columns.find(c => c.name === 'connect');
119
121
  if (connectColum) {
120
122
  connectColum.target = this.deviceId;
@@ -125,14 +127,17 @@ export class RemoteAccessConfigurationListComponent {
125
127
  }
126
128
  async reload() {
127
129
  const configurations = await this.remoteAccess.listConfigurations(this.deviceId);
128
- const supportedProtocols = this.protocolProviders.map(p => p.protocolName);
129
- this.rows = configurations.filter(c => supportedProtocols.includes(c.protocol));
130
+ const supportedProtocols = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);
131
+ const supportedProtocolsNames = supportedProtocols.map(p => p.protocolName);
132
+ this.rows = configurations.filter(c => supportedProtocolsNames.includes(c.protocol.toUpperCase()));
130
133
  }
131
134
  async addEndpoint() {
132
135
  let provider = null;
133
136
  if (this.protocolProviders.length > 1) {
134
137
  try {
135
- const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent);
138
+ const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent, {
139
+ initialState: { protocolProviders: this.protocolProviders }
140
+ });
136
141
  provider = await modalRef.content.result;
137
142
  }
138
143
  catch (e) {
@@ -200,7 +205,7 @@ export class RemoteAccessConfigurationListComponent {
200
205
  return null;
201
206
  }
202
207
  }
203
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.ActivatedRoute }, { token: i3.ModalService }, { token: i3.AlertService }, { token: i4.BsModalService }, { token: i5.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
208
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.ActivatedRoute }, { token: i3.ModalService }, { token: i3.AlertService }, { token: i4.BsModalService }, { token: i5.TranslateService }, { token: i3.ContextRouteService }], target: i0.ɵɵFactoryTarget.Component }); }
204
209
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessConfigurationListComponent, isStandalone: true, selector: "c8y-remote-access-configuration-list", ngImport: i0, template: "<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n (click)=\"addEndpoint()\"\n [disabled]=\"!protocolProviders?.length\"\n [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n >\n <i [c8yIcon]=\"'plus-circle'\"></i>\n <span translate>Add endpoint</span>\n </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n <c8y-data-grid\n [title]=\"gridTitle\"\n [rows]=\"rows\"\n [columns]=\"columns\"\n [pagination]=\"pagination\"\n (onReload)=\"reload()\"\n [actionControls]=\"actionControls\"\n [displayOptions]=\"displayOptions\"\n >\n <c8y-ui-empty-state\n [icon]=\"'window-restore'\"\n [title]=\"'No endpoints configured.' | translate\"\n [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n [horizontal]=\"false\"\n >\n <button\n class=\"btn btn-default\"\n (click)=\"addEndpoint()\"\n translate\n >\n Add endpoint\n </button>\n </c8y-ui-empty-state>\n </c8y-data-grid>\n</div>\n", dependencies: [{ kind: "ngmodule", type: DataGridModule }, { kind: "component", type: i3.DataGridComponent, selector: "c8y-data-grid", inputs: ["title", "loadMoreItemsLabel", "loadingItemsLabel", "showSearch", "refresh", "columns", "rows", "pagination", "infiniteScroll", "serverSideDataCallback", "selectable", "singleSelection", "selectionPrimaryKey", "displayOptions", "actionControls", "bulkActionControls", "headerActionControls", "searchText", "configureColumnsEnabled", "showCounterWarning", "activeClassName", "expandableRows", "hideReload"], outputs: ["rowMouseOver", "rowMouseLeave", "rowClick", "onConfigChange", "onBeforeFilter", "onBeforeSearch", "onFilter", "itemsSelect", "onReload", "onAddCustomColumn", "onRemoveCustomColumn", "onColumnFilterReset", "onSort", "onPageSizeChange", "onColumnReordered", "onColumnVisibilityChange"] }, { kind: "component", type: TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "component", type: ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "component", type: EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
205
210
  }
206
211
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, decorators: [{
@@ -214,5 +219,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
214
219
  EmptyStateComponent,
215
220
  C8yTranslatePipe
216
221
  ], template: "<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n (click)=\"addEndpoint()\"\n [disabled]=\"!protocolProviders?.length\"\n [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n >\n <i [c8yIcon]=\"'plus-circle'\"></i>\n <span translate>Add endpoint</span>\n </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n <c8y-data-grid\n [title]=\"gridTitle\"\n [rows]=\"rows\"\n [columns]=\"columns\"\n [pagination]=\"pagination\"\n (onReload)=\"reload()\"\n [actionControls]=\"actionControls\"\n [displayOptions]=\"displayOptions\"\n >\n <c8y-ui-empty-state\n [icon]=\"'window-restore'\"\n [title]=\"'No endpoints configured.' | translate\"\n [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n [horizontal]=\"false\"\n >\n <button\n class=\"btn btn-default\"\n (click)=\"addEndpoint()\"\n translate\n >\n Add endpoint\n </button>\n </c8y-ui-empty-state>\n </c8y-data-grid>\n</div>\n" }]
217
- }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.ActivatedRoute }, { type: i3.ModalService }, { type: i3.AlertService }, { type: i4.BsModalService }, { type: i5.TranslateService }] });
218
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"remote-access-configuration-list.component.js","sourceRoot":"","sources":["../../../../../remote-access/configurations/remote-access-configuration-list/remote-access-configuration-list.component.ts","../../../../../remote-access/configurations/remote-access-configuration-list/remote-access-configuration-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport {\n  ActionBarItemComponent,\n  ActionControl,\n  AlertService,\n  BuiltInActionType,\n  C8yTranslateDirective,\n  C8yTranslatePipe,\n  Column,\n  DataGridModule,\n  DisplayOptions,\n  EmptyComponent,\n  EmptyStateComponent,\n  gettext,\n  IconDirective,\n  ModalService,\n  Pagination,\n  Status,\n  TitleComponent\n} from '@c8y/ngx-components';\nimport {\n  RemoteAccessConfiguration,\n  RemoteAccessProtocolProvider,\n  RemoteAccessService\n} from '@c8y/ngx-components/remote-access/data';\nimport { RemoteAccessConnectButtonComponent } from './remote-access-connect-button/remote-access-connect-button.component';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { RemoteAccessPickProtocolModalComponent } from '../remote-access-pick-protocol-modal/remote-access-pick-protocol-modal.component';\nimport { IconDeviceGridColumn } from '@c8y/ngx-components/device-grid';\nimport { TranslateService } from '@ngx-translate/core';\n\n@Component({\n  selector: 'c8y-remote-access-configuration-list',\n  templateUrl: './remote-access-configuration-list.component.html',\n  standalone: true,\n  imports: [\n    DataGridModule,\n    TitleComponent,\n    C8yTranslateDirective,\n    ActionBarItemComponent,\n    IconDirective,\n    EmptyStateComponent,\n    C8yTranslatePipe\n  ]\n})\nexport class RemoteAccessConfigurationListComponent implements OnInit {\n  rows: RemoteAccessConfiguration[] = [];\n  columns: Column[] = [\n    new IconDeviceGridColumn(\n      {\n        headerCellRendererComponent: EmptyComponent\n      },\n      context => {\n        const provider = this.protocolProviders.find(p => p.protocolName === context.item.protocol);\n        return provider?.protocolIcon || 'laptop';\n      }\n    ),\n    {\n      name: 'name',\n      header: gettext('Endpoint'),\n      path: 'name'\n    },\n    {\n      name: 'hostname',\n      header: gettext('Host'),\n      path: 'hostname'\n    },\n    {\n      name: 'port',\n      header: gettext('Port'),\n      path: 'port'\n    },\n    {\n      name: 'protocol',\n      header: gettext('Protocol'),\n      path: 'protocol'\n    },\n    {\n      name: 'connect',\n      header: gettext('Connect'),\n      cellRendererComponent: RemoteAccessConnectButtonComponent\n    }\n  ];\n  actionControls: ActionControl[] = [\n    {\n      type: BuiltInActionType.Edit,\n      callback: async (entry: RemoteAccessConfiguration) => {\n        const provider = this.getProtocolProviderForConfiguration(entry);\n\n        try {\n          let result: RemoteAccessConfiguration;\n          if (provider.editEndpoint) {\n            result = await provider.editEndpoint(this.deviceId, entry);\n          } else {\n            result = await this.defaultEditEndpoint(this.deviceId, entry);\n          }\n\n          if (!result) {\n            return;\n          }\n\n          this.alert.success(gettext('Endpoint saved.'));\n        } catch (e) {\n          console.warn(e);\n          this.alert.danger(gettext('Failed to save endpoint.'), e);\n        }\n        this.reload();\n      }\n    },\n    {\n      type: BuiltInActionType.Delete,\n      callback: async (entry: RemoteAccessConfiguration) => {\n        const result = await this.modalService.confirm(\n          gettext('Delete endpoint'),\n          this.translateService.instant(\n            gettext(\n              `You are about to delete endpoint \"{{ endpointName }}\". Do you want to proceed?`\n            ),\n            { endpointName: entry.name }\n          ),\n          Status.DANGER,\n          { cancel: gettext('Cancel'), ok: gettext('Delete') }\n        );\n        if (!result) {\n          return;\n        }\n\n        const provider = this.getProtocolProviderForConfiguration(entry);\n\n        try {\n          await provider.removeEndpoint(this.deviceId, entry);\n          this.alert.success(gettext('Endpoint deleted.'));\n        } catch (e) {\n          console.warn(e);\n          this.alert.danger(gettext('Failed to delete endpoint.'), e);\n        }\n        this.reload();\n      }\n    }\n  ];\n  deviceId: string;\n  pagination: Pagination = {\n    pageSize: 10,\n    currentPage: 1\n  };\n  displayOptions: DisplayOptions = {\n    bordered: false,\n    striped: true,\n    filter: true,\n    gridHeader: true,\n    hover: true\n  };\n  protocolProviders: RemoteAccessProtocolProvider[] = [];\n  gridTitle = gettext('Remote access endpoints');\n\n  constructor(\n    private remoteAccess: RemoteAccessService,\n    private activatedRoute: ActivatedRoute,\n    private modalService: ModalService,\n    private alert: AlertService,\n    private bsModalService: BsModalService,\n    private translateService: TranslateService\n  ) {\n    this.protocolProviders = this.remoteAccess.getProtocolProviders();\n    this.deviceId = this.activatedRoute.parent.snapshot.params.id;\n    const connectColum = this.columns.find(c => c.name === 'connect');\n    if (connectColum) {\n      (<any>connectColum).target = this.deviceId;\n    }\n  }\n\n  ngOnInit() {\n    this.reload();\n  }\n\n  async reload() {\n    const configurations = await this.remoteAccess.listConfigurations(this.deviceId);\n    const supportedProtocols = this.protocolProviders.map(p => p.protocolName);\n    this.rows = configurations.filter(c => supportedProtocols.includes(c.protocol));\n  }\n\n  async addEndpoint() {\n    let provider: RemoteAccessProtocolProvider = null;\n    if (this.protocolProviders.length > 1) {\n      try {\n        const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent);\n        provider = await modalRef.content.result;\n      } catch (e) {\n        // modal closed\n        return;\n      }\n    } else if (this.protocolProviders.length === 1) {\n      provider = this.protocolProviders[0];\n    } else {\n      return;\n    }\n\n    try {\n      let addModalResult: RemoteAccessConfiguration = null;\n      if (provider.addEndpoint) {\n        addModalResult = await provider.addEndpoint(this.deviceId);\n      } else {\n        addModalResult = await this.defaultAddEndpoint(this.deviceId, provider);\n      }\n      if (!addModalResult) {\n        return;\n      }\n      this.alert.success(gettext('Endpoint added.'));\n    } catch (e) {\n      console.warn(e);\n      this.alert.danger(gettext('Failed to add endpoint.'), e);\n    }\n    this.reload();\n  }\n\n  getProtocolProviderForConfiguration(configuration: RemoteAccessConfiguration) {\n    return this.protocolProviders.find(p => p.protocolName === configuration.protocol);\n  }\n\n  async defaultEditEndpoint(deviceId: string, configuration: RemoteAccessConfiguration) {\n    const modalResult = await this.openBasicEndpointModal(configuration);\n    if (!modalResult) {\n      return modalResult;\n    }\n\n    return this.remoteAccess.updateConfiguration(deviceId, modalResult);\n  }\n\n  async defaultAddEndpoint(deviceId: string, provider: RemoteAccessProtocolProvider) {\n    let defaultConfig = { protocol: provider.protocolName };\n    if (provider.getDefaultEndpointConfiguration) {\n      const defaultConfigFromProvider = await provider.getDefaultEndpointConfiguration();\n      defaultConfig = { ...defaultConfig, ...defaultConfigFromProvider };\n    }\n    const modalResult = await this.openBasicEndpointModal(defaultConfig);\n    if (!modalResult) {\n      return modalResult;\n    }\n\n    return this.remoteAccess.addConfiguration(deviceId, modalResult);\n  }\n\n  async openBasicEndpointModal(\n    currentConfiguration: Partial<RemoteAccessConfiguration>\n  ): Promise<RemoteAccessConfiguration | null> {\n    const { RemoteAccessBasicEndpointModalComponent } = await import(\n      '@c8y/ngx-components/remote-access/shared'\n    );\n\n    try {\n      const modalRef = this.bsModalService.show(RemoteAccessBasicEndpointModalComponent, {\n        initialState: { currentValue: currentConfiguration }\n      });\n      return (await modalRef.content.result) as RemoteAccessConfiguration;\n    } catch (e) {\n      // modal closed\n      return null;\n    }\n  }\n}\n","<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n  <button\n    class=\"btn btn-link\"\n    (click)=\"addEndpoint()\"\n    [disabled]=\"!protocolProviders?.length\"\n    [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n  >\n    <i [c8yIcon]=\"'plus-circle'\"></i>\n    <span translate>Add endpoint</span>\n  </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n  <c8y-data-grid\n    [title]=\"gridTitle\"\n    [rows]=\"rows\"\n    [columns]=\"columns\"\n    [pagination]=\"pagination\"\n    (onReload)=\"reload()\"\n    [actionControls]=\"actionControls\"\n    [displayOptions]=\"displayOptions\"\n  >\n    <c8y-ui-empty-state\n      [icon]=\"'window-restore'\"\n      [title]=\"'No endpoints configured.' | translate\"\n      [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n      [horizontal]=\"false\"\n    >\n      <button\n        class=\"btn btn-default\"\n        (click)=\"addEndpoint()\"\n        translate\n      >\n        Add endpoint\n      </button>\n    </c8y-ui-empty-state>\n  </c8y-data-grid>\n</div>\n"]}
222
+ }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.ActivatedRoute }, { type: i3.ModalService }, { type: i3.AlertService }, { type: i4.BsModalService }, { type: i5.TranslateService }, { type: i3.ContextRouteService }] });
223
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"remote-access-configuration-list.component.js","sourceRoot":"","sources":["../../../../../remote-access/configurations/remote-access-configuration-list/remote-access-configuration-list.component.ts","../../../../../remote-access/configurations/remote-access-configuration-list/remote-access-configuration-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport {\n  ActionBarItemComponent,\n  ActionControl,\n  AlertService,\n  BuiltInActionType,\n  C8yTranslateDirective,\n  C8yTranslatePipe,\n  Column,\n  ContextRouteService,\n  DataGridModule,\n  DisplayOptions,\n  EmptyComponent,\n  EmptyStateComponent,\n  gettext,\n  IconDirective,\n  ModalService,\n  Pagination,\n  Status,\n  TitleComponent\n} from '@c8y/ngx-components';\nimport {\n  RemoteAccessConfiguration,\n  RemoteAccessProtocolProvider,\n  RemoteAccessService\n} from '@c8y/ngx-components/remote-access/data';\nimport { RemoteAccessConnectButtonComponent } from './remote-access-connect-button/remote-access-connect-button.component';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { RemoteAccessPickProtocolModalComponent } from '../remote-access-pick-protocol-modal/remote-access-pick-protocol-modal.component';\nimport { IconDeviceGridColumn } from '@c8y/ngx-components/device-grid';\nimport { TranslateService } from '@ngx-translate/core';\nimport { IManagedObject } from '@c8y/client';\n\n@Component({\n  selector: 'c8y-remote-access-configuration-list',\n  templateUrl: './remote-access-configuration-list.component.html',\n  standalone: true,\n  imports: [\n    DataGridModule,\n    TitleComponent,\n    C8yTranslateDirective,\n    ActionBarItemComponent,\n    IconDirective,\n    EmptyStateComponent,\n    C8yTranslatePipe\n  ]\n})\nexport class RemoteAccessConfigurationListComponent implements OnInit {\n  rows: RemoteAccessConfiguration[] = [];\n  columns: Column[] = [\n    new IconDeviceGridColumn(\n      {\n        headerCellRendererComponent: EmptyComponent\n      },\n      context => {\n        const provider = this.protocolProviders.find(p => p.protocolName === context.item.protocol);\n        return provider?.protocolIcon || 'laptop';\n      }\n    ),\n    {\n      name: 'name',\n      header: gettext('Endpoint'),\n      path: 'name'\n    },\n    {\n      name: 'hostname',\n      header: gettext('Host'),\n      path: 'hostname'\n    },\n    {\n      name: 'port',\n      header: gettext('Port'),\n      path: 'port'\n    },\n    {\n      name: 'protocol',\n      header: gettext('Protocol'),\n      path: 'protocol'\n    },\n    {\n      name: 'connect',\n      header: gettext('Connect'),\n      cellRendererComponent: RemoteAccessConnectButtonComponent\n    }\n  ];\n  actionControls: ActionControl[] = [\n    {\n      type: BuiltInActionType.Edit,\n      callback: async (entry: RemoteAccessConfiguration) => {\n        const provider = this.getProtocolProviderForConfiguration(entry);\n\n        try {\n          let result: RemoteAccessConfiguration;\n          if (provider.editEndpoint) {\n            result = await provider.editEndpoint(this.deviceId, entry);\n          } else {\n            result = await this.defaultEditEndpoint(this.deviceId, entry);\n          }\n\n          if (!result) {\n            return;\n          }\n\n          this.alert.success(gettext('Endpoint saved.'));\n        } catch (e) {\n          console.warn(e);\n          this.alert.danger(gettext('Failed to save endpoint.'), e);\n        }\n        this.reload();\n      }\n    },\n    {\n      type: BuiltInActionType.Delete,\n      callback: async (entry: RemoteAccessConfiguration) => {\n        const result = await this.modalService.confirm(\n          gettext('Delete endpoint'),\n          this.translateService.instant(\n            gettext(\n              `You are about to delete endpoint \"{{ endpointName }}\". Do you want to proceed?`\n            ),\n            { endpointName: entry.name }\n          ),\n          Status.DANGER,\n          { cancel: gettext('Cancel'), ok: gettext('Delete') }\n        );\n        if (!result) {\n          return;\n        }\n\n        const provider = this.getProtocolProviderForConfiguration(entry);\n\n        try {\n          await provider.removeEndpoint(this.deviceId, entry);\n          this.alert.success(gettext('Endpoint deleted.'));\n        } catch (e) {\n          console.warn(e);\n          this.alert.danger(gettext('Failed to delete endpoint.'), e);\n        }\n        this.reload();\n      }\n    }\n  ];\n  deviceId: string;\n  pagination: Pagination = {\n    pageSize: 10,\n    currentPage: 1\n  };\n  displayOptions: DisplayOptions = {\n    bordered: false,\n    striped: true,\n    filter: true,\n    gridHeader: true,\n    hover: true\n  };\n  protocolProviders: RemoteAccessProtocolProvider[] = [];\n  gridTitle = gettext('Remote access endpoints');\n\n  private device: IManagedObject;\n\n  constructor(\n    private remoteAccess: RemoteAccessService,\n    private activatedRoute: ActivatedRoute,\n    private modalService: ModalService,\n    private alert: AlertService,\n    private bsModalService: BsModalService,\n    private translateService: TranslateService,\n    private ctx: ContextRouteService\n  ) {\n    this.device = this.ctx.getContextData(this.activatedRoute)?.contextData as IManagedObject;\n    this.deviceId = this.device?.id as string;\n    this.protocolProviders = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);\n    const connectColum = this.columns.find(c => c.name === 'connect');\n    if (connectColum) {\n      (<any>connectColum).target = this.deviceId;\n    }\n  }\n\n  ngOnInit() {\n    this.reload();\n  }\n\n  async reload() {\n    const configurations = await this.remoteAccess.listConfigurations(this.deviceId);\n    const supportedProtocols = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);\n    const supportedProtocolsNames = supportedProtocols.map(p => p.protocolName);\n    this.rows = configurations.filter(c =>\n      supportedProtocolsNames.includes(c.protocol.toUpperCase())\n    );\n  }\n\n  async addEndpoint() {\n    let provider: RemoteAccessProtocolProvider = null;\n    if (this.protocolProviders.length > 1) {\n      try {\n        const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent, {\n          initialState: { protocolProviders: this.protocolProviders }\n        });\n        provider = await modalRef.content.result;\n      } catch (e) {\n        // modal closed\n        return;\n      }\n    } else if (this.protocolProviders.length === 1) {\n      provider = this.protocolProviders[0];\n    } else {\n      return;\n    }\n\n    try {\n      let addModalResult: RemoteAccessConfiguration = null;\n      if (provider.addEndpoint) {\n        addModalResult = await provider.addEndpoint(this.deviceId);\n      } else {\n        addModalResult = await this.defaultAddEndpoint(this.deviceId, provider);\n      }\n      if (!addModalResult) {\n        return;\n      }\n      this.alert.success(gettext('Endpoint added.'));\n    } catch (e) {\n      console.warn(e);\n      this.alert.danger(gettext('Failed to add endpoint.'), e);\n    }\n    this.reload();\n  }\n\n  getProtocolProviderForConfiguration(configuration: RemoteAccessConfiguration) {\n    return this.protocolProviders.find(p => p.protocolName === configuration.protocol);\n  }\n\n  async defaultEditEndpoint(deviceId: string, configuration: RemoteAccessConfiguration) {\n    const modalResult = await this.openBasicEndpointModal(configuration);\n    if (!modalResult) {\n      return modalResult;\n    }\n\n    return this.remoteAccess.updateConfiguration(deviceId, modalResult);\n  }\n\n  async defaultAddEndpoint(deviceId: string, provider: RemoteAccessProtocolProvider) {\n    let defaultConfig = { protocol: provider.protocolName };\n    if (provider.getDefaultEndpointConfiguration) {\n      const defaultConfigFromProvider = await provider.getDefaultEndpointConfiguration();\n      defaultConfig = { ...defaultConfig, ...defaultConfigFromProvider };\n    }\n    const modalResult = await this.openBasicEndpointModal(defaultConfig);\n    if (!modalResult) {\n      return modalResult;\n    }\n\n    return this.remoteAccess.addConfiguration(deviceId, modalResult);\n  }\n\n  async openBasicEndpointModal(\n    currentConfiguration: Partial<RemoteAccessConfiguration>\n  ): Promise<RemoteAccessConfiguration | null> {\n    const { RemoteAccessBasicEndpointModalComponent } = await import(\n      '@c8y/ngx-components/remote-access/shared'\n    );\n\n    try {\n      const modalRef = this.bsModalService.show(RemoteAccessBasicEndpointModalComponent, {\n        initialState: { currentValue: currentConfiguration }\n      });\n      return (await modalRef.content.result) as RemoteAccessConfiguration;\n    } catch (e) {\n      // modal closed\n      return null;\n    }\n  }\n}\n","<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n  <button\n    class=\"btn btn-link\"\n    (click)=\"addEndpoint()\"\n    [disabled]=\"!protocolProviders?.length\"\n    [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n  >\n    <i [c8yIcon]=\"'plus-circle'\"></i>\n    <span translate>Add endpoint</span>\n  </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n  <c8y-data-grid\n    [title]=\"gridTitle\"\n    [rows]=\"rows\"\n    [columns]=\"columns\"\n    [pagination]=\"pagination\"\n    (onReload)=\"reload()\"\n    [actionControls]=\"actionControls\"\n    [displayOptions]=\"displayOptions\"\n  >\n    <c8y-ui-empty-state\n      [icon]=\"'window-restore'\"\n      [title]=\"'No endpoints configured.' | translate\"\n      [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n      [horizontal]=\"false\"\n    >\n      <button\n        class=\"btn btn-default\"\n        (click)=\"addEndpoint()\"\n        translate\n      >\n        Add endpoint\n      </button>\n    </c8y-ui-empty-state>\n  </c8y-data-grid>\n</div>\n"]}
@@ -1,23 +1,20 @@
1
1
  import { Component } from '@angular/core';
2
2
  import { CoreModule, gettext } from '@c8y/ngx-components';
3
- import { RemoteAccessService } from '@c8y/ngx-components/remote-access/data';
4
3
  import { BsModalRef } from 'ngx-bootstrap/modal';
5
4
  import * as i0 from "@angular/core";
6
- import * as i1 from "@c8y/ngx-components/remote-access/data";
7
- import * as i2 from "ngx-bootstrap/modal";
8
- import * as i3 from "@c8y/ngx-components";
9
- import * as i4 from "@angular/common";
5
+ import * as i1 from "ngx-bootstrap/modal";
6
+ import * as i2 from "@c8y/ngx-components";
7
+ import * as i3 from "@angular/common";
10
8
  export class RemoteAccessPickProtocolModalComponent {
11
- constructor(remoteAccess, bsModalRef) {
12
- this.remoteAccess = remoteAccess;
9
+ constructor(bsModalRef) {
13
10
  this.bsModalRef = bsModalRef;
14
11
  this.title = gettext('Select a protocol');
15
12
  this.result = new Promise((resolve, reject) => {
16
13
  this._resolve = resolve;
17
14
  this._reject = reject;
18
15
  });
16
+ // set via initialState
19
17
  this.protocolProviders = [];
20
- this.protocolProviders = this.remoteAccess.getProtocolProviders();
21
18
  }
22
19
  cancel() {
23
20
  this._reject();
@@ -26,11 +23,11 @@ export class RemoteAccessPickProtocolModalComponent {
26
23
  this.bsModalRef.hide();
27
24
  this._resolve(protocol);
28
25
  }
29
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.BsModalRef }], target: i0.ɵɵFactoryTarget.Component }); }
30
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessPickProtocolModalComponent, isStandalone: true, selector: "c8y-remote-access-pick-protocol-modal", ngImport: i0, template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n", dependencies: [{ kind: "ngmodule", type: CoreModule }, { kind: "directive", type: i3.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "pipe", type: i3.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i3.ModalComponent, selector: "c8y-modal", inputs: ["disabled", "close", "dismiss", "title", "body", "customFooter", "headerClasses", "labels"], outputs: ["onDismiss", "onClose"] }, { kind: "component", type: i3.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i3.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i3.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }] }); }
26
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, deps: [{ token: i1.BsModalRef }], target: i0.ɵɵFactoryTarget.Component }); }
27
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessPickProtocolModalComponent, isStandalone: true, selector: "c8y-remote-access-pick-protocol-modal", ngImport: i0, template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n", dependencies: [{ kind: "ngmodule", type: CoreModule }, { kind: "directive", type: i2.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "pipe", type: i2.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i2.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.ModalComponent, selector: "c8y-modal", inputs: ["disabled", "close", "dismiss", "title", "body", "customFooter", "headerClasses", "labels"], outputs: ["onDismiss", "onClose"] }, { kind: "component", type: i2.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i2.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i2.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }] }); }
31
28
  }
32
29
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, decorators: [{
33
30
  type: Component,
34
31
  args: [{ selector: 'c8y-remote-access-pick-protocol-modal', standalone: true, imports: [CoreModule], template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n" }]
35
- }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.BsModalRef }] });
36
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVtb3RlLWFjY2Vzcy1waWNrLXByb3RvY29sLW1vZGFsLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3JlbW90ZS1hY2Nlc3MvY29uZmlndXJhdGlvbnMvcmVtb3RlLWFjY2Vzcy1waWNrLXByb3RvY29sLW1vZGFsL3JlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9yZW1vdGUtYWNjZXNzL2NvbmZpZ3VyYXRpb25zL3JlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbC9yZW1vdGUtYWNjZXNzLXBpY2stcHJvdG9jb2wtbW9kYWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMxQyxPQUFPLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQzFELE9BQU8sRUFFTCxtQkFBbUIsRUFDcEIsTUFBTSx3Q0FBd0MsQ0FBQztBQUNoRCxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0scUJBQXFCLENBQUM7Ozs7OztBQVFqRCxNQUFNLE9BQU8sc0NBQXNDO0lBWWpELFlBQ1UsWUFBaUMsRUFDakMsVUFBc0I7UUFEdEIsaUJBQVksR0FBWixZQUFZLENBQXFCO1FBQ2pDLGVBQVUsR0FBVixVQUFVLENBQVk7UUFiaEMsVUFBSyxHQUFHLE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO1FBQ3JDLFdBQU0sR0FBRyxJQUFJLE9BQU8sQ0FBK0IsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEVBQUU7WUFDckUsSUFBSSxDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUM7WUFDeEIsSUFBSSxDQUFDLE9BQU8sR0FBRyxNQUFNLENBQUM7UUFDeEIsQ0FBQyxDQUFDLENBQUM7UUFFSCxzQkFBaUIsR0FBbUMsRUFBRSxDQUFDO1FBU3JELElBQUksQ0FBQyxpQkFBaUIsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLG9CQUFvQixFQUFFLENBQUM7SUFDcEUsQ0FBQztJQUVELE1BQU07UUFDSixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVELGNBQWMsQ0FBQyxRQUFzQztRQUNuRCxJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ3ZCLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDMUIsQ0FBQzsrR0ExQlUsc0NBQXNDO21HQUF0QyxzQ0FBc0MsaUdDZG5ELHlrQ0FvQ0EsMkNEeEJZLFVBQVU7OzRGQUVULHNDQUFzQztrQkFObEQsU0FBUzsrQkFDRSx1Q0FBdUMsY0FFckMsSUFBSSxXQUNQLENBQUMsVUFBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb3JlTW9kdWxlLCBnZXR0ZXh0IH0gZnJvbSAnQGM4eS9uZ3gtY29tcG9uZW50cyc7XG5pbXBvcnQge1xuICBSZW1vdGVBY2Nlc3NQcm90b2NvbFByb3ZpZGVyLFxuICBSZW1vdGVBY2Nlc3NTZXJ2aWNlXG59IGZyb20gJ0BjOHkvbmd4LWNvbXBvbmVudHMvcmVtb3RlLWFjY2Vzcy9kYXRhJztcbmltcG9ydCB7IEJzTW9kYWxSZWYgfSBmcm9tICduZ3gtYm9vdHN0cmFwL21vZGFsJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnYzh5LXJlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9yZW1vdGUtYWNjZXNzLXBpY2stcHJvdG9jb2wtbW9kYWwuY29tcG9uZW50Lmh0bWwnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29yZU1vZHVsZV1cbn0pXG5leHBvcnQgY2xhc3MgUmVtb3RlQWNjZXNzUGlja1Byb3RvY29sTW9kYWxDb21wb25lbnQge1xuICB0aXRsZSA9IGdldHRleHQoJ1NlbGVjdCBhIHByb3RvY29sJyk7XG4gIHJlc3VsdCA9IG5ldyBQcm9taXNlPFJlbW90ZUFjY2Vzc1Byb3RvY29sUHJvdmlkZXI+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICB0aGlzLl9yZXNvbHZlID0gcmVzb2x2ZTtcbiAgICB0aGlzLl9yZWplY3QgPSByZWplY3Q7XG4gIH0pO1xuXG4gIHByb3RvY29sUHJvdmlkZXJzOiBSZW1vdGVBY2Nlc3NQcm90b2NvbFByb3ZpZGVyW10gPSBbXTtcblxuICBwcml2YXRlIF9yZXNvbHZlOiAodmFsdWU6IFJlbW90ZUFjY2Vzc1Byb3RvY29sUHJvdmlkZXIpID0+IHZvaWQ7XG4gIHByaXZhdGUgX3JlamVjdDogKHJlYXNvbj86IGFueSkgPT4gdm9pZDtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIHJlbW90ZUFjY2VzczogUmVtb3RlQWNjZXNzU2VydmljZSxcbiAgICBwcml2YXRlIGJzTW9kYWxSZWY6IEJzTW9kYWxSZWZcbiAgKSB7XG4gICAgdGhpcy5wcm90b2NvbFByb3ZpZGVycyA9IHRoaXMucmVtb3RlQWNjZXNzLmdldFByb3RvY29sUHJvdmlkZXJzKCk7XG4gIH1cblxuICBjYW5jZWwoKSB7XG4gICAgdGhpcy5fcmVqZWN0KCk7XG4gIH1cblxuICBzZWxlY3RQcm90b2NvbChwcm90b2NvbDogUmVtb3RlQWNjZXNzUHJvdG9jb2xQcm92aWRlcikge1xuICAgIHRoaXMuYnNNb2RhbFJlZi5oaWRlKCk7XG4gICAgdGhpcy5fcmVzb2x2ZShwcm90b2NvbCk7XG4gIH1cbn1cbiIsIjxjOHktbW9kYWxcbiAgW3RpdGxlXT1cInRpdGxlXCJcbiAgW2hlYWRlckNsYXNzZXNdPVwiJ2RpYWxvZy1oZWFkZXInXCJcbiAgKG9uRGlzbWlzcyk9XCJjYW5jZWwoKVwiXG4gIFtsYWJlbHNdPVwieyBjYW5jZWw6ICdDYW5jZWwnIH1cIlxuPlxuICA8bmctY29udGFpbmVyIGM4eS1tb2RhbC10aXRsZT5cbiAgICA8c3BhbiBbYzh5SWNvbl09XCInbGFwdG9wJ1wiPjwvc3Bhbj5cbiAgPC9uZy1jb250YWluZXI+XG4gIDxkaXYgY2xhc3M9XCJwLTI0XCI+XG4gICAgPGM4eS1saSAqbmdGb3I9XCJsZXQgcHJvdmlkZXIgb2YgcHJvdG9jb2xQcm92aWRlcnNcIj5cbiAgICAgIDxjOHktbGktaWNvbj5cbiAgICAgICAgPGkgW2M4eUljb25dPVwicHJvdmlkZXIucHJvdG9jb2xJY29uXCI+PC9pPlxuICAgICAgPC9jOHktbGktaWNvbj5cbiAgICAgIDxjOHktbGktYm9keT5cbiAgICAgICAgPGRpdiBjbGFzcz1cImQtZmxleCBqLWMtYmV0d2VlblwiPlxuICAgICAgICAgIDxkaXY+XG4gICAgICAgICAgICA8cD57eyBwcm92aWRlci5wcm90b2NvbExhYmVsIHwgdHJhbnNsYXRlIH19PC9wPlxuICAgICAgICAgICAgPHA+XG4gICAgICAgICAgICAgIDxzbWFsbD57eyBwcm92aWRlci5wcm90b2NvbERlc2NyaXB0aW9uIHwgdHJhbnNsYXRlIH19PC9zbWFsbD5cbiAgICAgICAgICAgIDwvcD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICBjbGFzcz1cImJ0biBidG4tcHJpbWFyeVwiXG4gICAgICAgICAgICAgIChjbGljayk9XCJzZWxlY3RQcm90b2NvbChwcm92aWRlcilcIlxuICAgICAgICAgICAgICBbYXR0ci5kYXRhLWN5XT1cIidyZW1vdGVBY2Nlc3NQaWNrUHJvdG9jb2xNb2RhbC0tc2VsZWN0LScgKyBwcm92aWRlci5wcm90b2NvbE5hbWVcIlxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICA8c3BhbiB0cmFuc2xhdGU+U2VsZWN0PC9zcGFuPlxuICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9jOHktbGktYm9keT5cbiAgICA8L2M4eS1saT5cbiAgPC9kaXY+XG48L2M4eS1tb2RhbD5cbiJdfQ==
32
+ }], ctorParameters: () => [{ type: i1.BsModalRef }] });
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVtb3RlLWFjY2Vzcy1waWNrLXByb3RvY29sLW1vZGFsLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3JlbW90ZS1hY2Nlc3MvY29uZmlndXJhdGlvbnMvcmVtb3RlLWFjY2Vzcy1waWNrLXByb3RvY29sLW1vZGFsL3JlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9yZW1vdGUtYWNjZXNzL2NvbmZpZ3VyYXRpb25zL3JlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbC9yZW1vdGUtYWNjZXNzLXBpY2stcHJvdG9jb2wtbW9kYWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMxQyxPQUFPLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBRTFELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQzs7Ozs7QUFRakQsTUFBTSxPQUFPLHNDQUFzQztJQWFqRCxZQUFvQixVQUFzQjtRQUF0QixlQUFVLEdBQVYsVUFBVSxDQUFZO1FBWjFDLFVBQUssR0FBRyxPQUFPLENBQUMsbUJBQW1CLENBQUMsQ0FBQztRQUNyQyxXQUFNLEdBQUcsSUFBSSxPQUFPLENBQStCLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxFQUFFO1lBQ3JFLElBQUksQ0FBQyxRQUFRLEdBQUcsT0FBTyxDQUFDO1lBQ3hCLElBQUksQ0FBQyxPQUFPLEdBQUcsTUFBTSxDQUFDO1FBQ3hCLENBQUMsQ0FBQyxDQUFDO1FBRUgsdUJBQXVCO1FBQ3ZCLHNCQUFpQixHQUFtQyxFQUFFLENBQUM7SUFLVixDQUFDO0lBRTlDLE1BQU07UUFDSixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVELGNBQWMsQ0FBQyxRQUFzQztRQUNuRCxJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ3ZCLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDMUIsQ0FBQzsrR0F0QlUsc0NBQXNDO21HQUF0QyxzQ0FBc0MsaUdDWG5ELHlrQ0FvQ0EsMkNEM0JZLFVBQVU7OzRGQUVULHNDQUFzQztrQkFObEQsU0FBUzsrQkFDRSx1Q0FBdUMsY0FFckMsSUFBSSxXQUNQLENBQUMsVUFBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb3JlTW9kdWxlLCBnZXR0ZXh0IH0gZnJvbSAnQGM4eS9uZ3gtY29tcG9uZW50cyc7XG5pbXBvcnQgeyBSZW1vdGVBY2Nlc3NQcm90b2NvbFByb3ZpZGVyIH0gZnJvbSAnQGM4eS9uZ3gtY29tcG9uZW50cy9yZW1vdGUtYWNjZXNzL2RhdGEnO1xuaW1wb3J0IHsgQnNNb2RhbFJlZiB9IGZyb20gJ25neC1ib290c3RyYXAvbW9kYWwnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjOHktcmVtb3RlLWFjY2Vzcy1waWNrLXByb3RvY29sLW1vZGFsJyxcbiAgdGVtcGxhdGVVcmw6ICcuL3JlbW90ZS1hY2Nlc3MtcGljay1wcm90b2NvbC1tb2RhbC5jb21wb25lbnQuaHRtbCcsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb3JlTW9kdWxlXVxufSlcbmV4cG9ydCBjbGFzcyBSZW1vdGVBY2Nlc3NQaWNrUHJvdG9jb2xNb2RhbENvbXBvbmVudCB7XG4gIHRpdGxlID0gZ2V0dGV4dCgnU2VsZWN0IGEgcHJvdG9jb2wnKTtcbiAgcmVzdWx0ID0gbmV3IFByb21pc2U8UmVtb3RlQWNjZXNzUHJvdG9jb2xQcm92aWRlcj4oKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIHRoaXMuX3Jlc29sdmUgPSByZXNvbHZlO1xuICAgIHRoaXMuX3JlamVjdCA9IHJlamVjdDtcbiAgfSk7XG5cbiAgLy8gc2V0IHZpYSBpbml0aWFsU3RhdGVcbiAgcHJvdG9jb2xQcm92aWRlcnM6IFJlbW90ZUFjY2Vzc1Byb3RvY29sUHJvdmlkZXJbXSA9IFtdO1xuXG4gIHByaXZhdGUgX3Jlc29sdmU6ICh2YWx1ZTogUmVtb3RlQWNjZXNzUHJvdG9jb2xQcm92aWRlcikgPT4gdm9pZDtcbiAgcHJpdmF0ZSBfcmVqZWN0OiAocmVhc29uPzogYW55KSA9PiB2b2lkO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgYnNNb2RhbFJlZjogQnNNb2RhbFJlZikge31cblxuICBjYW5jZWwoKSB7XG4gICAgdGhpcy5fcmVqZWN0KCk7XG4gIH1cblxuICBzZWxlY3RQcm90b2NvbChwcm90b2NvbDogUmVtb3RlQWNjZXNzUHJvdG9jb2xQcm92aWRlcikge1xuICAgIHRoaXMuYnNNb2RhbFJlZi5oaWRlKCk7XG4gICAgdGhpcy5fcmVzb2x2ZShwcm90b2NvbCk7XG4gIH1cbn1cbiIsIjxjOHktbW9kYWxcbiAgW3RpdGxlXT1cInRpdGxlXCJcbiAgW2hlYWRlckNsYXNzZXNdPVwiJ2RpYWxvZy1oZWFkZXInXCJcbiAgKG9uRGlzbWlzcyk9XCJjYW5jZWwoKVwiXG4gIFtsYWJlbHNdPVwieyBjYW5jZWw6ICdDYW5jZWwnIH1cIlxuPlxuICA8bmctY29udGFpbmVyIGM4eS1tb2RhbC10aXRsZT5cbiAgICA8c3BhbiBbYzh5SWNvbl09XCInbGFwdG9wJ1wiPjwvc3Bhbj5cbiAgPC9uZy1jb250YWluZXI+XG4gIDxkaXYgY2xhc3M9XCJwLTI0XCI+XG4gICAgPGM4eS1saSAqbmdGb3I9XCJsZXQgcHJvdmlkZXIgb2YgcHJvdG9jb2xQcm92aWRlcnNcIj5cbiAgICAgIDxjOHktbGktaWNvbj5cbiAgICAgICAgPGkgW2M4eUljb25dPVwicHJvdmlkZXIucHJvdG9jb2xJY29uXCI+PC9pPlxuICAgICAgPC9jOHktbGktaWNvbj5cbiAgICAgIDxjOHktbGktYm9keT5cbiAgICAgICAgPGRpdiBjbGFzcz1cImQtZmxleCBqLWMtYmV0d2VlblwiPlxuICAgICAgICAgIDxkaXY+XG4gICAgICAgICAgICA8cD57eyBwcm92aWRlci5wcm90b2NvbExhYmVsIHwgdHJhbnNsYXRlIH19PC9wPlxuICAgICAgICAgICAgPHA+XG4gICAgICAgICAgICAgIDxzbWFsbD57eyBwcm92aWRlci5wcm90b2NvbERlc2NyaXB0aW9uIHwgdHJhbnNsYXRlIH19PC9zbWFsbD5cbiAgICAgICAgICAgIDwvcD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgICBjbGFzcz1cImJ0biBidG4tcHJpbWFyeVwiXG4gICAgICAgICAgICAgIChjbGljayk9XCJzZWxlY3RQcm90b2NvbChwcm92aWRlcilcIlxuICAgICAgICAgICAgICBbYXR0ci5kYXRhLWN5XT1cIidyZW1vdGVBY2Nlc3NQaWNrUHJvdG9jb2xNb2RhbC0tc2VsZWN0LScgKyBwcm92aWRlci5wcm90b2NvbE5hbWVcIlxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICA8c3BhbiB0cmFuc2xhdGU+U2VsZWN0PC9zcGFuPlxuICAgICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9jOHktbGktYm9keT5cbiAgICA8L2M4eS1saT5cbiAgPC9kaXY+XG48L2M4eS1tb2RhbD5cbiJdfQ==
@@ -3,6 +3,7 @@ import { FetchClient } from '@c8y/client';
3
3
  import { ContextRouteService, Permissions, ServiceRegistry, ViewContext } from '@c8y/ngx-components';
4
4
  import { gettext } from '@c8y/ngx-components/gettext';
5
5
  import { defer, shareReplay } from 'rxjs';
6
+ import { uniqWith, intersectionWith, isEqual } from 'lodash';
6
7
  import * as i0 from "@angular/core";
7
8
  import * as i1 from "@c8y/client";
8
9
  import * as i2 from "@c8y/ngx-components";
@@ -124,6 +125,16 @@ export class RemoteAccessService {
124
125
  getProtocolProviders() {
125
126
  return this.serviceRegistry.get('remoteAccessProtocolHook');
126
127
  }
128
+ /**
129
+ * Retrieves all supported protocol providers for a given device.
130
+ * Based on the declarations in the fragment c8y_RemoteAccessSupportedProtocols of the device managed object.
131
+ */
132
+ getSupportedProtocolProvidersFor(device) {
133
+ const { c8y_RemoteAccessSupportedProtocols = [] } = device;
134
+ const uniqueInput = uniqWith(c8y_RemoteAccessSupportedProtocols.map(p => p.toUpperCase()), isEqual);
135
+ const supportedProviders = intersectionWith(this.getProtocolProviders(), uniqueInput, ({ protocolName }, protocol) => protocolName?.toUpperCase() === protocol);
136
+ return supportedProviders?.length > 0 ? supportedProviders : this.getProtocolProviders();
137
+ }
127
138
  /**
128
139
  * Creates a new configuration for a given device.
129
140
  */
@@ -197,4 +208,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
197
208
  providedIn: 'root'
198
209
  }]
199
210
  }], ctorParameters: () => [{ type: i1.FetchClient }, { type: i2.ServiceRegistry }] });
200
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"remote-access.service.js","sourceRoot":"","sources":["../../../../remote-access/data/remote-access.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivateFn } from '@angular/router';\nimport { FetchClient, IManagedObject } from '@c8y/client';\nimport {\n  ContextRouteService,\n  Permissions,\n  ServiceRegistry,\n  ViewContext\n} from '@c8y/ngx-components';\nimport { gettext } from '@c8y/ngx-components/gettext';\nimport { Observable, defer, shareReplay } from 'rxjs';\n\nexport interface RemoteAccessConfiguration {\n  id: string;\n  name: string;\n  hostname: string;\n  port: number;\n  protocol: string;\n  attrs?: any;\n  credentials?: any;\n  credentialsType?: string;\n}\n\nexport const CREDENTIALS_TYPES = {\n  NONE: {\n    name: 'NONE',\n    value: 'NONE',\n    label: gettext('No password')\n  },\n  USER_PASS: {\n    name: 'USER_PASS',\n    value: 'USER_PASS',\n    label: gettext('Username and password')\n  },\n  PASS_ONLY: {\n    name: 'PASS_ONLY',\n    value: 'PASS_ONLY',\n    label: gettext('Password only')\n  },\n  KEY_PAIR: {\n    name: 'KEY_PAIR',\n    value: 'KEY_PAIR',\n    label: gettext('Public/private keys')\n  },\n  CERTIFICATE: {\n    name: 'CERTIFICATE',\n    value: 'CERTIFICATE',\n    label: gettext('Certificate')\n  }\n} as const;\n\nexport const canActivateRemoteAccess: CanActivateFn = (route: ActivatedRouteSnapshot) => {\n  const permissions = inject(Permissions);\n  const remoteAccessService = inject(RemoteAccessService);\n  const contextRouteService = inject(ContextRouteService);\n  if (!permissions.hasRole(Permissions.ROLE_REMOTE_ACCESS_ADMIN)) {\n    return false;\n  }\n\n  const contextDetails = contextRouteService.getContextData(route);\n  if (contextDetails.context !== ViewContext.Device) {\n    return false;\n  }\n\n  const device = contextDetails.contextData as IManagedObject;\n  if (!device || !Array.isArray(device.c8y_SupportedOperations)) {\n    return false;\n  }\n  const supportedOperations: string[] = device.c8y_SupportedOperations;\n  if (!supportedOperations.includes('c8y_RemoteAccessConnect')) {\n    return false;\n  }\n  return remoteAccessService.isAvailable$();\n};\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class RemoteAccessService {\n  private cachedIsAvailable$: Observable<boolean>;\n  readonly baseUrl = '/service/remoteaccess';\n  constructor(\n    private fetchClient: FetchClient,\n    private serviceRegistry: ServiceRegistry\n  ) {}\n\n  /**\n   * Verifies if the remote access service is available by sending a HEAD request to is's health endpoint.\n   * @returns cached Observable that emits true if the service is available, false otherwise.\n   */\n  isAvailable$(): Observable<boolean> {\n    if (!this.cachedIsAvailable$) {\n      this.cachedIsAvailable$ = defer(() => this.healthEndpointAvailable()).pipe(shareReplay(1));\n    }\n\n    return this.cachedIsAvailable$;\n  }\n\n  /**\n   * misses the leading ? for the query params\n   */\n  getAuthQueryParamsForWebsocketConnection() {\n    const { headers } = this.fetchClient.getFetchOptions();\n    const params = new URLSearchParams();\n\n    if (headers) {\n      const xsrfToken = headers['X-XSRF-TOKEN'];\n      const auth = headers['Authorization'];\n\n      if (xsrfToken) {\n        params.append('XSRF-TOKEN', xsrfToken);\n      }\n      if (auth) {\n        params.append('token', auth.replace('Bearer ', '').replace('Basic ', ''));\n      }\n    }\n\n    const paramsString = params.toString();\n    return paramsString;\n  }\n\n  /**\n   * Returns the URI for the websocket connection to the remote access service.\n   */\n  getWebSocketUri<K extends string, I extends string>(deviceId: K, configurationId: I) {\n    const authQueryParams = this.getAuthQueryParamsForWebsocketConnection();\n    const protocol = window.location.protocol === 'http:' ? 'ws' : 'wss';\n    const pathName =\n      `${protocol}://${window.location.host}${this.baseUrl}/client/${deviceId}/configurations/${configurationId}` as const;\n    return authQueryParams ? (`${pathName}?${authQueryParams}` as const) : pathName;\n  }\n\n  /**\n   * Retrieves all configurations for a given device.\n   */\n  async listConfigurations(deviceId: string): Promise<RemoteAccessConfiguration[]> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations`\n    );\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to fetch configurations for device ${deviceId}`);\n  }\n\n  /**\n   * Deletes a configuration for a given device.\n   */\n  async deleteConfiguration(deviceId: string, configurationId: string) {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations/${configurationId}`,\n      { method: 'DELETE' }\n    );\n\n    if (response.ok) {\n      return;\n    }\n\n    throw new Error(`Failed to delete configuration for device ${deviceId}`);\n  }\n\n  /**\n   * Retrieves all available remote access protocol providers.\n   */\n  getProtocolProviders() {\n    return this.serviceRegistry.get('remoteAccessProtocolHook');\n  }\n\n  /**\n   * Creates a new configuration for a given device.\n   */\n  async addConfiguration(\n    deviceId: string,\n    configuration: Omit<RemoteAccessConfiguration, 'id'>\n  ): Promise<RemoteAccessConfiguration> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations`,\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json'\n        },\n        body: JSON.stringify(configuration)\n      }\n    );\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to add configuration for device ${configuration.attrs.deviceId}`);\n  }\n\n  /**\n   * Updates a configuration for a given device.\n   */\n  async updateConfiguration(\n    deviceId: string,\n    configuration: RemoteAccessConfiguration\n  ): Promise<RemoteAccessConfiguration> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations/${configuration.id}`,\n      {\n        method: 'PUT',\n        headers: {\n          'Content-Type': 'application/json'\n        },\n        body: JSON.stringify(configuration)\n      }\n    );\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to update configuration for device ${configuration.attrs.deviceId}`);\n  }\n\n  /**\n   * Generates a SSH key pair for a given hostname.\n   */\n  async generateKeyPair(hostname: string): Promise<{ publicKey: string; privateKey: string }> {\n    const response = await this.fetchClient.fetch(`${this.baseUrl}/keypair/generate`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json'\n      },\n      body: JSON.stringify({ hostname })\n    });\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to generate key pair for ${hostname}`);\n  }\n\n  private async healthEndpointAvailable(): Promise<boolean> {\n    try {\n      const response = await this.fetchClient.fetch(`${this.baseUrl}/health`, {\n        method: 'HEAD',\n        headers: {\n          'Content-Type': 'application/json'\n        }\n      });\n\n      if (response.ok) {\n        return !!response.ok;\n      }\n    } catch (e) {\n      return false;\n    }\n  }\n}\n"]}
211
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"remote-access.service.js","sourceRoot":"","sources":["../../../../remote-access/data/remote-access.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEnD,OAAO,EAAE,WAAW,EAAkB,MAAM,aAAa,CAAC;AAC1D,OAAO,EACL,mBAAmB,EACnB,WAAW,EACX,eAAe,EACf,WAAW,EACZ,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EAAE,OAAO,EAAE,MAAM,6BAA6B,CAAC;AACtD,OAAO,EAAc,KAAK,EAAE,WAAW,EAAE,MAAM,MAAM,CAAC;AAEtD,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,OAAO,EAAE,MAAM,QAAQ,CAAC;;;;AAa7D,MAAM,CAAC,MAAM,iBAAiB,GAAG;IAC/B,IAAI,EAAE;QACJ,IAAI,EAAE,MAAM;QACZ,KAAK,EAAE,MAAM;QACb,KAAK,EAAE,OAAO,CAAC,aAAa,CAAC;KAC9B;IACD,SAAS,EAAE;QACT,IAAI,EAAE,WAAW;QACjB,KAAK,EAAE,WAAW;QAClB,KAAK,EAAE,OAAO,CAAC,uBAAuB,CAAC;KACxC;IACD,SAAS,EAAE;QACT,IAAI,EAAE,WAAW;QACjB,KAAK,EAAE,WAAW;QAClB,KAAK,EAAE,OAAO,CAAC,eAAe,CAAC;KAChC;IACD,QAAQ,EAAE;QACR,IAAI,EAAE,UAAU;QAChB,KAAK,EAAE,UAAU;QACjB,KAAK,EAAE,OAAO,CAAC,qBAAqB,CAAC;KACtC;IACD,WAAW,EAAE;QACX,IAAI,EAAE,aAAa;QACnB,KAAK,EAAE,aAAa;QACpB,KAAK,EAAE,OAAO,CAAC,aAAa,CAAC;KAC9B;CACO,CAAC;AAEX,MAAM,CAAC,MAAM,uBAAuB,GAAkB,CAAC,KAA6B,EAAE,EAAE;IACtF,MAAM,WAAW,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;IACxC,MAAM,mBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACxD,MAAM,mBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACxD,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,WAAW,CAAC,wBAAwB,CAAC,EAAE,CAAC;QAC/D,OAAO,KAAK,CAAC;IACf,CAAC;IAED,MAAM,cAAc,GAAG,mBAAmB,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;IACjE,IAAI,cAAc,CAAC,OAAO,KAAK,WAAW,CAAC,MAAM,EAAE,CAAC;QAClD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,MAAM,MAAM,GAAG,cAAc,CAAC,WAA6B,CAAC;IAC5D,IAAI,CAAC,MAAM,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,uBAAuB,CAAC,EAAE,CAAC;QAC9D,OAAO,KAAK,CAAC;IACf,CAAC;IACD,MAAM,mBAAmB,GAAa,MAAM,CAAC,uBAAuB,CAAC;IACrE,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,yBAAyB,CAAC,EAAE,CAAC;QAC7D,OAAO,KAAK,CAAC;IACf,CAAC;IACD,OAAO,mBAAmB,CAAC,YAAY,EAAE,CAAC;AAC5C,CAAC,CAAC;AAKF,MAAM,OAAO,mBAAmB;IAG9B,YACU,WAAwB,EACxB,eAAgC;QADhC,gBAAW,GAAX,WAAW,CAAa;QACxB,oBAAe,GAAf,eAAe,CAAiB;QAHjC,YAAO,GAAG,uBAAuB,CAAC;IAIxC,CAAC;IAEJ;;;OAGG;IACH,YAAY;QACV,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC7B,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,uBAAuB,EAAE,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7F,CAAC;QAED,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,wCAAwC;QACtC,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,eAAe,EAAE,CAAC;QACvD,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QAErC,IAAI,OAAO,EAAE,CAAC;YACZ,MAAM,SAAS,GAAG,OAAO,CAAC,cAAc,CAAC,CAAC;YAC1C,MAAM,IAAI,GAAG,OAAO,CAAC,eAAe,CAAC,CAAC;YAEtC,IAAI,SAAS,EAAE,CAAC;gBACd,MAAM,CAAC,MAAM,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC;YACzC,CAAC;YACD,IAAI,IAAI,EAAE,CAAC;gBACT,MAAM,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;YAC5E,CAAC;QACH,CAAC;QAED,MAAM,YAAY,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;QACvC,OAAO,YAAY,CAAC;IACtB,CAAC;IAED;;OAEG;IACH,eAAe,CAAqC,QAAW,EAAE,eAAkB;QACjF,MAAM,eAAe,GAAG,IAAI,CAAC,wCAAwC,EAAE,CAAC;QACxE,MAAM,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC;QACrE,MAAM,QAAQ,GACZ,GAAG,QAAQ,MAAM,MAAM,CAAC,QAAQ,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,WAAW,QAAQ,mBAAmB,eAAe,EAAW,CAAC;QACvH,OAAO,eAAe,CAAC,CAAC,CAAE,GAAG,QAAQ,IAAI,eAAe,EAAY,CAAC,CAAC,CAAC,QAAQ,CAAC;IAClF,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,kBAAkB,CAAC,QAAgB;QACvC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAC3C,GAAG,IAAI,CAAC,OAAO,YAAY,QAAQ,iBAAiB,CACrD,CAAC;QACF,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;QACzB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,6CAA6C,QAAQ,EAAE,CAAC,CAAC;IAC3E,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,mBAAmB,CAAC,QAAgB,EAAE,eAAuB;QACjE,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAC3C,GAAG,IAAI,CAAC,OAAO,YAAY,QAAQ,mBAAmB,eAAe,EAAE,EACvE,EAAE,MAAM,EAAE,QAAQ,EAAE,CACrB,CAAC;QAEF,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,OAAO;QACT,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,6CAA6C,QAAQ,EAAE,CAAC,CAAC;IAC3E,CAAC;IAED;;OAEG;IACH,oBAAoB;QAClB,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;IAC9D,CAAC;IAED;;;OAGG;IACH,gCAAgC,CAAC,MAAsB;QACrD,MAAM,EAAE,kCAAkC,GAAG,EAAE,EAAE,GAAG,MAAM,CAAC;QAC3D,MAAM,WAAW,GAAG,QAAQ,CAC1B,kCAAkC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,EAC5D,OAAO,CACR,CAAC;QACF,MAAM,kBAAkB,GAAG,gBAAgB,CACzC,IAAI,CAAC,oBAAoB,EAAE,EAC3B,WAAW,EACX,CAAC,EAAE,YAAY,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,YAAY,EAAE,WAAW,EAAE,KAAK,QAAQ,CACzE,CAAC;QACF,OAAO,kBAAkB,EAAE,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAC3F,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,gBAAgB,CACpB,QAAgB,EAChB,aAAoD;QAEpD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAC3C,GAAG,IAAI,CAAC,OAAO,YAAY,QAAQ,iBAAiB,EACpD;YACE,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;aACnC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC;SACpC,CACF,CAAC;QAEF,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;QACzB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,0CAA0C,aAAa,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC5F,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,mBAAmB,CACvB,QAAgB,EAChB,aAAwC;QAExC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAC3C,GAAG,IAAI,CAAC,OAAO,YAAY,QAAQ,mBAAmB,aAAa,CAAC,EAAE,EAAE,EACxE;YACE,MAAM,EAAE,KAAK;YACb,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;aACnC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC;SACpC,CACF,CAAC;QAEF,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;QACzB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,6CAA6C,aAAa,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC/F,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,eAAe,CAAC,QAAgB;QACpC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,OAAO,mBAAmB,EAAE;YAChF,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;aACnC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC;SACnC,CAAC,CAAC;QAEH,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;QACzB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,mCAAmC,QAAQ,EAAE,CAAC,CAAC;IACjE,CAAC;IAEO,KAAK,CAAC,uBAAuB;QACnC,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,OAAO,SAAS,EAAE;gBACtE,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE;oBACP,cAAc,EAAE,kBAAkB;iBACnC;aACF,CAAC,CAAC;YAEH,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;gBAChB,OAAO,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC;YACvB,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;+GAjMU,mBAAmB;mHAAnB,mBAAmB,cAFlB,MAAM;;4FAEP,mBAAmB;kBAH/B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivateFn } from '@angular/router';\nimport { FetchClient, IManagedObject } from '@c8y/client';\nimport {\n  ContextRouteService,\n  Permissions,\n  ServiceRegistry,\n  ViewContext\n} from '@c8y/ngx-components';\nimport { gettext } from '@c8y/ngx-components/gettext';\nimport { Observable, defer, shareReplay } from 'rxjs';\nimport { RemoteAccessProtocolProvider } from './remote-access-protocol-provider';\nimport { uniqWith, intersectionWith, isEqual } from 'lodash';\n\nexport interface RemoteAccessConfiguration {\n  id: string;\n  name: string;\n  hostname: string;\n  port: number;\n  protocol: string;\n  attrs?: any;\n  credentials?: any;\n  credentialsType?: string;\n}\n\nexport const CREDENTIALS_TYPES = {\n  NONE: {\n    name: 'NONE',\n    value: 'NONE',\n    label: gettext('No password')\n  },\n  USER_PASS: {\n    name: 'USER_PASS',\n    value: 'USER_PASS',\n    label: gettext('Username and password')\n  },\n  PASS_ONLY: {\n    name: 'PASS_ONLY',\n    value: 'PASS_ONLY',\n    label: gettext('Password only')\n  },\n  KEY_PAIR: {\n    name: 'KEY_PAIR',\n    value: 'KEY_PAIR',\n    label: gettext('Public/private keys')\n  },\n  CERTIFICATE: {\n    name: 'CERTIFICATE',\n    value: 'CERTIFICATE',\n    label: gettext('Certificate')\n  }\n} as const;\n\nexport const canActivateRemoteAccess: CanActivateFn = (route: ActivatedRouteSnapshot) => {\n  const permissions = inject(Permissions);\n  const remoteAccessService = inject(RemoteAccessService);\n  const contextRouteService = inject(ContextRouteService);\n  if (!permissions.hasRole(Permissions.ROLE_REMOTE_ACCESS_ADMIN)) {\n    return false;\n  }\n\n  const contextDetails = contextRouteService.getContextData(route);\n  if (contextDetails.context !== ViewContext.Device) {\n    return false;\n  }\n\n  const device = contextDetails.contextData as IManagedObject;\n  if (!device || !Array.isArray(device.c8y_SupportedOperations)) {\n    return false;\n  }\n  const supportedOperations: string[] = device.c8y_SupportedOperations;\n  if (!supportedOperations.includes('c8y_RemoteAccessConnect')) {\n    return false;\n  }\n  return remoteAccessService.isAvailable$();\n};\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class RemoteAccessService {\n  private cachedIsAvailable$: Observable<boolean>;\n  readonly baseUrl = '/service/remoteaccess';\n  constructor(\n    private fetchClient: FetchClient,\n    private serviceRegistry: ServiceRegistry\n  ) {}\n\n  /**\n   * Verifies if the remote access service is available by sending a HEAD request to is's health endpoint.\n   * @returns cached Observable that emits true if the service is available, false otherwise.\n   */\n  isAvailable$(): Observable<boolean> {\n    if (!this.cachedIsAvailable$) {\n      this.cachedIsAvailable$ = defer(() => this.healthEndpointAvailable()).pipe(shareReplay(1));\n    }\n\n    return this.cachedIsAvailable$;\n  }\n\n  /**\n   * misses the leading ? for the query params\n   */\n  getAuthQueryParamsForWebsocketConnection() {\n    const { headers } = this.fetchClient.getFetchOptions();\n    const params = new URLSearchParams();\n\n    if (headers) {\n      const xsrfToken = headers['X-XSRF-TOKEN'];\n      const auth = headers['Authorization'];\n\n      if (xsrfToken) {\n        params.append('XSRF-TOKEN', xsrfToken);\n      }\n      if (auth) {\n        params.append('token', auth.replace('Bearer ', '').replace('Basic ', ''));\n      }\n    }\n\n    const paramsString = params.toString();\n    return paramsString;\n  }\n\n  /**\n   * Returns the URI for the websocket connection to the remote access service.\n   */\n  getWebSocketUri<K extends string, I extends string>(deviceId: K, configurationId: I) {\n    const authQueryParams = this.getAuthQueryParamsForWebsocketConnection();\n    const protocol = window.location.protocol === 'http:' ? 'ws' : 'wss';\n    const pathName =\n      `${protocol}://${window.location.host}${this.baseUrl}/client/${deviceId}/configurations/${configurationId}` as const;\n    return authQueryParams ? (`${pathName}?${authQueryParams}` as const) : pathName;\n  }\n\n  /**\n   * Retrieves all configurations for a given device.\n   */\n  async listConfigurations(deviceId: string): Promise<RemoteAccessConfiguration[]> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations`\n    );\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to fetch configurations for device ${deviceId}`);\n  }\n\n  /**\n   * Deletes a configuration for a given device.\n   */\n  async deleteConfiguration(deviceId: string, configurationId: string) {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations/${configurationId}`,\n      { method: 'DELETE' }\n    );\n\n    if (response.ok) {\n      return;\n    }\n\n    throw new Error(`Failed to delete configuration for device ${deviceId}`);\n  }\n\n  /**\n   * Retrieves all available remote access protocol providers.\n   */\n  getProtocolProviders() {\n    return this.serviceRegistry.get('remoteAccessProtocolHook');\n  }\n\n  /**\n   * Retrieves all supported protocol providers for a given device.\n   * Based on the declarations in the fragment c8y_RemoteAccessSupportedProtocols of the device managed object.\n   */\n  getSupportedProtocolProvidersFor(device: IManagedObject): RemoteAccessProtocolProvider[] {\n    const { c8y_RemoteAccessSupportedProtocols = [] } = device;\n    const uniqueInput = uniqWith(\n      c8y_RemoteAccessSupportedProtocols.map(p => p.toUpperCase()),\n      isEqual\n    );\n    const supportedProviders = intersectionWith(\n      this.getProtocolProviders(),\n      uniqueInput,\n      ({ protocolName }, protocol) => protocolName?.toUpperCase() === protocol\n    );\n    return supportedProviders?.length > 0 ? supportedProviders : this.getProtocolProviders();\n  }\n\n  /**\n   * Creates a new configuration for a given device.\n   */\n  async addConfiguration(\n    deviceId: string,\n    configuration: Omit<RemoteAccessConfiguration, 'id'>\n  ): Promise<RemoteAccessConfiguration> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations`,\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json'\n        },\n        body: JSON.stringify(configuration)\n      }\n    );\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to add configuration for device ${configuration.attrs.deviceId}`);\n  }\n\n  /**\n   * Updates a configuration for a given device.\n   */\n  async updateConfiguration(\n    deviceId: string,\n    configuration: RemoteAccessConfiguration\n  ): Promise<RemoteAccessConfiguration> {\n    const response = await this.fetchClient.fetch(\n      `${this.baseUrl}/devices/${deviceId}/configurations/${configuration.id}`,\n      {\n        method: 'PUT',\n        headers: {\n          'Content-Type': 'application/json'\n        },\n        body: JSON.stringify(configuration)\n      }\n    );\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to update configuration for device ${configuration.attrs.deviceId}`);\n  }\n\n  /**\n   * Generates a SSH key pair for a given hostname.\n   */\n  async generateKeyPair(hostname: string): Promise<{ publicKey: string; privateKey: string }> {\n    const response = await this.fetchClient.fetch(`${this.baseUrl}/keypair/generate`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json'\n      },\n      body: JSON.stringify({ hostname })\n    });\n\n    if (response.ok) {\n      return response.json();\n    }\n\n    throw new Error(`Failed to generate key pair for ${hostname}`);\n  }\n\n  private async healthEndpointAvailable(): Promise<boolean> {\n    try {\n      const response = await this.fetchClient.fetch(`${this.baseUrl}/health`, {\n        method: 'HEAD',\n        headers: {\n          'Content-Type': 'application/json'\n        }\n      });\n\n      if (response.ok) {\n        return !!response.ok;\n      }\n    } catch (e) {\n      return false;\n    }\n  }\n}\n"]}
@@ -1,4 +1,4 @@
1
- import * as i3 from '@c8y/ngx-components';
1
+ import * as i2 from '@c8y/ngx-components';
2
2
  import { IconDirective, C8yTranslatePipe, gettext, CoreModule, EmptyComponent, BuiltInActionType, Status, DataGridModule, TitleComponent, C8yTranslateDirective, ActionBarItemComponent, EmptyStateComponent, hookRoute, ViewContext } from '@c8y/ngx-components';
3
3
  import { gettext as gettext$1 } from '@c8y/ngx-components/gettext';
4
4
  import * as i0 from '@angular/core';
@@ -7,9 +7,9 @@ import * as i2$1 from '@angular/router';
7
7
  import { RouterLinkWithHref } from '@angular/router';
8
8
  import * as i1 from '@c8y/ngx-components/remote-access/data';
9
9
  import { canActivateRemoteAccess } from '@c8y/ngx-components/remote-access/data';
10
- import * as i4 from '@angular/common';
10
+ import * as i3 from '@angular/common';
11
11
  import { NgIf } from '@angular/common';
12
- import * as i2 from 'ngx-bootstrap/modal';
12
+ import * as i1$1 from 'ngx-bootstrap/modal';
13
13
  import { IconDeviceGridColumn } from '@c8y/ngx-components/device-grid';
14
14
  import * as i5 from '@ngx-translate/core';
15
15
 
@@ -55,27 +55,26 @@ class RemoteAccessConnectButtonComponent {
55
55
  }, { allowSignalWrites: true });
56
56
  return resultSignal.asReadonly();
57
57
  }
58
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConnectButtonComponent, deps: [{ token: i1.RemoteAccessService }, { token: i3.CellRendererContext, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
58
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConnectButtonComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.CellRendererContext, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
59
59
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: RemoteAccessConnectButtonComponent, isStandalone: true, selector: "c8y-remote-access-connect-button", inputs: { deviceId: { classPropertyName: "deviceId", publicName: "deviceId", isSignal: true, isRequired: false, transformFunction: null }, configurationId: { classPropertyName: "configurationId", publicName: "configurationId", isSignal: true, isRequired: false, transformFunction: null }, protocol: { classPropertyName: "protocol", publicName: "protocol", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<a\n *ngIf=\"connectHref() as endpointHref\"\n [title]=\"'Connect' | translate\"\n [routerLink]=\"endpointHref\"\n [attr.data-cy]=\"'remoteAccessConnectButton'\"\n>\n <i [c8yIcon]=\"'connected'\"></i>\n</a>\n", dependencies: [{ kind: "directive", type: RouterLinkWithHref, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
60
60
  }
61
61
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConnectButtonComponent, decorators: [{
62
62
  type: Component,
63
63
  args: [{ selector: 'c8y-remote-access-connect-button', standalone: true, imports: [RouterLinkWithHref, IconDirective, NgIf, C8yTranslatePipe], template: "<a\n *ngIf=\"connectHref() as endpointHref\"\n [title]=\"'Connect' | translate\"\n [routerLink]=\"endpointHref\"\n [attr.data-cy]=\"'remoteAccessConnectButton'\"\n>\n <i [c8yIcon]=\"'connected'\"></i>\n</a>\n" }]
64
- }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i3.CellRendererContext, decorators: [{
64
+ }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.CellRendererContext, decorators: [{
65
65
  type: Optional
66
66
  }] }] });
67
67
 
68
68
  class RemoteAccessPickProtocolModalComponent {
69
- constructor(remoteAccess, bsModalRef) {
70
- this.remoteAccess = remoteAccess;
69
+ constructor(bsModalRef) {
71
70
  this.bsModalRef = bsModalRef;
72
71
  this.title = gettext('Select a protocol');
73
72
  this.result = new Promise((resolve, reject) => {
74
73
  this._resolve = resolve;
75
74
  this._reject = reject;
76
75
  });
76
+ // set via initialState
77
77
  this.protocolProviders = [];
78
- this.protocolProviders = this.remoteAccess.getProtocolProviders();
79
78
  }
80
79
  cancel() {
81
80
  this._reject();
@@ -84,22 +83,23 @@ class RemoteAccessPickProtocolModalComponent {
84
83
  this.bsModalRef.hide();
85
84
  this._resolve(protocol);
86
85
  }
87
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.BsModalRef }], target: i0.ɵɵFactoryTarget.Component }); }
88
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessPickProtocolModalComponent, isStandalone: true, selector: "c8y-remote-access-pick-protocol-modal", ngImport: i0, template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n", dependencies: [{ kind: "ngmodule", type: CoreModule }, { kind: "directive", type: i3.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "pipe", type: i3.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i3.ModalComponent, selector: "c8y-modal", inputs: ["disabled", "close", "dismiss", "title", "body", "customFooter", "headerClasses", "labels"], outputs: ["onDismiss", "onClose"] }, { kind: "component", type: i3.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i3.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i3.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }] }); }
86
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, deps: [{ token: i1$1.BsModalRef }], target: i0.ɵɵFactoryTarget.Component }); }
87
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessPickProtocolModalComponent, isStandalone: true, selector: "c8y-remote-access-pick-protocol-modal", ngImport: i0, template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n", dependencies: [{ kind: "ngmodule", type: CoreModule }, { kind: "directive", type: i2.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "pipe", type: i2.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i2.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.ModalComponent, selector: "c8y-modal", inputs: ["disabled", "close", "dismiss", "title", "body", "customFooter", "headerClasses", "labels"], outputs: ["onDismiss", "onClose"] }, { kind: "component", type: i2.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i2.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i2.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }] }); }
89
88
  }
90
89
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessPickProtocolModalComponent, decorators: [{
91
90
  type: Component,
92
91
  args: [{ selector: 'c8y-remote-access-pick-protocol-modal', standalone: true, imports: [CoreModule], template: "<c8y-modal\n [title]=\"title\"\n [headerClasses]=\"'dialog-header'\"\n (onDismiss)=\"cancel()\"\n [labels]=\"{ cancel: 'Cancel' }\"\n>\n <ng-container c8y-modal-title>\n <span [c8yIcon]=\"'laptop'\"></span>\n </ng-container>\n <div class=\"p-24\">\n <c8y-li *ngFor=\"let provider of protocolProviders\">\n <c8y-li-icon>\n <i [c8yIcon]=\"provider.protocolIcon\"></i>\n </c8y-li-icon>\n <c8y-li-body>\n <div class=\"d-flex j-c-between\">\n <div>\n <p>{{ provider.protocolLabel | translate }}</p>\n <p>\n <small>{{ provider.protocolDescription | translate }}</small>\n </p>\n </div>\n <div>\n <button\n class=\"btn btn-primary\"\n (click)=\"selectProtocol(provider)\"\n [attr.data-cy]=\"'remoteAccessPickProtocolModal--select-' + provider.protocolName\"\n >\n <span translate>Select</span>\n </button>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </div>\n</c8y-modal>\n" }]
93
- }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.BsModalRef }] });
92
+ }], ctorParameters: () => [{ type: i1$1.BsModalRef }] });
94
93
 
95
94
  class RemoteAccessConfigurationListComponent {
96
- constructor(remoteAccess, activatedRoute, modalService, alert, bsModalService, translateService) {
95
+ constructor(remoteAccess, activatedRoute, modalService, alert, bsModalService, translateService, ctx) {
97
96
  this.remoteAccess = remoteAccess;
98
97
  this.activatedRoute = activatedRoute;
99
98
  this.modalService = modalService;
100
99
  this.alert = alert;
101
100
  this.bsModalService = bsModalService;
102
101
  this.translateService = translateService;
102
+ this.ctx = ctx;
103
103
  this.rows = [];
104
104
  this.columns = [
105
105
  new IconDeviceGridColumn({
@@ -192,8 +192,9 @@ class RemoteAccessConfigurationListComponent {
192
192
  };
193
193
  this.protocolProviders = [];
194
194
  this.gridTitle = gettext('Remote access endpoints');
195
- this.protocolProviders = this.remoteAccess.getProtocolProviders();
196
- this.deviceId = this.activatedRoute.parent.snapshot.params.id;
195
+ this.device = this.ctx.getContextData(this.activatedRoute)?.contextData;
196
+ this.deviceId = this.device?.id;
197
+ this.protocolProviders = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);
197
198
  const connectColum = this.columns.find(c => c.name === 'connect');
198
199
  if (connectColum) {
199
200
  connectColum.target = this.deviceId;
@@ -204,14 +205,17 @@ class RemoteAccessConfigurationListComponent {
204
205
  }
205
206
  async reload() {
206
207
  const configurations = await this.remoteAccess.listConfigurations(this.deviceId);
207
- const supportedProtocols = this.protocolProviders.map(p => p.protocolName);
208
- this.rows = configurations.filter(c => supportedProtocols.includes(c.protocol));
208
+ const supportedProtocols = this.remoteAccess.getSupportedProtocolProvidersFor(this.device);
209
+ const supportedProtocolsNames = supportedProtocols.map(p => p.protocolName);
210
+ this.rows = configurations.filter(c => supportedProtocolsNames.includes(c.protocol.toUpperCase()));
209
211
  }
210
212
  async addEndpoint() {
211
213
  let provider = null;
212
214
  if (this.protocolProviders.length > 1) {
213
215
  try {
214
- const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent);
216
+ const modalRef = this.bsModalService.show(RemoteAccessPickProtocolModalComponent, {
217
+ initialState: { protocolProviders: this.protocolProviders }
218
+ });
215
219
  provider = await modalRef.content.result;
216
220
  }
217
221
  catch (e) {
@@ -279,8 +283,8 @@ class RemoteAccessConfigurationListComponent {
279
283
  return null;
280
284
  }
281
285
  }
282
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2$1.ActivatedRoute }, { token: i3.ModalService }, { token: i3.AlertService }, { token: i2.BsModalService }, { token: i5.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
283
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessConfigurationListComponent, isStandalone: true, selector: "c8y-remote-access-configuration-list", ngImport: i0, template: "<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n (click)=\"addEndpoint()\"\n [disabled]=\"!protocolProviders?.length\"\n [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n >\n <i [c8yIcon]=\"'plus-circle'\"></i>\n <span translate>Add endpoint</span>\n </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n <c8y-data-grid\n [title]=\"gridTitle\"\n [rows]=\"rows\"\n [columns]=\"columns\"\n [pagination]=\"pagination\"\n (onReload)=\"reload()\"\n [actionControls]=\"actionControls\"\n [displayOptions]=\"displayOptions\"\n >\n <c8y-ui-empty-state\n [icon]=\"'window-restore'\"\n [title]=\"'No endpoints configured.' | translate\"\n [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n [horizontal]=\"false\"\n >\n <button\n class=\"btn btn-default\"\n (click)=\"addEndpoint()\"\n translate\n >\n Add endpoint\n </button>\n </c8y-ui-empty-state>\n </c8y-data-grid>\n</div>\n", dependencies: [{ kind: "ngmodule", type: DataGridModule }, { kind: "component", type: i3.DataGridComponent, selector: "c8y-data-grid", inputs: ["title", "loadMoreItemsLabel", "loadingItemsLabel", "showSearch", "refresh", "columns", "rows", "pagination", "infiniteScroll", "serverSideDataCallback", "selectable", "singleSelection", "selectionPrimaryKey", "displayOptions", "actionControls", "bulkActionControls", "headerActionControls", "searchText", "configureColumnsEnabled", "showCounterWarning", "activeClassName", "expandableRows", "hideReload"], outputs: ["rowMouseOver", "rowMouseLeave", "rowClick", "onConfigChange", "onBeforeFilter", "onBeforeSearch", "onFilter", "itemsSelect", "onReload", "onAddCustomColumn", "onRemoveCustomColumn", "onColumnFilterReset", "onSort", "onPageSizeChange", "onColumnReordered", "onColumnVisibilityChange"] }, { kind: "component", type: TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "component", type: ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "component", type: EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
286
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2$1.ActivatedRoute }, { token: i2.ModalService }, { token: i2.AlertService }, { token: i1$1.BsModalService }, { token: i5.TranslateService }, { token: i2.ContextRouteService }], target: i0.ɵɵFactoryTarget.Component }); }
287
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RemoteAccessConfigurationListComponent, isStandalone: true, selector: "c8y-remote-access-configuration-list", ngImport: i0, template: "<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n (click)=\"addEndpoint()\"\n [disabled]=\"!protocolProviders?.length\"\n [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n >\n <i [c8yIcon]=\"'plus-circle'\"></i>\n <span translate>Add endpoint</span>\n </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n <c8y-data-grid\n [title]=\"gridTitle\"\n [rows]=\"rows\"\n [columns]=\"columns\"\n [pagination]=\"pagination\"\n (onReload)=\"reload()\"\n [actionControls]=\"actionControls\"\n [displayOptions]=\"displayOptions\"\n >\n <c8y-ui-empty-state\n [icon]=\"'window-restore'\"\n [title]=\"'No endpoints configured.' | translate\"\n [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n [horizontal]=\"false\"\n >\n <button\n class=\"btn btn-default\"\n (click)=\"addEndpoint()\"\n translate\n >\n Add endpoint\n </button>\n </c8y-ui-empty-state>\n </c8y-data-grid>\n</div>\n", dependencies: [{ kind: "ngmodule", type: DataGridModule }, { kind: "component", type: i2.DataGridComponent, selector: "c8y-data-grid", inputs: ["title", "loadMoreItemsLabel", "loadingItemsLabel", "showSearch", "refresh", "columns", "rows", "pagination", "infiniteScroll", "serverSideDataCallback", "selectable", "singleSelection", "selectionPrimaryKey", "displayOptions", "actionControls", "bulkActionControls", "headerActionControls", "searchText", "configureColumnsEnabled", "showCounterWarning", "activeClassName", "expandableRows", "hideReload"], outputs: ["rowMouseOver", "rowMouseLeave", "rowClick", "onConfigChange", "onBeforeFilter", "onBeforeSearch", "onFilter", "itemsSelect", "onReload", "onAddCustomColumn", "onRemoveCustomColumn", "onColumnFilterReset", "onSort", "onPageSizeChange", "onColumnReordered", "onColumnVisibilityChange"] }, { kind: "component", type: TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "component", type: ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "component", type: EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
284
288
  }
285
289
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RemoteAccessConfigurationListComponent, decorators: [{
286
290
  type: Component,
@@ -293,7 +297,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
293
297
  EmptyStateComponent,
294
298
  C8yTranslatePipe
295
299
  ], template: "<c8y-title translate>Remote access</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n (click)=\"addEndpoint()\"\n [disabled]=\"!protocolProviders?.length\"\n [attr.data-cy]=\"'remoteAccessConfigurationList--add-endpoint'\"\n >\n <i [c8yIcon]=\"'plus-circle'\"></i>\n <span translate>Add endpoint</span>\n </button>\n</c8y-action-bar-item>\n\n<div class=\"content-fullpage d-flex d-col border-top\">\n <c8y-data-grid\n [title]=\"gridTitle\"\n [rows]=\"rows\"\n [columns]=\"columns\"\n [pagination]=\"pagination\"\n (onReload)=\"reload()\"\n [actionControls]=\"actionControls\"\n [displayOptions]=\"displayOptions\"\n >\n <c8y-ui-empty-state\n [icon]=\"'window-restore'\"\n [title]=\"'No endpoints configured.' | translate\"\n [subtitle]=\"'Click below to add your first endpoint.' | translate\"\n [horizontal]=\"false\"\n >\n <button\n class=\"btn btn-default\"\n (click)=\"addEndpoint()\"\n translate\n >\n Add endpoint\n </button>\n </c8y-ui-empty-state>\n </c8y-data-grid>\n</div>\n" }]
296
- }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2$1.ActivatedRoute }, { type: i3.ModalService }, { type: i3.AlertService }, { type: i2.BsModalService }, { type: i5.TranslateService }] });
300
+ }], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2$1.ActivatedRoute }, { type: i2.ModalService }, { type: i2.AlertService }, { type: i1$1.BsModalService }, { type: i5.TranslateService }, { type: i2.ContextRouteService }] });
297
301
 
298
302
  const remoteAccessConfigurationListProviders = [
299
303
  hookRoute({