Skip to content

Commit

Permalink
Migrating to apache commons configuration 1.10 for parsing ETL jobs c…
Browse files Browse the repository at this point in the history
…onfigurations.
  • Loading branch information
ShridharSattur committed Nov 10, 2017
1 parent 62b60de commit 81df270
Show file tree
Hide file tree
Showing 4 changed files with 62 additions and 68 deletions.
1 change: 1 addition & 0 deletions gradle/scripts/dependency.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ ext.externalDependency = [
"jsoup" : "org.jsoup:jsoup:1.8.3",
"commons_io" : "commons-io:commons-io:2.5",
"commons_lang3" : "org.apache.commons:commons-lang3:3.6",
"commons_config" : "commons-configuration:commons-configuration:1.10",

"jackson_databind" : "com.fasterxml.jackson.core:jackson-databind:2.8.9",
"jackson_core" : "com.fasterxml.jackson.core:jackson-core:2.8.9",
Expand Down
1 change: 1 addition & 0 deletions wherehows-common/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ dependencies {
compile externalDependency.jackson_databind
compile externalDependency.guava
compile externalDependency.lombok
compile externalDependency.commons_config

testCompile externalDependency.testng
}
Expand Down
86 changes: 37 additions & 49 deletions wherehows-common/src/main/java/wherehows/common/utils/JobsUtil.java
Original file line number Diff line number Diff line change
Expand Up @@ -19,60 +19,22 @@
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import org.apache.commons.configuration.Configuration;
import org.apache.commons.configuration.ConfigurationException;
import org.apache.commons.configuration.PropertiesConfiguration;
import wherehows.common.Constant;


public class JobsUtil {

// Patterns for environmental variables resolution in jobs file.
private static final List<Pattern> ENV_VAR_PATTERNS = ImmutableList.<Pattern>builder()
.add(Pattern.compile("\\$(.+)")) // $ENV_VAR
.add(Pattern.compile("\\$\\{(.+)\\}")) // ${ENV_VAR}
.build();

/**
* Reads {@link Properties} from the given file and resolves all environmental variables denoted by ${ENV_VAR_NAME}.
*
* @param jobFile Path to the property file.
* @return Resolved {@link Properties} or null if failed to read from file.
*/
public static Properties getResolvedProperties(Path jobFile) {
Properties prop = new Properties();
try (BufferedReader reader = Files.newBufferedReader(jobFile)) {
prop.load(reader);
} catch (IOException ex) {
return null;
}

for (Map.Entry<Object, Object> entry : prop.entrySet()) {
String value = (String) entry.getValue();
prop.setProperty((String) entry.getKey(), resolveEnviornmentalVariable(value));
}
return prop;
}

/**
* Resolves the value to the corresponding environmental variable if possible, otherwise returns the original value.
*/
private static String resolveEnviornmentalVariable(String value) {
for (Pattern pattern : ENV_VAR_PATTERNS) {
Matcher matcher = pattern.matcher(value);
if (matcher.matches()) {
String resolved = System.getenv(matcher.group(1));
if (resolved != null) {
return resolved;
}
}
}
return value;
}

/**
* Get job name without file extension
* @param file File
Expand Down Expand Up @@ -103,11 +65,11 @@ public static String jobNameFromFileName(String filename) {
/**
* Returns a map of job name to job properties for all scheduled and enabled jobs.
*/
public static Map<String, Properties> getScheduledJobs(String dir) {
public static Map<String, Properties> getScheduledJobs(String dir) throws ConfigurationException {
Map<String, Properties> jobs = new HashMap<>();
for (File file : new File(dir).listFiles()) {
if (file.getAbsolutePath().endsWith(".job")) {
Properties prop = getResolvedProperties(file.toPath());
Properties prop = getJobConfigProperties(file);
if (!prop.containsKey(Constant.JOB_DISABLED_KEY) && prop.containsKey(Constant.JOB_CRON_EXPR_KEY)) {
// job name = file name without the extension.
jobs.put(jobNameFromFile(file), prop);
Expand All @@ -120,11 +82,12 @@ public static Map<String, Properties> getScheduledJobs(String dir) {
/**
* Returns a map of job name to job properties which are enabled.
*/
public static Map<String, Properties> getEnabledJobs(String dir) {
public static Map<String, Properties> getEnabledJobs(String dir) throws ConfigurationException {
Map<String, Properties> jobs = new HashMap<>();
for (File file : new File(dir).listFiles()) {
if (file.getAbsolutePath().endsWith(".job")) {
Properties prop = getResolvedProperties(file.toPath());
Configuration jobParam = new PropertiesConfiguration(file.getAbsolutePath());
Properties prop = getJobConfigProperties(file);
if (!prop.containsKey(Constant.JOB_DISABLED_KEY)) {
// job name = file name without the extension.
jobs.put(jobNameFromFile(file), prop);
Expand All @@ -137,17 +100,42 @@ public static Map<String, Properties> getEnabledJobs(String dir) {
/**
* Returns a map of job name to job properties which are enabled and of certain type.
*/
public static Map<String, Properties> getEnabledJobsByType(String dir, String type) {
public static Map<String, Properties> getEnabledJobsByType(String dir, String type) throws ConfigurationException {
Map<String, Properties> jobs = new HashMap<>();
for (File file : new File(dir).listFiles()) {
if (file.getAbsolutePath().endsWith(".job")) {
Properties prop = getResolvedProperties(file.toPath());
Properties prop = getJobConfigProperties(file);
if (!prop.containsKey(Constant.JOB_DISABLED_KEY) && prop.getProperty(Constant.JOB_TYPE_KEY, "").equals(type)) {
// job name = file name without the extension.
jobs.put(jobNameFromFile(file), prop);
}
}
}

return jobs;
}

public static Properties getJobConfigProperties(File jobConfigFile) throws ConfigurationException {
Configuration jobParam = new PropertiesConfiguration(jobConfigFile.getAbsolutePath());
Properties prop = new Properties();

if (jobParam != null) {
Iterator<String> keyit = jobParam.getKeys();
while (keyit.hasNext()) {
String key = keyit.next();
if (key != null) {
Object value = jobParam.getProperty(key);

if (value != null && value instanceof String[]) {
prop.setProperty(key, String.join(",", (String[]) value));
} else if (value != null && value instanceof List<?>) {
prop.setProperty(key, String.join(",", (List<String>) value));
} else if (value != null) {
prop.setProperty(key, value.toString());
}
}
}
}
return prop;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
import java.nio.file.Path;
import java.util.Map;
import java.util.Properties;
import org.apache.commons.configuration.ConfigurationException;
import org.testng.Assert;
import org.testng.annotations.Test;
import wherehows.common.Constant;
Expand All @@ -30,23 +31,7 @@
public class JobsUtilTest {

@Test
public void testEnvVarResolution() throws IOException {
String propertyStr = "var1=foo\n" + "var2=$JAVA_HOME\n" + "var3=${JAVA_HOME}";

Path path = createPropertiesFile(propertyStr);

Properties props = getResolvedProperties(path);

Assert.assertNotEquals(props, null);
Assert.assertEquals(props.getProperty("var1", ""), "foo");
Assert.assertTrue(props.getProperty("var2").length() > 0);
Assert.assertEquals(props.getProperty("var2"), props.getProperty("var3"));

Files.deleteIfExists(path);
}

@Test
public void testGetScheduledJobs() throws IOException {
public void testGetScheduledJobs() throws IOException, ConfigurationException {
String propertyStr1 = "job.class=test\n" + "job.cron.expr=0 0 1 * * ? *\n" + "#job.disabled=1\n" + "job.type=TEST1";
String propertyStr2 = "job.class=test\n" + "job.cron.expr=0 0 1 * * ? *\n" + "job.disabled=1\n" + "job.type=TEST2";
String propertyStr3 = "job.class=test\n" + "#job.disabled=1\n" + "job.type=TEST3";
Expand All @@ -71,7 +56,7 @@ public void testGetScheduledJobs() throws IOException {
}

@Test
public void testGetEnabledJobs() throws IOException {
public void testGetEnabledJobs() throws IOException, ConfigurationException {
String propertyStr1 = "job.class=test\n" + "job.cron.expr=0 0 1 * * ? *\n" + "#job.disabled=1\n" + "job.type=TEST1";
String propertyStr2 = "job.class=test\n" + "#job.disabled=1\n";

Expand All @@ -95,7 +80,7 @@ public void testGetEnabledJobs() throws IOException {
}

@Test
public void testGetEnabledJobsByType() throws IOException {
public void testGetEnabledJobsByType() throws IOException, ConfigurationException {
String propertyStr1 = "job.class=test\n" + "job.cron.expr=0 0 1 * * ? *\n" + "#job.disabled=1\n" + "job.type=TEST1";
String propertyStr2 = "job.class=test\n" + "job.cron.expr=0 0 1 * * ? *\n" + "#job.disabled=1\n" + "job.type=TEST2";

Expand All @@ -116,6 +101,25 @@ public void testGetEnabledJobsByType() throws IOException {
Files.deleteIfExists(path2);
}

@Test
public void testMultipleLinesSameProperties() throws IOException, ConfigurationException {
String dir = "/tmp/";
String propertyStr =
"var1=foo\n" + "var1=foo\n" + "var1=foo\n" + "var2=foo\n" + "var2=foo\n" + "var3=foo\n" + "var1=foo\n"
+ "var4=foo\n";
Properties props = new Properties();
Path path = createPropertiesFile(propertyStr);

props = getJobConfigProperties(path.toFile());

Assert.assertNotEquals(props, null);
Assert.assertEquals(props.getProperty("var1", ""), "foo,foo,foo,foo");
Assert.assertTrue(props.getProperty("var2").length() > 0);
Assert.assertEquals(props.getProperty("var3"), props.getProperty("var4"));

Files.deleteIfExists(path);
}

private Path createPropertiesFile(String content) throws IOException {
File propertyFile = File.createTempFile("temp", ".job");
FileWriter writer = new FileWriter(propertyFile);
Expand Down

0 comments on commit 81df270

Please sign in to comment.