mirror of
https://github.com/goharbor/harbor.git
synced 2025-01-28 02:21:24 +01:00
Update storage display (#14807)
Signed-off-by: AllForNothing <sshijun@vmware.com>
This commit is contained in:
parent
dce3522b4e
commit
3322716bc6
@ -18,7 +18,6 @@ import { ListProjectComponent } from "./list-project/list-project.component";
|
|||||||
import { CreateProjectComponent } from "./create-project/create-project.component";
|
import { CreateProjectComponent } from "./create-project/create-project.component";
|
||||||
import { RouterModule, Routes } from "@angular/router";
|
import { RouterModule, Routes } from "@angular/router";
|
||||||
import { StatisticsPanelComponent } from "./statictics/statistics-panel.component";
|
import { StatisticsPanelComponent } from "./statictics/statistics-panel.component";
|
||||||
import { StatisticsComponent } from "./statictics/statistics.component";
|
|
||||||
|
|
||||||
const routes: Routes = [
|
const routes: Routes = [
|
||||||
{
|
{
|
||||||
@ -36,8 +35,7 @@ const routes: Routes = [
|
|||||||
ProjectsComponent,
|
ProjectsComponent,
|
||||||
ListProjectComponent,
|
ListProjectComponent,
|
||||||
CreateProjectComponent,
|
CreateProjectComponent,
|
||||||
StatisticsPanelComponent,
|
StatisticsPanelComponent
|
||||||
StatisticsComponent
|
|
||||||
],
|
],
|
||||||
providers: []
|
providers: []
|
||||||
})
|
})
|
||||||
|
@ -1,44 +1,46 @@
|
|||||||
<div class="row flex-items-xs-between flex-items-xs-middle">
|
<div class="clr-row flex-end">
|
||||||
<div></div>
|
<div class="card">
|
||||||
<div id="right_statistic_panel">
|
<div class="card-block">
|
||||||
<div class="statistic-block">
|
<h4 class="head">{{'STATISTICS.PRO_ITEM' | translate }}</h4>
|
||||||
<div class="statistic-column-block">
|
<div class="clr-row" *ngIf="isValidSession">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_PRIVATE" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.private_project_count}}</div>
|
||||||
|
</div>
|
||||||
|
<div class="clr-row">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_PUB" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.public_project_count}}</div>
|
||||||
|
</div>
|
||||||
|
<div class="clr-row" *ngIf="isValidSession">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_TOTAL" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.total_project_count}}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-block">
|
||||||
|
<h4 class="head">{{'STATISTICS.REPO_ITEM' | translate }}</h4>
|
||||||
|
<div class="clr-row" *ngIf="isValidSession">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_PRIVATE" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.private_repo_count}}</div>
|
||||||
|
</div>
|
||||||
|
<div class="clr-row">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_PUB" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.public_repo_count}}</div>
|
||||||
|
</div>
|
||||||
|
<div class="clr-row" *ngIf="isValidSession">
|
||||||
|
<div class="clr-col">{{"STATISTICS.INDEX_TOTAL" | translate}}</div>
|
||||||
|
<div class="clr-col font-weight-700">{{originalCopy?.total_repo_count}}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="card" *ngIf="isValidSession">
|
||||||
|
<div class="card-block container">
|
||||||
|
<h4 class="head">{{'STATISTICS.STORAGE_USED' | translate }}</h4>
|
||||||
|
<div class="storage-used font-weight-700">
|
||||||
<div>
|
<div>
|
||||||
<span class="statistic-column-title statistic-column-title-pro">{{'STATISTICS.PRO_ITEM' | translate }}</span>
|
<span class="size-number margin-right-5px">{{getSizeNumber()}}</span><span *ngIf="getSizeNumber()">{{getSizeUnit()}}</span>
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<span class="statistic-column-title statistic-column-title-repo">{{'STATISTICS.REPO_ITEM' | translate }}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="statistic-column-block">
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.private_project_count' [label]='"STATISTICS.INDEX_PRIVATE" | translate'></statistics>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.private_repo_count' [label]='"STATISTICS.INDEX_PRIVATE" | translate'></statistics>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="statistic-column-block">
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.public_project_count' [label]='"STATISTICS.INDEX_PUB" | translate'></statistics>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.public_repo_count' [label]='"STATISTICS.INDEX_PUB" | translate'></statistics>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="statistic-column-block">
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.total_project_count' [label]='"STATISTICS.INDEX_TOTAL" | translate' *ngIf="isValidSession"></statistics>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<statistics [data]='originalCopy.total_repo_count' [label]='"STATISTICS.INDEX_TOTAL" | translate' *ngIf="isValidSession"></statistics>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="statistic-item-divider" [hidden]="!isValidSession || !isValidStorage"></div>
|
|
||||||
<div class="statistic-block" [hidden]="!isValidSession || !isValidStorage">
|
|
||||||
<esxc-gauge [free]="freeStorage" [threasHold]="totalStorage" [title]='"STATISTICS.STORAGE"' [animate]="true">
|
|
||||||
</esxc-gauge>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -0,0 +1,37 @@
|
|||||||
|
.head {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0.25rem;
|
||||||
|
}
|
||||||
|
.card{
|
||||||
|
width: 10rem;
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card:last-child{
|
||||||
|
width: 10rem;
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
.flex-end {
|
||||||
|
justify-content: flex-end;
|
||||||
|
}
|
||||||
|
.font-weight-700 {
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
.storage-used {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
.size-number {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
}
|
||||||
|
.container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.margin-right-5px {
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
@ -1,30 +1,36 @@
|
|||||||
import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing';
|
import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
import { TranslateModule, TranslateService } from '@ngx-translate/core';
|
|
||||||
import { StatisticsPanelComponent } from './statistics-panel.component';
|
import { StatisticsPanelComponent } from './statistics-panel.component';
|
||||||
import { StatisticsComponent } from './statistics.component';
|
|
||||||
import { CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
|
import { CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
|
||||||
import { BrowserAnimationsModule, NoopAnimationsModule } from '@angular/platform-browser/animations';
|
|
||||||
import { ClarityModule } from '@clr/angular';
|
|
||||||
import { FormsModule } from '@angular/forms';
|
|
||||||
import { RouterTestingModule } from '@angular/router/testing';
|
|
||||||
import { of } from 'rxjs';
|
import { of } from 'rxjs';
|
||||||
import { HttpClientTestingModule } from '@angular/common/http/testing';
|
|
||||||
import { StatisticsService } from "./statistics.service";
|
|
||||||
import { SessionService } from "../../../../shared/services/session.service";
|
import { SessionService } from "../../../../shared/services/session.service";
|
||||||
import { MessageHandlerService } from "../../../../shared/services/message-handler.service";
|
import { MessageHandlerService } from "../../../../shared/services/message-handler.service";
|
||||||
import { StatisticHandler } from "./statistic-handler.service";
|
import { StatisticHandler } from "./statistic-handler.service";
|
||||||
import { AppConfigService } from "../../../../services/app-config.service";
|
import { AppConfigService } from "../../../../services/app-config.service";
|
||||||
import { Statistics } from './statistics';
|
import { Statistic } from "../../../../../../ng-swagger-gen/models/statistic";
|
||||||
import { Volumes } from './volumes';
|
import { SharedTestingModule } from "../../../../shared/shared.module";
|
||||||
|
import { StatisticService } from "../../../../../../ng-swagger-gen/services/statistic.service";
|
||||||
|
|
||||||
describe('StatisticsPanelComponent', () => {
|
describe('StatisticsPanelComponent', () => {
|
||||||
|
const mockedStatistic: Statistic = {
|
||||||
|
"private_project_count": 2,
|
||||||
|
"private_repo_count": 0,
|
||||||
|
"public_project_count": 3,
|
||||||
|
"public_repo_count": 1,
|
||||||
|
"total_project_count": 5,
|
||||||
|
"total_repo_count": 1,
|
||||||
|
"total_storage_consumption": 4564
|
||||||
|
};
|
||||||
let component: StatisticsPanelComponent;
|
let component: StatisticsPanelComponent;
|
||||||
let fixture: ComponentFixture<StatisticsPanelComponent>;
|
let fixture: ComponentFixture<StatisticsPanelComponent>;
|
||||||
const mockStatisticsService = {
|
const mockStatisticsService = {
|
||||||
getStatistics: () => of(new Statistics()),
|
getStatistic: () => of(mockedStatistic),
|
||||||
getVolumes: () => of(new Volumes()),
|
|
||||||
};
|
};
|
||||||
const mockSessionService = {
|
const mockSessionService = {
|
||||||
getCurrentUser: () => { }
|
getCurrentUser: () => {
|
||||||
|
return {
|
||||||
|
has_admin_role: true
|
||||||
|
};
|
||||||
|
}
|
||||||
};
|
};
|
||||||
const mockAppConfigService = {
|
const mockAppConfigService = {
|
||||||
getConfig: () => {
|
getConfig: () => {
|
||||||
@ -34,34 +40,25 @@ describe('StatisticsPanelComponent', () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
const mockMessageHandlerService = {
|
const mockMessageHandlerService = {
|
||||||
handleError: () => { }
|
handleError: () => {
|
||||||
|
}
|
||||||
};
|
};
|
||||||
const mockStatisticHandler = {
|
const mockStatisticHandler = {
|
||||||
refreshChan$: of(null)
|
refreshChan$: of(null)
|
||||||
};
|
};
|
||||||
const mockRouter = {
|
|
||||||
navigate: () => { }
|
|
||||||
};
|
|
||||||
beforeEach(waitForAsync(() => {
|
beforeEach(waitForAsync(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
schemas: [
|
schemas: [
|
||||||
CUSTOM_ELEMENTS_SCHEMA
|
CUSTOM_ELEMENTS_SCHEMA
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
BrowserAnimationsModule,
|
SharedTestingModule
|
||||||
ClarityModule,
|
|
||||||
TranslateModule.forRoot(),
|
|
||||||
FormsModule,
|
|
||||||
RouterTestingModule,
|
|
||||||
NoopAnimationsModule,
|
|
||||||
HttpClientTestingModule
|
|
||||||
],
|
],
|
||||||
declarations: [StatisticsPanelComponent, StatisticsComponent],
|
declarations: [StatisticsPanelComponent],
|
||||||
providers: [
|
providers: [
|
||||||
TranslateService,
|
|
||||||
{provide: SessionService, useValue: mockSessionService},
|
{provide: SessionService, useValue: mockSessionService},
|
||||||
{provide: AppConfigService, useValue: mockAppConfigService},
|
{provide: AppConfigService, useValue: mockAppConfigService},
|
||||||
{ provide: StatisticsService, useValue: mockStatisticsService },
|
{provide: StatisticService, useValue: mockStatisticsService},
|
||||||
{provide: StatisticHandler, useValue: mockStatisticHandler},
|
{provide: StatisticHandler, useValue: mockStatisticHandler},
|
||||||
{provide: MessageHandlerService, useValue: mockMessageHandlerService}
|
{provide: MessageHandlerService, useValue: mockMessageHandlerService}
|
||||||
]
|
]
|
||||||
@ -77,4 +74,16 @@ describe('StatisticsPanelComponent', () => {
|
|||||||
it('should create', () => {
|
it('should create', () => {
|
||||||
expect(component).toBeTruthy();
|
expect(component).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
it('should have 3 cards', async () => {
|
||||||
|
fixture.detectChanges();
|
||||||
|
await fixture.whenStable();
|
||||||
|
const cards = fixture.nativeElement.querySelectorAll('.card');
|
||||||
|
expect(cards.length).toEqual(3);
|
||||||
|
});
|
||||||
|
it('should display right size number', async () => {
|
||||||
|
fixture.detectChanges();
|
||||||
|
await fixture.whenStable();
|
||||||
|
const sizeHtml: HTMLSpanElement = fixture.nativeElement.querySelector('.size-number');
|
||||||
|
expect(sizeHtml.innerText).toEqual('4.46');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -13,35 +13,28 @@
|
|||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
import { Component, OnInit, OnDestroy } from "@angular/core";
|
import { Component, OnInit, OnDestroy } from "@angular/core";
|
||||||
import { Subscription } from "rxjs";
|
import { Subscription } from "rxjs";
|
||||||
|
|
||||||
import { StatisticsService } from "./statistics.service";
|
|
||||||
import { Statistics } from "./statistics";
|
|
||||||
|
|
||||||
import { SessionService } from "../../../../shared/services/session.service";
|
import { SessionService } from "../../../../shared/services/session.service";
|
||||||
import { Volumes } from "./volumes";
|
|
||||||
|
|
||||||
import { MessageHandlerService } from "../../../../shared/services/message-handler.service";
|
import { MessageHandlerService } from "../../../../shared/services/message-handler.service";
|
||||||
import { StatisticHandler } from "./statistic-handler.service";
|
import { StatisticHandler } from "./statistic-handler.service";
|
||||||
import { AppConfigService } from "../../../../services/app-config.service";
|
import { Statistic } from "../../../../../../ng-swagger-gen/models/statistic";
|
||||||
|
import { StatisticService } from "../../../../../../ng-swagger-gen/services/statistic.service";
|
||||||
|
import { getSizeNumber, getSizeUnit } from "../../../../shared/units/utils";
|
||||||
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: "statistics-panel",
|
selector: "statistics-panel",
|
||||||
templateUrl: "statistics-panel.component.html",
|
templateUrl: "statistics-panel.component.html",
|
||||||
styleUrls: ["statistics.component.scss"],
|
styleUrls: ["statistics-panel.component.scss"],
|
||||||
providers: [StatisticsService]
|
|
||||||
})
|
})
|
||||||
|
|
||||||
export class StatisticsPanelComponent implements OnInit, OnDestroy {
|
export class StatisticsPanelComponent implements OnInit, OnDestroy {
|
||||||
|
|
||||||
originalCopy: Statistics = new Statistics();
|
originalCopy: Statistic;
|
||||||
volumesInfo: Volumes = new Volumes();
|
|
||||||
refreshSub: Subscription;
|
refreshSub: Subscription;
|
||||||
constructor(
|
constructor(
|
||||||
private statistics: StatisticsService,
|
private statistics: StatisticService,
|
||||||
private msgHandler: MessageHandlerService,
|
private msgHandler: MessageHandlerService,
|
||||||
private session: SessionService,
|
private session: SessionService,
|
||||||
private appConfigService: AppConfigService,
|
|
||||||
private statisticHandler: StatisticHandler) {
|
private statisticHandler: StatisticHandler) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -54,10 +47,6 @@ export class StatisticsPanelComponent implements OnInit, OnDestroy {
|
|||||||
if (this.session.getCurrentUser()) {
|
if (this.session.getCurrentUser()) {
|
||||||
this.getStatistics();
|
this.getStatistics();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.isValidSession) {
|
|
||||||
this.getVolumes();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
@ -65,60 +54,27 @@ export class StatisticsPanelComponent implements OnInit, OnDestroy {
|
|||||||
this.refreshSub.unsubscribe();
|
this.refreshSub.unsubscribe();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
getStatistics(): void {
|
||||||
public get totalStorage(): number {
|
this.statistics.getStatistic()
|
||||||
let count: number = 0;
|
|
||||||
if (this.volumesInfo && this.volumesInfo.storage && this.volumesInfo.storage.length) {
|
|
||||||
this.volumesInfo.storage.forEach(item => {
|
|
||||||
count += item.total ? item.total : 0;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return this.getGBFromBytes(count);
|
|
||||||
}
|
|
||||||
|
|
||||||
public get freeStorage(): number {
|
|
||||||
let count: number = 0;
|
|
||||||
if (this.volumesInfo && this.volumesInfo.storage && this.volumesInfo.storage.length) {
|
|
||||||
this.volumesInfo.storage.forEach(item => {
|
|
||||||
count += item.free ? item.free : 0;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return this.getGBFromBytes(count);
|
|
||||||
}
|
|
||||||
|
|
||||||
public getStatistics(): void {
|
|
||||||
this.statistics.getStatistics()
|
|
||||||
.subscribe(statistics => this.originalCopy = statistics
|
.subscribe(statistics => this.originalCopy = statistics
|
||||||
, error => {
|
, error => {
|
||||||
this.msgHandler.handleError(error);
|
this.msgHandler.handleError(error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
get isValidSession(): boolean {
|
||||||
public getVolumes(): void {
|
|
||||||
this.statistics.getVolumes()
|
|
||||||
.subscribe(volumes => this.volumesInfo = volumes
|
|
||||||
, error => {
|
|
||||||
this.msgHandler.handleError(error);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public get isValidSession(): boolean {
|
|
||||||
let user = this.session.getCurrentUser();
|
let user = this.session.getCurrentUser();
|
||||||
return user && user.has_admin_role;
|
return user && user.has_admin_role;
|
||||||
}
|
}
|
||||||
|
getSizeNumber(): number | string {
|
||||||
public get isValidStorage(): boolean {
|
if (this.originalCopy) {
|
||||||
let count: number = 0;
|
return getSizeNumber(this.originalCopy.total_storage_consumption);
|
||||||
if (this.volumesInfo && this.volumesInfo.storage && this.volumesInfo.storage.length) {
|
|
||||||
this.volumesInfo.storage.forEach(item => {
|
|
||||||
count += item.total ? item.total : 0;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
return count !== 0 &&
|
return null;
|
||||||
this.appConfigService.getConfig().registry_storage_provider_name === "filesystem";
|
|
||||||
}
|
}
|
||||||
|
getSizeUnit(): number | string {
|
||||||
getGBFromBytes(bytes: number): number {
|
if (this.originalCopy) {
|
||||||
return Math.round((bytes / (1024 * 1024 * 1024)));
|
return getSizeUnit(this.originalCopy.total_storage_consumption);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +0,0 @@
|
|||||||
<div class="statistic-wrapper">
|
|
||||||
<span class="statistic-data">{{data}}</span>
|
|
||||||
<span class="statistic-text">{{label}}</span>
|
|
||||||
</div>
|
|
@ -1,75 +0,0 @@
|
|||||||
.statistic-wrapper {
|
|
||||||
padding: 4px;
|
|
||||||
margin: 4px;
|
|
||||||
text-align: right;
|
|
||||||
vertical-align: middle;
|
|
||||||
height: 30px;
|
|
||||||
display: inline-block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-data {
|
|
||||||
font-size: 24px;
|
|
||||||
font-weight: 600;
|
|
||||||
font-family: Metropolis, "Avenir Next", "Helvetica Neue", Arial, sans-serif;
|
|
||||||
line-height: 24px;
|
|
||||||
margin-right: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-text {
|
|
||||||
font-size: 14px;
|
|
||||||
line-height: 14px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
font-family: Metropolis, "Avenir Next", "Helvetica Neue", Arial, sans-serif;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-column-block {
|
|
||||||
display: inline-block;
|
|
||||||
text-align: right;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-column-title {
|
|
||||||
position: relative;
|
|
||||||
text-transform: uppercase;
|
|
||||||
font-size: 14px;
|
|
||||||
font-family: Metropolis, "Avenir Next", "Helvetica Neue", Arial, sans-serif;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-column-title-pro {
|
|
||||||
top: -10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-column-title-repo {
|
|
||||||
top: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-item-divider {
|
|
||||||
height: 54px;
|
|
||||||
display: inline-block;
|
|
||||||
width: 2px;
|
|
||||||
background-color: #bdbdbd;
|
|
||||||
opacity: 0.55;
|
|
||||||
margin-left: 4px;
|
|
||||||
margin-right: 12px;
|
|
||||||
position: relative;
|
|
||||||
top: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.statistic-block {
|
|
||||||
display: inline-block;
|
|
||||||
}
|
|
||||||
|
|
||||||
#right_statistic_panel {
|
|
||||||
margin-right: 18px;
|
|
||||||
margin-top: 8px;
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-end;
|
|
||||||
div:nth-of-type(2) {
|
|
||||||
margin-left: 16px;
|
|
||||||
}
|
|
||||||
div:nth-of-type(3) {
|
|
||||||
margin-left: 28px;
|
|
||||||
}
|
|
||||||
div:nth-of-type(4) {
|
|
||||||
margin-left: 28px;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,25 +0,0 @@
|
|||||||
import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing';
|
|
||||||
|
|
||||||
import { StatisticsComponent } from './statistics.component';
|
|
||||||
|
|
||||||
describe('StatisticsComponent', () => {
|
|
||||||
let component: StatisticsComponent;
|
|
||||||
let fixture: ComponentFixture<StatisticsComponent>;
|
|
||||||
|
|
||||||
beforeEach(waitForAsync(() => {
|
|
||||||
TestBed.configureTestingModule({
|
|
||||||
declarations: [StatisticsComponent]
|
|
||||||
})
|
|
||||||
.compileComponents();
|
|
||||||
}));
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
fixture = TestBed.createComponent(StatisticsComponent);
|
|
||||||
component = fixture.componentInstance;
|
|
||||||
fixture.detectChanges();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should create', () => {
|
|
||||||
expect(component).toBeTruthy();
|
|
||||||
});
|
|
||||||
});
|
|
@ -1,25 +0,0 @@
|
|||||||
// Copyright Project Harbor Authors
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
import { Component, Input } from '@angular/core';
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
selector: 'statistics',
|
|
||||||
templateUrl: "statistics.component.html",
|
|
||||||
styleUrls: ['statistics.component.scss']
|
|
||||||
})
|
|
||||||
|
|
||||||
export class StatisticsComponent {
|
|
||||||
@Input() label: string;
|
|
||||||
@Input() data: number = 0;
|
|
||||||
}
|
|
@ -1,18 +0,0 @@
|
|||||||
import { TestBed, inject } from '@angular/core/testing';
|
|
||||||
import { HttpClientTestingModule } from '@angular/common/http/testing';
|
|
||||||
import { StatisticsService } from './statistics.service';
|
|
||||||
|
|
||||||
describe('StatisticsService', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
TestBed.configureTestingModule({
|
|
||||||
imports: [
|
|
||||||
HttpClientTestingModule
|
|
||||||
],
|
|
||||||
providers: [StatisticsService]
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should be created', inject([StatisticsService], (service: StatisticsService) => {
|
|
||||||
expect(service).toBeTruthy();
|
|
||||||
}));
|
|
||||||
});
|
|
@ -1,48 +0,0 @@
|
|||||||
// Copyright (c) 2017 VMware, Inc. All Rights Reserved.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { HttpClient } from '@angular/common/http';
|
|
||||||
import { map, catchError } from "rxjs/operators";
|
|
||||||
import { Observable, throwError as observableThrowError } from "rxjs";
|
|
||||||
import { Statistics } from './statistics';
|
|
||||||
import { Volumes } from './volumes';
|
|
||||||
import { CURRENT_BASE_HREF, HTTP_GET_OPTIONS } from "../../../../shared/units/utils";
|
|
||||||
|
|
||||||
|
|
||||||
const statisticsEndpoint = CURRENT_BASE_HREF + "/statistics";
|
|
||||||
const volumesEndpoint = CURRENT_BASE_HREF + "/systeminfo/volumes";
|
|
||||||
/**
|
|
||||||
* Declare service to handle the top repositories
|
|
||||||
*
|
|
||||||
*
|
|
||||||
**
|
|
||||||
* class GlobalSearchService
|
|
||||||
*/
|
|
||||||
@Injectable()
|
|
||||||
export class StatisticsService {
|
|
||||||
|
|
||||||
constructor(private http: HttpClient) { }
|
|
||||||
|
|
||||||
getStatistics(): Observable<Statistics> {
|
|
||||||
return this.http.get(statisticsEndpoint, HTTP_GET_OPTIONS)
|
|
||||||
.pipe(map(response => response as Statistics)
|
|
||||||
, catchError(error => observableThrowError(error)));
|
|
||||||
}
|
|
||||||
|
|
||||||
getVolumes(): Observable<Volumes> {
|
|
||||||
return this.http.get(volumesEndpoint, HTTP_GET_OPTIONS)
|
|
||||||
.pipe(map(response => response as Volumes)
|
|
||||||
, catchError(error => observableThrowError(error)));
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,23 +0,0 @@
|
|||||||
// Copyright Project Harbor Authors
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
export class Statistics {
|
|
||||||
constructor() {}
|
|
||||||
|
|
||||||
private_project_count: number;
|
|
||||||
private_repo_count: number;
|
|
||||||
public_project_count: number;
|
|
||||||
public_repo_count: number;
|
|
||||||
total_project_count: number;
|
|
||||||
total_repo_count: number;
|
|
||||||
}
|
|
@ -1,30 +0,0 @@
|
|||||||
// Copyright Project Harbor Authors
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
export class Volumes {
|
|
||||||
constructor() {
|
|
||||||
this.storage = [new Storage()];
|
|
||||||
}
|
|
||||||
|
|
||||||
storage: Storage[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export class Storage {
|
|
||||||
constructor() {
|
|
||||||
this.total = 0;
|
|
||||||
this.free = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
total: number;
|
|
||||||
free: number;
|
|
||||||
}
|
|
@ -1,4 +1,4 @@
|
|||||||
import { delUrlParam, getQueryString, getSortingString, isSameArrayValue, isSameObject } from "./utils";
|
import { delUrlParam, getQueryString, getSizeNumber, getSizeUnit, getSortingString, isSameArrayValue, isSameObject } from "./utils";
|
||||||
import { ClrDatagridStateInterface } from "@clr/angular";
|
import { ClrDatagridStateInterface } from "@clr/angular";
|
||||||
|
|
||||||
describe('functions in utils.ts should work', () => {
|
describe('functions in utils.ts should work', () => {
|
||||||
@ -53,4 +53,20 @@ describe('functions in utils.ts should work', () => {
|
|||||||
};
|
};
|
||||||
expect(getQueryString(state)).toEqual(encodeURIComponent('name=~test,url=~http://test.com'));
|
expect(getQueryString(state)).toEqual(encodeURIComponent('name=~test,url=~http://test.com'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('function getSizeNumber() should work', () => {
|
||||||
|
expect(getSizeNumber).toBeTruthy();
|
||||||
|
expect(getSizeNumber(4564)).toEqual('4.46');
|
||||||
|
expect(getSizeNumber(10)).toEqual(10);
|
||||||
|
expect(getSizeNumber(456400)).toEqual('445.70');
|
||||||
|
expect(getSizeNumber(45640000)).toEqual('43.53');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('function getSizeUnit() should work', () => {
|
||||||
|
expect(getSizeUnit).toBeTruthy();
|
||||||
|
expect(getSizeUnit(4564)).toEqual('KB');
|
||||||
|
expect(getSizeUnit(10)).toEqual('B');
|
||||||
|
expect(getSizeUnit(4564000)).toEqual('MB');
|
||||||
|
expect(getSizeUnit(4564000000)).toEqual('GB');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -579,6 +579,38 @@ export function formatSize(tagSize: string): string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get size number of target size (in byte)
|
||||||
|
* @param size
|
||||||
|
*/
|
||||||
|
export function getSizeNumber(size: number): string | number {
|
||||||
|
if (Math.pow(1024, 1) <= size && size < Math.pow(1024, 2)) {
|
||||||
|
return (size / Math.pow(1024, 1)).toFixed(2);
|
||||||
|
} else if (Math.pow(1024, 2) <= size && size < Math.pow(1024, 3)) {
|
||||||
|
return (size / Math.pow(1024, 2)).toFixed(2);
|
||||||
|
} else if (Math.pow(1024, 3) <= size && size < Math.pow(1024, 4)) {
|
||||||
|
return (size / Math.pow(1024, 3)).toFixed(2);
|
||||||
|
} else {
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get size unit of target size (in byte)
|
||||||
|
* @param size
|
||||||
|
*/
|
||||||
|
export function getSizeUnit(size: number): string {
|
||||||
|
if (Math.pow(1024, 1) <= size && size < Math.pow(1024, 2)) {
|
||||||
|
return "KB";
|
||||||
|
} else if (Math.pow(1024, 2) <= size && size < Math.pow(1024, 3)) {
|
||||||
|
return "MB";
|
||||||
|
} else if (Math.pow(1024, 3) <= size && size < Math.pow(1024, 4)) {
|
||||||
|
return "GB";
|
||||||
|
} else {
|
||||||
|
return "B";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simple object check.
|
* Simple object check.
|
||||||
* @param item
|
* @param item
|
||||||
|
@ -991,16 +991,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Beliebte Repositories",
|
"TOP_REPO": "Beliebte Repositories",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "STATISTIKEN",
|
|
||||||
"PRO_ITEM": "PROJEKTE",
|
"PRO_ITEM": "PROJEKTE",
|
||||||
"REPO_ITEM": "REPOSITORIES",
|
"REPO_ITEM": "REPOSITORIES",
|
||||||
"INDEX_PRIVATE": "PRIVAT",
|
"INDEX_PRIVATE": "PRIVAT",
|
||||||
"INDEX_MY_PROJECTS": "MEINE PROJEKTE",
|
|
||||||
"INDEX_MY_REPOSITORIES": "MEINE REPOSITORIES",
|
|
||||||
"INDEX_PUB": "ÖFFENTLICH",
|
"INDEX_PUB": "ÖFFENTLICH",
|
||||||
"INDEX_TOTAL": "GESAMT",
|
"INDEX_TOTAL": "GESAMT",
|
||||||
"STORAGE": "SPEICHER",
|
"STORAGE": "SPEICHER",
|
||||||
"LIMIT": "Limit"
|
"LIMIT": "Limit",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Suche...",
|
"IN_PROGRESS": "Suche...",
|
||||||
|
@ -991,16 +991,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Popular Repositories",
|
"TOP_REPO": "Popular Repositories",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "STATISTICS",
|
"PRO_ITEM": "Projects",
|
||||||
"PRO_ITEM": "PROJECTS",
|
"REPO_ITEM": "Repositories",
|
||||||
"REPO_ITEM": "REPOSITORIES",
|
"INDEX_PRIVATE": "Private",
|
||||||
"INDEX_PRIVATE": "PRIVATE",
|
"INDEX_PUB": "Public",
|
||||||
"INDEX_MY_PROJECTS": "MY PROJECTS",
|
"INDEX_TOTAL": "Total",
|
||||||
"INDEX_MY_REPOSITORIES": "MY REPOSITORIES",
|
|
||||||
"INDEX_PUB": "PUBLIC",
|
|
||||||
"INDEX_TOTAL": "TOTAL",
|
|
||||||
"STORAGE": "STORAGE",
|
"STORAGE": "STORAGE",
|
||||||
"LIMIT": "Limit"
|
"LIMIT": "Limit",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Search...",
|
"IN_PROGRESS": "Search...",
|
||||||
|
@ -991,16 +991,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Repositorios Populares",
|
"TOP_REPO": "Repositorios Populares",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "ESTADÍSTICAS",
|
|
||||||
"PRO_ITEM": "PROYECTOS",
|
"PRO_ITEM": "PROYECTOS",
|
||||||
"REPO_ITEM": "REPOSITORIOS",
|
"REPO_ITEM": "REPOSITORIOS",
|
||||||
"INDEX_PRIVATE": "PRIVADO",
|
"INDEX_PRIVATE": "PRIVADO",
|
||||||
"INDEX_MY_PROJECTS": "MY PROJECTS",
|
|
||||||
"INDEX_MY_REPOSITORIES": "MY REPOSITORIES",
|
|
||||||
"INDEX_PUB": "PÚBLICO",
|
"INDEX_PUB": "PÚBLICO",
|
||||||
"INDEX_TOTAL": "TOTAL",
|
"INDEX_TOTAL": "TOTAL",
|
||||||
"STORAGE": "ALMACENAMIENTO",
|
"STORAGE": "ALMACENAMIENTO",
|
||||||
"LIMIT": "Límite"
|
"LIMIT": "Límite",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Buscar...",
|
"IN_PROGRESS": "Buscar...",
|
||||||
|
@ -964,16 +964,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Dépôts Populaires",
|
"TOP_REPO": "Dépôts Populaires",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "STATISTIQUES",
|
|
||||||
"PRO_ITEM": "PROJETS",
|
"PRO_ITEM": "PROJETS",
|
||||||
"REPO_ITEM": "DÉPÔTS",
|
"REPO_ITEM": "DÉPÔTS",
|
||||||
"INDEX_PRIVATE": "PRIVÉ",
|
"INDEX_PRIVATE": "PRIVÉ",
|
||||||
"INDEX_MY_PROJECTS": "MES PROJETS",
|
|
||||||
"INDEX_MY_REPOSITORIES": "MES DÉPÔTS",
|
|
||||||
"INDEX_PUB": "PUBLIC",
|
"INDEX_PUB": "PUBLIC",
|
||||||
"INDEX_TOTAL": "TOTAL",
|
"INDEX_TOTAL": "TOTAL",
|
||||||
"STORAGE": "STOCKAGE",
|
"STORAGE": "STOCKAGE",
|
||||||
"LIMIT": "Limite"
|
"LIMIT": "Limite",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Rechercher...",
|
"IN_PROGRESS": "Rechercher...",
|
||||||
|
@ -987,16 +987,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Repositórios Populares",
|
"TOP_REPO": "Repositórios Populares",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "ESTATÍSTICAS",
|
|
||||||
"PRO_ITEM": "PROJETOS",
|
"PRO_ITEM": "PROJETOS",
|
||||||
"REPO_ITEM": "REPOSITÓRIOS",
|
"REPO_ITEM": "REPOSITÓRIOS",
|
||||||
"INDEX_PRIVATE": "PRIVADO",
|
"INDEX_PRIVATE": "PRIVADO",
|
||||||
"INDEX_MY_PROJECTS": "MEUS PROJETOS",
|
|
||||||
"INDEX_MY_REPOSITORIES": "MEUS REPOSITÓRIOS",
|
|
||||||
"INDEX_PUB": "PUBLICO",
|
"INDEX_PUB": "PUBLICO",
|
||||||
"INDEX_TOTAL": "TOTAL",
|
"INDEX_TOTAL": "TOTAL",
|
||||||
"STORAGE": "ARMAZENAMENTO",
|
"STORAGE": "ARMAZENAMENTO",
|
||||||
"LIMIT": "Limite"
|
"LIMIT": "Limite",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Buscando...",
|
"IN_PROGRESS": "Buscando...",
|
||||||
|
@ -991,16 +991,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "Popüler Depolar",
|
"TOP_REPO": "Popüler Depolar",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "İSTATİSTİK",
|
|
||||||
"PRO_ITEM": "PROJELER",
|
"PRO_ITEM": "PROJELER",
|
||||||
"REPO_ITEM": "DEPOLAR",
|
"REPO_ITEM": "DEPOLAR",
|
||||||
"INDEX_PRIVATE": "ÖZEL",
|
"INDEX_PRIVATE": "ÖZEL",
|
||||||
"INDEX_MY_PROJECTS": "BENİM PROJELERİM",
|
|
||||||
"INDEX_MY_REPOSITORIES": "BENİM DEPOLARIM",
|
|
||||||
"INDEX_PUB": "GENEL",
|
"INDEX_PUB": "GENEL",
|
||||||
"INDEX_TOTAL": "TOPLAM",
|
"INDEX_TOTAL": "TOPLAM",
|
||||||
"STORAGE": "DEPOLAMA",
|
"STORAGE": "DEPOLAMA",
|
||||||
"LIMIT": "Limit"
|
"LIMIT": "Limit",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "Ara...",
|
"IN_PROGRESS": "Ara...",
|
||||||
|
@ -991,16 +991,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "受欢迎的镜像仓库",
|
"TOP_REPO": "受欢迎的镜像仓库",
|
||||||
"STATISTICS": {
|
"STATISTICS": {
|
||||||
"TITLE": "统计",
|
|
||||||
"PRO_ITEM": "项目",
|
"PRO_ITEM": "项目",
|
||||||
"REPO_ITEM": "镜像仓库",
|
"REPO_ITEM": "镜像仓库",
|
||||||
"INDEX_PRIVATE": "私有",
|
"INDEX_PRIVATE": "私有",
|
||||||
"INDEX_MY_PROJECTS": "我的项目",
|
|
||||||
"INDEX_MY_REPOSITORIES": "我的镜像仓库",
|
|
||||||
"INDEX_PUB": "公开",
|
"INDEX_PUB": "公开",
|
||||||
"INDEX_TOTAL": "总计",
|
"INDEX_TOTAL": "总计",
|
||||||
"STORAGE": "存储",
|
"STORAGE": "存储",
|
||||||
"LIMIT": "容量"
|
"LIMIT": "容量",
|
||||||
|
"STORAGE_USED": "已使用的存储空间"
|
||||||
},
|
},
|
||||||
"SEARCH": {
|
"SEARCH": {
|
||||||
"IN_PROGRESS": "搜索中...",
|
"IN_PROGRESS": "搜索中...",
|
||||||
|
@ -987,16 +987,14 @@
|
|||||||
},
|
},
|
||||||
"TOP_REPO": "受歡迎的鏡像倉庫",
|
"TOP_REPO": "受歡迎的鏡像倉庫",
|
||||||
"STATISTICS":{
|
"STATISTICS":{
|
||||||
"TITLE": "統計",
|
|
||||||
"PRO_ITEM": "項目",
|
"PRO_ITEM": "項目",
|
||||||
"REPO_ITEM": "鏡像倉庫",
|
"REPO_ITEM": "鏡像倉庫",
|
||||||
"INDEX_PRIVATE": "私有",
|
"INDEX_PRIVATE": "私有",
|
||||||
"INDEX_MY_PROJECTS": "我的項目",
|
|
||||||
"INDEX_MY_REPOSITORIES": "我的鏡像倉庫",
|
|
||||||
"INDEX_PUB": "公開",
|
"INDEX_PUB": "公開",
|
||||||
"INDEX_TOTAL": "總計",
|
"INDEX_TOTAL": "總計",
|
||||||
"STORAGE": "存儲",
|
"STORAGE": "存儲",
|
||||||
"LIMIT": "容量"
|
"LIMIT": "容量",
|
||||||
|
"STORAGE_USED": "Storage used"
|
||||||
},
|
},
|
||||||
"SEARCH":{
|
"SEARCH":{
|
||||||
"IN_PROGRESS": "搜索中...",
|
"IN_PROGRESS": "搜索中...",
|
||||||
|
Loading…
Reference in New Issue
Block a user