From e15e0ef193af38d7d7210ea3f2ac74087081aa33 Mon Sep 17 00:00:00 2001 From: Oleg Zhurakousky Date: Tue, 30 Oct 2018 14:44:08 +0100 Subject: [PATCH] Sync docs from v2.1.0.RC1 to gh-pages --- .../2.1.0.RC1/Guardfile | 20 + .../2.1.0.RC1/css/highlight.css | 35 + .../2.1.0.RC1/css/manual-multipage.css | 9 + .../2.1.0.RC1/css/manual-singlepage.css | 6 + .../2.1.0.RC1/css/manual.css | 344 ++++ .../2.1.0.RC1/ghpages.sh | 330 ++++ .../2.1.0.RC1/images/background.png | Bin 0 -> 18255 bytes .../2.1.0.RC1/images/callouts/1.png | Bin 0 -> 329 bytes .../2.1.0.RC1/images/callouts/2.png | Bin 0 -> 353 bytes .../2.1.0.RC1/images/callouts/3.png | Bin 0 -> 350 bytes .../2.1.0.RC1/images/caution.png | Bin 0 -> 2099 bytes .../2.1.0.RC1/images/important.png | Bin 0 -> 2085 bytes .../2.1.0.RC1/images/logo.png | Bin 0 -> 4387 bytes .../2.1.0.RC1/images/note.png | Bin 0 -> 2257 bytes .../2.1.0.RC1/images/part-bindings.png | Bin 0 -> 61765 bytes .../2.1.0.RC1/images/part-exchange.png | Bin 0 -> 24024 bytes .../2.1.0.RC1/images/part-queues.png | Bin 0 -> 46910 bytes .../2.1.0.RC1/images/rabbit-binder.png | Bin 0 -> 12440 bytes .../2.1.0.RC1/images/tip.png | Bin 0 -> 931 bytes .../2.1.0.RC1/images/warning.png | Bin 0 -> 2130 bytes .../2.1.0.RC1/index.html | 117 ++ .../2.1.0.RC1/multi/css/highlight.css | 35 + .../2.1.0.RC1/multi/css/manual-multipage.css | 9 + .../2.1.0.RC1/multi/css/manual-singlepage.css | 6 + .../2.1.0.RC1/multi/css/manual.css | 344 ++++ .../2.1.0.RC1/multi/images/background.png | Bin 0 -> 18255 bytes .../2.1.0.RC1/multi/images/callouts/1.png | Bin 0 -> 329 bytes .../2.1.0.RC1/multi/images/callouts/2.png | Bin 0 -> 353 bytes .../2.1.0.RC1/multi/images/callouts/3.png | Bin 0 -> 350 bytes .../2.1.0.RC1/multi/images/caution.png | Bin 0 -> 2099 bytes .../2.1.0.RC1/multi/images/important.png | Bin 0 -> 2085 bytes .../2.1.0.RC1/multi/images/logo.png | Bin 0 -> 4387 bytes .../2.1.0.RC1/multi/images/note.png | Bin 0 -> 2257 bytes .../2.1.0.RC1/multi/images/tip.png | Bin 0 -> 931 bytes .../2.1.0.RC1/multi/images/warning.png | Bin 0 -> 2130 bytes .../2.1.0.RC1/multi/multi__appendices.html | 3 + .../multi/multi__configuration_options.html | 92 + ...partitioning_with_the_rabbitmq_binder.html | 80 + .../multi__rabbitmq_binder_overview.html | 21 + .../multi/multi__reference_guide.html | 4 + ...multi__retry_with_the_rabbitmq_binder.html | 42 + .../2.1.0.RC1/multi/multi__usage.html | 9 + .../2.1.0.RC1/multi/multi_building.html | 33 + .../2.1.0.RC1/multi/multi_contributing.html | 26 + .../multi/multi_rabbit-dlq-processing.html | 204 +++ .../multi/multi_rabbit-error-channels.html | 6 + ...lti_spring-cloud-stream-binder-rabbit.html | 3 + .../2.1.0.RC1/single/css/highlight.css | 35 + .../2.1.0.RC1/single/css/manual-multipage.css | 9 + .../single/css/manual-singlepage.css | 6 + .../2.1.0.RC1/single/css/manual.css | 344 ++++ .../2.1.0.RC1/single/images/background.png | Bin 0 -> 18255 bytes .../2.1.0.RC1/single/images/callouts/1.png | Bin 0 -> 329 bytes .../2.1.0.RC1/single/images/callouts/2.png | Bin 0 -> 353 bytes .../2.1.0.RC1/single/images/callouts/3.png | Bin 0 -> 350 bytes .../2.1.0.RC1/single/images/caution.png | Bin 0 -> 2099 bytes .../2.1.0.RC1/single/images/important.png | Bin 0 -> 2085 bytes .../2.1.0.RC1/single/images/logo.png | Bin 0 -> 4387 bytes .../2.1.0.RC1/single/images/note.png | Bin 0 -> 2257 bytes .../2.1.0.RC1/single/images/tip.png | Bin 0 -> 931 bytes .../2.1.0.RC1/single/images/warning.png | Bin 0 -> 2130 bytes .../spring-cloud-stream-binder-rabbit.html | 490 ++++++ .../spring-cloud-stream-binder-rabbit.xml | 1477 +++++++++++++++++ 63 files changed, 4139 insertions(+) create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/Guardfile create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/highlight.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-multipage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-singlepage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/ghpages.sh create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/background.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/1.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/2.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/3.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/caution.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/important.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/logo.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/note.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/part-bindings.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/part-exchange.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/part-queues.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/rabbit-binder.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/tip.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/warning.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/index.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/highlight.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-multipage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-singlepage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/background.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/1.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/2.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/3.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/caution.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/important.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/logo.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/note.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/tip.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/warning.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__appendices.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__configuration_options.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__partitioning_with_the_rabbitmq_binder.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__rabbitmq_binder_overview.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__reference_guide.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__retry_with_the_rabbitmq_binder.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__usage.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_building.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_contributing.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-dlq-processing.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-error-channels.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_spring-cloud-stream-binder-rabbit.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/highlight.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-multipage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-singlepage.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual.css create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/background.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/1.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/2.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/3.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/caution.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/important.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/logo.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/note.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/tip.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/warning.png create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/spring-cloud-stream-binder-rabbit.html create mode 100644 spring-cloud-stream-binder-rabbit/2.1.0.RC1/spring-cloud-stream-binder-rabbit.xml diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/Guardfile b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/Guardfile new file mode 100644 index 00000000..bdd4d729 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/Guardfile @@ -0,0 +1,20 @@ +require 'asciidoctor' +require 'erb' + +guard 'shell' do + watch(/.*\.adoc$/) {|m| + Asciidoctor.render_file('index.adoc', \ + :in_place => true, \ + :safe => Asciidoctor::SafeMode::UNSAFE, \ + :attributes=> { \ + 'source-highlighter' => 'prettify', \ + 'icons' => 'font', \ + 'linkcss'=> 'true', \ + 'copycss' => 'true', \ + 'doctype' => 'book'}) + } +end + +guard 'livereload' do + watch(%r{^.+\.(css|js|html)$}) +end diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/highlight.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/highlight.css new file mode 100644 index 00000000..ffefef72 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/highlight.css @@ -0,0 +1,35 @@ +/* + code highlight CSS resemblign the Eclipse IDE default color schema + @author Costin Leau +*/ + +.hl-keyword { + color: #7F0055; + font-weight: bold; +} + +.hl-comment { + color: #3F5F5F; + font-style: italic; +} + +.hl-multiline-comment { + color: #3F5FBF; + font-style: italic; +} + +.hl-tag { + color: #3F7F7F; +} + +.hl-attribute { + color: #7F007F; +} + +.hl-value { + color: #2A00FF; +} + +.hl-string { + color: #2A00FF; +} \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-multipage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-multipage.css new file mode 100644 index 00000000..0c484531 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-multipage.css @@ -0,0 +1,9 @@ +@IMPORT url("manual.css"); + +body.firstpage { + background: url("../images/background.png") no-repeat center top; +} + +div.part h1 { + border-top: none; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-singlepage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-singlepage.css new file mode 100644 index 00000000..4a7fd140 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual-singlepage.css @@ -0,0 +1,6 @@ +@IMPORT url("manual.css"); + +body { + background: url("../images/background.png") no-repeat center top; +} + diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual.css new file mode 100644 index 00000000..0ecbe2e8 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/css/manual.css @@ -0,0 +1,344 @@ +@IMPORT url("highlight.css"); + +html { + padding: 0pt; + margin: 0pt; +} + +body { + color: #333333; + margin: 15px 30px; + font-family: Helvetica, Arial, Freesans, Clean, Sans-serif; + line-height: 1.6; + -webkit-font-smoothing: antialiased; +} + +code { + font-size: 16px; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +:not(a)>code { + color: #6D180B; +} + +:not(pre)>code { + background-color: #F2F2F2; + border: 1px solid #CCCCCC; + border-radius: 4px; + padding: 1px 3px 0; + text-shadow: none; + white-space: nowrap; +} + +body>*:first-child { + margin-top: 0 !important; +} + +div { + margin: 0pt; +} + +hr { + border: 1px solid #CCCCCC; + background: #CCCCCC; +} + +h1,h2,h3,h4,h5,h6 { + color: #000000; + cursor: text; + font-weight: bold; + margin: 30px 0 10px; + padding: 0; +} + +h1,h2,h3 { + margin: 40px 0 10px; +} + +h1 { + margin: 70px 0 30px; + padding-top: 20px; +} + +div.part h1 { + border-top: 1px dotted #CCCCCC; +} + +h1,h1 code { + font-size: 32px; +} + +h2,h2 code { + font-size: 24px; +} + +h3,h3 code { + font-size: 20px; +} + +h4,h1 code,h5,h5 code,h6,h6 code { + font-size: 18px; +} + +div.book,div.chapter,div.appendix,div.part,div.preface { + min-width: 300px; + max-width: 1200px; + margin: 0 auto; +} + +p.releaseinfo { + font-weight: bold; + margin-bottom: 40px; + margin-top: 40px; +} + +div.authorgroup { + line-height: 1; +} + +p.copyright { + line-height: 1; + margin-bottom: -5px; +} + +.legalnotice p { + font-style: italic; + font-size: 14px; + line-height: 1; +} + +div.titlepage+p,div.titlepage+p { + margin-top: 0; +} + +pre { + line-height: 1.0; + color: black; +} + +a { + color: #4183C4; + text-decoration: none; +} + +p { + margin: 15px 0; + text-align: left; +} + +ul,ol { + padding-left: 30px; +} + +li p { + margin: 0; +} + +div.table { + margin: 1em; + padding: 0.5em; + text-align: center; +} + +div.table table,div.informaltable table { + display: table; + width: 100%; +} + +div.table td { + padding-left: 7px; + padding-right: 7px; +} + +.sidebar { + line-height: 1.4; + padding: 0 20px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; +} + +.sidebar p.title { + color: #6D180B; +} + +pre.programlisting,pre.screen { + font-size: 15px; + padding: 6px 10px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; + clear: both; + overflow: auto; + line-height: 1.4; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +table { + border-collapse: collapse; + border-spacing: 0; + border: 1px solid #DDDDDD !important; + border-radius: 4px !important; + border-collapse: separate !important; + line-height: 1.6; +} + +table thead { + background: #F5F5F5; +} + +table tr { + border: none; + border-bottom: none; +} + +table th { + font-weight: bold; +} + +table th,table td { + border: none !important; + padding: 6px 13px; +} + +table tr:nth-child(2n) { + background-color: #F8F8F8; +} + +td p { + margin: 0 0 15px 0; +} + +div.table-contents td p { + margin: 0; +} + +div.important *,div.note *,div.tip *,div.warning *,div.navheader *,div.navfooter *,div.calloutlist * + { + border: none !important; + background: none !important; + margin: 0; +} + +div.important p,div.note p,div.tip p,div.warning p { + color: #6F6F6F; + line-height: 1.6; +} + +div.important code,div.note code,div.tip code,div.warning code { + background-color: #F2F2F2 !important; + border: 1px solid #CCCCCC !important; + border-radius: 4px !important; + padding: 1px 3px 0 !important; + text-shadow: none !important; + white-space: nowrap !important; +} + +.note th,.tip th,.warning th { + display: none; +} + +.note tr:first-child td,.tip tr:first-child td,.warning tr:first-child td + { + border-right: 1px solid #CCCCCC !important; + padding-top: 10px; +} + +div.calloutlist p,div.calloutlist td { + padding: 0; + margin: 0; +} + +div.calloutlist>table>tbody>tr>td:first-child { + padding-left: 10px; + width: 30px !important; +} + +div.important,div.note,div.tip,div.warning { + margin-left: 0px !important; + margin-right: 20px !important; + margin-top: 20px; + margin-bottom: 20px; + padding-top: 10px; + padding-bottom: 10px; +} + +div.toc { + line-height: 1.2; +} + +dl,dt { + margin-top: 1px; + margin-bottom: 0; +} + +div.toc>dl>dt { + font-size: 32px; + font-weight: bold; + margin: 30px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dt { + font-size: 24px; + font-weight: bold; + margin: 20px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dd>dl>dt { + font-weight: bold; + font-size: 20px; + margin: 10px 0 0 0; +} + +tbody.footnotes * { + border: none !important; +} + +div.footnote p { + margin: 0; + line-height: 1; +} + +div.footnote p sup { + margin-right: 6px; + vertical-align: middle; +} + +div.navheader { + border-bottom: 1px solid #CCCCCC; +} + +div.navfooter { + border-top: 1px solid #CCCCCC; +} + +.title { + margin-left: -1em; + padding-left: 1em; +} + +.title>a { + position: absolute; + visibility: hidden; + display: block; + font-size: 0.85em; + margin-top: 0.05em; + margin-left: -1em; + vertical-align: text-top; + color: black; +} + +.title>a:before { + content: "\00A7"; +} + +.title:hover>a,.title>a:hover,.title:hover>a:hover { + visibility: visible; +} + +.title:focus>a,.title>a:focus,.title:focus>a:focus { + outline: 0; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/ghpages.sh b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/ghpages.sh new file mode 100644 index 00000000..6795216e --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/ghpages.sh @@ -0,0 +1,330 @@ +#!/bin/bash -x + +set -e + +# Set default props like MAVEN_PATH, ROOT_FOLDER etc. +function set_default_props() { + # The script should be executed from the root folder + ROOT_FOLDER=`pwd` + echo "Current folder is ${ROOT_FOLDER}" + + if [[ ! -e "${ROOT_FOLDER}/.git" ]]; then + echo "You're not in the root folder of the project!" + exit 1 + fi + + # Prop that will let commit the changes + COMMIT_CHANGES="no" + MAVEN_PATH=${MAVEN_PATH:-} + echo "Path to Maven is [${MAVEN_PATH}]" + REPO_NAME=${PWD##*/} + echo "Repo name is [${REPO_NAME}]" + SPRING_CLOUD_STATIC_REPO=${SPRING_CLOUD_STATIC_REPO:-git@github.com:spring-cloud/spring-cloud-static.git} + echo "Spring Cloud Static repo is [${SPRING_CLOUD_STATIC_REPO}" +} + +# Check if gh-pages exists and docs have been built +function check_if_anything_to_sync() { + git remote set-url --push origin `git config remote.origin.url | sed -e 's/^git:/https:/'` + + if ! (git remote set-branches --add origin gh-pages && git fetch -q); then + echo "No gh-pages, so not syncing" + exit 0 + fi + + if ! [ -d docs/target/generated-docs ] && ! [ "${BUILD}" == "yes" ]; then + echo "No gh-pages sources in docs/target/generated-docs, so not syncing" + exit 0 + fi +} + +function retrieve_current_branch() { + # Code getting the name of the current branch. For master we want to publish as we did until now + # http://stackoverflow.com/questions/1593051/how-to-programmatically-determine-the-current-checked-out-git-branch + # If there is a branch already passed will reuse it - otherwise will try to find it + CURRENT_BRANCH=${BRANCH} + if [[ -z "${CURRENT_BRANCH}" ]] ; then + CURRENT_BRANCH=$(git symbolic-ref -q HEAD) + CURRENT_BRANCH=${CURRENT_BRANCH##refs/heads/} + CURRENT_BRANCH=${CURRENT_BRANCH:-HEAD} + fi + echo "Current branch is [${CURRENT_BRANCH}]" + git checkout ${CURRENT_BRANCH} || echo "Failed to check the branch... continuing with the script" +} + +# Switches to the provided value of the release version. We always prefix it with `v` +function switch_to_tag() { + git checkout v${VERSION} +} + +# Build the docs if switch is on +function build_docs_if_applicable() { + if [[ "${BUILD}" == "yes" ]] ; then + ./mvnw clean install -P docs -pl docs -DskipTests + fi +} + +# Get the name of the `docs.main` property +# Get whitelisted branches - assumes that a `docs` module is available under `docs` profile +function retrieve_doc_properties() { + MAIN_ADOC_VALUE=$("${MAVEN_PATH}"mvn -q \ + -Dexec.executable="echo" \ + -Dexec.args='${docs.main}' \ + --non-recursive \ + org.codehaus.mojo:exec-maven-plugin:1.3.1:exec) + echo "Extracted 'main.adoc' from Maven build [${MAIN_ADOC_VALUE}]" + + + WHITELIST_PROPERTY=${WHITELIST_PROPERTY:-"docs.whitelisted.branches"} + WHITELISTED_BRANCHES_VALUE=$("${MAVEN_PATH}"mvn -q \ + -Dexec.executable="echo" \ + -Dexec.args="\${${WHITELIST_PROPERTY}}" \ + org.codehaus.mojo:exec-maven-plugin:1.3.1:exec \ + -P docs \ + -pl docs) + echo "Extracted '${WHITELIST_PROPERTY}' from Maven build [${WHITELISTED_BRANCHES_VALUE}]" +} + +# Stash any outstanding changes +function stash_changes() { + git diff-index --quiet HEAD && dirty=$? || (echo "Failed to check if the current repo is dirty. Assuming that it is." && dirty="1") + if [ "$dirty" != "0" ]; then git stash; fi +} + +# Switch to gh-pages branch to sync it with current branch +function add_docs_from_target() { + local DESTINATION_REPO_FOLDER + if [[ -z "${DESTINATION}" && -z "${CLONE}" ]] ; then + DESTINATION_REPO_FOLDER=${ROOT_FOLDER} + elif [[ "${CLONE}" == "yes" ]]; then + mkdir -p ${ROOT_FOLDER}/target + local clonedStatic=${ROOT_FOLDER}/target/spring-cloud-static + if [[ ! -e "${clonedStatic}/.git" ]]; then + echo "Cloning Spring Cloud Static to target" + git clone ${SPRING_CLOUD_STATIC_REPO} ${clonedStatic} && git checkout gh-pages + else + echo "Spring Cloud Static already cloned - will pull changes" + cd ${clonedStatic} && git checkout gh-pages && git pull origin gh-pages + fi + DESTINATION_REPO_FOLDER=${clonedStatic}/${REPO_NAME} + mkdir -p ${DESTINATION_REPO_FOLDER} + else + if [[ ! -e "${DESTINATION}/.git" ]]; then + echo "[${DESTINATION}] is not a git repository" + exit 1 + fi + DESTINATION_REPO_FOLDER=${DESTINATION}/${REPO_NAME} + mkdir -p ${DESTINATION_REPO_FOLDER} + echo "Destination was provided [${DESTINATION}]" + fi + cd ${DESTINATION_REPO_FOLDER} + git checkout gh-pages + git pull origin gh-pages + + # Add git branches + ################################################################### + if [[ -z "${VERSION}" ]] ; then + copy_docs_for_current_version + else + copy_docs_for_provided_version + fi + commit_changes_if_applicable +} + + +# Copies the docs by using the retrieved properties from Maven build +function copy_docs_for_current_version() { + if [[ "${CURRENT_BRANCH}" == "master" ]] ; then + echo -e "Current branch is master - will copy the current docs only to the root folder" + for f in docs/target/generated-docs/*; do + file=${f#docs/target/generated-docs/*} + if ! git ls-files -i -o --exclude-standard --directory | grep -q ^$file$; then + # Not ignored... + cp -rf $f ${ROOT_FOLDER}/ + git add -A ${ROOT_FOLDER}/$file + fi + done + COMMIT_CHANGES="yes" + else + echo -e "Current branch is [${CURRENT_BRANCH}]" + # http://stackoverflow.com/questions/29300806/a-bash-script-to-check-if-a-string-is-present-in-a-comma-separated-list-of-strin + if [[ ",${WHITELISTED_BRANCHES_VALUE}," = *",${CURRENT_BRANCH},"* ]] ; then + mkdir -p ${ROOT_FOLDER}/${CURRENT_BRANCH} + echo -e "Branch [${CURRENT_BRANCH}] is whitelisted! Will copy the current docs to the [${CURRENT_BRANCH}] folder" + for f in docs/target/generated-docs/*; do + file=${f#docs/target/generated-docs/*} + if ! git ls-files -i -o --exclude-standard --directory | grep -q ^$file$; then + # Not ignored... + # We want users to access 2.0.0.BUILD-SNAPSHOT/ instead of 1.0.0.RELEASE/spring-cloud.sleuth.html + if [[ "${file}" == "${MAIN_ADOC_VALUE}.html" ]] ; then + # We don't want to copy the spring-cloud-sleuth.html + # we want it to be converted to index.html + cp -rf $f ${ROOT_FOLDER}/${CURRENT_BRANCH}/index.html + git add -A ${ROOT_FOLDER}/${CURRENT_BRANCH}/index.html + else + cp -rf $f ${ROOT_FOLDER}/${CURRENT_BRANCH} + git add -A ${ROOT_FOLDER}/${CURRENT_BRANCH}/$file + fi + fi + done + COMMIT_CHANGES="yes" + else + echo -e "Branch [${CURRENT_BRANCH}] is not on the white list! Check out the Maven [${WHITELIST_PROPERTY}] property in + [docs] module available under [docs] profile. Won't commit any changes to gh-pages for this branch." + fi + fi +} + +# Copies the docs by using the explicitly provided version +function copy_docs_for_provided_version() { + local FOLDER=${DESTINATION_REPO_FOLDER}/${VERSION} + mkdir -p ${FOLDER} + echo -e "Current tag is [v${VERSION}] Will copy the current docs to the [${FOLDER}] folder" + for f in ${ROOT_FOLDER}/docs/target/generated-docs/*; do + file=${f#${ROOT_FOLDER}/docs/target/generated-docs/*} + copy_docs_for_branch ${file} ${FOLDER} + done + COMMIT_CHANGES="yes" + CURRENT_BRANCH="v${VERSION}" +} + +# Copies the docs from target to the provided destination +# Params: +# $1 - file from target +# $2 - destination to which copy the files +function copy_docs_for_branch() { + local file=$1 + local destination=$2 + if ! git ls-files -i -o --exclude-standard --directory | grep -q ^${file}$; then + # Not ignored... + # We want users to access 2.0.0.BUILD-SNAPSHOT/ instead of 1.0.0.RELEASE/spring-cloud.sleuth.html + if [[ ("${file}" == "${MAIN_ADOC_VALUE}.html") || ("${file}" == "${REPO_NAME}.html") ]] ; then + # We don't want to copy the spring-cloud-sleuth.html + # we want it to be converted to index.html + cp -rf $f ${destination}/index.html + git add -A ${destination}/index.html + else + cp -rf $f ${destination} + git add -A ${destination}/$file + fi + fi +} + +function commit_changes_if_applicable() { + if [[ "${COMMIT_CHANGES}" == "yes" ]] ; then + COMMIT_SUCCESSFUL="no" + git commit -a -m "Sync docs from ${CURRENT_BRANCH} to gh-pages" && COMMIT_SUCCESSFUL="yes" || echo "Failed to commit changes" + + # Uncomment the following push if you want to auto push to + # the gh-pages branch whenever you commit to master locally. + # This is a little extreme. Use with care! + ################################################################### + if [[ "${COMMIT_SUCCESSFUL}" == "yes" ]] ; then + git push origin gh-pages + fi + fi +} + +# Switch back to the previous branch and exit block +function checkout_previous_branch() { + # If -version was provided we need to come back to root project + cd ${ROOT_FOLDER} + git checkout ${CURRENT_BRANCH} || echo "Failed to check the branch... continuing with the script" + if [ "$dirty" != "0" ]; then git stash pop; fi + exit 0 +} + +# Assert if properties have been properly passed +function assert_properties() { +echo "VERSION [${VERSION}], DESTINATION [${DESTINATION}], CLONE [${CLONE}]" +if [[ "${VERSION}" != "" && (-z "${DESTINATION}" && -z "${CLONE}") ]] ; then echo "Version was set but destination / clone was not!"; exit 1;fi +if [[ ("${DESTINATION}" != "" && "${CLONE}" != "") && -z "${VERSION}" ]] ; then echo "Destination / clone was set but version was not!"; exit 1;fi +if [[ "${DESTINATION}" != "" && "${CLONE}" == "yes" ]] ; then echo "Destination and clone was set. Pick one!"; exit 1;fi +} + +# Prints the usage +function print_usage() { +cat </` +- if the destination switch is passed (-d) then the script will check if the provided dir is a git repo and then will + switch to gh-pages of that repo and copy the generated docs to `docs//` + +USAGE: + +You can use the following options: + +-v|--version - the script will apply the whole procedure for a particular library version +-d|--destination - the root of destination folder where the docs should be copied. You have to use the full path. + E.g. point to spring-cloud-static folder. Can't be used with (-c) +-b|--build - will run the standard build process after checking out the branch +-c|--clone - will automatically clone the spring-cloud-static repo instead of providing the destination. + Obviously can't be used with (-d) + +EOF +} + + +# ========================================== +# ____ ____ _____ _____ _____ _______ +# / ____|/ ____| __ \|_ _| __ \__ __| +# | (___ | | | |__) | | | | |__) | | | +# \___ \| | | _ / | | | ___/ | | +# ____) | |____| | \ \ _| |_| | | | +# |_____/ \_____|_| \_\_____|_| |_| +# +# ========================================== + +while [[ $# > 0 ]] +do +key="$1" +case ${key} in + -v|--version) + VERSION="$2" + shift # past argument + ;; + -d|--destination) + DESTINATION="$2" + shift # past argument + ;; + -b|--build) + BUILD="yes" + ;; + -c|--clone) + CLONE="yes" + ;; + -h|--help) + print_usage + exit 0 + ;; + *) + echo "Invalid option: [$1]" + print_usage + exit 1 + ;; +esac +shift # past argument or value +done + +assert_properties +set_default_props +check_if_anything_to_sync +if [[ -z "${VERSION}" ]] ; then + retrieve_current_branch +else + switch_to_tag +fi +build_docs_if_applicable +retrieve_doc_properties +stash_changes +add_docs_from_target +checkout_previous_branch \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/background.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/background.png new file mode 100644 index 0000000000000000000000000000000000000000..15dca6fbe2669fae3609605e49c69cc414f1b6ed GIT binary patch literal 18255 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 EKU&lfD*ylh literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/1.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/1.png new file mode 100644 index 0000000000000000000000000000000000000000..7d473430b7bec514f7de12f5769fe7c5859e8c5d GIT binary patch literal 329 zcmeAS@N?(olHy`uVBq!ia0vp^JRr;gBp8b2n5}^nQC}X^4DKU-G|w_t}fLBA)Suv#nrW z!^h2QnY_`l!BOq-UXEX{m2up>JTQkX)2m zTvF+fTUlI^nXH#utd~++ke^qgmzgTe~DWM4ffP81J literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/2.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/callouts/2.png new file mode 100644 index 0000000000000000000000000000000000000000..5d09341b2f6d2ea2d1d5dad5d980f14b4b05dfd2 GIT binary patch literal 353 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 literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/logo.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..ade2ce6ed9d9e9f2f4d9c5729a252ee618a0a5a7 GIT binary patch literal 4387 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 literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/rabbit-binder.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/rabbit-binder.png new file mode 100644 index 0000000000000000000000000000000000000000..aabf698b3a9873395764d1d974996a22436188a9 GIT binary patch literal 12440 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 literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/tip.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/tip.png new file mode 100644 index 0000000000000000000000000000000000000000..6530abb4b5a5d68e3b1c98de512d264c1f8ba883 GIT binary patch literal 931 zcmV;U16=%xP)bSEogsT%F|x0xuT7xdRv7>Rev?4wv{qrDN}+xS$8V5!!ga&#Y1*BgqL?&c}jPc zG_JlfMSD5I%DQQcHXTbGWQtKpeL6yAB|UI5CQ=~#`}=c}Um;E%R)9u^qI0>&GHQ-g zOm;DCkym+{WF$}@UWrV1mtnTPtu!WtY$r7BOpo|N_#mqWGhK#KR0MD7eW*yPaY&xBTRfcG-E5p&`2dq z875XFdy+3GStd(wD_Mg`dys8Xd_houJju_&*4)mZt2Tk1H)DTRJY^_lf>>*ZU2Th5 zWQ3Ly{;kf91GM2s4Vfv8a-fcsXpb+4t> zmM%11X*>M&PQZNVdARf4d*2x!aq1>jOzQ?>>R)(Ok;sOJ)7jfk$Fdif23? z-}3V78&9qod*O;uGk%fEW^;|k`Lo>bOq2iF72o-IGb2gTw+4B~#iYz(oL}sS7|$R2 zDGfrR{|@~AQJ&v(#4u|ZtJP}t520N48P!$8U;|Vfuq=8>E$w`o2Jf`%eqhqbr%IH1zV?O3uDWqKZId-wMQ*MFefpD5X*w@ zok{kNA?%%$F{M!OUcE^^x{~(wkHK|_*9Yg`KNS88FaVH_sda1Xfs6nE002ovPDHLk FV1jwin)(0$ literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/warning.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/images/warning.png new file mode 100644 index 0000000000000000000000000000000000000000..0d5b5244605adbb7ab05a1549746a9c35490f95b GIT binary patch literal 2130 zcmbVNYg7|w8V(4q($)50y>JmGlLW#g$xLn}Vd2Si)}#|ptPAQp3Bp-3!-lL0;i^LY?;i#f0m5s49g z3h?3rDQg~EDPlm?FKkgKIcWEK-3X6YU0uzs7H|nq84s39r2!5;pF?SI$QqXy^Ko1x zV~zpENvp@<_Bsd`5MabC#3rvCq&$5dg43yGR zAdy0-rWjC#a1N_+kzUMY#pmogD7!DP(9dEKr3c5ngvUq_6>}Y+w-a81v=eSXnIi_+ zI?U>D1q2C!0zHox#XXKH+@|&rPT*OF5yvY$5J|)WwLqnU)c-5;=UChSlQkaY3@^|g z|J5#YBB}=i+n3Ex9bS$P?xJSKLk)-HHII@;3qG#TG^!+aj>0QkO~7kB0+|yM;YrGB zF>Ge@isQ#73%Tp#k_(tInh3U$uJWY-+DND*o}L-CB5g@#9YWWw~pxZ!Obm>yN#ZHFvL0zA3vNCTJ=t?)~`2O1M{L6un=ml<2x zQEYfifmA?Pz0tqRs^2Utt1pU0BQB1eIY0U_I}c1Y#PP7Ci5s7#IJn}xK{G+{_v^Z+1V#$Erodv>d}ew>RP0wzw+GWkGCBlS1Oj5Z!5NK zUuSR6>pa}RSEZ;qE_w1l#`hQX4E67U6NeP zHZ`T&wj0_H)t|Y1thUqnhub?%e)Y07;a^Tq%FO&iQkR&`qG!dh*2WfW(HuRQj*_DI zeCD1bUA|tMwjOb8E|FRIn$pzEU!2j_N}1Z2ig)vbr5xA0rjC70cmI6*%Uf->hWzaZ z{33HABO5q)vRhzDly2l691@+q3O{}NbSzx+I*k@|L4&3leK#$>u+T#TvTELfKb|IH zc23atf3vmfCa0&TWY@iuoA_Pm<0~ttEC*ut`isb^jXS>X^Sp=l?RXN_OJ*&usGXg$ zTlTv;Ch^vhLDf&+62jl64*&D+=+`sN_dap_1W%p|KQVYIdgPL5uRE9(c4On)DXndL zLNq?%!-u*zmMxyYc4m4Nr>>=A=s}PkJkqr&E^b9>5g+}c1%X}3|WKcg&spcJQ)05zI<<5LTBhNKRg$XRTi2{j)yl_ zj4~kKOvr+m(;vw~9zVh(zC-y9jqQnguz5r7FRq^MyKuXAENp(TAzUVtg&Ts+B_s7) zGn+fN0sG>9GjsPLKTRrvCd`71IZulJ1_1jO9KWbD&_@2UC+LTNpxxrdz#E|j|2nA9 zzB!UTyfAEJ&#%$pQ>QX#8vk@_a5iqukMF;8`wRKe{BI}5$H%OROs4J1#j)|?p|YSh z_SpR^e`VE#F52;WL{!+L(yZLRh40*KS;@box;9(-tE)`mcVp27O*>Z{_Lb*5T3cJA yr~0nPHtg2+UHi&&$8ha;`+hiaUmw&!n@8(?5PqF(KE5>Ym)EG)p&uyBP5%a8^# + + + + + + +spring-cloud-stream-binder-rabbit + + + + + + + +

+ + + + + + \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/highlight.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/highlight.css new file mode 100644 index 00000000..ffefef72 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/highlight.css @@ -0,0 +1,35 @@ +/* + code highlight CSS resemblign the Eclipse IDE default color schema + @author Costin Leau +*/ + +.hl-keyword { + color: #7F0055; + font-weight: bold; +} + +.hl-comment { + color: #3F5F5F; + font-style: italic; +} + +.hl-multiline-comment { + color: #3F5FBF; + font-style: italic; +} + +.hl-tag { + color: #3F7F7F; +} + +.hl-attribute { + color: #7F007F; +} + +.hl-value { + color: #2A00FF; +} + +.hl-string { + color: #2A00FF; +} \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-multipage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-multipage.css new file mode 100644 index 00000000..0c484531 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-multipage.css @@ -0,0 +1,9 @@ +@IMPORT url("manual.css"); + +body.firstpage { + background: url("../images/background.png") no-repeat center top; +} + +div.part h1 { + border-top: none; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-singlepage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-singlepage.css new file mode 100644 index 00000000..4a7fd140 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual-singlepage.css @@ -0,0 +1,6 @@ +@IMPORT url("manual.css"); + +body { + background: url("../images/background.png") no-repeat center top; +} + diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual.css new file mode 100644 index 00000000..0ecbe2e8 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/css/manual.css @@ -0,0 +1,344 @@ +@IMPORT url("highlight.css"); + +html { + padding: 0pt; + margin: 0pt; +} + +body { + color: #333333; + margin: 15px 30px; + font-family: Helvetica, Arial, Freesans, Clean, Sans-serif; + line-height: 1.6; + -webkit-font-smoothing: antialiased; +} + +code { + font-size: 16px; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +:not(a)>code { + color: #6D180B; +} + +:not(pre)>code { + background-color: #F2F2F2; + border: 1px solid #CCCCCC; + border-radius: 4px; + padding: 1px 3px 0; + text-shadow: none; + white-space: nowrap; +} + +body>*:first-child { + margin-top: 0 !important; +} + +div { + margin: 0pt; +} + +hr { + border: 1px solid #CCCCCC; + background: #CCCCCC; +} + +h1,h2,h3,h4,h5,h6 { + color: #000000; + cursor: text; + font-weight: bold; + margin: 30px 0 10px; + padding: 0; +} + +h1,h2,h3 { + margin: 40px 0 10px; +} + +h1 { + margin: 70px 0 30px; + padding-top: 20px; +} + +div.part h1 { + border-top: 1px dotted #CCCCCC; +} + +h1,h1 code { + font-size: 32px; +} + +h2,h2 code { + font-size: 24px; +} + +h3,h3 code { + font-size: 20px; +} + +h4,h1 code,h5,h5 code,h6,h6 code { + font-size: 18px; +} + +div.book,div.chapter,div.appendix,div.part,div.preface { + min-width: 300px; + max-width: 1200px; + margin: 0 auto; +} + +p.releaseinfo { + font-weight: bold; + margin-bottom: 40px; + margin-top: 40px; +} + +div.authorgroup { + line-height: 1; +} + +p.copyright { + line-height: 1; + margin-bottom: -5px; +} + +.legalnotice p { + font-style: italic; + font-size: 14px; + line-height: 1; +} + +div.titlepage+p,div.titlepage+p { + margin-top: 0; +} + +pre { + line-height: 1.0; + color: black; +} + +a { + color: #4183C4; + text-decoration: none; +} + +p { + margin: 15px 0; + text-align: left; +} + +ul,ol { + padding-left: 30px; +} + +li p { + margin: 0; +} + +div.table { + margin: 1em; + padding: 0.5em; + text-align: center; +} + +div.table table,div.informaltable table { + display: table; + width: 100%; +} + +div.table td { + padding-left: 7px; + padding-right: 7px; +} + +.sidebar { + line-height: 1.4; + padding: 0 20px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; +} + +.sidebar p.title { + color: #6D180B; +} + +pre.programlisting,pre.screen { + font-size: 15px; + padding: 6px 10px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; + clear: both; + overflow: auto; + line-height: 1.4; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +table { + border-collapse: collapse; + border-spacing: 0; + border: 1px solid #DDDDDD !important; + border-radius: 4px !important; + border-collapse: separate !important; + line-height: 1.6; +} + +table thead { + background: #F5F5F5; +} + +table tr { + border: none; + border-bottom: none; +} + +table th { + font-weight: bold; +} + +table th,table td { + border: none !important; + padding: 6px 13px; +} + +table tr:nth-child(2n) { + background-color: #F8F8F8; +} + +td p { + margin: 0 0 15px 0; +} + +div.table-contents td p { + margin: 0; +} + +div.important *,div.note *,div.tip *,div.warning *,div.navheader *,div.navfooter *,div.calloutlist * + { + border: none !important; + background: none !important; + margin: 0; +} + +div.important p,div.note p,div.tip p,div.warning p { + color: #6F6F6F; + line-height: 1.6; +} + +div.important code,div.note code,div.tip code,div.warning code { + background-color: #F2F2F2 !important; + border: 1px solid #CCCCCC !important; + border-radius: 4px !important; + padding: 1px 3px 0 !important; + text-shadow: none !important; + white-space: nowrap !important; +} + +.note th,.tip th,.warning th { + display: none; +} + +.note tr:first-child td,.tip tr:first-child td,.warning tr:first-child td + { + border-right: 1px solid #CCCCCC !important; + padding-top: 10px; +} + +div.calloutlist p,div.calloutlist td { + padding: 0; + margin: 0; +} + +div.calloutlist>table>tbody>tr>td:first-child { + padding-left: 10px; + width: 30px !important; +} + +div.important,div.note,div.tip,div.warning { + margin-left: 0px !important; + margin-right: 20px !important; + margin-top: 20px; + margin-bottom: 20px; + padding-top: 10px; + padding-bottom: 10px; +} + +div.toc { + line-height: 1.2; +} + +dl,dt { + margin-top: 1px; + margin-bottom: 0; +} + +div.toc>dl>dt { + font-size: 32px; + font-weight: bold; + margin: 30px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dt { + font-size: 24px; + font-weight: bold; + margin: 20px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dd>dl>dt { + font-weight: bold; + font-size: 20px; + margin: 10px 0 0 0; +} + +tbody.footnotes * { + border: none !important; +} + +div.footnote p { + margin: 0; + line-height: 1; +} + +div.footnote p sup { + margin-right: 6px; + vertical-align: middle; +} + +div.navheader { + border-bottom: 1px solid #CCCCCC; +} + +div.navfooter { + border-top: 1px solid #CCCCCC; +} + +.title { + margin-left: -1em; + padding-left: 1em; +} + +.title>a { + position: absolute; + visibility: hidden; + display: block; + font-size: 0.85em; + margin-top: 0.05em; + margin-left: -1em; + vertical-align: text-top; + color: black; +} + +.title>a:before { + content: "\00A7"; +} + +.title:hover>a,.title>a:hover,.title:hover>a:hover { + visibility: visible; +} + +.title:focus>a,.title>a:focus,.title:focus>a:focus { + outline: 0; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/background.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/background.png new file mode 100644 index 0000000000000000000000000000000000000000..15dca6fbe2669fae3609605e49c69cc414f1b6ed GIT binary patch literal 18255 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 EKU&lfD*ylh literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/1.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/1.png new file mode 100644 index 0000000000000000000000000000000000000000..7d473430b7bec514f7de12f5769fe7c5859e8c5d GIT binary patch literal 329 zcmeAS@N?(olHy`uVBq!ia0vp^JRr;gBp8b2n5}^nQC}X^4DKU-G|w_t}fLBA)Suv#nrW z!^h2QnY_`l!BOq-UXEX{m2up>JTQkX)2m zTvF+fTUlI^nXH#utd~++ke^qgmzgTe~DWM4ffP81J literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/2.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/callouts/2.png new file mode 100644 index 0000000000000000000000000000000000000000..5d09341b2f6d2ea2d1d5dad5d980f14b4b05dfd2 GIT binary patch literal 353 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 literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/logo.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..ade2ce6ed9d9e9f2f4d9c5729a252ee618a0a5a7 GIT binary patch literal 4387 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 FV1jwin)(0$ literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/warning.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/images/warning.png new file mode 100644 index 0000000000000000000000000000000000000000..0d5b5244605adbb7ab05a1549746a9c35490f95b GIT binary patch literal 2130 zcmbVNYg7|w8V(4q($)50y>JmGlLW#g$xLn}Vd2Si)}#|ptPAQp3Bp-3!-lL0;i^LY?;i#f0m5s49g z3h?3rDQg~EDPlm?FKkgKIcWEK-3X6YU0uzs7H|nq84s39r2!5;pF?SI$QqXy^Ko1x zV~zpENvp@<_Bsd`5MabC#3rvCq&$5dg43yGR zAdy0-rWjC#a1N_+kzUMY#pmogD7!DP(9dEKr3c5ngvUq_6>}Y+w-a81v=eSXnIi_+ zI?U>D1q2C!0zHox#XXKH+@|&rPT*OF5yvY$5J|)WwLqnU)c-5;=UChSlQkaY3@^|g z|J5#YBB}=i+n3Ex9bS$P?xJSKLk)-HHII@;3qG#TG^!+aj>0QkO~7kB0+|yM;YrGB zF>Ge@isQ#73%Tp#k_(tInh3U$uJWY-+DND*o}L-CB5g@#9YWWw~pxZ!Obm>yN#ZHFvL0zA3vNCTJ=t?)~`2O1M{L6un=ml<2x zQEYfifmA?Pz0tqRs^2Utt1pU0BQB1eIY0U_I}c1Y#PP7Ci5s7#IJn}xK{G+{_v^Z+1V#$Erodv>d}ew>RP0wzw+GWkGCBlS1Oj5Z!5NK zUuSR6>pa}RSEZ;qE_w1l#`hQX4E67U6NeP zHZ`T&wj0_H)t|Y1thUqnhub?%e)Y07;a^Tq%FO&iQkR&`qG!dh*2WfW(HuRQj*_DI zeCD1bUA|tMwjOb8E|FRIn$pzEU!2j_N}1Z2ig)vbr5xA0rjC70cmI6*%Uf->hWzaZ z{33HABO5q)vRhzDly2l691@+q3O{}NbSzx+I*k@|L4&3leK#$>u+T#TvTELfKb|IH zc23atf3vmfCa0&TWY@iuoA_Pm<0~ttEC*ut`isb^jXS>X^Sp=l?RXN_OJ*&usGXg$ zTlTv;Ch^vhLDf&+62jl64*&D+=+`sN_dap_1W%p|KQVYIdgPL5uRE9(c4On)DXndL zLNq?%!-u*zmMxyYc4m4Nr>>=A=s}PkJkqr&E^b9>5g+}c1%X}3|WKcg&spcJQ)05zI<<5LTBhNKRg$XRTi2{j)yl_ zj4~kKOvr+m(;vw~9zVh(zC-y9jqQnguz5r7FRq^MyKuXAENp(TAzUVtg&Ts+B_s7) zGn+fN0sG>9GjsPLKTRrvCd`71IZulJ1_1jO9KWbD&_@2UC+LTNpxxrdz#E|j|2nA9 zzB!UTyfAEJ&#%$pQ>QX#8vk@_a5iqukMF;8`wRKe{BI}5$H%OROs4J1#j)|?p|YSh z_SpR^e`VE#F52;WL{!+L(yZLRh40*KS;@box;9(-tE)`mcVp27O*>Z{_Lb*5T3cJA yr~0nPHtg2+UHi&&$8ha;`+hiaUmw&!n@8(?5PqF(KE5>Ym)EG)p&uyBP5%a8^# + + Part II. Appendices

Part II. Appendices

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__configuration_options.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__configuration_options.html new file mode 100644 index 00000000..5954916a --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__configuration_options.html @@ -0,0 +1,92 @@ + + + 3. Configuration Options

3. Configuration Options

This section contains settings specific to the RabbitMQ Binder and bound channels.

For general binding configuration options and properties, see the Spring Cloud Stream core documentation.

3.1 RabbitMQ Binder Properties

By default, the RabbitMQ binder uses Spring Boot’s ConnectionFactory. +Conseuqently, it supports all Spring Boot configuration options for RabbitMQ. +(For reference, see the Spring Boot documentation). +RabbitMQ configuration options use the spring.rabbitmq prefix.

In addition to Spring Boot options, the RabbitMQ binder supports the following properties:

spring.cloud.stream.rabbit.binder.adminAddresses

A comma-separated list of RabbitMQ management plugin URLs. +Only used when nodes contains more than one entry. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information.

Default: empty.

spring.cloud.stream.rabbit.binder.nodes

A comma-separated list of RabbitMQ node names. +When more than one entry, used to locate the server address where a queue is located. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information.

Default: empty.

spring.cloud.stream.rabbit.binder.compressionLevel

The compression level for compressed bindings. +See java.util.zip.Deflater.

Default: 1 (BEST_LEVEL).

spring.cloud.stream.binder.connection-name-prefix

A connection name prefix used to name the connection(s) created by this binder. +The name is this prefix followed by #n, where n increments each time a new connection is opened.

Default: none (Spring AMQP default).

3.2 RabbitMQ Consumer Properties

The following properties are available for Rabbit consumers only and must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.consumer..

acknowledgeMode

The acknowledge mode.

Default: AUTO.

autoBindDlq

Whether to automatically declare the DLQ and bind it to the binder DLX.

Default: false.

bindingRoutingKey

The routing key with which to bind the queue to the exchange (if bindQueue is true). +For partitioned destinations, -<instanceIndex> is appended.

Default: #.

bindQueue

Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue.

Default: true.

consumerTagPrefix

Used to create the consumer tag(s); will be appended by #n where n increments for each consumer created. +Example: ${spring.application.name}-${spring.cloud.stream.bindings.input.group}-${spring.cloud.stream.instance-index}.

Default: none - the broker will generate random consumer tags.

deadLetterQueueName

The name of the DLQ

Default: prefix+destination.dlq

deadLetterExchange

A DLX to assign to the queue. +Relevant only if autoBindDlq is true.

Default: 'prefix+DLX'

deadLetterExchangeType

The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true.

Default: 'direct'

deadLetterRoutingKey

A dead letter routing key to assign to the queue. +Relevant only if autoBindDlq is true.

Default: destination

declareDlx

Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX.

Default: true.

declareExchange

Whether to declare the exchange for the destination.

Default: true.

delayedExchange

Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType.

Default: false.

dlqDeadLetterExchange

If a DLQ is declared, a DLX to assign to that queue.

Default: none

dlqDeadLetterRoutingKey

If a DLQ is declared, a dead letter routing key to assign to that queue.

Default: none

dlqExpires

How long before an unused dead letter queue is deleted (in milliseconds).

Default: no expiration

dlqLazy

Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue.

Default: false.

dlqMaxLength

Maximum number of messages in the dead letter queue.

Default: no limit

dlqMaxLengthBytes

Maximum number of total bytes in the dead letter queue from all messages.

Default: no limit

dlqMaxPriority

Maximum priority of messages in the dead letter queue (0-255).

Default: none

dlqOverflowBehavior

Action to take when dlqMaxLength or dlqMaxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation.

Default: none

dlqTtl

Default time to live to apply to the dead letter queue when declared (in milliseconds).

Default: no limit

durableSubscription

Whether the subscription should be durable. +Only effective if group is also set.

Default: true.

exchangeAutoDelete

If declareExchange is true, whether the exchange should be auto-deleted (that is, removed after the last queue is removed).

Default: true.

exchangeDurable

If declareExchange is true, whether the exchange should be durable (that is, it survives broker restart).

Default: true.

exchangeType

The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations.

Default: topic.

exclusive

Whether to create an exclusive consumer. +Concurrency should be 1 when this is true. +Often used when strict ordering is required but enabling a hot standby instance to take over after a failure. +See recoveryInterval, which controls how often a standby instance attempts to consume.

Default: false.

expires

How long before an unused queue is deleted (in milliseconds).

Default: no expiration

failedDeclarationRetryInterval

The interval (in milliseconds) between attempts to consume from a queue if it is missing.

Default: 5000

frameMaxHeadroom

The number of bytes to reserve for other headers when adding the stack trace to a DLQ message header. +All headers must fit within the frame_max size configured on the broker. +Stack traces can be large; if the size plus this property exceeds frame_max then the stack trace will be truncated. +A WARN log will be written; consider increasing the frame_max or reducing the stack trace by catching the exception and throwing one with a smaller stack trace.

Default: 20000

headerPatterns

Patterns for headers to be mapped from inbound messages.

Default: ['*'] (all headers).

lazy

Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue.

Default: false.

maxConcurrency

The maximum number of consumers.

Default: 1.

maxLength

The maximum number of messages in the queue.

Default: no limit

maxLengthBytes

The maximum number of total bytes in the queue from all messages.

Default: no limit

maxPriority

The maximum priority of messages in the queue (0-255).

Default: none

missingQueuesFatal

When the queue cannot be found, whether to treat the condition as fatal and stop the listener container. +Defaults to false so that the container keeps trying to consume from the queue — for example, when using a cluster and the node hosting a non-HA queue is down.

Default: false

overflowBehavior

Action to take when maxLength or maxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation.

Default: none

prefetch

Prefetch count.

Default: 1.

prefix

A prefix to be added to the name of the destination and queues.

Default: "".

queueDeclarationRetries

The number of times to retry consuming from a queue if it is missing. +Relevant only when missingQueuesFatal is true. +Otherwise, the container keeps retrying indefinitely.

Default: 3

queueNameGroupOnly

When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue.

Default: false.

recoveryInterval

The interval between connection recovery attempts, in milliseconds.

Default: 5000.

requeueRejected

Whether delivery failures should be re-queued when retry is disabled or republishToDlq is false.

Default: false.

republishDeliveryMode

When republishToDlq is true, specifies the delivery mode of the republished message.

Default: DeliveryMode.PERSISTENT

republishToDlq

By default, messages that fail after retries are exhausted are rejected. +If a dead-letter queue (DLQ) is configured, RabbitMQ routes the failed message (unchanged) to the DLQ. +If set to true, the binder republishs failed messages to the DLQ with additional headers, including the exception message and stack trace from the cause of the final failure. +Also see the frameMaxHeadroom property.

Default: false

transacted

Whether to use transacted channels.

Default: false.

ttl

Default time to live to apply to the queue when declared (in milliseconds).

Default: no limit

txSize

The number of deliveries between acks.

Default: 1.

3.3 Advanced Listener Container Configuration

To set listener container properties that are not exposed as binder or binding properties, add a single bean of type ListenerContainerCustomizer to the application context. +The binder and binding properties will be set and then the customizer will be called. +The customizer (configure() method) is provided with the queue name as well as the consumer group as arguments.

3.4 Rabbit Producer Properties

The following properties are available for Rabbit producers only and +must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.producer..

autoBindDlq

Whether to automatically declare the DLQ and bind it to the binder DLX.

Default: false.

batchingEnabled

Whether to enable message batching by producers. +Messages are batched into one message according to the following properties (described in the next three entries in this list): 'batchSize', batchBufferLimit, and batchTimeout. +See Batching for more information.

Default: false.

batchSize

The number of messages to buffer when batching is enabled.

Default: 100.

batchBufferLimit

The maximum buffer size when batching is enabled.

Default: 10000.

batchTimeout

The batch timeout when batching is enabled.

Default: 5000.

bindingRoutingKey

The routing key with which to bind the queue to the exchange (if bindQueue is true). +Only applies to non-partitioned destinations. +Only applies if requiredGroups are provided and then only to those groups.

Default: #.

bindQueue

Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue. +Only applies if requiredGroups are provided and then only to those groups.

Default: true.

compress

Whether data should be compressed when sent.

Default: false.

confirmAckChannel

When errorChannelEnabled is true, a channel to which to send positive delivery acknowledgments (aka publisher confirms). +If the channel does not exist, a DirectChannel is registered with this name. +The connection factory must be configured to enable publisher confirms.

Default: nullChannel (acks are discarded).

deadLetterQueueName

The name of the DLQ +Only applies if requiredGroups are provided and then only to those groups.

Default: prefix+destination.dlq

deadLetterExchange

A DLX to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: 'prefix+DLX'

deadLetterExchangeType

The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: 'direct'

deadLetterRoutingKey

A dead letter routing key to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: destination

declareDlx

Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX. +Applies only when requiredGroups are provided and then only to those groups.

Default: true.

declareExchange

Whether to declare the exchange for the destination.

Default: true.

delayExpression

A SpEL expression to evaluate the delay to apply to the message (x-delay header). +It has no effect if the exchange is not a delayed message exchange.

Default: No x-delay header is set.

delayedExchange

Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType.

Default: false.

deliveryMode

The delivery mode.

Default: PERSISTENT.

dlqDeadLetterExchange

When a DLQ is declared, a DLX to assign to that queue. +Applies only if requiredGroups are provided and then only to those groups.

Default: none

dlqDeadLetterRoutingKey

When a DLQ is declared, a dead letter routing key to assign to that queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: none

dlqExpires

How long (in milliseconds) before an unused dead letter queue is deleted. +Applies only when requiredGroups are provided and then only to those groups.

Default: no expiration

dlqLazy
Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups.
dlqMaxLength

Maximum number of messages in the dead letter queue. +Applies only if requiredGroups are provided and then only to those groups.

Default: no limit

dlqMaxLengthBytes

Maximum number of total bytes in the dead letter queue from all messages. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

dlqMaxPriority

Maximum priority of messages in the dead letter queue (0-255) +Applies only when requiredGroups are provided and then only to those groups.

Default: none

dlqTtl

Default time (in milliseconds) to live to apply to the dead letter queue when declared. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

exchangeAutoDelete

If declareExchange is true, whether the exchange should be auto-delete (it is removed after the last queue is removed).

Default: true.

exchangeDurable

If declareExchange is true, whether the exchange should be durable (survives broker restart).

Default: true.

exchangeType

The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations.

Default: topic.

expires

How long (in milliseconds) before an unused queue is deleted. +Applies only when requiredGroups are provided and then only to those groups.

Default: no expiration

headerPatterns

Patterns for headers to be mapped to outbound messages.

Default: ['*'] (all headers).

lazy

Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: false.

maxLength

Maximum number of messages in the queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

maxLengthBytes

Maximum number of total bytes in the queue from all messages. +Only applies if requiredGroups are provided and then only to those groups.

Default: no limit

maxPriority

Maximum priority of messages in the queue (0-255). +Only applies if requiredGroups are provided and then only to those groups.

Default: none

prefix

A prefix to be added to the name of the destination exchange.

Default: "".

queueNameGroupOnly

When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: false.

routingKeyExpression

A SpEL expression to determine the routing key to use when publishing messages. +For a fixed routing key, use a literal expression, such as routingKeyExpression='my.routingKey' in a properties file or routingKeyExpression: '''my.routingKey''' in a YAML file.

Default: destination or destination-<partition> for partitioned destinations.

transacted

Whether to use transacted channels.

Default: false.

ttl

Default time (in milliseconds) to live to apply to the queue when declared. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

[Note]Note

In the case of RabbitMQ, content type headers can be set by external applications. +Spring Cloud Stream supports them as part of an extended internal protocol used for any type of transport — including transports, such as Kafka (prior to 0.11), that do not natively support headers.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__partitioning_with_the_rabbitmq_binder.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__partitioning_with_the_rabbitmq_binder.html new file mode 100644 index 00000000..b0bbeab1 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__partitioning_with_the_rabbitmq_binder.html @@ -0,0 +1,80 @@ + + + 7. Partitioning with the RabbitMQ Binder

7. Partitioning with the RabbitMQ Binder

RabbitMQ does not support partitioning natively.

Sometimes, it is advantageous to send data to specific partitions — for example, when you want to strictly order message processing, all messages for a particular customer should go to the same partition.

The RabbitMessageChannelBinder provides partitioning by binding a queue for each partition to the destination exchange.

The following Java and YAML examples show how to configure the producer:

Producer.  +

@SpringBootApplication
+@EnableBinding(Source.class)
+public class RabbitPartitionProducerApplication {
+
+    private static final Random RANDOM = new Random(System.currentTimeMillis());
+
+    private static final String[] data = new String[] {
+            "abc1", "def1", "qux1",
+            "abc2", "def2", "qux2",
+            "abc3", "def3", "qux3",
+            "abc4", "def4", "qux4",
+            };
+
+    public static void main(String[] args) {
+        new SpringApplicationBuilder(RabbitPartitionProducerApplication.class)
+            .web(false)
+            .run(args);
+    }
+
+    @InboundChannelAdapter(channel = Source.OUTPUT, poller = @Poller(fixedRate = "5000"))
+    public Message<?> generate() {
+        String value = data[RANDOM.nextInt(data.length)];
+        System.out.println("Sending: " + value);
+        return MessageBuilder.withPayload(value)
+                .setHeader("partitionKey", value)
+                .build();
+    }
+
+}

+

application.yml.  +

    spring:
+      cloud:
+        stream:
+          bindings:
+            output:
+              destination: partitioned.destination
+              producer:
+                partitioned: true
+                partition-key-expression: headers['partitionKey']
+                partition-count: 2
+                required-groups:
+                - myGroup

+

[Note]Note

The configuration in the prececing example uses the default partitioning (key.hashCode() % partitionCount). +This may or may not provide a suitably balanced algorithm, depending on the key values. +You can override this default by using the partitionSelectorExpression or partitionSelectorClass properties.

The required-groups property is required only if you need the consumer queues to be provisioned when the producer is deployed. +Otherwise, any messages sent to a partition are lost until the corresponding consumer is deployed.

The following configuration provisions a topic exchange:

part exchange

The following queues are bound to that exchange:

part queues

The following bindings associate the queues to the exchange:

part bindings

The following Java and YAML examples continue the previous examples and show how to configure the consumer:

Consumer.  +

@SpringBootApplication
+@EnableBinding(Sink.class)
+public class RabbitPartitionConsumerApplication {
+
+    public static void main(String[] args) {
+        new SpringApplicationBuilder(RabbitPartitionConsumerApplication.class)
+            .web(false)
+            .run(args);
+    }
+
+    @StreamListener(Sink.INPUT)
+    public void listen(@Payload String in, @Header(AmqpHeaders.CONSUMER_QUEUE) String queue) {
+        System.out.println(in + " received from queue " + queue);
+    }
+
+}

+

application.yml.  +

    spring:
+      cloud:
+        stream:
+          bindings:
+            input:
+              destination: partitioned.destination
+              group: myGroup
+              consumer:
+                partitioned: true
+                instance-index: 0

+

[Important]Important

The RabbitMessageChannelBinder does not support dynamic scaling. +There must be at least one consumer per partition. +The consumer’s instanceIndex is used to indicate which partition is consumed. +Platforms such as Cloud Foundry can have only one instance with an instanceIndex.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__rabbitmq_binder_overview.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__rabbitmq_binder_overview.html new file mode 100644 index 00000000..e80728de --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__rabbitmq_binder_overview.html @@ -0,0 +1,21 @@ + + + 2. RabbitMQ Binder Overview

2. RabbitMQ Binder Overview

The following simplified diagram shows how the RabbitMQ binder operates:

Figure 2.1. RabbitMQ Binder

rabbit binder

By default, the RabbitMQ Binder implementation maps each destination to a TopicExchange. +For each consumer group, a Queue is bound to that TopicExchange. +Each consumer instance has a corresponding RabbitMQ Consumer instance for its group’s Queue. +For partitioned producers and consumers, the queues are suffixed with the partition index and use the partition index as the routing key. +For anonymous consumers (those with no group property), an auto-delete queue (with a randomized unique name) is used.

By using the optional autoBindDlq option, you can configure the binder to create and configure dead-letter queues (DLQs) (and a dead-letter exchange DLX, as well as routing infrastructure). +By default, the dead letter queue has the name of the destination, appended with .dlq. +If retry is enabled (maxAttempts > 1), failed messages are delivered to the DLQ after retries are exhausted. +If retry is disabled (maxAttempts = 1), you should set requeueRejected to false (the default) so that failed messages are routed to the DLQ, instead of being re-queued. +In addition, republishToDlq causes the binder to publish a failed message to the DLQ (instead of rejecting it). +This feature lets additional information (such as the stack trace in the x-exception-stacktrace header) be added to the message in headers. +See the frameMaxHeadroom property for information about truncated stack traces. +This option does not need retry enabled. +You can republish a failed message after just one attempt. +Starting with version 1.2, you can configure the delivery mode of republished messages. +See the republishDeliveryMode property.

If the stream listener throws an ImmediateAcknowledgeAmqpException, the DLQ is bypassed and the message simply discarded. +Starting with version 2.1, this is true regardless of the setting of republishToDlq; previously it was only the case when republishToDlq was false.

[Important]Important

Setting requeueRejected to true (with republishToDlq=false ) causes the message to be re-queued and redelivered continually, which is likely not what you want unless the reason for the failure is transient. +In general, you should enable retry within the binder by setting maxAttempts to greater than one or by setting republishToDlq to true.

See Section 3.1, “RabbitMQ Binder Properties” for more information about these properties.

The framework does not provide any standard mechanism to consume dead-letter messages (or to re-route them back to the primary queue). +Some options are described in Chapter 6, Dead-Letter Queue Processing.

[Note]Note

When multiple RabbitMQ binders are used in a Spring Cloud Stream application, it is important to disable 'RabbitAutoConfiguration' to avoid the same configuration from RabbitAutoConfiguration being applied to the two binders. +You can exclude the class by using the @SpringBootApplication annotation.

Starting with version 2.0, the RabbitMessageChannelBinder sets the RabbitTemplate.userPublisherConnection property to true so that the non-transactional producers avoid deadlocks on consumers, which can happen if cached connections are blocked because of a memory alarm on the broker.

[Note]Note

Currently, a multiplex consumer (a single consumer listening to multiple queues) is only supported for message-driven conssumers; polled consumers can only retrieve messages from a single queue.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__reference_guide.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__reference_guide.html new file mode 100644 index 00000000..dc904356 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__reference_guide.html @@ -0,0 +1,4 @@ + + + Part I. Reference Guide

Part I. Reference Guide

This guide describes the RabbitMQ implementation of the Spring Cloud Stream Binder. +It contains information about its design, usage and configuration options, as well as information on how the Stream Cloud Stream concepts map into RabbitMQ specific constructs.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__retry_with_the_rabbitmq_binder.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__retry_with_the_rabbitmq_binder.html new file mode 100644 index 00000000..c8b5488e --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__retry_with_the_rabbitmq_binder.html @@ -0,0 +1,42 @@ + + + 4. Retry With the RabbitMQ Binder

4. Retry With the RabbitMQ Binder

When retry is enabled within the binder, the listener container thread is suspended for any back off periods that are configured. +This might be important when strict ordering is required with a single consumer. However, for other use cases, it prevents other messages from being processed on that thread. +An alternative to using binder retry is to set up dead lettering with time to live on the dead-letter queue (DLQ) as well as dead-letter configuration on the DLQ itself. +See Section 3.1, “RabbitMQ Binder Properties” for more information about the properties discussed here. +You can use the following example configuration to enable this feature:

  • Set autoBindDlq to true. +The binder create a DLQ. +Optionally, you can specify a name in deadLetterQueueName.
  • Set dlqTtl to the back off time you want to wait between redeliveries.
  • Set the dlqDeadLetterExchange to the default exchange. +Expired messages from the DLQ are routed to the original queue, because the default deadLetterRoutingKey is the queue name (destination.group). +Setting to the default exchange is achieved by setting the property with no value, as shown in the next example.

To force a message to be dead-lettered, either throw an AmqpRejectAndDontRequeueException or set requeueRejected to true (the default) and throw any exception.

The loop continue without end, which is fine for transient problems, but you may want to give up after some number of attempts. +Fortunately, RabbitMQ provides the x-death header, which lets you determine how many cycles have occurred.

To acknowledge a message after giving up, throw an ImmediateAcknowledgeAmqpException.

4.1 Putting it All Together

The following configuration creates an exchange myDestination with queue myDestination.consumerGroup bound to a topic exchange with a wildcard routing key #:

---
+spring.cloud.stream.bindings.input.destination=myDestination
+spring.cloud.stream.bindings.input.group=consumerGroup
+#disable binder retries
+spring.cloud.stream.bindings.input.consumer.max-attempts=1
+#dlx/dlq setup
+spring.cloud.stream.rabbit.bindings.input.consumer.auto-bind-dlq=true
+spring.cloud.stream.rabbit.bindings.input.consumer.dlq-ttl=5000
+spring.cloud.stream.rabbit.bindings.input.consumer.dlq-dead-letter-exchange=
+---

This configuration creates a DLQ bound to a direct exchange (DLX) with a routing key of myDestination.consumerGroup. +When messages are rejected, they are routed to the DLQ. +After 5 seconds, the message expires and is routed to the original queue by using the queue name as the routing key, as shown in the following example:

Spring Boot application.  +

@SpringBootApplication
+@EnableBinding(Sink.class)
+public class XDeathApplication {
+
+    public static void main(String[] args) {
+        SpringApplication.run(XDeathApplication.class, args);
+    }
+
+    @StreamListener(Sink.INPUT)
+    public void listen(String in, @Header(name = "x-death", required = false) Map<?,?> death) {
+        if (death != null && death.get("count").equals(3L)) {
+            // giving up - don't send to DLX
+            throw new ImmediateAcknowledgeAmqpException("Failed after 4 attempts");
+        }
+        throw new AmqpRejectAndDontRequeueException("failed");
+    }
+
+}

+

Notice that the count property in the x-death header is a Long.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__usage.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__usage.html new file mode 100644 index 00000000..02c16615 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi__usage.html @@ -0,0 +1,9 @@ + + + 1. Usage

1. Usage

To use the RabbitMQ binder, you can add it to your Spring Cloud Stream application, by using the following Maven coordinates:

<dependency>
+  <groupId>org.springframework.cloud</groupId>
+  <artifactId>spring-cloud-stream-binder-rabbit</artifactId>
+</dependency>

Alternatively, you can use the Spring Cloud Stream RabbitMQ Starter, as follows:

<dependency>
+  <groupId>org.springframework.cloud</groupId>
+  <artifactId>spring-cloud-starter-stream-rabbit</artifactId>
+</dependency>
\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_building.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_building.html new file mode 100644 index 00000000..94ad1b0a --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_building.html @@ -0,0 +1,33 @@ + + + Appendix A. Building

Appendix A. Building

A.1 Basic Compile and Test

To build the source you will need to install JDK 1.8.

The build uses the Maven wrapper so you don’t have to install a specific +version of Maven. To enable the tests, you should have RabbitMQ server running +on localhost and the default port (5672) +before building.

The main build command is

$ ./mvnw clean install

You can also add '-DskipTests' if you like, to avoid running the tests.

[Note]Note

You can also install Maven (>=3.3.3) yourself and run the mvn command +in place of ./mvnw in the examples below. If you do that you also +might need to add -P spring if your local Maven settings do not +contain repository declarations for spring pre-release artifacts.

[Note]Note

Be aware that you might need to increase the amount of memory +available to Maven by setting a MAVEN_OPTS environment variable with +a value like -Xmx512m -XX:MaxPermSize=128m. We try to cover this in +the .mvn configuration, so if you find you have to do it to make a +build succeed, please raise a ticket to get the settings added to +source control.

The projects that require middleware generally include a +docker-compose.yml, so consider using +Docker Compose to run the middeware servers +in Docker containers.

A.2 Documentation

There is a "full" profile that will generate documentation.

A.3 Working with the code

If you don’t have an IDE preference we would recommend that you use +Spring Tools Suite or +Eclipse when working with the code. We use the +m2eclipe eclipse plugin for maven support. Other IDEs and tools +should also work without issue.

A.3.1 Importing into eclipse with m2eclipse

We recommend the m2eclipe eclipse plugin when working with +eclipse. If you don’t already have m2eclipse installed it is available from the "eclipse +marketplace".

Unfortunately m2e does not yet support Maven 3.3, so once the projects +are imported into Eclipse you will also need to tell m2eclipse to use +the .settings.xml file for the projects. If you do not do this you +may see many different errors related to the POMs in the +projects. Open your Eclipse preferences, expand the Maven +preferences, and select User Settings. In the User Settings field +click Browse and navigate to the Spring Cloud project you imported +selecting the .settings.xml file in that project. Click Apply and +then OK to save the preference changes.

[Note]Note

Alternatively you can copy the repository settings from .settings.xml into your own ~/.m2/settings.xml.

A.3.2 Importing into eclipse without m2eclipse

If you prefer not to use m2eclipse you can generate eclipse project metadata using the +following command:

$ ./mvnw eclipse:eclipse

The generated eclipse projects can be imported by selecting import existing projects +from the file menu.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_contributing.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_contributing.html new file mode 100644 index 00000000..f208057b --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_contributing.html @@ -0,0 +1,26 @@ + + + Appendix B. Contributing

Appendix B. Contributing

Spring Cloud is released under the non-restrictive Apache 2.0 license, +and follows a very standard Github development process, using Github +tracker for issues and merging pull requests into master. If you want +to contribute even something trivial please do not hesitate, but +follow the guidelines below.

B.1 Sign the Contributor License Agreement

Before we accept a non-trivial patch or pull request we will need you to sign the +contributor’s agreement. +Signing the contributor’s agreement does not grant anyone commit rights to the main +repository, but it does mean that we can accept your contributions, and you will get an +author credit if we do. Active contributors might be asked to join the core team, and +given the ability to merge pull requests.

B.2 Code Conventions and Housekeeping

None of these is essential for a pull request, but they will all help. They can also be +added after the original pull request but before a merge.

  • Use the Spring Framework code format conventions. If you use Eclipse +you can import formatter settings using the +eclipse-code-formatter.xml file from the +Spring +Cloud Build project. If using IntelliJ, you can use the +Eclipse Code Formatter +Plugin to import the same file.
  • Make sure all new .java files to have a simple Javadoc class comment with at least an +@author tag identifying you, and preferably at least a paragraph on what the class is +for.
  • Add the ASF license header comment to all new .java files (copy from existing files +in the project)
  • Add yourself as an @author to the .java files that you modify substantially (more +than cosmetic changes).
  • Add some Javadocs and, if you change the namespace, some XSD doc elements.
  • A few unit tests would help a lot as well — someone has to do it.
  • If no-one else is using your branch, please rebase it against the current master (or +other target branch in the main project).
  • When writing a commit message please follow these conventions, +if you are fixing an existing issue please add Fixes gh-XXXX at the end of the commit +message (where XXXX is the issue number).
\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-dlq-processing.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-dlq-processing.html new file mode 100644 index 00000000..6b80a6f3 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-dlq-processing.html @@ -0,0 +1,204 @@ + + + 6. Dead-Letter Queue Processing

6. Dead-Letter Queue Processing

Because you cannot anticipate how users would want to dispose of dead-lettered messages, the framework does not provide any standard mechanism to handle them. +If the reason for the dead-lettering is transient, you may wish to route the messages back to the original queue. +However, if the problem is a permanent issue, that could cause an infinite loop. +The following Spring Boot application shows an example of how to route those messages back to the original queue but moves them to a third parking lot queue after three attempts. +The second example uses the RabbitMQ Delayed Message Exchange to introduce a delay to the re-queued message. +In this example, the delay increases for each attempt. +These examples use a @RabbitListener to receive messages from the DLQ. +You could also use RabbitTemplate.receive() in a batch process.

The examples assume the original destination is so8400in and the consumer group is so8400.

6.1 Non-Partitioned Destinations

The first two examples are for when the destination is not partitioned:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+    private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+    private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+    private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+    private static final String X_RETRIES_HEADER = "x-retries";
+
+    public static void main(String[] args) throws Exception {
+        ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+        System.out.println("Hit enter to terminate");
+        System.in.read();
+        context.close();
+    }
+
+    @Autowired
+    private RabbitTemplate rabbitTemplate;
+
+    @RabbitListener(queues = DLQ)
+    public void rePublish(Message failedMessage) {
+        Integer retriesHeader = (Integer) failedMessage.getMessageProperties().getHeaders().get(X_RETRIES_HEADER);
+        if (retriesHeader == null) {
+            retriesHeader = Integer.valueOf(0);
+        }
+        if (retriesHeader < 3) {
+            failedMessage.getMessageProperties().getHeaders().put(X_RETRIES_HEADER, retriesHeader + 1);
+            this.rabbitTemplate.send(ORIGINAL_QUEUE, failedMessage);
+        }
+        else {
+            this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+        }
+    }
+
+    @Bean
+    public Queue parkingLot() {
+        return new Queue(PARKING_LOT);
+    }
+
+}
@SpringBootApplication
+public class ReRouteDlqApplication {
+
+    private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+    private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+    private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+    private static final String X_RETRIES_HEADER = "x-retries";
+
+    private static final String DELAY_EXCHANGE = "dlqReRouter";
+
+    public static void main(String[] args) throws Exception {
+        ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+        System.out.println("Hit enter to terminate");
+        System.in.read();
+        context.close();
+    }
+
+    @Autowired
+    private RabbitTemplate rabbitTemplate;
+
+    @RabbitListener(queues = DLQ)
+    public void rePublish(Message failedMessage) {
+        Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+        Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+        if (retriesHeader == null) {
+            retriesHeader = Integer.valueOf(0);
+        }
+        if (retriesHeader < 3) {
+            headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+            headers.put("x-delay", 5000 * retriesHeader);
+            this.rabbitTemplate.send(DELAY_EXCHANGE, ORIGINAL_QUEUE, failedMessage);
+        }
+        else {
+            this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+        }
+    }
+
+    @Bean
+    public DirectExchange delayExchange() {
+        DirectExchange exchange = new DirectExchange(DELAY_EXCHANGE);
+        exchange.setDelayed(true);
+        return exchange;
+    }
+
+    @Bean
+    public Binding bindOriginalToDelay() {
+        return BindingBuilder.bind(new Queue(ORIGINAL_QUEUE)).to(delayExchange()).with(ORIGINAL_QUEUE);
+    }
+
+    @Bean
+    public Queue parkingLot() {
+        return new Queue(PARKING_LOT);
+    }
+
+}

6.2 Partitioned Destinations

With partitioned destinations, there is one DLQ for all partitions. We determine the original queue from the headers.

6.2.1 republishToDlq=false

When republishToDlq is false, RabbitMQ publishes the message to the DLX/DLQ with an x-death header containing information about the original destination, as shown in the following example:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+	private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+	private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+	private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+	private static final String X_DEATH_HEADER = "x-death";
+
+	private static final String X_RETRIES_HEADER = "x-retries";
+
+	public static void main(String[] args) throws Exception {
+		ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+		System.out.println("Hit enter to terminate");
+		System.in.read();
+		context.close();
+	}
+
+	@Autowired
+	private RabbitTemplate rabbitTemplate;
+
+	@SuppressWarnings("unchecked")
+	@RabbitListener(queues = DLQ)
+	public void rePublish(Message failedMessage) {
+		Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+		Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+		if (retriesHeader == null) {
+			retriesHeader = Integer.valueOf(0);
+		}
+		if (retriesHeader < 3) {
+			headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+			List<Map<String, ?>> xDeath = (List<Map<String, ?>>) headers.get(X_DEATH_HEADER);
+			String exchange = (String) xDeath.get(0).get("exchange");
+			List<String> routingKeys = (List<String>) xDeath.get(0).get("routing-keys");
+			this.rabbitTemplate.send(exchange, routingKeys.get(0), failedMessage);
+		}
+		else {
+			this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+		}
+	}
+
+	@Bean
+	public Queue parkingLot() {
+		return new Queue(PARKING_LOT);
+	}
+
+}

