mirror of
https://github.com/Ride-The-Lightning/RTL.git
synced 2025-02-21 22:11:54 +01:00
Base Config Page
Base Config Page
This commit is contained in:
parent
dd180cba25
commit
6945d084b4
23 changed files with 339 additions and 40 deletions
|
@ -62,7 +62,7 @@ export const getInfo = (req, res, next) => {
|
|||
req.session.selectedNode.ln_version = body.version || '';
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Connecting to the Core Lightning\'s Websocket Server.' });
|
||||
clWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}
|
||||
|
|
|
@ -22,7 +22,7 @@ export const listOfferBookmarks = (req, res, next) => {
|
|||
};
|
||||
export const deleteOfferBookmark = (req, res, next) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Offers', msg: 'Deleting Offer Bookmark..' });
|
||||
databaseService.destroy(req.session.selectedNode, CollectionsEnum.OFFERS, CollectionFieldsEnum.BOLT12, req.params.offerStr).then((deleteRes) => {
|
||||
databaseService.remove(req.session.selectedNode, CollectionsEnum.OFFERS, CollectionFieldsEnum.BOLT12, req.params.offerStr).then((deleteRes) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Offers', msg: 'Offer Bookmark Deleted', data: deleteRes });
|
||||
res.status(204).json(req.params.offerStr);
|
||||
}).catch((errRes) => {
|
||||
|
|
|
@ -38,7 +38,7 @@ export const getInfo = (req, res, next) => {
|
|||
body.lnImplementation = 'Eclair';
|
||||
req.session.selectedNode.ln_version = body.version.split('-')[0] || '';
|
||||
eclWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}).catch((errRes) => {
|
||||
|
|
|
@ -45,7 +45,7 @@ export const getInfo = (req, res, next) => {
|
|||
else {
|
||||
req.session.selectedNode.ln_version = body.version.split('-')[0] || '';
|
||||
lndWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}
|
||||
|
|
|
@ -19,7 +19,7 @@ export const updateSelectedNode = (req, res, next) => {
|
|||
if (req.headers && req.headers.authorization && req.headers.authorization !== '') {
|
||||
wsServer.updateLNWSClientDetails(req.session.id, +req.session.selectedNode.index, +req.params.prevNodeIndex);
|
||||
if (req.params.prevNodeIndex !== -1) {
|
||||
databaseService.unloadDatabase(req.params.prevNodeIndex);
|
||||
databaseService.unloadDatabase(req.params.prevNodeIndex, req.session.id);
|
||||
}
|
||||
}
|
||||
const responseVal = !req.session.selectedNode.ln_node ? '' : req.session.selectedNode.ln_node;
|
||||
|
|
|
@ -124,7 +124,7 @@ export const resetPassword = (req, res, next) => {
|
|||
export const logoutUser = (req, res, next) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Authenticate', msg: 'Logged out' });
|
||||
if (req.session.selectedNode && req.session.selectedNode.index) {
|
||||
databaseService.unloadDatabase(+req.session.selectedNode.index);
|
||||
databaseService.unloadDatabase(+req.session.selectedNode.index, req.session.id);
|
||||
}
|
||||
req.session.destroy((err) => {
|
||||
res.clearCookie('connect.sid');
|
||||
|
|
|
@ -11,19 +11,23 @@ export class DatabaseService {
|
|||
this.dbDirectory = join(dirname(fileURLToPath(import.meta.url)), '..', '..', 'database');
|
||||
this.nodeDatabase = {};
|
||||
}
|
||||
loadDatabase(selectedNode) {
|
||||
loadDatabase(session) {
|
||||
const { id, selectedNode } = session;
|
||||
try {
|
||||
if (!this.nodeDatabase[selectedNode.index]) {
|
||||
this.nodeDatabase[selectedNode.index] = { adapter: null, data: null };
|
||||
this.nodeDatabase[selectedNode.index].adapter = new DatabaseAdapter(this.dbDirectory, 'rtldb', selectedNode, id);
|
||||
this.nodeDatabase[selectedNode.index].data = this.nodeDatabase[selectedNode.index].adapter.fetchData();
|
||||
}
|
||||
else {
|
||||
this.nodeDatabase[selectedNode.index].adapter.insertSession(id);
|
||||
}
|
||||
this.nodeDatabase[selectedNode.index].adapter = new DatabaseAdapter(this.dbDirectory, 'rtldb', selectedNode);
|
||||
this.nodeDatabase[selectedNode.index].data = this.nodeDatabase[selectedNode.index].adapter.fetchData();
|
||||
}
|
||||
catch (err) {
|
||||
this.logger.log({ selectedNode: selectedNode, level: 'ERROR', fileName: 'Database', msg: 'Database Load Error', error: err });
|
||||
}
|
||||
}
|
||||
create(selectedNode, collectionName, newDocument) {
|
||||
insert(selectedNode, collectionName, newDocument) {
|
||||
return new Promise((resolve, reject) => {
|
||||
try {
|
||||
if (!selectedNode || !selectedNode.index) {
|
||||
|
@ -105,7 +109,7 @@ export class DatabaseService {
|
|||
}
|
||||
});
|
||||
}
|
||||
destroy(selectedNode, collectionName, documentFieldName, documentFieldValue) {
|
||||
remove(selectedNode, collectionName, documentFieldName, documentFieldValue) {
|
||||
return new Promise((resolve, reject) => {
|
||||
try {
|
||||
if (!selectedNode || !selectedNode.index) {
|
||||
|
@ -154,18 +158,27 @@ export class DatabaseService {
|
|||
return new Error(err);
|
||||
}
|
||||
}
|
||||
unloadDatabase(nodeIndex) {
|
||||
this.saveDatabase(nodeIndex);
|
||||
this.nodeDatabase[nodeIndex] = null;
|
||||
unloadDatabase(nodeIndex, sessionID) {
|
||||
if (nodeIndex > 0) {
|
||||
if (this.nodeDatabase[nodeIndex] && this.nodeDatabase[nodeIndex].adapter) {
|
||||
this.nodeDatabase[nodeIndex].adapter.removeSession(sessionID);
|
||||
if (this.nodeDatabase[nodeIndex].adapter.userSessions && this.nodeDatabase[nodeIndex].adapter.userSessions.length <= 0) {
|
||||
delete this.nodeDatabase[nodeIndex];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
export class DatabaseAdapter {
|
||||
constructor(dbDirectoryPath, fileName, selNode = null) {
|
||||
constructor(dbDirectoryPath, fileName, selNode = null, id = '') {
|
||||
this.dbDirectoryPath = dbDirectoryPath;
|
||||
this.fileName = fileName;
|
||||
this.selNode = selNode;
|
||||
this.id = id;
|
||||
this.dbFile = '';
|
||||
this.userSessions = [];
|
||||
this.dbFile = dbDirectoryPath + sep + fileName + '-node-' + selNode.index + '.json';
|
||||
this.insertSession(id);
|
||||
}
|
||||
fetchData() {
|
||||
try {
|
||||
|
@ -208,5 +221,11 @@ export class DatabaseAdapter {
|
|||
return new Error('Database Write Error ' + JSON.stringify(err));
|
||||
}
|
||||
}
|
||||
insertSession(id = '') {
|
||||
this.userSessions.push(id);
|
||||
}
|
||||
removeSession(sessionID = '') {
|
||||
this.userSessions.splice(this.userSessions.findIndex((sId) => sId === sessionID), 1);
|
||||
}
|
||||
}
|
||||
export const Database = new DatabaseService();
|
||||
|
|
|
@ -56,7 +56,7 @@ export const getInfo = (req, res, next) => {
|
|||
req.session.selectedNode.ln_version = body.version || '';
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Connecting to the Core Lightning\'s Websocket Server.' });
|
||||
clWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}
|
||||
|
|
|
@ -25,7 +25,7 @@ export const listOfferBookmarks = (req, res, next) => {
|
|||
|
||||
export const deleteOfferBookmark = (req, res, next) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Offers', msg: 'Deleting Offer Bookmark..' });
|
||||
databaseService.destroy(req.session.selectedNode, CollectionsEnum.OFFERS, CollectionFieldsEnum.BOLT12, req.params.offerStr).then((deleteRes) => {
|
||||
databaseService.remove(req.session.selectedNode, CollectionsEnum.OFFERS, CollectionFieldsEnum.BOLT12, req.params.offerStr).then((deleteRes) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Offers', msg: 'Offer Bookmark Deleted', data: deleteRes });
|
||||
res.status(204).json(req.params.offerStr);
|
||||
}).catch((errRes) => {
|
||||
|
|
|
@ -36,7 +36,7 @@ export const getInfo = (req, res, next) => {
|
|||
body.lnImplementation = 'Eclair';
|
||||
req.session.selectedNode.ln_version = body.version.split('-')[0] || '';
|
||||
eclWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}).catch((errRes) => {
|
||||
|
|
|
@ -40,7 +40,7 @@ export const getInfo = (req, res, next) => {
|
|||
} else {
|
||||
req.session.selectedNode.ln_version = body.version.split('-')[0] || '';
|
||||
lndWsClient.updateSelectedNode(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session.selectedNode);
|
||||
databaseService.loadDatabase(req.session);
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'GetInfo', msg: 'Node Information Received', data: body });
|
||||
return res.status(200).json(body);
|
||||
}
|
||||
|
|
|
@ -22,7 +22,7 @@ export const updateSelectedNode = (req, res, next) => {
|
|||
if (req.headers && req.headers.authorization && req.headers.authorization !== '') {
|
||||
wsServer.updateLNWSClientDetails(req.session.id, +req.session.selectedNode.index, +req.params.prevNodeIndex);
|
||||
if (req.params.prevNodeIndex !== -1) {
|
||||
databaseService.unloadDatabase(req.params.prevNodeIndex);
|
||||
databaseService.unloadDatabase(req.params.prevNodeIndex, req.session.id);
|
||||
}
|
||||
}
|
||||
const responseVal = !req.session.selectedNode.ln_node ? '' : req.session.selectedNode.ln_node;
|
||||
|
|
|
@ -120,7 +120,7 @@ export const resetPassword = (req, res, next) => {
|
|||
export const logoutUser = (req, res, next) => {
|
||||
logger.log({ selectedNode: req.session.selectedNode, level: 'INFO', fileName: 'Authenticate', msg: 'Logged out' });
|
||||
if (req.session.selectedNode && req.session.selectedNode.index) {
|
||||
databaseService.unloadDatabase(+req.session.selectedNode.index);
|
||||
databaseService.unloadDatabase(+req.session.selectedNode.index, req.session.id);
|
||||
}
|
||||
req.session.destroy((err) => {
|
||||
res.clearCookie('connect.sid');
|
||||
|
|
|
@ -15,19 +15,22 @@ export class DatabaseService {
|
|||
|
||||
constructor() { }
|
||||
|
||||
loadDatabase(selectedNode: CommonSelectedNode) {
|
||||
loadDatabase(session: any) {
|
||||
const { id, selectedNode } = session;
|
||||
try {
|
||||
if (!this.nodeDatabase[selectedNode.index]) {
|
||||
this.nodeDatabase[selectedNode.index] = { adapter: null, data: null };
|
||||
this.nodeDatabase[selectedNode.index].adapter = new DatabaseAdapter(this.dbDirectory, 'rtldb', selectedNode, id);
|
||||
this.nodeDatabase[selectedNode.index].data = this.nodeDatabase[selectedNode.index].adapter.fetchData();
|
||||
} else {
|
||||
this.nodeDatabase[selectedNode.index].adapter.insertSession(id);
|
||||
}
|
||||
this.nodeDatabase[selectedNode.index].adapter = new DatabaseAdapter(this.dbDirectory, 'rtldb', selectedNode);
|
||||
this.nodeDatabase[selectedNode.index].data = this.nodeDatabase[selectedNode.index].adapter.fetchData();
|
||||
} catch (err) {
|
||||
this.logger.log({ selectedNode: selectedNode, level: 'ERROR', fileName: 'Database', msg: 'Database Load Error', error: err });
|
||||
}
|
||||
}
|
||||
|
||||
create(selectedNode: CommonSelectedNode, collectionName: CollectionsEnum, newDocument: any) {
|
||||
insert(selectedNode: CommonSelectedNode, collectionName: CollectionsEnum, newDocument: any) {
|
||||
return new Promise((resolve, reject) => {
|
||||
try {
|
||||
if (!selectedNode || !selectedNode.index) {
|
||||
|
@ -105,7 +108,7 @@ export class DatabaseService {
|
|||
});
|
||||
}
|
||||
|
||||
destroy(selectedNode: CommonSelectedNode, collectionName: CollectionsEnum, documentFieldName: string, documentFieldValue: string) {
|
||||
remove(selectedNode: CommonSelectedNode, collectionName: CollectionsEnum, documentFieldName: string, documentFieldValue: string) {
|
||||
return new Promise((resolve, reject) => {
|
||||
try {
|
||||
if (!selectedNode || !selectedNode.index) {
|
||||
|
@ -155,9 +158,15 @@ export class DatabaseService {
|
|||
}
|
||||
}
|
||||
|
||||
unloadDatabase(nodeIndex: number) {
|
||||
this.saveDatabase(nodeIndex);
|
||||
this.nodeDatabase[nodeIndex] = null;
|
||||
unloadDatabase(nodeIndex: number, sessionID: string) {
|
||||
if (nodeIndex > 0) {
|
||||
if (this.nodeDatabase[nodeIndex] && this.nodeDatabase[nodeIndex].adapter) {
|
||||
this.nodeDatabase[nodeIndex].adapter.removeSession(sessionID);
|
||||
if (this.nodeDatabase[nodeIndex].adapter.userSessions && this.nodeDatabase[nodeIndex].adapter.userSessions.length <= 0) {
|
||||
delete this.nodeDatabase[nodeIndex];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -165,9 +174,11 @@ export class DatabaseService {
|
|||
export class DatabaseAdapter {
|
||||
|
||||
private dbFile = '';
|
||||
private userSessions = [];
|
||||
|
||||
constructor(public dbDirectoryPath: string, public fileName: string, private selNode: CommonSelectedNode = null) {
|
||||
constructor(public dbDirectoryPath: string, public fileName: string, private selNode: CommonSelectedNode = null, private id: string = '') {
|
||||
this.dbFile = dbDirectoryPath + sep + fileName + '-node-' + selNode.index + '.json';
|
||||
this.insertSession(id);
|
||||
}
|
||||
|
||||
fetchData() {
|
||||
|
@ -210,6 +221,14 @@ export class DatabaseAdapter {
|
|||
}
|
||||
}
|
||||
|
||||
insertSession(id: string = '') {
|
||||
this.userSessions.push(id);
|
||||
}
|
||||
|
||||
removeSession(sessionID: string = '') {
|
||||
this.userSessions.splice(this.userSessions.findIndex((sId) => sId === sessionID), 1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
export const Database = new DatabaseService();
|
||||
|
|
|
@ -8,6 +8,7 @@ import { BitcoinConfigComponent } from './shared/components/settings/bitcoin-con
|
|||
import { NodeConfigComponent } from './shared/components/node-config/node-config.component';
|
||||
import { LNPConfigComponent } from './shared/components/node-config/lnp-config/lnp-config.component';
|
||||
import { NodeSettingsComponent } from './shared/components/node-config/node-settings/node-settings.component';
|
||||
import { PageSettingsComponent } from './shared/components/node-config/page-settings/page-settings.component';
|
||||
import { ServicesSettingsComponent } from './shared/components/node-config/services-settings/services-settings.component';
|
||||
import { LoopServiceSettingsComponent } from './shared/components/node-config/services-settings/loop-service-settings/loop-service-settings.component';
|
||||
import { BoltzServiceSettingsComponent } from './shared/components/node-config/services-settings/boltz-service-settings/boltz-service-settings.component';
|
||||
|
@ -42,8 +43,9 @@ export const routes: Routes = [
|
|||
},
|
||||
{
|
||||
path: 'config', component: NodeConfigComponent, canActivate: [AuthGuard], children: [
|
||||
{ path: '', pathMatch: 'full', redirectTo: 'layout' },
|
||||
{ path: 'layout', component: NodeSettingsComponent, canActivate: [AuthGuard] },
|
||||
{ path: '', pathMatch: 'full', redirectTo: 'applayout' },
|
||||
{ path: 'applayout', component: NodeSettingsComponent, canActivate: [AuthGuard] },
|
||||
{ path: 'pglayout', component: PageSettingsComponent, canActivate: [AuthGuard] },
|
||||
{
|
||||
path: 'services', component: ServicesSettingsComponent, canActivate: [AuthGuard], children: [
|
||||
{ path: '', pathMatch: 'full', redirectTo: 'loop' },
|
||||
|
|
|
@ -22,7 +22,7 @@ import { LNDEffects } from '../../store/lnd.effects';
|
|||
import { RTLEffects } from '../../../store/rtl.effects';
|
||||
import { RTLState } from '../../../store/rtl.state';
|
||||
import { openAlert, openConfirmation } from '../../../store/rtl.actions';
|
||||
import { fetchPayments, sendPayment } from '../../store/lnd.actions';
|
||||
import { sendPayment } from '../../store/lnd.actions';
|
||||
import { lndNodeInformation, lndNodeSettings, payments, peers } from '../../store/lnd.selector';
|
||||
|
||||
@Component({
|
||||
|
|
|
@ -7,10 +7,11 @@
|
|||
<mat-card-content fxLayout="column">
|
||||
<nav mat-tab-nav-bar>
|
||||
<div role="tab" mat-tab-link class="mat-tab-label" [active]="activeLink === links[0].link" (click)="activeLink = links[0].link" routerLink="{{links[0].link}}">{{links[0].name}}</div>
|
||||
<!-- <div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'LND' || selNode?.lnImplementation?.toUpperCase() === 'CLN'" class="mat-tab-label" [active]="activeLink === links[1].link" (click)="activeLink = links[1].link" routerLink="{{links[1].link}}" [state]="{ initial: false }">{{links[1].name}}</div> -->
|
||||
<div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'LND'" class="mat-tab-label" [active]="activeLink === links[1].link" (click)="activeLink = links[1].link" routerLink="{{links[1].link}}" [state]="{ initial: false }">{{links[1].name}}</div>
|
||||
<div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'CLN'" class="mat-tab-label" [active]="activeLink === links[2].link" (click)="activeLink = links[2].link" routerLink="{{links[2].link}}">{{links[2].name}}</div>
|
||||
<div role="tab" mat-tab-link *ngIf="showLnConfig" class="mat-tab-label" [active]="activeLink === links[3].link" (click)="showLnConfigClicked()">{{links[3].name}}</div>
|
||||
<div role="tab" mat-tab-link class="mat-tab-label" [active]="activeLink === links[1].link" (click)="activeLink = links[1].link" routerLink="{{links[1].link}}">{{links[1].name}}</div>
|
||||
<!-- <div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'LND' || selNode?.lnImplementation?.toUpperCase() === 'CLN'" class="mat-tab-label" [active]="activeLink === links[2].link" (click)="activeLink = links[2].link" routerLink="{{links[2].link}}" [state]="{ initial: false }">{{links[2].name}}</div> -->
|
||||
<div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'LND'" class="mat-tab-label" [active]="activeLink === links[2].link" (click)="activeLink = links[2].link" routerLink="{{links[2].link}}" [state]="{ initial: false }">{{links[2].name}}</div>
|
||||
<div role="tab" mat-tab-link *ngIf="selNode?.lnImplementation?.toUpperCase() === 'CLN'" class="mat-tab-label" [active]="activeLink === links[3].link" (click)="activeLink = links[3].link" routerLink="{{links[3].link}}">{{links[3].name}}</div>
|
||||
<div role="tab" mat-tab-link *ngIf="showLnConfig" class="mat-tab-label" [active]="activeLink === links[4].link" (click)="showLnConfigClicked()">{{links[4].name}}</div>
|
||||
</nav>
|
||||
<div fxLayout="column" fxFlex="100" fxLayoutAlign="space-between stretch" class="mat-tab-body-wrapper mb-2">
|
||||
<router-outlet></router-outlet>
|
||||
|
|
|
@ -23,7 +23,7 @@ export class NodeConfigComponent implements OnInit, OnDestroy {
|
|||
public showLnConfig = false;
|
||||
public selNode: ConfigSettingsNode | any;
|
||||
public lnImplementationStr = '';
|
||||
public links = [{ link: 'layout', name: 'Layout' }, { link: 'services', name: 'Services' }, { link: 'experimental', name: 'Experimental' }, { link: 'lnconfig', name: this.lnImplementationStr }];
|
||||
public links = [{ link: 'applayout', name: 'App Layout' }, { link: 'pglayout', name: 'Page Layout' }, { link: 'services', name: 'Services' }, { link: 'experimental', name: 'Experimental' }, { link: 'lnconfig', name: this.lnImplementationStr }];
|
||||
public activeLink = '';
|
||||
private unSubs: Array<Subject<void>> = [new Subject(), new Subject(), new Subject()];
|
||||
|
||||
|
@ -56,7 +56,7 @@ export class NodeConfigComponent implements OnInit, OnDestroy {
|
|||
break;
|
||||
}
|
||||
if (this.selNode.authentication && this.selNode.authentication.configPath && this.selNode.authentication.configPath.trim() !== '') {
|
||||
this.links[3].name = this.lnImplementationStr;
|
||||
this.links[4].name = this.lnImplementationStr;
|
||||
this.showLnConfig = true;
|
||||
}
|
||||
});
|
||||
|
@ -73,7 +73,7 @@ export class NodeConfigComponent implements OnInit, OnDestroy {
|
|||
}));
|
||||
this.rtlEffects.closeAlert.pipe(takeUntil(this.unSubs[1])).subscribe((alertRes) => {
|
||||
if (alertRes) {
|
||||
this.activeLink = this.links[3].link;
|
||||
this.activeLink = this.links[4].link;
|
||||
this.router.navigate(['./' + this.activeLink], { relativeTo: this.activatedRoute });
|
||||
}
|
||||
});
|
||||
|
|
|
@ -0,0 +1,75 @@
|
|||
<div [perfectScrollbar] fxLayout="column" fxFlex="100">
|
||||
<form fxLayout="column" fxLayoutAlign="start stretch" class="settings-container page-sub-title-container mt-1" #form="ngForm">
|
||||
<div fxLayout="row">
|
||||
<fa-icon [icon]="faPenRuler" class="page-title-img mr-1"></fa-icon>
|
||||
<span class="page-title">Page Settings</span>
|
||||
</div>
|
||||
<!-- <div fxLayout="column" fxLayoutAlign="start stretch" class="mt-1 bordered-box padding-gap-large">
|
||||
<div fxFlex="100" class="alert alert-warn">
|
||||
<fa-icon [icon]="faExclamationTriangle" class="mr-1 alert-icon"></fa-icon>
|
||||
<span>Fiat conversion calls <strong><a href="https://www.blockchain.com/api/exchange_rates_api" target="blank">Blockchain.com</a></strong> API to get conversion rates.</span>
|
||||
</div>
|
||||
<div fxLayout="row wrap" fxLayoutAlign="start center">
|
||||
<mat-slide-toggle tabindex="2" color="primary" [(ngModel)]="selNode.settings.fiatConversion" (change)="selNode.settings.currencyUnit = !$event.checked ? null : selNode.settings.currencyUnit" name="fiatConversion">Enable Fiat Conversion</mat-slide-toggle>
|
||||
<mat-form-field>
|
||||
<mat-select autoFocus [(ngModel)]="selNode.settings.currencyUnit" (selectionChange)="onCurrencyChange($event)" placeholder="Fiat Currency" [disabled]="!selNode.settings.fiatConversion" tabindex="3" [required]="selNode.settings.fiatConversion" name="currencyUnit" #currencyUnit="ngModel">
|
||||
<mat-option *ngFor="let currencyUnit of currencyUnits" [value]="currencyUnit.id">
|
||||
{{currencyUnit.id}}
|
||||
</mat-option>
|
||||
</mat-select>
|
||||
<mat-error *ngIf="selNode.settings.fiatConversion && !selNode.settings.currencyUnit">Currency unit is required.</mat-error>
|
||||
</mat-form-field>
|
||||
</div>
|
||||
</div>
|
||||
<div fxLayout="column" fxFlex="100" fxLayoutAlign="start stretch">
|
||||
<div fxLayout="row wrap" fxLayoutAlign="start start" fxLayout.gt-sm="column" fxFlex="100" fxLayoutAlign.gt-sm="space-between stretch" class="settings-container page-sub-title-container mt-1">
|
||||
<div class="mt-1">
|
||||
<fa-icon [icon]="faPaintBrush" class="page-title-img mr-1"></fa-icon>
|
||||
<span class="page-title">Customization</span>
|
||||
</div>
|
||||
<div fxLayout="column" fxLayoutAlign="start stretch" class="mt-1 bordered-box padding-gap-large">
|
||||
<div fxLayout="column" fxLayoutAlign="start stretch" fxFlex="100">
|
||||
<div fxLayout="row" fxFlex="100" class="alert alert-info mb-0">
|
||||
<fa-icon [icon]="faInfoCircle" class="mr-1 alert-icon"></fa-icon>
|
||||
<span>Dashboard layout will be tailored based on the role selected to better serve its needs.</span>
|
||||
</div>
|
||||
<div fxLayout="column" fxLayoutAlign="start start" fxFlex="100">
|
||||
<h4>Dashboard Layout</h4>
|
||||
<mat-radio-group color="primary" [(ngModel)]="selNode.settings.userPersona" tabindex="1" name="userPersona">
|
||||
<mat-radio-button *ngFor="let userPersona of userPersonas" [value]="userPersona" [checked]="selNode.settings.userPersona === userPersona" class="mr-4">
|
||||
{{userPersona | titlecase}}
|
||||
</mat-radio-button>
|
||||
</mat-radio-group>
|
||||
</div>
|
||||
</div>
|
||||
<mat-divider [inset]="true" class="mt-1"></mat-divider>
|
||||
<div fxLayout="column" fxLayout.gt-xs="row" fxFlex="100" fxLayoutAlign="space-between stretch" fxLayoutAlign.gt-xs="start stretch">
|
||||
<div fxFlex.gt-xs="20" fxFlex.gt-md="15" fxLayout="column" fxLayoutAlign="space-between stretch">
|
||||
<h4>Mode</h4>
|
||||
<mat-radio-group color="primary" [(ngModel)]="selectedThemeMode" (change)="chooseThemeMode()" name="themeMode">
|
||||
<mat-radio-button tabindex="5" *ngFor="let themeMode of themeModes" [value]="themeMode" [ngClass]="{'mr-4': screenSize === screenSizeEnum.XS || screenSize === screenSizeEnum.SM}">{{themeMode.name}}
|
||||
</mat-radio-button>
|
||||
</mat-radio-group>
|
||||
</div>
|
||||
</div>
|
||||
<mat-divider [inset]="true" class="mt-1"></mat-divider>
|
||||
<div fxLayout="column" fxLayout.gt-xs="row" fxFlex="100" fxLayoutAlign="space-between stretch" fxLayoutAlign.gt-xs="start stretch">
|
||||
<div fxLayout="column" fxFlex.gt-xs="50" fxFlex.gt-md="40" fxLayoutAlign="space-between stretch">
|
||||
<h4>Themes</h4>
|
||||
<div fxLayout="row" fxFlex="100" fxLayoutAlign="space-between start">
|
||||
<span *ngFor="let themeColor of themeColors" fxLayout="row" class="theme-name">
|
||||
<div tabindex="9" [class]="themeColor.id | lowercase" [ngClass]="{'skin': true, 'selected-color': selectedThemeColor === themeColor.id}" (click)="changeThemeColor(themeColor.id)"></div>
|
||||
{{themeColor.name}}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div> -->
|
||||
</form>
|
||||
<div fxLayout="row" class="mt-1">
|
||||
<button class="mr-1" mat-stroked-button color="primary" (click)="onResetSettings()" tabindex="10">Reset</button>
|
||||
<button mat-flat-button color="primary" (click)="onUpdateSettings()" tabindex="11">Update</button>
|
||||
</div>
|
||||
</div>
|
|
@ -0,0 +1,51 @@
|
|||
import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
import { StoreModule } from '@ngrx/store';
|
||||
|
||||
import { RootReducer } from '../../../../store/rtl.reducers';
|
||||
import { LNDReducer } from '../../../../lnd/store/lnd.reducers';
|
||||
import { CLNReducer } from '../../../../cln/store/cln.reducers';
|
||||
import { ECLReducer } from '../../../../eclair/store/ecl.reducers';
|
||||
import { CommonService } from '../../../services/common.service';
|
||||
import { LoggerService } from '../../../services/logger.service';
|
||||
|
||||
import { PageSettingsComponent } from './page-settings.component';
|
||||
import { mockDataService, mockLoggerService } from '../../../test-helpers/mock-services';
|
||||
import { SharedModule } from '../../../shared.module';
|
||||
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
|
||||
import { DataService } from '../../../services/data.service';
|
||||
|
||||
describe('PageSettingsComponent', () => {
|
||||
let component: PageSettingsComponent;
|
||||
let fixture: ComponentFixture<PageSettingsComponent>;
|
||||
|
||||
beforeEach(waitForAsync(() => {
|
||||
TestBed.configureTestingModule({
|
||||
declarations: [PageSettingsComponent],
|
||||
imports: [
|
||||
BrowserAnimationsModule,
|
||||
SharedModule,
|
||||
StoreModule.forRoot({ root: RootReducer, lnd: LNDReducer, cln: CLNReducer, ecl: ECLReducer })
|
||||
],
|
||||
providers: [
|
||||
CommonService,
|
||||
{ provide: LoggerService, useClass: mockLoggerService },
|
||||
{ provide: DataService, useClass: mockDataService }
|
||||
]
|
||||
}).
|
||||
compileComponents();
|
||||
}));
|
||||
|
||||
beforeEach(() => {
|
||||
fixture = TestBed.createComponent(PageSettingsComponent);
|
||||
component = fixture.componentInstance;
|
||||
fixture.detectChanges();
|
||||
});
|
||||
|
||||
it('should create', () => {
|
||||
expect(component).toBeTruthy();
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
TestBed.resetTestingModule();
|
||||
});
|
||||
});
|
|
@ -0,0 +1,129 @@
|
|||
import { Component, OnInit, OnDestroy } from '@angular/core';
|
||||
import { Subject } from 'rxjs';
|
||||
import { takeUntil } from 'rxjs/operators';
|
||||
import { Store } from '@ngrx/store';
|
||||
import { faPenRuler } from '@fortawesome/free-solid-svg-icons';
|
||||
|
||||
import { CURRENCY_UNITS, UserPersonaEnum, ScreenSizeEnum, FIAT_CURRENCY_UNITS, NODE_SETTINGS, UI_MESSAGES } from '../../../services/consts-enums-functions';
|
||||
import { ConfigSettingsNode, Settings } from '../../../models/RTLconfig';
|
||||
import { LoggerService } from '../../../services/logger.service';
|
||||
import { CommonService } from '../../../services/common.service';
|
||||
import { RTLState } from '../../../../store/rtl.state';
|
||||
import { saveSettings, setSelectedNode } from '../../../../store/rtl.actions';
|
||||
import { setChildNodeSettingsECL } from '../../../../eclair/store/ecl.actions';
|
||||
import { setChildNodeSettingsCL } from '../../../../cln/store/cln.actions';
|
||||
import { setChildNodeSettingsLND } from '../../../../lnd/store/lnd.actions';
|
||||
import { rootSelectedNode } from '../../../../store/rtl.selector';
|
||||
|
||||
@Component({
|
||||
selector: 'rtl-page-settings',
|
||||
templateUrl: './page-settings.component.html',
|
||||
styleUrls: ['./page-settings.component.scss']
|
||||
})
|
||||
export class PageSettingsComponent implements OnInit, OnDestroy {
|
||||
|
||||
public faPenRuler = faPenRuler;
|
||||
public selNode: ConfigSettingsNode | any;
|
||||
public userPersonas = [UserPersonaEnum.OPERATOR, UserPersonaEnum.MERCHANT];
|
||||
public currencyUnits = FIAT_CURRENCY_UNITS;
|
||||
public themeModes = NODE_SETTINGS.modes;
|
||||
public themeColors = NODE_SETTINGS.themes;
|
||||
public selectedThemeMode = NODE_SETTINGS.modes[0];
|
||||
public selectedThemeColor = NODE_SETTINGS.themes[0].id;
|
||||
public currencyUnit = 'BTC';
|
||||
public smallerCurrencyUnit = 'Sats';
|
||||
public showSettingOption = true;
|
||||
public previousSettings: Settings;
|
||||
public screenSize = '';
|
||||
public screenSizeEnum = ScreenSizeEnum;
|
||||
unSubs: Array<Subject<void>> = [new Subject(), new Subject()];
|
||||
|
||||
constructor(private logger: LoggerService, private commonService: CommonService, private store: Store<RTLState>) {
|
||||
this.screenSize = this.commonService.getScreenSize();
|
||||
}
|
||||
|
||||
ngOnInit() {
|
||||
this.store.select(rootSelectedNode).pipe(takeUntil(this.unSubs[0])).subscribe((selNode) => {
|
||||
this.selNode = selNode;
|
||||
this.selectedThemeMode = this.themeModes.find((themeMode) => this.selNode.settings.themeMode === themeMode.id) || this.themeModes[0];
|
||||
this.selectedThemeColor = this.selNode.settings.themeColor;
|
||||
if (!this.selNode.settings.fiatConversion) {
|
||||
this.selNode.settings.currencyUnit = '';
|
||||
}
|
||||
this.previousSettings = JSON.parse(JSON.stringify(this.selNode.settings));
|
||||
this.logger.info(selNode);
|
||||
});
|
||||
}
|
||||
|
||||
onCurrencyChange(event: any) {
|
||||
this.selNode.settings.currencyUnits = [...CURRENCY_UNITS, event.value];
|
||||
this.store.dispatch(setChildNodeSettingsLND({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: event.value, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion,
|
||||
lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
this.store.dispatch(setChildNodeSettingsCL({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: event.value, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion, lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
this.store.dispatch(setChildNodeSettingsECL({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: event.value, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion, lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
}
|
||||
|
||||
toggleSettings(toggleField: string, event?: any) {
|
||||
this.selNode.settings[toggleField] = !this.selNode.settings[toggleField];
|
||||
}
|
||||
|
||||
changeThemeColor(newThemeColor: string) {
|
||||
this.selectedThemeColor = newThemeColor;
|
||||
this.selNode.settings.themeColor = newThemeColor;
|
||||
}
|
||||
|
||||
chooseThemeMode() {
|
||||
this.selNode.settings.themeMode = this.selectedThemeMode.id;
|
||||
}
|
||||
|
||||
onUpdateSettings(): boolean | void {
|
||||
if (this.selNode.settings.fiatConversion && !this.selNode.settings.currencyUnit) {
|
||||
return true;
|
||||
}
|
||||
this.logger.info(this.selNode.settings);
|
||||
this.store.dispatch(saveSettings({ payload: { uiMessage: UI_MESSAGES.UPDATE_NODE_SETTINGS, settings: this.selNode.settings } }));
|
||||
this.store.dispatch(setChildNodeSettingsLND({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: this.selNode.settings.currencyUnit, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion, lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
this.store.dispatch(setChildNodeSettingsCL({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: this.selNode.settings.currencyUnit, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion, lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
this.store.dispatch(setChildNodeSettingsECL({
|
||||
payload: {
|
||||
userPersona: this.selNode.settings.userPersona, channelBackupPath: this.selNode.settings.channelBackupPath, selCurrencyUnit: this.selNode.settings.currencyUnit, currencyUnits: this.selNode.settings.currencyUnits, fiatConversion: this.selNode.settings.fiatConversion, lnImplementation: this.selNode.lnImplementation, swapServerUrl: this.selNode.settings.swapServerUrl, boltzServerUrl: this.selNode.settings.boltzServerUrl
|
||||
}
|
||||
}));
|
||||
}
|
||||
|
||||
onResetSettings() {
|
||||
const prevIndex = this.selNode.index || -1;
|
||||
this.selNode.settings = this.previousSettings;
|
||||
this.selectedThemeMode = this.themeModes.find((themeMode) => themeMode.id === this.previousSettings.themeMode) || this.themeModes[0];
|
||||
this.selectedThemeColor = this.previousSettings.themeColor;
|
||||
this.store.dispatch(setSelectedNode({ payload: { uiMessage: UI_MESSAGES.NO_SPINNER, prevLnNodeIndex: +prevIndex, currentLnNode: this.selNode, isInitialSetup: true } }));
|
||||
}
|
||||
|
||||
ngOnDestroy() {
|
||||
this.unSubs.forEach((unsub) => {
|
||||
unsub.next();
|
||||
unsub.complete();
|
||||
});
|
||||
}
|
||||
|
||||
}
|
|
@ -62,6 +62,7 @@ import { AppSettingsComponent } from './components/settings/app-settings/app-set
|
|||
import { NodeConfigComponent } from './components/node-config/node-config.component';
|
||||
import { LNPConfigComponent } from './components/node-config/lnp-config/lnp-config.component';
|
||||
import { NodeSettingsComponent } from './components/node-config/node-settings/node-settings.component';
|
||||
import { PageSettingsComponent } from './components/node-config/page-settings/page-settings.component';
|
||||
import { ServicesSettingsComponent } from './components/node-config/services-settings/services-settings.component';
|
||||
import { LoopServiceSettingsComponent } from './components/node-config/services-settings/loop-service-settings/loop-service-settings.component';
|
||||
import { BoltzServiceSettingsComponent } from './components/node-config/services-settings/boltz-service-settings/boltz-service-settings.component';
|
||||
|
@ -248,6 +249,7 @@ export const DEFAULT_DATE_FORMAT: MatDateFormats = {
|
|||
NodeConfigComponent,
|
||||
LNPConfigComponent,
|
||||
NodeSettingsComponent,
|
||||
PageSettingsComponent,
|
||||
ServicesSettingsComponent,
|
||||
LoopServiceSettingsComponent,
|
||||
BoltzServiceSettingsComponent,
|
||||
|
@ -292,6 +294,7 @@ export const DEFAULT_DATE_FORMAT: MatDateFormats = {
|
|||
NodeConfigComponent,
|
||||
LNPConfigComponent,
|
||||
NodeSettingsComponent,
|
||||
PageSettingsComponent,
|
||||
ServicesSettingsComponent,
|
||||
LoopServiceSettingsComponent,
|
||||
BoltzServiceSettingsComponent,
|
||||
|
|
Loading…
Add table
Reference in a new issue