Show signed push verification result
Uses colored icons to reflect the signed push verification result. Bug: Issue 9723 Change-Id: Ia6426f128a89992c68d27be8924634bc8093518d
This commit is contained in:

committed by
David Ostrovsky

parent
5486249ce7
commit
c7c05db8bb
@@ -27,6 +27,7 @@ limitations under the License.
|
||||
<link rel="import" href="../../shared/gr-account-link/gr-account-link.html">
|
||||
<link rel="import" href="../../shared/gr-date-formatter/gr-date-formatter.html">
|
||||
<link rel="import" href="../../shared/gr-editable-label/gr-editable-label.html">
|
||||
<link rel="import" href="../../shared/gr-icons/gr-icons.html">
|
||||
<link rel="import" href="../../shared/gr-limited-text/gr-limited-text.html">
|
||||
<link rel="import" href="../../shared/gr-linked-chip/gr-linked-chip.html">
|
||||
<link rel="import" href="../../shared/gr-tooltip-content/gr-tooltip-content.html">
|
||||
@@ -114,6 +115,19 @@ limitations under the License.
|
||||
#parentNotCurrentMessage {
|
||||
display: none;
|
||||
}
|
||||
.icon {
|
||||
margin: -.25em 0;
|
||||
}
|
||||
.icon.help,
|
||||
.icon.notTrusted {
|
||||
color: #FFA62F;
|
||||
}
|
||||
.icon.invalid {
|
||||
color: var(--vote-text-color-disliked);
|
||||
}
|
||||
.icon.trusted {
|
||||
color: var(--vote-text-color-recommended);
|
||||
}
|
||||
.parentList.notCurrent.nonMerge #parentNotCurrentMessage {
|
||||
--arrow-color: #ffa62f;
|
||||
display: inline-block;
|
||||
@@ -137,6 +151,16 @@ limitations under the License.
|
||||
<span class="title">Owner</span>
|
||||
<span class="value">
|
||||
<gr-account-link account="[[change.owner]]"></gr-account-link>
|
||||
<template is="dom-if" if="[[_pushCertificateValidation]]">
|
||||
<gr-tooltip-content
|
||||
has-tooltip
|
||||
title$="[[_pushCertificateValidation.message]]">
|
||||
<iron-icon
|
||||
class$="icon [[_pushCertificateValidation.class]]"
|
||||
icon="[[_pushCertificateValidation.icon]]">
|
||||
</iron-icon>
|
||||
</gr-tooltip-content>
|
||||
</template>
|
||||
</span>
|
||||
</section>
|
||||
<section class$="[[_computeShowRoleClass(change, _CHANGE_ROLE.UPLOADER)]]">
|
||||
|
@@ -17,6 +17,17 @@
|
||||
(function() {
|
||||
'use strict';
|
||||
|
||||
const Defs = {};
|
||||
|
||||
/**
|
||||
* @typedef {{
|
||||
* message: string,
|
||||
* icon: string,
|
||||
* class: string,
|
||||
* }}
|
||||
*/
|
||||
Defs.PushCertificateValidation;
|
||||
|
||||
const HASHTAG_ADD_MESSAGE = 'Add Hashtag';
|
||||
|
||||
const SubmitTypeLabel = {
|
||||
@@ -30,6 +41,24 @@
|
||||
|
||||
const NOT_CURRENT_MESSAGE = 'Not current - rebase possible';
|
||||
|
||||
/**
|
||||
* @enum {string}
|
||||
*/
|
||||
const CertificateStatus = {
|
||||
/**
|
||||
* This certificate status is bad.
|
||||
*/
|
||||
BAD: 'BAD',
|
||||
/**
|
||||
* This certificate status is OK.
|
||||
*/
|
||||
OK: 'OK',
|
||||
/**
|
||||
* This certificate status is TRUSTED.
|
||||
*/
|
||||
TRUSTED: 'TRUSTED',
|
||||
};
|
||||
|
||||
Polymer({
|
||||
is: 'gr-change-metadata',
|
||||
|
||||
@@ -76,6 +105,13 @@
|
||||
type: Boolean,
|
||||
computed: '_computeShowReviewersByState(serverConfig)',
|
||||
},
|
||||
/**
|
||||
* @type {Defs.PushCertificateValidation}
|
||||
*/
|
||||
_pushCertificateValidation: {
|
||||
type: Object,
|
||||
computed: '_computePushCertificateValidation(serverConfig, change)',
|
||||
},
|
||||
_showRequirements: {
|
||||
type: Boolean,
|
||||
computed: '_computeShowRequirements(change)',
|
||||
@@ -260,6 +296,59 @@
|
||||
return hasRequirements || hasLabels || !!change.work_in_progress;
|
||||
},
|
||||
|
||||
/**
|
||||
* @return {?Defs.PushCertificateValidation} object representing data for
|
||||
* the push validation.
|
||||
*/
|
||||
_computePushCertificateValidation(serverConfig, change) {
|
||||
if (!serverConfig || !serverConfig.receive ||
|
||||
!serverConfig.receive.enable_signed_push) {
|
||||
return null;
|
||||
}
|
||||
const rev = change.revisions[change.current_revision];
|
||||
if (!rev.push_certificate || !rev.push_certificate.key) {
|
||||
return {
|
||||
class: 'help',
|
||||
icon: 'gr-icons:help',
|
||||
message: 'This patch set was created without a push certificate',
|
||||
};
|
||||
}
|
||||
|
||||
const key = rev.push_certificate.key;
|
||||
switch (key.status) {
|
||||
case CertificateStatus.BAD:
|
||||
return {
|
||||
class: 'invalid',
|
||||
icon: 'gr-icons:close',
|
||||
message: this._problems('Push certificate is invalid', key),
|
||||
};
|
||||
case CertificateStatus.OK:
|
||||
return {
|
||||
class: 'notTrusted',
|
||||
icon: 'gr-icons:info',
|
||||
message: this._problems(
|
||||
'Push certificate is valid, but key is not trusted', key),
|
||||
};
|
||||
case CertificateStatus.TRUSTED:
|
||||
return {
|
||||
class: 'trusted',
|
||||
icon: 'gr-icons:check',
|
||||
message: this._problems(
|
||||
'Push certificate is valid and key is trusted', key),
|
||||
};
|
||||
default:
|
||||
throw new Error(`unknown certificate status: ${key.status}`);
|
||||
}
|
||||
},
|
||||
|
||||
_problems(msg, key) {
|
||||
if (!key || !key.problems || key.problems.length === 0) {
|
||||
return msg;
|
||||
}
|
||||
|
||||
return [msg + ':'].concat(key.problems).join('\n');
|
||||
},
|
||||
|
||||
_computeProjectURL(project) {
|
||||
return Gerrit.Nav.getUrlForProjectChanges(project);
|
||||
},
|
||||
|
@@ -311,6 +311,107 @@ limitations under the License.
|
||||
});
|
||||
});
|
||||
|
||||
test('Push Certificate Validation test BAD', () => {
|
||||
const serverConfig = {
|
||||
receive: {
|
||||
enable_signed_push: true,
|
||||
},
|
||||
};
|
||||
const change = {
|
||||
change_id: 'Iad9dc96274af6946f3632be53b106ef80f7ba6ca',
|
||||
owner: {
|
||||
_account_id: 1019328,
|
||||
},
|
||||
revisions: {
|
||||
rev1: {
|
||||
_number: 1,
|
||||
push_certificate: {
|
||||
key: {
|
||||
status: 'BAD',
|
||||
problems: [
|
||||
'No public keys found for key ID E5E20E52',
|
||||
],
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
current_revision: 'rev1',
|
||||
status: 'NEW',
|
||||
labels: {},
|
||||
mergeable: true,
|
||||
};
|
||||
const result =
|
||||
element._computePushCertificateValidation(serverConfig, change);
|
||||
assert.equal(result.message,
|
||||
'Push certificate is invalid:\n' +
|
||||
'No public keys found for key ID E5E20E52');
|
||||
assert.equal(result.icon, 'gr-icons:close');
|
||||
assert.equal(result.class, 'invalid');
|
||||
});
|
||||
|
||||
test('Push Certificate Validation test TRUSTED', () => {
|
||||
const serverConfig = {
|
||||
receive: {
|
||||
enable_signed_push: true,
|
||||
},
|
||||
};
|
||||
const change = {
|
||||
change_id: 'Iad9dc96274af6946f3632be53b106ef80f7ba6ca',
|
||||
owner: {
|
||||
_account_id: 1019328,
|
||||
},
|
||||
revisions: {
|
||||
rev1: {
|
||||
_number: 1,
|
||||
push_certificate: {
|
||||
key: {
|
||||
status: 'TRUSTED',
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
current_revision: 'rev1',
|
||||
status: 'NEW',
|
||||
labels: {},
|
||||
mergeable: true,
|
||||
};
|
||||
const result =
|
||||
element._computePushCertificateValidation(serverConfig, change);
|
||||
assert.equal(result.message,
|
||||
'Push certificate is valid and key is trusted');
|
||||
assert.equal(result.icon, 'gr-icons:check');
|
||||
assert.equal(result.class, 'trusted');
|
||||
});
|
||||
|
||||
test('Push Certificate Validation is missing test', () => {
|
||||
const serverConfig = {
|
||||
receive: {
|
||||
enable_signed_push: true,
|
||||
},
|
||||
};
|
||||
const change = {
|
||||
change_id: 'Iad9dc96274af6946f3632be53b106ef80f7ba6ca',
|
||||
owner: {
|
||||
_account_id: 1019328,
|
||||
},
|
||||
revisions: {
|
||||
rev1: {
|
||||
_number: 1,
|
||||
},
|
||||
},
|
||||
current_revision: 'rev1',
|
||||
status: 'NEW',
|
||||
labels: {},
|
||||
mergeable: true,
|
||||
};
|
||||
const result =
|
||||
element._computePushCertificateValidation(serverConfig, change);
|
||||
assert.equal(result.message,
|
||||
'This patch set was created without a push certificate');
|
||||
assert.equal(result.icon, 'gr-icons:help');
|
||||
assert.equal(result.class, 'help');
|
||||
});
|
||||
|
||||
test('_computeParents', () => {
|
||||
const parents = [{commit: '123', subject: 'abc'}];
|
||||
assert.isUndefined(element._computeParents(
|
||||
|
@@ -48,6 +48,10 @@ limitations under the License.
|
||||
<g id="publishEdit"><path d="M5 4v2h14V4H5zm0 10h4v6h6v-6h4l-7-7-7 7z"/></g>
|
||||
<!-- This SVG is a copy from iron-icons https://github.com/PolymerElements/iron-icons/blob/master/editor-icons.html -->
|
||||
<g id="delete"><path d="M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z"/></g>
|
||||
<!-- This SVG is a copy from iron-icons https://github.com/PolymerElements/iron-icons/blob/master/iron-icons.js -->
|
||||
<g id="help"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 17h-2v-2h2v2zm2.07-7.75l-.9.92C13.45 12.9 13 13.5 13 15h-2v-.5c0-1.1.45-2.1 1.17-2.83l1.24-1.26c.37-.36.59-.86.59-1.41 0-1.1-.9-2-2-2s-2 .9-2 2H8c0-2.21 1.79-4 4-4s4 1.79 4 4c0 .88-.36 1.68-.93 2.25z"></path></g>
|
||||
<!-- This SVG is a copy from iron-icons https://github.com/PolymerElements/iron-icons/blob/master/iron-icons.js -->
|
||||
<g id="info"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"></path></g>
|
||||
<!-- This SVG is a copy from material.io https://material.io/icons/#ic_hourglass_full-->
|
||||
<g id="hourglass"><path d="M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6z"/><path d="M0 0h24v24H0V0z" fill="none"/></g>
|
||||
<!-- This is a custom PolyGerrit SVG -->
|
||||
|
Reference in New Issue
Block a user