6.2.2 republishToDlq=true

When republishToDlq is true, the republishing recoverer adds the original exchange and routing key to headers, as shown in the following example:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+	private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+	private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+	private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+	private static final String X_RETRIES_HEADER = "x-retries";
+
+	private static final String X_ORIGINAL_EXCHANGE_HEADER = RepublishMessageRecoverer.X_ORIGINAL_EXCHANGE;
+
+	private static final String X_ORIGINAL_ROUTING_KEY_HEADER = RepublishMessageRecoverer.X_ORIGINAL_ROUTING_KEY;
+
+	public static void main(String[] args) throws Exception {
+		ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+		System.out.println("Hit enter to terminate");
+		System.in.read();
+		context.close();
+	}
+
+	@Autowired
+	private RabbitTemplate rabbitTemplate;
+
+	@RabbitListener(queues = DLQ)
+	public void rePublish(Message failedMessage) {
+		Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+		Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+		if (retriesHeader == null) {
+			retriesHeader = Integer.valueOf(0);
+		}
+		if (retriesHeader < 3) {
+			headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+			String exchange = (String) headers.get(X_ORIGINAL_EXCHANGE_HEADER);
+			String originalRoutingKey = (String) headers.get(X_ORIGINAL_ROUTING_KEY_HEADER);
+			this.rabbitTemplate.send(exchange, originalRoutingKey, failedMessage);
+		}
+		else {
+			this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+		}
+	}
+
+	@Bean
+	public Queue parkingLot() {
+		return new Queue(PARKING_LOT);
+	}
+
+}
\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-error-channels.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-error-channels.html new file mode 100644 index 00000000..87d8b8e0 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_rabbit-error-channels.html @@ -0,0 +1,6 @@ + + + 5. Error Channels

