mirror of
https://github.com/avatao-content/frontend-tutorial-framework
synced 2025-04-03 11:52:40 +00:00
commit
9d6a819606
@ -3,11 +3,16 @@
|
|||||||
|
|
||||||
import { Component, OnInit } from '@angular/core';
|
import { Component, OnInit } from '@angular/core';
|
||||||
import { WebSocketService } from '../services/websocket.service';
|
import { WebSocketService } from '../services/websocket.service';
|
||||||
import { ConsoleContentCommand, RewriteContentCommand, ShowLiveLogsCommand } from '../message-types/console-commands';
|
import { WebSocketMessage } from '../message-types/websocket-message';
|
||||||
|
import {
|
||||||
|
ConsoleReadMessage,
|
||||||
|
ConsoleWriteMessage,
|
||||||
|
ConsoleLiveLogsMessage,
|
||||||
|
ConsoleRewriteContentMessage
|
||||||
|
} from '../message-types/console-messages';
|
||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
import { ProcessLogService } from '../services/processlog.service';
|
import { ProcessLogService } from '../services/processlog.service';
|
||||||
import { LogMessage } from '../message-types/log-message';
|
import { LogMessage } from '../message-types/log-message';
|
||||||
import { CommandMessage } from '../message-types/command-message';
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-console',
|
selector: 'app-console',
|
||||||
@ -19,10 +24,10 @@ export class ConsoleComponent implements OnInit {
|
|||||||
rewriteContentWithProcessLogsOnDeploy: string = config.console.rewriteContentWithProcessLogsOnDeploy;
|
rewriteContentWithProcessLogsOnDeploy: string = config.console.rewriteContentWithProcessLogsOnDeploy;
|
||||||
|
|
||||||
command_handlers = {
|
command_handlers = {
|
||||||
'write': this.writeHandler.bind(this),
|
'console.read': this.readHandler.bind(this),
|
||||||
'read': this.readHandler.bind(this),
|
'console.write': this.writeHandler.bind(this),
|
||||||
'showLiveLogs': this.showLiveLogsHandler.bind(this),
|
'console.showLiveLogs': this.showLiveLogsHandler.bind(this),
|
||||||
'rewriteContentWithProcessLogsOnDeploy': this.rewriteContentWithProcessLogsOnDeployHandler.bind(this)
|
'console.rewriteContentWithProcessLogsOnDeploy': this.rewriteContentWithProcessLogsOnDeployHandler.bind(this)
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(private webSocketService: WebSocketService,
|
constructor(private webSocketService: WebSocketService,
|
||||||
@ -30,18 +35,18 @@ export class ConsoleComponent implements OnInit {
|
|||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.webSocketService.connect();
|
this.webSocketService.connect();
|
||||||
this.webSocketService.observeKey<CommandMessage>('console').subscribe(
|
this.webSocketService.observeKey<WebSocketMessage>('console').subscribe(
|
||||||
(event) => this.command_handlers[event.data.command](event.data)
|
message => this.command_handlers[message.key](message)
|
||||||
);
|
);
|
||||||
this.processLogService.newLogs.subscribe((data) => this.newLogsHandler(data));
|
this.processLogService.newLogs.subscribe((data) => this.newLogsHandler(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
writeHandler(data: ConsoleContentCommand) {
|
readHandler(message: ConsoleReadMessage) {
|
||||||
this.setContent(data.content);
|
this.sendContent(this.console_content);
|
||||||
}
|
}
|
||||||
|
|
||||||
readHandler(data: ConsoleContentCommand) {
|
writeHandler(message: ConsoleWriteMessage) {
|
||||||
this.sendContent(this.console_content);
|
this.setContent(message.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
newLogsHandler(logs: LogMessage) {
|
newLogsHandler(logs: LogMessage) {
|
||||||
@ -53,12 +58,12 @@ export class ConsoleComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
showLiveLogsHandler(data: ShowLiveLogsCommand) {
|
showLiveLogsHandler(message: ConsoleLiveLogsMessage) {
|
||||||
this.processLogService.showLiveLogs = data.value;
|
this.processLogService.showLiveLogs = message.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
rewriteContentWithProcessLogsOnDeployHandler(data: RewriteContentCommand) {
|
rewriteContentWithProcessLogsOnDeployHandler(message: ConsoleRewriteContentMessage) {
|
||||||
this.rewriteContentWithProcessLogsOnDeploy = data.value;
|
this.rewriteContentWithProcessLogsOnDeploy = message.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
setContent(content: string) {
|
setContent(content: string) {
|
||||||
@ -66,9 +71,9 @@ export class ConsoleComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
sendContent(content: string) {
|
sendContent(content: string) {
|
||||||
this.webSocketService.send('console', {
|
this.webSocketService.sendJSON({
|
||||||
'command': 'read',
|
'key': 'console.content',
|
||||||
'content': content
|
'value': content
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,11 +5,11 @@ import { Component, OnDestroy, OnInit, ChangeDetectorRef, ElementRef, ViewChild
|
|||||||
import { DeploymentNotificationService } from '../services/deployment-notification.service';
|
import { DeploymentNotificationService } from '../services/deployment-notification.service';
|
||||||
import { Subscription } from 'rxjs';
|
import { Subscription } from 'rxjs';
|
||||||
import { WebSocketService } from '../services/websocket.service';
|
import { WebSocketService } from '../services/websocket.service';
|
||||||
import { HideMessagesCommand, LayoutCommand, TerminalMenuItemCommand } from '../message-types/dashboard-commands';
|
import { WebSocketMessage } from '../message-types/websocket-message';
|
||||||
|
import { HideMessagesMessage, LayoutMessage, TerminalMenuItemMessage } from '../message-types/dashboard-messages';
|
||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
import { ProcessLogService } from '../services/processlog.service';
|
import { ProcessLogService } from '../services/processlog.service';
|
||||||
import { LogMessage } from '../message-types/log-message';
|
import { LogMessage } from '../message-types/log-message';
|
||||||
import { CommandMessage } from '../message-types/command-message';
|
|
||||||
import { HttpClient } from '@angular/common/http';
|
import { HttpClient } from '@angular/common/http';
|
||||||
import { delay, retryWhen, tap } from 'rxjs/operators';
|
import { delay, retryWhen, tap } from 'rxjs/operators';
|
||||||
|
|
||||||
@ -35,11 +35,11 @@ export class DashboardComponent implements OnInit, OnDestroy {
|
|||||||
iframeReloadSubscription: Subscription;
|
iframeReloadSubscription: Subscription;
|
||||||
|
|
||||||
command_handlers = {
|
command_handlers = {
|
||||||
'layout': this.layoutHandler.bind(this),
|
'dashboard.layout': this.layoutHandler.bind(this),
|
||||||
'hideMessages': this.hideMessagesHandler.bind(this),
|
'dashboard.hideMessages': this.hideMessagesHandler.bind(this),
|
||||||
'terminalMenuItem': this.terminalMenuItemHandler.bind(this),
|
'dashboard.terminalMenuItem': this.terminalMenuItemHandler.bind(this),
|
||||||
'reloadFrontend': this.reloadFrontendHandlder.bind(this),
|
'dashboard.reloadFrontend': this.reloadFrontendHandlder.bind(this),
|
||||||
'reloadIframe': this.reloadIframeHandler.bind(this)
|
'dashboard.reloadIframe': this.reloadIframeHandler.bind(this)
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(private deploymentNotificationService: DeploymentNotificationService,
|
constructor(private deploymentNotificationService: DeploymentNotificationService,
|
||||||
@ -57,8 +57,8 @@ export class DashboardComponent implements OnInit, OnDestroy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
initCommandHandling() {
|
initCommandHandling() {
|
||||||
this.webSocketService.observeKey<CommandMessage>('dashboard').subscribe((event) => {
|
this.webSocketService.observeKey<WebSocketMessage>('dashboard').subscribe(message => {
|
||||||
this.command_handlers[event.data.command](event.data);
|
this.command_handlers[message.key](message);
|
||||||
this.changeDetectorRef.detectChanges();
|
this.changeDetectorRef.detectChanges();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -80,11 +80,8 @@ export class DashboardComponent implements OnInit, OnDestroy {
|
|||||||
if (config.dashboard.triggerFirstFSMStep) {
|
if (config.dashboard.triggerFirstFSMStep) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.webSocketService.sendJSON({
|
this.webSocketService.sendJSON({
|
||||||
'key': 'fsm',
|
'key': 'fsm.step',
|
||||||
'data': {
|
'trigger': config.dashboard.triggerFirstFSMStep
|
||||||
'command': 'trigger',
|
|
||||||
'value': config.dashboard.triggerFirstFSMStep
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -96,27 +93,27 @@ export class DashboardComponent implements OnInit, OnDestroy {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
layoutHandler(data: LayoutCommand) {
|
layoutHandler(message: LayoutMessage) {
|
||||||
if (config.dashboard.enabledLayouts.includes(data.value)) {
|
if (config.dashboard.enabledLayouts.includes(message.value)) {
|
||||||
this.setLayout(data.value);
|
this.setLayout(message.value);
|
||||||
} else {
|
} else {
|
||||||
console.log('Invalid ide layout "' + data.value + '" received!');
|
console.log('Invalid ide layout "' + message.value + '" received!');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
hideMessagesHandler(data: HideMessagesCommand) {
|
hideMessagesHandler(message: HideMessagesMessage) {
|
||||||
this.hideMessages = data.value;
|
this.hideMessages = message.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
terminalMenuItemHandler(data: TerminalMenuItemCommand) {
|
terminalMenuItemHandler(message: TerminalMenuItemMessage) {
|
||||||
this.selectTerminalMenuItem(data.value);
|
this.selectTerminalMenuItem(message.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
reloadFrontendHandlder(data: CommandMessage) {
|
reloadFrontendHandlder(message: WebSocketMessage) {
|
||||||
setTimeout(() => window.location.reload(), 2000);
|
setTimeout(() => window.location.reload(), 2000);
|
||||||
}
|
}
|
||||||
|
|
||||||
reloadIframeHandler(data: CommandMessage) {
|
reloadIframeHandler(message: WebSocketMessage) {
|
||||||
setTimeout(() => this.reloadIframeNoSubmit(), 200);
|
setTimeout(() => this.reloadIframeNoSubmit(), 200);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10,9 +10,9 @@
|
|||||||
[class.disabled]="filename === file"
|
[class.disabled]="filename === file"
|
||||||
[disabled]="filename === file"
|
[disabled]="filename === file"
|
||||||
[class.tao-tab-btn-saved]="filename === file && codeState === CodeState.SAVED">
|
[class.tao-tab-btn-saved]="filename === file && codeState === CodeState.SAVED">
|
||||||
<span *ngIf="filename !== file">{{file}}</span>
|
<span *ngIf="filename !== file">{{pathBasename(file)}}</span>
|
||||||
<span *ngIf="filename === file"
|
<span *ngIf="filename === file"
|
||||||
[class.underline]="codeState === CodeState.DIRTY">{{file}}</span>
|
[class.underline]="codeState === CodeState.DIRTY">{{pathBasename(file)}}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -39,6 +39,6 @@
|
|||||||
|
|
||||||
<ngx-monaco-editor [(ngModel)]="code"
|
<ngx-monaco-editor [(ngModel)]="code"
|
||||||
class="tfw-ide-editor"
|
class="tfw-ide-editor"
|
||||||
(ngModelChange)="editorWriteHanlder()"
|
(keyup)="editorWriteHandler()"
|
||||||
[options]="editorOptions"
|
[options]="editorOptions"
|
||||||
></ngx-monaco-editor>
|
></ngx-monaco-editor>
|
||||||
|
@ -3,16 +3,15 @@
|
|||||||
|
|
||||||
import { ChangeDetectorRef, Component, EventEmitter, OnInit, Output } from '@angular/core';
|
import { ChangeDetectorRef, Component, EventEmitter, OnInit, Output } from '@angular/core';
|
||||||
|
|
||||||
import { IDECommand } from '../message-types/ide-command';
|
|
||||||
import { WebSocketService } from '../services/websocket.service';
|
import { WebSocketService } from '../services/websocket.service';
|
||||||
|
import { WebSocketMessage } from '../message-types/websocket-message';
|
||||||
|
import { IDEMessage } from '../message-types/ide-message';
|
||||||
import { ProcessManagerService } from '../services/processmanager.service';
|
import { ProcessManagerService } from '../services/processmanager.service';
|
||||||
import { DeploymentNotificationService } from '../services/deployment-notification.service';
|
import { DeploymentNotificationService } from '../services/deployment-notification.service';
|
||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
import { CommandMessage } from '../message-types/command-message';
|
|
||||||
import { LanguageMap } from './language-map';
|
import { LanguageMap } from './language-map';
|
||||||
import { filter, first } from 'rxjs/operators';
|
import { filter, first } from 'rxjs/operators';
|
||||||
|
|
||||||
|
|
||||||
enum DeployButtonState {
|
enum DeployButtonState {
|
||||||
DEPLOYED,
|
DEPLOYED,
|
||||||
DEPLOYING,
|
DEPLOYING,
|
||||||
@ -20,13 +19,11 @@ enum DeployButtonState {
|
|||||||
TODEPLOY
|
TODEPLOY
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
enum CodeState {
|
enum CodeState {
|
||||||
SAVED,
|
SAVED,
|
||||||
DIRTY
|
DIRTY
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-ide',
|
selector: 'app-ide',
|
||||||
templateUrl: './ide.component.html',
|
templateUrl: './ide.component.html',
|
||||||
@ -36,10 +33,8 @@ export class IdeComponent implements OnInit {
|
|||||||
CodeState = CodeState;
|
CodeState = CodeState;
|
||||||
DeployButtonState = DeployButtonState;
|
DeployButtonState = DeployButtonState;
|
||||||
|
|
||||||
key_id = 'ide';
|
|
||||||
files: string[];
|
files: string[];
|
||||||
filename = '';
|
filename = '';
|
||||||
directory = '';
|
|
||||||
code: string = config.ide.defaultCode;
|
code: string = config.ide.defaultCode;
|
||||||
|
|
||||||
codeState = CodeState.SAVED;
|
codeState = CodeState.SAVED;
|
||||||
@ -58,11 +53,9 @@ export class IdeComponent implements OnInit {
|
|||||||
@Output() newLogs = new EventEmitter<any>();
|
@Output() newLogs = new EventEmitter<any>();
|
||||||
|
|
||||||
command_handlers = {
|
command_handlers = {
|
||||||
'reload': this.reloadHandler.bind(this),
|
'ide.reload': this.reloadHandler.bind(this),
|
||||||
'read': this.readHandler.bind(this),
|
'ide.read': this.readHandler.bind(this),
|
||||||
'select': this.selectHandler.bind(this),
|
'ide.write': this.writeHandler.bind(this),
|
||||||
'write': this.writeHandler.bind(this),
|
|
||||||
'selectdir': this.selectdirHandler.bind(this)
|
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(private webSocketService: WebSocketService,
|
constructor(private webSocketService: WebSocketService,
|
||||||
@ -80,17 +73,14 @@ export class IdeComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
subscribeWS() {
|
subscribeWS() {
|
||||||
this.webSocketService.observeKey<CommandMessage>(this.key_id).subscribe((event) => {
|
this.webSocketService.observeKey<WebSocketMessage>('ide').subscribe(message => {
|
||||||
this.command_handlers[event.data.command](event.data);
|
this.command_handlers[message.key](message);
|
||||||
this.changeDetectorRef.detectChanges();
|
this.changeDetectorRef.detectChanges();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
subscribeFirstLanguageDetection() {
|
subscribeFirstLanguageDetection() {
|
||||||
this.webSocketService.observeKey<CommandMessage>(this.key_id).pipe(
|
this.webSocketService.observeKey<IDEMessage>('ide.read').pipe(first()).subscribe(
|
||||||
filter(message => message.data.command === 'read'),
|
|
||||||
first()
|
|
||||||
).subscribe(
|
|
||||||
() => this.autoDetectEditorLanguageIfEnabled(this.filename)
|
() => this.autoDetectEditorLanguageIfEnabled(this.filename)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -99,18 +89,18 @@ export class IdeComponent implements OnInit {
|
|||||||
this.processManagerService.init();
|
this.processManagerService.init();
|
||||||
this.processManagerService.subscribeCallback(
|
this.processManagerService.subscribeCallback(
|
||||||
config.ide.deployProcessName,
|
config.ide.deployProcessName,
|
||||||
(event) => {
|
message => {
|
||||||
this.deploymentNotificationService.deploying.next(false);
|
this.deploymentNotificationService.deploying.next(false);
|
||||||
this.newLogs.emit({
|
this.newLogs.emit({
|
||||||
stdout: event.data.stdout,
|
stdout: message.stdout,
|
||||||
stderr: event.data.stderr
|
stderr: message.stderr
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
this.processManagerService.subscribeSuccessCallback(
|
this.processManagerService.subscribeSuccessCallback(
|
||||||
config.ide.deployProcessName,
|
config.ide.deployProcessName,
|
||||||
(event) => this.setDeployButtonState(DeployButtonState.DEPLOYED)
|
message => this.setDeployButtonState(DeployButtonState.DEPLOYED)
|
||||||
);
|
);
|
||||||
|
|
||||||
this.processManagerService.subscribeErrorCallback(
|
this.processManagerService.subscribeErrorCallback(
|
||||||
@ -119,36 +109,24 @@ export class IdeComponent implements OnInit {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
updateFileData(data: IDECommand) {
|
reloadHandler(message: WebSocketMessage) {
|
||||||
this.filename = data.filename;
|
|
||||||
this.directory = data.directory;
|
|
||||||
this.code = (data.content != null) ? data.content : this.code;
|
|
||||||
this.files = data.files;
|
|
||||||
}
|
|
||||||
|
|
||||||
selectHandler(data: IDECommand) {
|
|
||||||
this.updateFileData(data);
|
|
||||||
this.autoDetectEditorLanguageIfEnabled(this.filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
reloadHandler(data: CommandMessage) {
|
|
||||||
this.requestCode();
|
this.requestCode();
|
||||||
}
|
}
|
||||||
|
|
||||||
readHandler(data: IDECommand) {
|
readHandler(message: IDEMessage) {
|
||||||
if (this.codeState === CodeState.SAVED) {
|
if (this.codeState === CodeState.SAVED) {
|
||||||
this.updateFileData(data);
|
if (this.filename != message.filename) {
|
||||||
|
this.filename = message.filename;
|
||||||
|
}
|
||||||
|
this.code = (message.content != null) ? message.content : this.code;
|
||||||
|
this.files = message.files;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
writeHandler(data: CommandMessage) {
|
writeHandler(message: IDEMessage) {
|
||||||
this.setCodeState(CodeState.SAVED);
|
this.setCodeState(CodeState.SAVED);
|
||||||
}
|
}
|
||||||
|
|
||||||
selectdirHandler(data: IDECommand) {
|
|
||||||
this.updateFileData(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
autoDetectEditorLanguageIfEnabled(filename: string) {
|
autoDetectEditorLanguageIfEnabled(filename: string) {
|
||||||
if (!config.ide.autoDetectFileLanguage) {
|
if (!config.ide.autoDetectFileLanguage) {
|
||||||
return;
|
return;
|
||||||
@ -181,11 +159,12 @@ export class IdeComponent implements OnInit {
|
|||||||
if (this.codeState === CodeState.DIRTY) {
|
if (this.codeState === CodeState.DIRTY) {
|
||||||
this.sendCodeContents();
|
this.sendCodeContents();
|
||||||
}
|
}
|
||||||
this.selectCode(file);
|
this.filename = file;
|
||||||
this.requestCode();
|
this.requestCode();
|
||||||
|
this.autoDetectEditorLanguageIfEnabled(this.filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
editorWriteHanlder() {
|
editorWriteHandler() {
|
||||||
this.setCodeState(CodeState.DIRTY);
|
this.setCodeState(CodeState.DIRTY);
|
||||||
this.setDeployButtonState(DeployButtonState.TODEPLOY);
|
this.setDeployButtonState(DeployButtonState.TODEPLOY);
|
||||||
this.resetAutoSaveCountdown();
|
this.resetAutoSaveCountdown();
|
||||||
@ -211,23 +190,22 @@ export class IdeComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pathBasename(path: string) {
|
||||||
|
return path.split('/').reverse()[0];
|
||||||
|
}
|
||||||
|
|
||||||
sendCodeContents() {
|
sendCodeContents() {
|
||||||
this.webSocketService.send(this.key_id, {
|
this.webSocketService.sendJSON({
|
||||||
'command': 'write',
|
'key': 'ide.write',
|
||||||
|
'filename': this.filename,
|
||||||
'content': this.code
|
'content': this.code
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
requestCode() {
|
requestCode() {
|
||||||
this.webSocketService.send(this.key_id, {
|
this.webSocketService.sendJSON({
|
||||||
'command': 'read'
|
'key': 'ide.read',
|
||||||
});
|
'filename': this.filename
|
||||||
}
|
|
||||||
|
|
||||||
selectCode(filename: string) {
|
|
||||||
this.webSocketService.send(this.key_id, {
|
|
||||||
'command': 'select',
|
|
||||||
'filename': filename
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
23
src/app/message-types/bot-messages.ts
Normal file
23
src/app/message-types/bot-messages.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
|
export interface MessageConfig extends WebSocketMessage {
|
||||||
|
originator?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MessageMetadata {
|
||||||
|
originator?: string;
|
||||||
|
timestamp?: Date;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MessageData extends MessageMetadata {
|
||||||
|
message: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Message extends MessageData, WebSocketMessage {}
|
||||||
|
|
||||||
|
export interface MessageQueue extends WebSocketMessage {
|
||||||
|
value: Array<MessageData>;
|
||||||
|
}
|
@ -1,6 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
export interface CommandMessage {
|
|
||||||
readonly command: string;
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
|
||||||
import { SetValueCommand } from './set-value-command';
|
|
||||||
|
|
||||||
export interface ConsoleContentCommand extends CommandMessage {
|
|
||||||
content?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ShowLiveLogsCommand extends CommandMessage, SetValueCommand<boolean> {}
|
|
||||||
|
|
||||||
export interface RewriteContentCommand extends CommandMessage, SetValueCommand<string> {}
|
|
15
src/app/message-types/console-messages.ts
Normal file
15
src/app/message-types/console-messages.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
|
import { SetValueCommand } from './set-value-command';
|
||||||
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
|
export interface ConsoleReadMessage extends WebSocketMessage {}
|
||||||
|
|
||||||
|
export interface ConsoleWriteMessage extends WebSocketMessage {
|
||||||
|
value: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ConsoleLiveLogsMessage extends WebSocketMessage, SetValueCommand<boolean> {}
|
||||||
|
|
||||||
|
export interface ConsoleRewriteContentMessage extends WebSocketMessage, SetValueCommand<string> {}
|
@ -1,11 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
|
||||||
import { SetValueCommand } from './set-value-command';
|
|
||||||
|
|
||||||
export interface LayoutCommand extends CommandMessage, SetValueCommand<string> {}
|
|
||||||
|
|
||||||
export interface HideMessagesCommand extends CommandMessage, SetValueCommand<boolean> {}
|
|
||||||
|
|
||||||
export interface TerminalMenuItemCommand extends CommandMessage, SetValueCommand<string> {}
|
|
11
src/app/message-types/dashboard-messages.ts
Normal file
11
src/app/message-types/dashboard-messages.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
|
import { SetValueCommand } from './set-value-command';
|
||||||
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
|
export interface LayoutMessage extends WebSocketMessage, SetValueCommand<string> {}
|
||||||
|
|
||||||
|
export interface HideMessagesMessage extends WebSocketMessage, SetValueCommand<boolean> {}
|
||||||
|
|
||||||
|
export interface TerminalMenuItemMessage extends WebSocketMessage, SetValueCommand<string> {}
|
@ -1,7 +1,9 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
export class FSMUpdate {
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
|
export interface FSMUpdateMessage extends WebSocketMessage {
|
||||||
current_state: string;
|
current_state: string;
|
||||||
valid_transitions: object;
|
valid_transitions: object;
|
||||||
}
|
}
|
@ -1,11 +1,10 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
export interface IDECommand extends CommandMessage {
|
export interface IDEMessage extends WebSocketMessage {
|
||||||
filename: string;
|
filename: string;
|
||||||
content?: string;
|
content?: string;
|
||||||
files: string[];
|
files?: string[];
|
||||||
directory: string;
|
|
||||||
}
|
}
|
@ -1,7 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
export class MessageMetadata {
|
|
||||||
originator: string;
|
|
||||||
timestamp?: Date;
|
|
||||||
}
|
|
@ -1,8 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import { MessagesMessage } from './messages-message';
|
|
||||||
|
|
||||||
export class MessageQueueMessage {
|
|
||||||
messages: Array<MessagesMessage>;
|
|
||||||
}
|
|
@ -1,7 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
|
||||||
import { SetValueCommand } from './set-value-command';
|
|
||||||
|
|
||||||
export interface MessagesControlCommand extends CommandMessage, SetValueCommand<boolean> {}
|
|
@ -1,8 +0,0 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import { MessageMetadata } from './message-metadata';
|
|
||||||
|
|
||||||
export class MessagesMessage extends MessageMetadata {
|
|
||||||
message: string;
|
|
||||||
}
|
|
@ -1,7 +1,7 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
|
||||||
import { LogMessage } from './log-message';
|
import { LogMessage } from './log-message';
|
||||||
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
export interface ProcessLogCommand extends CommandMessage, LogMessage {}
|
export interface ProcessLogMessage extends WebSocketMessage, LogMessage {}
|
@ -1,10 +1,10 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
|
||||||
import { LogMessage } from './log-message';
|
import { LogMessage } from './log-message';
|
||||||
|
import { WebSocketMessage } from './websocket-message';
|
||||||
|
|
||||||
export interface ProcessCommand extends CommandMessage, LogMessage {
|
export interface ProcessMessage extends WebSocketMessage, LogMessage {
|
||||||
process_name: string;
|
name: string;
|
||||||
error?: string;
|
error?: string;
|
||||||
}
|
}
|
@ -1,8 +1,6 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
import { CommandMessage } from './command-message';
|
export interface SetValueCommand<T> {
|
||||||
|
|
||||||
export interface SetValueCommand<T> extends CommandMessage {
|
|
||||||
value: T;
|
value: T;
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
// Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
export class WebSocketMessage<T> {
|
export interface WebSocketMessage {
|
||||||
key: string;
|
key: string;
|
||||||
trigger?: string;
|
|
||||||
data: T;
|
|
||||||
}
|
}
|
||||||
|
@ -17,11 +17,4 @@
|
|||||||
<div class="jumping-circle" id="jc2"></div>
|
<div class="jumping-circle" id="jc2"></div>
|
||||||
<div class="jumping-circle" id="jc3"></div>
|
<div class="jumping-circle" id="jc3"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="tfw-next-button">
|
|
||||||
<button *ngIf="showNextButton"
|
|
||||||
(click)="stepFSM()"
|
|
||||||
class="tao-btn-rainbow">
|
|
||||||
Next
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -2,14 +2,11 @@
|
|||||||
// All Rights Reserved. See LICENSE file for details.
|
// All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
import { ChangeDetectorRef, Component, OnInit, EventEmitter, Output } from '@angular/core';
|
import { ChangeDetectorRef, Component, OnInit, EventEmitter, Output } from '@angular/core';
|
||||||
|
import { MessageConfig, MessageData, Message, MessageQueue } from '../message-types/bot-messages';
|
||||||
import { MarkdownService } from '../services/markdown.service';
|
import { MarkdownService } from '../services/markdown.service';
|
||||||
import { WebSocketService } from '../services/websocket.service';
|
import { WebSocketService } from '../services/websocket.service';
|
||||||
|
|
||||||
import { MessagesMessage } from '../message-types/messages-message';
|
|
||||||
import { MessagesControlCommand } from '../message-types/messages-control-command';
|
|
||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
import { CommandMessage } from '../message-types/command-message';
|
|
||||||
import { MessageQueueMessage } from '../message-types/message-queue-message';
|
|
||||||
import { Subject, Observer, BehaviorSubject } from 'rxjs';
|
import { Subject, Observer, BehaviorSubject } from 'rxjs';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -19,15 +16,17 @@ import { Subject, Observer, BehaviorSubject } from 'rxjs';
|
|||||||
})
|
})
|
||||||
export class MessagesComponent implements OnInit {
|
export class MessagesComponent implements OnInit {
|
||||||
@Output() newMessageEvent: EventEmitter<any> = new EventEmitter();
|
@Output() newMessageEvent: EventEmitter<any> = new EventEmitter();
|
||||||
newMessage: Subject<MessagesMessage> = new Subject<MessagesMessage>();
|
newMessage: Subject<MessageData> = new Subject<MessageData>();
|
||||||
messageInQueue = true;
|
messageInQueue = true;
|
||||||
|
|
||||||
messages: MessagesMessage[] = [];
|
originator: string = 'avataobot';
|
||||||
|
messages: MessageData[] = [];
|
||||||
messageQueueAttender: MessageQueueAttender;
|
messageQueueAttender: MessageQueueAttender;
|
||||||
showNextButton: boolean = config.messages.showNextButton;
|
|
||||||
|
|
||||||
command_handlers = {
|
command_handlers = {
|
||||||
'showNextButton': this.showButtonHandler.bind(this)
|
'message.queue': this.handleQueueMessage.bind(this),
|
||||||
|
'message.config': this.configureMessages.bind(this),
|
||||||
|
'message.send': () => {}
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@ -40,7 +39,7 @@ export class MessagesComponent implements OnInit {
|
|||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.newMessage.subscribe(
|
this.newMessage.subscribe(
|
||||||
(message) => {
|
message => {
|
||||||
this.writeMessage(message);
|
this.writeMessage(message);
|
||||||
this.newMessageEvent.emit();
|
this.newMessageEvent.emit();
|
||||||
});
|
});
|
||||||
@ -49,45 +48,43 @@ export class MessagesComponent implements OnInit {
|
|||||||
);
|
);
|
||||||
|
|
||||||
this.websocketService.connect();
|
this.websocketService.connect();
|
||||||
this.websocketService.observeKey<MessagesMessage>('message').subscribe(
|
this.websocketService.observeKey<Message>('message.send').subscribe(
|
||||||
(event) => this.newMessage.next(event.data)
|
message => this.newMessage.next(message)
|
||||||
);
|
);
|
||||||
this.websocketService.observeKey<MessageQueueMessage>('queueMessages').subscribe(
|
this.websocketService.observeKey<MessageQueue>('message').subscribe(
|
||||||
(event) => this.handleQueueMessage(event.data)
|
message => this.command_handlers[message.key](message)
|
||||||
);
|
|
||||||
this.websocketService.observeKey<CommandMessage>('messagecontrol').subscribe(
|
|
||||||
(event) => this.command_handlers[event.data.command](event.data)
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
writeMessage(message: MessagesMessage) {
|
writeMessage(message: MessageData) {
|
||||||
this.transformMessage(message);
|
this.transformMessage(message);
|
||||||
this.messages.push(message);
|
this.messages.push(message);
|
||||||
this.changeDetectorRef.detectChanges();
|
this.changeDetectorRef.detectChanges();
|
||||||
}
|
}
|
||||||
|
|
||||||
transformMessage(message: MessagesMessage) {
|
transformMessage(message: MessageData) {
|
||||||
message.message = this.convertMarkdownToHTML(message.message);
|
message.message = this.convertMarkdownToHTML(message.message);
|
||||||
|
if (!message.originator) {
|
||||||
|
message.originator = this.originator;
|
||||||
|
}
|
||||||
if (!message.timestamp) {
|
if (!message.timestamp) {
|
||||||
message.timestamp = new Date();
|
message.timestamp = new Date();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
handleQueueMessage(data: MessageQueueMessage) {
|
handleQueueMessage(message: MessageQueue) {
|
||||||
this.messageQueueAttender.queueMessages(data.messages);
|
this.messageQueueAttender.queueMessages(message.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
configureMessages(message: MessageConfig) {
|
||||||
|
if (message.originator) {
|
||||||
|
this.originator = message.originator;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
convertMarkdownToHTML(text: string) {
|
convertMarkdownToHTML(text: string) {
|
||||||
return this.markdownService.convertToHtml(text);
|
return this.markdownService.convertToHtml(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
showButtonHandler(data: MessagesControlCommand) {
|
|
||||||
this.showNextButton = data.value;
|
|
||||||
}
|
|
||||||
|
|
||||||
stepFSM() {
|
|
||||||
this.websocketService.sendJSON({key: '', trigger: 'step_next'});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -96,14 +93,14 @@ class MessageQueueAttender {
|
|||||||
|
|
||||||
private readonly charPerSecond: number;
|
private readonly charPerSecond: number;
|
||||||
private lastMessageLength = 0;
|
private lastMessageLength = 0;
|
||||||
private queue: MessagesMessage[] = [];
|
private queue: MessageData[] = [];
|
||||||
|
|
||||||
constructor(private messageEmitter: Observer<MessagesMessage>, wordPerMinute: number = config.messages.messageQueueWPM) {
|
constructor(private messageEmitter: Observer<MessageData>, wordPerMinute: number = config.messages.messageQueueWPM) {
|
||||||
const charPerMinute = wordPerMinute * 5;
|
const charPerMinute = wordPerMinute * 5;
|
||||||
this.charPerSecond = charPerMinute / 60;
|
this.charPerSecond = charPerMinute / 60;
|
||||||
}
|
}
|
||||||
|
|
||||||
queueMessages(messages: Array<MessagesMessage>) {
|
queueMessages(messages: Array<MessageData>) {
|
||||||
this.queue = this.queue.concat(messages);
|
this.queue = this.queue.concat(messages);
|
||||||
this.attendQueue();
|
this.attendQueue();
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
import { WebSocketService } from './websocket.service';
|
import { WebSocketService } from './websocket.service';
|
||||||
|
|
||||||
import { FSMUpdate } from '../message-types/fsm-update';
|
import { FSMUpdateMessage } from '../message-types/fsm-update-message';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class FSMUpdateService {
|
export class FSMUpdateService {
|
||||||
@ -14,9 +14,9 @@ export class FSMUpdateService {
|
|||||||
constructor(private websocketService: WebSocketService) {}
|
constructor(private websocketService: WebSocketService) {}
|
||||||
|
|
||||||
public init(): void {
|
public init(): void {
|
||||||
this.websocketService.observeKey<FSMUpdate>('FSMUpdate').subscribe((event) => {
|
this.websocketService.observeKey<FSMUpdateMessage>('fsm.update').subscribe(message => {
|
||||||
this.current_state = event.data.current_state;
|
this.current_state = message.current_state;
|
||||||
this.valid_transitions = event.data.valid_transitions;
|
this.valid_transitions = message.valid_transitions;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,25 +3,20 @@
|
|||||||
|
|
||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
import { WebSocketService } from './websocket.service';
|
import { WebSocketService } from './websocket.service';
|
||||||
import { ProcessLogCommand } from '../message-types/process-log-command';
|
import { LogMessage } from '../message-types/log-message';
|
||||||
|
import { ProcessLogMessage } from '../message-types/process-log-message';
|
||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
import { BehaviorSubject } from 'rxjs';
|
import { BehaviorSubject } from 'rxjs';
|
||||||
import { LogMessage } from '../message-types/log-message';
|
|
||||||
import { CommandMessage } from '../message-types/command-message';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ProcessLogService {
|
export class ProcessLogService {
|
||||||
newLogs = new BehaviorSubject<LogMessage>(config.console.defaultLogs);
|
newLogs = new BehaviorSubject<LogMessage>(config.console.defaultLogs);
|
||||||
showLiveLogs = config.console.showLiveLogs;
|
showLiveLogs = config.console.showLiveLogs;
|
||||||
|
|
||||||
command_handlers = {
|
|
||||||
'new_log': this.newLogsHandler.bind(this)
|
|
||||||
};
|
|
||||||
|
|
||||||
constructor(private webSocketService: WebSocketService) {
|
constructor(private webSocketService: WebSocketService) {
|
||||||
this.webSocketService.connect();
|
this.webSocketService.connect();
|
||||||
this.webSocketService.observeKey<CommandMessage>('processlog').subscribe(
|
this.webSocketService.observeKey<ProcessLogMessage>('process.log.new').subscribe(
|
||||||
(event) => this.command_handlers[event.data.command](event.data)
|
message => this.newLogsHandler(message)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -31,11 +26,11 @@ export class ProcessLogService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
newLogsHandler(data: ProcessLogCommand) {
|
newLogsHandler(message: ProcessLogMessage) {
|
||||||
if (this.showLiveLogs) {
|
if (this.showLiveLogs) {
|
||||||
this.newLogs.next({
|
this.newLogs.next({
|
||||||
stdout: data.stdout,
|
stdout: message.stdout,
|
||||||
stderr: data.stderr
|
stderr: message.stderr
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,14 +4,11 @@
|
|||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
|
|
||||||
import { WebSocketService } from './websocket.service';
|
import { WebSocketService } from './websocket.service';
|
||||||
import { ProcessCommand } from '../message-types/process-command';
|
import { ProcessMessage } from '../message-types/process-message';
|
||||||
import { filter } from 'rxjs/operators';
|
import { filter } from 'rxjs/operators';
|
||||||
import { WebSocketMessage } from '../message-types/websocket-message';
|
|
||||||
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ProcessManagerService {
|
export class ProcessManagerService {
|
||||||
key = 'processmanager';
|
|
||||||
process_name: string;
|
process_name: string;
|
||||||
|
|
||||||
constructor(private webSocketService: WebSocketService) {}
|
constructor(private webSocketService: WebSocketService) {}
|
||||||
@ -20,25 +17,21 @@ export class ProcessManagerService {
|
|||||||
this.webSocketService.connect();
|
this.webSocketService.connect();
|
||||||
}
|
}
|
||||||
|
|
||||||
subscribeCallback(process_name: string, callback: (event: WebSocketMessage<ProcessCommand>) => void) {
|
subscribeCallback(process_name: string, callback: (message: ProcessMessage) => void) {
|
||||||
this.observeProcessMessage(process_name).subscribe(callback);
|
this.observeProcessMessage(process_name).subscribe(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
subscribeSuccessCallback(process_name: string, callback: (event: WebSocketMessage<ProcessCommand>) => void) {
|
subscribeSuccessCallback(process_name: string, callback: (message: ProcessMessage) => void) {
|
||||||
this.observeProcessMessage(process_name).pipe(filter(message => !('error' in message.data))).subscribe(callback);
|
this.observeProcessMessage(process_name).pipe(filter(message => !('error' in message))).subscribe(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
subscribeErrorCallback(process_name: string, callback: (event: WebSocketMessage<ProcessCommand>) => void) {
|
subscribeErrorCallback(process_name: string, callback: (message: ProcessMessage) => void) {
|
||||||
this.observeProcessMessage(process_name).pipe(filter(message => 'error' in message.data)).subscribe(callback);
|
this.observeProcessMessage(process_name).pipe(filter(message => 'error' in message)).subscribe(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
observeProcessMessage(process_name: string) {
|
observeProcessMessage(process_name: string) {
|
||||||
return this.webSocketService.observeKey<ProcessCommand>(this.key)
|
return this.webSocketService.observeKey<ProcessMessage>('process')
|
||||||
.pipe(filter(message => message.data.process_name === process_name));
|
.pipe(filter(message => message.name === process_name));
|
||||||
}
|
|
||||||
|
|
||||||
sendCommand(command: string, process_name: string) {
|
|
||||||
this.webSocketService.send(this.key, {'command': command, 'process_name': process_name});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
startProcess(process_name: string) {
|
startProcess(process_name: string) {
|
||||||
@ -52,4 +45,11 @@ export class ProcessManagerService {
|
|||||||
restartProcess(process_name: string) {
|
restartProcess(process_name: string) {
|
||||||
this.sendCommand('restart', process_name);
|
this.sendCommand('restart', process_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sendCommand(command: string, process_name: string) {
|
||||||
|
this.webSocketService.sendJSON({
|
||||||
|
'key': 'process.'+command,
|
||||||
|
'name': process_name
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ function jsonWebsocketConnect(url: string, input: Observable<object>, protocols?
|
|||||||
@Injectable()
|
@Injectable()
|
||||||
export class WebSocketService {
|
export class WebSocketService {
|
||||||
private uplink: QueueingSubject<object>;
|
private uplink: QueueingSubject<object>;
|
||||||
public downlink: Observable<WebSocketMessage<undefined>>;
|
public downlink: Observable<WebSocketMessage>;
|
||||||
|
|
||||||
constructor() {}
|
constructor() {}
|
||||||
|
|
||||||
@ -33,14 +33,14 @@ export class WebSocketService {
|
|||||||
wsproto + window.location.host + '/ws',
|
wsproto + window.location.host + '/ws',
|
||||||
this.uplink = new QueueingSubject<object>()
|
this.uplink = new QueueingSubject<object>()
|
||||||
).messages.pipe(
|
).messages.pipe(
|
||||||
map(message => <WebSocketMessage<undefined>> message),
|
map(message => <WebSocketMessage> message),
|
||||||
share()
|
share()
|
||||||
);
|
);
|
||||||
console.log('ws connected');
|
console.log('ws connected');
|
||||||
}
|
}
|
||||||
|
|
||||||
public observeKey<T>(key: string): Observable<WebSocketMessage<T>> {
|
public observeKey<T extends WebSocketMessage>(key: string): Observable<T> {
|
||||||
return this.downlink.pipe(filter(message => message.key === key));
|
return this.downlink.pipe(filter(message => message.key.startsWith(key)), map(message => <T> message));
|
||||||
}
|
}
|
||||||
|
|
||||||
public send(key: string, data?: any, trigger?: any): void {
|
public send(key: string, data?: any, trigger?: any): void {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user