harbor/src/ui_ng/lib
2017-09-08 19:15:50 +08:00
..
pkg upgrade ui lib version to 0.4.2 2017-08-04 19:21:38 +08:00
src enable server end pagination for project list 2017-09-08 19:15:50 +08:00
angular-cli.json Enable UI UT test framework (karma and jasmine) 2017-04-24 19:44:12 +08:00
AUTHORS Setup UI shared components libarary 2017-04-20 14:58:31 +08:00
index.ts Setup UI shared components libarary 2017-04-20 14:58:31 +08:00
karma.conf.js Enable UI UT test framework (karma and jasmine) 2017-04-24 19:44:12 +08:00
karma.conf.win.js fixed issue #2688 2017-07-11 21:23:32 -07:00
LICENSE Setup UI shared components libarary 2017-04-20 14:58:31 +08:00
package.json update the library version 2017-08-09 15:03:33 +08:00
README.md add CVE link 2017-08-14 15:56:55 +08:00
rollup.config.js add third party lib 2017-06-21 14:49:29 +08:00
tsconfig.json Enable server-driven pagination for recent logs 2017-06-07 14:36:55 +08:00
tslint.json Setup UI shared components libarary 2017-04-20 14:58:31 +08:00
typings.json Setup UI shared components libarary 2017-04-20 14:58:31 +08:00

Harbor UI library

NOTES: Odd version is development version and may be not stable. Even version is release version which should be stable.

Wrap the following Harbor UI components into a sharable library and published as npm package for other third-party applications to import and reuse.

  • Repository and tag management view
  • Replication rules and jobs management view
  • Replication endpoints management view
  • Access log list view
  • Vulnerability scanning result bar chart and list view (Embedded in tag management view)
  • Registry(Harbor) related configuration options

The Harbor UI library is built on Angular 4.x and Clarity 0.9.x .

The library is published to the public npm repository with name harbor-ui.

Build & Test

Build library with command:

npm run build

Execute the testing specs with command:

npm run test

Install the package

npm install harbor-ui[@version]

Usage

Add dependency to application

Execute install command to add dependency to package.json

npm install harbor-ui --save

//OR
npm install harbor-ui@0.2.x --save

The latest version of the library will be installed.

Import the library module into the root Angular module

import { HarborLibraryModule } from 'harbor-ui';

@NgModule({
    declarations: [...],
    imports: [
        HarborLibraryModule.forRoot()
    ],
    providers: [...],
    bootstrap: [...]
})
export class AppModule {
}

If no parameters are passed to 'forRoot', the module will be initialized with default configurations. If re-configuration required, please refer the 'Configurations' parts.

Enable components via tags

  • Registry log view

Use withTitle to set whether self-contained a header with title or not. Default is false, that means no header is existing.

<hbr-log [withTitle]="..."></hbr-log>
  • Replication Management View

Support two different display scope mode: under specific project or whole system.

If projectId is set to the id of specified project, then only show the replication rules bound with the project. Otherwise, show all the rules of the whole system.

withReplicationJob is used to determine whether or not show the replication jobs which are relevant with the selected replication rule.

readonly is to disable all the create/edit/delete actions.

<hbr-replication [projectId]="..." [withReplicationJob]='...' [readonly]="..."></hbr-replication>
  • Endpoint Management View
//No @Input properties

<hbr-endpoint></hbr-endpoint>
  • Repository and Tag Management View

projectId is used to specify which projects the repositories are from.

projectName is used to generate the related commands for pushing images.

hasSignedIn is a user session related property to determined whether a valid user signed in session existing. This component supports anonymous user.

hasProjectAdminRole is a user session related property to determined whether the current user has project administrator role. Some action menus might be disabled based on this property.

tagClickEvent is an @output event emitter for you to catch the tag click events.

<hbr-repository-stackview [projectId]="..." [projectName]="" [hasSignedIn]="..." [hasProjectAdminRole]="..." (tagClickEvent)="watchTagClickEvent($event)"></hbr-repository-stackview>

...

watchTagClickEvent(tag: Tag): void {
    //Process tag
    ...
}

  • Tag detail view

This view is linked by the repository stack view only when the Clair is enabled in Harbor.

tagId is an @Input property and used to specify the tag of which details are displayed.

repositoryId is an @Input property and used to specified the repository to which the tag is belonged.