5. Error Channels

Starting with version 1.3, the binder unconditionally sends exceptions to an error channel for each consumer destination and can also be configured to send async producer send failures to an error channel. +See ??? for more information.

RabbitMQ has two types of send failures:

The latter is rare. +According to the RabbitMQ documentation "[A nack] will only be delivered if an internal error occurs in the Erlang process responsible for a queue.".

As well as enabling producer error channels (as described in ???), the RabbitMQ binder only sends messages to the channels if the connection factory is appropriately configured, as follows:

  • ccf.setPublisherConfirms(true);
  • ccf.setPublisherReturns(true);

When using Spring Boot configuration for the connection factory, set the following properties:

  • spring.rabbitmq.publisher-confirms
  • spring.rabbitmq.publisher-returns

The payload of the ErrorMessage for a returned message is a ReturnedAmqpMessageException with the following properties:

  • failedMessage: The spring-messaging Message<?> that failed to be sent.
  • amqpMessage: The raw spring-amqp Message.
  • replyCode: An integer value indicating the reason for the failure (for example, 312 - No route).
  • replyText: A text value indicating the reason for the failure (for example, NO_ROUTE).
  • exchange: The exchange to which the message was published.
  • routingKey: The routing key used when the message was published.

For negatively acknowledged confirmations, the payload is a NackedAmqpMessageException with the following properties:

  • failedMessage: The spring-messaging Message<?> that failed to be sent.
  • nackReason: A reason (if available — you may need to examine the broker logs for more information).

