2020-02-19 09:17:52 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 Red Hat, Inc. and/or its affiliates
|
|
|
|
* and other contributors as indicated by the @author tags.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2018-01-17 07:55:24 +00:00
|
|
|
package org.keycloak.theme;
|
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
import org.jboss.logging.Logger;
|
|
|
|
import org.keycloak.Config;
|
|
|
|
import org.keycloak.common.Version;
|
|
|
|
import org.keycloak.common.util.StringPropertyReplacer;
|
|
|
|
import org.keycloak.common.util.SystemEnvProperties;
|
2018-01-17 07:55:24 +00:00
|
|
|
import org.keycloak.models.KeycloakSession;
|
|
|
|
import org.keycloak.models.ThemeManager;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
2020-02-19 09:17:52 +00:00
|
|
|
import java.io.InputStream;
|
|
|
|
import java.net.URL;
|
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.HashSet;
|
|
|
|
import java.util.LinkedList;
|
2018-01-17 07:55:24 +00:00
|
|
|
import java.util.List;
|
2020-02-19 09:17:52 +00:00
|
|
|
import java.util.ListIterator;
|
|
|
|
import java.util.Locale;
|
|
|
|
import java.util.Properties;
|
2018-01-17 07:55:24 +00:00
|
|
|
import java.util.Set;
|
2020-02-19 09:17:52 +00:00
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
2020-11-18 12:44:54 +00:00
|
|
|
import org.keycloak.common.Profile;
|
2018-01-17 07:55:24 +00:00
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
/**
|
|
|
|
* @author <a href="mailto:sthorger@redhat.com">Stian Thorgersen</a>
|
|
|
|
*/
|
2018-01-17 07:55:24 +00:00
|
|
|
public class DefaultThemeManager implements ThemeManager {
|
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
private static final Logger log = Logger.getLogger(DefaultThemeManager.class);
|
|
|
|
|
|
|
|
private final DefaultThemeManagerFactory factory;
|
|
|
|
private final KeycloakSession session;
|
|
|
|
private List<ThemeProvider> providers;
|
2021-07-20 12:29:15 +00:00
|
|
|
private final String defaultTheme;
|
2018-01-17 07:55:24 +00:00
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
public DefaultThemeManager(DefaultThemeManagerFactory factory, KeycloakSession session) {
|
|
|
|
this.factory = factory;
|
2018-01-17 07:55:24 +00:00
|
|
|
this.session = session;
|
2020-02-19 09:17:52 +00:00
|
|
|
this.defaultTheme = Config.scope("theme").get("default", Version.NAME.toLowerCase());
|
2018-01-17 07:55:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-02-19 09:17:52 +00:00
|
|
|
public Theme getTheme(Theme.Type type) {
|
2018-01-17 07:55:24 +00:00
|
|
|
String name = session.getProvider(ThemeSelectorProvider.class).getThemeName(type);
|
|
|
|
return getTheme(name, type);
|
|
|
|
}
|
|
|
|
|
2020-11-18 12:44:54 +00:00
|
|
|
private String typeBasedDefault(Theme.Type type) {
|
2021-07-20 12:29:15 +00:00
|
|
|
boolean isProduct = Profile.isProduct();
|
|
|
|
|
2020-11-19 18:26:24 +00:00
|
|
|
if ((type == Theme.Type.ACCOUNT) && isAccount2Enabled()) {
|
2021-07-20 12:29:15 +00:00
|
|
|
return isProduct ? "rh-sso.v2" : "keycloak.v2";
|
2020-11-18 12:44:54 +00:00
|
|
|
}
|
2021-07-20 12:29:15 +00:00
|
|
|
|
|
|
|
return isProduct ? "rh-sso" : "keycloak";
|
2020-11-18 12:44:54 +00:00
|
|
|
}
|
|
|
|
|
2018-01-17 07:55:24 +00:00
|
|
|
@Override
|
2020-02-19 09:17:52 +00:00
|
|
|
public Theme getTheme(String name, Theme.Type type) {
|
|
|
|
if (name == null) {
|
|
|
|
name = defaultTheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
Theme theme = factory.getCachedTheme(name, type);
|
|
|
|
if (theme == null) {
|
|
|
|
theme = loadTheme(name, type);
|
|
|
|
if (theme == null) {
|
2020-11-18 12:44:54 +00:00
|
|
|
theme = loadTheme(typeBasedDefault(type), type);
|
2020-02-19 09:17:52 +00:00
|
|
|
if (theme == null) {
|
|
|
|
theme = loadTheme("base", type);
|
|
|
|
}
|
|
|
|
log.errorv("Failed to find {0} theme {1}, using built-in themes", type, name);
|
|
|
|
} else {
|
|
|
|
theme = factory.addCachedTheme(name, type, theme);
|
|
|
|
}
|
|
|
|
}
|
2020-11-18 12:44:54 +00:00
|
|
|
|
2020-11-19 18:26:24 +00:00
|
|
|
if (!isAccount2Enabled() && theme.getName().equals("keycloak.v2")) {
|
2020-11-18 12:44:54 +00:00
|
|
|
theme = loadTheme("keycloak", type);
|
|
|
|
}
|
2021-07-20 12:29:15 +00:00
|
|
|
|
|
|
|
if (!isAccount2Enabled() && theme.getName().equals("rh-sso.v2")) {
|
|
|
|
theme = loadTheme("rh-sso", type);
|
2020-11-18 12:44:54 +00:00
|
|
|
}
|
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
return theme;
|
2018-01-17 07:55:24 +00:00
|
|
|
}
|
2020-11-18 12:44:54 +00:00
|
|
|
|
2018-01-17 07:55:24 +00:00
|
|
|
@Override
|
|
|
|
public Set<String> nameSet(Theme.Type type) {
|
2020-02-19 09:17:52 +00:00
|
|
|
Set<String> themes = new HashSet<String>();
|
|
|
|
for (ThemeProvider p : getProviders()) {
|
|
|
|
themes.addAll(p.nameSet(type));
|
|
|
|
}
|
|
|
|
return themes;
|
|
|
|
}
|
|
|
|
|
2020-10-21 07:02:19 +00:00
|
|
|
@Override
|
|
|
|
public boolean isCacheEnabled() {
|
|
|
|
return factory.isCacheEnabled();
|
|
|
|
}
|
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
@Override
|
|
|
|
public void clearCache() {
|
|
|
|
factory.clearCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
private Theme loadTheme(String name, Theme.Type type) {
|
|
|
|
Theme theme = findTheme(name, type);
|
2021-01-27 10:09:23 +00:00
|
|
|
if (theme == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-02-19 09:17:52 +00:00
|
|
|
List<Theme> themes = new LinkedList<>();
|
|
|
|
themes.add(theme);
|
|
|
|
|
|
|
|
if (theme.getImportName() != null) {
|
|
|
|
String[] s = theme.getImportName().split("/");
|
|
|
|
themes.add(findTheme(s[1], Theme.Type.valueOf(s[0].toUpperCase())));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (theme.getParentName() != null) {
|
|
|
|
for (String parentName = theme.getParentName(); parentName != null; parentName = theme.getParentName()) {
|
|
|
|
theme = findTheme(parentName, type);
|
|
|
|
themes.add(theme);
|
|
|
|
|
|
|
|
if (theme.getImportName() != null) {
|
|
|
|
String[] s = theme.getImportName().split("/");
|
|
|
|
themes.add(findTheme(s[1], Theme.Type.valueOf(s[0].toUpperCase())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new ExtendingTheme(themes, session.getAllProviders(ThemeResourceProvider.class));
|
|
|
|
}
|
|
|
|
|
|
|
|
private Theme findTheme(String name, Theme.Type type) {
|
|
|
|
for (ThemeProvider p : getProviders()) {
|
|
|
|
if (p.hasTheme(name, type)) {
|
|
|
|
try {
|
|
|
|
return p.getTheme(name, type);
|
|
|
|
} catch (IOException e) {
|
|
|
|
log.errorv(e, p.getClass() + " failed to load theme, type={0}, name={1}", type, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class ExtendingTheme implements Theme {
|
|
|
|
|
|
|
|
private List<Theme> themes;
|
|
|
|
private Set<ThemeResourceProvider> themeResourceProviders;
|
|
|
|
|
|
|
|
private Properties properties;
|
|
|
|
|
|
|
|
private ConcurrentHashMap<String, ConcurrentHashMap<Locale, Properties>> messages = new ConcurrentHashMap<>();
|
|
|
|
|
|
|
|
public ExtendingTheme(List<Theme> themes, Set<ThemeResourceProvider> themeResourceProviders) {
|
|
|
|
this.themes = themes;
|
|
|
|
this.themeResourceProviders = themeResourceProviders;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String getName() {
|
|
|
|
return themes.get(0).getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String getParentName() {
|
|
|
|
return themes.get(0).getParentName();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String getImportName() {
|
|
|
|
return themes.get(0).getImportName();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Type getType() {
|
|
|
|
return themes.get(0).getType();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public URL getTemplate(String name) throws IOException {
|
|
|
|
for (Theme t : themes) {
|
|
|
|
URL template = t.getTemplate(name);
|
|
|
|
if (template != null) {
|
|
|
|
return template;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ThemeResourceProvider t : themeResourceProviders) {
|
|
|
|
URL template = t.getTemplate(name);
|
|
|
|
if (template != null) {
|
|
|
|
return template;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public InputStream getResourceAsStream(String path) throws IOException {
|
|
|
|
for (Theme t : themes) {
|
|
|
|
InputStream resource = t.getResourceAsStream(path);
|
|
|
|
if (resource != null) {
|
|
|
|
return resource;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ThemeResourceProvider t : themeResourceProviders) {
|
|
|
|
InputStream resource = t.getResourceAsStream(path);
|
|
|
|
if (resource != null) {
|
|
|
|
return resource;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Properties getMessages(Locale locale) throws IOException {
|
|
|
|
return getMessages("messages", locale);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Properties getMessages(String baseBundlename, Locale locale) throws IOException {
|
|
|
|
if (messages.get(baseBundlename) == null || messages.get(baseBundlename).get(locale) == null) {
|
|
|
|
Properties messages = new Properties();
|
|
|
|
|
|
|
|
Locale parent = getParent(locale);
|
|
|
|
|
|
|
|
if (parent != null) {
|
|
|
|
messages.putAll(getMessages(baseBundlename, parent));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ThemeResourceProvider t : themeResourceProviders ){
|
|
|
|
messages.putAll(t.getMessages(baseBundlename, locale));
|
|
|
|
}
|
|
|
|
|
|
|
|
ListIterator<Theme> itr = themes.listIterator(themes.size());
|
|
|
|
while (itr.hasPrevious()) {
|
|
|
|
Properties m = itr.previous().getMessages(baseBundlename, locale);
|
|
|
|
if (m != null) {
|
|
|
|
messages.putAll(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.messages.putIfAbsent(baseBundlename, new ConcurrentHashMap<Locale, Properties>());
|
|
|
|
this.messages.get(baseBundlename).putIfAbsent(locale, messages);
|
|
|
|
|
|
|
|
return messages;
|
|
|
|
} else {
|
|
|
|
return messages.get(baseBundlename).get(locale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Properties getProperties() throws IOException {
|
|
|
|
if (properties == null) {
|
|
|
|
Properties properties = new Properties();
|
|
|
|
ListIterator<Theme> itr = themes.listIterator(themes.size());
|
|
|
|
while (itr.hasPrevious()) {
|
|
|
|
Properties p = itr.previous().getProperties();
|
|
|
|
if (p != null) {
|
|
|
|
properties.putAll(p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
substituteProperties(properties);
|
|
|
|
this.properties = properties;
|
|
|
|
return properties;
|
|
|
|
} else {
|
|
|
|
return properties;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over all string properties defined in "theme.properties" then substitute the value with system property or environment variables.
|
|
|
|
* See {@link StringPropertyReplacer#replaceProperties} for details about the different formats.
|
|
|
|
*/
|
|
|
|
private void substituteProperties(final Properties properties) {
|
|
|
|
for (final String propertyName : properties.stringPropertyNames()) {
|
|
|
|
properties.setProperty(propertyName, StringPropertyReplacer.replaceProperties(properties.getProperty(propertyName), new SystemEnvProperties()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Locale getParent(Locale locale) {
|
|
|
|
if (Locale.ENGLISH.equals(locale)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (locale.getVariant() != null && !locale.getVariant().isEmpty()) {
|
|
|
|
return new Locale(locale.getLanguage(), locale.getCountry());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (locale.getCountry() != null && !locale.getCountry().isEmpty()) {
|
|
|
|
return new Locale(locale.getLanguage());
|
|
|
|
}
|
|
|
|
|
|
|
|
return Locale.ENGLISH;
|
|
|
|
}
|
|
|
|
|
|
|
|
private List<ThemeProvider> getProviders() {
|
|
|
|
if (providers == null) {
|
|
|
|
providers = new LinkedList(session.getAllProviders(ThemeProvider.class));
|
|
|
|
Collections.sort(providers, (o1, o2) -> o2.getProviderPriority() - o1.getProviderPriority());
|
|
|
|
}
|
|
|
|
|
|
|
|
return providers;
|
2018-01-17 07:55:24 +00:00
|
|
|
}
|
2020-02-19 09:17:52 +00:00
|
|
|
|
2020-11-19 18:26:24 +00:00
|
|
|
private static boolean isAccount2Enabled() {
|
|
|
|
return Profile.isFeatureEnabled(Profile.Feature.ACCOUNT2);
|
|
|
|
}
|
|
|
|
|
2018-01-17 07:55:24 +00:00
|
|
|
}
|