Feature: Add helmchart version result in global search result

Include helmchart to global search resault related to ##5791

Signed-off-by: Qian Deng <dengq@vmware.com>
This commit is contained in:
Qian Deng 2018-10-22 19:57:23 +08:00
parent 0c901a475f
commit 2eff6a794a
14 changed files with 125 additions and 14 deletions

View File

@ -296,7 +296,11 @@ export interface ScrollPosition {
sT: number;
cH: number;
}
export interface HelmChartSearchResultItem {
Name: string;
Score: number;
Chart: HelmChartVersion;
}
export interface HelmChartItem {
name: string;
total_versions: number;

View File

@ -10,5 +10,7 @@
<list-project-ro [projects]="searchResults.project"></list-project-ro>
<h2>{{'PROJECT_DETAIL.REPOSITORIES' | translate}}</h2>
<list-repository-ro [repositories]="searchResults.repository"></list-repository-ro>
<h2>{{'HELM_CHART.HELMCHARTS' | translate}}</h2>
<list-chart-version-ro [charts]="searchResults.Chart"></list-chart-version-ro>
</div>
</div>

View File

@ -76,7 +76,7 @@ export class SearchResultComponent implements OnInit, OnDestroy {
if (src) {
src.project.forEach(pro => res.project.push(Object.assign({}, pro)));
src.repository.forEach(repo => res.repository.push(Object.assign({}, repo)));
src.Chart.forEach(chart => res.Chart.push(JSON.parse(JSON.stringify(chart))));
return res;
}
@ -135,7 +135,7 @@ export class SearchResultComponent implements OnInit, OnDestroy {
this.search.doSearch(term)
.then(searchResults => {
this.onGoing = false;
this.originalCopy = searchResults; // Keeo the original data
this.originalCopy = searchResults; // Keep the original data
this.searchResults = this.clone(searchResults);
})
.catch(error => {

View File

@ -12,14 +12,16 @@
// See the License for the specific language governing permissions and
// limitations under the License.
import { Project } from "../../project/project";
import { Repository } from "@harbor/ui";
import { Repository, HelmChartSearchResultItem } from "@harbor/ui";
export class SearchResults {
constructor() {
this.project = [];
this.repository = [];
this.Chart = [];
}
project: Project[];
repository: Repository[];
Chart: HelmChartSearchResultItem[];
}

View File

@ -35,7 +35,7 @@ export class ProjectService {
catchError(error => observableThrowError(error)), );
}
listProjects(name: string, isPublic: number, page?: number, pageSize?: number): Observable<any> {
listProjects(name: string, isPublic?: number, page?: number, pageSize?: number): Observable<any> {
let params = new URLSearchParams();
if (page && pageSize) {
params.set('page', page + '');
@ -47,7 +47,6 @@ export class ProjectService {
if (isPublic !== undefined) {
params.set('public', '' + isPublic);
}
return this.http
.get(`/api/projects`, buildHttpRequestOptions(params)).pipe(
map(response => response),

View File

@ -0,0 +1,25 @@
<clr-datagrid>
<clr-dg-column>{{'HELM_CHART.NAME' | translate}}</clr-dg-column>
<clr-dg-column>{{'HELM_CHART.VERSION' | translate}}</clr-dg-column>
<clr-dg-column>{{'HELM_CHART.STATUS' | translate }}</clr-dg-column>
<clr-dg-column>{{'HELM_CHART.MAINTAINERS' | translate }}</clr-dg-column>
<clr-dg-column>{{'HELM_CHART.CREATED' | translate }}</clr-dg-column>
<clr-dg-placeholder>{{'HELM_CHART.NO_VERSION_PLACEHOLDER' | translate }}</clr-dg-placeholder>
<clr-dg-row *clrDgItems="let chart of charts" [clrDgItem]="chart">
<clr-dg-cell>
<a href="javascript:void(0)" (click)="gotoChartVersion(chart.Chart)">{{ chart.Name }}</a>
</clr-dg-cell>
<clr-dg-cell>{{ chart.Chart.version }}</clr-dg-cell>
<clr-dg-cell>{{ getStatusString(chart.Chart) | translate }}</clr-dg-cell>
<clr-dg-cell>{{ getMaintainerString(chart.Chart.maintainers) | translate : getMaintainerTranslateInfo(chart.Chart.maintainers) }}</clr-dg-cell>
<clr-dg-cell>{{ chart.Chart.created | date}}</clr-dg-cell>
</clr-dg-row>
<clr-dg-footer>
<clr-dg-pagination #pagination [clrDgPageSize]="5" [clrDgTotalItems]="pagination.totalItems">
<span *ngIf="pagination.totalItems">
{{pagination.firstItem + 1}} - {{pagination.lastItem + 1}} {{'HELM_CHART.OF' | translate}}
</span>
{{pagination.totalItems}} {{'HELM_CHART.ITEMS'| translate}}
</clr-dg-pagination>
</clr-dg-footer>
</clr-datagrid>

View File

@ -0,0 +1,69 @@
import { extractJson } from './../shared.utils';
import { Router } from '@angular/router';
import { Component, OnInit, Input } from '@angular/core';
import { HelmChartSearchResultItem, HelmChartVersion, HelmChartMaintainer } from '@harbor/ui';
import { SearchTriggerService } from '../../base/global-search/search-trigger.service';
import { ProjectService } from '../../project/project.service';
@Component({
selector: 'list-chart-version-ro',
templateUrl: './list-chart-version-ro.component.html'
})
export class ListChartVersionRoComponent implements OnInit {
@Input() projectId: number;
@Input() charts: HelmChartSearchResultItem;
constructor(
private searchTrigger: SearchTriggerService,
private projectService: ProjectService,
private router: Router) { }
ngOnInit() {
}
getStatusString(chart: HelmChartVersion) {
if (chart.deprecated) {
return "HELM_CHART.DEPRECATED";
} else {
return "HELM_CHART.ACTIVE";
}
}
getMaintainerString(maintainers: HelmChartMaintainer[]) {
if (!maintainers || maintainers.length < 1) {
return "";
}
let maintainer_string = maintainers[0].name;
if (maintainers.length > 1) {
maintainer_string = "HELM_CHART.OTHER_MAINTAINERS";
}
return maintainer_string;
}
getMaintainerTranslateInfo(maintainers: HelmChartMaintainer[]) {
if (!maintainers || maintainers.length < 1) {
return {};
}
let name = maintainers[0].name;
let number = maintainers.length;
return {name: name, number: number};
}
gotoChartVersion(chartVersion: HelmChartVersion) {
this.searchTrigger.closeSearch(true);
let [projectName, chartName] = chartVersion.name.split('/');
this.projectService.listProjects(projectName).subscribe( res => {
let projects = extractJson(res);
if (projects || projects.length >= 1) {
let linkUrl = ['harbor', 'projects', projects[0].project_id, 'helm-charts', chartName, 'versions', chartVersion.version];
this.router.navigate(linkUrl);
} else {
return;
}
});
}
}

View File

@ -30,11 +30,7 @@ export class ListProjectROComponent {
constructor(
private searchTrigger: SearchTriggerService,
private router: Router,
private ref: ChangeDetectorRef) {
let hnd = setInterval(() => ref.markForCheck(), 100);
setTimeout(() => clearInterval(hnd), 1000);
}
private router: Router) {}
goToLink(proId: number): void {
this.searchTrigger.closeSearch(true);

View File

@ -52,6 +52,7 @@ import { GaugeComponent } from "./gauge/gauge.component";
import { ConfirmationDialogComponent } from "./confirmation-dialog/confirmation-dialog.component";
import { ConfirmationDialogService } from "./confirmation-dialog/confirmation-dialog.service";
import { MessageHandlerService } from "./message-handler/message-handler.service";
import { ListChartVersionRoComponent } from './list-chart-version-ro/list-chart-version-ro.component';
const uiLibConfig: IServiceConfig = {
enablei18Support: true,
@ -103,7 +104,8 @@ const uiLibConfig: IServiceConfig = {
ListProjectROComponent,
ListRepositoryROComponent,
GaugeComponent,
DateValidatorDirective
DateValidatorDirective,
ListChartVersionRoComponent
],
exports: [
CoreModule,
@ -124,7 +126,8 @@ const uiLibConfig: IServiceConfig = {
GaugeComponent,
DateValidatorDirective,
FormsModule,
ReactiveFormsModule
ReactiveFormsModule,
ListChartVersionRoComponent
],
providers: [
SessionService,

View File

@ -12,7 +12,7 @@
// See the License for the specific language governing permissions and
// limitations under the License.
import { NgForm } from '@angular/forms';
import {RequestOptions, Headers} from "@angular/http";
import {RequestOptions, Headers, Response} from "@angular/http";
import { Comparator, State } from '@clr/angular';
import {RequestQueryParams} from "@harbor/ui";
@ -301,3 +301,10 @@ export function doSorting<T extends { [key: string]: any | any[] }>(items: T[],
return comp;
});
}
export const extractJson = (res: Response) => {
if (res.text() === '') {
return [];
}
return (res.json() || []);
};

View File

@ -490,6 +490,7 @@
"SRC_REPO": "Source Repository",
"CREATED": "Created Time",
"MAINTAINERS": "Maintainers",
"OTHER_MAINTAINERS": "{{ name }} and {{ number }} others",
"PULLS": "Pull Count",
"VERSION": "Version",
"APP_VERSION": "Application Version",

View File

@ -489,6 +489,7 @@
"SRC_REPO": "Source Repository",
"CREATED": "Created Time",
"MAINTAINERS": "Maintainers",
"OTHER_MAINTAINERS": "{{ name }} and {{ number }} others",
"PULLS": "Pull Count",
"VERSION": "Version",
"APP_VERSION": "Application Version",

View File

@ -466,6 +466,7 @@
"SRC_REPO": "Source Repository",
"CREATED": "Created Time",
"MAINTAINERS": "Maintainers",
"OTHER_MAINTAINERS": "{{ name }} and {{ number }} others",
"PULLS": "Pull Count",
"VERSION": "Version",
"APP_VERSION": "Application Version",

View File

@ -489,6 +489,7 @@
"SRC_REPO": "源仓库",
"CREATED": "创建时间",
"MAINTAINERS": "维护者",
"OTHER_MAINTAINERS": "{{ name }} 与其他 {{ number }} 位",
"PULLS": "拉取数",
"VERSION": "版本",
"INSTALL": "安装",