There is no automatic handling of these exceptions (such as sending to a dead-letter queue). +You can consume these exceptions with your own Spring Integration flow.

\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_spring-cloud-stream-binder-rabbit.html b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_spring-cloud-stream-binder-rabbit.html new file mode 100644 index 00000000..b1ab5714 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/multi/multi_spring-cloud-stream-binder-rabbit.html @@ -0,0 +1,3 @@ + + + Spring Cloud Stream RabbitMQ Binder Reference Guide \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/highlight.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/highlight.css new file mode 100644 index 00000000..ffefef72 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/highlight.css @@ -0,0 +1,35 @@ +/* + code highlight CSS resemblign the Eclipse IDE default color schema + @author Costin Leau +*/ + +.hl-keyword { + color: #7F0055; + font-weight: bold; +} + +.hl-comment { + color: #3F5F5F; + font-style: italic; +} + +.hl-multiline-comment { + color: #3F5FBF; + font-style: italic; +} + +.hl-tag { + color: #3F7F7F; +} + +.hl-attribute { + color: #7F007F; +} + +.hl-value { + color: #2A00FF; +} + +.hl-string { + color: #2A00FF; +} \ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-multipage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-multipage.css new file mode 100644 index 00000000..0c484531 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-multipage.css @@ -0,0 +1,9 @@ +@IMPORT url("manual.css"); + +body.firstpage { + background: url("../images/background.png") no-repeat center top; +} + +div.part h1 { + border-top: none; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-singlepage.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-singlepage.css new file mode 100644 index 00000000..4a7fd140 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual-singlepage.css @@ -0,0 +1,6 @@ +@IMPORT url("manual.css"); + +body { + background: url("../images/background.png") no-repeat center top; +} + diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual.css b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual.css new file mode 100644 index 00000000..0ecbe2e8 --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/css/manual.css @@ -0,0 +1,344 @@ +@IMPORT url("highlight.css"); + +html { + padding: 0pt; + margin: 0pt; +} + +body { + color: #333333; + margin: 15px 30px; + font-family: Helvetica, Arial, Freesans, Clean, Sans-serif; + line-height: 1.6; + -webkit-font-smoothing: antialiased; +} + +code { + font-size: 16px; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +:not(a)>code { + color: #6D180B; +} + +:not(pre)>code { + background-color: #F2F2F2; + border: 1px solid #CCCCCC; + border-radius: 4px; + padding: 1px 3px 0; + text-shadow: none; + white-space: nowrap; +} + +body>*:first-child { + margin-top: 0 !important; +} + +div { + margin: 0pt; +} + +hr { + border: 1px solid #CCCCCC; + background: #CCCCCC; +} + +h1,h2,h3,h4,h5,h6 { + color: #000000; + cursor: text; + font-weight: bold; + margin: 30px 0 10px; + padding: 0; +} + +h1,h2,h3 { + margin: 40px 0 10px; +} + +h1 { + margin: 70px 0 30px; + padding-top: 20px; +} + +div.part h1 { + border-top: 1px dotted #CCCCCC; +} + +h1,h1 code { + font-size: 32px; +} + +h2,h2 code { + font-size: 24px; +} + +h3,h3 code { + font-size: 20px; +} + +h4,h1 code,h5,h5 code,h6,h6 code { + font-size: 18px; +} + +div.book,div.chapter,div.appendix,div.part,div.preface { + min-width: 300px; + max-width: 1200px; + margin: 0 auto; +} + +p.releaseinfo { + font-weight: bold; + margin-bottom: 40px; + margin-top: 40px; +} + +div.authorgroup { + line-height: 1; +} + +p.copyright { + line-height: 1; + margin-bottom: -5px; +} + +.legalnotice p { + font-style: italic; + font-size: 14px; + line-height: 1; +} + +div.titlepage+p,div.titlepage+p { + margin-top: 0; +} + +pre { + line-height: 1.0; + color: black; +} + +a { + color: #4183C4; + text-decoration: none; +} + +p { + margin: 15px 0; + text-align: left; +} + +ul,ol { + padding-left: 30px; +} + +li p { + margin: 0; +} + +div.table { + margin: 1em; + padding: 0.5em; + text-align: center; +} + +div.table table,div.informaltable table { + display: table; + width: 100%; +} + +div.table td { + padding-left: 7px; + padding-right: 7px; +} + +.sidebar { + line-height: 1.4; + padding: 0 20px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; +} + +.sidebar p.title { + color: #6D180B; +} + +pre.programlisting,pre.screen { + font-size: 15px; + padding: 6px 10px; + background-color: #F8F8F8; + border: 1px solid #CCCCCC; + border-radius: 3px 3px 3px 3px; + clear: both; + overflow: auto; + line-height: 1.4; + font-family: Consolas, "Liberation Mono", Courier, monospace; +} + +table { + border-collapse: collapse; + border-spacing: 0; + border: 1px solid #DDDDDD !important; + border-radius: 4px !important; + border-collapse: separate !important; + line-height: 1.6; +} + +table thead { + background: #F5F5F5; +} + +table tr { + border: none; + border-bottom: none; +} + +table th { + font-weight: bold; +} + +table th,table td { + border: none !important; + padding: 6px 13px; +} + +table tr:nth-child(2n) { + background-color: #F8F8F8; +} + +td p { + margin: 0 0 15px 0; +} + +div.table-contents td p { + margin: 0; +} + +div.important *,div.note *,div.tip *,div.warning *,div.navheader *,div.navfooter *,div.calloutlist * + { + border: none !important; + background: none !important; + margin: 0; +} + +div.important p,div.note p,div.tip p,div.warning p { + color: #6F6F6F; + line-height: 1.6; +} + +div.important code,div.note code,div.tip code,div.warning code { + background-color: #F2F2F2 !important; + border: 1px solid #CCCCCC !important; + border-radius: 4px !important; + padding: 1px 3px 0 !important; + text-shadow: none !important; + white-space: nowrap !important; +} + +.note th,.tip th,.warning th { + display: none; +} + +.note tr:first-child td,.tip tr:first-child td,.warning tr:first-child td + { + border-right: 1px solid #CCCCCC !important; + padding-top: 10px; +} + +div.calloutlist p,div.calloutlist td { + padding: 0; + margin: 0; +} + +div.calloutlist>table>tbody>tr>td:first-child { + padding-left: 10px; + width: 30px !important; +} + +div.important,div.note,div.tip,div.warning { + margin-left: 0px !important; + margin-right: 20px !important; + margin-top: 20px; + margin-bottom: 20px; + padding-top: 10px; + padding-bottom: 10px; +} + +div.toc { + line-height: 1.2; +} + +dl,dt { + margin-top: 1px; + margin-bottom: 0; +} + +div.toc>dl>dt { + font-size: 32px; + font-weight: bold; + margin: 30px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dt { + font-size: 24px; + font-weight: bold; + margin: 20px 0 10px 0; + display: block; +} + +div.toc>dl>dd>dl>dd>dl>dt { + font-weight: bold; + font-size: 20px; + margin: 10px 0 0 0; +} + +tbody.footnotes * { + border: none !important; +} + +div.footnote p { + margin: 0; + line-height: 1; +} + +div.footnote p sup { + margin-right: 6px; + vertical-align: middle; +} + +div.navheader { + border-bottom: 1px solid #CCCCCC; +} + +div.navfooter { + border-top: 1px solid #CCCCCC; +} + +.title { + margin-left: -1em; + padding-left: 1em; +} + +.title>a { + position: absolute; + visibility: hidden; + display: block; + font-size: 0.85em; + margin-top: 0.05em; + margin-left: -1em; + vertical-align: text-top; + color: black; +} + +.title>a:before { + content: "\00A7"; +} + +.title:hover>a,.title>a:hover,.title:hover>a:hover { + visibility: visible; +} + +.title:focus>a,.title>a:focus,.title:focus>a:focus { + outline: 0; +} diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/background.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/background.png new file mode 100644 index 0000000000000000000000000000000000000000..15dca6fbe2669fae3609605e49c69cc414f1b6ed GIT binary patch literal 18255 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 EKU&lfD*ylh literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/1.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/1.png new file mode 100644 index 0000000000000000000000000000000000000000..7d473430b7bec514f7de12f5769fe7c5859e8c5d GIT binary patch literal 329 zcmeAS@N?(olHy`uVBq!ia0vp^JRr;gBp8b2n5}^nQC}X^4DKU-G|w_t}fLBA)Suv#nrW z!^h2QnY_`l!BOq-UXEX{m2up>JTQkX)2m zTvF+fTUlI^nXH#utd~++ke^qgmzgTe~DWM4ffP81J literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/2.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/callouts/2.png new file mode 100644 index 0000000000000000000000000000000000000000..5d09341b2f6d2ea2d1d5dad5d980f14b4b05dfd2 GIT binary patch literal 353 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 literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/logo.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..ade2ce6ed9d9e9f2f4d9c5729a252ee618a0a5a7 GIT binary patch literal 4387 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 FV1jwin)(0$ literal 0 HcmV?d00001 diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/warning.png b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/single/images/warning.png new file mode 100644 index 0000000000000000000000000000000000000000..0d5b5244605adbb7ab05a1549746a9c35490f95b GIT binary patch literal 2130 zcmbVNYg7|w8V(4q($)50y>JmGlLW#g$xLn}Vd2Si)}#|ptPAQp3Bp-3!-lL0;i^LY?;i#f0m5s49g z3h?3rDQg~EDPlm?FKkgKIcWEK-3X6YU0uzs7H|nq84s39r2!5;pF?SI$QqXy^Ko1x zV~zpENvp@<_Bsd`5MabC#3rvCq&$5dg43yGR zAdy0-rWjC#a1N_+kzUMY#pmogD7!DP(9dEKr3c5ngvUq_6>}Y+w-a81v=eSXnIi_+ zI?U>D1q2C!0zHox#XXKH+@|&rPT*OF5yvY$5J|)WwLqnU)c-5;=UChSlQkaY3@^|g z|J5#YBB}=i+n3Ex9bS$P?xJSKLk)-HHII@;3qG#TG^!+aj>0QkO~7kB0+|yM;YrGB zF>Ge@isQ#73%Tp#k_(tInh3U$uJWY-+DND*o}L-CB5g@#9YWWw~pxZ!Obm>yN#ZHFvL0zA3vNCTJ=t?)~`2O1M{L6un=ml<2x zQEYfifmA?Pz0tqRs^2Utt1pU0BQB1eIY0U_I}c1Y#PP7Ci5s7#IJn}xK{G+{_v^Z+1V#$Erodv>d}ew>RP0wzw+GWkGCBlS1Oj5Z!5NK zUuSR6>pa}RSEZ;qE_w1l#`hQX4E67U6NeP zHZ`T&wj0_H)t|Y1thUqnhub?%e)Y07;a^Tq%FO&iQkR&`qG!dh*2WfW(HuRQj*_DI zeCD1bUA|tMwjOb8E|FRIn$pzEU!2j_N}1Z2ig)vbr5xA0rjC70cmI6*%Uf->hWzaZ z{33HABO5q)vRhzDly2l691@+q3O{}NbSzx+I*k@|L4&3leK#$>u+T#TvTELfKb|IH zc23atf3vmfCa0&TWY@iuoA_Pm<0~ttEC*ut`isb^jXS>X^Sp=l?RXN_OJ*&usGXg$ zTlTv;Ch^vhLDf&+62jl64*&D+=+`sN_dap_1W%p|KQVYIdgPL5uRE9(c4On)DXndL zLNq?%!-u*zmMxyYc4m4Nr>>=A=s}PkJkqr&E^b9>5g+}c1%X}3|WKcg&spcJQ)05zI<<5LTBhNKRg$XRTi2{j)yl_ zj4~kKOvr+m(;vw~9zVh(zC-y9jqQnguz5r7FRq^MyKuXAENp(TAzUVtg&Ts+B_s7) zGn+fN0sG>9GjsPLKTRrvCd`71IZulJ1_1jO9KWbD&_@2UC+LTNpxxrdz#E|j|2nA9 zzB!UTyfAEJ&#%$pQ>QX#8vk@_a5iqukMF;8`wRKe{BI}5$H%OROs4J1#j)|?p|YSh z_SpR^e`VE#F52;WL{!+L(yZLRh40*KS;@box;9(-tE)`mcVp27O*>Z{_Lb*5T3cJA yr~0nPHtg2+UHi&&$8ha;`+hiaUmw&!n@8(?5PqF(KE5>Ym)EG)p&uyBP5%a8^# + + Spring Cloud Stream RabbitMQ Binder Reference Guide

Spring Cloud Stream RabbitMQ Binder Reference Guide

Sabby Anandan, Marius Bogoevici, Eric Bottard, Mark Fisher, Ilayaperumal Gopinathan, Gunnar Hillert, Mark Pollack, Patrick Peralta, Glenn Renfro, Thomas Risberg, Dave Syer, David Turanski, Janne Valkealahti, Benjamin Klein, Gary Russell, Jay Bryant

Part I. Reference Guide

This guide describes the RabbitMQ implementation of the Spring Cloud Stream Binder. +It contains information about its design, usage and configuration options, as well as information on how the Stream Cloud Stream concepts map into RabbitMQ specific constructs.

1. Usage

To use the RabbitMQ binder, you can add it to your Spring Cloud Stream application, by using the following Maven coordinates:

<dependency>
+  <groupId>org.springframework.cloud</groupId>
+  <artifactId>spring-cloud-stream-binder-rabbit</artifactId>
+</dependency>

Alternatively, you can use the Spring Cloud Stream RabbitMQ Starter, as follows:

<dependency>
+  <groupId>org.springframework.cloud</groupId>
+  <artifactId>spring-cloud-starter-stream-rabbit</artifactId>
+</dependency>

2. RabbitMQ Binder Overview

The following simplified diagram shows how the RabbitMQ binder operates:

Figure 2.1. RabbitMQ Binder

rabbit binder

By default, the RabbitMQ Binder implementation maps each destination to a TopicExchange. +For each consumer group, a Queue is bound to that TopicExchange. +Each consumer instance has a corresponding RabbitMQ Consumer instance for its group’s Queue. +For partitioned producers and consumers, the queues are suffixed with the partition index and use the partition index as the routing key. +For anonymous consumers (those with no group property), an auto-delete queue (with a randomized unique name) is used.

By using the optional autoBindDlq option, you can configure the binder to create and configure dead-letter queues (DLQs) (and a dead-letter exchange DLX, as well as routing infrastructure). +By default, the dead letter queue has the name of the destination, appended with .dlq. +If retry is enabled (maxAttempts > 1), failed messages are delivered to the DLQ after retries are exhausted. +If retry is disabled (maxAttempts = 1), you should set requeueRejected to false (the default) so that failed messages are routed to the DLQ, instead of being re-queued. +In addition, republishToDlq causes the binder to publish a failed message to the DLQ (instead of rejecting it). +This feature lets additional information (such as the stack trace in the x-exception-stacktrace header) be added to the message in headers. +See the frameMaxHeadroom property for information about truncated stack traces. +This option does not need retry enabled. +You can republish a failed message after just one attempt. +Starting with version 1.2, you can configure the delivery mode of republished messages. +See the republishDeliveryMode property.

If the stream listener throws an ImmediateAcknowledgeAmqpException, the DLQ is bypassed and the message simply discarded. +Starting with version 2.1, this is true regardless of the setting of republishToDlq; previously it was only the case when republishToDlq was false.

[Important]Important

Setting requeueRejected to true (with republishToDlq=false ) causes the message to be re-queued and redelivered continually, which is likely not what you want unless the reason for the failure is transient. +In general, you should enable retry within the binder by setting maxAttempts to greater than one or by setting republishToDlq to true.

See Section 3.1, “RabbitMQ Binder Properties” for more information about these properties.

The framework does not provide any standard mechanism to consume dead-letter messages (or to re-route them back to the primary queue). +Some options are described in Chapter 6, Dead-Letter Queue Processing.

[Note]Note

When multiple RabbitMQ binders are used in a Spring Cloud Stream application, it is important to disable 'RabbitAutoConfiguration' to avoid the same configuration from RabbitAutoConfiguration being applied to the two binders. +You can exclude the class by using the @SpringBootApplication annotation.

Starting with version 2.0, the RabbitMessageChannelBinder sets the RabbitTemplate.userPublisherConnection property to true so that the non-transactional producers avoid deadlocks on consumers, which can happen if cached connections are blocked because of a memory alarm on the broker.

[Note]Note

Currently, a multiplex consumer (a single consumer listening to multiple queues) is only supported for message-driven conssumers; polled consumers can only retrieve messages from a single queue.

3. Configuration Options

This section contains settings specific to the RabbitMQ Binder and bound channels.

For general binding configuration options and properties, see the Spring Cloud Stream core documentation.

3.1 RabbitMQ Binder Properties

By default, the RabbitMQ binder uses Spring Boot’s ConnectionFactory. +Conseuqently, it supports all Spring Boot configuration options for RabbitMQ. +(For reference, see the Spring Boot documentation). +RabbitMQ configuration options use the spring.rabbitmq prefix.

In addition to Spring Boot options, the RabbitMQ binder supports the following properties:

spring.cloud.stream.rabbit.binder.adminAddresses

A comma-separated list of RabbitMQ management plugin URLs. +Only used when nodes contains more than one entry. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information.

Default: empty.

spring.cloud.stream.rabbit.binder.nodes

A comma-separated list of RabbitMQ node names. +When more than one entry, used to locate the server address where a queue is located. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information.

Default: empty.

spring.cloud.stream.rabbit.binder.compressionLevel

The compression level for compressed bindings. +See java.util.zip.Deflater.

Default: 1 (BEST_LEVEL).

spring.cloud.stream.binder.connection-name-prefix

A connection name prefix used to name the connection(s) created by this binder. +The name is this prefix followed by #n, where n increments each time a new connection is opened.

Default: none (Spring AMQP default).

3.2 RabbitMQ Consumer Properties

The following properties are available for Rabbit consumers only and must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.consumer..

acknowledgeMode

The acknowledge mode.

Default: AUTO.

autoBindDlq

Whether to automatically declare the DLQ and bind it to the binder DLX.

Default: false.

bindingRoutingKey

The routing key with which to bind the queue to the exchange (if bindQueue is true). +For partitioned destinations, -<instanceIndex> is appended.

Default: #.

bindQueue

Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue.

Default: true.

consumerTagPrefix

Used to create the consumer tag(s); will be appended by #n where n increments for each consumer created. +Example: ${spring.application.name}-${spring.cloud.stream.bindings.input.group}-${spring.cloud.stream.instance-index}.

Default: none - the broker will generate random consumer tags.

deadLetterQueueName

The name of the DLQ

Default: prefix+destination.dlq

deadLetterExchange

A DLX to assign to the queue. +Relevant only if autoBindDlq is true.

Default: 'prefix+DLX'

deadLetterExchangeType

The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true.

Default: 'direct'

deadLetterRoutingKey

A dead letter routing key to assign to the queue. +Relevant only if autoBindDlq is true.

Default: destination

declareDlx

Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX.

Default: true.

declareExchange

Whether to declare the exchange for the destination.

Default: true.

delayedExchange

Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType.

Default: false.

dlqDeadLetterExchange

If a DLQ is declared, a DLX to assign to that queue.

Default: none

dlqDeadLetterRoutingKey

If a DLQ is declared, a dead letter routing key to assign to that queue.

Default: none

dlqExpires

How long before an unused dead letter queue is deleted (in milliseconds).

Default: no expiration

dlqLazy

Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue.

Default: false.

dlqMaxLength

Maximum number of messages in the dead letter queue.

Default: no limit

dlqMaxLengthBytes

Maximum number of total bytes in the dead letter queue from all messages.

Default: no limit

dlqMaxPriority

Maximum priority of messages in the dead letter queue (0-255).

Default: none

dlqOverflowBehavior

Action to take when dlqMaxLength or dlqMaxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation.

Default: none

dlqTtl

Default time to live to apply to the dead letter queue when declared (in milliseconds).

Default: no limit

durableSubscription

Whether the subscription should be durable. +Only effective if group is also set.

Default: true.

exchangeAutoDelete

If declareExchange is true, whether the exchange should be auto-deleted (that is, removed after the last queue is removed).

Default: true.

exchangeDurable

If declareExchange is true, whether the exchange should be durable (that is, it survives broker restart).

Default: true.

exchangeType

The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations.

Default: topic.

exclusive

Whether to create an exclusive consumer. +Concurrency should be 1 when this is true. +Often used when strict ordering is required but enabling a hot standby instance to take over after a failure. +See recoveryInterval, which controls how often a standby instance attempts to consume.

Default: false.

expires

How long before an unused queue is deleted (in milliseconds).

Default: no expiration

failedDeclarationRetryInterval

The interval (in milliseconds) between attempts to consume from a queue if it is missing.

Default: 5000

frameMaxHeadroom

The number of bytes to reserve for other headers when adding the stack trace to a DLQ message header. +All headers must fit within the frame_max size configured on the broker. +Stack traces can be large; if the size plus this property exceeds frame_max then the stack trace will be truncated. +A WARN log will be written; consider increasing the frame_max or reducing the stack trace by catching the exception and throwing one with a smaller stack trace.

Default: 20000

headerPatterns

Patterns for headers to be mapped from inbound messages.

Default: ['*'] (all headers).

lazy

Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue.

Default: false.

maxConcurrency

The maximum number of consumers.

Default: 1.

maxLength

The maximum number of messages in the queue.

Default: no limit

maxLengthBytes

The maximum number of total bytes in the queue from all messages.

Default: no limit

maxPriority

The maximum priority of messages in the queue (0-255).

Default: none

missingQueuesFatal

When the queue cannot be found, whether to treat the condition as fatal and stop the listener container. +Defaults to false so that the container keeps trying to consume from the queue — for example, when using a cluster and the node hosting a non-HA queue is down.

Default: false

overflowBehavior

Action to take when maxLength or maxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation.

Default: none

prefetch

Prefetch count.

Default: 1.

prefix

A prefix to be added to the name of the destination and queues.

Default: "".

queueDeclarationRetries

The number of times to retry consuming from a queue if it is missing. +Relevant only when missingQueuesFatal is true. +Otherwise, the container keeps retrying indefinitely.

Default: 3

queueNameGroupOnly

When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue.

Default: false.

recoveryInterval

The interval between connection recovery attempts, in milliseconds.

Default: 5000.

requeueRejected

Whether delivery failures should be re-queued when retry is disabled or republishToDlq is false.

Default: false.

republishDeliveryMode

When republishToDlq is true, specifies the delivery mode of the republished message.

Default: DeliveryMode.PERSISTENT

republishToDlq

By default, messages that fail after retries are exhausted are rejected. +If a dead-letter queue (DLQ) is configured, RabbitMQ routes the failed message (unchanged) to the DLQ. +If set to true, the binder republishs failed messages to the DLQ with additional headers, including the exception message and stack trace from the cause of the final failure. +Also see the frameMaxHeadroom property.

Default: false

transacted

Whether to use transacted channels.

Default: false.

ttl

Default time to live to apply to the queue when declared (in milliseconds).

Default: no limit

txSize

The number of deliveries between acks.

Default: 1.

3.3 Advanced Listener Container Configuration

To set listener container properties that are not exposed as binder or binding properties, add a single bean of type ListenerContainerCustomizer to the application context. +The binder and binding properties will be set and then the customizer will be called. +The customizer (configure() method) is provided with the queue name as well as the consumer group as arguments.

3.4 Rabbit Producer Properties

The following properties are available for Rabbit producers only and +must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.producer..

autoBindDlq

Whether to automatically declare the DLQ and bind it to the binder DLX.

Default: false.

batchingEnabled

Whether to enable message batching by producers. +Messages are batched into one message according to the following properties (described in the next three entries in this list): 'batchSize', batchBufferLimit, and batchTimeout. +See Batching for more information.

Default: false.

batchSize

The number of messages to buffer when batching is enabled.

Default: 100.

batchBufferLimit

The maximum buffer size when batching is enabled.

Default: 10000.

batchTimeout

The batch timeout when batching is enabled.

Default: 5000.

bindingRoutingKey

The routing key with which to bind the queue to the exchange (if bindQueue is true). +Only applies to non-partitioned destinations. +Only applies if requiredGroups are provided and then only to those groups.

Default: #.

bindQueue

Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue. +Only applies if requiredGroups are provided and then only to those groups.

Default: true.

compress

Whether data should be compressed when sent.

Default: false.

confirmAckChannel

When errorChannelEnabled is true, a channel to which to send positive delivery acknowledgments (aka publisher confirms). +If the channel does not exist, a DirectChannel is registered with this name. +The connection factory must be configured to enable publisher confirms.

Default: nullChannel (acks are discarded).

deadLetterQueueName

The name of the DLQ +Only applies if requiredGroups are provided and then only to those groups.

Default: prefix+destination.dlq

deadLetterExchange

A DLX to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: 'prefix+DLX'

deadLetterExchangeType

The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: 'direct'

deadLetterRoutingKey

A dead letter routing key to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups.

Default: destination

declareDlx

Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX. +Applies only when requiredGroups are provided and then only to those groups.

Default: true.

declareExchange

Whether to declare the exchange for the destination.

Default: true.

delayExpression

A SpEL expression to evaluate the delay to apply to the message (x-delay header). +It has no effect if the exchange is not a delayed message exchange.

Default: No x-delay header is set.

delayedExchange

Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType.

Default: false.

deliveryMode

The delivery mode.

Default: PERSISTENT.

dlqDeadLetterExchange

When a DLQ is declared, a DLX to assign to that queue. +Applies only if requiredGroups are provided and then only to those groups.

Default: none

dlqDeadLetterRoutingKey

When a DLQ is declared, a dead letter routing key to assign to that queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: none

dlqExpires

How long (in milliseconds) before an unused dead letter queue is deleted. +Applies only when requiredGroups are provided and then only to those groups.

Default: no expiration

dlqLazy
Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups.
dlqMaxLength

Maximum number of messages in the dead letter queue. +Applies only if requiredGroups are provided and then only to those groups.

Default: no limit

dlqMaxLengthBytes

Maximum number of total bytes in the dead letter queue from all messages. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

dlqMaxPriority

Maximum priority of messages in the dead letter queue (0-255) +Applies only when requiredGroups are provided and then only to those groups.

Default: none

dlqTtl

Default time (in milliseconds) to live to apply to the dead letter queue when declared. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

exchangeAutoDelete

If declareExchange is true, whether the exchange should be auto-delete (it is removed after the last queue is removed).

Default: true.

exchangeDurable

If declareExchange is true, whether the exchange should be durable (survives broker restart).

Default: true.

exchangeType

The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations.

Default: topic.

expires

How long (in milliseconds) before an unused queue is deleted. +Applies only when requiredGroups are provided and then only to those groups.

Default: no expiration

headerPatterns

Patterns for headers to be mapped to outbound messages.

Default: ['*'] (all headers).

lazy

Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: false.

maxLength

Maximum number of messages in the queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

maxLengthBytes

Maximum number of total bytes in the queue from all messages. +Only applies if requiredGroups are provided and then only to those groups.

Default: no limit

maxPriority

Maximum priority of messages in the queue (0-255). +Only applies if requiredGroups are provided and then only to those groups.

Default: none

prefix

A prefix to be added to the name of the destination exchange.

Default: "".

queueNameGroupOnly

When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue. +Applies only when requiredGroups are provided and then only to those groups.

Default: false.

routingKeyExpression

A SpEL expression to determine the routing key to use when publishing messages. +For a fixed routing key, use a literal expression, such as routingKeyExpression='my.routingKey' in a properties file or routingKeyExpression: '''my.routingKey''' in a YAML file.

Default: destination or destination-<partition> for partitioned destinations.

transacted

Whether to use transacted channels.

Default: false.

ttl

Default time (in milliseconds) to live to apply to the queue when declared. +Applies only when requiredGroups are provided and then only to those groups.

Default: no limit

[Note]Note

In the case of RabbitMQ, content type headers can be set by external applications. +Spring Cloud Stream supports them as part of an extended internal protocol used for any type of transport — including transports, such as Kafka (prior to 0.11), that do not natively support headers.

4. Retry With the RabbitMQ Binder

When retry is enabled within the binder, the listener container thread is suspended for any back off periods that are configured. +This might be important when strict ordering is required with a single consumer. However, for other use cases, it prevents other messages from being processed on that thread. +An alternative to using binder retry is to set up dead lettering with time to live on the dead-letter queue (DLQ) as well as dead-letter configuration on the DLQ itself. +See Section 3.1, “RabbitMQ Binder Properties” for more information about the properties discussed here. +You can use the following example configuration to enable this feature:

  • Set autoBindDlq to true. +The binder create a DLQ. +Optionally, you can specify a name in deadLetterQueueName.
  • Set dlqTtl to the back off time you want to wait between redeliveries.
  • Set the dlqDeadLetterExchange to the default exchange. +Expired messages from the DLQ are routed to the original queue, because the default deadLetterRoutingKey is the queue name (destination.group). +Setting to the default exchange is achieved by setting the property with no value, as shown in the next example.

To force a message to be dead-lettered, either throw an AmqpRejectAndDontRequeueException or set requeueRejected to true (the default) and throw any exception.

The loop continue without end, which is fine for transient problems, but you may want to give up after some number of attempts. +Fortunately, RabbitMQ provides the x-death header, which lets you determine how many cycles have occurred.

To acknowledge a message after giving up, throw an ImmediateAcknowledgeAmqpException.

4.1 Putting it All Together

The following configuration creates an exchange myDestination with queue myDestination.consumerGroup bound to a topic exchange with a wildcard routing key #:

---
+spring.cloud.stream.bindings.input.destination=myDestination
+spring.cloud.stream.bindings.input.group=consumerGroup
+#disable binder retries
+spring.cloud.stream.bindings.input.consumer.max-attempts=1
+#dlx/dlq setup
+spring.cloud.stream.rabbit.bindings.input.consumer.auto-bind-dlq=true
+spring.cloud.stream.rabbit.bindings.input.consumer.dlq-ttl=5000
+spring.cloud.stream.rabbit.bindings.input.consumer.dlq-dead-letter-exchange=
+---

This configuration creates a DLQ bound to a direct exchange (DLX) with a routing key of myDestination.consumerGroup. +When messages are rejected, they are routed to the DLQ. +After 5 seconds, the message expires and is routed to the original queue by using the queue name as the routing key, as shown in the following example:

Spring Boot application.  +

@SpringBootApplication
+@EnableBinding(Sink.class)
+public class XDeathApplication {
+
+    public static void main(String[] args) {
+        SpringApplication.run(XDeathApplication.class, args);
+    }
+
+    @StreamListener(Sink.INPUT)
+    public void listen(String in, @Header(name = "x-death", required = false) Map<?,?> death) {
+        if (death != null && death.get("count").equals(3L)) {
+            // giving up - don't send to DLX
+            throw new ImmediateAcknowledgeAmqpException("Failed after 4 attempts");
+        }
+        throw new AmqpRejectAndDontRequeueException("failed");
+    }
+
+}

+

Notice that the count property in the x-death header is a Long.

5. Error Channels

Starting with version 1.3, the binder unconditionally sends exceptions to an error channel for each consumer destination and can also be configured to send async producer send failures to an error channel. +See ??? for more information.

RabbitMQ has two types of send failures:

The latter is rare. +According to the RabbitMQ documentation "[A nack] will only be delivered if an internal error occurs in the Erlang process responsible for a queue.".

As well as enabling producer error channels (as described in ???), the RabbitMQ binder only sends messages to the channels if the connection factory is appropriately configured, as follows:

  • ccf.setPublisherConfirms(true);
  • ccf.setPublisherReturns(true);

When using Spring Boot configuration for the connection factory, set the following properties:

  • spring.rabbitmq.publisher-confirms
  • spring.rabbitmq.publisher-returns

The payload of the ErrorMessage for a returned message is a ReturnedAmqpMessageException with the following properties:

  • failedMessage: The spring-messaging Message<?> that failed to be sent.
  • amqpMessage: The raw spring-amqp Message.
  • replyCode: An integer value indicating the reason for the failure (for example, 312 - No route).
  • replyText: A text value indicating the reason for the failure (for example, NO_ROUTE).
  • exchange: The exchange to which the message was published.
  • routingKey: The routing key used when the message was published.

For negatively acknowledged confirmations, the payload is a NackedAmqpMessageException with the following properties:

  • failedMessage: The spring-messaging Message<?> that failed to be sent.
  • nackReason: A reason (if available — you may need to examine the broker logs for more information).

There is no automatic handling of these exceptions (such as sending to a dead-letter queue). +You can consume these exceptions with your own Spring Integration flow.

6. Dead-Letter Queue Processing

Because you cannot anticipate how users would want to dispose of dead-lettered messages, the framework does not provide any standard mechanism to handle them. +If the reason for the dead-lettering is transient, you may wish to route the messages back to the original queue. +However, if the problem is a permanent issue, that could cause an infinite loop. +The following Spring Boot application shows an example of how to route those messages back to the original queue but moves them to a third parking lot queue after three attempts. +The second example uses the RabbitMQ Delayed Message Exchange to introduce a delay to the re-queued message. +In this example, the delay increases for each attempt. +These examples use a @RabbitListener to receive messages from the DLQ. +You could also use RabbitTemplate.receive() in a batch process.

The examples assume the original destination is so8400in and the consumer group is so8400.

6.1 Non-Partitioned Destinations

The first two examples are for when the destination is not partitioned:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+    private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+    private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+    private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+    private static final String X_RETRIES_HEADER = "x-retries";
+
+    public static void main(String[] args) throws Exception {
+        ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+        System.out.println("Hit enter to terminate");
+        System.in.read();
+        context.close();
+    }
+
+    @Autowired
+    private RabbitTemplate rabbitTemplate;
+
+    @RabbitListener(queues = DLQ)
+    public void rePublish(Message failedMessage) {
+        Integer retriesHeader = (Integer) failedMessage.getMessageProperties().getHeaders().get(X_RETRIES_HEADER);
+        if (retriesHeader == null) {
+            retriesHeader = Integer.valueOf(0);
+        }
+        if (retriesHeader < 3) {
+            failedMessage.getMessageProperties().getHeaders().put(X_RETRIES_HEADER, retriesHeader + 1);
+            this.rabbitTemplate.send(ORIGINAL_QUEUE, failedMessage);
+        }
+        else {
+            this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+        }
+    }
+
+    @Bean
+    public Queue parkingLot() {
+        return new Queue(PARKING_LOT);
+    }
+
+}
@SpringBootApplication
+public class ReRouteDlqApplication {
+
+    private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+    private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+    private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+    private static final String X_RETRIES_HEADER = "x-retries";
+
+    private static final String DELAY_EXCHANGE = "dlqReRouter";
+
+    public static void main(String[] args) throws Exception {
+        ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+        System.out.println("Hit enter to terminate");
+        System.in.read();
+        context.close();
+    }
+
+    @Autowired
+    private RabbitTemplate rabbitTemplate;
+
+    @RabbitListener(queues = DLQ)
+    public void rePublish(Message failedMessage) {
+        Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+        Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+        if (retriesHeader == null) {
+            retriesHeader = Integer.valueOf(0);
+        }
+        if (retriesHeader < 3) {
+            headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+            headers.put("x-delay", 5000 * retriesHeader);
+            this.rabbitTemplate.send(DELAY_EXCHANGE, ORIGINAL_QUEUE, failedMessage);
+        }
+        else {
+            this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+        }
+    }
+
+    @Bean
+    public DirectExchange delayExchange() {
+        DirectExchange exchange = new DirectExchange(DELAY_EXCHANGE);
+        exchange.setDelayed(true);
+        return exchange;
+    }
+
+    @Bean
+    public Binding bindOriginalToDelay() {
+        return BindingBuilder.bind(new Queue(ORIGINAL_QUEUE)).to(delayExchange()).with(ORIGINAL_QUEUE);
+    }
+
+    @Bean
+    public Queue parkingLot() {
+        return new Queue(PARKING_LOT);
+    }
+
+}

