Important Update Effective February 1, 2024!
Due to recent changes in Jira and Confluence, we've made the tough decision to discontinue the OpenID Connect (OIDC)/OAuth app and no longer provide new versions for the newest Jira/Confluence releases as of January 31, 2024.
This is due to some necessary components no longer shipping with Jira/Confluence, which would require some extensive rewrites of the OIDC App.
Important Update! This app will be discontinued soon!
Due to recent changes in Jira, which no longer ships with some components required for our Read Receipts app to run, we've made the tough decision to discontinue the app, as of Februar 5, 2025.
Important Update! This app will be discontinued soon!
We've made the tough business decision to discontinue the app, as of January 11, 2025.
IdP Selection Page Examples
On this page, we'll add some more examples of how to customize the IdP Selection Page Template in the app's Page Template section.
Page templates are using Velocity (http://velocity.apache.org/) and support HTML, CSS, JS, and images.
Images can only be embedded directly, as Base64 blobs. You can convert them here: https://www.base64-image.de/
Jira (IdP Buttons)
This is for Jira only, looks like the normal login page with IdPs added as Buttons
<html>
<head>
<title>Select Identity Provider</title>
$webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
<meta name="decorator" content="atl.general">
</head>
<body class="aui-layout aui-theme-default page-type-message">
<section id="content" role="main">
<div class="aui-page-panel">
<div class="aui-page-panel-inner">
<section class="aui-page-panel-content">
<header class="aui-page-header">
<div class="aui-page-header-inner">
<div class="aui-page-header-main">
<h1>Welcome to Jira</h1>
</div>
</div>
</header>
<form action="/login.jsp" class="aui" id="login-form" method="post">
<div class="form-body">
<div class="aui-group jira-login-method">
<div class="aui-item jira-login-item">
<div class="field-group">
<label accesskey="u" for="login-form-username"><u>U</u>sername</label>
<input class="text medium-field" id="login-form-username" name="os_username"
type="text" value="">
</div>
<div class="field-group">
<label accesskey="p" for="login-form-password" id="passwordlabel"><u>P</u>assword</label>
<input id="login-form-password" class="text medium-field" name="os_password"
type="password">
</div>
<fieldset class="group ">
<div class="checkbox" resolved="">
<input class="checkbox" id="login-form-remember-me" name="os_cookie"
type="checkbox" value="true" resolved=""><span
class="aui-form-glyph"></span>
<label for="login-form-remember-me" accesskey="r"><u>R</u>emember my
login on this computer</label>
</div>
</fieldset>
<!-- // .group -->
<div id="sign-up-hint" class="field-group">
Not a member? To request an account, please contact your Jira
administrators.
</div>
</div>
</div>
<div class="hidden">
<input name="os_destination" id="redirect_to" type="hidden" >
<script>
const queryString = window.location.search;
const urlParams = new URLSearchParams(queryString);
if (urlParams.has('redirectTo')){
document.getElementById('redirect_to').value = urlParams.get('redirectTo');
}
</script>
</div>
<div class="hidden">
<input name="user_role" type="hidden">
</div>
<div class="hidden">
<input name="atl_token" type="hidden">
</div>
</div>
<div class="buttons-container form-footer">
<div class="buttons">
<input accesskey="s" class="aui-button aui-button-primary" id="login-form-submit"
name="login" title="Press Ctrl+Alt+s to submit this form" type="submit"
value="Log In" resolved="">
<a accesskey="`" class="aui-button aui-button-link cancel"
href="/secure/ForgotLoginDetails.jspa" id="login-form-cancel"
title="Press Ctrl+Alt+` to cancel" resolved="">Can't access your account?</a>
</div>
</div>
</form>
</div>
<div class="buttons-container form-footer">
<div class="buttons" align="center">
#foreach($idp in $idps)
<button title="Click to login with $idp.name" class="aui-button"
onclick="window.location.href = '$idp.ssoUrl';">Login with $idp.name
</button>
#end
</div>
<p> </p>
</div>
</form>
<!-- // .aui #login-form -->
</section>
<!-- .aui-page-panel-content -->
</div>
<!-- .aui-page-panel-inner -->
</div>
<!-- .aui-page-panel -->
</section>
</div>
</div>
</section>
</body>
</html>
Template with embedded Images and CSS/ JS
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Your page title</title>
<script src="https://code.jquery.com/jquery-3.3.1.min.js" integrity="sha256-FgpCb/KJQlLNfOu91ta32o/NMZxltwRo8QtmkMRdAu8=" crossorigin="anonymous"></script>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">
<style type="text/css">
body {
background-image: url('');
background-repeat: repeat;
}
* {
font-family: 'Blue Sky Standard', sans-serif;
}
.content-container {
/*margin-top: 120px;*/
top: 30%;
position: absolute;
background-color: transparent !important;
color: #fff;
}
.heading {
font-size: 25px;
font-weight: lighter;
}
.select-box {
margin-top: 10px;
max-width: 650px;
padding: 0;
color: #fff;
background: rgba(256, 256, 256, 0.2);
border-radius: 4px;
font-weight: lighter;
}
.select-box > .inner-select:first-child {
border-bottom: 1px solid #000;
}
.inner-select {
padding: 30px;
transition: 0.5s all;
color: #fff;
}
.select-type {
margin-bottom: 0;
font-size: 15px;
}
.click {
font-size: 10px;
}
.tab {
padding: 0;
}
.inner-select:hover {
cursor: pointer;
background: rgba(256, 256, 256, 0.9);
transition: 0.5s all;
color: black;
font-weight: normal;
}
.desc-container {
padding: 0;
max-width: 500px;
}
.description {
font-size: smaller;
font-weight: lighter;
}
.logo {
width: 75px;
}
.logo-container {
padding: 20px;
}
.arrow {
padding: 7px 0;
}
.click-arrow {
font-size: 25px;
}
</style>
</head>
<body>
<div class="container-fluid">
<div class="logo-container">
<img src="" class="logo">
</div>
<div class="content-container col-sm-4 col-sm-offset-4">
<h1 class="heading">Your heading</h1>
#if($idpSelected)
<p>Select or wait 3 seconds to use $selectedName <span class="aui-icon aui-icon-wait"></span></p>
<script>
var timeout = setTimeout("location.href = '$selectedUrl';", 3000);
window.onclick= function () { clearTimeout(timeout); }
</script>
#end
<div class="select-box col-xs-12">
#foreach($idp in $idps)
<a class="inner-select col-xs-12" href="$idp.ssoUrl">
<div class="tab col-xs-11">
<span class="select-type">$idp.name</span><br/>
<span class="click">CLICK HERE</span>
</div>
<div class="arrow col-xs-1 pull-right">
<i class="fa fa-angle-right click-arrow"></i>
</div>
</a>
#end
</div>
<div class="select-box col-xs-12">
<a class="inner-select col-xs-12" href="/login.jsp?nosso">
<div class="tab col-xs-11">
<span class="select-type">Login with local password</span><br/>
<span class="click">CLICK HERE</span>
</div>
<div class="arrow col-xs-1 pull-right">
<i class="fa fa-angle-right click-arrow"></i>
</div>
</a>
</div> <div class="desc-container col-xs-12">
<h1 class="heading">Status</h1>
<span class="description">More text to be displayed here</span>
</div>
</div>
</div>
</body>
</html>
Change the decorator
Atlassian's applications support standard page decorators, allowing our plugin to generate new web pages with consistent decoration by the host application across the Atlassian products.
Default we are using the 'atl.general' decorator in our page templates. The general decorator can be used for the header and footer of general pages outside the administration UI.
atl.general decorator
If you want to remove the header and footer from the page, you need to change the decorator type. We will use the 'atl.popup' decorator for content that can be placed in a new browser popup window.
Just change the decorator type to hide the header and footer.
In the “IdP Selection Page Template”, replace the line:
<meta name="decorator" content="atl.general">
to
<meta name="decorator" content="atl.popup">
Look & Feel
general decorator
popup decorator
Click-thru Terms & Conditions Sign-up
Would it be possible to provide some kind of "Click-thru" agreement to a user which requires that they sign up to the terms of the site before getting general access?
For this scenario, we use the 'IdP Selection Page Template' (SAML Single Sign-On Plugin Configuration → Page Templates) and modify the default IdP Selection Page Template.
When a user now accesses Jira/Confluence for the first time, this page is displayed to him. By clicking "Okay", he will then be redirected to the first configured Identity Provider.
In doing so, we set a cookie that stores this. If the page is now visited a second time, it will be redirected directly to the Identity Provider as long as the cookie is still set.
So this means that each of your users will see this banner once (if the cookie is lost, the page is displayed again).
Below you will find the template, please use it in the Page Templates (IdP Selection Page) section. You can easily customize the text or button text to your needs, as well as the color via CSS.
<html>
<head>
<title>Terms of Service</title>
$webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
<meta name="decorator" content="atl.general">
</head>
<body class="aui-layout aui-theme-default page-type-message">
<style>
.alert {
padding: 20px;
background-color: #0000ff;
color: white;
}
.closebtn {
margin-left: 15px;
color: white;
font-weight: bold;
float: right;
font-size: 22px;
line-height: 20px;
cursor: pointer;
transition: 0.3s;
}
.closebtn:hover {
color: black;
}
</style>
<div class="alert" id="message">
Your data will be processed for things. You will see this message only once. Click 'Okay' to acknowledge and proceed.<br></br>
<button onclick="function myFunction() {
window.location.href = '$idps.get(0).ssoUrl'
}
myFunction()"
type="button">
Okay
</button>
#if($idpSelected)
<script>
document.getElementById("message").style.display = "none";
window.location.href = '$idps.get(0).ssoUrl'
</script>
#end
</div>
</body>
</html>
Multi-language IdP Selection Page
Would it be possible to have a multi-language version of the 'IdP Selection Page Template' (SAML Single Sign On Plugin Configuration → Page Templates)?
We currently don't have an option in the UI to support multi-language templates. However, you can use JavaScript inside the IdP Selection templates to achieve your goal.
Below you will find an example based on our default page template. Our example contains the languages German (de) and French (fr), however you can change or add the languages to your needs.
Multilanguage Template
<html>
<head>
<title>Select Identity Provider</title>
$webResourceManager.requireResource("com.atlassian.auiplugin:aui-spinner")
$webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
<meta name="decorator" content="atl.general">
</head>
<body class="aui-layout aui-theme-default page-type-message" >
<section id="content" role="main">
<div class="aui-page-panel"><div class="aui-page-panel-inner">
<section class="aui-page-panel-content">
<div style="display: flex; justify-content: center;" id="initial-load-spinner">
<aui-spinner size="large"></aui-spinner>
</div>
<div class="form-body" style="display: none;" id="form-body">
<h1 data-translate="title">Select your Identity Provider</h1>
#if($idpSelected)
<p><span data-translate="wait">Select or wait 3 seconds to use $selectedName</span> <span class="aui-icon aui-icon-wait"></span></p>
<script>
var timeout = setTimeout("location.href = '$selectedUrl';", 3000);
window.onclick= function () { clearTimeout(timeout); }
</script>
#end
#foreach($idp in $idps)
<p>
<a href="$idp.ssoUrl" data-translate="$idp.name">$idp.name</a> $idp.description
</p>
#end
#if($loginurl) <p>
<a href="$loginurl" data-translate="nosso">Login with username and password</a>
</p>
#end
</div>
</section>
</div>
</div>
</section>
<script>
AJS.toInit(function() {
// TRANSLATION SETTINGS
// The key to this template is that elements above whose text is supposed to be translated all have this data-translate attribute, where they get a translation key
// The following map contains mappings from languages to translation keys to the actual translations.
// Add your translations here. The keys ('de', 'fr') can be either locales ('de-DE', 'de-AT') or languages ('de', 'fr'). When in doubt, just use the language it's going to be used as a fallback
var translations = {
de: {
title: "Wählen Sie Ihren Identity Provider",
wait: "Wählen Sie einen Identity Provider aus oder warten Sie 3 Sekunden um $selectedName zu benutzen",
nosso: "Login mit Benutzername und Passwort",
// IdP names are allowed as well
// If they contain non-alphanumeric characters, you'll have to write the name in "quotes"
// the keys are case-sensitive!!!
"Your IdP name": "Ihr IdP-Name"
},
fr: { // This translation has been generously provided by Google Translate, I would not trust it, but it's here as an example.
title: "Choisissez votre fournisseur d'identité",
wait: "Sélectionnez un fournisseur d'identité ou attendez 3 secondes pour utiliser $selectedName",
nosso: "Connectez-vous avec nom d'utilisateur et mot de passe",
"Your IdP name": "Votre nom IdP"
}
};
// This is supposed to be the language present in the fields that are given above
var defaultLanguage = "en";
// TRANSLATION CODE
// This is where the magic happens. Ideally, you shouldn't need to touch the code after this.
// Detect user language with fallbacks for different levels of old browsers
// This will be ordered from most preferred to least preferred
var userLanguages = navigator.languages || [navigator.language] || [navigator.userLanguage] || [];
// ensure default language can also match.
translations[defaultLanguage] = {};
var matchedLanguage = null;
// Check if one the user's preferred languages is available
userLanguages.forEach(function (languageString) {
if (matchedLanguage === null && languageString in translations) {
matchedLanguage = languageString
}
});
// if no match was found, try converting the locale strings into language strings and see if that helps
if (matchedLanguage === null) {
userLanguages.forEach(function (languageString) {
try {
var splitLanguageString = languageString.split('-')[0];
if (matchedLanguage === null && typeof splitLanguageString === 'string' && splitLanguageString in translations) {
matchedLanguage = splitLanguageString;
}
} catch (e) {
console.warn("Error preparing user language");
}
});
}
// if any (non-default language) match was found, we set the translated strings in the UI
if (matchedLanguage !== null && matchedLanguage !== defaultLanguage) {
var translationsToUse = translations[matchedLanguage];
AJS.$('[data-translate]').each(function() {
var elem = AJS.$(this);
var translationToUse = translationsToUse[elem.attr('data-translate')];
if (!translationToUse) {
console.warn("Did not find a", matchedLanguage, "translation for", elem.attr('data-translate'));
return;
}
elem.text(translationToUse);
})
}
// and now display the UI
AJS.$("#initial-load-spinner").hide();
AJS.$("#form-body").show();
})
</script>
</body>
</html>
A little more background/explanation
The script basically extends every element with text in it to have an attribute (data-translate) that contains a “translation key”, while the text is kept as the default language. After loading the page, it executes a JavaScript block. This contains the translations you want to have for each language and the translation key (you can modify this according to the example and your need). Afterwards, there’s code that detects the user’s preferred language and does the substitution if necessary.
IdP Selection By Email Page with a Button
Applying the following template to the "IdP Selection By Email Page Template" would add a button that only upon clicking would trigger redirection.
IdP Selection by Email with a Button
#disable_html_escaping()
<html>
<head>
<title>Enter your Email Address</title>
$webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
<meta name="decorator" content="atl.general">
<script type="text/javascript">
var emailDomains = $emailDomainMap;
var ssoURL = "$ssourl";
AJS.toInit(function () {
// Hide these items
jQuery('#notFound').hide();
jQuery('#redirecting').hide();
// The form is never submitted
jQuery("#emailAddressForm").submit(function (e) {
checkEmailField();
e.preventDefault();
});
});
function checkEmailField() {
jQuery('#notFound').hide();
jQuery('#redirecting').hide();
// The input must have an @ and a . behind it to be ready to be checked
var addressPattern = new RegExp(".*@.*\\...+");
var enteredEmail = jQuery("#emailAddressField").val();
if (!addressPattern.test(enteredEmail)) {
return;
}
var atPosition = enteredEmail.indexOf('@');
if (atPosition < 0) {
return;
}
var emailDomain = enteredEmail.substring(atPosition + 1).toLowerCase();
if (!emailDomain) {
return;
}
var restEndpointUrl = emailDomains.__emailDomainCheck;
jQuery.ajax({
url: restEndpointUrl + emailDomain,
dataType: "json",
success: function (data) {
console.log("Successfully retrieved IdP for given email domain", data);
var idpId = data.idpId;
if (idpId !== 0) {
jQuery('#notFound').hide();
jQuery('#redirecting').show();
jQuery('#emailAddressField').prop('disabled', true);
redirectUrl = ssoURL.replace('IDPID', idpId);
// This avoids a warning message when starting the redirect
window.onbeforeunload = function () {
};
window.location = redirectUrl;
} else {
jQuery('#notFound').show();
}
},
error: function (xhr, errorText, err) {
console.error("Error checking email domain: " + errorText, err);
jQuery('#notFound').show();
}
});
}
</script>
</head>
<body class="aui-layout aui-theme-default page-type-message">
<section id="content" role="main">
<div class="aui-page-panel">
<div class="aui-page-panel-inner">
<section class="aui-page-panel-content">
<h1>Single Sign On</h1>
<form class="aui top-label" id="emailAddressForm" >
<div class="field-group">
<label for="emailAddressField">Enter your email address</label>
<input type="text" class="text" id="emailAddressField" name="emailAddressField">
<input type="submit" class="aui-button aui-button-primary" value="Do SSO">
<div class="description">
<div id="notFound" >No SSO destination found for this address</div>
<div id="redirecting"><span class="aui-icon aui-icon-wait"></span>Redirecting...</div>
</div>
</div>
#if($loginurl)
<p><a href="$loginurl">Login with username and password</a></p>
#end
</form>
</section>
</div>
</div>
</section>
</body>
</html>
IdP Selection page only at the first login
From version 6.3.0 of the plugin, it is also possible to provide users the choice of whether they want to see the IdP selection page again or if they only want to pick their IdP ones. So far the user's choice was already stored in a cookie but the IdP selection was always displayed for 3 seconds (a default value that can be amended) until the previous IdP was automatically selected. This gives the user the choice to switch the IdP within that timeframe.
Please note, the option described below will only work if you set the timeout variable (setTimeout) to 0 instead of 3000.
Now in this example, when the "Do you want to save your selection?" is selected the IdP selection page will not be shown the next time a user logs in. This is especially useful in an environment where the majority of users only use one IdP and you have some users that need to switch the IdPs for logging into the application.
The IdPs can be displayed in a meatball or drop-down menu, or as a list. Our template below contains all options (meatball, drop-down menu, or as a list). Every option is marked with a comment. Please modify the template if you only want to display the IdPs with one option.
IdP Selection page only at the first login
<html>
<head>
<title>Select Identity Provider</title>
$webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
<meta name="decorator" content="atl.general">
</head>
<body class="aui-layout aui-theme-default page-type-message" >
<section id="content" role="main">
<div class="aui-page-panel"><div class="aui-page-panel-inner">
<section class="aui-page-panel-content">
<div class="form-body">
<h1>Select your Identity Provider</h1>
#if($idpSelected)
<!-- Change the value setTimeout 3000 to 0, otherwise the IdP selection page will be shown every time -->
<p>Select or wait 3 seconds to use $selectedName <span class="aui-icon aui-icon-wait"></span></p>
<script>
var timeout = setTimeout("location.href = '$selectedUrl';", 3000);
window.onclick= function () { clearTimeout(timeout); }
</script>
#end
<!-- This is the meatball option -->
<form action="/plugins/servlet/samlsso" method="get" >
<label for="idp">Choose an Idp:</label>
<fieldset id="idp">
#foreach($idp in $idps)
<input type="radio" value="$idp.idpId" name="idp" id="$idp.idpId">
<label for="$idp.idpId">$idp.name $idp.description</label>
<br>
#end
</fieldset>
<!-- This is the drop-down menu option -->
<select name="idp">
#foreach($idp in $idps)
<option value="$idp.idpId">$idp.name $idp.description</option>
#end
</select>
<input type="hidden" name="tracker" value="$trackerId">
<input type="hidden" name="redirectTo" value="$redirectTo">
<br><br>
<input type="hidden" name="saveIdpSelection" value="false">
<input type="checkbox" name="saveIdpSelection" id="saveIdpCookie" value="true" checked>
<label for="saveIdpCookie">Do you want to save your selection?</label>
<br><br>
<input type="submit" value="Go">
</form>
<!-- This is the list option -->
#foreach($idp in $idps)
<p>
<a href="$idp.ssoUrl">$idp.name</a> $idp.description
</p>
#end
#if($loginurl) <p>
<a href="$loginurl">Login with username and password</a>
</p>
#end
</div>
</section>
</div>
</div>
</section>
</body>
</html>