org.openide.util.EditableProperties Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*/
package org.openide.util;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.nio.charset.StandardCharsets;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
// XXX: consider adding getInitialComment() and setInitialComment() methods
// (useful e.g. for GeneratedFilesHelper)
/**
* Similar to {@link java.util.Properties} but designed to retain additional
* information needed for safe hand-editing.
* Useful for various *.properties
in a project:
*
* - Can associate comments with particular entries.
*
- Order of entries preserved during modifications whenever possible.
*
- VCS-friendly: lines which are not semantically modified are not textually modified.
*
- Can automatically insert line breaks in new or modified values at positions
* that are likely to be semantically meaningful, e.g. between path components
*
* The file format (including encoding etc.) is compatible with the regular JRE implementation.
* Only (non-null) String is supported for keys and values.
* This class is not thread-safe; use only from a single thread, or use {@link java.util.Collections#synchronizedMap}.
* @author Jesse Glick, David Konecny
* @since org.openide.util 7.26
*/
public final class EditableProperties extends AbstractMap implements Cloneable {
private static class State {
/** whether multiple EP instances are currently linking to this */
boolean shared;
/** List of Item instances as read from the properties file. Order is important.
* Saving properties will save then in this order. */
final LinkedList- items;
/** Map of [property key, Item instance] for faster access. */
final Map
itemIndex;
/** create fresh state */
State() {
items = new LinkedList- ();
itemIndex = new HashMap
();
}
/** duplicate state */
State(State original) {
items = new LinkedList- ();
itemIndex = new HashMap
(original.items.size() * 4 / 3 + 1);
for (Item _i : original.items) {
Item i = (Item) _i.clone();
items.add(i);
if(i.getKey() != null) {
itemIndex.put(i.getKey(), i);
}
}
}
}
private State state;
private final boolean alphabetize;
private static final String INDENT = " ";
// parse states:
private static final int WAITING_FOR_KEY_VALUE = 1;
private static final int READING_KEY_VALUE = 2;
/**
* Creates empty instance.
* @param alphabetize alphabetize new items according to key or not
*/
public EditableProperties(boolean alphabetize) {
this.alphabetize = alphabetize;
state = new State();
}
/**
* Creates new instance from an existing one.
* @param ep an instance of EditableProperties
*/
private EditableProperties(EditableProperties ep) {
// #64174: use a simple deep copy for speed
alphabetize = ep.alphabetize;
state = ep.state;
state.shared = true;
}
private void writeOperation() {
if (state.shared) {
state = new State(state);
}
}
/**
* Returns a set view of the mappings ordered according to their file
* position. Each element in this set is a Map.Entry. See
* {@link AbstractMap#entrySet} for more details.
* @return set with Map.Entry instances.
*/
public Set> entrySet() {
return new SetImpl();
}
/**
* Load properties from a stream.
* @param stream an input stream
* @throws IOException if the contents are malformed or the stream could not be read
*/
public void load(InputStream stream) throws IOException {
int parseState = WAITING_FOR_KEY_VALUE;
BufferedReader input = new BufferedReader(new InputStreamReader(stream, StandardCharsets.ISO_8859_1));
List tempList = new LinkedList();
String line;
int commentLinesCount = 0;
// Read block of lines and create instance of Item for each.
// Separator is: either empty line or valid end of proeprty declaration
while (null != (line = input.readLine())) {
tempList.add(line);
boolean empty = isEmpty(line);
boolean comment = isComment(line);
if (parseState == WAITING_FOR_KEY_VALUE) {
if (empty) {
// empty line: create Item without any key
createNonKeyItem(tempList);
commentLinesCount = 0;
} else {
if (comment) {
commentLinesCount++;
} else {
parseState = READING_KEY_VALUE;
}
}
}
if (parseState == READING_KEY_VALUE && !isContinue(line)) {
// valid end of property declaration: create Item for it
createKeyItem(tempList, commentLinesCount);
parseState = WAITING_FOR_KEY_VALUE;
commentLinesCount = 0;
}
}
if (tempList.size() > 0) {
if (parseState == READING_KEY_VALUE) {
// value was not ended correctly? ignore.
createKeyItem(tempList, commentLinesCount);
} else {
createNonKeyItem(tempList);
}
}
}
/**
* Store properties to a stream.
* @param stream an output stream
* @throws IOException if the stream could not be written to
*/
public void store(OutputStream stream) throws IOException {
boolean previousLineWasEmpty = true;
BufferedWriter output = new BufferedWriter(new OutputStreamWriter(stream, StandardCharsets.ISO_8859_1));
for (Item item : state.items) {
if (item.isSeparate() && !previousLineWasEmpty) {
output.newLine();
}
String line = null;
Iterator it = item.getRawData().iterator();
while (it.hasNext()) {
line = it.next();
output.write(line);
output.newLine();
}
if (line != null) {
previousLineWasEmpty = isEmpty(line);
}
}
output.flush();
}
@Override
public String get(Object key) {
if (!(key instanceof String)) {
return null;
}
Item item = state.itemIndex.get((String) key);
return item != null ? item.getValue() : null;
}
@Override
public String put(String key, String value) {
Parameters.notNull("key", key);
Parameters.notNull(key, value);
writeOperation();
Item item = state.itemIndex.get(key);
String result = null;
if (item != null) {
result = item.getValue();
item.setValue(value);
} else {
item = new Item(key, value);
addItem(item, alphabetize);
}
return result;
}
/**
* Convenience method to get a property as a string.
* Same as {@link #get}; only here because of pre-generic code.
* @param key a property name; cannot be null nor empty
* @return the property value, or null if it was not defined
*/
public String getProperty(String key) {
return get(key);
}
/**
* Convenience method to set a property.
* Same as {@link #put}; only here because of pre-generic code.
* @param key a property name; cannot be null nor empty
* @param value the desired value; cannot be null
* @return previous value of the property or null if there was not any
*/
public String setProperty(String key, String value) {
return put(key, value);
}
/**
* Sets a property to a value broken into segments for readability.
* Same behavior as {@link #setProperty(String,String)} with the difference that each item
* will be stored on its own line of text. {@link #getProperty} will simply concatenate
* all the items into one string, so generally separators
* (such as :
for path-like properties) must be included in
* the items (for example, at the end of all but the last item).
* @param key a property name; cannot be null nor empty
* @param value the desired value; cannot be null; can be empty array
* @return previous value of the property or null if there was not any
*/
public String setProperty(String key, String[] value) {
String result = get(key);
if (key == null || value == null) {
throw new NullPointerException();
}
List valueList = Arrays.asList(value);
writeOperation();
Item item = state.itemIndex.get(key);
if (item != null) {
item.setValue(valueList);
} else {
addItem(new Item(key, valueList), alphabetize);
}
return result;
}
/**
* Returns comment associated with the property. The comment lines are
* returned as defined in properties file, that is comment delimiter is
* included. Comment for property is defined as: continuous block of lines
* starting with comment delimiter which are followed by property
* declaration (no empty line separator allowed).
* @param key a property name; cannot be null nor empty
* @return array of String lines as specified in properties file; comment
* delimiter character is included
*/
public String[] getComment(String key) {
Item item = state.itemIndex.get(key);
if (item == null) {
return new String[0];
}
return item.getComment();
}
/**
* Create comment for the property.
* Note: if a comment includes non-ISO-8859-1 characters, they will be written
* to disk using Unicode escapes (and {@link #getComment} will interpret
* such escapes), but of course they will be unreadable for humans.
* @param key a property name; cannot be null nor empty
* @param comment lines of comment which will be written just above
* the property; no reformatting; comment lines must start with
* comment delimiter; cannot be null; cannot be emty array
* @param separate whether the comment should be separated from previous
* item by empty line
*/
public void setComment(String key, String[] comment, boolean separate) {
// XXX: check validity of comment parameter
writeOperation();
Item item = state.itemIndex.get(key);
if (item == null) {
throw new IllegalArgumentException("Cannot set comment for non-existing property "+key);
}
item.setComment(comment, separate);
}
@Override
public Object clone() {
return cloneProperties();
}
/**
* Create an exact copy of this properties object.
* @return a clone of this object
*/
public EditableProperties cloneProperties() {
return new EditableProperties(this);
}
// non-key item is block of empty lines/comment not associated with any property
private void createNonKeyItem(List lines) {
writeOperation();
// First check that previous item is not non-key item.
if (!state.items.isEmpty()) {
Item item = state.items.getLast();
if (item.getKey() == null) {
// it is non-key item: merge them
item.addCommentLines(lines);
lines.clear();
return;
}
}
// create new non-key item
Item item = new Item(lines);
addItem(item, false);
lines.clear();
}
// opposite to non-key item: item with valid property declaration and
// perhaps some comment lines
private void createKeyItem(List lines, int commentLinesCount) {
Item item = new Item(lines.subList(0, commentLinesCount), lines.subList(commentLinesCount, lines.size()));
addItem(item, false);
lines.clear();
}
private void addItem(Item item, boolean sort) {
writeOperation();
String key = item.getKey();
if (sort) {
assert key != null;
ListIterator- it = state.items.listIterator();
while (it.hasNext()) {
String k = it.next().getKey();
if (k != null && k.compareToIgnoreCase(key) > 0) {
it.previous();
it.add(item);
state.itemIndex.put(key, item);
return;
}
}
}
state.items.add(item);
if (key != null) {
state.itemIndex.put(key, item);
}
}
// does property declaration continue on next line?
private boolean isContinue(String line) {
int index = line.length() - 1;
int slashCount = 0;
while (index >= 0 && line.charAt(index) == '\\') {
slashCount++;
index--;
}
// if line ends with odd number of backslash then property definition
// continues on next line
return (slashCount % 2 != 0);
}
// does line start with comment delimiter? (whitespaces are ignored)
private static boolean isComment(String line) {
line = trimLeft(line);
if (line.length() > 0) {
switch (line.charAt(0)) {
case '#':
case '!':
return true;
}
}
return false;
}
// is line empty? (whitespaces are ignored)
private static boolean isEmpty(String line) {
return trimLeft(line).length() == 0;
}
// remove all whitespaces from left
private static String trimLeft(String line) {
int start = 0;
int len = line.length();
NONWS: while (start < len) {
switch (line.charAt(start)) {
case ' ':
case '\t':
case '\r':
case '\n':
case '\f':
start++;
break;
default:
break NONWS;
}
}
return line.substring(start);
}
/**
* Representation of one item read from properties file. It can be either
* valid property declaration with associated comment or chunk of empty
* lines or lines with comment which are not associated with any property.
*/
private static class Item implements Cloneable {
private static List
EMPTY_LIST = Collections.emptyList();
/** Lines of comment as read from properties file and as they will be
* written back to properties file. */
private List commentLines;
/** Lines with property name and value declaration as read from
* properties file and as they will be written back to properties file. */
private List keyValueLines;
/** Property key */
private String key;
/** Property value */
private String value;
/** Should this property be separated from previous one by at least
* one empty line. */
private boolean separate;
// constructor only for cloning
private Item() {
}
/**
* Create instance which does not have any key and value - just
* some empty or comment lines. This item is READ-ONLY.
*/
public Item(List commentLines) {
this.commentLines = commentLines.isEmpty() ? EMPTY_LIST : new ArrayList(commentLines);
}
/**
* Create instance from the lines of comment and property declaration.
* Property name and value will be split.
*/
public Item(List commentLines, List keyValueLines) {
this.commentLines = commentLines.isEmpty() ? EMPTY_LIST : new ArrayList(commentLines);
this.keyValueLines = keyValueLines.isEmpty() ? EMPTY_LIST : new ArrayList(keyValueLines);
parse(keyValueLines);
}
/**
* Create new instance with key and value.
*/
public Item(String key, String value) {
this.key = key;
this.value = value;
}
/**
* Create new instance with key and value.
*/
public Item(String key, List value) {
this.key = key;
setValue(value);
}
// backdoor for merging non-key items
void addCommentLines(List lines) {
assert key == null;
if (commentLines == EMPTY_LIST) {
if (!lines.isEmpty()) {
commentLines = new ArrayList(lines);
}
} else {
commentLines.addAll(lines);
}
}
public String[] getComment() {
String[] res = new String[commentLines.size()];
for (int i = 0; i < res.length; i++) {
// #60249: the comment might have Unicode chars in escapes.
res[i] = decodeUnicode(commentLines.get(i));
}
return res;
}
public void setComment(String[] commentLines, boolean separate) {
this.separate = separate;
if (commentLines.length > 0) {
this.commentLines = new ArrayList(commentLines.length);
for (int i = 0; i < commentLines.length; i++) {
// #60249 again - write only ISO-8859-1.
this.commentLines.add(encodeUnicode(commentLines[i]));
}
} else {
this.commentLines = EMPTY_LIST;
}
}
public String getKey() {
return key;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
keyValueLines = null;
}
public void setValue(List value) {
StringBuilder val = new StringBuilder();
List l = new ArrayList();
if (!value.isEmpty()) {
l.add(encode(key, true) + "=\\"); // NOI18N
Iterator it = value.iterator();
while (it.hasNext()) {
String s = it.next();
val.append(s);
s = encode(s, false);
l.add(it.hasNext() ? INDENT + s + '\\' : INDENT + s); // NOI18N
}
} else {
// #45061: for no vals, use just "prop="
l.add(encode(key, true) + '='); // NOI18N
}
this.value = val.toString();
keyValueLines = l.isEmpty() ? EMPTY_LIST : l;
}
public boolean isSeparate() {
return separate;
}
/**
* Returns persistent image of this property.
*/
public List getRawData() {
List l = new ArrayList();
if (commentLines != null) {
l.addAll(commentLines);
}
if (keyValueLines == null) {
keyValueLines = new ArrayList();
if (key != null && value != null) {
keyValueLines.add(encode(key, true)+"="+encode(value, false));
}
}
l.addAll(keyValueLines);
return l;
}
private void parse(List keyValueLines) {
// merge lines into one:
String line = mergeLines(keyValueLines);
// split key and value
splitKeyValue(line);
}
private static String mergeLines(List lines) {
if (lines.size() == 1) {
return trimLeft(lines.get(0));
}
StringBuilder line = new StringBuilder();
Iterator it = lines.iterator();
while (it.hasNext()) {
String l = trimLeft(it.next());
// if this is not the last line then remove last backslash
if (it.hasNext()) {
assert l.endsWith("\\") : lines;
l = l.substring(0, l.length()-1);
}
line.append(l);
}
return line.toString();
}
private void splitKeyValue(String line) {
int separatorIndex = 0;
int len = line.length();
POS: while (separatorIndex < len) {
char ch = line.charAt(separatorIndex);
if (ch == '\\') {
// ignore next one character
separatorIndex++;
} else {
switch (ch) {
case '=':
case ':':
case ' ':
case '\t':
case '\r':
case '\n':
case '\f':
break POS;
}
}
separatorIndex++;
}
key = decode(line.substring(0, separatorIndex));
line = trimLeft(line.substring(separatorIndex));
if (line.length() == 0) {
value = "";
return;
}
switch (line.charAt(0)) {
case '=':
case ':':
line = trimLeft(line.substring(1));
}
value = decode(line);
}
private static String decode(String input) {
if (input.indexOf('\\') == -1) {
return input; // shortcut
}
char ch;
int len = input.length();
StringBuilder output = new StringBuilder(len);
for (int x=0; xlen) {
// unicode character not finished? syntax error: ignore
output.append(input.substring(x-1));
x += 4;
continue;
}
String val = input.substring(x+1, x+5);
try {
output.append((char)Integer.parseInt(val, 16));
} catch (NumberFormatException e) {
// #46234: handle gracefully
output.append(input.substring(x - 1, x + 5));
}
x += 4;
} else {
if (ch == 't') ch = '\t';
else if (ch == 'r') ch = '\r';
else if (ch == 'n') ch = '\n';
else if (ch == 'f') ch = '\f';
output.append(ch);
}
}
return output.toString();
}
private static String encode(String input, boolean forKey) {
int len = input.length();
StringBuilder output = new StringBuilder(len*2);
for(int x=0; x 0x007e)) {
output.append("\\u");
String hex = Integer.toHexString(ch);
for (int i = 0; i < 4 - hex.length(); i++) {
output.append('0');
}
output.append(hex);
} else {
output.append(ch);
}
}
}
return output.toString();
}
private static String decodeUnicode(String input) {
char ch;
int len = input.length();
StringBuilder output = new StringBuilder(len);
for (int x = 0; x < len; x++) {
ch = input.charAt(x);
if (ch != '\\') {
output.append(ch);
continue;
}
x++;
if (x==len) {
// backslash at the end? syntax error: ignore it
continue;
}
ch = input.charAt(x);
if (ch == 'u') {
if (x+5>len) {
// unicode character not finished? syntax error: ignore
output.append(input.substring(x-1));
x += 4;
continue;
}
String val = input.substring(x+1, x+5);
try {
output.append((char)Integer.parseInt(val, 16));
} catch (NumberFormatException e) {
// #46234: handle gracefully
output.append(input.substring(x - 1, x + 5));
}
x += 4;
} else {
output.append(ch);
}
}
return output.toString();
}
private static String encodeUnicode(String input) {
int len = input.length();
StringBuilder output = new StringBuilder(len * 2);
for (int x = 0; x < len; x++) {
char ch = input.charAt(x);
if ((ch < 0x0020) || (ch > 0x007e)) {
output.append("\\u"); // NOI18N
String hex = Integer.toHexString(ch);
for (int i = 0; i < 4 - hex.length(); i++) {
output.append('0');
}
output.append(hex);
} else {
output.append(ch);
}
}
return output.toString();
}
@Override
public Object clone() {
Item item = new Item();
if (keyValueLines != null) {
item.keyValueLines = this.keyValueLines.isEmpty() ? EMPTY_LIST : new ArrayList(keyValueLines);
}
if (commentLines != null) {
item.commentLines = this.commentLines.isEmpty() ? EMPTY_LIST : new ArrayList(commentLines);
}
item.key = key;
item.value = value;
item.separate = separate;
return item;
}
}
private class SetImpl extends AbstractSet> {
public SetImpl() {}
public Iterator> iterator() {
return new IteratorImpl();
}
public int size() {
return state.itemIndex.size();
}
}
private class IteratorImpl implements Iterator> {
private ListIterator- delegate;
public IteratorImpl() {
delegate = state.items.listIterator();
}
public boolean hasNext() {
return findNext() != null;
}
public Map.Entry
next() {
Item item = findNext();
if (item == null) {
throw new NoSuchElementException();
}
delegate.next();
return new MapEntryImpl(item);
}
@Override
public void remove() {
delegate.previous();
Item item = findNext();
if (item == null) {
throw new IllegalStateException();
}
int index = delegate.nextIndex();
writeOperation();
Item removed = state.items.remove(index);
assert removed.getKey().equals(item.getKey());
state.itemIndex.remove(item.getKey());
delegate = state.items.listIterator(index);
}
private Item findNext() {
while (delegate.hasNext()) {
Item item = delegate.next();
if (item.getKey() != null && item.getValue() != null) {
// Found one. Back up!
delegate.previous();
return item;
}
}
return null;
}
}
private class MapEntryImpl implements Map.Entry {
private Item item;
public MapEntryImpl(Item item) {
this.item = item;
}
public String getKey() {
return item.getKey();
}
public String getValue() {
return item.getValue();
}
public String setValue(String value) {
writeOperation();
item = state.itemIndex.get(item.getKey());
String result = item.getValue();
item.setValue(value);
return result;
}
}
}