mirror of
https://github.com/corda/corda.git
synced 2025-01-08 14:03:06 +00:00
87b02eb949
Previously, I used a shell script to extract modification date ranges from the Git history, but that was complicated and unreliable, so now every file just gets the same year range in its copyright header. If someone needs to know when a specific file was modified and by whom, they can look at the Git history themselves; no need to include it redundantly in the header.
210 lines
4.7 KiB
Java
210 lines
4.7 KiB
Java
/* Copyright (c) 2008-2013, Avian Contributors
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
for any purpose with or without fee is hereby granted, provided
|
|
that the above copyright notice and this permission notice appear
|
|
in all copies.
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
details. */
|
|
|
|
package java.util;
|
|
|
|
import java.io.InputStream;
|
|
import java.io.OutputStream;
|
|
import java.io.PrintStream;
|
|
import java.io.IOException;
|
|
import java.io.Reader;
|
|
|
|
public class Properties extends Hashtable {
|
|
public void load(InputStream in) throws IOException {
|
|
new InputStreamParser(in).parse(this);
|
|
}
|
|
|
|
public void load(Reader reader) throws IOException {
|
|
new ReaderParser(reader).parse(this);
|
|
}
|
|
|
|
public void store(OutputStream out, String comment) throws IOException {
|
|
PrintStream os = new PrintStream(out);
|
|
os.println("# " + comment);
|
|
for (Iterator it = entrySet().iterator();
|
|
it.hasNext();) {
|
|
Map.Entry entry = (Map.Entry)it.next();
|
|
os.print(entry.getKey());
|
|
os.print('=');
|
|
os.println(entry.getValue());
|
|
}
|
|
os.flush();
|
|
}
|
|
|
|
public String getProperty(String key) {
|
|
return (String)get(key);
|
|
}
|
|
|
|
public String getProperty(String key, String defaultValue) {
|
|
String value = (String) get(key);
|
|
if (value == null) {
|
|
return defaultValue;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
public Object setProperty(String key, String value) {
|
|
return put(key, value);
|
|
}
|
|
|
|
public Enumeration<?> propertyNames() {
|
|
return keys();
|
|
}
|
|
|
|
private abstract static class Parser {
|
|
private StringBuilder key = null;
|
|
private StringBuilder value = null;
|
|
private StringBuilder current = null;
|
|
|
|
private void append(int c) {
|
|
if (current == null) {
|
|
if (key == null) {
|
|
current = key = new StringBuilder();
|
|
} else {
|
|
current = value = new StringBuilder();
|
|
}
|
|
}
|
|
|
|
current.append((char) c);
|
|
}
|
|
|
|
private void finishLine(Map<String, Object> map) {
|
|
if (key != null) {
|
|
map.put(key.toString(),
|
|
(value == null ? "" : value.toString().trim()));
|
|
}
|
|
|
|
key = value = current = null;
|
|
}
|
|
|
|
abstract int readCharacter() throws IOException;
|
|
|
|
char readUtf16() throws IOException {
|
|
char c = 0;
|
|
for (int i = 0; i < 4; ++i) {
|
|
int digit = Character.digit((char)readCharacter(), 16);
|
|
if (digit == -1) throw new IOException("Invalid Unicode escape encountered.");
|
|
c <<= 4;
|
|
c |= digit;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
void parse(Map map)
|
|
throws IOException
|
|
{
|
|
boolean escaped = false;
|
|
|
|
int c;
|
|
while ((c = readCharacter()) != -1) {
|
|
if (c == '\\') {
|
|
if (escaped) {
|
|
escaped = false;
|
|
append(c);
|
|
} else {
|
|
escaped = true;
|
|
}
|
|
} else {
|
|
switch (c) {
|
|
case '#':
|
|
case '!':
|
|
if (key == null) {
|
|
while ((c = readCharacter()) != -1 && c != '\n');
|
|
} else {
|
|
append(c);
|
|
}
|
|
break;
|
|
|
|
case ' ':
|
|
case '\r':
|
|
case '\t':
|
|
if (escaped || (current != null && value == current)) {
|
|
append(c);
|
|
} else if (key == current) {
|
|
current = null;
|
|
}
|
|
break;
|
|
|
|
case ':':
|
|
case '=':
|
|
if (escaped || (current != null && value == current)) {
|
|
append(c);
|
|
} else {
|
|
if (key == null) {
|
|
key = new StringBuilder();
|
|
}
|
|
current = null;
|
|
}
|
|
break;
|
|
|
|
case '\n':
|
|
if (escaped) {
|
|
append(c);
|
|
} else {
|
|
finishLine(map);
|
|
}
|
|
break;
|
|
case 'n':
|
|
if (escaped) {
|
|
append('\n');
|
|
} else {
|
|
append(c);
|
|
}
|
|
break;
|
|
|
|
case 'u':
|
|
if (escaped) {
|
|
append(readUtf16());
|
|
} else {
|
|
append(c);
|
|
} break;
|
|
|
|
default:
|
|
append(c);
|
|
break;
|
|
}
|
|
|
|
escaped = false;
|
|
}
|
|
}
|
|
|
|
finishLine(map);
|
|
}
|
|
}
|
|
|
|
static class InputStreamParser extends Parser {
|
|
InputStream in;
|
|
|
|
|
|
public InputStreamParser(InputStream in) {
|
|
this.in = in;
|
|
}
|
|
|
|
@Override
|
|
int readCharacter() throws IOException {
|
|
return in.read();
|
|
}
|
|
}
|
|
|
|
static class ReaderParser extends Parser {
|
|
Reader in;
|
|
|
|
public ReaderParser(Reader in) {
|
|
this.in = in;
|
|
}
|
|
|
|
@Override
|
|
int readCharacter() throws IOException {
|
|
return in.read();
|
|
}
|
|
}
|
|
|
|
}
|