From 1d2a5b5f06f98e00bddee4f8ecb4a4eba4dc11b2 Mon Sep 17 00:00:00 2001 From: Lukasz Antoniak Date: Sun, 13 Jan 2013 11:54:25 -0800 Subject: [PATCH] INTEXT-36 - Add Voldemort Module --- spring-integration-voldemort/README.md | 56 ++++ spring-integration-voldemort/build.gradle | 266 +++++++++++++++ .../gradle.properties | 1 + .../gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 39770 bytes .../gradle/wrapper/gradle-wrapper.properties | 6 + spring-integration-voldemort/gradlew | 164 +++++++++ spring-integration-voldemort/gradlew.bat | 90 +++++ .../publish-maven.gradle | 61 ++++ .../src/api/overview.html | 22 ++ .../src/dist/changelog.txt | 10 + .../src/dist/license.txt | 201 +++++++++++ .../src/dist/notice.txt | 21 ++ .../src/dist/readme.txt | 13 + .../VoldemortInboundChannelAdapterParser.java | 53 +++ .../config/xml/VoldemortNamespaceHandler.java | 31 ++ ...VoldemortOutboundChannelAdapterParser.java | 65 ++++ .../config/xml/VoldemortParserUtils.java | 47 +++ .../voldemort/config/xml/package-info.java | 4 + .../inbound/VoldemortMessageSource.java | 101 ++++++ .../voldemort/inbound/package-info.java | 4 + .../VoldemortStoringMessageHandler.java | 112 +++++++ .../voldemort/outbound/package-info.java | 4 + .../integration/voldemort/package-info.java | 4 + .../store/VoldemortMessageStore.java | 234 +++++++++++++ .../voldemort/support/PersistMode.java | 26 ++ .../voldemort/support/VoldemortHeaders.java | 36 ++ .../voldemort/support/package-info.java | 4 + .../main/resources/META-INF/spring.handlers | 1 + .../main/resources/META-INF/spring.schemas | 2 + .../main/resources/META-INF/spring.tooling | 4 + .../xml/spring-integration-voldemort-1.0.xsd | 187 +++++++++++ .../xml/spring-integration-voldemort.gif | Bin 0 -> 572 bytes .../src/reference/docbook/history.xml | 28 ++ .../src/reference/docbook/images/logo.png | Bin 0 -> 9627 bytes .../src/reference/docbook/index.xml | 63 ++++ .../src/reference/docbook/resources.xml | 35 ++ .../src/reference/docbook/voldemort.xml | 214 ++++++++++++ .../src/reference/docbook/whats-new.xml | 35 ++ .../test/BaseFunctionalTestCase.java | 101 ++++++ .../voldemort/test/domain/Car.java | 123 +++++++ .../voldemort/test/domain/Person.java | 89 +++++ .../voldemort/test/inbound/ObjectKeyTest.java | 66 ++++ .../inbound/VoldemortInboundAdapterTest.java | 101 ++++++ .../MessageUpdatingServiceActivator.java | 38 +++ .../test/outbound/ObjectKeyTest.java | 58 ++++ .../VoldemortOutboundAdapterTest.java | 177 ++++++++++ .../store/BaseStoreFunctionalTestCase.java | 44 +++ .../store/VoldemortMessageGroupStoreTest.java | 316 ++++++++++++++++++ .../VoldemortMessageStoreAggregationTest.java | 70 ++++ .../test/store/VoldemortMessageStoreTest.java | 187 +++++++++++ .../src/test/resources/cluster.xml | 12 + .../src/test/resources/log4j.properties | 8 + .../voldemort/test/common-test-context.xml | 16 + .../test/inbound/ObjectKeyTest-context.xml | 25 ++ .../VoldemortInboundAdapterTest-context.xml | 30 ++ .../test/outbound/ObjectKeyTest-context.xml | 19 ++ .../VoldemortOutboundAdapterTest-context.xml | 42 +++ ...ortMessageStoreAggregationTest-context.xml | 14 + .../test/store/store-test-context.xml | 25 ++ .../voldemort/test/store/stores.xml | 19 ++ .../src/test/resources/stores.xml | 34 ++ 61 files changed, 3819 insertions(+) create mode 100755 spring-integration-voldemort/README.md create mode 100755 spring-integration-voldemort/build.gradle create mode 100755 spring-integration-voldemort/gradle.properties create mode 100755 spring-integration-voldemort/gradle/wrapper/gradle-wrapper.jar create mode 100755 spring-integration-voldemort/gradle/wrapper/gradle-wrapper.properties create mode 100755 spring-integration-voldemort/gradlew create mode 100755 spring-integration-voldemort/gradlew.bat create mode 100755 spring-integration-voldemort/publish-maven.gradle create mode 100755 spring-integration-voldemort/src/api/overview.html create mode 100755 spring-integration-voldemort/src/dist/changelog.txt create mode 100755 spring-integration-voldemort/src/dist/license.txt create mode 100755 spring-integration-voldemort/src/dist/notice.txt create mode 100755 spring-integration-voldemort/src/dist/readme.txt create mode 100644 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortInboundChannelAdapterParser.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortNamespaceHandler.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortOutboundChannelAdapterParser.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortParserUtils.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/package-info.java create mode 100644 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/VoldemortMessageSource.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/package-info.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/VoldemortStoringMessageHandler.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/package-info.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/package-info.java create mode 100644 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/store/VoldemortMessageStore.java create mode 100644 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/PersistMode.java create mode 100644 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/VoldemortHeaders.java create mode 100755 spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/package-info.java create mode 100755 spring-integration-voldemort/src/main/resources/META-INF/spring.handlers create mode 100755 spring-integration-voldemort/src/main/resources/META-INF/spring.schemas create mode 100755 spring-integration-voldemort/src/main/resources/META-INF/spring.tooling create mode 100755 spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd create mode 100755 spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort.gif create mode 100755 spring-integration-voldemort/src/reference/docbook/history.xml create mode 100755 spring-integration-voldemort/src/reference/docbook/images/logo.png create mode 100755 spring-integration-voldemort/src/reference/docbook/index.xml create mode 100755 spring-integration-voldemort/src/reference/docbook/resources.xml create mode 100755 spring-integration-voldemort/src/reference/docbook/voldemort.xml create mode 100755 spring-integration-voldemort/src/reference/docbook/whats-new.xml create mode 100755 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/BaseFunctionalTestCase.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Car.java create mode 100755 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Person.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/MessageUpdatingServiceActivator.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest.java create mode 100755 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/BaseStoreFunctionalTestCase.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageGroupStoreTest.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest.java create mode 100644 spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreTest.java create mode 100755 spring-integration-voldemort/src/test/resources/cluster.xml create mode 100755 spring-integration-voldemort/src/test/resources/log4j.properties create mode 100755 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/common-test-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest-context.xml create mode 100755 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/store-test-context.xml create mode 100644 spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml create mode 100755 spring-integration-voldemort/src/test/resources/stores.xml diff --git a/spring-integration-voldemort/README.md b/spring-integration-voldemort/README.md new file mode 100755 index 0000000..965caee --- /dev/null +++ b/spring-integration-voldemort/README.md @@ -0,0 +1,56 @@ +Spring Integration Voldemort Adapter +==================================== + +The Voldemort extension for Spring Integration (SI) project includes inbound +and outbound channel adapters. + +Inbound channel adapter: +----------------------------------------------------------------------------- +Inbound channel adapter is used to retrieve data out of Voldemort database +and transfer objects into Spring Integration's channel. Component expects +user to provide Voldemort store client, message converter and desired object's +key. + +### Example: +~~~~~xml + + ... + + + + ... + +~~~~~ + +For more implementation details please review documentation and integration +test cases. + +Outbound channel adapter: +----------------------------------------------------------------------------- +Outbound channel adapter is used to insert data into Voldemort database +from Spring Integration's channel. Component expects user to provide +Voldemort store client and message converter. + +### Example: +~~~~~xml + + ... + + ... + +~~~~~ + +For more implementation details please review documentation and integration +test cases. + +Build +----------------------------------------------------------------------------- +For build instructions visit [Spring Integration on GitHub](https://github.com/SpringSource/spring-integration). \ No newline at end of file diff --git a/spring-integration-voldemort/build.gradle b/spring-integration-voldemort/build.gradle new file mode 100755 index 0000000..61618f8 --- /dev/null +++ b/spring-integration-voldemort/build.gradle @@ -0,0 +1,266 @@ +description = 'Spring Integration Voldemort Adapter' + +buildscript { + repositories { + maven { url 'https://repo.springsource.org/plugins-snapshot' } + } + dependencies { + classpath 'org.springframework.build.gradle:docbook-reference-plugin:0.1.5' + } +} + +apply plugin: 'java' +apply from: "${rootProject.projectDir}/publish-maven.gradle" +apply plugin: 'eclipse' +apply plugin: 'idea' + +group = 'org.springframework.integration.voldemort' + +repositories { + maven { url 'http://repo.springsource.org/libs-milestone' } + maven { url 'http://repo.springsource.org/plugins-release' } +} + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +ext { + junitVersion = '4.10' + log4jVersion = '1.2.12' + mockitoVersion = '1.9.0' + springVersion = '3.1.3.RELEASE' + springIntegrationVersion = '2.2.0.RELEASE' + voldemortVersion = '0.96' + + shortName = 'voldemort' +} + +eclipse { + project { + natures += 'org.springframework.ide.eclipse.core.springnature' + } +} + +sourceSets { + test { + resources { + srcDirs = ['src/test/resources', 'src/test/java'] + } + } +} + +// See http://www.gradle.org/docs/current/userguide/dependency_management.html#sub:configurations +// and http://www.gradle.org/docs/current/dsl/org.gradle.api.artifacts.ConfigurationContainer.html +configurations { + jacoco //Configuration Group used by Sonar to provide Code Coverage using JaCoCo +} + +dependencies { + compile "org.springframework.integration:spring-integration-core:$springIntegrationVersion" + compile "voldemort:voldemort:$voldemortVersion" + testCompile "org.springframework.integration:spring-integration-test:$springIntegrationVersion" + testCompile "junit:junit-dep:$junitVersion" + testCompile "log4j:log4j:$log4jVersion" + testCompile "org.mockito:mockito-all:$mockitoVersion" + testCompile "org.springframework:spring-test:$springVersion" + jacoco group: "org.jacoco", name: "org.jacoco.agent", version: "0.5.6.201201232323", classifier: "runtime" +} + + +// enable all compiler warnings; individual projects may customize further +ext.xLintArg = '-Xlint:all' +[compileJava, compileTestJava]*.options*.compilerArgs = [xLintArg] + +test { + // suppress all console output during testing unless running `gradle -i` + logging.captureStandardOutput(LogLevel.INFO) + // JVM settings from http://www.project-voldemort.com/voldemort/configuration.html + jvmArgs "-XX:+UseConcMarkSweepGC", "-XX:+UseParNewGC", "-javaagent:${configurations.jacoco.asPath}=destfile=${buildDir}/jacoco.exec,includes=*" +} + +task sourcesJar(type: Jar) { + classifier = 'sources' + from sourceSets.main.allJava +} + +task javadocJar(type: Jar) { + classifier = 'javadoc' + from javadoc +} + +artifacts { + archives sourcesJar + archives javadocJar +} + +apply plugin: 'docbook-reference' + +reference { + sourceDir = file('src/reference/docbook') +} + +apply plugin: 'sonar' + +sonar { + + if (rootProject.hasProperty('sonarHostUrl')) { + server.url = rootProject.sonarHostUrl + } + + database { + if (rootProject.hasProperty('sonarJdbcUrl')) { + url = rootProject.sonarJdbcUrl + } + if (rootProject.hasProperty('sonarJdbcDriver')) { + driverClassName = rootProject.sonarJdbcDriver + } + if (rootProject.hasProperty('sonarJdbcUsername')) { + username = rootProject.sonarJdbcUsername + } + if (rootProject.hasProperty('sonarJdbcPassword')) { + password = rootProject.sonarJdbcPassword + } + } + + project { + dynamicAnalysis = "reuseReports" + withProjectProperties { props -> + props["sonar.core.codeCoveragePlugin"] = "jacoco" + props["sonar.jacoco.reportPath"] = "${buildDir.name}/jacoco.exec" + } + } + + logger.info("Sonar parameters used: server.url='${server.url}'; database.url='${database.url}'; database.driverClassName='${database.driverClassName}'; database.username='${database.username}'") +} + +task api(type: Javadoc) { + group = 'Documentation' + description = 'Generates the Javadoc API documentation.' + title = "${rootProject.description} ${version} API" + options.memberLevel = org.gradle.external.javadoc.JavadocMemberLevel.PROTECTED + options.author = true + options.header = rootProject.description + options.overview = 'src/api/overview.html' + + source = sourceSets.main.allJava + classpath = project.sourceSets.main.compileClasspath + destinationDir = new File(buildDir, "api") +} + +task schemaZip(type: Zip) { + group = 'Distribution' + classifier = 'schema' + description = "Builds -${classifier} archive containing all " + + "XSDs for deployment at static.springframework.org/schema." + + def Properties schemas = new Properties(); + + project.sourceSets.main.resources.find { + it.path.endsWith('META-INF/spring.schemas') + }?.withInputStream { schemas.load(it) } + + for (def key : schemas.keySet()) { + File xsdFile = project.sourceSets.main.resources.find { + it.path.endsWith(schemas.get(key)) + } + assert xsdFile != null + into ("integration/${shortName}") { + from xsdFile.path + } + } + +} + +task docsZip(type: Zip) { + group = 'Distribution' + classifier = 'docs' + description = "Builds -${classifier} archive containing api and reference " + + "for deployment at static.springframework.org/spring-integration/docs." + + from('src/dist') { + include 'changelog.txt' + } + + from (api) { + into 'api' + } + + from (reference) { + into 'reference' + } +} + +task distZip(type: Zip, dependsOn: [docsZip, schemaZip]) { + group = 'Distribution' + classifier = 'dist' + description = "Builds -${classifier} archive, containing all jars and docs, " + + "suitable for community download page." + + ext.baseDir = "${project.name}-${project.version}"; + + from('src/dist') { + include 'readme.txt' + include 'license.txt' + include 'notice.txt' + into "${baseDir}" + } + + from(zipTree(docsZip.archivePath)) { + into "${baseDir}/docs" + } + + from(zipTree(schemaZip.archivePath)) { + into "${baseDir}/schema" + } + + into ("${baseDir}/libs") { + from project.jar + from project.sourcesJar + from project.javadocJar + } +} + +// Create an optional "with dependencies" distribution. +// Not published by default; only for use when building from source. +task depsZip(type: Zip, dependsOn: distZip) { zipTask -> + group = 'Distribution' + classifier = 'dist-with-deps' + description = "Builds -${classifier} archive, containing everything " + + "in the -${distZip.classifier} archive plus all dependencies." + + from zipTree(distZip.archivePath) + + gradle.taskGraph.whenReady { taskGraph -> + if (taskGraph.hasTask(":${zipTask.name}")) { + def projectName = rootProject.name + def artifacts = new HashSet() + + rootProject.configurations.runtime.resolvedConfiguration.resolvedArtifacts.each { artifact -> + def dependency = artifact.moduleVersion.id + if (!projectName.equals(dependency.name)) { + artifacts << artifact.file + } + } + + zipTask.from(artifacts) { + into "${distZip.baseDir}/deps" + } + } + } +} + +artifacts { + archives distZip + archives docsZip + archives schemaZip +} + +task dist(dependsOn: assemble) { + group = 'Distribution' + description = 'Builds -dist, -docs and -schema distribution archives.' +} + +task wrapper(type: Wrapper) { + description = 'Generates gradlew[.bat] scripts' + gradleVersion = '1.3' +} diff --git a/spring-integration-voldemort/gradle.properties b/spring-integration-voldemort/gradle.properties new file mode 100755 index 0000000..bebfcbc --- /dev/null +++ b/spring-integration-voldemort/gradle.properties @@ -0,0 +1 @@ +version=1.0.0.BUILD-SNAPSHOT diff --git a/spring-integration-voldemort/gradle/wrapper/gradle-wrapper.jar b/spring-integration-voldemort/gradle/wrapper/gradle-wrapper.jar new file mode 100755 index 0000000000000000000000000000000000000000..81dcde627d5cfbc12c96dc5113aac79b66a15118 GIT binary patch literal 39770 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 \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" +APP_HOME="`pwd -P`" +cd "$SAVED" + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query businessSystem maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin, switch paths to Windows format before running java +if $cygwin ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=$((i+1)) + done + case $i in + (0) set -- ;; + (1) set -- "$args0" ;; + (2) set -- "$args0" "$args1" ;; + (3) set -- "$args0" "$args1" "$args2" ;; + (4) set -- "$args0" "$args1" "$args2" "$args3" ;; + (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules +function splitJvmOpts() { + JVM_OPTS=("$@") +} +eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS +JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" + +exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" diff --git a/spring-integration-voldemort/gradlew.bat b/spring-integration-voldemort/gradlew.bat new file mode 100755 index 0000000..8a0b282 --- /dev/null +++ b/spring-integration-voldemort/gradlew.bat @@ -0,0 +1,90 @@ +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS= + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto init + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto init + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:init +@rem Get command-line arguments, handling Windowz variants + +if not "%OS%" == "Windows_NT" goto win9xME_args +if "%@eval[2+2]" == "4" goto 4NT_args + +:win9xME_args +@rem Slurp the command line arguments. +set CMD_LINE_ARGS= +set _SKIP=2 + +:win9xME_args_slurp +if "x%~1" == "x" goto execute + +set CMD_LINE_ARGS=%* +goto execute + +:4NT_args +@rem Get arguments from the 4NT Shell from JP Software +set CMD_LINE_ARGS=%$ + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/spring-integration-voldemort/publish-maven.gradle b/spring-integration-voldemort/publish-maven.gradle new file mode 100755 index 0000000..0ba656a --- /dev/null +++ b/spring-integration-voldemort/publish-maven.gradle @@ -0,0 +1,61 @@ +apply plugin: 'maven' + +ext.optionalDeps = [] +ext.providedDeps = [] + +ext.optional = { optionalDeps << it } +ext.provided = { providedDeps << it } + +install { + repositories.mavenInstaller { + customizePom(pom, project) + } +} + +def customizePom(pom, gradleProject) { + pom.whenConfigured { generatedPom -> + // respect 'optional' and 'provided' dependencies + gradleProject.optionalDeps.each { dep -> + generatedPom.dependencies.find { it.artifactId == dep.name }?.optional = true + } + gradleProject.providedDeps.each { dep -> + generatedPom.dependencies.find { it.artifactId == dep.name }?.scope = 'provided' + } + + // eliminate test-scoped dependencies (no need in maven central poms) + generatedPom.dependencies.removeAll { dep -> + dep.scope == 'test' + } + + // add all items necessary for maven central publication + generatedPom.project { + name = gradleProject.description + description = gradleProject.description + url = 'https://github.com/SpringSource/spring-integration-extensions' + organization { + name = 'SpringSource' + url = 'http://springsource.org' + } + licenses { + license { + name 'The Apache Software License, Version 2.0' + url 'http://www.apache.org/licenses/LICENSE-2.0.txt' + distribution 'repo' + } + } + scm { + url = 'https://github.com/SpringSource/spring-integration-extensions' + connection = 'scm:git:git://github.com/SpringSource/spring-integration-extensions' + developerConnection = 'scm:git:git://github.com/SpringSource/spring-integration-extensions' + } + + developers { + developer { + id = 'lukasz-antoniak' + name = 'Lukasz Antoniak' + email = 'Lukasz.Antoniak@gmail.com' + } + } + } + } +} diff --git a/spring-integration-voldemort/src/api/overview.html b/spring-integration-voldemort/src/api/overview.html new file mode 100755 index 0000000..fb0198b --- /dev/null +++ b/spring-integration-voldemort/src/api/overview.html @@ -0,0 +1,22 @@ + + +This document is the API specification for Spring Integration +
+
+