6.2 Partitioned Destinations

With partitioned destinations, there is one DLQ for all partitions. We determine the original queue from the headers.

6.2.1 republishToDlq=false

When republishToDlq is false, RabbitMQ publishes the message to the DLX/DLQ with an x-death header containing information about the original destination, as shown in the following example:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+	private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+	private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+	private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+	private static final String X_DEATH_HEADER = "x-death";
+
+	private static final String X_RETRIES_HEADER = "x-retries";
+
+	public static void main(String[] args) throws Exception {
+		ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+		System.out.println("Hit enter to terminate");
+		System.in.read();
+		context.close();
+	}
+
+	@Autowired
+	private RabbitTemplate rabbitTemplate;
+
+	@SuppressWarnings("unchecked")
+	@RabbitListener(queues = DLQ)
+	public void rePublish(Message failedMessage) {
+		Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+		Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+		if (retriesHeader == null) {
+			retriesHeader = Integer.valueOf(0);
+		}
+		if (retriesHeader < 3) {
+			headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+			List<Map<String, ?>> xDeath = (List<Map<String, ?>>) headers.get(X_DEATH_HEADER);
+			String exchange = (String) xDeath.get(0).get("exchange");
+			List<String> routingKeys = (List<String>) xDeath.get(0).get("routing-keys");
+			this.rabbitTemplate.send(exchange, routingKeys.get(0), failedMessage);
+		}
+		else {
+			this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+		}
+	}
+
+	@Bean
+	public Queue parkingLot() {
+		return new Queue(PARKING_LOT);
+	}
+
+}

