Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[JEP-234] Customizable header proposal #380

Merged
merged 26 commits into from
Nov 23, 2021
Merged
Changes from 4 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
125bbd3
[JEP-401] Customizable header proposal
imonteroperez Nov 11, 2021
781a489
Fix JEP reference
imonteroperez Nov 11, 2021
da44e56
Relocate
imonteroperez Nov 11, 2021
b7a2d3b
Mention on no API compatibility on backward compatibility section
imonteroperez Nov 12, 2021
4ad4a02
Apply suggestions from code review
imonteroperez Nov 12, 2021
1ef0b93
Fix broken link
imonteroperez Nov 12, 2021
2f170e6
Testing source block
imonteroperez Nov 12, 2021
080272a
Use source,java block
imonteroperez Nov 12, 2021
6dcf673
Use the right acronym
imonteroperez Nov 12, 2021
0d9f3aa
Remove complicated/redundant sentence
imonteroperez Nov 12, 2021
ba6d169
Removed redundant section
imonteroperez Nov 12, 2021
d24ed80
Apply suggestions from code review
imonteroperez Nov 15, 2021
e1d7333
Relocated to 0000 folder
imonteroperez Nov 16, 2021
955f61a
Update some code changes
imonteroperez Nov 16, 2021
2499ba3
Reorder content
imonteroperez Nov 16, 2021
6dadb21
Remove Integer.MIN_VALUE on JenkinsHeader
imonteroperez Nov 16, 2021
9e62a7c
Setup explicitly that code on Reasoning section may differ from the f…
imonteroperez Nov 16, 2021
6231d10
One-line
imonteroperez Nov 16, 2021
6aea87c
Explain why source code is there
imonteroperez Nov 16, 2021
e93d9a8
Refer to full and partial header on Backward Compatibility
imonteroperez Nov 16, 2021
cc194ad
Some Jesse's comment/feedback
imonteroperez Nov 19, 2021
0a9b76a
Apply suggestions from code review
imonteroperez Nov 22, 2021
b6b353c
Applied one-sentence-per-line
imonteroperez Nov 22, 2021
d1f9b73
Fix reference implementation description
imonteroperez Nov 22, 2021
46cb96c
Customizable Jenkins header is now JEP-234
MarkEWaite Nov 23, 2021
85b00c3
Accept JEP-234 as a draft JEP
MarkEWaite Nov 23, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
266 changes: 266 additions & 0 deletions jep/234/README.adoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,266 @@
= JEP-234: Customizable Jenkins header
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
:toc: preamble
:toclevels: 3
ifdef::env-github[]
:tip-caption: :bulb:
:note-caption: :information_source:
:important-caption: :heavy_exclamation_mark:
:caution-caption: :fire:
:warning-caption: :warning:
endif::[]

.Metadata
[cols="2"]
|===
| JEP
| 234
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

| Title
| Customizable Jenkins header

| Sponsor
| link:https://github.com/imonteroperez[Ildefonso Montero]

// Use the script `set-jep-status <jep-number> <status>` to update the status.
| Status
| Draft :speech_balloon:

| Type
| Standards

| Created
| 2021-11-28

//
//
// Uncomment if there is an associated placeholder JIRA issue.
//| JIRA
//| :bulb: link:https://issues.jenkins-ci.org/browse/JENKINS-nnnnn[JENKINS-nnnnn] :bulb:
//
//
// Uncomment if there will be a BDFL delegate for this JEP.
//| BDFL-Delegate
//| :bulb: Link to github user page :bulb:
//
//
// Uncomment if discussion will occur in forum other than jenkinsci-dev@ mailing list.
//| Discussions-To
//| :bulb: Link to where discussion and final status announcement will occur :bulb:
//
//
// Uncomment if this JEP depends on one or more other JEPs.
//| Requires
//| :bulb: JEP-NUMBER, JEP-NUMBER... :bulb:
//
//
// Uncomment and fill if this JEP is rendered obsolete by a later JEP
//| Superseded-By
//| :bulb: JEP-NUMBER :bulb:
//
//
// Uncomment when this JEP status is set to Accepted, Rejected or Withdrawn.
//| Resolution
//| :bulb: Link to relevant post in the jenkinsci-dev@ mailing list archives :bulb:

|===

== Abstract

Jenkins does not provide a customization mechanism for header.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

Unique existing approach based on the https://plugins.jenkins.io/simple-theme-plugin/[simple-theme-plugin] has a limited capabilities (see Reasoning section for more details).
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

It makes it difficult:

* to provide branding capabilities to include custom logos, styles or other elements, and
* to make feasible to rewrite some functionality or include additional one to some elements like the search bar.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

