@c8y/ngx-components 1018.0.215 → 1018.0.219
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.
- package/ecosystem/application-plugins/plugin-list.component.d.ts +1 -0
- package/esm2020/core/header/title/title-outlet.component.mjs +3 -3
- package/esm2020/ecosystem/application-plugins/plugin-list.component.mjs +27 -13
- package/esm2020/ecosystem/packages/deploy-application/deploy-application.component.mjs +14 -3
- package/fesm2015/c8y-ngx-components-ecosystem-application-plugins.mjs +19 -13
- package/fesm2015/c8y-ngx-components-ecosystem-application-plugins.mjs.map +1 -1
- package/fesm2015/c8y-ngx-components-ecosystem.mjs +30 -20
- package/fesm2015/c8y-ngx-components-ecosystem.mjs.map +1 -1
- package/fesm2015/c8y-ngx-components.mjs +2 -2
- package/fesm2015/c8y-ngx-components.mjs.map +1 -1
- package/fesm2020/c8y-ngx-components-ecosystem-application-plugins.mjs +26 -13
- package/fesm2020/c8y-ngx-components-ecosystem-application-plugins.mjs.map +1 -1
- package/fesm2020/c8y-ngx-components-ecosystem.mjs +38 -15
- package/fesm2020/c8y-ngx-components-ecosystem.mjs.map +1 -1
- package/fesm2020/c8y-ngx-components.mjs +2 -2
- package/fesm2020/c8y-ngx-components.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -35,6 +35,7 @@ export declare class PluginListComponent {
|
|
|
35
35
|
installPlugin(plugin: ApplicationPlugin): Promise<void>;
|
|
36
36
|
uninstallPlugin(plugin: ApplicationPlugin): Promise<void>;
|
|
37
37
|
private updateAppRemotes;
|
|
38
|
+
private onUpdateEventHandleGS;
|
|
38
39
|
private getAppsForUpdate;
|
|
39
40
|
private isPluginInstalledInApp;
|
|
40
41
|
private getPluginContextPathWithoutVersion;
|
|
@@ -23,9 +23,9 @@ export class TitleOutletComponent {
|
|
|
23
23
|
}
|
|
24
24
|
}
|
|
25
25
|
TitleOutletComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: TitleOutletComponent, deps: [{ token: i1.HeaderService }, { token: i0.ElementRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
|
|
26
|
-
TitleOutletComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: TitleOutletComponent, selector: "c8y-title-outlet", ngImport: i0, template: "<div class=\"c8y-ui-title\">\n <div class=\"title\" *c8yOutlet=\"title\"></div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.OutletDirective, selector: "[c8yOutlet]", inputs: ["c8yOutlet", "c8yOutletProperties", "c8yOutletInjector"] }] });
|
|
26
|
+
TitleOutletComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: TitleOutletComponent, selector: "c8y-title-outlet", ngImport: i0, template: "<div class=\"c8y-ui-title\" data-cy=\"c8y-title--title-outlet\">\n <div class=\"title\" *c8yOutlet=\"title\"></div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.OutletDirective, selector: "[c8yOutlet]", inputs: ["c8yOutlet", "c8yOutletProperties", "c8yOutletInjector"] }] });
|
|
27
27
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: TitleOutletComponent, decorators: [{
|
|
28
28
|
type: Component,
|
|
29
|
-
args: [{ selector: 'c8y-title-outlet', template: "<div class=\"c8y-ui-title\">\n <div class=\"title\" *c8yOutlet=\"title\"></div>\n</div>\n" }]
|
|
29
|
+
args: [{ selector: 'c8y-title-outlet', template: "<div class=\"c8y-ui-title\" data-cy=\"c8y-title--title-outlet\">\n <div class=\"title\" *c8yOutlet=\"title\"></div>\n</div>\n" }]
|
|
30
30
|
}], ctorParameters: function () { return [{ type: i1.HeaderService }, { type: i0.ElementRef }, { type: i0.NgZone }]; } });
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGl0bGUtb3V0bGV0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2NvcmUvaGVhZGVyL3RpdGxlL3RpdGxlLW91dGxldC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9jb3JlL2hlYWRlci90aXRsZS90aXRsZS1vdXRsZXQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNsRCxPQUFPLEVBQUUsWUFBWSxFQUFFLEdBQUcsRUFBRSxNQUFNLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7OztBQU0zRCxNQUFNLE9BQU8sb0JBQW9CO0lBSS9CLFlBQ1UsYUFBNEIsRUFDNUIsT0FBbUIsRUFDbkIsSUFBWTtRQUZaLGtCQUFhLEdBQWIsYUFBYSxDQUFlO1FBQzVCLFlBQU8sR0FBUCxPQUFPLENBQVk7UUFDbkIsU0FBSSxHQUFKLElBQUksQ0FBUTtRQU50QixVQUFLLEdBQUcsU0FBUyxDQUFDO1FBQ2xCLG9CQUFlLEdBQUcsSUFBSSxDQUFDO1FBT3JCLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVELE1BQU0sQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUVoRCxNQUFNLGdCQUFnQixHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLGVBQWUsQ0FBQyxDQUFDO1FBQ2hGLGdCQUFnQixDQUFDLFNBQVMsQ0FBQyxlQUFlLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLGVBQWUsR0FBRyxlQUFlLENBQUMsQ0FBQyxDQUFDO1FBRXhGLElBQUksQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsR0FBRyxFQUFFO1lBQy9CLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTTtpQkFDdEIsSUFBSSxDQUNILFlBQVksQ0FBQyxHQUFHLENBQUMsRUFDakIsTUFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLGVBQWUsQ0FBQyxFQUN0QyxZQUFZLENBQUMsR0FBRyxDQUFDLEVBQ2pCLEdBQUcsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLENBQUMsRUFDekQsTUFBTSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxLQUFLLElBQUksQ0FBQyxFQUN6QixHQUFHLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsU0FBUyxDQUFDLEVBQ3ZCLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FDaEI7aUJBQ0EsU0FBUyxDQUFDLENBQUMsS0FBYSxFQUFFLEVBQUUsQ0FBQyxhQUFhLENBQUMsZUFBZSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7UUFDeEUsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDOztpSEE1QlUsb0JBQW9CO3FHQUFwQixvQkFBb0Isd0RDUmpDLGdJQUdBOzJGREthLG9CQUFvQjtrQkFKaEMsU0FBUzsrQkFDRSxrQkFBa0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIEVsZW1lbnRSZWYsIE5nWm9uZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSGVhZGVyU2VydmljZSB9IGZyb20gJy4uL2hlYWRlci5zZXJ2aWNlJztcbmltcG9ydCB7IGRlYm91bmNlVGltZSwgbWFwLCBmaWx0ZXIgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2M4eS10aXRsZS1vdXRsZXQnLFxuICB0ZW1wbGF0ZVVybDogJy4vdGl0bGUtb3V0bGV0LmNvbXBvbmVudC5odG1sJ1xufSlcbmV4cG9ydCBjbGFzcyBUaXRsZU91dGxldENvbXBvbmVudCB7XG4gIHRpdGxlID0gdW5kZWZpbmVkO1xuICBwYWdlVGl0bGVVcGRhdGUgPSB0cnVlO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgaGVhZGVyU2VydmljZTogSGVhZGVyU2VydmljZSxcbiAgICBwcml2YXRlIGVsZW1lbnQ6IEVsZW1lbnRSZWYsXG4gICAgcHJpdmF0ZSB6b25lOiBOZ1pvbmVcbiAgKSB7XG4gICAgY29uc3QgdGl0bGUkID0gdGhpcy5oZWFkZXJTZXJ2aWNlLm1hcChzdGF0ZSA9PiBzdGF0ZS50aXRsZSk7XG4gICAgdGl0bGUkLnN1YnNjcmliZSh0aXRsZSA9PiAodGhpcy50aXRsZSA9IHRpdGxlKSk7XG5cbiAgICBjb25zdCBwYWdlVGl0bGVVcGRhdGUkID0gdGhpcy5oZWFkZXJTZXJ2aWNlLm1hcChzdGF0ZSA9PiBzdGF0ZS5wYWdlVGl0bGVVcGRhdGUpO1xuICAgIHBhZ2VUaXRsZVVwZGF0ZSQuc3Vic2NyaWJlKHBhZ2VUaXRsZVVwZGF0ZSA9PiAodGhpcy5wYWdlVGl0bGVVcGRhdGUgPSBwYWdlVGl0bGVVcGRhdGUpKTtcblxuICAgIHRoaXMuem9uZS5ydW5PdXRzaWRlQW5ndWxhcigoKSA9PiB7XG4gICAgICB0aGlzLmhlYWRlclNlcnZpY2Uuc3RhdGUkXG4gICAgICAgIC5waXBlKFxuICAgICAgICAgIGRlYm91bmNlVGltZSgxMDApLFxuICAgICAgICAgIGZpbHRlcihzdGF0ZSA9PiBzdGF0ZS5wYWdlVGl0bGVVcGRhdGUpLFxuICAgICAgICAgIGRlYm91bmNlVGltZSgxMDApLFxuICAgICAgICAgIG1hcCgoKSA9PiB0aGlzLmVsZW1lbnQubmF0aXZlRWxlbWVudC5xdWVyeVNlbGVjdG9yKCdoMScpKSxcbiAgICAgICAgICBmaWx0ZXIoZWwgPT4gZWwgIT09IG51bGwpLFxuICAgICAgICAgIG1hcChlbCA9PiBlbC5pbm5lclRleHQpLFxuICAgICAgICAgIGZpbHRlcihCb29sZWFuKVxuICAgICAgICApXG4gICAgICAgIC5zdWJzY3JpYmUoKHRpdGxlOiBzdHJpbmcpID0+IGhlYWRlclNlcnZpY2UuY2hhbmdlUGFnZVRpdGxlKHRpdGxlKSk7XG4gICAgfSk7XG4gIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJjOHktdWktdGl0bGVcIiBkYXRhLWN5PVwiYzh5LXRpdGxlLS10aXRsZS1vdXRsZXRcIj5cbiAgPGRpdiBjbGFzcz1cInRpdGxlXCIgKmM4eU91dGxldD1cInRpdGxlXCI+PC9kaXY+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -6,6 +6,7 @@ import { TranslateService } from '@ngx-translate/core';
|
|
|
6
6
|
import { BsModalService } from 'ngx-bootstrap/modal';
|
|
7
7
|
import { BehaviorSubject, Observable } from 'rxjs';
|
|
8
8
|
import { AppsToUpdateRemotesSelectComponent } from './apps-to-update-remotes-select.component';
|
|
9
|
+
import { pick } from 'lodash-es';
|
|
9
10
|
import * as i0 from "@angular/core";
|
|
10
11
|
import * as i1 from "@c8y/ngx-components/ecosystem/shared";
|
|
11
12
|
import * as i2 from "ngx-bootstrap/modal";
|
|
@@ -42,11 +43,6 @@ export class PluginListComponent {
|
|
|
42
43
|
await this.updateAppRemotes(plugin, 'install');
|
|
43
44
|
}
|
|
44
45
|
async uninstallPlugin(plugin) {
|
|
45
|
-
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {
|
|
46
|
-
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
47
|
-
action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.INSTALL_PLUGIN,
|
|
48
|
-
url: this.CURRENT_LOCATION
|
|
49
|
-
});
|
|
50
46
|
await this.updateAppRemotes(plugin, 'uninstall');
|
|
51
47
|
}
|
|
52
48
|
async updateAppRemotes(plugin, updateType) {
|
|
@@ -93,22 +89,40 @@ export class PluginListComponent {
|
|
|
93
89
|
})
|
|
94
90
|
: this.translateService.instant(gettext('Plugin uninstalled from app "{{ appName }}".'), { appName: app.name });
|
|
95
91
|
this.alertService.success(successText);
|
|
96
|
-
this.
|
|
97
|
-
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
98
|
-
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED,
|
|
99
|
-
url: this.CURRENT_LOCATION
|
|
100
|
-
});
|
|
92
|
+
this.onUpdateEventHandleGS(plugin, app, updateType);
|
|
101
93
|
}
|
|
102
|
-
catch {
|
|
94
|
+
catch (error) {
|
|
103
95
|
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {
|
|
104
96
|
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
105
97
|
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,
|
|
106
|
-
url: this.CURRENT_LOCATION
|
|
98
|
+
url: this.CURRENT_LOCATION,
|
|
99
|
+
error
|
|
107
100
|
});
|
|
108
101
|
}
|
|
109
102
|
}
|
|
110
103
|
this.updatingPluginId[updateType] = '';
|
|
111
104
|
}
|
|
105
|
+
onUpdateEventHandleGS(plugin, app, updateType) {
|
|
106
|
+
const pluginCustomEventInfo = pick(plugin, [
|
|
107
|
+
'name',
|
|
108
|
+
'contextPath',
|
|
109
|
+
'module',
|
|
110
|
+
'version',
|
|
111
|
+
'type',
|
|
112
|
+
'id'
|
|
113
|
+
]);
|
|
114
|
+
const gsEventResult = updateType === 'install'
|
|
115
|
+
? PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED
|
|
116
|
+
: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_REMOVED;
|
|
117
|
+
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {
|
|
118
|
+
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
119
|
+
result: gsEventResult,
|
|
120
|
+
url: this.CURRENT_LOCATION,
|
|
121
|
+
...pluginCustomEventInfo,
|
|
122
|
+
targetApplicationName: app.name,
|
|
123
|
+
targetApplicationContextPath: app.contextPath
|
|
124
|
+
});
|
|
125
|
+
}
|
|
112
126
|
async getAppsForUpdate(plugin, updateType) {
|
|
113
127
|
let apps = (await this.ecosystemService.getWebApplications()).filter(app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL);
|
|
114
128
|
if (updateType === 'install') {
|
|
@@ -213,4 +227,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
213
227
|
}], selectedItems: [{
|
|
214
228
|
type: Output
|
|
215
229
|
}] } });
|
|
216
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-list.component.js","sourceRoot":"","sources":["../../../../ecosystem/application-plugins/plugin-list.component.ts","../../../../ecosystem/application-plugins/plugin-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { ApplicationRemotePlugins, ApplicationType, IApplication } from '@c8y/client';\nimport {\n  AlertService,\n  ApplicationPlugin,\n  GainsightService,\n  PluginsService,\n  gettext\n} from '@c8y/ngx-components';\nimport { EcosystemService, PRODUCT_EXPERIENCE } from '@c8y/ngx-components/ecosystem/shared';\nimport { TranslateService } from '@ngx-translate/core';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { AppsToUpdateRemotesSelectComponent } from './apps-to-update-remotes-select.component';\nimport { UpdateType } from './apps-to-update-remotes-select.model';\n\n@Component({\n  selector: 'c8y-plugin-list',\n  templateUrl: './plugin-list.component.html'\n})\nexport class PluginListComponent {\n  CURRENT_LOCATION = location.href;\n\n  @Input() plugins$: Observable<ApplicationPlugin[]>;\n  @Input() emptyListText = '';\n  @Input() selectable: boolean;\n  /**\n   * Shows the install button for each plugin separately. Currently used in package-details view.\n   */\n  @Input() installable = false;\n  @Input() package: IApplication;\n  @Output() selectedItems: EventEmitter<ApplicationPlugin[]> = new EventEmitter();\n  remotePlugins$: BehaviorSubject<ApplicationRemotePlugins> = new BehaviorSubject({});\n  selectedPlugins: { [key: string]: ApplicationPlugin } = {};\n  updatingPluginId: Record<UpdateType, string> = { install: '', uninstall: '' };\n  appsDisabled: Set<IApplication['id']> = new Set<IApplication['id']>();\n\n  constructor(\n    private ecosystemService: EcosystemService,\n    private bsModalService: BsModalService,\n    private pluginsService: PluginsService,\n    private alertService: AlertService,\n    private translateService: TranslateService,\n    private gainsightService: GainsightService\n  ) {}\n\n  updateSelectedItems(selected: boolean, plugin: ApplicationPlugin) {\n    this.selectedPlugins[plugin.id] = selected ? plugin : undefined;\n    const onlyInstalledPlugins = Object.values(this.selectedPlugins).filter(Boolean);\n    this.selectedItems.emit(onlyInstalledPlugins);\n  }\n\n  async installPlugin(plugin: ApplicationPlugin) {\n    await this.updateAppRemotes(plugin, 'install');\n  }\n\n  async uninstallPlugin(plugin: ApplicationPlugin) {\n    this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {\n      component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n      action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.INSTALL_PLUGIN,\n      url: this.CURRENT_LOCATION\n    });\n    await this.updateAppRemotes(plugin, 'uninstall');\n  }\n\n  private async updateAppRemotes(plugin: ApplicationPlugin, updateType: UpdateType) {\n    this.updatingPluginId[updateType] = plugin?.id;\n    let initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >;\n    try {\n      const apps = await this.getAppsForUpdate(plugin, updateType);\n      initialState = {\n        apps,\n        updateType,\n        pluginName: plugin.name,\n        appsDisabled: this.appsDisabled\n      };\n    } catch (e) {\n      this.alertService.addServerFailure(e);\n      this.updatingPluginId[updateType] = '';\n      return;\n    }\n\n    let selectedApps: IApplication[];\n    try {\n      selectedApps = await this.selectApps(initialState);\n      if (!selectedApps) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    } catch {\n      // unreached\n    }\n\n    if (updateType === 'install') {\n      const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([plugin]);\n      if (!licensesVerifiedByUser) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    }\n\n    for (const app of selectedApps) {\n      try {\n        await this.handleRemotesUpdate(app, plugin, updateType);\n        const successText =\n          updateType === 'install'\n            ? this.translateService.instant(gettext('Plugin installed to app \"{{ appName }}\".'), {\n                appName: app.name\n              })\n            : this.translateService.instant(\n                gettext('Plugin uninstalled from app \"{{ appName }}\".'),\n                { appName: app.name }\n              );\n        this.alertService.success(successText);\n        this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED,\n          url: this.CURRENT_LOCATION\n        });\n      } catch {\n        this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,\n          url: this.CURRENT_LOCATION\n        });\n      }\n    }\n    this.updatingPluginId[updateType] = '';\n  }\n\n  private async getAppsForUpdate(plugin: ApplicationPlugin, updateType: UpdateType) {\n    let apps = (await this.ecosystemService.getWebApplications()).filter(\n      app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL\n    );\n\n    if (updateType === 'install') {\n      this.appsDisabled.clear();\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          this.appsDisabled.add(app.id);\n        }\n      }\n    }\n\n    if (updateType === 'uninstall') {\n      const installedApps: IApplication[] = [];\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          installedApps.push(app);\n        }\n      }\n      apps = installedApps;\n    }\n    return apps;\n  }\n\n  private isPluginInstalledInApp(plugin: ApplicationPlugin, app: IApplication): boolean {\n    const appRemotes = this.pluginsService.getMFRemotes(app) || {};\n\n    for (const [remoteName, modules] of Object.entries(appRemotes)) {\n      const pluginFromThisPackageIsInstalled =\n        this.getPluginContextPathWithoutVersion(remoteName) === plugin.contextPath;\n      const specificPluginModuleIsInstalled = modules.some(module => module === plugin.module);\n      if (pluginFromThisPackageIsInstalled && specificPluginModuleIsInstalled) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  private getPluginContextPathWithoutVersion(remote: string) {\n    return remote.split('@')[0];\n  }\n\n  private async handleRemotesUpdate(\n    application: IApplication,\n    plugin: ApplicationPlugin,\n    updateType: UpdateType\n  ) {\n    try {\n      // When remotes object is not set in the configuration object of an application.\n      // Fallback to setInitialRemotes is triggered.\n      let updatedRemotes = await (updateType === 'install'\n        ? this.pluginsService.addRemotes(application, plugin)\n        : this.pluginsService.removeRemotes(application, this.getAllPluginsToRemove(plugin)));\n      if (!updatedRemotes) {\n        // TODO discuss if we need to handle it like that.\n        // Right now remotes from the cumulocity.json are taken into account when remotes object is missing in the configuration.\n        updatedRemotes = await this.pluginsService.setInitialRemotes(application);\n        // Fresh application MO is needed, after initial state was set.\n        const app = await this.ecosystemService.getApplication(application.id);\n        updatedRemotes = await this.pluginsService.addRemotes(app, plugin);\n      }\n      return this.emitRemotes(updatedRemotes);\n    } catch (er) {\n      if (er) {\n        this.alertService.addServerFailure(er);\n      }\n      throw er;\n    }\n  }\n\n  private getAllPluginsToRemove(plugin: ApplicationPlugin): ApplicationPlugin[] {\n    return this.package.applicationVersions.map(av => ({\n      id: `${plugin.contextPath}@${av.version}/${plugin.module}`,\n      module: plugin.module,\n      path: plugin.path\n    }));\n  }\n\n  private emitRemotes(remotes: ApplicationRemotePlugins): ApplicationRemotePlugins {\n    this.remotePlugins$.next(remotes);\n    return { ...this.remotePlugins$.value };\n  }\n\n  private async selectApps(\n    initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >\n  ): Promise<IApplication[]> {\n    try {\n      return await (\n        this.bsModalService.show(AppsToUpdateRemotesSelectComponent, {\n          class: 'modal-sm',\n          ariaDescribedby: 'modal-body',\n          ariaLabelledBy: 'modal-title',\n          initialState,\n          ignoreBackdropClick: true,\n          keyboard: false\n        }).content as AppsToUpdateRemotesSelectComponent\n      ).result;\n    } catch (er) {\n      return;\n    }\n  }\n}\n","<c8y-list-group class=\"bg-inherit\">\n  <ng-container *ngIf=\"(plugins$ | async)?.length !== 0; else emptyList\">\n    <ng-container *ngFor=\"let plugin of plugins$ | async\">\n      <c8y-li [ngClass]=\"{ disabled: plugin.installed }\" class=\"bg-inherit\">\n        <c8y-plugin-list-item\n          (isItemSelected)=\"updateSelectedItems($event, plugin)\"\n          [plugin]=\"plugin\"\n          [selectable]=\"selectable\"\n          class=\"d-flex\"\n        ></c8y-plugin-list-item>\n        <div class=\"p-l-40 m-t-4\">\n          <button\n            *ngIf=\"installable\"\n            (click)=\"uninstallPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.uninstall }\"\n            [disabled]=\"updatingPluginId.uninstall && plugin.id !== updatingPluginId.uninstall\"\n            class=\"btn btn-danger btn-sm m-l-4\"\n            title=\"{{ 'Uninstall plugin' | translate }}\"\n            translate\n          >\n            Uninstall plugin\n          </button>\n          <button\n            *ngIf=\"installable\"\n            (click)=\"installPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.install }\"\n            [disabled]=\"updatingPluginId.install && plugin.id !== updatingPluginId.install\"\n            class=\"btn btn-default btn-sm m-l-8\"\n            title=\"{{ 'Install plugin' | translate }}\"\n            translate\n          >\n            Install plugin\n          </button>\n        </div>\n      </c8y-li>\n    </ng-container>\n  </ng-container>\n</c8y-list-group>\n<ng-template #emptyList>\n  <div class=\"c8y-empty-state text-left\" *ngIf=\"emptyListText\">\n    <h1 c8yIcon=\"plugin\"></h1>\n    <p>\n      {{ emptyListText | translate }}\n    </p>\n  </div>\n</ng-template>\n"]}
|
|
230
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-list.component.js","sourceRoot":"","sources":["../../../../ecosystem/application-plugins/plugin-list.component.ts","../../../../ecosystem/application-plugins/plugin-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { ApplicationRemotePlugins, ApplicationType, IApplication } from '@c8y/client';\nimport {\n  AlertService,\n  ApplicationPlugin,\n  GainsightService,\n  PluginsService,\n  gettext\n} from '@c8y/ngx-components';\nimport { EcosystemService, PRODUCT_EXPERIENCE } from '@c8y/ngx-components/ecosystem/shared';\nimport { TranslateService } from '@ngx-translate/core';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { AppsToUpdateRemotesSelectComponent } from './apps-to-update-remotes-select.component';\nimport { UpdateType } from './apps-to-update-remotes-select.model';\nimport { pick } from 'lodash-es';\n\n@Component({\n  selector: 'c8y-plugin-list',\n  templateUrl: './plugin-list.component.html'\n})\nexport class PluginListComponent {\n  CURRENT_LOCATION = location.href;\n\n  @Input() plugins$: Observable<ApplicationPlugin[]>;\n  @Input() emptyListText = '';\n  @Input() selectable: boolean;\n  /**\n   * Shows the install button for each plugin separately. Currently used in package-details view.\n   */\n  @Input() installable = false;\n  @Input() package: IApplication;\n  @Output() selectedItems: EventEmitter<ApplicationPlugin[]> = new EventEmitter();\n  remotePlugins$: BehaviorSubject<ApplicationRemotePlugins> = new BehaviorSubject({});\n  selectedPlugins: { [key: string]: ApplicationPlugin } = {};\n  updatingPluginId: Record<UpdateType, string> = { install: '', uninstall: '' };\n  appsDisabled: Set<IApplication['id']> = new Set<IApplication['id']>();\n\n  constructor(\n    private ecosystemService: EcosystemService,\n    private bsModalService: BsModalService,\n    private pluginsService: PluginsService,\n    private alertService: AlertService,\n    private translateService: TranslateService,\n    private gainsightService: GainsightService\n  ) {}\n\n  updateSelectedItems(selected: boolean, plugin: ApplicationPlugin) {\n    this.selectedPlugins[plugin.id] = selected ? plugin : undefined;\n    const onlyInstalledPlugins = Object.values(this.selectedPlugins).filter(Boolean);\n    this.selectedItems.emit(onlyInstalledPlugins);\n  }\n\n  async installPlugin(plugin: ApplicationPlugin) {\n    await this.updateAppRemotes(plugin, 'install');\n  }\n\n  async uninstallPlugin(plugin: ApplicationPlugin) {\n    await this.updateAppRemotes(plugin, 'uninstall');\n  }\n\n  private async updateAppRemotes(plugin: ApplicationPlugin, updateType: UpdateType) {\n    this.updatingPluginId[updateType] = plugin?.id;\n    let initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >;\n    try {\n      const apps = await this.getAppsForUpdate(plugin, updateType);\n      initialState = {\n        apps,\n        updateType,\n        pluginName: plugin.name,\n        appsDisabled: this.appsDisabled\n      };\n    } catch (e) {\n      this.alertService.addServerFailure(e);\n      this.updatingPluginId[updateType] = '';\n      return;\n    }\n\n    let selectedApps: IApplication[];\n    try {\n      selectedApps = await this.selectApps(initialState);\n      if (!selectedApps) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    } catch {\n      // unreached\n    }\n\n    if (updateType === 'install') {\n      const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([plugin]);\n      if (!licensesVerifiedByUser) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    }\n\n    for (const app of selectedApps) {\n      try {\n        await this.handleRemotesUpdate(app, plugin, updateType);\n        const successText =\n          updateType === 'install'\n            ? this.translateService.instant(gettext('Plugin installed to app \"{{ appName }}\".'), {\n                appName: app.name\n              })\n            : this.translateService.instant(\n                gettext('Plugin uninstalled from app \"{{ appName }}\".'),\n                { appName: app.name }\n              );\n        this.alertService.success(successText);\n        this.onUpdateEventHandleGS(plugin, app, updateType);\n      } catch (error) {\n        this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,\n          url: this.CURRENT_LOCATION,\n          error\n        });\n      }\n    }\n    this.updatingPluginId[updateType] = '';\n  }\n\n  private onUpdateEventHandleGS(\n    plugin: ApplicationPlugin,\n    app: IApplication,\n    updateType: UpdateType\n  ) {\n    const pluginCustomEventInfo = pick(plugin, [\n      'name',\n      'contextPath',\n      'module',\n      'version',\n      'type',\n      'id'\n    ]);\n    const gsEventResult =\n      updateType === 'install'\n        ? PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED\n        : PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_REMOVED;\n\n    this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {\n      component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n      result: gsEventResult,\n      url: this.CURRENT_LOCATION,\n      ...pluginCustomEventInfo,\n      targetApplicationName: app.name,\n      targetApplicationContextPath: app.contextPath\n    });\n  }\n\n  private async getAppsForUpdate(plugin: ApplicationPlugin, updateType: UpdateType) {\n    let apps = (await this.ecosystemService.getWebApplications()).filter(\n      app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL\n    );\n\n    if (updateType === 'install') {\n      this.appsDisabled.clear();\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          this.appsDisabled.add(app.id);\n        }\n      }\n    }\n\n    if (updateType === 'uninstall') {\n      const installedApps: IApplication[] = [];\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          installedApps.push(app);\n        }\n      }\n      apps = installedApps;\n    }\n    return apps;\n  }\n\n  private isPluginInstalledInApp(plugin: ApplicationPlugin, app: IApplication): boolean {\n    const appRemotes = this.pluginsService.getMFRemotes(app) || {};\n\n    for (const [remoteName, modules] of Object.entries(appRemotes)) {\n      const pluginFromThisPackageIsInstalled =\n        this.getPluginContextPathWithoutVersion(remoteName) === plugin.contextPath;\n      const specificPluginModuleIsInstalled = modules.some(module => module === plugin.module);\n      if (pluginFromThisPackageIsInstalled && specificPluginModuleIsInstalled) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  private getPluginContextPathWithoutVersion(remote: string) {\n    return remote.split('@')[0];\n  }\n\n  private async handleRemotesUpdate(\n    application: IApplication,\n    plugin: ApplicationPlugin,\n    updateType: UpdateType\n  ) {\n    try {\n      // When remotes object is not set in the configuration object of an application.\n      // Fallback to setInitialRemotes is triggered.\n      let updatedRemotes = await (updateType === 'install'\n        ? this.pluginsService.addRemotes(application, plugin)\n        : this.pluginsService.removeRemotes(application, this.getAllPluginsToRemove(plugin)));\n      if (!updatedRemotes) {\n        // TODO discuss if we need to handle it like that.\n        // Right now remotes from the cumulocity.json are taken into account when remotes object is missing in the configuration.\n        updatedRemotes = await this.pluginsService.setInitialRemotes(application);\n        // Fresh application MO is needed, after initial state was set.\n        const app = await this.ecosystemService.getApplication(application.id);\n        updatedRemotes = await this.pluginsService.addRemotes(app, plugin);\n      }\n      return this.emitRemotes(updatedRemotes);\n    } catch (er) {\n      if (er) {\n        this.alertService.addServerFailure(er);\n      }\n      throw er;\n    }\n  }\n\n  private getAllPluginsToRemove(plugin: ApplicationPlugin): ApplicationPlugin[] {\n    return this.package.applicationVersions.map(av => ({\n      id: `${plugin.contextPath}@${av.version}/${plugin.module}`,\n      module: plugin.module,\n      path: plugin.path\n    }));\n  }\n\n  private emitRemotes(remotes: ApplicationRemotePlugins): ApplicationRemotePlugins {\n    this.remotePlugins$.next(remotes);\n    return { ...this.remotePlugins$.value };\n  }\n\n  private async selectApps(\n    initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >\n  ): Promise<IApplication[]> {\n    try {\n      return await (\n        this.bsModalService.show(AppsToUpdateRemotesSelectComponent, {\n          class: 'modal-sm',\n          ariaDescribedby: 'modal-body',\n          ariaLabelledBy: 'modal-title',\n          initialState,\n          ignoreBackdropClick: true,\n          keyboard: false\n        }).content as AppsToUpdateRemotesSelectComponent\n      ).result;\n    } catch (er) {\n      return;\n    }\n  }\n}\n","<c8y-list-group class=\"bg-inherit\">\n  <ng-container *ngIf=\"(plugins$ | async)?.length !== 0; else emptyList\">\n    <ng-container *ngFor=\"let plugin of plugins$ | async\">\n      <c8y-li [ngClass]=\"{ disabled: plugin.installed }\" class=\"bg-inherit\">\n        <c8y-plugin-list-item\n          (isItemSelected)=\"updateSelectedItems($event, plugin)\"\n          [plugin]=\"plugin\"\n          [selectable]=\"selectable\"\n          class=\"d-flex\"\n        ></c8y-plugin-list-item>\n        <div class=\"p-l-40 m-t-4\">\n          <button\n            *ngIf=\"installable\"\n            (click)=\"uninstallPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.uninstall }\"\n            [disabled]=\"updatingPluginId.uninstall && plugin.id !== updatingPluginId.uninstall\"\n            class=\"btn btn-danger btn-sm m-l-4\"\n            title=\"{{ 'Uninstall plugin' | translate }}\"\n            translate\n          >\n            Uninstall plugin\n          </button>\n          <button\n            *ngIf=\"installable\"\n            (click)=\"installPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.install }\"\n            [disabled]=\"updatingPluginId.install && plugin.id !== updatingPluginId.install\"\n            class=\"btn btn-default btn-sm m-l-8\"\n            title=\"{{ 'Install plugin' | translate }}\"\n            translate\n          >\n            Install plugin\n          </button>\n        </div>\n      </c8y-li>\n    </ng-container>\n  </ng-container>\n</c8y-list-group>\n<ng-template #emptyList>\n  <div class=\"c8y-empty-state text-left\" *ngIf=\"emptyListText\">\n    <h1 c8yIcon=\"plugin\"></h1>\n    <p>\n      {{ emptyListText | translate }}\n    </p>\n  </div>\n</ng-template>\n"]}
|
|
@@ -3,6 +3,7 @@ import { Router } from '@angular/router';
|
|
|
3
3
|
import { GainsightService, PluginsService, ViewContext, WizardComponent, gettext } from '@c8y/ngx-components';
|
|
4
4
|
import { ApplicationPropertiesFormComponent, EcosystemService, PRODUCT_EXPERIENCE } from '@c8y/ngx-components/ecosystem/shared';
|
|
5
5
|
import { TranslateService } from '@ngx-translate/core';
|
|
6
|
+
import { pick } from 'lodash-es';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "@c8y/ngx-components/ecosystem/shared";
|
|
8
9
|
import * as i2 from "@c8y/ngx-components";
|
|
@@ -61,12 +62,21 @@ export class DeployApplicationComponent {
|
|
|
61
62
|
}
|
|
62
63
|
try {
|
|
63
64
|
this.deployedApp = await this.ecosystemService.deployApp(this.package, formGroupValue, this.model);
|
|
65
|
+
const applicationCustomEventInfo = pick(this.package, [
|
|
66
|
+
'id',
|
|
67
|
+
'name',
|
|
68
|
+
'contextPath',
|
|
69
|
+
'label',
|
|
70
|
+
'key'
|
|
71
|
+
]);
|
|
64
72
|
this.deployedWithSuccess = true;
|
|
65
73
|
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.DEPLOY_APPLICATION, {
|
|
66
74
|
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,
|
|
67
75
|
action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,
|
|
68
76
|
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.DEPLOYED,
|
|
69
|
-
url: this.CURRENT_LOCATION
|
|
77
|
+
url: this.CURRENT_LOCATION,
|
|
78
|
+
...applicationCustomEventInfo,
|
|
79
|
+
package: this.deployedApp?.manifest?.package ?? null
|
|
70
80
|
});
|
|
71
81
|
}
|
|
72
82
|
catch (error) {
|
|
@@ -75,7 +85,8 @@ export class DeployApplicationComponent {
|
|
|
75
85
|
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,
|
|
76
86
|
action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,
|
|
77
87
|
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,
|
|
78
|
-
url: this.CURRENT_LOCATION
|
|
88
|
+
url: this.CURRENT_LOCATION,
|
|
89
|
+
error
|
|
79
90
|
});
|
|
80
91
|
}
|
|
81
92
|
finally {
|
|
@@ -114,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
114
125
|
type: ViewChild,
|
|
115
126
|
args: [ApplicationPropertiesFormComponent]
|
|
116
127
|
}] } });
|
|
117
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-application.component.js","sourceRoot":"","sources":["../../../../../ecosystem/packages/deploy-application/deploy-application.component.ts","../../../../../ecosystem/packages/deploy-application/deploy-application.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, ViewChild } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { IApplication, IApplicationVersion } from '@c8y/client';\nimport {\n  GainsightService,\n  PluginsService,\n  ViewContext,\n  WizardComponent,\n  gettext\n} from '@c8y/ngx-components';\nimport {\n  ApplicationPropertiesFormComponent,\n  EcosystemService,\n  LicensedApplicationPlugin,\n  PRODUCT_EXPERIENCE\n} from '@c8y/ngx-components/ecosystem/shared';\nimport { TranslateService } from '@ngx-translate/core';\n\n@Component({\n  selector: 'c8y-deploy-application',\n  templateUrl: './deploy-application.component.html'\n})\nexport class DeployApplicationComponent implements OnInit {\n  CURRENT_LOCATION = location.href;\n\n  inProgress = true;\n  package: IApplication;\n  isDeployed = false;\n  deployedWithSuccess = false;\n  newAppConfig: IApplication;\n  deployedApp: IApplication;\n  applicationHref: string;\n\n  model = {\n    selected: undefined,\n    binary: {\n      id: undefined\n    }\n  };\n  canDeploy = false;\n\n  @ViewChild(ApplicationPropertiesFormComponent)\n  applicationPropertiesForm: ApplicationPropertiesFormComponent;\n\n  readonly descriptionTemplate: string = gettext(\n    'Deploy application using \"{{ packageName }}\" package'\n  );\n  readonly successMessageTemplate = gettext('Application \"{{ packageName }}\" created');\n  readonly doneLabel = gettext('Done');\n  readonly cancelLabel = gettext('Cancel');\n  headerText: string;\n\n  constructor(\n    private ecosystemService: EcosystemService,\n    private wizardComponent: WizardComponent,\n    private translate: TranslateService,\n    private pluginService: PluginsService,\n    private gainsightService: GainsightService,\n    private router: Router\n  ) {\n    this.package = (this.wizardComponent as any).package;\n  }\n\n  async ngOnInit() {\n    const apps = await this.ecosystemService.getHostedAndPackageApplications();\n    this.newAppConfig = this.ecosystemService.getUniqueAppConfig(this.package, apps);\n    this.headerText = this.getHeaderText();\n    this.inProgress = false;\n  }\n\n  async deployApp() {\n    this.inProgress = true;\n    const formGroupValue = this.applicationPropertiesForm.formGroup.getRawValue();\n    this.package.manifest.version = this.model.selected.version;\n\n    const { contextPath, license, name, manifest } = this.package;\n    const type = this.pluginService.getPackageType(this.package);\n    const licensedApp: LicensedApplicationPlugin = {\n      contextPath,\n      license: license || manifest.license,\n      name,\n      type,\n      version: this.model.selected.version\n    };\n\n    const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([licensedApp]);\n    if (!licensesVerifiedByUser) {\n      this.cancel();\n      return;\n    }\n\n    try {\n      this.deployedApp = await this.ecosystemService.deployApp(\n        this.package,\n        formGroupValue,\n        this.model\n      );\n      this.deployedWithSuccess = true;\n      this.gainsightService.triggerEvent(\n        PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.DEPLOY_APPLICATION,\n        {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,\n          action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.DEPLOYED,\n          url: this.CURRENT_LOCATION\n        }\n      );\n    } catch (error) {\n      this.ecosystemService.alertError(error);\n      this.gainsightService.triggerEvent(\n        PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.DEPLOY_APPLICATION,\n        {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,\n          action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,\n          url: this.CURRENT_LOCATION\n        }\n      );\n    } finally {\n      this.markAsDeployed();\n    }\n  }\n\n  cancel() {\n    this.wizardComponent.close();\n  }\n\n  onAppVersionSelect(appVersion: IApplicationVersion) {\n    Object.assign(this.model, {\n      selected: appVersion\n    });\n    this.canDeploy = true;\n  }\n\n  open() {\n    this.router.navigateByUrl(ViewContext.Application.replace(':id', `${this.deployedApp.id}`));\n    this.cancel();\n  }\n\n  private markAsDeployed() {\n    this.isDeployed = true;\n    this.inProgress = false;\n  }\n\n  private getHeaderText() {\n    return this.translate.instant(this.descriptionTemplate, {\n      packageName: this.package.name\n    });\n  }\n}\n","<c8y-wizard-header>\n  <div class=\"modal-header dialog-header\">\n    <i c8yIcon=\"output\"></i>\n    <h4 id=\"modal-title\">{{ 'Deploy application' | translate }}</h4>\n  </div>\n</c8y-wizard-header>\n\n<c8y-wizard-body id=\"modal-body\">\n  <ng-container *ngIf=\"!isDeployed\">\n    <div class=\"fadeIn animated d-flex a-i-center j-c-center d-col\" style=\"min-height: 309px\">\n      <p\n        class=\"bg-level-0 fit-w p-16 text-center text-medium sticky-top bg-level-0 separator-bottom\"\n        *ngIf=\"!inProgress\"\n      >\n        {{ headerText | translate }}\n      </p>\n      <c8y-application-properties-form\n        *ngIf=\"!inProgress\"\n        [application]=\"newAppConfig\"\n        class=\"d-block fit-w bg-level-1\"\n      ></c8y-application-properties-form>\n\n      <ng-container *ngIf=\"!inProgress\">\n        <div [ngStyle]=\"{ padding: '0 16px' }\" class=\"d-block fit-w bg-gray-white\">\n          <c8y-package-version-select\n            [ngModel]=\"model.selected\"\n            (ngModelChange)=\"onAppVersionSelect($event)\"\n            [packageId]=\"package?.id\"\n            [label]=\"'Use extension package version' | translate\"\n          ></c8y-package-version-select>\n        </div>\n      </ng-container>\n\n      <c8y-progress-bar\n        *ngIf=\"inProgress\"\n        [message]=\"'Deploying…' | translate\"\n        class=\"text-center\"\n      ></c8y-progress-bar>\n    </div>\n  </ng-container>\n\n  <ng-container *ngIf=\"isDeployed\">\n    <div\n      *ngIf=\"deployedWithSuccess; else failedDeploy\"\n      class=\"modal-body fadeIn animated\"\n      style=\"min-height: 309px\"\n    >\n      <div class=\"d-flex a-i-center j-c-center d-col\">\n        <c8y-operation-result\n          type=\"success\"\n          [size]=\"84\"\n          [vertical]=\"true\"\n          [text]=\"successMessageTemplate | translate: { packageName: package.name }\"\n          class=\"lead d-block m-b-16\"\n        ></c8y-operation-result>\n      </div>\n    </div>\n    <ng-template #failedDeploy>\n      <div class=\"modal-body fadeIn animated text-center\" style=\"min-height: 257px\">\n        <c8y-operation-result\n          type=\"error\"\n          [size]=\"84\"\n          [vertical]=\"true\"\n          text=\"{{ 'Application creation failed' | translate }}\"\n          class=\"lead\"\n        ></c8y-operation-result>\n      </div>\n    </ng-template>\n  </ng-container>\n</c8y-wizard-body>\n\n<c8y-wizard-footer>\n  <button\n    (click)=\"cancel()\"\n    type=\"button\"\n    class=\"btn btn-default\"\n    title=\"{{ (isDeployed && deployedWithSuccess ? doneLabel : cancelLabel) | translate }}\"\n  >\n    {{ (isDeployed && deployedWithSuccess ? doneLabel : cancelLabel) | translate }}\n  </button>\n\n  <button\n    (click)=\"deployApp()\"\n    *ngIf=\"!isDeployed\"\n    [disabled]=\"inProgress || !canDeploy\"\n    [ngClass]=\"{ 'btn-pending': inProgress }\"\n    class=\"btn btn-primary\"\n    type=\"button\"\n    title=\"{{ 'Deploy' | translate }}\"\n  >\n    {{ 'Deploy' | translate }}\n  </button>\n\n  <button\n    (click)=\"open()\"\n    type=\"button\"\n    class=\"btn btn-primary\"\n    *ngIf=\"isDeployed && deployedWithSuccess\"\n    title=\"{{ 'Open the application details' | translate }}\"\n    translate\n  >\n    Open\n  </button>\n</c8y-wizard-footer>\n"]}
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-application.component.js","sourceRoot":"","sources":["../../../../../ecosystem/packages/deploy-application/deploy-application.component.ts","../../../../../ecosystem/packages/deploy-application/deploy-application.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, ViewChild } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { IApplication, IApplicationVersion } from '@c8y/client';\nimport {\n  GainsightService,\n  PluginsService,\n  ViewContext,\n  WizardComponent,\n  gettext\n} from '@c8y/ngx-components';\nimport {\n  ApplicationPropertiesFormComponent,\n  EcosystemService,\n  LicensedApplicationPlugin,\n  PRODUCT_EXPERIENCE\n} from '@c8y/ngx-components/ecosystem/shared';\nimport { TranslateService } from '@ngx-translate/core';\nimport { pick } from 'lodash-es';\n\n@Component({\n  selector: 'c8y-deploy-application',\n  templateUrl: './deploy-application.component.html'\n})\nexport class DeployApplicationComponent implements OnInit {\n  CURRENT_LOCATION = location.href;\n\n  inProgress = true;\n  package: IApplication;\n  isDeployed = false;\n  deployedWithSuccess = false;\n  newAppConfig: IApplication;\n  deployedApp: IApplication;\n  applicationHref: string;\n\n  model = {\n    selected: undefined,\n    binary: {\n      id: undefined\n    }\n  };\n  canDeploy = false;\n\n  @ViewChild(ApplicationPropertiesFormComponent)\n  applicationPropertiesForm: ApplicationPropertiesFormComponent;\n\n  readonly descriptionTemplate: string = gettext(\n    'Deploy application using \"{{ packageName }}\" package'\n  );\n  readonly successMessageTemplate = gettext('Application \"{{ packageName }}\" created');\n  readonly doneLabel = gettext('Done');\n  readonly cancelLabel = gettext('Cancel');\n  headerText: string;\n\n  constructor(\n    private ecosystemService: EcosystemService,\n    private wizardComponent: WizardComponent,\n    private translate: TranslateService,\n    private pluginService: PluginsService,\n    private gainsightService: GainsightService,\n    private router: Router\n  ) {\n    this.package = (this.wizardComponent as any).package;\n  }\n\n  async ngOnInit() {\n    const apps = await this.ecosystemService.getHostedAndPackageApplications();\n    this.newAppConfig = this.ecosystemService.getUniqueAppConfig(this.package, apps);\n    this.headerText = this.getHeaderText();\n    this.inProgress = false;\n  }\n\n  async deployApp() {\n    this.inProgress = true;\n    const formGroupValue = this.applicationPropertiesForm.formGroup.getRawValue();\n    this.package.manifest.version = this.model.selected.version;\n\n    const { contextPath, license, name, manifest } = this.package;\n    const type = this.pluginService.getPackageType(this.package);\n    const licensedApp: LicensedApplicationPlugin = {\n      contextPath,\n      license: license || manifest.license,\n      name,\n      type,\n      version: this.model.selected.version\n    };\n\n    const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([licensedApp]);\n    if (!licensesVerifiedByUser) {\n      this.cancel();\n      return;\n    }\n\n    try {\n      this.deployedApp = await this.ecosystemService.deployApp(\n        this.package,\n        formGroupValue,\n        this.model\n      );\n      const applicationCustomEventInfo = pick(this.package, [\n        'id',\n        'name',\n        'contextPath',\n        'label',\n        'key'\n      ]);\n\n      this.deployedWithSuccess = true;\n      this.gainsightService.triggerEvent(\n        PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.DEPLOY_APPLICATION,\n        {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,\n          action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.DEPLOYED,\n          url: this.CURRENT_LOCATION,\n          ...applicationCustomEventInfo,\n          package: this.deployedApp?.manifest?.package ?? null\n        }\n      );\n    } catch (error) {\n      this.ecosystemService.alertError(error);\n      this.gainsightService.triggerEvent(\n        PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.DEPLOY_APPLICATION,\n        {\n          component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.DEPLOY_APPLICATION,\n          action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.DEPLOY_APPLICATION,\n          result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,\n          url: this.CURRENT_LOCATION,\n          error\n        }\n      );\n    } finally {\n      this.markAsDeployed();\n    }\n  }\n\n  cancel() {\n    this.wizardComponent.close();\n  }\n\n  onAppVersionSelect(appVersion: IApplicationVersion) {\n    Object.assign(this.model, {\n      selected: appVersion\n    });\n    this.canDeploy = true;\n  }\n\n  open() {\n    this.router.navigateByUrl(ViewContext.Application.replace(':id', `${this.deployedApp.id}`));\n    this.cancel();\n  }\n\n  private markAsDeployed() {\n    this.isDeployed = true;\n    this.inProgress = false;\n  }\n\n  private getHeaderText() {\n    return this.translate.instant(this.descriptionTemplate, {\n      packageName: this.package.name\n    });\n  }\n}\n","<c8y-wizard-header>\n  <div class=\"modal-header dialog-header\">\n    <i c8yIcon=\"output\"></i>\n    <h4 id=\"modal-title\">{{ 'Deploy application' | translate }}</h4>\n  </div>\n</c8y-wizard-header>\n\n<c8y-wizard-body id=\"modal-body\">\n  <ng-container *ngIf=\"!isDeployed\">\n    <div class=\"fadeIn animated d-flex a-i-center j-c-center d-col\" style=\"min-height: 309px\">\n      <p\n        class=\"bg-level-0 fit-w p-16 text-center text-medium sticky-top bg-level-0 separator-bottom\"\n        *ngIf=\"!inProgress\"\n      >\n        {{ headerText | translate }}\n      </p>\n      <c8y-application-properties-form\n        *ngIf=\"!inProgress\"\n        [application]=\"newAppConfig\"\n        class=\"d-block fit-w bg-level-1\"\n      ></c8y-application-properties-form>\n\n      <ng-container *ngIf=\"!inProgress\">\n        <div [ngStyle]=\"{ padding: '0 16px' }\" class=\"d-block fit-w bg-gray-white\">\n          <c8y-package-version-select\n            [ngModel]=\"model.selected\"\n            (ngModelChange)=\"onAppVersionSelect($event)\"\n            [packageId]=\"package?.id\"\n            [label]=\"'Use extension package version' | translate\"\n          ></c8y-package-version-select>\n        </div>\n      </ng-container>\n\n      <c8y-progress-bar\n        *ngIf=\"inProgress\"\n        [message]=\"'Deploying…' | translate\"\n        class=\"text-center\"\n      ></c8y-progress-bar>\n    </div>\n  </ng-container>\n\n  <ng-container *ngIf=\"isDeployed\">\n    <div\n      *ngIf=\"deployedWithSuccess; else failedDeploy\"\n      class=\"modal-body fadeIn animated\"\n      style=\"min-height: 309px\"\n    >\n      <div class=\"d-flex a-i-center j-c-center d-col\">\n        <c8y-operation-result\n          type=\"success\"\n          [size]=\"84\"\n          [vertical]=\"true\"\n          [text]=\"successMessageTemplate | translate: { packageName: package.name }\"\n          class=\"lead d-block m-b-16\"\n        ></c8y-operation-result>\n      </div>\n    </div>\n    <ng-template #failedDeploy>\n      <div class=\"modal-body fadeIn animated text-center\" style=\"min-height: 257px\">\n        <c8y-operation-result\n          type=\"error\"\n          [size]=\"84\"\n          [vertical]=\"true\"\n          text=\"{{ 'Application creation failed' | translate }}\"\n          class=\"lead\"\n        ></c8y-operation-result>\n      </div>\n    </ng-template>\n  </ng-container>\n</c8y-wizard-body>\n\n<c8y-wizard-footer>\n  <button\n    (click)=\"cancel()\"\n    type=\"button\"\n    class=\"btn btn-default\"\n    title=\"{{ (isDeployed && deployedWithSuccess ? doneLabel : cancelLabel) | translate }}\"\n  >\n    {{ (isDeployed && deployedWithSuccess ? doneLabel : cancelLabel) | translate }}\n  </button>\n\n  <button\n    (click)=\"deployApp()\"\n    *ngIf=\"!isDeployed\"\n    [disabled]=\"inProgress || !canDeploy\"\n    [ngClass]=\"{ 'btn-pending': inProgress }\"\n    class=\"btn btn-primary\"\n    type=\"button\"\n    title=\"{{ 'Deploy' | translate }}\"\n  >\n    {{ 'Deploy' | translate }}\n  </button>\n\n  <button\n    (click)=\"open()\"\n    type=\"button\"\n    class=\"btn btn-primary\"\n    *ngIf=\"isDeployed && deployedWithSuccess\"\n    title=\"{{ 'Open the application details' | translate }}\"\n    translate\n  >\n    Open\n  </button>\n</c8y-wizard-footer>\n"]}
|
|
@@ -12,7 +12,7 @@ import * as i4 from '@angular/common';
|
|
|
12
12
|
import * as i4$1 from '@angular/forms';
|
|
13
13
|
import { ApplicationType } from '@c8y/client';
|
|
14
14
|
import * as i2 from '@ngx-translate/core';
|
|
15
|
-
import { uniq } from 'lodash-es';
|
|
15
|
+
import { pick, uniq } from 'lodash-es';
|
|
16
16
|
import * as i1$2 from '@angular/router';
|
|
17
17
|
|
|
18
18
|
class AppStatePipe {
|
|
@@ -140,11 +140,6 @@ class PluginListComponent {
|
|
|
140
140
|
}
|
|
141
141
|
uninstallPlugin(plugin) {
|
|
142
142
|
return __awaiter(this, void 0, void 0, function* () {
|
|
143
|
-
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {
|
|
144
|
-
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
145
|
-
action: PRODUCT_EXPERIENCE.APPLICATIONS.ACTIONS.INSTALL_PLUGIN,
|
|
146
|
-
url: this.CURRENT_LOCATION
|
|
147
|
-
});
|
|
148
143
|
yield this.updateAppRemotes(plugin, 'uninstall');
|
|
149
144
|
});
|
|
150
145
|
}
|
|
@@ -193,23 +188,34 @@ class PluginListComponent {
|
|
|
193
188
|
})
|
|
194
189
|
: this.translateService.instant(gettext('Plugin uninstalled from app "{{ appName }}".'), { appName: app.name });
|
|
195
190
|
this.alertService.success(successText);
|
|
196
|
-
this.
|
|
197
|
-
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
198
|
-
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED,
|
|
199
|
-
url: this.CURRENT_LOCATION
|
|
200
|
-
});
|
|
191
|
+
this.onUpdateEventHandleGS(plugin, app, updateType);
|
|
201
192
|
}
|
|
202
|
-
catch (
|
|
193
|
+
catch (error) {
|
|
203
194
|
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, {
|
|
204
195
|
component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
|
|
205
196
|
result: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.SERVER_FAILURE,
|
|
206
|
-
url: this.CURRENT_LOCATION
|
|
197
|
+
url: this.CURRENT_LOCATION,
|
|
198
|
+
error
|
|
207
199
|
});
|
|
208
200
|
}
|
|
209
201
|
}
|
|
210
202
|
this.updatingPluginId[updateType] = '';
|
|
211
203
|
});
|
|
212
204
|
}
|
|
205
|
+
onUpdateEventHandleGS(plugin, app, updateType) {
|
|
206
|
+
const pluginCustomEventInfo = pick(plugin, [
|
|
207
|
+
'name',
|
|
208
|
+
'contextPath',
|
|
209
|
+
'module',
|
|
210
|
+
'version',
|
|
211
|
+
'type',
|
|
212
|
+
'id'
|
|
213
|
+
]);
|
|
214
|
+
const gsEventResult = updateType === 'install'
|
|
215
|
+
? PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_INSTALLED
|
|
216
|
+
: PRODUCT_EXPERIENCE.APPLICATIONS.RESULTS.PLUGIN_REMOVED;
|
|
217
|
+
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, Object.assign(Object.assign({ component: PRODUCT_EXPERIENCE.APPLICATIONS.COMPONENTS.PLUGIN_LIST, result: gsEventResult, url: this.CURRENT_LOCATION }, pluginCustomEventInfo), { targetApplicationName: app.name, targetApplicationContextPath: app.contextPath }));
|
|
218
|
+
}
|
|
213
219
|
getAppsForUpdate(plugin, updateType) {
|
|
214
220
|
return __awaiter(this, void 0, void 0, function* () {
|
|
215
221
|
let apps = (yield this.ecosystemService.getWebApplications()).filter(app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL);
|