6.2.2 republishToDlq=true

When republishToDlq is true, the republishing recoverer adds the original exchange and routing key to headers, as shown in the following example:

@SpringBootApplication
+public class ReRouteDlqApplication {
+
+	private static final String ORIGINAL_QUEUE = "so8400in.so8400";
+
+	private static final String DLQ = ORIGINAL_QUEUE + ".dlq";
+
+	private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot";
+
+	private static final String X_RETRIES_HEADER = "x-retries";
+
+	private static final String X_ORIGINAL_EXCHANGE_HEADER = RepublishMessageRecoverer.X_ORIGINAL_EXCHANGE;
+
+	private static final String X_ORIGINAL_ROUTING_KEY_HEADER = RepublishMessageRecoverer.X_ORIGINAL_ROUTING_KEY;
+
+	public static void main(String[] args) throws Exception {
+		ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args);
+		System.out.println("Hit enter to terminate");
+		System.in.read();
+		context.close();
+	}
+
+	@Autowired
+	private RabbitTemplate rabbitTemplate;
+
+	@RabbitListener(queues = DLQ)
+	public void rePublish(Message failedMessage) {
+		Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders();
+		Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER);
+		if (retriesHeader == null) {
+			retriesHeader = Integer.valueOf(0);
+		}
+		if (retriesHeader < 3) {
+			headers.put(X_RETRIES_HEADER, retriesHeader + 1);
+			String exchange = (String) headers.get(X_ORIGINAL_EXCHANGE_HEADER);
+			String originalRoutingKey = (String) headers.get(X_ORIGINAL_ROUTING_KEY_HEADER);
+			this.rabbitTemplate.send(exchange, originalRoutingKey, failedMessage);
+		}
+		else {
+			this.rabbitTemplate.send(PARKING_LOT, failedMessage);
+		}
+	}
+
+	@Bean
+	public Queue parkingLot() {
+		return new Queue(PARKING_LOT);
+	}
+
+}

7. Partitioning with the RabbitMQ Binder

RabbitMQ does not support partitioning natively.

Sometimes, it is advantageous to send data to specific partitions — for example, when you want to strictly order message processing, all messages for a particular customer should go to the same partition.

The RabbitMessageChannelBinder provides partitioning by binding a queue for each partition to the destination exchange.

The following Java and YAML examples show how to configure the producer:

Producer.  +

@SpringBootApplication
+@EnableBinding(Source.class)
+public class RabbitPartitionProducerApplication {
+
+    private static final Random RANDOM = new Random(System.currentTimeMillis());
+
+    private static final String[] data = new String[] {
+            "abc1", "def1", "qux1",
+            "abc2", "def2", "qux2",
+            "abc3", "def3", "qux3",
+            "abc4", "def4", "qux4",
+            };
+
+    public static void main(String[] args) {
+        new SpringApplicationBuilder(RabbitPartitionProducerApplication.class)
+            .web(false)
+            .run(args);
+    }
+
+    @InboundChannelAdapter(channel = Source.OUTPUT, poller = @Poller(fixedRate = "5000"))
+    public Message<?> generate() {
+        String value = data[RANDOM.nextInt(data.length)];
+        System.out.println("Sending: " + value);
+        return MessageBuilder.withPayload(value)
+                .setHeader("partitionKey", value)
+                .build();
+    }
+
+}

+

application.yml.  +

    spring:
+      cloud:
+        stream:
+          bindings:
+            output:
+              destination: partitioned.destination
+              producer:
+                partitioned: true
+                partition-key-expression: headers['partitionKey']
+                partition-count: 2
+                required-groups:
+                - myGroup

+

[Note]Note

The configuration in the prececing example uses the default partitioning (key.hashCode() % partitionCount). +This may or may not provide a suitably balanced algorithm, depending on the key values. +You can override this default by using the partitionSelectorExpression or partitionSelectorClass properties.

The required-groups property is required only if you need the consumer queues to be provisioned when the producer is deployed. +Otherwise, any messages sent to a partition are lost until the corresponding consumer is deployed.

The following configuration provisions a topic exchange:

part exchange

The following queues are bound to that exchange:

part queues

The following bindings associate the queues to the exchange:

part bindings

The following Java and YAML examples continue the previous examples and show how to configure the consumer:

Consumer.  +

@SpringBootApplication
+@EnableBinding(Sink.class)
+public class RabbitPartitionConsumerApplication {
+
+    public static void main(String[] args) {
+        new SpringApplicationBuilder(RabbitPartitionConsumerApplication.class)
+            .web(false)
+            .run(args);
+    }
+
+    @StreamListener(Sink.INPUT)
+    public void listen(@Payload String in, @Header(AmqpHeaders.CONSUMER_QUEUE) String queue) {
+        System.out.println(in + " received from queue " + queue);
+    }
+
+}

+

application.yml.  +

    spring:
+      cloud:
+        stream:
+          bindings:
+            input:
+              destination: partitioned.destination
+              group: myGroup
+              consumer:
+                partitioned: true
+                instance-index: 0

+

[Important]Important

The RabbitMessageChannelBinder does not support dynamic scaling. +There must be at least one consumer per partition. +The consumer’s instanceIndex is used to indicate which partition is consumed. +Platforms such as Cloud Foundry can have only one instance with an instanceIndex.

Part II. Appendices

Appendix A. Building

A.1 Basic Compile and Test

To build the source you will need to install JDK 1.8.

The build uses the Maven wrapper so you don’t have to install a specific +version of Maven. To enable the tests, you should have RabbitMQ server running +on localhost and the default port (5672) +before building.

The main build command is

$ ./mvnw clean install

You can also add '-DskipTests' if you like, to avoid running the tests.

[Note]Note

