mirror of https://github.com/apache/lucene.git
SOLR-79: Add system property ${<sys.prop>[:<default>]} substitution for
configuration files loaded, including schema.xml and solrconfig.xml. (Erik Hatcher with inspiration from Andrew Saar) git-svn-id: https://svn.apache.org/repos/asf/lucene/solr/trunk@508485 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
7f130dd630
commit
33c0b65e47
|
@ -82,6 +82,10 @@ New Features
|
||||||
|
|
||||||
10. SOLR-116: IndexInfoRequestHandler added. (Erik Hatcher)
|
10. SOLR-116: IndexInfoRequestHandler added. (Erik Hatcher)
|
||||||
|
|
||||||
|
11. SOLR-79: Add system property ${<sys.prop>[:<default>]} substitution for
|
||||||
|
configuration files loaded, including schema.xml and solrconfig.xml.
|
||||||
|
(Erik Hatcher with inspiration from Andrew Saar)
|
||||||
|
|
||||||
Changes in runtime behavior
|
Changes in runtime behavior
|
||||||
1. Highlighting using DisMax will only pick up terms from the main
|
1. Highlighting using DisMax will only pick up terms from the main
|
||||||
user query, not boost or filter queries (klaas).
|
user query, not boost or filter queries (klaas).
|
||||||
|
|
|
@ -282,6 +282,9 @@
|
||||||
<syspropertyset>
|
<syspropertyset>
|
||||||
<propertyref prefix="solr" />
|
<propertyref prefix="solr" />
|
||||||
</syspropertyset>
|
</syspropertyset>
|
||||||
|
<!-- solr.test.sys.prop1/2 used by TestConfig -->
|
||||||
|
<sysproperty key="solr.test.sys.prop1" value="propone"/>
|
||||||
|
<sysproperty key="solr.test.sys.prop2" value="proptwo"/>
|
||||||
<classpath refid="test.run.classpath"/>
|
<classpath refid="test.run.classpath"/>
|
||||||
<formatter type="xml"/>
|
<formatter type="xml"/>
|
||||||
<batchtest fork="yes" todir="${junit.output.dir}" unless="testcase">
|
<batchtest fork="yes" todir="${junit.output.dir}" unless="testcase">
|
||||||
|
|
|
@ -59,6 +59,8 @@ public class Config {
|
||||||
|
|
||||||
javax.xml.parsers.DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder();
|
javax.xml.parsers.DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder();
|
||||||
doc = builder.parse(is);
|
doc = builder.parse(is);
|
||||||
|
|
||||||
|
DOMUtil.substituteSystemProperties(doc);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Document getDocument() {
|
public Document getDocument() {
|
||||||
|
@ -124,7 +126,7 @@ public class Config {
|
||||||
if (nd==null) return null;
|
if (nd==null) return null;
|
||||||
|
|
||||||
String txt = DOMUtil.getText(nd);
|
String txt = DOMUtil.getText(nd);
|
||||||
|
|
||||||
log.fine(name + ' '+path+'='+txt);
|
log.fine(name + ' '+path+'='+txt);
|
||||||
return txt;
|
return txt;
|
||||||
|
|
||||||
|
@ -255,7 +257,7 @@ public class Config {
|
||||||
|
|
||||||
/** Singleton classloader loading resources specified in any configs */
|
/** Singleton classloader loading resources specified in any configs */
|
||||||
private static ClassLoader classLoader = null;
|
private static ClassLoader classLoader = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the singleton classloader to be use when loading resources
|
* Returns the singleton classloader to be use when loading resources
|
||||||
* specified in any configs.
|
* specified in any configs.
|
||||||
|
@ -269,7 +271,7 @@ public class Config {
|
||||||
static ClassLoader getClassLoader() {
|
static ClassLoader getClassLoader() {
|
||||||
if (null == classLoader) {
|
if (null == classLoader) {
|
||||||
classLoader = Thread.currentThread().getContextClassLoader();
|
classLoader = Thread.currentThread().getContextClassLoader();
|
||||||
|
|
||||||
File f = new File(getInstanceDir() + "lib/");
|
File f = new File(getInstanceDir() + "lib/");
|
||||||
if (f.canRead() && f.isDirectory()) {
|
if (f.canRead() && f.isDirectory()) {
|
||||||
File[] jarFiles = f.listFiles();
|
File[] jarFiles = f.listFiles();
|
||||||
|
|
|
@ -110,7 +110,7 @@ public abstract class AbstractSolrTestCase extends TestCase {
|
||||||
* is set.
|
* is set.
|
||||||
*/
|
*/
|
||||||
public void tearDown() throws Exception {
|
public void tearDown() throws Exception {
|
||||||
h.close();
|
if (h != null) { h.close(); }
|
||||||
String skip = System.getProperty("solr.test.leavedatadir");
|
String skip = System.getProperty("solr.test.leavedatadir");
|
||||||
if (null != skip && 0 != skip.trim().length()) {
|
if (null != skip && 0 != skip.trim().length()) {
|
||||||
System.err.println("NOTE: per solr.test.leavedatadir, dataDir will not be removed: " + dataDir.getAbsolutePath());
|
System.err.println("NOTE: per solr.test.leavedatadir, dataDir will not be removed: " + dataDir.getAbsolutePath());
|
||||||
|
|
|
@ -21,11 +21,14 @@ import org.w3c.dom.Element;
|
||||||
import org.w3c.dom.NamedNodeMap;
|
import org.w3c.dom.NamedNodeMap;
|
||||||
import org.w3c.dom.Node;
|
import org.w3c.dom.Node;
|
||||||
import org.w3c.dom.NodeList;
|
import org.w3c.dom.NodeList;
|
||||||
|
import org.apache.solr.core.SolrException;
|
||||||
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Enumeration;
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author yonik
|
* @author yonik
|
||||||
|
@ -164,12 +167,12 @@ public class DOMUtil {
|
||||||
public static String getText(Node nd) {
|
public static String getText(Node nd) {
|
||||||
|
|
||||||
short type = nd.getNodeType();
|
short type = nd.getNodeType();
|
||||||
|
|
||||||
// for most node types, we can defer to the recursive helper method,
|
// for most node types, we can defer to the recursive helper method,
|
||||||
// but when asked for the text of these types, we must return null
|
// but when asked for the text of these types, we must return null
|
||||||
// (Not the empty string)
|
// (Not the empty string)
|
||||||
switch (type) {
|
switch (type) {
|
||||||
|
|
||||||
case Node.DOCUMENT_NODE: /* fall through */
|
case Node.DOCUMENT_NODE: /* fall through */
|
||||||
case Node.DOCUMENT_TYPE_NODE: /* fall through */
|
case Node.DOCUMENT_TYPE_NODE: /* fall through */
|
||||||
case Node.NOTATION_NODE: /* fall through */
|
case Node.NOTATION_NODE: /* fall through */
|
||||||
|
@ -183,15 +186,15 @@ public class DOMUtil {
|
||||||
|
|
||||||
/** @see #getText(Node) */
|
/** @see #getText(Node) */
|
||||||
private static void getText(Node nd, StringBuilder buf) {
|
private static void getText(Node nd, StringBuilder buf) {
|
||||||
|
|
||||||
short type = nd.getNodeType();
|
short type = nd.getNodeType();
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
|
|
||||||
case Node.ELEMENT_NODE: /* fall through */
|
case Node.ELEMENT_NODE: /* fall through */
|
||||||
case Node.ENTITY_NODE: /* fall through */
|
case Node.ENTITY_NODE: /* fall through */
|
||||||
case Node.ENTITY_REFERENCE_NODE: /* fall through */
|
case Node.ENTITY_REFERENCE_NODE: /* fall through */
|
||||||
case Node.DOCUMENT_FRAGMENT_NODE:
|
case Node.DOCUMENT_FRAGMENT_NODE:
|
||||||
NodeList childs = nd.getChildNodes();
|
NodeList childs = nd.getChildNodes();
|
||||||
for (int i = 0; i < childs.getLength(); i++) {
|
for (int i = 0; i < childs.getLength(); i++) {
|
||||||
Node child = childs.item(i);
|
Node child = childs.item(i);
|
||||||
|
@ -202,7 +205,7 @@ public class DOMUtil {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Node.ATTRIBUTE_NODE: /* fall through */
|
case Node.ATTRIBUTE_NODE: /* fall through */
|
||||||
/* Putting Attribute nodes in this section does not exactly
|
/* Putting Attribute nodes in this section does not exactly
|
||||||
match the definition of how textContent should behave
|
match the definition of how textContent should behave
|
||||||
|
@ -232,4 +235,131 @@ public class DOMUtil {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Replaces ${system.property[:default value]} references in all attributes
|
||||||
|
* and text nodes of supplied node. If the system property is not defined, an empty string
|
||||||
|
* is substituted or the default value if provided.
|
||||||
|
*
|
||||||
|
* @param node DOM node to walk for substitutions
|
||||||
|
*/
|
||||||
|
public static void substituteSystemProperties(Node node) {
|
||||||
|
// loop through child nodes
|
||||||
|
Node child;
|
||||||
|
Node next = node.getFirstChild();
|
||||||
|
while ((child = next) != null) {
|
||||||
|
|
||||||
|
// set next before we change anything
|
||||||
|
next = child.getNextSibling();
|
||||||
|
|
||||||
|
// handle child by node type
|
||||||
|
if (child.getNodeType() == Node.TEXT_NODE) {
|
||||||
|
child.setNodeValue(substituteSystemProperty(child.getNodeValue()));
|
||||||
|
} else if (child.getNodeType() == Node.ELEMENT_NODE) {
|
||||||
|
// handle child elements with recursive call
|
||||||
|
NamedNodeMap attributes = child.getAttributes();
|
||||||
|
for (int i = 0; i < attributes.getLength(); i++) {
|
||||||
|
Node attribute = attributes.item(i);
|
||||||
|
attribute.setNodeValue(substituteSystemProperty(attribute.getNodeValue()));
|
||||||
|
}
|
||||||
|
substituteSystemProperties(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This method borrowed from Ant's PropertyHelper.replaceProperties:
|
||||||
|
* http://svn.apache.org/repos/asf/ant/core/trunk/src/main/org/apache/tools/ant/PropertyHelper.java
|
||||||
|
*/
|
||||||
|
private static String substituteSystemProperty(String value) {
|
||||||
|
if (value == null || value.indexOf('$') == -1) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<String> fragments = new ArrayList<String>();
|
||||||
|
List<String> propertyRefs = new ArrayList<String>();
|
||||||
|
parsePropertyString(value, fragments, propertyRefs);
|
||||||
|
|
||||||
|
StringBuffer sb = new StringBuffer();
|
||||||
|
Iterator<String> i = fragments.iterator();
|
||||||
|
Iterator<String> j = propertyRefs.iterator();
|
||||||
|
|
||||||
|
while (i.hasNext()) {
|
||||||
|
String fragment = i.next();
|
||||||
|
if (fragment == null) {
|
||||||
|
String propertyName = j.next();
|
||||||
|
String defaultValue = null;
|
||||||
|
int colon_index = propertyName.indexOf(':');
|
||||||
|
if (colon_index > -1) {
|
||||||
|
defaultValue = propertyName.substring(colon_index + 1);
|
||||||
|
propertyName = propertyName.substring(0,colon_index);
|
||||||
|
}
|
||||||
|
fragment = System.getProperty(propertyName,defaultValue);
|
||||||
|
if (fragment == null) {
|
||||||
|
throw new SolrException(500, "No system property or default value specified for " + propertyName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sb.append(fragment);
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This method borrowed from Ant's PropertyHelper.parsePropertyStringDefault:
|
||||||
|
* http://svn.apache.org/repos/asf/ant/core/trunk/src/main/org/apache/tools/ant/PropertyHelper.java
|
||||||
|
*/
|
||||||
|
private static void parsePropertyString(String value, List<String> fragments, List<String> propertyRefs) {
|
||||||
|
int prev = 0;
|
||||||
|
int pos;
|
||||||
|
//search for the next instance of $ from the 'prev' position
|
||||||
|
while ((pos = value.indexOf("$", prev)) >= 0) {
|
||||||
|
|
||||||
|
//if there was any text before this, add it as a fragment
|
||||||
|
//TODO, this check could be modified to go if pos>prev;
|
||||||
|
//seems like this current version could stick empty strings
|
||||||
|
//into the list
|
||||||
|
if (pos > 0) {
|
||||||
|
fragments.add(value.substring(prev, pos));
|
||||||
|
}
|
||||||
|
//if we are at the end of the string, we tack on a $
|
||||||
|
//then move past it
|
||||||
|
if (pos == (value.length() - 1)) {
|
||||||
|
fragments.add("$");
|
||||||
|
prev = pos + 1;
|
||||||
|
} else if (value.charAt(pos + 1) != '{') {
|
||||||
|
//peek ahead to see if the next char is a property or not
|
||||||
|
//not a property: insert the char as a literal
|
||||||
|
/*
|
||||||
|
fragments.addElement(value.substring(pos + 1, pos + 2));
|
||||||
|
prev = pos + 2;
|
||||||
|
*/
|
||||||
|
if (value.charAt(pos + 1) == '$') {
|
||||||
|
//backwards compatibility two $ map to one mode
|
||||||
|
fragments.add("$");
|
||||||
|
prev = pos + 2;
|
||||||
|
} else {
|
||||||
|
//new behaviour: $X maps to $X for all values of X!='$'
|
||||||
|
fragments.add(value.substring(pos, pos + 2));
|
||||||
|
prev = pos + 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
//property found, extract its name or bail on a typo
|
||||||
|
int endName = value.indexOf('}', pos);
|
||||||
|
if (endName < 0) {
|
||||||
|
throw new RuntimeException("Syntax error in property: " + value);
|
||||||
|
}
|
||||||
|
String propertyName = value.substring(pos + 2, endName);
|
||||||
|
fragments.add(null);
|
||||||
|
propertyRefs.add(propertyName);
|
||||||
|
prev = endName + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//no more $ signs found
|
||||||
|
//if there is any tail to the string, append it
|
||||||
|
if (prev < value.length()) {
|
||||||
|
fragments.add(value.substring(prev));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
/**
|
||||||
|
* 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.apache.solr.core;
|
||||||
|
|
||||||
|
import org.apache.solr.util.AbstractSolrTestCase;
|
||||||
|
import org.apache.solr.util.TestHarness;
|
||||||
|
import org.w3c.dom.Node;
|
||||||
|
import org.w3c.dom.NodeList;
|
||||||
|
|
||||||
|
import javax.xml.xpath.XPathConstants;
|
||||||
|
import java.io.File;
|
||||||
|
|
||||||
|
public class TestBadConfig extends AbstractSolrTestCase {
|
||||||
|
|
||||||
|
public String getSchemaFile() { return "schema.xml"; }
|
||||||
|
public String getSolrConfigFile() { return "bad_solrconfig.xml"; }
|
||||||
|
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
|
||||||
|
dataDir = new File(System.getProperty("java.io.tmpdir")
|
||||||
|
+ System.getProperty("file.separator")
|
||||||
|
+ getClass().getName() + "-" + getName() + "-"
|
||||||
|
+ System.currentTimeMillis());
|
||||||
|
dataDir.mkdirs();
|
||||||
|
try {
|
||||||
|
h = new TestHarness(dataDir.getAbsolutePath(),
|
||||||
|
getSolrConfigFile(),
|
||||||
|
getSchemaFile());
|
||||||
|
fail("Exception should have been thrown");
|
||||||
|
} catch (Exception e) {
|
||||||
|
assertTrue(e.getMessage().contains("unset.sys.property"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void testNothing() {
|
||||||
|
// Empty test case as the real test is that the initialization of the TestHarness fails
|
||||||
|
assertTrue(true);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,53 @@
|
||||||
|
/**
|
||||||
|
* 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.apache.solr.core;
|
||||||
|
|
||||||
|
import org.apache.solr.util.AbstractSolrTestCase;
|
||||||
|
import org.w3c.dom.Node;
|
||||||
|
import org.w3c.dom.NodeList;
|
||||||
|
|
||||||
|
import javax.xml.xpath.XPathConstants;
|
||||||
|
|
||||||
|
public class TestConfig extends AbstractSolrTestCase {
|
||||||
|
|
||||||
|
public String getSchemaFile() { return "schema.xml"; }
|
||||||
|
public String getSolrConfigFile() { return "solrconfig.xml"; }
|
||||||
|
|
||||||
|
public void testJavaProperty() {
|
||||||
|
// property values defined in build.xml
|
||||||
|
|
||||||
|
String s = SolrConfig.config.get("propTest");
|
||||||
|
assertEquals("prefix-proptwo-suffix", s);
|
||||||
|
|
||||||
|
s = SolrConfig.config.get("propTest/@attr1", "default");
|
||||||
|
assertEquals("propone-${literal}", s);
|
||||||
|
|
||||||
|
s = SolrConfig.config.get("propTest/@attr2", "default");
|
||||||
|
assertEquals("default-from-config", s);
|
||||||
|
|
||||||
|
s = SolrConfig.config.get("propTest[@attr2='default-from-config']", "default");
|
||||||
|
assertEquals("prefix-proptwo-suffix", s);
|
||||||
|
|
||||||
|
NodeList nl = (NodeList) SolrConfig.config.evaluate("propTest", XPathConstants.NODESET);
|
||||||
|
assertEquals(1, nl.getLength());
|
||||||
|
assertEquals("prefix-proptwo-suffix", nl.item(0).getTextContent());
|
||||||
|
|
||||||
|
Node node = SolrConfig.config.getNode("propTest", true);
|
||||||
|
assertEquals("prefix-proptwo-suffix", node.getTextContent());
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,29 @@
|
||||||
|
<?xml version="1.0" ?>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- $Id: solrconfig.xml 382610 2006-03-03 01:43:03Z yonik $
|
||||||
|
$Source$
|
||||||
|
$Name$
|
||||||
|
-->
|
||||||
|
|
||||||
|
<config>
|
||||||
|
<indexDefaults>
|
||||||
|
<useCompoundFile>${unset.sys.property}</useCompoundFile>
|
||||||
|
</indexDefaults>
|
||||||
|
</config>
|
|
@ -261,6 +261,9 @@
|
||||||
<gettableFiles>solrconfig.xml scheam.xml admin-extra.html</gettableFiles>
|
<gettableFiles>solrconfig.xml scheam.xml admin-extra.html</gettableFiles>
|
||||||
</admin>
|
</admin>
|
||||||
|
|
||||||
|
<!-- test getting system property -->
|
||||||
|
<propTest attr1="${solr.test.sys.prop1}-$${literal}"
|
||||||
|
attr2="${non.existent.sys.prop:default-from-config}">prefix-${solr.test.sys.prop2}-suffix</propTest>
|
||||||
|
|
||||||
|
|
||||||
</config>
|
</config>
|
||||||
|
|
Loading…
Reference in New Issue