Sometimes, this limited customization capabilities implies a barrier on Jenkins adoption inside enterprises.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

Having the ability to customize the header (not only from the UI POV) will help to avoid that situation.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

This proposal provides a customization mechanism for a better integration that also reduces current tech debt.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

== Specification

The main aspect of the change is the introduction of a new extension point `Header` as an interface (or an abstract class) that provides capabilities to render a specific header and a default implementation of that, named `JenkinsHeader` that is enabled by default always.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

All headers will provide a prioritization technique, via https://javadoc.jenkins.io/hudson/Extension.html[ordinal] though the `Extension` annotation, that will help to override, based on its value, which header will be rendered. Default one will provide a `Integer.MIN_VALUE` value.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

On `pageHeader.jelly` file we will perform a refactor to make it more modular injecting the `headerContent.jelly` that will provide the content of the header.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
Thus, `pageHeader.jelly` should look like:

```xml
<?jelly escape-by-default='true'?>
<j:jelly xmlns:j="jelly:core" xmlns:st="jelly:stapler" xmlns:d="jelly:define" xmlns:l="/lib/layout" xmlns:i="jelly:fmt" xmlns:x="jelly:xml">
<j:invokeStatic var="header" className="jenkins.views.JenkinsHeader" method="get"/>
<st:include it="${header}" page="headerContent.jelly"/>
</j:jelly>
```

The `get()` method is provided in the default implementation `jenkins.views.JenkinsHeader` and it will retrieve the available headers via `Extension` lookup that are enabled and for those obtained it will provide the one with max priority (ordinal value). In case two or more headers are provided via external plugins with the same priority (and max), then the default one will be provided and a warning message would be provided on the logs.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

In terms of simplicity, proposal aims to use only one extension to do a full replacement of the header. As a follow up, any given implementation of the extension can provide custom extension points for further or more granular customization capabilities.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

=== Extensibility

Jenkins users will be able to interact with the `Header` extension as follows in case want to customize the header of their instances by creating a custom plugin. For that purpose, they need just to extend `Header` and provide an implementation of the priority method using the ordinal value to specify desired priority value.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved


== Motivation

Jenkins uses `pageHeader.jelly` file to specify the content of the header. Although it is valid, if a user wants to modify the header to perform some branding operation, Jenkins header branding capabilities are limited.

There exist some plugins, like: https://plugins.jenkins.io/simple-theme-plugin/[simple-theme-plugin], that allow Jenkins users to customize some parts like CSS and/or Javascript. On the other hand, if a Jenkins user wants to customize/modify some additional business functionality on some menus and search bar, then there is no approach beyond updating/overriding `pageHeader.jelly` from the Jenkins core, which would be a problem on updating the instance due to conflicts. In addition, if the user wants to customize these behaviors it would be good to have those features as REST endpoints.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

Also, other alternatives has been evaluated as workaround like the https://github.com/stephenc/diffpatch-maven-plugin[diffpatch-maven-plugin] with no satisfactory results. See Reasoning section for futher details.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

So, this approach is about providing not only UI capabilities, it is also providing extra functionality and better integration.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

== Reasoning

Let's consider the following dummy example to illustrate reasoning on why particular design decisions were made and also why current approaches were discarded.

> A Jenkins user wants to modify its current Jenkins instance header to provide a configurable message (default: `Hello World!`) with the account username of the logged user, as well as two (why not?) search bars

There exists some options to perform parts of the required actions mentioned above:
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

* Update the `pageHeader.jelly` content of his Jenkins instance. Discarded in terms of looking for better mechanisms to customize the instance that does not require to override/update its original source code.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
* Use https://plugins.jenkins.io/simple-theme-plugin/[simple-theme-plugin]. It could help us to modify some CSS and Javascript elements, and could be used to reach our goals, but it would be so hacky and will not be able to retrieve programatically the configurable message to be included with the username of the logged user.
* Use https://github.com/stephenc/diffpatch-maven-plugin[diffpatch-maven-plugin] to override the content of the header using patches. It would help us to have a repeated search box, but not to have the configurable message due to it only will help for static content.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

Given existing approaches does not fullfill this example, we will explore the alternative of a Jenkins user that wants to update the Jenkins header through an ad hoc Jenkins plugin that follows the principles provided on Specification section:
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

=== Jenkins core

* Let’s consider the following definition of the `Header` on: `core/src/main/java/jenkins/views/Header.java`

```
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
package jenkins.views;

import hudson.ExtensionPoint;

public interface Header extends ExtensionPoint {

/**
* Check if the header is enabled. By default it is if installed,
* but the logic is deferred in the plugins.
* @return
*/
boolean isHeaderEnabled();

}
```