You can also install Maven (>=3.3.3) yourself and run the mvn command +in place of ./mvnw in the examples below. If you do that you also +might need to add -P spring if your local Maven settings do not +contain repository declarations for spring pre-release artifacts.

[Note]Note

Be aware that you might need to increase the amount of memory +available to Maven by setting a MAVEN_OPTS environment variable with +a value like -Xmx512m -XX:MaxPermSize=128m. We try to cover this in +the .mvn configuration, so if you find you have to do it to make a +build succeed, please raise a ticket to get the settings added to +source control.

The projects that require middleware generally include a +docker-compose.yml, so consider using +Docker Compose to run the middeware servers +in Docker containers.

A.2 Documentation

There is a "full" profile that will generate documentation.

A.3 Working with the code

If you don’t have an IDE preference we would recommend that you use +Spring Tools Suite or +Eclipse when working with the code. We use the +m2eclipe eclipse plugin for maven support. Other IDEs and tools +should also work without issue.

A.3.1 Importing into eclipse with m2eclipse

We recommend the m2eclipe eclipse plugin when working with +eclipse. If you don’t already have m2eclipse installed it is available from the "eclipse +marketplace".

Unfortunately m2e does not yet support Maven 3.3, so once the projects +are imported into Eclipse you will also need to tell m2eclipse to use +the .settings.xml file for the projects. If you do not do this you +may see many different errors related to the POMs in the +projects. Open your Eclipse preferences, expand the Maven +preferences, and select User Settings. In the User Settings field +click Browse and navigate to the Spring Cloud project you imported +selecting the .settings.xml file in that project. Click Apply and +then OK to save the preference changes.

[Note]Note

Alternatively you can copy the repository settings from .settings.xml into your own ~/.m2/settings.xml.

A.3.2 Importing into eclipse without m2eclipse

If you prefer not to use m2eclipse you can generate eclipse project metadata using the +following command:

$ ./mvnw eclipse:eclipse

The generated eclipse projects can be imported by selecting import existing projects +from the file menu.

Appendix B. Contributing

Spring Cloud is released under the non-restrictive Apache 2.0 license, +and follows a very standard Github development process, using Github +tracker for issues and merging pull requests into master. If you want +to contribute even something trivial please do not hesitate, but +follow the guidelines below.

B.1 Sign the Contributor License Agreement

Before we accept a non-trivial patch or pull request we will need you to sign the +contributor’s agreement. +Signing the contributor’s agreement does not grant anyone commit rights to the main +repository, but it does mean that we can accept your contributions, and you will get an +author credit if we do. Active contributors might be asked to join the core team, and +given the ability to merge pull requests.

B.2 Code Conventions and Housekeeping

None of these is essential for a pull request, but they will all help. They can also be +added after the original pull request but before a merge.

  • Use the Spring Framework code format conventions. If you use Eclipse +you can import formatter settings using the +eclipse-code-formatter.xml file from the +Spring +Cloud Build project. If using IntelliJ, you can use the +Eclipse Code Formatter +Plugin to import the same file.
  • Make sure all new .java files to have a simple Javadoc class comment with at least an +@author tag identifying you, and preferably at least a paragraph on what the class is +for.
  • Add the ASF license header comment to all new .java files (copy from existing files +in the project)
  • Add yourself as an @author to the .java files that you modify substantially (more +than cosmetic changes).
  • Add some Javadocs and, if you change the namespace, some XSD doc elements.
  • A few unit tests would help a lot as well — someone has to do it.
  • If no-one else is using your branch, please rebase it against the current master (or +other target branch in the main project).
  • When writing a commit message please follow these conventions, +if you are fixing an existing issue please add Fixes gh-XXXX at the end of the commit +message (where XXXX is the issue number).
