
After change 290410, we have converted gr-rest-api into a service and no longer require this import. Change-Id: I7990253fc5c25bab8d9d28ee594754a2eb78947d
441 lines
12 KiB
TypeScript
441 lines
12 KiB
TypeScript
/**
|
|
* @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-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 {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';
|
|
import {appContext} from '../../../services/app-context';
|
|
|
|
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',
|
|
},
|
|
};
|
|
|
|
@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;
|
|
|
|
private restApiService = appContext.restApiService;
|
|
|
|
/** @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.restApiService.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.restApiService.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.restApiService.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.restApiService.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.restApiService.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') {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
const repoConfigObj: any = repoConfig[key];
|
|
if (repoConfigObj.configured_value) {
|
|
configInputObj[key as keyof ConfigInput] =
|
|
repoConfigObj.configured_value;
|
|
}
|
|
} else {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
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.restApiService
|
|
.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;
|
|
}
|
|
}
|