High severityNVD Advisory· Published Mar 2, 2023· Updated Mar 5, 2025
XWiki-Platform vulnerable to stored Cross-site Scripting via the HTML displayer in Live Data
CVE-2023-26480
Description
XWiki Platform is a generic wiki platform. Starting in version 12.10, a user without script rights can introduce a stored cross-site scripting by using the Live Data macro. This has been patched in XWiki 14.9, 14.4.7, and 13.10.10. There are no known workarounds.
Affected packages
Versions sourced from the GitHub Security Advisory.
| Package | Affected versions | Patched versions |
|---|---|---|
org.xwiki.platform:xwiki-platform-livedata-macroMaven | >= 12.10, < 13.10.10 | 13.10.10 |
org.xwiki.platform:xwiki-platform-livedata-macroMaven | >= 14.0, < 14.4.7 | 14.4.7 |
org.xwiki.platform:xwiki-platform-livedata-macroMaven | >= 14.5, < 14.9 | 14.9 |
Affected products
1- Range: >= 12.10, < 13.10.10
Patches
223d5ea9b23e8XWIKI-20143: Improved Live Data escaping
1 file changed · +3 −1
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/test/java/org/xwiki/livedata/internal/macro/LiveDataMacroComponentList.java+3 −1 modified@@ -49,7 +49,9 @@ LiveDataMacro.class, DefaultLiveDataConfigurationResolver.class, StringLiveDataConfigurationResolver.class, - JsFileSkinExtension.class + JsFileSkinExtension.class, + LiveDataMacroConfiguration.class, + LiveDataMacroRights.class }) @Inherited public @interface LiveDataMacroComponentList
556e7823260bXWIKI-20143: Improved Live Data escaping
23 files changed · +455 −206
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/pom.xml+5 −0 modified@@ -58,6 +58,11 @@ <artifactId>xwiki-platform-skin-api</artifactId> <version>${project.version}</version> </dependency> + <dependency> + <groupId>org.xwiki.platform</groupId> + <artifactId>xwiki-platform-bridge</artifactId> + <version>${project.version}</version> + </dependency> <!-- Test dependencies --> <dependency> <groupId>org.xwiki.rendering</groupId>
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/java/org/xwiki/livedata/internal/macro/LiveDataMacroConfiguration.java+233 −0 added@@ -0,0 +1,233 @@ +/* + * See the NOTICE file distributed with this work for additional + * information regarding copyright ownership. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ +package org.xwiki.livedata.internal.macro; + +import java.net.URL; +import java.net.URLDecoder; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.regex.Pattern; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import javax.inject.Inject; +import javax.inject.Singleton; + +import org.apache.commons.lang3.StringUtils; +import org.xwiki.component.annotation.Component; +import org.xwiki.livedata.LiveDataConfiguration; +import org.xwiki.livedata.LiveDataConfigurationResolver; +import org.xwiki.livedata.LiveDataLayoutDescriptor; +import org.xwiki.livedata.LiveDataMeta; +import org.xwiki.livedata.LiveDataPaginationConfiguration; +import org.xwiki.livedata.LiveDataQuery; +import org.xwiki.livedata.internal.JSONMerge; +import org.xwiki.livedata.macro.LiveDataMacroParameters; + +/** + * Provides services to manipulate configurations for the {@link LiveDataMacroConfiguration}. + * + * @version $Id$ + * @since 14.9 + * @since 14.4.7 + * @since 13.10.10 + */ +@Component(roles = LiveDataMacroConfiguration.class) +@Singleton +public class LiveDataMacroConfiguration +{ + private static final Pattern PATTERN_COMMA = Pattern.compile("\\s*,\\s*"); + + private static final String UTF8 = "UTF-8"; + + /** + * Used to read the Live Data configuration from the macro content. + */ + @Inject + private LiveDataConfigurationResolver<String> stringLiveDataConfigResolver; + + /** + * Used to merge the Live Data configuration built from the macro parameters with the live data configuration read + * from the macro content. + */ + private JSONMerge jsonMerge = new JSONMerge(); + + /** + * Resolve a complete Live Data configuration from a json advanced configuration (the content) and a set of macro + * parameters. + * + * @param content the string representation of the json live data advanced configuration + * @param parameters the Live Data macro parameters + * @return the complete Live Data configuration + * @throws Exception in case of error when resolving the configuration + */ + public LiveDataConfiguration getLiveDataConfiguration(String content, LiveDataMacroParameters parameters) + throws Exception + { + String json = StringUtils.defaultIfBlank(content, "{}"); + LiveDataConfiguration advancedConfig = this.stringLiveDataConfigResolver.resolve(json); + LiveDataConfiguration basicConfig = getLiveDataConfiguration(parameters); + // Make sure both configurations have the same id so that they are properly merged. + advancedConfig.setId(basicConfig.getId()); + return this.jsonMerge.merge(advancedConfig, basicConfig); + } + + private LiveDataConfiguration getLiveDataConfiguration(LiveDataMacroParameters parameters) throws Exception + { + LiveDataConfiguration liveDataConfig = new LiveDataConfiguration(); + liveDataConfig.setId(parameters.getId()); + liveDataConfig.setQuery(getQuery(parameters)); + liveDataConfig.setMeta(getMeta(parameters)); + return liveDataConfig; + } + + private LiveDataQuery getQuery(LiveDataMacroParameters parameters) throws Exception + { + LiveDataQuery query = new LiveDataQuery(); + query.setProperties(getProperties(parameters.getProperties())); + query.setSource(new LiveDataQuery.Source(parameters.getSource())); + query.getSource().getParameters().putAll(getSourceParameters(parameters.getSourceParameters())); + query.setSort(getSortEntries(parameters.getSort())); + query.setFilters(getFilters(parameters.getFilters())); + query.setLimit(parameters.getLimit()); + query.setOffset(parameters.getOffset()); + return query; + } + + private List<String> getProperties(String properties) + { + if (properties == null) { + return null; + } else { + return Stream.of(PATTERN_COMMA.split(properties)).collect(Collectors.toList()); + } + } + + private Map<String, Object> getSourceParameters(String sourceParametersString) throws Exception + { + if (StringUtils.isEmpty(sourceParametersString)) { + return Collections.emptyMap(); + } + + Map<String, List<String>> urlParams = getURLParameters('?' + sourceParametersString); + Map<String, Object> sourceParams = new HashMap<>(); + for (Map.Entry<String, List<String>> entry : urlParams.entrySet()) { + if (entry.getValue().size() > 1) { + sourceParams.put(entry.getKey(), entry.getValue()); + } else { + sourceParams.put(entry.getKey(), entry.getValue().get(0)); + } + } + return sourceParams; + } + + private List<LiveDataQuery.SortEntry> getSortEntries(String sort) + { + if (sort == null) { + return null; + } else { + return Stream.of(PATTERN_COMMA.split(sort)).filter(StringUtils::isNotEmpty).map(this::getSortEntry) + .collect(Collectors.toList()); + } + } + + private LiveDataQuery.SortEntry getSortEntry(String sortEntryString) + { + LiveDataQuery.SortEntry sortEntry = new LiveDataQuery.SortEntry(); + sortEntry.setDescending(sortEntryString.endsWith(":desc")); + if (sortEntry.isDescending() || sortEntryString.endsWith(":asc")) { + sortEntry.setProperty(StringUtils.substringBeforeLast(sortEntryString, ":")); + } else { + sortEntry.setProperty(sortEntryString); + } + return sortEntry; + } + + private List<LiveDataQuery.Filter> getFilters(String filtersString) throws Exception + { + List<LiveDataQuery.Filter> filters = + getURLParameters('?' + StringUtils.defaultString(filtersString)).entrySet().stream() + .map(this::getFilter).collect(Collectors.toList()); + return filters.isEmpty() ? null : filters; + } + + private LiveDataQuery.Filter getFilter(Map.Entry<String, List<String>> entry) + { + LiveDataQuery.Filter filter = new LiveDataQuery.Filter(); + filter.setProperty(entry.getKey()); + filter.getConstraints() + .addAll(entry.getValue().stream().map(LiveDataQuery.Constraint::new).collect(Collectors.toList())); + return filter; + } + + private LiveDataMeta getMeta(LiveDataMacroParameters parameters) + { + LiveDataMeta meta = new LiveDataMeta(); + meta.setLayouts(getLayouts(parameters)); + meta.setPagination(getPagination(parameters)); + return meta; + } + + private List<LiveDataLayoutDescriptor> getLayouts(LiveDataMacroParameters parameters) + { + if (parameters.getLayouts() == null) { + return null; + } else { + return Stream.of(PATTERN_COMMA.split(parameters.getLayouts())).map(LiveDataLayoutDescriptor::new) + .collect(Collectors.toList()); + } + } + + private LiveDataPaginationConfiguration getPagination(LiveDataMacroParameters parameters) + { + LiveDataPaginationConfiguration pagination = new LiveDataPaginationConfiguration(); + pagination.setShowPageSizeDropdown(parameters.getShowPageSizeDropdown()); + if (parameters.getPageSizes() != null) { + pagination.setPageSizes(Stream.of(PATTERN_COMMA.split(parameters.getPageSizes())).map(Integer::parseInt) + .collect(Collectors.toList())); + } + return pagination; + } + + private Map<String, List<String>> getURLParameters(String url) throws Exception + { + URL baseURL = new URL("http://www.xwiki.org"); + String queryString = new URL(baseURL, url).getQuery(); + Map<String, List<String>> parameters = new HashMap<>(); + for (String entry : queryString.split("&")) { + String[] parts = entry.split("=", 2); + String key = URLDecoder.decode(parts[0], UTF8); + if (key.isEmpty()) { + continue; + } + String value = parts.length == 2 ? URLDecoder.decode(parts[1], UTF8) : ""; + List<String> values = parameters.get(key); + if (values == null) { + values = new ArrayList<>(); + parameters.put(key, values); + } + values.add(value); + } + return parameters; + } +}
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/java/org/xwiki/livedata/internal/macro/LiveDataMacro.java+16 −181 modified@@ -19,17 +19,9 @@ */ package org.xwiki.livedata.internal.macro; -import java.net.URL; -import java.net.URLDecoder; -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Set; -import java.util.regex.Pattern; -import java.util.stream.Collectors; -import java.util.stream.Stream; import javax.inject.Inject; import javax.inject.Named; @@ -39,15 +31,6 @@ import org.xwiki.component.annotation.Component; import org.xwiki.livedata.LiveDataConfiguration; import org.xwiki.livedata.LiveDataConfigurationResolver; -import org.xwiki.livedata.LiveDataLayoutDescriptor; -import org.xwiki.livedata.LiveDataMeta; -import org.xwiki.livedata.LiveDataPaginationConfiguration; -import org.xwiki.livedata.LiveDataQuery; -import org.xwiki.livedata.LiveDataQuery.Constraint; -import org.xwiki.livedata.LiveDataQuery.Filter; -import org.xwiki.livedata.LiveDataQuery.SortEntry; -import org.xwiki.livedata.LiveDataQuery.Source; -import org.xwiki.livedata.internal.JSONMerge; import org.xwiki.livedata.macro.LiveDataMacroParameters; import org.xwiki.rendering.block.Block; import org.xwiki.rendering.block.GroupBlock; @@ -59,6 +42,9 @@ import com.fasterxml.jackson.databind.ObjectMapper; +import static java.util.Collections.singletonList; +import static java.util.Collections.singletonMap; + /** * Display dynamic lists of data. * @@ -70,27 +56,17 @@ @Singleton public class LiveDataMacro extends AbstractMacro<LiveDataMacroParameters> { - private static final Pattern PATTERN_COMMA = Pattern.compile("\\s*,\\s*"); - - private static final String UTF8 = "UTF-8"; - /** * Used to add default Live Data configuration values. */ @Inject private LiveDataConfigurationResolver<LiveDataConfiguration> defaultLiveDataConfigResolver; - /** - * Used to read the Live Data configuration from the macro content. - */ @Inject - private LiveDataConfigurationResolver<String> stringLiveDataConfigResolver; + private LiveDataMacroConfiguration liveDataMacroConfiguration; - /** - * Used to merge the Live Data configuration built from the macro parameters with the live data configuration read - * from the macro content. - */ - private JSONMerge jsonMerge = new JSONMerge(); + @Inject + private LiveDataMacroRights liveDataMacroRights; /** * The component used to load the JavaScript code of the Live Data widget. @@ -115,7 +91,7 @@ public List<Block> execute(LiveDataMacroParameters parameters, String content, M throws MacroExecutionException { // Load the JavaScript code of the Live Data widget. - Map<String, Object> skinExtensionParameters = Collections.singletonMap("forceSkinAction", Boolean.TRUE); + Map<String, Object> skinExtensionParameters = singletonMap("forceSkinAction", Boolean.TRUE); this.jsfx.use("uicomponents/widgets/liveData.js", skinExtensionParameters); GroupBlock output = new GroupBlock(); @@ -125,169 +101,28 @@ public List<Block> execute(LiveDataMacroParameters parameters, String content, M } try { // Compute the live data configuration based on the macro parameters. - LiveDataConfiguration liveDataConfig = getLiveDataConfiguration(content, parameters); + LiveDataConfiguration liveDataConfig = this.liveDataMacroConfiguration.getLiveDataConfiguration(content, + parameters); // Add the default values. liveDataConfig = this.defaultLiveDataConfigResolver.resolve(liveDataConfig); // Serialize as JSON. ObjectMapper objectMapper = new ObjectMapper(); output.setParameter("data-config", objectMapper.writeValueAsString(liveDataConfig)); + // The content is trusted if the author has script right, or if no advanced configuration is used (i.e., + // no macro content), and we are running in a trusted context. + boolean trustedContent = + StringUtils.isBlank(content) || (this.liveDataMacroRights.authorHasScriptRight() + && !context.getTransformationContext().isRestricted()); + output.setParameter("data-config-content-trusted", Boolean.toString(trustedContent)); } catch (Exception e) { throw new MacroExecutionException("Failed to generate live data configuration from macro parameters.", e); } - return Collections.singletonList(output); + return singletonList(output); } @Override public boolean supportsInlineMode() { return false; } - - private LiveDataConfiguration getLiveDataConfiguration(String content, LiveDataMacroParameters parameters) - throws Exception - { - String json = StringUtils.defaultIfBlank(content, "{}"); - LiveDataConfiguration advancedConfig = this.stringLiveDataConfigResolver.resolve(json); - LiveDataConfiguration basicConfig = getLiveDataConfiguration(parameters); - // Make sure both configurations have the same id so that they are properly merged. - advancedConfig.setId(basicConfig.getId()); - return this.jsonMerge.merge(advancedConfig, basicConfig); - } - - private LiveDataConfiguration getLiveDataConfiguration(LiveDataMacroParameters parameters) throws Exception - { - LiveDataConfiguration liveDataConfig = new LiveDataConfiguration(); - liveDataConfig.setId(parameters.getId()); - liveDataConfig.setQuery(getQuery(parameters)); - liveDataConfig.setMeta(getMeta(parameters)); - return liveDataConfig; - } - - private LiveDataQuery getQuery(LiveDataMacroParameters parameters) throws Exception - { - LiveDataQuery query = new LiveDataQuery(); - query.setProperties(getProperties(parameters.getProperties())); - query.setSource(new Source(parameters.getSource())); - query.getSource().getParameters().putAll(getSourceParameters(parameters.getSourceParameters())); - query.setSort(getSortEntries(parameters.getSort())); - query.setFilters(getFilters(parameters.getFilters())); - query.setLimit(parameters.getLimit()); - query.setOffset(parameters.getOffset()); - return query; - } - - private List<String> getProperties(String properties) - { - if (properties == null) { - return null; - } else { - return Stream.of(PATTERN_COMMA.split(properties)).collect(Collectors.toList()); - } - } - - private Map<String, Object> getSourceParameters(String sourceParametersString) throws Exception - { - if (StringUtils.isEmpty(sourceParametersString)) { - return Collections.emptyMap(); - } - - Map<String, List<String>> urlParams = getURLParameters('?' + sourceParametersString); - Map<String, Object> sourceParams = new HashMap<>(); - for (Map.Entry<String, List<String>> entry : urlParams.entrySet()) { - if (entry.getValue().size() > 1) { - sourceParams.put(entry.getKey(), entry.getValue()); - } else { - sourceParams.put(entry.getKey(), entry.getValue().get(0)); - } - } - return sourceParams; - } - - private List<SortEntry> getSortEntries(String sort) - { - if (sort == null) { - return null; - } else { - return Stream.of(PATTERN_COMMA.split(sort)).filter(StringUtils::isNotEmpty).map(this::getSortEntry) - .collect(Collectors.toList()); - } - } - - private SortEntry getSortEntry(String sortEntryString) - { - SortEntry sortEntry = new SortEntry(); - sortEntry.setDescending(sortEntryString.endsWith(":desc")); - if (sortEntry.isDescending() || sortEntryString.endsWith(":asc")) { - sortEntry.setProperty(StringUtils.substringBeforeLast(sortEntryString, ":")); - } else { - sortEntry.setProperty(sortEntryString); - } - return sortEntry; - } - - private List<Filter> getFilters(String filtersString) throws Exception - { - List<Filter> filters = getURLParameters('?' + StringUtils.defaultString(filtersString)).entrySet().stream() - .map(this::getFilter).collect(Collectors.toList()); - return filters.isEmpty() ? null : filters; - } - - private Filter getFilter(Map.Entry<String, List<String>> entry) - { - Filter filter = new Filter(); - filter.setProperty(entry.getKey()); - filter.getConstraints().addAll(entry.getValue().stream().map(Constraint::new).collect(Collectors.toList())); - return filter; - } - - private LiveDataMeta getMeta(LiveDataMacroParameters parameters) - { - LiveDataMeta meta = new LiveDataMeta(); - meta.setLayouts(getLayouts(parameters)); - meta.setPagination(getPagination(parameters)); - return meta; - } - - private List<LiveDataLayoutDescriptor> getLayouts(LiveDataMacroParameters parameters) - { - if (parameters.getLayouts() == null) { - return null; - } else { - return Stream.of(PATTERN_COMMA.split(parameters.getLayouts())).map(LiveDataLayoutDescriptor::new) - .collect(Collectors.toList()); - } - } - - private LiveDataPaginationConfiguration getPagination(LiveDataMacroParameters parameters) - { - LiveDataPaginationConfiguration pagination = new LiveDataPaginationConfiguration(); - pagination.setShowPageSizeDropdown(parameters.getShowPageSizeDropdown()); - if (parameters.getPageSizes() != null) { - pagination.setPageSizes(Stream.of(PATTERN_COMMA.split(parameters.getPageSizes())).map(Integer::parseInt) - .collect(Collectors.toList())); - } - return pagination; - } - - private Map<String, List<String>> getURLParameters(String url) throws Exception - { - URL baseURL = new URL("http://www.xwiki.org"); - String queryString = new URL(baseURL, url).getQuery(); - Map<String, List<String>> parameters = new HashMap<>(); - for (String entry : queryString.split("&")) { - String[] parts = entry.split("=", 2); - String key = URLDecoder.decode(parts[0], UTF8); - if (key.isEmpty()) { - continue; - } - String value = parts.length == 2 ? URLDecoder.decode(parts[1], UTF8) : ""; - List<String> values = parameters.get(key); - if (values == null) { - values = new ArrayList<>(); - parameters.put(key, values); - } - values.add(value); - } - return parameters; - } }
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/java/org/xwiki/livedata/internal/macro/LiveDataMacroRights.java+60 −0 added@@ -0,0 +1,60 @@ +/* + * See the NOTICE file distributed with this work for additional + * information regarding copyright ownership. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ +package org.xwiki.livedata.internal.macro; + +import javax.inject.Inject; +import javax.inject.Singleton; + +import org.xwiki.bridge.DocumentAccessBridge; +import org.xwiki.component.annotation.Component; +import org.xwiki.model.reference.DocumentReference; +import org.xwiki.security.authorization.AuthorizationManager; + +import static org.xwiki.security.authorization.Right.SCRIPT; + +/** + * Provides the services related to rights for {@link LiveDataMacro}. + * + * @version $Id$ + * @since 14.9 + * @since 14.4.7 + * @since 13.10.10 + */ +@Component(roles = LiveDataMacroRights.class) +@Singleton +public class LiveDataMacroRights +{ + @Inject + private DocumentAccessBridge documentAccessBridge; + + @Inject + private AuthorizationManager authorizationManager; + + /** + * @return {@code true} if the last author of the document where the Live Data macro is executed has script rights + * on said document, {@code false} otherwise + */ + public boolean authorHasScriptRight() + { + DocumentReference currentDocumentReference = this.documentAccessBridge.getCurrentDocumentReference(); + DocumentReference currentAuthorReference = this.documentAccessBridge.getCurrentAuthorReference(); + return this.authorizationManager.hasAccess(SCRIPT, currentAuthorReference, currentDocumentReference); + } +}
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/resources/ApplicationResources_de.properties+1 −1 modified@@ -87,7 +87,7 @@ livedata.panel.sort.direction.ascending=Aufsteigend livedata.panel.sort.direction.descending=Absteigend livedata.panel.sort.add=Sortierung hinzuf\u00FCgen livedata.panel.sort.delete=Sortierung l\u00F6schen -livedata.displayer.emptyValue=n.v.<sup>*</sup> +livedata.displayer.emptyValue=n.v. livedata.displayer.link.noValue=(kein Wert) livedata.displayer.boolean.true=Wahr livedata.displayer.boolean.false=Falsch
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/resources/ApplicationResources_fr.properties+1 −1 modified@@ -87,7 +87,7 @@ livedata.panel.sort.direction.ascending=Croissant livedata.panel.sort.direction.descending=D\u00E9croissant livedata.panel.sort.add=Ajouter un tri livedata.panel.sort.delete=Supprimer le tri -livedata.displayer.emptyValue=SO<sup>*</sup> +livedata.displayer.emptyValue=SO livedata.displayer.link.noValue=(aucune valeur) livedata.displayer.boolean.true=Vrai livedata.displayer.boolean.false=Faux
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/resources/ApplicationResources.properties+1 −1 modified@@ -106,7 +106,7 @@ livedata.panel.sort.direction.descending=Descending livedata.panel.sort.add=Add sort livedata.panel.sort.delete=Delete sort -livedata.displayer.emptyValue=N/A<sup>*</sup> +livedata.displayer.emptyValue=N/A livedata.displayer.link.noValue=(no value)
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/main/resources/META-INF/components.txt+2 −0 modified@@ -1 +1,3 @@ org.xwiki.livedata.internal.macro.LiveDataMacro +org.xwiki.livedata.internal.macro.LiveDataMacroConfiguration +org.xwiki.livedata.internal.macro.LiveDataMacroRights
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-macro/src/test/java/org/xwiki/livedata/internal/macro/LiveDataMacroTest.java+17 −6 modified@@ -27,6 +27,7 @@ import org.junit.jupiter.api.Test; import org.mockito.invocation.InvocationOnMock; import org.mockito.stubbing.Answer; +import org.xwiki.bridge.DocumentAccessBridge; import org.xwiki.configuration.internal.RestrictedConfigurationSourceProvider; import org.xwiki.context.internal.DefaultExecution; import org.xwiki.livedata.LiveDataConfiguration; @@ -43,6 +44,7 @@ import org.xwiki.rendering.internal.renderer.xhtml.link.DefaultXHTMLLinkRenderer; import org.xwiki.rendering.internal.renderer.xhtml.link.DefaultXHTMLLinkTypeRenderer; import org.xwiki.rendering.renderer.PrintRendererFactory; +import org.xwiki.security.authorization.AuthorizationManager; import org.xwiki.test.annotation.ComponentList; import org.xwiki.test.junit5.mockito.ComponentTest; import org.xwiki.test.junit5.mockito.InjectMockComponents; @@ -80,7 +82,9 @@ HTMLDefinitions.class, MathMLDefinitions.class, SVGDefinitions.class, - DefaultExecution.class + DefaultExecution.class, + LiveDataMacroConfiguration.class, + LiveDataMacroRights.class }) class LiveDataMacroTest { @@ -93,6 +97,12 @@ class LiveDataMacroTest @MockComponent private LiveDataConfigurationResolver<String> stringConfigResolver; + @MockComponent + private DocumentAccessBridge documentAccessBridge; + + @MockComponent + private AuthorizationManager authorizationManager; + private PrintRendererFactory rendererFactory; @BeforeEach @@ -117,7 +127,8 @@ public LiveDataConfiguration answer(InvocationOnMock invocation) throws Throwabl void executeWithoutParams() throws Exception { String expectedConfig = json("{'query':{'source':{}},'meta':{'pagination':{}}}"); - String expected = "<div class=\"liveData loading\" data-config=\"" + escapeXML(expectedConfig) + "\"></div>"; + String expected = "<div class=\"liveData loading\" data-config=\"" + escapeXML(expectedConfig) + "\" " + + "data-config-content-trusted=\"true\"></div>"; List<Block> blocks = this.liveDataMacro.execute(new LiveDataMacroParameters(), null, null); assertBlocks(expected, blocks, this.rendererFactory); @@ -155,8 +166,8 @@ void execute() throws Exception expectedConfig.append(" }".trim()); expectedConfig.append("}"); - String expected = "<div class=\"liveData loading\" id=\"test\" data-config=\"" - + escapeXML(json(expectedConfig.toString())) + "\"></div>"; + String expected = String.format("<div class=\"liveData loading\" id=\"test\" data-config=\"%s\" " + + "data-config-content-trusted=\"true\"></div>", escapeXML(json(expectedConfig.toString()))); LiveDataMacroParameters parameters = new LiveDataMacroParameters(); parameters.setId("test"); @@ -214,8 +225,8 @@ void executeWithContent() throws Exception expectedConfig.append(" }".trim()); expectedConfig.append("}"); - String expected = "<div class=\"liveData loading\" id=\"test\" data-config=\"" - + escapeXML(json(expectedConfig.toString())) + "\"></div>"; + String expected = String.format("<div class=\"liveData loading\" id=\"test\" data-config=\"%s\" " + + "data-config-content-trusted=\"false\"></div>", escapeXML(json(expectedConfig.toString()))); List<Block> blocks = this.liveDataMacro.execute(parameters, "{...}", null); assertBlocks(expected, blocks, this.rendererFactory);
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/config/package.json+2 −1 modified@@ -19,7 +19,8 @@ "u-node": "0.0.2", "vuedraggable": "2.24.3", "vue-tippy": "4.16.0", - "vue2-touch-events": "3.2.2" + "vue2-touch-events": "3.2.2", + "dompurify": "2.4.0" }, "devDependencies": { "@vue/cli-plugin-babel": "5.0.8",
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/config/package-lock.json+11 −0 modified@@ -8,6 +8,7 @@ "name": "xwiki-livedata-vue", "version": "1.0.0", "dependencies": { + "dompurify": "2.4.0", "lz-string": "1.4.4", "u-node": "0.0.2", "vue": "2.7.10", @@ -6475,6 +6476,11 @@ "url": "https://github.com/fb55/domhandler?sponsor=1" } }, + "node_modules/dompurify": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-2.4.0.tgz", + "integrity": "sha512-Be9tbQMZds4a3C6xTmz68NlMfeONA//4dOavl/1rNw50E+/QO0KVpbcU0PcaW0nsQxurXls9ZocqFxk8R2mWEA==" + }, "node_modules/domutils": { "version": "2.8.0", "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", @@ -21827,6 +21833,11 @@ "domelementtype": "^2.2.0" } }, + "dompurify": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-2.4.0.tgz", + "integrity": "sha512-Be9tbQMZds4a3C6xTmz68NlMfeONA//4dOavl/1rNw50E+/QO0KVpbcU0PcaW0nsQxurXls9ZocqFxk8R2mWEA==" + }, "domutils": { "version": "2.8.0", "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz",
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/tests/unit/displayers/BaseDisplayer.spec.js+1 −1 modified@@ -81,7 +81,7 @@ describe('BaseDisplayer.vue', () => { } }); - expect(wrapper.find('div.view > div').html()).toBe('<div><span></span> <span>livedata.displayer.emptyValue</span></div>'); + expect(wrapper.find('div.view > div').text()).toBe('livedata.displayer.emptyValue*'); }) it('Renders a viewable entry with an empty content', () => {
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/tests/unit/displayers/DisplayerDocTitle.spec.js+2 −1 modified@@ -37,7 +37,8 @@ describe('DisplayerDocTitle.vue', () => { return { propertyHref: 'colorHref' }; - } + }, + isContentTrusted: () => true } }); expect(wrapper.find('a').html()).toBe('<a href="entryLink" class="">Test <sup>1</sup></a>');
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/tests/unit/displayers/DisplayerHtml.spec.js+18 −0 modified@@ -28,6 +28,24 @@ describe('DisplayerHtml.vue', () => { entry: { color: '<strong>some content</strong>' } + }, + logic: { + isContentTrusted: () => true + } + }) + expect(wrapper.find('.html-wrapper').html()) + .toBe('<div class="html-wrapper"><strong>some content</strong></div>') + }) + + it('Renders an entry in view mode with untrusted content', () => { + const wrapper = initWrapper(DisplayerHtml, { + props: { + entry: { + color: '<strong>some content<script>console.log("hello world")</script></strong>' + } + }, + logic: { + isContentTrusted: () => false } }) expect(wrapper.find('.html-wrapper').html())
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/tests/unit/displayers/DisplayerLink.spec.js+35 −0 modified@@ -43,6 +43,41 @@ describe('DisplayerLink.vue', () => { expect(wrapper.find('a').element.href).toBe('http://localhost/entryLink'); }) + it('Renders an entry in view mode with untrusted content', () => { + const logic = { + getDisplayerDescriptor() { + return { + propertyHref: 'colorHref' + }; + }, + isContentTrusted: () => false + }; + const wrapperHttpLink = initWrapper(DisplayerLink, { + props: { + entry: { + color: 'yellow<script>console.log("hello")</script>', + colorHref: 'http://test.com' + } + }, + logic + }); + expect(wrapperHttpLink.text()).toMatch('yellow') + expect(wrapperHttpLink.find('a').element.href).toBe('http://test.com/'); + + const wrapperJavascriptLink = initWrapper(DisplayerLink, { + props: { + entry: { + color: 'yellow<script>console.log("hello")</script>', + colorHref: 'javascript:console.log("world")' + } + }, + logic + }); + + expect(wrapperJavascriptLink.text()).toMatch('yellow') + expect(wrapperJavascriptLink.find('a').element.href).toBe('http://localhost/#'); + }) + it('Renders an entry in view mode with an empty content', () => { const wrapper = initWrapper(DisplayerLink, { props: {
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/tests/unit/displayers/displayerTestsHelper.js+2 −1 modified@@ -158,7 +158,8 @@ export function initWrapper(displayer, {props, logic, editBus, mocks}) put() {}, reset() {}, list() { return []; } - } + }, + isContentTrusted: () => true }, logic), }, mocks: Object.assign({
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/BaseDisplayer.vue+4 −5 modified@@ -71,10 +71,9 @@ --> <span>{{ value }}</span> </slot> - <span - v-if="!isViewable" - v-html="$t('livedata.displayer.emptyValue')" - ></span> + <span v-if="!isViewable"> + {{ $t('livedata.displayer.emptyValue') }}<sup>*</sup> + </span> </div> <!-- The slot containing the displayer Editor widget --> @@ -116,7 +115,7 @@ :close-popover="closePopover" /> <ActionFollowLink - :displayer="{ href }" + :displayer="{ href: sanitizeUrl(href) }" v-if="href" :close-popover="closePopover" />
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/DisplayerActions.vue+1 −1 modified@@ -46,7 +46,7 @@ :key="action.id" :class="'action action_' + action.id" :title="action.description" - :href="entry[action.urlProperty] || '#'" + :href="sanitizeUrl(entry[action.urlProperty]) || '#'" > <XWikiIcon :iconDescriptor="action.icon" /><span class="action-name">{{ action.name }}</span> </a>
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/DisplayerHtml.vue+2 −1 modified@@ -33,7 +33,8 @@ view-only> <!-- Provide the Html Viewer widget to the `viewer` slot --> <template #viewer> - <div class="html-wrapper" v-html="value"></div> + <div class="html-wrapper" + v-html="safeValue"/> </template> </BaseDisplayer> </template>
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/DisplayerLink.vue+3 −3 modified@@ -45,11 +45,11 @@ so we create an explicit "no value" message in that case --> <a v-if="linkContent && hasViewRight" - :href="href" + :href="sanitizeUrl(href)" :class="{'explicit-empty-value': !html && !htmlValue}" - v-html="linkContent" + v-html="sanitizeHtml(linkContent)" ></a> - <span v-else v-html="linkContent"></span> + <span v-else v-html="sanitizeHtml(linkContent)"></span> </template>
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/displayerMixin.js+20 −0 modified@@ -18,6 +18,7 @@ * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ +import * as DOMPurify from 'dompurify'; /** * The displayerMixin is a vue mixin containing all the needed @@ -59,6 +60,9 @@ export default { value () { return this.entry[this.propertyId]; }, + safeValue() { + return this.sanitizeHtml(this.value) + }, // The property descriptor of `this.propertyId` propertyDescriptor () { return this.logic.getPropertyDescriptor(this.propertyId); @@ -106,6 +110,22 @@ export default { genericSave(value) { const savedValue = value || this.editedValue; this.logic.getEditBus().save(this.entry, this.propertyId, {[this.propertyId]: savedValue}) + }, + sanitizeHtml(value) { + if (!this.logic.isContentTrusted()) { + // TODO: Take into account xml.htmlElementSanitizer properties when sanitizing (see XWIKI-20249). + return DOMPurify.sanitize(value); + } else { + return value; + } + }, + sanitizeUrl(url, subtitute) { + // TODO: Take into account xml.htmlElementSanitizer properties when sanitizing (see XWIKI-20249). + if (this.logic.isContentTrusted() || DOMPurify.isValidAttribute('a', 'href', url)) { + return url; + } else { + return (subtitute || '#'); + } } },
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/vue/displayers/DisplayerXObjectProperty.vue+5 −2 modified@@ -35,12 +35,15 @@ <!-- Provide the Html Viewer widget to the `viewer` slot --> <template #viewer> - <div :class="['html-wrapper', isLoading ? 'disabled' : '']" v-html="value" ref="xObjectPropertyView"></div> + <div :class="['html-wrapper', isLoading ? 'disabled' : '']" + v-html="sanitizeHtml(value)" + ref="xObjectPropertyView"/> </template> <!-- Provide the Html Editor widget to the `editor` slot --> <template #editor> - <div v-html="editField" ref="xObjectPropertyEdit"></div> + <div v-html="sanitizeHtml(editField)" + ref="xObjectPropertyEdit"/> </template> </BaseDisplayer> </template>
xwiki-platform-core/xwiki-platform-livedata/xwiki-platform-livedata-webjar/src/main/webjar/Logic.js+13 −0 modified@@ -105,6 +105,7 @@ define('xwiki-livedata', [ const Logic = function (element) { this.element = element; this.data = JSON.parse(element.getAttribute("data-config") || "{}"); + this.contentTrusted = element.getAttribute("data-config-content-trusted") === "true"; this.data.entries = Object.freeze(this.data.entries); // Reactive properties must be initialized before Vue is instantiated. @@ -1519,6 +1520,18 @@ define('xwiki-livedata', [ } else { this.panels.splice(index, 0, panel); } + }, + + // + // Content status + // + + /** + * @returns {boolean} when false, the content is not trusted will be sanitized whenever Vue integrated escaping + * is not enough. When true, the content is never sanitized + */ + isContentTrusted() { + return this.contentTrusted; } };
Vulnerability mechanics
Generated by null/stub on May 9, 2026. Inputs: CWE entries + fix-commit diffs from this CVE's patches. Citations validated against bundle.
References
6- github.com/advisories/GHSA-32fq-m2q5-h83gghsaADVISORY
- nvd.nist.gov/vuln/detail/CVE-2023-26480ghsaADVISORY
- github.com/xwiki/xwiki-platform/commit/23d5ea9b23e84b5f3d1f1b2d5673fe8c774d0d79ghsax_refsource_MISCWEB
- github.com/xwiki/xwiki-platform/commit/556e7823260b826f344c1a6e95d935774587e028ghsax_refsource_MISCWEB
- github.com/xwiki/xwiki-platform/security/advisories/GHSA-32fq-m2q5-h83gghsax_refsource_CONFIRMWEB
- jira.xwiki.org/browse/XWIKI-20143ghsax_refsource_MISCWEB
News mentions
0No linked articles in our index yet.