\ No newline at end of file diff --git a/spring-cloud-stream-binder-rabbit/2.1.0.RC1/spring-cloud-stream-binder-rabbit.xml b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/spring-cloud-stream-binder-rabbit.xml new file mode 100644 index 00000000..be86890b --- /dev/null +++ b/spring-cloud-stream-binder-rabbit/2.1.0.RC1/spring-cloud-stream-binder-rabbit.xml @@ -0,0 +1,1477 @@ + + + + + +Spring Cloud Stream RabbitMQ Binder Reference Guide +2018-10-30 + + +Sabby Anandan, Marius Bogoevici, Eric Bottard, Mark Fisher, Ilayaperumal Gopinathan, Gunnar Hillert, Mark Pollack, Patrick Peralta, Glenn Renfro, Thomas Risberg, Dave Syer, David Turanski, Janne Valkealahti, Benjamin Klein, Gary Russell, Jay Bryant + + +S + + +Reference Guide + +This guide describes the RabbitMQ implementation of the Spring Cloud Stream Binder. +It contains information about its design, usage and configuration options, as well as information on how the Stream Cloud Stream concepts map into RabbitMQ specific constructs. + + +Usage +To use the RabbitMQ binder, you can add it to your Spring Cloud Stream application, by using the following Maven coordinates: +<dependency> + <groupId>org.springframework.cloud</groupId> + <artifactId>spring-cloud-stream-binder-rabbit</artifactId> +</dependency> +Alternatively, you can use the Spring Cloud Stream RabbitMQ Starter, as follows: +<dependency> + <groupId>org.springframework.cloud</groupId> + <artifactId>spring-cloud-starter-stream-rabbit</artifactId> +</dependency> + + +RabbitMQ Binder Overview +The following simplified diagram shows how the RabbitMQ binder operates: +
+RabbitMQ Binder + + + + +rabbit binder + +
+By default, the RabbitMQ Binder implementation maps each destination to a TopicExchange. +For each consumer group, a Queue is bound to that TopicExchange. +Each consumer instance has a corresponding RabbitMQ Consumer instance for its group’s Queue. +For partitioned producers and consumers, the queues are suffixed with the partition index and use the partition index as the routing key. +For anonymous consumers (those with no group property), an auto-delete queue (with a randomized unique name) is used. +By using the optional autoBindDlq option, you can configure the binder to create and configure dead-letter queues (DLQs) (and a dead-letter exchange DLX, as well as routing infrastructure). +By default, the dead letter queue has the name of the destination, appended with .dlq. +If retry is enabled (maxAttempts > 1), failed messages are delivered to the DLQ after retries are exhausted. +If retry is disabled (maxAttempts = 1), you should set requeueRejected to false (the default) so that failed messages are routed to the DLQ, instead of being re-queued. +In addition, republishToDlq causes the binder to publish a failed message to the DLQ (instead of rejecting it). +This feature lets additional information (such as the stack trace in the x-exception-stacktrace header) be added to the message in headers. +See the frameMaxHeadroom property for information about truncated stack traces. +This option does not need retry enabled. +You can republish a failed message after just one attempt. +Starting with version 1.2, you can configure the delivery mode of republished messages. +See the republishDeliveryMode property. +If the stream listener throws an ImmediateAcknowledgeAmqpException, the DLQ is bypassed and the message simply discarded. +Starting with version 2.1, this is true regardless of the setting of republishToDlq; previously it was only the case when republishToDlq was false. + +Setting requeueRejected to true (with republishToDlq=false ) causes the message to be re-queued and redelivered continually, which is likely not what you want unless the reason for the failure is transient. +In general, you should enable retry within the binder by setting maxAttempts to greater than one or by setting republishToDlq to true. + +See for more information about these properties. +The framework does not provide any standard mechanism to consume dead-letter messages (or to re-route them back to the primary queue). +Some options are described in . + +When multiple RabbitMQ binders are used in a Spring Cloud Stream application, it is important to disable 'RabbitAutoConfiguration' to avoid the same configuration from RabbitAutoConfiguration being applied to the two binders. +You can exclude the class by using the @SpringBootApplication annotation. + +Starting with version 2.0, the RabbitMessageChannelBinder sets the RabbitTemplate.userPublisherConnection property to true so that the non-transactional producers avoid deadlocks on consumers, which can happen if cached connections are blocked because of a memory alarm on the broker. + +Currently, a multiplex consumer (a single consumer listening to multiple queues) is only supported for message-driven conssumers; polled consumers can only retrieve messages from a single queue. + +
+ +Configuration Options +This section contains settings specific to the RabbitMQ Binder and bound channels. +For general binding configuration options and properties, see the Spring Cloud Stream core documentation. +
+RabbitMQ Binder Properties +By default, the RabbitMQ binder uses Spring Boot’s ConnectionFactory. +Conseuqently, it supports all Spring Boot configuration options for RabbitMQ. +(For reference, see the Spring Boot documentation). +RabbitMQ configuration options use the spring.rabbitmq prefix. +In addition to Spring Boot options, the RabbitMQ binder supports the following properties: + + +spring.cloud.stream.rabbit.binder.adminAddresses + +A comma-separated list of RabbitMQ management plugin URLs. +Only used when nodes contains more than one entry. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information. +Default: empty. + + + +spring.cloud.stream.rabbit.binder.nodes + +A comma-separated list of RabbitMQ node names. +When more than one entry, used to locate the server address where a queue is located. +Each entry in this list must have a corresponding entry in spring.rabbitmq.addresses. +Only needed if you use a RabbitMQ cluster and wish to consume from the node that hosts the queue. +See Queue Affinity and the LocalizedQueueConnectionFactory for more information. +Default: empty. + + + +spring.cloud.stream.rabbit.binder.compressionLevel + +The compression level for compressed bindings. +See java.util.zip.Deflater. +Default: 1 (BEST_LEVEL). + + + +spring.cloud.stream.binder.connection-name-prefix + +A connection name prefix used to name the connection(s) created by this binder. +The name is this prefix followed by #n, where n increments each time a new connection is opened. +Default: none (Spring AMQP default). + + + +
+
+RabbitMQ Consumer Properties +The following properties are available for Rabbit consumers only and must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.consumer.. + + +acknowledgeMode + +The acknowledge mode. +Default: AUTO. + + + +autoBindDlq + +Whether to automatically declare the DLQ and bind it to the binder DLX. +Default: false. + + + +bindingRoutingKey + +The routing key with which to bind the queue to the exchange (if bindQueue is true). +For partitioned destinations, -<instanceIndex> is appended. +Default: #. + + + +bindQueue + +Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue. +Default: true. + + + +consumerTagPrefix + +Used to create the consumer tag(s); will be appended by #n where n increments for each consumer created. +Example: ${spring.application.name}-${spring.cloud.stream.bindings.input.group}-${spring.cloud.stream.instance-index}. +Default: none - the broker will generate random consumer tags. + + + +deadLetterQueueName + +The name of the DLQ +Default: prefix+destination.dlq + + + +deadLetterExchange + +A DLX to assign to the queue. +Relevant only if autoBindDlq is true. +Default: 'prefix+DLX' + + + +deadLetterExchangeType + +The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true. +Default: 'direct' + + + +deadLetterRoutingKey + +A dead letter routing key to assign to the queue. +Relevant only if autoBindDlq is true. +Default: destination + + + +declareDlx + +Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX. +Default: true. + + + +declareExchange + +Whether to declare the exchange for the destination. +Default: true. + + + +delayedExchange + +Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType. +Default: false. + + + +dlqDeadLetterExchange + +If a DLQ is declared, a DLX to assign to that queue. +Default: none + + + +dlqDeadLetterRoutingKey + +If a DLQ is declared, a dead letter routing key to assign to that queue. +Default: none + + + +dlqExpires + +How long before an unused dead letter queue is deleted (in milliseconds). +Default: no expiration + + + +dlqLazy + +Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Default: false. + + + +dlqMaxLength + +Maximum number of messages in the dead letter queue. +Default: no limit + + + +dlqMaxLengthBytes + +Maximum number of total bytes in the dead letter queue from all messages. +Default: no limit + + + +dlqMaxPriority + +Maximum priority of messages in the dead letter queue (0-255). +Default: none + + + +dlqOverflowBehavior + +Action to take when dlqMaxLength or dlqMaxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation. +Default: none + + + +dlqTtl + +Default time to live to apply to the dead letter queue when declared (in milliseconds). +Default: no limit + + + +durableSubscription + +Whether the subscription should be durable. +Only effective if group is also set. +Default: true. + + + +exchangeAutoDelete + +If declareExchange is true, whether the exchange should be auto-deleted (that is, removed after the last queue is removed). +Default: true. + + + +exchangeDurable + +If declareExchange is true, whether the exchange should be durable (that is, it survives broker restart). +Default: true. + + + +exchangeType + +The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations. +Default: topic. + + + +exclusive + +Whether to create an exclusive consumer. +Concurrency should be 1 when this is true. +Often used when strict ordering is required but enabling a hot standby instance to take over after a failure. +See recoveryInterval, which controls how often a standby instance attempts to consume. +Default: false. + + + +expires + +How long before an unused queue is deleted (in milliseconds). +Default: no expiration + + + +failedDeclarationRetryInterval + +The interval (in milliseconds) between attempts to consume from a queue if it is missing. +Default: 5000 + + + + + +frameMaxHeadroom + +The number of bytes to reserve for other headers when adding the stack trace to a DLQ message header. +All headers must fit within the frame_max size configured on the broker. +Stack traces can be large; if the size plus this property exceeds frame_max then the stack trace will be truncated. +A WARN log will be written; consider increasing the frame_max or reducing the stack trace by catching the exception and throwing one with a smaller stack trace. +Default: 20000 + + + +headerPatterns + +Patterns for headers to be mapped from inbound messages. +Default: ['*'] (all headers). + + + +lazy + +Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Default: false. + + + +maxConcurrency + +The maximum number of consumers. +Default: 1. + + + +maxLength + +The maximum number of messages in the queue. +Default: no limit + + + +maxLengthBytes + +The maximum number of total bytes in the queue from all messages. +Default: no limit + + + +maxPriority + +The maximum priority of messages in the queue (0-255). +Default: none + + + +missingQueuesFatal + +When the queue cannot be found, whether to treat the condition as fatal and stop the listener container. +Defaults to false so that the container keeps trying to consume from the queue — for example, when using a cluster and the node hosting a non-HA queue is down. +Default: false + + + +overflowBehavior + +Action to take when maxLength or maxLengthBytes is exceeded; currently drop-head or reject-publish but refer to the RabbitMQ documentation. +Default: none + + + +prefetch + +Prefetch count. +Default: 1. + + + +prefix + +A prefix to be added to the name of the destination and queues. +Default: "". + + + +queueDeclarationRetries + +The number of times to retry consuming from a queue if it is missing. +Relevant only when missingQueuesFatal is true. +Otherwise, the container keeps retrying indefinitely. +Default: 3 + + + +queueNameGroupOnly + +When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue. +Default: false. + + + +recoveryInterval + +The interval between connection recovery attempts, in milliseconds. +Default: 5000. + + + +requeueRejected + +Whether delivery failures should be re-queued when retry is disabled or republishToDlq is false. +Default: false. + + + + + +republishDeliveryMode + +When republishToDlq is true, specifies the delivery mode of the republished message. +Default: DeliveryMode.PERSISTENT + + + +republishToDlq + +By default, messages that fail after retries are exhausted are rejected. +If a dead-letter queue (DLQ) is configured, RabbitMQ routes the failed message (unchanged) to the DLQ. +If set to true, the binder republishs failed messages to the DLQ with additional headers, including the exception message and stack trace from the cause of the final failure. +Also see the frameMaxHeadroom property. +Default: false + + + +transacted + +Whether to use transacted channels. +Default: false. + + + +ttl + +Default time to live to apply to the queue when declared (in milliseconds). +Default: no limit + + + +txSize + +The number of deliveries between acks. +Default: 1. + + + +
+
+Advanced Listener Container Configuration +To set listener container properties that are not exposed as binder or binding properties, add a single bean of type ListenerContainerCustomizer to the application context. +The binder and binding properties will be set and then the customizer will be called. +The customizer (configure() method) is provided with the queue name as well as the consumer group as arguments. +
+
+Rabbit Producer Properties +The following properties are available for Rabbit producers only and +must be prefixed with spring.cloud.stream.rabbit.bindings.<channelName>.producer.. + + +autoBindDlq + +Whether to automatically declare the DLQ and bind it to the binder DLX. +Default: false. + + + +batchingEnabled + +Whether to enable message batching by producers. +Messages are batched into one message according to the following properties (described in the next three entries in this list): 'batchSize', batchBufferLimit, and batchTimeout. +See Batching for more information. +Default: false. + + + +batchSize + +The number of messages to buffer when batching is enabled. +Default: 100. + + + +batchBufferLimit + +The maximum buffer size when batching is enabled. +Default: 10000. + + + +batchTimeout + +The batch timeout when batching is enabled. +Default: 5000. + + + +bindingRoutingKey + +The routing key with which to bind the queue to the exchange (if bindQueue is true). +Only applies to non-partitioned destinations. +Only applies if requiredGroups are provided and then only to those groups. +Default: #. + + + +bindQueue + +Whether to declare the queue and bind it to the destination exchange. +Set it to false if you have set up your own infrastructure and have previously created and bound the queue. +Only applies if requiredGroups are provided and then only to those groups. +Default: true. + + + +compress + +Whether data should be compressed when sent. +Default: false. + + + +confirmAckChannel + +When errorChannelEnabled is true, a channel to which to send positive delivery acknowledgments (aka publisher confirms). +If the channel does not exist, a DirectChannel is registered with this name. +The connection factory must be configured to enable publisher confirms. +Default: nullChannel (acks are discarded). + + + +deadLetterQueueName + +The name of the DLQ +Only applies if requiredGroups are provided and then only to those groups. +Default: prefix+destination.dlq + + + +deadLetterExchange + +A DLX to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups. +Default: 'prefix+DLX' + + + +deadLetterExchangeType + +The type of the DLX to assign to the queue. +Relevant only if autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups. +Default: 'direct' + + + +deadLetterRoutingKey + +A dead letter routing key to assign to the queue. +Relevant only when autoBindDlq is true. +Applies only when requiredGroups are provided and then only to those groups. +Default: destination + + + +declareDlx + +Whether to declare the dead letter exchange for the destination. +Relevant only if autoBindDlq is true. +Set to false if you have a pre-configured DLX. +Applies only when requiredGroups are provided and then only to those groups. +Default: true. + + + +declareExchange + +Whether to declare the exchange for the destination. +Default: true. + + + +delayExpression + +A SpEL expression to evaluate the delay to apply to the message (x-delay header). +It has no effect if the exchange is not a delayed message exchange. +Default: No x-delay header is set. + + + +delayedExchange + +Whether to declare the exchange as a Delayed Message Exchange. +Requires the delayed message exchange plugin on the broker. +The x-delayed-type argument is set to the exchangeType. +Default: false. + + + +deliveryMode + +The delivery mode. +Default: PERSISTENT. + + + +dlqDeadLetterExchange + +When a DLQ is declared, a DLX to assign to that queue. +Applies only if requiredGroups are provided and then only to those groups. +Default: none + + + +dlqDeadLetterRoutingKey + +When a DLQ is declared, a dead letter routing key to assign to that queue. +Applies only when requiredGroups are provided and then only to those groups. +Default: none + + + +dlqExpires + +How long (in milliseconds) before an unused dead letter queue is deleted. +Applies only when requiredGroups are provided and then only to those groups. +Default: no expiration + + + +dlqLazy + +Declare the dead letter queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups. + + + +dlqMaxLength + +Maximum number of messages in the dead letter queue. +Applies only if requiredGroups are provided and then only to those groups. +Default: no limit + + + +dlqMaxLengthBytes + +Maximum number of total bytes in the dead letter queue from all messages. +Applies only when requiredGroups are provided and then only to those groups. +Default: no limit + + + +dlqMaxPriority + +Maximum priority of messages in the dead letter queue (0-255) +Applies only when requiredGroups are provided and then only to those groups. +Default: none + + + +dlqTtl + +Default time (in milliseconds) to live to apply to the dead letter queue when declared. +Applies only when requiredGroups are provided and then only to those groups. +Default: no limit + + + +exchangeAutoDelete + +If declareExchange is true, whether the exchange should be auto-delete (it is removed after the last queue is removed). +Default: true. + + + +exchangeDurable + +If declareExchange is true, whether the exchange should be durable (survives broker restart). +Default: true. + + + +exchangeType + +The exchange type: direct, fanout or topic for non-partitioned destinations and direct or topic for partitioned destinations. +Default: topic. + + + +expires + +How long (in milliseconds) before an unused queue is deleted. +Applies only when requiredGroups are provided and then only to those groups. +Default: no expiration + + + +headerPatterns + +Patterns for headers to be mapped to outbound messages. +Default: ['*'] (all headers). + + + +lazy + +Declare the queue with the x-queue-mode=lazy argument. +See Lazy Queues. +Consider using a policy instead of this setting, because using a policy allows changing the setting without deleting the queue. +Applies only when requiredGroups are provided and then only to those groups. +Default: false. + + + +maxLength + +Maximum number of messages in the queue. +Applies only when requiredGroups are provided and then only to those groups. +Default: no limit + + + +maxLengthBytes + +Maximum number of total bytes in the queue from all messages. +Only applies if requiredGroups are provided and then only to those groups. +Default: no limit + + + +maxPriority + +Maximum priority of messages in the queue (0-255). +Only applies if requiredGroups are provided and then only to those groups. +Default: none + + + +prefix + +A prefix to be added to the name of the destination exchange. +Default: "". + + + +queueNameGroupOnly + +When true, consume from a queue with a name equal to the group. +Otherwise the queue name is destination.group. +This is useful, for example, when using Spring Cloud Stream to consume from an existing RabbitMQ queue. +Applies only when requiredGroups are provided and then only to those groups. +Default: false. + + + +routingKeyExpression + +A SpEL expression to determine the routing key to use when publishing messages. +For a fixed routing key, use a literal expression, such as routingKeyExpression='my.routingKey' in a properties file or routingKeyExpression: '''my.routingKey''' in a YAML file. +Default: destination or destination-<partition> for partitioned destinations. + + + +transacted + +Whether to use transacted channels. +Default: false. + + + +ttl + +Default time (in milliseconds) to live to apply to the queue when declared. +Applies only when requiredGroups are provided and then only to those groups. +Default: no limit + + + + +In the case of RabbitMQ, content type headers can be set by external applications. +Spring Cloud Stream supports them as part of an extended internal protocol used for any type of transport — including transports, such as Kafka (prior to 0.11), that do not natively support headers. + +
+
+ +Retry With the RabbitMQ Binder +When retry is enabled within the binder, the listener container thread is suspended for any back off periods that are configured. +This might be important when strict ordering is required with a single consumer. However, for other use cases, it prevents other messages from being processed on that thread. +An alternative to using binder retry is to set up dead lettering with time to live on the dead-letter queue (DLQ) as well as dead-letter configuration on the DLQ itself. +See for more information about the properties discussed here. +You can use the following example configuration to enable this feature: + + +Set autoBindDlq to true. +The binder create a DLQ. +Optionally, you can specify a name in deadLetterQueueName. + + +Set dlqTtl to the back off time you want to wait between redeliveries. + + +Set the dlqDeadLetterExchange to the default exchange. +Expired messages from the DLQ are routed to the original queue, because the default deadLetterRoutingKey is the queue name (destination.group). +Setting to the default exchange is achieved by setting the property with no value, as shown in the next example. + + +To force a message to be dead-lettered, either throw an AmqpRejectAndDontRequeueException or set requeueRejected to true (the default) and throw any exception. +The loop continue without end, which is fine for transient problems, but you may want to give up after some number of attempts. +Fortunately, RabbitMQ provides the x-death header, which lets you determine how many cycles have occurred. +To acknowledge a message after giving up, throw an ImmediateAcknowledgeAmqpException. +
+Putting it All Together +The following configuration creates an exchange myDestination with queue myDestination.consumerGroup bound to a topic exchange with a wildcard routing key #: +--- +spring.cloud.stream.bindings.input.destination=myDestination +spring.cloud.stream.bindings.input.group=consumerGroup +#disable binder retries +spring.cloud.stream.bindings.input.consumer.max-attempts=1 +#dlx/dlq setup +spring.cloud.stream.rabbit.bindings.input.consumer.auto-bind-dlq=true +spring.cloud.stream.rabbit.bindings.input.consumer.dlq-ttl=5000 +spring.cloud.stream.rabbit.bindings.input.consumer.dlq-dead-letter-exchange= +--- +This configuration creates a DLQ bound to a direct exchange (DLX) with a routing key of myDestination.consumerGroup. +When messages are rejected, they are routed to the DLQ. +After 5 seconds, the message expires and is routed to the original queue by using the queue name as the routing key, as shown in the following example: + +Spring Boot application + +@SpringBootApplication +@EnableBinding(Sink.class) +public class XDeathApplication { + + public static void main(String[] args) { + SpringApplication.run(XDeathApplication.class, args); + } + + @StreamListener(Sink.INPUT) + public void listen(String in, @Header(name = "x-death", required = false) Map<?,?> death) { + if (death != null && death.get("count").equals(3L)) { + // giving up - don't send to DLX + throw new ImmediateAcknowledgeAmqpException("Failed after 4 attempts"); + } + throw new AmqpRejectAndDontRequeueException("failed"); + } + +} + + +Notice that the count property in the x-death header is a Long. +
+
+ +Error Channels +Starting with version 1.3, the binder unconditionally sends exceptions to an error channel for each consumer destination and can also be configured to send async producer send failures to an error channel. +See for more information. +RabbitMQ has two types of send failures: + + +Returned messages, + + +Negatively acknowledged Publisher Confirms. + + +The latter is rare. +According to the RabbitMQ documentation "[A nack] will only be delivered if an internal error occurs in the Erlang process responsible for a queue.". +As well as enabling producer error channels (as described in ), the RabbitMQ binder only sends messages to the channels if the connection factory is appropriately configured, as follows: + + +ccf.setPublisherConfirms(true); + + +ccf.setPublisherReturns(true); + + +When using Spring Boot configuration for the connection factory, set the following properties: + + +spring.rabbitmq.publisher-confirms + + +spring.rabbitmq.publisher-returns + + +The payload of the ErrorMessage for a returned message is a ReturnedAmqpMessageException with the following properties: + + +failedMessage: The spring-messaging Message<?> that failed to be sent. + + +amqpMessage: The raw spring-amqp Message. + + +replyCode: An integer value indicating the reason for the failure (for example, 312 - No route). + + +replyText: A text value indicating the reason for the failure (for example, NO_ROUTE). + + +exchange: The exchange to which the message was published. + + +routingKey: The routing key used when the message was published. + + +For negatively acknowledged confirmations, the payload is a NackedAmqpMessageException with the following properties: + + +failedMessage: The spring-messaging Message<?> that failed to be sent. + + +nackReason: A reason (if available — you may need to examine the broker logs for more information). + + +There is no automatic handling of these exceptions (such as sending to a dead-letter queue). +You can consume these exceptions with your own Spring Integration flow. + + +Dead-Letter Queue Processing +Because you cannot anticipate how users would want to dispose of dead-lettered messages, the framework does not provide any standard mechanism to handle them. +If the reason for the dead-lettering is transient, you may wish to route the messages back to the original queue. +However, if the problem is a permanent issue, that could cause an infinite loop. +The following Spring Boot application shows an example of how to route those messages back to the original queue but moves them to a third parking lot queue after three attempts. +The second example uses the RabbitMQ Delayed Message Exchange to introduce a delay to the re-queued message. +In this example, the delay increases for each attempt. +These examples use a @RabbitListener to receive messages from the DLQ. +You could also use RabbitTemplate.receive() in a batch process. +The examples assume the original destination is so8400in and the consumer group is so8400. +
+Non-Partitioned Destinations +The first two examples are for when the destination is not partitioned: +@SpringBootApplication +public class ReRouteDlqApplication { + + private static final String ORIGINAL_QUEUE = "so8400in.so8400"; + + private static final String DLQ = ORIGINAL_QUEUE + ".dlq"; + + private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot"; + + private static final String X_RETRIES_HEADER = "x-retries"; + + public static void main(String[] args) throws Exception { + ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args); + System.out.println("Hit enter to terminate"); + System.in.read(); + context.close(); + } + + @Autowired + private RabbitTemplate rabbitTemplate; + + @RabbitListener(queues = DLQ) + public void rePublish(Message failedMessage) { + Integer retriesHeader = (Integer) failedMessage.getMessageProperties().getHeaders().get(X_RETRIES_HEADER); + if (retriesHeader == null) { + retriesHeader = Integer.valueOf(0); + } + if (retriesHeader < 3) { + failedMessage.getMessageProperties().getHeaders().put(X_RETRIES_HEADER, retriesHeader + 1); + this.rabbitTemplate.send(ORIGINAL_QUEUE, failedMessage); + } + else { + this.rabbitTemplate.send(PARKING_LOT, failedMessage); + } + } + + @Bean + public Queue parkingLot() { + return new Queue(PARKING_LOT); + } + +} +@SpringBootApplication +public class ReRouteDlqApplication { + + private static final String ORIGINAL_QUEUE = "so8400in.so8400"; + + private static final String DLQ = ORIGINAL_QUEUE + ".dlq"; + + private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot"; + + private static final String X_RETRIES_HEADER = "x-retries"; + + private static final String DELAY_EXCHANGE = "dlqReRouter"; + + public static void main(String[] args) throws Exception { + ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args); + System.out.println("Hit enter to terminate"); + System.in.read(); + context.close(); + } + + @Autowired + private RabbitTemplate rabbitTemplate; + + @RabbitListener(queues = DLQ) + public void rePublish(Message failedMessage) { + Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders(); + Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER); + if (retriesHeader == null) { + retriesHeader = Integer.valueOf(0); + } + if (retriesHeader < 3) { + headers.put(X_RETRIES_HEADER, retriesHeader + 1); + headers.put("x-delay", 5000 * retriesHeader); + this.rabbitTemplate.send(DELAY_EXCHANGE, ORIGINAL_QUEUE, failedMessage); + } + else { + this.rabbitTemplate.send(PARKING_LOT, failedMessage); + } + } + + @Bean + public DirectExchange delayExchange() { + DirectExchange exchange = new DirectExchange(DELAY_EXCHANGE); + exchange.setDelayed(true); + return exchange; + } + + @Bean + public Binding bindOriginalToDelay() { + return BindingBuilder.bind(new Queue(ORIGINAL_QUEUE)).to(delayExchange()).with(ORIGINAL_QUEUE); + } + + @Bean + public Queue parkingLot() { + return new Queue(PARKING_LOT); + } + +} +
+
+Partitioned Destinations +With partitioned destinations, there is one DLQ for all partitions. We determine the original queue from the headers. +
+<literal>republishToDlq=false</literal> +When republishToDlq is false, RabbitMQ publishes the message to the DLX/DLQ with an x-death header containing information about the original destination, as shown in the following example: +@SpringBootApplication +public class ReRouteDlqApplication { + + private static final String ORIGINAL_QUEUE = "so8400in.so8400"; + + private static final String DLQ = ORIGINAL_QUEUE + ".dlq"; + + private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot"; + + private static final String X_DEATH_HEADER = "x-death"; + + private static final String X_RETRIES_HEADER = "x-retries"; + + public static void main(String[] args) throws Exception { + ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args); + System.out.println("Hit enter to terminate"); + System.in.read(); + context.close(); + } + + @Autowired + private RabbitTemplate rabbitTemplate; + + @SuppressWarnings("unchecked") + @RabbitListener(queues = DLQ) + public void rePublish(Message failedMessage) { + Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders(); + Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER); + if (retriesHeader == null) { + retriesHeader = Integer.valueOf(0); + } + if (retriesHeader < 3) { + headers.put(X_RETRIES_HEADER, retriesHeader + 1); + List<Map<String, ?>> xDeath = (List<Map<String, ?>>) headers.get(X_DEATH_HEADER); + String exchange = (String) xDeath.get(0).get("exchange"); + List<String> routingKeys = (List<String>) xDeath.get(0).get("routing-keys"); + this.rabbitTemplate.send(exchange, routingKeys.get(0), failedMessage); + } + else { + this.rabbitTemplate.send(PARKING_LOT, failedMessage); + } + } + + @Bean + public Queue parkingLot() { + return new Queue(PARKING_LOT); + } + +} +
+
+<literal>republishToDlq=true</literal> +When republishToDlq is true, the republishing recoverer adds the original exchange and routing key to headers, as shown in the following example: +@SpringBootApplication +public class ReRouteDlqApplication { + + private static final String ORIGINAL_QUEUE = "so8400in.so8400"; + + private static final String DLQ = ORIGINAL_QUEUE + ".dlq"; + + private static final String PARKING_LOT = ORIGINAL_QUEUE + ".parkingLot"; + + private static final String X_RETRIES_HEADER = "x-retries"; + + private static final String X_ORIGINAL_EXCHANGE_HEADER = RepublishMessageRecoverer.X_ORIGINAL_EXCHANGE; + + private static final String X_ORIGINAL_ROUTING_KEY_HEADER = RepublishMessageRecoverer.X_ORIGINAL_ROUTING_KEY; + + public static void main(String[] args) throws Exception { + ConfigurableApplicationContext context = SpringApplication.run(ReRouteDlqApplication.class, args); + System.out.println("Hit enter to terminate"); + System.in.read(); + context.close(); + } + + @Autowired + private RabbitTemplate rabbitTemplate; + + @RabbitListener(queues = DLQ) + public void rePublish(Message failedMessage) { + Map<String, Object> headers = failedMessage.getMessageProperties().getHeaders(); + Integer retriesHeader = (Integer) headers.get(X_RETRIES_HEADER); + if (retriesHeader == null) { + retriesHeader = Integer.valueOf(0); + } + if (retriesHeader < 3) { + headers.put(X_RETRIES_HEADER, retriesHeader + 1); + String exchange = (String) headers.get(X_ORIGINAL_EXCHANGE_HEADER); + String originalRoutingKey = (String) headers.get(X_ORIGINAL_ROUTING_KEY_HEADER); + this.rabbitTemplate.send(exchange, originalRoutingKey, failedMessage); + } + else { + this.rabbitTemplate.send(PARKING_LOT, failedMessage); + } + } + + @Bean + public Queue parkingLot() { + return new Queue(PARKING_LOT); + } + +} +
+
+
+ +Partitioning with the RabbitMQ Binder +RabbitMQ does not support partitioning natively. +Sometimes, it is advantageous to send data to specific partitions — for example, when you want to strictly order message processing, all messages for a particular customer should go to the same partition. +The RabbitMessageChannelBinder provides partitioning by binding a queue for each partition to the destination exchange. +The following Java and YAML examples show how to configure the producer: + +Producer + +@SpringBootApplication +@EnableBinding(Source.class) +public class RabbitPartitionProducerApplication { + + private static final Random RANDOM = new Random(System.currentTimeMillis()); + + private static final String[] data = new String[] { + "abc1", "def1", "qux1", + "abc2", "def2", "qux2", + "abc3", "def3", "qux3", + "abc4", "def4", "qux4", + }; + + public static void main(String[] args) { + new SpringApplicationBuilder(RabbitPartitionProducerApplication.class) + .web(false) + .run(args); + } + + @InboundChannelAdapter(channel = Source.OUTPUT, poller = @Poller(fixedRate = "5000")) + public Message<?> generate() { + String value = data[RANDOM.nextInt(data.length)]; + System.out.println("Sending: " + value); + return MessageBuilder.withPayload(value) + .setHeader("partitionKey", value) + .build(); + } + +} + + + +application.yml + + spring: + cloud: + stream: + bindings: + output: + destination: partitioned.destination + producer: + partitioned: true + partition-key-expression: headers['partitionKey'] + partition-count: 2 + required-groups: + - myGroup + + + +The configuration in the prececing example uses the default partitioning (key.hashCode() % partitionCount). +This may or may not provide a suitably balanced algorithm, depending on the key values. +You can override this default by using the partitionSelectorExpression or partitionSelectorClass properties. +The required-groups property is required only if you need the consumer queues to be provisioned when the producer is deployed. +Otherwise, any messages sent to a partition are lost until the corresponding consumer is deployed. + +The following configuration provisions a topic exchange: + + + + + +part exchange + + +The following queues are bound to that exchange: + + + + + +part queues + + +The following bindings associate the queues to the exchange: + + + + + +part bindings + + +The following Java and YAML examples continue the previous examples and show how to configure the consumer: + +Consumer + +@SpringBootApplication +@EnableBinding(Sink.class) +public class RabbitPartitionConsumerApplication { + + public static void main(String[] args) { + new SpringApplicationBuilder(RabbitPartitionConsumerApplication.class) + .web(false) + .run(args); + } + + @StreamListener(Sink.INPUT) + public void listen(@Payload String in, @Header(AmqpHeaders.CONSUMER_QUEUE) String queue) { + System.out.println(in + " received from queue " + queue); + } + +} + + + +application.yml + + spring: + cloud: + stream: + bindings: + input: + destination: partitioned.destination + group: myGroup + consumer: + partitioned: true + instance-index: 0 + + + +The RabbitMessageChannelBinder does not support dynamic scaling. +There must be at least one consumer per partition. +The consumer’s instanceIndex is used to indicate which partition is consumed. +Platforms such as Cloud Foundry can have only one instance with an instanceIndex. + + +
+ +Appendices + +Building +
+Basic Compile and Test +To build the source you will need to install JDK 1.8. +The build uses the Maven wrapper so you don’t have to install a specific +version of Maven. To enable the tests, you should have RabbitMQ server running +on localhost and the default port (5672) +before building. +The main build command is +$ ./mvnw clean install +You can also add '-DskipTests' if you like, to avoid running the tests. + +You can also install Maven (>=3.3.3) yourself and run the mvn command +in place of ./mvnw in the examples below. If you do that you also +might need to add -P spring if your local Maven settings do not +contain repository declarations for spring pre-release artifacts. + + +Be aware that you might need to increase the amount of memory +available to Maven by setting a MAVEN_OPTS environment variable with +a value like -Xmx512m -XX:MaxPermSize=128m. We try to cover this in +the .mvn configuration, so if you find you have to do it to make a +build succeed, please raise a ticket to get the settings added to +source control. + +The projects that require middleware generally include a +docker-compose.yml, so consider using +Docker Compose to run the middeware servers +in Docker containers. +
+
+Documentation +There is a "full" profile that will generate documentation. +
+
+Working with the code +If you don’t have an IDE preference we would recommend that you use +Spring Tools Suite or +Eclipse when working with the code. We use the +m2eclipe eclipse plugin for maven support. Other IDEs and tools +should also work without issue. +
+Importing into eclipse with m2eclipse +We recommend the m2eclipe eclipse plugin when working with +eclipse. If you don’t already have m2eclipse installed it is available from the "eclipse +marketplace". +Unfortunately m2e does not yet support Maven 3.3, so once the projects +are imported into Eclipse you will also need to tell m2eclipse to use +the .settings.xml file for the projects. If you do not do this you +may see many different errors related to the POMs in the +projects. Open your Eclipse preferences, expand the Maven +preferences, and select User Settings. In the User Settings field +click Browse and navigate to the Spring Cloud project you imported +selecting the .settings.xml file in that project. Click Apply and +then OK to save the preference changes. + +Alternatively you can copy the repository settings from .settings.xml into your own ~/.m2/settings.xml. + +
+
+Importing into eclipse without m2eclipse +If you prefer not to use m2eclipse you can generate eclipse project metadata using the +following command: +$ ./mvnw eclipse:eclipse +The generated eclipse projects can be imported by selecting import existing projects +from the file menu. +
+
+
+ +Contributing +Spring Cloud is released under the non-restrictive Apache 2.0 license, +and follows a very standard Github development process, using Github +tracker for issues and merging pull requests into master. If you want +to contribute even something trivial please do not hesitate, but +follow the guidelines below. +
+Sign the Contributor License Agreement +Before we accept a non-trivial patch or pull request we will need you to sign the +contributor’s agreement. +Signing the contributor’s agreement does not grant anyone commit rights to the main +repository, but it does mean that we can accept your contributions, and you will get an +author credit if we do. Active contributors might be asked to join the core team, and +given the ability to merge pull requests. +
+
+Code Conventions and Housekeeping +None of these is essential for a pull request, but they will all help. They can also be +added after the original pull request but before a merge. + + +Use the Spring Framework code format conventions. If you use Eclipse +you can import formatter settings using the +eclipse-code-formatter.xml file from the +Spring +Cloud Build project. If using IntelliJ, you can use the +Eclipse Code Formatter +Plugin to import the same file. + + +Make sure all new .java files to have a simple Javadoc class comment with at least an +@author tag identifying you, and preferably at least a paragraph on what the class is +for. + + +Add the ASF license header comment to all new .java files (copy from existing files +in the project) + + +Add yourself as an @author to the .java files that you modify substantially (more +than cosmetic changes). + + +Add some Javadocs and, if you change the namespace, some XSD doc elements. + + +A few unit tests would help a lot as well — someone has to do it. + + +If no-one else is using your branch, please rebase it against the current master (or +other target branch in the main project). + + +When writing a commit message please follow these conventions, +if you are fixing an existing issue please add Fixes gh-XXXX at the end of the commit +message (where XXXX is the issue number). + + +
+
+
+
\ No newline at end of file