backEvt is an @Output event emitter and used to distribute the click event of the back arrow in the detail page.

<hbr-tag-detail (backEvt)="goBack($event)" [tagId]="..." [repositoryId]="..."></hbr-tag-detail>
  • Registry related configuration

This component provides some options for registry(Harbor) related configurations.

hasAdminRole is an @Input property to indicate if the current logged user has administrator role.

<hbr-registry-config [hasAdminRole]="***"></hbr-registry-config>

Configurations

All the related configurations are defined in the HarborModuleConfig interface.

1. config The base configuration for the module. Mainly used to define the relevant endpoints of services which are in charge of retrieving data from backend APIs. It's a 'OpaqueToken' and defined by 'IServiceConfig' interface. If config is not set, the default value will be used.

export const DefaultServiceConfig: IServiceConfig = {
  systemInfoEndpoint: "/api/systeminfo",
  repositoryBaseEndpoint: "/api/repositories",
  logBaseEndpoint: "/api/logs",
  targetBaseEndpoint: "/api/targets",
  replicationRuleEndpoint: "/api/policies/replication",
  replicationJobEndpoint: "/api/jobs/replication",
  vulnerabilityScanningBaseEndpoint: "/api/repositories",
  configurationEndpoint: "/api/configurations",
  enablei18Support: false,
  defaultLang: DEFAULT_LANG, //'en-us'
  langCookieKey: DEFAULT_LANG_COOKIE_KEY, //'harbor-lang'
  supportedLangs: DEFAULT_SUPPORTING_LANGS,//['en-us','zh-cn','es-es']
  langMessageLoader: "local",
  langMessagePathForHttpLoader: "i18n/langs/",
  langMessageFileSuffixForHttpLoader: "-lang.json",
  localI18nMessageVariableMap: {}
};

If you want to override the related items, declare your own 'IServiceConfig' interface and define the configuration value. E.g: Override 'repositoryBaseEndpoint'

export const MyServiceConfig: IServiceConfig = {
    repositoryBaseEndpoint: "/api/wrap/repositories"
}

...
HarborLibraryModule.forRoot({
    config: { provide: SERVICE_CONFIG, useValue: MyServiceConfig }
})
...

It supports partially overriding. For the items not overridden, default values will be adopted. The items contained in config are:

  • systemInfoEndpoint: The base endpoint of the service used to get the related system configurations. Default value is "/api/systeminfo".

  • repositoryBaseEndpoint: The base endpoint of the service used to handle the repositories of registry and/or tags of repository. Default value is "/api/repositories".

  • logBaseEndpoint: The base endpoint of the service used to handle the recent access logs. Default is "/api/logs".

  • targetBaseEndpoint: The base endpoint of the service used to handle the registry endpoints. Default is "/api/targets".

  • replicationRuleEndpoint: The base endpoint of the service used to handle the replication rules. Default is "/api/policies/replication".

  • replicationJobEndpoint: The base endpoint of the service used to handle the replication jobs. Default is "/api/jobs/replication".

  • vulnerabilityScanningBaseEndpoint: The base endpoint of the service used to handle the vulnerability scanning results.Default value is "/api/repositories".

  • configurationEndpoint: The base endpoint of the service used to configure registry related options. Default is "/api/configurations".

  • langCookieKey: The cookie key used to store the current used language preference. Default is "harbor-lang".

  • supportedLangs: Declare what languages are supported. Default is ['en-us', 'zh-cn', 'es-es'].

  • enablei18Support: To determine whether or not to enable the i18 multiple languages supporting. Default is false.

  • langMessageLoader: To determine which loader will be used to load the required lang messages. Support two loaders: One is 'http', use async http to load json files with the specified url/path. Another is 'local', use local json variable to store the lang message.

  • langMessagePathForHttpLoader: Define the basic url/path prefix for the loader to find the json files if the 'langMessageLoader' is set to 'http'. E.g: 'src/i18n/langs'.

  • langMessageFileSuffixForHttpLoader: Define the suffix of the json file names without lang name if 'langMessageLoader' is set to 'http'. For example, '-lang.json' is suffix of message file 'en-us-lang.json'.

  • localI18nMessageVariableMap: If configuration property 'langMessageLoader' is set to 'local' to load the i18n messages, this property must be defined to tell local JSON loader where to get the related messages. E.g: If declare the following messages storage variables,

        export const EN_US_LANG: any = {
            "APP_TITLE": {
                "VMW_HARBOR": "VMware Harbor",
                "HARBOR": "Harbor"
            }
        }
        
        export const ZH_CN_LANG: any = {
            "APP_TITLE": {
                "VMW_HARBOR": "VMware Harbor中文版",
                "HARBOR": "Harbor"
            }
        }

