| /** |
| * @license |
| * Copyright (C) 2017 The Android Open Source Project |
| * |
| * 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 '@polymer/iron-autogrow-textarea/iron-autogrow-textarea'; |
| import '@polymer/iron-input/iron-input'; |
| import '../../plugins/gr-endpoint-decorator/gr-endpoint-decorator'; |
| import '../../plugins/gr-endpoint-param/gr-endpoint-param'; |
| import '../../shared/gr-download-commands/gr-download-commands'; |
| import '../../shared/gr-rest-api-interface/gr-rest-api-interface'; |
| import '../../shared/gr-select/gr-select'; |
| import '../../../styles/gr-form-styles'; |
| import '../../../styles/gr-subpage-styles'; |
| import '../../../styles/shared-styles'; |
| import '../gr-repo-plugin-config/gr-repo-plugin-config'; |
| import {GestureEventListeners} from '@polymer/polymer/lib/mixins/gesture-event-listeners'; |
| import {LegacyElementMixin} from '@polymer/polymer/lib/legacy/legacy-element-mixin'; |
| import {PolymerElement} from '@polymer/polymer/polymer-element'; |
| import {htmlTemplate} from './gr-repo_html'; |
| import {GerritNav} from '../../core/gr-navigation/gr-navigation'; |
| import {customElement, property, observe} from '@polymer/decorators'; |
| import { |
| RestApiService, |
| ErrorCallback, |
| } from '../../../services/services/gr-rest-api/gr-rest-api'; |
| import { |
| ConfigInfo, |
| RepoName, |
| InheritedBooleanInfo, |
| SchemesInfoMap, |
| ConfigInput, |
| PluginParameterToConfigParameterInfoMap, |
| PluginNameToPluginParametersMap, |
| } from '../../../types/common'; |
| import {PluginData} from '../gr-repo-plugin-config/gr-repo-plugin-config'; |
| import {ProjectState} from '../../../constants/constants'; |
| import {PolymerDeepPropertyChange} from '@polymer/polymer/interfaces'; |
| import {hasOwnProperty} from '../../../utils/common-util'; |
| import {firePageError, fireTitleChange} from '../../../utils/event-util'; |
| |
| const STATES = { |
| active: {value: ProjectState.ACTIVE, label: 'Active'}, |
| readOnly: {value: ProjectState.READ_ONLY, label: 'Read Only'}, |
| hidden: {value: ProjectState.HIDDEN, label: 'Hidden'}, |
| }; |
| |
| const SUBMIT_TYPES = { |
| // Exclude INHERIT, which is handled specially. |
| mergeIfNecessary: { |
| value: 'MERGE_IF_NECESSARY', |
| label: 'Merge if necessary', |
| }, |
| fastForwardOnly: { |
| value: 'FAST_FORWARD_ONLY', |
| label: 'Fast forward only', |
| }, |
| rebaseAlways: { |
| value: 'REBASE_ALWAYS', |
| label: 'Rebase Always', |
| }, |
| rebaseIfNecessary: { |
| value: 'REBASE_IF_NECESSARY', |
| label: 'Rebase if necessary', |
| }, |
| mergeAlways: { |
| value: 'MERGE_ALWAYS', |
| label: 'Merge always', |
| }, |
| cherryPick: { |
| value: 'CHERRY_PICK', |
| label: 'Cherry pick', |
| }, |
| }; |
| |
| export interface GrRepo { |
| $: { |
| restAPI: RestApiService & Element; |
| }; |
| } |
| @customElement('gr-repo') |
| export class GrRepo extends GestureEventListeners( |
| LegacyElementMixin(PolymerElement) |
| ) { |
| static get template() { |
| return htmlTemplate; |
| } |
| |
| @property({type: String}) |
| repo?: RepoName; |
| |
| @property({type: Boolean}) |
| _configChanged = false; |
| |
| @property({type: Boolean}) |
| _loading = true; |
| |
| @property({type: Boolean, observer: '_loggedInChanged'}) |
| _loggedIn = false; |
| |
| @property({type: Object}) |
| _repoConfig?: ConfigInfo; |
| |
| @property({ |
| type: Array, |
| computed: '_computePluginData(_repoConfig.plugin_config.*)', |
| }) |
| _pluginData?: PluginData[]; |
| |
| @property({type: Boolean}) |
| _readOnly = true; |
| |
| @property({type: Array}) |
| _states = Object.values(STATES); |
| |
| @property({ |
| type: Array, |
| computed: '_computeSchemes(_schemesDefault, _schemesObj)', |
| observer: '_schemesChanged', |
| }) |
| _schemes: string[] = []; |
| |
| // This is workaround to have _schemes with default value [], |
| // because assignment doesn't work when property has a computed attribute. |
| @property({type: Array}) |
| _schemesDefault: string[] = []; |
| |
| @property({type: String}) |
| _selectedCommand = 'Clone'; |
| |
| @property({type: String}) |
| _selectedScheme?: string; |
| |
| @property({type: Object}) |
| _schemesObj?: SchemesInfoMap; |
| |
| /** @override */ |
| attached() { |
| super.attached(); |
| this._loadRepo(); |
| |
| fireTitleChange(this, `${this.repo}`); |
| } |
| |
| _computePluginData( |
| configRecord: PolymerDeepPropertyChange< |
| PluginNameToPluginParametersMap, |
| PluginNameToPluginParametersMap |
| > |
| ) { |
| if (!configRecord || !configRecord.base) { |
| return []; |
| } |
| |
| const pluginConfig = configRecord.base; |
| return Object.keys(pluginConfig).map(name => { |
| return {name, config: pluginConfig[name]}; |
| }); |
| } |
| |
| _loadRepo() { |
| if (!this.repo) { |
| return Promise.resolve(); |
| } |
| |
| const promises = []; |
| |
| const errFn: ErrorCallback = response => { |
| firePageError(this, response); |
| }; |
| |
| promises.push( |
| this._getLoggedIn().then(loggedIn => { |
| this._loggedIn = loggedIn; |
| if (loggedIn) { |
| const repo = this.repo; |
| if (!repo) throw new Error('undefined repo'); |
| this.$.restAPI.getRepoAccess(repo).then(access => { |
| if (!access || this.repo !== repo) { |
| return; |
| } |
| |
| // If the user is not an owner, is_owner is not a property. |
| this._readOnly = !access[repo].is_owner; |
| }); |
| } |
| }) |
| ); |
| |
| promises.push( |
| this.$.restAPI.getProjectConfig(this.repo, errFn).then(config => { |
| if (!config) { |
| return; |
| } |
| |
| if (config.default_submit_type) { |
| // The gr-select is bound to submit_type, which needs to be the |
| // *configured* submit type. When default_submit_type is |
| // present, the server reports the *effective* submit type in |
| // submit_type, so we need to overwrite it before storing the |
| // config in this. |
| config.submit_type = config.default_submit_type.configured_value; |
| } |
| if (!config.state) { |
| config.state = STATES.active.value; |
| } |
| this._repoConfig = config; |
| this._loading = false; |
| }) |
| ); |
| |
| promises.push( |
| this.$.restAPI.getConfig().then(config => { |
| if (!config) { |
| return; |
| } |
| |
| this._schemesObj = config.download.schemes; |
| }) |
| ); |
| |
| return Promise.all(promises); |
| } |
| |
| _computeLoadingClass(loading: boolean) { |
| return loading ? 'loading' : ''; |
| } |
| |
| _computeHideClass(arr?: PluginData[] | string[]) { |
| return !arr || !arr.length ? 'hide' : ''; |
| } |
| |
| _loggedInChanged(_loggedIn?: boolean) { |
| if (!_loggedIn) { |
| return; |
| } |
| this.$.restAPI.getPreferences().then(prefs => { |
| if (prefs?.download_scheme) { |
| // Note (issue 5180): normalize the download scheme with lower-case. |
| this._selectedScheme = prefs.download_scheme.toLowerCase(); |
| } |
| }); |
| } |
| |
| _formatBooleanSelect(item: InheritedBooleanInfo) { |
| if (!item) { |
| return; |
| } |
| let inheritLabel = 'Inherit'; |
| if (!(item.inherited_value === undefined)) { |
| inheritLabel = `Inherit (${item.inherited_value})`; |
| } |
| return [ |
| { |
| label: inheritLabel, |
| value: 'INHERIT', |
| }, |
| { |
| label: 'True', |
| value: 'TRUE', |
| }, |
| { |
| label: 'False', |
| value: 'FALSE', |
| }, |
| ]; |
| } |
| |
| _formatSubmitTypeSelect(projectConfig: ConfigInfo) { |
| if (!projectConfig) { |
| return; |
| } |
| const allValues = Object.values(SUBMIT_TYPES); |
| const type = projectConfig.default_submit_type; |
| if (!type) { |
| // Server is too old to report default_submit_type, so assume INHERIT |
| // is not a valid value. |
| return allValues; |
| } |
| |
| let inheritLabel = 'Inherit'; |
| if (type.inherited_value) { |
| inheritLabel = `Inherit (${type.inherited_value})`; |
| for (const val of allValues) { |
| if (val.value === type.inherited_value) { |
| inheritLabel = `Inherit (${val.label})`; |
| break; |
| } |
| } |
| } |
| return [ |
| { |
| label: inheritLabel, |
| value: 'INHERIT', |
| }, |
| ...allValues, |
| ]; |
| } |
| |
| _isLoading() { |
| return this._loading || this._loading === undefined; |
| } |
| |
| _getLoggedIn() { |
| return this.$.restAPI.getLoggedIn(); |
| } |
| |
| _formatRepoConfigForSave(repoConfig: ConfigInfo): ConfigInput { |
| const configInputObj: ConfigInput = {}; |
| for (const configKey of Object.keys(repoConfig)) { |
| const key = configKey as keyof ConfigInfo; |
| if (key === 'default_submit_type') { |
| // default_submit_type is not in the input type, and the |
| // configured value was already copied to submit_type by |
| // _loadProject. Omit this property when saving. |
| continue; |
| } |
| if (key === 'plugin_config') { |
| configInputObj.plugin_config_values = repoConfig.plugin_config; |
| } else if (typeof repoConfig[key] === 'object') { |
| const repoConfigObj: any = repoConfig[key]; |
| if (repoConfigObj.configured_value) { |
| configInputObj[key as keyof ConfigInput] = |
| repoConfigObj.configured_value; |
| } |
| } else { |
| configInputObj[key as keyof ConfigInput] = repoConfig[key] as any; |
| } |
| } |
| return configInputObj; |
| } |
| |
| _handleSaveRepoConfig() { |
| if (!this._repoConfig || !this.repo) |
| return Promise.reject(new Error('undefined repoConfig or repo')); |
| return this.$.restAPI |
| .saveRepoConfig( |
| this.repo, |
| this._formatRepoConfigForSave(this._repoConfig) |
| ) |
| .then(() => { |
| this._configChanged = false; |
| }); |
| } |
| |
| @observe('_repoConfig.*') |
| _handleConfigChanged() { |
| if (this._isLoading()) { |
| return; |
| } |
| this._configChanged = true; |
| } |
| |
| _computeButtonDisabled(readOnly: boolean, configChanged: boolean) { |
| return readOnly || !configChanged; |
| } |
| |
| _computeHeaderClass(configChanged: boolean) { |
| return configChanged ? 'edited' : ''; |
| } |
| |
| _computeSchemes(schemesDefault: string[], schemesObj?: SchemesInfoMap) { |
| return !schemesObj ? schemesDefault : Object.keys(schemesObj); |
| } |
| |
| _schemesChanged(schemes: string[]) { |
| if (schemes.length === 0) { |
| return; |
| } |
| if (!this._selectedScheme || !schemes.includes(this._selectedScheme)) { |
| this._selectedScheme = schemes.sort()[0]; |
| } |
| } |
| |
| _computeCommands( |
| repo?: RepoName, |
| schemesObj?: SchemesInfoMap, |
| _selectedScheme?: string |
| ) { |
| if (!schemesObj || !repo || !_selectedScheme) { |
| return []; |
| } |
| const commands = []; |
| let commandObj: {[title: string]: string} = {}; |
| if (hasOwnProperty(schemesObj, _selectedScheme)) { |
| commandObj = schemesObj[_selectedScheme].clone_commands; |
| } |
| for (const title in commandObj) { |
| if (!hasOwnProperty(commandObj, title)) { |
| continue; |
| } |
| commands.push({ |
| title, |
| command: commandObj[title] |
| .replace(/\${project}/gi, encodeURI(repo)) |
| .replace( |
| /\${project-base-name}/gi, |
| encodeURI(repo.substring(repo.lastIndexOf('/') + 1)) |
| ), |
| }); |
| } |
| return commands; |
| } |
| |
| _computeRepositoriesClass(config: InheritedBooleanInfo) { |
| return config ? 'showConfig' : ''; |
| } |
| |
| _computeChangesUrl(name: RepoName) { |
| return GerritNav.getUrlForProjectChanges(name); |
| } |
| |
| _handlePluginConfigChanged({ |
| detail: {name, config, notifyPath}, |
| }: { |
| detail: { |
| name: string; |
| config: PluginParameterToConfigParameterInfoMap; |
| notifyPath: string; |
| }; |
| }) { |
| if (this._repoConfig?.plugin_config) { |
| this._repoConfig.plugin_config[name] = config; |
| this.notifyPath('_repoConfig.plugin_config.' + notifyPath); |
| } |
| } |
| } |
| |
| declare global { |
| interface HTMLElementTagNameMap { |
| 'gr-repo': GrRepo; |
| } |
| } |