+ For further API reference and developer documentation, see the + Spring + Integration reference documentation. + That documentation contains more detailed, developer-targeted + descriptions, with conceptual overviews, definitions of terms, + workarounds, and working code examples. +

+ +

+ If you are interested in commercial training, consultancy, and + support for Spring Integration, please visit + http://www.springsource.com +

+
+ + diff --git a/spring-integration-voldemort/src/dist/changelog.txt b/spring-integration-voldemort/src/dist/changelog.txt new file mode 100755 index 0000000..5a45c4e --- /dev/null +++ b/spring-integration-voldemort/src/dist/changelog.txt @@ -0,0 +1,10 @@ +Spring Integration Voldemort Adapter Changelog +============================================== + +Changes in version 1.0.0 (13 Jan 2013) +------------------------------------- + +** New Feature + * Voldemort Inbound Adapter + * Voldemort Outbound Adapter + * Voldemort Message Store \ No newline at end of file diff --git a/spring-integration-voldemort/src/dist/license.txt b/spring-integration-voldemort/src/dist/license.txt new file mode 100755 index 0000000..261eeb9 --- /dev/null +++ b/spring-integration-voldemort/src/dist/license.txt @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + 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 + + 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. diff --git a/spring-integration-voldemort/src/dist/notice.txt b/spring-integration-voldemort/src/dist/notice.txt new file mode 100755 index 0000000..f62045a --- /dev/null +++ b/spring-integration-voldemort/src/dist/notice.txt @@ -0,0 +1,21 @@ + ======================================================================== + == NOTICE file corresponding to section 4 d of the Apache License, == + == Version 2.0, in this case for the Spring Integration distribution. == + ======================================================================== + + This product includes software developed by + the Apache Software Foundation (http://www.apache.org). + + The end-user documentation included with a redistribution, if any, + must include the following acknowledgement: + + "This product includes software developed by the Spring Framework + Project (http://www.springframework.org)." + + Alternatively, this acknowledgement may appear in the software itself, + if and wherever such third-party acknowledgements normally appear. + + The names "Spring", "Spring Framework", and "Spring Integration" must + not be used to endorse or promote products derived from this software + without prior written permission. For written permission, please contact + enquiries@springsource.com. diff --git a/spring-integration-voldemort/src/dist/readme.txt b/spring-integration-voldemort/src/dist/readme.txt new file mode 100755 index 0000000..edccc0c --- /dev/null +++ b/spring-integration-voldemort/src/dist/readme.txt @@ -0,0 +1,13 @@ +Spring Integration Voldemort Adapter +------------------------------------ + +To find out what has changed since any earlier releases, see 'changelog.txt'. + +Please consult the documentation located within the 'docs/reference' directory +of this release and also visit the official Spring Integration home at +http://www.springsource.org/spring-integration + +There you will find links to the forum, issue tracker, and several other resources. + +See https://github.com/SpringSource/spring-integration#readme for additional +information including instructions on building from source. diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortInboundChannelAdapterParser.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortInboundChannelAdapterParser.java new file mode 100644 index 0000000..aa4fc98 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortInboundChannelAdapterParser.java @@ -0,0 +1,53 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.config.xml; + +import org.springframework.beans.BeanMetadataElement; +import org.springframework.beans.factory.support.BeanDefinitionBuilder; +import org.springframework.beans.factory.support.RootBeanDefinition; +import org.springframework.beans.factory.xml.ParserContext; +import org.springframework.integration.config.xml.AbstractPollingInboundChannelAdapterParser; +import org.springframework.integration.config.xml.IntegrationNamespaceUtils; +import org.springframework.integration.voldemort.inbound.VoldemortMessageSource; +import org.w3c.dom.Element; + +/** + * Parses Voldemort inbound adapter XML definition. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class VoldemortInboundChannelAdapterParser extends AbstractPollingInboundChannelAdapterParser { + /** + * Produces "int-voldemort:inbound-channel-adapter" bean definition. + *

+ * {@inheritDoc} + */ + @Override + protected BeanMetadataElement parseSource(Element element, ParserContext parserContext) { + final BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition( VoldemortMessageSource.class ); + VoldemortParserUtils.processCommonAttributes( element, builder ); + final RootBeanDefinition queryExpressionDef = + IntegrationNamespaceUtils.createExpressionDefinitionFromValueOrExpression( + VoldemortParserUtils.SEARCH_KEY, VoldemortParserUtils.SEARCH_KEY_EXPRESSION, + parserContext, element, true + ); + builder.addPropertyValue( VoldemortParserUtils.KEY_EXPRESSION_PROPERTY, queryExpressionDef ); + IntegrationNamespaceUtils.setValueIfAttributeDefined( builder, element, VoldemortParserUtils.DELETE_AFTER_POLL ); + IntegrationNamespaceUtils.setValueIfAttributeDefined( builder, element, VoldemortParserUtils.EXTRACT_VALUE ); + return builder.getBeanDefinition(); + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortNamespaceHandler.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortNamespaceHandler.java new file mode 100755 index 0000000..85ab0fe --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortNamespaceHandler.java @@ -0,0 +1,31 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.config.xml; + +import org.springframework.integration.config.xml.AbstractIntegrationNamespaceHandler; + +/** + * The handler for Voldemort extension namespace. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class VoldemortNamespaceHandler extends AbstractIntegrationNamespaceHandler { + public void init() { + registerBeanDefinitionParser( "inbound-channel-adapter", new VoldemortInboundChannelAdapterParser() ); + registerBeanDefinitionParser( "outbound-channel-adapter", new VoldemortOutboundChannelAdapterParser() ); + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortOutboundChannelAdapterParser.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortOutboundChannelAdapterParser.java new file mode 100755 index 0000000..3ee51aa --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortOutboundChannelAdapterParser.java @@ -0,0 +1,65 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.config.xml; + +import org.springframework.beans.factory.config.TypedStringValue; +import org.springframework.beans.factory.support.AbstractBeanDefinition; +import org.springframework.beans.factory.support.BeanDefinitionBuilder; +import org.springframework.beans.factory.support.RootBeanDefinition; +import org.springframework.beans.factory.xml.ParserContext; +import org.springframework.integration.config.ExpressionFactoryBean; +import org.springframework.integration.config.xml.AbstractOutboundChannelAdapterParser; +import org.springframework.integration.config.xml.IntegrationNamespaceUtils; +import org.springframework.integration.voldemort.outbound.VoldemortStoringMessageHandler; +import org.w3c.dom.Element; + +/** + * Parses Voldemort outbound adapter XML definition. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class VoldemortOutboundChannelAdapterParser extends AbstractOutboundChannelAdapterParser { + /** + * Produces "int-voldemort:outbound-channel-adapter" bean definition. + *

+ * {@inheritDoc} + */ + @Override + protected AbstractBeanDefinition parseConsumer(Element element, ParserContext parserContext) { + final BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition( VoldemortStoringMessageHandler.class ); + VoldemortParserUtils.processCommonAttributes( element, builder ); + final boolean hasKey = element.hasAttribute( VoldemortParserUtils.STORE_KEY ); + final boolean hasKeyExpression = element.hasAttribute( VoldemortParserUtils.STORE_KEY_EXPRESSION ); + if ( hasKey && hasKeyExpression ) { + parserContext.getReaderContext().error( "At most one of '" + VoldemortParserUtils.STORE_KEY + + "' or '" + VoldemortParserUtils.STORE_KEY_EXPRESSION + "' is allowed.", element ); + } + if ( hasKey ) { + builder.addPropertyValue( + VoldemortParserUtils.KEY_PROPERTY, + new TypedStringValue( element.getAttribute( VoldemortParserUtils.STORE_KEY ) ) + ); + } + if ( hasKeyExpression ) { + RootBeanDefinition expressionDef = new RootBeanDefinition( ExpressionFactoryBean.class ); + expressionDef.getConstructorArgumentValues().addGenericArgumentValue( element.getAttribute( VoldemortParserUtils.STORE_KEY_EXPRESSION ) ); + builder.addPropertyValue( VoldemortParserUtils.KEY_EXPRESSION_PROPERTY, expressionDef ); + } + IntegrationNamespaceUtils.setValueIfAttributeDefined( builder, element, VoldemortParserUtils.PERSIST_MODE ); + return builder.getBeanDefinition(); + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortParserUtils.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortParserUtils.java new file mode 100755 index 0000000..1762115 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/VoldemortParserUtils.java @@ -0,0 +1,47 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.config.xml; + +import org.springframework.beans.factory.support.BeanDefinitionBuilder; +import org.w3c.dom.Element; + +/** + * Contains various utility methods for parsing Voldemort Adapter specific namesspace elements. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +abstract class VoldemortParserUtils { + static final String STORE_CLIENT = "store-client"; + static final String PERSIST_MODE = "persist-mode"; + static final String SEARCH_KEY = "search-key"; + static final String SEARCH_KEY_EXPRESSION = "search-key-expression"; + static final String STORE_KEY = "store-key"; + static final String STORE_KEY_EXPRESSION = "store-key-expression"; + static final String DELETE_AFTER_POLL = "delete-after-poll"; + static final String EXTRACT_VALUE = "extract-value"; + + static final String KEY_PROPERTY = "key"; + static final String KEY_EXPRESSION_PROPERTY = "keyExpression"; + + /** + * Handles "store-client" and "message-converter" attributes. + */ + static void processCommonAttributes(Element element, BeanDefinitionBuilder builder) { + final String storeClient = element.getAttribute( VoldemortParserUtils.STORE_CLIENT ); + builder.addConstructorArgReference( storeClient ); + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/package-info.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/package-info.java new file mode 100755 index 0000000..8c9202d --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/config/xml/package-info.java @@ -0,0 +1,4 @@ +/** + * Provides parser classes to manage XML namespace for the Voldemort components. + */ +package org.springframework.integration.voldemort.config.xml; diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/VoldemortMessageSource.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/VoldemortMessageSource.java new file mode 100644 index 0000000..d6524e3 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/VoldemortMessageSource.java @@ -0,0 +1,101 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.inbound; + +import org.springframework.expression.Expression; +import org.springframework.expression.spel.support.StandardEvaluationContext; +import org.springframework.integration.Message; +import org.springframework.integration.context.IntegrationObjectSupport; +import org.springframework.integration.core.MessageSource; +import org.springframework.integration.expression.ExpressionUtils; +import org.springframework.integration.support.MessageBuilder; +import org.springframework.integration.voldemort.support.VoldemortHeaders; +import voldemort.client.StoreClient; +import voldemort.versioning.Versioned; + +/** + * Voldemort polling inbound adapter implementation. Regularly tries to retrieve object with a given key. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class VoldemortMessageSource extends IntegrationObjectSupport implements MessageSource { + private final StoreClient client; + + /** + * Key expression which will be evaluated on every call to the {@link #receive()} method. + */ + private volatile Expression keyExpression; + private volatile StandardEvaluationContext evaluationContext; + private volatile boolean extractValue = true; + private volatile boolean deleteAfterPoll = false; + + /** + * Creates new message source. + * + * @param client Voldemort store client. + */ + public VoldemortMessageSource(StoreClient client) { + this.client = client; + } + + @Override + protected void onInit() throws Exception { + super.onInit(); + if ( getBeanFactory() != null ) { + evaluationContext = ExpressionUtils.createStandardEvaluationContext( getBeanFactory() ); + } + else { + evaluationContext = ExpressionUtils.createStandardEvaluationContext(); + } + } + + @Override + @SuppressWarnings("unchecked") + public Message receive() { + final Object key = keyExpression.getValue( evaluationContext, Object.class ); + final Versioned value = client.get( key ); + if ( value != null ) { + if ( deleteAfterPoll ) { + client.delete( key ); + } + return MessageBuilder.withPayload( extractValue ? value.getValue() : value ) + .setHeader( VoldemortHeaders.KEY, key ).build(); + } + return null; + } + + @Override + public String getComponentType() { + return "voldemort:inbound-channel-adapter"; + } + + public void setDeleteAfterPoll(boolean deleteAfterPoll) { + this.deleteAfterPoll = deleteAfterPoll; + } + + public Expression getKeyExpression() { + return keyExpression; + } + + public void setKeyExpression(Expression keyExpression) { + this.keyExpression = keyExpression; + } + + public void setExtractValue(boolean extractValue) { + this.extractValue = extractValue; + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/package-info.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/package-info.java new file mode 100755 index 0000000..5c18a3c --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/inbound/package-info.java @@ -0,0 +1,4 @@ +/** + * Provides inbound Spring Integration Voldemort components. + */ +package org.springframework.integration.voldemort.inbound; diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/VoldemortStoringMessageHandler.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/VoldemortStoringMessageHandler.java new file mode 100755 index 0000000..8704746 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/VoldemortStoringMessageHandler.java @@ -0,0 +1,112 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.outbound; + +import org.springframework.expression.Expression; +import org.springframework.expression.common.LiteralExpression; +import org.springframework.expression.spel.standard.SpelExpressionParser; +import org.springframework.expression.spel.support.StandardEvaluationContext; +import org.springframework.integration.Message; +import org.springframework.integration.expression.ExpressionUtils; +import org.springframework.integration.handler.AbstractMessageHandler; +import org.springframework.integration.voldemort.support.PersistMode; +import org.springframework.integration.voldemort.support.VoldemortHeaders; +import voldemort.client.StoreClient; + +/** + * Voldemort outbound adapter implementation. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class VoldemortStoringMessageHandler extends AbstractMessageHandler { + private final StoreClient client; + + private volatile StandardEvaluationContext evaluationContext; + private volatile Expression keyExpression = new SpelExpressionParser().parseExpression( "headers." + VoldemortHeaders.KEY ); + + private volatile PersistMode persistMode = PersistMode.PUT; + + /** + * Creates new message sender. + * + * @param client Voldemort store client. + */ + public VoldemortStoringMessageHandler(StoreClient client) { + this.client = client; + } + + @Override + protected void onInit() throws Exception { + super.onInit(); + if ( getBeanFactory() != null ) { + evaluationContext = ExpressionUtils.createStandardEvaluationContext( getBeanFactory() ); + } + else { + evaluationContext = ExpressionUtils.createStandardEvaluationContext(); + } + } + + @Override + @SuppressWarnings("unchecked") + protected void handleMessageInternal(Message message) throws Exception { + final Object key = keyExpression.getValue( evaluationContext, message, Object.class ); + switch ( determinePersistMode( message ) ) { + case PUT: + client.put( key, message.getPayload() ); + break; + case DELETE: + client.delete( key ); + break; + } + } + + /** + * Computes desired persist mode for a given message. Default output adapter's configuration + * can be overridden with {@link VoldemortHeaders#PERSIST_MODE} message header which supports + * direct or text representation of {@link PersistMode} enumeration. + * + * @param message Spring Integration message. + * @return Persist mode. + */ + private PersistMode determinePersistMode(Message message) { + final Object confValue = message.getHeaders().get( VoldemortHeaders.PERSIST_MODE ); + if ( confValue instanceof PersistMode ) { + return (PersistMode) confValue; + } + else if ( confValue instanceof String ) { + return PersistMode.valueOf( (String) confValue ); + } + return persistMode; + } + + @Override + public String getComponentType() { + return "voldemort:outbound-channel-adapter"; + } + + public void setKey(String key) { + setKeyExpression( new LiteralExpression( key ) ); + } + + public void setKeyExpression(Expression keyExpression) { + this.keyExpression = keyExpression; + } + + public void setPersistMode(PersistMode persistMode) { + this.persistMode = persistMode; + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/package-info.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/package-info.java new file mode 100755 index 0000000..a77b939 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/outbound/package-info.java @@ -0,0 +1,4 @@ +/** + * Provides Spring Integration Voldemort components for doing outbound operations. + */ +package org.springframework.integration.voldemort.outbound; diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/package-info.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/package-info.java new file mode 100755 index 0000000..c4eebeb --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/package-info.java @@ -0,0 +1,4 @@ +/** + * Root package of the Voldemort Module. + */ +package org.springframework.integration.voldemort; diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/store/VoldemortMessageStore.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/store/VoldemortMessageStore.java new file mode 100644 index 0000000..6caf424 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/store/VoldemortMessageStore.java @@ -0,0 +1,234 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.store; + +import java.util.Collection; +import java.util.Collections; +import java.util.HashSet; +import java.util.Set; +import java.util.concurrent.locks.Lock; + +import org.springframework.beans.factory.InitializingBean; +import org.springframework.integration.store.AbstractKeyValueMessageStore; +import org.springframework.integration.store.MessageGroupStore; +import org.springframework.integration.store.MessageStore; +import org.springframework.integration.util.DefaultLockRegistry; +import org.springframework.integration.util.LockRegistry; +import org.springframework.integration.util.UUIDConverter; +import voldemort.client.StoreClient; +import voldemort.client.UpdateAction; +import voldemort.serialization.SerializationException; +import voldemort.versioning.Versioned; + +/** + * Voldemort implementation of the key-value style {@link MessageStore} and {@link MessageGroupStore}. + * Implementation note: message identifiers are persisted as {@link String}s. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortMessageStore extends AbstractKeyValueMessageStore implements InitializingBean { + /** + * Key under which message store tracks all currently saved message identifiers. + */ + protected final static String MESSAGE_KEY_LIST = "MESSAGE_KEY_LIST"; + + /** + * Key under which message store tracks all currently saved message group identifiers. + */ + protected final static String MESSAGE_GROUP_KEY_LIST = "MESSAGE_GROUP_KEY_LIST"; + + private final static LockRegistry LOCK_REGISTRY = new DefaultLockRegistry(); + + private final StoreClient client; + + public VoldemortMessageStore(StoreClient client) { + this.client = client; + } + + @Override + public void afterPropertiesSet() throws Exception { + // Initialize empty set of message and message group identifiers if needed. + synchronized ( getClass() ) { + if ( client.get( MESSAGE_KEY_LIST ) == null ) { + client.put( MESSAGE_KEY_LIST, new HashSet() ); + } + if ( client.get( MESSAGE_GROUP_KEY_LIST ) == null ) { + client.put( MESSAGE_GROUP_KEY_LIST, new HashSet() ); + } + } + } + + @Override + protected Object doRetrieve(Object id) { + final Versioned version = client.get( id ); + if ( version != null ) { + return version.getValue(); + } + return null; + } + + @Override + protected void doStore(final Object id, final Object objectToStore) { + final Lock messageLock = doLock( id ); + try { + client.put( id, objectToStore ); + // Keeping track of stored message identifiers for efficient implementation of #doListKeys(String) method. + // Voldemort does not provide API to list all keys matching specified pattern. + final String keyListKey = computeKeyListKey( id ); + final Lock keyListLock = doLock( keyListKey ); + try { + client.applyUpdate( new AddKeyUpdateAction( id, keyListKey ) ); + } + finally { + keyListLock.unlock(); + } + } + catch ( SerializationException e ) { + throw new IllegalArgumentException( "Voldemort failed to serialize message with id: " + id + "." ); + } + finally { + messageLock.unlock(); + } + } + + @Override + protected Object doRemove(final Object id) { + final Object message = doRetrieve( id ); + final Lock messageLock = doLock( id ); + try { + client.delete( id ); + // Keeping track of stored message identifiers for efficient implementation of #doListKeys(String) method. + // Voldemort does not provide API to list all keys matching specified pattern. + final String keyListKey = computeKeyListKey( id ); + final Lock keyListLock = doLock( keyListKey ); + try { + client.applyUpdate( new RemoveKeyUpdateAction( id, keyListKey ) ); + } + finally { + keyListLock.unlock(); + } + return message; + } + finally { + messageLock.unlock(); + } + } + + @Override + protected Collection doListKeys(String keyPattern) { + return Collections.unmodifiableSet( (Set) client.get( computeKeyListKey( keyPattern ) ).getValue() ); + } + + /** + * @param id Message or message group identifier. {@link String} type required. + * @return Identifier under which set of all currently persisted message or message group keys is saved. + */ + private String computeKeyListKey(Object id) { + final String key = (String) id; + if ( isMessageGroupKey( key ) ) { + return MESSAGE_GROUP_KEY_LIST; + } + else if ( isMessageKey( key ) ) { + return MESSAGE_KEY_LIST; + } + else { + throw new IllegalArgumentException("Unsupported identifier: " + key + "."); + } + } + + /** + * @param key Message or message group identifier. + * @return {@code true} in case of message identifier, {@code false} otherwise. + */ + private boolean isMessageKey(String key) { + return key.startsWith( AbstractKeyValueMessageStore.MESSAGE_KEY_PREFIX ); + } + + /** + * @param key Message or message group identifier. + * @return {@code true} in case of message group identifier, {@code false} otherwise. + */ + private boolean isMessageGroupKey(String key) { + return key.startsWith( AbstractKeyValueMessageStore.MESSAGE_GROUP_KEY_PREFIX ); + } + + /** + * Acquire JVM wide lock on the given object. + * @param obj Object. + * @return Lock. + */ + private Lock doLock(Object obj) { + final Lock lock = LOCK_REGISTRY.obtain( UUIDConverter.getUUID( obj ).toString() ); + lock.lock(); + return lock; + } + + /** + * Voldemort update action that adds given key to the list of currently saved identifiers. + */ + private static class AddKeyUpdateAction extends UpdateAction { + private final Object id; + private final String keyListKey; + + /** + * The only constructor. + * + * @param id Message or message group identifier. + * @param keyListKey Key under which Voldemort stores set of all currently persisted identifiers. + */ + private AddKeyUpdateAction(Object id, String keyListKey) { + this.id = id; + this.keyListKey = keyListKey; + } + + @Override + public void update(StoreClient storeClient) { + final Set keys = (Set) storeClient.get( keyListKey ).getValue(); + if ( keys.add( id ) ) { + storeClient.put( keyListKey, keys ); + } + } + } + + /** + * Voldemort update action that removes given key from the list of currently saved identifiers. + */ + private static class RemoveKeyUpdateAction extends UpdateAction { + private final Object id; + private final String keyListKey; + + /** + * The only constructor. + * + * @param id Message or message group identifier. + * @param keyListKey Key under which Voldemort stores set of all currently persisted identifiers. + */ + private RemoveKeyUpdateAction(Object id, String keyListKey) { + this.id = id; + this.keyListKey = keyListKey; + } + + @Override + public void update(StoreClient storeClient) { + final Set keys = (Set) storeClient.get( keyListKey ).getValue(); + if ( keys.remove( id ) ) { + storeClient.put( keyListKey, keys ); + } + } + } +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/PersistMode.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/PersistMode.java new file mode 100644 index 0000000..c4583c7 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/PersistMode.java @@ -0,0 +1,26 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.support; + +/** + * Indicates whether records shall be added or removed from Voldemort database. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public enum PersistMode { + PUT, DELETE +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/VoldemortHeaders.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/VoldemortHeaders.java new file mode 100644 index 0000000..ca92796 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/VoldemortHeaders.java @@ -0,0 +1,36 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.support; + +/** + * Voldemort adapter specific message headers. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public abstract class VoldemortHeaders { + private static final String PREFIX = "voldemort_"; + + /** + * Header representing key under which message payload should be stored. + */ + public static final String KEY = PREFIX + "_key"; + + /** + * Overrides default output adapter's persist mode for a given message. + */ + public static final String PERSIST_MODE = PREFIX + "persistMode"; +} diff --git a/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/package-info.java b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/package-info.java new file mode 100755 index 0000000..2cf0e46 --- /dev/null +++ b/spring-integration-voldemort/src/main/java/org/springframework/integration/voldemort/support/package-info.java @@ -0,0 +1,4 @@ +/** + * Provides various support classes used across Spring Integration Voldemort components. + */ +package org.springframework.integration.voldemort.support; diff --git a/spring-integration-voldemort/src/main/resources/META-INF/spring.handlers b/spring-integration-voldemort/src/main/resources/META-INF/spring.handlers new file mode 100755 index 0000000..74e9341 --- /dev/null +++ b/spring-integration-voldemort/src/main/resources/META-INF/spring.handlers @@ -0,0 +1 @@ +http\://www.springframework.org/schema/integration/voldemort=org.springframework.integration.voldemort.config.xml.VoldemortNamespaceHandler diff --git a/spring-integration-voldemort/src/main/resources/META-INF/spring.schemas b/spring-integration-voldemort/src/main/resources/META-INF/spring.schemas new file mode 100755 index 0000000..e2e301d --- /dev/null +++ b/spring-integration-voldemort/src/main/resources/META-INF/spring.schemas @@ -0,0 +1,2 @@ +http\://www.springframework.org/schema/integration/voldemort/spring-integration-voldemort-1.0.xsd=org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd +http\://www.springframework.org/schema/integration/voldemort/spring-integration-voldemort.xsd=org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd diff --git a/spring-integration-voldemort/src/main/resources/META-INF/spring.tooling b/spring-integration-voldemort/src/main/resources/META-INF/spring.tooling new file mode 100755 index 0000000..c93fd86 --- /dev/null +++ b/spring-integration-voldemort/src/main/resources/META-INF/spring.tooling @@ -0,0 +1,4 @@ +# Tooling related information for the integration Voldemort namespace +http\://www.springframework.org/schema/integration/voldemort@name=integration Voldemort Namespace +http\://www.springframework.org/schema/integration/voldemort@prefix=int-voldemort +http\://www.springframework.org/schema/integration/voldemort@icon=org/springframework/integration/config/xml/spring-integration-voldemort.gif diff --git a/spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd b/spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd new file mode 100755 index 0000000..9b053db --- /dev/null +++ b/spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort-1.0.xsd @@ -0,0 +1,187 @@ + + + + + + + + + + + + + + + Defines an Outbound Channel Adapter that writes messages to + Voldemort key-value store. + + + + + + + + Specifies the key for Voldemort store. If an expression + is needed, use 'store-key-expression' attribute. + Both options are mutually exclusive. + + + + + + + Spring Expression Language statement used to determine + desired entry key in Voldemort store. + Default expression equals 'headers.voldemort_key'. + This attribute is mutually exclusive with 'store-key' attribute. + + + + + + + Defines default persist mode for outbound-channel-adapter. + By default values are added or updated in the underlying + Voldemort database. Hereby behavior can be overridden using + VoldemortHeaders.PERSIST_MODE message header. + + + + + + + + + + Specifies the order for invocation when this endpoint is connected as a + subscriber to SubscribableChannel. + + + + + + + + + + Defines an Inbound Channel Adapter that reads records from Voldemort + key-value store. + + + + + + + + + + + String key that is being searched. + This attribute is mutually exclusive + with 'search-key-expression' attribute. + + + + + + + Spring Expression Language statement that returns + desired entry key. + This attribute is mutually exclusive with 'search-key' attribute. + + + + + + + Indicates whether returned object should be removed + from Voldemort database after polling into Spring + Integration channel. + + + + + + + + + + Flag controlling value extraction from Voldemort's + Versioned object. + + + + + + + + + + + + + Common configuration for Voldemort adapters. + + + + + + Identifies the underlying Spring bean definition. + + + + + + + + Reference to a MessageChannel that will be utilized to + "receive from" or "send to" messages depending on the + adapter type (e.g. inbound or outbound). + + + + + + + + + + + + Native Voldemort store client. + + + + + + + + + + + Flag to indicate that the component should start automatically + on startup (default true). + + + + + + + + + + + + + + + diff --git a/spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort.gif b/spring-integration-voldemort/src/main/resources/org/springframework/integration/config/xml/spring-integration-voldemort.gif new file mode 100755 index 0000000000000000000000000000000000000000..41b369fece0e576ecd920400b221fd33abbaf5c4 GIT binary patch literal 572 zcmZ?wbhEHb6krfwc*Xz%|NsBj^POrMHnn5wI!WKA$M#t>l?H4BPzO?b&{cRWSFFSB< z^|`x}P0QyVy|U`SiF40CoO%A?(2mZdyE;$q?mn}(`^J&Ji~DRU^KH0`_S=*f zu`(xWd1~172%niTfzv{QrUeE70}32q)Fc#tvM@3*q%i1!3doJ z$QZroAYKk;o)!}g_c$BZP}??kHkSD;%JZ2d6u9}BT8+&ujM`ePV%-d#6T`zi);F^_ zIR!9@O3I6^oV(z-tdN`(pX{#l&0dZgror~-%^VCYeSN$`0@iCL`334n#7Fs`w{uBO x(B)aWd{Qfm5=#n`xT2uIN}aG328D}QI)&6(d>(9QbkfS;6w=UeP!nLV1^^EL*)#wE literal 0 HcmV?d00001 diff --git a/spring-integration-voldemort/src/reference/docbook/history.xml b/spring-integration-voldemort/src/reference/docbook/history.xml new file mode 100755 index 0000000..4e9aaf7 --- /dev/null +++ b/spring-integration-voldemort/src/reference/docbook/history.xml @@ -0,0 +1,28 @@ + + + Change History + + Change History + + + + Release + Date + Changes + + + + + 1.0.0 + 1 March 2013 + + Initial release providing basic capabilities for message store, + inbound and outbound adapters. + + + + +
+
diff --git a/spring-integration-voldemort/src/reference/docbook/images/logo.png b/spring-integration-voldemort/src/reference/docbook/images/logo.png new file mode 100755 index 0000000000000000000000000000000000000000..4f2804f849424ee722de5062adc04a4022539c5d GIT binary patch literal 9627 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 NsOTtHDcVF3{|~1+q%Qyf literal 0 HcmV?d00001 diff --git a/spring-integration-voldemort/src/reference/docbook/index.xml b/spring-integration-voldemort/src/reference/docbook/index.xml new file mode 100755 index 0000000..238b413 --- /dev/null +++ b/spring-integration-voldemort/src/reference/docbook/index.xml @@ -0,0 +1,63 @@ + + + + Spring Integration Voldemort Adapter + Voldemort Adapter ${version} + Spring Integration + ${version} + + + + + + + + + + + + + + + Lukasz Antoniak + + + + © SpringSource Inc., 2013 + + + + + + + What's new? + + + If you are interested in the changes and features, that were introduced in + earlier versions, please take a look at chapter . + + + + + + + Integration Adapters + + + Spring Integration adapter for Voldemort enables users to persist data + transferred through message channel in the destination key-value store, + as well as to poll records from the Voldemort database and process them + inside Spring Integration flow. Hereby extension provides also the + implementation of a message store. + + + + + + Appendices + + + + diff --git a/spring-integration-voldemort/src/reference/docbook/resources.xml b/spring-integration-voldemort/src/reference/docbook/resources.xml new file mode 100755 index 0000000..9369e3a --- /dev/null +++ b/spring-integration-voldemort/src/reference/docbook/resources.xml @@ -0,0 +1,35 @@ + + + Additional Resources + +
+ Spring Integration Home + + The definitive source of information about Spring Integration is the + Spring Integration Home + at http://www.springsource.org. That site serves as a hub + of information and is the best place to find up-to-date announcements about the project as well as links to + articles, blogs, and new sample applications. + +
+ +
+ Project Voldemort Home + + If you are not familiar with Voldemort distributed key-value store, please visit + Project Voldemort homepage. Hereby site contains + complete design overview, list of available configuration options and documentation of client API. Links + to source code, bug tracker and wiki can be found as well. + +
+ +
+ Voldemort Adapter Integration Tests + + Integration tests of Voldemort adapter are available together with the entire source code on GitHub + (review content of src/test/java and src/test/resources packages). + +
+ +
diff --git a/spring-integration-voldemort/src/reference/docbook/voldemort.xml b/spring-integration-voldemort/src/reference/docbook/voldemort.xml new file mode 100755 index 0000000..167cf27 --- /dev/null +++ b/spring-integration-voldemort/src/reference/docbook/voldemort.xml @@ -0,0 +1,214 @@ + + + Voldemort Adapter + + The Voldemort extension for Spring Integration provides: + + + + + + Outbound Channel Adapter + + + + + + + Inbound Channel Adapter + + + + + + + Message Store + + + + + + + Each one of the components listed above requires to pass an implementation of + voldemort.client.StoreClient as constructor parameter. This way + Voldemort Adapter does not need to introduce all configuration options required to build native store client + and enables developers to choose the concrete implementation of hereby interface. + + + + + + + + + + + + +]]> + +
+ Common Configuration Attributes + + Certain configuration parameters that are shared amongst inbound and + outbound adapters have been described below. + + + + Common Adapter Attributes + + + + + + Attribute Name + Description + + + + + id + + Identifies the underlying Spring bean definition, which is an instance of either + EventDrivenConsumer or PollingConsumer + depending on whether the component's input channel is a + SubscribableChannel or PollableChannel. + Optional. + + + + channel + + Reference to a message channel that will be utilized to "receive from" or "send to" + messages depending on the adapter type (e.g. inbound or outbound). + Required. + + + + store-client + + Native Voldemort store client (implementation of + voldemort.client.StoreClient interface). + Required. + + + + auto-startup + + Lifecycle attribute signaling if this component should be started during application + context startup. Defaults to true. Optional. + + + + +
+
+ +
+ Outbound Channel Adapter + + The Voldemort Outbound channel adapter allows to persist data transferred through message channel in + the underlying key-value store. + + <int-voldemort:outbound-channel-adapter id="voldemortOutbound" + channel="voldemortOutboundChannel" + store-client="storeClient" + store-key="constant-key" + store-key-expression="payload.id" + persist-mode="DELETE" + order="1" + auto-startup="true" /> + + + + Specifies the key for Voldemort store. If an expression is needed, use + store-key-expression attribute. Both options are mutually exclusive. + Required. + + + + + Spring Expression Language statement used to determine desired entry key in Voldemort store. + Default expression equals headers.voldemort_key + (see VoldemortHeaders.KEY). This option is mutually exclusive with + store-key attribute. Required. + + + + + Defines default persist mode for outbound channel adapter. Normally values are added + or updated in the Voldemort database. Hereby behavior can be overridden on a message level using + VoldemortHeaders.PERSIST_MODE header (see PersistMode + enumeration). Supported options: PUT or DELETE. + Optional. + + + + + Specifies the order for invocation when this endpoint is connected as a subscriber to + SubscribableChannel. Optional. + + + +
+
+ Inbound Channel Adapter + + An inbound channel adapter allows to poll records from the Voldemort database and process them inside + Spring Integration flow. + + <int-voldemort:inbound-channel-adapter id="voldemortInbound" + channel="voldemortInboundChannel" + store-client="storeClient" + message-converter="messageConverter" + search-key="lukasz" + search-key-expression="@keyProducer.value" + delete-after-poll="true" + extract-value="true" + auto-startup="true" /> + + + + String key that is being searched. This option is mutually exclusive with + search-key-expression attribute. Required. + + + + + Spring Expression Language statement that returns desired entry key. Useful for keys of non string + type. This option is mutually exclusive with search-key attribute. + Required. + + + + + Indicates whether returned object should be removed from Voldemort database after polling into + Spring Integration channel. Defaults to false. Optional. + + + + + Flag controlling value extraction from Voldemort's + voldemort.versioning.Versioned object. + Defaults to true. Optional. + + + +
+
+ Message Store + + Message Store allows to persist Spring Integration messages in Voldemort database. Hereby module provides + VoldemortMessageStore which is an implementation of both the + MessageStore strategy (mainly used by QueueChannel and + ClaimCheck patterns) and the MessageGroupStore strategy + (mainly used by Aggregator and Resequencer patterns). + + + + + +]]> +
+
diff --git a/spring-integration-voldemort/src/reference/docbook/whats-new.xml b/spring-integration-voldemort/src/reference/docbook/whats-new.xml new file mode 100755 index 0000000..eab323f --- /dev/null +++ b/spring-integration-voldemort/src/reference/docbook/whats-new.xml @@ -0,0 +1,35 @@ + + + Changes in 1.0.0 + + This chapter provides an overview of new features and improvements + that have been added to the Voldemort Adapter. + + + + Inbound Channel Adapter allows to poll data from Voldemort key-value store. + Supported features: + + Periodically poll value persisted under given key. + Specify search key directly or as a Spring Expression Language statement. + Automatically remove data from Voldemort store after putting message into + Spring Integration channel. + Control adapter startup time. + + + + Outbound Channel Adapter allows to insert, update and delete data from + Voldemort key-value store. Supported features: + + Define operation type (e.g. put or delete) on adapter and message level. + Order execution when chaining with other adapters. + Control adapter startup time. + + + + Message store allows to persist Spring Integration messages in Voldemort database. + + + diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/BaseFunctionalTestCase.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/BaseFunctionalTestCase.java new file mode 100755 index 0000000..4e30167 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/BaseFunctionalTestCase.java @@ -0,0 +1,101 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test; + +import java.io.File; +import java.util.Properties; + +import org.apache.commons.io.FileUtils; +import org.junit.After; +import org.junit.Before; +import voldemort.server.VoldemortConfig; +import voldemort.server.VoldemortServer; + +/** + * Base class for functional test cases. Handles embedded Voldemort server startup and shutdown. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public abstract class BaseFunctionalTestCase { + protected VoldemortServer server = null; + + /** + * Starts Voldemort embedded server. + */ + @Before + public void setUp() throws Exception { + final VoldemortConfig config = configureServer(); + server = new VoldemortServer( config ); + server.start(); + } + + /** + * Stops Voldemort embedded server. + */ + @After + public void tearDown() throws Exception { + if ( server != null && server.isStarted() ) { + server.stop(); + } + } + + /** + * Review cluster.xml and stores.xml configuration files. + * + * @return Voldemort embedded server configuration. + */ + protected VoldemortConfig configureServer() throws Exception { + final File voldemortHome = new File( System.getProperty( "java.io.tmpdir" ), "voldemort" ); + FileUtils.deleteDirectory( voldemortHome ); + + final Properties properties = new Properties(); + properties.put( "node.id", "0" ); + properties.put( "voldemort.home", voldemortHome.getAbsolutePath() ); + addConfigOptions( properties ); + final VoldemortConfig config = new VoldemortConfig( properties ); + + final File metadata = new File( config.getMetadataDirectory() ); + FileUtils.forceMkdir( metadata ); + + FileUtils.copyFileToDirectory( getClusterConfiguration(), metadata ); + FileUtils.copyFileToDirectory( getStoreConfiguration(), metadata ); + + return config; + } + + /** + * @return Voldemort cluster configuration descriptor. + */ + protected File getClusterConfiguration() { + return new File( "src/test/resources/cluster.xml" ); + } + + /** + * @return Voldemort store configuration descriptor. + */ + protected File getStoreConfiguration() { + return new File( "src/test/resources/stores.xml" ); + } + + /** + * Subclasses may want to setup specific server configuration parameters. + * + * @param properties Voldemort server configuration properties. + */ + protected void addConfigOptions(Properties properties) { + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Car.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Car.java new file mode 100644 index 0000000..0b81119 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Car.java @@ -0,0 +1,123 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.domain; + +import java.io.Serializable; + +/** + * Sample object with composite key persisted in Voldemort database. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class Car implements Serializable { + private static final long serialVersionUID = -891019943116582242L; + + private CarId id; + private String model; + + public Car(CarId id, String model) { + this.id = id; + this.model = model; + } + + @Override + public boolean equals(Object o) { + if ( this == o ) return true; + if ( ! ( o instanceof Car ) ) return false; + + Car car = (Car) o; + + if ( id != null ? !id.equals( car.id ) : car.id != null ) return false; + if ( model != null ? !model.equals( car.model ) : car.model != null ) return false; + + return true; + } + + @Override + public int hashCode() { + int result = id != null ? id.hashCode() : 0; + result = 31 * result + ( model != null ? model.hashCode() : 0 ); + return result; + } + + @Override + public String toString() { + return "Car(id = " + id + ", model = " + model + ")"; + } + + public CarId getId() { + return id; + } + + public void setId(CarId id) { + this.id = id; + } + + public String getModel() { + return model; + } + + public void setModel(String model) { + this.model = model; + } + + /** + * Car identifier wrapper. + * + * @author Lukasz Antoniak + * @since 1.0 + */ + public static class CarId implements Serializable { + private static final long serialVersionUID = -5586075844887213095L; + + private Integer id; + + public CarId(Integer id) { + this.id = id; + } + + @Override + public boolean equals(Object o) { + if ( this == o ) return true; + if ( ! ( o instanceof CarId ) ) return false; + + CarId carId = (CarId) o; + + if ( id != null ? !id.equals( carId.id ) : carId.id != null ) return false; + + return true; + } + + @Override + public int hashCode() { + return id != null ? id.hashCode() : 0; + } + + @Override + public String toString() { + return "CarId(id = " + id + ")"; + } + + public Integer getId() { + return id; + } + + public void setId(Integer id) { + this.id = id; + } + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Person.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Person.java new file mode 100755 index 0000000..071279d --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/domain/Person.java @@ -0,0 +1,89 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.domain; + +import java.io.Serializable; + +/** + * Sample object persisted in Voldemort database. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class Person implements Serializable { + private static final long serialVersionUID = -9092199331950213292L; + + private String id; + private String firstName; + private String lastName; + + public Person(String id, String firstName, String lastName) { + this.id = id; + this.firstName = firstName; + this.lastName = lastName; + } + + @Override + public boolean equals(Object o) { + if ( this == o ) return true; + if ( !( o instanceof Person ) ) return false; + + Person person = (Person) o; + + if ( id != null ? !id.equals( person.id ) : person.id != null ) return false; + if ( firstName != null ? !firstName.equals( person.firstName ) : person.firstName != null ) return false; + if ( lastName != null ? !lastName.equals( person.lastName ) : person.lastName != null ) return false; + + return true; + } + + @Override + public int hashCode() { + int result = id != null ? id.hashCode() : 0; + result = 31 * result + ( firstName != null ? firstName.hashCode() : 0 ); + result = 31 * result + ( lastName != null ? lastName.hashCode() : 0 ); + return result; + } + + @Override + public String toString() { + return "Person(id = " + id + ", firstName = " + firstName + ", lastName = " + lastName + ")"; + } + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public String getFirstName() { + return firstName; + } + + public void setFirstName(String firstName) { + this.firstName = firstName; + } + + public String getLastName() { + return lastName; + } + + public void setLastName(String lastName) { + this.lastName = lastName; + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest.java new file mode 100644 index 0000000..ad6471e --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest.java @@ -0,0 +1,66 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.inbound; + +import junit.framework.Assert; +import org.junit.Test; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.Message; +import org.springframework.integration.core.PollableChannel; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; +import org.springframework.integration.voldemort.test.domain.Car; +import voldemort.client.StoreClient; + +/** + * Test key of object type. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class ObjectKeyTest extends BaseFunctionalTestCase { + @Test + public void testReceiveMessageByObjectKey() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "ObjectKeyTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "objectStoreClient", StoreClient.class ); + final PollableChannel inboundChannel = context.getBean( "voldemortInboundChannel", PollableChannel.class ); + + // given + final Car.CarId carId = new Car.CarId( 1 ); + final Car car = new Car( carId, "Ford Mustang" ); + storeClient.put( carId, car ); + + // when + final Message received = (Message) inboundChannel.receive(); + + // then + Assert.assertEquals( car, received.getPayload() ); + + context.close(); + } + + /** + * Produces key of object type. + * + * @author Lukasz Antoniak + * @since 1.0 + */ + public static class ObjectKeyProducer { + public Object getValue() { + return new Car.CarId( 1 ); + } + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest.java new file mode 100644 index 0000000..e52617e --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest.java @@ -0,0 +1,101 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.inbound; + +import junit.framework.Assert; +import org.junit.Test; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.Message; +import org.springframework.integration.core.PollableChannel; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; +import org.springframework.integration.voldemort.test.domain.Person; +import voldemort.client.StoreClient; +import voldemort.versioning.Versioned; + +/** + * Voldemort basic inbound adapter tests. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortInboundAdapterTest extends BaseFunctionalTestCase { + /** + * Tests inbound adapter configured with "search-key" attribute. + */ + @Test + public void testReceiveMessageKey() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortInboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final PollableChannel inboundChannel = context.getBean( "voldemortInboundChannel", PollableChannel.class ); + + // given + final Person lukasz = new Person( "lukasz", "Lukasz", "Antoniak" ); + storeClient.put( lukasz.getId(), lukasz ); + + // when + final Message received = (Message) inboundChannel.receive(); + + // then + Assert.assertEquals( lukasz, received.getPayload() ); + + context.close(); + } + + /** + * Tests inbound adapter configured with "search-key-expression" attribute. + */ + @Test + public void testReceiveMessageExpr() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortInboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final PollableChannel inboundChannel = context.getBean( "voldemortInboundChannel", PollableChannel.class ); + + // given + final Person kinga = new Person( "kinga", "Kinga", "Mroz" ); + storeClient.put( kinga.getId(), kinga ); + + // when + final Message> received = (Message>) inboundChannel.receive(); + + // then + final Versioned found = storeClient.get( kinga.getId() ); + Assert.assertEquals( found, received.getPayload() ); + + context.close(); + } + + @Test + public void testDeleteAfterPoll() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortInboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final PollableChannel inboundChannel = context.getBean( "voldemortInboundChannel", PollableChannel.class ); + + // given + final Person robert = new Person( "robert", "Robert", "Antoniak" ); + storeClient.put( robert.getId(), robert ); + + // when + final Message received = (Message) inboundChannel.receive(); + + // then + Assert.assertEquals( robert, received.getPayload() ); + final Versioned found = storeClient.get( robert.getId() ); + Assert.assertNull( found ); + + context.close(); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/MessageUpdatingServiceActivator.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/MessageUpdatingServiceActivator.java new file mode 100644 index 0000000..520940c --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/MessageUpdatingServiceActivator.java @@ -0,0 +1,38 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.outbound; + +import org.springframework.integration.Message; +import org.springframework.integration.annotation.ServiceActivator; +import org.springframework.integration.voldemort.test.domain.Person; + +/** + * Sample service activator used to verify order of output adapters execution. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public class MessageUpdatingServiceActivator { + @ServiceActivator + public Message updateMessage(Message message) { + updatePerson( message.getPayload() ); + return message; + } + + protected void updatePerson(Person person) { + person.setFirstName( "Robert" ); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest.java new file mode 100644 index 0000000..bdec184 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest.java @@ -0,0 +1,58 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.outbound; + +import junit.framework.Assert; +import org.junit.Test; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.Message; +import org.springframework.integration.MessageChannel; +import org.springframework.integration.support.MessageBuilder; +import org.springframework.integration.voldemort.support.VoldemortHeaders; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; +import org.springframework.integration.voldemort.test.domain.Car; +import voldemort.client.StoreClient; +import voldemort.versioning.Versioned; + +/** + * Test key of object type. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class ObjectKeyTest extends BaseFunctionalTestCase { + @Test + public void testCompositeKey() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "ObjectKeyTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "objectStoreClient", StoreClient.class ); + final MessageChannel voldemortOutboundPutChannel = context.getBean( "voldemortOutboundPutChannel", MessageChannel.class ); + + // given + final Car.CarId carId = new Car.CarId( 1 ); + final Car car = new Car( carId, "Ford Mustang" ); + + // when + final Message message = MessageBuilder.withPayload( car ).setHeader( VoldemortHeaders.KEY, carId ).build(); + voldemortOutboundPutChannel.send( message ); + + // then + final Versioned found = storeClient.get( carId ); + Assert.assertEquals( car, found.getValue() ); + + context.close(); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest.java new file mode 100755 index 0000000..588cba8 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest.java @@ -0,0 +1,177 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.outbound; + +import junit.framework.Assert; +import org.junit.Test; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.Message; +import org.springframework.integration.MessageChannel; +import org.springframework.integration.MessageDeliveryException; +import org.springframework.integration.support.MessageBuilder; +import org.springframework.integration.voldemort.support.PersistMode; +import org.springframework.integration.voldemort.support.VoldemortHeaders; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; +import org.springframework.integration.voldemort.test.domain.Person; +import voldemort.client.StoreClient; +import voldemort.versioning.Versioned; + +/** + * Voldemort basic outbound adapter tests. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortOutboundAdapterTest extends BaseFunctionalTestCase { + @Test + public void testPutObject() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final MessageChannel voldemortOutboundPutChannel = context.getBean( "voldemortOutboundPutChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "1", "Lukasz", "Antoniak" ); + + // when + final Message message = MessageBuilder.withPayload( lukasz ).setHeader( VoldemortHeaders.KEY, lukasz.getId() ).build(); + voldemortOutboundPutChannel.send( message ); + + // then + final Versioned found = storeClient.get( lukasz.getId() ); + Assert.assertEquals( lukasz, found.getValue() ); + + context.close(); + } + + @Test + public void testPutObjectsConstantKey() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final MessageChannel voldemortOutboundChannel = context.getBean( "voldemortOutboundPutConstantKeyChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "1", "Lukasz", "Antoniak" ); + + // when + final Message firstMessage = MessageBuilder.withPayload( lukasz ).build(); + voldemortOutboundChannel.send( firstMessage ); + + // then + Assert.assertEquals( lukasz, storeClient.get( "constant-key" ).getValue() ); + + // given + final Person tomasz = new Person( "2", "Tomasz", "Antoniak" ); + + // when + final Message secondMessage = MessageBuilder.withPayload( tomasz ).build(); + voldemortOutboundChannel.send( secondMessage ); + + // then + Assert.assertEquals( tomasz, storeClient.get( "constant-key" ).getValue() ); + + context.close(); + } + + @Test + public void testDeleteObject() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final MessageChannel voldemortOutboundDeleteChannel = context.getBean( "voldemortOutboundDeleteChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "1", "Lukasz", "Antoniak" ); + storeClient.put( lukasz.getId(), lukasz ); + + // when + final Message message = MessageBuilder.withPayload( lukasz ).build(); + voldemortOutboundDeleteChannel.send( message ); + + // then + final Versioned found = storeClient.get( lukasz.getId() ); + Assert.assertNull( found ); + + context.close(); + } + + @Test + public void testOverridePersistMode() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final MessageChannel voldemortOutboundDeleteChannel = context.getBean( "voldemortOutboundDeleteChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "1", "Lukasz", "Antoniak" ); + + // when + // Overriding output adapter's persist mode. + final Message message = MessageBuilder.withPayload( lukasz ) + .setHeader( VoldemortHeaders.PERSIST_MODE, PersistMode.PUT ).build(); + voldemortOutboundDeleteChannel.send( message ); + + // then + final Versioned found = storeClient.get( lukasz.getId() ); + Assert.assertEquals( lukasz, found.getValue() ); + + context.close(); + } + + @Test + public void testStoppedAdapter() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final MessageChannel voldemortStoppedChannel = context.getBean( "voldemortStoppedChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "1", "Lukasz", "Antoniak" ); + + // when + final Message message = MessageBuilder.withPayload( lukasz ).build(); + try { + voldemortStoppedChannel.send( message ); + } + catch ( MessageDeliveryException e ) { + return; + } + finally { + context.close(); + } + + Assert.fail(); + } + + @Test + public void testOrder() { + final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortOutboundAdapterTest-context.xml", getClass() ); + final StoreClient storeClient = context.getBean( "storeClient", StoreClient.class ); + final MessageUpdatingServiceActivator messageUpdater = context.getBean( "messageUpdater", MessageUpdatingServiceActivator.class ); + final MessageChannel voldemortOrderChannel = context.getBean( "voldemortOrderChannel", MessageChannel.class ); + + // given + final Person lukasz = new Person( "lukasz", "Lukasz", "Antoniak" ); + final Person copy = new Person( "lukasz", "Lukasz", "Antoniak" ); + + // when + final Message message = MessageBuilder.withPayload( lukasz ).build(); + voldemortOrderChannel.send( message ); + + // then + messageUpdater.updatePerson( copy ); + final Versioned found = storeClient.get( lukasz.getId() ); + Assert.assertEquals( copy, found.getValue() ); + + context.close(); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/BaseStoreFunctionalTestCase.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/BaseStoreFunctionalTestCase.java new file mode 100644 index 0000000..8fc8641 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/BaseStoreFunctionalTestCase.java @@ -0,0 +1,44 @@ +package org.springframework.integration.voldemort.test.store; + +import java.io.File; + +import org.junit.After; +import org.junit.Before; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.voldemort.store.VoldemortMessageStore; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; + +/** + * Base class for message store test cases. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +public abstract class BaseStoreFunctionalTestCase extends BaseFunctionalTestCase { + protected ClassPathXmlApplicationContext context = null; + protected VoldemortMessageStore store = null; + + @Override + protected File getStoreConfiguration() { + return new File( "src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml" ); + } + + @Before + public void contextSetup() { + context = new ClassPathXmlApplicationContext( "store-test-context.xml", getClass() ); + store = context.getBean( "voldemortMessageStore", VoldemortMessageStore.class ); + } + + @After + public void contextDestroy() { + store = null; + if ( context != null ) { + context.close(); + context = null; + } + } + + protected VoldemortMessageStore createNewStoreClient() { + return context.getBean( "voldemortMessageStorePrototype", VoldemortMessageStore.class ); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageGroupStoreTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageGroupStoreTest.java new file mode 100644 index 0000000..40e1f0f --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageGroupStoreTest.java @@ -0,0 +1,316 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.store; + +import java.util.Iterator; +import java.util.Properties; +import java.util.Random; +import java.util.UUID; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; + +import org.junit.Assert; +import org.junit.Test; +import org.springframework.integration.Message; +import org.springframework.integration.channel.DirectChannel; +import org.springframework.integration.history.MessageHistory; +import org.springframework.integration.message.GenericMessage; +import org.springframework.integration.store.MessageGroup; +import org.springframework.integration.store.SimpleMessageGroup; +import org.springframework.integration.support.MessageBuilder; +import org.springframework.integration.voldemort.store.VoldemortMessageStore; + +/** + * Voldemort message store tests based on Redis module. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortMessageGroupStoreTest extends BaseStoreFunctionalTestCase { + @Test + public void testNonExistingEmptyMessageGroup() { + final MessageGroup messageGroup = store.getMessageGroup( 1 ); + + Assert.assertNotNull( messageGroup ); + Assert.assertTrue( messageGroup instanceof SimpleMessageGroup ); + Assert.assertEquals( 0, messageGroup.size() ); + } + + @Test + public void testUpdatedDateChangesWithEachAddedMessage() throws InterruptedException { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( 1, message ); + Assert.assertEquals( 1, messageGroup.size() ); + + long createdTimestamp = messageGroup.getTimestamp(); + long updatedTimestamp = messageGroup.getLastModified(); + Assert.assertEquals( createdTimestamp, updatedTimestamp ); + Thread.sleep( 1000 ); + + message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( 1, message ); + createdTimestamp = messageGroup.getTimestamp(); + updatedTimestamp = messageGroup.getLastModified(); + Assert.assertTrue( updatedTimestamp > createdTimestamp ); + + // use another message store instance + VoldemortMessageStore newStore = createNewStoreClient(); + + messageGroup = newStore.getMessageGroup( 1 ); + Assert.assertEquals( 2, messageGroup.size() ); + } + + @Test + public void testMessageGroupAddOperation() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( 1, message ); + Assert.assertEquals( 1, messageGroup.size() ); + + // use another message store instance + VoldemortMessageStore newStore = createNewStoreClient(); + + messageGroup = newStore.getMessageGroup( 1 ); + Assert.assertEquals( 1, messageGroup.size() ); + } + + @Test + public void testMessageGroupWithAddedMessageUUIDGroupIdAndUUIDHeader() { + final Object id = UUID.randomUUID(); + MessageGroup messageGroup = store.getMessageGroup( id ); + final UUID uuidA = UUID.randomUUID(); + Message messageA = MessageBuilder.withPayload( "A" ).setHeader( "foo", uuidA ).build(); + final UUID uuidB = UUID.randomUUID(); + Message messageB = MessageBuilder.withPayload( "B" ).setHeader( "foo", uuidB ).build(); + store.addMessageToGroup( id, messageA ); + messageGroup = store.addMessageToGroup( id, messageB ); + Assert.assertEquals( 2, messageGroup.size() ); + Message retrievedMessage = store.getMessage( messageA.getHeaders().getId() ); + Assert.assertNotNull( retrievedMessage ); + Assert.assertEquals( retrievedMessage.getHeaders().getId(), messageA.getHeaders().getId() ); + Object fooHeader = retrievedMessage.getHeaders().get( "foo" ); + Assert.assertTrue( fooHeader instanceof UUID ); + Assert.assertEquals( uuidA, fooHeader ); + } + + @Test + public void testCountMessagesInGroup() { + Message messageA = new GenericMessage( "A" ); + Message messageB = new GenericMessage( "B" ); + store.addMessageToGroup( 1, messageA ); + store.addMessageToGroup( 1, messageB ); + Assert.assertEquals( 2, store.messageGroupSize( 1 ) ); + } + + @Test + public void testRemoveMessageGroup() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( messageGroup.getGroupId(), message ); + Assert.assertEquals( 1, messageGroup.size() ); + + store.removeMessageGroup( 1 ); + + MessageGroup messageGroupA = store.getMessageGroup( 1 ); + Assert.assertNotSame( messageGroup, messageGroupA ); + Assert.assertEquals( 0, messageGroupA.getMessages().size() ); + Assert.assertEquals( 0, messageGroupA.size() ); + + // use another message store instance + VoldemortMessageStore newStore = createNewStoreClient(); + + messageGroup = newStore.getMessageGroup( 1 ); + + Assert.assertEquals( 0, messageGroup.getMessages().size() ); + Assert.assertEquals( 0, messageGroup.size() ); + } + + @Test + public void testCompleteMessageGroup() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( messageGroup.getGroupId(), message ); + store.completeGroup( messageGroup.getGroupId() ); + messageGroup = store.getMessageGroup( 1 ); + Assert.assertTrue( messageGroup.isComplete() ); + } + + @Test + public void testLastReleasedSequenceNumber() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + messageGroup = store.addMessageToGroup( messageGroup.getGroupId(), message ); + store.setLastReleasedSequenceNumberForGroup( messageGroup.getGroupId(), 5 ); + messageGroup = store.getMessageGroup( 1 ); + Assert.assertEquals( 5, messageGroup.getLastReleasedMessageSequenceNumber() ); + } + + @Test + public void testRemoveMessageFromTheGroup() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "2" ); + store.addMessageToGroup( messageGroup.getGroupId(), new GenericMessage( "1" ) ); + store.addMessageToGroup( messageGroup.getGroupId(), message ); + messageGroup = store.addMessageToGroup( messageGroup.getGroupId(), new GenericMessage( "3" ) ); + Assert.assertEquals( 3, messageGroup.size() ); + + messageGroup = store.removeMessageFromGroup( 1, message ); + Assert.assertEquals( 2, messageGroup.size() ); + + // use another message store instance + VoldemortMessageStore newStore = createNewStoreClient(); + + messageGroup = newStore.getMessageGroup( 1 ); + Assert.assertEquals( 2, messageGroup.size() ); + } + + @Test + public void testWithMessageHistory() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + Message message = new GenericMessage( "Hello" ); + DirectChannel fooChannel = new DirectChannel(); + fooChannel.setBeanName( "fooChannel" ); + DirectChannel barChannel = new DirectChannel(); + barChannel.setBeanName( "barChannel" ); + + message = MessageHistory.write( message, fooChannel ); + message = MessageHistory.write( message, barChannel ); + store.addMessageToGroup( 1, message ); + message = store.getMessageGroup( 1 ).getMessages().iterator().next(); + MessageHistory messageHistory = MessageHistory.read( message ); + + Assert.assertNotNull( messageHistory ); + Assert.assertEquals( 2, messageHistory.size() ); + + Properties fooChannelHistory = messageHistory.get( 0 ); + + Assert.assertEquals( "fooChannel", fooChannelHistory.get( "name" ) ); + Assert.assertEquals( "channel", fooChannelHistory.get( "type" ) ); + } + + @Test + public void testRemoveNonExistingMessageFromGroup() { + MessageGroup messageGroup = store.getMessageGroup( 1 ); + store.addMessageToGroup( messageGroup.getGroupId(), new GenericMessage( "1" ) ); + store.removeMessageFromGroup( messageGroup.getGroupId(), new GenericMessage( "2" ) ); + } + + @Test + public void testRemoveNonExistingMessageFromNonExistingGroup() { + store.removeMessageFromGroup( 1, new GenericMessage( "2" ) ); + } + + @Test + public void testMultipleInstancesOfGroupStore() { + VoldemortMessageStore store1 = createNewStoreClient(); + VoldemortMessageStore store2 = createNewStoreClient(); + + Message message = new GenericMessage( "1" ); + store1.addMessageToGroup( 1, message ); + MessageGroup messageGroup = store2.addMessageToGroup( 1, new GenericMessage( "2" ) ); + Assert.assertEquals( 2, messageGroup.getMessages().size() ); + + VoldemortMessageStore store3 = createNewStoreClient(); + messageGroup = store3.removeMessageFromGroup( 1, message ); + Assert.assertEquals( 1, messageGroup.getMessages().size() ); + } + + @Test + public void testIteratorOfMessageGroups() { + VoldemortMessageStore store1 = createNewStoreClient(); + VoldemortMessageStore store2 = createNewStoreClient(); + + store1.addMessageToGroup( 1, new GenericMessage( "1" ) ); + store2.addMessageToGroup( 2, new GenericMessage( "2" ) ); + store1.addMessageToGroup( 3, new GenericMessage( "3" ) ); + store2.addMessageToGroup( 3, new GenericMessage( "3A" ) ); + + Iterator messageGroups = store1.iterator(); + int counter = 0; + while ( messageGroups.hasNext() ) { + final MessageGroup group = messageGroups.next(); + final String groupId = (String) group.getGroupId(); + if ( "1".equals( groupId ) ) { + Assert.assertEquals( 1, group.getMessages().size() ); + } + else if ( "2".equals( groupId ) ) { + Assert.assertEquals( 1, group.getMessages().size() ); + } + else if ( "3".equals( groupId ) ) { + Assert.assertEquals( 2, group.getMessages().size() ); + } + ++counter; + } + Assert.assertEquals( 3, counter ); + + store2.removeMessageGroup( 3 ); + + messageGroups = store1.iterator(); + counter = 0; + while ( messageGroups.hasNext() ) { + messageGroups.next(); + ++counter; + } + Assert.assertEquals( 2, counter ); + } + + @Test + public void testConcurrentModifications() throws InterruptedException { + final VoldemortMessageStore store1 = createNewStoreClient(); + final VoldemortMessageStore store2 = createNewStoreClient(); + + final ExecutorService executor = Executors.newCachedThreadPool(); + final Counter errorCounter = new Counter(); + final Random randomGenerator = new Random(); + for ( int i = 0; i < 10; ++i ) { + executor.execute( new Runnable() { + public void run() { + try { + final Message message = new GenericMessage( UUID.randomUUID() ); + MessageGroup group = store1.addMessageToGroup( 1, message ); + + Thread.sleep( randomGenerator.nextInt( 100 ) ); + + group = store2.removeMessageFromGroup( 1, message ); + } + catch ( Exception e ) { + errorCounter.increment(); + } + } + }); + } + executor.shutdown(); + executor.awaitTermination( 10, TimeUnit.SECONDS ); + + Assert.assertEquals( 0, errorCounter.getValue() ); + Assert.assertEquals( 0, store1.getMessageCount() ); + } + + private static class Counter { + private int value = 0; + + public synchronized int increment() { + return ++value; + } + + public int getValue() { + return value; + } + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest.java new file mode 100644 index 0000000..c38016e --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest.java @@ -0,0 +1,70 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.store; + +import java.io.File; + +import org.junit.Assert; +import org.junit.Test; +import org.springframework.context.support.ClassPathXmlApplicationContext; +import org.springframework.integration.Message; +import org.springframework.integration.MessageChannel; +import org.springframework.integration.channel.QueueChannel; +import org.springframework.integration.support.MessageBuilder; +import org.springframework.integration.voldemort.test.BaseFunctionalTestCase; + +/** + * Voldemort message store tests based on Redis module. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortMessageStoreAggregationTest extends BaseFunctionalTestCase { + @Override + protected File getStoreConfiguration() { + return new File( "src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml" ); + } + + @Test + public void testAggregatorWithShutdown() { + ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext( "VoldemortMessageStoreAggregationTest-context.xml", getClass() ); + MessageChannel input = context.getBean( "inputChannel", MessageChannel.class ); + QueueChannel output = context.getBean( "outputChannel", QueueChannel.class ); + + Message message1 = MessageBuilder.withPayload( "1" ) + .setSequenceNumber( 1 ).setSequenceSize( 3 ).setCorrelationId( 1 ).build(); + Message message2 = MessageBuilder.withPayload( "2" ) + .setSequenceNumber( 2 ).setSequenceSize( 3 ).setCorrelationId( 1 ).build(); + input.send( message1 ); + Assert.assertNull( output.receive( 1000 ) ); + input.send( message2 ); + Assert.assertNull( output.receive( 1000 ) ); + + context.close(); + + context = new ClassPathXmlApplicationContext( "VoldemortMessageStoreAggregationTest-context.xml", getClass() ); + input = context.getBean( "inputChannel", MessageChannel.class ); + output = context.getBean( "outputChannel", QueueChannel.class ); + + Message message3 = MessageBuilder.withPayload( "3" ) + .setSequenceNumber( 3 ).setSequenceSize( 3 ).setCorrelationId( 1 ).build(); + input.send( message3 ); + Assert.assertNotNull( output.receive( 1000 ) ); + + context.close(); + } +} diff --git a/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreTest.java b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreTest.java new file mode 100644 index 0000000..415e604 --- /dev/null +++ b/spring-integration-voldemort/src/test/java/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreTest.java @@ -0,0 +1,187 @@ +/* + * Copyright 2002-2013 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 + * + * 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.springframework.integration.voldemort.test.store; + +import java.io.Serializable; +import java.util.Properties; +import java.util.UUID; + +import org.junit.Assert; +import org.junit.Test; +import org.springframework.integration.Message; +import org.springframework.integration.channel.DirectChannel; +import org.springframework.integration.history.MessageHistory; +import org.springframework.integration.message.GenericMessage; + +/** + * Voldemort message store tests based on Redis module. + * + * @author Lukasz Antoniak + * @since 1.0 + */ +@SuppressWarnings("unchecked") +public class VoldemortMessageStoreTest extends BaseStoreFunctionalTestCase { + @Test + public void testGetNonExistingMessage() { + final Message message = store.getMessage( UUID.randomUUID() ); + + Assert.assertNull( message ); + } + + @Test + public void testGetMessageCountWhenEmpty() { + Assert.assertEquals( 0, store.getMessageCount() ); + } + + @Test + public void testAddStringMessage() { + final Message stringMessage = new GenericMessage( "Hello Voldemort" ); + + final Message storedMessage = store.addMessage( stringMessage ); + + Assert.assertNotSame( stringMessage, storedMessage ); + Assert.assertEquals( stringMessage.getPayload(), storedMessage.getPayload() ); + } + + @Test + public void testAddSerializableObjectMessage() { + final Address address = new Address( "1600 Pennsylvania Av, Washington, DC" ); + final Person person = new Person( "Barack Obama", address ); + Message objectMessage = new GenericMessage( person ); + + Message storedMessage = store.addMessage( objectMessage ); + + Assert.assertNotSame( objectMessage, storedMessage ); + Assert.assertEquals( person.getName(), storedMessage.getPayload().getName() ); + Assert.assertEquals( person.getAddress().getAddress(), storedMessage.getPayload().getAddress().getAddress() ); + } + + @Test(expected = IllegalArgumentException.class) + public void testAddNonSerializableObjectMessage() { + Message objectMessage = new GenericMessage( new Foo() ); + store.addMessage( objectMessage ); + } + + @Test + public void testAddAndGetStringMessage() { + final Message stringMessage = new GenericMessage( "Hello Voldemort" ); + + store.addMessage( stringMessage ); + final Message retrievedMessage = (Message) store.getMessage( stringMessage.getHeaders().getId() ); + + Assert.assertNotNull( retrievedMessage ); + Assert.assertEquals( stringMessage.getPayload(), retrievedMessage.getPayload() ); + } + + @Test + public void testAddAndRemoveStringMessage() { + final Message stringMessage = new GenericMessage( "Hello Voldemort" ); + + store.addMessage(stringMessage); + Message retrievedMessage = (Message) store.removeMessage( stringMessage.getHeaders().getId() ); + + Assert.assertNotNull( retrievedMessage ); + Assert.assertEquals( stringMessage.getPayload(), retrievedMessage.getPayload() ); + Assert.assertNull( store.getMessage( stringMessage.getHeaders().getId() ) ); + } + + @Test + public void testMessageCount() { + final Message stringMessage1 = new GenericMessage( "Hello Voldemort" ); + final Message stringMessage2 = new GenericMessage( "Hello World" ); + + store.addMessage( stringMessage1 ); + Assert.assertEquals( 1, store.getMessageCount() ); + + store.addMessage( stringMessage2 ); + Assert.assertEquals( 2, store.getMessageCount() ); + + store.removeMessage( stringMessage1.getHeaders().getId() ); + Assert.assertEquals( 1, store.getMessageCount() ); + } + + @Test + public void testWithMessageHistory() { + Message message = new GenericMessage( "Hello" ); + final DirectChannel fooChannel = new DirectChannel(); + fooChannel.setBeanName( "fooChannel" ); + final DirectChannel barChannel = new DirectChannel(); + barChannel.setBeanName( "barChannel" ); + + message = MessageHistory.write( message, fooChannel ); + message = MessageHistory.write( message, barChannel ); + store.addMessage( message ); + message = store.getMessage( message.getHeaders().getId() ); + MessageHistory messageHistory = MessageHistory.read( message ); + + Assert.assertNotNull( messageHistory ); + Assert.assertEquals( 2, messageHistory.size() ); + + Properties fooChannelHistory = messageHistory.get( 0 ); + + Assert.assertEquals( "fooChannel", fooChannelHistory.get( "name" ) ); + Assert.assertEquals( "channel", fooChannelHistory.get( "type" ) ); + } + + public static class Person implements Serializable { + private static final long serialVersionUID = 6109955909562732898L; + + private String name; + private Address address; + + public Person(String name, Address address) { + this.name = name; + this.address = address; + } + + public Address getAddress() { + return address; + } + + public void setAddress(Address address) { + this.address = address; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + } + + public static class Address implements Serializable { + private static final long serialVersionUID = 2382619388682259472L; + + private String address; + + public Address(String address) { + this.address = address; + } + + public String getAddress() { + return address; + } + + public void setAddress(String address) { + this.address = address; + } + } + + public static class Foo { + } +} diff --git a/spring-integration-voldemort/src/test/resources/cluster.xml b/spring-integration-voldemort/src/test/resources/cluster.xml new file mode 100755 index 0000000..69f4a64 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/cluster.xml @@ -0,0 +1,12 @@ + + + test-cluster + + 0 + localhost + 8080 + 6666 + 6667 + 0, 1 + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/log4j.properties b/spring-integration-voldemort/src/test/resources/log4j.properties new file mode 100755 index 0000000..4130143 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/log4j.properties @@ -0,0 +1,8 @@ +log4j.rootCategory=WARN, stdout + +log4j.appender.stdout=org.apache.log4j.ConsoleAppender +log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +log4j.appender.stdout.layout.ConversionPattern=%d{HH:mm:ss.SSS} %-5p [%t][%c] %m%n + +log4j.category.org.springframework.integration=WARN +log4j.category.org.springframework.integration.voldemort=INFO diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/common-test-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/common-test-context.xml new file mode 100755 index 0000000..2995ee6 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/common-test-context.xml @@ -0,0 +1,16 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest-context.xml new file mode 100644 index 0000000..bda59a3 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/ObjectKeyTest-context.xml @@ -0,0 +1,25 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest-context.xml new file mode 100644 index 0000000..f5ac708 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/inbound/VoldemortInboundAdapterTest-context.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest-context.xml new file mode 100644 index 0000000..a0e7eb6 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/ObjectKeyTest-context.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest-context.xml new file mode 100755 index 0000000..1c0b337 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/outbound/VoldemortOutboundAdapterTest-context.xml @@ -0,0 +1,42 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest-context.xml new file mode 100644 index 0000000..c861068 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/VoldemortMessageStoreAggregationTest-context.xml @@ -0,0 +1,14 @@ + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/store-test-context.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/store-test-context.xml new file mode 100644 index 0000000..83f1ba9 --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/store-test-context.xml @@ -0,0 +1,25 @@ + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml new file mode 100644 index 0000000..afa6fbf --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/org/springframework/integration/voldemort/test/store/stores.xml @@ -0,0 +1,19 @@ + + + + test-message-store + lukasz.antoniak@gmail.com + bdb + client + 1 + 1 + 1 + + string + UTF-8 + + + java-serialization + + + \ No newline at end of file diff --git a/spring-integration-voldemort/src/test/resources/stores.xml b/spring-integration-voldemort/src/test/resources/stores.xml new file mode 100755 index 0000000..013c15f --- /dev/null +++ b/spring-integration-voldemort/src/test/resources/stores.xml @@ -0,0 +1,34 @@ + + + + test-store + lukasz.antoniak@gmail.com + bdb + client + 1 + 1 + 1 + + string + UTF-8 + + + java-serialization + + + + test-object-store + lukasz.antoniak@gmail.com + bdb + client + 1 + 1 + 1 + + java-serialization + + + java-serialization + + + \ No newline at end of file