then this property should be set to:

        {
            "en-us": EN_US_LANG,
            "zh-cn": ZH_CN_LANG
        };

2. errorHandler UI components in the library use this interface to pass the errors/warnings/infos/logs to the top component or page. The top component or page can display those information in their message panel or notification system. If not set, the console will be used as default output approach.

@Injectable()
export class MyErrorHandler extends ErrorHandler {
    public error(error: any): void {
        ...
    }

    public warning(warning: any): void {
        ...
    }

    public info(info: any): void {
        ...
    }

    public log(log: any): void {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    errorHandler: { provide: ErrorHandler, useClass: MyErrorHandler }
})
...

3. user session Some components may need the user authorization and authentication information to display different views. The following way of handing user session is supported by the library.

  • Use @Input properties or interface to let top component or page to pass the required user session information in.
//In the above repository stack view, the user session informations are passed via @input properties.
[hasSignedIn]="..." [hasProjectAdminRole]="..."

4. services The library has its own service implementations to communicate with backend APIs and transfer data. If you want to use your own data handling logic, you can implement your own services based on the defined interfaces.

  • AccessLogService: Define service methods to handle the access log related things.
@Injectable()
export class MyAccessLogService extends AccessLogService {
     /**
     * Get the audit logs for the specified project.
     * Set query parameters through 'queryParams', support:
     *  - page
     *  - pageSize
     * 
     * @param {(number | string)} projectId
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<AccessLog[]> | Promise<AccessLog[]> | AccessLog[])}
     * 
     * @memberOf AccessLogService
     */
    getAuditLogs(projectId: number | string, queryParams?: RequestQueryParams): Observable<AccessLog[]> | Promise<AccessLog[]> | AccessLog[]{
        ...
    }

    /**
     * Get the recent logs.
     * 
     * @param {number} lines : Specify how many lines should be returned.
     * @returns {(Observable<AccessLog[]> | Promise<AccessLog[]> | AccessLog[])}
     * 
     * @memberOf AccessLogService
     */
    getRecentLogs(lines: number): Observable<AccessLog[]> | Promise<AccessLog[]> | AccessLog[]{
        ...
    }
}

...
HarborLibraryModule.forRoot({
    logService: { provide: AccessLogService, useClass: MyAccessLogService }
})
...

  • EndpointService: Define the service methods to handle the endpoint related things.
@Injectable()
export class MyEndpointService extends EndpointService {
    /**
     * Get all the endpoints.
     * Set the argument 'endpointName' to return only the endpoints match the name pattern.
     * 
     * @param {string} [endpointName]
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<Endpoint[]> | Endpoint[])}
     * 
     * @memberOf EndpointService
     */
    getEndpoints(endpointName?: string, queryParams?: RequestQueryParams): Observable<Endpoint[]> | Promise<Endpoint[]> | Endpoint[] {
        ...
    }

    /**
     * Get the specified endpoint.
     * 
     * @param {(number | string)} endpointId
     * @returns {(Observable<Endpoint> | Endpoint)}
     * 
     * @memberOf EndpointService
     */
    getEndpoint(endpointId: number | string): Observable<Endpoint> | Promise<Endpoint> | Endpoint {
        ...
    }

    /**
     * Create new endpoint.
     * 
     * @param {Endpoint} endpoint
     * @returns {(Observable<any> | any)}
     * 
     * @memberOf EndpointService
     */
    createEndpoint(endpoint: Endpoint): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Update the specified endpoint.
     * 
     * @param {(number | string)} endpointId
     * @param {Endpoint} endpoint
     * @returns {(Observable<any> | any)}
     * 
     * @memberOf EndpointService
     */
    updateEndpoint(endpointId: number | string, endpoint: Endpoint): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Delete the specified endpoint.
     * 
     * @param {(number | string)} endpointId
     * @returns {(Observable<any> | any)}
     * 
     * @memberOf EndpointService
     */
    deleteEndpoint(endpointId: number | string): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Ping the specified endpoint.
     * 
     * @param {Endpoint} endpoint
     * @returns {(Observable<any> | any)}
     * 
     * @memberOf EndpointService
     */
    pingEndpoint(endpoint: Endpoint): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Check endpoint whether in used with specific replication rule.
     * 
     * @param {{number | string}} endpointId
     * @returns {{Observable<any> | any}}
     */
    getEndpointWithReplicationRules(endpointId: number | string): Observable<any> | Promise<any> | any {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    endpointService: { provide: EndpointService, useClass: MyEndpointService }
})
...

  • ReplicationService: Define the service methods to handle the replication (rule and job) related things.
