提交 c39c4211 编写于 作者: S Sam Brannen

Introduce support for sorted Properties

This commit introduces an internal SortedProperties class that is a
specialization of java.util.Properties which sorts properties
alphanumerically based on their keys.

This can be useful when storing a java.util.Properties instance in a
properties file, since it allows such files to be generated in a
repeatable manner with consistent ordering of properties.

Comments in generated properties files can also be optionally omitted.

An instance of SortedProperties can be created via two new
createSortedProperties() factory methods in
org.springframework.core.CollectionFactory.

Closes gh-23018
上级 98079a36
......@@ -346,6 +346,50 @@ public final class CollectionFactory {
};
}
/**
* Create a variant of {@link java.util.Properties} that sorts properties
* alphanumerically based on their keys.
*
* <p>This can be useful when storing the {@link Properties} instance in a
* properties file, since it allows such files to be generated in a repeatable
* manner with consistent ordering of properties. Comments in generated
* properties files can also be optionally omitted.
*
* @param omitComments {@code true} if comments should be omitted when
* storing properties in a file
* @return a new {@code Properties} instance
* @since 5.2
* @see #createSortedProperties(Properties, boolean)
*/
public static Properties createSortedProperties(boolean omitComments) {
return new SortedProperties(omitComments);
}
/**
* Create a variant of {@link java.util.Properties} that sorts properties
* alphanumerically based on their keys.
*
* <p>This can be useful when storing the {@code Properties} instance in a
* properties file, since it allows such files to be generated in a repeatable
* manner with consistent ordering of properties. Comments in generated
* properties files can also be optionally omitted.
*
* <p>The returned {@code Properties} instance will be populated with
* properties from the supplied {@code properties} object, but default
* properties from the supplied {@code properties} object will not be copied.
*
* @param properties the {@code Properties} object from which to copy the
* initial properties
* @param omitComments {@code true} if comments should be omitted when
* storing properties in a file
* @return a new {@code Properties} instance
* @since 5.2
* @see #createSortedProperties(boolean)
*/
public static Properties createSortedProperties(Properties properties, boolean omitComments) {
return new SortedProperties(properties, omitComments);
}
/**
* Cast the given type to a subtype of {@link Enum}.
* @param enumType the enum type, never {@code null}
......
/*
* Copyright 2002-2019 the original author or authors.
*
* 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
*
* https://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.
*/
package org.springframework.core;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.StringWriter;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.Comparator;
import java.util.Enumeration;
import java.util.Map.Entry;
import java.util.Properties;
import java.util.Set;
import java.util.TreeSet;
import org.springframework.util.StringUtils;
/**
* Specialization of {@link Properties} that sorts properties alphanumerically
* based on their keys.
*
* <p>This can be useful when storing the {@link Properties} instance in a
* properties file, since it allows such files to be generated in a repeatable
* manner with consistent ordering of properties.
*
* <p>Comments in generated properties files can also be optionally omitted.
*
* @author Sam Brannen
* @since 5.2
* @see java.util.Properties
*/
@SuppressWarnings("serial")
class SortedProperties extends Properties {
static final String EOL = System.getProperty("line.separator");
private static final Comparator<Object> keyComparator = //
(key1, key2) -> String.valueOf(key1).compareTo(String.valueOf(key2));
private static final Comparator<Entry<Object, Object>> entryComparator = //
Entry.comparingByKey(keyComparator);
private final boolean omitComments;
/**
* Construct a new {@code SortedProperties} instance that honors the supplied
* {@code omitComments} flag.
*
* @param omitComments {@code true} if comments should be omitted when
* storing properties in a file
*/
SortedProperties(boolean omitComments) {
this.omitComments = omitComments;
}
/**
* Construct a new {@code SortedProperties} instance with properties populated
* from the supplied {@link Properties} object and honoring the supplied
* {@code omitComments} flag.
*
* <p>Default properties from the supplied {@code Properties} object will
* not be copied.
*
* @param properties the {@code Properties} object from which to copy the
* initial properties
* @param omitComments {@code true} if comments should be omitted when
* storing properties in a file
*/
SortedProperties(Properties properties, boolean omitComments) {
this(omitComments);
putAll(properties);
}
@Override
public void store(OutputStream out, String comments) throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
super.store(baos, (this.omitComments ? null : comments));
String contents = new String(baos.toByteArray(), StandardCharsets.ISO_8859_1);
for (String line : StringUtils.tokenizeToStringArray(contents, EOL)) {
if (!this.omitComments || !line.startsWith("#")) {
out.write((line + EOL).getBytes(StandardCharsets.ISO_8859_1));
}
}
}
@Override
public void store(Writer writer, String comments) throws IOException {
StringWriter stringWriter = new StringWriter();
super.store(stringWriter, (this.omitComments ? null : comments));
String contents = stringWriter.toString();
for (String line : StringUtils.tokenizeToStringArray(contents, EOL)) {
if (!this.omitComments || !line.startsWith("#")) {
writer.write(line + EOL);
}
}
}
@Override
public void storeToXML(OutputStream out, String comments) throws IOException {
super.storeToXML(out, (this.omitComments ? null : comments));
}
@Override
public void storeToXML(OutputStream out, String comments, String encoding) throws IOException {
super.storeToXML(out, (this.omitComments ? null : comments), encoding);
}
/**
* Return a sorted enumeration of the keys in this {@link Properties} object.
* @see #keySet()
*/
@Override
public synchronized Enumeration<Object> keys() {
return Collections.enumeration(keySet());
}
/**
* Return a sorted set of the keys in this {@link Properties} object.
* <p>The keys will be converted to strings if necessary using
* {@link String#valueOf(Object)} and sorted alphanumerically according to
* the natural order of strings.
*/
@Override
public Set<Object> keySet() {
Set<Object> sortedKeys = new TreeSet<>(keyComparator);
sortedKeys.addAll(super.keySet());
return Collections.synchronizedSet(sortedKeys);
}
/**
* Return a sorted set of the entries in this {@link Properties} object.
* <p>The entries will be sorted based on their keys, and the keys will be
* converted to strings if necessary using {@link String#valueOf(Object)}
* and compared alphanumerically according to the natural order of strings.
*/
@Override
public Set<Entry<Object, Object>> entrySet() {
Set<Entry<Object, Object>> sortedEntries = new TreeSet<>(entryComparator);
sortedEntries.addAll(super.entrySet());
return Collections.synchronizedSet(sortedEntries);
}
}
/*
* Copyright 2002-2019 the original author or authors.
*
* 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
*
* https://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.
*/
package org.springframework.core;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.StringWriter;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.Properties;
import org.junit.Test;
import static java.util.Arrays.stream;
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.entry;
/**
* Unit tests for {@link SortedProperties}.
*
* @author Sam Brannen
* @since 5.2
*/
public class SortedPropertiesTests {
@Test
public void keys() {
assertKeys(createSortedProps());
}
@Test
public void keysFromPrototype() {
assertKeys(createSortedPropsFromPrototype());
}
@Test
public void keySet() {
assertKeySet(createSortedProps());
}
@Test
public void keySetFromPrototype() {
assertKeySet(createSortedPropsFromPrototype());
}
@Test
public void entrySet() {
assertEntrySet(createSortedProps());
}
@Test
public void entrySetFromPrototype() {
assertEntrySet(createSortedPropsFromPrototype());
}
@Test
public void sortsPropertiesUsingOutputStream() throws IOException {
SortedProperties sortedProperties = createSortedProps();
ByteArrayOutputStream baos = new ByteArrayOutputStream();
sortedProperties.store(baos, "custom comment");
String[] lines = lines(baos);
assertThat(lines).hasSize(7);
assertThat(lines[0]).isEqualTo("#custom comment");
assertThat(lines[1]).as("timestamp").startsWith("#");
assertPropsAreSorted(lines);
}
@Test
public void sortsPropertiesUsingWriter() throws IOException {
SortedProperties sortedProperties = createSortedProps();
StringWriter writer = new StringWriter();
sortedProperties.store(writer, "custom comment");
String[] lines = lines(writer);
assertThat(lines).hasSize(7);
assertThat(lines[0]).isEqualTo("#custom comment");
assertThat(lines[1]).as("timestamp").startsWith("#");
assertPropsAreSorted(lines);
}
@Test
public void sortsPropertiesAndOmitsCommentsUsingOutputStream() throws IOException {
SortedProperties sortedProperties = createSortedProps(true);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
sortedProperties.store(baos, "custom comment");
String[] lines = lines(baos);
assertThat(lines).hasSize(5);
assertPropsAreSorted(lines);
}
@Test
public void sortsPropertiesAndOmitsCommentsUsingWriter() throws IOException {
SortedProperties sortedProperties = createSortedProps(true);
StringWriter writer = new StringWriter();
sortedProperties.store(writer, "custom comment");
String[] lines = lines(writer);
assertThat(lines).hasSize(5);
assertPropsAreSorted(lines);
}
@Test
public void storingAsXmlSortsPropertiesAndOmitsComments() throws IOException {
SortedProperties sortedProperties = createSortedProps(true);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
sortedProperties.storeToXML(baos, "custom comment");
String[] lines = lines(baos);
assertThat(lines).containsExactly( //
"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>", //
"<!DOCTYPE properties SYSTEM \"http://java.sun.com/dtd/properties.dtd\">", //
"<properties>", //
"<entry key=\"color\">blue</entry>", //
"<entry key=\"fragrance\">sweet</entry>", //
"<entry key=\"fruit\">apple</entry>", //
"<entry key=\"size\">medium</entry>", //
"<entry key=\"vehicle\">car</entry>", //
"</properties>" //
);
}
private SortedProperties createSortedProps() {
return createSortedProps(false);
}
private SortedProperties createSortedProps(boolean omitComments) {
SortedProperties sortedProperties = new SortedProperties(omitComments);
populateProperties(sortedProperties);
return sortedProperties;
}
private SortedProperties createSortedPropsFromPrototype() {
Properties properties = new Properties();
populateProperties(properties);
return new SortedProperties(properties, false);
}
private void populateProperties(Properties properties) {
properties.setProperty("color", "blue");
properties.setProperty("fragrance", "sweet");
properties.setProperty("fruit", "apple");
properties.setProperty("size", "medium");
properties.setProperty("vehicle", "car");
}
private String[] lines(ByteArrayOutputStream baos) {
return lines(new String(baos.toByteArray(), StandardCharsets.ISO_8859_1));
}
private String[] lines(StringWriter writer) {
return lines(writer.toString());
}
private String[] lines(String input) {
return input.trim().split(SortedProperties.EOL);
}
private void assertKeys(Properties properties) {
assertThat(Collections.list(properties.keys())) //
.containsExactly("color", "fragrance", "fruit", "size", "vehicle");
}
private void assertKeySet(Properties properties) {
assertThat(properties.keySet()).containsExactly("color", "fragrance", "fruit", "size", "vehicle");
}
private void assertEntrySet(Properties properties) {
assertThat(properties.entrySet()).containsExactly( //
entry("color", "blue"), //
entry("fragrance", "sweet"), //
entry("fruit", "apple"), //
entry("size", "medium"), //
entry("vehicle", "car") //
);
}
private void assertPropsAreSorted(String[] lines) {
assertThat(stream(lines).filter(s -> !s.startsWith("#"))).containsExactly( //
"color=blue", //
"fragrance=sweet", //
"fruit=apple", //
"size=medium", //
"vehicle=car"//
);
}
}
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册