Wide implementation

This commit is contained in:
everoddandeven 2024-10-03 22:35:30 +02:00
parent 131199113d
commit 91b38d1fb4
17 changed files with 795 additions and 595 deletions

View file

@ -122,7 +122,13 @@ function getMonerodVersion(monerodFilePath: string): void {
} }
function startMoneroDaemon(commandOptions: string[]): ChildProcessWithoutNullStreams { function startMoneroDaemon(commandOptions: string[]): ChildProcessWithoutNullStreams {
const monerodPath = getMonerodPath(); //const monerodPath = getMonerodPath();
const monerodPath = commandOptions.shift();
if (!monerodPath) {
win?.webContents.send('monero-sterr', `Invalid monerod path provided: ${monerodPath}`);
throw Error("Invalid monerod path provided");
}
console.log("Starting monerod daemon with options: " + commandOptions.join(" ")); console.log("Starting monerod daemon with options: " + commandOptions.join(" "));

View file

@ -46,21 +46,6 @@ export class AppComponent {
private async load(): Promise<void> { private async load(): Promise<void> {
this.loading = true; this.loading = true;
if (!window.indexedDB) {
console.log("Il tuo browser non supporta indexedDB");
}
else {
console.log("Browser supports IndexedDB");
var request = window.indexedDB.open("dati", 1);
console.log(request);
request.onsuccess = function(event: Event) {
if(event.target instanceof IDBOpenDBRequest) {
console.log(event.target.result)
}
};
}
try { try {
this.daemonRunning = await this.daemonService.isRunning(); this.daemonRunning = await this.daemonService.isRunning();
} }

View file

@ -1,6 +1,6 @@
import { EventEmitter, Injectable } from '@angular/core'; import { EventEmitter, Injectable } from '@angular/core';
import { DaemonService } from './daemon.service'; import { DaemonService } from './daemon.service';
import { BlockCount, BlockHeader, DaemonInfo, SyncInfo } from '../../../../common'; import { BlockCount, BlockHeader, Chain, DaemonInfo, SyncInfo } from '../../../../common';
@Injectable({ @Injectable({
providedIn: 'root' providedIn: 'root'
@ -31,6 +31,9 @@ export class DaemonDataService {
private _lastBlockHeader?: BlockHeader; private _lastBlockHeader?: BlockHeader;
private _gettingLastBlockHeader: boolean = false; private _gettingLastBlockHeader: boolean = false;
private _altChains: Chain[] = [];
private _gettingAltChains: boolean = false;
public readonly syncStart: EventEmitter<void> = new EventEmitter<void>(); public readonly syncStart: EventEmitter<void> = new EventEmitter<void>();
public readonly syncEnd: EventEmitter<void> = new EventEmitter<void>(); public readonly syncEnd: EventEmitter<void> = new EventEmitter<void>();
public readonly syncError: EventEmitter<Error> = new EventEmitter<Error>(); public readonly syncError: EventEmitter<Error> = new EventEmitter<Error>();
@ -115,6 +118,14 @@ export class DaemonDataService {
return this._gettingLastBlockHeader; return this._gettingLastBlockHeader;
} }
public get AltChains(): Chain[] {
return this._altChains;
}
public get gettingAltChains(): boolean {
return this._gettingAltChains;
}
public setRefreshTimeout(ms: number = 5000): void { public setRefreshTimeout(ms: number = 5000): void {
this.refreshTimeoutMs = ms; this.refreshTimeoutMs = ms;
} }
@ -144,6 +155,10 @@ export class DaemonDataService {
return Date.now() - this._lastRefresh <= this.refreshTimeoutMs; return Date.now() - this._lastRefresh <= this.refreshTimeoutMs;
} }
private async getInfo(): Promise<void> {
}
private async refresh(): Promise<void> { private async refresh(): Promise<void> {
if (this.refreshing || this.tooEarlyForRefresh) { if (this.refreshing || this.tooEarlyForRefresh) {
return; return;
@ -157,6 +172,11 @@ export class DaemonDataService {
this._daemonRunning = await this.daemonService.isRunning(); this._daemonRunning = await this.daemonService.isRunning();
this._firstRefresh = false; this._firstRefresh = false;
if (!this._daemonRunning) {
this.syncEnd.emit();
return;
}
this._gettingDaemonInfo = true; this._gettingDaemonInfo = true;
this._daemonInfo = await this.daemonService.getInfo(); this._daemonInfo = await this.daemonService.getInfo();
this._gettingDaemonInfo = false; this._gettingDaemonInfo = false;
@ -179,6 +199,10 @@ export class DaemonDataService {
if (firstRefresh) this._isBlockchainPruned = (await this.daemonService.pruneBlockchain(true)).pruned; if (firstRefresh) this._isBlockchainPruned = (await this.daemonService.pruneBlockchain(true)).pruned;
this._gettingIsBlockchainPruned = false; this._gettingIsBlockchainPruned = false;
this._gettingAltChains = true;
this._altChains = await this.daemonService.getAlternateChains();
this._gettingAltChains = false;
this._lastRefresh = Date.now(); this._lastRefresh = Date.now();
} catch(error) { } catch(error) {
console.error(error); console.error(error);
@ -187,6 +211,7 @@ export class DaemonDataService {
this._gettingBlockCount = false; this._gettingBlockCount = false;
this._gettingLastBlockHeader = false; this._gettingLastBlockHeader = false;
this._gettingIsBlockchainPruned = false; this._gettingIsBlockchainPruned = false;
this._gettingAltChains = false;
this.syncError.emit(<Error>error); this.syncError.emit(<Error>error);

View file

@ -98,6 +98,8 @@ export class DaemonService {
public readonly onDaemonStopStart: EventEmitter<void> = new EventEmitter<void>(); public readonly onDaemonStopStart: EventEmitter<void> = new EventEmitter<void>();
public readonly onDaemonStopEnd: EventEmitter<void> = new EventEmitter<void>(); public readonly onDaemonStopEnd: EventEmitter<void> = new EventEmitter<void>();
private isRunningPromise?: Promise<boolean>;
private readonly headers: { [key: string]: string } = { private readonly headers: { [key: string]: string } = {
"Access-Control-Allow-Headers": "*", // this will allow all CORS requests "Access-Control-Allow-Headers": "*", // this will allow all CORS requests
"Access-Control-Allow-Methods": 'POST,GET' // this states the allowed methods "Access-Control-Allow-Methods": 'POST,GET' // this states the allowed methods
@ -237,13 +239,17 @@ export class DaemonService {
return response; return response;
} }
catch (error) { catch (error) {
if (error instanceof HttpErrorResponse && error.status == 0) { if (error instanceof HttpErrorResponse) {
const wasRunning = this.daemonRunning; if (error.status == 0) {
this.daemonRunning = false; const wasRunning = this.daemonRunning;
this.daemonRunning = false;
if (wasRunning) { if (wasRunning) {
this.onDaemonStatusChanged.emit(false); this.onDaemonStatusChanged.emit(false);
}
} }
throw new Error(error.message);
} }
throw error; throw error;
@ -282,26 +288,37 @@ export class DaemonService {
} }
public async isRunning(force: boolean = false): Promise<boolean> { private async checkDaemonIsRunning(): Promise<boolean> {
try { try {
if (!force && this.daemonRunning != undefined) {
return this.daemonRunning;
}
await this.callRpc(new EmptyRpcRequest()); await this.callRpc(new EmptyRpcRequest());
} }
catch(error) { catch(error) {
if (error instanceof MethodNotFoundError) { if (error instanceof MethodNotFoundError) {
this.daemonRunning = true; return true;
return this.daemonRunning;
} }
console.error(error); console.error(error);
} }
this.daemonRunning = false; return false;
return this.daemonRunning; }
public async isRunning(force: boolean = false): Promise<boolean> {
if (this.isRunningPromise) {
return await this.isRunningPromise;
}
if (!force && this.daemonRunning != undefined) {
return this.daemonRunning;
}
this.isRunningPromise = this.checkDaemonIsRunning();
this.daemonRunning = await this.isRunningPromise;
this.isRunningPromise = undefined;
return this.daemonRunning;
} }
public async getBlock(heightOrHash: number | string, fillPowHash: boolean = false): Promise<Block> { public async getBlock(heightOrHash: number | string, fillPowHash: boolean = false): Promise<Block> {
@ -678,6 +695,10 @@ export class DaemonService {
public async isKeyImageSpent(...keyImages: string[]): Promise<number[]> { public async isKeyImageSpent(...keyImages: string[]): Promise<number[]> {
const response = await this.callRpc(new IsKeyImageSpentRequest(keyImages)); const response = await this.callRpc(new IsKeyImageSpentRequest(keyImages));
if (response.status != 'OK') {
throw new Error(response.status);
}
return response.spent_status; return response.spent_status;
} }

View file

@ -423,6 +423,32 @@
</div> </div>
<div class="tab-pane fade" id="pills-prune-blockchain" role="tabpanel" aria-labelledby="pills-prune-blockchain-tab" tabindex="0">
<div *ngIf="pruneBlockchainError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert">
<h4><i class="bi bi-exclamation-triangle m-2"></i></h4>&nbsp;&nbsp;
<div>
{{pruneBlockchainError}}
</div>
</div>
<div *ngIf="blockchainPruned" class="alert alert-success d-flex align-items-center justify-content-center text-center" role="alert">
<h4><i class="bi bi-check-circle m-2"></i></h4>&nbsp;&nbsp;
<div>
Successfully initiated blockchain pruning
</div>
</div>
<div class="alert alert-info d-flex text-center" role="alert">
<div>
'Pruning' allows node operators to save 2/3 of storage space while keeping the full transaction history </div>
</div>
<hr class="my-4">
<button *ngIf="!pruningBlockchain" class="w-100 btn btn-primary btn-lg" type="button" (click)="pruneBlockchain()">Prune Blockchain</button>
<button *ngIf="pruningBlockchain" class="w-100 btn btn-primary btn-lg" type="button" disabled>Pruning Blockchain ...</button>
</div>
<div class="tab-pane fade" id="pills-save-bc" role="tabpanel" aria-labelledby="pills-save-bc-tab" tabindex="0"> <div class="tab-pane fade" id="pills-save-bc" role="tabpanel" aria-labelledby="pills-save-bc-tab" tabindex="0">
<div *ngIf="saveBlockchainError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert"> <div *ngIf="saveBlockchainError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert">

View file

@ -39,13 +39,18 @@ export class BlockchainComponent implements AfterViewInit {
public saveBlockchainError: string = ''; public saveBlockchainError: string = '';
public blockchainSaved: boolean = false; public blockchainSaved: boolean = false;
public pruningBlockchain: boolean = false;
public pruneBlockchainError: string = '';
public blockchainPruned: boolean = false;
constructor(private daemonService: DaemonService, private ngZone: NgZone) { constructor(private daemonService: DaemonService, private ngZone: NgZone) {
this.navbarLinks = [ this.navbarLinks = [
new NavbarLink('pills-last-block-header-tab', '#pills-last-block-header', 'pills-last-block-header', true, 'Last Block Header', true), new NavbarLink('pills-last-block-header-tab', '#pills-last-block-header', 'pills-last-block-header', true, 'Last Block Header'),
new NavbarLink('pills-get-block-tab', '#pills-get-block', 'pills-get-block', false, 'Get Block', true), new NavbarLink('pills-get-block-tab', '#pills-get-block', 'pills-get-block', false, 'Get Block'),
new NavbarLink('pills-get-block-header-tab', '#pills-get-block-header', 'pills-get-block-header', false, 'Get Block Header', true), new NavbarLink('pills-get-block-header-tab', '#pills-get-block-header', 'pills-get-block-header', false, 'Get Block Header'),
new NavbarLink('pills-pop-blocks-tab', '#pills-pop-blocks', 'pills-pop-blocks', false, 'Pop Blocks', true), new NavbarLink('pills-pop-blocks-tab', '#pills-pop-blocks', 'pills-pop-blocks', false, 'Pop Blocks'),
new NavbarLink('pills-save-bc-tab', '#pills-save-bc', 'pills-save-bc', false, 'Save Blockchain', true) new NavbarLink('pills-prune-blockchain-tab', '#pills-prune-blockchain', 'pills-prune-blockchain', false, 'Prune'),
new NavbarLink('pills-save-bc-tab', '#pills-save-bc', 'pills-save-bc', false, 'Save')
]; ];
this.daemonService.onDaemonStatusChanged.subscribe((running) => { this.daemonService.onDaemonStatusChanged.subscribe((running) => {
@ -148,4 +153,18 @@ export class BlockchainComponent implements AfterViewInit {
this.savingBlockchain = false; this.savingBlockchain = false;
} }
public async pruneBlockchain(): Promise<void> {
this.pruningBlockchain = true;
try {
await this.daemonService.pruneBlockchain(false);
this.blockchainPruned = true;
} catch(error) {
this.pruneBlockchainError = `${error}`;
this.blockchainPruned = false;
}
this.pruningBlockchain = false;
}
} }

View file

@ -34,7 +34,13 @@ export class DetailComponent implements AfterViewInit {
} }
private get targetHeight(): number { private get targetHeight(): number {
return this.daemonData.syncInfo ? this.daemonData.syncInfo.targetHeight : 0; const value = this.daemonData.syncInfo ? this.daemonData.syncInfo.targetHeight : 0;
if (value == 0 && this.height > 0) {
return this.height;
}
return value;
} }
private get nextNeededPruningSeed(): number { private get nextNeededPruningSeed(): number {
@ -90,7 +96,11 @@ export class DetailComponent implements AfterViewInit {
} }
private get syncProgress(): string { private get syncProgress(): string {
return `${(this.height*100/this.targetHeight).toFixed(2)} %`; const targetHeight = this.targetHeight;
const height = this.height;
console.log(`Sync progress, height ${height},targetHeight ${targetHeight}`)
return `${(height*100/targetHeight).toFixed(2)} %`;
} }
//#endregion //#endregion

View file

@ -32,7 +32,137 @@
</table> </table>
</div> </div>
<div class="tab-pane fade" id="pills-outputs-histogram" role="tabpanel" aria-labelledby="pills-outputs-histogram-tab" tabindex="0">
<div *ngIf="getOutDistributionError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert">
<h4><i class="bi bi-exclamation-triangle m-2"></i></h4>&nbsp;&nbsp;
<div>
{{getOutHistogramError}}
</div>
</div>
<div *ngIf="getOutHistogramResult != undefined" class="alert alert-success d-flex align-items-center justify-content-center text-center" role="alert">
</div>
<div class="row g-5 p-2">
<div class="cold-md-7 col-lg-12">
<div class="row gy-3">
<div class="col-sm-12">
<h4 class="mb-3">Get a histogram of output amounts. For all amounts (possibly filtered by parameters), gives the number of outputs on the chain for that amount. RingCT outputs counts as 0 amount.</h4>
<form class="needs-validation" novalidate="">
<div class="row g-3">
<div class="col-12">
<label for="get-out-histogram-amounts" class="form-label">Amounts</label>
<textarea type="text" class="form-control" id="get-out-histogram-amounts" placeholder="[
'4323154546',
'5423442423',
... ,
'2346534525'
]"
rows="15" cols="15" [(ngModel)]="getOutHistogramAmountsJsonString" [ngModelOptions]="{standalone: true}"></textarea>
<small class="text-body-secondary">Array of unsigned int</small>
</div>
</div>
</form>
</div>
<div class="col-sm-6">
<label for="get-out-histogram-min-count" class="form-label">Min count</label>
<input type="number" min="0" class="form-control" id="get-out-histogram-min-count" placeholder="" [(ngModel)]="getOutHistogramMinCount" [ngModelOptions]="{standalone: true}">
<small class="text-body-secondary"></small>
</div>
<div class="col-sm-6">
<label for="get-out-histogram-max-count" class="form-label">Max count</label>
<input type="number" min="0" class="form-control" id="get-out-histogram-max-count" placeholder="" [(ngModel)]="getOutHistogramMaxCount" [ngModelOptions]="{standalone: true}">
<small class="text-body-secondary"></small>
</div>
<div class="form-check form-switch col-md-6">
<label for="get-out-histogram-unlocked" class="form-check-label">Unlocked</label>
<input class="form-control form-check-input" type="checkbox" role="switch" id="get-out-histogram-unlocked" [checked]="getOutHistogramUnlocked" [(ngModel)]="getOutHistogramUnlocked" [ngModelOptions]="{standalone: true}">
<br>
<small class="text-body-secondary"></small>
</div>
<div class="col-sm-6">
<label for="get-out-histogram-recent-cutoff" class="form-label">Recent cutoff</label>
<input type="number" min="0" class="form-control" id="get-out-histogram-recent-cutoff" placeholder="" [(ngModel)]="getOutHistogramRecentCutoff" [ngModelOptions]="{standalone: true}">
<small class="text-body-secondary"></small>
</div>
</div>
</div>
</div>
<button class="w-100 btn btn-primary btn-lg" type="button" [disabled]="!validOutDistributionAmounts" (click)="getOutHistogram()">Get Output Histogram</button>
</div>
<div class="tab-pane fade" id="pills-outputs-distribution" role="tabpanel" aria-labelledby="pills-outputs-distribution-tab" tabindex="0">
<div *ngIf="getOutDistributionError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert">
<h4><i class="bi bi-exclamation-triangle m-2"></i></h4>&nbsp;&nbsp;
<div>
{{getOutDistributionError}}
</div>
</div>
<div *ngIf="getOutDistributionResult != undefined" class="alert alert-success d-flex align-items-center justify-content-center text-center" role="alert">
</div>
<div class="row g-5 p-2">
<div class="cold-md-7 col-lg-12">
<div class="row gy-3">
<div class="col-sm-12">
<h4 class="mb-3">Get Outputs</h4>
<form class="needs-validation" novalidate="">
<div class="row g-3">
<div class="col-12">
<label for="get-out-distribution" class="form-label">Amounts</label>
<textarea type="text" class="form-control" id="get-out-distribution" placeholder="[
'4323154546',
'5423442423',
... ,
'2346534525'
]"
rows="15" cols="15" [(ngModel)]="getOutDistributionAmountsJsonString" [ngModelOptions]="{standalone: true}"></textarea>
<small class="text-body-secondary">Array of unsigned int, amounts to look for</small>
</div>
</div>
</form>
</div>
<div class="col-sm-6">
<label for="get-out-distribution-from-height" class="form-label">From Height</label>
<input type="number" min="0" class="form-control" id="get-out-distribution-from-height" placeholder="" [(ngModel)]="getOutDistributionFromHeight" [ngModelOptions]="{standalone: true}">
<small class="text-body-secondary">Starting height to check from</small>
</div>
<div class="col-sm-6">
<label for="get-out-distribution-to-height" class="form-label">To Height</label>
<input type="number" min="0" class="form-control" id="get-out-distribution-to-height" placeholder="" [(ngModel)]="getOutDistributionToHeight" [ngModelOptions]="{standalone: true}">
<small class="text-body-secondary">Ending height to check up to</small>
</div>
<div class="form-check form-switch col-md-6">
<label for="get-out-distribution-cumulative" class="form-check-label">Cumulative</label>
<input class="form-control form-check-input" type="checkbox" role="switch" id="get-out-distribution-cumulative" [checked]="getOutDistributionCumulative" [(ngModel)]="getOutDistributionCumulative" [ngModelOptions]="{standalone: true}">
<br>
<small class="text-body-secondary">Cumulative result</small>
</div>
</div>
</div>
</div>
<button class="w-100 btn btn-primary btn-lg" type="button" [disabled]="!validOutDistributionAmounts" (click)="getOutDistribution()">Get Out Distribution</button>
</div>
<div class="tab-pane fade" id="pills-is-key-image-spent" role="tabpanel" aria-labelledby="pills-is-key-image-spent-tab" tabindex="0"> <div class="tab-pane fade" id="pills-is-key-image-spent" role="tabpanel" aria-labelledby="pills-is-key-image-spent-tab" tabindex="0">
<div *ngIf="isKeyImageSpentError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert"> <div *ngIf="isKeyImageSpentError != ''" class="alert alert-danger d-flex align-items-center justify-content-center text-center" role="alert">
<h4><i class="bi bi-exclamation-triangle m-2"></i></h4>&nbsp;&nbsp; <h4><i class="bi bi-exclamation-triangle m-2"></i></h4>&nbsp;&nbsp;
<div> <div>
@ -40,6 +170,25 @@
</div> </div>
</div> </div>
<div [hidden]="!isKeyImageSpentResult">
<h4 class="mb-3">Key images spent status</h4>
<table
id="keyImagesTable"
data-toggle="keyImagesTable"
data-toolbar="#toolbar"
>
<thead>
<tr>
<th data-field="keyImage">Key Image</th>
<th data-field="spentStatus">Status</th>
</tr>
</thead>
</table>
<hr class="my-4">
</div>
<div class="row g-5 p-2"> <div class="row g-5 p-2">
<div class="cold-md-7 col-lg-12"> <div class="cold-md-7 col-lg-12">
<div class="row gy-3"> <div class="row gy-3">
@ -47,8 +196,9 @@
<div class="col-sm-12"> <div class="col-sm-12">
<label for="key-images" class="form-label">Key images</label> <label for="key-images" class="form-label">Key images</label>
<textarea [(ngModel)]="keyImages" [ngModelOptions]="{standalone: true}" type="text" class="form-control" id="key-images" placeholder="" <textarea [(ngModel)]="keyImagesJsonString" [ngModelOptions]="{standalone: true}" type="text" class="form-control" id="key-images" placeholder=""
rows="15" cols="15" ></textarea> rows="15" cols="15" ></textarea>
<small class="text-body-secondary">List of key image hex strings to check.</small>
<div class="invalid-feedback"> <div class="invalid-feedback">
Invalid key images. Invalid key images.
</div> </div>

View file

@ -2,7 +2,7 @@ import { AfterViewInit, Component, NgZone } from '@angular/core';
import { NavbarLink } from '../../shared/components/navbar/navbar.model'; import { NavbarLink } from '../../shared/components/navbar/navbar.model';
import { DaemonService } from '../../core/services/daemon/daemon.service'; import { DaemonService } from '../../core/services/daemon/daemon.service';
import { NavbarService } from '../../shared/components/navbar/navbar.service'; import { NavbarService } from '../../shared/components/navbar/navbar.service';
import { Output } from '../../../common'; import { HistogramEntry, Output, OutputDistribution } from '../../../common';
@Component({ @Component({
selector: 'app-outputs', selector: 'app-outputs',
@ -17,11 +17,63 @@ export class OutputsComponent implements AfterViewInit {
public getOutsJsonString: string = ''; public getOutsJsonString: string = '';
public getOutsGetTxId: boolean = false; public getOutsGetTxId: boolean = false;
public keyImages: string = ''; public keyImagesJsonString: string = '';
public isKeyImageSpentError: string = ''; public isKeyImageSpentError: string = '';
public isKeyImageSpentResult?: boolean; public isKeyImageSpentResult?: { keyImage: string, spentStatus: string }[];
public gettingKeyImages: boolean = false; public gettingKeyImages: boolean = false;
public get validKeyImages(): boolean {
try {
const keyImages: string[] = JSON.parse(this.keyImagesJsonString);
if (!Array.isArray(keyImages)) {
return false;
}
keyImages.forEach((keyImage: string) => {
if (typeof keyImage != 'string') {
throw new Error();
}
});
return true;
} catch(error) {
return false;
}
}
public get keyImages(): string[] {
if (!this.validKeyImages) {
return [];
}
return JSON.parse(this.keyImagesJsonString);
}
public getOutHistogramAmountsJsonString: string = '';
public getOutHistogramMinCount: number = 0;
public getOutHistogramMaxCount: number = 0;
public getOutHistogramUnlocked: boolean = false;
public getOutHistogramRecentCutoff: number = 0;
public getOutHistogramResult?: HistogramEntry[];
public getOutHistogramError: string = '';
public gettingOutHistogram: boolean = false;
public getOutDistributionAmountsJsonString: string = '';
public getOutDistributionFromHeight: number = 0;
public getOutDistributionToHeight: number = 0;
public getOutDistributionCumulative: boolean = false;
public getOutDistributionResult?: OutputDistribution[];
public getOutDistributionError: string = '';
public get getOutDistributionAmounts(): number[] {
if (!this.validOutDistributionAmounts) {
return [];
}
return JSON.parse(this.getOutDistributionAmountsJsonString);
}
constructor(private daemonService: DaemonService, private navbarService: NavbarService, private ngZone: NgZone) { constructor(private daemonService: DaemonService, private navbarService: NavbarService, private ngZone: NgZone) {
this.navbarLinks = [ this.navbarLinks = [
new NavbarLink('pills-outputs-overview-tab', '#pills-outputs-overview', 'outputs-overview', true, 'Overview'), new NavbarLink('pills-outputs-overview-tab', '#pills-outputs-overview', 'outputs-overview', true, 'Overview'),
@ -67,7 +119,7 @@ export class OutputsComponent implements AfterViewInit {
} }
} }
public validOuts(): boolean { public get validOuts(): boolean {
try { try {
const _outs: any[] = JSON.parse(this.getOutsJsonString); const _outs: any[] = JSON.parse(this.getOutsJsonString);
@ -93,11 +145,80 @@ export class OutputsComponent implements AfterViewInit {
} }
public async isKeyImageSpent(): Promise<void> { public get validOutDistributionAmounts(): boolean {
try {
const amounts: number[] = JSON.parse(this.getOutDistributionAmountsJsonString);
if(!Array.isArray(amounts)) {
return false;
}
amounts.forEach((amount) => {
if (typeof amount != 'number' || amount <= 0) throw new Error("");
})
return true;
}
catch(error) {
return false;
}
}
public async getOutDistribution(): Promise<void> {
try
{
const amounts = this.getOutDistributionAmounts;
const cumulative = this.getOutDistributionCumulative;
const fromHeight = this.getOutDistributionFromHeight;
const toHeight = this.getOutDistributionToHeight;
this.getOutDistributionResult = await this.daemonService.getOutputDistribution(amounts, cumulative, fromHeight, toHeight);
this.getOutDistributionError = '';
}
catch(error) {
this.getOutDistributionError = `${error}`;
}
}
public async getOutHistogram(): Promise<void> {
}
public async isKeyImageSpent(): Promise<void> {
this.gettingKeyImages = true;
try {
const keyImages: string[] = this.keyImages;
const spentList: number[] = await this.daemonService.isKeyImageSpent(...keyImages);
if (keyImages.length != spentList.length) {
throw new Error("Invalid spent list size response");
}
this.isKeyImageSpentResult = [];
for(let i = 0; i < keyImages.length; i++) {
const ki = keyImages[i];
const spentStatus = spentList[i];
this.isKeyImageSpentResult.push({
keyImage: ki,
spentStatus: spentStatus == 0 ? 'unspent' : spentStatus == 1 ? 'spent in blockchain' : spentStatus == 2 ? 'spent in tx pool' : 'unknown'
})
const $table = $('#keyImagesTable');
$table.bootstrapTable({});
$table.bootstrapTable('load', this.isKeyImageSpentResult);
this.isKeyImageSpentError = '';
}
} catch(error) {
this.isKeyImageSpentError = `${error}`;
this.isKeyImageSpentResult = undefined;
}
this.gettingKeyImages = false;
} }
public async load() { public async load() {
await this.getOuts();
} }
} }

View file

@ -1,20 +1,48 @@
<div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom"> <div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
<h1 class="h2">Settings</h1> <h1 class="h2">Settings</h1>
<div class="btn-toolbar mb-2 mb-md-0"> <div class="btn-toolbar mb-2 mb-md-0">
<ul class="nav nav-pills m-3" id="pills-tab" role="tablist"> <div class="btn-group me-2">
@for(navbarLink of navbarLinks; track navbarLink.name) { <ul class="nav nav-pills m-3" id="pills-tab" role="tablist">
<li class="nav-item mr-2" role="presentation"> @for(navbarLink of navbarLinks; track navbarLink.name) {
<button [class]="navbarLink.selected ? 'nav-link active btn-sm' : 'nav-link btn-sm'" [id]="navbarLink.id" data-bs-toggle="pill" [attr.data-bs-target]="navbarLink.target" type="button" role="tab" [attr.aria-controls]="navbarLink.controls" [attr.aria-selected]="navbarLink.selected" [disabled]="navbarLink.disabled">{{navbarLink.name}}</button> <li class="nav-item mr-2" role="presentation">
</li> <button [class]="navbarLink.selected ? 'nav-link active btn-sm' : 'nav-link btn-sm'" [id]="navbarLink.id" data-bs-toggle="pill" [attr.data-bs-target]="navbarLink.target" type="button" role="tab" [attr.aria-controls]="navbarLink.controls" [attr.aria-selected]="navbarLink.selected" [disabled]="navbarLink.disabled">{{navbarLink.name}}</button>
} </li>
</ul> }
</ul>
</div>
</div> </div>
</div> </div>
<div *ngIf="!loading" class="tab-content" id="pills-settings-tabContent"> <div *ngIf="!loading" class="tab-content" id="pills-settings-tabContent">
<div class="tab-pane fade show active" id="pills-rpc" role="tabpanel" aria-labelledby="pills-rpc-tab" tabindex="0">
<div class="tab-pane fade show active" id="pills-general" role="tabpanel" aria-labelledby="pills-general-tab" tabindex="0">
<div class="row g-5 p-2">
<div class="col-md-7 col-lg-10">
<h4 class="mb-3">Node</h4>
<div class="row gy-3">
<div class="col-md-12">
<label for="general-monerod-path" class="form-label">Monerod path</label>
<input type="file" class="form-control" id="general-monerod-path" [(ngModel)]="currentSettings.monerodPath" [ngModelOptions]="{standalone: true}" (change)="onMonerodPathChange()" placeholder="AAA">
<small class="text-body-secondary">Path to monerod executable</small>
</div>
</div>
<br>
<div class="row gy-3">
<div class="col-md-12">
<label for="general-xmrig-path" class="form-label">XMRig path</label>
<input type="file" class="form-control" id="general-xmrig-path" [(ngModel)]="currentSettings.monerodPath" [ngModelOptions]="{standalone: true}" (change)="onMonerodPathChange()" placeholder="AAA">
<small class="text-body-secondary">Path to XMRig executable</small>
</div>
</div>
</div>
</div>
</div>
<div class="tab-pane fade" id="pills-rpc" role="tabpanel" aria-labelledby="pills-rpc-tab" tabindex="0">
<div class="row g-5 m-2"> <div class="row g-5 m-2">
<div class="col-md-7 col-lg-10"> <div class="col-md-7 col-lg-10">
<h4 class="mb-3">General</h4> <h4 class="mb-3">General</h4>

View file

@ -1,9 +1,8 @@
import { AfterViewInit, Component } from '@angular/core'; import { AfterViewInit, Component } from '@angular/core';
import { NavbarService } from '../../shared/components/navbar/navbar.service'; import { NavbarService } from '../../shared/components/navbar/navbar.service';
import { NavigationEnd, NavigationStart, Router } from '@angular/router'; import { Router } from '@angular/router';
import { NavbarLink } from '../../shared/components/navbar/navbar.model'; import { NavbarLink } from '../../shared/components/navbar/navbar.model';
import { DaemonSettings } from '../../../common/DaemonSettings'; import { DaemonSettings } from '../../../common/DaemonSettings';
import { FormsModule, NgModel } from '@angular/forms';
import { DaemonService } from '../../core/services/daemon/daemon.service'; import { DaemonService } from '../../core/services/daemon/daemon.service';
@Component({ @Component({
@ -27,7 +26,8 @@ export class SettingsComponent implements AfterViewInit {
this.loading = true; this.loading = true;
this.navbarLinks = [ this.navbarLinks = [
new NavbarLink('pills-rpc-tab', '#pills-rpc', 'pills-rpc', true, 'RPC'), new NavbarLink('pills-general-tab', '#pills-general', 'pills-general', true, 'General'),
new NavbarLink('pills-rpc-tab', '#pills-rpc', 'pills-rpc', false, 'RPC'),
new NavbarLink('pills-p2p-tab', '#pills-p2p', 'pills-p2p', false, 'P2P'), new NavbarLink('pills-p2p-tab', '#pills-p2p', 'pills-p2p', false, 'P2P'),
new NavbarLink('pills-blockchain-tab', '#pills-blockchain', 'pills-blockchain', false, 'Blockchain'), new NavbarLink('pills-blockchain-tab', '#pills-blockchain', 'pills-blockchain', false, 'Blockchain'),
new NavbarLink('pills-mining-tab', '#pills-mining', 'pills-mining', false, 'Mining'), new NavbarLink('pills-mining-tab', '#pills-mining', 'pills-mining', false, 'Mining'),
@ -46,13 +46,6 @@ export class SettingsComponent implements AfterViewInit {
this.rpcLoginPassword = ''; this.rpcLoginPassword = '';
} }
this.router.events.subscribe((event) => {
if (event instanceof NavigationEnd) {
if (event.url != '/settings') return;
this.onNavigationEnd();
}
});
this.load(); this.load();
} }
@ -70,12 +63,12 @@ export class SettingsComponent implements AfterViewInit {
return true; return true;
} }
return false; return false;
} }
ngAfterViewInit(): void { ngAfterViewInit(): void {
this.navbarService.setLinks(this.navbarLinks); this.navbarService.setLinks(this.navbarLinks);
this.navbarService.enableLinks();
} }
public OnOfflineChange() { public OnOfflineChange() {
@ -136,8 +129,13 @@ export class SettingsComponent implements AfterViewInit {
} }
} }
private onNavigationEnd(): void { public onMonerodPathChange(): void {
if (document) {
const element = <HTMLInputElement>document.getElementById('general-monerod-path');
if (element.files) {
this.currentSettings.monerodPath = element.files[0].path;
}
}
} }
public async OnSave(): Promise<void> { public async OnSave(): Promise<void> {
@ -150,220 +148,3 @@ export class SettingsComponent implements AfterViewInit {
this.originalSettings = this.currentSettings.clone(); this.originalSettings = this.currentSettings.clone();
} }
} }
/**
* --log-file arg (=/home/sidney/.bitmonero/bitmonero.log, /home/sidney/.bitmonero/testnet/bitmonero.log if 'testnet', /home/sidney/.bitmonero/stagenet/bitmonero.log if 'stagenet')
Specify log file
--log-level arg
--max-log-file-size arg (=104850000) Specify maximum log file size [B]
--max-log-files arg (=50) Specify maximum number of rotated log
files to be saved (no limit by setting
to 0)
--max-concurrency arg (=0) Max number of threads to use for a
parallel job
--proxy arg Network communication through proxy:
<socks-ip:port> i.e. "127.0.0.1:9050"
--proxy-allow-dns-leaks Allow DNS leaks outside of proxy
--public-node Allow other users to use the node as a
remote (restricted RPC mode, view-only
commands) and advertise it over P2P
--zmq-rpc-bind-ip arg (=127.0.0.1) IP for ZMQ RPC server to listen on
--zmq-rpc-bind-port arg (=18082, 28082 if 'testnet', 38082 if 'stagenet')
Port for ZMQ RPC server to listen on
--zmq-pub arg Address for ZMQ pub - tcp://ip:port or
ipc://path
--no-zmq Disable ZMQ RPC server
--data-dir arg (=/home/sidney/.bitmonero, /home/sidney/.bitmonero/testnet if 'testnet', /home/sidney/.bitmonero/stagenet if 'stagenet')
Specify data directory
--test-drop-download For net tests: in download, discard ALL
blocks instead checking/saving them
(very fast)
--test-drop-download-height arg (=0) Like test-drop-download but discards
only after around certain height
--testnet Run on testnet. The wallet must be
launched with --testnet flag.
--stagenet Run on stagenet. The wallet must be
launched with --stagenet flag.
--regtest Run in a regression testing mode.
--keep-fakechain Don't delete any existing database when
in fakechain mode.
--fixed-difficulty arg (=0) Fixed difficulty used for testing.
--enforce-dns-checkpointing checkpoints from DNS server will be
enforced
--prep-blocks-threads arg (=4) Max number of threads to use when
preparing block hashes in groups.
--fast-block-sync arg (=1) Sync up most of the way by using
embedded, known block hashes.
--show-time-stats arg (=0) Show time-stats when processing
blocks/txs and disk synchronization.
--block-sync-size arg (=0) How many blocks to sync at once during
chain synchronization (0 = adaptive).
--check-updates arg (=notify) Check for new versions of monero:
[disabled|notify|download|update]
--fluffy-blocks Relay blocks as fluffy blocks
(obsolete, now default)
--no-fluffy-blocks Relay blocks as normal blocks
--test-dbg-lock-sleep arg (=0) Sleep time in ms, defaults to 0 (off),
used to debug before/after locking
mutex. Values 100 to 1000 are good for
tests.
--offline Do not listen for peers, nor connect to
any
--disable-dns-checkpoints Do not retrieve checkpoints from DNS
--block-download-max-size arg (=0) Set maximum size of block download
queue in bytes (0 for default)
--sync-pruned-blocks Allow syncing from nodes with only
pruned blocks
--max-txpool-weight arg (=648000000) Set maximum txpool weight in bytes.
--block-notify arg Run a program for each new block, '%s'
will be replaced by the block hash
--prune-blockchain Prune blockchain
--reorg-notify arg Run a program for each reorg, '%s' will
be replaced by the split height, '%h'
will be replaced by the new blockchain
height, '%n' will be replaced by the
number of new blocks in the new chain,
and '%d' will be replaced by the number
of blocks discarded from the old chain
--block-rate-notify arg Run a program when the block rate
undergoes large fluctuations. This
might be a sign of large amounts of
hash rate going on and off the Monero
network, and thus be of potential
interest in predicting attacks. %t will
be replaced by the number of minutes
for the observation window, %b by the
number of blocks observed within that
window, and %e by the number of blocks
that was expected in that window. It is
suggested that this notification is
used to automatically increase the
number of confirmations required before
a payment is acted upon.
--keep-alt-blocks Keep alternative blocks on restart
--extra-messages-file arg Specify file for extra messages to
include into coinbase transactions
--start-mining arg Specify wallet address to mining for
--mining-threads arg Specify mining threads count
--bg-mining-enable enable background mining
--bg-mining-ignore-battery if true, assumes plugged in when unable
to query system power status
--bg-mining-min-idle-interval arg Specify min lookback interval in
seconds for determining idle state
--bg-mining-idle-threshold arg Specify minimum avg idle percentage
over lookback interval
--bg-mining-miner-target arg Specify maximum percentage cpu use by
miner(s)
--db-sync-mode arg (=fast:async:250000000bytes)
Specify sync option, using format
[safe|fast|fastest]:[sync|async]:[<nblo
cks_per_sync>[blocks]|<nbytes_per_sync>
[bytes]].
--db-salvage Try to salvage a blockchain database if
it seems corrupted
--p2p-bind-ip arg (=0.0.0.0) Interface for p2p network protocol
(IPv4)
--p2p-bind-ipv6-address arg (=::) Interface for p2p network protocol
(IPv6)
--p2p-bind-port arg (=18080, 28080 if 'testnet', 38080 if 'stagenet')
Port for p2p network protocol (IPv4)
--p2p-bind-port-ipv6 arg (=18080, 28080 if 'testnet', 38080 if 'stagenet')
Port for p2p network protocol (IPv6)
--p2p-use-ipv6 Enable IPv6 for p2p
--p2p-ignore-ipv4 Ignore unsuccessful IPv4 bind for p2p
--p2p-external-port arg (=0) External port for p2p network protocol
(if port forwarding used with NAT)
--allow-local-ip Allow local ip add to peer list, mostly
in debug purposes
--add-peer arg Manually add peer to local peerlist
--add-priority-node arg Specify list of peers to connect to and
attempt to keep the connection open
--add-exclusive-node arg Specify list of peers to connect to
only. If this option is given the
options add-priority-node and seed-node
are ignored
--seed-node arg Connect to a node to retrieve peer
addresses, and disconnect
--tx-proxy arg Send local txes through proxy:
<network-type>,<socks-ip:port>[,max_con
nections][,disable_noise] i.e.
"tor,127.0.0.1:9050,100,disable_noise"
--anonymous-inbound arg <hidden-service-address>,<[bind-ip:]por
t>[,max_connections] i.e.
"x.onion,127.0.0.1:18083,100"
--ban-list arg Specify ban list file, one IP address
per line
--hide-my-port Do not announce yourself as peerlist
candidate
--no-sync Don't synchronize the blockchain with
other peers
--enable-dns-blocklist Apply realtime blocklist from DNS
--no-igd Disable UPnP port mapping
--igd arg (=delayed) UPnP port mapping (disabled, enabled,
delayed)
--out-peers arg (=-1) set max number of out peers
--in-peers arg (=-1) set max number of in peers
--tos-flag arg (=-1) set TOS flag
--limit-rate-up arg (=2048) set limit-rate-up [kB/s]
--limit-rate-down arg (=8192) set limit-rate-down [kB/s]
--limit-rate arg (=-1) set limit-rate [kB/s]
--pad-transactions Pad relayed transactions to help defend
against traffic volume analysis
--max-connections-per-ip arg (=1) Maximum number of connections allowed
from the same IP address
--rpc-bind-port arg (=18081, 28081 if 'testnet', 38081 if 'stagenet')
Port for RPC server
--rpc-restricted-bind-port arg Port for restricted RPC server
--restricted-rpc Restrict RPC to view only commands and
do not return privacy sensitive data in
RPC calls
--bootstrap-daemon-address arg URL of a 'bootstrap' remote daemon that
the connected wallets can use while
this daemon is still not fully synced.
Use 'auto' to enable automatic public
nodes discovering and bootstrap daemon
switching
--bootstrap-daemon-login arg Specify username:password for the
bootstrap daemon login
--bootstrap-daemon-proxy arg <ip>:<port> socks proxy to use for
bootstrap daemon connections
--rpc-bind-ip arg (=127.0.0.1) Specify IP to bind RPC server
--rpc-bind-ipv6-address arg (=::1) Specify IPv6 address to bind RPC server
--rpc-restricted-bind-ip arg (=127.0.0.1)
Specify IP to bind restricted RPC
server
--rpc-restricted-bind-ipv6-address arg (=::1)
Specify IPv6 address to bind restricted
RPC server
--rpc-use-ipv6 Allow IPv6 for RPC
--rpc-ignore-ipv4 Ignore unsuccessful IPv4 bind for RPC
--rpc-login arg Specify username[:password] required
for RPC server
--confirm-external-bind Confirm rpc-bind-ip value is NOT a
loopback (local) IP
--rpc-access-control-origins arg Specify a comma separated list of
origins to allow cross origin resource
sharing
--rpc-ssl arg (=autodetect) Enable SSL on RPC connections:
enabled|disabled|autodetect
--rpc-ssl-private-key arg Path to a PEM format private key
--rpc-ssl-certificate arg Path to a PEM format certificate
--rpc-ssl-ca-certificates arg Path to file containing concatenated
PEM format certificate(s) to replace
system CA(s).
--rpc-ssl-allowed-fingerprints arg List of certificate fingerprints to
allow
--rpc-ssl-allow-chained Allow user (via --rpc-ssl-certificates)
chain certificates
--disable-rpc-ban Do not ban hosts on RPC errors
--rpc-ssl-allow-any-cert Allow any peer certificate
--rpc-payment-address arg Restrict RPC to clients sending
micropayment to this address
--rpc-payment-difficulty arg (=1000) Restrict RPC to clients sending
micropayment at this difficulty
--rpc-payment-credits arg (=100) Restrict RPC to clients sending
micropayment, yields that many credits
per payment
--rpc-payment-allow-free-loopback Allow free access from the loopback
address (ie, the local host)
*/

View file

@ -1,11 +1,13 @@
<div *ngIf="!daemonRunning" class="h-100 p-5 text-bg-dark rounded-3 m-4 text-center"> <div *ngIf="!daemonRunning" class="h-100 p-5 text-bg-dark rounded-3 m-4 text-center">
<h2 *ngIf="!startingDaemon"><i class="bi bi-exclamation-diamond m-4"></i> Daemon not running</h2> <h2 *ngIf="!startingDaemon && daemonConfigured"><i class="bi bi-exclamation-diamond m-4"></i> Daemon not running</h2>
<p *ngIf="!startingDaemon">Start monero daemon</p> <h2 *ngIf="!startingDaemon && !daemonConfigured"><i class="bi bi-exclamation-diamond m-4"></i> Daemon not configured</h2>
<p *ngIf="!startingDaemon && daemonConfigured">Start monero daemon</p>
<p *ngIf="!startingDaemon && !daemonConfigured">Configure monero daemon</p>
<h2 *ngIf="startingDaemon"><i class="bi bi-play-fill m-4"></i> Daemon is starting</h2> <h2 *ngIf="startingDaemon"><i class="bi bi-play-fill m-4"></i> Daemon is starting</h2>
<p *ngIf="startingDaemon">Starting monero daemon</p> <p *ngIf="startingDaemon">Starting monero daemon</p>
<button *ngIf="!startingDaemon" class="btn btn-outline-light" type="button" (click)="startDaemon()"><i class="bi bi-play-fill"></i> Start</button> <button *ngIf="!startingDaemon && daemonConfigured" class="btn btn-outline-light" type="button" (click)="startDaemon()"><i class="bi bi-play-fill"></i> Start</button>
<button *ngIf="startingDaemon" class="btn btn-outline-light" type="button" disabled> <button *ngIf="startingDaemon" class="btn btn-outline-light" type="button" disabled>
<span class="spinner-border spinner-border-sm" role="status" aria-hidden="true"></span> <span class="spinner-border spinner-border-sm" role="status" aria-hidden="true"></span>
Starting monerod Starting monerod

View file

@ -9,6 +9,7 @@ import { DaemonService } from '../../../core/services/daemon/daemon.service';
export class DaemonNotRunningComponent { export class DaemonNotRunningComponent {
public daemonRunning: boolean = false; public daemonRunning: boolean = false;
public daemonConfigured: boolean = false;
public get startingDaemon(): boolean { public get startingDaemon(): boolean {
return this.daemonService.starting; return this.daemonService.starting;
@ -24,6 +25,9 @@ export class DaemonNotRunningComponent {
}); });
this.daemonService.isRunning().then((running: boolean) => { this.daemonService.isRunning().then((running: boolean) => {
this.ngZone.run(() => this.daemonRunning = running); this.ngZone.run(() => this.daemonRunning = running);
});
this.daemonService.getSettings().then((settings) => {
this.daemonConfigured = settings.monerodPath != '';
}) })
} }

View file

@ -18,6 +18,12 @@ export class NavbarService {
this.daemonRunning = running; this.daemonRunning = running;
if (!running) this.disableLinks(); if (!running) this.disableLinks();
if (running) this.enableLinks(); if (running) this.enableLinks();
});
this.daemonService.isRunning().then((running: boolean) => {
this.daemonRunning = running;
if (!running) this.disableLinks();
if (running) this.enableLinks();
}) })
} }
@ -44,7 +50,10 @@ export class NavbarService {
} }
public enableLinks(): void { public enableLinks(): void {
this._navbarLinks.forEach((link) => link.disabled = false); this._navbarLinks.forEach((link, index: number) => {
link.disabled = false
link.selected = index == 0;
});
} }
} }

View file

@ -1,64 +1,73 @@
body { body {
min-height: 100vh; min-height: 100vh;
min-height: -webkit-fill-available; min-height: -webkit-fill-available;
} }
html { html {
height: -webkit-fill-available; height: -webkit-fill-available;
} }
main { main {
height: 100vh; height: 100vh;
height: -webkit-fill-available; height: -webkit-fill-available;
max-height: 100vh; max-height: 100vh;
overflow-x: auto; overflow-x: auto;
overflow-y: hidden; overflow-y: hidden;
} }
.dropdown-toggle { outline: 0; } .dropdown-toggle { outline: 0; }
.btn-toggle { .btn-toggle {
padding: .25rem .5rem; padding: .25rem .5rem;
font-weight: 600; font-weight: 600;
color: var(--bs-emphasis-color); color: var(--bs-emphasis-color);
background-color: transparent; background-color: transparent;
} }
.btn-toggle:hover, .btn-toggle:hover,
.btn-toggle:focus { .btn-toggle:focus {
color: rgba(var(--bs-emphasis-color-rgb), .85); color: rgba(var(--bs-emphasis-color-rgb), .85);
background-color: var(--bs-tertiary-bg); background-color: var(--bs-tertiary-bg);
} }
.btn-toggle::before { .btn-toggle::before {
width: 1.25em; width: 1.25em;
line-height: 0; line-height: 0;
content: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='rgba%280,0,0,.5%29' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M5 14l6-6-6-6'/%3e%3c/svg%3e"); content: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='rgba%280,0,0,.5%29' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M5 14l6-6-6-6'/%3e%3c/svg%3e");
transition: transform .35s ease; transition: transform .35s ease;
transform-origin: .5em 50%; transform-origin: .5em 50%;
} }
[data-bs-theme="dark"] .btn-toggle::before { [data-bs-theme="dark"] .btn-toggle::before {
content: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='rgba%28255,255,255,.5%29' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M5 14l6-6-6-6'/%3e%3c/svg%3e"); content: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='rgba%28255,255,255,.5%29' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M5 14l6-6-6-6'/%3e%3c/svg%3e");
} }
.btn-toggle[aria-expanded="true"] { .btn-toggle[aria-expanded="true"] {
color: rgba(var(--bs-emphasis-color-rgb), .85); color: rgba(var(--bs-emphasis-color-rgb), .85);
} }
.btn-toggle[aria-expanded="true"]::before { .btn-toggle[aria-expanded="true"]::before {
transform: rotate(90deg); transform: rotate(90deg);
} }
.btn-toggle-nav a { .btn-toggle-nav a {
padding: .1875rem .5rem; padding: .1875rem .5rem;
margin-top: .125rem; margin-top: .125rem;
margin-left: 1.25rem; margin-left: 1.25rem;
} }
.btn-toggle-nav a:hover, .btn-toggle-nav a:hover,
.btn-toggle-nav a:focus { .btn-toggle-nav a:focus {
background-color: var(--bs-tertiary-bg); background-color: var(--bs-tertiary-bg);
} }
.scrollarea { .scrollarea {
overflow-y: auto; overflow-y: auto;
} }
.icon-xr {
display: inline-block;
width: 16px;
height: 16px;
background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" viewBox="0 0 16 16"><text x="0" y="12" font-family="Arial, sans-serif" font-size="12" fill="%23ff9A85" font-weight="bold">Xr</text></svg>');
background-size: contain;
background-repeat: no-repeat;
background-position: center;
}

View file

@ -37,6 +37,7 @@ export class SidebarComponent implements OnChanges {
new NavLink('Outputs', '/outputs', 'bi bi-circle-fill'), new NavLink('Outputs', '/outputs', 'bi bi-circle-fill'),
new NavLink('Mining', '/mining', 'bi bi-minecart-loaded'), new NavLink('Mining', '/mining', 'bi bi-minecart-loaded'),
new NavLink('Hard Fork Info', '/hardforkinfo', 'bi bi-signpost-split'), new NavLink('Hard Fork Info', '/hardforkinfo', 'bi bi-signpost-split'),
new NavLink('XMRig', '/xmrig', 'icon-xr text-primary'),
new NavLink('Bans', '/bans', 'bi bi-ban', 'bottom'), new NavLink('Bans', '/bans', 'bi bi-ban', 'bottom'),
new NavLink('Logs', '/logs', 'bi bi-terminal', 'bottom'), new NavLink('Logs', '/logs', 'bi bi-terminal', 'bottom'),
new NavLink('Version', '/version', 'bi bi-git', 'bottom'), new NavLink('Version', '/version', 'bi bi-git', 'bottom'),

View file

@ -1,249 +1,252 @@
export class DaemonSettings { export class DaemonSettings {
public logFile: string = ''; public monerodPath: string = '';
public logLevel: number = 0;
public maxLogFileSize: number = 104850000;
public maxLogFiles: number = 50;
public maxConcurrency: number = 0; public logFile: string = '';
public proxy: string = ''; public logLevel: number = 0;
public proxyAllowDnsLeaks: boolean = false; public maxLogFileSize: number = 104850000;
public publicNode: boolean = false; public maxLogFiles: number = 50;
public zmqRpcBindIp: string = ''; public maxConcurrency: number = 0;
public zmqRpcBindPort: number = 0; public proxy: string = '';
public zmqPub: string = ''; public proxyAllowDnsLeaks: boolean = false;
public noZmq: boolean = false; public publicNode: boolean = false;
public testDropDownload: boolean = false; public zmqRpcBindIp: string = '';
public testDropDownloadHeight: number = 0; public zmqRpcBindPort: number = 0;
public zmqPub: string = '';
public noZmq: boolean = false;
public testDbgLockSleep: number = 0; public testDropDownload: boolean = false;
public testDropDownloadHeight: number = 0;
public testnet: boolean = false; public testDbgLockSleep: number = 0;
public mainnet: boolean = true;
public stagenet: boolean = false;
public regtest: boolean = false; public testnet: boolean = false;
public mainnet: boolean = true;
public stagenet: boolean = false;
public keepFakeChain: boolean = false; public regtest: boolean = false;
public fixedDifficulty: number = 0;
public enforceDnsCheckpoint: boolean = false;
public prepBlocksThreads: number = 0;
public fastBlockSync: boolean = false;
public showTimeStats: boolean = false;
public blockSyncSize: number = 0;
public checkUpdates: 'disabled' | 'notify' | 'download' | 'update' = 'notify';
public fluffyBlocks: boolean = true;
public noFluffyBlocks: boolean = false;
public offline: boolean = false; public keepFakeChain: boolean = false;
public disableDnsCheckpoints: boolean = false; public fixedDifficulty: number = 0;
public enforceDnsCheckpoint: boolean = false;
public prepBlocksThreads: number = 0;
public fastBlockSync: boolean = false;
public showTimeStats: boolean = false;
public blockSyncSize: number = 0;
public checkUpdates: 'disabled' | 'notify' | 'download' | 'update' = 'notify';
public fluffyBlocks: boolean = true;
public noFluffyBlocks: boolean = false;
public blockDownloadMaxSize: number = 0; public offline: boolean = false;
public syncPrunedBlocks: boolean = false; public disableDnsCheckpoints: boolean = false;
public maxTxPoolWeight: number = 648000000;
public blockNotify: string = '';
public pruneBlockchain: boolean = false;
public reorgNotify: string = ''; public blockDownloadMaxSize: number = 0;
public blockRateNotify: string = ''; public syncPrunedBlocks: boolean = false;
public maxTxPoolWeight: number = 648000000;
public blockNotify: string = '';
public pruneBlockchain: boolean = false;
public keepAltBlocks: boolean = false; public reorgNotify: string = '';
public extraMessagesFile: string = ''; public blockRateNotify: string = '';
public startMining: string = ''; public keepAltBlocks: boolean = false;
public miningThreds: number = 0; public extraMessagesFile: string = '';
public bgMiningEnable: boolean = false; public startMining: string = '';
public bgMiningIgnoreBattery: boolean = false; public miningThreds: number = 0;
public bgMiningMinIdleInterval: number = 0;
public bgMiningIdleThreshold: number = 0;
public bgMiningMinerTarget: number = 0;
public dbSyncMode: string = 'fast:async:250000000bytes'; public bgMiningEnable: boolean = false;
public dbSalvage: boolean = false; public bgMiningIgnoreBattery: boolean = false;
public bgMiningMinIdleInterval: number = 0;
public bgMiningIdleThreshold: number = 0;
public bgMiningMinerTarget: number = 0;
public p2pBindIp: string = '0.0.0.0'; public dbSyncMode: string = 'fast:async:250000000bytes';
public p2pBindIpv6Address: string = "::"; public dbSalvage: boolean = false;
public p2pBindPort: number = 0;
public p2pBindPortIpv6: number = 0;
public p2pUseIpv6: boolean = false;
public p2pIgnoreIpv4: boolean = false;
public p2pExternalPort: number = 0;
public allowLocalIp: boolean = false;
public addPeer: string = '';
public addPriorityNode: string = '';
public addExclusiveNode: string = '';
public seedNode: string = ''; public p2pBindIp: string = '0.0.0.0';
public txProxy: string = ''; public p2pBindIpv6Address: string = "::";
public anonymousInbound: string = ''; public p2pBindPort: number = 0;
public p2pBindPortIpv6: number = 0;
public p2pUseIpv6: boolean = false;
public p2pIgnoreIpv4: boolean = false;
public p2pExternalPort: number = 0;
public allowLocalIp: boolean = false;
public addPeer: string = '';
public addPriorityNode: string = '';
public addExclusiveNode: string = '';
public banList: string = ''; public seedNode: string = '';
public hideMyPort: boolean = false; public txProxy: string = '';
public anonymousInbound: string = '';
public noSync: boolean = false; public banList: string = '';
public hideMyPort: boolean = false;
public enableDnsBlocklist: boolean = false; public noSync: boolean = false;
public noIgd: boolean = false;
public igd: 'disable' | 'enabled' | 'delayed' = 'delayed';
public outPeers: number = -1;
public inPeers: number = -1;
public tosFlag: number = -1;
public limitRateUp: number = 2048; public enableDnsBlocklist: boolean = false;
public limitRateDown: number = 8192; public noIgd: boolean = false;
public limitRate: number = -1; public igd: 'disable' | 'enabled' | 'delayed' = 'delayed';
public outPeers: number = -1;
public inPeers: number = -1;
public tosFlag: number = -1;
public padTransactions: boolean = false; public limitRateUp: number = 2048;
public maxConnectionsPerIp: number = 1; public limitRateDown: number = 8192;
public limitRate: number = -1;
public rpcBindPort: number = 0; public padTransactions: boolean = false;
public restrictedBindPort: number = 0; public maxConnectionsPerIp: number = 1;
public restrictedRpc: boolean = false;
public bootstrapDaemonAddress: string = ''; public rpcBindPort: number = 0;
public bootstrapDaemonLogin: string = ''; public restrictedBindPort: number = 0;
public bootstrapDaemonProxy: string = ''; public restrictedRpc: boolean = false;
public confirmExternalBind: boolean = false; public bootstrapDaemonAddress: string = '';
public bootstrapDaemonLogin: string = '';
public bootstrapDaemonProxy: string = '';
public rpcBindIp: string = '127.0.0.1'; public confirmExternalBind: boolean = false;
public rpcBindIpv6Address: string = '::1';
public rpcRestrictedBindIp: string = '';
public rpcUseIpv6: boolean = false;
public rpcIgnoreIpv4: boolean = false;
public rpcLogin: string = '';
public rpcAccessControlOrigins: string = '';
public rpcSsl: 'autodetect' | 'enabled' | 'disabled' = 'autodetect';
public rpcSslPrivateKey: string = '';
public rpcSslCertificate: string = '';
public rpcSslCACertificates: string = '';
public rpcAllowedFingerprints: string = '';
public rpcSslAllowChained: boolean = false;
public rpcSslAllowAnyCert: boolean = false;
public rpcPaymentAddress: string = '';
public rpcPaymentDifficuly: number = 1000;
public rpcPaymentCredits: number = 100;
public rpcPaymentAllowFreeLoopback: boolean = false;
public disableRpcBan: boolean = false;
public equals(settings: DaemonSettings): boolean { public rpcBindIp: string = '127.0.0.1';
return this.toCommandOptions().join('') == settings.toCommandOptions().join(''); public rpcBindIpv6Address: string = '::1';
} public rpcRestrictedBindIp: string = '';
public rpcUseIpv6: boolean = false;
public rpcIgnoreIpv4: boolean = false;
public rpcLogin: string = '';
public rpcAccessControlOrigins: string = '';
public rpcSsl: 'autodetect' | 'enabled' | 'disabled' = 'autodetect';
public rpcSslPrivateKey: string = '';
public rpcSslCertificate: string = '';
public rpcSslCACertificates: string = '';
public rpcAllowedFingerprints: string = '';
public rpcSslAllowChained: boolean = false;
public rpcSslAllowAnyCert: boolean = false;
public rpcPaymentAddress: string = '';
public rpcPaymentDifficuly: number = 1000;
public rpcPaymentCredits: number = 100;
public rpcPaymentAllowFreeLoopback: boolean = false;
public disableRpcBan: boolean = false;
public clone(): DaemonSettings { public equals(settings: DaemonSettings): boolean {
return Object.assign(new DaemonSettings(), this); return this.toCommandOptions().join('') == settings.toCommandOptions().join('');
} }
public static parse(data: any): DaemonSettings { public clone(): DaemonSettings {
const settings = new DaemonSettings(); return Object.assign(new DaemonSettings(), this);
Object.assign(settings, data); }
return settings;
}
public toCommandOptions(): string[] { public static parse(data: any): DaemonSettings {
const options: string[] = []; const settings = new DaemonSettings();
Object.assign(settings, data);
return settings;
}
if (this.mainnet) options.push(`--mainnet`); public toCommandOptions(): string[] {
else if (this.testnet) options.push(`--testnet`); const options: string[] = [];
else if (this.stagenet) options.push(`--stagenet`); if (this.monerodPath != '') options.push(this.monerodPath);
if (this.logFile != '') options.push('--log-file', this.logFile); if (this.mainnet) options.push(`--mainnet`);
if (this.logLevel >= 0 && this.logLevel <= 4) options.push('--log-level', `${this.logLevel}`); else if (this.testnet) options.push(`--testnet`);
if (this.maxLogFileSize) options.push(`--max-log-file-size=${this.maxLogFileSize}`); else if (this.stagenet) options.push(`--stagenet`);
if (this.maxLogFiles) options.push(`--max-log-files=${this.maxLogFiles}`);
if (this.maxConcurrency) options.push(`--max-concurrency=${this.maxConcurrency}`);
if (this.proxy != '') options.push(`--proxy=${this.proxy}`);
if (this.proxyAllowDnsLeaks) options.push(`--proxy-allow-dns-leaks`);
if (this.publicNode) options.push(`--public-node`);
if (this.noZmq) options.push(`--no-zmq`);
if (!this.noZmq && this.zmqRpcBindIp != '') options.push(`--zmq-rpc-bind-ip`, this.zmqRpcBindIp);
if (!this.noZmq && this.zmqRpcBindPort) options.push(`--zmq-rpc-bind-port`, `${this.zmqRpcBindPort}`);
if (!this.noZmq && this.zmqPub != '') options.push(`--zmq-pub`, this.zmqPub);
if (this.testDropDownload) options.push(`--test-drop-download`);
if (this.testDropDownload && this.testDropDownloadHeight) options.push(`--test-drop-download-height`);
if (this.testDbgLockSleep) options.push(`--tet-dbg-lock-sleep`, `${this.testDbgLockSleep}`);
if (this.regtest) options.push(`--regtest`);
if (this.keepFakeChain) options.push(`--keep-fakechain`);
if (this.fixedDifficulty) options.push(`--fixed-difficulty`, `${this.fixedDifficulty}`);
if (this.enforceDnsCheckpoint) options.push(`--enforce-dns-checkpoint`);
if (this.prepBlocksThreads) options.push(`--prep-block-threads`, `${this.prepBlocksThreads}`);
if (this.fastBlockSync) options.push(`--fast-block-sync`, `1`);
if (this.showTimeStats) options.push(`--show-time-stats`);
if (this.blockSyncSize) options.push(`--block-sync-size`, `${this.blockSyncSize}`);
if (this.checkUpdates) options.push(`--check-updates`, this.checkUpdates);
if (this.noFluffyBlocks) options.push(`--no-fluffy-blocks`);
if (this.offline) options.push(`--offline`);
if (this.disableDnsCheckpoints) options.push(`--disable-dns-checkpoints`);
if (this.blockDownloadMaxSize) options.push(`--block-download-max-size`, `${this.blockDownloadMaxSize}`);
if (this.syncPrunedBlocks) options.push(`--sync-pruned-blocks`);
if (this.maxTxPoolWeight) options.push(`--max-txpool-weight`, `${this.maxTxPoolWeight}`);
if (this.blockNotify != '') options.push(`--block-notify`, this.blockNotify);
if (this.pruneBlockchain) options.push('--prune-blockchain');
if (this.reorgNotify != '') options.push(`--reorg-notify`, this.reorgNotify);
if (this.blockRateNotify != '') options.push(`--block-rate-notify`, this.blockRateNotify);
if (this.keepAltBlocks) options.push(`--keep-alt-blocks`);
if (this.extraMessagesFile != '') options.push(`--extra-messages-file`, this.extraMessagesFile);
if (this.startMining != '') options.push(`--start-mining`, this.startMining);
if (this.miningThreds) options.push(`--mining-threads`, `${this.miningThreds}`);
if (this.bgMiningEnable) options.push(`--bg-mining-enable`);
if (this.bgMiningIgnoreBattery) options.push(`--bg-mining-ignore-battery`);
if (this.bgMiningIdleThreshold) options.push(`--bg-mining-idle-threshold`, `${this.bgMiningIdleThreshold}`);
if (this.bgMiningMinIdleInterval) options.push(`--bg-mining-idle-interval`, `${this.bgMiningMinIdleInterval}`);
if (this.bgMiningMinerTarget) options.push(`--bg-mining-miner-target`, `${this.bgMiningMinerTarget}`);
if (this.dbSyncMode != '') options.push(`--db-sync-mode`, `${this.dbSyncMode}`);
if (this.dbSalvage) options.push(`--db-salvage`);
if (this.p2pBindIp != '') options.push(`--p2p-bind-ip`, this.p2pBindIp);
if (this.p2pBindIpv6Address != '') options.push(`--p2p-bind-ipv6-address`, this.p2pBindIpv6Address);
if (this.p2pBindPort) options.push(`--p2p-bind-port`, `${this.p2pBindPort}`);
if (this.p2pBindPortIpv6) options.push(`--p2p-bind-port-ipv6`, `${this.p2pBindPortIpv6}`);
if (this.p2pUseIpv6) options.push(`--p2p-use-ipv6`);
if (this.p2pIgnoreIpv4) options.push(`--p2p-ignore-ipv4`);
if (this.p2pExternalPort) options.push(`--p2p-external-port`, `${this.p2pExternalPort}`);
if (this.allowLocalIp) options.push(`--allow-local-ip`);
if (this.addPeer != '') options.push('--add-peer', this.addPeer);
if (this.addPriorityNode != '') options.push(`--add-priority-node`, this.addPriorityNode);
if (this.addExclusiveNode != '') options.push(`--add-exlcusive-node`, this.addExclusiveNode);
if (this.seedNode != '') options.push(`--seed-node`, this.seedNode);
if (this.txProxy != '') options.push(`--tx-proxy`, this.txProxy);
if (this.anonymousInbound != '') options.push(`--anonymous-inbound`, this.anonymousInbound);
if (this.banList != '') options.push(`--ban-list`, this.banList);
if (this.hideMyPort) options.push(`--hide-my-port`);
if (this.noSync) options.push(`--no-sync`);
if (this.enableDnsBlocklist) options.push(`--enable-dns-block-list`);
if (this.noIgd) options.push(`--no-igd`);
if (this.outPeers >= 0) options.push(`--out-peers`, `${this.outPeers}`);
if (this.inPeers >= 0) options.push(`--in-peers`, `${this.inPeers}`);
if (this.tosFlag >= 0) options.push(`--tos-flag`, `${this.tosFlag}`);
if (this.limitRate >= 0) options.push(`--limit-rate`, `${this.limitRate}`);
if (this.limitRateUp >= 0) options.push(`--limit-rate-up`, `${this.limitRateUp}`);
if (this.limitRateDown >= 0) options.push(`--limit-rate-down`, `${this.limitRateDown}`);
if (this.padTransactions) options.push(`--pad-transactions`);
if (this.maxConnectionsPerIp >= 0) options.push(`--max-connections-per-ip`, `${this.maxConnectionsPerIp}`);
if (this.rpcBindIp != '') options.push(`--rpc-bind-ip`, `${this.rpcBindIp}`);
if (this.rpcBindPort) options.push(`--rpc-bind-ip`, `${this.rpcBindIp}`);
if (this.restrictedBindPort) options.push(`--restricted-bind-port`, `${this.restrictedBindPort}`);
if (this.restrictedRpc) options.push(`--restricted-rpc`);
if (this.bootstrapDaemonAddress != '') options.push(`--bootstrap-daemon-address`, this.bootstrapDaemonAddress);
if (this.bootstrapDaemonLogin != '') options.push(`--bootstrap-daemon-login`, this.bootstrapDaemonLogin);
if (this.bootstrapDaemonProxy != '') options.push(`--bootstrap-daemon-proxy`, this.bootstrapDaemonProxy);
if (this.confirmExternalBind) options.push(`--confirm-external-bind`);
if (this.rpcAccessControlOrigins != '') options.push(`--rpc-access-control-origins=${this.rpcAccessControlOrigins}`);
if (this.rpcSsl) options.push(`--rpc-ssl`, this.rpcSsl);
if (this.rpcSslPrivateKey) options.push(`--rpc-ssl-private-key`, this.rpcSslPrivateKey);
if (this.rpcSslCertificate) options.push(`--rpc-ssl-certificate`, this.rpcSslCertificate);
if (this.rpcSslCACertificates) options.push(`--rpc-ssl-ca-certificates`, this.rpcSslCACertificates);
if (this.rpcAllowedFingerprints) options.push(`--rpc-allowed-fingerprints`, this.rpcAllowedFingerprints);
if (this.rpcSslAllowChained) options.push(`--rpc-ssl-allow-chained`);
if (this.rpcSslAllowAnyCert) options.push(`--rpc-ssl-allow-any-cert`);
if (this.rpcPaymentAddress != '') options.push(`--rpc-payment-address`, this.rpcPaymentAddress); if (this.logFile != '') options.push('--log-file', this.logFile);
if (this.rpcPaymentDifficuly) options.push(`--rpc-payment-difficulty`, `${this.rpcPaymentDifficuly}`); if (this.logLevel >= 0 && this.logLevel <= 4) options.push('--log-level', `${this.logLevel}`);
if (this.rpcPaymentCredits) options.push(`--rpc-payment-credits`, `${this.rpcPaymentCredits}`); if (this.maxLogFileSize) options.push(`--max-log-file-size=${this.maxLogFileSize}`);
if (this.rpcPaymentAllowFreeLoopback) options.push(`--rpc-payment-allow-free-loopback`); if (this.maxLogFiles) options.push(`--max-log-files=${this.maxLogFiles}`);
if (this.disableRpcBan) options.push(`--disable-rpc-ban`); if (this.maxConcurrency) options.push(`--max-concurrency=${this.maxConcurrency}`);
if (this.proxy != '') options.push(`--proxy=${this.proxy}`);
if (this.proxyAllowDnsLeaks) options.push(`--proxy-allow-dns-leaks`);
if (this.publicNode) options.push(`--public-node`);
if (this.noZmq) options.push(`--no-zmq`);
if (!this.noZmq && this.zmqRpcBindIp != '') options.push(`--zmq-rpc-bind-ip`, this.zmqRpcBindIp);
if (!this.noZmq && this.zmqRpcBindPort) options.push(`--zmq-rpc-bind-port`, `${this.zmqRpcBindPort}`);
if (!this.noZmq && this.zmqPub != '') options.push(`--zmq-pub`, this.zmqPub);
if (this.testDropDownload) options.push(`--test-drop-download`);
if (this.testDropDownload && this.testDropDownloadHeight) options.push(`--test-drop-download-height`);
if (this.testDbgLockSleep) options.push(`--tet-dbg-lock-sleep`, `${this.testDbgLockSleep}`);
if (this.regtest) options.push(`--regtest`);
if (this.keepFakeChain) options.push(`--keep-fakechain`);
if (this.fixedDifficulty) options.push(`--fixed-difficulty`, `${this.fixedDifficulty}`);
if (this.enforceDnsCheckpoint) options.push(`--enforce-dns-checkpoint`);
if (this.prepBlocksThreads) options.push(`--prep-block-threads`, `${this.prepBlocksThreads}`);
if (this.fastBlockSync) options.push(`--fast-block-sync`, `1`);
if (this.showTimeStats) options.push(`--show-time-stats`);
if (this.blockSyncSize) options.push(`--block-sync-size`, `${this.blockSyncSize}`);
if (this.checkUpdates) options.push(`--check-updates`, this.checkUpdates);
if (this.noFluffyBlocks) options.push(`--no-fluffy-blocks`);
if (this.offline) options.push(`--offline`);
if (this.disableDnsCheckpoints) options.push(`--disable-dns-checkpoints`);
if (this.blockDownloadMaxSize) options.push(`--block-download-max-size`, `${this.blockDownloadMaxSize}`);
if (this.syncPrunedBlocks) options.push(`--sync-pruned-blocks`);
if (this.maxTxPoolWeight) options.push(`--max-txpool-weight`, `${this.maxTxPoolWeight}`);
if (this.blockNotify != '') options.push(`--block-notify`, this.blockNotify);
if (this.pruneBlockchain) options.push('--prune-blockchain');
if (this.reorgNotify != '') options.push(`--reorg-notify`, this.reorgNotify);
if (this.blockRateNotify != '') options.push(`--block-rate-notify`, this.blockRateNotify);
if (this.keepAltBlocks) options.push(`--keep-alt-blocks`);
if (this.extraMessagesFile != '') options.push(`--extra-messages-file`, this.extraMessagesFile);
if (this.startMining != '') options.push(`--start-mining`, this.startMining);
if (this.miningThreds) options.push(`--mining-threads`, `${this.miningThreds}`);
if (this.bgMiningEnable) options.push(`--bg-mining-enable`);
if (this.bgMiningIgnoreBattery) options.push(`--bg-mining-ignore-battery`);
if (this.bgMiningIdleThreshold) options.push(`--bg-mining-idle-threshold`, `${this.bgMiningIdleThreshold}`);
if (this.bgMiningMinIdleInterval) options.push(`--bg-mining-idle-interval`, `${this.bgMiningMinIdleInterval}`);
if (this.bgMiningMinerTarget) options.push(`--bg-mining-miner-target`, `${this.bgMiningMinerTarget}`);
if (this.dbSyncMode != '') options.push(`--db-sync-mode`, `${this.dbSyncMode}`);
if (this.dbSalvage) options.push(`--db-salvage`);
if (this.p2pBindIp != '') options.push(`--p2p-bind-ip`, this.p2pBindIp);
if (this.p2pBindIpv6Address != '') options.push(`--p2p-bind-ipv6-address`, this.p2pBindIpv6Address);
if (this.p2pBindPort) options.push(`--p2p-bind-port`, `${this.p2pBindPort}`);
if (this.p2pBindPortIpv6) options.push(`--p2p-bind-port-ipv6`, `${this.p2pBindPortIpv6}`);
if (this.p2pUseIpv6) options.push(`--p2p-use-ipv6`);
if (this.p2pIgnoreIpv4) options.push(`--p2p-ignore-ipv4`);
if (this.p2pExternalPort) options.push(`--p2p-external-port`, `${this.p2pExternalPort}`);
if (this.allowLocalIp) options.push(`--allow-local-ip`);
if (this.addPeer != '') options.push('--add-peer', this.addPeer);
if (this.addPriorityNode != '') options.push(`--add-priority-node`, this.addPriorityNode);
if (this.addExclusiveNode != '') options.push(`--add-exlcusive-node`, this.addExclusiveNode);
if (this.seedNode != '') options.push(`--seed-node`, this.seedNode);
if (this.txProxy != '') options.push(`--tx-proxy`, this.txProxy);
if (this.anonymousInbound != '') options.push(`--anonymous-inbound`, this.anonymousInbound);
if (this.banList != '') options.push(`--ban-list`, this.banList);
if (this.hideMyPort) options.push(`--hide-my-port`);
if (this.noSync) options.push(`--no-sync`);
if (this.enableDnsBlocklist) options.push(`--enable-dns-block-list`);
if (this.noIgd) options.push(`--no-igd`);
if (this.outPeers >= 0) options.push(`--out-peers`, `${this.outPeers}`);
if (this.inPeers >= 0) options.push(`--in-peers`, `${this.inPeers}`);
if (this.tosFlag >= 0) options.push(`--tos-flag`, `${this.tosFlag}`);
if (this.limitRate >= 0) options.push(`--limit-rate`, `${this.limitRate}`);
if (this.limitRateUp >= 0) options.push(`--limit-rate-up`, `${this.limitRateUp}`);
if (this.limitRateDown >= 0) options.push(`--limit-rate-down`, `${this.limitRateDown}`);
if (this.padTransactions) options.push(`--pad-transactions`);
if (this.maxConnectionsPerIp >= 0) options.push(`--max-connections-per-ip`, `${this.maxConnectionsPerIp}`);
if (this.rpcBindIp != '') options.push(`--rpc-bind-ip`, `${this.rpcBindIp}`);
if (this.rpcBindPort) options.push(`--rpc-bind-ip`, `${this.rpcBindIp}`);
if (this.restrictedBindPort) options.push(`--restricted-bind-port`, `${this.restrictedBindPort}`);
if (this.restrictedRpc) options.push(`--restricted-rpc`);
if (this.bootstrapDaemonAddress != '') options.push(`--bootstrap-daemon-address`, this.bootstrapDaemonAddress);
if (this.bootstrapDaemonLogin != '') options.push(`--bootstrap-daemon-login`, this.bootstrapDaemonLogin);
if (this.bootstrapDaemonProxy != '') options.push(`--bootstrap-daemon-proxy`, this.bootstrapDaemonProxy);
if (this.confirmExternalBind) options.push(`--confirm-external-bind`);
if (this.rpcAccessControlOrigins != '') options.push(`--rpc-access-control-origins=${this.rpcAccessControlOrigins}`);
if (this.rpcSsl) options.push(`--rpc-ssl`, this.rpcSsl);
if (this.rpcSslPrivateKey) options.push(`--rpc-ssl-private-key`, this.rpcSslPrivateKey);
if (this.rpcSslCertificate) options.push(`--rpc-ssl-certificate`, this.rpcSslCertificate);
if (this.rpcSslCACertificates) options.push(`--rpc-ssl-ca-certificates`, this.rpcSslCACertificates);
if (this.rpcAllowedFingerprints) options.push(`--rpc-allowed-fingerprints`, this.rpcAllowedFingerprints);
if (this.rpcSslAllowChained) options.push(`--rpc-ssl-allow-chained`);
if (this.rpcSslAllowAnyCert) options.push(`--rpc-ssl-allow-any-cert`);
return options; if (this.rpcPaymentAddress != '') options.push(`--rpc-payment-address`, this.rpcPaymentAddress);
} if (this.rpcPaymentDifficuly) options.push(`--rpc-payment-difficulty`, `${this.rpcPaymentDifficuly}`);
if (this.rpcPaymentCredits) options.push(`--rpc-payment-credits`, `${this.rpcPaymentCredits}`);
if (this.rpcPaymentAllowFreeLoopback) options.push(`--rpc-payment-allow-free-loopback`);
if (this.disableRpcBan) options.push(`--disable-rpc-ban`);
return options;
}
} }