@Injectable()
export class MyReplicationService extends ReplicationService {
     /**
     * Get the replication rules.
     * Set the argument 'projectId' to limit the data scope to the specified project;
     * set the argument 'ruleName' to return the rule only match the name pattern;
     * if pagination needed, use the queryParams to add query parameters.
     * 
     * @param {(number | string)} [projectId]
     * @param {string} [ruleName]
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<ReplicationRule[]> | Promise<ReplicationRule[]> | ReplicationRule[])}
     * 
     * @memberOf ReplicationService
     */
    getReplicationRules(projectId?: number | string, ruleName?: string, queryParams?: RequestQueryParams): Observable<ReplicationRule[]> | Promise<ReplicationRule[]> | ReplicationRule[] {
        ...
    }

    /**
     * Get the specified replication rule.
     * 
     * @param {(number | string)} ruleId
     * @returns {(Observable<ReplicationRule> | Promise<ReplicationRule> | ReplicationRule)}
     * 
     * @memberOf ReplicationService
     */
    getReplicationRule(ruleId: number | string): Observable<ReplicationRule> | Promise<ReplicationRule> | ReplicationRule {
        ...
    }

    /**
     * Create new replication rule.
     * 
     * @param {ReplicationRule} replicationRule
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ReplicationService
     */
    createReplicationRule(replicationRule: ReplicationRule): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Update the specified replication rule.
     * 
     * @param {ReplicationRule} replicationRule
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ReplicationService
     */
    updateReplicationRule(replicationRule: ReplicationRule): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Delete the specified replication rule.
     * 
     * @param {(number | string)} ruleId
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ReplicationService
     */
    deleteReplicationRule(ruleId: number | string): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Enable the specified replication rule.
     * 
     * @param {(number | string)} ruleId
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ReplicationService
     */
    enableReplicationRule(ruleId: number | string, enablement: number): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Disable the specified replication rule.
     * 
     * @param {(number | string)} ruleId
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ReplicationService
     */
    disableReplicationRule(ruleId: number | string): Observable<any> | Promise<any> | any {
        ...
    }

    /**
     * Get the jobs for the specified replication rule.
     * Set query parameters through 'queryParams', support:
     *   - status
     *   - repository
     *   - startTime and endTime
     *   - page
     *   - pageSize
     * 
     * @param {(number | string)} ruleId
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<ReplicationJob> | Promise<ReplicationJob[]> | ReplicationJob)}
     * 
     * @memberOf ReplicationService
     */
    getJobs(ruleId: number | string, queryParams?: RequestQueryParams): Observable<ReplicationJob[]> | Promise<ReplicationJob[]> | ReplicationJob[] {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    replicationService: { provide: ReplicationService, useClass: MyReplicationService }
})
...

  • RepositoryService: Define service methods for handling the repository related things.
@Injectable()
export class MyRepositoryService extends RepositoryService {
    /**
     * List all the repositories in the specified project.
     * Specify the 'repositoryName' to only return the repositories which match the name pattern.
     * If pagination needed, set the following parameters in queryParams:
     *   'page': current page,
     *   'page_size': page size.
     * 
     * @param {(number | string)} projectId
     * @param {string} repositoryName
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<Repository[]> | Promise<Repository[]> | Repository[])}
     * 
     * @memberOf RepositoryService
     */
    getRepositories(projectId: number | string, repositoryName?: string, queryParams?: RequestQueryParams): Observable<Repository[]> | Promise<Repository[]> | Repository[] {
        ...
    }

    /**
     * DELETE the specified repository.
     * 
     * @param {string} repositoryName
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf RepositoryService
     */
    deleteRepository(repositoryName: string): Observable<any> | Promise<any> | any {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    repositoryService: { provide: RepositoryService, useClass: MyRepositoryService }
})
...