* Let’s consider the following implementation of the Jenkins header on: `core/src/main/java/jenkins/views/JenkinsHeader.java`

```
package jenkins.views;

import hudson.Extension;

@Extension(ordinal = Integer.MIN_VALUE)
public class JenkinsHeader extends Header {

@Override
public boolean isHeaderEnabled() {
return true;
}
[...]
}
```

* As mentioned before, method `get()` from `JenkinsHeader` will retrieve the available headers via `Extension` lookup that are enabled and for those obtained it will provide the one with max priority (ordinal value)

```
[...]
@Restricted(NoExternalUse.class)
@CheckForNull
public static Header get() {
List<Header> headers = ExtensionList.lookup(Header.class).stream()
.filter(header -> header.isHeaderEnabled())
.collect(Collectors.toList());
if (headers.size() > 0) {
if (headers.size() > 1) {
LOGGER.warning("More than one configured header. This should not happen. Serving the Jenkins default header and please review");
} else {
return headers.get(0);
}
}
return new JenkinsHeader();
}
```

* Once we launch Jenkins with the proposed changes on the core, we will obtain the expected/current header working without any issue

=== Custom UI plugin

* Create a new plugin following the usual procedure
* Provide an implementation of the custom Header (i.e: `src/main/java/org/jenkinsci/plugins/custom/header/CustomHeader.java`)
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

```
[...]
@Extension(ordinal = 100)
public class CustomHeader extends Header {

@Override
public boolean isHeaderEnabled() {
// Disable/enable the header based on an ENV var and/or system property
boolean isDisabled = System.getProperty(CustomHeader.class.getName() + ".disable") != null ?
"true".equalsIgnoreCase(System.getProperty(CustomHeader.class.getName() + ".disable")) :
"true".equalsIgnoreCase(System.getenv("CUSTOM_HEADER_DISABLE"));
return !isDisabled;
}
}
```

* Provide a method in the custom header to retrieve the label which will be with the username. Current code is just an example, but the label could be obtained from the https://javadoc.jenkins.io/jenkins/model/GlobalConfiguration.html[GlobalConfiguration].

```
public static String getHeaderLabel(){
// This label content could be retrieved programatically. Not coded in aims of simplicity.
return "Hello World!";
}
```

* Provide the jelly file to override the `headerContent`. For that purpose, use the common location convention. For the previous example: `src/main/resources/org/jenkinsci/plugins/custom/header/CustomHeader/`. Retrieve the customizable label to be rendered with the username on the `headerContent` file.

```xml
<j:invokeStatic var="label" className="org.jenkinsci.plugins.custom.header.CustomHeader" method="getHeaderLabel"/>
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
<span class="hidden-xs hidden-sm">${label} - ${userName}</span>
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved
```

* See the sample implementation provided in the Reference Implementation section.

== Backwards Compatibility

Given this proposal relies on replacement/injection of the `pageHeader` and `headerContent` and the content of that source relies also on UI elements (CSS identifiers, Javascript, etc.) backward compatibility cannot be guaranteed (as happens with themes - documented as https://www.jenkins.io/doc/book/managing/ui-themes/#themes-support-policy[no API compatibility]).

To deal with these incompatibilities:

* Consider to place all your required CSS and Javascript code inside your custom plugins if you are going to do a complete refactor of the header.
* Consider to be up-to-date with the latest sources/updates on the `headerContent` in case you were doing minimal changes through your custom header plugin.
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

== Security

No specific security considerations

== Infrastructure Requirements

No impact on the Jenkins project infrastructure

== Testing

To write tests specific to the header (also using a patched core via https://github.com/stephenc/diffpatch-maven-plugin[diffpatch-maven-plugin] are currently difficult. Proposed solution will solve these issues: if a customized header is an extension in a plugin then having this plugin on your test classpath will suffice to let UI tests run in the expected way, regardless of core provenance.

== Reference Implementation

* Proposed changes on Jenkins core: https://github.com/jenkinsci/jenkins/pull/5909
* Prototype of a https://github.com/imonteroperez/custom-header-plugin[Custom Header plugin]. This plugin is modifying the current Jenkins header including an extra search box (just for clarification purposes).
imonteroperez marked this conversation as resolved.
Show resolved Hide resolved

== References

Relevant data

* jenkins-dev: https://groups.google.com/g/jenkinsci-dev/c/1tDvSioCaF0
* Jenkins UX SIG meeting Nov 24: https://docs.google.com/document/d/1QttPwdimNP_120JukigKsRuBvMr34KZhVfsbgq1HFLM/edit#