@Injectable()
export class MyTagService extends TagService {
    /**
     * Get all the tags under the specified repository.
     * NOTES: If the Notary is enabled, the signatures should be included in the returned data.
     *  
     * @param {string} repositoryName
     * @param {RequestQueryParams} [queryParams]
     * @returns {(Observable<Tag[]> | Promise<Tag[]> | Tag[])}
     * 
     * @memberOf TagService
     */
    getTags(repositoryName: string, queryParams?: RequestQueryParams): Observable<Tag[]> | Promise<Tag[]> | Tag[] {
        ...
    }

    /**
     * Delete the specified tag.
     * 
     * @param {string} repositoryName
     * @param {string} tag
     * @returns {(Observable<any> | any)}
     * 
     * @memberOf TagService
     */
    deleteTag(repositoryName: string, tag: string): Observable<any> | Promise<Tag> | any {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    tagService: { provide: TagService, useClass: MyTagService }
})
...

  • ScanningResultService: Get the vulnerabilities scanning results for the specified tag.
@Injectable()
/**
 * Get the vulnerabilities scanning results for the specified tag.
 * 
 * @export
 * @class ScanningResultService
 */
export class MyScanningResultService extends ScanningResultService {
    /**
     * Get the summary of vulnerability scanning result.
     * 
     * @param {string} tagId
     * @returns {(Observable<VulnerabilitySummary> | Promise<VulnerabilitySummary> | VulnerabilitySummary)}
     * 
     * @memberOf ScanningResultService
     */
    getVulnerabilityScanningSummary(repoName: string, tagId: string, queryParams?: RequestQueryParams): Observable<VulnerabilitySummary> | Promise<VulnerabilitySummary> | VulnerabilitySummary{
        ...
    }

    /**
     * Get the detailed vulnerabilities scanning results.
     * 
     * @param {string} tagId
     * @returns {(Observable<VulnerabilityItem[]> | Promise<VulnerabilityItem[]> | VulnerabilityItem[])}
     * 
     * @memberOf ScanningResultService
     */
    getVulnerabilityScanningResults(repoName: string, tagId: string, queryParams?: RequestQueryParams): Observable<VulnerabilityItem[]> | Promise<VulnerabilityItem[]> | VulnerabilityItem[]{
        ...
    }


    /**
     * Start a new vulnerability scanning
     * 
     * @param {string} repoName
     * @param {string} tagId
     * @returns {(Observable<any> | Promise<any> | any)}
     * 
     * @memberOf ScanningResultService
     */
    startVulnerabilityScanning(repoName: string, tagId: string): Observable<any> | Promise<any> | any {
        ...
    }
}

...
HarborLibraryModule.forRoot({
    scanningService: { provide: ScanningResultService, useClass: MyScanningResultService }
})
...

  • SystemInfoService: Get related system configurations.
/**
 * Get System information about current backend server.
 * @class
 */
export class MySystemInfoService extends SystemInfoService {
  /**
   *  Get global system information.
   *  @returns 
   */
   getSystemInfo(): Observable<SystemInfo> | Promise<SystemInfo> | SystemInfo {
       ...
   }
}

...
HarborLibraryModule.forRoot({
    systemInfoService: { provide: SystemInfoService, useClass: MySystemInfoService }
})
...

  • ConfigurationService: Get and save the registry related configuration options.
/**
 * Service used to get and save registry-related configurations.
 * 
 * @export
 * @class MyConfigurationService
 */
export class MyConfigurationService extends ConfigurationService{

    /**
     * Get configurations.
     * 
    
     * @returns {(Observable<Configuration> | Promise<Configuration> | Configuration)}
     * 
     * @memberOf ConfigurationService
     */
    getConfigurations(): Observable<Configuration> | Promise<Configuration> | Configuration{
        ...
    }

    /**
     * Save configurations.
     * 
    
     * @returns {(Observable<Configuration> | Promise<Configuration> | Configuration)}
     * 
     * @memberOf ConfigurationService
     */
    saveConfigurations(changedConfigs: any | { [key: string]: any | any[] }): Observable<any> | Promise<any> | any{
        ...
    }
}

...
HarborLibraryModule.forRoot({
    config.configService || { provide: ConfigurationService, useClass: ConfigurationDefaultService }
})
...